cmd_pmc440.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558
  1. /*
  2. * (C) Copyright 2007
  3. * Matthias Fuchs, esd Gmbh, matthias.fuchs@esd-electronics.com.
  4. *
  5. * See file CREDITS for list of people who contributed to this
  6. * project.
  7. *
  8. * This program is free software; you can redistribute it and/or
  9. * modify it under the terms of the GNU General Public License as
  10. * published by the Free Software Foundation; either version 2 of
  11. * the License, or (at your option) any later version.
  12. *
  13. * This program is distributed in the hope that it will be useful,
  14. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. * GNU General Public License for more details.
  17. *
  18. * You should have received a copy of the GNU General Public License
  19. * along with this program; if not, write to the Free Software
  20. * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
  21. * MA 02111-1307 USA
  22. *
  23. */
  24. #include <common.h>
  25. #include <command.h>
  26. #include <asm/io.h>
  27. #include <asm/cache.h>
  28. #include <asm/processor.h>
  29. #include "pmc440.h"
  30. int is_monarch(void);
  31. int bootstrap_eeprom_write(unsigned dev_addr, unsigned offset, uchar *buffer, unsigned cnt);
  32. int eeprom_write_enable(unsigned dev_addr, int state);
  33. DECLARE_GLOBAL_DATA_PTR;
  34. #if defined(CONFIG_CMD_BSP)
  35. static int got_fifoirq;
  36. static int got_hcirq;
  37. int fpga_interrupt(u32 arg)
  38. {
  39. pmc440_fpga_t *fpga = (pmc440_fpga_t *)arg;
  40. int rc = -1; /* not for us */
  41. u32 status = FPGA_IN32(&fpga->status);
  42. /* check for interrupt from fifo module */
  43. if (status & STATUS_FIFO_ISF) {
  44. /* disable this int source */
  45. FPGA_OUT32(&fpga->hostctrl, HOSTCTRL_FIFOIE_GATE);
  46. rc = 0;
  47. got_fifoirq = 1; /* trigger backend */
  48. }
  49. if (status & STATUS_HOST_ISF) {
  50. FPGA_OUT32(&fpga->hostctrl, HOSTCTRL_HCINT_GATE);
  51. rc = 0;
  52. got_hcirq = 1;
  53. }
  54. return rc;
  55. }
  56. int do_waithci(cmd_tbl_t *cmdtp, int flag, int argc, char *argv[])
  57. {
  58. pmc440_fpga_t *fpga = (pmc440_fpga_t *)FPGA_BA;
  59. got_hcirq = 0;
  60. FPGA_CLRBITS(&fpga->ctrla, CTRL_HOST_IE);
  61. FPGA_OUT32(&fpga->hostctrl, HOSTCTRL_HCINT_GATE);
  62. irq_install_handler(IRQ0_FPGA,
  63. (interrupt_handler_t *)fpga_interrupt,
  64. fpga);
  65. FPGA_SETBITS(&fpga->ctrla, CTRL_HOST_IE);
  66. while (!got_hcirq) {
  67. /* Abort if ctrl-c was pressed */
  68. if (ctrlc()) {
  69. puts("\nAbort\n");
  70. break;
  71. }
  72. }
  73. if (got_hcirq)
  74. printf("Got interrupt!\n");
  75. FPGA_CLRBITS(&fpga->ctrla, CTRL_HOST_IE);
  76. irq_free_handler(IRQ0_FPGA);
  77. return 0;
  78. }
  79. U_BOOT_CMD(
  80. waithci, 1, 1, do_waithci,
  81. "waithci - Wait for host control interrupt\n",
  82. NULL
  83. );
  84. void dump_fifo(pmc440_fpga_t *fpga, int f, int *n)
  85. {
  86. u32 ctrl;
  87. while (!((ctrl = FPGA_IN32(&fpga->fifo[f].ctrl)) & FIFO_EMPTY)) {
  88. printf("%5d %d %3d %08x",
  89. (*n)++, f, ctrl & (FIFO_LEVEL_MASK | FIFO_FULL),
  90. FPGA_IN32(&fpga->fifo[f].data));
  91. if (ctrl & FIFO_OVERFLOW) {
  92. printf(" OVERFLOW\n");
  93. FPGA_CLRBITS(&fpga->fifo[f].ctrl, FIFO_OVERFLOW);
  94. } else
  95. printf("\n");
  96. }
  97. }
  98. int do_fifo(cmd_tbl_t *cmdtp, int flag, int argc, char *argv[])
  99. {
  100. pmc440_fpga_t *fpga = (pmc440_fpga_t *)FPGA_BA;
  101. int i;
  102. int n = 0;
  103. u32 ctrl, data, f;
  104. char str[] = "\\|/-";
  105. int abort = 0;
  106. int count = 0;
  107. int count2 = 0;
  108. switch (argc) {
  109. case 1:
  110. /* print all fifos status information */
  111. printf("fifo level status\n");
  112. printf("______________________________\n");
  113. for (i=0; i<FIFO_COUNT; i++) {
  114. ctrl = FPGA_IN32(&fpga->fifo[i].ctrl);
  115. printf(" %d %3d %s%s%s %s\n",
  116. i, ctrl & (FIFO_LEVEL_MASK | FIFO_FULL),
  117. ctrl & FIFO_FULL ? "FULL " : "",
  118. ctrl & FIFO_EMPTY ? "EMPTY " : "",
  119. ctrl & (FIFO_FULL|FIFO_EMPTY) ? "" : "NOT EMPTY",
  120. ctrl & FIFO_OVERFLOW ? "OVERFLOW" : "");
  121. }
  122. break;
  123. case 2:
  124. /* completely read out fifo 'n' */
  125. if (!strcmp(argv[1],"read")) {
  126. printf(" # fifo level data\n");
  127. printf("______________________________\n");
  128. for (i=0; i<FIFO_COUNT; i++)
  129. dump_fifo(fpga, i, &n);
  130. } else if (!strcmp(argv[1],"wait")) {
  131. got_fifoirq = 0;
  132. irq_install_handler(IRQ0_FPGA,
  133. (interrupt_handler_t *)fpga_interrupt,
  134. fpga);
  135. printf(" # fifo level data\n");
  136. printf("______________________________\n");
  137. /* enable all fifo interrupts */
  138. FPGA_OUT32(&fpga->hostctrl,
  139. HOSTCTRL_FIFOIE_GATE | HOSTCTRL_FIFOIE_FLAG);
  140. for (i=0; i<FIFO_COUNT; i++) {
  141. /* enable interrupts from all fifos */
  142. FPGA_SETBITS(&fpga->fifo[i].ctrl, FIFO_IE);
  143. }
  144. while (1) {
  145. /* wait loop */
  146. while (!got_fifoirq) {
  147. count++;
  148. if (!(count % 100)) {
  149. count2++;
  150. putc(0x08); /* backspace */
  151. putc(str[count2 % 4]);
  152. }
  153. /* Abort if ctrl-c was pressed */
  154. if ((abort = ctrlc())) {
  155. puts("\nAbort\n");
  156. break;
  157. }
  158. udelay(1000);
  159. }
  160. if (abort)
  161. break;
  162. /* simple fifo backend */
  163. if (got_fifoirq) {
  164. for (i=0; i<FIFO_COUNT; i++)
  165. dump_fifo(fpga, i, &n);
  166. got_fifoirq = 0;
  167. /* unmask global fifo irq */
  168. FPGA_OUT32(&fpga->hostctrl,
  169. HOSTCTRL_FIFOIE_GATE | HOSTCTRL_FIFOIE_FLAG);
  170. }
  171. }
  172. /* disable all fifo interrupts */
  173. FPGA_OUT32(&fpga->hostctrl, HOSTCTRL_FIFOIE_GATE);
  174. for (i=0; i<FIFO_COUNT; i++)
  175. FPGA_CLRBITS(&fpga->fifo[i].ctrl, FIFO_IE);
  176. irq_free_handler(IRQ0_FPGA);
  177. } else {
  178. printf("Usage:\nfifo %s\n", cmdtp->help);
  179. return 1;
  180. }
  181. break;
  182. case 4:
  183. case 5:
  184. if (!strcmp(argv[1],"write")) {
  185. /* get fifo number or fifo address */
  186. f = simple_strtoul(argv[2], NULL, 16);
  187. /* data paramter */
  188. data = simple_strtoul(argv[3], NULL, 16);
  189. /* get optional count parameter */
  190. n = 1;
  191. if (argc >= 5)
  192. n = (int)simple_strtoul(argv[4], NULL, 10);
  193. if (f < FIFO_COUNT) {
  194. printf("writing %d x %08x to fifo %d\n",
  195. n, data, f);
  196. for (i=0; i<n; i++)
  197. FPGA_OUT32(&fpga->fifo[f].data, data);
  198. } else {
  199. printf("writing %d x %08x to fifo port at address %08x\n",
  200. n, data, f);
  201. for (i=0; i<n; i++)
  202. out32(f, data);
  203. }
  204. } else {
  205. printf("Usage:\nfifo %s\n", cmdtp->help);
  206. return 1;
  207. }
  208. break;
  209. default:
  210. printf("Usage:\nfifo %s\n", cmdtp->help);
  211. return 1;
  212. }
  213. return 0;
  214. }
  215. U_BOOT_CMD(
  216. fifo, 5, 1, do_fifo,
  217. "fifo - Fifo module operations\n",
  218. "wait\nfifo read\n"
  219. "fifo write fifo(0..3) data [cnt=1]\n"
  220. "fifo write address(>=4) data [cnt=1]\n"
  221. " - without arguments: print all fifo's status\n"
  222. " - with 'wait' argument: interrupt driven read from all fifos\n"
  223. " - with 'read' argument: read current contents from all fifos\n"
  224. " - with 'write' argument: write 'data' 'cnt' times to 'fifo' or 'address'\n"
  225. );
  226. int do_setup_bootstrap_eeprom(cmd_tbl_t *cmdtp, int flag, int argc, char *argv[])
  227. {
  228. ulong sdsdp[5];
  229. ulong delay;
  230. int count=16;
  231. if (argc < 2) {
  232. printf("Usage:\nsbe %s\n", cmdtp->help);
  233. return -1;
  234. }
  235. if (argc > 1) {
  236. if (!strcmp(argv[1], "400")) {
  237. /* PLB=133MHz, PLB/PCI=4 */
  238. printf("Bootstrapping for 400MHz\n");
  239. sdsdp[0]=0x8678624e;
  240. sdsdp[1]=0x0947a030;
  241. sdsdp[2]=0x40082350;
  242. sdsdp[3]=0x0d050000;
  243. } else if (!strcmp(argv[1], "533")) {
  244. /* PLB=133MHz, PLB/PCI=3 */
  245. printf("Bootstrapping for 533MHz\n");
  246. sdsdp[0]=0x87788252;
  247. sdsdp[1]=0x095fa030;
  248. sdsdp[2]=0x40082350;
  249. sdsdp[3]=0x0d050000;
  250. } else if (!strcmp(argv[1], "667")) {
  251. /* PLB=133MHz, PLB/PCI=4 */
  252. printf("Bootstrapping for 667MHz\n");
  253. sdsdp[0]=0x8778a256;
  254. sdsdp[1]=0x0947a030;
  255. sdsdp[2]=0x40082350;
  256. sdsdp[3]=0x0d050000;
  257. } else if (!strcmp(argv[1], "test")) {
  258. /* TODO: this will replace the 667 MHz config above.
  259. * But it needs some more testing on a real 667 MHz CPU.
  260. */
  261. printf("Bootstrapping for test (667MHz PLB=133PLB PLB/PCI=3)\n");
  262. sdsdp[0]=0x8778a256;
  263. sdsdp[1]=0x095fa030;
  264. sdsdp[2]=0x40082350;
  265. sdsdp[3]=0x0d050000;
  266. } else {
  267. printf("Usage:\nsbe %s\n", cmdtp->help);
  268. return -1;
  269. }
  270. }
  271. if (argc > 2) {
  272. sdsdp[4] = 0;
  273. if (argv[2][0]=='1')
  274. sdsdp[4]=0x19750100;
  275. else if (argv[2][0]=='0')
  276. sdsdp[4]=0x19750000;
  277. if (sdsdp[4])
  278. count += 4;
  279. }
  280. if (argc > 3) {
  281. delay = simple_strtoul(argv[3], NULL, 10);
  282. if (delay > 20)
  283. delay = 20;
  284. sdsdp[4] |= delay;
  285. }
  286. printf("Writing boot EEPROM ...\n");
  287. if (bootstrap_eeprom_write(CFG_I2C_BOOT_EEPROM_ADDR,
  288. 0, (uchar*)sdsdp, count) != 0)
  289. printf("bootstrap_eeprom_write failed\n");
  290. else
  291. printf("done (dump via 'i2c md 52 0.1 14')\n");
  292. return 0;
  293. }
  294. U_BOOT_CMD(
  295. sbe, 4, 0, do_setup_bootstrap_eeprom,
  296. "sbe - setup bootstrap eeprom\n",
  297. "<cpufreq:400|533|667> [<console-uart:0|1> [<bringup delay (0..20s)>]]"
  298. );
  299. #if defined(CONFIG_PRAM)
  300. #include <environment.h>
  301. extern env_t *env_ptr;
  302. int do_painit(cmd_tbl_t *cmdtp, int flag, int argc, char *argv[])
  303. {
  304. u32 memsize;
  305. u32 pram, env_base;
  306. char *v;
  307. u32 param;
  308. ulong *lptr;
  309. memsize = gd->bd->bi_memsize;
  310. v = getenv("pram");
  311. if (v)
  312. pram = simple_strtoul(v, NULL, 10);
  313. else {
  314. printf("Error: pram undefined. Please define pram in KiB\n");
  315. return 1;
  316. }
  317. param = memsize - (pram << 10);
  318. printf("PARAM: @%08x\n", param);
  319. memset((void*)param, 0, (pram << 10));
  320. env_base = memsize - 4096 - ((CFG_ENV_SIZE + 4096) & ~(4096-1));
  321. memcpy((void*)env_base, env_ptr, CFG_ENV_SIZE);
  322. lptr = (ulong*)memsize;
  323. *(--lptr) = CFG_ENV_SIZE;
  324. *(--lptr) = memsize - env_base;
  325. *(--lptr) = crc32(0, (void*)(memsize - 0x08), 0x08);
  326. *(--lptr) = 0;
  327. /* make sure data can be accessed through PCI */
  328. flush_dcache_range(param, param + (pram << 10) - 1);
  329. return 0;
  330. }
  331. U_BOOT_CMD(
  332. painit, 1, 1, do_painit,
  333. "painit - prepare PciAccess system\n",
  334. NULL
  335. );
  336. #endif /* CONFIG_PRAM */
  337. int do_selfreset(cmd_tbl_t *cmdtp, int flag, int argc, char *argv[])
  338. {
  339. if (argc > 1) {
  340. if (argv[1][0] == '0') {
  341. /* assert */
  342. printf("self-reset# asserted\n");
  343. out_be32((void*)GPIO0_TCR,
  344. in_be32((void*)GPIO0_TCR) | GPIO0_SELF_RST);
  345. } else {
  346. /* deassert */
  347. printf("self-reset# deasserted\n");
  348. out_be32((void*)GPIO0_TCR,
  349. in_be32((void*)GPIO0_TCR) & ~GPIO0_SELF_RST);
  350. }
  351. } else {
  352. printf("self-reset# is %s\n",
  353. in_be32((void*)GPIO0_TCR) & GPIO0_SELF_RST ?
  354. "active" : "inactive");
  355. }
  356. return 0;
  357. }
  358. U_BOOT_CMD(
  359. selfreset, 2, 1, do_selfreset,
  360. "selfreset- assert self-reset# signal\n",
  361. NULL
  362. );
  363. int do_resetout(cmd_tbl_t *cmdtp, int flag, int argc, char *argv[])
  364. {
  365. pmc440_fpga_t *fpga = (pmc440_fpga_t *)FPGA_BA;
  366. /* requiers bootet FPGA and PLD_IOEN_N active */
  367. if (in_be32((void*)GPIO1_OR) & GPIO1_IOEN_N) {
  368. printf("Error: resetout requires a bootet FPGA\n");
  369. return -1;
  370. }
  371. if (argc > 1) {
  372. if (argv[1][0] == '0') {
  373. /* assert */
  374. printf("PMC-RESETOUT# asserted\n");
  375. FPGA_OUT32(&fpga->hostctrl,
  376. HOSTCTRL_PMCRSTOUT_GATE);
  377. } else {
  378. /* deassert */
  379. printf("PMC-RESETOUT# deasserted\n");
  380. FPGA_OUT32(&fpga->hostctrl,
  381. HOSTCTRL_PMCRSTOUT_GATE | HOSTCTRL_PMCRSTOUT_FLAG);
  382. }
  383. } else {
  384. printf("PMC-RESETOUT# is %s\n",
  385. FPGA_IN32(&fpga->hostctrl) & HOSTCTRL_PMCRSTOUT_FLAG ?
  386. "inactive" : "active");
  387. }
  388. return 0;
  389. }
  390. U_BOOT_CMD(
  391. resetout, 2, 1, do_resetout,
  392. "resetout - assert PMC-RESETOUT# signal\n",
  393. NULL
  394. );
  395. int do_inta(cmd_tbl_t *cmdtp, int flag, int argc, char *argv[])
  396. {
  397. if (is_monarch()) {
  398. printf("This command is only supported in non-monarch mode\n");
  399. return -1;
  400. }
  401. if (argc > 1) {
  402. if (argv[1][0] == '0') {
  403. /* assert */
  404. printf("inta# asserted\n");
  405. out_be32((void*)GPIO1_TCR,
  406. in_be32((void*)GPIO1_TCR) | GPIO1_INTA_FAKE);
  407. } else {
  408. /* deassert */
  409. printf("inta# deasserted\n");
  410. out_be32((void*)GPIO1_TCR,
  411. in_be32((void*)GPIO1_TCR) & ~GPIO1_INTA_FAKE);
  412. }
  413. } else {
  414. printf("inta# is %s\n", in_be32((void*)GPIO1_TCR) & GPIO1_INTA_FAKE ? "active" : "inactive");
  415. }
  416. return 0;
  417. }
  418. U_BOOT_CMD(
  419. inta, 2, 1, do_inta,
  420. "inta - Assert/Deassert or query INTA# state in non-monarch mode\n",
  421. NULL
  422. );
  423. /* test-only */
  424. int do_pmm(cmd_tbl_t *cmdtp, int flag, int argc, char *argv[])
  425. {
  426. ulong pciaddr;
  427. if (argc > 1) {
  428. pciaddr = simple_strtoul(argv[1], NULL, 16);
  429. pciaddr &= 0xf0000000;
  430. /* map PCI address at 0xc0000000 in PLB space */
  431. out32r(PCIX0_PMM1MA, 0x00000000); /* PMM1 Mask/Attribute - disabled b4 setting */
  432. out32r(PCIX0_PMM1LA, 0xc0000000); /* PMM1 Local Address */
  433. out32r(PCIX0_PMM1PCILA, pciaddr); /* PMM1 PCI Low Address */
  434. out32r(PCIX0_PMM1PCIHA, 0x00000000); /* PMM1 PCI High Address */
  435. out32r(PCIX0_PMM1MA, 0xf0000001); /* 256MB + No prefetching, and enable region */
  436. } else {
  437. printf("Usage:\npmm %s\n", cmdtp->help);
  438. }
  439. return 0;
  440. }
  441. U_BOOT_CMD(
  442. pmm, 2, 1, do_pmm,
  443. "pmm - Setup pmm[1] registers\n",
  444. "<pciaddr> (pciaddr will be aligned to 256MB)\n"
  445. );
  446. #if defined(CFG_EEPROM_WREN)
  447. int do_eep_wren(cmd_tbl_t *cmdtp, int flag, int argc, char *argv[])
  448. {
  449. int query = argc == 1;
  450. int state = 0;
  451. if (query) {
  452. /* Query write access state. */
  453. state = eeprom_write_enable(CFG_I2C_EEPROM_ADDR, -1);
  454. if (state < 0) {
  455. puts("Query of write access state failed.\n");
  456. } else {
  457. printf("Write access for device 0x%0x is %sabled.\n",
  458. CFG_I2C_EEPROM_ADDR, state ? "en" : "dis");
  459. state = 0;
  460. }
  461. } else {
  462. if ('0' == argv[1][0]) {
  463. /* Disable write access. */
  464. state = eeprom_write_enable(CFG_I2C_EEPROM_ADDR, 0);
  465. } else {
  466. /* Enable write access. */
  467. state = eeprom_write_enable(CFG_I2C_EEPROM_ADDR, 1);
  468. }
  469. if (state < 0) {
  470. puts("Setup of write access state failed.\n");
  471. }
  472. }
  473. return state;
  474. }
  475. U_BOOT_CMD(eepwren, 2, 0, do_eep_wren,
  476. "eepwren - Enable / disable / query EEPROM write access\n",
  477. NULL);
  478. #endif /* #if defined(CFG_EEPROM_WREN) */
  479. #endif /* CONFIG_CMD_BSP */