cmd_universe.c 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384
  1. /*
  2. * (C) Copyright 2003 Stefan Roese, stefan.roese@esd-electronics.com
  3. *
  4. * See file CREDITS for list of people who contributed to this
  5. * project.
  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. #include <common.h>
  23. #include <command.h>
  24. #include <malloc.h>
  25. #include <asm/io.h>
  26. #include <pci.h>
  27. #include <universe.h>
  28. #define PCI_VENDOR PCI_VENDOR_ID_TUNDRA
  29. #define PCI_DEVICE PCI_DEVICE_ID_TUNDRA_CA91C042
  30. typedef struct _UNI_DEV UNI_DEV;
  31. struct _UNI_DEV {
  32. int bus;
  33. pci_dev_t busdevfn;
  34. UNIVERSE *uregs;
  35. unsigned int pci_bs;
  36. };
  37. static UNI_DEV *dev;
  38. int universe_init(void)
  39. {
  40. int j, result;
  41. pci_dev_t busdevfn;
  42. unsigned int val;
  43. busdevfn = pci_find_device(PCI_VENDOR, PCI_DEVICE, 0);
  44. if (busdevfn == -1) {
  45. puts("No Tundra Universe found!\n");
  46. return -1;
  47. }
  48. /* Lets turn Latency off */
  49. pci_write_config_dword(busdevfn, 0x0c, 0);
  50. dev = malloc(sizeof(*dev));
  51. if (NULL == dev) {
  52. puts("UNIVERSE: No memory!\n");
  53. result = -1;
  54. goto break_20;
  55. }
  56. memset(dev, 0, sizeof(*dev));
  57. dev->busdevfn = busdevfn;
  58. pci_read_config_dword(busdevfn, PCI_BASE_ADDRESS_1, &val);
  59. if (val & 1) {
  60. pci_read_config_dword(busdevfn, PCI_BASE_ADDRESS_0, &val);
  61. }
  62. val &= ~0xf;
  63. dev->uregs = (UNIVERSE *)val;
  64. debug ("UNIVERSE-Base : %p\n", dev->uregs);
  65. /* check mapping */
  66. debug (" Read via mapping, PCI_ID = %08X\n", readl(&dev->uregs->pci_id));
  67. if (((PCI_DEVICE <<16) | PCI_VENDOR) != readl(&dev->uregs->pci_id)) {
  68. printf ("UNIVERSE: Cannot read PCI-ID via Mapping: %08x\n",
  69. readl(&dev->uregs->pci_id));
  70. result = -1;
  71. goto break_30;
  72. }
  73. debug ("PCI_BS = %08X\n", readl(&dev->uregs->pci_bs));
  74. dev->pci_bs = readl(&dev->uregs->pci_bs);
  75. /* turn off windows */
  76. for (j=0; j <4; j ++) {
  77. writel(0x00800000, &dev->uregs->lsi[j].ctl);
  78. writel(0x00800000, &dev->uregs->vsi[j].ctl);
  79. }
  80. /*
  81. * Write to Misc Register
  82. * Set VME Bus Time-out
  83. * Arbitration Mode
  84. * DTACK Enable
  85. */
  86. writel(0x15040000 | (readl(&dev->uregs->misc_ctl) & 0x00020000), &dev->uregs->misc_ctl);
  87. if (readl(&dev->uregs->misc_ctl) & 0x00020000) {
  88. debug ("System Controller!\n"); /* test-only */
  89. } else {
  90. debug ("Not System Controller!\n"); /* test-only */
  91. }
  92. /*
  93. * Lets turn off interrupts
  94. */
  95. writel(0x00000000,&dev->uregs->lint_en); /* Disable interrupts in the Universe first */
  96. writel(0x0000FFFF,&dev->uregs->lint_stat); /* Clear Any Pending Interrupts */
  97. eieio();
  98. writel(0x0000, &dev->uregs->lint_map0); /* Map all ints to 0 */
  99. writel(0x0000, &dev->uregs->lint_map1); /* Map all ints to 0 */
  100. eieio();
  101. return 0;
  102. break_30:
  103. free(dev);
  104. break_20:
  105. return result;
  106. }
  107. /*
  108. * Create pci slave window (access: pci -> vme)
  109. */
  110. int universe_pci_slave_window(unsigned int pciAddr, unsigned int vmeAddr, int size, int vam, int pms, int vdw)
  111. {
  112. int result, i;
  113. unsigned int ctl = 0;
  114. if (NULL == dev) {
  115. result = -1;
  116. goto exit_10;
  117. }
  118. for (i = 0; i < 4; i++) {
  119. if (0x00800000 == readl(&dev->uregs->lsi[i].ctl))
  120. break;
  121. }
  122. if (i == 4) {
  123. printf ("universe: No Image available\n");
  124. result = -1;
  125. goto exit_10;
  126. }
  127. debug ("universe: Using image %d\n", i);
  128. writel(pciAddr , &dev->uregs->lsi[i].bs);
  129. writel((pciAddr + size), &dev->uregs->lsi[i].bd);
  130. writel((vmeAddr - pciAddr), &dev->uregs->lsi[i].to);
  131. switch (vam & VME_AM_Axx) {
  132. case VME_AM_A16:
  133. ctl = 0x00000000;
  134. break;
  135. case VME_AM_A24:
  136. ctl = 0x00010000;
  137. break;
  138. case VME_AM_A32:
  139. ctl = 0x00020000;
  140. break;
  141. }
  142. switch (vam & VME_AM_Mxx) {
  143. case VME_AM_DATA:
  144. ctl |= 0x00000000;
  145. break;
  146. case VME_AM_PROG:
  147. ctl |= 0x00008000;
  148. break;
  149. }
  150. if (vam & VME_AM_SUP) {
  151. ctl |= 0x00001000;
  152. }
  153. switch (vdw & VME_FLAG_Dxx) {
  154. case VME_FLAG_D8:
  155. ctl |= 0x00000000;
  156. break;
  157. case VME_FLAG_D16:
  158. ctl |= 0x00400000;
  159. break;
  160. case VME_FLAG_D32:
  161. ctl |= 0x00800000;
  162. break;
  163. }
  164. switch (pms & PCI_MS_Mxx) {
  165. case PCI_MS_MEM:
  166. ctl |= 0x00000000;
  167. break;
  168. case PCI_MS_IO:
  169. ctl |= 0x00000001;
  170. break;
  171. case PCI_MS_CONFIG:
  172. ctl |= 0x00000002;
  173. break;
  174. }
  175. ctl |= 0x80000000; /* enable */
  176. writel(ctl, &dev->uregs->lsi[i].ctl);
  177. debug ("universe: window-addr=%p\n", &dev->uregs->lsi[i].ctl);
  178. debug ("universe: pci slave window[%d] ctl=%08x\n", i, readl(&dev->uregs->lsi[i].ctl));
  179. debug ("universe: pci slave window[%d] bs=%08x\n", i, readl(&dev->uregs->lsi[i].bs));
  180. debug ("universe: pci slave window[%d] bd=%08x\n", i, readl(&dev->uregs->lsi[i].bd));
  181. debug ("universe: pci slave window[%d] to=%08x\n", i, readl(&dev->uregs->lsi[i].to));
  182. return 0;
  183. exit_10:
  184. return -result;
  185. }
  186. /*
  187. * Create vme slave window (access: vme -> pci)
  188. */
  189. int universe_vme_slave_window(unsigned int vmeAddr, unsigned int pciAddr, int size, int vam, int pms)
  190. {
  191. int result, i;
  192. unsigned int ctl = 0;
  193. if (NULL == dev) {
  194. result = -1;
  195. goto exit_10;
  196. }
  197. for (i = 0; i < 4; i++) {
  198. if (0x00800000 == readl(&dev->uregs->vsi[i].ctl))
  199. break;
  200. }
  201. if (i == 4) {
  202. printf ("universe: No Image available\n");
  203. result = -1;
  204. goto exit_10;
  205. }
  206. debug ("universe: Using image %d\n", i);
  207. writel(vmeAddr , &dev->uregs->vsi[i].bs);
  208. writel((vmeAddr + size), &dev->uregs->vsi[i].bd);
  209. writel((pciAddr - vmeAddr), &dev->uregs->vsi[i].to);
  210. switch (vam & VME_AM_Axx) {
  211. case VME_AM_A16:
  212. ctl = 0x00000000;
  213. break;
  214. case VME_AM_A24:
  215. ctl = 0x00010000;
  216. break;
  217. case VME_AM_A32:
  218. ctl = 0x00020000;
  219. break;
  220. }
  221. switch (vam & VME_AM_Mxx) {
  222. case VME_AM_DATA:
  223. ctl |= 0x00000000;
  224. break;
  225. case VME_AM_PROG:
  226. ctl |= 0x00800000;
  227. break;
  228. }
  229. if (vam & VME_AM_SUP) {
  230. ctl |= 0x00100000;
  231. }
  232. switch (pms & PCI_MS_Mxx) {
  233. case PCI_MS_MEM:
  234. ctl |= 0x00000000;
  235. break;
  236. case PCI_MS_IO:
  237. ctl |= 0x00000001;
  238. break;
  239. case PCI_MS_CONFIG:
  240. ctl |= 0x00000002;
  241. break;
  242. }
  243. ctl |= 0x80f00000; /* enable */
  244. writel(ctl, &dev->uregs->vsi[i].ctl);
  245. debug ("universe: window-addr=%p\n", &dev->uregs->vsi[i].ctl);
  246. debug ("universe: vme slave window[%d] ctl=%08x\n", i, readl(&dev->uregs->vsi[i].ctl));
  247. debug ("universe: vme slave window[%d] bs=%08x\n", i, readl(&dev->uregs->vsi[i].bs));
  248. debug ("universe: vme slave window[%d] bd=%08x\n", i, readl(&dev->uregs->vsi[i].bd));
  249. debug ("universe: vme slave window[%d] to=%08x\n", i, readl(&dev->uregs->vsi[i].to));
  250. return 0;
  251. exit_10:
  252. return -result;
  253. }
  254. /*
  255. * Tundra Universe configuration
  256. */
  257. int do_universe(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
  258. {
  259. ulong addr1 = 0, addr2 = 0, size = 0, vam = 0, pms = 0, vdw = 0;
  260. char cmd = 'x';
  261. /* get parameter */
  262. if (argc > 1)
  263. cmd = argv[1][0];
  264. if (argc > 2)
  265. addr1 = simple_strtoul(argv[2], NULL, 16);
  266. if (argc > 3)
  267. addr2 = simple_strtoul(argv[3], NULL, 16);
  268. if (argc > 4)
  269. size = simple_strtoul(argv[4], NULL, 16);
  270. if (argc > 5)
  271. vam = simple_strtoul(argv[5], NULL, 16);
  272. if (argc > 6)
  273. pms = simple_strtoul(argv[6], NULL, 16);
  274. if (argc > 7)
  275. vdw = simple_strtoul(argv[7], NULL, 16);
  276. switch (cmd) {
  277. case 'i': /* init */
  278. universe_init();
  279. break;
  280. case 'v': /* vme */
  281. printf("Configuring Universe VME Slave Window (VME->PCI):\n");
  282. printf(" vme=%08lx pci=%08lx size=%08lx vam=%02lx pms=%02lx\n",
  283. addr1, addr2, size, vam, pms);
  284. universe_vme_slave_window(addr1, addr2, size, vam, pms);
  285. break;
  286. case 'p': /* pci */
  287. printf("Configuring Universe PCI Slave Window (PCI->VME):\n");
  288. printf(" pci=%08lx vme=%08lx size=%08lx vam=%02lx pms=%02lx vdw=%02lx\n",
  289. addr1, addr2, size, vam, pms, vdw);
  290. universe_pci_slave_window(addr1, addr2, size, vam, pms, vdw);
  291. break;
  292. default:
  293. printf("Universe command %s not supported!\n", argv[1]);
  294. }
  295. return 0;
  296. }
  297. U_BOOT_CMD(
  298. universe, 8, 1, do_universe,
  299. "initialize and configure Turndra Universe",
  300. "init\n"
  301. " - initialize universe\n"
  302. "universe vme [vme_addr] [pci_addr] [size] [vam] [pms]\n"
  303. " - create vme slave window (access: vme->pci)\n"
  304. "universe pci [pci_addr] [vme_addr] [size] [vam] [pms] [vdw]\n"
  305. " - create pci slave window (access: pci->vme)\n"
  306. " [vam] = VMEbus Address-Modifier: 01 -> A16 Address Space\n"
  307. " 02 -> A24 Address Space\n"
  308. " 03 -> A32 Address Space\n"
  309. " 04 -> Supervisor AM Code\n"
  310. " 10 -> Data AM Code\n"
  311. " 20 -> Program AM Code\n"
  312. " [pms] = PCI Memory Space: 01 -> Memory Space\n"
  313. " 02 -> I/O Space\n"
  314. " 03 -> Configuration Space\n"
  315. " [vdw] = VMEbus Maximum Datawidth: 01 -> D8 Data Width\n"
  316. " 02 -> D16 Data Width\n"
  317. " 03 -> D32 Data Width"
  318. );