Commit 76e48896 authored by Kieran Bingham's avatar Kieran Bingham Committed by Mauro Carvalho Chehab

[media] v4l: vsp1: Support chained display lists

When display lists are linked in a chain, they will be processed
automatically by the hardware, with each list linking to the next. Only
on the last display list will the frame end interrupt be fired to mark
the completion event.

Upon frame-end, the chain will be iterated to release each display list
back to the free list.

The chained lists use case (image partitioning) can require up to 64
lists per frame in the worst case scenario, bump up the number of
preallocated lists.
Signed-off-by: default avatarKieran Bingham <kieran+renesas@bingham.xyz>
Signed-off-by: default avatarLaurent Pinchart <laurent.pinchart+renesas@ideasonboard.com>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab@s-opensource.com>
parent 8ddf3784
...@@ -21,7 +21,6 @@ ...@@ -21,7 +21,6 @@
#include "vsp1_dl.h" #include "vsp1_dl.h"
#define VSP1_DL_NUM_ENTRIES 256 #define VSP1_DL_NUM_ENTRIES 256
#define VSP1_DL_NUM_LISTS 3
#define VSP1_DLH_INT_ENABLE (1 << 1) #define VSP1_DLH_INT_ENABLE (1 << 1)
#define VSP1_DLH_AUTO_START (1 << 0) #define VSP1_DLH_AUTO_START (1 << 0)
...@@ -71,6 +70,7 @@ struct vsp1_dl_body { ...@@ -71,6 +70,7 @@ struct vsp1_dl_body {
* @dma: DMA address for the header * @dma: DMA address for the header
* @body0: first display list body * @body0: first display list body
* @fragments: list of extra display list bodies * @fragments: list of extra display list bodies
* @chain: entry in the display list partition chain
*/ */
struct vsp1_dl_list { struct vsp1_dl_list {
struct list_head list; struct list_head list;
...@@ -81,6 +81,9 @@ struct vsp1_dl_list { ...@@ -81,6 +81,9 @@ struct vsp1_dl_list {
struct vsp1_dl_body body0; struct vsp1_dl_body body0;
struct list_head fragments; struct list_head fragments;
bool has_chain;
struct list_head chain;
}; };
enum vsp1_dl_mode { enum vsp1_dl_mode {
...@@ -262,7 +265,6 @@ static struct vsp1_dl_list *vsp1_dl_list_alloc(struct vsp1_dl_manager *dlm) ...@@ -262,7 +265,6 @@ static struct vsp1_dl_list *vsp1_dl_list_alloc(struct vsp1_dl_manager *dlm)
memset(dl->header, 0, sizeof(*dl->header)); memset(dl->header, 0, sizeof(*dl->header));
dl->header->lists[0].addr = dl->body0.dma; dl->header->lists[0].addr = dl->body0.dma;
dl->header->flags = VSP1_DLH_INT_ENABLE;
} }
return dl; return dl;
...@@ -293,6 +295,11 @@ struct vsp1_dl_list *vsp1_dl_list_get(struct vsp1_dl_manager *dlm) ...@@ -293,6 +295,11 @@ struct vsp1_dl_list *vsp1_dl_list_get(struct vsp1_dl_manager *dlm)
if (!list_empty(&dlm->free)) { if (!list_empty(&dlm->free)) {
dl = list_first_entry(&dlm->free, struct vsp1_dl_list, list); dl = list_first_entry(&dlm->free, struct vsp1_dl_list, list);
list_del(&dl->list); list_del(&dl->list);
/* The display list chain must be initialised to ensure every
* display list can assert list_empty() if it is not in a chain.
*/
INIT_LIST_HEAD(&dl->chain);
} }
spin_unlock_irqrestore(&dlm->lock, flags); spin_unlock_irqrestore(&dlm->lock, flags);
...@@ -303,9 +310,21 @@ struct vsp1_dl_list *vsp1_dl_list_get(struct vsp1_dl_manager *dlm) ...@@ -303,9 +310,21 @@ struct vsp1_dl_list *vsp1_dl_list_get(struct vsp1_dl_manager *dlm)
/* This function must be called with the display list manager lock held.*/ /* This function must be called with the display list manager lock held.*/
static void __vsp1_dl_list_put(struct vsp1_dl_list *dl) static void __vsp1_dl_list_put(struct vsp1_dl_list *dl)
{ {
struct vsp1_dl_list *dl_child;
if (!dl) if (!dl)
return; return;
/* Release any linked display-lists which were chained for a single
* hardware operation.
*/
if (dl->has_chain) {
list_for_each_entry(dl_child, &dl->chain, chain)
__vsp1_dl_list_put(dl_child);
}
dl->has_chain = false;
/* We can't free fragments here as DMA memory can only be freed in /* We can't free fragments here as DMA memory can only be freed in
* interruptible context. Move all fragments to the display list * interruptible context. Move all fragments to the display list
* manager's list of fragments to be freed, they will be * manager's list of fragments to be freed, they will be
...@@ -383,27 +402,46 @@ int vsp1_dl_list_add_fragment(struct vsp1_dl_list *dl, ...@@ -383,27 +402,46 @@ int vsp1_dl_list_add_fragment(struct vsp1_dl_list *dl,
return 0; return 0;
} }
void vsp1_dl_list_commit(struct vsp1_dl_list *dl) /**
* vsp1_dl_list_add_chain - Add a display list to a chain
* @head: The head display list
* @dl: The new display list
*
* Add a display list to an existing display list chain. The chained lists
* will be automatically processed by the hardware without intervention from
* the CPU. A display list end interrupt will only complete after the last
* display list in the chain has completed processing.
*
* Adding a display list to a chain passes ownership of the display list to
* the head display list item. The chain is released when the head dl item is
* put back with __vsp1_dl_list_put().
*
* Chained display lists are only usable in header mode. Attempts to add a
* display list to a chain in header-less mode will return an error.
*/
int vsp1_dl_list_add_chain(struct vsp1_dl_list *head,
struct vsp1_dl_list *dl)
{ {
struct vsp1_dl_manager *dlm = dl->dlm; /* Chained lists are only available in header mode. */
struct vsp1_device *vsp1 = dlm->vsp1; if (head->dlm->mode != VSP1_DL_MODE_HEADER)
unsigned long flags; return -EINVAL;
bool update;
spin_lock_irqsave(&dlm->lock, flags); head->has_chain = true;
list_add_tail(&dl->chain, &head->chain);
return 0;
}
if (dl->dlm->mode == VSP1_DL_MODE_HEADER) { static void vsp1_dl_list_fill_header(struct vsp1_dl_list *dl, bool is_last)
{
struct vsp1_dl_header_list *hdr = dl->header->lists; struct vsp1_dl_header_list *hdr = dl->header->lists;
struct vsp1_dl_body *dlb; struct vsp1_dl_body *dlb;
unsigned int num_lists = 0; unsigned int num_lists = 0;
/* Fill the header with the display list bodies addresses and /* Fill the header with the display list bodies addresses and sizes. The
* sizes. The address of the first body has already been filled * address of the first body has already been filled when the display
* when the display list was allocated. * list was allocated.
*
* In header mode the caller guarantees that the hardware is
* idle at this point.
*/ */
hdr->num_bytes = dl->body0.num_entries hdr->num_bytes = dl->body0.num_entries
* sizeof(*dl->header->lists); * sizeof(*dl->header->lists);
...@@ -417,6 +455,49 @@ void vsp1_dl_list_commit(struct vsp1_dl_list *dl) ...@@ -417,6 +455,49 @@ void vsp1_dl_list_commit(struct vsp1_dl_list *dl)
} }
dl->header->num_lists = num_lists; dl->header->num_lists = num_lists;
/* If this display list's chain is not empty, we are on a list, where
* the next item in the list is the display list entity which should be
* automatically queued by the hardware.
*/
if (!list_empty(&dl->chain) && !is_last) {
struct vsp1_dl_list *next = list_next_entry(dl, chain);
dl->header->next_header = next->dma;
dl->header->flags = VSP1_DLH_AUTO_START;
} else {
dl->header->flags = VSP1_DLH_INT_ENABLE;
}
}
void vsp1_dl_list_commit(struct vsp1_dl_list *dl)
{
struct vsp1_dl_manager *dlm = dl->dlm;
struct vsp1_device *vsp1 = dlm->vsp1;
unsigned long flags;
bool update;
spin_lock_irqsave(&dlm->lock, flags);
if (dl->dlm->mode == VSP1_DL_MODE_HEADER) {
struct vsp1_dl_list *dl_child;
/* In header mode the caller guarantees that the hardware is
* idle at this point.
*/
/* Fill the header for the head and chained display lists. */
vsp1_dl_list_fill_header(dl, list_empty(&dl->chain));
list_for_each_entry(dl_child, &dl->chain, chain) {
bool last = list_is_last(&dl_child->chain, &dl->chain);
vsp1_dl_list_fill_header(dl_child, last);
}
/* Commit the head display list to hardware. Chained headers
* will auto-start.
*/
vsp1_write(vsp1, VI6_DL_HDR_ADDR(dlm->index), dl->dma); vsp1_write(vsp1, VI6_DL_HDR_ADDR(dlm->index), dl->dma);
dlm->active = dl; dlm->active = dl;
......
...@@ -41,5 +41,6 @@ void vsp1_dl_fragment_free(struct vsp1_dl_body *dlb); ...@@ -41,5 +41,6 @@ void vsp1_dl_fragment_free(struct vsp1_dl_body *dlb);
void vsp1_dl_fragment_write(struct vsp1_dl_body *dlb, u32 reg, u32 data); void vsp1_dl_fragment_write(struct vsp1_dl_body *dlb, u32 reg, u32 data);
int vsp1_dl_list_add_fragment(struct vsp1_dl_list *dl, int vsp1_dl_list_add_fragment(struct vsp1_dl_list *dl,
struct vsp1_dl_body *dlb); struct vsp1_dl_body *dlb);
int vsp1_dl_list_add_chain(struct vsp1_dl_list *head, struct vsp1_dl_list *dl);
#endif /* __VSP1_DL_H__ */ #endif /* __VSP1_DL_H__ */
...@@ -364,7 +364,7 @@ struct vsp1_rwpf *vsp1_wpf_create(struct vsp1_device *vsp1, unsigned int index) ...@@ -364,7 +364,7 @@ struct vsp1_rwpf *vsp1_wpf_create(struct vsp1_device *vsp1, unsigned int index)
return ERR_PTR(ret); return ERR_PTR(ret);
/* Initialize the display list manager. */ /* Initialize the display list manager. */
wpf->dlm = vsp1_dlm_create(vsp1, index, 4); wpf->dlm = vsp1_dlm_create(vsp1, index, 64);
if (!wpf->dlm) { if (!wpf->dlm) {
ret = -ENOMEM; ret = -ENOMEM;
goto error; goto error;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment