blob: d9b2af856c08feb18adda8934452483800f83bb4 [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
Simon Glassd6eedf72014-10-13 23:42:06 -060012#include <dm.h> /* Because we dereference struct udevice here */
Mike Frysingere4c698b2009-11-03 11:36:39 -050013#include <linux/types.h>
Vignesh R14510412019-02-05 11:29:23 +053014#include <linux/mtd/spi-nor.h>
Haavard Skinnemoen2f5bfb72008-05-16 11:10:33 +020015
Patrick Delaunayf8e932e2019-02-27 15:20:38 +010016/* by default ENV use the same parameters than SF command */
17#ifndef CONFIG_ENV_SPI_BUS
18# define CONFIG_ENV_SPI_BUS CONFIG_SF_DEFAULT_BUS
19#endif
20#ifndef CONFIG_ENV_SPI_CS
21# define CONFIG_ENV_SPI_CS CONFIG_SF_DEFAULT_CS
22#endif
23#ifndef CONFIG_ENV_SPI_MAX_HZ
24# define CONFIG_ENV_SPI_MAX_HZ CONFIG_SF_DEFAULT_SPEED
25#endif
26#ifndef CONFIG_ENV_SPI_MODE
27# define CONFIG_ENV_SPI_MODE CONFIG_SF_DEFAULT_MODE
28#endif
29
Simon Glassd34b4562014-10-13 23:42:04 -060030struct spi_slave;
Jagannadha Sutradharudu Teki20e45bc2013-12-23 23:34:42 +053031
Simon Glassd6eedf72014-10-13 23:42:06 -060032struct dm_spi_flash_ops {
33 int (*read)(struct udevice *dev, u32 offset, size_t len, void *buf);
34 int (*write)(struct udevice *dev, u32 offset, size_t len,
35 const void *buf);
36 int (*erase)(struct udevice *dev, u32 offset, size_t len);
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
42#ifdef CONFIG_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
50 * @return 0 if OK, -ve on error
51 */
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
61 * @return 0 if OK, -ve on error
62 */
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
74 * @return 0 if OK, -ve on error
75 */
76int spi_flash_erase_dm(struct udevice *dev, u32 offset, size_t len);
77
Simon Glass36eee8c2018-11-06 15:21:41 -070078/**
Simon Glassa104b642019-12-06 21:42:50 -070079 * spi_flash_std_probe() - Probe a SPI flash device
80 *
81 * This is the standard internal method for probing a SPI flash device to
82 * determine its type. It can be used in chip-specific drivers which need to
83 * do this, typically with of-platdata
84 *
85 * @dev: SPI-flash device to probe
86 * @return 0 if OK, -ve on error
87 */
88int spi_flash_std_probe(struct udevice *dev);
89
Simon Glassd6eedf72014-10-13 23:42:06 -060090int spi_flash_probe_bus_cs(unsigned int busnum, unsigned int cs,
91 unsigned int max_hz, unsigned int spi_mode,
92 struct udevice **devp);
93
94/* Compatibility function - this is the old U-Boot API */
95struct spi_flash *spi_flash_probe(unsigned int bus, unsigned int cs,
96 unsigned int max_hz, unsigned int spi_mode);
97
98/* Compatibility function - this is the old U-Boot API */
99void spi_flash_free(struct spi_flash *flash);
100
Simon Glassd6eedf72014-10-13 23:42:06 -0600101static inline int spi_flash_read(struct spi_flash *flash, u32 offset,
Simon Glass1c369232015-03-26 09:29:25 -0600102 size_t len, void *buf)
Simon Glassd6eedf72014-10-13 23:42:06 -0600103{
Simon Glass1c369232015-03-26 09:29:25 -0600104 return spi_flash_read_dm(flash->dev, offset, len, buf);
Simon Glassd6eedf72014-10-13 23:42:06 -0600105}
106
107static inline int spi_flash_write(struct spi_flash *flash, u32 offset,
Simon Glass1c369232015-03-26 09:29:25 -0600108 size_t len, const void *buf)
Simon Glassd6eedf72014-10-13 23:42:06 -0600109{
Simon Glass1c369232015-03-26 09:29:25 -0600110 return spi_flash_write_dm(flash->dev, offset, len, buf);
Simon Glassd6eedf72014-10-13 23:42:06 -0600111}
112
113static inline int spi_flash_erase(struct spi_flash *flash, u32 offset,
Simon Glass1c369232015-03-26 09:29:25 -0600114 size_t len)
Simon Glassd6eedf72014-10-13 23:42:06 -0600115{
Simon Glass1c369232015-03-26 09:29:25 -0600116 return spi_flash_erase_dm(flash->dev, offset, len);
Simon Glassd6eedf72014-10-13 23:42:06 -0600117}
118
119struct sandbox_state;
120
121int sandbox_sf_bind_emul(struct sandbox_state *state, int busnum, int cs,
Simon Glassf3548632018-06-11 13:07:16 -0600122 struct udevice *bus, ofnode node, const char *spec);
Simon Glassd6eedf72014-10-13 23:42:06 -0600123
124void sandbox_sf_unbind_emul(struct sandbox_state *state, int busnum, int cs);
125
126#else
Haavard Skinnemoen2f5bfb72008-05-16 11:10:33 +0200127struct spi_flash *spi_flash_probe(unsigned int bus, unsigned int cs,
128 unsigned int max_hz, unsigned int spi_mode);
Simon Glass49e9d2c2013-12-03 16:43:24 -0700129
Haavard Skinnemoen2f5bfb72008-05-16 11:10:33 +0200130void spi_flash_free(struct spi_flash *flash);
131
132static inline int spi_flash_read(struct spi_flash *flash, u32 offset,
133 size_t len, void *buf)
134{
Vignesh R14510412019-02-05 11:29:23 +0530135 struct mtd_info *mtd = &flash->mtd;
136 size_t retlen;
137
138 return mtd->_read(mtd, offset, len, &retlen, buf);
Haavard Skinnemoen2f5bfb72008-05-16 11:10:33 +0200139}
140
141static inline int spi_flash_write(struct spi_flash *flash, u32 offset,
142 size_t len, const void *buf)
143{
Vignesh R14510412019-02-05 11:29:23 +0530144 struct mtd_info *mtd = &flash->mtd;
145 size_t retlen;
146
147 return mtd->_write(mtd, offset, len, &retlen, buf);
Haavard Skinnemoen2f5bfb72008-05-16 11:10:33 +0200148}
149
150static inline int spi_flash_erase(struct spi_flash *flash, u32 offset,
151 size_t len)
152{
Vignesh R14510412019-02-05 11:29:23 +0530153 struct mtd_info *mtd = &flash->mtd;
154 struct erase_info instr;
155
156 if (offset % mtd->erasesize || len % mtd->erasesize) {
157 printf("SF: Erase offset/length not multiple of erase size\n");
158 return -EINVAL;
159 }
160
161 memset(&instr, 0, sizeof(instr));
162 instr.addr = offset;
163 instr.len = len;
164
165 return mtd->_erase(mtd, &instr);
Haavard Skinnemoen2f5bfb72008-05-16 11:10:33 +0200166}
Simon Glassd6eedf72014-10-13 23:42:06 -0600167#endif
Haavard Skinnemoen2f5bfb72008-05-16 11:10:33 +0200168
Fabio Estevam1cd87612015-11-05 12:43:42 -0200169static inline int spi_flash_protect(struct spi_flash *flash, u32 ofs, u32 len,
170 bool prot)
171{
Bin Meng66528f62015-11-13 02:46:26 -0800172 if (!flash->flash_lock || !flash->flash_unlock)
Fabio Estevam1cd87612015-11-05 12:43:42 -0200173 return -EOPNOTSUPP;
174
175 if (prot)
176 return flash->flash_lock(flash, ofs, len);
177 else
178 return flash->flash_unlock(flash, ofs, len);
179}
180
Haavard Skinnemoen2f5bfb72008-05-16 11:10:33 +0200181#endif /* _SPI_FLASH_H_ */