blob: d86d86798415852ad58e0109c7d588759db748b7 [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
wdenkeeb1b772004-03-23 22:53:55 +00002/*
3 * (C) Copyright 2004
4 * Pierre Aubert, Staubli Faverges , <p.aubert@staubli.com>
Timur Tabia5dbdc82011-03-21 16:38:49 -05005 * Copyright 2011 Freescale Semiconductor, Inc.
wdenkeeb1b772004-03-23 22:53:55 +00006 */
7
8/************************************************************************
9 Get Parameters for the video mode:
Tom Rinia3fda0d2023-01-10 11:19:28 -050010 The default video mode is set to 0x301
wdenkeeb1b772004-03-23 22:53:55 +000011 Parameters can be set via the variable "videomode" in the environment.
12 2 diferent ways are possible:
13 "videomode=301" - 301 is a hexadecimal number describing the VESA
14 mode. Following modes are implemented:
15
16 Colors 640x480 800x600 1024x768 1152x864 1280x1024
17 --------+---------------------------------------------
18 8 bits | 0x301 0x303 0x305 0x161 0x307
19 15 bits | 0x310 0x313 0x316 0x162 0x319
20 16 bits | 0x311 0x314 0x317 0x163 0x31A
21 24 bits | 0x312 0x315 0x318 ? 0x31B
22 --------+---------------------------------------------
23 "videomode=bootargs"
24 - the parameters are parsed from the bootargs.
25 The format is "NAME:VALUE,NAME:VALUE" etc.
26 Ex.:
27 "bootargs=video=ctfb:x:800,y:600,depth:16,pclk:25000"
28 Parameters not included in the list will be taken from
29 the default mode, which is one of the following:
30 mode:0 640x480x24
31 mode:1 800x600x16
32 mode:2 1024x768x8
33 mode:3 960x720x24
34 mode:4 1152x864x16
35 mode:5 1280x1024x8
36
37 if "mode" is not provided within the parameter list,
38 mode:0 is assumed.
39 Following parameters are supported:
40 x xres = visible resolution horizontal
41 y yres = visible resolution vertical
42 pclk pixelclocks in pico sec
43 le left_marging time from sync to picture in pixelclocks
44 ri right_marging time from picture to sync in pixelclocks
45 up upper_margin time from sync to picture
46 lo lower_margin
47 hs hsync_len length of horizontal sync
48 vs vsync_len length of vertical sync
49 sync see FB_SYNC_*
50 vmode see FB_VMODE_*
51 depth Color depth in bits per pixel
52 All other parameters in the variable bootargs are ignored.
53 It is also possible to set the parameters direct in the
54 variable "videomode", or in another variable i.e.
55 "myvideo" and setting the variable "videomode=myvideo"..
56****************************************************************************/
57
Hans de Goedeb7ce12d2014-12-19 15:47:37 +010058#include <edid.h>
Simon Glass7b51b572019-08-01 09:46:52 -060059#include <env.h>
Hans de Goedeb7ce12d2014-12-19 15:47:37 +010060#include <errno.h>
Simon Glass401d1c42020-10-30 21:38:53 -060061#include <fdtdec.h>
Timur Tabia5dbdc82011-03-21 16:38:49 -050062#include <linux/ctype.h>
63
wdenkeeb1b772004-03-23 22:53:55 +000064#include "videomodes.h"
65
66const struct ctfb_vesa_modes vesa_modes[VESA_MODES_COUNT] = {
67 {0x301, RES_MODE_640x480, 8},
68 {0x310, RES_MODE_640x480, 15},
69 {0x311, RES_MODE_640x480, 16},
70 {0x312, RES_MODE_640x480, 24},
71 {0x303, RES_MODE_800x600, 8},
72 {0x313, RES_MODE_800x600, 15},
73 {0x314, RES_MODE_800x600, 16},
74 {0x315, RES_MODE_800x600, 24},
75 {0x305, RES_MODE_1024x768, 8},
76 {0x316, RES_MODE_1024x768, 15},
77 {0x317, RES_MODE_1024x768, 16},
78 {0x318, RES_MODE_1024x768, 24},
79 {0x161, RES_MODE_1152x864, 8},
80 {0x162, RES_MODE_1152x864, 15},
81 {0x163, RES_MODE_1152x864, 16},
82 {0x307, RES_MODE_1280x1024, 8},
83 {0x319, RES_MODE_1280x1024, 15},
84 {0x31A, RES_MODE_1280x1024, 16},
85 {0x31B, RES_MODE_1280x1024, 24},
86};
87const struct ctfb_res_modes res_mode_init[RES_MODES_COUNT] = {
Hans de Goede0c91d2572014-12-19 10:38:49 +010088 /* x y hz pixclk ps/kHz le ri up lo hs vs s vmode */
Hans de Goede92a88c32014-12-19 11:11:52 +010089#ifndef CONFIG_VIDEO_STD_TIMINGS
Hans de Goede0c91d2572014-12-19 10:38:49 +010090 { 640, 480, 60, 39721, 25180, 40, 24, 32, 11, 96, 2, 0, FB_VMODE_NONINTERLACED},
91 { 800, 600, 60, 27778, 36000, 64, 24, 22, 1, 72, 2, 0, FB_VMODE_NONINTERLACED},
92 {1024, 768, 60, 15384, 65000, 168, 8, 29, 3, 144, 4, 0, FB_VMODE_NONINTERLACED},
93 { 960, 720, 80, 13100, 76335, 160, 40, 32, 8, 80, 4, 0, FB_VMODE_NONINTERLACED},
94 {1152, 864, 60, 12004, 83300, 200, 64, 32, 16, 80, 4, 0, FB_VMODE_NONINTERLACED},
95 {1280, 1024, 60, 9090, 110000, 200, 48, 26, 1, 184, 3, 0, FB_VMODE_NONINTERLACED},
Hans de Goede92a88c32014-12-19 11:11:52 +010096#else
97 { 640, 480, 60, 39683, 25200, 48, 16, 33, 10, 96, 2, 0, FB_VMODE_NONINTERLACED},
98 { 800, 600, 60, 25000, 40000, 88, 40, 23, 1, 128, 4, FB_SYNC_HOR_HIGH_ACT | FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED},
99 {1024, 768, 60, 15384, 65000, 160, 24, 29, 3, 136, 6, 0, FB_VMODE_NONINTERLACED},
100 { 960, 720, 75, 13468, 74250, 176, 72, 27, 1, 112, 2, 0, FB_VMODE_NONINTERLACED},
101 {1152, 864, 75, 9259, 108000, 256, 64, 32, 1, 128, 3, FB_SYNC_HOR_HIGH_ACT | FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED},
102 {1280, 1024, 60, 9259, 108000, 248, 48, 38, 1, 112, 3, FB_SYNC_HOR_HIGH_ACT | FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED},
103#endif
Hans de Goede59bb6102014-12-19 11:45:19 +0100104 {1280, 720, 60, 13468, 74250, 220, 110, 20, 5, 40, 5, FB_SYNC_HOR_HIGH_ACT | FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED},
105 {1360, 768, 60, 11696, 85500, 256, 64, 17, 3, 112, 7, 0, FB_VMODE_NONINTERLACED},
106 {1920, 1080, 60, 6734, 148500, 148, 88, 36, 4, 44, 5, FB_SYNC_HOR_HIGH_ACT | FB_SYNC_VERT_HIGH_ACT, FB_VMODE_NONINTERLACED},
107 {1920, 1200, 60, 6494, 154000, 80, 48, 26, 3, 32, 6, FB_SYNC_HOR_HIGH_ACT, FB_VMODE_NONINTERLACED},
wdenkeeb1b772004-03-23 22:53:55 +0000108};
109
110/************************************************************************
111 * Get Parameters for the video mode:
112 */
113/*********************************************************************
114 * returns the length to the next seperator
115 */
116static int
Hans de Goedeeb3c0cf2014-12-19 14:27:46 +0100117video_get_param_len(const char *start, char sep)
wdenkeeb1b772004-03-23 22:53:55 +0000118{
119 int i = 0;
120 while ((*start != 0) && (*start != sep)) {
121 start++;
122 i++;
123 }
124 return i;
125}
126
127static int
128video_search_param (char *start, char *param)
129{
130 int len, totallen, i;
131 char *p = start;
132 len = strlen (param);
133 totallen = len + strlen (start);
134 for (i = 0; i < totallen; i++) {
135 if (strncmp (p++, param, len) == 0)
136 return (i);
137 }
138 return -1;
139}
140
141/***************************************************************
142 * Get parameter via the environment as it is done for the
143 * linux kernel i.e:
144 * video=ctfb:x:800,xv:1280,y:600,yv:1024,depth:16,mode:0,pclk:25000,
145 * le:56,ri:48,up:26,lo:5,hs:152,vs:2,sync:0,vmode:0,accel:0
146 *
147 * penv is a pointer to the environment, containing the string, or the name of
148 * another environment variable. It could even be the term "bootargs"
149 */
150
151#define GET_OPTION(name,var) \
152 if(strncmp(p,name,strlen(name))==0) { \
153 val_s=p+strlen(name); \
154 var=simple_strtoul(val_s, NULL, 10); \
155 }
156
157int video_get_params (struct ctfb_res_modes *pPar, char *penv)
158{
159 char *p, *s, *val_s;
Wolfgang Denk40ac78a2011-11-04 15:55:14 +0000160 int i = 0;
wdenkeeb1b772004-03-23 22:53:55 +0000161 int bpp;
162 int mode;
Wolfgang Denk40ac78a2011-11-04 15:55:14 +0000163
wdenkeeb1b772004-03-23 22:53:55 +0000164 /* first search for the environment containing the real param string */
165 s = penv;
Wolfgang Denk40ac78a2011-11-04 15:55:14 +0000166
Simon Glass00caae62017-08-03 12:22:12 -0600167 p = env_get(s);
168 if (p)
wdenkeeb1b772004-03-23 22:53:55 +0000169 s = p;
Wolfgang Denk40ac78a2011-11-04 15:55:14 +0000170
171 /*
172 * in case of the bootargs line, we have to start
wdenkeeb1b772004-03-23 22:53:55 +0000173 * after "video=ctfb:"
174 */
175 i = video_search_param (s, "video=ctfb:");
176 if (i >= 0) {
177 s += i;
178 s += strlen ("video=ctfb:");
179 }
180 /* search for mode as a default value */
181 p = s;
wdenkeeb1b772004-03-23 22:53:55 +0000182 mode = 0; /* default */
Wolfgang Denk40ac78a2011-11-04 15:55:14 +0000183
wdenkeeb1b772004-03-23 22:53:55 +0000184 while ((i = video_get_param_len (p, ',')) != 0) {
185 GET_OPTION ("mode:", mode)
186 p += i;
187 if (*p != 0)
188 p++; /* skip ',' */
189 }
Wolfgang Denk40ac78a2011-11-04 15:55:14 +0000190
wdenkeeb1b772004-03-23 22:53:55 +0000191 if (mode >= RES_MODES_COUNT)
192 mode = 0;
Wolfgang Denk40ac78a2011-11-04 15:55:14 +0000193
wdenkeeb1b772004-03-23 22:53:55 +0000194 *pPar = res_mode_init[mode]; /* copy default values */
195 bpp = 24 - ((mode % 3) * 8);
196 p = s; /* restart */
Wolfgang Denk40ac78a2011-11-04 15:55:14 +0000197
wdenkeeb1b772004-03-23 22:53:55 +0000198 while ((i = video_get_param_len (p, ',')) != 0) {
199 GET_OPTION ("x:", pPar->xres)
200 GET_OPTION ("y:", pPar->yres)
Hans de Goede0c91d2572014-12-19 10:38:49 +0100201 GET_OPTION ("refresh:", pPar->refresh)
wdenkeeb1b772004-03-23 22:53:55 +0000202 GET_OPTION ("le:", pPar->left_margin)
203 GET_OPTION ("ri:", pPar->right_margin)
204 GET_OPTION ("up:", pPar->upper_margin)
205 GET_OPTION ("lo:", pPar->lower_margin)
206 GET_OPTION ("hs:", pPar->hsync_len)
207 GET_OPTION ("vs:", pPar->vsync_len)
208 GET_OPTION ("sync:", pPar->sync)
209 GET_OPTION ("vmode:", pPar->vmode)
210 GET_OPTION ("pclk:", pPar->pixclock)
Hans de Goede0c91d2572014-12-19 10:38:49 +0100211 GET_OPTION ("pclk_khz:", pPar->pixclock_khz)
wdenkeeb1b772004-03-23 22:53:55 +0000212 GET_OPTION ("depth:", bpp)
213 p += i;
214 if (*p != 0)
215 p++; /* skip ',' */
216 }
217 return bpp;
218}
Timur Tabia5dbdc82011-03-21 16:38:49 -0500219
220/*
221 * Parse the 'video-mode' environment variable
222 *
223 * Example: "video-mode=fslfb:1280x1024-32@60,monitor=dvi". See
224 * doc/README.video for more information on how to set the variable.
225 *
226 * @xres: returned value of X-resolution
227 * @yres: returned value of Y-resolution
228 * @depth: returned value of color depth
229 * @freq: returned value of monitor frequency
230 * @options: pointer to any remaining options, or NULL
231 *
232 * Returns 1 if valid values were found, 0 otherwise
233 */
234int video_get_video_mode(unsigned int *xres, unsigned int *yres,
235 unsigned int *depth, unsigned int *freq, const char **options)
236{
Simon Glass00caae62017-08-03 12:22:12 -0600237 char *p = env_get("video-mode");
Timur Tabia5dbdc82011-03-21 16:38:49 -0500238 if (!p)
239 return 0;
240
241 /* Skip over the driver name, which we don't care about. */
242 p = strchr(p, ':');
243 if (!p)
244 return 0;
245
246 /* Get the X-resolution*/
247 while (*p && !isdigit(*p))
248 p++;
249 *xres = simple_strtoul(p, &p, 10);
250 if (!*xres)
251 return 0;
252
253 /* Get the Y-resolution */
254 while (*p && !isdigit(*p))
255 p++;
256 *yres = simple_strtoul(p, &p, 10);
257 if (!*yres)
258 return 0;
259
260 /* Get the depth */
261 while (*p && !isdigit(*p))
262 p++;
263 *depth = simple_strtoul(p, &p, 10);
264 if (!*depth)
265 return 0;
266
267 /* Get the frequency */
268 while (*p && !isdigit(*p))
269 p++;
270 *freq = simple_strtoul(p, &p, 10);
271 if (!*freq)
272 return 0;
273
274 /* Find the extra options, if any */
275 p = strchr(p, ',');
276 *options = p ? p + 1 : NULL;
277
278 return 1;
279}
Hans de Goedee976b862014-12-19 13:22:47 +0100280
281/*
282 * Parse the 'video-mode' environment variable using video_get_video_mode()
283 * and lookup the matching ctfb_res_modes in res_mode_init.
284 *
285 * @default_mode: RES_MODE_##x## define for the mode to store in mode_ret
286 * when 'video-mode' is not set or does not contain a valid mode
287 * @default_depth: depth to set when 'video-mode' is not set
288 * @mode_ret: pointer where the mode will be stored
289 * @depth_ret: pointer where the depth will be stored
290 * @options: pointer to any remaining options, or NULL
291 */
292void video_get_ctfb_res_modes(int default_mode, unsigned int default_depth,
293 const struct ctfb_res_modes **mode_ret,
294 unsigned int *depth_ret,
295 const char **options)
296{
297 unsigned int i, xres, yres, depth, refresh;
298
299 *mode_ret = &res_mode_init[default_mode];
300 *depth_ret = default_depth;
301 *options = NULL;
302
303 if (!video_get_video_mode(&xres, &yres, &depth, &refresh, options))
304 return;
305
306 for (i = 0; i < RES_MODES_COUNT; i++) {
307 if (res_mode_init[i].xres == xres &&
308 res_mode_init[i].yres == yres &&
309 res_mode_init[i].refresh == refresh) {
310 *mode_ret = &res_mode_init[i];
311 *depth_ret = depth;
312 return;
313 }
314 }
315
316 printf("video-mode %dx%d-%d@%d not available, falling back to %dx%d-%d@%d\n",
317 xres, yres, depth, refresh, (*mode_ret)->xres,
318 (*mode_ret)->yres, *depth_ret, (*mode_ret)->refresh);
319}
Hans de Goedeeb3c0cf2014-12-19 14:27:46 +0100320
321/*
322 * Find the named string option within the ',' separated options string, and
323 * store its value in dest.
324 *
325 * @options: ',' separated options string
326 * @name: name of the option to look for
327 * @dest: destination buffer to store the value of the option in
328 * @dest_len: length of dest
329 * @def: value to store in dest if the option is not present in options
330 */
331void video_get_option_string(const char *options, const char *name,
332 char *dest, int dest_len, const char *def)
333{
334 const char *p = options;
335 const int name_len = strlen(name);
336 int i, len;
337
338 while (p && (i = video_get_param_len(p, ',')) != 0) {
339 if (strncmp(p, name, name_len) == 0 && p[name_len] == '=') {
340 len = i - (name_len + 1);
341 if (len >= dest_len)
342 len = dest_len - 1;
343 memcpy(dest, &p[name_len + 1], len);
344 dest[len] = 0;
345 return;
346 }
347 p += i;
348 if (*p != 0)
349 p++; /* skip ',' */
350 }
351 strcpy(dest, def);
352}
353
354/*
355 * Find the named integer option within the ',' separated options string, and
356 * return its value.
357 *
358 * @options: ',' separated options string
359 * @name: name of the option to look for
360 * @def: value to return if the option is not present in options
361 */
362int video_get_option_int(const char *options, const char *name, int def)
363{
364 const char *p = options;
365 const int name_len = strlen(name);
366 int i;
367
368 while (p && (i = video_get_param_len(p, ',')) != 0) {
369 if (strncmp(p, name, name_len) == 0 && p[name_len] == '=')
370 return simple_strtoul(&p[name_len + 1], NULL, 10);
371
372 p += i;
373 if (*p != 0)
374 p++; /* skip ',' */
375 }
376 return def;
377}
Hans de Goedeb7ce12d2014-12-19 15:47:37 +0100378
379/**
380 * Convert an EDID detailed timing to a struct ctfb_res_modes
381 *
382 * @param t The EDID detailed timing to be converted
383 * @param mode Returns the converted timing
384 *
Heinrich Schuchardt185f8122022-01-19 18:05:50 +0100385 * Return: 0 on success, or a negative errno on error
Hans de Goedeb7ce12d2014-12-19 15:47:37 +0100386 */
387int video_edid_dtd_to_ctfb_res_modes(struct edid_detailed_timing *t,
388 struct ctfb_res_modes *mode)
389{
390 int margin, h_total, v_total;
391
392 /* Check all timings are non 0 */
393 if (EDID_DETAILED_TIMING_PIXEL_CLOCK(*t) == 0 ||
394 EDID_DETAILED_TIMING_HORIZONTAL_ACTIVE(*t) == 0 ||
395 EDID_DETAILED_TIMING_HORIZONTAL_BLANKING(*t) == 0 ||
396 EDID_DETAILED_TIMING_VERTICAL_ACTIVE(*t) == 0 ||
397 EDID_DETAILED_TIMING_VERTICAL_BLANKING(*t) == 0 ||
398 EDID_DETAILED_TIMING_HSYNC_OFFSET(*t) == 0 ||
Hans de Goedeb7ce12d2014-12-19 15:47:37 +0100399 EDID_DETAILED_TIMING_VSYNC_OFFSET(*t) == 0 ||
Marcel Ziswiler7b48e2b2019-03-25 17:24:51 +0100400 /* 3d formats are not supported */
Hans de Goedeb7ce12d2014-12-19 15:47:37 +0100401 EDID_DETAILED_TIMING_FLAG_STEREO(*t) != 0)
402 return -EINVAL;
403
404 mode->xres = EDID_DETAILED_TIMING_HORIZONTAL_ACTIVE(*t);
405 mode->yres = EDID_DETAILED_TIMING_VERTICAL_ACTIVE(*t);
406
407 h_total = mode->xres + EDID_DETAILED_TIMING_HORIZONTAL_BLANKING(*t);
408 v_total = mode->yres + EDID_DETAILED_TIMING_VERTICAL_BLANKING(*t);
409 mode->refresh = EDID_DETAILED_TIMING_PIXEL_CLOCK(*t) /
410 (h_total * v_total);
411
412 mode->pixclock_khz = EDID_DETAILED_TIMING_PIXEL_CLOCK(*t) / 1000;
413 mode->pixclock = 1000000000L / mode->pixclock_khz;
414
415 mode->right_margin = EDID_DETAILED_TIMING_HSYNC_OFFSET(*t);
416 mode->hsync_len = EDID_DETAILED_TIMING_HSYNC_PULSE_WIDTH(*t);
417 margin = EDID_DETAILED_TIMING_HORIZONTAL_BLANKING(*t) -
418 (mode->right_margin + mode->hsync_len);
419 if (margin <= 0)
420 return -EINVAL;
421
422 mode->left_margin = margin;
423
424 mode->lower_margin = EDID_DETAILED_TIMING_VSYNC_OFFSET(*t);
425 mode->vsync_len = EDID_DETAILED_TIMING_VSYNC_PULSE_WIDTH(*t);
426 margin = EDID_DETAILED_TIMING_VERTICAL_BLANKING(*t) -
427 (mode->lower_margin + mode->vsync_len);
428 if (margin <= 0)
429 return -EINVAL;
430
431 mode->upper_margin = margin;
432
433 mode->sync = 0;
434 if (EDID_DETAILED_TIMING_FLAG_HSYNC_POLARITY(*t))
435 mode->sync |= FB_SYNC_HOR_HIGH_ACT;
436 if (EDID_DETAILED_TIMING_FLAG_VSYNC_POLARITY(*t))
437 mode->sync |= FB_SYNC_VERT_HIGH_ACT;
438
439 if (EDID_DETAILED_TIMING_FLAG_INTERLACED(*t))
440 mode->vmode = FB_VMODE_INTERLACED;
441 else
442 mode->vmode = FB_VMODE_NONINTERLACED;
443
444 return 0;
445}
Giulio Benetti10374da2020-04-08 17:10:11 +0200446
447void video_ctfb_mode_to_display_timing(const struct ctfb_res_modes *mode,
448 struct display_timing *timing)
449{
450 timing->pixelclock.typ = mode->pixclock_khz * 1000;
451
452 timing->hactive.typ = mode->xres;
453 timing->hfront_porch.typ = mode->right_margin;
454 timing->hback_porch.typ = mode->left_margin;
455 timing->hsync_len.typ = mode->hsync_len;
456
457 timing->vactive.typ = mode->yres;
458 timing->vfront_porch.typ = mode->lower_margin;
459 timing->vback_porch.typ = mode->upper_margin;
460 timing->vsync_len.typ = mode->vsync_len;
461
462 timing->flags = 0;
463
464 if (mode->sync & FB_SYNC_HOR_HIGH_ACT)
465 timing->flags |= DISPLAY_FLAGS_HSYNC_HIGH;
466 else
467 timing->flags |= DISPLAY_FLAGS_HSYNC_LOW;
468 if (mode->sync & FB_SYNC_VERT_HIGH_ACT)
469 timing->flags |= DISPLAY_FLAGS_VSYNC_HIGH;
470 else
471 timing->flags |= DISPLAY_FLAGS_VSYNC_LOW;
472 if (mode->vmode == FB_VMODE_INTERLACED)
473 timing->flags |= DISPLAY_FLAGS_INTERLACED;
474}