xref: /aosp_15_r20/external/flashrom/sst28sf040.c (revision 0d6140be3aa665ecc836e8907834fcd3e3b018fc)
1*0d6140beSAndroid Build Coastguard Worker /*
2*0d6140beSAndroid Build Coastguard Worker  * This file is part of the flashrom project.
3*0d6140beSAndroid Build Coastguard Worker  *
4*0d6140beSAndroid Build Coastguard Worker  * Copyright (C) 2000 Silicon Integrated System Corporation
5*0d6140beSAndroid Build Coastguard Worker  * Copyright (C) 2005 coresystems GmbH <[email protected]>
6*0d6140beSAndroid Build Coastguard Worker  * Copyright (C) 2009 Sean Nelson <[email protected]>
7*0d6140beSAndroid Build Coastguard Worker  *
8*0d6140beSAndroid Build Coastguard Worker  * This program is free software; you can redistribute it and/or modify
9*0d6140beSAndroid Build Coastguard Worker  * it under the terms of the GNU General Public License as published by
10*0d6140beSAndroid Build Coastguard Worker  * the Free Software Foundation; either version 2 of the License, or
11*0d6140beSAndroid Build Coastguard Worker  * (at your option) any later version.
12*0d6140beSAndroid Build Coastguard Worker  *
13*0d6140beSAndroid Build Coastguard Worker  * This program is distributed in the hope that it will be useful,
14*0d6140beSAndroid Build Coastguard Worker  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15*0d6140beSAndroid Build Coastguard Worker  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16*0d6140beSAndroid Build Coastguard Worker  * GNU General Public License for more details.
17*0d6140beSAndroid Build Coastguard Worker  */
18*0d6140beSAndroid Build Coastguard Worker 
19*0d6140beSAndroid Build Coastguard Worker #include "flash.h"
20*0d6140beSAndroid Build Coastguard Worker #include "chipdrivers.h"
21*0d6140beSAndroid Build Coastguard Worker 
22*0d6140beSAndroid Build Coastguard Worker #define AUTO_PG_ERASE1		0x20
23*0d6140beSAndroid Build Coastguard Worker #define AUTO_PG_ERASE2		0xD0
24*0d6140beSAndroid Build Coastguard Worker #define AUTO_PGRM		0x10
25*0d6140beSAndroid Build Coastguard Worker #define CHIP_ERASE		0x30
26*0d6140beSAndroid Build Coastguard Worker #define RESET			0xFF
27*0d6140beSAndroid Build Coastguard Worker #define READ_ID			0x90
28*0d6140beSAndroid Build Coastguard Worker 
protect_28sf040(struct flashctx * flash)29*0d6140beSAndroid Build Coastguard Worker int protect_28sf040(struct flashctx *flash)
30*0d6140beSAndroid Build Coastguard Worker {
31*0d6140beSAndroid Build Coastguard Worker 	chipaddr bios = flash->virtual_memory;
32*0d6140beSAndroid Build Coastguard Worker 
33*0d6140beSAndroid Build Coastguard Worker 	chip_readb(flash, bios + 0x1823);
34*0d6140beSAndroid Build Coastguard Worker 	chip_readb(flash, bios + 0x1820);
35*0d6140beSAndroid Build Coastguard Worker 	chip_readb(flash, bios + 0x1822);
36*0d6140beSAndroid Build Coastguard Worker 	chip_readb(flash, bios + 0x0418);
37*0d6140beSAndroid Build Coastguard Worker 	chip_readb(flash, bios + 0x041B);
38*0d6140beSAndroid Build Coastguard Worker 	chip_readb(flash, bios + 0x0419);
39*0d6140beSAndroid Build Coastguard Worker 	chip_readb(flash, bios + 0x040A);
40*0d6140beSAndroid Build Coastguard Worker 
41*0d6140beSAndroid Build Coastguard Worker 	return 0;
42*0d6140beSAndroid Build Coastguard Worker }
43*0d6140beSAndroid Build Coastguard Worker 
unprotect_28sf040(struct flashctx * flash)44*0d6140beSAndroid Build Coastguard Worker int unprotect_28sf040(struct flashctx *flash)
45*0d6140beSAndroid Build Coastguard Worker {
46*0d6140beSAndroid Build Coastguard Worker 	chipaddr bios = flash->virtual_memory;
47*0d6140beSAndroid Build Coastguard Worker 
48*0d6140beSAndroid Build Coastguard Worker 	chip_readb(flash, bios + 0x1823);
49*0d6140beSAndroid Build Coastguard Worker 	chip_readb(flash, bios + 0x1820);
50*0d6140beSAndroid Build Coastguard Worker 	chip_readb(flash, bios + 0x1822);
51*0d6140beSAndroid Build Coastguard Worker 	chip_readb(flash, bios + 0x0418);
52*0d6140beSAndroid Build Coastguard Worker 	chip_readb(flash, bios + 0x041B);
53*0d6140beSAndroid Build Coastguard Worker 	chip_readb(flash, bios + 0x0419);
54*0d6140beSAndroid Build Coastguard Worker 	chip_readb(flash, bios + 0x041A);
55*0d6140beSAndroid Build Coastguard Worker 
56*0d6140beSAndroid Build Coastguard Worker 	return 0;
57*0d6140beSAndroid Build Coastguard Worker }
58*0d6140beSAndroid Build Coastguard Worker 
erase_sector_28sf040(struct flashctx * flash,unsigned int address,unsigned int sector_size)59*0d6140beSAndroid Build Coastguard Worker int erase_sector_28sf040(struct flashctx *flash, unsigned int address,
60*0d6140beSAndroid Build Coastguard Worker 			 unsigned int sector_size)
61*0d6140beSAndroid Build Coastguard Worker {
62*0d6140beSAndroid Build Coastguard Worker 	chipaddr bios = flash->virtual_memory;
63*0d6140beSAndroid Build Coastguard Worker 
64*0d6140beSAndroid Build Coastguard Worker 	/* This command sequence is very similar to erase_block_82802ab. */
65*0d6140beSAndroid Build Coastguard Worker 	chip_writeb(flash, AUTO_PG_ERASE1, bios);
66*0d6140beSAndroid Build Coastguard Worker 	chip_writeb(flash, AUTO_PG_ERASE2, bios + address);
67*0d6140beSAndroid Build Coastguard Worker 
68*0d6140beSAndroid Build Coastguard Worker 	/* wait for Toggle bit ready */
69*0d6140beSAndroid Build Coastguard Worker 	toggle_ready_jedec(flash, bios);
70*0d6140beSAndroid Build Coastguard Worker 
71*0d6140beSAndroid Build Coastguard Worker 	/* FIXME: Check the status register for errors. */
72*0d6140beSAndroid Build Coastguard Worker 	return 0;
73*0d6140beSAndroid Build Coastguard Worker }
74*0d6140beSAndroid Build Coastguard Worker 
75*0d6140beSAndroid Build Coastguard Worker /* chunksize is 1 */
write_28sf040(struct flashctx * flash,const uint8_t * src,unsigned int start,unsigned int len)76*0d6140beSAndroid Build Coastguard Worker int write_28sf040(struct flashctx *flash, const uint8_t *src, unsigned int start, unsigned int len)
77*0d6140beSAndroid Build Coastguard Worker {
78*0d6140beSAndroid Build Coastguard Worker 	unsigned int i;
79*0d6140beSAndroid Build Coastguard Worker 	chipaddr bios = flash->virtual_memory;
80*0d6140beSAndroid Build Coastguard Worker 	chipaddr dst = flash->virtual_memory + start;
81*0d6140beSAndroid Build Coastguard Worker 
82*0d6140beSAndroid Build Coastguard Worker 	for (i = 0; i < len; i++) {
83*0d6140beSAndroid Build Coastguard Worker 		/* transfer data from source to destination */
84*0d6140beSAndroid Build Coastguard Worker 		if (*src == 0xFF) {
85*0d6140beSAndroid Build Coastguard Worker 			dst++, src++;
86*0d6140beSAndroid Build Coastguard Worker 			/* If the data is 0xFF, don't program it */
87*0d6140beSAndroid Build Coastguard Worker 			continue;
88*0d6140beSAndroid Build Coastguard Worker 		}
89*0d6140beSAndroid Build Coastguard Worker 		/*issue AUTO PROGRAM command */
90*0d6140beSAndroid Build Coastguard Worker 		chip_writeb(flash, AUTO_PGRM, dst);
91*0d6140beSAndroid Build Coastguard Worker 		chip_writeb(flash, *src++, dst++);
92*0d6140beSAndroid Build Coastguard Worker 
93*0d6140beSAndroid Build Coastguard Worker 		/* wait for Toggle bit ready */
94*0d6140beSAndroid Build Coastguard Worker 		toggle_ready_jedec(flash, bios);
95*0d6140beSAndroid Build Coastguard Worker 		update_progress(flash, FLASHROM_PROGRESS_WRITE, i + 1, len);
96*0d6140beSAndroid Build Coastguard Worker 	}
97*0d6140beSAndroid Build Coastguard Worker 
98*0d6140beSAndroid Build Coastguard Worker 	return 0;
99*0d6140beSAndroid Build Coastguard Worker }
100*0d6140beSAndroid Build Coastguard Worker 
erase_28sf040(struct flashctx * flash)101*0d6140beSAndroid Build Coastguard Worker static int erase_28sf040(struct flashctx *flash)
102*0d6140beSAndroid Build Coastguard Worker {
103*0d6140beSAndroid Build Coastguard Worker 	chipaddr bios = flash->virtual_memory;
104*0d6140beSAndroid Build Coastguard Worker 
105*0d6140beSAndroid Build Coastguard Worker 	chip_writeb(flash, CHIP_ERASE, bios);
106*0d6140beSAndroid Build Coastguard Worker 	chip_writeb(flash, CHIP_ERASE, bios);
107*0d6140beSAndroid Build Coastguard Worker 
108*0d6140beSAndroid Build Coastguard Worker 	programmer_delay(flash, 10);
109*0d6140beSAndroid Build Coastguard Worker 	toggle_ready_jedec(flash, bios);
110*0d6140beSAndroid Build Coastguard Worker 
111*0d6140beSAndroid Build Coastguard Worker 	/* FIXME: Check the status register for errors. */
112*0d6140beSAndroid Build Coastguard Worker 	return 0;
113*0d6140beSAndroid Build Coastguard Worker }
114*0d6140beSAndroid Build Coastguard Worker 
erase_chip_28sf040(struct flashctx * flash,unsigned int addr,unsigned int blocklen)115*0d6140beSAndroid Build Coastguard Worker int erase_chip_28sf040(struct flashctx *flash, unsigned int addr,
116*0d6140beSAndroid Build Coastguard Worker 		       unsigned int blocklen)
117*0d6140beSAndroid Build Coastguard Worker {
118*0d6140beSAndroid Build Coastguard Worker 	if ((addr != 0) || (blocklen != flash->chip->total_size * 1024)) {
119*0d6140beSAndroid Build Coastguard Worker 		msg_cerr("%s called with incorrect arguments\n",
120*0d6140beSAndroid Build Coastguard Worker 			__func__);
121*0d6140beSAndroid Build Coastguard Worker 		return -1;
122*0d6140beSAndroid Build Coastguard Worker 	}
123*0d6140beSAndroid Build Coastguard Worker 	return erase_28sf040(flash);
124*0d6140beSAndroid Build Coastguard Worker }
125