cmd_universe.c 9.2 KB

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