blob: 55ecdf10127ed6e976e7906dd2707d89381e0835 [file] [log] [blame]
Dirk Eibacha605ea72010-10-21 10:50:05 +02001/*
2 * (C) Copyright 2010
3 * Dirk Eibach, Guntermann & Drunck GmbH, eibach@gdsys.de
4 *
Wolfgang Denk1a459662013-07-08 09:37:19 +02005 * SPDX-License-Identifier: GPL-2.0+
Dirk Eibacha605ea72010-10-21 10:50:05 +02006 */
7
8#include <common.h>
Dirk Eibachaba27ac2013-06-26 16:04:26 +02009#include <i2c.h>
Dirk Eibache50e8962013-07-25 19:28:13 +020010#include <malloc.h>
Dirk Eibacha605ea72010-10-21 10:50:05 +020011
Dirk Eibach3a990bf2014-07-03 09:28:22 +020012#include "dp501.h"
Dirk Eibach2da0fc02011-01-21 09:31:21 +010013#include <gdsys_fpga.h>
Dirk Eibacha605ea72010-10-21 10:50:05 +020014
15#define CH7301_I2C_ADDR 0x75
16
Dirk Eibach2da0fc02011-01-21 09:31:21 +010017#define ICS8N3QV01_I2C_ADDR 0x6E
Dirk Eibach6853cc42011-04-06 13:53:43 +020018#define ICS8N3QV01_FREF 114285000
19#define ICS8N3QV01_FREF_LL 114285000LL
20#define ICS8N3QV01_F_DEFAULT_0 156250000LL
21#define ICS8N3QV01_F_DEFAULT_1 125000000LL
22#define ICS8N3QV01_F_DEFAULT_2 100000000LL
23#define ICS8N3QV01_F_DEFAULT_3 25175000LL
Dirk Eibach2da0fc02011-01-21 09:31:21 +010024
25#define SIL1178_MASTER_I2C_ADDRESS 0x38
26#define SIL1178_SLAVE_I2C_ADDRESS 0x39
27
Dirk Eibach3a990bf2014-07-03 09:28:22 +020028#define DP501_I2C_ADDR 0x08
29
Dirk Eibacha605ea72010-10-21 10:50:05 +020030#define PIXCLK_640_480_60 25180000
31
Dirk Eibacha605ea72010-10-21 10:50:05 +020032enum {
Dirk Eibacha605ea72010-10-21 10:50:05 +020033 CH7301_CM = 0x1c, /* Clock Mode Register */
34 CH7301_IC = 0x1d, /* Input Clock Register */
35 CH7301_GPIO = 0x1e, /* GPIO Control Register */
36 CH7301_IDF = 0x1f, /* Input Data Format Register */
37 CH7301_CD = 0x20, /* Connection Detect Register */
38 CH7301_DC = 0x21, /* DAC Control Register */
39 CH7301_HPD = 0x23, /* Hot Plug Detection Register */
40 CH7301_TCTL = 0x31, /* DVI Control Input Register */
41 CH7301_TPCP = 0x33, /* DVI PLL Charge Pump Ctrl Register */
42 CH7301_TPD = 0x34, /* DVI PLL Divide Register */
43 CH7301_TPVT = 0x35, /* DVI PLL Supply Control Register */
44 CH7301_TPF = 0x36, /* DVI PLL Filter Register */
45 CH7301_TCT = 0x37, /* DVI Clock Test Register */
46 CH7301_TSTP = 0x48, /* Test Pattern Register */
47 CH7301_PM = 0x49, /* Power Management register */
48 CH7301_VID = 0x4a, /* Version ID Register */
49 CH7301_DID = 0x4b, /* Device ID Register */
50 CH7301_DSP = 0x56, /* DVI Sync polarity Register */
51};
52
Dirk Eibach0f0c1022013-06-26 16:04:30 +020053unsigned int base_width;
54unsigned int base_height;
55size_t bufsize;
56u16 *buf;
57
Dirk Eibache50e8962013-07-25 19:28:13 +020058unsigned int max_osd_screen = CONFIG_SYS_OSD_SCREENS - 1;
59
Dirk Eibach3a990bf2014-07-03 09:28:22 +020060#ifdef CONFIG_SYS_ICS8N3QV01_I2C
Dirk Eibachedfe9fe2014-07-03 09:28:17 +020061int ics8n3qv01_i2c[] = CONFIG_SYS_ICS8N3QV01_I2C;
62#endif
Dirk Eibach2da0fc02011-01-21 09:31:21 +010063
Dirk Eibach3a990bf2014-07-03 09:28:22 +020064#ifdef CONFIG_SYS_CH7301_I2C
65int ch7301_i2c[] = CONFIG_SYS_CH7301_I2C;
66#endif
67
68#ifdef CONFIG_SYS_SIL1178_I2C
Dirk Eibachedfe9fe2014-07-03 09:28:17 +020069int sil1178_i2c[] = CONFIG_SYS_SIL1178_I2C;
Dirk Eibach2da0fc02011-01-21 09:31:21 +010070#endif
71
Dirk Eibach3a990bf2014-07-03 09:28:22 +020072#ifdef CONFIG_SYS_DP501_I2C
73int dp501_i2c[] = CONFIG_SYS_DP501_I2C;
74#endif
75
76
Dirk Eibach2da0fc02011-01-21 09:31:21 +010077#ifdef CONFIG_SYS_MPC92469AC
Dirk Eibacha605ea72010-10-21 10:50:05 +020078static void mpc92469ac_calc_parameters(unsigned int fout,
79 unsigned int *post_div, unsigned int *feedback_div)
80{
81 unsigned int n = *post_div;
82 unsigned int m = *feedback_div;
83 unsigned int a;
84 unsigned int b = 14745600 / 16;
85
86 if (fout < 50169600)
87 n = 8;
88 else if (fout < 100339199)
89 n = 4;
90 else if (fout < 200678399)
91 n = 2;
92 else
93 n = 1;
94
95 a = fout * n + (b / 2); /* add b/2 for proper rounding */
96
97 m = a / b;
98
99 *post_div = n;
100 *feedback_div = m;
101}
102
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100103static void mpc92469ac_set(unsigned screen, unsigned int fout)
Dirk Eibacha605ea72010-10-21 10:50:05 +0200104{
105 unsigned int n;
106 unsigned int m;
107 unsigned int bitval = 0;
108 mpc92469ac_calc_parameters(fout, &n, &m);
109
110 switch (n) {
111 case 1:
112 bitval = 0x00;
113 break;
114 case 2:
115 bitval = 0x01;
116 break;
117 case 4:
118 bitval = 0x02;
119 break;
120 case 8:
121 bitval = 0x03;
122 break;
123 }
124
Dirk Eibachaba27ac2013-06-26 16:04:26 +0200125 FPGA_SET_REG(screen, mpc3w_control, (bitval << 9) | m);
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100126}
127#endif
128
Dirk Eibach3a990bf2014-07-03 09:28:22 +0200129#ifdef CONFIG_SYS_ICS8N3QV01_I2C
Dirk Eibach6853cc42011-04-06 13:53:43 +0200130
Dirk Eibachedfe9fe2014-07-03 09:28:17 +0200131static unsigned int ics8n3qv01_get_fout_calc(unsigned index)
Dirk Eibach6853cc42011-04-06 13:53:43 +0200132{
133 unsigned long long n;
134 unsigned long long mint;
135 unsigned long long mfrac;
136 u8 reg_a, reg_b, reg_c, reg_d, reg_f;
137 unsigned long long fout_calc;
138
139 if (index > 3)
140 return 0;
141
Dirk Eibachedfe9fe2014-07-03 09:28:17 +0200142 reg_a = i2c_reg_read(ICS8N3QV01_I2C_ADDR, 0 + index);
143 reg_b = i2c_reg_read(ICS8N3QV01_I2C_ADDR, 4 + index);
144 reg_c = i2c_reg_read(ICS8N3QV01_I2C_ADDR, 8 + index);
145 reg_d = i2c_reg_read(ICS8N3QV01_I2C_ADDR, 12 + index);
146 reg_f = i2c_reg_read(ICS8N3QV01_I2C_ADDR, 20 + index);
Dirk Eibach6853cc42011-04-06 13:53:43 +0200147
148 mint = ((reg_a >> 1) & 0x1f) | (reg_f & 0x20);
149 mfrac = ((reg_a & 0x01) << 17) | (reg_b << 9) | (reg_c << 1)
150 | (reg_d >> 7);
151 n = reg_d & 0x7f;
152
153 fout_calc = (mint * ICS8N3QV01_FREF_LL
154 + mfrac * ICS8N3QV01_FREF_LL / 262144LL
155 + ICS8N3QV01_FREF_LL / 524288LL
156 + n / 2)
157 / n
158 * 1000000
159 / (1000000 - 100);
160
161 return fout_calc;
162}
163
164
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100165static void ics8n3qv01_calc_parameters(unsigned int fout,
166 unsigned int *_mint, unsigned int *_mfrac,
167 unsigned int *_n)
168{
169 unsigned int n;
170 unsigned int foutiic;
171 unsigned int fvcoiic;
172 unsigned int mint;
173 unsigned long long mfrac;
174
Dirk Eibach6853cc42011-04-06 13:53:43 +0200175 n = (2215000000U + fout / 2) / fout;
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100176 if ((n & 1) && (n > 5))
177 n -= 1;
178
179 foutiic = fout - (fout / 10000);
180 fvcoiic = foutiic * n;
181
182 mint = fvcoiic / 114285000;
183 if ((mint < 17) || (mint > 63))
184 printf("ics8n3qv01_calc_parameters: cannot determine mint\n");
185
186 mfrac = ((unsigned long long)fvcoiic % 114285000LL) * 262144LL
187 / 114285000LL;
188
189 *_mint = mint;
190 *_mfrac = mfrac;
191 *_n = n;
Dirk Eibacha605ea72010-10-21 10:50:05 +0200192}
193
Dirk Eibachedfe9fe2014-07-03 09:28:17 +0200194static void ics8n3qv01_set(unsigned int fout)
Dirk Eibacha605ea72010-10-21 10:50:05 +0200195{
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100196 unsigned int n;
197 unsigned int mint;
198 unsigned int mfrac;
Dirk Eibach6853cc42011-04-06 13:53:43 +0200199 unsigned int fout_calc;
200 unsigned long long fout_prog;
201 long long off_ppm;
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100202 u8 reg0, reg4, reg8, reg12, reg18, reg20;
203
Dirk Eibachedfe9fe2014-07-03 09:28:17 +0200204 fout_calc = ics8n3qv01_get_fout_calc(1);
Dirk Eibach6853cc42011-04-06 13:53:43 +0200205 off_ppm = (fout_calc - ICS8N3QV01_F_DEFAULT_1) * 1000000
206 / ICS8N3QV01_F_DEFAULT_1;
207 printf(" PLL is off by %lld ppm\n", off_ppm);
208 fout_prog = (unsigned long long)fout * (unsigned long long)fout_calc
209 / ICS8N3QV01_F_DEFAULT_1;
210 ics8n3qv01_calc_parameters(fout_prog, &mint, &mfrac, &n);
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100211
Dirk Eibachedfe9fe2014-07-03 09:28:17 +0200212 reg0 = i2c_reg_read(ICS8N3QV01_I2C_ADDR, 0) & 0xc0;
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100213 reg0 |= (mint & 0x1f) << 1;
214 reg0 |= (mfrac >> 17) & 0x01;
Dirk Eibachedfe9fe2014-07-03 09:28:17 +0200215 i2c_reg_write(ICS8N3QV01_I2C_ADDR, 0, reg0);
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100216
217 reg4 = mfrac >> 9;
Dirk Eibachedfe9fe2014-07-03 09:28:17 +0200218 i2c_reg_write(ICS8N3QV01_I2C_ADDR, 4, reg4);
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100219
220 reg8 = mfrac >> 1;
Dirk Eibachedfe9fe2014-07-03 09:28:17 +0200221 i2c_reg_write(ICS8N3QV01_I2C_ADDR, 8, reg8);
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100222
223 reg12 = mfrac << 7;
224 reg12 |= n & 0x7f;
Dirk Eibachedfe9fe2014-07-03 09:28:17 +0200225 i2c_reg_write(ICS8N3QV01_I2C_ADDR, 12, reg12);
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100226
Dirk Eibachedfe9fe2014-07-03 09:28:17 +0200227 reg18 = i2c_reg_read(ICS8N3QV01_I2C_ADDR, 18) & 0x03;
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100228 reg18 |= 0x20;
Dirk Eibachedfe9fe2014-07-03 09:28:17 +0200229 i2c_reg_write(ICS8N3QV01_I2C_ADDR, 18, reg18);
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100230
Dirk Eibachedfe9fe2014-07-03 09:28:17 +0200231 reg20 = i2c_reg_read(ICS8N3QV01_I2C_ADDR, 20) & 0x1f;
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100232 reg20 |= mint & (1 << 5);
Dirk Eibachedfe9fe2014-07-03 09:28:17 +0200233 i2c_reg_write(ICS8N3QV01_I2C_ADDR, 20, reg20);
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100234}
235#endif
236
237static int osd_write_videomem(unsigned screen, unsigned offset,
238 u16 *data, size_t charcount)
239{
Dirk Eibacha605ea72010-10-21 10:50:05 +0200240 unsigned int k;
241
242 for (k = 0; k < charcount; ++k) {
Dirk Eibach0f0c1022013-06-26 16:04:30 +0200243 if (offset + k >= bufsize)
Dirk Eibacha605ea72010-10-21 10:50:05 +0200244 return -1;
Dirk Eibachaba27ac2013-06-26 16:04:26 +0200245 FPGA_SET_REG(screen, videomem[offset + k], data[k]);
Dirk Eibacha605ea72010-10-21 10:50:05 +0200246 }
247
248 return charcount;
249}
250
251static int osd_print(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
252{
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100253 unsigned screen;
Dirk Eibacha605ea72010-10-21 10:50:05 +0200254
Dirk Eibache50e8962013-07-25 19:28:13 +0200255 for (screen = 0; screen <= max_osd_screen; ++screen) {
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100256 unsigned x;
257 unsigned y;
258 unsigned charcount;
259 unsigned len;
260 u8 color;
261 unsigned int k;
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100262 char *text;
263 int res;
264
265 if (argc < 5) {
266 cmd_usage(cmdtp);
267 return 1;
268 }
269
270 x = simple_strtoul(argv[1], NULL, 16);
271 y = simple_strtoul(argv[2], NULL, 16);
272 color = simple_strtoul(argv[3], NULL, 16);
273 text = argv[4];
274 charcount = strlen(text);
Dirk Eibach0f0c1022013-06-26 16:04:30 +0200275 len = (charcount > bufsize) ? bufsize : charcount;
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100276
277 for (k = 0; k < len; ++k)
278 buf[k] = (text[k] << 8) | color;
279
Dirk Eibach0f0c1022013-06-26 16:04:30 +0200280 res = osd_write_videomem(screen, y * base_width + x, buf, len);
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100281 if (res < 0)
282 return res;
Dirk Eibacha605ea72010-10-21 10:50:05 +0200283 }
284
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100285 return 0;
Dirk Eibacha605ea72010-10-21 10:50:05 +0200286}
287
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100288int osd_probe(unsigned screen)
Dirk Eibacha605ea72010-10-21 10:50:05 +0200289{
Dirk Eibachaba27ac2013-06-26 16:04:26 +0200290 u16 version;
291 u16 features;
Dirk Eibache50e8962013-07-25 19:28:13 +0200292 int old_bus = i2c_get_bus_num();
Dirk Eibach3a990bf2014-07-03 09:28:22 +0200293 bool pixclock_present = false;
294 bool output_driver_present = false;
Dirk Eibacha605ea72010-10-21 10:50:05 +0200295
Dirk Eibachaba27ac2013-06-26 16:04:26 +0200296 FPGA_GET_REG(0, osd.version, &version);
297 FPGA_GET_REG(0, osd.features, &features);
298
Dirk Eibach0f0c1022013-06-26 16:04:30 +0200299 base_width = ((features & 0x3f00) >> 8) + 1;
300 base_height = (features & 0x001f) + 1;
301 bufsize = base_width * base_height;
302 buf = malloc(sizeof(u16) * bufsize);
303 if (!buf)
304 return -1;
Dirk Eibacha605ea72010-10-21 10:50:05 +0200305
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100306 printf("OSD%d: Digital-OSD version %01d.%02d, %d" "x%d characters\n",
Dirk Eibach0f0c1022013-06-26 16:04:30 +0200307 screen, version/100, version%100, base_width, base_height);
Dirk Eibacha605ea72010-10-21 10:50:05 +0200308
Dirk Eibach3a990bf2014-07-03 09:28:22 +0200309 /* setup pixclock */
Dirk Eibacha605ea72010-10-21 10:50:05 +0200310
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100311#ifdef CONFIG_SYS_MPC92469AC
Dirk Eibach3a990bf2014-07-03 09:28:22 +0200312 pixclock_present = true;
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100313 mpc92469ac_set(screen, PIXCLK_640_480_60);
314#endif
Dirk Eibacha605ea72010-10-21 10:50:05 +0200315
Dirk Eibach3a990bf2014-07-03 09:28:22 +0200316#ifdef CONFIG_SYS_ICS8N3QV01_I2C
Dirk Eibachedfe9fe2014-07-03 09:28:17 +0200317 i2c_set_bus_num(ics8n3qv01_i2c[screen]);
Dirk Eibach3a990bf2014-07-03 09:28:22 +0200318 if (!i2c_probe(ICS8N3QV01_I2C_ADDR)) {
319 ics8n3qv01_set(PIXCLK_640_480_60);
320 pixclock_present = true;
321 }
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100322#endif
323
Dirk Eibach3a990bf2014-07-03 09:28:22 +0200324 if (!pixclock_present)
325 printf(" no pixelclock found\n");
326
327 /* setup output driver */
328
329#ifdef CONFIG_SYS_CH7301_I2C
330 i2c_set_bus_num(ch7301_i2c[screen]);
331 if (!i2c_probe(CH7301_I2C_ADDR)) {
Dirk Eibacha61762d2014-11-13 19:21:15 +0100332 u8 value = i2c_reg_read(CH7301_I2C_ADDR, CH7301_DID);
333
Dirk Eibach3a990bf2014-07-03 09:28:22 +0200334 if (value == 0x17) {
335 i2c_reg_write(CH7301_I2C_ADDR, CH7301_TPCP, 0x08);
336 i2c_reg_write(CH7301_I2C_ADDR, CH7301_TPD, 0x16);
337 i2c_reg_write(CH7301_I2C_ADDR, CH7301_TPF, 0x60);
338 i2c_reg_write(CH7301_I2C_ADDR, CH7301_DC, 0x09);
339 i2c_reg_write(CH7301_I2C_ADDR, CH7301_PM, 0xc0);
340 output_driver_present = true;
341 }
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100342 }
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100343#endif
344
Dirk Eibach3a990bf2014-07-03 09:28:22 +0200345#ifdef CONFIG_SYS_SIL1178_I2C
346 i2c_set_bus_num(sil1178_i2c[screen]);
347 if (!i2c_probe(SIL1178_SLAVE_I2C_ADDRESS)) {
Dirk Eibacha61762d2014-11-13 19:21:15 +0100348 if (i2c_reg_read(SIL1178_SLAVE_I2C_ADDRESS, 0x02) == 0x06) {
Dirk Eibach3a990bf2014-07-03 09:28:22 +0200349 /*
350 * magic initialization sequence,
351 * adapted from datasheet
352 */
353 i2c_reg_write(SIL1178_SLAVE_I2C_ADDRESS, 0x08, 0x36);
354 i2c_reg_write(SIL1178_MASTER_I2C_ADDRESS, 0x0f, 0x44);
355 i2c_reg_write(SIL1178_MASTER_I2C_ADDRESS, 0x0f, 0x4c);
356 i2c_reg_write(SIL1178_MASTER_I2C_ADDRESS, 0x0e, 0x10);
357 i2c_reg_write(SIL1178_MASTER_I2C_ADDRESS, 0x0a, 0x80);
358 i2c_reg_write(SIL1178_MASTER_I2C_ADDRESS, 0x09, 0x30);
359 i2c_reg_write(SIL1178_MASTER_I2C_ADDRESS, 0x0c, 0x89);
360 i2c_reg_write(SIL1178_MASTER_I2C_ADDRESS, 0x0d, 0x60);
361 i2c_reg_write(SIL1178_MASTER_I2C_ADDRESS, 0x08, 0x36);
362 i2c_reg_write(SIL1178_MASTER_I2C_ADDRESS, 0x08, 0x37);
363 output_driver_present = true;
364 }
365 }
366#endif
367
368#ifdef CONFIG_SYS_DP501_I2C
369 i2c_set_bus_num(dp501_i2c[screen]);
370 if (!i2c_probe(DP501_I2C_ADDR)) {
371 dp501_powerup(DP501_I2C_ADDR);
372 output_driver_present = true;
373 }
374#endif
375
376 if (!output_driver_present)
377 printf(" no output driver found\n");
378
Dirk Eibachaba27ac2013-06-26 16:04:26 +0200379 FPGA_SET_REG(screen, osd.control, 0x0049);
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100380
Dirk Eibachaba27ac2013-06-26 16:04:26 +0200381 FPGA_SET_REG(screen, osd.xy_size, ((32 - 1) << 8) | (16 - 1));
382 FPGA_SET_REG(screen, osd.x_pos, 0x007f);
383 FPGA_SET_REG(screen, osd.y_pos, 0x005f);
384
Dirk Eibache50e8962013-07-25 19:28:13 +0200385 if (screen > max_osd_screen)
386 max_osd_screen = screen;
Dirk Eibacha605ea72010-10-21 10:50:05 +0200387
Dirk Eibachedfe9fe2014-07-03 09:28:17 +0200388 i2c_set_bus_num(old_bus);
389
Dirk Eibacha605ea72010-10-21 10:50:05 +0200390 return 0;
391}
392
393int osd_write(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
394{
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100395 unsigned screen;
Dirk Eibacha605ea72010-10-21 10:50:05 +0200396
Dirk Eibache50e8962013-07-25 19:28:13 +0200397 for (screen = 0; screen <= max_osd_screen; ++screen) {
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100398 unsigned x;
399 unsigned y;
400 unsigned k;
Dirk Eibach0f0c1022013-06-26 16:04:30 +0200401 u16 buffer[base_width];
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100402 char *rp;
403 u16 *wp = buffer;
404 unsigned count = (argc > 4) ?
405 simple_strtoul(argv[4], NULL, 16) : 1;
Dirk Eibacha605ea72010-10-21 10:50:05 +0200406
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100407 if ((argc < 4) || (strlen(argv[3]) % 4)) {
408 cmd_usage(cmdtp);
409 return 1;
410 }
411
412 x = simple_strtoul(argv[1], NULL, 16);
413 y = simple_strtoul(argv[2], NULL, 16);
414 rp = argv[3];
Dirk Eibacha605ea72010-10-21 10:50:05 +0200415
416
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100417 while (*rp) {
418 char substr[5];
Dirk Eibacha605ea72010-10-21 10:50:05 +0200419
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100420 memcpy(substr, rp, 4);
421 substr[4] = 0;
422 *wp = simple_strtoul(substr, NULL, 16);
Dirk Eibacha605ea72010-10-21 10:50:05 +0200423
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100424 rp += 4;
425 wp++;
Dirk Eibach0f0c1022013-06-26 16:04:30 +0200426 if (wp - buffer > base_width)
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100427 break;
428 }
Dirk Eibacha605ea72010-10-21 10:50:05 +0200429
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100430 for (k = 0; k < count; ++k) {
431 unsigned offset =
Dirk Eibach0f0c1022013-06-26 16:04:30 +0200432 y * base_width + x + k * (wp - buffer);
Dirk Eibach2da0fc02011-01-21 09:31:21 +0100433 osd_write_videomem(screen, offset, buffer,
434 wp - buffer);
435 }
Dirk Eibacha605ea72010-10-21 10:50:05 +0200436 }
437
438 return 0;
439}
440
441U_BOOT_CMD(
442 osdw, 5, 0, osd_write,
443 "write 16-bit hex encoded buffer to osd memory",
444 "pos_x pos_y buffer count\n"
445);
446
447U_BOOT_CMD(
448 osdp, 5, 0, osd_print,
449 "write ASCII buffer to osd memory",
450 "pos_x pos_y color text\n"
451);