[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <0cd56538-ffca-4660-ad69-0447b0eb92b0@collabora.com>
Date: Wed, 7 Feb 2024 11:01:34 +0100
From: Benjamin Gaignard <benjamin.gaignard@...labora.com>
To: Devarsh Thakkar <devarsht@...com>, mchehab@...nel.org,
robh+dt@...nel.org, krzysztof.kozlowski+dt@...aro.org, conor+dt@...nel.org,
hverkuil-cisco@...all.nl, linux-media@...r.kernel.org,
devicetree@...r.kernel.org, linux-kernel@...r.kernel.org
Cc: laurent.pinchart@...asonboard.com, praneeth@...com, nm@...com,
vigneshr@...com, a-bhatia1@...com, j-luthra@...com, b-brnich@...com,
detheridge@...com, p-mantena@...com, vijayp@...com, andrzej.p@...labora.com,
nicolas@...fresne.ca
Subject: Re: [PATCH v4 3/3] media: imagination: Add E5010 JPEG Encoder driver
Le 07/02/2024 à 10:40, Devarsh Thakkar a écrit :
> Hi Benjamin,
>
> Thanks for the review.
>
> On 05/02/24 18:41, Benjamin Gaignard wrote:
>>
>> Le 05/02/2024 à 12:42, Devarsh Thakkar a écrit :
>>> This adds support for stateful V4L2 M2M based driver for Imagination
>>> E5010
>>> JPEG Encoder [1] which supports baseline encoding with two different
>>> quantization tables and compression ratio as demanded.
>>>
>>> Support for both contiguous and non-contiguous YUV420 and YUV422
>>> semiplanar
>>> formats is added along with alignment restrictions as required by the
>>> hardware.
>>>
>>> System and runtime PM hooks are added in the driver along with v4l2
>>> crop
>>> and selection API support.
>>>
>>> Minimum resolution supported is 64x64 and
>>> Maximum resolution supported is 8192x8192.
>>>
>>> All v4l2-compliance tests are passing [2] :
>>> v4l2-compliance -s -f -a -d /dev/video0 -e /dev/video1
>>>
>>> Total for e5010 device /dev/video0: 78, Succeeded: 78, Failed: 0,
>>> Warnings: 0
>>>
>>> NOTE: video1 here is VIVID test pattern generator
>>>
>>> Also tests [3] were run manually to verify below driver features:
>>> - Runtime Power Management
>>> - Multi-instance JPEG Encoding
>>> - DMABUF import, export support
>>> - NV12, NV21, NV16, NV61 video format support
>>> - Compression quality S_CTRL
>>>
>>> Existing V4L2 M2M based JPEG drivers namely s5p-jpeg, imx-jpeg and
>>> rcar_jpu
>>> were referred while making this.
>>>
>>> TODO:
>>> Add MMU and memory tiling support
>>>
>>> [1]: AM62A TRM (Section 7.6 is for JPEG Encoder) :
>>> Link: https://www.ti.com/lit/pdf/spruj16
>>>
>>> [2]: v4l2-compliance test :
>>> Link: https://gist.github.com/devarsht/867b1d646bca3f3877edb1f3638aae31
>>>
>>> [3]: E5010 JPEG Encoder Manual tests :
>>> Performance:
>>> Link: https://gist.github.com/devarsht/63d835093195181866ae9a85de880d3c
>>>
>>> Functionality:
>>> Link: https://gist.github.com/devarsht/253e485e86661a3051e711f97ec3c5ac
>>>
>>> Compression Quality:
>>> Link: https://gist.github.com/devarsht/e16d8b73c8107ac78cc77ad79fd6299f
>>>
>>> Multi Instance:
>>> Link: https://gist.github.com/devarsht/58af9ccf822963d2bf8f0c2f4481438a
>>>
>>> Co-developed-by: David Huang <d-huang@...com>
>>> Signed-off-by: David Huang <d-huang@...com>
>>> Signed-off-by: Devarsh Thakkar <devarsht@...com>
>>> ---
>>> V2:
>>> No change
>>>
>>> V3:
>>> - Correct license headers
>>> - Use more generic name core instead of jasper for base registers
>>> - Add Comment for forward declarations
>>> - Simplify quantization table calculations
>>> - Use v4l2_apply_frmsize_constraints for updating framesize and
>>> remove unrequired functions
>>> - Place TODO at top of file and in commit message too
>>> - Use dev_err_probe helper in probe function
>>> - Fix return value checking for failure scenarios in probe function
>>> - Use v4l2_err/info/warn helpers instead of dev_err/info/warn helpers
>>> - Fix unexpected indentation
>>> - Correct commit message
>>> - Remove dependency on ARCH_K3 as driver is not specific to that
>>>
>>> V4:
>>> - Fix issue with default params setting
>>> - Correct v4l2 error prints
>>> - Simplify register write functions with single statement return values
>>> - Remove unrequired error checks from get_queue()
>>> - Drop explicit device_caps setting as it is already taken care by v4l2
>>> core
>>> - Remove unrequired multiplanar checks and memset from s_fmt, g_fmt
>>> callback
>>> functions
>>> - Fix try_fmt callback to not update the queues
>>> - Remove unrequired contiguous format attribute from queue_init
>>> - Use dynamic allocation for video_device and remove unrequired
>>> assignments in probe()
>>> - Remove unrequired checks from queue_setup function
>>> - Return queued buffers back if start_streaming fails
>>> - Use ARRAY_SIZE in place of hard-coding
>>> - Use huffman and quantization tables from reference header file
>>>
>>> Link to previous commit:
>>> https://lore.kernel.org/all/20230816152210.4080779-3-devarsht@ti.com/
>>> ---
>>> MAINTAINERS | 2 +
>>> drivers/media/platform/Kconfig | 1 +
>>> drivers/media/platform/Makefile | 1 +
>>> drivers/media/platform/imagination/Kconfig | 12 +
>>> drivers/media/platform/imagination/Makefile | 3 +
>>> .../platform/imagination/e5010-core-regs.h | 585 +++++++
>>> .../platform/imagination/e5010-jpeg-enc-hw.c | 267 +++
>>> .../platform/imagination/e5010-jpeg-enc-hw.h | 42 +
>>> .../platform/imagination/e5010-jpeg-enc.c | 1552
>>> +++++++++++++++++
>>> .../platform/imagination/e5010-jpeg-enc.h | 169 ++
>>> .../platform/imagination/e5010-mmu-regs.h | 311 ++++
>>> 11 files changed, 2945 insertions(+)
>>> create mode 100644 drivers/media/platform/imagination/Kconfig
>>> create mode 100644 drivers/media/platform/imagination/Makefile
>>> create mode 100644
>>> drivers/media/platform/imagination/e5010-core-regs.h
>>> create mode 100644
>>> drivers/media/platform/imagination/e5010-jpeg-enc-hw.c
>>> create mode 100644
>>> drivers/media/platform/imagination/e5010-jpeg-enc-hw.h
>>> create mode 100644
>>> drivers/media/platform/imagination/e5010-jpeg-enc.c
>>> create mode 100644
>>> drivers/media/platform/imagination/e5010-jpeg-enc.h
>>> create mode 100644
>>> drivers/media/platform/imagination/e5010-mmu-regs.h
>>>
>>> diff --git a/MAINTAINERS b/MAINTAINERS
>>> index d0f8c46d3ce9..34353d913650 100644
>>> --- a/MAINTAINERS
>>> +++ b/MAINTAINERS
>>> @@ -10463,8 +10463,10 @@ F: drivers/auxdisplay/img-ascii-lcd.c
>>> IMGTEC JPEG ENCODER DRIVER
>>> M: Devarsh Thakkar <devarsht@...com>
>>> +L: linux-media@...r.kernel.org
>>> S: Supported
>>> F: Documentation/devicetree/bindings/media/img,e5010-jpeg-enc.yaml
>>> +F: drivers/media/platform/imagination/e5010*
>>> IMGTEC IR DECODER DRIVER
>>> S: Orphan
>>> diff --git a/drivers/media/platform/Kconfig
>>> b/drivers/media/platform/Kconfig
>>> index 91e54215de3a..fe5621ef5a7b 100644
>>> --- a/drivers/media/platform/Kconfig
>>> +++ b/drivers/media/platform/Kconfig
>>> @@ -69,6 +69,7 @@ source "drivers/media/platform/aspeed/Kconfig"
>>> source "drivers/media/platform/atmel/Kconfig"
>>> source "drivers/media/platform/cadence/Kconfig"
>>> source "drivers/media/platform/chips-media/Kconfig"
>>> +source "drivers/media/platform/imagination/Kconfig"
>>> source "drivers/media/platform/intel/Kconfig"
>>> source "drivers/media/platform/marvell/Kconfig"
>>> source "drivers/media/platform/mediatek/Kconfig"
>>> diff --git a/drivers/media/platform/Makefile
>>> b/drivers/media/platform/Makefile
>>> index 3296ec1ebe16..cb9f3afb837c 100644
>>> --- a/drivers/media/platform/Makefile
>>> +++ b/drivers/media/platform/Makefile
>>> @@ -12,6 +12,7 @@ obj-y += aspeed/
>>> obj-y += atmel/
>>> obj-y += cadence/
>>> obj-y += chips-media/
>>> +obj-y += imagination/
>>> obj-y += intel/
>>> obj-y += marvell/
>>> obj-y += mediatek/
>>> diff --git a/drivers/media/platform/imagination/Kconfig
>>> b/drivers/media/platform/imagination/Kconfig
>>> new file mode 100644
>>> index 000000000000..d8d79266ad5d
>>> --- /dev/null
>>> +++ b/drivers/media/platform/imagination/Kconfig
>>> @@ -0,0 +1,12 @@
>>> +# SPDX-License-Identifier: GPL-2.0
>>> +config VIDEO_E5010_JPEG_ENC
>>> + tristate "Imagination E5010 JPEG Encoder Driver"
>>> + depends on VIDEO_DEV
>>> + select VIDEOBUF2_DMA_CONTIG
>>> + select VIDEOBUF2_VMALLOC
>>> + select V4L2_MEM2MEM_DEV
>>> + help
>>> + This is a video4linux2 M2M driver for Imagination E5010 JPEG
>>> encoder,
>>> + which supports JPEG and MJPEG baseline encoding of YUV422 and
>>> YUV420
>>> + semiplanar video formats, with resolution ranging from 64x64
>>> to 8K x 8K
>>> + pixels. The module will be named as e5010_jpeg_enc.
>>> diff --git a/drivers/media/platform/imagination/Makefile
>>> b/drivers/media/platform/imagination/Makefile
>>> new file mode 100644
>>> index 000000000000..d45b85b88575
>>> --- /dev/null
>>> +++ b/drivers/media/platform/imagination/Makefile
>>> @@ -0,0 +1,3 @@
>>> +# SPDX-License-Identifier: GPL-2.0
>>> +e5010_jpeg_enc-objs := e5010-jpeg-enc-hw.o e5010-jpeg-enc.o
>>> +obj-$(CONFIG_VIDEO_E5010_JPEG_ENC) += e5010_jpeg_enc.o
>>> diff --git a/drivers/media/platform/imagination/e5010-core-regs.h
>>> b/drivers/media/platform/imagination/e5010-core-regs.h
>>> new file mode 100644
>>> index 000000000000..aaec498fe83f
>>> --- /dev/null
>>> +++ b/drivers/media/platform/imagination/e5010-core-regs.h
>>> @@ -0,0 +1,585 @@
>>> +/* SPDX-License-Identifier: GPL-2.0 */
>>> +/*
>>> + * Imagination E5010 JPEG Encoder driver.
>>> + *
>>> + * Copyright (C) 2023 Texas Instruments Incorporated -
>>> https://www.ti.com/
>>> + *
>>> + * Author: David Huang <d-huang@...com>
>>> + * Author: Devarsh Thakkar <devarsht@...com>
>>> + */
>>> +
>>> +#ifndef _E5010_CORE_REGS_H
>>> +#define _E5010_CORE_REGS_H
>>> +
>>> +#define JASPER_CORE_ID_OFFSET (0x0000)
>>> +#define JASPER_CORE_ID_CR_GROUP_ID_MASK (0xFF000000)
>>> +#define JASPER_CORE_ID_CR_GROUP_ID_SHIFT (24)
>>> +#define JASPER_CORE_ID_CR_CORE_ID_MASK (0x00FF0000)
>>> +#define JASPER_CORE_ID_CR_CORE_ID_SHIFT (16)
>>> +#define JASPER_CORE_ID_CR_UNIQUE_NUM_MASK (0x0000FFF8)
>>> +#define JASPER_CORE_ID_CR_UNIQUE_NUM_SHIFT (3)
>>> +#define JASPER_CORE_ID_CR_PELS_PER_CYCLE_MASK (0x00000007)
>>> +#define JASPER_CORE_ID_CR_PELS_PER_CYCLE_SHIFT (0)
>>> +
>>> +#define JASPER_CORE_REV_OFFSET (0x0004)
>>> +#define JASPER_CORE_REV_CR_JASPER_DESIGNER_MASK (0xFF000000)
>>> +#define JASPER_CORE_REV_CR_JASPER_DESIGNER_SHIFT (24)
>>> +#define JASPER_CORE_REV_CR_JASPER_MAJOR_REV_MASK (0x00FF0000)
>>> +#define JASPER_CORE_REV_CR_JASPER_MAJOR_REV_SHIFT (16)
>>> +#define JASPER_CORE_REV_CR_JASPER_MINOR_REV_MASK (0x0000FF00)
>>> +#define JASPER_CORE_REV_CR_JASPER_MINOR_REV_SHIFT (8)
>>> +#define JASPER_CORE_REV_CR_JASPER_MAINT_REV_MASK (0x000000FF)
>>> +#define JASPER_CORE_REV_CR_JASPER_MAINT_REV_SHIFT (0)
>>> +
>>> +#define JASPER_INTERRUPT_MASK_OFFSET (0x0008)
>>> +#define JASPER_INTERRUPT_MASK_CR_OUTPUT_ADDRESS_ERROR_ENABLE_MASK
>>> (0x00000002)
>>> +#define JASPER_INTERRUPT_MASK_CR_OUTPUT_ADDRESS_ERROR_ENABLE_SHIFT (1)
>>> +#define JASPER_INTERRUPT_MASK_CR_PICTURE_DONE_ENABLE_MASK (0x00000001)
>>> +#define JASPER_INTERRUPT_MASK_CR_PICTURE_DONE_ENABLE_SHIFT (0)
>>> +
>>> +#define JASPER_INTERRUPT_STATUS_OFFSET (0x000C)
>>> +#define JASPER_INTERRUPT_STATUS_CR_OUTPUT_ADDRESS_ERROR_IRQ_MASK
>>> (0x00000002)
>>> +#define JASPER_INTERRUPT_STATUS_CR_OUTPUT_ADDRESS_ERROR_IRQ_SHIFT (1)
>>> +#define JASPER_INTERRUPT_STATUS_CR_PICTURE_DONE_IRQ_MASK (0x00000001)
>>> +#define JASPER_INTERRUPT_STATUS_CR_PICTURE_DONE_IRQ_SHIFT (0)
>>> +
>>> +#define JASPER_INTERRUPT_CLEAR_OFFSET (0x0010)
>>> +#define JASPER_INTERRUPT_CLEAR_CR_OUTPUT_ERROR_CLEAR_MASK (0x00000002)
>>> +#define JASPER_INTERRUPT_CLEAR_CR_OUTPUT_ERROR_CLEAR_SHIFT (1)
>>> +#define JASPER_INTERRUPT_CLEAR_CR_PICTURE_DONE_CLEAR_MASK (0x00000001)
>>> +#define JASPER_INTERRUPT_CLEAR_CR_PICTURE_DONE_CLEAR_SHIFT (0)
>>> +
>>> +#define JASPER_CLK_CONTROL_OFFSET (0x0014)
>>> +#define JASPER_CLK_CONTROL_CR_JASPER_AUTO_CLKG_ENABLE_MASK
>>> (0x00000002)
>>> +#define JASPER_CLK_CONTROL_CR_JASPER_AUTO_CLKG_ENABLE_SHIFT (1)
>>> +#define JASPER_CLK_CONTROL_CR_JASPER_MAN_CLKG_ENABLE_MASK (0x00000001)
>>> +#define JASPER_CLK_CONTROL_CR_JASPER_MAN_CLKG_ENABLE_SHIFT (0)
>>> +
>>> +#define JASPER_CLK_STATUS_OFFSET (0x0018)
>>> +#define JASPER_CLK_STATUS_CR_JASPER_CLKG_STATUS_MASK (0x00000001)
>>> +#define JASPER_CLK_STATUS_CR_JASPER_CLKG_STATUS_SHIFT (0)
>>> +
>>> +#define JASPER_RESET_OFFSET (0x001C)
>>> +#define JASPER_RESET_CR_SYS_RESET_MASK (0x00000002)
>>> +#define JASPER_RESET_CR_SYS_RESET_SHIFT (1)
>>> +#define JASPER_RESET_CR_CORE_RESET_MASK (0x00000001)
>>> +#define JASPER_RESET_CR_CORE_RESET_SHIFT (0)
>>> +
>>> +#define JASPER_CORE_CTRL_OFFSET (0x0020)
>>> +#define JASPER_CORE_CTRL_CR_JASPER_ENCODE_START_MASK (0x00000001)
>>> +#define JASPER_CORE_CTRL_CR_JASPER_ENCODE_START_SHIFT (0)
>>> +
>>> +#define JASPER_STATUS_OFFSET (0x0024)
>>> +#define JASPER_STATUS_CR_FLUSH_MODE_MASK (0x00000002)
>>> +#define JASPER_STATUS_CR_FLUSH_MODE_SHIFT (1)
>>> +#define JASPER_STATUS_CR_JASPER_BUSY_MASK (0x00000001)
>>> +#define JASPER_STATUS_CR_JASPER_BUSY_SHIFT (0)
>>> +
>>> +#define JASPER_CRC_CLEAR_OFFSET (0x0028)
>>> +#define JASPER_CRC_CLEAR_CR_FRONT_END_CRC_CLEAR_MASK (0x00000001)
>>> +#define JASPER_CRC_CLEAR_CR_FRONT_END_CRC_CLEAR_SHIFT (0)
>>> +#define JASPER_CRC_CLEAR_CR_DCT_CRC_CLEAR_MASK (0x00000002)
>>> +#define JASPER_CRC_CLEAR_CR_DCT_CRC_CLEAR_SHIFT (1)
>>> +#define JASPER_CRC_CLEAR_CR_ZZ_CRC_CLEAR_MASK (0x00000004)
>>> +#define JASPER_CRC_CLEAR_CR_ZZ_CRC_CLEAR_SHIFT (2)
>>> +#define JASPER_CRC_CLEAR_CR_QUANT_CRC_CLEAR_MASK (0x00000008)
>>> +#define JASPER_CRC_CLEAR_CR_QUANT_CRC_CLEAR_SHIFT (3)
>>> +#define JASPER_CRC_CLEAR_CR_ENTROPY_ENCODER_CRC_CLEAR_MASK
>>> (0x00000010)
>>> +#define JASPER_CRC_CLEAR_CR_ENTROPY_ENCODER_CRC_CLEAR_SHIFT (4)
>>> +#define JASPER_CRC_CLEAR_CR_PACKING_BUFFER_CRC_CLEAR_MASK (0x00000020)
>>> +#define JASPER_CRC_CLEAR_CR_PACKING_BUFFER_CRC_CLEAR_SHIFT (5)
>>> +
>>> +#define JASPER_INPUT_CTRL0_OFFSET (0x002C)
>>> +#define JASPER_INPUT_CTRL0_CR_INPUT_CHROMA_ORDER_MASK (0x01000000)
>>> +#define JASPER_INPUT_CTRL0_CR_INPUT_CHROMA_ORDER_SHIFT (24)
>>> +#define JASPER_INPUT_CTRL0_CR_INPUT_SUBSAMPLING_MASK (0x00030000)
>>> +#define JASPER_INPUT_CTRL0_CR_INPUT_SUBSAMPLING_SHIFT (16)
>>> +#define JASPER_INPUT_CTRL0_CR_INPUT_SOURCE_MASK (0x00000004)
>>> +#define JASPER_INPUT_CTRL0_CR_INPUT_SOURCE_SHIFT (2)
>>> +
>>> +#define JASPER_INPUT_CTRL1_OFFSET (0x0030)
>>> +#define JASPER_INPUT_CTRL1_CR_INPUT_LUMA_STRIDE_MASK (0x1FC00000)
>>> +#define JASPER_INPUT_CTRL1_CR_INPUT_LUMA_STRIDE_SHIFT (22)
>>> +#define JASPER_INPUT_CTRL1_CR_INPUT_CHROMA_STRIDE_MASK (0x00001FC0)
>>> +#define JASPER_INPUT_CTRL1_CR_INPUT_CHROMA_STRIDE_SHIFT (6)
>>> +
>>> +#define JASPER_MMU_CTRL_OFFSET (0x0034)
>>> +#define JASPER_MMU_CTRL_CR_JASPER_TILING_SCHEME_MASK (0x00000002)
>>> +#define JASPER_MMU_CTRL_CR_JASPER_TILING_SCHEME_SHIFT (1)
>>> +#define JASPER_MMU_CTRL_CR_JASPER_TILING_ENABLE_MASK (0x00000001)
>>> +#define JASPER_MMU_CTRL_CR_JASPER_TILING_ENABLE_SHIFT (0)
>>> +
>>> +#define JASPER_IMAGE_SIZE_OFFSET (0x0038)
>>> +#define JASPER_IMAGE_SIZE_CR_IMAGE_VERTICAL_SIZE_MASK (0x1FFF0000)
>>> +#define JASPER_IMAGE_SIZE_CR_IMAGE_VERTICAL_SIZE_SHIFT (16)
>>> +#define JASPER_IMAGE_SIZE_CR_IMAGE_HORIZONTAL_SIZE_MASK (0x00001FFF)
>>> +#define JASPER_IMAGE_SIZE_CR_IMAGE_HORIZONTAL_SIZE_SHIFT (0)
>>> +
>>> +#define INPUT_LUMA_BASE_OFFSET (0x003C)
>>> +#define INPUT_LUMA_BASE_CR_INPUT_LUMA_BASE_MASK (0xFFFFFFC0)
>>> +#define INPUT_LUMA_BASE_CR_INPUT_LUMA_BASE_SHIFT (6)
>>> +
>>> +#define INPUT_CHROMA_BASE_OFFSET (0x0040)
>>> +#define INPUT_CHROMA_BASE_CR_INPUT_CHROMA_BASE_MASK (0xFFFFFFC0)
>>> +#define INPUT_CHROMA_BASE_CR_INPUT_CHROMA_BASE_SHIFT (6)
>>> +
>>> +#define JASPER_OUTPUT_BASE_OFFSET (0x0044)
>>> +#define JASPER_OUTPUT_BASE_CR_OUTPUT_BASE_MASK (0xFFFFFFFF)
>>> +#define JASPER_OUTPUT_BASE_CR_OUTPUT_BASE_SHIFT (0)
>>> +
>>> +#define JASPER_OUTPUT_SIZE_OFFSET (0x0048)
>>> +#define JASPER_OUTPUT_SIZE_CR_OUTPUT_SIZE_MASK (0xFFFFFFFF)
>>> +#define JASPER_OUTPUT_SIZE_CR_OUTPUT_SIZE_SHIFT (0)
>>> +#define JASPER_OUTPUT_MAX_SIZE_OFFSET (0x004C)
>>> +#define JASPER_OUTPUT_MAX_SIZE_CR_OUTPUT_MAX_SIZE_MASK (0xFFFFFFFF)
>>> +#define JASPER_OUTPUT_MAX_SIZE_CR_OUTPUT_MAX_SIZE_SHIFT (0)
>>> +
>>> +#define JASPER_LUMA_QUANTIZATION_TABLE0_OFFSET (0x0050)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE0_CR_LUMA_QUANTIZATION_TABLE_03_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE0_CR_LUMA_QUANTIZATION_TABLE_03_SHIFT
>>> (24)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE0_CR_LUMA_QUANTIZATION_TABLE_02_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE0_CR_LUMA_QUANTIZATION_TABLE_02_SHIFT
>>> (16)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE0_CR_LUMA_QUANTIZATION_TABLE_01_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE0_CR_LUMA_QUANTIZATION_TABLE_01_SHIFT (8)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE0_CR_LUMA_QUANTIZATION_TABLE_00_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE0_CR_LUMA_QUANTIZATION_TABLE_00_SHIFT (0)
>>> +#define JASPER_LUMA_QUANTIZATION_TABLE1_OFFSET (0x0054)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE1_CR_LUMA_QUANTIZATION_TABLE_07_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE1_CR_LUMA_QUANTIZATION_TABLE_07_SHIFT
>>> (24)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE1_CR_LUMA_QUANTIZATION_TABLE_06_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE1_CR_LUMA_QUANTIZATION_TABLE_06_SHIFT
>>> (16)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE1_CR_LUMA_QUANTIZATION_TABLE_05_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE1_CR_LUMA_QUANTIZATION_TABLE_05_SHIFT (8)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE1_CR_LUMA_QUANTIZATION_TABLE_04_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE1_CR_LUMA_QUANTIZATION_TABLE_04_SHIFT (0)
>>> +#define JASPER_LUMA_QUANTIZATION_TABLE2_OFFSET (0x0058)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE2_CR_LUMA_QUANTIZATION_TABLE_13_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE2_CR_LUMA_QUANTIZATION_TABLE_13_SHIFT
>>> (24)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE2_CR_LUMA_QUANTIZATION_TABLE_12_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE2_CR_LUMA_QUANTIZATION_TABLE_12_SHIFT
>>> (16)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE2_CR_LUMA_QUANTIZATION_TABLE_11_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE2_CR_LUMA_QUANTIZATION_TABLE_11_SHIFT (8)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE2_CR_LUMA_QUANTIZATION_TABLE_10_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE2_CR_LUMA_QUANTIZATION_TABLE_10_SHIFT (0)
>>> +#define JASPER_LUMA_QUANTIZATION_TABLE3_OFFSET (0x005C)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE3_CR_LUMA_QUANTIZATION_TABLE_17_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE3_CR_LUMA_QUANTIZATION_TABLE_17_SHIFT
>>> (24)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE3_CR_LUMA_QUANTIZATION_TABLE_16_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE3_CR_LUMA_QUANTIZATION_TABLE_16_SHIFT
>>> (16)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE3_CR_LUMA_QUANTIZATION_TABLE_15_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE3_CR_LUMA_QUANTIZATION_TABLE_15_SHIFT (8)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE3_CR_LUMA_QUANTIZATION_TABLE_14_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE3_CR_LUMA_QUANTIZATION_TABLE_14_SHIFT (0)
>>> +#define JASPER_LUMA_QUANTIZATION_TABLE4_OFFSET (0x0060)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE4_CR_LUMA_QUANTIZATION_TABLE_21_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE4_CR_LUMA_QUANTIZATION_TABLE_21_SHIFT (8)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE4_CR_LUMA_QUANTIZATION_TABLE_20_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE4_CR_LUMA_QUANTIZATION_TABLE_20_SHIFT (0)
>>> +
>>> +#define JASPER_LUMA_QUANTIZATION_TABLE5_OFFSET (0x0064)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE5_CR_LUMA_QUANTIZATION_TABLE_27_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE5_CR_LUMA_QUANTIZATION_TABLE_27_SHIFT
>>> (24)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE5_CR_LUMA_QUANTIZATION_TABLE_26_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE5_CR_LUMA_QUANTIZATION_TABLE_26_SHIFT
>>> (16)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE5_CR_LUMA_QUANTIZATION_TABLE_25_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE5_CR_LUMA_QUANTIZATION_TABLE_25_SHIFT (8)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE5_CR_LUMA_QUANTIZATION_TABLE_24_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE5_CR_LUMA_QUANTIZATION_TABLE_24_SHIFT (0)
>>> +
>>> +#define JASPER_LUMA_QUANTIZATION_TABLE6_OFFSET (0x0068)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE6_CR_LUMA_QUANTIZATION_TABLE_33_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE6_CR_LUMA_QUANTIZATION_TABLE_33_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE6_CR_LUMA_QUANTIZATION_TABLE_32_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE6_CR_LUMA_QUANTIZATION_TABLE_32_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE6_CR_LUMA_QUANTIZATION_TABLE_31_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE6_CR_LUMA_QUANTIZATION_TABLE_31_SHIFT (8)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE6_CR_LUMA_QUANTIZATION_TABLE_30_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE6_CR_LUMA_QUANTIZATION_TABLE_30_SHIFT (0)
>>> +
>>> +#define JASPER_LUMA_QUANTIZATION_TABLE7_OFFSET (0x006C)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE7_CR_LUMA_QUANTIZATION_TABLE_37_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE7_CR_LUMA_QUANTIZATION_TABLE_37_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE7_CR_LUMA_QUANTIZATION_TABLE_36_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE7_CR_LUMA_QUANTIZATION_TABLE_36_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE7_CR_LUMA_QUANTIZATION_TABLE_35_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE7_CR_LUMA_QUANTIZATION_TABLE_35_SHIFT (8)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE7_CR_LUMA_QUANTIZATION_TABLE_34_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE7_CR_LUMA_QUANTIZATION_TABLE_34_SHIFT (0)
>>> +
>>> +#define JASPER_LUMA_QUANTIZATION_TABLE8_OFFSET (0x0070)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE8_CR_LUMA_QUANTIZATION_TABLE_43_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE8_CR_LUMA_QUANTIZATION_TABLE_43_SHIFT
>>> (24)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE8_CR_LUMA_QUANTIZATION_TABLE_42_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE8_CR_LUMA_QUANTIZATION_TABLE_42_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE8_CR_LUMA_QUANTIZATION_TABLE_41_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE8_CR_LUMA_QUANTIZATION_TABLE_41_SHIFT (8)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE8_CR_LUMA_QUANTIZATION_TABLE_40_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE8_CR_LUMA_QUANTIZATION_TABLE_40_SHIFT (0)
>>> +
>>> +#define JASPER_LUMA_QUANTIZATION_TABLE9_OFFSET (0x0074)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE9_CR_LUMA_QUANTIZATION_TABLE_47_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE9_CR_LUMA_QUANTIZATION_TABLE_47_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE9_CR_LUMA_QUANTIZATION_TABLE_46_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE9_CR_LUMA_QUANTIZATION_TABLE_46_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE9_CR_LUMA_QUANTIZATION_TABLE_45_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE9_CR_LUMA_QUANTIZATION_TABLE_45_SHIFT (8)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE9_CR_LUMA_QUANTIZATION_TABLE_44_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE9_CR_LUMA_QUANTIZATION_TABLE_44_SHIFT (0)
>>> +
>>> +#define JASPER_LUMA_QUANTIZATION_TABLE10_OFFSET (0x0078)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE10_CR_LUMA_QUANTIZATION_TABLE_53_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE10_CR_LUMA_QUANTIZATION_TABLE_53_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE10_CR_LUMA_QUANTIZATION_TABLE_52_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE10_CR_LUMA_QUANTIZATION_TABLE_52_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE10_CR_LUMA_QUANTIZATION_TABLE_51_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE10_CR_LUMA_QUANTIZATION_TABLE_51_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE10_CR_LUMA_QUANTIZATION_TABLE_50_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE10_CR_LUMA_QUANTIZATION_TABLE_50_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_LUMA_QUANTIZATION_TABLE11_OFFSET (0x007C)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE11_CR_LUMA_QUANTIZATION_TABLE_57_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE11_CR_LUMA_QUANTIZATION_TABLE_57_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE11_CR_LUMA_QUANTIZATION_TABLE_56_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE11_CR_LUMA_QUANTIZATION_TABLE_56_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE11_CR_LUMA_QUANTIZATION_TABLE_55_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE11_CR_LUMA_QUANTIZATION_TABLE_55_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE11_CR_LUMA_QUANTIZATION_TABLE_54_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE11_CR_LUMA_QUANTIZATION_TABLE_54_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_LUMA_QUANTIZATION_TABLE12_OFFSET (0x0080)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE12_CR_LUMA_QUANTIZATION_TABLE_63_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE12_CR_LUMA_QUANTIZATION_TABLE_63_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE12_CR_LUMA_QUANTIZATION_TABLE_62_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE12_CR_LUMA_QUANTIZATION_TABLE_62_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE12_CR_LUMA_QUANTIZATION_TABLE_61_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE12_CR_LUMA_QUANTIZATION_TABLE_61_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE12_CR_LUMA_QUANTIZATION_TABLE_60_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE12_CR_LUMA_QUANTIZATION_TABLE_60_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_LUMA_QUANTIZATION_TABLE13_OFFSET (0x0084)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE13_CR_LUMA_QUANTIZATION_TABLE_67_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE13_CR_LUMA_QUANTIZATION_TABLE_67_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE13_CR_LUMA_QUANTIZATION_TABLE_66_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE13_CR_LUMA_QUANTIZATION_TABLE_66_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE13_CR_LUMA_QUANTIZATION_TABLE_65_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE13_CR_LUMA_QUANTIZATION_TABLE_65_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE13_CR_LUMA_QUANTIZATION_TABLE_64_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE13_CR_LUMA_QUANTIZATION_TABLE_64_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_LUMA_QUANTIZATION_TABLE14_OFFSET (0x0088)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE14_CR_LUMA_QUANTIZATION_TABLE_73_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE14_CR_LUMA_QUANTIZATION_TABLE_73_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE14_CR_LUMA_QUANTIZATION_TABLE_72_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE14_CR_LUMA_QUANTIZATION_TABLE_72_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE14_CR_LUMA_QUANTIZATION_TABLE_71_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE14_CR_LUMA_QUANTIZATION_TABLE_71_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE14_CR_LUMA_QUANTIZATION_TABLE_70_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE14_CR_LUMA_QUANTIZATION_TABLE_70_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_LUMA_QUANTIZATION_TABLE15_OFFSET (0x008C)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE15_CR_LUMA_QUANTIZATION_TABLE_77_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE15_CR_LUMA_QUANTIZATION_TABLE_77_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE15_CR_LUMA_QUANTIZATION_TABLE_76_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE15_CR_LUMA_QUANTIZATION_TABLE_76_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE15_CR_LUMA_QUANTIZATION_TABLE_75_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE15_CR_LUMA_QUANTIZATION_TABLE_75_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE15_CR_LUMA_QUANTIZATION_TABLE_74_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_LUMA_QUANTIZATION_TABLE15_CR_LUMA_QUANTIZATION_TABLE_74_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_CHROMA_QUANTIZATION_TABLE0_OFFSET (0x0090)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE0_CR_CHROMA_QUANTIZATION_TABLE_03_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE0_CR_CHROMA_QUANTIZATION_TABLE_03_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE0_CR_CHROMA_QUANTIZATION_TABLE_02_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE0_CR_CHROMA_QUANTIZATION_TABLE_02_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE0_CR_CHROMA_QUANTIZATION_TABLE_01_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE0_CR_CHROMA_QUANTIZATION_TABLE_01_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE0_CR_CHROMA_QUANTIZATION_TABLE_00_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE0_CR_CHROMA_QUANTIZATION_TABLE_00_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_CHROMA_QUANTIZATION_TABLE1_OFFSET (0x0094)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE1_CR_CHROMA_QUANTIZATION_TABLE_07_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE1_CR_CHROMA_QUANTIZATION_TABLE_07_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE1_CR_CHROMA_QUANTIZATION_TABLE_06_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE1_CR_CHROMA_QUANTIZATION_TABLE_06_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE1_CR_CHROMA_QUANTIZATION_TABLE_05_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE1_CR_CHROMA_QUANTIZATION_TABLE_05_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE1_CR_CHROMA_QUANTIZATION_TABLE_04_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE1_CR_CHROMA_QUANTIZATION_TABLE_04_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_CHROMA_QUANTIZATION_TABLE2_OFFSET (0x0098)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE2_CR_CHROMA_QUANTIZATION_TABLE_13_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE2_CR_CHROMA_QUANTIZATION_TABLE_13_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE2_CR_CHROMA_QUANTIZATION_TABLE_12_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE2_CR_CHROMA_QUANTIZATION_TABLE_12_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE2_CR_CHROMA_QUANTIZATION_TABLE_11_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE2_CR_CHROMA_QUANTIZATION_TABLE_11_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE2_CR_CHROMA_QUANTIZATION_TABLE_10_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE2_CR_CHROMA_QUANTIZATION_TABLE_10_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_CHROMA_QUANTIZATION_TABLE3_OFFSET (0x009C)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE3_CR_CHROMA_QUANTIZATION_TABLE_17_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE3_CR_CHROMA_QUANTIZATION_TABLE_17_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE3_CR_CHROMA_QUANTIZATION_TABLE_16_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE3_CR_CHROMA_QUANTIZATION_TABLE_16_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE3_CR_CHROMA_QUANTIZATION_TABLE_15_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE3_CR_CHROMA_QUANTIZATION_TABLE_15_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE3_CR_CHROMA_QUANTIZATION_TABLE_14_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE3_CR_CHROMA_QUANTIZATION_TABLE_14_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_CHROMA_QUANTIZATION_TABLE4_OFFSET (0x00A0)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE4_CR_CHROMA_QUANTIZATION_TABLE_23_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE4_CR_CHROMA_QUANTIZATION_TABLE_23_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE4_CR_CHROMA_QUANTIZATION_TABLE_22_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE4_CR_CHROMA_QUANTIZATION_TABLE_22_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE4_CR_CHROMA_QUANTIZATION_TABLE_21_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE4_CR_CHROMA_QUANTIZATION_TABLE_21_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE4_CR_CHROMA_QUANTIZATION_TABLE_20_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE4_CR_CHROMA_QUANTIZATION_TABLE_20_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_CHROMA_QUANTIZATION_TABLE5_OFFSET (0x00A4)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE5_CR_CHROMA_QUANTIZATION_TABLE_27_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE5_CR_CHROMA_QUANTIZATION_TABLE_27_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE5_CR_CHROMA_QUANTIZATION_TABLE_26_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE5_CR_CHROMA_QUANTIZATION_TABLE_26_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE5_CR_CHROMA_QUANTIZATION_TABLE_25_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE5_CR_CHROMA_QUANTIZATION_TABLE_25_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE5_CR_CHROMA_QUANTIZATION_TABLE_24_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE5_CR_CHROMA_QUANTIZATION_TABLE_24_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_CHROMA_QUANTIZATION_TABLE6_OFFSET (0x00A8)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE6_CR_CHROMA_QUANTIZATION_TABLE_33_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE6_CR_CHROMA_QUANTIZATION_TABLE_33_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE6_CR_CHROMA_QUANTIZATION_TABLE_32_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE6_CR_CHROMA_QUANTIZATION_TABLE_32_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE6_CR_CHROMA_QUANTIZATION_TABLE_31_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE6_CR_CHROMA_QUANTIZATION_TABLE_31_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE6_CR_CHROMA_QUANTIZATION_TABLE_30_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE6_CR_CHROMA_QUANTIZATION_TABLE_30_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_CHROMA_QUANTIZATION_TABLE7_OFFSET (0x00AC)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE7_CR_CHROMA_QUANTIZATION_TABLE_37_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE7_CR_CHROMA_QUANTIZATION_TABLE_37_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE7_CR_CHROMA_QUANTIZATION_TABLE_36_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE7_CR_CHROMA_QUANTIZATION_TABLE_36_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE7_CR_CHROMA_QUANTIZATION_TABLE_35_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE7_CR_CHROMA_QUANTIZATION_TABLE_35_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE7_CR_CHROMA_QUANTIZATION_TABLE_34_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE7_CR_CHROMA_QUANTIZATION_TABLE_34_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_CHROMA_QUANTIZATION_TABLE8_OFFSET (0x00B0)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE8_CR_CHROMA_QUANTIZATION_TABLE_43_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE8_CR_CHROMA_QUANTIZATION_TABLE_43_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE8_CR_CHROMA_QUANTIZATION_TABLE_42_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE8_CR_CHROMA_QUANTIZATION_TABLE_42_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE8_CR_CHROMA_QUANTIZATION_TABLE_41_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE8_CR_CHROMA_QUANTIZATION_TABLE_41_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE8_CR_CHROMA_QUANTIZATION_TABLE_40_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE8_CR_CHROMA_QUANTIZATION_TABLE_40_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_CHROMA_QUANTIZATION_TABLE9_OFFSET (0x00B4)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE9_CR_CHROMA_QUANTIZATION_TABLE_47_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE9_CR_CHROMA_QUANTIZATION_TABLE_47_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE9_CR_CHROMA_QUANTIZATION_TABLE_46_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE9_CR_CHROMA_QUANTIZATION_TABLE_46_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE9_CR_CHROMA_QUANTIZATION_TABLE_45_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE9_CR_CHROMA_QUANTIZATION_TABLE_45_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE9_CR_CHROMA_QUANTIZATION_TABLE_44_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE9_CR_CHROMA_QUANTIZATION_TABLE_44_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_CHROMA_QUANTIZATION_TABLE10_OFFSET (0x00B8)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE10_CR_CHROMA_QUANTIZATION_TABLE_53_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE10_CR_CHROMA_QUANTIZATION_TABLE_53_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE10_CR_CHROMA_QUANTIZATION_TABLE_52_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE10_CR_CHROMA_QUANTIZATION_TABLE_52_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE10_CR_CHROMA_QUANTIZATION_TABLE_51_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE10_CR_CHROMA_QUANTIZATION_TABLE_51_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE10_CR_CHROMA_QUANTIZATION_TABLE_50_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE10_CR_CHROMA_QUANTIZATION_TABLE_50_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_CHROMA_QUANTIZATION_TABLE11_OFFSET (0x00BC)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE11_CR_CHROMA_QUANTIZATION_TABLE_57_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE11_CR_CHROMA_QUANTIZATION_TABLE_57_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE11_CR_CHROMA_QUANTIZATION_TABLE_56_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE11_CR_CHROMA_QUANTIZATION_TABLE_56_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE11_CR_CHROMA_QUANTIZATION_TABLE_55_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE11_CR_CHROMA_QUANTIZATION_TABLE_55_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE11_CR_CHROMA_QUANTIZATION_TABLE_54_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE11_CR_CHROMA_QUANTIZATION_TABLE_54_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_CHROMA_QUANTIZATION_TABLE12_OFFSET (0x00C0)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE12_CR_CHROMA_QUANTIZATION_TABLE_63_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE12_CR_CHROMA_QUANTIZATION_TABLE_63_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE12_CR_CHROMA_QUANTIZATION_TABLE_62_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE12_CR_CHROMA_QUANTIZATION_TABLE_62_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE12_CR_CHROMA_QUANTIZATION_TABLE_61_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE12_CR_CHROMA_QUANTIZATION_TABLE_61_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE12_CR_CHROMA_QUANTIZATION_TABLE_60_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE12_CR_CHROMA_QUANTIZATION_TABLE_60_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_CHROMA_QUANTIZATION_TABLE13_OFFSET (0x00C4)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE13_CR_CHROMA_QUANTIZATION_TABLE_67_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE13_CR_CHROMA_QUANTIZATION_TABLE_67_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE13_CR_CHROMA_QUANTIZATION_TABLE_66_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE13_CR_CHROMA_QUANTIZATION_TABLE_66_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE13_CR_CHROMA_QUANTIZATION_TABLE_65_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE13_CR_CHROMA_QUANTIZATION_TABLE_65_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE13_CR_CHROMA_QUANTIZATION_TABLE_64_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE13_CR_CHROMA_QUANTIZATION_TABLE_64_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_CHROMA_QUANTIZATION_TABLE14_OFFSET (0x00C8)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE14_CR_CHROMA_QUANTIZATION_TABLE_73_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE14_CR_CHROMA_QUANTIZATION_TABLE_73_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE14_CR_CHROMA_QUANTIZATION_TABLE_72_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE14_CR_CHROMA_QUANTIZATION_TABLE_72_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE14_CR_CHROMA_QUANTIZATION_TABLE_71_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE14_CR_CHROMA_QUANTIZATION_TABLE_71_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE14_CR_CHROMA_QUANTIZATION_TABLE_70_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE14_CR_CHROMA_QUANTIZATION_TABLE_70_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_CHROMA_QUANTIZATION_TABLE15_OFFSET (0x00CC)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE15_CR_CHROMA_QUANTIZATION_TABLE_77_MASK
>>> (0xFF000000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE15_CR_CHROMA_QUANTIZATION_TABLE_77_SHIFT
>>> (24)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE15_CR_CHROMA_QUANTIZATION_TABLE_76_MASK
>>> (0x00FF0000)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE15_CR_CHROMA_QUANTIZATION_TABLE_76_SHIFT
>>> (16)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE15_CR_CHROMA_QUANTIZATION_TABLE_75_MASK
>>> (0x0000FF00)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE15_CR_CHROMA_QUANTIZATION_TABLE_75_SHIFT
>>> (8)
>>> +
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE15_CR_CHROMA_QUANTIZATION_TABLE_74_MASK
>>> (0x000000FF)
>>> +#define
>>> JASPER_CHROMA_QUANTIZATION_TABLE15_CR_CHROMA_QUANTIZATION_TABLE_74_SHIFT
>>> (0)
>>> +
>>> +#define JASPER_CRC_CTRL_OFFSET (0x00D0)
>>> +#define JASPER_CRC_CTRL_JASPER_CRC_ENABLE_MASK (0x00000001)
>>> +#define JASPER_CRC_CTRL_JASPER_CRC_ENABLE_SHIFT (0)
>>> +
>>> +#define JASPER_FRONT_END_CRC_OFFSET (0x00D4)
>>> +#define JASPER_FRONT_END_CRC_CR_JASPER_FRONT_END_CRC_OUT_MASK
>>> (0xFFFFFFFF)
>>> +#define JASPER_FRONT_END_CRC_CR_JASPER_FRONT_END_CRC_OUT_SHIFT (0)
>>> +
>>> +#define JASPER_DCT_CRC_OFFSET (0x00D8)
>>> +#define JASPER_DCT_CRC_CR_JASPER_DCT_CRC_OUT_MASK (0xFFFFFFFF)
>>> +#define JASPER_DCT_CRC_CR_JASPER_DCT_CRC_OUT_SHIFT (0)
>>> +
>>> +#define JASPER_ZZ_CRC_OFFSET (0x00DC)
>>> +#define JASPER_ZZ_CRC_CR_JASPER_ZZ_CRC_OUT_MASK (0xFFFFFFFF)
>>> +#define JASPER_ZZ_CRC_CR_JASPER_ZZ_CRC_OUT_SHIFT (0)
>>> +
>>> +#define JASPER_QUANT_CRC_OFFSET (0x00E0)
>>> +#define JASPER_QUANT_CRC_CR_JASPER_QUANT_CRC_OUT_MASK (0xFFFFFFFF)
>>> +#define JASPER_QUANT_CRC_CR_JASPER_QUANT_CRC_OUT_SHIFT (0)
>>> +
>>> +#define JASPER_ENTROPY_ENCODER_CRC_OFFSET (0x00E4)
>>> +#define JASPER_ENTROPY_ENCODER_CRC_CR_JASPER_ENTROPY_CRC_OUT_MASK
>>> (0xFFFFFFFF)
>>> +#define JASPER_ENTROPY_ENCODER_CRC_CR_JASPER_ENTROPY_CRC_OUT_SHIFT (0)
>>> +
>>> +#define JASPER_PACKING_BUFFER_DATA_CRC_OFFSET (0x00E8)
>>> +#define
>>> JASPER_PACKING_BUFFER_DATA_CRC_CR_JASPER_PACKING_DATA_CRC_OUT_MASK
>>> (0xFFFFFFFF)
>>> +#define
>>> JASPER_PACKING_BUFFER_DATA_CRC_CR_JASPER_PACKING_DATA_CRC_OUT_SHIFT (0)
>>> +
>>> +#define JASPER_PACKING_BUFFER_ADDR_CRC_OFFSET (0x00EC)
>>> +#define
>>> JASPER_PACKING_BUFFER_ADDR_CRC_CR_JASPER_PACKING_ADDR_OUT_CRC_MASK
>>> (0xFFFFFFFF)
>>> +#define
>>> JASPER_PACKING_BUFFER_ADDR_CRC_CR_JASPER_PACKING_ADDR_OUT_CRC_SHIFT (0)
>>> +
>>> +#define JASPER_CORE_BYTE_SIZE (0x00F0)
>>> +
>>> +#endif
>>> diff --git a/drivers/media/platform/imagination/e5010-jpeg-enc-hw.c
>>> b/drivers/media/platform/imagination/e5010-jpeg-enc-hw.c
>>> new file mode 100644
>>> index 000000000000..56d5941020fa
>>> --- /dev/null
>>> +++ b/drivers/media/platform/imagination/e5010-jpeg-enc-hw.c
>>> @@ -0,0 +1,267 @@
>>> +// SPDX-License-Identifier: GPL-2.0
>>> +/*
>>> + * Imagination E5010 JPEG Encoder driver.
>>> + *
>>> + * Copyright (C) 2023 Texas Instruments Incorporated -
>>> https://www.ti.com/
>>> + *
>>> + * Author: David Huang <d-huang@...com>
>>> + * Author: Devarsh Thakkar <devarsht@...com>
>>> + */
>>> +
>>> +#include <linux/io.h>
>>> +#include <linux/iopoll.h>
>>> +#include <linux/dev_printk.h>
>>> +#include "e5010-jpeg-enc-hw.h"
>>> +
>>> +static void write_reg_field(void __iomem *base, unsigned int
>>> offset, u32 mask,
>>> + unsigned int shift, u32 value)
>>> +{
>>> + u32 reg;
>>> +
>>> + value <<= shift;
>>> + if (mask != 0xffffffff) {
>>> + reg = readl(base + offset);
>>> + value = (value & mask) | (reg & ~mask);
>>> + }
>>> + writel(value, (base + offset));
>>> +}
>>> +
>>> +static int write_reg_field_not_busy(void __iomem *jasper_base, void
>>> __iomem *wr_base,
>>> + unsigned int offset, u32 mask, unsigned int shift,
>>> + u32 value)
>>> +{
>>> + int ret;
>>> + u32 val;
>>> +
>>> + ret = readl_poll_timeout_atomic(jasper_base +
>>> JASPER_STATUS_OFFSET, val,
>>> + (val & JASPER_STATUS_CR_JASPER_BUSY_MASK) == 0,
>>> + 2000, 50000);
>>> + if (ret)
>>> + return ret;
>>> +
>>> + write_reg_field(wr_base, offset, mask, shift, value);
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +void e5010_reset(struct device *dev, void __iomem *core_base, void
>>> __iomem *mmu_base)
>>> +{
>>> + int ret = 0;
>>> + u32 val;
>>> +
>>> + write_reg_field(core_base, JASPER_RESET_OFFSET,
>>> + JASPER_RESET_CR_CORE_RESET_MASK,
>>> + JASPER_RESET_CR_CORE_RESET_SHIFT, 1);
>>> +
>>> + write_reg_field(mmu_base, MMU_MMU_CONTROL1_OFFSET,
>>> + MMU_MMU_CONTROL1_MMU_SOFT_RESET_MASK,
>>> + MMU_MMU_CONTROL1_MMU_SOFT_RESET_SHIFT, 1);
>>> +
>>> + ret = readl_poll_timeout_atomic(mmu_base +
>>> MMU_MMU_CONTROL1_OFFSET, val,
>>> + (val & MMU_MMU_CONTROL1_MMU_SOFT_RESET_MASK) == 0,
>>> + 2000, 50000);
>>> + if (ret)
>>> + dev_warn(dev, "MMU soft reset timed out, forcing system
>>> soft reset\n");
>>> +
>>> + write_reg_field(core_base, JASPER_RESET_OFFSET,
>>> + JASPER_RESET_CR_SYS_RESET_MASK,
>>> + JASPER_RESET_CR_SYS_RESET_SHIFT, 1);
>>> +}
>>> +
>>> +void e5010_hw_bypass_mmu(void __iomem *mmu_base, u32 enable)
>>> +{
>>> + /* Bypass MMU */
>>> + write_reg_field(mmu_base,
>>> + MMU_MMU_ADDRESS_CONTROL_OFFSET,
>>> + MMU_MMU_ADDRESS_CONTROL_MMU_BYPASS_MASK,
>>> + MMU_MMU_ADDRESS_CONTROL_MMU_BYPASS_SHIFT,
>>> + enable);
>>> +}
>>> +
>>> +int e5010_hw_enable_output_address_error_irq(void __iomem
>>> *core_base, u32 enable)
>>> +{
>>> + return write_reg_field_not_busy(core_base, core_base,
>>> + JASPER_INTERRUPT_MASK_OFFSET,
>>> + JASPER_INTERRUPT_MASK_CR_OUTPUT_ADDRESS_ERROR_ENABLE_MASK,
>>> + JASPER_INTERRUPT_MASK_CR_OUTPUT_ADDRESS_ERROR_ENABLE_SHIFT,
>>> + enable);
>>> +}
>>> +
>>> +bool e5010_hw_pic_done_irq(void __iomem *core_base)
>>> +{
>>> + u32 reg;
>>> +
>>> + reg = readl(core_base + JASPER_INTERRUPT_STATUS_OFFSET);
>>> + return reg & JASPER_INTERRUPT_STATUS_CR_PICTURE_DONE_IRQ_MASK;
>>> +}
>>> +
>>> +bool e5010_hw_output_address_irq(void __iomem *core_base)
>>> +{
>>> + u32 reg;
>>> +
>>> + reg = readl(core_base + JASPER_INTERRUPT_STATUS_OFFSET);
>>> + return reg &
>>> JASPER_INTERRUPT_STATUS_CR_OUTPUT_ADDRESS_ERROR_IRQ_MASK;
>>> +}
>>> +
>>> +int e5010_hw_enable_picture_done_irq(void __iomem *core_base, u32
>>> enable)
>>> +{
>>> + return write_reg_field_not_busy(core_base, core_base,
>>> + JASPER_INTERRUPT_MASK_OFFSET,
>>> + JASPER_INTERRUPT_MASK_CR_PICTURE_DONE_ENABLE_MASK,
>>> + JASPER_INTERRUPT_MASK_CR_PICTURE_DONE_ENABLE_SHIFT,
>>> + enable);
>>> +}
>>> +
>>> +int e5010_hw_enable_auto_clock_gating(void __iomem *core_base, u32
>>> enable)
>>> +{
>>> + return write_reg_field_not_busy(core_base, core_base,
>>> + JASPER_CLK_CONTROL_OFFSET,
>>> + JASPER_CLK_CONTROL_CR_JASPER_AUTO_CLKG_ENABLE_MASK,
>>> + JASPER_CLK_CONTROL_CR_JASPER_AUTO_CLKG_ENABLE_SHIFT,
>>> + enable);
>>> +}
>>> +
>>> +int e5010_hw_enable_manual_clock_gating(void __iomem *core_base,
>>> u32 enable)
>>> +{
>>> + return write_reg_field_not_busy(core_base, core_base,
>>> + JASPER_CLK_CONTROL_OFFSET,
>>> + JASPER_CLK_CONTROL_CR_JASPER_MAN_CLKG_ENABLE_MASK,
>>> + JASPER_CLK_CONTROL_CR_JASPER_MAN_CLKG_ENABLE_SHIFT, 0);
>>> +}
>>> +
>>> +int e5010_hw_enable_crc_check(void __iomem *core_base, u32 enable)
>>> +{
>>> + return write_reg_field_not_busy(core_base, core_base,
>>> + JASPER_CRC_CTRL_OFFSET,
>>> + JASPER_CRC_CTRL_JASPER_CRC_ENABLE_MASK,
>>> + JASPER_CRC_CTRL_JASPER_CRC_ENABLE_SHIFT, enable);
>>> +}
>>> +
>>> +int e5010_hw_set_input_source_to_memory(void __iomem *core_base,
>>> u32 set)
>>> +{
>>> + return write_reg_field_not_busy(core_base, core_base,
>>> + JASPER_INPUT_CTRL0_OFFSET,
>>> + JASPER_INPUT_CTRL0_CR_INPUT_SOURCE_MASK,
>>> + JASPER_INPUT_CTRL0_CR_INPUT_SOURCE_SHIFT, set);
>>> +}
>>> +
>>> +int e5010_hw_set_input_luma_addr(void __iomem *core_base, u32 val)
>>> +{
>>> + return write_reg_field_not_busy(core_base, core_base,
>>> + INPUT_LUMA_BASE_OFFSET,
>>> + INPUT_LUMA_BASE_CR_INPUT_LUMA_BASE_MASK, 0, val);
>>> +}
>>> +
>>> +int e5010_hw_set_input_chroma_addr(void __iomem *core_base, u32 val)
>>> +{
>>> + return write_reg_field_not_busy(core_base, core_base,
>>> + INPUT_CHROMA_BASE_OFFSET,
>>> + INPUT_CHROMA_BASE_CR_INPUT_CHROMA_BASE_MASK, 0, val);
>>> +}
>>> +
>>> +int e5010_hw_set_output_base_addr(void __iomem *core_base, u32 val)
>>> +{
>>> + return write_reg_field_not_busy(core_base, core_base,
>>> + JASPER_OUTPUT_BASE_OFFSET,
>>> + JASPER_OUTPUT_BASE_CR_OUTPUT_BASE_MASK,
>>> + JASPER_OUTPUT_BASE_CR_OUTPUT_BASE_SHIFT, val);
>>> +}
>>> +
>>> +int e5010_hw_set_horizontal_size(void __iomem *core_base, u32 val)
>>> +{
>>> + return write_reg_field_not_busy(core_base, core_base,
>>> + JASPER_IMAGE_SIZE_OFFSET,
>>> + JASPER_IMAGE_SIZE_CR_IMAGE_HORIZONTAL_SIZE_MASK,
>>> + JASPER_IMAGE_SIZE_CR_IMAGE_HORIZONTAL_SIZE_SHIFT,
>>> + val);
>>> +}
>>> +
>>> +int e5010_hw_set_vertical_size(void __iomem *core_base, u32 val)
>>> +{
>>> + return write_reg_field_not_busy(core_base, core_base,
>>> + JASPER_IMAGE_SIZE_OFFSET,
>>> + JASPER_IMAGE_SIZE_CR_IMAGE_VERTICAL_SIZE_MASK,
>>> + JASPER_IMAGE_SIZE_CR_IMAGE_VERTICAL_SIZE_SHIFT,
>>> + val);
>>> +}
>>> +
>>> +int e5010_hw_set_luma_stride(void __iomem *core_base, u32
>>> bytesperline)
>>> +{
>>> + u32 val = bytesperline / 64;
>>> +
>>> + return write_reg_field_not_busy(core_base, core_base,
>>> + JASPER_INPUT_CTRL1_OFFSET,
>>> + JASPER_INPUT_CTRL1_CR_INPUT_LUMA_STRIDE_MASK,
>>> + JASPER_INPUT_CTRL1_CR_INPUT_LUMA_STRIDE_SHIFT,
>>> + val);
>>> +}
>>> +
>>> +int e5010_hw_set_chroma_stride(void __iomem *core_base, u32
>>> bytesperline)
>>> +{
>>> + u32 val = bytesperline / 64;
>>> +
>>> + return write_reg_field_not_busy(core_base, core_base,
>>> + JASPER_INPUT_CTRL1_OFFSET,
>>> + JASPER_INPUT_CTRL1_CR_INPUT_CHROMA_STRIDE_MASK,
>>> + JASPER_INPUT_CTRL1_CR_INPUT_CHROMA_STRIDE_SHIFT,
>>> + val);
>>> +}
>>> +
>>> +int e5010_hw_set_input_subsampling(void __iomem *core_base, u32 val)
>>> +{
>>> + return write_reg_field_not_busy(core_base, core_base,
>>> + JASPER_INPUT_CTRL0_OFFSET,
>>> + JASPER_INPUT_CTRL0_CR_INPUT_SUBSAMPLING_MASK,
>>> + JASPER_INPUT_CTRL0_CR_INPUT_SUBSAMPLING_SHIFT,
>>> + val);
>>> +}
>>> +
>>> +int e5010_hw_set_chroma_order(void __iomem *core_base, u32 val)
>>> +{
>>> + return write_reg_field_not_busy(core_base, core_base,
>>> + JASPER_INPUT_CTRL0_OFFSET,
>>> + JASPER_INPUT_CTRL0_CR_INPUT_CHROMA_ORDER_MASK,
>>> + JASPER_INPUT_CTRL0_CR_INPUT_CHROMA_ORDER_SHIFT,
>>> + val);
>>> +}
>>> +
>>> +void e5010_hw_set_output_max_size(void __iomem *core_base, u32 val)
>>> +{
>>> + write_reg_field(core_base, JASPER_OUTPUT_MAX_SIZE_OFFSET,
>>> + JASPER_OUTPUT_MAX_SIZE_CR_OUTPUT_MAX_SIZE_MASK,
>>> + JASPER_OUTPUT_MAX_SIZE_CR_OUTPUT_MAX_SIZE_SHIFT,
>>> + val);
>>> +}
>>> +
>>> +int e5010_hw_set_qpvalue(void __iomem *core_base, u32 offset, u32 val)
>>> +{
>>> + return write_reg_field_not_busy(core_base, core_base, offset,
>>> 0xffffffff, 0, val);
>>> +}
>>> +
>>> +void e5010_hw_clear_output_error(void __iomem *core_base, u32 clear)
>>> +{
>>> + /* Make sure interrupts are clear */
>>> + write_reg_field(core_base, JASPER_INTERRUPT_CLEAR_OFFSET,
>>> + JASPER_INTERRUPT_CLEAR_CR_OUTPUT_ERROR_CLEAR_MASK,
>>> + JASPER_INTERRUPT_CLEAR_CR_OUTPUT_ERROR_CLEAR_SHIFT, clear);
>>> +}
>>> +
>>> +void e5010_hw_clear_picture_done(void __iomem *core_base, u32 clear)
>>> +{
>>> + write_reg_field(core_base,
>>> + JASPER_INTERRUPT_CLEAR_OFFSET,
>>> + JASPER_INTERRUPT_CLEAR_CR_PICTURE_DONE_CLEAR_MASK,
>>> + JASPER_INTERRUPT_CLEAR_CR_PICTURE_DONE_CLEAR_SHIFT, clear);
>>> +}
>>> +
>>> +int e5010_hw_get_output_size(void __iomem *core_base)
>>> +{
>>> + return readl(core_base + JASPER_OUTPUT_SIZE_OFFSET);
>>> +}
>>> +
>>> +void e5010_hw_encode_start(void __iomem *core_base, u32 start)
>>> +{
>>> + write_reg_field(core_base, JASPER_CORE_CTRL_OFFSET,
>>> + JASPER_CORE_CTRL_CR_JASPER_ENCODE_START_MASK,
>>> + JASPER_CORE_CTRL_CR_JASPER_ENCODE_START_SHIFT, start);
>>> +}
>>> diff --git a/drivers/media/platform/imagination/e5010-jpeg-enc-hw.h
>>> b/drivers/media/platform/imagination/e5010-jpeg-enc-hw.h
>>> new file mode 100644
>>> index 000000000000..781d353c3226
>>> --- /dev/null
>>> +++ b/drivers/media/platform/imagination/e5010-jpeg-enc-hw.h
>>> @@ -0,0 +1,42 @@
>>> +/* SPDX-License-Identifier: GPL-2.0 */
>>> +/*
>>> + * Imagination E5010 JPEG Encoder driver.
>>> + *
>>> + * Copyright (C) 2023 Texas Instruments Incorporated -
>>> https://www.ti.com/
>>> + *
>>> + * Author: David Huang <d-huang@...com>
>>> + * Author: Devarsh Thakkar <devarsht@...com>
>>> + */
>>> +
>>> +#ifndef _E5010_JPEG_ENC_HW_H
>>> +#define _E5010_JPEG_ENC_HW_H
>>> +
>>> +#include "e5010-core-regs.h"
>>> +#include "e5010-mmu-regs.h"
>>> +
>>> +int e5010_hw_enable_output_address_error_irq(void __iomem
>>> *core_offset, u32 enable);
>>> +int e5010_hw_enable_picture_done_irq(void __iomem *core_offset, u32
>>> enable);
>>> +int e5010_hw_enable_auto_clock_gating(void __iomem *core_offset,
>>> u32 enable);
>>> +int e5010_hw_enable_manual_clock_gating(void __iomem *core_offset,
>>> u32 enable);
>>> +int e5010_hw_enable_crc_check(void __iomem *core_offset, u32 enable);
>>> +int e5010_hw_set_input_source_to_memory(void __iomem *core_offset,
>>> u32 set);
>>> +int e5010_hw_set_input_luma_addr(void __iomem *core_offset, u32 val);
>>> +int e5010_hw_set_input_chroma_addr(void __iomem *core_offset, u32
>>> val);
>>> +int e5010_hw_set_output_base_addr(void __iomem *core_offset, u32 val);
>>> +int e5010_hw_get_output_size(void __iomem *core_offset);
>>> +int e5010_hw_set_horizontal_size(void __iomem *core_offset, u32 val);
>>> +int e5010_hw_set_vertical_size(void __iomem *core_offset, u32 val);
>>> +int e5010_hw_set_luma_stride(void __iomem *core_offset, u32
>>> bytesperline);
>>> +int e5010_hw_set_chroma_stride(void __iomem *core_offset, u32
>>> bytesperline);
>>> +int e5010_hw_set_input_subsampling(void __iomem *core_offset, u32
>>> val);
>>> +int e5010_hw_set_chroma_order(void __iomem *core_offset, u32 val);
>>> +int e5010_hw_set_qpvalue(void __iomem *core_offset, u32 offset, u32
>>> value);
>>> +void e5010_reset(struct device *dev, void __iomem *core_offset,
>>> void __iomem *mmu_offset);
>>> +void e5010_hw_set_output_max_size(void __iomem *core_offset, u32 val);
>>> +void e5010_hw_clear_picture_done(void __iomem *core_offset, u32
>>> clear);
>>> +void e5010_hw_encode_start(void __iomem *core_offset, u32 start);
>>> +void e5010_hw_clear_output_error(void __iomem *core_offset, u32
>>> clear);
>>> +void e5010_hw_bypass_mmu(void __iomem *mmu_base, u32 enable);
>>> +bool e5010_hw_pic_done_irq(void __iomem *core_base);
>>> +bool e5010_hw_output_address_irq(void __iomem *core_base);
>>> +#endif
>>> diff --git a/drivers/media/platform/imagination/e5010-jpeg-enc.c
>>> b/drivers/media/platform/imagination/e5010-jpeg-enc.c
>>> new file mode 100644
>>> index 000000000000..4a929a19c83b
>>> --- /dev/null
>>> +++ b/drivers/media/platform/imagination/e5010-jpeg-enc.c
>>> @@ -0,0 +1,1552 @@
>>> +// SPDX-License-Identifier: GPL-2.0
>>> +/*
>>> + * Imagination E5010 JPEG Encoder driver.
>>> + *
>>> + * TODO: Add MMU and memory tiling support
>>> + *
>>> + * Copyright (C) 2023 Texas Instruments Incorporated -
>>> https://www.ti.com/
>>> + *
>>> + * Author: David Huang <d-huang@...com>
>>> + * Author: Devarsh Thakkar <devarsht@...com>
>>> + */
>>> +
>>> +#include <linux/module.h>
>>> +#include <linux/of_device.h>
>>> +#include <linux/ioctl.h>
>>> +#include <linux/dma-mapping.h>
>>> +#include <linux/err.h>
>>> +#include <linux/pm_runtime.h>
>>> +#include <linux/interrupt.h>
>>> +#include <linux/clk.h>
>>> +#include <media/v4l2-common.h>
>>> +#include <media/v4l2-ctrls.h>
>>> +#include <media/v4l2-device.h>
>>> +#include <media/v4l2-event.h>
>>> +#include <media/v4l2-ioctl.h>
>>> +#include <media/v4l2-mem2mem.h>
>>> +#include <media/jpeg-enc-reftables.h>
>>
>> The driver doesn't compile because this include file name is wrong.
>
>
> I don't see anything wrong with the naming here, The header file
> jpeg-enc-reftables.h is introduced in previous commit, did you include
> [PATCH 2/3] media: jpeg: Add reference quantization and huffman tables
> while trying to compile ?
In this previous patch you introduce jpeg*_*enc*_*reftables.h with underscores:
create mode 100644 include/media/jpeg_enc_reftables.h
I guess you still have in your local tree the file without the underscores.
Regards,
Benjamin
>
> To re-check I downloaded the series again from lore and re-compiled
> and it works fine for me :
>
> ✗ b4 am
> https://lore.kernel.org/all/20240205114239.924697-1-devarsht@ti.com/
> Grabbing thread from
> lore.kernel.org/all/20240205114239.924697-1-devarsht%40ti.com/t.mbox.gz
> Analyzing 8 messages in the thread
> Checking attestation on all messages, may take a moment...
> ---
> ✗ [PATCH v4 1/3] media: dt-bindings: Add Imagination E5010 JPEG Encoder
> + Reviewed-by: Rob Herring <robh@...nel.org> (✗ DKIM/kernel.org)
> ✗ [PATCH v4 2/3] media: jpeg: Add reference quantization and huffman
> tables
> ✗ [PATCH v4 3/3] media: imagination: Add E5010 JPEG Encoder driver
> ---
> ✗ BADSIG: DKIM/ti.com
> ---
> NOTE: install patatt for end-to-end signature verification
> ---
> Total patches: 3
> ---
> Cover:
> ./v4_20240205_devarsht_add_v4l2_m2m_driver_for_e5010_jpeg_encoder.cover
> Link: https://lore.kernel.org/r/20240205114239.924697-1-devarsht@ti.com
> Base: applies clean to current tree
> git am
> ./v4_20240205_devarsht_add_v4l2_m2m_driver_for_e5010_jpeg_encoder.mbx
> ➜ linux-next git:(v4_real_jpeg_new) ✗ git am
> ./v4_20240205_devarsht_add_v4l2_m2m_driver_for_e5010_jpeg_encoder.mbx
> Applying: media: dt-bindings: Add Imagination E5010 JPEG Encoder
> Applying: media: jpeg: Add reference quantization and huffman tables
> Applying: media: imagination: Add E5010 JPEG Encoder driver
>
> rm drivers/media/platform/imagination/*.o
> ➜ linux-next git:(v4_real_jpeg_new) ✗ make modules -j20
> CALL scripts/checksyscalls.sh
> CC [M] drivers/media/platform/imagination/e5010-jpeg-enc-hw.o
> CC [M] drivers/media/platform/imagination/e5010-jpeg-enc.o
> LD [M] drivers/media/platform/imagination/e5010_jpeg_enc.o
> MODPOST Module.symvers
> CC [M] drivers/media/platform/imagination/e5010_jpeg_enc.mod.o
> LD [M] drivers/media/platform/imagination/e5010_jpeg_enc.ko
> ➜ linux-next git:(v4_real_jpeg_new) ✗
>
>
>> Please also put the include in alphabetic order.
>>
>
> Sure, will do.
>
>>> +#include <media/videobuf2-v4l2.h>
>>> +#include <media/videobuf2-dma-contig.h>
>>> +#include <media/jpeg.h>
>>> +#include "e5010-jpeg-enc.h"
>>> +#include "e5010-jpeg-enc-hw.h"
>>> +
>>> +/* forward declarations */
>>> +static const struct of_device_id e5010_of_match[];
>>> +
>>> +static const struct v4l2_file_operations e5010_fops;
>>> +
>>> +static const struct v4l2_ioctl_ops e5010_ioctl_ops;
>>> +
>>> +static const struct vb2_ops e5010_video_ops;
>>> +
>>> +static const struct v4l2_m2m_ops e5010_m2m_ops;
>>
>> You can remove these declaration by reordering your code below.
>>
>
> I had gone with forward declarations to keep all the above structs at
> one place for better readability.
>
> But yes it is technically feasible to get rid of these forward
> declarations although that will require moving these structs in
> between the functions i.e after the functions they are referring in
> their structs and before the funcs using those structs which I
> personally did not prefer, but I can do that if that's the more
> popular opinion. Please do let me know.
>
>>> +
>>> +static struct e5010_fmt e5010_formats[] = {
>>> + {
>>> + .fourcc = V4L2_PIX_FMT_NV12,
>>> + .num_planes = 1,
>>> + .type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE,
>>> + .subsampling = V4L2_JPEG_CHROMA_SUBSAMPLING_420,
>>> + .chroma_order = CHROMA_ORDER_CB_CR,
>>> + .frmsize = { MIN_DIMENSION, MAX_DIMENSION, 64,
>>> + MIN_DIMENSION, MAX_DIMENSION, 8 },
>>> + },
>>> + {
>>> + .fourcc = V4L2_PIX_FMT_NV12M,
>>> + .num_planes = 2,
>>> + .type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE,
>>> + .subsampling = V4L2_JPEG_CHROMA_SUBSAMPLING_420,
>>> + .chroma_order = CHROMA_ORDER_CB_CR,
>>> + .frmsize = { MIN_DIMENSION, MAX_DIMENSION, 64,
>>> + MIN_DIMENSION, MAX_DIMENSION, 8 },
>>> + },
>>> + {
>>> + .fourcc = V4L2_PIX_FMT_NV21,
>>> + .num_planes = 1,
>>> + .type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE,
>>> + .subsampling = V4L2_JPEG_CHROMA_SUBSAMPLING_420,
>>> + .chroma_order = CHROMA_ORDER_CR_CB,
>>> + .frmsize = { MIN_DIMENSION, MAX_DIMENSION, 64,
>>> + MIN_DIMENSION, MAX_DIMENSION, 8 },
>>> + },
>>> + {
>>> + .fourcc = V4L2_PIX_FMT_NV21M,
>>> + .num_planes = 2,
>>> + .type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE,
>>> + .subsampling = V4L2_JPEG_CHROMA_SUBSAMPLING_420,
>>> + .chroma_order = CHROMA_ORDER_CR_CB,
>>> + .frmsize = { MIN_DIMENSION, MAX_DIMENSION, 64,
>>> + MIN_DIMENSION, MAX_DIMENSION, 8 },
>>> + },
>>> + {
>>> + .fourcc = V4L2_PIX_FMT_NV16,
>>> + .num_planes = 1,
>>> + .type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE,
>>> + .subsampling = V4L2_JPEG_CHROMA_SUBSAMPLING_422,
>>> + .chroma_order = CHROMA_ORDER_CB_CR,
>>> + .frmsize = { MIN_DIMENSION, MAX_DIMENSION, 64,
>>> + MIN_DIMENSION, MAX_DIMENSION, 8 },
>>> + },
>>> + {
>>> + .fourcc = V4L2_PIX_FMT_NV16M,
>>> + .num_planes = 2,
>>> + .type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE,
>>> + .subsampling = V4L2_JPEG_CHROMA_SUBSAMPLING_422,
>>> + .chroma_order = CHROMA_ORDER_CB_CR,
>>> + .frmsize = { MIN_DIMENSION, MAX_DIMENSION, 64,
>>> + MIN_DIMENSION, MAX_DIMENSION, 8 },
>>> +
>>> + },
>>> + {
>>> + .fourcc = V4L2_PIX_FMT_NV61,
>>> + .num_planes = 1,
>>> + .type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE,
>>> + .subsampling = V4L2_JPEG_CHROMA_SUBSAMPLING_422,
>>> + .chroma_order = CHROMA_ORDER_CR_CB,
>>> + .frmsize = { MIN_DIMENSION, MAX_DIMENSION, 64,
>>> + MIN_DIMENSION, MAX_DIMENSION, 8 },
>>> + },
>>> + {
>>> + .fourcc = V4L2_PIX_FMT_NV61M,
>>> + .num_planes = 2,
>>> + .type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE,
>>> + .subsampling = V4L2_JPEG_CHROMA_SUBSAMPLING_422,
>>> + .chroma_order = CHROMA_ORDER_CR_CB,
>>> + .frmsize = { MIN_DIMENSION, MAX_DIMENSION, 64,
>>> + MIN_DIMENSION, MAX_DIMENSION, 8 },
>>> + },
>>> + {
>>> + .fourcc = V4L2_PIX_FMT_JPEG,
>>> + .num_planes = 1,
>>> + .type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE,
>>> + .subsampling = 0,
>>> + .chroma_order = 0,
>>> + .frmsize = { MIN_DIMENSION, MAX_DIMENSION, 16,
>>> + MIN_DIMENSION, MAX_DIMENSION, 8 },
>>> + },
>>> +};
>>> +
>>> +static unsigned int debug;
>>> +module_param(debug, uint, 0644);
>>> +MODULE_PARM_DESC(debug, "debug level");
>>> +
>>> +#define dprintk(dev, lvl, fmt, arg...) \
>>> + v4l2_dbg(lvl, debug, &(dev)->v4l2_dev, "%s: " fmt, __func__, ##
>>> arg)
>>> +
>>> +static const struct v4l2_event e5010_eos_event = {
>>> + .type = V4L2_EVENT_EOS
>>> +};
>>> +
>>> +static const char *type_name(enum v4l2_buf_type type)
>>> +{
>>> + switch (type) {
>>> + case V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE:
>>> + return "Output";
>>> + case V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE:
>>> + return "Capture";
>>> + default:
>>> + return "Invalid";
>>> + }
>>> +}
>>> +
>>> +static struct e5010_q_data *get_queue(struct e5010_context *ctx,
>>> enum v4l2_buf_type type)
>>> +{
>>> + return (type == V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE) ?
>>> &ctx->out_queue : &ctx->cap_queue;
>>> +}
>>> +
>>> +static void calculate_qp_tables(struct e5010_context *ctx)
>>> +{
>>> + long long luminosity, contrast;
>>> + int quality, i;
>>> +
>>> + quality = 50 - ctx->quality;
>>> +
>>> + luminosity = LUMINOSITY * quality / 50;
>>> + contrast = CONTRAST * quality / 50;
>>> +
>>> + if (quality > 0) {
>>> + luminosity *= INCREASE;
>>> + contrast *= INCREASE;
>>> + }
>>> +
>>> + for (i = 0; i < ARRAY_SIZE(luma_q_table); i++) {
>>> + long long delta = chroma_q_table[i] * contrast + luminosity;
>>> + int val = (int)(chroma_q_table[i] + delta);
>>> +
>>> + clamp(val, 1, 255);
>>> + ctx->chroma_qp[i] = quality == -50 ? 1 : val;
>>> +
>>> + delta = luma_q_table[i] * contrast + luminosity;
>>> + val = (int)(luma_q_table[i] + delta);
>>> + clamp(val, 1, 255);
>>> + ctx->luma_qp[i] = quality == -50 ? 1 : val;
>>> + }
>>> +
>>> + ctx->update_qp = true;
>>> +}
>>> +
>>> +static int update_qp_tables(struct e5010_context *ctx)
>>> +{
>>> + struct e5010_dev *e5010 = ctx->e5010;
>>> + int i, ret = 0;
>>> + u32 lvalue, cvalue;
>>> +
>>> + lvalue = 0;
>>> + cvalue = 0;
>>> +
>>> + for (i = 0; i < QP_TABLE_SIZE; i++) {
>>> + lvalue |= ctx->luma_qp[i] << (8 * (i % 4));
>>> + cvalue |= ctx->chroma_qp[i] << (8 * (i % 4));
>>> + if (i % 4 == 3) {
>>> + ret |= e5010_hw_set_qpvalue(e5010->core_base,
>>> + JASPER_LUMA_QUANTIZATION_TABLE0_OFFSET
>>> + + QP_TABLE_FIELD_OFFSET * ((i - 3) / 4),
>>> + lvalue);
>>> + ret |= e5010_hw_set_qpvalue(e5010->core_base,
>>> + JASPER_CHROMA_QUANTIZATION_TABLE0_OFFSET
>>> + + QP_TABLE_FIELD_OFFSET * ((i - 3) / 4),
>>> + cvalue);
>>> + lvalue = 0;
>>> + cvalue = 0;
>>> + }
>>> + }
>>> +
>>> + return ret;
>>> +}
>>> +
>>> +static int e5010_set_input_subsampling(void __iomem *core_base, int
>>> subsampling)
>>> +{
>>> + switch (subsampling) {
>>> + case V4L2_JPEG_CHROMA_SUBSAMPLING_420:
>>> + return e5010_hw_set_input_subsampling(core_base,
>>> SUBSAMPLING_420);
>>> + case V4L2_JPEG_CHROMA_SUBSAMPLING_422:
>>> + return e5010_hw_set_input_subsampling(core_base,
>>> SUBSAMPLING_422);
>>> + default:
>>> + return -EINVAL;
>>> + };
>>> +}
>>> +
>>> +static int e5010_querycap(struct file *file, void *priv, struct
>>> v4l2_capability *cap)
>>> +{
>>> + strscpy(cap->driver, E5010_MODULE_NAME, sizeof(cap->driver));
>>> + strscpy(cap->card, E5010_MODULE_NAME, sizeof(cap->card));
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +static struct e5010_fmt *find_format(struct v4l2_format *f)
>>> +{
>>> + int i;
>>> +
>>> + for (i = 0; i < ARRAY_SIZE(e5010_formats); ++i) {
>>> + if (e5010_formats[i].fourcc == f->fmt.pix_mp.pixelformat &&
>>> + e5010_formats[i].type == f->type)
>>> + return &e5010_formats[i];
>>> + }
>>> +
>>> + return NULL;
>>> +}
>>> +
>>> +static int e5010_enum_fmt(struct file *file, void *priv, struct
>>> v4l2_fmtdesc *f)
>>> +{
>>> + int i, index = 0;
>>> + struct e5010_fmt *fmt = NULL;
>>> + struct e5010_context *ctx = file->private_data;
>>> +
>>> + if (!V4L2_TYPE_IS_MULTIPLANAR(f->type)) {
>>> + v4l2_err(&ctx->e5010->v4l2_dev, "ENUMFMT with Invalid type:
>>> %d\n", f->type);
>>> + return -EINVAL;
>>> + }
>>> +
>>> + for (i = 0; i < ARRAY_SIZE(e5010_formats); ++i) {
>>> + if (e5010_formats[i].type == f->type) {
>>> + if (index == f->index) {
>>> + fmt = &e5010_formats[i];
>>> + break;
>>> + }
>>> + index++;
>>> + }
>>> + }
>>> +
>>> + if (!fmt)
>>> + return -EINVAL;
>>> +
>>> + f->pixelformat = fmt->fourcc;
>>> + return 0;
>>> +}
>>> +
>>> +static int e5010_g_fmt(struct file *file, void *priv, struct
>>> v4l2_format *f)
>>> +{
>>> + struct e5010_context *ctx = file->private_data;
>>> + struct e5010_q_data *queue;
>>> + int i;
>>> + struct v4l2_pix_format_mplane *pix_mp = &f->fmt.pix_mp;
>>> + struct v4l2_plane_pix_format *plane_fmt = pix_mp->plane_fmt;
>>> +
>>> + queue = get_queue(ctx, f->type);
>>> +
>>> + pix_mp->flags = 0;
>>> + pix_mp->field = V4L2_FIELD_NONE;
>>> + pix_mp->pixelformat = queue->fmt->fourcc;
>>> + pix_mp->width = queue->width_adjusted;
>>> + pix_mp->height = queue->height_adjusted;
>>> + pix_mp->num_planes = queue->fmt->num_planes;
>>> +
>>> + if (V4L2_TYPE_IS_OUTPUT(f->type)) {
>>> + if (!pix_mp->colorspace)
>>> + pix_mp->colorspace = V4L2_COLORSPACE_SRGB;
>>> +
>>> + for (i = 0; i < queue->fmt->num_planes; i++) {
>>> + plane_fmt[i].sizeimage = queue->sizeimage[i];
>>> + plane_fmt[i].bytesperline = queue->bytesperline[i];
>>> + }
>>> +
>>> + } else {
>>> + pix_mp->colorspace = V4L2_COLORSPACE_JPEG;
>>> + plane_fmt[0].bytesperline = 0;
>>> + plane_fmt[0].sizeimage = queue->sizeimage[0];
>>> + }
>>> + pix_mp->ycbcr_enc = V4L2_YCBCR_ENC_DEFAULT;
>>> + pix_mp->xfer_func = V4L2_XFER_FUNC_DEFAULT;
>>> + pix_mp->quantization = V4L2_QUANTIZATION_DEFAULT;
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +static int e5010_jpeg_try_fmt(struct v4l2_format *f, struct
>>> e5010_context *ctx)
>>> +{
>>> + struct e5010_fmt *fmt;
>>> + struct v4l2_pix_format_mplane *pix_mp = &f->fmt.pix_mp;
>>> + struct v4l2_plane_pix_format *plane_fmt = pix_mp->plane_fmt;
>>> +
>>> + fmt = find_format(f);
>>> + if (!fmt) {
>>> + if (V4L2_TYPE_IS_OUTPUT(f->type))
>>> + pix_mp->pixelformat = V4L2_PIX_FMT_NV12;
>>> + else
>>> + pix_mp->pixelformat = V4L2_PIX_FMT_JPEG;
>>> + fmt = find_format(f);
>>> + if (!fmt)
>>> + return -EINVAL;
>>> + }
>>> +
>>> + if (V4L2_TYPE_IS_OUTPUT(f->type)) {
>>> + if (!pix_mp->colorspace)
>>> + pix_mp->colorspace = V4L2_COLORSPACE_JPEG;
>>> + if (!pix_mp->ycbcr_enc)
>>> + pix_mp->ycbcr_enc = V4L2_YCBCR_ENC_DEFAULT;
>>> + if (!pix_mp->quantization)
>>> + pix_mp->quantization = V4L2_QUANTIZATION_DEFAULT;
>>> + if (!pix_mp->xfer_func)
>>> + pix_mp->xfer_func = V4L2_XFER_FUNC_DEFAULT;
>>> +
>>> + v4l2_apply_frmsize_constraints(&pix_mp->width,
>>> + &pix_mp->height,
>>> + &fmt->frmsize);
>>> +
>>> + v4l2_fill_pixfmt_mp(pix_mp, pix_mp->pixelformat,
>>> + pix_mp->width, pix_mp->height);
>>> +
>>> + } else {
>>> + pix_mp->colorspace = V4L2_COLORSPACE_JPEG;
>>> + pix_mp->ycbcr_enc = V4L2_YCBCR_ENC_DEFAULT;
>>> + pix_mp->quantization = V4L2_QUANTIZATION_DEFAULT;
>>> + pix_mp->xfer_func = V4L2_XFER_FUNC_DEFAULT;
>>> + v4l2_apply_frmsize_constraints(&pix_mp->width,
>>> + &pix_mp->height,
>>> + &fmt->frmsize);
>>> + plane_fmt[0].sizeimage = pix_mp->width * pix_mp->height *
>>> JPEG_MAX_BYTES_PER_PIXEL;
>>> + plane_fmt[0].sizeimage += HEADER_SIZE;
>>> + plane_fmt[0].bytesperline = 0;
>>> + pix_mp->pixelformat = fmt->fourcc;
>>> + pix_mp->num_planes = fmt->num_planes;
>>> + }
>>> + pix_mp->flags = 0;
>>> + pix_mp->field = V4L2_FIELD_NONE;
>>> +
>>> + dprintk(ctx->e5010, 2,
>>> + "ctx: 0x%p: format type %s:, wxh: %dx%d (plane0 : %d bytes,
>>> plane1 : %d bytes),fmt: %c%c%c%c\n",
>>> + ctx, type_name(f->type), pix_mp->width, pix_mp->height,
>>> + plane_fmt[0].sizeimage, plane_fmt[1].sizeimage,
>>> + (fmt->fourcc & 0xff),
>>> + (fmt->fourcc >> 8) & 0xff,
>>> + (fmt->fourcc >> 16) & 0xff,
>>> + (fmt->fourcc >> 24) & 0xff);
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +static int e5010_try_fmt(struct file *file, void *priv, struct
>>> v4l2_format *f)
>>> +{
>>> + struct e5010_context *ctx = file->private_data;
>>> +
>>> + return e5010_jpeg_try_fmt(f, ctx);
>>> +}
>>> +
>>> +static int e5010_s_fmt(struct file *file, void *priv, struct
>>> v4l2_format *f)
>>> +{
>>> + struct e5010_context *ctx = file->private_data;
>>> + struct vb2_queue *vq;
>>> + int ret = 0, i = 0;
>>> + struct v4l2_pix_format_mplane *pix_mp = &f->fmt.pix_mp;
>>> + struct v4l2_plane_pix_format *plane_fmt = pix_mp->plane_fmt;
>>> + struct e5010_q_data *queue;
>>> + struct e5010_fmt *fmt;
>>> +
>>> + vq = v4l2_m2m_get_vq(ctx->fh.m2m_ctx, f->type);
>>> + if (!vq)
>>> + return -EINVAL;
>>> +
>>> + if (vb2_is_busy(vq)) {
>>> + v4l2_err(&ctx->e5010->v4l2_dev, "queue busy\n");
>>> + return -EBUSY;
>>> + }
>>> +
>>> + ret = e5010_jpeg_try_fmt(f, ctx);
>>> + if (ret)
>>> + return ret;
>>> +
>>> + fmt = find_format(f);
>>> + queue = get_queue(ctx, f->type);
>>> +
>>> + queue->fmt = fmt;
>>> + queue->width = pix_mp->width;
>>> + queue->height = pix_mp->height;
>>> +
>>> + if (V4L2_TYPE_IS_OUTPUT(f->type)) {
>>> + for (i = 0; i < fmt->num_planes; i++) {
>>> + queue->bytesperline[i] = plane_fmt[i].bytesperline;
>>> + queue->sizeimage[i] = plane_fmt[i].sizeimage;
>>> + }
>>> + } else {
>>> + queue->sizeimage[0] = plane_fmt[0].sizeimage;
>>> + queue->sizeimage[1] = 0;
>>> + queue->bytesperline[0] = 0;
>>> + queue->bytesperline[1] = 0;
>>> + }
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +static int e5010_enum_framesizes(struct file *file, void *priv,
>>> struct v4l2_frmsizeenum *fsize)
>>> +{
>>> + struct v4l2_format f;
>>> + struct e5010_fmt *fmt;
>>> +
>>> + if (fsize->index != 0)
>>> + return -EINVAL;
>>> +
>>> + f.fmt.pix_mp.pixelformat = fsize->pixel_format;
>>> + if (f.fmt.pix_mp.pixelformat == V4L2_PIX_FMT_JPEG)
>>> + f.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
>>> + else
>>> + f.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
>>> +
>>> + fmt = find_format(&f);
>>> + if (!fmt)
>>> + return -EINVAL;
>>> +
>>> + fsize->type = V4L2_FRMSIZE_TYPE_STEPWISE;
>>> + fsize->stepwise = fmt->frmsize;
>>> + fsize->reserved[0] = 0;
>>> + fsize->reserved[1] = 0;
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +static int e5010_g_selection(struct file *file, void *fh, struct
>>> v4l2_selection *s)
>>> +{
>>> + struct e5010_context *ctx = file->private_data;
>>> + struct e5010_q_data *queue;
>>> +
>>> + if (s->type != V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE)
>>> + return -EINVAL;
>>> +
>>> + queue = get_queue(ctx, s->type);
>>> +
>>> + switch (s->target) {
>>> + case V4L2_SEL_TGT_CROP_DEFAULT:
>>> + case V4L2_SEL_TGT_CROP_BOUNDS:
>>> + s->r.left = 0;
>>> + s->r.top = 0;
>>> + s->r.width = queue->width;
>>> + s->r.height = queue->height;
>>> + break;
>>> + case V4L2_SEL_TGT_CROP:
>>> + s->r = queue->crop;
>>> + break;
>>> + default:
>>> + return -EINVAL;
>>> + }
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +static int e5010_s_selection(struct file *file, void *fh, struct
>>> v4l2_selection *s)
>>> +{
>>> + struct e5010_context *ctx = file->private_data;
>>> + struct e5010_q_data *queue;
>>> +
>>> + if (s->type != V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE)
>>> + return -EINVAL;
>>> +
>>> + queue = get_queue(ctx, s->type);
>>> +
>>> + queue->crop.left = 0;
>>> + queue->crop.top = 0;
>>> + queue->crop.width = s->r.width;
>>> + queue->crop.height = s->r.height;
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +static int e5010_subscribe_event(struct v4l2_fh *fh, const struct
>>> v4l2_event_subscription *sub)
>>> +{
>>> + switch (sub->type) {
>>> + case V4L2_EVENT_EOS:
>>> + return v4l2_event_subscribe(fh, sub, 0, NULL);
>>> + case V4L2_EVENT_CTRL:
>>> + return v4l2_ctrl_subscribe_event(fh, sub);
>>> + default:
>>> + return -EINVAL;
>>> + }
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +static int queue_init(void *priv, struct vb2_queue *src_vq, struct
>>> vb2_queue *dst_vq)
>>> +{
>>> + struct e5010_context *ctx = priv;
>>> + struct e5010_dev *e5010 = ctx->e5010;
>>> + int ret = 0;
>>> +
>>> + /* src_vq */
>>> + memset(src_vq, 0, sizeof(*src_vq));
>>> + src_vq->type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
>>> + src_vq->io_modes = VB2_MMAP | VB2_DMABUF;
>>> + src_vq->drv_priv = ctx;
>>> + src_vq->buf_struct_size = sizeof(struct e5010_buffer);
>>> + src_vq->ops = &e5010_video_ops;
>>> + src_vq->mem_ops = &vb2_dma_contig_memops;
>>> + src_vq->timestamp_flags = V4L2_BUF_FLAG_TIMESTAMP_COPY;
>>> + src_vq->lock = &e5010->mutex;
>>> + src_vq->dev = e5010->v4l2_dev.dev;
>>> +
>>> + ret = vb2_queue_init(src_vq);
>>> + if (ret)
>>> + return ret;
>>> +
>>> + /* dst_vq */
>>> + memset(dst_vq, 0, sizeof(*dst_vq));
>>> + dst_vq->type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
>>> + dst_vq->io_modes = VB2_MMAP | VB2_DMABUF;
>>> + dst_vq->drv_priv = ctx;
>>> + dst_vq->buf_struct_size = sizeof(struct e5010_buffer);
>>> + dst_vq->ops = &e5010_video_ops;
>>> + dst_vq->mem_ops = &vb2_dma_contig_memops;
>>> + dst_vq->timestamp_flags = V4L2_BUF_FLAG_TIMESTAMP_COPY;
>>> + dst_vq->lock = &e5010->mutex;
>>> + dst_vq->dev = e5010->v4l2_dev.dev;
>>> +
>>> + ret = vb2_queue_init(dst_vq);
>>> + if (ret) {
>>> + vb2_queue_release(src_vq);
>>> + return ret;
>>> + }
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +static int e5010_s_ctrl(struct v4l2_ctrl *ctrl)
>>> +{
>>> + struct e5010_context *ctx =
>>> + container_of(ctrl->handler, struct e5010_context,
>>> ctrl_handler);
>>> +
>>> + switch (ctrl->id) {
>>> + case V4L2_CID_JPEG_COMPRESSION_QUALITY:
>>> + ctx->quality = ctrl->val;
>>> + calculate_qp_tables(ctx);
>>> + break;
>>> + default:
>>> + return -EINVAL;
>>> + }
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +static const struct v4l2_ctrl_ops e5010_ctrl_ops = {
>>> + .s_ctrl = e5010_s_ctrl,
>>> +};
>>> +
>>> +static void e5010_encode_ctrls(struct e5010_context *ctx)
>>> +{
>>> + v4l2_ctrl_new_std(&ctx->ctrl_handler, &e5010_ctrl_ops,
>>> + V4L2_CID_JPEG_COMPRESSION_QUALITY, 1, 100, 1, 75);
>>> +}
>>> +
>>> +static int e5010_ctrls_setup(struct e5010_context *ctx)
>>> +{
>>> + int err;
>>> +
>>> + v4l2_ctrl_handler_init(&ctx->ctrl_handler, 1);
>>> +
>>> + e5010_encode_ctrls(ctx);
>>> +
>>> + if (ctx->ctrl_handler.error) {
>>> + err = ctx->ctrl_handler.error;
>>> + v4l2_ctrl_handler_free(&ctx->ctrl_handler);
>>> +
>>> + return err;
>>> + }
>>> +
>>> + err = v4l2_ctrl_handler_setup(&ctx->ctrl_handler);
>>> + if (err)
>>> + v4l2_ctrl_handler_free(&ctx->ctrl_handler);
>>> +
>>> + return err;
>>> +}
>>> +
>>> +static void e5010_jpeg_set_default_params(struct e5010_context *ctx)
>>> +{
>>> + struct e5010_q_data *queue;
>>> + struct v4l2_format f;
>>> + struct e5010_fmt *fmt;
>>> + struct v4l2_pix_format_mplane *pix_mp = &f.fmt.pix_mp;
>>> + struct v4l2_plane_pix_format *plane_fmt = pix_mp->plane_fmt;
>>> + int i = 0;
>>> +
>>> + f.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
>>> + f.fmt.pix_mp.pixelformat = V4L2_PIX_FMT_NV12;
>>> + fmt = find_format(&f);
>>> + queue = &ctx->out_queue;
>>> + queue->fmt = fmt;
>>> + queue->width = DEFAULT_WIDTH;
>>> + queue->height = DEFAULT_HEIGHT;
>>> + pix_mp->width = queue->width;
>>> + pix_mp->height = queue->height;
>>> + v4l2_apply_frmsize_constraints(&pix_mp->width,
>>> + &pix_mp->height,
>>> + &fmt->frmsize);
>>> + v4l2_fill_pixfmt_mp(pix_mp, pix_mp->pixelformat,
>>> + pix_mp->width, pix_mp->height);
>>> + for (i = 0; i < fmt->num_planes; i++) {
>>> + queue->bytesperline[i] = plane_fmt[i].bytesperline;
>>> + queue->sizeimage[i] = plane_fmt[i].sizeimage;
>>> + }
>>> + queue->width_adjusted = pix_mp->width;
>>> + queue->height_adjusted = pix_mp->height;
>>> + queue->format_set = false;
>>> + queue->streaming = false;
>>> +
>>> + f.type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE;
>>> + f.fmt.pix_mp.pixelformat = V4L2_PIX_FMT_JPEG;
>>> + fmt = find_format(&f);
>>> + queue = &ctx->cap_queue;
>>> + queue->fmt = fmt;
>>> + queue->width = DEFAULT_WIDTH;
>>> + queue->height = DEFAULT_HEIGHT;
>>> + pix_mp->width = queue->width;
>>> + pix_mp->height = queue->height;
>>> + v4l2_apply_frmsize_constraints(&pix_mp->width,
>>> + &pix_mp->height,
>>> + &fmt->frmsize);
>>> + queue->sizeimage[0] = pix_mp->width * pix_mp->height *
>>> JPEG_MAX_BYTES_PER_PIXEL;
>>> + queue->sizeimage[0] += HEADER_SIZE;
>>> + queue->sizeimage[1] = 0;
>>> + queue->bytesperline[0] = 0;
>>> + queue->bytesperline[1] = 0;
>>> + queue->format_set = false;
>>> + queue->streaming = false;
>>> + queue->width_adjusted = pix_mp->width;
>>> + queue->height_adjusted = pix_mp->height;
>>> +}
>>> +
>>> +static int e5010_open(struct file *file)
>>> +{
>>> + struct e5010_dev *e5010 = video_drvdata(file);
>>> + struct video_device *vdev = video_devdata(file);
>>> + struct e5010_context *ctx;
>>> + int ret = 0;
>>> +
>>> + ctx = kzalloc(sizeof(*ctx), GFP_KERNEL);
>>> + if (!ctx)
>>> + return -ENOMEM;
>>> +
>>> + if (mutex_lock_interruptible(&e5010->mutex)) {
>>> + ret = -ERESTARTSYS;
>>> + goto free;
>>> + }
>>> +
>>> + v4l2_fh_init(&ctx->fh, vdev);
>>> + file->private_data = ctx;
>>> + v4l2_fh_add(&ctx->fh);
>>> +
>>> + ctx->e5010 = e5010;
>>> + ctx->fh.m2m_ctx = v4l2_m2m_ctx_init(e5010->m2m_dev, ctx,
>>> queue_init);
>>> + if (IS_ERR(ctx->fh.m2m_ctx)) {
>>> + v4l2_err(&e5010->v4l2_dev, "failed to init m2m ctx\n");
>>> + ret = PTR_ERR(ctx->fh.m2m_ctx);
>>> + goto exit;
>>> + }
>>> +
>>> + ret = e5010_ctrls_setup(ctx);
>>> + if (ret) {
>>> + v4l2_err(&e5010->v4l2_dev, "failed to setup e5010 jpeg
>>> controls\n");
>>> + goto err_ctrls_setup;
>>> + }
>>> + ctx->fh.ctrl_handler = &ctx->ctrl_handler;
>>> +
>>> + e5010_jpeg_set_default_params(ctx);
>>> +
>>> + dprintk(e5010, 1, "Created instance: 0x%p, m2m_ctx: 0x%p\n",
>>> ctx, ctx->fh.m2m_ctx);
>>> +
>>> + mutex_unlock(&e5010->mutex);
>>> + return 0;
>>> +
>>> +err_ctrls_setup:
>>> + v4l2_m2m_ctx_release(ctx->fh.m2m_ctx);
>>> +exit:
>>> + v4l2_fh_del(&ctx->fh);
>>> + v4l2_fh_exit(&ctx->fh);
>>> + mutex_unlock(&e5010->mutex);
>>> +free:
>>> + kfree(ctx);
>>> + return ret;
>>> +}
>>> +
>>> +static int e5010_release(struct file *file)
>>> +{
>>> + struct e5010_dev *e5010 = video_drvdata(file);
>>> + struct e5010_context *ctx = file->private_data;
>>> +
>>> + dprintk(e5010, 1, "Releasing instance: 0x%p, m2m_ctx: 0x%p\n",
>>> ctx, ctx->fh.m2m_ctx);
>>> + mutex_lock(&e5010->mutex);
>>> + v4l2_ctrl_handler_free(&ctx->ctrl_handler);
>>> + v4l2_m2m_ctx_release(ctx->fh.m2m_ctx);
>>> + v4l2_fh_del(&ctx->fh);
>>> + v4l2_fh_exit(&ctx->fh);
>>> + kfree(ctx);
>>> + mutex_unlock(&e5010->mutex);
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +static void header_write(struct e5010_context *ctx, u8 *addr,
>>> unsigned int *offset,
>>> + unsigned int no_bytes, unsigned long bits)
>>> +{
>>> + u8 *w_addr = addr + *offset;
>>> + int i;
>>> +
>>> + if ((*offset + no_bytes) > HEADER_SIZE) {
>>> + v4l2_warn(&ctx->e5010->v4l2_dev, "%s: %s: %d: Problem
>>> writing header. %d > HEADER_SIZE %d\n",
>>> + __FILE__, __func__, __LINE__, *offset + no_bytes,
>>> HEADER_SIZE);
>>> + return;
>>> + }
>>> +
>>> + for (i = no_bytes - 1; i >= 0; i--)
>>> + *(w_addr++) = ((u8 *)&bits)[i];
>>> +
>>> + *offset += no_bytes;
>>> +}
>>> +
>>> +static void encode_marker_segment(struct e5010_context *ctx, void
>>> *addr, unsigned int *offset)
>>> +{
>>> + u8 *buffer = (u8 *)addr;
>>> + int i;
>>> +
>>> + header_write(ctx, buffer, offset, 2, START_OF_IMAGE);
>>> + header_write(ctx, buffer, offset, 2, DQT_MARKER);
>>> + header_write(ctx, buffer, offset, 3, LQPQ << 4);
>>> + for (i = 0; i < PELS_IN_BLOCK; i++)
>>> + header_write(ctx, buffer, offset, 1, ctx->luma_qp[zigzag[i]]);
>>> +
>>> + header_write(ctx, buffer, offset, 2, DQT_MARKER);
>>> + header_write(ctx, buffer, offset, 3, (LQPQ << 4) | 1);
>>> + for (i = 0; i < PELS_IN_BLOCK; i++)
>>> + header_write(ctx, buffer, offset, 1,
>>> ctx->chroma_qp[zigzag[i]]);
>>> +
>>> + /* Huffman tables */
>>> + header_write(ctx, buffer, offset, 2, DHT_MARKER);
>>> + header_write(ctx, buffer, offset, 2, LH_DC);
>>> + header_write(ctx, buffer, offset, 1, JPEG_LUM_HT | JPEG_DC_HT);
>>> + for (i = 0 ; i < ARRAY_SIZE(luma_dc_table); i++)
>>> + header_write(ctx, buffer, offset, 1, luma_dc_table[i]);
>>> +
>>> + header_write(ctx, buffer, offset, 2, DHT_MARKER);
>>> + header_write(ctx, buffer, offset, 2, LH_AC);
>>> + header_write(ctx, buffer, offset, 1, JPEG_LUM_HT | JPEG_AC_HT);
>>> + for (i = 0 ; i < ARRAY_SIZE(luma_ac_table); i++)
>>> + header_write(ctx, buffer, offset, 1, luma_ac_table[i]);
>>> +
>>> + header_write(ctx, buffer, offset, 2, DHT_MARKER);
>>> + header_write(ctx, buffer, offset, 2, LH_DC);
>>> + header_write(ctx, buffer, offset, 1, JPEG_CHR_HT | JPEG_DC_HT);
>>> + for (i = 0 ; i < ARRAY_SIZE(chroma_dc_table); i++)
>>> + header_write(ctx, buffer, offset, 1, chroma_dc_table[i]);
>>> +
>>> + header_write(ctx, buffer, offset, 2, DHT_MARKER);
>>> + header_write(ctx, buffer, offset, 2, LH_AC);
>>> + header_write(ctx, buffer, offset, 1, JPEG_CHR_HT | JPEG_AC_HT);
>>> + for (i = 0 ; i < ARRAY_SIZE(chroma_ac_table); i++)
>>> + header_write(ctx, buffer, offset, 1, chroma_ac_table[i]);
>>> +}
>>> +
>>> +static void encode_frame_header(struct e5010_context *ctx, void
>>> *addr, unsigned int *offset)
>>> +{
>>> + u8 *buffer = (u8 *)addr;
>>> +
>>> + header_write(ctx, buffer, offset, 2, SOF_BASELINE_DCT);
>>> + header_write(ctx, buffer, offset, 2, 8 + (3 * UC_NUM_COMP));
>>> + header_write(ctx, buffer, offset, 1, PRECISION);
>>> + header_write(ctx, buffer, offset, 2, ctx->out_queue.height);
>>> + header_write(ctx, buffer, offset, 2, ctx->out_queue.width);
>>> + header_write(ctx, buffer, offset, 1, UC_NUM_COMP);
>>> +
>>> + /* Luma details */
>>> + header_write(ctx, buffer, offset, 1, 1);
>>> + if (ctx->out_queue.fmt->subsampling ==
>>> V4L2_JPEG_CHROMA_SUBSAMPLING_422)
>>> + header_write(ctx, buffer, offset, 1,
>>> + HORZ_SAMPLING_FACTOR | (VERT_SAMPLING_FACTOR_422));
>>> + else
>>> + header_write(ctx, buffer, offset, 1,
>>> + HORZ_SAMPLING_FACTOR | (VERT_SAMPLING_FACTOR_420));
>>> + header_write(ctx, buffer, offset, 1, 0);
>>> + /* Chroma details */
>>> + header_write(ctx, buffer, offset, 1, 2);
>>> + header_write(ctx, buffer, offset, 1, (HORZ_SAMPLING_FACTOR >>
>>> 1) | 1);
>>> + header_write(ctx, buffer, offset, 1, 1);
>>> + header_write(ctx, buffer, offset, 1, 3);
>>> + header_write(ctx, buffer, offset, 1, (HORZ_SAMPLING_FACTOR >>
>>> 1) | 1);
>>> + header_write(ctx, buffer, offset, 1, 1);
>>> +
>>> + header_write(ctx, buffer, offset, 1, 0xFF);
>>> +}
>>> +
>>> +static void jpg_encode_sos_header(struct e5010_context *ctx, void
>>> *addr, unsigned int *offset)
>>> +{
>>> + u8 *buffer = (u8 *)addr;
>>> + int i;
>>> +
>>> + header_write(ctx, buffer, offset, 2, START_OF_SCAN);
>>> + header_write(ctx, buffer, offset, 2, 6 + (COMPONENTS_IN_SCAN <<
>>> 1));
>>> + header_write(ctx, buffer, offset, 1, COMPONENTS_IN_SCAN);
>>> +
>>> + for (i = 0; i < COMPONENTS_IN_SCAN; i++) {
>>> + header_write(ctx, buffer, offset, 1, i + 1);
>>> + if (i == 0)
>>> + header_write(ctx, buffer, offset, 1, 0);
>>> + else
>>> + header_write(ctx, buffer, offset, 1, 17);
>>> + }
>>> +
>>> + header_write(ctx, buffer, offset, 1, 0);
>>> + header_write(ctx, buffer, offset, 1, 63);
>>> + header_write(ctx, buffer, offset, 1, 0);
>>> +}
>>> +
>>> +static void write_header(struct e5010_context *ctx, void *addr)
>>> +{
>>> + unsigned int offset = 0;
>>> +
>>> + encode_marker_segment(ctx, addr, &offset);
>>> + encode_frame_header(ctx, addr, &offset);
>>> + jpg_encode_sos_header(ctx, addr, &offset);
>>> +}
>>> +
>>> +static irqreturn_t e5010_irq(int irq, void *data)
>>> +{
>>> + struct e5010_dev *e5010 = data;
>>> + struct e5010_context *ctx;
>>> + int output_size;
>>> + struct vb2_v4l2_buffer *src_buf, *dst_buf;
>>> + bool pic_done, out_addr_err;
>>> +
>>> + spin_lock(&e5010->hw_lock);
>>> + pic_done = e5010_hw_pic_done_irq(e5010->core_base);
>>> + out_addr_err = e5010_hw_output_address_irq(e5010->core_base);
>>> +
>>> + if (!pic_done && !out_addr_err) {
>>> + spin_unlock(&e5010->hw_lock);
>>> + return IRQ_NONE;
>>> + }
>>> +
>>> + ctx = v4l2_m2m_get_curr_priv(e5010->m2m_dev);
>>> + if (WARN_ON(!ctx))
>>> + goto job_unlock;
>>> +
>>> + dst_buf = v4l2_m2m_dst_buf_remove(ctx->fh.m2m_ctx);
>>> + src_buf = v4l2_m2m_src_buf_remove(ctx->fh.m2m_ctx);
>>> + if (!dst_buf || !src_buf) {
>>> + v4l2_err(&e5010->v4l2_dev, "ctx: 0x%p No source or
>>> destination buffer\n", ctx);
>>> + goto job_unlock;
>>> + }
>>> +
>>> + if (out_addr_err) {
>>> + e5010_hw_clear_output_error(e5010->core_base, 1);
>>> + v4l2_warn(&e5010->v4l2_dev,
>>> + "ctx: 0x%p Output bitstream size exceeded max
>>> size\n", ctx);
>>> + v4l2_m2m_buf_done(src_buf, VB2_BUF_STATE_ERROR);
>>> + vb2_set_plane_payload(&dst_buf->vb2_buf, 0,
>>> dst_buf->planes[0].length);
>>> + v4l2_m2m_buf_done(dst_buf, VB2_BUF_STATE_ERROR);
>>> + if (v4l2_m2m_is_last_draining_src_buf(ctx->fh.m2m_ctx,
>>> src_buf)) {
>>> + dst_buf->flags |= V4L2_BUF_FLAG_LAST;
>>> + v4l2_m2m_mark_stopped(ctx->fh.m2m_ctx);
>>> + v4l2_event_queue_fh(&ctx->fh, &e5010_eos_event);
>>> + dprintk(e5010, 2, "ctx: 0x%p Sending EOS\n", ctx);
>>> + }
>>> + }
>>> +
>>> + if (pic_done) {
>>> + e5010_hw_clear_picture_done(e5010->core_base, 1);
>>> + dprintk(e5010, 3, "ctx: 0x%p Got output bitstream of size
>>> %d bytes\n",
>>> + ctx, readl(e5010->core_base + JASPER_OUTPUT_SIZE_OFFSET));
>>> +
>>> + if (v4l2_m2m_is_last_draining_src_buf(ctx->fh.m2m_ctx,
>>> src_buf)) {
>>> + dst_buf->flags |= V4L2_BUF_FLAG_LAST;
>>> + v4l2_m2m_mark_stopped(ctx->fh.m2m_ctx);
>>> + v4l2_event_queue_fh(&ctx->fh, &e5010_eos_event);
>>> + dprintk(e5010, 2, "ctx: 0x%p Sending EOS\n", ctx);
>>> + }
>>> + v4l2_m2m_buf_done(src_buf, VB2_BUF_STATE_DONE);
>>> + output_size = e5010_hw_get_output_size(e5010->core_base);
>>> + vb2_set_plane_payload(&dst_buf->vb2_buf, 0, output_size +
>>> HEADER_SIZE);
>>> + v4l2_m2m_buf_done(dst_buf, VB2_BUF_STATE_DONE);
>>> + dprintk(e5010, 3,
>>> + "ctx: 0x%p frame done for dst_buf->sequence: %d
>>> src_buf->sequence: %d\n",
>>> + ctx, dst_buf->sequence, src_buf->sequence);
>>> + }
>>> +
>>> + v4l2_m2m_job_finish(e5010->m2m_dev, ctx->fh.m2m_ctx);
>>> + dprintk(e5010, 3, "ctx: 0x%p Finish job\n", ctx);
>>
>> That is a lot to do in interrupt handler, you should consider using
>> threaded irq for all this.
>>
>
> I had preferred to go with hard irq since there were no iterative code
> added and no sleeping function calls made in ISR (so it did not look
> quite a lot to me :) ), and besides I didn't see any issues/unexpected
> perfs with multi-stream tests [1].
>
> Also I referred some existing jpeg drivers [2] which used hard irq
> with similar calls as mine and they too seem to be quite stable.
>
> [1] :
> https://gist.github.com/devarsht/58af9ccf822963d2bf8f0c2f4481438a
>
> [2] :
> https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git/tree/drivers/media/platform/renesas/rcar_jpu.c?h=v6.6#n1606
>
>
> https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git/tree/drivers/media/platform/samsung/s5p-jpeg/jpeg-core.c?h=v6.6#n2876
>
>
> Regards
> Devarsh
>
>> Regards,
>> Benjamin
>>
>>> +
>>> +job_unlock:
>>> + spin_unlock(&e5010->hw_lock);
>>> + return IRQ_HANDLED;
>>> +}
>>> +
>>> +static int e5010_init_device(struct e5010_dev *e5010)
>>> +{
>>> + int ret = 0;
>>> +
>>> + /*TODO: Set MMU in bypass mode until support for the same is
>>> added in driver*/
>>> + e5010_hw_bypass_mmu(e5010->mmu_base, 1);
>>> +
>>> + if (e5010_hw_enable_auto_clock_gating(e5010->core_base, 1))
>>> + v4l2_warn(&e5010->v4l2_dev, "failed to enable auto clock
>>> gating\n");
>>> +
>>> + if (e5010_hw_enable_manual_clock_gating(e5010->core_base, 0))
>>> + v4l2_warn(&e5010->v4l2_dev, "failed to disable manual clock
>>> gating\n");
>>> +
>>> + if (e5010_hw_enable_crc_check(e5010->core_base, 0))
>>> + v4l2_warn(&e5010->v4l2_dev, "failed to disable CRC check\n");
>>> +
>>> + if (e5010_hw_enable_output_address_error_irq(e5010->core_base, 1))
>>> + v4l2_err(&e5010->v4l2_dev, "failed to enable Output Address
>>> Error interrupts\n");
>>> +
>>> + ret = e5010_hw_set_input_source_to_memory(e5010->core_base, 1);
>>> + if (ret) {
>>> + v4l2_err(&e5010->v4l2_dev, "failed to set input source to
>>> memory\n");
>>> + return ret;
>>> + }
>>> +
>>> + ret = e5010_hw_enable_picture_done_irq(e5010->core_base, 1);
>>> + if (ret)
>>> + v4l2_err(&e5010->v4l2_dev, "failed to enable Picture Done
>>> interrupts\n");
>>> +
>>> + return ret;
>>> +}
>>> +
>>> +static int e5010_probe(struct platform_device *pdev)
>>> +{
>>> + struct e5010_dev *e5010;
>>> + int irq, ret = 0;
>>> +
>>> + ret = dma_set_mask(&pdev->dev, DMA_BIT_MASK(32));
>>> + if (ret)
>>> + return dev_err_probe(e5010->dev, ret, "32-bit consistent
>>> DMA enable failed\n");
>>> +
>>> + e5010 = devm_kzalloc(&pdev->dev, sizeof(*e5010), GFP_KERNEL);
>>> + if (!e5010)
>>> + return -ENOMEM;
>>> +
>>> + platform_set_drvdata(pdev, e5010);
>>> +
>>> + e5010->dev = &pdev->dev;
>>> +
>>> + mutex_init(&e5010->mutex);
>>> + spin_lock_init(&e5010->hw_lock);
>>> +
>>> + e5010->vdev = video_device_alloc();
>>> + if (!e5010->vdev) {
>>> + dev_err(e5010->dev, "failed to allocate video device\n");
>>> + return -ENOMEM;
>>> + }
>>> +
>>> + snprintf(e5010->vdev->name, sizeof(e5010->vdev->name), "%s",
>>> E5010_MODULE_NAME);
>>> + e5010->vdev->fops = &e5010_fops;
>>> + e5010->vdev->ioctl_ops = &e5010_ioctl_ops;
>>> + e5010->vdev->minor = -1;
>>> + e5010->vdev->release = video_device_release;
>>> + e5010->vdev->vfl_dir = VFL_DIR_M2M;
>>> + e5010->vdev->device_caps = V4L2_CAP_VIDEO_M2M_MPLANE |
>>> V4L2_CAP_STREAMING;
>>> + e5010->vdev->v4l2_dev = &e5010->v4l2_dev;
>>> + e5010->vdev->lock = &e5010->mutex;
>>> +
>>> + ret = v4l2_device_register(&pdev->dev, &e5010->v4l2_dev);
>>> + if (ret)
>>> + return dev_err_probe(e5010->dev, ret, "failed to register
>>> v4l2 device\n");
>>> +
>>> + e5010->m2m_dev = v4l2_m2m_init(&e5010_m2m_ops);
>>> + if (IS_ERR(e5010->m2m_dev)) {
>>> + ret = PTR_ERR(e5010->m2m_dev);
>>> + e5010->m2m_dev = NULL;
>>> + dev_err_probe(e5010->dev, ret, "failed to init mem2mem
>>> device\n");
>>> + goto fail_after_v4l2_register;
>>> + }
>>> +
>>> + video_set_drvdata(e5010->vdev, e5010);
>>> +
>>> + e5010->core_base = devm_platform_ioremap_resource_byname(pdev,
>>> "core");
>>> + if (IS_ERR(e5010->core_base)) {
>>> + ret = PTR_ERR(e5010->core_base);
>>> + dev_err_probe(e5010->dev, ret, "Missing 'core' resources
>>> area\n");
>>> + goto fail_after_v4l2_register;
>>> + }
>>> +
>>> + e5010->mmu_base = devm_platform_ioremap_resource_byname(pdev,
>>> "mmu");
>>> + if (IS_ERR(e5010->mmu_base)) {
>>> + ret = PTR_ERR(e5010->mmu_base);
>>> + dev_err_probe(e5010->dev, ret, "Missing 'mmu' resources
>>> area\n");
>>> + goto fail_after_v4l2_register;
>>> + }
>>> +
>>> + e5010->last_context_run = NULL;
>>> +
>>> + irq = platform_get_irq(pdev, 0);
>>> + ret = devm_request_irq(e5010->dev, irq, e5010_irq, 0,
>>> + E5010_MODULE_NAME, e5010);
>>> + if (ret) {
>>> + dev_err_probe(e5010->dev, ret, "failed to register IRQ
>>> %d\n", irq);
>>> + goto fail_after_v4l2_register;
>>> + }
>>> +
>>> + e5010->clk = devm_clk_get(&pdev->dev, NULL);
>>> + if (IS_ERR(e5010->clk)) {
>>> + ret = PTR_ERR(e5010->clk);
>>> + dev_err_probe(e5010->dev, ret, "failed to get clock\n");
>>> + goto fail_after_v4l2_register;
>>> + }
>>> +
>>> + pm_runtime_enable(e5010->dev);
>>> +
>>> + ret = video_register_device(e5010->vdev, VFL_TYPE_VIDEO, 0);
>>> + if (ret) {
>>> + dev_err_probe(e5010->dev, ret, "failed to register video
>>> device\n");
>>> + goto fail_after_video_register_device;
>>> + }
>>> +
>>> + v4l2_info(&e5010->v4l2_dev, "Device registered as /dev/video%d\n",
>>> + e5010->vdev->num);
>>> +
>>> + return 0;
>>> +
>>> +fail_after_video_register_device:
>>> + v4l2_m2m_release(e5010->m2m_dev);
>>> +fail_after_v4l2_register:
>>> + v4l2_device_unregister(&e5010->v4l2_dev);
>>> + return ret;
>>> +}
>>> +
>>> +static int e5010_remove(struct platform_device *pdev)
>>> +{
>>> + struct e5010_dev *e5010 = platform_get_drvdata(pdev);
>>> +
>>> + pm_runtime_disable(e5010->dev);
>>> + video_unregister_device(e5010->vdev);
>>> + v4l2_m2m_release(e5010->m2m_dev);
>>> + v4l2_device_unregister(&e5010->v4l2_dev);
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +static void e5010_vb2_buffers_return(struct vb2_queue *q, enum
>>> vb2_buffer_state state)
>>> +{
>>> + struct vb2_v4l2_buffer *vbuf;
>>> + struct e5010_context *ctx = vb2_get_drv_priv(q);
>>> +
>>> + if (V4L2_TYPE_IS_OUTPUT(q->type)) {
>>> + while ((vbuf = v4l2_m2m_src_buf_remove(ctx->fh.m2m_ctx))) {
>>> + dprintk(ctx->e5010, 2, "ctx: 0x%p, buf type %s | index
>>> %d\n",
>>> + ctx, type_name(vbuf->vb2_buf.type),
>>> vbuf->vb2_buf.index);
>>> + v4l2_m2m_buf_done(vbuf, state);
>>> + }
>>> + } else {
>>> + while ((vbuf = v4l2_m2m_dst_buf_remove(ctx->fh.m2m_ctx))) {
>>> + dprintk(ctx->e5010, 2, "ctx: 0x%p, buf type %s | index
>>> %d\n",
>>> + ctx, type_name(vbuf->vb2_buf.type),
>>> vbuf->vb2_buf.index);
>>> + vb2_set_plane_payload(&vbuf->vb2_buf, 0, 0);
>>> + v4l2_m2m_buf_done(vbuf, state);
>>> + }
>>> + }
>>> +}
>>> +
>>> +static int e5010_queue_setup(struct vb2_queue *vq, unsigned int
>>> *nbuffers, unsigned int *nplanes,
>>> + unsigned int sizes[], struct device *alloc_devs[])
>>> +{
>>> + struct e5010_context *ctx = vb2_get_drv_priv(vq);
>>> + struct e5010_q_data *queue;
>>> + int i;
>>> +
>>> + queue = get_queue(ctx, vq->type);
>>> +
>>> + if (*nplanes) {
>>> + if (*nplanes != queue->fmt->num_planes)
>>> + return -EINVAL;
>>> + for (i = 0; i < *nplanes; i++) {
>>> + if (sizes[i] < queue->sizeimage[i])
>>> + return -EINVAL;
>>> + }
>>> + return 0;
>>> + }
>>> +
>>> + *nplanes = queue->fmt->num_planes;
>>> + for (i = 0; i < *nplanes; i++)
>>> + sizes[i] = queue->sizeimage[i];
>>> +
>>> + dprintk(ctx->e5010, 2,
>>> + "ctx: 0x%p, type %s, buffer(s): %d, planes %d, plane1:
>>> bytes %d plane2: %d bytes\n",
>>> + ctx, type_name(vq->type), *nbuffers, *nplanes, sizes[0],
>>> sizes[1]);
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +static void e5010_buf_finish(struct vb2_buffer *vb)
>>> +{
>>> + struct e5010_context *ctx = vb2_get_drv_priv(vb->vb2_queue);
>>> + void *d_addr;
>>> +
>>> + if (vb->state != VB2_BUF_STATE_DONE ||
>>> V4L2_TYPE_IS_OUTPUT(vb->vb2_queue->type))
>>> + return;
>>> +
>>> + d_addr = vb2_plane_vaddr(vb, 0);
>>> + write_header(ctx, d_addr);
>>> +}
>>> +
>>> +static int e5010_buf_out_validate(struct vb2_buffer *vb)
>>> +{
>>> + struct vb2_v4l2_buffer *vbuf = to_vb2_v4l2_buffer(vb);
>>> + struct e5010_context *ctx = vb2_get_drv_priv(vb->vb2_queue);
>>> +
>>> + if (vbuf->field != V4L2_FIELD_NONE)
>>> + dprintk(ctx->e5010, 1, "ctx: 0x%p, field isn't
>>> supported\n", ctx);
>>> +
>>> + vbuf->field = V4L2_FIELD_NONE;
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +static int e5010_buf_prepare(struct vb2_buffer *vb)
>>> +{
>>> + struct e5010_context *ctx = vb2_get_drv_priv(vb->vb2_queue);
>>> + struct vb2_v4l2_buffer *vbuf = to_vb2_v4l2_buffer(vb);
>>> + struct e5010_q_data *queue;
>>> + int i;
>>> +
>>> + vbuf->field = V4L2_FIELD_NONE;
>>> +
>>> + queue = get_queue(ctx, vb->vb2_queue->type);
>>> +
>>> + for (i = 0; i < queue->fmt->num_planes; i++) {
>>> + if (vb2_plane_size(vb, i) < (unsigned
>>> long)queue->sizeimage[i]) {
>>> + v4l2_err(&ctx->e5010->v4l2_dev, "plane %d too small
>>> (%lu < %lu)", i,
>>> + vb2_plane_size(vb, i), (unsigned
>>> long)queue->sizeimage[i]);
>>> +
>>> + return -EINVAL;
>>> + }
>>> + }
>>> +
>>> + if (V4L2_TYPE_IS_CAPTURE(vb->vb2_queue->type)) {
>>> + vb2_set_plane_payload(vb, 0, 0);
>>> + vb2_set_plane_payload(vb, 1, 0);
>>> + }
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +static void e5010_buf_queue(struct vb2_buffer *vb)
>>> +{
>>> + struct e5010_context *ctx = vb2_get_drv_priv(vb->vb2_queue);
>>> + struct vb2_v4l2_buffer *vbuf = to_vb2_v4l2_buffer(vb);
>>> +
>>> + if (V4L2_TYPE_IS_CAPTURE(vb->vb2_queue->type) &&
>>> + vb2_is_streaming(vb->vb2_queue) &&
>>> + v4l2_m2m_dst_buf_is_last(ctx->fh.m2m_ctx)) {
>>> + struct e5010_q_data *queue = get_queue(ctx,
>>> V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE);
>>> +
>>> + vbuf->sequence = queue->sequence++;
>>> + v4l2_m2m_last_buffer_done(ctx->fh.m2m_ctx, vbuf);
>>> + v4l2_event_queue_fh(&ctx->fh, &e5010_eos_event);
>>> + return;
>>> + }
>>> +
>>> + v4l2_m2m_buf_queue(ctx->fh.m2m_ctx, vbuf);
>>> +}
>>> +
>>> +static int e5010_encoder_cmd(struct file *file, void *priv,
>>> + struct v4l2_encoder_cmd *cmd)
>>> +{
>>> + struct e5010_context *ctx = file->private_data;
>>> + int ret;
>>> + struct vb2_queue *cap_vq;
>>> +
>>> + cap_vq = v4l2_m2m_get_vq(ctx->fh.m2m_ctx,
>>> V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE);
>>> +
>>> + ret = v4l2_m2m_ioctl_try_encoder_cmd(file, &ctx->fh, cmd);
>>> + if (ret < 0)
>>> + return ret;
>>> +
>>> + if (!vb2_is_streaming(v4l2_m2m_get_src_vq(ctx->fh.m2m_ctx)) ||
>>> + !vb2_is_streaming(v4l2_m2m_get_dst_vq(ctx->fh.m2m_ctx)))
>>> + return 0;
>>> +
>>> + ret = v4l2_m2m_ioctl_encoder_cmd(file, &ctx->fh, cmd);
>>> + if (ret < 0)
>>> + return ret;
>>> +
>>> + if (cmd->cmd == V4L2_ENC_CMD_STOP &&
>>> + v4l2_m2m_has_stopped(ctx->fh.m2m_ctx))
>>> + v4l2_event_queue_fh(&ctx->fh, &e5010_eos_event);
>>> +
>>> + if (cmd->cmd == V4L2_ENC_CMD_START &&
>>> + v4l2_m2m_has_stopped(ctx->fh.m2m_ctx))
>>> + vb2_clear_last_buffer_dequeued(cap_vq);
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +static int e5010_start_streaming(struct vb2_queue *q, unsigned int
>>> count)
>>> +{
>>> + struct e5010_context *ctx = vb2_get_drv_priv(q);
>>> + int ret;
>>> +
>>> + struct e5010_q_data *queue = get_queue(ctx, q->type);
>>> +
>>> + queue->streaming = true;
>>> + v4l2_m2m_update_start_streaming_state(ctx->fh.m2m_ctx, q);
>>> + queue->sequence = 0;
>>> +
>>> + ret = pm_runtime_resume_and_get(ctx->e5010->dev);
>>> + if (ret < 0) {
>>> + v4l2_err(&ctx->e5010->v4l2_dev, "failed to power up jpeg\n");
>>> + goto fail;
>>> + }
>>> +
>>> + ret = e5010_init_device(ctx->e5010);
>>> + if (ret) {
>>> + v4l2_err(&ctx->e5010->v4l2_dev, "failed to Enable e5010
>>> device\n");
>>> + goto fail;
>>> + }
>>> +
>>> + return 0;
>>> +
>>> +fail:
>>> + e5010_vb2_buffers_return(q, VB2_BUF_STATE_QUEUED);
>>> +
>>> + return ret;
>>> +}
>>> +
>>> +static void e5010_stop_streaming(struct vb2_queue *q)
>>> +{
>>> + struct e5010_context *ctx = vb2_get_drv_priv(q);
>>> + struct e5010_q_data *queue;
>>> +
>>> + queue = get_queue(ctx, q->type);
>>> +
>>> + queue->streaming = false;
>>> +
>>> + e5010_vb2_buffers_return(q, VB2_BUF_STATE_ERROR);
>>> +
>>> + if (V4L2_TYPE_IS_OUTPUT(q->type))
>>> + v4l2_m2m_update_stop_streaming_state(ctx->fh.m2m_ctx, q);
>>> +
>>> + if (V4L2_TYPE_IS_OUTPUT(q->type) &&
>>> + v4l2_m2m_has_stopped(ctx->fh.m2m_ctx)) {
>>> + v4l2_event_queue_fh(&ctx->fh, &e5010_eos_event);
>>> + }
>>> +
>>> + pm_runtime_put_sync(ctx->e5010->dev);
>>> +}
>>> +
>>> +static void e5010_device_run(void *priv)
>>> +{
>>> + struct e5010_context *ctx = priv;
>>> + struct e5010_dev *e5010 = ctx->e5010;
>>> + struct vb2_v4l2_buffer *s_vb, *d_vb;
>>> + u32 reg = 0;
>>> + int ret = 0;
>>> + unsigned long flags;
>>> + int num_planes = ctx->out_queue.fmt->num_planes;
>>> +
>>> + spin_lock_irqsave(&e5010->hw_lock, flags);
>>> + s_vb = v4l2_m2m_next_src_buf(ctx->fh.m2m_ctx);
>>> + WARN_ON(!s_vb);
>>> + d_vb = v4l2_m2m_next_dst_buf(ctx->fh.m2m_ctx);
>>> + WARN_ON(!d_vb);
>>> + if (!s_vb || !d_vb)
>>> + goto no_ready_buf_err;
>>> +
>>> + s_vb->sequence = ctx->out_queue.sequence++;
>>> + d_vb->sequence = ctx->cap_queue.sequence++;
>>> +
>>> + v4l2_m2m_buf_copy_metadata(s_vb, d_vb, false);
>>> +
>>> + if (ctx != e5010->last_context_run || ctx->update_qp) {
>>> + dprintk(e5010, 1, "ctx updated: 0x%p -> 0x%p, updating qp
>>> tables\n",
>>> + e5010->last_context_run, ctx);
>>> + ret = update_qp_tables(ctx);
>>> + }
>>> +
>>> + if (ret) {
>>> + ctx->update_qp = true;
>>> + v4l2_err(&e5010->v4l2_dev, "failed to update QP tables\n");
>>> + goto device_busy_err;
>>> + } else {
>>> + e5010->last_context_run = ctx;
>>> + ctx->update_qp = false;
>>> + }
>>> +
>>> + /* Set I/O Buffer addresses */
>>> + reg = (u32)vb2_dma_contig_plane_dma_addr(&s_vb->vb2_buf, 0);
>>> + ret = e5010_hw_set_input_luma_addr(e5010->core_base, reg);
>>> + if (ret || !reg) {
>>> + v4l2_err(&e5010->v4l2_dev, "failed to set input luma
>>> address\n");
>>> + goto device_busy_err;
>>> + }
>>> +
>>> + if (num_planes == 1)
>>> + reg += (ctx->out_queue.bytesperline[0]) *
>>> (ctx->out_queue.height);
>>> + else
>>> + reg = (u32)vb2_dma_contig_plane_dma_addr(&s_vb->vb2_buf, 1);
>>> +
>>> + dprintk(e5010, 3,
>>> + "ctx: 0x%p, luma_addr: 0x%x, chroma_addr: 0x%x, out_addr:
>>> 0x%x\n",
>>> + ctx, (u32)vb2_dma_contig_plane_dma_addr(&s_vb->vb2_buf, 0),
>>> reg,
>>> + (u32)vb2_dma_contig_plane_dma_addr(&d_vb->vb2_buf, 0));
>>> +
>>> + dprintk(e5010, 3,
>>> + "ctx: 0x%p, buf indices: src_index: %d, dst_index: %d\n",
>>> + ctx, s_vb->vb2_buf.index, d_vb->vb2_buf.index);
>>> +
>>> + ret = e5010_hw_set_input_chroma_addr(e5010->core_base, reg);
>>> + if (ret || !reg) {
>>> + v4l2_err(&e5010->v4l2_dev, "failed to set input chroma
>>> address\n");
>>> + goto device_busy_err;
>>> + }
>>> +
>>> + reg = (u32)vb2_dma_contig_plane_dma_addr(&d_vb->vb2_buf, 0);
>>> + reg += HEADER_SIZE;
>>> + ret = e5010_hw_set_output_base_addr(e5010->core_base, reg);
>>> + if (ret || !reg) {
>>> + v4l2_err(&e5010->v4l2_dev, "failed to set output base
>>> address\n");
>>> + goto device_busy_err;
>>> + }
>>> +
>>> + /* Set input settings */
>>> + ret = e5010_hw_set_horizontal_size(e5010->core_base,
>>> ctx->out_queue.width - 1);
>>> + if (ret) {
>>> + v4l2_err(&e5010->v4l2_dev, "failed to set input width\n");
>>> + goto device_busy_err;
>>> + }
>>> +
>>> + ret = e5010_hw_set_vertical_size(e5010->core_base,
>>> ctx->out_queue.height - 1);
>>> + if (ret) {
>>> + v4l2_err(&e5010->v4l2_dev, "failed to set input width\n");
>>> + goto device_busy_err;
>>> + }
>>> +
>>> + ret = e5010_hw_set_luma_stride(e5010->core_base,
>>> ctx->out_queue.bytesperline[0]);
>>> + if (ret) {
>>> + v4l2_err(&e5010->v4l2_dev, "failed to set luma stride\n");
>>> + goto device_busy_err;
>>> + }
>>> +
>>> + ret = e5010_hw_set_chroma_stride(e5010->core_base,
>>> ctx->out_queue.bytesperline[0]);
>>> + if (ret) {
>>> + v4l2_err(&e5010->v4l2_dev, "failed to set chroma stride\n");
>>> + goto device_busy_err;
>>> + }
>>> +
>>> + ret = e5010_set_input_subsampling(e5010->core_base,
>>> ctx->out_queue.fmt->subsampling);
>>> + if (ret) {
>>> + v4l2_err(&e5010->v4l2_dev, "failed to set input
>>> subsampling\n");
>>> + goto device_busy_err;
>>> + }
>>> +
>>> + ret = e5010_hw_set_chroma_order(e5010->core_base,
>>> ctx->out_queue.fmt->chroma_order);
>>> + if (ret) {
>>> + v4l2_err(&e5010->v4l2_dev, "failed to set chroma order\n");
>>> + goto device_busy_err;
>>> + }
>>> +
>>> + e5010_hw_set_output_max_size(e5010->core_base,
>>> d_vb->planes[0].length);
>>> + e5010_hw_encode_start(e5010->core_base, 1);
>>> +
>>> + spin_unlock_irqrestore(&e5010->hw_lock, flags);
>>> +
>>> + return;
>>> +
>>> +device_busy_err:
>>> + e5010_reset(e5010->dev, e5010->core_base, e5010->mmu_base);
>>> +
>>> +no_ready_buf_err:
>>> + if (s_vb) {
>>> + v4l2_m2m_src_buf_remove_by_buf(ctx->fh.m2m_ctx, s_vb);
>>> + v4l2_m2m_buf_done(s_vb, VB2_BUF_STATE_ERROR);
>>> + }
>>> +
>>> + if (d_vb) {
>>> + v4l2_m2m_dst_buf_remove_by_buf(ctx->fh.m2m_ctx, d_vb);
>>> + /* Payload set to 1 since 0 payload can trigger EOS */
>>> + vb2_set_plane_payload(&d_vb->vb2_buf, 0, 1);
>>> + v4l2_m2m_buf_done(d_vb, VB2_BUF_STATE_ERROR);
>>> + }
>>> + v4l2_m2m_job_finish(e5010->m2m_dev, ctx->fh.m2m_ctx);
>>> + spin_unlock_irqrestore(&e5010->hw_lock, flags);
>>> +}
>>> +
>>> +#ifdef CONFIG_PM
>>> +static int e5010_runtime_resume(struct device *dev)
>>> +{
>>> + struct e5010_dev *e5010 = dev_get_drvdata(dev);
>>> + int ret;
>>> +
>>> + ret = clk_prepare_enable(e5010->clk);
>>> + if (ret < 0) {
>>> + v4l2_err(&e5010->v4l2_dev, "failed to enable clock\n");
>>> + return ret;
>>> + }
>>> +
>>> + return 0;
>>> +}
>>> +
>>> +static int e5010_runtime_suspend(struct device *dev)
>>> +{
>>> + struct e5010_dev *e5010 = dev_get_drvdata(dev);
>>> +
>>> + clk_disable_unprepare(e5010->clk);
>>> +
>>> + return 0;
>>> +}
>>> +#endif
>>> +
>>> +#ifdef CONFIG_PM_SLEEP
>>> +static int e5010_suspend(struct device *dev)
>>> +{
>>> + struct e5010_dev *e5010 = dev_get_drvdata(dev);
>>> +
>>> + v4l2_m2m_suspend(e5010->m2m_dev);
>>> +
>>> + return pm_runtime_force_suspend(dev);
>>> +}
>>> +
>>> +static int e5010_resume(struct device *dev)
>>> +{
>>> + struct e5010_dev *e5010 = dev_get_drvdata(dev);
>>> + int ret;
>>> +
>>> + ret = pm_runtime_force_resume(dev);
>>> + if (ret < 0)
>>> + return ret;
>>> +
>>> + v4l2_m2m_resume(e5010->m2m_dev);
>>> +
>>> + return ret;
>>> +}
>>> +#endif
>>> +
>>> +static const struct dev_pm_ops e5010_pm_ops = {
>>> + SET_RUNTIME_PM_OPS(e5010_runtime_suspend,
>>> + e5010_runtime_resume, NULL)
>>> + SET_SYSTEM_SLEEP_PM_OPS(e5010_suspend, e5010_resume)
>>> +};
>>> +
>>> +static const struct v4l2_ioctl_ops e5010_ioctl_ops = {
>>> + .vidioc_querycap = e5010_querycap,
>>> +
>>> + .vidioc_enum_fmt_vid_cap = e5010_enum_fmt,
>>> + .vidioc_g_fmt_vid_cap_mplane = e5010_g_fmt,
>>> + .vidioc_try_fmt_vid_cap_mplane = e5010_try_fmt,
>>> + .vidioc_s_fmt_vid_cap_mplane = e5010_s_fmt,
>>> +
>>> + .vidioc_enum_fmt_vid_out = e5010_enum_fmt,
>>> + .vidioc_g_fmt_vid_out_mplane = e5010_g_fmt,
>>> + .vidioc_try_fmt_vid_out_mplane = e5010_try_fmt,
>>> + .vidioc_s_fmt_vid_out_mplane = e5010_s_fmt,
>>> +
>>> + .vidioc_g_selection = e5010_g_selection,
>>> + .vidioc_s_selection = e5010_s_selection,
>>> +
>>> + .vidioc_reqbufs = v4l2_m2m_ioctl_reqbufs,
>>> + .vidioc_querybuf = v4l2_m2m_ioctl_querybuf,
>>> + .vidioc_qbuf = v4l2_m2m_ioctl_qbuf,
>>> + .vidioc_dqbuf = v4l2_m2m_ioctl_dqbuf,
>>> + .vidioc_expbuf = v4l2_m2m_ioctl_expbuf,
>>> + .vidioc_create_bufs = v4l2_m2m_ioctl_create_bufs,
>>> + .vidioc_prepare_buf = v4l2_m2m_ioctl_prepare_buf,
>>> +
>>> + .vidioc_streamon = v4l2_m2m_ioctl_streamon,
>>> + .vidioc_streamoff = v4l2_m2m_ioctl_streamoff,
>>> + .vidioc_log_status = v4l2_ctrl_log_status,
>>> +
>>> + .vidioc_subscribe_event = e5010_subscribe_event,
>>> + .vidioc_unsubscribe_event = v4l2_event_unsubscribe,
>>> + .vidioc_try_encoder_cmd = v4l2_m2m_ioctl_try_encoder_cmd,
>>> + .vidioc_encoder_cmd = e5010_encoder_cmd,
>>> +
>>> + .vidioc_enum_framesizes = e5010_enum_framesizes,
>>> +};
>>> +
>>> +static const struct vb2_ops e5010_video_ops = {
>>> + .queue_setup = e5010_queue_setup,
>>> + .buf_queue = e5010_buf_queue,
>>> + .buf_finish = e5010_buf_finish,
>>> + .buf_prepare = e5010_buf_prepare,
>>> + .buf_out_validate = e5010_buf_out_validate,
>>> + .wait_prepare = vb2_ops_wait_prepare,
>>> + .wait_finish = vb2_ops_wait_finish,
>>> + .start_streaming = e5010_start_streaming,
>>> + .stop_streaming = e5010_stop_streaming,
>>> +};
>>> +
>>> +static const struct v4l2_file_operations e5010_fops = {
>>> + .owner = THIS_MODULE,
>>> + .open = e5010_open,
>>> + .release = e5010_release,
>>> + .poll = v4l2_m2m_fop_poll,
>>> + .unlocked_ioctl = video_ioctl2,
>>> + .mmap = v4l2_m2m_fop_mmap,
>>> +};
>>> +
>>> +static const struct v4l2_m2m_ops e5010_m2m_ops = {
>>> + .device_run = e5010_device_run,
>>> +};
>>> +
>>> +static const struct of_device_id e5010_of_match[] = {
>>> + {.compatible = "img,e5010-jpeg-enc"}, { /* end */},
>>> +};
>>> +MODULE_DEVICE_TABLE(of, e5010_of_match);
>>> +
>>> +static struct platform_driver e5010_driver = {
>>> + .probe = e5010_probe,
>>> + .remove = e5010_remove,
>>> + .driver = {
>>> + .name = E5010_MODULE_NAME,
>>> + .of_match_table = e5010_of_match,
>>> + .pm = &e5010_pm_ops,
>>> + },
>>> +};
>>> +module_platform_driver(e5010_driver);
>>> +
>>> +MODULE_LICENSE("GPL");
>>> +MODULE_DESCRIPTION("Imagination E5010 JPEG encoder driver");
>>> diff --git a/drivers/media/platform/imagination/e5010-jpeg-enc.h
>>> b/drivers/media/platform/imagination/e5010-jpeg-enc.h
>>> new file mode 100644
>>> index 000000000000..3a47b0327101
>>> --- /dev/null
>>> +++ b/drivers/media/platform/imagination/e5010-jpeg-enc.h
>>> @@ -0,0 +1,169 @@
>>> +/* SPDX-License-Identifier: GPL-2.0 */
>>> +/*
>>> + * Imagination E5010 JPEG Encoder driver.
>>> + *
>>> + * Copyright (C) 2023 Texas Instruments Incorporated -
>>> https://www.ti.com/
>>> + *
>>> + * Author: David Huang <d-huang@...com>
>>> + * Author: Devarsh Thakkar <devarsht@...com>
>>> + */
>>> +
>>> +#include <media/v4l2-ctrls.h>
>>> +#include <media/v4l2-device.h>
>>> +#include <media/v4l2-fh.h>
>>> +
>>> +#ifndef _E5010_JPEG_ENC_H
>>> +#define _E5010_JPEG_ENC_H
>>> +
>>> +#define MAX_PLANES 2
>>> +#define HEADER_SIZE 0x025E
>>> +#define MIN_DIMENSION 64
>>> +#define MAX_DIMENSION 8192
>>> +#define DEFAULT_WIDTH 640
>>> +#define DEFAULT_HEIGHT 480
>>> +#define E5010_MODULE_NAME "e5010"
>>> +#define JPEG_MAX_BYTES_PER_PIXEL 2
>>> +
>>> +/* JPEG marker definitions */
>>> +#define START_OF_IMAGE 0xFFD8
>>> +#define SOF_BASELINE_DCT 0xFFC0
>>> +#define END_OF_IMAGE 0xFFD9
>>> +#define START_OF_SCAN 0xFFDA
>>> +
>>> +/* Definitions for the huffman table specification in the Marker
>>> segment */
>>> +#define DHT_MARKER 0xFFC4
>>> +#define LH_DC 0x001F
>>> +#define LH_AC 0x00B5
>>> +
>>> +/* Definitions for the quantization table specification in the
>>> Marker segment */
>>> +#define DQT_MARKER 0xFFDB
>>> +#define ACMAX 0x03FF
>>> +#define DCMAX 0x07FF
>>> +
>>> +/* Length and precision of the quantization table parameters */
>>> +#define LQPQ 0x00430
>>> +#define QMAX 255
>>> +
>>> +/* Misc JPEG header definitions */
>>> +#define UC_NUM_COMP 3
>>> +#define PRECISION 8
>>> +#define HORZ_SAMPLING_FACTOR (2 << 4)
>>> +#define VERT_SAMPLING_FACTOR_422 1
>>> +#define VERT_SAMPLING_FACTOR_420 2
>>> +#define COMPONENTS_IN_SCAN 3
>>> +#define PELS_IN_BLOCK 64
>>> +
>>> +/* Used for Qp table generation */
>>> +#define LUMINOSITY 10
>>> +#define CONTRAST 1
>>> +#define INCREASE 2
>>> +#define QP_TABLE_SIZE (8 * 8)
>>> +#define QP_TABLE_FIELD_OFFSET 0x04
>>> +
>>> +/*
>>> + * vb2 queue structure
>>> + * contains queue data information
>>> + *
>>> + * @fmt: format info
>>> + * @width: frame width
>>> + * @height: frame height
>>> + * @bytesperline: bytes per line in memory
>>> + * @size_image: image size in memory
>>> + */
>>> +struct e5010_q_data {
>>> + struct e5010_fmt *fmt;
>>> + u32 width;
>>> + u32 height;
>>> + u32 width_adjusted;
>>> + u32 height_adjusted;
>>> + u32 sizeimage[MAX_PLANES];
>>> + u32 bytesperline[MAX_PLANES];
>>> + bool format_set;
>>> + bool streaming;
>>> + u32 sequence;
>>> + struct v4l2_rect crop;
>>> +};
>>> +
>>> +/*
>>> + * Driver device structure
>>> + * Holds all memory handles and global parameters
>>> + * Shared by all instances
>>> + */
>>> +struct e5010_dev {
>>> + struct device *dev;
>>> + struct v4l2_device v4l2_dev;
>>> + struct v4l2_m2m_dev *m2m_dev;
>>> + struct video_device *vdev;
>>> + void __iomem *core_base;
>>> + void __iomem *mmu_base;
>>> + struct clk *clk;
>>> + struct e5010_context *last_context_run;
>>> + /* Protect access to device data */
>>> + struct mutex mutex;
>>> + /* Protect access to hardware*/
>>> + spinlock_t hw_lock;
>>> +};
>>> +
>>> +/*
>>> + * Driver context structure
>>> + * One of these exists for every m2m context
>>> + * Holds context specific data
>>> + */
>>> +struct e5010_context {
>>> + struct v4l2_fh fh;
>>> + struct e5010_dev *e5010;
>>> + struct e5010_q_data out_queue;
>>> + struct e5010_q_data cap_queue;
>>> + int quality;
>>> + bool update_qp;
>>> + struct v4l2_ctrl_handler ctrl_handler;
>>> + u8 luma_qp[QP_TABLE_SIZE];
>>> + u8 chroma_qp[QP_TABLE_SIZE];
>>> +};
>>> +
>>> +/*
>>> + * Buffer structure
>>> + * Contains info for all buffers
>>> + */
>>> +struct e5010_buffer {
>>> + struct v4l2_m2m_buffer buffer;
>>> +};
>>> +
>>> +enum {
>>> + CHROMA_ORDER_CB_CR = 0, //UV ordering
>>> + CHROMA_ORDER_CR_CB = 1, //VU ordering
>>> +};
>>> +
>>> +enum {
>>> + SUBSAMPLING_420 = 1,
>>> + SUBSAMPLING_422 = 2,
>>> +};
>>> +
>>> +/*
>>> + * e5010 format structure
>>> + * contains format information
>>> + */
>>> +struct e5010_fmt {
>>> + u32 fourcc;
>>> + unsigned int num_planes;
>>> + unsigned int type;
>>> + u32 subsampling;
>>> + u32 chroma_order;
>>> + const struct v4l2_frmsize_stepwise frmsize;
>>> +};
>>> +
>>> +/*
>>> + * struct e5010_ctrl - contains info for each supported v4l2 control
>>> + */
>>> +struct e5010_ctrl {
>>> + unsigned int cid;
>>> + enum v4l2_ctrl_type type;
>>> + unsigned char name[32];
>>> + int minimum;
>>> + int maximum;
>>> + int step;
>>> + int default_value;
>>> + unsigned char compound;
>>> +};
>>> +
>>> +#endif
>>> diff --git a/drivers/media/platform/imagination/e5010-mmu-regs.h
>>> b/drivers/media/platform/imagination/e5010-mmu-regs.h
>>> new file mode 100644
>>> index 000000000000..bfba06956cf2
>>> --- /dev/null
>>> +++ b/drivers/media/platform/imagination/e5010-mmu-regs.h
>>> @@ -0,0 +1,311 @@
>>> +/* SPDX-License-Identifier: GPL-2.0 */
>>> +/*
>>> + * Imagination E5010 JPEG Encoder driver.
>>> + *
>>> + * Copyright (C) 2023 Texas Instruments Incorporated -
>>> https://www.ti.com/
>>> + *
>>> + * Author: David Huang <d-huang@...com>
>>> + * Author: Devarsh Thakkar <devarsht@...com>
>>> + */
>>> +
>>> +#ifndef _E5010_MMU_REGS_H
>>> +#define _E5010_MMU_REGS_H
>>> +
>>> +#define MMU_MMU_DIR_BASE_ADDR_OFFSET (0x0020)
>>> +#define MMU_MMU_DIR_BASE_ADDR_STRIDE (4)
>>> +#define MMU_MMU_DIR_BASE_ADDR_NO_ENTRIES (4)
>>> +
>>> +#define MMU_MMU_DIR_BASE_ADDR_MMU_DIR_BASE_ADDR_MASK (0xFFFFFFFF)
>>> +#define MMU_MMU_DIR_BASE_ADDR_MMU_DIR_BASE_ADDR_SHIFT (0)
>>> +
>>> +#define MMU_MMU_TILE_CFG_OFFSET (0x0040)
>>> +#define MMU_MMU_TILE_CFG_STRIDE (4)
>>> +#define MMU_MMU_TILE_CFG_NO_ENTRIES (4)
>>> +
>>> +#define MMU_MMU_TILE_CFG_TILE_128INTERLEAVE_MASK (0x00000010)
>>> +#define MMU_MMU_TILE_CFG_TILE_128INTERLEAVE_SHIFT (4)
>>> +
>>> +#define MMU_MMU_TILE_CFG_TILE_ENABLE_MASK (0x00000008)
>>> +#define MMU_MMU_TILE_CFG_TILE_ENABLE_SHIFT (3)
>>> +
>>> +#define MMU_MMU_TILE_CFG_TILE_STRIDE_MASK (0x00000007)
>>> +#define MMU_MMU_TILE_CFG_TILE_STRIDE_SHIFT (0)
>>> +
>>> +#define MMU_MMU_TILE_MIN_ADDR_OFFSET (0x0050)
>>> +#define MMU_MMU_TILE_MIN_ADDR_STRIDE (4)
>>> +#define MMU_MMU_TILE_MIN_ADDR_NO_ENTRIES (4)
>>> +
>>> +#define MMU_MMU_TILE_MIN_ADDR_TILE_MIN_ADDR_MASK (0xFFFFFFFF)
>>> +#define MMU_MMU_TILE_MIN_ADDR_TILE_MIN_ADDR_SHIFT (0)
>>> +
>>> +#define MMU_MMU_TILE_MAX_ADDR_OFFSET (0x0060)
>>> +#define MMU_MMU_TILE_MAX_ADDR_STRIDE (4)
>>> +#define MMU_MMU_TILE_MAX_ADDR_NO_ENTRIES (4)
>>> +
>>> +#define MMU_MMU_TILE_MAX_ADDR_TILE_MAX_ADDR_MASK (0xFFFFFFFF)
>>> +#define MMU_MMU_TILE_MAX_ADDR_TILE_MAX_ADDR_SHIFT (0)
>>> +
>>> +#define MMU_MMU_CONTROL0_OFFSET (0x0000)
>>> +
>>> +#define MMU_MMU_CONTROL0_MMU_TILING_SCHEME_MASK (0x00000001)
>>> +#define MMU_MMU_CONTROL0_MMU_TILING_SCHEME_SHIFT (0)
>>> +
>>> +#define MMU_MMU_CONTROL0_MMU_CACHE_POLICY_MASK (0x00000100)
>>> +#define MMU_MMU_CONTROL0_MMU_CACHE_POLICY_SHIFT (8)
>>> +
>>> +#define MMU_MMU_CONTROL0_FORCE_CACHE_POLICY_BYPASS_MASK (0x00000200)
>>> +#define MMU_MMU_CONTROL0_FORCE_CACHE_POLICY_BYPASS_SHIFT (9)
>>> +
>>> +#define MMU_MMU_CONTROL0_STALL_ON_PROTOCOL_FAULT_MASK (0x00001000)
>>> +#define MMU_MMU_CONTROL0_STALL_ON_PROTOCOL_FAULT_SHIFT (12)
>>> +
>>> +#define MMU_MMU_CONTROL1_OFFSET (0x0008)
>>> +
>>> +#define MMU_MMU_CONTROL1_MMU_FLUSH_MASK (0x00000008)
>>> +#define MMU_MMU_CONTROL1_MMU_FLUSH_SHIFT (3)
>>> +#define MMU_MMU_CONTROL1_MMU_FLUSH_NO_REPS (4)
>>> +#define MMU_MMU_CONTROL1_MMU_FLUSH_SIZE (1)
>>> +
>>> +#define MMU_MMU_CONTROL1_MMU_INVALDC_MASK (0x00000800)
>>> +#define MMU_MMU_CONTROL1_MMU_INVALDC_SHIFT (11)
>>> +#define MMU_MMU_CONTROL1_MMU_INVALDC_NO_REPS (4)
>>> +#define MMU_MMU_CONTROL1_MMU_INVALDC_SIZE (1)
>>> +
>>> +#define MMU_MMU_CONTROL1_MMU_FAULT_CLEAR_MASK (0x00010000)
>>> +#define MMU_MMU_CONTROL1_MMU_FAULT_CLEAR_SHIFT (16)
>>> +
>>> +#define MMU_MMU_CONTROL1_PROTOCOL_FAULT_CLEAR_MASK (0x00100000)
>>> +#define MMU_MMU_CONTROL1_PROTOCOL_FAULT_CLEAR_SHIFT (20)
>>> +
>>> +#define MMU_MMU_CONTROL1_MMU_PAUSE_SET_MASK (0x01000000)
>>> +#define MMU_MMU_CONTROL1_MMU_PAUSE_SET_SHIFT (24)
>>> +
>>> +#define MMU_MMU_CONTROL1_MMU_PAUSE_CLEAR_MASK (0x02000000)
>>> +#define MMU_MMU_CONTROL1_MMU_PAUSE_CLEAR_SHIFT (25)
>>> +
>>> +#define MMU_MMU_CONTROL1_MMU_SOFT_RESET_MASK (0x10000000)
>>> +#define MMU_MMU_CONTROL1_MMU_SOFT_RESET_SHIFT (28)
>>> +
>>> +#define MMU_MMU_BANK_INDEX_OFFSET (0x0010)
>>> +
>>> +#define MMU_MMU_BANK_INDEX_MMU_BANK_INDEX_MASK (0xC0000000)
>>> +#define MMU_MMU_BANK_INDEX_MMU_BANK_INDEX_SHIFT (30)
>>> +#define MMU_MMU_BANK_INDEX_MMU_BANK_INDEX_NO_REPS (16)
>>> +#define MMU_MMU_BANK_INDEX_MMU_BANK_INDEX_SIZE (2)
>>> +
>>> +#define MMU_REQUEST_PRIORITY_ENABLE_OFFSET (0x0018)
>>> +
>>> +#define MMU_REQUEST_PRIORITY_ENABLE_CMD_PRIORITY_ENABLE_MASK
>>> (0x00008000)
>>> +#define MMU_REQUEST_PRIORITY_ENABLE_CMD_PRIORITY_ENABLE_SHIFT (15)
>>> +#define MMU_REQUEST_PRIORITY_ENABLE_CMD_PRIORITY_ENABLE_NO_REPS (16)
>>> +#define MMU_REQUEST_PRIORITY_ENABLE_CMD_PRIORITY_ENABLE_SIZE (1)
>>> +
>>> +#define MMU_REQUEST_PRIORITY_ENABLE_CMD_MMU_PRIORITY_ENABLE_MASK
>>> (0x00010000)
>>> +#define MMU_REQUEST_PRIORITY_ENABLE_CMD_MMU_PRIORITY_ENABLE_SHIFT (16)
>>> +
>>> +#define MMU_REQUEST_LIMITED_THROUGHPUT_OFFSET (0x001C)
>>> +
>>> +#define MMU_REQUEST_LIMITED_THROUGHPUT_LIMITED_WORDS_MASK (0x000003FF)
>>> +#define MMU_REQUEST_LIMITED_THROUGHPUT_LIMITED_WORDS_SHIFT (0)
>>> +
>>> +#define MMU_REQUEST_LIMITED_THROUGHPUT_REQUEST_GAP_MASK (0x0FFF0000)
>>> +#define MMU_REQUEST_LIMITED_THROUGHPUT_REQUEST_GAP_SHIFT (16)
>>> +
>>> +#define MMU_MMU_ADDRESS_CONTROL_OFFSET (0x0070)
>>> +#define MMU_MMU_ADDRESS_CONTROL_TRUSTED (IMG_TRUE)
>>> +
>>> +#define MMU_MMU_ADDRESS_CONTROL_MMU_BYPASS_MASK (0x00000001)
>>> +#define MMU_MMU_ADDRESS_CONTROL_MMU_BYPASS_SHIFT (0)
>>> +
>>> +#define MMU_MMU_ADDRESS_CONTROL_MMU_ENABLE_EXT_ADDRESSING_MASK
>>> (0x00000010)
>>> +#define MMU_MMU_ADDRESS_CONTROL_MMU_ENABLE_EXT_ADDRESSING_SHIFT (4)
>>> +
>>> +#define MMU_MMU_ADDRESS_CONTROL_UPPER_ADDRESS_FIXED_MASK (0x00FF0000)
>>> +#define MMU_MMU_ADDRESS_CONTROL_UPPER_ADDRESS_FIXED_SHIFT (16)
>>> +
>>> +#define MMU_MMU_CONFIG0_OFFSET (0x0080)
>>> +
>>> +#define MMU_MMU_CONFIG0_NUM_REQUESTORS_MASK (0x0000000F)
>>> +#define MMU_MMU_CONFIG0_NUM_REQUESTORS_SHIFT (0)
>>> +
>>> +#define MMU_MMU_CONFIG0_EXTENDED_ADDR_RANGE_MASK (0x000000F0)
>>> +#define MMU_MMU_CONFIG0_EXTENDED_ADDR_RANGE_SHIFT (4)
>>> +
>>> +#define MMU_MMU_CONFIG0_GROUP_OVERRIDE_SIZE_MASK (0x00000700)
>>> +#define MMU_MMU_CONFIG0_GROUP_OVERRIDE_SIZE_SHIFT (8)
>>> +
>>> +#define MMU_MMU_CONFIG0_ADDR_COHERENCY_SUPPORTED_MASK (0x00001000)
>>> +#define MMU_MMU_CONFIG0_ADDR_COHERENCY_SUPPORTED_SHIFT (12)
>>> +
>>> +#define MMU_MMU_CONFIG0_MMU_SUPPORTED_MASK (0x00002000)
>>> +#define MMU_MMU_CONFIG0_MMU_SUPPORTED_SHIFT (13)
>>> +
>>> +#define MMU_MMU_CONFIG0_TILE_ADDR_GRANULARITY_MASK (0x001F0000)
>>> +#define MMU_MMU_CONFIG0_TILE_ADDR_GRANULARITY_SHIFT (16)
>>> +
>>> +#define MMU_MMU_CONFIG0_NO_READ_REORDER_MASK (0x00200000)
>>> +#define MMU_MMU_CONFIG0_NO_READ_REORDER_SHIFT (21)
>>> +
>>> +#define MMU_MMU_CONFIG0_TAGS_SUPPORTED_MASK (0xFFC00000)
>>> +#define MMU_MMU_CONFIG0_TAGS_SUPPORTED_SHIFT (22)
>>> +
>>> +#define MMU_MMU_CONFIG1_OFFSET (0x0084)
>>> +
>>> +#define MMU_MMU_CONFIG1_PAGE_SIZE_MASK (0x0000000F)
>>> +#define MMU_MMU_CONFIG1_PAGE_SIZE_SHIFT (0)
>>> +
>>> +#define MMU_MMU_CONFIG1_PAGE_CACHE_ENTRIES_MASK (0x0000FF00)
>>> +#define MMU_MMU_CONFIG1_PAGE_CACHE_ENTRIES_SHIFT (8)
>>> +
>>> +#define MMU_MMU_CONFIG1_DIR_CACHE_ENTRIES_MASK (0x001F0000)
>>> +#define MMU_MMU_CONFIG1_DIR_CACHE_ENTRIES_SHIFT (16)
>>> +
>>> +#define MMU_MMU_CONFIG1_BANDWIDTH_COUNT_SUPPORTED_MASK (0x01000000)
>>> +#define MMU_MMU_CONFIG1_BANDWIDTH_COUNT_SUPPORTED_SHIFT (24)
>>> +
>>> +#define MMU_MMU_CONFIG1_STALL_COUNT_SUPPORTED_MASK (0x02000000)
>>> +#define MMU_MMU_CONFIG1_STALL_COUNT_SUPPORTED_SHIFT (25)
>>> +
>>> +#define MMU_MMU_CONFIG1_LATENCY_COUNT_SUPPORTED_MASK (0x04000000)
>>> +#define MMU_MMU_CONFIG1_LATENCY_COUNT_SUPPORTED_SHIFT (26)
>>> +
>>> +#define MMU_MMU_STATUS0_OFFSET (0x0088)
>>> +
>>> +#define MMU_MMU_STATUS0_MMU_PF_N_RW_MASK (0x00000001)
>>> +#define MMU_MMU_STATUS0_MMU_PF_N_RW_SHIFT (0)
>>> +
>>> +#define MMU_MMU_STATUS0_MMU_FAULT_ADDR_MASK (0xFFFFF000)
>>> +#define MMU_MMU_STATUS0_MMU_FAULT_ADDR_SHIFT (12)
>>> +
>>> +#define MMU_MMU_STATUS1_OFFSET (0x008C)
>>> +
>>> +#define MMU_MMU_STATUS1_MMU_FAULT_REQ_STAT_MASK (0x0000FFFF)
>>> +#define MMU_MMU_STATUS1_MMU_FAULT_REQ_STAT_SHIFT (0)
>>> +
>>> +#define MMU_MMU_STATUS1_MMU_FAULT_REQ_ID_MASK (0x000F0000)
>>> +#define MMU_MMU_STATUS1_MMU_FAULT_REQ_ID_SHIFT (16)
>>> +
>>> +#define MMU_MMU_STATUS1_MMU_FAULT_INDEX_MASK (0x03000000)
>>> +#define MMU_MMU_STATUS1_MMU_FAULT_INDEX_SHIFT (24)
>>> +
>>> +#define MMU_MMU_STATUS1_MMU_FAULT_RNW_MASK (0x10000000)
>>> +#define MMU_MMU_STATUS1_MMU_FAULT_RNW_SHIFT (28)
>>> +
>>> +#define MMU_MMU_MEM_REQ_OFFSET (0x0090)
>>> +
>>> +#define MMU_MMU_MEM_REQ_TAG_OUTSTANDING_MASK (0x000003FF)
>>> +#define MMU_MMU_MEM_REQ_TAG_OUTSTANDING_SHIFT (0)
>>> +
>>> +#define MMU_MMU_MEM_REQ_EXT_WRRESP_FAULT_MASK (0x00001000)
>>> +#define MMU_MMU_MEM_REQ_EXT_WRRESP_FAULT_SHIFT (12)
>>> +
>>> +#define MMU_MMU_MEM_REQ_EXT_RDRESP_FAULT_MASK (0x00002000)
>>> +#define MMU_MMU_MEM_REQ_EXT_RDRESP_FAULT_SHIFT (13)
>>> +
>>> +#define MMU_MMU_MEM_REQ_EXT_READ_BURST_FAULT_MASK (0x00004000)
>>> +#define MMU_MMU_MEM_REQ_EXT_READ_BURST_FAULT_SHIFT (14)
>>> +
>>> +#define MMU_MMU_MEM_REQ_INT_PROTOCOL_FAULT_MASK (0x80000000)
>>> +#define MMU_MMU_MEM_REQ_INT_PROTOCOL_FAULT_SHIFT (31)
>>> +#define MMU_MMU_MEM_REQ_INT_PROTOCOL_FAULT_NO_REPS (16)
>>> +#define MMU_MMU_MEM_REQ_INT_PROTOCOL_FAULT_SIZE (1)
>>> +
>>> +#define MMU_MMU_FAULT_SELECT_OFFSET (0x00A0)
>>> +
>>> +#define MMU_MMU_FAULT_SELECT_MMU_FAULT_SELECT_MASK (0x0000000F)
>>> +#define MMU_MMU_FAULT_SELECT_MMU_FAULT_SELECT_SHIFT (0)
>>> +
>>> +#define MMU_PROTOCOL_FAULT_OFFSET (0x00A8)
>>> +
>>> +#define MMU_PROTOCOL_FAULT_FAULT_PAGE_BREAK_MASK (0x00000001)
>>> +#define MMU_PROTOCOL_FAULT_FAULT_PAGE_BREAK_SHIFT (0)
>>> +
>>> +#define MMU_PROTOCOL_FAULT_FAULT_WRITE_MASK (0x00000010)
>>> +#define MMU_PROTOCOL_FAULT_FAULT_WRITE_SHIFT (4)
>>> +
>>> +#define MMU_PROTOCOL_FAULT_FAULT_READ_MASK (0x00000020)
>>> +#define MMU_PROTOCOL_FAULT_FAULT_READ_SHIFT (5)
>>> +
>>> +#define MMU_TOTAL_READ_REQ_OFFSET (0x0100)
>>> +
>>> +#define MMU_TOTAL_READ_REQ_TOTAL_READ_REQ_MASK (0xFFFFFFFF)
>>> +#define MMU_TOTAL_READ_REQ_TOTAL_READ_REQ_SHIFT (0)
>>> +
>>> +#define MMU_TOTAL_WRITE_REQ_OFFSET (0x0104)
>>> +
>>> +#define MMU_TOTAL_WRITE_REQ_TOTAL_WRITE_REQ_MASK (0xFFFFFFFF)
>>> +#define MMU_TOTAL_WRITE_REQ_TOTAL_WRITE_REQ_SHIFT (0)
>>> +
>>> +#define MMU_READS_LESS_64_REQ_OFFSET (0x0108)
>>> +
>>> +#define MMU_READS_LESS_64_REQ_READS_LESS_64_REQ_MASK (0xFFFFFFFF)
>>> +#define MMU_READS_LESS_64_REQ_READS_LESS_64_REQ_SHIFT (0)
>>> +
>>> +#define MMU_WRITES_LESS_64_REQ_OFFSET (0x010C)
>>> +
>>> +#define MMU_WRITES_LESS_64_REQ_WRITES_LESS_64_REQ_MASK (0xFFFFFFFF)
>>> +#define MMU_WRITES_LESS_64_REQ_WRITES_LESS_64_REQ_SHIFT (0)
>>> +
>>> +#define MMU_EXT_CMD_STALL_OFFSET (0x0120)
>>> +
>>> +#define MMU_EXT_CMD_STALL_EXT_CMD_STALL_MASK (0xFFFFFFFF)
>>> +#define MMU_EXT_CMD_STALL_EXT_CMD_STALL_SHIFT (0)
>>> +
>>> +#define MMU_WRITE_REQ_STALL_OFFSET (0x0124)
>>> +
>>> +#define MMU_WRITE_REQ_STALL_WRITE_REQ_STALL_MASK (0xFFFFFFFF)
>>> +#define MMU_WRITE_REQ_STALL_WRITE_REQ_STALL_SHIFT (0)
>>> +
>>> +#define MMU_MMU_MISS_STALL_OFFSET (0x0128)
>>> +
>>> +#define MMU_MMU_MISS_STALL_MMU_MISS_STALL_MASK (0xFFFFFFFF)
>>> +#define MMU_MMU_MISS_STALL_MMU_MISS_STALL_SHIFT (0)
>>> +
>>> +#define MMU_ADDRESS_STALL_OFFSET (0x012C)
>>> +
>>> +#define MMU_ADDRESS_STALL_ADDRESS_STALL_MASK (0xFFFFFFFF)
>>> +#define MMU_ADDRESS_STALL_ADDRESS_STALL_SHIFT (0)
>>> +
>>> +#define MMU_TAG_STALL_OFFSET (0x0130)
>>> +
>>> +#define MMU_TAG_STALL_TAG_STALL_MASK (0xFFFFFFFF)
>>> +#define MMU_TAG_STALL_TAG_STALL_SHIFT (0)
>>> +
>>> +#define MMU_PEAK_READ_OUTSTANDING_OFFSET (0x0140)
>>> +
>>> +#define MMU_PEAK_READ_OUTSTANDING_PEAK_TAG_OUTSTANDING_MASK
>>> (0x000003FF)
>>> +#define MMU_PEAK_READ_OUTSTANDING_PEAK_TAG_OUTSTANDING_SHIFT (0)
>>> +
>>> +#define MMU_PEAK_READ_OUTSTANDING_PEAK_READ_LATENCY_MASK (0xFFFF0000)
>>> +#define MMU_PEAK_READ_OUTSTANDING_PEAK_READ_LATENCY_SHIFT (16)
>>> +
>>> +#define MMU_AVERAGE_READ_LATENCY_OFFSET (0x0144)
>>> +
>>> +#define MMU_AVERAGE_READ_LATENCY_AVERAGE_READ_LATENCY_MASK
>>> (0xFFFFFFFF)
>>> +#define MMU_AVERAGE_READ_LATENCY_AVERAGE_READ_LATENCY_SHIFT (0)
>>> +
>>> +#define MMU_STATISTICS_CONTROL_OFFSET (0x0160)
>>> +
>>> +#define MMU_STATISTICS_CONTROL_BANDWIDTH_STATS_INIT_MASK (0x00000001)
>>> +#define MMU_STATISTICS_CONTROL_BANDWIDTH_STATS_INIT_SHIFT (0)
>>> +
>>> +#define MMU_STATISTICS_CONTROL_STALL_STATS_INIT_MASK (0x00000002)
>>> +#define MMU_STATISTICS_CONTROL_STALL_STATS_INIT_SHIFT (1)
>>> +
>>> +#define MMU_STATISTICS_CONTROL_LATENCY_STATS_INIT_MASK (0x00000004)
>>> +#define MMU_STATISTICS_CONTROL_LATENCY_STATS_INIT_SHIFT (2)
>>> +
>>> +#define MMU_MMU_VERSION_OFFSET (0x01D0)
>>> +
>>> +#define MMU_MMU_VERSION_MMU_MAJOR_REV_MASK (0x00FF0000)
>>> +#define MMU_MMU_VERSION_MMU_MAJOR_REV_SHIFT (16)
>>> +
>>> +#define MMU_MMU_VERSION_MMU_MINOR_REV_MASK (0x0000FF00)
>>> +#define MMU_MMU_VERSION_MMU_MINOR_REV_SHIFT (8)
>>> +
>>> +#define MMU_MMU_VERSION_MMU_MAINT_REV_MASK (0x000000FF)
>>> +#define MMU_MMU_VERSION_MMU_MAINT_REV_SHIFT (0)
>>> +
>>> +#define MMU_BYTE_SIZE (0x01D4)
>>> +
>>> +#endif
>>
>
Powered by blists - more mailing lists