aboutsummaryrefslogtreecommitdiffstats
path: root/target/linux/bcm27xx/patches-5.15/950-0410-media-rpivid-Fix-H265-aux-ent-reuse-of-the-same-slot.patch
diff options
context:
space:
mode:
Diffstat (limited to 'target/linux/bcm27xx/patches-5.15/950-0410-media-rpivid-Fix-H265-aux-ent-reuse-of-the-same-slot.patch')
-rw-r--r--target/linux/bcm27xx/patches-5.15/950-0410-media-rpivid-Fix-H265-aux-ent-reuse-of-the-same-slot.patch151
1 files changed, 151 insertions, 0 deletions
diff --git a/target/linux/bcm27xx/patches-5.15/950-0410-media-rpivid-Fix-H265-aux-ent-reuse-of-the-same-slot.patch b/target/linux/bcm27xx/patches-5.15/950-0410-media-rpivid-Fix-H265-aux-ent-reuse-of-the-same-slot.patch
new file mode 100644
index 0000000000..318f2bf0f5
--- /dev/null
+++ b/target/linux/bcm27xx/patches-5.15/950-0410-media-rpivid-Fix-H265-aux-ent-reuse-of-the-same-slot.patch
@@ -0,0 +1,151 @@
+From a873fa8e44651d31d005199053a16cfc440ffc49 Mon Sep 17 00:00:00 2001
+From: John Cox <jc@kynesim.co.uk>
+Date: Thu, 24 Jun 2021 14:43:49 +0100
+Subject: [PATCH] media: rpivid: Fix H265 aux ent reuse of the same
+ slot
+
+It is legitimate, though unusual, for an aux ent associated with a slot
+to be selected in phase 0 before a previous selection has been used and
+released in phase 2. Fix such that if the slot is found to be in use
+that the aux ent associated with it is reused rather than an new aux
+ent being created. This fixes a problem where when the first aux ent
+was released the second was lost track of.
+
+This bug spotted in Nick's testing. It may explain some other occasional,
+unreliable decode error reports where dmesg included "Missing DPB AUX
+ent" logging.
+
+Signed-off-by: John Cox <jc@kynesim.co.uk>
+---
+ drivers/staging/media/rpivid/rpivid_h265.c | 75 ++++++++++++++--------
+ 1 file changed, 49 insertions(+), 26 deletions(-)
+
+--- a/drivers/staging/media/rpivid/rpivid_h265.c
++++ b/drivers/staging/media/rpivid/rpivid_h265.c
+@@ -371,7 +371,8 @@ static void aux_q_free(struct rpivid_ctx
+ kfree(aq);
+ }
+
+-static struct rpivid_q_aux *aux_q_alloc(struct rpivid_ctx *const ctx)
++static struct rpivid_q_aux *aux_q_alloc(struct rpivid_ctx *const ctx,
++ const unsigned int q_index)
+ {
+ struct rpivid_dev *const dev = ctx->dev;
+ struct rpivid_q_aux *const aq = kzalloc(sizeof(*aq), GFP_KERNEL);
+@@ -379,11 +380,17 @@ static struct rpivid_q_aux *aux_q_alloc(
+ if (!aq)
+ return NULL;
+
+- aq->refcount = 1;
+ if (gptr_alloc(dev, &aq->col, ctx->colmv_picsize,
+ DMA_ATTR_FORCE_CONTIGUOUS | DMA_ATTR_NO_KERNEL_MAPPING))
+ goto fail;
+
++ /*
++ * Spinlock not required as called in P0 only and
++ * aux checks done by _new
++ */
++ aq->refcount = 1;
++ aq->q_index = q_index;
++ ctx->aux_ents[q_index] = aq;
+ return aq;
+
+ fail:
+@@ -398,22 +405,38 @@ static struct rpivid_q_aux *aux_q_new(st
+ unsigned long lockflags;
+
+ spin_lock_irqsave(&ctx->aux_lock, lockflags);
+- aq = ctx->aux_free;
+- if (aq) {
++ /*
++ * If we already have this allocated to a slot then use that
++ * and assume that it will all work itself out in the pipeline
++ */
++ if ((aq = ctx->aux_ents[q_index]) != NULL) {
++ ++aq->refcount;
++ } else if ((aq = ctx->aux_free) != NULL) {
+ ctx->aux_free = aq->next;
+ aq->next = NULL;
+ aq->refcount = 1;
++ aq->q_index = q_index;
++ ctx->aux_ents[q_index] = aq;
+ }
+ spin_unlock_irqrestore(&ctx->aux_lock, lockflags);
+
+- if (!aq) {
+- aq = aux_q_alloc(ctx);
+- if (!aq)
+- return NULL;
+- }
++ if (!aq)
++ aq = aux_q_alloc(ctx, q_index);
++
++ return aq;
++}
++
++static struct rpivid_q_aux *aux_q_ref_idx(struct rpivid_ctx *const ctx,
++ const int q_index)
++{
++ unsigned long lockflags;
++ struct rpivid_q_aux *aq;
++
++ spin_lock_irqsave(&ctx->aux_lock, lockflags);
++ if ((aq = ctx->aux_ents[q_index]) != NULL)
++ ++aq->refcount;
++ spin_unlock_irqrestore(&ctx->aux_lock, lockflags);
+
+- aq->q_index = q_index;
+- ctx->aux_ents[q_index] = aq;
+ return aq;
+ }
+
+@@ -436,21 +459,21 @@ static void aux_q_release(struct rpivid_
+ struct rpivid_q_aux **const paq)
+ {
+ struct rpivid_q_aux *const aq = *paq;
+- *paq = NULL;
+-
+- if (aq) {
+- unsigned long lockflags;
++ unsigned long lockflags;
+
+- spin_lock_irqsave(&ctx->aux_lock, lockflags);
++ if (!aq)
++ return;
+
+- if (--aq->refcount == 0) {
+- aq->next = ctx->aux_free;
+- ctx->aux_free = aq;
+- ctx->aux_ents[aq->q_index] = NULL;
+- }
++ *paq = NULL;
+
+- spin_unlock_irqrestore(&ctx->aux_lock, lockflags);
++ spin_lock_irqsave(&ctx->aux_lock, lockflags);
++ if (--aq->refcount == 0) {
++ aq->next = ctx->aux_free;
++ ctx->aux_free = aq;
++ ctx->aux_ents[aq->q_index] = NULL;
++ aq->q_index = ~0U;
+ }
++ spin_unlock_irqrestore(&ctx->aux_lock, lockflags);
+ }
+
+ static void aux_q_init(struct rpivid_ctx *const ctx)
+@@ -1958,12 +1981,12 @@ static void rpivid_h265_setup(struct rpi
+ }
+
+ if (use_aux) {
+- dpb_q_aux[i] = aux_q_ref(ctx,
+- ctx->aux_ents[buffer_index]);
++ dpb_q_aux[i] = aux_q_ref_idx(ctx, buffer_index);
+ if (!dpb_q_aux[i])
+ v4l2_warn(&dev->v4l2_dev,
+- "Missing DPB AUX ent %d index=%d\n",
+- i, buffer_index);
++ "Missing DPB AUX ent %d, timestamp=%lld, index=%d\n",
++ i, (long long)sh->dpb[i].timestamp,
++ buffer_index);
+ }
+
+ de->ref_addrs[i] =