blob: c7621ef49c5149973ac9286314cc938a88af43aa [file] [log] [blame]
Stefan Bosch5ed5ad42020-07-10 19:07:36 +02001// SPDX-License-Identifier: GPL-2.0+
2/*
3 * Copyright (C) 2016 Nexell Co., Ltd.
4 *
5 * Author: junghyun, kim <jhkim@nexell.co.kr>
6 *
7 * Copyright (C) 2020 Stefan Bosch <stefan_b@posteo.net>
8 */
9
10#include <config.h>
11#include <common.h>
12#include <command.h>
13#include <dm.h>
14#include <mapmem.h>
15#include <malloc.h>
16#include <linux/compat.h>
17#include <linux/err.h>
Dario Binacchi2ec85772021-01-23 19:43:52 +010018#include <video.h> /* For struct video_uc_plat */
Stefan Bosch5ed5ad42020-07-10 19:07:36 +020019#include <video_fb.h>
20#include <lcd.h>
21#include <asm/global_data.h>
22#include <asm/io.h>
23#include <asm/arch/display.h>
24#include <asm/arch/display_dev.h>
25#include "videomodes.h"
26
27DECLARE_GLOBAL_DATA_PTR;
28
29#if !defined(CONFIG_DM) && !defined(CONFIG_OF_CONTROL)
30static struct nx_display_dev *dp_dev;
31#endif
32
33static char *const dp_dev_str[] = {
34 [DP_DEVICE_RESCONV] = "RESCONV",
35 [DP_DEVICE_RGBLCD] = "LCD",
36 [DP_DEVICE_HDMI] = "HDMI",
37 [DP_DEVICE_MIPI] = "MiPi",
38 [DP_DEVICE_LVDS] = "LVDS",
39 [DP_DEVICE_CVBS] = "TVOUT",
40 [DP_DEVICE_DP0] = "DP0",
41 [DP_DEVICE_DP1] = "DP1",
42};
43
44#if CONFIG_IS_ENABLED(OF_CONTROL)
45static void nx_display_parse_dp_sync(ofnode node, struct dp_sync_info *sync)
46{
47 sync->h_active_len = ofnode_read_s32_default(node, "h_active_len", 0);
48 sync->h_sync_width = ofnode_read_s32_default(node, "h_sync_width", 0);
49 sync->h_back_porch = ofnode_read_s32_default(node, "h_back_porch", 0);
50 sync->h_front_porch = ofnode_read_s32_default(node, "h_front_porch", 0);
51 sync->h_sync_invert = ofnode_read_s32_default(node, "h_sync_invert", 0);
52 sync->v_active_len = ofnode_read_s32_default(node, "v_active_len", 0);
53 sync->v_sync_width = ofnode_read_s32_default(node, "v_sync_width", 0);
54 sync->v_back_porch = ofnode_read_s32_default(node, "v_back_porch", 0);
55 sync->v_front_porch = ofnode_read_s32_default(node, "v_front_porch", 0);
56 sync->v_sync_invert = ofnode_read_s32_default(node, "v_sync_invert", 0);
57 sync->pixel_clock_hz = ofnode_read_s32_default(node, "pixel_clock_hz", 0);
58
59 debug("DP: sync ->\n");
60 debug("ha:%d, hs:%d, hb:%d, hf:%d, hi:%d\n",
61 sync->h_active_len, sync->h_sync_width,
62 sync->h_back_porch, sync->h_front_porch, sync->h_sync_invert);
63 debug("va:%d, vs:%d, vb:%d, vf:%d, vi:%d\n",
64 sync->v_active_len, sync->v_sync_width,
65 sync->v_back_porch, sync->v_front_porch, sync->v_sync_invert);
66}
67
68static void nx_display_parse_dp_ctrl(ofnode node, struct dp_ctrl_info *ctrl)
69{
70 /* clock gen */
71 ctrl->clk_src_lv0 = ofnode_read_s32_default(node, "clk_src_lv0", 0);
72 ctrl->clk_div_lv0 = ofnode_read_s32_default(node, "clk_div_lv0", 0);
73 ctrl->clk_src_lv1 = ofnode_read_s32_default(node, "clk_src_lv1", 0);
74 ctrl->clk_div_lv1 = ofnode_read_s32_default(node, "clk_div_lv1", 0);
75
76 /* scan format */
77 ctrl->interlace = ofnode_read_s32_default(node, "interlace", 0);
78
79 /* syncgen format */
80 ctrl->out_format = ofnode_read_s32_default(node, "out_format", 0);
81 ctrl->invert_field = ofnode_read_s32_default(node, "invert_field", 0);
82 ctrl->swap_RB = ofnode_read_s32_default(node, "swap_RB", 0);
83 ctrl->yc_order = ofnode_read_s32_default(node, "yc_order", 0);
84
85 /* extern sync delay */
86 ctrl->delay_mask = ofnode_read_s32_default(node, "delay_mask", 0);
87 ctrl->d_rgb_pvd = ofnode_read_s32_default(node, "d_rgb_pvd", 0);
88 ctrl->d_hsync_cp1 = ofnode_read_s32_default(node, "d_hsync_cp1", 0);
89 ctrl->d_vsync_fram = ofnode_read_s32_default(node, "d_vsync_fram", 0);
90 ctrl->d_de_cp2 = ofnode_read_s32_default(node, "d_de_cp2", 0);
91
92 /* extern sync delay */
93 ctrl->vs_start_offset =
94 ofnode_read_s32_default(node, "vs_start_offset", 0);
95 ctrl->vs_end_offset = ofnode_read_s32_default(node, "vs_end_offset", 0);
96 ctrl->ev_start_offset =
97 ofnode_read_s32_default(node, "ev_start_offset", 0);
98 ctrl->ev_end_offset = ofnode_read_s32_default(node, "ev_end_offset", 0);
99
100 /* pad clock seletor */
101 ctrl->vck_select = ofnode_read_s32_default(node, "vck_select", 0);
102 ctrl->clk_inv_lv0 = ofnode_read_s32_default(node, "clk_inv_lv0", 0);
103 ctrl->clk_delay_lv0 = ofnode_read_s32_default(node, "clk_delay_lv0", 0);
104 ctrl->clk_inv_lv1 = ofnode_read_s32_default(node, "clk_inv_lv1", 0);
105 ctrl->clk_delay_lv1 = ofnode_read_s32_default(node, "clk_delay_lv1", 0);
106 ctrl->clk_sel_div1 = ofnode_read_s32_default(node, "clk_sel_div1", 0);
107
108 debug("DP: ctrl [%s] ->\n",
109 ctrl->interlace ? "Interlace" : " Progressive");
110 debug("cs0:%d, cd0:%d, cs1:%d, cd1:%d\n",
111 ctrl->clk_src_lv0, ctrl->clk_div_lv0,
112 ctrl->clk_src_lv1, ctrl->clk_div_lv1);
113 debug("fmt:0x%x, inv:%d, swap:%d, yb:0x%x\n",
114 ctrl->out_format, ctrl->invert_field,
115 ctrl->swap_RB, ctrl->yc_order);
116 debug("dm:0x%x, drp:%d, dhs:%d, dvs:%d, dde:0x%x\n",
117 ctrl->delay_mask, ctrl->d_rgb_pvd,
118 ctrl->d_hsync_cp1, ctrl->d_vsync_fram, ctrl->d_de_cp2);
119 debug("vss:%d, vse:%d, evs:%d, eve:%d\n",
120 ctrl->vs_start_offset, ctrl->vs_end_offset,
121 ctrl->ev_start_offset, ctrl->ev_end_offset);
122 debug("sel:%d, i0:%d, d0:%d, i1:%d, d1:%d, s1:%d\n",
123 ctrl->vck_select, ctrl->clk_inv_lv0, ctrl->clk_delay_lv0,
124 ctrl->clk_inv_lv1, ctrl->clk_delay_lv1, ctrl->clk_sel_div1);
125}
126
127static void nx_display_parse_dp_top_layer(ofnode node, struct dp_plane_top *top)
128{
129 top->screen_width = ofnode_read_s32_default(node, "screen_width", 0);
130 top->screen_height = ofnode_read_s32_default(node, "screen_height", 0);
131 top->video_prior = ofnode_read_s32_default(node, "video_prior", 0);
132 top->interlace = ofnode_read_s32_default(node, "interlace", 0);
133 top->back_color = ofnode_read_s32_default(node, "back_color", 0);
134 top->plane_num = DP_PLANS_NUM;
135
136 debug("DP: top [%s] ->\n",
137 top->interlace ? "Interlace" : " Progressive");
138 debug("w:%d, h:%d, prior:%d, bg:0x%x\n",
139 top->screen_width, top->screen_height,
140 top->video_prior, top->back_color);
141}
142
143static void nx_display_parse_dp_layer(ofnode node, struct dp_plane_info *plane)
144{
145 plane->left = ofnode_read_s32_default(node, "left", 0);
146 plane->width = ofnode_read_s32_default(node, "width", 0);
147 plane->top = ofnode_read_s32_default(node, "top", 0);
148 plane->height = ofnode_read_s32_default(node, "height", 0);
149 plane->pixel_byte = ofnode_read_s32_default(node, "pixel_byte", 0);
150 plane->format = ofnode_read_s32_default(node, "format", 0);
151 plane->alpha_on = ofnode_read_s32_default(node, "alpha_on", 0);
152 plane->alpha_depth = ofnode_read_s32_default(node, "alpha", 0);
153 plane->tp_on = ofnode_read_s32_default(node, "tp_on", 0);
154 plane->tp_color = ofnode_read_s32_default(node, "tp_color", 0);
155
156 /* enable layer */
157 if (plane->fb_base)
158 plane->enable = 1;
159 else
160 plane->enable = 0;
161
162 if (plane->fb_base == 0) {
163 printf("fail : dp plane.%d invalid fb base [0x%x] ->\n",
164 plane->layer, plane->fb_base);
165 return;
166 }
167
168 debug("DP: plane.%d [0x%x] ->\n", plane->layer, plane->fb_base);
169 debug("f:0x%x, l:%d, t:%d, %d * %d, bpp:%d, a:%d(%d), t:%d(0x%x)\n",
170 plane->format, plane->left, plane->top, plane->width,
171 plane->height, plane->pixel_byte, plane->alpha_on,
172 plane->alpha_depth, plane->tp_on, plane->tp_color);
173}
174
175static void nx_display_parse_dp_planes(ofnode node,
176 struct nx_display_dev *dp,
Simon Glassb75b15b2020-12-03 16:55:23 -0700177 struct video_uc_plat *plat)
Stefan Bosch5ed5ad42020-07-10 19:07:36 +0200178{
179 const char *name;
180 ofnode subnode;
181
182 ofnode_for_each_subnode(subnode, node) {
183 name = ofnode_get_name(subnode);
184
185 if (strcmp(name, "layer_top") == 0)
186 nx_display_parse_dp_top_layer(subnode, &dp->top);
187
188 /*
189 * TODO: Is it sure that only one layer is used? Otherwise
190 * fb_base must be different?
191 */
192 if (strcmp(name, "layer_0") == 0) {
193 dp->planes[0].fb_base =
194 (uint)map_sysmem(plat->base, plat->size);
195 debug("%s(): dp->planes[0].fb_base == 0x%x\n", __func__,
196 (uint)dp->planes[0].fb_base);
197 nx_display_parse_dp_layer(subnode, &dp->planes[0]);
198 }
199
200 if (strcmp(name, "layer_1") == 0) {
201 dp->planes[1].fb_base =
202 (uint)map_sysmem(plat->base, plat->size);
203 debug("%s(): dp->planes[1].fb_base == 0x%x\n", __func__,
204 (uint)dp->planes[1].fb_base);
205 nx_display_parse_dp_layer(subnode, &dp->planes[1]);
206 }
207
208 if (strcmp(name, "layer_2") == 0) {
209 dp->planes[2].fb_base =
210 (uint)map_sysmem(plat->base, plat->size);
211 debug("%s(): dp->planes[2].fb_base == 0x%x\n", __func__,
212 (uint)dp->planes[2].fb_base);
213 nx_display_parse_dp_layer(subnode, &dp->planes[2]);
214 }
215 }
216}
217
218static int nx_display_parse_dp_lvds(ofnode node, struct nx_display_dev *dp)
219{
220 struct dp_lvds_dev *dev = kzalloc(sizeof(*dev), GFP_KERNEL);
221
222 if (!dev) {
223 printf("failed to allocate display LVDS object.\n");
224 return -ENOMEM;
225 }
226
227 dp->device = dev;
228
229 dev->lvds_format = ofnode_read_s32_default(node, "format", 0);
230 dev->pol_inv_hs = ofnode_read_s32_default(node, "pol_inv_hs", 0);
231 dev->pol_inv_vs = ofnode_read_s32_default(node, "pol_inv_vs", 0);
232 dev->pol_inv_de = ofnode_read_s32_default(node, "pol_inv_de", 0);
233 dev->pol_inv_ck = ofnode_read_s32_default(node, "pol_inv_ck", 0);
234 dev->voltage_level = ofnode_read_s32_default(node, "voltage_level", 0);
235
236 if (!dev->voltage_level)
237 dev->voltage_level = DEF_VOLTAGE_LEVEL;
238
239 debug("DP: LVDS -> %s, voltage LV:0x%x\n",
240 dev->lvds_format == DP_LVDS_FORMAT_VESA ? "VESA" :
241 dev->lvds_format == DP_LVDS_FORMAT_JEIDA ? "JEIDA" : "LOC",
242 dev->voltage_level);
243 debug("pol inv hs:%d, vs:%d, de:%d, ck:%d\n",
244 dev->pol_inv_hs, dev->pol_inv_vs,
245 dev->pol_inv_de, dev->pol_inv_ck);
246
247 return 0;
248}
249
250static int nx_display_parse_dp_rgb(ofnode node, struct nx_display_dev *dp)
251{
252 struct dp_rgb_dev *dev = kzalloc(sizeof(*dev), GFP_KERNEL);
253
254 if (!dev) {
255 printf("failed to allocate display RGB LCD object.\n");
256 return -ENOMEM;
257 }
258 dp->device = dev;
259
260 dev->lcd_mpu_type = ofnode_read_s32_default(node, "lcd_mpu_type", 0);
261
262 debug("DP: RGB -> MPU[%s]\n", dev->lcd_mpu_type ? "O" : "X");
263 return 0;
264}
265
266static int nx_display_parse_dp_mipi(ofnode node, struct nx_display_dev *dp)
267{
268 struct dp_mipi_dev *dev = kzalloc(sizeof(*dev), GFP_KERNEL);
269
270 if (!dev) {
271 printf("failed to allocate display MiPi object.\n");
272 return -ENOMEM;
273 }
274 dp->device = dev;
275
276 dev->lp_bitrate = ofnode_read_s32_default(node, "lp_bitrate", 0);
277 dev->hs_bitrate = ofnode_read_s32_default(node, "hs_bitrate", 0);
278 dev->lpm_trans = 1;
279 dev->command_mode = 0;
280
281 debug("DP: MIPI ->\n");
282 debug("lp:%dmhz, hs:%dmhz\n", dev->lp_bitrate, dev->hs_bitrate);
283
284 return 0;
285}
286
287static int nx_display_parse_dp_hdmi(ofnode node, struct nx_display_dev *dp)
288{
289 struct dp_hdmi_dev *dev = kzalloc(sizeof(*dev), GFP_KERNEL);
290
291 if (!dev) {
292 printf("failed to allocate display HDMI object.\n");
293 return -ENOMEM;
294 }
295 dp->device = dev;
296
297 dev->preset = ofnode_read_s32_default(node, "preset", 0);
298
299 debug("DP: HDMI -> %d\n", dev->preset);
300
301 return 0;
302}
303
304static int nx_display_parse_dp_lcds(ofnode node, const char *type,
305 struct nx_display_dev *dp)
306{
307 if (strcmp(type, "lvds") == 0) {
308 dp->dev_type = DP_DEVICE_LVDS;
309 return nx_display_parse_dp_lvds(node, dp);
310 } else if (strcmp(type, "rgb") == 0) {
311 dp->dev_type = DP_DEVICE_RGBLCD;
312 return nx_display_parse_dp_rgb(node, dp);
313 } else if (strcmp(type, "mipi") == 0) {
314 dp->dev_type = DP_DEVICE_MIPI;
315 return nx_display_parse_dp_mipi(node, dp);
316 } else if (strcmp(type, "hdmi") == 0) {
317 dp->dev_type = DP_DEVICE_HDMI;
318 return nx_display_parse_dp_hdmi(node, dp);
319 }
320
321 printf("%s: node %s unknown display type\n", __func__,
322 ofnode_get_name(node));
323 return -EINVAL;
324
325 return 0;
326}
327
328#define DT_SYNC (1 << 0)
329#define DT_CTRL (1 << 1)
330#define DT_PLANES (1 << 2)
331#define DT_DEVICE (1 << 3)
332
333static int nx_display_parse_dt(struct udevice *dev,
334 struct nx_display_dev *dp,
Simon Glassb75b15b2020-12-03 16:55:23 -0700335 struct video_uc_plat *plat)
Stefan Bosch5ed5ad42020-07-10 19:07:36 +0200336{
337 const char *name, *dtype;
338 int ret = 0;
339 unsigned int dt_status = 0;
340 ofnode subnode;
341
342 if (!dev)
343 return -ENODEV;
344
345 dp->module = dev_read_s32_default(dev, "module", -1);
346 if (dp->module == -1)
347 dp->module = dev_read_s32_default(dev, "index", 0);
348
349 dtype = dev_read_string(dev, "lcd-type");
350
351 ofnode_for_each_subnode(subnode, dev_ofnode(dev)) {
352 name = ofnode_get_name(subnode);
353
354 if (strcmp("dp-sync", name) == 0) {
355 dt_status |= DT_SYNC;
356 nx_display_parse_dp_sync(subnode, &dp->sync);
357 }
358
359 if (strcmp("dp-ctrl", name) == 0) {
360 dt_status |= DT_CTRL;
361 nx_display_parse_dp_ctrl(subnode, &dp->ctrl);
362 }
363
364 if (strcmp("dp-planes", name) == 0) {
365 dt_status |= DT_PLANES;
366 nx_display_parse_dp_planes(subnode, dp, plat);
367 }
368
369 if (strcmp("dp-device", name) == 0) {
370 dt_status |= DT_DEVICE;
371 ret = nx_display_parse_dp_lcds(subnode, dtype, dp);
372 }
373 }
374
375 if (dt_status != (DT_SYNC | DT_CTRL | DT_PLANES | DT_DEVICE)) {
376 printf("Not enough DT config for display [0x%x]\n", dt_status);
377 return -ENODEV;
378 }
379
380 return ret;
381}
382#endif
383
384__weak int nx_display_fixup_dp(struct nx_display_dev *dp)
385{
386 return 0;
387}
388
389static struct nx_display_dev *nx_display_setup(void)
390{
391 struct nx_display_dev *dp;
392 int i, ret;
393 int node = 0;
Simon Glassb75b15b2020-12-03 16:55:23 -0700394 struct video_uc_plat *plat = NULL;
Stefan Bosch5ed5ad42020-07-10 19:07:36 +0200395
396 struct udevice *dev;
397
398 /* call driver probe */
399 debug("DT: uclass device call...\n");
400
401 ret = uclass_get_device(UCLASS_VIDEO, 0, &dev);
402 if (ret) {
403 debug("%s(): uclass_get_device(UCLASS_VIDEO, 0, &dev) != 0 --> return NULL\n",
404 __func__);
405 return NULL;
406 }
Simon Glass71fa5b42020-12-03 16:55:18 -0700407 plat = dev_get_uclass_plat(dev);
Stefan Bosch5ed5ad42020-07-10 19:07:36 +0200408 if (!dev) {
Simon Glass71fa5b42020-12-03 16:55:18 -0700409 debug("%s(): dev_get_uclass_plat(dev) == NULL --> return NULL\n",
Stefan Bosch5ed5ad42020-07-10 19:07:36 +0200410 __func__);
411 return NULL;
412 }
413 dp = dev_get_priv(dev);
414 if (!dp) {
415 debug("%s(): dev_get_priv(dev) == NULL --> return NULL\n",
416 __func__);
417 return NULL;
418 }
Simon Glassa7ece582020-12-19 10:40:14 -0700419 node = dev_ofnode(dev).of_offset;
Stefan Bosch5ed5ad42020-07-10 19:07:36 +0200420
421 if (CONFIG_IS_ENABLED(OF_CONTROL)) {
422 ret = nx_display_parse_dt(dev, dp, plat);
423 if (ret)
424 goto err_setup;
425 }
426
427 nx_display_fixup_dp(dp);
428
429 for (i = 0; dp->top.plane_num > i; i++) {
430 dp->planes[i].layer = i;
431 if (dp->planes[i].enable && !dp->fb_plane) {
432 dp->fb_plane = &dp->planes[i];
433 dp->fb_addr = dp->fb_plane->fb_base;
434 dp->depth = dp->fb_plane->pixel_byte;
435 }
436 }
437
438 switch (dp->dev_type) {
439#ifdef CONFIG_VIDEO_NX_RGB
440 case DP_DEVICE_RGBLCD:
441 nx_rgb_display(dp->module,
442 &dp->sync, &dp->ctrl, &dp->top,
443 dp->planes, (struct dp_rgb_dev *)dp->device);
444 break;
445#endif
446#ifdef CONFIG_VIDEO_NX_LVDS
447 case DP_DEVICE_LVDS:
448 nx_lvds_display(dp->module,
449 &dp->sync, &dp->ctrl, &dp->top,
450 dp->planes, (struct dp_lvds_dev *)dp->device);
451 break;
452#endif
453#ifdef CONFIG_VIDEO_NX_MIPI
454 case DP_DEVICE_MIPI:
455 nx_mipi_display(dp->module,
456 &dp->sync, &dp->ctrl, &dp->top,
457 dp->planes, (struct dp_mipi_dev *)dp->device);
458 break;
459#endif
460#ifdef CONFIG_VIDEO_NX_HDMI
461 case DP_DEVICE_HDMI:
462 nx_hdmi_display(dp->module,
463 &dp->sync, &dp->ctrl, &dp->top,
464 dp->planes, (struct dp_hdmi_dev *)dp->device);
465 break;
466#endif
467 default:
468 printf("fail : not support lcd type %d !!!\n", dp->dev_type);
469 goto err_setup;
470 };
471
472 printf("LCD: [%s] dp.%d.%d %dx%d %dbpp FB:0x%08x\n",
473 dp_dev_str[dp->dev_type], dp->module, dp->fb_plane->layer,
474 dp->fb_plane->width, dp->fb_plane->height, dp->depth * 8,
475 dp->fb_addr);
476
477 return dp;
478
479err_setup:
480 kfree(dp);
481
482 return NULL;
483}
484
485#if defined CONFIG_LCD
486
487/* default lcd */
488struct vidinfo panel_info = {
489 .vl_col = 320, .vl_row = 240, .vl_bpix = 32,
490};
491
492void lcd_ctrl_init(void *lcdbase)
493{
494 vidinfo_t *pi = &panel_info;
495 struct nx_display_dev *dp;
496 int bpix;
497
498 dp = nx_display_setup();
499 if (!dp)
500 return NULL;
501
502 switch (dp->depth) {
503 case 2:
504 bpix = LCD_COLOR16;
505 break;
506 case 3:
507 case 4:
508 bpix = LCD_COLOR32;
509 break;
510 default:
511 printf("fail : not support LCD bit per pixel %d\n",
512 dp->depth * 8);
513 return NULL;
514 }
515
516 dp->panel_info = pi;
517
518 /* set resolution with config */
519 pi->vl_bpix = bpix;
520 pi->vl_col = dp->fb_plane->width;
521 pi->vl_row = dp->fb_plane->height;
522 pi->priv = dp;
523 gd->fb_base = dp->fb_addr;
524}
525
526void lcd_setcolreg(ushort regno, ushort red, ushort green, ushort blue)
527{
528}
529
530__weak void lcd_enable(void)
531{
532}
533#endif
534
535static int nx_display_probe(struct udevice *dev)
536{
Simon Glassb75b15b2020-12-03 16:55:23 -0700537 struct video_uc_plat *uc_plat = dev_get_uclass_plat(dev);
Stefan Bosch5ed5ad42020-07-10 19:07:36 +0200538 struct video_priv *uc_priv = dev_get_uclass_priv(dev);
Simon Glassb75b15b2020-12-03 16:55:23 -0700539 struct nx_display_plat *plat = dev_get_plat(dev);
Stefan Bosch5ed5ad42020-07-10 19:07:36 +0200540 static GraphicDevice *graphic_device;
541 char addr[64];
542
543 debug("%s()\n", __func__);
544
545 if (!dev)
546 return -EINVAL;
547
548 if (!uc_plat) {
Simon Glassb75b15b2020-12-03 16:55:23 -0700549 debug("%s(): video_uc_plat *plat == NULL --> return -EINVAL\n",
Stefan Bosch5ed5ad42020-07-10 19:07:36 +0200550 __func__);
551 return -EINVAL;
552 }
553
554 if (!uc_priv) {
555 debug("%s(): video_priv *uc_priv == NULL --> return -EINVAL\n",
556 __func__);
557 return -EINVAL;
558 }
559
560 if (!plat) {
Simon Glassb75b15b2020-12-03 16:55:23 -0700561 debug("%s(): nx_display_plat *plat == NULL --> return -EINVAL\n",
Stefan Bosch5ed5ad42020-07-10 19:07:36 +0200562 __func__);
563 return -EINVAL;
564 }
565
566 struct nx_display_dev *dp;
567 unsigned int pp_index = 0;
568
569 dp = nx_display_setup();
570 if (!dp) {
571 debug("%s(): nx_display_setup() == 0 --> return -EINVAL\n",
572 __func__);
573 return -EINVAL;
574 }
575
576 switch (dp->depth) {
577 case 2:
578 pp_index = GDF_16BIT_565RGB;
579 uc_priv->bpix = VIDEO_BPP16;
580 break;
581 case 3:
582 /* There is no VIDEO_BPP24 because these values are of
583 * type video_log2_bpp
584 */
585 case 4:
586 pp_index = GDF_32BIT_X888RGB;
587 uc_priv->bpix = VIDEO_BPP32;
588 break;
589 default:
590 printf("fail : not support LCD bit per pixel %d\n",
591 dp->depth * 8);
592 return -EINVAL;
593 }
594
595 uc_priv->xsize = dp->fb_plane->width;
596 uc_priv->ysize = dp->fb_plane->height;
597 uc_priv->rot = 0;
598
599 graphic_device = &dp->graphic_device;
600 graphic_device->frameAdrs = dp->fb_addr;
601 graphic_device->gdfIndex = pp_index;
602 graphic_device->gdfBytesPP = dp->depth;
603 graphic_device->winSizeX = dp->fb_plane->width;
604 graphic_device->winSizeY = dp->fb_plane->height;
605 graphic_device->plnSizeX =
606 graphic_device->winSizeX * graphic_device->gdfBytesPP;
607
608 /*
609 * set environment variable "fb_addr" (frame buffer address), required
610 * for splash image. Because drv_video_init() in common/stdio.c is only
611 * called when CONFIG_VIDEO is set (and not if CONFIG_DM_VIDEO is set).
612 */
613 sprintf(addr, "0x%x", dp->fb_addr);
614 debug("%s(): env_set(\"fb_addr\", %s) ...\n", __func__, addr);
615 env_set("fb_addr", addr);
616
617 return 0;
618}
619
620static int nx_display_bind(struct udevice *dev)
621{
Simon Glassb75b15b2020-12-03 16:55:23 -0700622 struct video_uc_plat *plat = dev_get_uclass_plat(dev);
Stefan Bosch5ed5ad42020-07-10 19:07:36 +0200623
624 debug("%s()\n", __func__);
625
626 /* Datasheet S5p4418:
627 * Resolution up to 2048 x 1280, up to 12 Bit per color (HDMI)
628 * Actual (max.) size is 0x1000000 because in U-Boot nanopi2-2016.01
629 * "#define CONFIG_FB_ADDR 0x77000000" and next address is
630 * "#define BMP_LOAD_ADDR 0x78000000"
631 */
632 plat->size = 0x1000000;
633
634 return 0;
635}
636
637static const struct udevice_id nx_display_ids[] = {
638 {.compatible = "nexell,nexell-display", },
639 {}
640};
641
642U_BOOT_DRIVER(nexell_display) = {
643 .name = "nexell-display",
644 .id = UCLASS_VIDEO,
645 .of_match = nx_display_ids,
Simon Glassb75b15b2020-12-03 16:55:23 -0700646 .plat_auto = sizeof(struct nx_display_plat),
Stefan Bosch5ed5ad42020-07-10 19:07:36 +0200647 .bind = nx_display_bind,
648 .probe = nx_display_probe,
Simon Glass8a2b47f2020-12-03 16:55:17 -0700649 .priv_auto = sizeof(struct nx_display_dev),
Stefan Bosch5ed5ad42020-07-10 19:07:36 +0200650};