systemace.c 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303
  1. /*
  2. * Copyright (c) 2004 Picture Elements, Inc.
  3. * Stephen Williams (XXXXXXXXXXXXXXXX)
  4. *
  5. * SPDX-License-Identifier: GPL-2.0+
  6. */
  7. /*
  8. * The Xilinx SystemACE chip support is activated by defining
  9. * CONFIG_SYSTEMACE to turn on support, and CONFIG_SYS_SYSTEMACE_BASE
  10. * to set the base address of the device. This code currently
  11. * assumes that the chip is connected via a byte-wide bus.
  12. *
  13. * The CONFIG_SYSTEMACE also adds to fat support the device class
  14. * "ace" that allows the user to execute "fatls ace 0" and the
  15. * like. This works by making the systemace_get_dev function
  16. * available to cmd_fat.c:get_dev and filling in a block device
  17. * description that has all the bits needed for FAT support to
  18. * read sectors.
  19. *
  20. * According to Xilinx technical support, before accessing the
  21. * SystemACE CF you need to set the following control bits:
  22. * FORCECFGMODE : 1
  23. * CFGMODE : 0
  24. * CFGSTART : 0
  25. */
  26. #include <common.h>
  27. #include <command.h>
  28. #include <dm.h>
  29. #include <part.h>
  30. #include <asm/io.h>
  31. /*
  32. * The ace_readw and writew functions read/write 16bit words, but the
  33. * offset value is the BYTE offset as most used in the Xilinx
  34. * datasheet for the SystemACE chip. The CONFIG_SYS_SYSTEMACE_BASE is defined
  35. * to be the base address for the chip, usually in the local
  36. * peripheral bus.
  37. */
  38. static u32 base = CONFIG_SYS_SYSTEMACE_BASE;
  39. static u32 width = CONFIG_SYS_SYSTEMACE_WIDTH;
  40. static void ace_writew(u16 val, unsigned off)
  41. {
  42. if (width == 8) {
  43. #if !defined(__BIG_ENDIAN)
  44. writeb(val >> 8, base + off);
  45. writeb(val, base + off + 1);
  46. #else
  47. writeb(val, base + off);
  48. writeb(val >> 8, base + off + 1);
  49. #endif
  50. } else
  51. out16(base + off, val);
  52. }
  53. static u16 ace_readw(unsigned off)
  54. {
  55. if (width == 8) {
  56. #if !defined(__BIG_ENDIAN)
  57. return (readb(base + off) << 8) | readb(base + off + 1);
  58. #else
  59. return readb(base + off) | (readb(base + off + 1) << 8);
  60. #endif
  61. }
  62. return in16(base + off);
  63. }
  64. #ifndef CONFIG_BLK
  65. static struct blk_desc systemace_dev = { 0 };
  66. #endif
  67. static int get_cf_lock(void)
  68. {
  69. int retry = 10;
  70. /* CONTROLREG = LOCKREG */
  71. unsigned val = ace_readw(0x18);
  72. val |= 0x0002;
  73. ace_writew((val & 0xffff), 0x18);
  74. /* Wait for MPULOCK in STATUSREG[15:0] */
  75. while (!(ace_readw(0x04) & 0x0002)) {
  76. if (retry < 0)
  77. return -1;
  78. udelay(100000);
  79. retry -= 1;
  80. }
  81. return 0;
  82. }
  83. static void release_cf_lock(void)
  84. {
  85. unsigned val = ace_readw(0x18);
  86. val &= ~(0x0002);
  87. ace_writew((val & 0xffff), 0x18);
  88. }
  89. /*
  90. * This function is called (by dereferencing the block_read pointer in
  91. * the dev_desc) to read blocks of data. The return value is the
  92. * number of blocks read. A zero return indicates an error.
  93. */
  94. #ifdef CONFIG_BLK
  95. static unsigned long systemace_read(struct udevice *dev, unsigned long start,
  96. lbaint_t blkcnt, void *buffer)
  97. #else
  98. static unsigned long systemace_read(struct blk_desc *block_dev,
  99. unsigned long start, lbaint_t blkcnt,
  100. void *buffer)
  101. #endif
  102. {
  103. int retry;
  104. unsigned blk_countdown;
  105. unsigned char *dp = buffer;
  106. unsigned val;
  107. if (get_cf_lock() < 0) {
  108. unsigned status = ace_readw(0x04);
  109. /* If CFDETECT is false, card is missing. */
  110. if (!(status & 0x0010)) {
  111. printf("** CompactFlash card not present. **\n");
  112. return 0;
  113. }
  114. printf("**** ACE locked away from me (STATUSREG=%04x)\n",
  115. status);
  116. return 0;
  117. }
  118. #ifdef DEBUG_SYSTEMACE
  119. printf("... systemace read %lu sectors at %lu\n", blkcnt, start);
  120. #endif
  121. retry = 2000;
  122. for (;;) {
  123. val = ace_readw(0x04);
  124. /* If CFDETECT is false, card is missing. */
  125. if (!(val & 0x0010)) {
  126. printf("**** ACE CompactFlash not found.\n");
  127. release_cf_lock();
  128. return 0;
  129. }
  130. /* If RDYFORCMD, then we are ready to go. */
  131. if (val & 0x0100)
  132. break;
  133. if (retry < 0) {
  134. printf("**** SystemACE not ready.\n");
  135. release_cf_lock();
  136. return 0;
  137. }
  138. udelay(1000);
  139. retry -= 1;
  140. }
  141. /* The SystemACE can only transfer 256 sectors at a time, so
  142. limit the current chunk of sectors. The blk_countdown
  143. variable is the number of sectors left to transfer. */
  144. blk_countdown = blkcnt;
  145. while (blk_countdown > 0) {
  146. unsigned trans = blk_countdown;
  147. if (trans > 256)
  148. trans = 256;
  149. #ifdef DEBUG_SYSTEMACE
  150. printf("... transfer %lu sector in a chunk\n", trans);
  151. #endif
  152. /* Write LBA block address */
  153. ace_writew((start >> 0) & 0xffff, 0x10);
  154. ace_writew((start >> 16) & 0x0fff, 0x12);
  155. /* NOTE: in the Write Sector count below, a count of 0
  156. causes a transfer of 256, so &0xff gives the right
  157. value for whatever transfer count we want. */
  158. /* Write sector count | ReadMemCardData. */
  159. ace_writew((trans & 0xff) | 0x0300, 0x14);
  160. /*
  161. * For FPGA configuration via SystemACE is reset unacceptable
  162. * CFGDONE bit in STATUSREG is not set to 1.
  163. */
  164. #ifndef SYSTEMACE_CONFIG_FPGA
  165. /* Reset the configruation controller */
  166. val = ace_readw(0x18);
  167. val |= 0x0080;
  168. ace_writew(val, 0x18);
  169. #endif
  170. retry = trans * 16;
  171. while (retry > 0) {
  172. int idx;
  173. /* Wait for buffer to become ready. */
  174. while (!(ace_readw(0x04) & 0x0020)) {
  175. udelay(100);
  176. }
  177. /* Read 16 words of 2bytes from the sector buffer. */
  178. for (idx = 0; idx < 16; idx += 1) {
  179. unsigned short val = ace_readw(0x40);
  180. *dp++ = val & 0xff;
  181. *dp++ = (val >> 8) & 0xff;
  182. }
  183. retry -= 1;
  184. }
  185. /* Clear the configruation controller reset */
  186. val = ace_readw(0x18);
  187. val &= ~0x0080;
  188. ace_writew(val, 0x18);
  189. /* Count the blocks we transfer this time. */
  190. start += trans;
  191. blk_countdown -= trans;
  192. }
  193. release_cf_lock();
  194. return blkcnt;
  195. }
  196. #ifdef CONFIG_BLK
  197. static int systemace_bind(struct udevice *dev)
  198. {
  199. struct blk_desc *bdesc;
  200. struct udevice *bdev;
  201. int ret;
  202. ret = blk_create_devicef(dev, "systemace_blk", "blk", IF_TYPE_SYSTEMACE,
  203. -1, 512, 0, &bdev);
  204. if (ret) {
  205. debug("Cannot create block device\n");
  206. return ret;
  207. }
  208. bdesc = dev_get_uclass_platdata(bdev);
  209. bdesc->removable = 1;
  210. bdesc->part_type = PART_TYPE_UNKNOWN;
  211. bdesc->log2blksz = LOG2(bdesc->blksz);
  212. /* Ensure the correct bus mode (8/16 bits) gets enabled */
  213. ace_writew(width == 8 ? 0 : 0x0001, 0);
  214. return 0;
  215. }
  216. static const struct blk_ops systemace_blk_ops = {
  217. .read = systemace_read,
  218. };
  219. U_BOOT_DRIVER(systemace_blk) = {
  220. .name = "systemace_blk",
  221. .id = UCLASS_BLK,
  222. .ops = &systemace_blk_ops,
  223. .bind = systemace_bind,
  224. };
  225. #else
  226. static int systemace_get_dev(int dev, struct blk_desc **descp)
  227. {
  228. /* The first time through this, the systemace_dev object is
  229. not yet initialized. In that case, fill it in. */
  230. if (systemace_dev.blksz == 0) {
  231. systemace_dev.if_type = IF_TYPE_UNKNOWN;
  232. systemace_dev.devnum = 0;
  233. systemace_dev.part_type = PART_TYPE_UNKNOWN;
  234. systemace_dev.type = DEV_TYPE_HARDDISK;
  235. systemace_dev.blksz = 512;
  236. systemace_dev.log2blksz = LOG2(systemace_dev.blksz);
  237. systemace_dev.removable = 1;
  238. systemace_dev.block_read = systemace_read;
  239. /*
  240. * Ensure the correct bus mode (8/16 bits) gets enabled
  241. */
  242. ace_writew(width == 8 ? 0 : 0x0001, 0);
  243. part_init(&systemace_dev);
  244. }
  245. *descp = &systemace_dev;
  246. return 0;
  247. }
  248. U_BOOT_LEGACY_BLK(systemace) = {
  249. .if_typename = "ace",
  250. .if_type = IF_TYPE_SYSTEMACE,
  251. .max_devs = 1,
  252. .get_dev = systemace_get_dev,
  253. };
  254. #endif