[PATCH v5 15/16] media: platform: Add mali-c55 parameters video node
Sakari Ailus
sakari.ailus at iki.fi
Fri Jun 14 11:53:22 PDT 2024
Hi Jacopo, Dan,
Thanks for the patch. Please see my comments below.
On Wed, May 29, 2024 at 04:28:57PM +0100, Daniel Scally wrote:
> Add a new code file to the mali-c55 driver that registers an output
> video node for userspace to queue buffers of parameters to. Handlers
> are included to program the statistics generation plus the white
> balance, black level correction and mesh shading correction blocks.
>
> Update the rest of the driver to register and link the new video node
>
> Acked-by: Nayden Kanchev <nayden.kanchev at arm.com>
> Co-developed-by: Jacopo Mondi <jacopo.mondi at ideasonboard.com>
> Signed-off-by: Jacopo Mondi <jacopo.mondi at ideasonboard.com>
> Signed-off-by: Daniel Scally <dan.scally at ideasonboard.com>
> ---
> Changes in v5:
>
> - New patch
>
> drivers/media/platform/arm/mali-c55/Makefile | 1 +
> .../platform/arm/mali-c55/mali-c55-common.h | 18 +
> .../platform/arm/mali-c55/mali-c55-core.c | 24 +
> .../platform/arm/mali-c55/mali-c55-isp.c | 16 +-
> .../platform/arm/mali-c55/mali-c55-params.c | 615 ++++++++++++++++++
> .../arm/mali-c55/mali-c55-registers.h | 104 +++
> 6 files changed, 777 insertions(+), 1 deletion(-)
> create mode 100644 drivers/media/platform/arm/mali-c55/mali-c55-params.c
>
> diff --git a/drivers/media/platform/arm/mali-c55/Makefile b/drivers/media/platform/arm/mali-c55/Makefile
> index cd5a64bf0c62..b2443f2d416a 100644
> --- a/drivers/media/platform/arm/mali-c55/Makefile
> +++ b/drivers/media/platform/arm/mali-c55/Makefile
> @@ -5,6 +5,7 @@ mali-c55-y := mali-c55-capture.o \
> mali-c55-isp.o \
> mali-c55-tpg.o \
> mali-c55-resizer.o \
> + mali-c55-params.o \
> mali-c55-stats.o
>
> obj-$(CONFIG_VIDEO_MALI_C55) += mali-c55.o
> diff --git a/drivers/media/platform/arm/mali-c55/mali-c55-common.h b/drivers/media/platform/arm/mali-c55/mali-c55-common.h
> index 44119e04009b..565d98acfcdd 100644
> --- a/drivers/media/platform/arm/mali-c55/mali-c55-common.h
> +++ b/drivers/media/platform/arm/mali-c55/mali-c55-common.h
> @@ -80,6 +80,7 @@ enum mali_c55_isp_pads {
> MALI_C55_ISP_PAD_SOURCE,
> MALI_C55_ISP_PAD_SOURCE_BYPASS,
> MALI_C55_ISP_PAD_SOURCE_3A,
> + MALI_C55_ISP_PAD_SINK_PARAMS,
> MALI_C55_ISP_NUM_PADS,
> };
>
> @@ -217,6 +218,19 @@ struct mali_c55_stats {
> } buffers;
> };
>
> +struct mali_c55_params {
> + struct mali_c55 *mali_c55;
> + struct video_device vdev;
> + struct vb2_queue queue;
> + struct media_pad pad;
> + struct mutex lock;
> +
> + struct {
> + spinlock_t lock;
> + struct list_head queue;
> + } buffers;
> +};
> +
> enum mali_c55_config_spaces {
> MALI_C55_CONFIG_PING,
> MALI_C55_CONFIG_PONG,
> @@ -247,6 +261,7 @@ struct mali_c55 {
> struct mali_c55_isp isp;
> struct mali_c55_resizer resizers[MALI_C55_NUM_RZRS];
> struct mali_c55_cap_dev cap_devs[MALI_C55_NUM_CAP_DEVS];
> + struct mali_c55_params params;
> struct mali_c55_stats stats;
>
> struct list_head contexts;
> @@ -271,6 +286,8 @@ int mali_c55_register_capture_devs(struct mali_c55 *mali_c55);
> void mali_c55_unregister_capture_devs(struct mali_c55 *mali_c55);
> int mali_c55_register_stats(struct mali_c55 *mali_c55);
> void mali_c55_unregister_stats(struct mali_c55 *mali_c55);
> +int mali_c55_register_params(struct mali_c55 *mali_c55);
> +void mali_c55_unregister_params(struct mali_c55 *mali_c55);
> struct mali_c55_ctx *mali_c55_get_active_context(struct mali_c55 *mali_c55);
> void mali_c55_set_plane_done(struct mali_c55_cap_dev *cap_dev,
> enum mali_c55_planes plane);
> @@ -290,5 +307,6 @@ bool mali_c55_isp_is_format_supported(unsigned int mbus_code);
> for ((fmt) = NULL; ((fmt) = mali_c55_isp_fmt_next((fmt)));)
> void mali_c55_stats_fill_buffer(struct mali_c55 *mali_c55,
> enum mali_c55_config_spaces cfg_space);
> +void mali_c55_params_write_config(struct mali_c55 *mali_c55);
>
> #endif /* _MALI_C55_COMMON_H */
> diff --git a/drivers/media/platform/arm/mali-c55/mali-c55-core.c b/drivers/media/platform/arm/mali-c55/mali-c55-core.c
> index 2cf8b1169604..6acee3edd03f 100644
> --- a/drivers/media/platform/arm/mali-c55/mali-c55-core.c
> +++ b/drivers/media/platform/arm/mali-c55/mali-c55-core.c
> @@ -347,6 +347,17 @@ static int mali_c55_create_links(struct mali_c55 *mali_c55)
> goto err_remove_links;
> }
>
> + ret = media_create_pad_link(&mali_c55->params.vdev.entity, 0,
> + &mali_c55->isp.sd.entity,
> + MALI_C55_ISP_PAD_SINK_PARAMS,
> + MEDIA_LNK_FL_ENABLED |
> + MEDIA_LNK_FL_IMMUTABLE);
> + if (ret) {
> + dev_err(mali_c55->dev,
> + "failed to link ISP and parameters video node\n");
> + goto err_remove_links;
> + }
> +
> return 0;
>
> err_remove_links:
> @@ -360,6 +371,7 @@ static void mali_c55_unregister_entities(struct mali_c55 *mali_c55)
> mali_c55_unregister_isp(mali_c55);
> mali_c55_unregister_resizers(mali_c55);
> mali_c55_unregister_capture_devs(mali_c55);
> + mali_c55_unregister_params(mali_c55);
> mali_c55_unregister_stats(mali_c55);
> }
>
> @@ -383,6 +395,10 @@ static int mali_c55_register_entities(struct mali_c55 *mali_c55)
> if (ret)
> goto err_unregister_entities;
>
> + ret = mali_c55_register_params(mali_c55);
> + if (ret)
> + goto err_unregister_entities;
> +
> ret = mali_c55_register_stats(mali_c55);
> if (ret)
> goto err_unregister_entities;
> @@ -474,6 +490,14 @@ static irqreturn_t mali_c55_isr(int irq, void *context)
> curr_config >>= ffs(MALI_C55_REG_PING_PONG_READ_MASK) - 1;
> next_config = curr_config ^ 1;
>
> + /*
> + * Write the configuration parameters received from
> + * userspace into the configuration buffer, which will
> + * be transferred to the 'next' active config space at
> + * by mali_c55_swap_next_config().
> + */
> + mali_c55_params_write_config(mali_c55);
> +
> /*
> * The ordering of these two is currently important as
> * mali_c55_stats_fill_buffer() is asynchronous whereas
> diff --git a/drivers/media/platform/arm/mali-c55/mali-c55-isp.c b/drivers/media/platform/arm/mali-c55/mali-c55-isp.c
> index 94876fba3353..8c2b45bfd82d 100644
> --- a/drivers/media/platform/arm/mali-c55/mali-c55-isp.c
> +++ b/drivers/media/platform/arm/mali-c55/mali-c55-isp.c
> @@ -146,6 +146,7 @@ static int mali_c55_isp_start(struct mali_c55 *mali_c55)
> cfg->encoding == V4L2_PIXEL_ENC_RGB ?
> MALI_C55_ISP_RAW_BYPASS_BYPASS_MASK : 0x00);
>
> + mali_c55_params_write_config(mali_c55);
> ret = mali_c55_config_write(ctx, MALI_C55_CONFIG_PING);
> if (ret) {
> dev_err(mali_c55->dev, "failed to DMA config\n");
> @@ -455,8 +456,20 @@ static const struct v4l2_subdev_internal_ops mali_c55_isp_internal_ops = {
> .init_state = mali_c55_isp_init_state,
> };
>
> +static int mali_c55_subdev_link_validate(struct media_link *link)
> +{
> + /*
> + * Skip validation for the parameters sink pad, as the source is not
> + * a subdevice.
> + */
> + if (link->sink->index == MALI_C55_ISP_PAD_SINK_PARAMS)
> + return 0;
> +
> + return v4l2_subdev_link_validate(link);
> +}
> +
> static const struct media_entity_operations mali_c55_isp_media_ops = {
> - .link_validate = v4l2_subdev_link_validate,
> + .link_validate = mali_c55_subdev_link_validate,
> };
>
> static int mali_c55_isp_notifier_bound(struct v4l2_async_notifier *notifier,
> @@ -565,6 +578,7 @@ int mali_c55_register_isp(struct mali_c55 *mali_c55)
> isp->pads[MALI_C55_ISP_PAD_SOURCE].flags = MEDIA_PAD_FL_SOURCE;
> isp->pads[MALI_C55_ISP_PAD_SOURCE_BYPASS].flags = MEDIA_PAD_FL_SOURCE;
> isp->pads[MALI_C55_ISP_PAD_SOURCE_3A].flags = MEDIA_PAD_FL_SOURCE;
> + isp->pads[MALI_C55_ISP_PAD_SINK_PARAMS].flags = MEDIA_PAD_FL_SINK;
>
> ret = media_entity_pads_init(&sd->entity, MALI_C55_ISP_NUM_PADS,
> isp->pads);
> diff --git a/drivers/media/platform/arm/mali-c55/mali-c55-params.c b/drivers/media/platform/arm/mali-c55/mali-c55-params.c
> new file mode 100644
> index 000000000000..049a7b8e4861
> --- /dev/null
> +++ b/drivers/media/platform/arm/mali-c55/mali-c55-params.c
> @@ -0,0 +1,615 @@
> +// SPDX-License-Identifier: GPL-2.0
> +/*
> + * ARM Mali-C55 ISP Driver - Configuration parameters output device
> + *
> + * Copyright (C) 2024 Ideas on Board Oy
> + */
> +#include <linux/media/arm/mali-c55-config.h>
> +
> +#include <media/media-entity.h>
> +#include <media/v4l2-dev.h>
> +#include <media/v4l2-event.h>
> +#include <media/v4l2-fh.h>
> +#include <media/v4l2-ioctl.h>
> +#include <media/videobuf2-core.h>
> +#include <media/videobuf2-dma-contig.h>
> +
> +#include "mali-c55-common.h"
> +#include "mali-c55-registers.h"
> +
> +typedef void (*mali_c55_block_handler)(struct mali_c55 *mali_c55,
You can wrap after the return type (including typedef). Same elsewhere.
> + struct mali_c55_params_block_header *block);
> +
> +struct mali_c55_block_handler {
> + size_t size;
> + mali_c55_block_handler handler;
> +};
> +
> +static void mali_c55_params_sensor_offs(struct mali_c55 *mali_c55,
> + struct mali_c55_params_block_header *block)
> +{
> + struct mali_c55_params_sensor_off_preshading *p =
> + (struct mali_c55_params_sensor_off_preshading *)block;
I wonder if an union could be used to make this a bit cleaner. You're doing
a lot of casting that I think could be avoided.
> + __u32 global_offset;
> +
> + if (!block->enabled)
> + return;
> +
> + if (!(p->chan00 || p->chan01 || p->chan10 || p->chan11))
> + return;
> +
> + mali_c55_write(mali_c55, MALI_C55_REG_SENSOR_OFF_PRE_SHA_00,
> + p->chan00 & MALI_C55_SENSOR_OFF_PRE_SHA_MASK);
> + mali_c55_write(mali_c55, MALI_C55_REG_SENSOR_OFF_PRE_SHA_01,
> + p->chan01 & MALI_C55_SENSOR_OFF_PRE_SHA_MASK);
> + mali_c55_write(mali_c55, MALI_C55_REG_SENSOR_OFF_PRE_SHA_10,
> + p->chan10 & MALI_C55_SENSOR_OFF_PRE_SHA_MASK);
> + mali_c55_write(mali_c55, MALI_C55_REG_SENSOR_OFF_PRE_SHA_11,
> + p->chan11 & MALI_C55_SENSOR_OFF_PRE_SHA_MASK);
> +
> + /*
> + * The average offset is applied as a global offset for the digital
> + * gain block
> + */
> + global_offset = (p->chan00 + p->chan01 + p->chan10 + p->chan11) >> 2;
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_DIGITAL_GAIN_OFFSET,
> + MALI_C55_DIGITAL_GAIN_OFFSET_MASK, global_offset);
> +
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_BYPASS_3,
> + MALI_C55_REG_BYPASS_3_SENSOR_OFFSET_PRE_SH, 0x00);
> +}
> +
> +static void mali_c55_params_aexp_hist(struct mali_c55 *mali_c55,
> + struct mali_c55_params_block_header *block)
> +{
> + u32 disable_mask = block->type == MALI_C55_PARAM_BLOCK_AEXP_HIST ?
> + MALI_C55_AEXP_HIST_DISABLE_MASK :
> + MALI_C55_AEXP_IHIST_DISABLE_MASK;
> + u32 base = block->type == MALI_C55_PARAM_BLOCK_AEXP_HIST ?
> + MALI_C55_REG_AEXP_HIST_BASE :
> + MALI_C55_REG_AEXP_IHIST_BASE;
> + struct mali_c55_params_aexp_hist *params =
> + (struct mali_c55_params_aexp_hist *)block;
> +
> + if (!block->enabled) {
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_METERING_CONFIG,
> + disable_mask, true);
> + return;
> + }
> +
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_METERING_CONFIG,
> + disable_mask, false);
> +
> + mali_c55_update_bits(mali_c55, base + MALI_C55_AEXP_HIST_SKIP_OFFSET,
> + MALI_C55_AEXP_HIST_SKIP_X_MASK, params->skip_x);
> + mali_c55_update_bits(mali_c55, base + MALI_C55_AEXP_HIST_SKIP_OFFSET,
> + MALI_C55_AEXP_HIST_OFFSET_X_MASK, params->offset_x);
> + mali_c55_update_bits(mali_c55, base + MALI_C55_AEXP_HIST_SKIP_OFFSET,
> + MALI_C55_AEXP_HIST_SKIP_Y_MASK, params->skip_y);
> + mali_c55_update_bits(mali_c55, base + MALI_C55_AEXP_HIST_SKIP_OFFSET,
> + MALI_C55_AEXP_HIST_OFFSET_Y_MASK, params->offset_y);
> +
> + mali_c55_update_bits(mali_c55, base + MALI_C55_AEXP_HIST_SCALE_OFFSET,
> + MALI_C55_AEXP_HIST_SCALE_BOTTOM_MASK, params->scale_bottom);
> + mali_c55_update_bits(mali_c55, base + MALI_C55_AEXP_HIST_SCALE_OFFSET,
> + MALI_C55_AEXP_HIST_SCALE_TOP_MASK, params->scale_top);
> +
> + mali_c55_update_bits(mali_c55, base + MALI_C55_AEXP_HIST_PLANE_MODE_OFFSET,
> + MALI_C55_AEXP_HIST_PLANE_MODE_MASK, params->plane_mode);
> +
> + if (block->type == MALI_C55_PARAM_BLOCK_AEXP_HIST)
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_METERING_CONFIG,
> + MALI_C55_AEXP_HIST_SWITCH_MASK,
> + params->tap_point);
> +}
> +
> +static void
> +mali_c55_params_aexp_hist_weights(struct mali_c55 *mali_c55,
> + struct mali_c55_params_block_header *block)
> +{
> + struct mali_c55_params_aexp_weights *params =
> + (struct mali_c55_params_aexp_weights *)block;
> + u32 base;
> +
> + if (!block->enabled)
> + return;
> +
> + base = block->type == MALI_C55_PARAM_BLOCK_AEXP_HIST_WEIGHTS ?
> + MALI_C55_REG_AEXP_HIST_BASE :
> + MALI_C55_REG_AEXP_IHIST_BASE;
> +
> + mali_c55_update_bits(mali_c55, base + MALI_C55_AEXP_HIST_NODES_USED_OFFSET,
> + MALI_C55_AEXP_HIST_NODES_USED_HORIZ_MASK, params->nodes_used_horiz);
> + mali_c55_update_bits(mali_c55, base + MALI_C55_AEXP_HIST_NODES_USED_OFFSET,
> + MALI_C55_AEXP_HIST_NODES_USED_VERT_MASK, params->nodes_used_vert);
> +
> + /*
> + * The zone weights array is a 225-element array of u8 values, but that
> + * is a bit annoying to handle given the ISP expects 32-bit writes. We
> + * just reinterpret it as a 57-element array of 32-bit values for the
> + * purposes of this transaction (the 3 bytes of additional space at the
> + * end of the write is just padding for the array of weights in the ISP
> + * memory space anyway, so there's no risk of overwriting other
> + * registers).
> + */
> + for (unsigned int i = 0; i < 57; i++) {
> + u32 val = ((u32 *)params->zone_weights)[i]
> + & MALI_C55_AEXP_HIST_ZONE_WEIGHT_MASK;
> + u32 addr = base + MALI_C55_AEXP_HIST_ZONE_WEIGHTS_OFFSET + (4 * i);
> +
> + mali_c55_write(mali_c55, addr, val);
> + }
> +}
> +
> +static void mali_c55_params_digital_gain(struct mali_c55 *mali_c55,
> + struct mali_c55_params_block_header *block)
> +{
> + struct mali_c55_params_digital_gain *dgain =
> + (struct mali_c55_params_digital_gain *)block;
> +
> + /*
> + * If the block is flagged as disabled we write a gain of 1.0, which in
> + * Q5.8 format is 256.
> + */
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_DIGITAL_GAIN,
> + MALI_C55_DIGITAL_GAIN_MASK,
> + block->enabled ? dgain->gain : 256);
> +}
> +
> +static void mali_c55_params_awb_gains(struct mali_c55 *mali_c55,
> + struct mali_c55_params_block_header *block)
> +{
> + struct mali_c55_params_awb_gains *gains =
> + (struct mali_c55_params_awb_gains *)block;
> +
> + /*
> + * There are two places AWB gains can be set in the ISP; one affects the
> + * image output data and the other affects the statistics for the
> + * AEXP-0 tap point.
> + */
> + u32 addr1 = block->type = MALI_C55_PARAM_BLOCK_AWB_GAINS ?
> + MALI_C55_REG_AWB_GAINS1 :
> + MALI_C55_REG_AWB_GAINS1_AEXP;
> + u32 addr2 = block->type = MALI_C55_PARAM_BLOCK_AWB_GAINS ?
> + MALI_C55_REG_AWB_GAINS2 :
> + MALI_C55_REG_AWB_GAINS2_AEXP;
> +
> + mali_c55_update_bits(mali_c55, addr1, MALI_C55_AWB_GAIN00_MASK,
> + gains->gain00);
> + mali_c55_update_bits(mali_c55, addr1, MALI_C55_AWB_GAIN01_MASK,
> + gains->gain01);
> + mali_c55_update_bits(mali_c55, addr2, MALI_C55_AWB_GAIN10_MASK,
> + gains->gain10);
> + mali_c55_update_bits(mali_c55, addr2, MALI_C55_AWB_GAIN11_MASK,
> + gains->gain11);
> +}
> +
> +static void mali_c55_params_awb_config(struct mali_c55 *mali_c55,
> + struct mali_c55_params_block_header *block)
> +{
> + struct mali_c55_params_awb_config *params =
> + (struct mali_c55_params_awb_config *)block;
> +
> + if (!block->enabled) {
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_METERING_CONFIG,
> + MALI_C55_AWB_DISABLE_MASK, true);
> + return;
> + }
> +
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_METERING_CONFIG,
> + MALI_C55_AWB_DISABLE_MASK, false);
> +
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_AWB_STATS_MODE,
> + MALI_C55_AWB_STATS_MODE_MASK, params->stats_mode);
> +
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_AWB_WHITE_LEVEL,
> + MALI_C55_AWB_WHITE_LEVEL_MASK, params->white_level);
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_AWB_BLACK_LEVEL,
> + MALI_C55_AWB_BLACK_LEVEL_MASK, params->black_level);
> +
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_AWB_CR_MAX,
> + MALI_C55_AWB_CR_MAX_MASK, params->cr_max);
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_AWB_CR_MIN,
> + MALI_C55_AWB_CR_MIN_MASK, params->cr_min);
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_AWB_CB_MAX,
> + MALI_C55_AWB_CB_MAX_MASK, params->cb_max);
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_AWB_CB_MIN,
> + MALI_C55_AWB_CB_MIN_MASK, params->cb_min);
> +
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_AWB_NODES_USED,
> + MALI_C55_AWB_NODES_USED_HORIZ_MASK,
> + params->nodes_used_horiz);
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_AWB_NODES_USED,
> + MALI_C55_AWB_NODES_USED_VERT_MASK,
> + params->nodes_used_vert);
> +
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_AWB_CR_HIGH,
> + MALI_C55_AWB_CR_HIGH_MASK, params->cr_high);
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_AWB_CR_LOW,
> + MALI_C55_AWB_CR_LOW_MASK, params->cr_low);
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_AWB_CB_HIGH,
> + MALI_C55_AWB_CB_HIGH_MASK, params->cb_high);
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_AWB_CB_LOW,
> + MALI_C55_AWB_CB_LOW_MASK, params->cb_low);
> +
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_METERING_CONFIG,
> + MALI_C55_AWB_SWITCH_MASK, params->tap_point);
> +}
> +
> +static void mali_c55_params_lsc_config(struct mali_c55 *mali_c55,
> + struct mali_c55_params_block_header *block)
> +{
> + struct mali_c55_params_mesh_shading_config *params =
> + (struct mali_c55_params_mesh_shading_config *)block;
> + unsigned int i;
> + u32 addr;
> +
> + if (!block->enabled) {
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_MESH_SHADING_CONFIG,
> + MALI_C55_MESH_SHADING_ENABLE_MASK, false);
> + return;
> + }
> +
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_MESH_SHADING_CONFIG,
> + MALI_C55_MESH_SHADING_ENABLE_MASK, true);
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_MESH_SHADING_CONFIG,
> + MALI_C55_MESH_SHADING_MESH_SHOW, params->mesh_show);
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_MESH_SHADING_CONFIG,
> + MALI_C55_MESH_SHADING_SCALE_MASK,
> + params->mesh_scale);
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_MESH_SHADING_CONFIG,
> + MALI_C55_MESH_SHADING_PAGE_R_MASK,
> + params->mesh_page_r);
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_MESH_SHADING_CONFIG,
> + MALI_C55_MESH_SHADING_PAGE_G_MASK,
> + params->mesh_page_g);
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_MESH_SHADING_CONFIG,
> + MALI_C55_MESH_SHADING_PAGE_B_MASK,
> + params->mesh_page_b);
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_MESH_SHADING_CONFIG,
> + MALI_C55_MESH_SHADING_MESH_WIDTH_MASK,
> + params->mesh_width);
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_MESH_SHADING_CONFIG,
> + MALI_C55_MESH_SHADING_MESH_HEIGHT_MASK,
> + params->mesh_height);
> +
> + for (i = 0; i < MALI_C55_NUM_MESH_SHADING_ELEMENTS; i++) {
> + addr = MALI_C55_REG_MESH_SHADING_TABLES + (i * 4);
> + mali_c55_write(mali_c55, addr, params->mesh[i]);
> + }
> +}
> +
> +static void mali_c55_params_lsc_selection(struct mali_c55 *mali_c55,
> + struct mali_c55_params_block_header *block)
> +{
> + struct mali_c55_params_mesh_shading_selection *params =
> + (struct mali_c55_params_mesh_shading_selection *)block;
> +
> + if (!block->enabled)
> + return;
> +
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_MESH_SHADING_ALPHA_BANK,
> + MALI_C55_MESH_SHADING_ALPHA_BANK_R_MASK,
> + params->mesh_alpha_bank_r);
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_MESH_SHADING_ALPHA_BANK,
> + MALI_C55_MESH_SHADING_ALPHA_BANK_G_MASK,
> + params->mesh_alpha_bank_g);
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_MESH_SHADING_ALPHA_BANK,
> + MALI_C55_MESH_SHADING_ALPHA_BANK_B_MASK,
> + params->mesh_alpha_bank_b);
> +
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_MESH_SHADING_ALPHA,
> + MALI_C55_MESH_SHADING_ALPHA_R_MASK,
> + params->mesh_alpha_r);
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_MESH_SHADING_ALPHA,
> + MALI_C55_MESH_SHADING_ALPHA_G_MASK,
> + params->mesh_alpha_g);
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_MESH_SHADING_ALPHA,
> + MALI_C55_MESH_SHADING_ALPHA_B_MASK,
> + params->mesh_alpha_b);
> +
> + mali_c55_update_bits(mali_c55, MALI_C55_REG_MESH_SHADING_MESH_STRENGTH,
> + MALI_c55_MESH_STRENGTH_MASK,
> + params->mesh_strength);
> +}
> +
> +static const struct mali_c55_block_handler mali_c55_block_handlers[] = {
> + [MALI_C55_PARAM_BLOCK_SENSOR_OFFS] = {
> + .size = sizeof(struct mali_c55_params_sensor_off_preshading),
> + .handler = &mali_c55_params_sensor_offs,
> + },
> + [MALI_C55_PARAM_BLOCK_AEXP_HIST] = {
> + .size = sizeof(struct mali_c55_params_aexp_hist),
> + .handler = &mali_c55_params_aexp_hist,
> + },
> + [MALI_C55_PARAM_BLOCK_AEXP_IHIST] = {
> + .size = sizeof(struct mali_c55_params_aexp_hist),
> + .handler = &mali_c55_params_aexp_hist,
> + },
> + [MALI_C55_PARAM_BLOCK_AEXP_HIST_WEIGHTS] = {
> + .size = sizeof(struct mali_c55_params_aexp_weights),
> + .handler = &mali_c55_params_aexp_hist_weights,
> + },
> + [MALI_C55_PARAM_BLOCK_AEXP_IHIST_WEIGHTS] = {
> + .size = sizeof(struct mali_c55_params_aexp_weights),
> + .handler = &mali_c55_params_aexp_hist_weights,
> + },
> + [MALI_C55_PARAM_BLOCK_DIGITAL_GAIN] = {
> + .size = sizeof(struct mali_c55_params_digital_gain),
> + .handler = &mali_c55_params_digital_gain,
> + },
> + [MALI_C55_PARAM_BLOCK_AWB_GAINS] = {
> + .size = sizeof(struct mali_c55_params_awb_gains),
> + .handler = &mali_c55_params_awb_gains,
> + },
> + [MALI_C55_PARAM_BLOCK_AWB_CONFIG] = {
> + .size = sizeof(struct mali_c55_params_awb_config),
> + .handler = &mali_c55_params_awb_config,
> + },
> + [MALI_C55_PARAM_BLOCK_AWB_GAINS_AEXP] = {
> + .size = sizeof(struct mali_c55_params_awb_gains),
> + .handler = &mali_c55_params_awb_gains,
> + },
> + [MALI_C55_PARAM_MESH_SHADING_CONFIG] = {
> + .size = sizeof(struct mali_c55_params_mesh_shading_config),
> + .handler = &mali_c55_params_lsc_config,
> + },
> + [MALI_C55_PARAM_MESH_SHADING_SELECTION] = {
> + .size = sizeof(struct mali_c55_params_mesh_shading_selection),
> + .handler = &mali_c55_params_lsc_selection,
> + },
> +};
> +
> +static int mali_c55_params_enum_fmt_meta_out(struct file *file, void *fh,
> + struct v4l2_fmtdesc *f)
> +{
> + if (f->index || f->type != V4L2_BUF_TYPE_META_OUTPUT)
The buffer type check has been done by the caller already.
> + return -EINVAL;
> +
> + f->pixelformat = V4L2_META_FMT_MALI_C55_PARAMS;
> +
> + return 0;
> +}
> +
> +static int mali_c55_params_g_fmt_meta_out(struct file *file, void *fh,
> + struct v4l2_format *f)
> +{
> + static const struct v4l2_meta_format mfmt = {
> + .dataformat = V4L2_META_FMT_MALI_C55_PARAMS,
> + .buffersize = sizeof(struct mali_c55_params_buffer),
> + };
> +
> + if (f->type != V4L2_BUF_TYPE_META_OUTPUT)
> + return -EINVAL;
Ditto.
Maybe check the other instances of format access functions in the driver,
too?
> +
> + f->fmt.meta = mfmt;
> +
> + return 0;
> +}
> +
> +static int mali_c55_params_querycap(struct file *file,
> + void *priv, struct v4l2_capability *cap)
> +{
> + strscpy(cap->driver, MALI_C55_DRIVER_NAME, sizeof(cap->driver));
> + strscpy(cap->card, "ARM Mali-C55 ISP", sizeof(cap->card));
> +
> + return 0;
> +}
> +
> +static const struct v4l2_ioctl_ops mali_c55_params_v4l2_ioctl_ops = {
> + .vidioc_reqbufs = vb2_ioctl_reqbufs,
> + .vidioc_querybuf = vb2_ioctl_querybuf,
> + .vidioc_create_bufs = vb2_ioctl_create_bufs,
> + .vidioc_qbuf = vb2_ioctl_qbuf,
> + .vidioc_expbuf = vb2_ioctl_expbuf,
> + .vidioc_dqbuf = vb2_ioctl_dqbuf,
> + .vidioc_prepare_buf = vb2_ioctl_prepare_buf,
> + .vidioc_streamon = vb2_ioctl_streamon,
> + .vidioc_streamoff = vb2_ioctl_streamoff,
> + .vidioc_enum_fmt_meta_out = mali_c55_params_enum_fmt_meta_out,
> + .vidioc_g_fmt_meta_out = mali_c55_params_g_fmt_meta_out,
> + .vidioc_s_fmt_meta_out = mali_c55_params_g_fmt_meta_out,
> + .vidioc_try_fmt_meta_out = mali_c55_params_g_fmt_meta_out,
> + .vidioc_querycap = mali_c55_params_querycap,
> + .vidioc_subscribe_event = v4l2_ctrl_subscribe_event,
> + .vidioc_unsubscribe_event = v4l2_event_unsubscribe,
> +};
> +
> +static const struct v4l2_file_operations mali_c55_params_v4l2_fops = {
> + .owner = THIS_MODULE,
> + .unlocked_ioctl = video_ioctl2,
> + .open = v4l2_fh_open,
> + .release = vb2_fop_release,
> + .poll = vb2_fop_poll,
> + .mmap = vb2_fop_mmap,
> +};
> +
> +static int
> +mali_c55_params_queue_setup(struct vb2_queue *q, unsigned int *num_buffers,
> + unsigned int *num_planes, unsigned int sizes[],
> + struct device *alloc_devs[])
> +{
> + if (*num_planes && *num_planes > 1)
> + return -EINVAL;
> +
> + if (sizes[0] && sizes[0] != sizeof(struct mali_c55_params_buffer))
> + return -EINVAL;
> +
> + *num_planes = 1;
> + sizes[0] = sizeof(struct mali_c55_params_buffer);
> +
> + return 0;
> +}
> +
> +static void mali_c55_params_buf_queue(struct vb2_buffer *vb)
> +{
> + struct mali_c55_params *params = vb2_get_drv_priv(vb->vb2_queue);
> + struct vb2_v4l2_buffer *vbuf = to_vb2_v4l2_buffer(vb);
> + struct mali_c55_buffer *buf = container_of(vbuf,
> + struct mali_c55_buffer, vb);
> +
> + vb2_set_plane_payload(vb, 0, sizeof(struct mali_c55_params_buffer));
> +
> + spin_lock(¶ms->buffers.lock);
> + list_add_tail(&buf->queue, ¶ms->buffers.queue);
> + spin_unlock(¶ms->buffers.lock);
> +}
> +
> +static void mali_c55_params_stop_streaming(struct vb2_queue *q)
> +{
> + struct mali_c55_params *params = vb2_get_drv_priv(q);
> + struct mali_c55_buffer *buf, *tmp;
> +
> + spin_lock(¶ms->buffers.lock);
> +
> + list_for_each_entry_safe(buf, tmp, ¶ms->buffers.queue, queue) {
> + list_del(&buf->queue);
> + vb2_buffer_done(&buf->vb.vb2_buf, VB2_BUF_STATE_ERROR);
> + }
> +
> + spin_unlock(¶ms->buffers.lock);
> +}
> +
> +static const struct vb2_ops mali_c55_params_vb2_ops = {
> + .queue_setup = mali_c55_params_queue_setup,
> + .buf_queue = mali_c55_params_buf_queue,
> + .wait_prepare = vb2_ops_wait_prepare,
> + .wait_finish = vb2_ops_wait_finish,
> + .stop_streaming = mali_c55_params_stop_streaming,
> +};
> +
> +void mali_c55_params_write_config(struct mali_c55 *mali_c55)
> +{
> + struct mali_c55_params *params = &mali_c55->params;
> + enum vb2_buffer_state state = VB2_BUF_STATE_DONE;
> + struct mali_c55_params_buffer *config;
> + struct mali_c55_buffer *buf;
> + size_t block_offset = 0;
> +
> + spin_lock(¶ms->buffers.lock);
> +
> + buf = list_first_entry_or_null(¶ms->buffers.queue,
> + struct mali_c55_buffer, queue);
> + if (buf)
> + list_del(&buf->queue);
> + spin_unlock(¶ms->buffers.lock);
> +
> + if (!buf)
> + return;
> +
> + buf->vb.sequence = mali_c55->isp.frame_sequence;
> + config = vb2_plane_vaddr(&buf->vb.vb2_buf, 0);
> +
> + if (config->total_size > MALI_C55_PARAMS_MAX_SIZE) {
> + dev_dbg(mali_c55->dev, "Invalid parameters buffer size %lu\n",
> + config->total_size);
> + state = VB2_BUF_STATE_ERROR;
> + goto err_buffer_done;
> + }
> +
> + /* Walk the list of parameter blocks and process them. */
> + while (block_offset < config->total_size) {
> + const struct mali_c55_block_handler *block_handler;
> + struct mali_c55_params_block_header *block;
> +
> + block = (struct mali_c55_params_block_header *)
> + &config->data[block_offset];
How do you ensure config->data does hold a full struct
mali_c33_params_block_header at block_offset (i.e. that the struct does not
exceed the memory available for config->data)?
> +
> + if (block->type >= MALI_C55_PARAM_BLOCK_SENTINEL) {
> + dev_dbg(mali_c55->dev, "Invalid parameters block type\n");
> + state = VB2_BUF_STATE_ERROR;
> + goto err_buffer_done;
> + }
> +
> + block_handler = &mali_c55_block_handlers[block->type];
> + if (block->size != block_handler->size) {
How do you ensure config->data has room for the block?
> + dev_dbg(mali_c55->dev, "Invalid parameters block size\n");
> + state = VB2_BUF_STATE_ERROR;
> + goto err_buffer_done;
> + }
> +
> + block_handler->handler(mali_c55, block);
> +
> + block_offset += block->size;
> + }
> +
> +err_buffer_done:
> + vb2_buffer_done(&buf->vb.vb2_buf, state);
> +}
> +
> +void mali_c55_unregister_params(struct mali_c55 *mali_c55)
> +{
> + struct mali_c55_params *params = &mali_c55->params;
> +
> + if (!video_is_registered(¶ms->vdev))
> + return;
> +
> + vb2_video_unregister_device(¶ms->vdev);
> + media_entity_cleanup(¶ms->vdev.entity);
> + mutex_destroy(¶ms->lock);
> +}
> +
> +int mali_c55_register_params(struct mali_c55 *mali_c55)
> +{
> + struct mali_c55_params *params = &mali_c55->params;
> + struct video_device *vdev = ¶ms->vdev;
> + struct vb2_queue *vb2q = ¶ms->queue;
> + int ret;
> +
> + mutex_init(¶ms->lock);
> + INIT_LIST_HEAD(¶ms->buffers.queue);
> +
> + params->pad.flags = MEDIA_PAD_FL_SOURCE;
> + ret = media_entity_pads_init(¶ms->vdev.entity, 1, ¶ms->pad);
> + if (ret)
> + goto err_destroy_mutex;
> +
> + vb2q->type = V4L2_BUF_TYPE_META_OUTPUT;
> + vb2q->io_modes = VB2_MMAP | VB2_DMABUF;
> + vb2q->drv_priv = params;
> + vb2q->mem_ops = &vb2_dma_contig_memops;
> + vb2q->ops = &mali_c55_params_vb2_ops;
> + vb2q->buf_struct_size = sizeof(struct mali_c55_buffer);
> + vb2q->min_queued_buffers = 1;
> + vb2q->timestamp_flags = V4L2_BUF_FLAG_TIMESTAMP_MONOTONIC;
> + vb2q->lock = ¶ms->lock;
> + vb2q->dev = mali_c55->dev;
> +
> + ret = vb2_queue_init(vb2q);
> + if (ret) {
> + dev_err(mali_c55->dev, "params vb2 queue init failed\n");
> + goto err_cleanup_entity;
> + }
> +
> + strscpy(params->vdev.name, "mali-c55 3a params",
> + sizeof(params->vdev.name));
> + vdev->release = video_device_release_empty;
> + vdev->fops = &mali_c55_params_v4l2_fops;
> + vdev->ioctl_ops = &mali_c55_params_v4l2_ioctl_ops;
> + vdev->lock = ¶ms->lock;
> + vdev->v4l2_dev = &mali_c55->v4l2_dev;
> + vdev->queue = ¶ms->queue;
> + vdev->device_caps = V4L2_CAP_META_OUTPUT | V4L2_CAP_STREAMING;
> + vdev->vfl_dir = VFL_DIR_TX;
> + video_set_drvdata(vdev, params);
> +
> + ret = video_register_device(vdev, VFL_TYPE_VIDEO, -1);
> + if (ret) {
> + dev_err(mali_c55->dev,
> + "failed to register params video device\n");
> + goto err_release_vb2q;
> + }
> +
> + params->mali_c55 = mali_c55;
> +
> + return 0;
> +
> +err_release_vb2q:
> + vb2_queue_release(vb2q);
> +err_cleanup_entity:
> + media_entity_cleanup(¶ms->vdev.entity);
> +err_destroy_mutex:
> + mutex_destroy(¶ms->lock);
> +
> + return ret;
> +}
> diff --git a/drivers/media/platform/arm/mali-c55/mali-c55-registers.h b/drivers/media/platform/arm/mali-c55/mali-c55-registers.h
> index eb3719245ec3..8e6a801077ed 100644
> --- a/drivers/media/platform/arm/mali-c55/mali-c55-registers.h
> +++ b/drivers/media/platform/arm/mali-c55/mali-c55-registers.h
> @@ -119,6 +119,19 @@
> #define MALI_C55_REG_ACTIVE_HEIGHT_MASK 0xffff0000
> #define MALI_C55_REG_BAYER_ORDER 0x18e8c
> #define MALI_C55_BAYER_ORDER_MASK GENMASK(1, 0)
> +
> +#define MALI_C55_REG_METERING_CONFIG 0x18ed0
> +#define MALI_C55_5BIN_HIST_DISABLE_MASK BIT(0)
> +#define MALI_C55_5BIN_HIST_SWITCH_MASK GENMASK(2, 1)
> +#define MALI_C55_AF_DISABLE_MASK BIT(4)
> +#define MALI_C55_AF_SWITCH_MASK BIT(5)
> +#define MALI_C55_AWB_DISABLE_MASK BIT(8)
> +#define MALI_C55_AWB_SWITCH_MASK BIT(9)
> +#define MALI_C55_AEXP_HIST_DISABLE_MASK BIT(12)
> +#define MALI_C55_AEXP_HIST_SWITCH_MASK GENMASK(14, 13)
> +#define MALI_C55_AEXP_IHIST_DISABLE_MASK BIT(16)
> +#define MALI_C55_AEXP_SRC_MASK BIT(24)
> +
> #define MALI_C55_REG_TPG_CH0 0x18ed8
> #define MALI_C55_TEST_PATTERN_ON_OFF BIT(0)
> #define MALI_C55_TEST_PATTERN_RGB_MASK BIT(1)
> @@ -138,6 +151,11 @@
> #define MALI_C55_REG_CONFIG_SPACES_OFFSET 0x0ab6c
> #define MALI_C55_CONFIG_SPACE_SIZE 0x1231c
>
> +#define MALI_C55_REG_DIGITAL_GAIN 0x1926c
> +#define MALI_C55_DIGITAL_GAIN_MASK GENMASK(12, 0)
> +#define MALI_C55_REG_DIGITAL_GAIN_OFFSET 0x19270
> +#define MALI_C55_DIGITAL_GAIN_OFFSET_MASK GENMASK(19, 0)
> +
> #define MALI_C55_REG_SINTER_CONFIG 0x19348
> #define MALI_C55_SINTER_VIEW_FILTER_MASK GENMASK(1, 0)
> #define MALI_C55_SINTER_SCALE_MODE_MASK GENMASK(3, 2)
> @@ -146,6 +164,46 @@
> #define MALI_C55_SINTER_INT_SELECT_MASK BIT(6)
> #define MALI_C55_SINTER_RM_ENABLE_MASK BIT(7)
>
> +/* Black Level Correction Configuration */
> +#define MALI_C55_REG_SENSOR_OFF_PRE_SHA_00 0x1abcc
> +#define MALI_C55_REG_SENSOR_OFF_PRE_SHA_01 0x1abd0
> +#define MALI_C55_REG_SENSOR_OFF_PRE_SHA_10 0x1abd4
> +#define MALI_C55_REG_SENSOR_OFF_PRE_SHA_11 0x1abd8
> +#define MALI_C55_SENSOR_OFF_PRE_SHA_MASK 0xfffff
> +
> +/* Lens Mesh Shading Configuration */
> +#define MALI_C55_REG_MESH_SHADING_TABLES 0x13074
> +#define MALI_C55_REG_MESH_SHADING_CONFIG 0x1abfc
> +#define MALI_C55_MESH_SHADING_ENABLE_MASK BIT(0)
> +#define MALI_C55_MESH_SHADING_MESH_SHOW BIT(1)
> +#define MALI_C55_MESH_SHADING_SCALE_MASK GENMASK(4, 2)
> +#define MALI_C55_MESH_SHADING_PAGE_R_MASK GENMASK(9, 8)
> +#define MALI_C55_MESH_SHADING_PAGE_G_MASK GENMASK(11, 10)
> +#define MALI_C55_MESH_SHADING_PAGE_B_MASK GENMASK(13, 12)
> +#define MALI_C55_MESH_SHADING_MESH_WIDTH_MASK GENMASK(21, 16)
> +#define MALI_C55_MESH_SHADING_MESH_HEIGHT_MASK GENMASK(29, 24)
> +
> +#define MALI_C55_REG_MESH_SHADING_ALPHA_BANK 0x1ac04
> +#define MALI_C55_MESH_SHADING_ALPHA_BANK_R_MASK GENMASK(2, 0)
> +#define MALI_C55_MESH_SHADING_ALPHA_BANK_G_MASK GENMASK(5, 3)
> +#define MALI_C55_MESH_SHADING_ALPHA_BANK_B_MASK GENMASK(8, 6)
> +#define MALI_C55_REG_MESH_SHADING_ALPHA 0x1ac08
> +#define MALI_C55_MESH_SHADING_ALPHA_R_MASK GENMASK(7, 0)
> +#define MALI_C55_MESH_SHADING_ALPHA_G_MASK GENMASK(15, 8)
> +#define MALI_C55_MESH_SHADING_ALPHA_B_MASK GENMASK(23, 16)
> +#define MALI_C55_REG_MESH_SHADING_MESH_STRENGTH 0x1ac0c
> +#define MALI_c55_MESH_STRENGTH_MASK GENMASK(15, 0)
> +
> +/* AWB Gains Configuration */
> +#define MALI_C55_REG_AWB_GAINS1 0x1ac10
> +#define MALI_C55_AWB_GAIN00_MASK GENMASK(11, 0)
> +#define MALI_C55_AWB_GAIN01_MASK GENMASK(27, 16)
> +#define MALI_C55_REG_AWB_GAINS2 0x1ac14
> +#define MALI_C55_AWB_GAIN10_MASK GENMASK(11, 0)
> +#define MALI_C55_AWB_GAIN11_MASK GENMASK(27, 16)
> +#define MALI_C55_REG_AWB_GAINS1_AEXP 0x1ac18
> +#define MALI_C55_REG_AWB_GAINS2_AEXP 0x1ac1c
> +
> /* Colour Correction Matrix Configuration */
> #define MALI_C55_REG_CCM_ENABLE 0x1b07c
> #define MALI_C55_CCM_ENABLE_MASK BIT(0)
> @@ -168,6 +226,52 @@
> #define MALI_C55_REG_CCM_ANTIFOG_OFFSET_B 0x1b0c8
> #define MALI_C55_CCM_ANTIFOG_OFFSET_MASK GENMASK(11, 0)
>
> +/* AWB Statistics Configuration */
> +#define MALI_C55_REG_AWB_STATS_MODE 0x1b29c
> +#define MALI_C55_AWB_STATS_MODE_MASK BIT(0)
> +#define MALI_C55_REG_AWB_WHITE_LEVEL 0x1b2a0
> +#define MALI_C55_AWB_WHITE_LEVEL_MASK GENMASK(9, 0)
> +#define MALI_C55_REG_AWB_BLACK_LEVEL 0x1b2a4
> +#define MALI_C55_AWB_BLACK_LEVEL_MASK GENMASK(9, 0)
> +#define MALI_C55_REG_AWB_CR_MAX 0x1b2a8
> +#define MALI_C55_AWB_CR_MAX_MASK GENMASK(11, 0)
> +#define MALI_C55_REG_AWB_CR_MIN 0x1b2ac
> +#define MALI_C55_AWB_CR_MIN_MASK GENMASK(11, 0)
> +#define MALI_C55_REG_AWB_CB_MAX 0x1b2b0
> +#define MALI_C55_REG_AWB_CB_MIN 0x1b2b4
> +#define MALI_C55_AWB_CB_MIN_MASK GENMASK(11, 0)
> +#define MALI_C55_REG_AWB_NODES_USED 0x1b2c4
> +#define MALI_C55_AWB_NODES_USED_HORIZ_MASK GENMASK(7, 0)
> +#define MALI_C55_AWB_NODES_USED_VERT_MASK GENMASK(15, 8)
> +#define MALI_C55_REG_AWB_CR_HIGH 0x1b2c8
> +#define MALI_C55_AWB_CR_HIGH_MASK GENMASK(11, 0)
> +#define MALI_C55_REG_AWB_CR_LOW 0x1b2cc
> +#define MALI_C55_AWB_CR_LOW_MASK GENMASK(11, 0)
> +#define MALI_C55_REG_AWB_CB_HIGH 0x1b2d0
> +#define MALI_C55_AWB_CB_HIGH_MASK GENMASK(11, 0)
> +#define MALI_C55_REG_AWB_CB_LOW 0x1b2d4
> +#define MALI_C55_AWB_CB_LOW_MASK GENMASK(11, 0)
> +
> +/* AEXP Metering Histogram Configuration */
> +#define MALI_C55_REG_AEXP_HIST_BASE 0x1b730
> +#define MALI_C55_REG_AEXP_IHIST_BASE 0x1bbac
> +#define MALI_C55_AEXP_HIST_SKIP_OFFSET 0
> +#define MALI_C55_AEXP_HIST_SKIP_X_MASK GENMASK(2, 0)
> +#define MALI_C55_AEXP_HIST_OFFSET_X_MASK BIT(3)
> +#define MALI_C55_AEXP_HIST_SKIP_Y_MASK GENMASK(6, 4)
> +#define MALI_C55_AEXP_HIST_OFFSET_Y_MASK BIT(7)
> +#define MALI_C55_AEXP_HIST_SCALE_OFFSET 4
> +#define MALI_C55_AEXP_HIST_SCALE_BOTTOM_MASK GENMASK(3, 0)
> +#define MALI_C55_AEXP_HIST_SCALE_TOP_MASK GENMASK(7, 4)
> +#define MALI_C55_AEXP_HIST_PLANE_MODE_OFFSET 16
> +#define MALI_C55_AEXP_HIST_PLANE_MODE_MASK GENMASK(2, 0)
> +#define MALI_C55_AEXP_HIST_NODES_USED_OFFSET 52
> +#define MALI_C55_AEXP_HIST_NODES_USED_HORIZ_MASK GENMASK(7, 0)
> +#define MALI_C55_AEXP_HIST_NODES_USED_VERT_MASK GENMASK(15, 8)
> +#define MALI_C55_AEXP_HIST_ZONE_WEIGHTS_OFFSET 56
> +#define MALI_C55_AEXP_HIST_ZONE_WEIGHT_MASK 0x0f0f0f0f
> +
> /*
> * The Mali-C55 ISP has up to two output pipes; known as full resolution and
> * down scaled. The register space for these is laid out identically, but offset
--
Kind regards,
Sakari Ailus
More information about the linux-arm-kernel
mailing list