flash.c 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359
  1. /*
  2. * (C) Copyright 2002
  3. * Kyle Harris, Nexus Technologies, Inc. kharris@nexus-tech.net
  4. *
  5. * (C) Copyright 2002
  6. * Sysgo Real-Time Solutions, GmbH <www.elinos.com>
  7. * Marius Groeger <mgroeger@sysgo.de>
  8. *
  9. * See file CREDITS for list of people who contributed to this
  10. * project.
  11. *
  12. * This program is free software; you can redistribute it and/or
  13. * modify it under the terms of the GNU General Public License as
  14. * published by the Free Software Foundation; either version 2 of
  15. * the License, or (at your option) any later version.
  16. *
  17. * This program is distributed in the hope that it will be useful,
  18. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  19. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  20. * GNU General Public License for more details.
  21. *
  22. * You should have received a copy of the GNU General Public License
  23. * along with this program; if not, write to the Free Software
  24. * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
  25. * MA 02111-1307 USA
  26. */
  27. #include <common.h>
  28. #define FLASH_BANK_SIZE 0x400000
  29. #define MAIN_SECT_SIZE 0x20000
  30. flash_info_t flash_info[CONFIG_SYS_MAX_FLASH_BANKS];
  31. /*-----------------------------------------------------------------------
  32. */
  33. ulong flash_init (void)
  34. {
  35. int i, j;
  36. ulong size = 0;
  37. for (i = 0; i < CONFIG_SYS_MAX_FLASH_BANKS; i++) {
  38. ulong flashbase = 0;
  39. flash_info[i].flash_id =
  40. (INTEL_MANUFACT & FLASH_VENDMASK) |
  41. (INTEL_ID_28F128J3 & FLASH_TYPEMASK);
  42. flash_info[i].size = FLASH_BANK_SIZE;
  43. flash_info[i].sector_count = CONFIG_SYS_MAX_FLASH_SECT;
  44. memset (flash_info[i].protect, 0, CONFIG_SYS_MAX_FLASH_SECT);
  45. switch (i) {
  46. case 0:
  47. flashbase = PHYS_FLASH_1;
  48. break;
  49. case 1:
  50. flashbase = PHYS_FLASH_2;
  51. break;
  52. default:
  53. panic ("configured too many flash banks!\n");
  54. break;
  55. }
  56. for (j = 0; j < flash_info[i].sector_count; j++) {
  57. flash_info[i].start[j] =
  58. flashbase + j * MAIN_SECT_SIZE;
  59. }
  60. size += flash_info[i].size;
  61. }
  62. /* Protect monitor and environment sectors
  63. */
  64. flash_protect (FLAG_PROTECT_SET,
  65. CONFIG_SYS_FLASH_BASE,
  66. CONFIG_SYS_FLASH_BASE + monitor_flash_len - 1,
  67. &flash_info[0]);
  68. flash_protect (FLAG_PROTECT_SET,
  69. CONFIG_ENV_ADDR,
  70. CONFIG_ENV_ADDR + CONFIG_ENV_SIZE - 1, &flash_info[0]);
  71. return size;
  72. }
  73. /*-----------------------------------------------------------------------
  74. */
  75. void flash_print_info (flash_info_t * info)
  76. {
  77. int i, j;
  78. for (j = 0; j < CONFIG_SYS_MAX_FLASH_BANKS; j++) {
  79. switch (info->flash_id & FLASH_VENDMASK) {
  80. case (INTEL_MANUFACT & FLASH_VENDMASK):
  81. printf ("Intel: ");
  82. break;
  83. default:
  84. printf ("Unknown Vendor ");
  85. break;
  86. }
  87. switch (info->flash_id & FLASH_TYPEMASK) {
  88. case (INTEL_ID_28F320J3A & FLASH_TYPEMASK):
  89. printf ("28F320J3A (32Mbit)\n");
  90. break;
  91. case (INTEL_ID_28F128J3 & FLASH_TYPEMASK):
  92. printf ("28F128J3 (128Mbit)\n");
  93. break;
  94. default:
  95. printf ("Unknown Chip Type\n");
  96. goto Done;
  97. break;
  98. }
  99. printf (" Size: %ld MB in %d Sectors\n",
  100. info->size >> 20, info->sector_count);
  101. printf (" Sector Start Addresses:");
  102. for (i = 0; i < info->sector_count; i++) {
  103. if ((i % 5) == 0) {
  104. printf ("\n ");
  105. }
  106. printf (" %08lX%s", info->start[i],
  107. info->protect[i] ? " (RO)" : " ");
  108. }
  109. printf ("\n");
  110. info++;
  111. }
  112. Done: ;
  113. }
  114. /*-----------------------------------------------------------------------
  115. */
  116. int flash_erase (flash_info_t * info, int s_first, int s_last)
  117. {
  118. int flag, prot, sect;
  119. int rc = ERR_OK;
  120. if (info->flash_id == FLASH_UNKNOWN)
  121. return ERR_UNKNOWN_FLASH_TYPE;
  122. if ((s_first < 0) || (s_first > s_last)) {
  123. return ERR_INVAL;
  124. }
  125. if ((info->flash_id & FLASH_VENDMASK) !=
  126. (INTEL_MANUFACT & FLASH_VENDMASK)) {
  127. return ERR_UNKNOWN_FLASH_VENDOR;
  128. }
  129. prot = 0;
  130. for (sect = s_first; sect <= s_last; ++sect) {
  131. if (info->protect[sect]) {
  132. prot++;
  133. }
  134. }
  135. if (prot)
  136. return ERR_PROTECTED;
  137. /*
  138. * Disable interrupts which might cause a timeout
  139. * here. Remember that our exception vectors are
  140. * at address 0 in the flash, and we don't want a
  141. * (ticker) exception to happen while the flash
  142. * chip is in programming mode.
  143. */
  144. flag = disable_interrupts ();
  145. /* Start erase on unprotected sectors */
  146. for (sect = s_first; sect <= s_last && !ctrlc (); sect++) {
  147. printf ("Erasing sector %2d ... ", sect);
  148. /* arm simple, non interrupt dependent timer */
  149. reset_timer_masked ();
  150. if (info->protect[sect] == 0) { /* not protected */
  151. vu_short *addr = (vu_short *) (info->start[sect]);
  152. *addr = 0x20; /* erase setup */
  153. *addr = 0xD0; /* erase confirm */
  154. while ((*addr & 0x80) != 0x80) {
  155. if (get_timer_masked () >
  156. CONFIG_SYS_FLASH_ERASE_TOUT) {
  157. *addr = 0xB0; /* suspend erase */
  158. *addr = 0xFF; /* reset to read mode */
  159. rc = ERR_TIMOUT;
  160. goto outahere;
  161. }
  162. }
  163. /* clear status register command */
  164. *addr = 0x50;
  165. /* reset to read mode */
  166. *addr = 0xFF;
  167. }
  168. printf ("ok.\n");
  169. }
  170. if (ctrlc ())
  171. printf ("User Interrupt!\n");
  172. outahere:
  173. /* allow flash to settle - wait 10 ms */
  174. udelay_masked (10000);
  175. if (flag)
  176. enable_interrupts ();
  177. return rc;
  178. }
  179. /*-----------------------------------------------------------------------
  180. * Copy memory to flash
  181. */
  182. static int write_word (flash_info_t * info, ulong dest, ushort data)
  183. {
  184. vu_short *addr = (vu_short *) dest, val;
  185. int rc = ERR_OK;
  186. int flag;
  187. /* Check if Flash is (sufficiently) erased
  188. */
  189. if ((*addr & data) != data)
  190. return ERR_NOT_ERASED;
  191. /*
  192. * Disable interrupts which might cause a timeout
  193. * here. Remember that our exception vectors are
  194. * at address 0 in the flash, and we don't want a
  195. * (ticker) exception to happen while the flash
  196. * chip is in programming mode.
  197. */
  198. flag = disable_interrupts ();
  199. /* clear status register command */
  200. *addr = 0x50;
  201. /* program set-up command */
  202. *addr = 0x40;
  203. /* latch address/data */
  204. *addr = data;
  205. /* arm simple, non interrupt dependent timer */
  206. reset_timer_masked ();
  207. /* wait while polling the status register */
  208. while (((val = *addr) & 0x80) != 0x80) {
  209. if (get_timer_masked () > CONFIG_SYS_FLASH_WRITE_TOUT) {
  210. rc = ERR_TIMOUT;
  211. /* suspend program command */
  212. *addr = 0xB0;
  213. goto outahere;
  214. }
  215. }
  216. if (val & 0x1A) { /* check for error */
  217. printf ("\nFlash write error %02x at address %08lx\n",
  218. (int) val, (unsigned long) dest);
  219. if (val & (1 << 3)) {
  220. printf ("Voltage range error.\n");
  221. rc = ERR_PROG_ERROR;
  222. goto outahere;
  223. }
  224. if (val & (1 << 1)) {
  225. printf ("Device protect error.\n");
  226. rc = ERR_PROTECTED;
  227. goto outahere;
  228. }
  229. if (val & (1 << 4)) {
  230. printf ("Programming error.\n");
  231. rc = ERR_PROG_ERROR;
  232. goto outahere;
  233. }
  234. rc = ERR_PROG_ERROR;
  235. goto outahere;
  236. }
  237. outahere:
  238. /* read array command */
  239. *addr = 0xFF;
  240. if (flag)
  241. enable_interrupts ();
  242. return rc;
  243. }
  244. /*-----------------------------------------------------------------------
  245. * Copy memory to flash.
  246. */
  247. int write_buff (flash_info_t * info, uchar * src, ulong addr, ulong cnt)
  248. {
  249. ulong cp, wp;
  250. ushort data;
  251. int l;
  252. int i, rc;
  253. wp = (addr & ~1); /* get lower word aligned address */
  254. /*
  255. * handle unaligned start bytes
  256. */
  257. if ((l = addr - wp) != 0) {
  258. data = 0;
  259. for (i = 0, cp = wp; i < l; ++i, ++cp) {
  260. data = (data >> 8) | (*(uchar *) cp << 8);
  261. }
  262. for (; i < 2 && cnt > 0; ++i) {
  263. data = (data >> 8) | (*src++ << 8);
  264. --cnt;
  265. ++cp;
  266. }
  267. for (; cnt == 0 && i < 2; ++i, ++cp) {
  268. data = (data >> 8) | (*(uchar *) cp << 8);
  269. }
  270. if ((rc = write_word (info, wp, data)) != 0) {
  271. return (rc);
  272. }
  273. wp += 2;
  274. }
  275. /*
  276. * handle word aligned part
  277. */
  278. while (cnt >= 2) {
  279. data = *((vu_short *) src);
  280. if ((rc = write_word (info, wp, data)) != 0) {
  281. return (rc);
  282. }
  283. src += 2;
  284. wp += 2;
  285. cnt -= 2;
  286. }
  287. if (cnt == 0) {
  288. return ERR_OK;
  289. }
  290. /*
  291. * handle unaligned tail bytes
  292. */
  293. data = 0;
  294. for (i = 0, cp = wp; i < 2 && cnt > 0; ++i, ++cp) {
  295. data = (data >> 8) | (*src++ << 8);
  296. --cnt;
  297. }
  298. for (; i < 2; ++i, ++cp) {
  299. data = (data >> 8) | (*(uchar *) cp << 8);
  300. }
  301. return write_word (info, wp, data);
  302. }