Commit f0bb8dd9 authored by Mauro Carvalho Chehab's avatar Mauro Carvalho Chehab

[media] doc-rst: add pxa_camera documentation

Convert pxa_camera to ReST format and add it to the
media/v4l-drivers book.
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab@s-opensource.com>
parent 1bb6f32a
...@@ -31,4 +31,5 @@ License". ...@@ -31,4 +31,5 @@ License".
omap3isp omap3isp
omap4_camera omap4_camera
pvrusb2 pvrusb2
pxa_camera
zr364xx zr364xx
PXA-Camera Host Driver PXA-Camera Host Driver
====================== ======================
Author: Robert Jarzmik <robert.jarzmik@free.fr>
Constraints Constraints
----------- -----------
a) Image size for YUV422P format
All YUV422P images are enforced to have width x height % 16 = 0. a) Image size for YUV422P format
This is due to DMA constraints, which transfers only planes of 8 byte All YUV422P images are enforced to have width x height % 16 = 0.
multiples. This is due to DMA constraints, which transfers only planes of 8 byte
multiples.
Global video workflow Global video workflow
--------------------- ---------------------
a) QCI stopped
Initialy, the QCI interface is stopped. a) QCI stopped
When a buffer is queued (pxa_videobuf_ops->buf_queue), the QCI starts. Initialy, the QCI interface is stopped.
When a buffer is queued (pxa_videobuf_ops->buf_queue), the QCI starts.
b) QCI started
More buffers can be queued while the QCI is started without halting the b) QCI started
capture. The new buffers are "appended" at the tail of the DMA chain, and More buffers can be queued while the QCI is started without halting the
smoothly captured one frame after the other. capture. The new buffers are "appended" at the tail of the DMA chain, and
smoothly captured one frame after the other.
Once a buffer is filled in the QCI interface, it is marked as "DONE" and
removed from the active buffers list. It can be then requeud or dequeued by Once a buffer is filled in the QCI interface, it is marked as "DONE" and
userland application. removed from the active buffers list. It can be then requeud or dequeued by
userland application.
Once the last buffer is filled in, the QCI interface stops.
Once the last buffer is filled in, the QCI interface stops.
c) Capture global finite state machine schema
c) Capture global finite state machine schema
+----+ +---+ +----+
| DQ | | Q | | DQ | .. code-block:: none
| v | v | v
+-----------+ +------------------------+ +----+ +---+ +----+
| STOP | | Wait for capture start | | DQ | | Q | | DQ |
+-----------+ Q +------------------------+ | v | v | v
+-> | QCI: stop | ------------------> | QCI: run | <------------+ +-----------+ +------------------------+
| | DMA: stop | | DMA: stop | | | STOP | | Wait for capture start |
| +-----------+ +-----> +------------------------+ | +-----------+ Q +------------------------+
| / | | +-> | QCI: stop | ------------------> | QCI: run | <------------+
| / +---+ +----+ | | | | DMA: stop | | DMA: stop | |
|capture list empty / | Q | | DQ | | QCI Irq EOF | | +-----------+ +-----> +------------------------+ |
| / | v | v v | | / | |
| +--------------------+ +----------------------+ | | / +---+ +----+ | |
| | DMA hotlink missed | | Capture running | | |capture list empty / | Q | | DQ | | QCI Irq EOF |
| +--------------------+ +----------------------+ | | / | v | v v |
| | QCI: run | +-----> | QCI: run | <-+ | | +--------------------+ +----------------------+ |
| | DMA: stop | / | DMA: run | | | | | DMA hotlink missed | | Capture running | |
| +--------------------+ / +----------------------+ | Other | | +--------------------+ +----------------------+ |
| ^ /DMA still | | channels | | | QCI: run | +-----> | QCI: run | <-+ |
| | capture list / running | DMA Irq End | not | | | DMA: stop | / | DMA: run | | |
| | not empty / | | finished | | +--------------------+ / +----------------------+ | Other |
| | / v | yet | | ^ /DMA still | | channels |
| +----------------------+ +----------------------+ | | | | capture list / running | DMA Irq End | not |
| | Videobuf released | | Channel completed | | | | | not empty / | | finished |
| +----------------------+ +----------------------+ | | | | / v | yet |
+-- | QCI: run | | QCI: run | --+ | | +----------------------+ +----------------------+ | |
| DMA: run | | DMA: run | | | | Videobuf released | | Channel completed | | |
+----------------------+ +----------------------+ | | +----------------------+ +----------------------+ | |
^ / | | +-- | QCI: run | | QCI: run | --+ |
| no overrun / | overrun | | DMA: run | | DMA: run | |
| / v | +----------------------+ +----------------------+ |
+--------------------+ / +----------------------+ | ^ / | |
| Frame completed | / | Frame overran | | | no overrun / | overrun |
+--------------------+ <-----+ +----------------------+ restart frame | | / v |
| QCI: run | | QCI: stop | --------------+ +--------------------+ / +----------------------+ |
| DMA: run | | DMA: stop | | Frame completed | / | Frame overran | |
+--------------------+ +----------------------+ +--------------------+ <-----+ +----------------------+ restart frame |
| QCI: run | | QCI: stop | --------------+
Legend: - each box is a FSM state | DMA: run | | DMA: stop |
- each arrow is the condition to transition to another state +--------------------+ +----------------------+
- an arrow with a comment is a mandatory transition (no condition)
- arrow "Q" means : a buffer was enqueued Legend: - each box is a FSM state
- arrow "DQ" means : a buffer was dequeued - each arrow is the condition to transition to another state
- "QCI: stop" means the QCI interface is not enabled - an arrow with a comment is a mandatory transition (no condition)
- "DMA: stop" means all 3 DMA channels are stopped - arrow "Q" means : a buffer was enqueued
- "DMA: run" means at least 1 DMA channel is still running - arrow "DQ" means : a buffer was dequeued
- "QCI: stop" means the QCI interface is not enabled
- "DMA: stop" means all 3 DMA channels are stopped
- "DMA: run" means at least 1 DMA channel is still running
DMA usage DMA usage
--------- ---------
a) DMA flow
a) DMA flow
- first buffer queued for capture - first buffer queued for capture
Once a first buffer is queued for capture, the QCI is started, but data Once a first buffer is queued for capture, the QCI is started, but data
transfer is not started. On "End Of Frame" interrupt, the irq handler transfer is not started. On "End Of Frame" interrupt, the irq handler
...@@ -93,22 +100,27 @@ DMA usage ...@@ -93,22 +100,27 @@ DMA usage
- finishing the last videobuffer - finishing the last videobuffer
On the DMA irq of the last videobuffer, the QCI is stopped. On the DMA irq of the last videobuffer, the QCI is stopped.
b) DMA prepared buffer will have this structure b) DMA prepared buffer will have this structure
.. code-block:: none
+------------+-----+---------------+-----------------+ +------------+-----+---------------+-----------------+
| desc-sg[0] | ... | desc-sg[last] | finisher/linker | | desc-sg[0] | ... | desc-sg[last] | finisher/linker |
+------------+-----+---------------+-----------------+ +------------+-----+---------------+-----------------+
This structure is pointed by dma->sg_cpu. This structure is pointed by dma->sg_cpu.
The descriptors are used as follows : The descriptors are used as follows:
- desc-sg[i]: i-th descriptor, transferring the i-th sg
element to the video buffer scatter gather - desc-sg[i]: i-th descriptor, transferring the i-th sg
- finisher: has ddadr=DADDR_STOP, dcmd=ENDIRQEN element to the video buffer scatter gather
- linker: has ddadr= desc-sg[0] of next video buffer, dcmd=0 - finisher: has ddadr=DADDR_STOP, dcmd=ENDIRQEN
- linker: has ddadr= desc-sg[0] of next video buffer, dcmd=0
For the next schema, let's assume d0=desc-sg[0] .. dN=desc-sg[N],
"f" stands for finisher and "l" for linker.
A typical running chain is :
For the next schema, let's assume d0=desc-sg[0] .. dN=desc-sg[N], .. code-block:: none
"f" stands for finisher and "l" for linker.
A typical running chain is :
Videobuffer 1 Videobuffer 2 Videobuffer 1 Videobuffer 2
+---------+----+---+ +----+----+----+---+ +---------+----+---+ +----+----+----+---+
...@@ -117,7 +129,9 @@ DMA usage ...@@ -117,7 +129,9 @@ DMA usage
| | | |
+----+ +----+
After the chaining is finished, the chain looks like : After the chaining is finished, the chain looks like :
.. code-block:: none
Videobuffer 1 Videobuffer 2 Videobuffer 3 Videobuffer 1 Videobuffer 2 Videobuffer 3
+---------+----+---+ +----+----+----+---+ +----+----+----+---+ +---------+----+---+ +----+----+----+---+ +----+----+----+---+
...@@ -127,15 +141,18 @@ DMA usage ...@@ -127,15 +141,18 @@ DMA usage
+----+ +----+ +----+ +----+
new_link new_link
c) DMA hot chaining timeslice issue c) DMA hot chaining timeslice issue
As DMA chaining is done while DMA _is_ running, the linking may be done
while the DMA jumps from one Videobuffer to another. On the schema, that
would be a problem if the following sequence is encountered :
As DMA chaining is done while DMA _is_ running, the linking may be done - DMA chain is Videobuffer1 + Videobuffer2
while the DMA jumps from one Videobuffer to another. On the schema, that - pxa_videobuf_queue() is called to queue Videobuffer3
would be a problem if the following sequence is encountered : - DMA controller finishes Videobuffer2, and DMA stops
.. code-block:: none
- DMA chain is Videobuffer1 + Videobuffer2
- pxa_videobuf_queue() is called to queue Videobuffer3
- DMA controller finishes Videobuffer2, and DMA stops
=> =>
Videobuffer 1 Videobuffer 2 Videobuffer 1 Videobuffer 2
+---------+----+---+ +----+----+----+---+ +---------+----+---+ +----+----+----+---+
...@@ -144,11 +161,13 @@ DMA usage ...@@ -144,11 +161,13 @@ DMA usage
| | | | | |
+----+ +-- DMA DDADR loads DDADR_STOP +----+ +-- DMA DDADR loads DDADR_STOP
- pxa_dma_add_tail_buf() is called, the Videobuffer2 "finisher" is - pxa_dma_add_tail_buf() is called, the Videobuffer2 "finisher" is
replaced by a "linker" to Videobuffer3 (creation of new_link) replaced by a "linker" to Videobuffer3 (creation of new_link)
- pxa_videobuf_queue() finishes - pxa_videobuf_queue() finishes
- the DMA irq handler is called, which terminates Videobuffer2 - the DMA irq handler is called, which terminates Videobuffer2
- Videobuffer3 capture is not scheduled on DMA chain (as it stopped !!!) - Videobuffer3 capture is not scheduled on DMA chain (as it stopped !!!)
.. code-block:: none
Videobuffer 1 Videobuffer 2 Videobuffer 3 Videobuffer 1 Videobuffer 2 Videobuffer 3
+---------+----+---+ +----+----+----+---+ +----+----+----+---+ +---------+----+---+ +----+----+----+---+ +----+----+----+---+
...@@ -159,16 +178,15 @@ DMA usage ...@@ -159,16 +178,15 @@ DMA usage
new_link new_link
DMA DDADR still is DDADR_STOP DMA DDADR still is DDADR_STOP
- pxa_camera_check_link_miss() is called - pxa_camera_check_link_miss() is called
This checks if the DMA is finished and a buffer is still on the This checks if the DMA is finished and a buffer is still on the
pcdev->capture list. If that's the case, the capture will be restarted, pcdev->capture list. If that's the case, the capture will be restarted,
and Videobuffer3 is scheduled on DMA chain. and Videobuffer3 is scheduled on DMA chain.
- the DMA irq handler finishes - the DMA irq handler finishes
Note: if DMA stops just after pxa_camera_check_link_miss() reads DDADR() .. note::
If DMA stops just after pxa_camera_check_link_miss() reads DDADR()
value, we have the guarantee that the DMA irq handler will be called back value, we have the guarantee that the DMA irq handler will be called back
when the DMA will finish the buffer, and pxa_camera_check_link_miss() will when the DMA will finish the buffer, and pxa_camera_check_link_miss() will
be called again, to reschedule Videobuffer3. be called again, to reschedule Videobuffer3.
--
Author: Robert Jarzmik <robert.jarzmik@free.fr>
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