2 * m29f400bt.c: driver for programming JEDEC standard flash parts
5 * Copyright 2000 Silicon Integrated System Corporation
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License, or
10 * (at your option) any later version.
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.
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., 675 Mass Ave, Cambridge, MA 02139, USA.
27 #include "m29f400bt.h"
30 void toggle_ready_m29f400bt(volatile uint8_t *dst)
37 while (i++ < 0xFFFFFF) {
46 void data_polling_m29f400bt(volatile uint8_t *dst, uint8_t data)
53 while (i++ < 0xFFFFFF) {
61 void protect_m29f400bt(volatile uint8_t *bios)
63 *(volatile uint8_t *)(bios + 0xAAA) = 0xAA;
64 *(volatile uint8_t *)(bios + 0x555) = 0x55;
65 *(volatile uint8_t *)(bios + 0xAAA) = 0xA0;
70 void write_page_m29f400bt(volatile uint8_t *bios, uint8_t *src,
71 volatile uint8_t *dst, int page_size)
75 for (i = 0; i < page_size; i++) {
76 *(volatile uint8_t *)(bios + 0xAAA) = 0xAA;
77 *(volatile uint8_t *)(bios + 0x555) = 0x55;
78 *(volatile uint8_t *)(bios + 0xAAA) = 0xA0;
80 /* transfer data from source to destination */
82 //*(volatile char *) (bios) = 0xF0;
84 toggle_ready_m29f400bt(dst);
86 ("Value in the flash at address %p = %#x, want %#x\n",
87 (uint8_t *) (dst - bios), *dst, *src);
93 int probe_m29f400bt(struct flashchip *flash)
95 volatile uint8_t *bios = flash->virtual_memory;
98 *(volatile uint8_t *)(bios + 0xAAA) = 0xAA;
99 *(volatile uint8_t *)(bios + 0x555) = 0x55;
100 *(volatile uint8_t *)(bios + 0xAAA) = 0x90;
104 id1 = *(volatile uint8_t *)bios;
105 id2 = *(volatile uint8_t *)(bios + 0x02);
107 *(volatile uint8_t *)(bios + 0xAAA) = 0xAA;
108 *(volatile uint8_t *)(bios + 0x555) = 0x55;
109 *(volatile uint8_t *)(bios + 0xAAA) = 0xF0;
113 printf_debug("%s: id1 0x%x, id2 0x%x\n", __FUNCTION__, id1, id2);
115 if (id1 == flash->manufacture_id && id2 == flash->model_id)
121 int erase_m29f400bt(struct flashchip *flash)
123 volatile uint8_t *bios = flash->virtual_memory;
125 *(volatile uint8_t *)(bios + 0xAAA) = 0xAA;
126 *(volatile uint8_t *)(bios + 0x555) = 0x55;
127 *(volatile uint8_t *)(bios + 0xAAA) = 0x80;
129 *(volatile uint8_t *)(bios + 0xAAA) = 0xAA;
130 *(volatile uint8_t *)(bios + 0x555) = 0x55;
131 *(volatile uint8_t *)(bios + 0xAAA) = 0x10;
134 toggle_ready_m29f400bt(bios);
139 int block_erase_m29f400bt(volatile uint8_t *bios, volatile uint8_t *dst)
142 *(volatile uint8_t *)(bios + 0xAAA) = 0xAA;
143 *(volatile uint8_t *)(bios + 0x555) = 0x55;
144 *(volatile uint8_t *)(bios + 0xAAA) = 0x80;
146 *(volatile uint8_t *)(bios + 0xAAA) = 0xAA;
147 *(volatile uint8_t *)(bios + 0x555) = 0x55;
148 //*(volatile uint8_t *) (bios + 0xAAA) = 0x10;
152 toggle_ready_m29f400bt(bios);
157 int write_m29f400bt(struct flashchip *flash, uint8_t *buf)
160 int total_size = flash->total_size * 1024;
161 int page_size = flash->page_size;
162 volatile uint8_t *bios = flash->virtual_memory;
164 //erase_m29f400bt (flash);
165 printf("Programming Page:\n ");
166 /*********************************
167 *Pages for M29F400BT:
168 * 16 0x7c000 0x7ffff TOP
175 *---------------------------------
179 * 64 0x00000 0x0ffff BOTTOM
180 *********************************/
181 printf("total_size/page_size = %d\n", total_size / page_size);
182 for (i = 0; i < (total_size / page_size) - 1; i++) {
183 printf("%04d at address: 0x%08x\n", i, i * page_size);
184 block_erase_m29f400bt(bios, bios + i * page_size);
185 write_page_m29f400bt(bios, buf + i * page_size,
186 bios + i * page_size, page_size);
187 printf("\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b\b");
190 printf("%04d at address: 0x%08x\n", 7, 0x70000);
191 block_erase_m29f400bt(bios, bios + 0x70000);
192 write_page_m29f400bt(bios, buf + 0x70000, bios + 0x70000, 32 * 1024);
194 printf("%04d at address: 0x%08x\n", 8, 0x78000);
195 block_erase_m29f400bt(bios, bios + 0x78000);
196 write_page_m29f400bt(bios, buf + 0x78000, bios + 0x78000, 8 * 1024);
198 printf("%04d at address: 0x%08x\n", 9, 0x7a000);
199 block_erase_m29f400bt(bios, bios + 0x7a000);
200 write_page_m29f400bt(bios, buf + 0x7a000, bios + 0x7a000, 8 * 1024);
202 printf("%04d at address: 0x%08x\n", 10, 0x7c000);
203 block_erase_m29f400bt(bios, bios + 0x7c000);
204 write_page_m29f400bt(bios, buf + 0x7c000, bios + 0x7c000, 16 * 1024);
207 //protect_m29f400bt (bios);
212 int write_linuxbios_m29f400bt(struct flashchip *flash, uint8_t *buf)
214 volatile uint8_t *bios = flash->virtual_memory;
216 printf("Programming Page:\n ");
217 /*********************************
218 *Pages for M29F400BT:
219 * 16 0x7c000 0x7ffff TOP
226 *---------------------------------
230 * 64 0x00000 0x0ffff BOTTOM
231 *********************************/
232 printf("%04d at address: 0x%08x\n", 7, 0x00000);
233 block_erase_m29f400bt(bios, bios + 0x00000);
234 write_page_m29f400bt(bios, buf + 0x00000, bios + 0x00000, 64 * 1024);
236 printf("%04d at address: 0x%08x\n", 7, 0x10000);
237 block_erase_m29f400bt(bios, bios + 0x10000);
238 write_page_m29f400bt(bios, buf + 0x10000, bios + 0x10000, 64 * 1024);
240 printf("%04d at address: 0x%08x\n", 7, 0x20000);
241 block_erase_m29f400bt(bios, bios + 0x20000);
242 write_page_m29f400bt(bios, buf + 0x20000, bios + 0x20000, 64 * 1024);
244 printf("%04d at address: 0x%08x\n", 7, 0x30000);
245 block_erase_m29f400bt(bios, bios + 0x30000);
246 write_page_m29f400bt(bios, buf + 0x30000, bios + 0x30000, 64 * 1024);
249 //protect_m29f400bt (bios);