blob: ea5a14b0bb3d91e75bdee14855a6d4cc559334d8 [file] [log] [blame]
Dave Liu7737d5c2006-11-03 12:11:15 -06001/*
2 * Copyright (C) 2006 Freescale Semiconductor, Inc.
3 *
4 * Dave Liu <daveliu@freescale.com>
5 * based on source code of Shlomi Gridish
6 *
7 * This program is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU General Public License as
9 * published by the Free Software Foundation; either version 2 of
10 * the License, or (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, write to the Free Software
19 * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
20 * MA 02111-1307 USA
21 */
22
23#include "common.h"
Timur Tabib8ec2382008-01-07 13:31:19 -060024#include <command.h>
Dave Liu7737d5c2006-11-03 12:11:15 -060025#include "asm/errno.h"
26#include "asm/io.h"
27#include "asm/immap_qe.h"
28#include "qe.h"
29
Dave Liu7737d5c2006-11-03 12:11:15 -060030qe_map_t *qe_immr = NULL;
31static qe_snum_t snums[QE_NUM_OF_SNUM];
32
Wolfgang Denk1218abf2007-09-15 20:48:41 +020033DECLARE_GLOBAL_DATA_PTR;
34
Dave Liu7737d5c2006-11-03 12:11:15 -060035void qe_issue_cmd(uint cmd, uint sbc, u8 mcn, u32 cmd_data)
36{
Wolfgang Denkd3a65322008-01-10 00:55:14 +010037 u32 cecr;
Dave Liu7737d5c2006-11-03 12:11:15 -060038
39 if (cmd == QE_RESET) {
40 out_be32(&qe_immr->cp.cecr,(u32) (cmd | QE_CR_FLG));
41 } else {
42 out_be32(&qe_immr->cp.cecdr, cmd_data);
43 out_be32(&qe_immr->cp.cecr, (sbc | QE_CR_FLG |
44 ((u32) mcn<<QE_CR_PROTOCOL_SHIFT) | cmd));
45 }
46 /* Wait for the QE_CR_FLG to clear */
47 do {
48 cecr = in_be32(&qe_immr->cp.cecr);
49 } while (cecr & QE_CR_FLG);
50
51 return;
52}
53
54uint qe_muram_alloc(uint size, uint align)
55{
Dave Liu7737d5c2006-11-03 12:11:15 -060056 uint retloc;
57 uint align_mask, off;
58 uint savebase;
59
60 align_mask = align - 1;
61 savebase = gd->mp_alloc_base;
62
63 if ((off = (gd->mp_alloc_base & align_mask)) != 0)
64 gd->mp_alloc_base += (align - off);
65
66 if ((off = size & align_mask) != 0)
67 size += (align - off);
68
69 if ((gd->mp_alloc_base + size) >= gd->mp_alloc_top) {
70 gd->mp_alloc_base = savebase;
71 printf("%s: ran out of ram.\n", __FUNCTION__);
72 }
73
74 retloc = gd->mp_alloc_base;
75 gd->mp_alloc_base += size;
76
77 memset((void *)&qe_immr->muram[retloc], 0, size);
78
79 __asm__ __volatile__("sync");
80
81 return retloc;
82}
83
84void *qe_muram_addr(uint offset)
85{
86 return (void *)&qe_immr->muram[offset];
87}
88
89static void qe_sdma_init(void)
90{
91 volatile sdma_t *p;
92 uint sdma_buffer_base;
93
94 p = (volatile sdma_t *)&qe_immr->sdma;
95
96 /* All of DMA transaction in bus 1 */
97 out_be32(&p->sdaqr, 0);
98 out_be32(&p->sdaqmr, 0);
99
100 /* Allocate 2KB temporary buffer for sdma */
Dave Liuff9658d2007-06-25 10:41:04 +0800101 sdma_buffer_base = qe_muram_alloc(2048, 4096);
Dave Liu7737d5c2006-11-03 12:11:15 -0600102 out_be32(&p->sdwbcr, sdma_buffer_base & QE_SDEBCR_BA_MASK);
103
104 /* Clear sdma status */
105 out_be32(&p->sdsr, 0x03000000);
106
107 /* Enable global mode on bus 1, and 2KB buffer size */
108 out_be32(&p->sdmr, QE_SDMR_GLB_1_MSK | (0x3 << QE_SDMR_CEN_SHIFT));
109}
110
111static u8 thread_snum[QE_NUM_OF_SNUM] = {
112 0x04, 0x05, 0x0c, 0x0d,
113 0x14, 0x15, 0x1c, 0x1d,
114 0x24, 0x25, 0x2c, 0x2d,
115 0x34, 0x35, 0x88, 0x89,
116 0x98, 0x99, 0xa8, 0xa9,
117 0xb8, 0xb9, 0xc8, 0xc9,
118 0xd8, 0xd9, 0xe8, 0xe9
119};
120
121static void qe_snums_init(void)
122{
123 int i;
124
125 for (i = 0; i < QE_NUM_OF_SNUM; i++) {
126 snums[i].state = QE_SNUM_STATE_FREE;
127 snums[i].num = thread_snum[i];
128 }
129}
130
131int qe_get_snum(void)
132{
133 int snum = -EBUSY;
134 int i;
135
136 for (i = 0; i < QE_NUM_OF_SNUM; i++) {
137 if (snums[i].state == QE_SNUM_STATE_FREE) {
138 snums[i].state = QE_SNUM_STATE_USED;
139 snum = snums[i].num;
140 break;
141 }
142 }
143
144 return snum;
145}
146
147void qe_put_snum(u8 snum)
148{
149 int i;
150
151 for (i = 0; i < QE_NUM_OF_SNUM; i++) {
152 if (snums[i].num == snum) {
153 snums[i].state = QE_SNUM_STATE_FREE;
154 break;
155 }
156 }
157}
158
159void qe_init(uint qe_base)
160{
Dave Liu7737d5c2006-11-03 12:11:15 -0600161 /* Init the QE IMMR base */
162 qe_immr = (qe_map_t *)qe_base;
163
Haiying Wang2d4de6a2009-03-26 17:01:49 -0400164#ifdef CONFIG_SYS_QE_FW_ADDR
165 /* Upload microcode to IRAM for those SOCs which do not have ROM in QE.
166 */
167 qe_upload_firmware((const struct qe_firmware *) CONFIG_SYS_QE_FW_ADDR);
168
169 /* enable the microcode in IRAM */
170 out_be32(&qe_immr->iram.iready,QE_IRAM_READY);
171#endif
172
Dave Liu7737d5c2006-11-03 12:11:15 -0600173 gd->mp_alloc_base = QE_DATAONLY_BASE;
174 gd->mp_alloc_top = gd->mp_alloc_base + QE_DATAONLY_SIZE;
175
176 qe_sdma_init();
177 qe_snums_init();
178}
179
180void qe_reset(void)
181{
182 qe_issue_cmd(QE_RESET, QE_CR_SUBBLOCK_INVALID,
183 (u8) QE_CR_PROTOCOL_UNSPECIFIED, 0);
184}
185
186void qe_assign_page(uint snum, uint para_ram_base)
187{
188 u32 cecr;
189
190 out_be32(&qe_immr->cp.cecdr, para_ram_base);
191 out_be32(&qe_immr->cp.cecr, ((u32) snum<<QE_CR_ASSIGN_PAGE_SNUM_SHIFT)
192 | QE_CR_FLG | QE_ASSIGN_PAGE);
193
194 /* Wait for the QE_CR_FLG to clear */
195 do {
196 cecr = in_be32(&qe_immr->cp.cecr);
197 } while (cecr & QE_CR_FLG );
198
199 return;
200}
201
202/*
203 * brg: 0~15 as BRG1~BRG16
204 rate: baud rate
205 * BRG input clock comes from the BRGCLK (internal clock generated from
206 the QE clock, it is one-half of the QE clock), If need the clock source
207 from CLKn pin, we have te change the function.
208 */
209
210#define BRG_CLK (gd->brg_clk)
211
212int qe_set_brg(uint brg, uint rate)
213{
Dave Liu7737d5c2006-11-03 12:11:15 -0600214 volatile uint *bp;
215 u32 divisor;
216 int div16 = 0;
217
218 if (brg >= QE_NUM_OF_BRGS)
219 return -EINVAL;
220 bp = (uint *)&qe_immr->brg.brgc1;
221 bp += brg;
222
223 divisor = (BRG_CLK / rate);
224 if (divisor > QE_BRGC_DIVISOR_MAX + 1) {
225 div16 = 1;
226 divisor /= 16;
227 }
228
229 *bp = ((divisor - 1) << QE_BRGC_DIVISOR_SHIFT) | QE_BRGC_ENABLE;
230 __asm__ __volatile__("sync");
231
232 if (div16) {
233 *bp |= QE_BRGC_DIV16;
234 __asm__ __volatile__("sync");
235 }
236
237 return 0;
238}
239
240/* Set ethernet MII clock master
241*/
242int qe_set_mii_clk_src(int ucc_num)
243{
244 u32 cmxgcr;
245
246 /* check if the UCC number is in range. */
247 if ((ucc_num > UCC_MAX_NUM - 1) || (ucc_num < 0)) {
248 printf("%s: ucc num not in ranges\n", __FUNCTION__);
249 return -EINVAL;
250 }
251
252 cmxgcr = in_be32(&qe_immr->qmx.cmxgcr);
253 cmxgcr &= ~QE_CMXGCR_MII_ENET_MNG_MASK;
254 cmxgcr |= (ucc_num <<QE_CMXGCR_MII_ENET_MNG_SHIFT);
255 out_be32(&qe_immr->qmx.cmxgcr, cmxgcr);
256
257 return 0;
258}
259
Timur Tabib8ec2382008-01-07 13:31:19 -0600260/* The maximum number of RISCs we support */
261#define MAX_QE_RISC 2
262
263/* Firmware information stored here for qe_get_firmware_info() */
264static struct qe_firmware_info qe_firmware_info;
265
266/*
267 * Set to 1 if QE firmware has been uploaded, and therefore
268 * qe_firmware_info contains valid data.
269 */
270static int qe_firmware_uploaded;
271
272/*
273 * Upload a QE microcode
274 *
275 * This function is a worker function for qe_upload_firmware(). It does
276 * the actual uploading of the microcode.
277 */
278static void qe_upload_microcode(const void *base,
279 const struct qe_microcode *ucode)
280{
281 const u32 *code = base + be32_to_cpu(ucode->code_offset);
282 unsigned int i;
283
284 if (ucode->major || ucode->minor || ucode->revision)
285 printf("QE: uploading microcode '%s' version %u.%u.%u\n",
286 ucode->id, ucode->major, ucode->minor, ucode->revision);
287 else
288 printf("QE: uploading microcode '%s'\n", ucode->id);
289
290 /* Use auto-increment */
291 out_be32(&qe_immr->iram.iadd, be32_to_cpu(ucode->iram_offset) |
292 QE_IRAM_IADD_AIE | QE_IRAM_IADD_BADDR);
293
294 for (i = 0; i < be32_to_cpu(ucode->count); i++)
295 out_be32(&qe_immr->iram.idata, be32_to_cpu(code[i]));
296}
297
298/*
299 * Upload a microcode to the I-RAM at a specific address.
300 *
301 * See docs/README.qe_firmware for information on QE microcode uploading.
302 *
303 * Currently, only version 1 is supported, so the 'version' field must be
304 * set to 1.
305 *
306 * The SOC model and revision are not validated, they are only displayed for
307 * informational purposes.
308 *
309 * 'calc_size' is the calculated size, in bytes, of the firmware structure and
310 * all of the microcode structures, minus the CRC.
311 *
312 * 'length' is the size that the structure says it is, including the CRC.
313 */
314int qe_upload_firmware(const struct qe_firmware *firmware)
315{
316 unsigned int i;
317 unsigned int j;
318 u32 crc;
319 size_t calc_size = sizeof(struct qe_firmware);
320 size_t length;
321 const struct qe_header *hdr;
322
323 if (!firmware) {
324 printf("Invalid address\n");
325 return -EINVAL;
326 }
327
328 hdr = &firmware->header;
329 length = be32_to_cpu(hdr->length);
330
331 /* Check the magic */
332 if ((hdr->magic[0] != 'Q') || (hdr->magic[1] != 'E') ||
333 (hdr->magic[2] != 'F')) {
334 printf("Not a microcode\n");
335 return -EPERM;
336 }
337
338 /* Check the version */
339 if (hdr->version != 1) {
340 printf("Unsupported version\n");
341 return -EPERM;
342 }
343
344 /* Validate some of the fields */
Timur Tabi491fb6d2008-03-03 09:58:52 -0600345 if ((firmware->count < 1) || (firmware->count > MAX_QE_RISC)) {
Timur Tabib8ec2382008-01-07 13:31:19 -0600346 printf("Invalid data\n");
347 return -EINVAL;
348 }
349
350 /* Validate the length and check if there's a CRC */
351 calc_size += (firmware->count - 1) * sizeof(struct qe_microcode);
352
353 for (i = 0; i < firmware->count; i++)
354 /*
355 * For situations where the second RISC uses the same microcode
356 * as the first, the 'code_offset' and 'count' fields will be
357 * zero, so it's okay to add those.
358 */
359 calc_size += sizeof(u32) *
360 be32_to_cpu(firmware->microcode[i].count);
361
362 /* Validate the length */
363 if (length != calc_size + sizeof(u32)) {
364 printf("Invalid length\n");
365 return -EPERM;
366 }
367
Wolfgang Denkd3a65322008-01-10 00:55:14 +0100368 /*
369 * Validate the CRC. We would normally call crc32_no_comp(), but that
370 * function isn't available unless you turn on JFFS support.
371 */
Timur Tabib8ec2382008-01-07 13:31:19 -0600372 crc = be32_to_cpu(*(u32 *)((void *)firmware + calc_size));
373 if (crc != (crc32(-1, (const void *) firmware, calc_size) ^ -1)) {
374 printf("Firmware CRC is invalid\n");
375 return -EIO;
376 }
377
378 /*
379 * If the microcode calls for it, split the I-RAM.
380 */
381 if (!firmware->split) {
382 out_be16(&qe_immr->cp.cercr,
383 in_be16(&qe_immr->cp.cercr) | QE_CP_CERCR_CIR);
384 }
385
386 if (firmware->soc.model)
387 printf("Firmware '%s' for %u V%u.%u\n",
388 firmware->id, be16_to_cpu(firmware->soc.model),
389 firmware->soc.major, firmware->soc.minor);
390 else
391 printf("Firmware '%s'\n", firmware->id);
392
393 /*
394 * The QE only supports one microcode per RISC, so clear out all the
395 * saved microcode information and put in the new.
396 */
397 memset(&qe_firmware_info, 0, sizeof(qe_firmware_info));
Dave Liu06c428b2008-01-14 11:12:01 +0800398 strcpy(qe_firmware_info.id, (char *)firmware->id);
Timur Tabib8ec2382008-01-07 13:31:19 -0600399 qe_firmware_info.extended_modes = firmware->extended_modes;
400 memcpy(qe_firmware_info.vtraps, firmware->vtraps,
401 sizeof(firmware->vtraps));
402 qe_firmware_uploaded = 1;
403
404 /* Loop through each microcode. */
405 for (i = 0; i < firmware->count; i++) {
406 const struct qe_microcode *ucode = &firmware->microcode[i];
407
408 /* Upload a microcode if it's present */
409 if (ucode->code_offset)
410 qe_upload_microcode(firmware, ucode);
411
412 /* Program the traps for this processor */
413 for (j = 0; j < 16; j++) {
414 u32 trap = be32_to_cpu(ucode->traps[j]);
415
416 if (trap)
417 out_be32(&qe_immr->rsp[i].tibcr[j], trap);
418 }
419
420 /* Enable traps */
421 out_be32(&qe_immr->rsp[i].eccr, be32_to_cpu(ucode->eccr));
422 }
423
424 return 0;
425}
426
427struct qe_firmware_info *qe_get_firmware_info(void)
428{
429 return qe_firmware_uploaded ? &qe_firmware_info : NULL;
430}
431
432static int qe_cmd(cmd_tbl_t *cmdtp, int flag, int argc, char *argv[])
433{
434 ulong addr;
435
436 if (argc < 3) {
Peter Tyser62c3ae72009-01-27 18:03:10 -0600437 cmd_usage(cmdtp);
Timur Tabib8ec2382008-01-07 13:31:19 -0600438 return 1;
439 }
440
441 if (strcmp(argv[1], "fw") == 0) {
442 addr = simple_strtoul(argv[2], NULL, 16);
443
444 if (!addr) {
445 printf("Invalid address\n");
446 return -EINVAL;
447 }
448
Wolfgang Denkd3a65322008-01-10 00:55:14 +0100449 /*
450 * If a length was supplied, compare that with the 'length'
451 * field.
452 */
Timur Tabib8ec2382008-01-07 13:31:19 -0600453
454 if (argc > 3) {
455 ulong length = simple_strtoul(argv[3], NULL, 16);
456 struct qe_firmware *firmware = (void *) addr;
457
458 if (length != be32_to_cpu(firmware->header.length)) {
459 printf("Length mismatch\n");
460 return -EINVAL;
461 }
462 }
463
464 return qe_upload_firmware((const struct qe_firmware *) addr);
465 }
466
Peter Tyser62c3ae72009-01-27 18:03:10 -0600467 cmd_usage(cmdtp);
Timur Tabib8ec2382008-01-07 13:31:19 -0600468 return 1;
469}
470
471U_BOOT_CMD(
472 qe, 4, 0, qe_cmd,
Peter Tyser2fb26042009-01-27 18:03:12 -0600473 "QUICC Engine commands",
Timur Tabib8ec2382008-01-07 13:31:19 -0600474 "fw <addr> [<length>] - Upload firmware binary at address <addr> to "
475 "the QE,\n\twith optional length <length> verification.\n"
476 );