On Wed, Jul 28, 2021 at 1:11 PM Viresh Kumar viresh.kumar@linaro.org wrote:
This patch adds support for interrupts to the virtio-gpio specification. This uses the feature bit 0 for the same.
Fixes: https://github.com/oasis-tcs/virtio-spec/issues/110 Signed-off-by: Viresh Kumar viresh.kumar@linaro.org
This also looks good to me overall, I have a few idea for clarifications below, but none of them are likely to change the actual behavior.
More importantly though, I would like to see a review by the linux irqchip maintainers (Marc Zyngier and/or Thomas Gleixner, both added to Cc here) before this gets merged into the virtio specification. I have a basic understanding of the subsystem but probably got something wrong myself.
That review is probably easier once you post the driver code for it but if Thomas or Marc want to comment on the spec before seeing the driver, the patch is available at [1], along with the base virtio-gpio spec.
+\subsubsection{requestq Operation: Set IRQ Type}\label{sec:Device Types / GPIO Device / requestq Operation / Set IRQ Type}
+The driver sends this message to request the device to set the IRQ trigger type, +to one of the \field{VIRTIO_GPIO_IRQ_TYPE_*} values, for a line configured for +input.
+This request is only allowed if the \field{VIRTIO_GPIO_F_IRQ} feature is +enabled by the device.
+The device MUST mask the interrupt for a GPIO line, if the trigger type is set +to \field{VIRTIO_GPIO_IRQ_TYPE_NONE} and also discard any latched interrupt +status associated with the line. The device MUST unmask the interrupt for any +other value of the trigger type.
I think the last sentence here is somewhat ambiguous, I'm not sure what it actually means to unmask a level triggered line if the driver has not queued an event request for that line.
For an edge triggered interrupt, I think it makes sense to unmask the interrupt at the device level and latch any incoming events until a request gets queued on the eventq.
+\item The driver MUST set the IRQ trigger type to
- \field{VIRTIO_GPIO_IRQ_TYPE_NONE} once it is done using the GPIO line,
- previously configured for a different trigger type.
\end{itemize}
I'm not entirely sure why this "MUST" be done. Can't the device clean up the irq when the line is set back to direction=none or the queues are shut down?
+\item The driver queues a pair of buffers, interrupt-request and
- interrupt-response, to the \field{eventq} virtqueue for the GPIO line.
+\item The driver notifies the device of the presence of new buffers on the
- \field{eventq} virtqueue.
+\item The driver sends the \field{VIRTIO_GPIO_MSG_SET_IRQ_TYPE} message over the
- \field{requestq} virtqueue and the device configures the GPIO line for the
- requested trigger type and unmasks the interrupt.
Ah, I guess that explains the flow. I was expecting the reverse order (first set-irq-type, then queue event request), but I suppose it works either way.
+\item The driver on receiving the notification from the device, processes the
- interrupt. The driver may try to update the trigger-type of the interrupt
- for the GPIO line over the \field{requestq} virtqueue.
+\item The driver may again queue, same or new, pair of buffers for that GPIO
- line and notify the device.
I feel there needs to be another step between these two, as this is where some of the magic happens that guarantees we don't get too many or not enough interrupts. How about:
\item In a typical guest operating system kernel, the virtio-gpio driver notifies another device driver that is associated with this gpio line to process the event.
\item In the case of a level triggered interrupt, the secondary device driver must fully process and acknowledge the event at its source to return the line to its inactive state.
Arnd
[1] https://lists.oasis-open.org/archives/virtio-dev/202107/msg00206.html