blob: 830ea80f4bcf79b1938d2278956c6a444fb0c0a2 [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
wdenkc6097192002-11-03 00:24:07 +00002/*
3 * (C) Copyright 2002 ELTEC Elektronik AG
4 * Frank Gottschling <fgottschling@eltec.de>
wdenkc6097192002-11-03 00:24:07 +00005 */
6
7/*
8 * cfb_console.c
9 *
10 * Color Framebuffer Console driver for 8/15/16/24/32 bits per pixel.
11 *
12 * At the moment only the 8x16 font is tested and the font fore- and
13 * background color is limited to black/white/gray colors. The Linux
14 * logo can be placed in the upper left corner and additional board
Wolfgang Denk64e40d72011-07-29 09:55:27 +000015 * information strings (that normally goes to serial port) can be drawn.
wdenkc6097192002-11-03 00:24:07 +000016 *
Simon Glass39f615e2015-11-11 10:05:47 -070017 * The console driver can use a keyboard interface for character input
18 * but this is deprecated. Only rk51 uses it.
19 *
20 * Character output goes to a memory-mapped video
wdenkc6097192002-11-03 00:24:07 +000021 * framebuffer with little or big-endian organisation.
22 * With environment setting 'console=serial' the console i/o can be
23 * forced to serial port.
Wolfgang Denk64e40d72011-07-29 09:55:27 +000024 *
25 * The driver uses graphic specific defines/parameters/functions:
26 *
27 * (for SMI LynxE graphic chip)
28 *
Wolfgang Denk64e40d72011-07-29 09:55:27 +000029 * VIDEO_FB_LITTLE_ENDIAN - framebuffer organisation default: big endian
30 * VIDEO_HW_RECTFILL - graphic driver supports hardware rectangle fill
31 * VIDEO_HW_BITBLT - graphic driver supports hardware bit blt
32 *
33 * Console Parameters are set by graphic drivers global struct:
34 *
35 * VIDEO_VISIBLE_COLS - x resolution
36 * VIDEO_VISIBLE_ROWS - y resolution
37 * VIDEO_PIXEL_SIZE - storage size in byte per pixel
38 * VIDEO_DATA_FORMAT - graphical data format GDF
39 * VIDEO_FB_ADRS - start of video memory
40 *
Wolfgang Denk64e40d72011-07-29 09:55:27 +000041 * VIDEO_KBD_INIT_FCT - init function for keyboard
42 * VIDEO_TSTC_FCT - keyboard_tstc function
43 * VIDEO_GETC_FCT - keyboard_getc function
44 *
Bastian Ruppert1e681f92012-09-13 22:29:01 +000045 * CONFIG_VIDEO_LOGO - display Linux Logo in upper left corner.
46 * Use CONFIG_SPLASH_SCREEN_ALIGN with
47 * environment variable "splashpos" to place
48 * the logo on other position. In this case
49 * no CONSOLE_EXTRA_INFO is possible.
Wolfgang Denk64e40d72011-07-29 09:55:27 +000050 * CONFIG_VIDEO_BMP_LOGO - use bmp_logo instead of linux_logo
51 * CONFIG_CONSOLE_EXTRA_INFO - display additional board information
52 * strings that normaly goes to serial
53 * port. This define requires a board
54 * specific function:
55 * video_drawstring (VIDEO_INFO_X,
56 * VIDEO_INFO_Y + i*VIDEO_FONT_HEIGHT,
57 * info);
58 * that fills a info buffer at i=row.
59 * s.a: board/eltec/bab7xx.
Wolfgang Denk64e40d72011-07-29 09:55:27 +000060 *
61 * CONFIG_VIDEO_SW_CURSOR: - Draws a cursor after the last
62 * character. No blinking is provided.
63 * Uses the macros CURSOR_SET and
64 * CURSOR_OFF.
Wolfgang Denk64e40d72011-07-29 09:55:27 +000065 */
wdenkc6097192002-11-03 00:24:07 +000066
67#include <common.h>
Simon Glass09140112020-05-10 11:40:03 -060068#include <command.h>
Simon Glass9edefc22019-11-14 12:57:37 -070069#include <cpu_func.h>
Simon Glass7b51b572019-08-01 09:46:52 -060070#include <env.h>
Simon Glass5692ccf2015-03-02 12:40:50 -070071#include <fdtdec.h>
Simon Glass0c670fc2019-08-01 09:46:36 -060072#include <gzip.h>
Simon Glassf7ae49f2020-05-10 11:40:05 -060073#include <log.h>
Pali Rohárbdfb6d72021-08-02 15:18:31 +020074#include <version_string.h>
wdenka6c7ad22002-12-03 21:28:10 +000075#include <malloc.h>
Simon Glass0a6eac82016-10-17 20:12:54 -060076#include <video.h>
Simon Glass401d1c42020-10-30 21:38:53 -060077#include <asm/global_data.h>
Wolfgang Denka9a62af2011-11-04 15:55:20 +000078#include <linux/compiler.h>
wdenka6c7ad22002-12-03 21:28:10 +000079
Simon Glassc370d382016-10-17 20:12:47 -060080#if defined(CONFIG_VIDEO_MXS)
Marek Vasutfb8ddc22013-04-28 09:20:03 +000081#define VIDEO_FB_16BPP_WORD_SWAP
82#endif
83
Wolfgang Denk64e40d72011-07-29 09:55:27 +000084/*
Helmut Raiger62a22dc2011-10-12 23:16:29 +000085 * Defines for the i.MX31 driver (mx3fb.c)
86 */
Fabio Estevam695af9a2012-05-31 07:23:56 +000087#if defined(CONFIG_VIDEO_MX3) || defined(CONFIG_VIDEO_IPUV3)
Helmut Raiger62a22dc2011-10-12 23:16:29 +000088#define VIDEO_FB_16BPP_WORD_SWAP
89#endif
90
91/*
Wolfgang Denk64e40d72011-07-29 09:55:27 +000092 * Include video_fb.h after definitions of VIDEO_HW_RECTFILL etc.
93 */
wdenkc6097192002-11-03 00:24:07 +000094#include <video_fb.h>
95
Robert Winklerdd4425e2013-06-17 11:31:29 -070096#include <splash.h>
97
Wolfgang Denk64e40d72011-07-29 09:55:27 +000098/*
99 * some Macros
100 */
wdenk4b248f32004-03-14 16:51:43 +0000101#define VIDEO_VISIBLE_COLS (pGD->winSizeX)
102#define VIDEO_VISIBLE_ROWS (pGD->winSizeY)
103#define VIDEO_PIXEL_SIZE (pGD->gdfBytesPP)
104#define VIDEO_DATA_FORMAT (pGD->gdfIndex)
105#define VIDEO_FB_ADRS (pGD->frameAdrs)
wdenkc6097192002-11-03 00:24:07 +0000106
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000107/*
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000108 * Console device
109 */
wdenkc6097192002-11-03 00:24:07 +0000110
wdenkc6097192002-11-03 00:24:07 +0000111#include <linux/types.h>
Jean-Christophe PLAGNIOL-VILLARD52cb4d42009-05-16 12:14:54 +0200112#include <stdio_dev.h>
wdenkc6097192002-11-03 00:24:07 +0000113#include <video_font.h>
wdenkc6097192002-11-03 00:24:07 +0000114
Jon Loeligerddb5d86f2007-07-10 11:13:21 -0500115#if defined(CONFIG_CMD_DATE)
116#include <rtc.h>
wdenkc6097192002-11-03 00:24:07 +0000117#endif
118
Jon Loeliger07d38a12007-07-09 17:30:01 -0500119#if defined(CONFIG_CMD_BMP) || defined(CONFIG_SPLASH_SCREEN)
wdenk4b248f32004-03-14 16:51:43 +0000120#include <watchdog.h>
121#include <bmp_layout.h>
Anatolij Gustschinff8fb562013-07-02 00:04:05 +0200122#include <splash.h>
Jon Loeliger07d38a12007-07-09 17:30:01 -0500123#endif
wdenk4b248f32004-03-14 16:51:43 +0000124
Simon Glassa4206572016-10-17 20:12:50 -0600125#if !defined(CONFIG_VIDEO_SW_CURSOR)
wdenkc6097192002-11-03 00:24:07 +0000126/* no Cursor defined */
127#define CURSOR_ON
128#define CURSOR_OFF
129#define CURSOR_SET
130#endif
131
Simon Glass7fe09332015-10-18 21:17:18 -0600132#if defined(CONFIG_VIDEO_SW_CURSOR)
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000133void console_cursor(int state);
134
Timur Tabi65618632010-08-27 15:45:47 -0500135#define CURSOR_ON console_cursor(1)
136#define CURSOR_OFF console_cursor(0)
Gabe Black03d31fc2011-11-30 13:50:50 +0000137#define CURSOR_SET video_set_cursor()
Simon Glass7fe09332015-10-18 21:17:18 -0600138#endif /* CONFIG_VIDEO_SW_CURSOR */
wdenkc6097192002-11-03 00:24:07 +0000139
wdenk4b248f32004-03-14 16:51:43 +0000140#ifdef CONFIG_VIDEO_LOGO
141#ifdef CONFIG_VIDEO_BMP_LOGO
wdenka6c7ad22002-12-03 21:28:10 +0000142#include <bmp_logo.h>
Che-Liang Chiouc2707302011-10-20 23:04:20 +0000143#include <bmp_logo_data.h>
wdenk4b248f32004-03-14 16:51:43 +0000144#define VIDEO_LOGO_WIDTH BMP_LOGO_WIDTH
145#define VIDEO_LOGO_HEIGHT BMP_LOGO_HEIGHT
146#define VIDEO_LOGO_LUT_OFFSET BMP_LOGO_OFFSET
147#define VIDEO_LOGO_COLORS BMP_LOGO_COLORS
wdenka6c7ad22002-12-03 21:28:10 +0000148
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000149#else /* CONFIG_VIDEO_BMP_LOGO */
wdenk4b248f32004-03-14 16:51:43 +0000150#define LINUX_LOGO_WIDTH 80
151#define LINUX_LOGO_HEIGHT 80
152#define LINUX_LOGO_COLORS 214
153#define LINUX_LOGO_LUT_OFFSET 0x20
wdenkc6097192002-11-03 00:24:07 +0000154#define __initdata
155#include <linux_logo.h>
wdenk4b248f32004-03-14 16:51:43 +0000156#define VIDEO_LOGO_WIDTH LINUX_LOGO_WIDTH
157#define VIDEO_LOGO_HEIGHT LINUX_LOGO_HEIGHT
158#define VIDEO_LOGO_LUT_OFFSET LINUX_LOGO_LUT_OFFSET
159#define VIDEO_LOGO_COLORS LINUX_LOGO_COLORS
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000160#endif /* CONFIG_VIDEO_BMP_LOGO */
wdenk4b248f32004-03-14 16:51:43 +0000161#define VIDEO_INFO_X (VIDEO_LOGO_WIDTH)
162#define VIDEO_INFO_Y (VIDEO_FONT_HEIGHT/2)
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000163#else /* CONFIG_VIDEO_LOGO */
wdenk4b248f32004-03-14 16:51:43 +0000164#define VIDEO_LOGO_WIDTH 0
165#define VIDEO_LOGO_HEIGHT 0
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000166#endif /* CONFIG_VIDEO_LOGO */
wdenkc6097192002-11-03 00:24:07 +0000167
wdenk4b248f32004-03-14 16:51:43 +0000168#define VIDEO_COLS VIDEO_VISIBLE_COLS
169#define VIDEO_ROWS VIDEO_VISIBLE_ROWS
Hans de Goedec67a8762015-08-04 12:15:39 +0200170#ifndef VIDEO_LINE_LEN
171#define VIDEO_LINE_LEN (VIDEO_COLS * VIDEO_PIXEL_SIZE)
172#endif
173#define VIDEO_SIZE (VIDEO_ROWS * VIDEO_LINE_LEN)
wdenk4b248f32004-03-14 16:51:43 +0000174#define VIDEO_BURST_LEN (VIDEO_COLS/8)
wdenkc6097192002-11-03 00:24:07 +0000175
wdenk4b248f32004-03-14 16:51:43 +0000176#ifdef CONFIG_VIDEO_LOGO
Matthias Weisserbe129aa2010-01-12 12:06:31 +0100177#define CONSOLE_ROWS ((VIDEO_ROWS - video_logo_height) / VIDEO_FONT_HEIGHT)
wdenkc6097192002-11-03 00:24:07 +0000178#else
wdenk4b248f32004-03-14 16:51:43 +0000179#define CONSOLE_ROWS (VIDEO_ROWS / VIDEO_FONT_HEIGHT)
wdenkc6097192002-11-03 00:24:07 +0000180#endif
181
wdenk4b248f32004-03-14 16:51:43 +0000182#define CONSOLE_COLS (VIDEO_COLS / VIDEO_FONT_WIDTH)
183#define CONSOLE_ROW_SIZE (VIDEO_FONT_HEIGHT * VIDEO_LINE_LEN)
184#define CONSOLE_ROW_FIRST (video_console_address)
185#define CONSOLE_ROW_SECOND (video_console_address + CONSOLE_ROW_SIZE)
186#define CONSOLE_ROW_LAST (video_console_address + CONSOLE_SIZE - CONSOLE_ROW_SIZE)
187#define CONSOLE_SIZE (CONSOLE_ROW_SIZE * CONSOLE_ROWS)
Simon Glass3c0b6682015-01-01 16:17:57 -0700188
189/* By default we scroll by a single line */
190#ifndef CONFIG_CONSOLE_SCROLL_LINES
191#define CONFIG_CONSOLE_SCROLL_LINES 1
192#endif
wdenkc6097192002-11-03 00:24:07 +0000193
194/* Macros */
wdenk4b248f32004-03-14 16:51:43 +0000195#ifdef VIDEO_FB_LITTLE_ENDIAN
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000196#define SWAP16(x) ((((x) & 0x00ff) << 8) | \
197 ((x) >> 8) \
198 )
199#define SWAP32(x) ((((x) & 0x000000ff) << 24) | \
200 (((x) & 0x0000ff00) << 8) | \
201 (((x) & 0x00ff0000) >> 8) | \
202 (((x) & 0xff000000) >> 24) \
203 )
204#define SHORTSWAP32(x) ((((x) & 0x000000ff) << 8) | \
205 (((x) & 0x0000ff00) >> 8) | \
206 (((x) & 0x00ff0000) << 8) | \
207 (((x) & 0xff000000) >> 8) \
208 )
wdenkc6097192002-11-03 00:24:07 +0000209#else
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000210#define SWAP16(x) (x)
211#define SWAP32(x) (x)
Wolfgang Grandegger229b6dc2009-10-23 12:03:15 +0200212#if defined(VIDEO_FB_16BPP_WORD_SWAP)
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000213#define SHORTSWAP32(x) (((x) >> 16) | ((x) << 16))
Andrew Dyercc347802008-08-29 12:30:39 -0500214#else
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000215#define SHORTSWAP32(x) (x)
Anatolij Gustschinbed53752008-01-11 14:30:01 +0100216#endif
wdenkc6097192002-11-03 00:24:07 +0000217#endif
218
Anatolij Gustschinbfd4be82012-06-05 09:19:18 +0200219DECLARE_GLOBAL_DATA_PTR;
220
wdenkc6097192002-11-03 00:24:07 +0000221/* Locals */
222static GraphicDevice *pGD; /* Pointer to Graphic array */
223
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000224static void *video_fb_address; /* frame buffer address */
wdenk4b248f32004-03-14 16:51:43 +0000225static void *video_console_address; /* console buffer start address */
wdenkc6097192002-11-03 00:24:07 +0000226
Matthias Weisserbe129aa2010-01-12 12:06:31 +0100227static int video_logo_height = VIDEO_LOGO_HEIGHT;
228
Anatolij Gustschina45adde2011-12-07 03:58:10 +0000229static int __maybe_unused cursor_state;
230static int __maybe_unused old_col;
231static int __maybe_unused old_row;
Gabe Black03d31fc2011-11-30 13:50:50 +0000232
Wolfgang Denk57912932011-07-30 12:48:09 +0000233static int console_col; /* cursor col */
234static int console_row; /* cursor row */
wdenkc6097192002-11-03 00:24:07 +0000235
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000236static u32 eorx, fgx, bgx; /* color pats */
wdenkc6097192002-11-03 00:24:07 +0000237
Anatolij Gustschinbfd4be82012-06-05 09:19:18 +0200238static int cfb_do_flush_cache;
239
Pali Rohár33a35bb2012-10-19 13:30:09 +0000240#ifdef CONFIG_CFB_CONSOLE_ANSI
241static char ansi_buf[10];
242static int ansi_buf_size;
243static int ansi_colors_need_revert;
244static int ansi_cursor_hidden;
245#endif
246
wdenkc6097192002-11-03 00:24:07 +0000247static const int video_font_draw_table8[] = {
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000248 0x00000000, 0x000000ff, 0x0000ff00, 0x0000ffff,
249 0x00ff0000, 0x00ff00ff, 0x00ffff00, 0x00ffffff,
250 0xff000000, 0xff0000ff, 0xff00ff00, 0xff00ffff,
251 0xffff0000, 0xffff00ff, 0xffffff00, 0xffffffff
252};
wdenkc6097192002-11-03 00:24:07 +0000253
254static const int video_font_draw_table15[] = {
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000255 0x00000000, 0x00007fff, 0x7fff0000, 0x7fff7fff
256};
wdenkc6097192002-11-03 00:24:07 +0000257
258static const int video_font_draw_table16[] = {
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000259 0x00000000, 0x0000ffff, 0xffff0000, 0xffffffff
260};
wdenkc6097192002-11-03 00:24:07 +0000261
262static const int video_font_draw_table24[16][3] = {
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000263 {0x00000000, 0x00000000, 0x00000000},
264 {0x00000000, 0x00000000, 0x00ffffff},
265 {0x00000000, 0x0000ffff, 0xff000000},
266 {0x00000000, 0x0000ffff, 0xffffffff},
267 {0x000000ff, 0xffff0000, 0x00000000},
268 {0x000000ff, 0xffff0000, 0x00ffffff},
269 {0x000000ff, 0xffffffff, 0xff000000},
270 {0x000000ff, 0xffffffff, 0xffffffff},
271 {0xffffff00, 0x00000000, 0x00000000},
272 {0xffffff00, 0x00000000, 0x00ffffff},
273 {0xffffff00, 0x0000ffff, 0xff000000},
274 {0xffffff00, 0x0000ffff, 0xffffffff},
275 {0xffffffff, 0xffff0000, 0x00000000},
276 {0xffffffff, 0xffff0000, 0x00ffffff},
277 {0xffffffff, 0xffffffff, 0xff000000},
278 {0xffffffff, 0xffffffff, 0xffffffff}
279};
wdenkc6097192002-11-03 00:24:07 +0000280
281static const int video_font_draw_table32[16][4] = {
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000282 {0x00000000, 0x00000000, 0x00000000, 0x00000000},
283 {0x00000000, 0x00000000, 0x00000000, 0x00ffffff},
284 {0x00000000, 0x00000000, 0x00ffffff, 0x00000000},
285 {0x00000000, 0x00000000, 0x00ffffff, 0x00ffffff},
286 {0x00000000, 0x00ffffff, 0x00000000, 0x00000000},
287 {0x00000000, 0x00ffffff, 0x00000000, 0x00ffffff},
288 {0x00000000, 0x00ffffff, 0x00ffffff, 0x00000000},
289 {0x00000000, 0x00ffffff, 0x00ffffff, 0x00ffffff},
290 {0x00ffffff, 0x00000000, 0x00000000, 0x00000000},
291 {0x00ffffff, 0x00000000, 0x00000000, 0x00ffffff},
292 {0x00ffffff, 0x00000000, 0x00ffffff, 0x00000000},
293 {0x00ffffff, 0x00000000, 0x00ffffff, 0x00ffffff},
294 {0x00ffffff, 0x00ffffff, 0x00000000, 0x00000000},
295 {0x00ffffff, 0x00ffffff, 0x00000000, 0x00ffffff},
296 {0x00ffffff, 0x00ffffff, 0x00ffffff, 0x00000000},
297 {0x00ffffff, 0x00ffffff, 0x00ffffff, 0x00ffffff}
298};
wdenkc6097192002-11-03 00:24:07 +0000299
Heiko Schocher2bc4aa52013-08-03 07:22:53 +0200300/*
301 * Implement a weak default function for boards that optionally
302 * need to skip the cfb initialization.
303 */
304__weak int board_cfb_skip(void)
305{
306 /* As default, don't skip cfb init */
307 return 0;
308}
309
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000310static void video_drawchars(int xx, int yy, unsigned char *s, int count)
wdenkc6097192002-11-03 00:24:07 +0000311{
wdenk4b248f32004-03-14 16:51:43 +0000312 u8 *cdat, *dest, *dest0;
313 int rows, offset, c;
wdenkc6097192002-11-03 00:24:07 +0000314
wdenk4b248f32004-03-14 16:51:43 +0000315 offset = yy * VIDEO_LINE_LEN + xx * VIDEO_PIXEL_SIZE;
316 dest0 = video_fb_address + offset;
wdenkc6097192002-11-03 00:24:07 +0000317
wdenk4b248f32004-03-14 16:51:43 +0000318 switch (VIDEO_DATA_FORMAT) {
319 case GDF__8BIT_INDEX:
320 case GDF__8BIT_332RGB:
321 while (count--) {
322 c = *s;
323 cdat = video_fontdata + c * VIDEO_FONT_HEIGHT;
324 for (rows = VIDEO_FONT_HEIGHT, dest = dest0;
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000325 rows--; dest += VIDEO_LINE_LEN) {
wdenk4b248f32004-03-14 16:51:43 +0000326 u8 bits = *cdat++;
wdenkc6097192002-11-03 00:24:07 +0000327
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000328 ((u32 *) dest)[0] =
329 (video_font_draw_table8[bits >> 4] &
330 eorx) ^ bgx;
Marek Vasutfd8cf992013-07-30 23:37:59 +0200331
332 if (VIDEO_FONT_WIDTH == 4)
333 continue;
334
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000335 ((u32 *) dest)[1] =
336 (video_font_draw_table8[bits & 15] &
337 eorx) ^ bgx;
wdenk4b248f32004-03-14 16:51:43 +0000338 }
339 dest0 += VIDEO_FONT_WIDTH * VIDEO_PIXEL_SIZE;
340 s++;
341 }
342 break;
wdenkc6097192002-11-03 00:24:07 +0000343
wdenk4b248f32004-03-14 16:51:43 +0000344 case GDF_15BIT_555RGB:
345 while (count--) {
346 c = *s;
347 cdat = video_fontdata + c * VIDEO_FONT_HEIGHT;
348 for (rows = VIDEO_FONT_HEIGHT, dest = dest0;
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000349 rows--; dest += VIDEO_LINE_LEN) {
wdenk4b248f32004-03-14 16:51:43 +0000350 u8 bits = *cdat++;
wdenkc6097192002-11-03 00:24:07 +0000351
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000352 ((u32 *) dest)[0] =
353 SHORTSWAP32((video_font_draw_table15
354 [bits >> 6] & eorx) ^
355 bgx);
356 ((u32 *) dest)[1] =
357 SHORTSWAP32((video_font_draw_table15
358 [bits >> 4 & 3] & eorx) ^
359 bgx);
Marek Vasutfd8cf992013-07-30 23:37:59 +0200360
361 if (VIDEO_FONT_WIDTH == 4)
362 continue;
363
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000364 ((u32 *) dest)[2] =
365 SHORTSWAP32((video_font_draw_table15
366 [bits >> 2 & 3] & eorx) ^
367 bgx);
368 ((u32 *) dest)[3] =
369 SHORTSWAP32((video_font_draw_table15
370 [bits & 3] & eorx) ^
371 bgx);
wdenk4b248f32004-03-14 16:51:43 +0000372 }
373 dest0 += VIDEO_FONT_WIDTH * VIDEO_PIXEL_SIZE;
374 s++;
375 }
376 break;
wdenkc6097192002-11-03 00:24:07 +0000377
wdenk4b248f32004-03-14 16:51:43 +0000378 case GDF_16BIT_565RGB:
379 while (count--) {
380 c = *s;
381 cdat = video_fontdata + c * VIDEO_FONT_HEIGHT;
382 for (rows = VIDEO_FONT_HEIGHT, dest = dest0;
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000383 rows--; dest += VIDEO_LINE_LEN) {
wdenk4b248f32004-03-14 16:51:43 +0000384 u8 bits = *cdat++;
385
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000386 ((u32 *) dest)[0] =
387 SHORTSWAP32((video_font_draw_table16
388 [bits >> 6] & eorx) ^
389 bgx);
390 ((u32 *) dest)[1] =
391 SHORTSWAP32((video_font_draw_table16
392 [bits >> 4 & 3] & eorx) ^
393 bgx);
Marek Vasutfd8cf992013-07-30 23:37:59 +0200394
395 if (VIDEO_FONT_WIDTH == 4)
396 continue;
397
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000398 ((u32 *) dest)[2] =
399 SHORTSWAP32((video_font_draw_table16
400 [bits >> 2 & 3] & eorx) ^
401 bgx);
402 ((u32 *) dest)[3] =
403 SHORTSWAP32((video_font_draw_table16
404 [bits & 3] & eorx) ^
405 bgx);
wdenk4b248f32004-03-14 16:51:43 +0000406 }
407 dest0 += VIDEO_FONT_WIDTH * VIDEO_PIXEL_SIZE;
408 s++;
409 }
410 break;
411
412 case GDF_32BIT_X888RGB:
413 while (count--) {
414 c = *s;
415 cdat = video_fontdata + c * VIDEO_FONT_HEIGHT;
416 for (rows = VIDEO_FONT_HEIGHT, dest = dest0;
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000417 rows--; dest += VIDEO_LINE_LEN) {
wdenk4b248f32004-03-14 16:51:43 +0000418 u8 bits = *cdat++;
419
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000420 ((u32 *) dest)[0] =
421 SWAP32((video_font_draw_table32
422 [bits >> 4][0] & eorx) ^ bgx);
423 ((u32 *) dest)[1] =
424 SWAP32((video_font_draw_table32
425 [bits >> 4][1] & eorx) ^ bgx);
426 ((u32 *) dest)[2] =
427 SWAP32((video_font_draw_table32
428 [bits >> 4][2] & eorx) ^ bgx);
429 ((u32 *) dest)[3] =
430 SWAP32((video_font_draw_table32
431 [bits >> 4][3] & eorx) ^ bgx);
Marek Vasutfd8cf992013-07-30 23:37:59 +0200432
433
434 if (VIDEO_FONT_WIDTH == 4)
435 continue;
436
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000437 ((u32 *) dest)[4] =
438 SWAP32((video_font_draw_table32
439 [bits & 15][0] & eorx) ^ bgx);
440 ((u32 *) dest)[5] =
441 SWAP32((video_font_draw_table32
442 [bits & 15][1] & eorx) ^ bgx);
443 ((u32 *) dest)[6] =
444 SWAP32((video_font_draw_table32
445 [bits & 15][2] & eorx) ^ bgx);
446 ((u32 *) dest)[7] =
447 SWAP32((video_font_draw_table32
448 [bits & 15][3] & eorx) ^ bgx);
wdenk4b248f32004-03-14 16:51:43 +0000449 }
450 dest0 += VIDEO_FONT_WIDTH * VIDEO_PIXEL_SIZE;
451 s++;
452 }
453 break;
454
455 case GDF_24BIT_888RGB:
456 while (count--) {
457 c = *s;
458 cdat = video_fontdata + c * VIDEO_FONT_HEIGHT;
459 for (rows = VIDEO_FONT_HEIGHT, dest = dest0;
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000460 rows--; dest += VIDEO_LINE_LEN) {
wdenk4b248f32004-03-14 16:51:43 +0000461 u8 bits = *cdat++;
462
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000463 ((u32 *) dest)[0] =
464 (video_font_draw_table24[bits >> 4][0]
465 & eorx) ^ bgx;
466 ((u32 *) dest)[1] =
467 (video_font_draw_table24[bits >> 4][1]
468 & eorx) ^ bgx;
469 ((u32 *) dest)[2] =
470 (video_font_draw_table24[bits >> 4][2]
471 & eorx) ^ bgx;
Marek Vasutfd8cf992013-07-30 23:37:59 +0200472
473 if (VIDEO_FONT_WIDTH == 4)
474 continue;
475
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000476 ((u32 *) dest)[3] =
477 (video_font_draw_table24[bits & 15][0]
478 & eorx) ^ bgx;
479 ((u32 *) dest)[4] =
480 (video_font_draw_table24[bits & 15][1]
481 & eorx) ^ bgx;
482 ((u32 *) dest)[5] =
483 (video_font_draw_table24[bits & 15][2]
484 & eorx) ^ bgx;
wdenk4b248f32004-03-14 16:51:43 +0000485 }
486 dest0 += VIDEO_FONT_WIDTH * VIDEO_PIXEL_SIZE;
487 s++;
488 }
489 break;
wdenk8bde7f72003-06-27 21:31:46 +0000490 }
wdenkc6097192002-11-03 00:24:07 +0000491}
492
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000493static inline void video_drawstring(int xx, int yy, unsigned char *s)
wdenkc6097192002-11-03 00:24:07 +0000494{
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000495 video_drawchars(xx, yy, s, strlen((char *) s));
wdenkc6097192002-11-03 00:24:07 +0000496}
497
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000498static void video_putchar(int xx, int yy, unsigned char c)
wdenkc6097192002-11-03 00:24:07 +0000499{
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000500 video_drawchars(xx, yy + video_logo_height, &c, 1);
wdenkc6097192002-11-03 00:24:07 +0000501}
502
Simon Glass7fe09332015-10-18 21:17:18 -0600503#if defined(CONFIG_VIDEO_SW_CURSOR)
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000504static void video_set_cursor(void)
wdenkc6097192002-11-03 00:24:07 +0000505{
Gabe Black03d31fc2011-11-30 13:50:50 +0000506 if (cursor_state)
507 console_cursor(0);
508 console_cursor(1);
wdenkc6097192002-11-03 00:24:07 +0000509}
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000510
Anatolij Gustschina45adde2011-12-07 03:58:10 +0000511static void video_invertchar(int xx, int yy)
512{
513 int firstx = xx * VIDEO_PIXEL_SIZE;
514 int lastx = (xx + VIDEO_FONT_WIDTH) * VIDEO_PIXEL_SIZE;
515 int firsty = yy * VIDEO_LINE_LEN;
516 int lasty = (yy + VIDEO_FONT_HEIGHT) * VIDEO_LINE_LEN;
517 int x, y;
518 for (y = firsty; y < lasty; y += VIDEO_LINE_LEN) {
519 for (x = firstx; x < lastx; x++) {
520 u8 *dest = (u8 *)(video_fb_address) + x + y;
521 *dest = ~*dest;
522 }
523 }
524}
Gabe Black03d31fc2011-11-30 13:50:50 +0000525
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000526void console_cursor(int state)
wdenkc6097192002-11-03 00:24:07 +0000527{
Gabe Black03d31fc2011-11-30 13:50:50 +0000528 if (cursor_state != state) {
529 if (cursor_state) {
530 /* turn off the cursor */
531 video_invertchar(old_col * VIDEO_FONT_WIDTH,
532 old_row * VIDEO_FONT_HEIGHT +
533 video_logo_height);
534 } else {
535 /* turn off the cursor and record where it is */
536 video_invertchar(console_col * VIDEO_FONT_WIDTH,
537 console_row * VIDEO_FONT_HEIGHT +
538 video_logo_height);
539 old_col = console_col;
540 old_row = console_row;
541 }
542 cursor_state = state;
wdenk4b248f32004-03-14 16:51:43 +0000543 }
Eric Nelsondb0d47d2013-05-06 14:27:28 +0200544 if (cfb_do_flush_cache)
545 flush_cache(VIDEO_FB_ADRS, VIDEO_SIZE);
wdenkc6097192002-11-03 00:24:07 +0000546}
547#endif
548
wdenkc6097192002-11-03 00:24:07 +0000549#ifndef VIDEO_HW_RECTFILL
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000550static void memsetl(int *p, int c, int v)
wdenkc6097192002-11-03 00:24:07 +0000551{
wdenk4b248f32004-03-14 16:51:43 +0000552 while (c--)
553 *(p++) = v;
wdenkc6097192002-11-03 00:24:07 +0000554}
555#endif
556
wdenkc6097192002-11-03 00:24:07 +0000557#ifndef VIDEO_HW_BITBLT
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000558static void memcpyl(int *d, int *s, int c)
wdenkc6097192002-11-03 00:24:07 +0000559{
wdenk4b248f32004-03-14 16:51:43 +0000560 while (c--)
561 *(d++) = *(s++);
wdenkc6097192002-11-03 00:24:07 +0000562}
563#endif
564
Pali Rohár90f60a82012-04-28 07:26:44 +0000565static void console_clear_line(int line, int begin, int end)
566{
567#ifdef VIDEO_HW_RECTFILL
568 video_hw_rectfill(VIDEO_PIXEL_SIZE, /* bytes per pixel */
569 VIDEO_FONT_WIDTH * begin, /* dest pos x */
570 video_logo_height +
571 VIDEO_FONT_HEIGHT * line, /* dest pos y */
572 VIDEO_FONT_WIDTH * (end - begin + 1), /* fr. width */
573 VIDEO_FONT_HEIGHT, /* frame height */
574 bgx /* fill color */
575 );
576#else
577 if (begin == 0 && (end + 1) == CONSOLE_COLS) {
578 memsetl(CONSOLE_ROW_FIRST +
579 CONSOLE_ROW_SIZE * line, /* offset of row */
580 CONSOLE_ROW_SIZE >> 2, /* length of row */
581 bgx /* fill color */
582 );
583 } else {
584 void *offset;
585 int i, size;
586
587 offset = CONSOLE_ROW_FIRST +
588 CONSOLE_ROW_SIZE * line + /* offset of row */
589 VIDEO_FONT_WIDTH *
590 VIDEO_PIXEL_SIZE * begin; /* offset of col */
591 size = VIDEO_FONT_WIDTH * VIDEO_PIXEL_SIZE * (end - begin + 1);
592 size >>= 2; /* length to end for memsetl() */
593 /* fill at col offset of i'th line using bgx as fill color */
594 for (i = 0; i < VIDEO_FONT_HEIGHT; i++)
595 memsetl(offset + i * VIDEO_LINE_LEN, size, bgx);
596 }
597#endif
598}
599
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000600static void console_scrollup(void)
wdenkc6097192002-11-03 00:24:07 +0000601{
Simon Glass3c0b6682015-01-01 16:17:57 -0700602 const int rows = CONFIG_CONSOLE_SCROLL_LINES;
603 int i;
604
wdenk4b248f32004-03-14 16:51:43 +0000605 /* copy up rows ignoring the first one */
wdenkc6097192002-11-03 00:24:07 +0000606
607#ifdef VIDEO_HW_BITBLT
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000608 video_hw_bitblt(VIDEO_PIXEL_SIZE, /* bytes per pixel */
609 0, /* source pos x */
610 video_logo_height +
Simon Glass3c0b6682015-01-01 16:17:57 -0700611 VIDEO_FONT_HEIGHT * rows, /* source pos y */
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000612 0, /* dest pos x */
613 video_logo_height, /* dest pos y */
614 VIDEO_VISIBLE_COLS, /* frame width */
615 VIDEO_VISIBLE_ROWS
616 - video_logo_height
Simon Glass3c0b6682015-01-01 16:17:57 -0700617 - VIDEO_FONT_HEIGHT * rows /* frame height */
wdenk4b248f32004-03-14 16:51:43 +0000618 );
wdenkc6097192002-11-03 00:24:07 +0000619#else
Simon Glass3c0b6682015-01-01 16:17:57 -0700620 memcpyl(CONSOLE_ROW_FIRST, CONSOLE_ROW_FIRST + rows * CONSOLE_ROW_SIZE,
621 (CONSOLE_SIZE - CONSOLE_ROW_SIZE * rows) >> 2);
wdenkc6097192002-11-03 00:24:07 +0000622#endif
wdenk4b248f32004-03-14 16:51:43 +0000623 /* clear the last one */
Simon Glass3c0b6682015-01-01 16:17:57 -0700624 for (i = 1; i <= rows; i++)
625 console_clear_line(CONSOLE_ROWS - i, 0, CONSOLE_COLS - 1);
626
627 /* Decrement row number */
628 console_row -= rows;
wdenkc6097192002-11-03 00:24:07 +0000629}
630
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000631static void console_back(void)
wdenkc6097192002-11-03 00:24:07 +0000632{
Timur Tabi65618632010-08-27 15:45:47 -0500633 console_col--;
wdenkc6097192002-11-03 00:24:07 +0000634
wdenk4b248f32004-03-14 16:51:43 +0000635 if (console_col < 0) {
636 console_col = CONSOLE_COLS - 1;
637 console_row--;
638 if (console_row < 0)
639 console_row = 0;
640 }
wdenkc6097192002-11-03 00:24:07 +0000641}
642
Pali Rohár33a35bb2012-10-19 13:30:09 +0000643#ifdef CONFIG_CFB_CONSOLE_ANSI
644
645static void console_clear(void)
wdenkc6097192002-11-03 00:24:07 +0000646{
Pali Rohár33a35bb2012-10-19 13:30:09 +0000647#ifdef VIDEO_HW_RECTFILL
648 video_hw_rectfill(VIDEO_PIXEL_SIZE, /* bytes per pixel */
649 0, /* dest pos x */
650 video_logo_height, /* dest pos y */
651 VIDEO_VISIBLE_COLS, /* frame width */
652 VIDEO_VISIBLE_ROWS, /* frame height */
653 bgx /* fill color */
654 );
655#else
656 memsetl(CONSOLE_ROW_FIRST, CONSOLE_SIZE, bgx);
657#endif
658}
659
660static void console_cursor_fix(void)
661{
662 if (console_row < 0)
663 console_row = 0;
664 if (console_row >= CONSOLE_ROWS)
665 console_row = CONSOLE_ROWS - 1;
666 if (console_col < 0)
667 console_col = 0;
668 if (console_col >= CONSOLE_COLS)
669 console_col = CONSOLE_COLS - 1;
670}
671
672static void console_cursor_up(int n)
673{
674 console_row -= n;
675 console_cursor_fix();
676}
677
678static void console_cursor_down(int n)
679{
680 console_row += n;
681 console_cursor_fix();
682}
683
684static void console_cursor_left(int n)
685{
686 console_col -= n;
687 console_cursor_fix();
688}
689
690static void console_cursor_right(int n)
691{
692 console_col += n;
693 console_cursor_fix();
694}
695
696static void console_cursor_set_position(int row, int col)
697{
698 if (console_row != -1)
699 console_row = row;
700 if (console_col != -1)
701 console_col = col;
702 console_cursor_fix();
703}
704
705static void console_previousline(int n)
706{
707 /* FIXME: also scroll terminal ? */
708 console_row -= n;
709 console_cursor_fix();
710}
711
712static void console_swap_colors(void)
713{
714 eorx = fgx;
715 fgx = bgx;
716 bgx = eorx;
717 eorx = fgx ^ bgx;
718}
719
720static inline int console_cursor_is_visible(void)
721{
722 return !ansi_cursor_hidden;
723}
724#else
725static inline int console_cursor_is_visible(void)
726{
727 return 1;
728}
729#endif
730
731static void console_newline(int n)
732{
733 console_row += n;
wdenk4b248f32004-03-14 16:51:43 +0000734 console_col = 0;
wdenkc6097192002-11-03 00:24:07 +0000735
wdenk4b248f32004-03-14 16:51:43 +0000736 /* Check if we need to scroll the terminal */
737 if (console_row >= CONSOLE_ROWS) {
738 /* Scroll everything up */
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000739 console_scrollup();
wdenk4b248f32004-03-14 16:51:43 +0000740 }
wdenkc6097192002-11-03 00:24:07 +0000741}
742
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000743static void console_cr(void)
Anatolij Gustschin20c450e2008-01-11 02:39:47 +0100744{
Timur Tabi65618632010-08-27 15:45:47 -0500745 console_col = 0;
Anatolij Gustschin20c450e2008-01-11 02:39:47 +0100746}
747
Pali Rohár33a35bb2012-10-19 13:30:09 +0000748static void parse_putc(const char c)
wdenkc6097192002-11-03 00:24:07 +0000749{
Anatolij Gustschin20c450e2008-01-11 02:39:47 +0100750 static int nl = 1;
751
Pali Rohár33a35bb2012-10-19 13:30:09 +0000752 if (console_cursor_is_visible())
753 CURSOR_OFF;
Gabe Black03d31fc2011-11-30 13:50:50 +0000754
wdenk4b248f32004-03-14 16:51:43 +0000755 switch (c) {
Anatolij Gustschin20c450e2008-01-11 02:39:47 +0100756 case 13: /* back to first column */
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000757 console_cr();
wdenk4b248f32004-03-14 16:51:43 +0000758 break;
wdenkc6097192002-11-03 00:24:07 +0000759
wdenk4b248f32004-03-14 16:51:43 +0000760 case '\n': /* next line */
Heinrich Schuchardt41ec1272018-03-18 14:24:39 +0100761 if (console_col || nl)
Pali Rohár33a35bb2012-10-19 13:30:09 +0000762 console_newline(1);
Anatolij Gustschin20c450e2008-01-11 02:39:47 +0100763 nl = 1;
wdenk4b248f32004-03-14 16:51:43 +0000764 break;
wdenkc6097192002-11-03 00:24:07 +0000765
wdenk4b248f32004-03-14 16:51:43 +0000766 case 9: /* tab 8 */
Timur Tabi65618632010-08-27 15:45:47 -0500767 console_col |= 0x0008;
wdenk4b248f32004-03-14 16:51:43 +0000768 console_col &= ~0x0007;
wdenkc6097192002-11-03 00:24:07 +0000769
wdenk4b248f32004-03-14 16:51:43 +0000770 if (console_col >= CONSOLE_COLS)
Pali Rohár33a35bb2012-10-19 13:30:09 +0000771 console_newline(1);
wdenk4b248f32004-03-14 16:51:43 +0000772 break;
wdenkc6097192002-11-03 00:24:07 +0000773
wdenk4b248f32004-03-14 16:51:43 +0000774 case 8: /* backspace */
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000775 console_back();
wdenk4b248f32004-03-14 16:51:43 +0000776 break;
wdenkc6097192002-11-03 00:24:07 +0000777
Pali Rohár24fe06c2012-04-28 07:26:47 +0000778 case 7: /* bell */
779 break; /* ignored */
780
wdenk4b248f32004-03-14 16:51:43 +0000781 default: /* draw the char */
Wolfgang Denk64e40d72011-07-29 09:55:27 +0000782 video_putchar(console_col * VIDEO_FONT_WIDTH,
783 console_row * VIDEO_FONT_HEIGHT, c);
wdenk4b248f32004-03-14 16:51:43 +0000784 console_col++;
wdenkc6097192002-11-03 00:24:07 +0000785
wdenk4b248f32004-03-14 16:51:43 +0000786 /* check for newline */
Anatolij Gustschin20c450e2008-01-11 02:39:47 +0100787 if (console_col >= CONSOLE_COLS) {
Pali Rohár33a35bb2012-10-19 13:30:09 +0000788 console_newline(1);
Anatolij Gustschin20c450e2008-01-11 02:39:47 +0100789 nl = 0;
790 }
wdenk4b248f32004-03-14 16:51:43 +0000791 }
Pali Rohár33a35bb2012-10-19 13:30:09 +0000792
793 if (console_cursor_is_visible())
794 CURSOR_SET;
795}
796
Simon Glass0a6eac82016-10-17 20:12:54 -0600797static void cfb_video_putc(struct stdio_dev *dev, const char c)
Pali Rohár33a35bb2012-10-19 13:30:09 +0000798{
799#ifdef CONFIG_CFB_CONSOLE_ANSI
800 int i;
801
802 if (c == 27) {
803 for (i = 0; i < ansi_buf_size; ++i)
804 parse_putc(ansi_buf[i]);
805 ansi_buf[0] = 27;
806 ansi_buf_size = 1;
807 return;
808 }
809
810 if (ansi_buf_size > 0) {
811 /*
812 * 0 - ESC
813 * 1 - [
814 * 2 - num1
815 * 3 - ..
816 * 4 - ;
817 * 5 - num2
818 * 6 - ..
819 * - cchar
820 */
821 int next = 0;
822
823 int flush = 0;
824 int fail = 0;
825
826 int num1 = 0;
827 int num2 = 0;
828 int cchar = 0;
829
830 ansi_buf[ansi_buf_size++] = c;
831
832 if (ansi_buf_size >= sizeof(ansi_buf))
833 fail = 1;
834
835 for (i = 0; i < ansi_buf_size; ++i) {
836 if (fail)
837 break;
838
839 switch (next) {
840 case 0:
841 if (ansi_buf[i] == 27)
842 next = 1;
843 else
844 fail = 1;
845 break;
846
847 case 1:
848 if (ansi_buf[i] == '[')
849 next = 2;
850 else
851 fail = 1;
852 break;
853
854 case 2:
855 if (ansi_buf[i] >= '0' && ansi_buf[i] <= '9') {
856 num1 = ansi_buf[i]-'0';
857 next = 3;
858 } else if (ansi_buf[i] != '?') {
859 --i;
860 num1 = 1;
861 next = 4;
862 }
863 break;
864
865 case 3:
866 if (ansi_buf[i] >= '0' && ansi_buf[i] <= '9') {
867 num1 *= 10;
868 num1 += ansi_buf[i]-'0';
869 } else {
870 --i;
871 next = 4;
872 }
873 break;
874
875 case 4:
876 if (ansi_buf[i] != ';') {
877 --i;
878 next = 7;
879 } else
880 next = 5;
881 break;
882
883 case 5:
884 if (ansi_buf[i] >= '0' && ansi_buf[i] <= '9') {
885 num2 = ansi_buf[i]-'0';
886 next = 6;
887 } else
888 fail = 1;
889 break;
890
891 case 6:
892 if (ansi_buf[i] >= '0' && ansi_buf[i] <= '9') {
893 num2 *= 10;
894 num2 += ansi_buf[i]-'0';
895 } else {
896 --i;
897 next = 7;
898 }
899 break;
900
901 case 7:
902 if ((ansi_buf[i] >= 'A' && ansi_buf[i] <= 'H')
903 || ansi_buf[i] == 'J'
904 || ansi_buf[i] == 'K'
905 || ansi_buf[i] == 'h'
906 || ansi_buf[i] == 'l'
907 || ansi_buf[i] == 'm') {
908 cchar = ansi_buf[i];
909 flush = 1;
910 } else
911 fail = 1;
912 break;
913 }
914 }
915
916 if (fail) {
917 for (i = 0; i < ansi_buf_size; ++i)
918 parse_putc(ansi_buf[i]);
919 ansi_buf_size = 0;
920 return;
921 }
922
923 if (flush) {
924 if (!ansi_cursor_hidden)
925 CURSOR_OFF;
926 ansi_buf_size = 0;
927 switch (cchar) {
928 case 'A':
929 /* move cursor num1 rows up */
930 console_cursor_up(num1);
931 break;
932 case 'B':
933 /* move cursor num1 rows down */
934 console_cursor_down(num1);
935 break;
936 case 'C':
937 /* move cursor num1 columns forward */
938 console_cursor_right(num1);
939 break;
940 case 'D':
941 /* move cursor num1 columns back */
942 console_cursor_left(num1);
943 break;
944 case 'E':
945 /* move cursor num1 rows up at begin of row */
946 console_previousline(num1);
947 break;
948 case 'F':
949 /* move cursor num1 rows down at begin of row */
950 console_newline(num1);
951 break;
952 case 'G':
953 /* move cursor to column num1 */
954 console_cursor_set_position(-1, num1-1);
955 break;
956 case 'H':
957 /* move cursor to row num1, column num2 */
958 console_cursor_set_position(num1-1, num2-1);
959 break;
960 case 'J':
961 /* clear console and move cursor to 0, 0 */
962 console_clear();
963 console_cursor_set_position(0, 0);
964 break;
965 case 'K':
966 /* clear line */
967 if (num1 == 0)
968 console_clear_line(console_row,
969 console_col,
970 CONSOLE_COLS-1);
971 else if (num1 == 1)
972 console_clear_line(console_row,
973 0, console_col);
974 else
975 console_clear_line(console_row,
976 0, CONSOLE_COLS-1);
977 break;
978 case 'h':
979 ansi_cursor_hidden = 0;
980 break;
981 case 'l':
982 ansi_cursor_hidden = 1;
983 break;
984 case 'm':
985 if (num1 == 0) { /* reset swapped colors */
986 if (ansi_colors_need_revert) {
987 console_swap_colors();
988 ansi_colors_need_revert = 0;
989 }
990 } else if (num1 == 7) { /* once swap colors */
991 if (!ansi_colors_need_revert) {
992 console_swap_colors();
993 ansi_colors_need_revert = 1;
994 }
995 }
996 break;
997 }
998 if (!ansi_cursor_hidden)
999 CURSOR_SET;
1000 }
1001 } else {
1002 parse_putc(c);
1003 }
1004#else
1005 parse_putc(c);
1006#endif
Eric Nelsondb0d47d2013-05-06 14:27:28 +02001007 if (cfb_do_flush_cache)
1008 flush_cache(VIDEO_FB_ADRS, VIDEO_SIZE);
Timur Tabi65618632010-08-27 15:45:47 -05001009}
wdenkc6097192002-11-03 00:24:07 +00001010
Simon Glass0a6eac82016-10-17 20:12:54 -06001011static void cfb_video_puts(struct stdio_dev *dev, const char *s)
wdenkc6097192002-11-03 00:24:07 +00001012{
Soeren Mochd37e96e2014-10-24 16:33:30 +02001013 int flush = cfb_do_flush_cache;
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001014 int count = strlen(s);
wdenkc6097192002-11-03 00:24:07 +00001015
Soeren Mochd37e96e2014-10-24 16:33:30 +02001016 /* temporarily disable cache flush */
1017 cfb_do_flush_cache = 0;
1018
wdenk4b248f32004-03-14 16:51:43 +00001019 while (count--)
Simon Glass0a6eac82016-10-17 20:12:54 -06001020 cfb_video_putc(dev, *s++);
Soeren Mochd37e96e2014-10-24 16:33:30 +02001021
1022 if (flush) {
1023 cfb_do_flush_cache = flush;
1024 flush_cache(VIDEO_FB_ADRS, VIDEO_SIZE);
1025 }
wdenkc6097192002-11-03 00:24:07 +00001026}
1027
Anatolij Gustschin10543822010-05-01 22:21:09 +02001028/*
1029 * Do not enforce drivers (or board code) to provide empty
1030 * video_set_lut() if they do not support 8 bpp format.
1031 * Implement weak default function instead.
1032 */
Jeroen Hofstee69d27542014-10-08 22:57:30 +02001033__weak void video_set_lut(unsigned int index, unsigned char r,
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001034 unsigned char g, unsigned char b)
Anatolij Gustschin10543822010-05-01 22:21:09 +02001035{
1036}
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001037
Jon Loeliger07d38a12007-07-09 17:30:01 -05001038#if defined(CONFIG_CMD_BMP) || defined(CONFIG_SPLASH_SCREEN)
wdenk4b248f32004-03-14 16:51:43 +00001039
1040#define FILL_8BIT_332RGB(r,g,b) { \
1041 *fb = ((r>>5)<<5) | ((g>>5)<<2) | (b>>6); \
1042 fb ++; \
1043}
1044
1045#define FILL_15BIT_555RGB(r,g,b) { \
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001046 *(unsigned short *)fb = \
1047 SWAP16((unsigned short)(((r>>3)<<10) | \
1048 ((g>>3)<<5) | \
1049 (b>>3))); \
wdenk4b248f32004-03-14 16:51:43 +00001050 fb += 2; \
1051}
1052
1053#define FILL_16BIT_565RGB(r,g,b) { \
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001054 *(unsigned short *)fb = \
1055 SWAP16((unsigned short)((((r)>>3)<<11)| \
1056 (((g)>>2)<<5) | \
1057 ((b)>>3))); \
wdenk4b248f32004-03-14 16:51:43 +00001058 fb += 2; \
1059}
1060
1061#define FILL_32BIT_X888RGB(r,g,b) { \
Andre Przywara1d4ed262017-03-06 01:13:38 +00001062 *(u32 *)fb = \
1063 SWAP32((unsigned int)(((r<<16) | \
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001064 (g<<8) | \
1065 b))); \
wdenk4b248f32004-03-14 16:51:43 +00001066 fb += 4; \
1067}
1068
1069#ifdef VIDEO_FB_LITTLE_ENDIAN
1070#define FILL_24BIT_888RGB(r,g,b) { \
1071 fb[0] = b; \
1072 fb[1] = g; \
1073 fb[2] = r; \
1074 fb += 3; \
1075}
1076#else
1077#define FILL_24BIT_888RGB(r,g,b) { \
1078 fb[0] = r; \
1079 fb[1] = g; \
1080 fb[2] = b; \
1081 fb += 3; \
1082}
1083#endif
1084
Anatolij Gustschine84d5682008-08-08 18:00:40 +02001085#if defined(VIDEO_FB_16BPP_PIXEL_SWAP)
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001086static inline void fill_555rgb_pswap(uchar *fb, int x, u8 r, u8 g, u8 b)
Anatolij Gustschine84d5682008-08-08 18:00:40 +02001087{
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001088 ushort *dst = (ushort *) fb;
1089 ushort color = (ushort) (((r >> 3) << 10) |
1090 ((g >> 3) << 5) |
1091 (b >> 3));
Anatolij Gustschine84d5682008-08-08 18:00:40 +02001092 if (x & 1)
1093 *(--dst) = color;
1094 else
1095 *(++dst) = color;
1096}
1097#endif
wdenk4b248f32004-03-14 16:51:43 +00001098
1099/*
Anatolij Gustschind5011762010-03-15 14:50:25 +01001100 * RLE8 bitmap support
1101 */
1102
1103#ifdef CONFIG_VIDEO_BMP_RLE8
1104/* Pre-calculated color table entry */
1105struct palette {
1106 union {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001107 unsigned short w; /* word */
1108 unsigned int dw; /* double word */
1109 } ce; /* color entry */
Anatolij Gustschind5011762010-03-15 14:50:25 +01001110};
1111
1112/*
1113 * Helper to draw encoded/unencoded run.
1114 */
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001115static void draw_bitmap(uchar **fb, uchar *bm, struct palette *p,
1116 int cnt, int enc)
Anatolij Gustschind5011762010-03-15 14:50:25 +01001117{
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001118 ulong addr = (ulong) *fb;
Anatolij Gustschind5011762010-03-15 14:50:25 +01001119 int *off;
1120 int enc_off = 1;
1121 int i;
1122
1123 /*
1124 * Setup offset of the color index in the bitmap.
1125 * Color index of encoded run is at offset 1.
1126 */
1127 off = enc ? &enc_off : &i;
1128
1129 switch (VIDEO_DATA_FORMAT) {
1130 case GDF__8BIT_INDEX:
1131 for (i = 0; i < cnt; i++)
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001132 *(unsigned char *) addr++ = bm[*off];
Anatolij Gustschind5011762010-03-15 14:50:25 +01001133 break;
1134 case GDF_15BIT_555RGB:
1135 case GDF_16BIT_565RGB:
1136 /* differences handled while pre-calculating palette */
1137 for (i = 0; i < cnt; i++) {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001138 *(unsigned short *) addr = p[bm[*off]].ce.w;
Anatolij Gustschind5011762010-03-15 14:50:25 +01001139 addr += 2;
1140 }
1141 break;
1142 case GDF_32BIT_X888RGB:
1143 for (i = 0; i < cnt; i++) {
Andre Przywara1d4ed262017-03-06 01:13:38 +00001144 *(u32 *) addr = p[bm[*off]].ce.dw;
Anatolij Gustschind5011762010-03-15 14:50:25 +01001145 addr += 4;
1146 }
1147 break;
1148 }
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001149 *fb = (uchar *) addr; /* return modified address */
Anatolij Gustschind5011762010-03-15 14:50:25 +01001150}
1151
Simon Glass1c3dbe52015-05-13 07:02:27 -06001152static int display_rle8_bitmap(struct bmp_image *img, int xoff, int yoff,
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001153 int width, int height)
Anatolij Gustschind5011762010-03-15 14:50:25 +01001154{
1155 unsigned char *bm;
1156 unsigned char *fbp;
1157 unsigned int cnt, runlen;
1158 int decode = 1;
1159 int x, y, bpp, i, ncolors;
1160 struct palette p[256];
Simon Glass1c3dbe52015-05-13 07:02:27 -06001161 struct bmp_color_table_entry cte;
Anatolij Gustschind5011762010-03-15 14:50:25 +01001162 int green_shift, red_off;
Hans de Goedec67a8762015-08-04 12:15:39 +02001163 int limit = (VIDEO_LINE_LEN / VIDEO_PIXEL_SIZE) * VIDEO_ROWS;
Anatolij Gustschin74446b62011-02-21 21:33:29 +01001164 int pixels = 0;
Anatolij Gustschind5011762010-03-15 14:50:25 +01001165
1166 x = 0;
1167 y = __le32_to_cpu(img->header.height) - 1;
1168 ncolors = __le32_to_cpu(img->header.colors_used);
1169 bpp = VIDEO_PIXEL_SIZE;
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001170 fbp = (unsigned char *) ((unsigned int) video_fb_address +
Hans de Goedec67a8762015-08-04 12:15:39 +02001171 (y + yoff) * VIDEO_LINE_LEN +
1172 xoff * bpp);
Anatolij Gustschind5011762010-03-15 14:50:25 +01001173
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001174 bm = (uchar *) img + __le32_to_cpu(img->header.data_offset);
Anatolij Gustschind5011762010-03-15 14:50:25 +01001175
1176 /* pre-calculate and setup palette */
1177 switch (VIDEO_DATA_FORMAT) {
1178 case GDF__8BIT_INDEX:
1179 for (i = 0; i < ncolors; i++) {
1180 cte = img->color_table[i];
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001181 video_set_lut(i, cte.red, cte.green, cte.blue);
Anatolij Gustschind5011762010-03-15 14:50:25 +01001182 }
1183 break;
1184 case GDF_15BIT_555RGB:
1185 case GDF_16BIT_565RGB:
1186 if (VIDEO_DATA_FORMAT == GDF_15BIT_555RGB) {
1187 green_shift = 3;
1188 red_off = 10;
1189 } else {
1190 green_shift = 2;
1191 red_off = 11;
1192 }
1193 for (i = 0; i < ncolors; i++) {
1194 cte = img->color_table[i];
1195 p[i].ce.w = SWAP16((unsigned short)
1196 (((cte.red >> 3) << red_off) |
1197 ((cte.green >> green_shift) << 5) |
1198 cte.blue >> 3));
1199 }
1200 break;
1201 case GDF_32BIT_X888RGB:
1202 for (i = 0; i < ncolors; i++) {
1203 cte = img->color_table[i];
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001204 p[i].ce.dw = SWAP32((cte.red << 16) |
1205 (cte.green << 8) |
Anatolij Gustschind5011762010-03-15 14:50:25 +01001206 cte.blue);
1207 }
1208 break;
1209 default:
1210 printf("RLE Bitmap unsupported in video mode 0x%x\n",
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001211 VIDEO_DATA_FORMAT);
Anatolij Gustschind5011762010-03-15 14:50:25 +01001212 return -1;
1213 }
1214
1215 while (decode) {
1216 switch (bm[0]) {
1217 case 0:
1218 switch (bm[1]) {
1219 case 0:
1220 /* scan line end marker */
1221 bm += 2;
1222 x = 0;
1223 y--;
1224 fbp = (unsigned char *)
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001225 ((unsigned int) video_fb_address +
Hans de Goedec67a8762015-08-04 12:15:39 +02001226 (y + yoff) * VIDEO_LINE_LEN +
1227 xoff * bpp);
Anatolij Gustschind5011762010-03-15 14:50:25 +01001228 continue;
1229 case 1:
1230 /* end of bitmap data marker */
1231 decode = 0;
1232 break;
1233 case 2:
1234 /* run offset marker */
1235 x += bm[2];
1236 y -= bm[3];
1237 fbp = (unsigned char *)
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001238 ((unsigned int) video_fb_address +
Hans de Goedec67a8762015-08-04 12:15:39 +02001239 (y + yoff) * VIDEO_LINE_LEN +
1240 xoff * bpp);
Anatolij Gustschind5011762010-03-15 14:50:25 +01001241 bm += 4;
1242 break;
1243 default:
1244 /* unencoded run */
1245 cnt = bm[1];
1246 runlen = cnt;
Anatolij Gustschin74446b62011-02-21 21:33:29 +01001247 pixels += cnt;
1248 if (pixels > limit)
1249 goto error;
1250
Anatolij Gustschind5011762010-03-15 14:50:25 +01001251 bm += 2;
1252 if (y < height) {
1253 if (x >= width) {
1254 x += runlen;
1255 goto next_run;
1256 }
1257 if (x + runlen > width)
1258 cnt = width - x;
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001259 draw_bitmap(&fbp, bm, p, cnt, 0);
Anatolij Gustschind5011762010-03-15 14:50:25 +01001260 x += runlen;
1261 }
1262next_run:
1263 bm += runlen;
1264 if (runlen & 1)
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001265 bm++; /* 0 padding if length is odd */
Anatolij Gustschind5011762010-03-15 14:50:25 +01001266 }
1267 break;
1268 default:
1269 /* encoded run */
Anatolij Gustschin74446b62011-02-21 21:33:29 +01001270 cnt = bm[0];
1271 runlen = cnt;
1272 pixels += cnt;
1273 if (pixels > limit)
1274 goto error;
1275
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001276 if (y < height) { /* only draw into visible area */
Anatolij Gustschind5011762010-03-15 14:50:25 +01001277 if (x >= width) {
1278 x += runlen;
1279 bm += 2;
1280 continue;
1281 }
1282 if (x + runlen > width)
1283 cnt = width - x;
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001284 draw_bitmap(&fbp, bm, p, cnt, 1);
Anatolij Gustschind5011762010-03-15 14:50:25 +01001285 x += runlen;
1286 }
1287 bm += 2;
1288 break;
1289 }
1290 }
Sébastien Szymanski47b11c82018-09-10 09:58:58 +02001291
1292 if (cfb_do_flush_cache)
1293 flush_cache(VIDEO_FB_ADRS, VIDEO_SIZE);
1294
Anatolij Gustschind5011762010-03-15 14:50:25 +01001295 return 0;
Anatolij Gustschin74446b62011-02-21 21:33:29 +01001296error:
1297 printf("Error: Too much encoded pixel data, validate your bitmap\n");
1298 return -1;
Anatolij Gustschind5011762010-03-15 14:50:25 +01001299}
1300#endif
1301
1302/*
wdenk4b248f32004-03-14 16:51:43 +00001303 * Display the BMP file located at address bmp_image.
wdenk4b248f32004-03-14 16:51:43 +00001304 */
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001305int video_display_bitmap(ulong bmp_image, int x, int y)
wdenk4b248f32004-03-14 16:51:43 +00001306{
1307 ushort xcount, ycount;
1308 uchar *fb;
Simon Glass1c3dbe52015-05-13 07:02:27 -06001309 struct bmp_image *bmp = (struct bmp_image *)bmp_image;
wdenk4b248f32004-03-14 16:51:43 +00001310 uchar *bmap;
1311 ushort padded_line;
1312 unsigned long width, height, bpp;
1313 unsigned colors;
1314 unsigned long compression;
Simon Glass1c3dbe52015-05-13 07:02:27 -06001315 struct bmp_color_table_entry cte;
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001316
Stefan Roese98f4a3d2005-09-22 09:04:17 +02001317#ifdef CONFIG_VIDEO_BMP_GZIP
1318 unsigned char *dst = NULL;
1319 ulong len;
1320#endif
wdenk4b248f32004-03-14 16:51:43 +00001321
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001322 WATCHDOG_RESET();
wdenk4b248f32004-03-14 16:51:43 +00001323
1324 if (!((bmp->header.signature[0] == 'B') &&
1325 (bmp->header.signature[1] == 'M'))) {
Stefan Roese98f4a3d2005-09-22 09:04:17 +02001326
1327#ifdef CONFIG_VIDEO_BMP_GZIP
1328 /*
1329 * Could be a gzipped bmp image, try to decrompress...
1330 */
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001331 len = CONFIG_SYS_VIDEO_LOGO_MAX_SIZE;
1332 dst = malloc(CONFIG_SYS_VIDEO_LOGO_MAX_SIZE);
Stefan Roesec29ab9d2005-10-08 10:19:07 +02001333 if (dst == NULL) {
1334 printf("Error: malloc in gunzip failed!\n");
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001335 return 1;
Stefan Roesec29ab9d2005-10-08 10:19:07 +02001336 }
Eric Nelson5ca05c82014-03-08 07:55:52 -07001337 /*
1338 * NB: we need to force offset of +2
1339 * See doc/README.displaying-bmps
1340 */
1341 if (gunzip(dst+2, CONFIG_SYS_VIDEO_LOGO_MAX_SIZE-2,
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001342 (uchar *) bmp_image,
1343 &len) != 0) {
1344 printf("Error: no valid bmp or bmp.gz image at %lx\n",
1345 bmp_image);
Stefan Roese98f4a3d2005-09-22 09:04:17 +02001346 free(dst);
1347 return 1;
1348 }
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001349 if (len == CONFIG_SYS_VIDEO_LOGO_MAX_SIZE) {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001350 printf("Image could be truncated "
1351 "(increase CONFIG_SYS_VIDEO_LOGO_MAX_SIZE)!\n");
Stefan Roesec29ab9d2005-10-08 10:19:07 +02001352 }
Stefan Roese98f4a3d2005-09-22 09:04:17 +02001353
1354 /*
1355 * Set addr to decompressed image
1356 */
Simon Glass1c3dbe52015-05-13 07:02:27 -06001357 bmp = (struct bmp_image *)(dst+2);
Stefan Roese98f4a3d2005-09-22 09:04:17 +02001358
1359 if (!((bmp->header.signature[0] == 'B') &&
1360 (bmp->header.signature[1] == 'M'))) {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001361 printf("Error: no valid bmp.gz image at %lx\n",
1362 bmp_image);
Matthias Fuchsa49e0d12008-04-21 11:19:04 +02001363 free(dst);
Stefan Roese98f4a3d2005-09-22 09:04:17 +02001364 return 1;
1365 }
1366#else
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001367 printf("Error: no valid bmp image at %lx\n", bmp_image);
wdenk4b248f32004-03-14 16:51:43 +00001368 return 1;
Stefan Roese98f4a3d2005-09-22 09:04:17 +02001369#endif /* CONFIG_VIDEO_BMP_GZIP */
wdenk4b248f32004-03-14 16:51:43 +00001370 }
1371
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001372 width = le32_to_cpu(bmp->header.width);
1373 height = le32_to_cpu(bmp->header.height);
1374 bpp = le16_to_cpu(bmp->header.bit_count);
1375 colors = le32_to_cpu(bmp->header.colors_used);
1376 compression = le32_to_cpu(bmp->header.compression);
wdenk4b248f32004-03-14 16:51:43 +00001377
Marek Vasut68da5b12011-10-21 14:17:04 +00001378 debug("Display-bmp: %ld x %ld with %d colors\n",
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001379 width, height, colors);
wdenk4b248f32004-03-14 16:51:43 +00001380
Anatolij Gustschind5011762010-03-15 14:50:25 +01001381 if (compression != BMP_BI_RGB
1382#ifdef CONFIG_VIDEO_BMP_RLE8
1383 && compression != BMP_BI_RLE8
1384#endif
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001385 ) {
1386 printf("Error: compression type %ld not supported\n",
1387 compression);
Matthias Fuchsa49e0d12008-04-21 11:19:04 +02001388#ifdef CONFIG_VIDEO_BMP_GZIP
1389 if (dst)
1390 free(dst);
1391#endif
wdenk4b248f32004-03-14 16:51:43 +00001392 return 1;
1393 }
1394
1395 padded_line = (((width * bpp + 7) / 8) + 3) & ~0x3;
1396
Matthias Weisser1ca298c2009-07-09 16:07:30 +02001397#ifdef CONFIG_SPLASH_SCREEN_ALIGN
1398 if (x == BMP_ALIGN_CENTER)
Masahiro Yamadab4141192014-11-07 03:03:31 +09001399 x = max(0, (int)(VIDEO_VISIBLE_COLS - width) / 2);
Matthias Weisser1ca298c2009-07-09 16:07:30 +02001400 else if (x < 0)
Masahiro Yamadab4141192014-11-07 03:03:31 +09001401 x = max(0, (int)(VIDEO_VISIBLE_COLS - width + x + 1));
Matthias Weisser1ca298c2009-07-09 16:07:30 +02001402
1403 if (y == BMP_ALIGN_CENTER)
Masahiro Yamadab4141192014-11-07 03:03:31 +09001404 y = max(0, (int)(VIDEO_VISIBLE_ROWS - height) / 2);
Matthias Weisser1ca298c2009-07-09 16:07:30 +02001405 else if (y < 0)
Masahiro Yamadab4141192014-11-07 03:03:31 +09001406 y = max(0, (int)(VIDEO_VISIBLE_ROWS - height + y + 1));
Matthias Weisser1ca298c2009-07-09 16:07:30 +02001407#endif /* CONFIG_SPLASH_SCREEN_ALIGN */
1408
Matthias Weisseracf3baa2013-02-15 03:35:12 +00001409 /*
1410 * Just ignore elements which are completely beyond screen
1411 * dimensions.
1412 */
1413 if ((x >= VIDEO_VISIBLE_COLS) || (y >= VIDEO_VISIBLE_ROWS))
1414 return 0;
1415
wdenk4b248f32004-03-14 16:51:43 +00001416 if ((x + width) > VIDEO_VISIBLE_COLS)
1417 width = VIDEO_VISIBLE_COLS - x;
1418 if ((y + height) > VIDEO_VISIBLE_ROWS)
1419 height = VIDEO_VISIBLE_ROWS - y;
1420
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001421 bmap = (uchar *) bmp + le32_to_cpu(bmp->header.data_offset);
wdenk4b248f32004-03-14 16:51:43 +00001422 fb = (uchar *) (video_fb_address +
Hans de Goedec67a8762015-08-04 12:15:39 +02001423 ((y + height - 1) * VIDEO_LINE_LEN) +
wdenk4b248f32004-03-14 16:51:43 +00001424 x * VIDEO_PIXEL_SIZE);
1425
Anatolij Gustschind5011762010-03-15 14:50:25 +01001426#ifdef CONFIG_VIDEO_BMP_RLE8
1427 if (compression == BMP_BI_RLE8) {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001428 return display_rle8_bitmap(bmp, x, y, width, height);
Anatolij Gustschind5011762010-03-15 14:50:25 +01001429 }
1430#endif
1431
Timur Tabi68f66182010-08-23 16:58:00 -05001432 /* We handle only 4, 8, or 24 bpp bitmaps */
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001433 switch (le16_to_cpu(bmp->header.bit_count)) {
Timur Tabi68f66182010-08-23 16:58:00 -05001434 case 4:
1435 padded_line -= width / 2;
1436 ycount = height;
1437
1438 switch (VIDEO_DATA_FORMAT) {
1439 case GDF_32BIT_X888RGB:
1440 while (ycount--) {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001441 WATCHDOG_RESET();
Timur Tabi68f66182010-08-23 16:58:00 -05001442 /*
1443 * Don't assume that 'width' is an
1444 * even number
1445 */
1446 for (xcount = 0; xcount < width; xcount++) {
1447 uchar idx;
1448
1449 if (xcount & 1) {
1450 idx = *bmap & 0xF;
1451 bmap++;
1452 } else
1453 idx = *bmap >> 4;
1454 cte = bmp->color_table[idx];
1455 FILL_32BIT_X888RGB(cte.red, cte.green,
1456 cte.blue);
1457 }
1458 bmap += padded_line;
Hans de Goedec67a8762015-08-04 12:15:39 +02001459 fb -= VIDEO_LINE_LEN + width *
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001460 VIDEO_PIXEL_SIZE;
Timur Tabi68f66182010-08-23 16:58:00 -05001461 }
1462 break;
1463 default:
1464 puts("4bpp bitmap unsupported with current "
1465 "video mode\n");
1466 break;
1467 }
1468 break;
1469
wdenk4b248f32004-03-14 16:51:43 +00001470 case 8:
1471 padded_line -= width;
1472 if (VIDEO_DATA_FORMAT == GDF__8BIT_INDEX) {
Anatolij Gustschin7c050f82010-06-19 20:41:56 +02001473 /* Copy colormap */
wdenk4b248f32004-03-14 16:51:43 +00001474 for (xcount = 0; xcount < colors; ++xcount) {
1475 cte = bmp->color_table[xcount];
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001476 video_set_lut(xcount, cte.red, cte.green,
1477 cte.blue);
wdenk4b248f32004-03-14 16:51:43 +00001478 }
1479 }
1480 ycount = height;
1481 switch (VIDEO_DATA_FORMAT) {
1482 case GDF__8BIT_INDEX:
1483 while (ycount--) {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001484 WATCHDOG_RESET();
wdenk4b248f32004-03-14 16:51:43 +00001485 xcount = width;
1486 while (xcount--) {
1487 *fb++ = *bmap++;
1488 }
1489 bmap += padded_line;
Hans de Goedec67a8762015-08-04 12:15:39 +02001490 fb -= VIDEO_LINE_LEN + width *
1491 VIDEO_PIXEL_SIZE;
wdenk4b248f32004-03-14 16:51:43 +00001492 }
1493 break;
1494 case GDF__8BIT_332RGB:
1495 while (ycount--) {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001496 WATCHDOG_RESET();
wdenk4b248f32004-03-14 16:51:43 +00001497 xcount = width;
1498 while (xcount--) {
1499 cte = bmp->color_table[*bmap++];
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001500 FILL_8BIT_332RGB(cte.red, cte.green,
1501 cte.blue);
wdenk4b248f32004-03-14 16:51:43 +00001502 }
1503 bmap += padded_line;
Hans de Goedec67a8762015-08-04 12:15:39 +02001504 fb -= VIDEO_LINE_LEN + width *
1505 VIDEO_PIXEL_SIZE;
wdenk4b248f32004-03-14 16:51:43 +00001506 }
1507 break;
1508 case GDF_15BIT_555RGB:
1509 while (ycount--) {
Anatolij Gustschine84d5682008-08-08 18:00:40 +02001510#if defined(VIDEO_FB_16BPP_PIXEL_SWAP)
1511 int xpos = x;
1512#endif
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001513 WATCHDOG_RESET();
wdenk4b248f32004-03-14 16:51:43 +00001514 xcount = width;
1515 while (xcount--) {
1516 cte = bmp->color_table[*bmap++];
Andrew Dyercc347802008-08-29 12:30:39 -05001517#if defined(VIDEO_FB_16BPP_PIXEL_SWAP)
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001518 fill_555rgb_pswap(fb, xpos++, cte.red,
1519 cte.green,
1520 cte.blue);
Anatolij Gustschine84d5682008-08-08 18:00:40 +02001521 fb += 2;
Andrew Dyercc347802008-08-29 12:30:39 -05001522#else
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001523 FILL_15BIT_555RGB(cte.red, cte.green,
1524 cte.blue);
Anatolij Gustschine84d5682008-08-08 18:00:40 +02001525#endif
wdenk4b248f32004-03-14 16:51:43 +00001526 }
1527 bmap += padded_line;
Hans de Goedec67a8762015-08-04 12:15:39 +02001528 fb -= VIDEO_LINE_LEN + width *
1529 VIDEO_PIXEL_SIZE;
wdenk4b248f32004-03-14 16:51:43 +00001530 }
1531 break;
1532 case GDF_16BIT_565RGB:
1533 while (ycount--) {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001534 WATCHDOG_RESET();
wdenk4b248f32004-03-14 16:51:43 +00001535 xcount = width;
1536 while (xcount--) {
1537 cte = bmp->color_table[*bmap++];
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001538 FILL_16BIT_565RGB(cte.red, cte.green,
1539 cte.blue);
wdenk4b248f32004-03-14 16:51:43 +00001540 }
1541 bmap += padded_line;
Hans de Goedec67a8762015-08-04 12:15:39 +02001542 fb -= VIDEO_LINE_LEN + width *
1543 VIDEO_PIXEL_SIZE;
wdenk4b248f32004-03-14 16:51:43 +00001544 }
1545 break;
1546 case GDF_32BIT_X888RGB:
1547 while (ycount--) {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001548 WATCHDOG_RESET();
wdenk4b248f32004-03-14 16:51:43 +00001549 xcount = width;
1550 while (xcount--) {
1551 cte = bmp->color_table[*bmap++];
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001552 FILL_32BIT_X888RGB(cte.red, cte.green,
1553 cte.blue);
wdenk4b248f32004-03-14 16:51:43 +00001554 }
1555 bmap += padded_line;
Hans de Goedec67a8762015-08-04 12:15:39 +02001556 fb -= VIDEO_LINE_LEN + width *
1557 VIDEO_PIXEL_SIZE;
wdenk4b248f32004-03-14 16:51:43 +00001558 }
1559 break;
1560 case GDF_24BIT_888RGB:
1561 while (ycount--) {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001562 WATCHDOG_RESET();
wdenk4b248f32004-03-14 16:51:43 +00001563 xcount = width;
1564 while (xcount--) {
1565 cte = bmp->color_table[*bmap++];
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001566 FILL_24BIT_888RGB(cte.red, cte.green,
1567 cte.blue);
wdenk4b248f32004-03-14 16:51:43 +00001568 }
1569 bmap += padded_line;
Hans de Goedec67a8762015-08-04 12:15:39 +02001570 fb -= VIDEO_LINE_LEN + width *
1571 VIDEO_PIXEL_SIZE;
wdenk4b248f32004-03-14 16:51:43 +00001572 }
1573 break;
1574 }
1575 break;
1576 case 24:
1577 padded_line -= 3 * width;
1578 ycount = height;
1579 switch (VIDEO_DATA_FORMAT) {
1580 case GDF__8BIT_332RGB:
1581 while (ycount--) {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001582 WATCHDOG_RESET();
wdenk4b248f32004-03-14 16:51:43 +00001583 xcount = width;
1584 while (xcount--) {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001585 FILL_8BIT_332RGB(bmap[2], bmap[1],
1586 bmap[0]);
wdenk4b248f32004-03-14 16:51:43 +00001587 bmap += 3;
1588 }
1589 bmap += padded_line;
Hans de Goedec67a8762015-08-04 12:15:39 +02001590 fb -= VIDEO_LINE_LEN + width *
1591 VIDEO_PIXEL_SIZE;
wdenk4b248f32004-03-14 16:51:43 +00001592 }
1593 break;
1594 case GDF_15BIT_555RGB:
1595 while (ycount--) {
Anatolij Gustschine84d5682008-08-08 18:00:40 +02001596#if defined(VIDEO_FB_16BPP_PIXEL_SWAP)
1597 int xpos = x;
1598#endif
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001599 WATCHDOG_RESET();
wdenk4b248f32004-03-14 16:51:43 +00001600 xcount = width;
1601 while (xcount--) {
Andrew Dyercc347802008-08-29 12:30:39 -05001602#if defined(VIDEO_FB_16BPP_PIXEL_SWAP)
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001603 fill_555rgb_pswap(fb, xpos++, bmap[2],
1604 bmap[1], bmap[0]);
Anatolij Gustschine84d5682008-08-08 18:00:40 +02001605 fb += 2;
Andrew Dyercc347802008-08-29 12:30:39 -05001606#else
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001607 FILL_15BIT_555RGB(bmap[2], bmap[1],
1608 bmap[0]);
Anatolij Gustschine84d5682008-08-08 18:00:40 +02001609#endif
wdenk4b248f32004-03-14 16:51:43 +00001610 bmap += 3;
1611 }
1612 bmap += padded_line;
Hans de Goedec67a8762015-08-04 12:15:39 +02001613 fb -= VIDEO_LINE_LEN + width *
1614 VIDEO_PIXEL_SIZE;
wdenk4b248f32004-03-14 16:51:43 +00001615 }
1616 break;
1617 case GDF_16BIT_565RGB:
1618 while (ycount--) {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001619 WATCHDOG_RESET();
wdenk4b248f32004-03-14 16:51:43 +00001620 xcount = width;
1621 while (xcount--) {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001622 FILL_16BIT_565RGB(bmap[2], bmap[1],
1623 bmap[0]);
wdenk4b248f32004-03-14 16:51:43 +00001624 bmap += 3;
1625 }
1626 bmap += padded_line;
Hans de Goedec67a8762015-08-04 12:15:39 +02001627 fb -= VIDEO_LINE_LEN + width *
1628 VIDEO_PIXEL_SIZE;
wdenk4b248f32004-03-14 16:51:43 +00001629 }
1630 break;
1631 case GDF_32BIT_X888RGB:
1632 while (ycount--) {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001633 WATCHDOG_RESET();
wdenk4b248f32004-03-14 16:51:43 +00001634 xcount = width;
1635 while (xcount--) {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001636 FILL_32BIT_X888RGB(bmap[2], bmap[1],
1637 bmap[0]);
wdenk4b248f32004-03-14 16:51:43 +00001638 bmap += 3;
1639 }
1640 bmap += padded_line;
Hans de Goedec67a8762015-08-04 12:15:39 +02001641 fb -= VIDEO_LINE_LEN + width *
1642 VIDEO_PIXEL_SIZE;
wdenk4b248f32004-03-14 16:51:43 +00001643 }
1644 break;
1645 case GDF_24BIT_888RGB:
1646 while (ycount--) {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001647 WATCHDOG_RESET();
wdenk4b248f32004-03-14 16:51:43 +00001648 xcount = width;
1649 while (xcount--) {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001650 FILL_24BIT_888RGB(bmap[2], bmap[1],
1651 bmap[0]);
wdenk4b248f32004-03-14 16:51:43 +00001652 bmap += 3;
1653 }
1654 bmap += padded_line;
Hans de Goedec67a8762015-08-04 12:15:39 +02001655 fb -= VIDEO_LINE_LEN + width *
1656 VIDEO_PIXEL_SIZE;
wdenk4b248f32004-03-14 16:51:43 +00001657 }
1658 break;
1659 default:
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001660 printf("Error: 24 bits/pixel bitmap incompatible "
1661 "with current video mode\n");
wdenk4b248f32004-03-14 16:51:43 +00001662 break;
1663 }
1664 break;
1665 default:
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001666 printf("Error: %d bit/pixel bitmaps not supported by U-Boot\n",
1667 le16_to_cpu(bmp->header.bit_count));
wdenk4b248f32004-03-14 16:51:43 +00001668 break;
1669 }
Stefan Roese98f4a3d2005-09-22 09:04:17 +02001670
1671#ifdef CONFIG_VIDEO_BMP_GZIP
1672 if (dst) {
1673 free(dst);
1674 }
1675#endif
1676
Eric Nelsondb0d47d2013-05-06 14:27:28 +02001677 if (cfb_do_flush_cache)
1678 flush_cache(VIDEO_FB_ADRS, VIDEO_SIZE);
wdenk4b248f32004-03-14 16:51:43 +00001679 return (0);
1680}
Jon Loeliger07d38a12007-07-09 17:30:01 -05001681#endif
wdenk4b248f32004-03-14 16:51:43 +00001682
wdenk4b248f32004-03-14 16:51:43 +00001683
wdenkc6097192002-11-03 00:24:07 +00001684#ifdef CONFIG_VIDEO_LOGO
Bastian Ruppert1e681f92012-09-13 22:29:01 +00001685static int video_logo_xpos;
1686static int video_logo_ypos;
1687
Hans de Goedec4c9e812015-08-04 12:21:40 +02001688static void plot_logo_or_black(void *screen, int x, int y, int black);
Bastian Ruppert4b7d3a02012-09-13 22:29:02 +00001689
Hans de Goedec4c9e812015-08-04 12:21:40 +02001690static void logo_plot(void *screen, int x, int y)
Bastian Ruppert4b7d3a02012-09-13 22:29:02 +00001691{
Hans de Goedec4c9e812015-08-04 12:21:40 +02001692 plot_logo_or_black(screen, x, y, 0);
Bastian Ruppert4b7d3a02012-09-13 22:29:02 +00001693}
1694
1695static void logo_black(void)
1696{
Hans de Goedec4c9e812015-08-04 12:21:40 +02001697 plot_logo_or_black(video_fb_address, video_logo_xpos, video_logo_ypos,
Bastian Ruppert4b7d3a02012-09-13 22:29:02 +00001698 1);
1699}
1700
Simon Glass09140112020-05-10 11:40:03 -06001701static int do_clrlogo(struct cmd_tbl *cmdtp, int flag, int argc,
1702 char *const argv[])
Bastian Ruppert4b7d3a02012-09-13 22:29:02 +00001703{
1704 if (argc != 1)
1705 return cmd_usage(cmdtp);
1706
1707 logo_black();
1708 return 0;
1709}
1710
1711U_BOOT_CMD(
1712 clrlogo, 1, 0, do_clrlogo,
1713 "fill the boot logo area with black",
1714 " "
1715 );
1716
Hans de Goedec4c9e812015-08-04 12:21:40 +02001717static void plot_logo_or_black(void *screen, int x, int y, int black)
wdenkc6097192002-11-03 00:24:07 +00001718{
wdenkc6097192002-11-03 00:24:07 +00001719
wdenk4b248f32004-03-14 16:51:43 +00001720 int xcount, i;
Hans de Goedec67a8762015-08-04 12:15:39 +02001721 int skip = VIDEO_LINE_LEN - VIDEO_LOGO_WIDTH * VIDEO_PIXEL_SIZE;
Matthias Weisserbe129aa2010-01-12 12:06:31 +01001722 int ycount = video_logo_height;
wdenk4b248f32004-03-14 16:51:43 +00001723 unsigned char r, g, b, *logo_red, *logo_blue, *logo_green;
1724 unsigned char *source;
Bastian Ruppert1e681f92012-09-13 22:29:01 +00001725 unsigned char *dest;
1726
1727#ifdef CONFIG_SPLASH_SCREEN_ALIGN
1728 if (x == BMP_ALIGN_CENTER)
Masahiro Yamadab4141192014-11-07 03:03:31 +09001729 x = max(0, (int)(VIDEO_VISIBLE_COLS - VIDEO_LOGO_WIDTH) / 2);
Bastian Ruppert1e681f92012-09-13 22:29:01 +00001730 else if (x < 0)
Masahiro Yamadab4141192014-11-07 03:03:31 +09001731 x = max(0, (int)(VIDEO_VISIBLE_COLS - VIDEO_LOGO_WIDTH + x + 1));
Bastian Ruppert1e681f92012-09-13 22:29:01 +00001732
1733 if (y == BMP_ALIGN_CENTER)
Masahiro Yamadab4141192014-11-07 03:03:31 +09001734 y = max(0, (int)(VIDEO_VISIBLE_ROWS - VIDEO_LOGO_HEIGHT) / 2);
Bastian Ruppert1e681f92012-09-13 22:29:01 +00001735 else if (y < 0)
Masahiro Yamadab4141192014-11-07 03:03:31 +09001736 y = max(0, (int)(VIDEO_VISIBLE_ROWS - VIDEO_LOGO_HEIGHT + y + 1));
Bastian Ruppert1e681f92012-09-13 22:29:01 +00001737#endif /* CONFIG_SPLASH_SCREEN_ALIGN */
1738
Hans de Goedec67a8762015-08-04 12:15:39 +02001739 dest = (unsigned char *)screen + y * VIDEO_LINE_LEN + x * VIDEO_PIXEL_SIZE;
wdenka6c7ad22002-12-03 21:28:10 +00001740
1741#ifdef CONFIG_VIDEO_BMP_LOGO
wdenk4b248f32004-03-14 16:51:43 +00001742 source = bmp_logo_bitmap;
wdenk8bde7f72003-06-27 21:31:46 +00001743
Anatolij Gustschin7c050f82010-06-19 20:41:56 +02001744 /* Allocate temporary space for computing colormap */
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001745 logo_red = malloc(BMP_LOGO_COLORS);
1746 logo_green = malloc(BMP_LOGO_COLORS);
1747 logo_blue = malloc(BMP_LOGO_COLORS);
Anatolij Gustschin7c050f82010-06-19 20:41:56 +02001748 /* Compute color map */
wdenk4b248f32004-03-14 16:51:43 +00001749 for (i = 0; i < VIDEO_LOGO_COLORS; i++) {
1750 logo_red[i] = (bmp_logo_palette[i] & 0x0f00) >> 4;
1751 logo_green[i] = (bmp_logo_palette[i] & 0x00f0);
1752 logo_blue[i] = (bmp_logo_palette[i] & 0x000f) << 4;
1753 }
wdenka6c7ad22002-12-03 21:28:10 +00001754#else
wdenk4b248f32004-03-14 16:51:43 +00001755 source = linux_logo;
1756 logo_red = linux_logo_red;
1757 logo_green = linux_logo_green;
1758 logo_blue = linux_logo_blue;
wdenka6c7ad22002-12-03 21:28:10 +00001759#endif
wdenk8bde7f72003-06-27 21:31:46 +00001760
wdenk4b248f32004-03-14 16:51:43 +00001761 if (VIDEO_DATA_FORMAT == GDF__8BIT_INDEX) {
1762 for (i = 0; i < VIDEO_LOGO_COLORS; i++) {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001763 video_set_lut(i + VIDEO_LOGO_LUT_OFFSET,
1764 logo_red[i], logo_green[i],
1765 logo_blue[i]);
wdenk4b248f32004-03-14 16:51:43 +00001766 }
wdenk8bde7f72003-06-27 21:31:46 +00001767 }
wdenkc6097192002-11-03 00:24:07 +00001768
wdenk4b248f32004-03-14 16:51:43 +00001769 while (ycount--) {
Anatolij Gustschine84d5682008-08-08 18:00:40 +02001770#if defined(VIDEO_FB_16BPP_PIXEL_SWAP)
1771 int xpos = x;
1772#endif
wdenk4b248f32004-03-14 16:51:43 +00001773 xcount = VIDEO_LOGO_WIDTH;
1774 while (xcount--) {
Bastian Ruppert4b7d3a02012-09-13 22:29:02 +00001775 if (black) {
1776 r = 0x00;
1777 g = 0x00;
1778 b = 0x00;
1779 } else {
1780 r = logo_red[*source - VIDEO_LOGO_LUT_OFFSET];
1781 g = logo_green[*source - VIDEO_LOGO_LUT_OFFSET];
1782 b = logo_blue[*source - VIDEO_LOGO_LUT_OFFSET];
1783 }
wdenk8bde7f72003-06-27 21:31:46 +00001784
wdenk4b248f32004-03-14 16:51:43 +00001785 switch (VIDEO_DATA_FORMAT) {
1786 case GDF__8BIT_INDEX:
1787 *dest = *source;
1788 break;
1789 case GDF__8BIT_332RGB:
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001790 *dest = ((r >> 5) << 5) |
1791 ((g >> 5) << 2) |
1792 (b >> 6);
wdenk4b248f32004-03-14 16:51:43 +00001793 break;
1794 case GDF_15BIT_555RGB:
Andrew Dyercc347802008-08-29 12:30:39 -05001795#if defined(VIDEO_FB_16BPP_PIXEL_SWAP)
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001796 fill_555rgb_pswap(dest, xpos++, r, g, b);
Andrew Dyercc347802008-08-29 12:30:39 -05001797#else
wdenk4b248f32004-03-14 16:51:43 +00001798 *(unsigned short *) dest =
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001799 SWAP16((unsigned short) (
1800 ((r >> 3) << 10) |
1801 ((g >> 3) << 5) |
1802 (b >> 3)));
Anatolij Gustschinbed53752008-01-11 14:30:01 +01001803#endif
wdenk4b248f32004-03-14 16:51:43 +00001804 break;
1805 case GDF_16BIT_565RGB:
1806 *(unsigned short *) dest =
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001807 SWAP16((unsigned short) (
1808 ((r >> 3) << 11) |
1809 ((g >> 2) << 5) |
1810 (b >> 3)));
wdenk4b248f32004-03-14 16:51:43 +00001811 break;
1812 case GDF_32BIT_X888RGB:
Andre Przywara1d4ed262017-03-06 01:13:38 +00001813 *(u32 *) dest =
1814 SWAP32((u32) (
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001815 (r << 16) |
1816 (g << 8) |
1817 b));
wdenk4b248f32004-03-14 16:51:43 +00001818 break;
1819 case GDF_24BIT_888RGB:
wdenkc6097192002-11-03 00:24:07 +00001820#ifdef VIDEO_FB_LITTLE_ENDIAN
wdenk4b248f32004-03-14 16:51:43 +00001821 dest[0] = b;
1822 dest[1] = g;
1823 dest[2] = r;
wdenkc6097192002-11-03 00:24:07 +00001824#else
wdenk4b248f32004-03-14 16:51:43 +00001825 dest[0] = r;
1826 dest[1] = g;
1827 dest[2] = b;
wdenkc6097192002-11-03 00:24:07 +00001828#endif
wdenk4b248f32004-03-14 16:51:43 +00001829 break;
1830 }
1831 source++;
1832 dest += VIDEO_PIXEL_SIZE;
1833 }
1834 dest += skip;
wdenk8bde7f72003-06-27 21:31:46 +00001835 }
wdenka6c7ad22002-12-03 21:28:10 +00001836#ifdef CONFIG_VIDEO_BMP_LOGO
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001837 free(logo_red);
1838 free(logo_green);
1839 free(logo_blue);
wdenka6c7ad22002-12-03 21:28:10 +00001840#endif
wdenkc6097192002-11-03 00:24:07 +00001841}
1842
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001843static void *video_logo(void)
wdenkc6097192002-11-03 00:24:07 +00001844{
wdenk4b248f32004-03-14 16:51:43 +00001845 char info[128];
Wolfgang Denka9a62af2011-11-04 15:55:20 +00001846 __maybe_unused int y_off = 0;
Bastian Ruppert1e681f92012-09-13 22:29:01 +00001847 __maybe_unused ulong addr;
1848 __maybe_unused char *s;
Anatolij Gustschinb4fc6f22016-11-30 14:30:59 +01001849 __maybe_unused int len, ret, space;
wdenkc6097192002-11-03 00:24:07 +00001850
Anatolij Gustschinff8fb562013-07-02 00:04:05 +02001851 splash_get_pos(&video_logo_xpos, &video_logo_ypos);
Matthias Weisser1ca298c2009-07-09 16:07:30 +02001852
Bastian Ruppert1e681f92012-09-13 22:29:01 +00001853#ifdef CONFIG_SPLASH_SCREEN
Simon Glass00caae62017-08-03 12:22:12 -06001854 s = env_get("splashimage");
Bastian Ruppert1e681f92012-09-13 22:29:01 +00001855 if (s != NULL) {
Anatolij Gustschinb4fc6f22016-11-30 14:30:59 +01001856 ret = splash_screen_prepare();
1857 if (ret < 0)
1858 return video_fb_address;
Simon Glass7e5f4602021-07-24 09:03:29 -06001859 addr = hextoul(s, NULL);
Bastian Ruppert1e681f92012-09-13 22:29:01 +00001860
Bastian Ruppert1e681f92012-09-13 22:29:01 +00001861 if (video_display_bitmap(addr,
1862 video_logo_xpos,
1863 video_logo_ypos) == 0) {
Matthias Weisserbe129aa2010-01-12 12:06:31 +01001864 video_logo_height = 0;
wdenk4b248f32004-03-14 16:51:43 +00001865 return ((void *) (video_fb_address));
1866 }
1867 }
1868#endif /* CONFIG_SPLASH_SCREEN */
1869
Hans de Goedec4c9e812015-08-04 12:21:40 +02001870 logo_plot(video_fb_address, video_logo_xpos, video_logo_ypos);
Bastian Ruppert1e681f92012-09-13 22:29:01 +00001871
1872#ifdef CONFIG_SPLASH_SCREEN_ALIGN
1873 /*
1874 * when using splashpos for video_logo, skip any info
1875 * output on video console if the logo is not at 0,0
1876 */
1877 if (video_logo_xpos || video_logo_ypos) {
1878 /*
1879 * video_logo_height is used in text and cursor offset
1880 * calculations. Since the console is below the logo,
1881 * we need to adjust the logo height
1882 */
1883 if (video_logo_ypos == BMP_ALIGN_CENTER)
Masahiro Yamadab4141192014-11-07 03:03:31 +09001884 video_logo_height += max(0, (int)(VIDEO_VISIBLE_ROWS -
Bastian Ruppert1e681f92012-09-13 22:29:01 +00001885 VIDEO_LOGO_HEIGHT) / 2);
1886 else if (video_logo_ypos > 0)
1887 video_logo_height += video_logo_ypos;
1888
1889 return video_fb_address + video_logo_height * VIDEO_LINE_LEN;
1890 }
1891#endif
Heiko Schocher2bc4aa52013-08-03 07:22:53 +02001892 if (board_cfb_skip())
1893 return 0;
wdenk4b248f32004-03-14 16:51:43 +00001894
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001895 sprintf(info, " %s", version_string);
Anatolij Gustschin3dcbe622009-04-23 12:35:22 +02001896
Tim Harveyadde4352016-05-24 14:59:59 -07001897#ifndef CONFIG_HIDE_LOGO_VERSION
Peng Fanf9d891f2018-01-02 15:25:37 +08001898 space = (VIDEO_COLS - VIDEO_INFO_X) / VIDEO_FONT_WIDTH;
Anatolij Gustschin3dcbe622009-04-23 12:35:22 +02001899 len = strlen(info);
1900
1901 if (len > space) {
Peng Fanf9d891f2018-01-02 15:25:37 +08001902 int xx = VIDEO_INFO_X, yy = VIDEO_INFO_Y;
1903 uchar *p = (uchar *) info;
1904
1905 while (len) {
1906 if (len > space) {
1907 video_drawchars(xx, yy, p, space);
1908 len -= space;
1909
1910 p = (uchar *)p + space;
1911
1912 if (!y_off) {
1913 xx += VIDEO_FONT_WIDTH;
1914 space--;
1915 }
1916 yy += VIDEO_FONT_HEIGHT;
1917
1918 y_off++;
1919 } else {
1920 video_drawchars(xx, yy, p, len);
1921 len = 0;
1922 }
1923 }
Anatolij Gustschin3dcbe622009-04-23 12:35:22 +02001924 } else
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001925 video_drawstring(VIDEO_INFO_X, VIDEO_INFO_Y, (uchar *) info);
wdenkc6097192002-11-03 00:24:07 +00001926
1927#ifdef CONFIG_CONSOLE_EXTRA_INFO
wdenk4b248f32004-03-14 16:51:43 +00001928 {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001929 int i, n =
1930 ((video_logo_height -
1931 VIDEO_FONT_HEIGHT) / VIDEO_FONT_HEIGHT);
wdenkc6097192002-11-03 00:24:07 +00001932
wdenk4b248f32004-03-14 16:51:43 +00001933 for (i = 1; i < n; i++) {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001934 video_get_info_str(i, info);
Anatolij Gustschin3dcbe622009-04-23 12:35:22 +02001935 if (!*info)
1936 continue;
1937
1938 len = strlen(info);
1939 if (len > space) {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001940 video_drawchars(VIDEO_INFO_X,
1941 VIDEO_INFO_Y +
1942 (i + y_off) *
1943 VIDEO_FONT_HEIGHT,
1944 (uchar *) info, space);
Anatolij Gustschin3dcbe622009-04-23 12:35:22 +02001945 y_off++;
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001946 video_drawchars(VIDEO_INFO_X +
1947 VIDEO_FONT_WIDTH,
1948 VIDEO_INFO_Y +
1949 (i + y_off) *
1950 VIDEO_FONT_HEIGHT,
1951 (uchar *) info + space,
1952 len - space);
Anatolij Gustschin3dcbe622009-04-23 12:35:22 +02001953 } else {
Wolfgang Denk64e40d72011-07-29 09:55:27 +00001954 video_drawstring(VIDEO_INFO_X,
1955 VIDEO_INFO_Y +
1956 (i + y_off) *
1957 VIDEO_FONT_HEIGHT,
1958 (uchar *) info);
Anatolij Gustschin3dcbe622009-04-23 12:35:22 +02001959 }
wdenk4b248f32004-03-14 16:51:43 +00001960 }
1961 }
wdenkc6097192002-11-03 00:24:07 +00001962#endif
Tim Harveyadde4352016-05-24 14:59:59 -07001963#endif
wdenkc6097192002-11-03 00:24:07 +00001964
Matthias Weisserbe129aa2010-01-12 12:06:31 +01001965 return (video_fb_address + video_logo_height * VIDEO_LINE_LEN);
wdenkc6097192002-11-03 00:24:07 +00001966}
1967#endif
1968
Anatolij Gustschinbfd4be82012-06-05 09:19:18 +02001969static int cfb_fb_is_in_dram(void)
1970{
Masahiro Yamadab75d8dc2020-06-26 15:13:33 +09001971 struct bd_info *bd = gd->bd;
Anatolij Gustschinbfd4be82012-06-05 09:19:18 +02001972 ulong start, end;
1973 int i;
1974
1975 for (i = 0; i < CONFIG_NR_DRAM_BANKS; ++i) {
1976 start = bd->bi_dram[i].start;
1977 end = bd->bi_dram[i].start + bd->bi_dram[i].size - 1;
1978 if ((ulong)video_fb_address >= start &&
1979 (ulong)video_fb_address < end)
1980 return 1;
1981 }
Stefan Roese063d5472020-08-12 13:32:41 +02001982
Anatolij Gustschinbfd4be82012-06-05 09:19:18 +02001983 return 0;
1984}
1985
Heiko Schocher45ae2542013-10-22 11:06:06 +02001986void video_clear(void)
1987{
1988 if (!video_fb_address)
1989 return;
1990#ifdef VIDEO_HW_RECTFILL
1991 video_hw_rectfill(VIDEO_PIXEL_SIZE, /* bytes per pixel */
1992 0, /* dest pos x */
1993 0, /* dest pos y */
1994 VIDEO_VISIBLE_COLS, /* frame width */
1995 VIDEO_VISIBLE_ROWS, /* frame height */
1996 bgx /* fill color */
1997 );
1998#else
1999 memsetl(video_fb_address,
2000 (VIDEO_VISIBLE_ROWS * VIDEO_LINE_LEN) / sizeof(int), bgx);
2001#endif
2002}
2003
Simon Glass0a6eac82016-10-17 20:12:54 -06002004static int cfg_video_init(void)
wdenkc6097192002-11-03 00:24:07 +00002005{
wdenk4b248f32004-03-14 16:51:43 +00002006 unsigned char color8;
wdenkc6097192002-11-03 00:24:07 +00002007
Wolfgang Denk57912932011-07-30 12:48:09 +00002008 pGD = video_hw_init();
2009 if (pGD == NULL)
wdenk4b248f32004-03-14 16:51:43 +00002010 return -1;
wdenkc6097192002-11-03 00:24:07 +00002011
wdenk4b248f32004-03-14 16:51:43 +00002012 video_fb_address = (void *) VIDEO_FB_ADRS;
wdenkc6097192002-11-03 00:24:07 +00002013
Anatolij Gustschinbfd4be82012-06-05 09:19:18 +02002014 cfb_do_flush_cache = cfb_fb_is_in_dram() && dcache_status();
2015
wdenk4b248f32004-03-14 16:51:43 +00002016 /* Init drawing pats */
2017 switch (VIDEO_DATA_FORMAT) {
2018 case GDF__8BIT_INDEX:
Simon Glass002f9672016-10-17 20:12:44 -06002019 video_set_lut(0x01, CONFIG_SYS_CONSOLE_FG_COL,
2020 CONFIG_SYS_CONSOLE_FG_COL,
2021 CONFIG_SYS_CONSOLE_FG_COL);
2022 video_set_lut(0x00, CONFIG_SYS_CONSOLE_BG_COL,
2023 CONFIG_SYS_CONSOLE_BG_COL,
2024 CONFIG_SYS_CONSOLE_BG_COL);
wdenk4b248f32004-03-14 16:51:43 +00002025 fgx = 0x01010101;
2026 bgx = 0x00000000;
2027 break;
2028 case GDF__8BIT_332RGB:
Simon Glass002f9672016-10-17 20:12:44 -06002029 color8 = ((CONFIG_SYS_CONSOLE_FG_COL & 0xe0) |
2030 ((CONFIG_SYS_CONSOLE_FG_COL >> 3) & 0x1c) |
2031 CONFIG_SYS_CONSOLE_FG_COL >> 6);
Wolfgang Denk64e40d72011-07-29 09:55:27 +00002032 fgx = (color8 << 24) | (color8 << 16) | (color8 << 8) |
2033 color8;
Simon Glass002f9672016-10-17 20:12:44 -06002034 color8 = ((CONFIG_SYS_CONSOLE_BG_COL & 0xe0) |
2035 ((CONFIG_SYS_CONSOLE_BG_COL >> 3) & 0x1c) |
2036 CONFIG_SYS_CONSOLE_BG_COL >> 6);
Wolfgang Denk64e40d72011-07-29 09:55:27 +00002037 bgx = (color8 << 24) | (color8 << 16) | (color8 << 8) |
2038 color8;
wdenk4b248f32004-03-14 16:51:43 +00002039 break;
2040 case GDF_15BIT_555RGB:
Simon Glass002f9672016-10-17 20:12:44 -06002041 fgx = (((CONFIG_SYS_CONSOLE_FG_COL >> 3) << 26) |
2042 ((CONFIG_SYS_CONSOLE_FG_COL >> 3) << 21) |
2043 ((CONFIG_SYS_CONSOLE_FG_COL >> 3) << 16) |
2044 ((CONFIG_SYS_CONSOLE_FG_COL >> 3) << 10) |
2045 ((CONFIG_SYS_CONSOLE_FG_COL >> 3) << 5) |
2046 (CONFIG_SYS_CONSOLE_FG_COL >> 3));
2047 bgx = (((CONFIG_SYS_CONSOLE_BG_COL >> 3) << 26) |
2048 ((CONFIG_SYS_CONSOLE_BG_COL >> 3) << 21) |
2049 ((CONFIG_SYS_CONSOLE_BG_COL >> 3) << 16) |
2050 ((CONFIG_SYS_CONSOLE_BG_COL >> 3) << 10) |
2051 ((CONFIG_SYS_CONSOLE_BG_COL >> 3) << 5) |
2052 (CONFIG_SYS_CONSOLE_BG_COL >> 3));
wdenk4b248f32004-03-14 16:51:43 +00002053 break;
2054 case GDF_16BIT_565RGB:
Simon Glass002f9672016-10-17 20:12:44 -06002055 fgx = (((CONFIG_SYS_CONSOLE_FG_COL >> 3) << 27) |
2056 ((CONFIG_SYS_CONSOLE_FG_COL >> 2) << 21) |
2057 ((CONFIG_SYS_CONSOLE_FG_COL >> 3) << 16) |
2058 ((CONFIG_SYS_CONSOLE_FG_COL >> 3) << 11) |
2059 ((CONFIG_SYS_CONSOLE_FG_COL >> 2) << 5) |
2060 (CONFIG_SYS_CONSOLE_FG_COL >> 3));
2061 bgx = (((CONFIG_SYS_CONSOLE_BG_COL >> 3) << 27) |
2062 ((CONFIG_SYS_CONSOLE_BG_COL >> 2) << 21) |
2063 ((CONFIG_SYS_CONSOLE_BG_COL >> 3) << 16) |
2064 ((CONFIG_SYS_CONSOLE_BG_COL >> 3) << 11) |
2065 ((CONFIG_SYS_CONSOLE_BG_COL >> 2) << 5) |
2066 (CONFIG_SYS_CONSOLE_BG_COL >> 3));
wdenk4b248f32004-03-14 16:51:43 +00002067 break;
2068 case GDF_32BIT_X888RGB:
Simon Glass002f9672016-10-17 20:12:44 -06002069 fgx = (CONFIG_SYS_CONSOLE_FG_COL << 16) |
2070 (CONFIG_SYS_CONSOLE_FG_COL << 8) |
2071 CONFIG_SYS_CONSOLE_FG_COL;
2072 bgx = (CONFIG_SYS_CONSOLE_BG_COL << 16) |
2073 (CONFIG_SYS_CONSOLE_BG_COL << 8) |
2074 CONFIG_SYS_CONSOLE_BG_COL;
wdenk4b248f32004-03-14 16:51:43 +00002075 break;
2076 case GDF_24BIT_888RGB:
Simon Glass002f9672016-10-17 20:12:44 -06002077 fgx = (CONFIG_SYS_CONSOLE_FG_COL << 24) |
2078 (CONFIG_SYS_CONSOLE_FG_COL << 16) |
2079 (CONFIG_SYS_CONSOLE_FG_COL << 8) |
2080 CONFIG_SYS_CONSOLE_FG_COL;
2081 bgx = (CONFIG_SYS_CONSOLE_BG_COL << 24) |
2082 (CONFIG_SYS_CONSOLE_BG_COL << 16) |
2083 (CONFIG_SYS_CONSOLE_BG_COL << 8) |
2084 CONFIG_SYS_CONSOLE_BG_COL;
wdenk4b248f32004-03-14 16:51:43 +00002085 break;
2086 }
2087 eorx = fgx ^ bgx;
wdenkc6097192002-11-03 00:24:07 +00002088
Rob Clark8ef05352017-08-03 12:47:01 -04002089 if (!CONFIG_IS_ENABLED(NO_FB_CLEAR))
2090 video_clear();
Heiko Schocher45ae2542013-10-22 11:06:06 +02002091
wdenkc6097192002-11-03 00:24:07 +00002092#ifdef CONFIG_VIDEO_LOGO
wdenk4b248f32004-03-14 16:51:43 +00002093 /* Plot the logo and get start point of console */
Wolfgang Denk72c65f62011-07-29 09:55:28 +00002094 debug("Video: Drawing the logo ...\n");
Wolfgang Denk64e40d72011-07-29 09:55:27 +00002095 video_console_address = video_logo();
wdenkc6097192002-11-03 00:24:07 +00002096#else
wdenk4b248f32004-03-14 16:51:43 +00002097 video_console_address = video_fb_address;
wdenkc6097192002-11-03 00:24:07 +00002098#endif
2099
wdenk4b248f32004-03-14 16:51:43 +00002100 /* Initialize the console */
2101 console_col = 0;
2102 console_row = 0;
wdenkc6097192002-11-03 00:24:07 +00002103
Eric Nelsondb0d47d2013-05-06 14:27:28 +02002104 if (cfb_do_flush_cache)
2105 flush_cache(VIDEO_FB_ADRS, VIDEO_SIZE);
2106
wdenk4b248f32004-03-14 16:51:43 +00002107 return 0;
wdenkc6097192002-11-03 00:24:07 +00002108}
2109
Wolfgang Denk6cc7ba92009-05-15 10:07:43 +02002110/*
2111 * Implement a weak default function for boards that optionally
2112 * need to skip the video initialization.
2113 */
Jeroen Hofstee69d27542014-10-08 22:57:30 +02002114__weak int board_video_skip(void)
Wolfgang Denk6cc7ba92009-05-15 10:07:43 +02002115{
2116 /* As default, don't skip test */
2117 return 0;
2118}
Wolfgang Denk6cc7ba92009-05-15 10:07:43 +02002119
Wolfgang Denk64e40d72011-07-29 09:55:27 +00002120int drv_video_init(void)
wdenkc6097192002-11-03 00:24:07 +00002121{
Jean-Christophe PLAGNIOL-VILLARD52cb4d42009-05-16 12:14:54 +02002122 struct stdio_dev console_dev;
Simon Glass5692ccf2015-03-02 12:40:50 -07002123 bool have_keyboard;
Bin Meng8ceb2422015-08-24 01:00:07 -07002124 bool __maybe_unused keyboard_ok = false;
wdenkc6097192002-11-03 00:24:07 +00002125
Wolfgang Denk6cc7ba92009-05-15 10:07:43 +02002126 /* Check if video initialization should be skipped */
2127 if (board_video_skip())
2128 return 0;
2129
wdenk81050922004-07-11 20:04:51 +00002130 /* Init video chip - returns with framebuffer cleared */
Simon Glass0a6eac82016-10-17 20:12:54 -06002131 if (cfg_video_init() == -1)
Bin Meng8ceb2422015-08-24 01:00:07 -07002132 return 0;
wdenk81050922004-07-11 20:04:51 +00002133
Heiko Schocher2bc4aa52013-08-03 07:22:53 +02002134 if (board_cfb_skip())
2135 return 0;
2136
Simon Glass5692ccf2015-03-02 12:40:50 -07002137#if defined(CONFIG_VGA_AS_SINGLE_DEVICE)
2138 have_keyboard = false;
2139#elif defined(CONFIG_OF_CONTROL)
2140 have_keyboard = !fdtdec_get_config_bool(gd->fdt_blob,
2141 "u-boot,no-keyboard");
2142#else
2143 have_keyboard = true;
Wolfgang Denkf62f6462009-05-15 10:07:42 +02002144#endif
Simon Glass5692ccf2015-03-02 12:40:50 -07002145 if (have_keyboard) {
2146 debug("KBD: Keyboard init ...\n");
2147#if !defined(CONFIG_VGA_AS_SINGLE_DEVICE)
Bin Meng8ceb2422015-08-24 01:00:07 -07002148 keyboard_ok = !(VIDEO_KBD_INIT_FCT == -1);
Simon Glass5692ccf2015-03-02 12:40:50 -07002149#endif
2150 }
wdenkc6097192002-11-03 00:24:07 +00002151
Wolfgang Denkf62f6462009-05-15 10:07:42 +02002152 /* Init vga device */
Wolfgang Denk64e40d72011-07-29 09:55:27 +00002153 memset(&console_dev, 0, sizeof(console_dev));
2154 strcpy(console_dev.name, "vga");
Bin Meng1caf9342015-11-03 23:23:37 -08002155 console_dev.flags = DEV_FLAGS_OUTPUT;
Simon Glass0a6eac82016-10-17 20:12:54 -06002156 console_dev.putc = cfb_video_putc; /* 'putc' function */
2157 console_dev.puts = cfb_video_puts; /* 'puts' function */
Wolfgang Denkf62f6462009-05-15 10:07:42 +02002158
2159#if !defined(CONFIG_VGA_AS_SINGLE_DEVICE)
Bin Meng8ceb2422015-08-24 01:00:07 -07002160 if (have_keyboard && keyboard_ok) {
Simon Glass5692ccf2015-03-02 12:40:50 -07002161 /* Also init console device */
2162 console_dev.flags |= DEV_FLAGS_INPUT;
2163 console_dev.tstc = VIDEO_TSTC_FCT; /* 'tstc' function */
2164 console_dev.getc = VIDEO_GETC_FCT; /* 'getc' function */
2165 }
2166#endif
Wolfgang Denkf62f6462009-05-15 10:07:42 +02002167
Wolfgang Denk64e40d72011-07-29 09:55:27 +00002168 if (stdio_register(&console_dev) != 0)
Wolfgang Denkf62f6462009-05-15 10:07:42 +02002169 return 0;
2170
2171 /* Return success */
2172 return 1;
wdenkc6097192002-11-03 00:24:07 +00002173}
Stefan Reinauerc20ee072012-09-28 15:11:12 +00002174
2175void video_position_cursor(unsigned col, unsigned row)
2176{
2177 console_col = min(col, CONSOLE_COLS - 1);
2178 console_row = min(row, CONSOLE_ROWS - 1);
2179}
2180
2181int video_get_pixel_width(void)
2182{
2183 return VIDEO_VISIBLE_COLS;
2184}
2185
2186int video_get_pixel_height(void)
2187{
2188 return VIDEO_VISIBLE_ROWS;
2189}
2190
2191int video_get_screen_rows(void)
2192{
2193 return CONSOLE_ROWS;
2194}
2195
2196int video_get_screen_columns(void)
2197{
2198 return CONSOLE_COLS;
2199}