1 // SPDX-License-Identifier: GPL-2.0-only
3 * Copyright (c) 2023, The Linux Foundation. All rights reserved.
6 #include <linux/bitfield.h>
8 #include <drm/drm_managed.h>
10 #include "dpu_hw_mdss.h"
11 #include "dpu_hw_util.h"
12 #include "dpu_hw_catalog.h"
13 #include "dpu_hw_cdm.h"
16 #define CDM_CSC_10_OPMODE 0x000
17 #define CDM_CSC_10_BASE 0x004
19 #define CDM_CDWN2_OP_MODE 0x100
20 #define CDM_CDWN2_CLAMP_OUT 0x104
21 #define CDM_CDWN2_PARAMS_3D_0 0x108
22 #define CDM_CDWN2_PARAMS_3D_1 0x10C
23 #define CDM_CDWN2_COEFF_COSITE_H_0 0x110
24 #define CDM_CDWN2_COEFF_COSITE_H_1 0x114
25 #define CDM_CDWN2_COEFF_COSITE_H_2 0x118
26 #define CDM_CDWN2_COEFF_OFFSITE_H_0 0x11C
27 #define CDM_CDWN2_COEFF_OFFSITE_H_1 0x120
28 #define CDM_CDWN2_COEFF_OFFSITE_H_2 0x124
29 #define CDM_CDWN2_COEFF_COSITE_V 0x128
30 #define CDM_CDWN2_COEFF_OFFSITE_V 0x12C
31 #define CDM_CDWN2_OUT_SIZE 0x130
33 #define CDM_HDMI_PACK_OP_MODE 0x200
34 #define CDM_CSC_10_MATRIX_COEFF_0 0x004
38 /* CDM CDWN2 sub-block bit definitions */
39 #define CDM_CDWN2_OP_MODE_EN BIT(0)
40 #define CDM_CDWN2_OP_MODE_ENABLE_H BIT(1)
41 #define CDM_CDWN2_OP_MODE_ENABLE_V BIT(2)
42 #define CDM_CDWN2_OP_MODE_BITS_OUT_8BIT BIT(7)
43 #define CDM_CDWN2_V_PIXEL_METHOD_MASK GENMASK(6, 5)
44 #define CDM_CDWN2_H_PIXEL_METHOD_MASK GENMASK(4, 3)
46 /* CDM CSC10 sub-block bit definitions */
47 #define CDM_CSC10_OP_MODE_EN BIT(0)
48 #define CDM_CSC10_OP_MODE_SRC_FMT_YUV BIT(1)
49 #define CDM_CSC10_OP_MODE_DST_FMT_YUV BIT(2)
51 /* CDM HDMI pack sub-block bit definitions */
52 #define CDM_HDMI_PACK_OP_MODE_EN BIT(0)
55 * Horizontal coefficients for cosite chroma downscale
56 * s13 representation of coefficients
58 static u32 cosite_h_coeff[] = {0x00000016, 0x000001cc, 0x0100009e};
61 * Horizontal coefficients for offsite chroma downscale
63 static u32 offsite_h_coeff[] = {0x000b0005, 0x01db01eb, 0x00e40046};
66 * Vertical coefficients for cosite chroma downscale
68 static u32 cosite_v_coeff[] = {0x00080004};
70 * Vertical coefficients for offsite chroma downscale
72 static u32 offsite_v_coeff[] = {0x00060002};
74 static int dpu_hw_cdm_setup_cdwn(struct dpu_hw_cdm *ctx, struct dpu_hw_cdm_cfg *cfg)
76 struct dpu_hw_blk_reg_map *c = &ctx->hw;
80 switch (cfg->h_cdwn_type) {
81 case CDM_CDWN_DISABLE:
84 case CDM_CDWN_PIXEL_DROP:
85 opmode = CDM_CDWN2_OP_MODE_ENABLE_H |
86 FIELD_PREP(CDM_CDWN2_H_PIXEL_METHOD_MASK,
87 CDM_CDWN2_METHOD_PIXEL_DROP);
90 opmode = CDM_CDWN2_OP_MODE_ENABLE_H |
91 FIELD_PREP(CDM_CDWN2_H_PIXEL_METHOD_MASK,
92 CDM_CDWN2_METHOD_AVG);
95 opmode = CDM_CDWN2_OP_MODE_ENABLE_H |
96 FIELD_PREP(CDM_CDWN2_H_PIXEL_METHOD_MASK,
97 CDM_CDWN2_METHOD_COSITE);
98 DPU_REG_WRITE(c, CDM_CDWN2_COEFF_COSITE_H_0,
100 DPU_REG_WRITE(c, CDM_CDWN2_COEFF_COSITE_H_1,
102 DPU_REG_WRITE(c, CDM_CDWN2_COEFF_COSITE_H_2,
105 case CDM_CDWN_OFFSITE:
106 opmode = CDM_CDWN2_OP_MODE_ENABLE_H |
107 FIELD_PREP(CDM_CDWN2_H_PIXEL_METHOD_MASK, CDM_CDWN2_METHOD_OFFSITE);
108 DPU_REG_WRITE(c, CDM_CDWN2_COEFF_OFFSITE_H_0,
110 DPU_REG_WRITE(c, CDM_CDWN2_COEFF_OFFSITE_H_1,
112 DPU_REG_WRITE(c, CDM_CDWN2_COEFF_OFFSITE_H_2,
116 DPU_ERROR("%s invalid horz down sampling type\n", __func__);
120 switch (cfg->v_cdwn_type) {
121 case CDM_CDWN_DISABLE:
122 /* if its only Horizontal downsample, we dont need to do anything here */
124 case CDM_CDWN_PIXEL_DROP:
125 opmode |= CDM_CDWN2_OP_MODE_ENABLE_V |
126 FIELD_PREP(CDM_CDWN2_V_PIXEL_METHOD_MASK,
127 CDM_CDWN2_METHOD_PIXEL_DROP);
130 opmode |= CDM_CDWN2_OP_MODE_ENABLE_V |
131 FIELD_PREP(CDM_CDWN2_V_PIXEL_METHOD_MASK,
132 CDM_CDWN2_METHOD_AVG);
134 case CDM_CDWN_COSITE:
135 opmode |= CDM_CDWN2_OP_MODE_ENABLE_V |
136 FIELD_PREP(CDM_CDWN2_V_PIXEL_METHOD_MASK,
137 CDM_CDWN2_METHOD_COSITE);
139 CDM_CDWN2_COEFF_COSITE_V,
142 case CDM_CDWN_OFFSITE:
143 opmode |= CDM_CDWN2_OP_MODE_ENABLE_V |
144 FIELD_PREP(CDM_CDWN2_V_PIXEL_METHOD_MASK,
145 CDM_CDWN2_METHOD_OFFSITE);
147 CDM_CDWN2_COEFF_OFFSITE_V,
154 if (cfg->output_bit_depth != CDM_CDWN_OUTPUT_10BIT)
155 opmode |= CDM_CDWN2_OP_MODE_BITS_OUT_8BIT;
157 if (cfg->v_cdwn_type || cfg->h_cdwn_type)
158 opmode |= CDM_CDWN2_OP_MODE_EN; /* EN CDWN module */
160 opmode &= ~CDM_CDWN2_OP_MODE_EN;
162 out_size = (cfg->output_width & 0xFFFF) | ((cfg->output_height & 0xFFFF) << 16);
163 DPU_REG_WRITE(c, CDM_CDWN2_OUT_SIZE, out_size);
164 DPU_REG_WRITE(c, CDM_CDWN2_OP_MODE, opmode);
165 DPU_REG_WRITE(c, CDM_CDWN2_CLAMP_OUT, ((0x3FF << 16) | 0x0));
170 static int dpu_hw_cdm_enable(struct dpu_hw_cdm *ctx, struct dpu_hw_cdm_cfg *cdm)
172 struct dpu_hw_blk_reg_map *c = &ctx->hw;
173 const struct msm_format *fmt;
180 fmt = cdm->output_fmt;
182 if (!MSM_FORMAT_IS_YUV(fmt))
185 dpu_hw_csc_setup(&ctx->hw, CDM_CSC_10_MATRIX_COEFF_0, cdm->csc_cfg, true);
186 dpu_hw_cdm_setup_cdwn(ctx, cdm);
188 if (cdm->output_type == CDM_CDWN_OUTPUT_HDMI) {
189 if (fmt->chroma_sample == CHROMA_H1V2)
190 return -EINVAL; /*unsupported format */
191 opmode = CDM_HDMI_PACK_OP_MODE_EN;
192 opmode |= (fmt->chroma_sample << 1);
195 csc |= CDM_CSC10_OP_MODE_DST_FMT_YUV;
196 csc &= ~CDM_CSC10_OP_MODE_SRC_FMT_YUV;
197 csc |= CDM_CSC10_OP_MODE_EN;
199 if (ctx && ctx->ops.bind_pingpong_blk)
200 ctx->ops.bind_pingpong_blk(ctx, cdm->pp_id);
202 DPU_REG_WRITE(c, CDM_CSC_10_OPMODE, csc);
203 DPU_REG_WRITE(c, CDM_HDMI_PACK_OP_MODE, opmode);
207 static void dpu_hw_cdm_bind_pingpong_blk(struct dpu_hw_cdm *ctx, const enum dpu_pingpong pp)
209 struct dpu_hw_blk_reg_map *c;
214 mux_cfg = DPU_REG_READ(c, CDM_MUX);
218 mux_cfg |= (pp - PINGPONG_0) & 0x7;
222 DPU_REG_WRITE(c, CDM_MUX, mux_cfg);
226 * dpu_hw_cdm_init - initializes the cdm hw driver object.
227 * should be called once before accessing every cdm.
228 * @dev: DRM device handle
229 * @cfg: CDM catalog entry for which driver object is required
230 * @addr : mapped register io address of MDSS
231 * @mdss_rev: mdss hw core revision
233 struct dpu_hw_cdm *dpu_hw_cdm_init(struct drm_device *dev,
234 const struct dpu_cdm_cfg *cfg, void __iomem *addr,
235 const struct dpu_mdss_version *mdss_rev)
237 struct dpu_hw_cdm *c;
239 c = drmm_kzalloc(dev, sizeof(*c), GFP_KERNEL);
241 return ERR_PTR(-ENOMEM);
243 c->hw.blk_addr = addr + cfg->base;
244 c->hw.log_mask = DPU_DBG_MASK_CDM;
250 c->ops.enable = dpu_hw_cdm_enable;
251 if (mdss_rev->core_major_ver >= 5)
252 c->ops.bind_pingpong_blk = dpu_hw_cdm_bind_pingpong_blk;