cmd_universe.c 9.0 KB

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