blob: 2e703e85250d437e2ba9e21415c3bd1b6eb335e8 [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001/* SPDX-License-Identifier: GPL-2.0 */
Haavard Skinnemoen2f5bfb72008-05-16 11:10:33 +02002/*
Jagannadha Sutradharudu Teki25dc86a2013-10-02 19:38:49 +05303 * Common SPI flash Interface
Haavard Skinnemoen2f5bfb72008-05-16 11:10:33 +02004 *
5 * Copyright (C) 2008 Atmel Corporation
Jagannadha Sutradharudu Teki25dc86a2013-10-02 19:38:49 +05306 * Copyright (C) 2013 Jagannadha Sutradharudu Teki, Xilinx Inc.
Haavard Skinnemoen2f5bfb72008-05-16 11:10:33 +02007 */
Jagannadha Sutradharudu Teki25dc86a2013-10-02 19:38:49 +05308
Haavard Skinnemoen2f5bfb72008-05-16 11:10:33 +02009#ifndef _SPI_FLASH_H_
10#define _SPI_FLASH_H_
11
Mike Frysingere4c698b2009-11-03 11:36:39 -050012#include <linux/types.h>
Vignesh R14510412019-02-05 11:29:23 +053013#include <linux/mtd/spi-nor.h>
Haavard Skinnemoen2f5bfb72008-05-16 11:10:33 +020014
Simon Glass7130b952020-07-19 10:15:40 -060015struct udevice;
16
Simon Glassd34b4562014-10-13 23:42:04 -060017struct spi_slave;
Jagannadha Sutradharudu Teki20e45bc2013-12-23 23:34:42 +053018
Simon Glassd6eedf72014-10-13 23:42:06 -060019struct dm_spi_flash_ops {
20 int (*read)(struct udevice *dev, u32 offset, size_t len, void *buf);
21 int (*write)(struct udevice *dev, u32 offset, size_t len,
22 const void *buf);
23 int (*erase)(struct udevice *dev, u32 offset, size_t len);
Simon Glass61d04642021-03-15 18:11:17 +130024 /**
25 * get_sw_write_prot() - Check state of software write-protect feature
26 *
27 * SPI flash chips can lock a region of the flash defined by a
28 * 'protected area'. This function checks if this protected area is
29 * defined.
30 *
31 * @dev: SPI flash device
32 * @return 0 if no region is write-protected, 1 if a region is
33 * write-protected, -ENOSYS if the driver does not implement this,
34 * other -ve value on error
35 */
36 int (*get_sw_write_prot)(struct udevice *dev);
Haavard Skinnemoen2f5bfb72008-05-16 11:10:33 +020037};
38
Simon Glassd6eedf72014-10-13 23:42:06 -060039/* Access the serial operations for a device */
40#define sf_get_ops(dev) ((struct dm_spi_flash_ops *)(dev)->driver->ops)
41
Lukasz Majewski76f442982020-06-04 23:11:53 +080042#if CONFIG_IS_ENABLED(DM_SPI_FLASH)
Simon Glass1c369232015-03-26 09:29:25 -060043/**
44 * spi_flash_read_dm() - Read data from SPI flash
45 *
46 * @dev: SPI flash device
47 * @offset: Offset into device in bytes to read from
48 * @len: Number of bytes to read
49 * @buf: Buffer to put the data that is read
Heinrich Schuchardt47b4c022022-01-19 18:05:50 +010050 * Return: 0 if OK, -ve on error
Simon Glass1c369232015-03-26 09:29:25 -060051 */
52int spi_flash_read_dm(struct udevice *dev, u32 offset, size_t len, void *buf);
53
54/**
55 * spi_flash_write_dm() - Write data to SPI flash
56 *
57 * @dev: SPI flash device
58 * @offset: Offset into device in bytes to write to
59 * @len: Number of bytes to write
60 * @buf: Buffer containing bytes to write
Heinrich Schuchardt47b4c022022-01-19 18:05:50 +010061 * Return: 0 if OK, -ve on error
Simon Glass1c369232015-03-26 09:29:25 -060062 */
63int spi_flash_write_dm(struct udevice *dev, u32 offset, size_t len,
64 const void *buf);
65
66/**
67 * spi_flash_erase_dm() - Erase blocks of the SPI flash
68 *
69 * Note that @len must be a muiltiple of the flash sector size.
70 *
71 * @dev: SPI flash device
72 * @offset: Offset into device in bytes to start erasing
73 * @len: Number of bytes to erase
Heinrich Schuchardt47b4c022022-01-19 18:05:50 +010074 * Return: 0 if OK, -ve on error
Simon Glass1c369232015-03-26 09:29:25 -060075 */
76int spi_flash_erase_dm(struct udevice *dev, u32 offset, size_t len);
77
Simon Glass36eee8c2018-11-06 15:21:41 -070078/**
Simon Glass61d04642021-03-15 18:11:17 +130079 * spl_flash_get_sw_write_prot() - Check state of software write-protect feature
80 *
81 * SPI flash chips can lock a region of the flash defined by a
82 * 'protected area'. This function checks if this protected area is
83 * defined.
84 *
85 * @dev: SPI flash device
Heinrich Schuchardt47b4c022022-01-19 18:05:50 +010086 * Return: 0 if no region is write-protected, 1 if a region is
Simon Glass61d04642021-03-15 18:11:17 +130087 * write-protected, -ENOSYS if the driver does not implement this,
88 * other -ve value on error
89 */
90int spl_flash_get_sw_write_prot(struct udevice *dev);
91
92/**
Simon Glassa104b642019-12-06 21:42:50 -070093 * spi_flash_std_probe() - Probe a SPI flash device
94 *
95 * This is the standard internal method for probing a SPI flash device to
96 * determine its type. It can be used in chip-specific drivers which need to
97 * do this, typically with of-platdata
98 *
99 * @dev: SPI-flash device to probe
Heinrich Schuchardt47b4c022022-01-19 18:05:50 +0100100 * Return: 0 if OK, -ve on error
Simon Glassa104b642019-12-06 21:42:50 -0700101 */
102int spi_flash_std_probe(struct udevice *dev);
103
Simon Glassd6eedf72014-10-13 23:42:06 -0600104int spi_flash_probe_bus_cs(unsigned int busnum, unsigned int cs,
Simon Glassd6eedf72014-10-13 23:42:06 -0600105 struct udevice **devp);
106
107/* Compatibility function - this is the old U-Boot API */
108struct spi_flash *spi_flash_probe(unsigned int bus, unsigned int cs,
109 unsigned int max_hz, unsigned int spi_mode);
110
111/* Compatibility function - this is the old U-Boot API */
Heinrich Schuchardt6acba5a2021-03-10 18:23:57 +0100112static inline void spi_flash_free(struct spi_flash *flash)
113{
114}
Simon Glassd6eedf72014-10-13 23:42:06 -0600115
Simon Glassd6eedf72014-10-13 23:42:06 -0600116static inline int spi_flash_read(struct spi_flash *flash, u32 offset,
Simon Glass1c369232015-03-26 09:29:25 -0600117 size_t len, void *buf)
Simon Glassd6eedf72014-10-13 23:42:06 -0600118{
Simon Glass1c369232015-03-26 09:29:25 -0600119 return spi_flash_read_dm(flash->dev, offset, len, buf);
Simon Glassd6eedf72014-10-13 23:42:06 -0600120}
121
122static inline int spi_flash_write(struct spi_flash *flash, u32 offset,
Simon Glass1c369232015-03-26 09:29:25 -0600123 size_t len, const void *buf)
Simon Glassd6eedf72014-10-13 23:42:06 -0600124{
Simon Glass1c369232015-03-26 09:29:25 -0600125 return spi_flash_write_dm(flash->dev, offset, len, buf);
Simon Glassd6eedf72014-10-13 23:42:06 -0600126}
127
128static inline int spi_flash_erase(struct spi_flash *flash, u32 offset,
Simon Glass1c369232015-03-26 09:29:25 -0600129 size_t len)
Simon Glassd6eedf72014-10-13 23:42:06 -0600130{
Simon Glass1c369232015-03-26 09:29:25 -0600131 return spi_flash_erase_dm(flash->dev, offset, len);
Simon Glassd6eedf72014-10-13 23:42:06 -0600132}
133
134struct sandbox_state;
135
136int sandbox_sf_bind_emul(struct sandbox_state *state, int busnum, int cs,
Simon Glassf3548632018-06-11 13:07:16 -0600137 struct udevice *bus, ofnode node, const char *spec);
Simon Glassd6eedf72014-10-13 23:42:06 -0600138
139void sandbox_sf_unbind_emul(struct sandbox_state *state, int busnum, int cs);
140
141#else
Neil Armstrong30b5c612024-10-01 18:06:11 +0200142/* Compatibility functions for when DM_SPI_FLASH is disabled */
143static inline int spi_flash_probe_bus_cs(unsigned int busnum, unsigned int cs,
144 struct udevice **devp)
145{
146 return -ENODEV;
147}
148
149static inline int spi_flash_read_dm(struct udevice *dev, u32 offset, size_t len,
150 void *buf)
151{
152 return -ENODEV;
153}
154
155static inline int spi_flash_write_dm(struct udevice *dev, u32 offset, size_t len,
156 const void *buf)
157{
158 return -ENODEV;
159}
160
161static inline int spi_flash_erase_dm(struct udevice *dev, u32 offset, size_t len)
162{
163 return -ENODEV;
164}
165
166static inline int spl_flash_get_sw_write_prot(struct udevice *dev)
167{
168 return -ENODEV;
169}
170
171static inline int spi_flash_std_probe(struct udevice *dev)
172{
173 return -ENODEV;
174}
175
Haavard Skinnemoen2f5bfb72008-05-16 11:10:33 +0200176struct spi_flash *spi_flash_probe(unsigned int bus, unsigned int cs,
177 unsigned int max_hz, unsigned int spi_mode);
Simon Glass49e9d2c2013-12-03 16:43:24 -0700178
Haavard Skinnemoen2f5bfb72008-05-16 11:10:33 +0200179void spi_flash_free(struct spi_flash *flash);
180
181static inline int spi_flash_read(struct spi_flash *flash, u32 offset,
182 size_t len, void *buf)
183{
Vignesh R14510412019-02-05 11:29:23 +0530184 struct mtd_info *mtd = &flash->mtd;
185 size_t retlen;
186
Marek Behún375027a2021-10-05 15:56:01 +0200187 if (!len)
188 return 0;
189
Vignesh R14510412019-02-05 11:29:23 +0530190 return mtd->_read(mtd, offset, len, &retlen, buf);
Haavard Skinnemoen2f5bfb72008-05-16 11:10:33 +0200191}
192
193static inline int spi_flash_write(struct spi_flash *flash, u32 offset,
194 size_t len, const void *buf)
195{
Vignesh R14510412019-02-05 11:29:23 +0530196 struct mtd_info *mtd = &flash->mtd;
197 size_t retlen;
198
Marek Behún375027a2021-10-05 15:56:01 +0200199 if (!len)
200 return 0;
201
Vignesh R14510412019-02-05 11:29:23 +0530202 return mtd->_write(mtd, offset, len, &retlen, buf);
Haavard Skinnemoen2f5bfb72008-05-16 11:10:33 +0200203}
204
205static inline int spi_flash_erase(struct spi_flash *flash, u32 offset,
206 size_t len)
207{
Vignesh R14510412019-02-05 11:29:23 +0530208 struct mtd_info *mtd = &flash->mtd;
209 struct erase_info instr;
210
211 if (offset % mtd->erasesize || len % mtd->erasesize) {
212 printf("SF: Erase offset/length not multiple of erase size\n");
213 return -EINVAL;
214 }
215
Marek Behún375027a2021-10-05 15:56:01 +0200216 if (!len)
217 return 0;
218
Vignesh R14510412019-02-05 11:29:23 +0530219 memset(&instr, 0, sizeof(instr));
220 instr.addr = offset;
221 instr.len = len;
222
223 return mtd->_erase(mtd, &instr);
Haavard Skinnemoen2f5bfb72008-05-16 11:10:33 +0200224}
Simon Glassd6eedf72014-10-13 23:42:06 -0600225#endif
Haavard Skinnemoen2f5bfb72008-05-16 11:10:33 +0200226
Fabio Estevam1cd87612015-11-05 12:43:42 -0200227static inline int spi_flash_protect(struct spi_flash *flash, u32 ofs, u32 len,
228 bool prot)
229{
Bin Meng66528f62015-11-13 02:46:26 -0800230 if (!flash->flash_lock || !flash->flash_unlock)
Fabio Estevam1cd87612015-11-05 12:43:42 -0200231 return -EOPNOTSUPP;
232
233 if (prot)
234 return flash->flash_lock(flash, ofs, len);
235 else
236 return flash->flash_unlock(flash, ofs, len);
237}
238
Haavard Skinnemoen2f5bfb72008-05-16 11:10:33 +0200239#endif /* _SPI_FLASH_H_ */