media: bcm2835-unicam: Handle a repeated frame start with no end
authorDave Stevenson <dave.stevenson@raspberrypi.com>
Fri, 4 Feb 2022 16:12:35 +0000 (16:12 +0000)
committerDom Cobley <popcornmix@gmail.com>
Mon, 21 Mar 2022 16:04:42 +0000 (16:04 +0000)
In the case of 2 frame starts being received with no frame end
between, the queued buffer held in next_frm was lost as the
pointer was overwritten with the dummy buffer.

Signed-off-by: Dave Stevenson <dave.stevenson@raspberrypi.com>
drivers/media/platform/bcm2835/bcm2835-unicam.c

index 4bba864b2366c1240bf0144f9842741aa9b7cec8..1e0101bb3af9ec8e21fc110020509a30daae1be5 100644 (file)
@@ -933,10 +933,14 @@ static irqreturn_t unicam_isr(int irq, void *dev)
                         * as complete, as the HW will reuse that buffer.
                         */
                        if (unicam->node[i].cur_frm &&
-                           unicam->node[i].cur_frm != unicam->node[i].next_frm)
+                           unicam->node[i].cur_frm != unicam->node[i].next_frm) {
                                unicam_process_buffer_complete(&unicam->node[i],
                                                               sequence);
-                       unicam->node[i].cur_frm = unicam->node[i].next_frm;
+                               unicam->node[i].cur_frm = unicam->node[i].next_frm;
+                               unicam->node[i].next_frm = NULL;
+                       } else {
+                               unicam->node[i].cur_frm = unicam->node[i].next_frm;
+                       }
                }
                unicam->sequence++;
        }
@@ -959,10 +963,25 @@ static irqreturn_t unicam_isr(int irq, void *dev)
                                           i);
                        /*
                         * Set the next frame output to go to a dummy frame
-                        * if we have not managed to obtain another frame
-                        * from the queue.
+                        * if no buffer currently queued.
                         */
-                       unicam_schedule_dummy_buffer(&unicam->node[i]);
+                       if (!unicam->node[i].next_frm ||
+                           unicam->node[i].next_frm == unicam->node[i].cur_frm) {
+                               unicam_schedule_dummy_buffer(&unicam->node[i]);
+                       } else if (unicam->node[i].cur_frm) {
+                               /*
+                                * Repeated FS without FE. Hardware will have
+                                * swapped buffers, but the cur_frm doesn't
+                                * contain valid data. Return cur_frm to the
+                                * queue.
+                                */
+                               spin_lock(&unicam->node[i].dma_queue_lock);
+                               list_add_tail(&unicam->node[i].cur_frm->list,
+                                             &unicam->node[i].dma_queue);
+                               spin_unlock(&unicam->node[i].dma_queue_lock);
+                               unicam->node[i].cur_frm = unicam->node[i].next_frm;
+                               unicam->node[i].next_frm = NULL;
+                       }
                }
 
                unicam_queue_event_sof(unicam);