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