blob: e54a4cacfabe2d3c8736a0c50b158644c01d7fa7 [file] [log] [blame]
Wolfgang Denkad5bb452007-03-06 18:08:43 +01001/*
2 * (C) Copyright 2002
3 * Wolfgang Denk, DENX Software Engineering, wd@denx.de.
4 *
Wolfgang Denk1a459662013-07-08 09:37:19 +02005 * SPDX-License-Identifier: GPL-2.0+
Wolfgang Denkad5bb452007-03-06 18:08:43 +01006 */
7
8#include <common.h>
9
10/*
11 * UART test
12 *
13 * The Serial Management Controllers (SMC) and the Serial Communication
14 * Controllers (SCC) listed in ctlr_list array below are tested in
15 * the loopback UART mode.
16 * The controllers are configured accordingly and several characters
17 * are transmitted. The configurable test parameters are:
18 * MIN_PACKET_LENGTH - minimum size of packet to transmit
19 * MAX_PACKET_LENGTH - maximum size of packet to transmit
20 * TEST_NUM - number of tests
21 */
22
Wolfgang Denkad5bb452007-03-06 18:08:43 +010023#include <post.h>
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +020024#if CONFIG_POST & CONFIG_SYS_POST_UART
Wolfgang Denkad5bb452007-03-06 18:08:43 +010025#if defined(CONFIG_8xx)
26#include <commproc.h>
27#elif defined(CONFIG_MPC8260)
28#include <asm/cpm_8260.h>
29#else
30#error "Apparently a bad configuration, please fix."
31#endif
32#include <command.h>
33#include <serial.h>
34
35DECLARE_GLOBAL_DATA_PTR;
36
37#define CTLR_SMC 0
38#define CTLR_SCC 1
39
40/* The list of controllers to test */
41#if defined(CONFIG_MPC823)
42static int ctlr_list[][2] =
43 { {CTLR_SMC, 0}, {CTLR_SMC, 1}, {CTLR_SCC, 1} };
44#else
45static int ctlr_list[][2] = { };
46#endif
47
Wolfgang Denkad5bb452007-03-06 18:08:43 +010048static struct {
49 void (*init) (int index);
50 void (*halt) (int index);
51 void (*putc) (int index, const char c);
52 int (*getc) (int index);
53} ctlr_proc[2];
54
55static char *ctlr_name[2] = { "SMC", "SCC" };
56
57static int proff_smc[] = { PROFF_SMC1, PROFF_SMC2 };
58static int proff_scc[] =
59 { PROFF_SCC1, PROFF_SCC2, PROFF_SCC3, PROFF_SCC4 };
60
61/*
62 * SMC callbacks
63 */
64
65static void smc_init (int smc_index)
66{
67 static int cpm_cr_ch[] = { CPM_CR_CH_SMC1, CPM_CR_CH_SMC2 };
68
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +020069 volatile immap_t *im = (immap_t *) CONFIG_SYS_IMMR;
Wolfgang Denkad5bb452007-03-06 18:08:43 +010070 volatile smc_t *sp;
71 volatile smc_uart_t *up;
72 volatile cbd_t *tbdf, *rbdf;
73 volatile cpm8xx_t *cp = &(im->im_cpm);
74 uint dpaddr;
75
76 /* initialize pointers to SMC */
77
78 sp = (smc_t *) & (cp->cp_smc[smc_index]);
79 up = (smc_uart_t *) & cp->cp_dparam[proff_smc[smc_index]];
80
81 /* Disable transmitter/receiver.
82 */
83 sp->smc_smcmr &= ~(SMCMR_REN | SMCMR_TEN);
84
85 /* Enable SDMA.
86 */
87 im->im_siu_conf.sc_sdcr = 1;
88
89 /* clear error conditions */
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +020090#ifdef CONFIG_SYS_SDSR
91 im->im_sdma.sdma_sdsr = CONFIG_SYS_SDSR;
Wolfgang Denkad5bb452007-03-06 18:08:43 +010092#else
93 im->im_sdma.sdma_sdsr = 0x83;
94#endif
95
96 /* clear SDMA interrupt mask */
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +020097#ifdef CONFIG_SYS_SDMR
98 im->im_sdma.sdma_sdmr = CONFIG_SYS_SDMR;
Wolfgang Denkad5bb452007-03-06 18:08:43 +010099#else
100 im->im_sdma.sdma_sdmr = 0x00;
101#endif
102
Wolfgang Denkad5bb452007-03-06 18:08:43 +0100103 /* Set the physical address of the host memory buffers in
104 * the buffer descriptors.
105 */
106
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200107#ifdef CONFIG_SYS_ALLOC_DPRAM
Wolfgang Denkad5bb452007-03-06 18:08:43 +0100108 dpaddr = dpram_alloc_align (sizeof (cbd_t) * 2 + 2, 8);
109#else
110 dpaddr = CPM_POST_BASE;
111#endif
112
113 /* Allocate space for two buffer descriptors in the DP ram.
114 * For now, this address seems OK, but it may have to
115 * change with newer versions of the firmware.
116 * damm: allocating space after the two buffers for rx/tx data
117 */
118
119 rbdf = (cbd_t *) & cp->cp_dpmem[dpaddr];
120 rbdf->cbd_bufaddr = (uint) (rbdf + 2);
121 rbdf->cbd_sc = 0;
122 tbdf = rbdf + 1;
123 tbdf->cbd_bufaddr = ((uint) (rbdf + 2)) + 1;
124 tbdf->cbd_sc = 0;
125
126 /* Set up the uart parameters in the parameter ram.
127 */
128 up->smc_rbase = dpaddr;
129 up->smc_tbase = dpaddr + sizeof (cbd_t);
130 up->smc_rfcr = SMC_EB;
131 up->smc_tfcr = SMC_EB;
132
Wolfgang Denkad5bb452007-03-06 18:08:43 +0100133 /* Set UART mode, 8 bit, no parity, one stop.
134 * Enable receive and transmit.
135 * Set local loopback mode.
136 */
137 sp->smc_smcmr = smcr_mk_clen (9) | SMCMR_SM_UART | (ushort) 0x0004;
138
139 /* Mask all interrupts and remove anything pending.
140 */
141 sp->smc_smcm = 0;
142 sp->smc_smce = 0xff;
143
144 /* Set up the baud rate generator.
145 */
146 cp->cp_simode = 0x00000000;
147
148 cp->cp_brgc1 =
149 (((gd->cpu_clk / 16 / gd->baudrate) -
150 1) << 1) | CPM_BRG_EN;
151
152 /* Make the first buffer the only buffer.
153 */
154 tbdf->cbd_sc |= BD_SC_WRAP;
155 rbdf->cbd_sc |= BD_SC_EMPTY | BD_SC_WRAP;
156
157 /* Single character receive.
158 */
159 up->smc_mrblr = 1;
160 up->smc_maxidl = 0;
161
162 /* Initialize Tx/Rx parameters.
163 */
164
165 while (cp->cp_cpcr & CPM_CR_FLG) /* wait if cp is busy */
166 ;
167
168 cp->cp_cpcr =
169 mk_cr_cmd (cpm_cr_ch[smc_index], CPM_CR_INIT_TRX) | CPM_CR_FLG;
170
171 while (cp->cp_cpcr & CPM_CR_FLG) /* wait if cp is busy */
172 ;
173
174 /* Enable transmitter/receiver.
175 */
176 sp->smc_smcmr |= SMCMR_REN | SMCMR_TEN;
177}
178
179static void smc_halt(int smc_index)
180{
181}
182
183static void smc_putc (int smc_index, const char c)
184{
185 volatile cbd_t *tbdf;
186 volatile char *buf;
187 volatile smc_uart_t *up;
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200188 volatile immap_t *im = (immap_t *) CONFIG_SYS_IMMR;
Wolfgang Denkad5bb452007-03-06 18:08:43 +0100189 volatile cpm8xx_t *cpmp = &(im->im_cpm);
190
191 up = (smc_uart_t *) & cpmp->cp_dparam[proff_smc[smc_index]];
192
193 tbdf = (cbd_t *) & cpmp->cp_dpmem[up->smc_tbase];
194
195 /* Wait for last character to go.
196 */
197
198 buf = (char *) tbdf->cbd_bufaddr;
199#if 0
200 __asm__ ("eieio");
201 while (tbdf->cbd_sc & BD_SC_READY)
202 __asm__ ("eieio");
203#endif
204
205 *buf = c;
206 tbdf->cbd_datlen = 1;
207 tbdf->cbd_sc |= BD_SC_READY;
208 __asm__ ("eieio");
209#if 1
210 while (tbdf->cbd_sc & BD_SC_READY)
211 __asm__ ("eieio");
212#endif
213}
214
215static int smc_getc (int smc_index)
216{
217 volatile cbd_t *rbdf;
218 volatile unsigned char *buf;
219 volatile smc_uart_t *up;
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200220 volatile immap_t *im = (immap_t *) CONFIG_SYS_IMMR;
Wolfgang Denkad5bb452007-03-06 18:08:43 +0100221 volatile cpm8xx_t *cpmp = &(im->im_cpm);
222 unsigned char c;
223 int i;
224
225 up = (smc_uart_t *) & cpmp->cp_dparam[proff_smc[smc_index]];
226
227 rbdf = (cbd_t *) & cpmp->cp_dpmem[up->smc_rbase];
228
229 /* Wait for character to show up.
230 */
231 buf = (unsigned char *) rbdf->cbd_bufaddr;
232#if 0
233 while (rbdf->cbd_sc & BD_SC_EMPTY);
234#else
235 for (i = 100; i > 0; i--) {
236 if (!(rbdf->cbd_sc & BD_SC_EMPTY))
237 break;
238 udelay (1000);
239 }
240
241 if (i == 0)
242 return -1;
243#endif
244 c = *buf;
245 rbdf->cbd_sc |= BD_SC_EMPTY;
246
247 return (c);
248}
249
250 /*
251 * SCC callbacks
252 */
253
254static void scc_init (int scc_index)
255{
256 static int cpm_cr_ch[] = {
257 CPM_CR_CH_SCC1,
258 CPM_CR_CH_SCC2,
259 CPM_CR_CH_SCC3,
260 CPM_CR_CH_SCC4,
261 };
262
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200263 volatile immap_t *im = (immap_t *) CONFIG_SYS_IMMR;
Wolfgang Denkad5bb452007-03-06 18:08:43 +0100264 volatile scc_t *sp;
265 volatile scc_uart_t *up;
266 volatile cbd_t *tbdf, *rbdf;
267 volatile cpm8xx_t *cp = &(im->im_cpm);
268 uint dpaddr;
269
270 /* initialize pointers to SCC */
271
272 sp = (scc_t *) & (cp->cp_scc[scc_index]);
273 up = (scc_uart_t *) & cp->cp_dparam[proff_scc[scc_index]];
274
275 /* Disable transmitter/receiver.
276 */
277 sp->scc_gsmrl &= ~(SCC_GSMRL_ENR | SCC_GSMRL_ENT);
278
279
280 /* Allocate space for two buffer descriptors in the DP ram.
281 */
282
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200283#ifdef CONFIG_SYS_ALLOC_DPRAM
Wolfgang Denkad5bb452007-03-06 18:08:43 +0100284 dpaddr = dpram_alloc_align (sizeof (cbd_t) * 2 + 2, 8);
285#else
286 dpaddr = CPM_POST_BASE;
287#endif
288
289 /* Enable SDMA.
290 */
291 im->im_siu_conf.sc_sdcr = 0x0001;
292
293 /* Set the physical address of the host memory buffers in
294 * the buffer descriptors.
295 */
296
297 rbdf = (cbd_t *) & cp->cp_dpmem[dpaddr];
298 rbdf->cbd_bufaddr = (uint) (rbdf + 2);
299 rbdf->cbd_sc = 0;
300 tbdf = rbdf + 1;
301 tbdf->cbd_bufaddr = ((uint) (rbdf + 2)) + 1;
302 tbdf->cbd_sc = 0;
303
304 /* Set up the baud rate generator.
305 */
306 cp->cp_sicr &= ~(0x000000FF << (8 * scc_index));
307 /* no |= needed, since BRG1 is 000 */
308
309 cp->cp_brgc1 =
310 (((gd->cpu_clk / 16 / gd->baudrate) -
311 1) << 1) | CPM_BRG_EN;
312
313 /* Set up the uart parameters in the parameter ram.
314 */
315 up->scc_genscc.scc_rbase = dpaddr;
316 up->scc_genscc.scc_tbase = dpaddr + sizeof (cbd_t);
317
318 /* Initialize Tx/Rx parameters.
319 */
320 while (cp->cp_cpcr & CPM_CR_FLG) /* wait if cp is busy */
321 ;
322 cp->cp_cpcr =
323 mk_cr_cmd (cpm_cr_ch[scc_index], CPM_CR_INIT_TRX) | CPM_CR_FLG;
324
325 while (cp->cp_cpcr & CPM_CR_FLG) /* wait if cp is busy */
326 ;
327
328 up->scc_genscc.scc_rfcr = SCC_EB | 0x05;
329 up->scc_genscc.scc_tfcr = SCC_EB | 0x05;
330
331 up->scc_genscc.scc_mrblr = 1; /* Single character receive */
332 up->scc_maxidl = 0; /* disable max idle */
333 up->scc_brkcr = 1; /* send one break character on stop TX */
334 up->scc_parec = 0;
335 up->scc_frmec = 0;
336 up->scc_nosec = 0;
337 up->scc_brkec = 0;
338 up->scc_uaddr1 = 0;
339 up->scc_uaddr2 = 0;
340 up->scc_toseq = 0;
341 up->scc_char1 = 0x8000;
342 up->scc_char2 = 0x8000;
343 up->scc_char3 = 0x8000;
344 up->scc_char4 = 0x8000;
345 up->scc_char5 = 0x8000;
346 up->scc_char6 = 0x8000;
347 up->scc_char7 = 0x8000;
348 up->scc_char8 = 0x8000;
349 up->scc_rccm = 0xc0ff;
350
351 /* Set low latency / small fifo.
352 */
353 sp->scc_gsmrh = SCC_GSMRH_RFW;
354
355 /* Set UART mode
356 */
357 sp->scc_gsmrl &= ~0xF;
358 sp->scc_gsmrl |= SCC_GSMRL_MODE_UART;
359
360 /* Set local loopback mode.
361 */
362 sp->scc_gsmrl &= ~SCC_GSMRL_DIAG_LE;
363 sp->scc_gsmrl |= SCC_GSMRL_DIAG_LOOP;
364
365 /* Set clock divider 16 on Tx and Rx
366 */
367 sp->scc_gsmrl |= (SCC_GSMRL_TDCR_16 | SCC_GSMRL_RDCR_16);
368
369 sp->scc_psmr |= SCU_PSMR_CL;
370
371 /* Mask all interrupts and remove anything pending.
372 */
373 sp->scc_sccm = 0;
374 sp->scc_scce = 0xffff;
375 sp->scc_dsr = 0x7e7e;
376 sp->scc_psmr = 0x3000;
377
378 /* Make the first buffer the only buffer.
379 */
380 tbdf->cbd_sc |= BD_SC_WRAP;
381 rbdf->cbd_sc |= BD_SC_EMPTY | BD_SC_WRAP;
382
383 /* Enable transmitter/receiver.
384 */
385 sp->scc_gsmrl |= (SCC_GSMRL_ENR | SCC_GSMRL_ENT);
386}
387
388static void scc_halt(int scc_index)
389{
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200390 volatile immap_t *im = (immap_t *) CONFIG_SYS_IMMR;
Wolfgang Denkad5bb452007-03-06 18:08:43 +0100391 volatile cpm8xx_t *cp = &(im->im_cpm);
392 volatile scc_t *sp = (scc_t *) & (cp->cp_scc[scc_index]);
393
394 sp->scc_gsmrl &= ~(SCC_GSMRL_ENR | SCC_GSMRL_ENT | SCC_GSMRL_DIAG_LE);
395}
396
397static void scc_putc (int scc_index, const char c)
398{
399 volatile cbd_t *tbdf;
400 volatile char *buf;
401 volatile scc_uart_t *up;
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200402 volatile immap_t *im = (immap_t *) CONFIG_SYS_IMMR;
Wolfgang Denkad5bb452007-03-06 18:08:43 +0100403 volatile cpm8xx_t *cpmp = &(im->im_cpm);
404
405 up = (scc_uart_t *) & cpmp->cp_dparam[proff_scc[scc_index]];
406
407 tbdf = (cbd_t *) & cpmp->cp_dpmem[up->scc_genscc.scc_tbase];
408
409 /* Wait for last character to go.
410 */
411
412 buf = (char *) tbdf->cbd_bufaddr;
413#if 0
414 __asm__ ("eieio");
415 while (tbdf->cbd_sc & BD_SC_READY)
416 __asm__ ("eieio");
417#endif
418
419 *buf = c;
420 tbdf->cbd_datlen = 1;
421 tbdf->cbd_sc |= BD_SC_READY;
422 __asm__ ("eieio");
423#if 1
424 while (tbdf->cbd_sc & BD_SC_READY)
425 __asm__ ("eieio");
426#endif
427}
428
429static int scc_getc (int scc_index)
430{
431 volatile cbd_t *rbdf;
432 volatile unsigned char *buf;
433 volatile scc_uart_t *up;
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200434 volatile immap_t *im = (immap_t *) CONFIG_SYS_IMMR;
Wolfgang Denkad5bb452007-03-06 18:08:43 +0100435 volatile cpm8xx_t *cpmp = &(im->im_cpm);
436 unsigned char c;
437 int i;
438
439 up = (scc_uart_t *) & cpmp->cp_dparam[proff_scc[scc_index]];
440
441 rbdf = (cbd_t *) & cpmp->cp_dpmem[up->scc_genscc.scc_rbase];
442
443 /* Wait for character to show up.
444 */
445 buf = (unsigned char *) rbdf->cbd_bufaddr;
446#if 0
447 while (rbdf->cbd_sc & BD_SC_EMPTY);
448#else
449 for (i = 100; i > 0; i--) {
450 if (!(rbdf->cbd_sc & BD_SC_EMPTY))
451 break;
452 udelay (1000);
453 }
454
455 if (i == 0)
456 return -1;
457#endif
458 c = *buf;
459 rbdf->cbd_sc |= BD_SC_EMPTY;
460
461 return (c);
462}
463
464 /*
465 * Test routines
466 */
467
468static int test_ctlr (int ctlr, int index)
469{
470 int res = -1;
471 char test_str[] = "*** UART Test String ***\r\n";
472 int i;
473
474 ctlr_proc[ctlr].init (index);
475
476 for (i = 0; i < sizeof (test_str) - 1; i++) {
477 ctlr_proc[ctlr].putc (index, test_str[i]);
478 if (ctlr_proc[ctlr].getc (index) != test_str[i])
479 goto Done;
480 }
481
482 res = 0;
483
484Done:
485 ctlr_proc[ctlr].halt (index);
486
487 if (res != 0) {
488 post_log ("uart %s%d test failed\n",
489 ctlr_name[ctlr], index + 1);
490 }
491
492 return res;
493}
494
495int uart_post_test (int flags)
496{
497 int res = 0;
498 int i;
499
500 ctlr_proc[CTLR_SMC].init = smc_init;
501 ctlr_proc[CTLR_SMC].halt = smc_halt;
502 ctlr_proc[CTLR_SMC].putc = smc_putc;
503 ctlr_proc[CTLR_SMC].getc = smc_getc;
504
505 ctlr_proc[CTLR_SCC].init = scc_init;
506 ctlr_proc[CTLR_SCC].halt = scc_halt;
507 ctlr_proc[CTLR_SCC].putc = scc_putc;
508 ctlr_proc[CTLR_SCC].getc = scc_getc;
509
Mike Frysingerd2397812011-05-10 07:28:35 +0000510 for (i = 0; i < ARRAY_SIZE(ctlr_list); i++) {
Wolfgang Denkad5bb452007-03-06 18:08:43 +0100511 if (test_ctlr (ctlr_list[i][0], ctlr_list[i][1]) != 0) {
512 res = -1;
513 }
514 }
515
516#if !defined(CONFIG_8xx_CONS_NONE)
517 serial_reinit_all ();
518#endif
519
520 return res;
521}
522
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200523#endif /* CONFIG_POST & CONFIG_SYS_POST_UART */