]> git.kernelconcepts.de Git - karo-tx-uboot.git/blob - drivers/mtd/spi/stmicro.c
Merge branch 'next' of git://git.denx.de/u-boot
[karo-tx-uboot.git] / drivers / mtd / spi / stmicro.c
1 /*
2  * (C) Copyright 2000-2002
3  * Wolfgang Denk, DENX Software Engineering, wd@denx.de.
4  *
5  * Copyright 2008, Network Appliance Inc.
6  * Jason McMullan <mcmullan@netapp.com>
7  *
8  * Copyright (C) 2004-2007 Freescale Semiconductor, Inc.
9  * TsiChung Liew (Tsi-Chung.Liew@freescale.com)
10  *
11  * See file CREDITS for list of people who contributed to this
12  * project.
13  *
14  * This program is free software; you can redistribute it and/or
15  * modify it under the terms of the GNU General Public License as
16  * published by the Free Software Foundation; either version 2 of
17  * the License, or (at your option) any later version.
18  *
19  * This program is distributed in the hope that it will be useful,
20  * but WITHOUT ANY WARRANTY; without even the implied warranty of
21  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
22  * GNU General Public License for more details.
23  *
24  * You should have received a copy of the GNU General Public License
25  * along with this program; if not, write to the Free Software
26  * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
27  * MA 02111-1307 USA
28  */
29
30 #include <common.h>
31 #include <malloc.h>
32 #include <spi_flash.h>
33
34 #include "spi_flash_internal.h"
35
36 /* M25Pxx-specific commands */
37 #define CMD_M25PXX_SE           0xd8    /* Sector Erase */
38 #define CMD_M25PXX_BE           0xc7    /* Bulk Erase */
39 #define CMD_M25PXX_RES          0xab    /* Release from DP, and Read Signature */
40
41 struct stmicro_spi_flash_params {
42         u8 idcode1;
43         u16 page_size;
44         u16 pages_per_sector;
45         u16 nr_sectors;
46         const char *name;
47 };
48
49 static const struct stmicro_spi_flash_params stmicro_spi_flash_table[] = {
50         {
51                 .idcode1 = 0x11,
52                 .page_size = 256,
53                 .pages_per_sector = 128,
54                 .nr_sectors = 4,
55                 .name = "M25P10",
56         },
57         {
58                 .idcode1 = 0x15,
59                 .page_size = 256,
60                 .pages_per_sector = 256,
61                 .nr_sectors = 32,
62                 .name = "M25P16",
63         },
64         {
65                 .idcode1 = 0x12,
66                 .page_size = 256,
67                 .pages_per_sector = 256,
68                 .nr_sectors = 4,
69                 .name = "M25P20",
70         },
71         {
72                 .idcode1 = 0x16,
73                 .page_size = 256,
74                 .pages_per_sector = 256,
75                 .nr_sectors = 64,
76                 .name = "M25P32",
77         },
78         {
79                 .idcode1 = 0x13,
80                 .page_size = 256,
81                 .pages_per_sector = 256,
82                 .nr_sectors = 8,
83                 .name = "M25P40",
84         },
85         {
86                 .idcode1 = 0x17,
87                 .page_size = 256,
88                 .pages_per_sector = 256,
89                 .nr_sectors = 128,
90                 .name = "M25P64",
91         },
92         {
93                 .idcode1 = 0x14,
94                 .page_size = 256,
95                 .pages_per_sector = 256,
96                 .nr_sectors = 16,
97                 .name = "M25P80",
98         },
99         {
100                 .idcode1 = 0x18,
101                 .page_size = 256,
102                 .pages_per_sector = 1024,
103                 .nr_sectors = 64,
104                 .name = "M25P128",
105         },
106 };
107
108 static int stmicro_erase(struct spi_flash *flash, u32 offset, size_t len)
109 {
110         return spi_flash_cmd_erase(flash, CMD_M25PXX_SE, offset, len);
111 }
112
113 struct spi_flash *spi_flash_probe_stmicro(struct spi_slave *spi, u8 * idcode)
114 {
115         const struct stmicro_spi_flash_params *params;
116         struct spi_flash *flash;
117         unsigned int i;
118
119         if (idcode[0] == 0xff) {
120                 i = spi_flash_cmd(spi, CMD_M25PXX_RES,
121                                   idcode, 4);
122                 if (i)
123                         return NULL;
124                 if ((idcode[3] & 0xf0) == 0x10) {
125                         idcode[0] = 0x20;
126                         idcode[1] = 0x20;
127                         idcode[2] = idcode[3] + 1;
128                 } else
129                         return NULL;
130         }
131
132         for (i = 0; i < ARRAY_SIZE(stmicro_spi_flash_table); i++) {
133                 params = &stmicro_spi_flash_table[i];
134                 if (params->idcode1 == idcode[2]) {
135                         break;
136                 }
137         }
138
139         if (i == ARRAY_SIZE(stmicro_spi_flash_table)) {
140                 debug("SF: Unsupported STMicro ID %02x\n", idcode[1]);
141                 return NULL;
142         }
143
144         flash = malloc(sizeof(*flash));
145         if (!flash) {
146                 debug("SF: Failed to allocate memory\n");
147                 return NULL;
148         }
149
150         flash->spi = spi;
151         flash->name = params->name;
152
153         flash->write = spi_flash_cmd_write_multi;
154         flash->erase = stmicro_erase;
155         flash->read = spi_flash_cmd_read_fast;
156         flash->page_size = params->page_size;
157         flash->sector_size = params->page_size * params->pages_per_sector;
158         flash->size = flash->sector_size * params->nr_sectors;
159
160         return flash;
161 }