1 /*
2 * Copyright (C) Jernej Skrabec <jernej.skrabec@siol.net>
3 *
4 * This program is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU General Public License as
6 * published by the Free Software Foundation; either version 2 of
7 * the License, or (at your option) any later version.
8 */
9
10 #include <drm/drm_atomic.h>
11 #include <drm/drm_atomic_helper.h>
12 #include <drm/drm_crtc.h>
13 #include <drm/drm_crtc_helper.h>
14 #include <drm/drm_fb_cma_helper.h>
15 #include <drm/drm_gem_cma_helper.h>
16 #include <drm/drm_plane_helper.h>
17 #include <drm/drmP.h>
18
19 #include "sun8i_vi_layer.h"
20 #include "sun8i_mixer.h"
21 #include "sun8i_vi_scaler.h"
22
sun8i_vi_layer_enable(struct sun8i_mixer * mixer,int channel,int overlay,bool enable,unsigned int zpos,unsigned int old_zpos)23 static void sun8i_vi_layer_enable(struct sun8i_mixer *mixer, int channel,
24 int overlay, bool enable, unsigned int zpos,
25 unsigned int old_zpos)
26 {
27 u32 val;
28
29 DRM_DEBUG_DRIVER("%sabling VI channel %d overlay %d\n",
30 enable ? "En" : "Dis", channel, overlay);
31
32 if (enable)
33 val = SUN8I_MIXER_CHAN_VI_LAYER_ATTR_EN;
34 else
35 val = 0;
36
37 regmap_update_bits(mixer->engine.regs,
38 SUN8I_MIXER_CHAN_VI_LAYER_ATTR(channel, overlay),
39 SUN8I_MIXER_CHAN_VI_LAYER_ATTR_EN, val);
40
41 if (!enable || zpos != old_zpos) {
42 regmap_update_bits(mixer->engine.regs,
43 SUN8I_MIXER_BLEND_PIPE_CTL,
44 SUN8I_MIXER_BLEND_PIPE_CTL_EN(old_zpos),
45 0);
46
47 regmap_update_bits(mixer->engine.regs,
48 SUN8I_MIXER_BLEND_ROUTE,
49 SUN8I_MIXER_BLEND_ROUTE_PIPE_MSK(old_zpos),
50 0);
51 }
52
53 if (enable) {
54 val = SUN8I_MIXER_BLEND_PIPE_CTL_EN(zpos);
55
56 regmap_update_bits(mixer->engine.regs,
57 SUN8I_MIXER_BLEND_PIPE_CTL, val, val);
58
59 val = channel << SUN8I_MIXER_BLEND_ROUTE_PIPE_SHIFT(zpos);
60
61 regmap_update_bits(mixer->engine.regs,
62 SUN8I_MIXER_BLEND_ROUTE,
63 SUN8I_MIXER_BLEND_ROUTE_PIPE_MSK(zpos),
64 val);
65 }
66 }
67
sun8i_vi_layer_update_coord(struct sun8i_mixer * mixer,int channel,int overlay,struct drm_plane * plane,unsigned int zpos)68 static int sun8i_vi_layer_update_coord(struct sun8i_mixer *mixer, int channel,
69 int overlay, struct drm_plane *plane,
70 unsigned int zpos)
71 {
72 struct drm_plane_state *state = plane->state;
73 const struct drm_format_info *format = state->fb->format;
74 u32 src_w, src_h, dst_w, dst_h;
75 u32 outsize, insize;
76 u32 hphase, vphase;
77 bool subsampled;
78
79 DRM_DEBUG_DRIVER("Updating VI channel %d overlay %d\n",
80 channel, overlay);
81
82 src_w = drm_rect_width(&state->src) >> 16;
83 src_h = drm_rect_height(&state->src) >> 16;
84 dst_w = drm_rect_width(&state->dst);
85 dst_h = drm_rect_height(&state->dst);
86
87 hphase = state->src.x1 & 0xffff;
88 vphase = state->src.y1 & 0xffff;
89
90 /* make coordinates dividable by subsampling factor */
91 if (format->hsub > 1) {
92 int mask, remainder;
93
94 mask = format->hsub - 1;
95 remainder = (state->src.x1 >> 16) & mask;
96 src_w = (src_w + remainder) & ~mask;
97 hphase += remainder << 16;
98 }
99
100 if (format->vsub > 1) {
101 int mask, remainder;
102
103 mask = format->vsub - 1;
104 remainder = (state->src.y1 >> 16) & mask;
105 src_h = (src_h + remainder) & ~mask;
106 vphase += remainder << 16;
107 }
108
109 insize = SUN8I_MIXER_SIZE(src_w, src_h);
110 outsize = SUN8I_MIXER_SIZE(dst_w, dst_h);
111
112 /* Set height and width */
113 DRM_DEBUG_DRIVER("Layer source offset X: %d Y: %d\n",
114 (state->src.x1 >> 16) & ~(format->hsub - 1),
115 (state->src.y1 >> 16) & ~(format->vsub - 1));
116 DRM_DEBUG_DRIVER("Layer source size W: %d H: %d\n", src_w, src_h);
117 regmap_write(mixer->engine.regs,
118 SUN8I_MIXER_CHAN_VI_LAYER_SIZE(channel, overlay),
119 insize);
120 regmap_write(mixer->engine.regs,
121 SUN8I_MIXER_CHAN_VI_OVL_SIZE(channel),
122 insize);
123
124 /*
125 * Scaler must be enabled for subsampled formats, so it scales
126 * chroma to same size as luma.
127 */
128 subsampled = format->hsub > 1 || format->vsub > 1;
129
130 if (insize != outsize || subsampled || hphase || vphase) {
131 u32 hscale, vscale;
132
133 DRM_DEBUG_DRIVER("HW scaling is enabled\n");
134
135 hscale = state->src_w / state->crtc_w;
136 vscale = state->src_h / state->crtc_h;
137
138 sun8i_vi_scaler_setup(mixer, channel, src_w, src_h, dst_w,
139 dst_h, hscale, vscale, hphase, vphase,
140 format);
141 sun8i_vi_scaler_enable(mixer, channel, true);
142 } else {
143 DRM_DEBUG_DRIVER("HW scaling is not needed\n");
144 sun8i_vi_scaler_enable(mixer, channel, false);
145 }
146
147 /* Set base coordinates */
148 DRM_DEBUG_DRIVER("Layer destination coordinates X: %d Y: %d\n",
149 state->dst.x1, state->dst.y1);
150 DRM_DEBUG_DRIVER("Layer destination size W: %d H: %d\n", dst_w, dst_h);
151 regmap_write(mixer->engine.regs,
152 SUN8I_MIXER_BLEND_ATTR_COORD(zpos),
153 SUN8I_MIXER_COORD(state->dst.x1, state->dst.y1));
154 regmap_write(mixer->engine.regs,
155 SUN8I_MIXER_BLEND_ATTR_INSIZE(zpos),
156 outsize);
157
158 return 0;
159 }
160
sun8i_vi_layer_update_formats(struct sun8i_mixer * mixer,int channel,int overlay,struct drm_plane * plane)161 static int sun8i_vi_layer_update_formats(struct sun8i_mixer *mixer, int channel,
162 int overlay, struct drm_plane *plane)
163 {
164 struct drm_plane_state *state = plane->state;
165 const struct de2_fmt_info *fmt_info;
166 u32 val;
167
168 fmt_info = sun8i_mixer_format_info(state->fb->format->format);
169 if (!fmt_info) {
170 DRM_DEBUG_DRIVER("Invalid format\n");
171 return -EINVAL;
172 }
173
174 val = fmt_info->de2_fmt << SUN8I_MIXER_CHAN_VI_LAYER_ATTR_FBFMT_OFFSET;
175 regmap_update_bits(mixer->engine.regs,
176 SUN8I_MIXER_CHAN_VI_LAYER_ATTR(channel, overlay),
177 SUN8I_MIXER_CHAN_VI_LAYER_ATTR_FBFMT_MASK, val);
178
179 if (fmt_info->csc != SUN8I_CSC_MODE_OFF) {
180 sun8i_csc_set_ccsc_coefficients(mixer, channel, fmt_info->csc);
181 sun8i_csc_enable_ccsc(mixer, channel, true);
182 } else {
183 sun8i_csc_enable_ccsc(mixer, channel, false);
184 }
185
186 if (fmt_info->rgb)
187 val = SUN8I_MIXER_CHAN_VI_LAYER_ATTR_RGB_MODE;
188 else
189 val = 0;
190
191 regmap_update_bits(mixer->engine.regs,
192 SUN8I_MIXER_CHAN_VI_LAYER_ATTR(channel, overlay),
193 SUN8I_MIXER_CHAN_VI_LAYER_ATTR_RGB_MODE, val);
194
195 return 0;
196 }
197
sun8i_vi_layer_update_buffer(struct sun8i_mixer * mixer,int channel,int overlay,struct drm_plane * plane)198 static int sun8i_vi_layer_update_buffer(struct sun8i_mixer *mixer, int channel,
199 int overlay, struct drm_plane *plane)
200 {
201 struct drm_plane_state *state = plane->state;
202 struct drm_framebuffer *fb = state->fb;
203 const struct drm_format_info *format = fb->format;
204 struct drm_gem_cma_object *gem;
205 u32 dx, dy, src_x, src_y;
206 dma_addr_t paddr;
207 int i;
208
209 /* Adjust x and y to be dividable by subsampling factor */
210 src_x = (state->src.x1 >> 16) & ~(format->hsub - 1);
211 src_y = (state->src.y1 >> 16) & ~(format->vsub - 1);
212
213 for (i = 0; i < format->num_planes; i++) {
214 /* Get the physical address of the buffer in memory */
215 gem = drm_fb_cma_get_gem_obj(fb, i);
216
217 DRM_DEBUG_DRIVER("Using GEM @ %pad\n", &gem->paddr);
218
219 /* Compute the start of the displayed memory */
220 paddr = gem->paddr + fb->offsets[i];
221
222 dx = src_x;
223 dy = src_y;
224
225 if (i > 0) {
226 dx /= format->hsub;
227 dy /= format->vsub;
228 }
229
230 /* Fixup framebuffer address for src coordinates */
231 paddr += dx * format->cpp[i];
232 paddr += dy * fb->pitches[i];
233
234 /* Set the line width */
235 DRM_DEBUG_DRIVER("Layer %d. line width: %d bytes\n",
236 i + 1, fb->pitches[i]);
237 regmap_write(mixer->engine.regs,
238 SUN8I_MIXER_CHAN_VI_LAYER_PITCH(channel,
239 overlay, i),
240 fb->pitches[i]);
241
242 DRM_DEBUG_DRIVER("Setting %d. buffer address to %pad\n",
243 i + 1, &paddr);
244
245 regmap_write(mixer->engine.regs,
246 SUN8I_MIXER_CHAN_VI_LAYER_TOP_LADDR(channel,
247 overlay, i),
248 lower_32_bits(paddr));
249 }
250
251 return 0;
252 }
253
sun8i_vi_layer_atomic_check(struct drm_plane * plane,struct drm_plane_state * state)254 static int sun8i_vi_layer_atomic_check(struct drm_plane *plane,
255 struct drm_plane_state *state)
256 {
257 struct sun8i_vi_layer *layer = plane_to_sun8i_vi_layer(plane);
258 struct drm_crtc *crtc = state->crtc;
259 struct drm_crtc_state *crtc_state;
260 int min_scale, max_scale;
261
262 if (!crtc)
263 return 0;
264
265 crtc_state = drm_atomic_get_existing_crtc_state(state->state, crtc);
266 if (WARN_ON(!crtc_state))
267 return -EINVAL;
268
269 min_scale = DRM_PLANE_HELPER_NO_SCALING;
270 max_scale = DRM_PLANE_HELPER_NO_SCALING;
271
272 if (layer->mixer->cfg->scaler_mask & BIT(layer->channel)) {
273 min_scale = SUN8I_VI_SCALER_SCALE_MIN;
274 max_scale = SUN8I_VI_SCALER_SCALE_MAX;
275 }
276
277 return drm_atomic_helper_check_plane_state(state, crtc_state,
278 min_scale, max_scale,
279 true, true);
280 }
281
sun8i_vi_layer_atomic_disable(struct drm_plane * plane,struct drm_plane_state * old_state)282 static void sun8i_vi_layer_atomic_disable(struct drm_plane *plane,
283 struct drm_plane_state *old_state)
284 {
285 struct sun8i_vi_layer *layer = plane_to_sun8i_vi_layer(plane);
286 unsigned int old_zpos = old_state->normalized_zpos;
287 struct sun8i_mixer *mixer = layer->mixer;
288
289 sun8i_vi_layer_enable(mixer, layer->channel, layer->overlay, false, 0,
290 old_zpos);
291 }
292
sun8i_vi_layer_atomic_update(struct drm_plane * plane,struct drm_plane_state * old_state)293 static void sun8i_vi_layer_atomic_update(struct drm_plane *plane,
294 struct drm_plane_state *old_state)
295 {
296 struct sun8i_vi_layer *layer = plane_to_sun8i_vi_layer(plane);
297 unsigned int zpos = plane->state->normalized_zpos;
298 unsigned int old_zpos = old_state->normalized_zpos;
299 struct sun8i_mixer *mixer = layer->mixer;
300
301 if (!plane->state->visible) {
302 sun8i_vi_layer_enable(mixer, layer->channel,
303 layer->overlay, false, 0, old_zpos);
304 return;
305 }
306
307 sun8i_vi_layer_update_coord(mixer, layer->channel,
308 layer->overlay, plane, zpos);
309 sun8i_vi_layer_update_formats(mixer, layer->channel,
310 layer->overlay, plane);
311 sun8i_vi_layer_update_buffer(mixer, layer->channel,
312 layer->overlay, plane);
313 sun8i_vi_layer_enable(mixer, layer->channel, layer->overlay,
314 true, zpos, old_zpos);
315 }
316
317 static struct drm_plane_helper_funcs sun8i_vi_layer_helper_funcs = {
318 .atomic_check = sun8i_vi_layer_atomic_check,
319 .atomic_disable = sun8i_vi_layer_atomic_disable,
320 .atomic_update = sun8i_vi_layer_atomic_update,
321 };
322
323 static const struct drm_plane_funcs sun8i_vi_layer_funcs = {
324 .atomic_destroy_state = drm_atomic_helper_plane_destroy_state,
325 .atomic_duplicate_state = drm_atomic_helper_plane_duplicate_state,
326 .destroy = drm_plane_cleanup,
327 .disable_plane = drm_atomic_helper_disable_plane,
328 .reset = drm_atomic_helper_plane_reset,
329 .update_plane = drm_atomic_helper_update_plane,
330 };
331
332 /*
333 * While all RGB formats are supported, VI planes don't support
334 * alpha blending, so there is no point having formats with alpha
335 * channel if their opaque analog exist.
336 */
337 static const u32 sun8i_vi_layer_formats[] = {
338 DRM_FORMAT_ABGR1555,
339 DRM_FORMAT_ABGR4444,
340 DRM_FORMAT_ARGB1555,
341 DRM_FORMAT_ARGB4444,
342 DRM_FORMAT_BGR565,
343 DRM_FORMAT_BGR888,
344 DRM_FORMAT_BGRA5551,
345 DRM_FORMAT_BGRA4444,
346 DRM_FORMAT_BGRX8888,
347 DRM_FORMAT_RGB565,
348 DRM_FORMAT_RGB888,
349 DRM_FORMAT_RGBA4444,
350 DRM_FORMAT_RGBA5551,
351 DRM_FORMAT_RGBX8888,
352 DRM_FORMAT_XBGR8888,
353 DRM_FORMAT_XRGB8888,
354
355 DRM_FORMAT_NV16,
356 DRM_FORMAT_NV12,
357 DRM_FORMAT_NV21,
358 DRM_FORMAT_NV61,
359 DRM_FORMAT_UYVY,
360 DRM_FORMAT_VYUY,
361 DRM_FORMAT_YUYV,
362 DRM_FORMAT_YVYU,
363 DRM_FORMAT_YUV411,
364 DRM_FORMAT_YUV420,
365 DRM_FORMAT_YUV422,
366 DRM_FORMAT_YUV444,
367 DRM_FORMAT_YVU411,
368 DRM_FORMAT_YVU420,
369 DRM_FORMAT_YVU422,
370 DRM_FORMAT_YVU444,
371 };
372
sun8i_vi_layer_init_one(struct drm_device * drm,struct sun8i_mixer * mixer,int index)373 struct sun8i_vi_layer *sun8i_vi_layer_init_one(struct drm_device *drm,
374 struct sun8i_mixer *mixer,
375 int index)
376 {
377 struct sun8i_vi_layer *layer;
378 unsigned int plane_cnt;
379 int ret;
380
381 layer = devm_kzalloc(drm->dev, sizeof(*layer), GFP_KERNEL);
382 if (!layer)
383 return ERR_PTR(-ENOMEM);
384
385 /* possible crtcs are set later */
386 ret = drm_universal_plane_init(drm, &layer->plane, 0,
387 &sun8i_vi_layer_funcs,
388 sun8i_vi_layer_formats,
389 ARRAY_SIZE(sun8i_vi_layer_formats),
390 NULL, DRM_PLANE_TYPE_OVERLAY, NULL);
391 if (ret) {
392 dev_err(drm->dev, "Couldn't initialize layer\n");
393 return ERR_PTR(ret);
394 }
395
396 plane_cnt = mixer->cfg->ui_num + mixer->cfg->vi_num;
397
398 ret = drm_plane_create_zpos_property(&layer->plane, index,
399 0, plane_cnt - 1);
400 if (ret) {
401 dev_err(drm->dev, "Couldn't add zpos property\n");
402 return ERR_PTR(ret);
403 }
404
405 drm_plane_helper_add(&layer->plane, &sun8i_vi_layer_helper_funcs);
406 layer->mixer = mixer;
407 layer->channel = index;
408 layer->overlay = 0;
409
410 return layer;
411 }
412