2 * Copyright (C) 2006-2009 Freescale Semiconductor, Inc.
4 * Dave Liu <daveliu@freescale.com>
5 * based on source code of Shlomi Gridish
7 * SPDX-License-Identifier: GPL-2.0+
12 #include "asm/errno.h"
14 #include "asm/immap_qe.h"
17 qe_map_t *qe_immr = NULL;
18 static qe_snum_t snums[QE_NUM_OF_SNUM];
20 DECLARE_GLOBAL_DATA_PTR;
22 void qe_issue_cmd(uint cmd, uint sbc, u8 mcn, u32 cmd_data)
26 if (cmd == QE_RESET) {
27 out_be32(&qe_immr->cp.cecr,(u32) (cmd | QE_CR_FLG));
29 out_be32(&qe_immr->cp.cecdr, cmd_data);
30 out_be32(&qe_immr->cp.cecr, (sbc | QE_CR_FLG |
31 ((u32) mcn<<QE_CR_PROTOCOL_SHIFT) | cmd));
33 /* Wait for the QE_CR_FLG to clear */
35 cecr = in_be32(&qe_immr->cp.cecr);
36 } while (cecr & QE_CR_FLG);
41 uint qe_muram_alloc(uint size, uint align)
47 align_mask = align - 1;
48 savebase = gd->arch.mp_alloc_base;
50 off = gd->arch.mp_alloc_base & align_mask;
52 gd->arch.mp_alloc_base += (align - off);
54 if ((off = size & align_mask) != 0)
55 size += (align - off);
57 if ((gd->arch.mp_alloc_base + size) >= gd->arch.mp_alloc_top) {
58 gd->arch.mp_alloc_base = savebase;
59 printf("%s: ran out of ram.\n", __FUNCTION__);
62 retloc = gd->arch.mp_alloc_base;
63 gd->arch.mp_alloc_base += size;
65 memset((void *)&qe_immr->muram[retloc], 0, size);
67 __asm__ __volatile__("sync");
72 void *qe_muram_addr(uint offset)
74 return (void *)&qe_immr->muram[offset];
77 static void qe_sdma_init(void)
80 uint sdma_buffer_base;
82 p = (volatile sdma_t *)&qe_immr->sdma;
84 /* All of DMA transaction in bus 1 */
85 out_be32(&p->sdaqr, 0);
86 out_be32(&p->sdaqmr, 0);
88 /* Allocate 2KB temporary buffer for sdma */
89 sdma_buffer_base = qe_muram_alloc(2048, 4096);
90 out_be32(&p->sdwbcr, sdma_buffer_base & QE_SDEBCR_BA_MASK);
92 /* Clear sdma status */
93 out_be32(&p->sdsr, 0x03000000);
95 /* Enable global mode on bus 1, and 2KB buffer size */
96 out_be32(&p->sdmr, QE_SDMR_GLB_1_MSK | (0x3 << QE_SDMR_CEN_SHIFT));
99 /* This table is a list of the serial numbers of the Threads, taken from the
100 * "SNUM Table" chart in the QE Reference Manual. The order is not important,
101 * we just need to know what the SNUMs are for the threads.
103 static u8 thread_snum[] = {
104 /* Evthreads 16-29 are not supported in MPC8309 */
105 #if !defined(CONFIG_MPC8309)
106 0x04, 0x05, 0x0c, 0x0d,
107 0x14, 0x15, 0x1c, 0x1d,
108 0x24, 0x25, 0x2c, 0x2d,
111 0x88, 0x89, 0x98, 0x99,
112 0xa8, 0xa9, 0xb8, 0xb9,
113 0xc8, 0xc9, 0xd8, 0xd9,
114 0xe8, 0xe9, 0x08, 0x09,
115 0x18, 0x19, 0x28, 0x29,
116 0x38, 0x39, 0x48, 0x49,
117 0x58, 0x59, 0x68, 0x69,
118 0x78, 0x79, 0x80, 0x81
121 static void qe_snums_init(void)
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];
131 int qe_get_snum(void)
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;
147 void qe_put_snum(u8 snum)
151 for (i = 0; i < QE_NUM_OF_SNUM; i++) {
152 if (snums[i].num == snum) {
153 snums[i].state = QE_SNUM_STATE_FREE;
159 void qe_init(uint qe_base)
161 /* Init the QE IMMR base */
162 qe_immr = (qe_map_t *)qe_base;
164 #ifdef CONFIG_SYS_QE_FMAN_FW_IN_NOR
166 * Upload microcode to IRAM for those SOCs which do not have ROM in QE.
168 qe_upload_firmware((const void *)CONFIG_SYS_QE_FMAN_FW_ADDR);
170 /* enable the microcode in IRAM */
171 out_be32(&qe_immr->iram.iready,QE_IRAM_READY);
174 gd->arch.mp_alloc_base = QE_DATAONLY_BASE;
175 gd->arch.mp_alloc_top = gd->arch.mp_alloc_base + QE_DATAONLY_SIZE;
183 qe_issue_cmd(QE_RESET, QE_CR_SUBBLOCK_INVALID,
184 (u8) QE_CR_PROTOCOL_UNSPECIFIED, 0);
187 void qe_assign_page(uint snum, uint para_ram_base)
191 out_be32(&qe_immr->cp.cecdr, para_ram_base);
192 out_be32(&qe_immr->cp.cecr, ((u32) snum<<QE_CR_ASSIGN_PAGE_SNUM_SHIFT)
193 | QE_CR_FLG | QE_ASSIGN_PAGE);
195 /* Wait for the QE_CR_FLG to clear */
197 cecr = in_be32(&qe_immr->cp.cecr);
198 } while (cecr & QE_CR_FLG );
204 * brg: 0~15 as BRG1~BRG16
206 * BRG input clock comes from the BRGCLK (internal clock generated from
207 the QE clock, it is one-half of the QE clock), If need the clock source
208 from CLKn pin, we have te change the function.
211 #define BRG_CLK (gd->arch.brg_clk)
213 int qe_set_brg(uint brg, uint rate)
219 if (brg >= QE_NUM_OF_BRGS)
221 bp = (uint *)&qe_immr->brg.brgc1;
224 divisor = (BRG_CLK / rate);
225 if (divisor > QE_BRGC_DIVISOR_MAX + 1) {
230 *bp = ((divisor - 1) << QE_BRGC_DIVISOR_SHIFT) | QE_BRGC_ENABLE;
231 __asm__ __volatile__("sync");
234 *bp |= QE_BRGC_DIV16;
235 __asm__ __volatile__("sync");
241 /* Set ethernet MII clock master
243 int qe_set_mii_clk_src(int ucc_num)
247 /* check if the UCC number is in range. */
248 if ((ucc_num > UCC_MAX_NUM - 1) || (ucc_num < 0)) {
249 printf("%s: ucc num not in ranges\n", __FUNCTION__);
253 cmxgcr = in_be32(&qe_immr->qmx.cmxgcr);
254 cmxgcr &= ~QE_CMXGCR_MII_ENET_MNG_MASK;
255 cmxgcr |= (ucc_num <<QE_CMXGCR_MII_ENET_MNG_SHIFT);
256 out_be32(&qe_immr->qmx.cmxgcr, cmxgcr);
261 /* Firmware information stored here for qe_get_firmware_info() */
262 static struct qe_firmware_info qe_firmware_info;
265 * Set to 1 if QE firmware has been uploaded, and therefore
266 * qe_firmware_info contains valid data.
268 static int qe_firmware_uploaded;
271 * Upload a QE microcode
273 * This function is a worker function for qe_upload_firmware(). It does
274 * the actual uploading of the microcode.
276 static void qe_upload_microcode(const void *base,
277 const struct qe_microcode *ucode)
279 const u32 *code = base + be32_to_cpu(ucode->code_offset);
282 if (ucode->major || ucode->minor || ucode->revision)
283 printf("QE: uploading microcode '%s' version %u.%u.%u\n",
284 ucode->id, ucode->major, ucode->minor, ucode->revision);
286 printf("QE: uploading microcode '%s'\n", ucode->id);
288 /* Use auto-increment */
289 out_be32(&qe_immr->iram.iadd, be32_to_cpu(ucode->iram_offset) |
290 QE_IRAM_IADD_AIE | QE_IRAM_IADD_BADDR);
292 for (i = 0; i < be32_to_cpu(ucode->count); i++)
293 out_be32(&qe_immr->iram.idata, be32_to_cpu(code[i]));
297 * Upload a microcode to the I-RAM at a specific address.
299 * See docs/README.qe_firmware for information on QE microcode uploading.
301 * Currently, only version 1 is supported, so the 'version' field must be
304 * The SOC model and revision are not validated, they are only displayed for
305 * informational purposes.
307 * 'calc_size' is the calculated size, in bytes, of the firmware structure and
308 * all of the microcode structures, minus the CRC.
310 * 'length' is the size that the structure says it is, including the CRC.
312 int qe_upload_firmware(const struct qe_firmware *firmware)
317 size_t calc_size = sizeof(struct qe_firmware);
319 const struct qe_header *hdr;
322 printf("Invalid address\n");
326 hdr = &firmware->header;
327 length = be32_to_cpu(hdr->length);
329 /* Check the magic */
330 if ((hdr->magic[0] != 'Q') || (hdr->magic[1] != 'E') ||
331 (hdr->magic[2] != 'F')) {
332 printf("Not a microcode\n");
336 /* Check the version */
337 if (hdr->version != 1) {
338 printf("Unsupported version\n");
342 /* Validate some of the fields */
343 if ((firmware->count < 1) || (firmware->count > MAX_QE_RISC)) {
344 printf("Invalid data\n");
348 /* Validate the length and check if there's a CRC */
349 calc_size += (firmware->count - 1) * sizeof(struct qe_microcode);
351 for (i = 0; i < firmware->count; i++)
353 * For situations where the second RISC uses the same microcode
354 * as the first, the 'code_offset' and 'count' fields will be
355 * zero, so it's okay to add those.
357 calc_size += sizeof(u32) *
358 be32_to_cpu(firmware->microcode[i].count);
360 /* Validate the length */
361 if (length != calc_size + sizeof(u32)) {
362 printf("Invalid length\n");
367 * Validate the CRC. We would normally call crc32_no_comp(), but that
368 * function isn't available unless you turn on JFFS support.
370 crc = be32_to_cpu(*(u32 *)((void *)firmware + calc_size));
371 if (crc != (crc32(-1, (const void *) firmware, calc_size) ^ -1)) {
372 printf("Firmware CRC is invalid\n");
377 * If the microcode calls for it, split the I-RAM.
379 if (!firmware->split) {
380 out_be16(&qe_immr->cp.cercr,
381 in_be16(&qe_immr->cp.cercr) | QE_CP_CERCR_CIR);
384 if (firmware->soc.model)
385 printf("Firmware '%s' for %u V%u.%u\n",
386 firmware->id, be16_to_cpu(firmware->soc.model),
387 firmware->soc.major, firmware->soc.minor);
389 printf("Firmware '%s'\n", firmware->id);
392 * The QE only supports one microcode per RISC, so clear out all the
393 * saved microcode information and put in the new.
395 memset(&qe_firmware_info, 0, sizeof(qe_firmware_info));
396 strcpy(qe_firmware_info.id, (char *)firmware->id);
397 qe_firmware_info.extended_modes = firmware->extended_modes;
398 memcpy(qe_firmware_info.vtraps, firmware->vtraps,
399 sizeof(firmware->vtraps));
400 qe_firmware_uploaded = 1;
402 /* Loop through each microcode. */
403 for (i = 0; i < firmware->count; i++) {
404 const struct qe_microcode *ucode = &firmware->microcode[i];
406 /* Upload a microcode if it's present */
407 if (ucode->code_offset)
408 qe_upload_microcode(firmware, ucode);
410 /* Program the traps for this processor */
411 for (j = 0; j < 16; j++) {
412 u32 trap = be32_to_cpu(ucode->traps[j]);
415 out_be32(&qe_immr->rsp[i].tibcr[j], trap);
419 out_be32(&qe_immr->rsp[i].eccr, be32_to_cpu(ucode->eccr));
425 struct qe_firmware_info *qe_get_firmware_info(void)
427 return qe_firmware_uploaded ? &qe_firmware_info : NULL;
430 static int qe_cmd(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
435 return cmd_usage(cmdtp);
437 if (strcmp(argv[1], "fw") == 0) {
438 addr = simple_strtoul(argv[2], NULL, 16);
441 printf("Invalid address\n");
446 * If a length was supplied, compare that with the 'length'
451 ulong length = simple_strtoul(argv[3], NULL, 16);
452 struct qe_firmware *firmware = (void *) addr;
454 if (length != be32_to_cpu(firmware->header.length)) {
455 printf("Length mismatch\n");
460 return qe_upload_firmware((const struct qe_firmware *) addr);
463 return cmd_usage(cmdtp);
468 "QUICC Engine commands",
469 "fw <addr> [<length>] - Upload firmware binary at address <addr> to "
471 "\twith optional length <length> verification."