blob: 478c543b06b6b80b6530096a17e79e77ebd50d02 [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001/* SPDX-License-Identifier: GPL-2.0 */
Haavard Skinnemoend25ce7d2008-05-16 11:10:33 +02002/*
Jagannadha Sutradharudu Tekia5e81992013-10-02 19:38:49 +05303 * Common SPI flash Interface
Haavard Skinnemoend25ce7d2008-05-16 11:10:33 +02004 *
5 * Copyright (C) 2008 Atmel Corporation
Jagannadha Sutradharudu Tekia5e81992013-10-02 19:38:49 +05306 * Copyright (C) 2013 Jagannadha Sutradharudu Teki, Xilinx Inc.
Haavard Skinnemoend25ce7d2008-05-16 11:10:33 +02007 */
Jagannadha Sutradharudu Tekia5e81992013-10-02 19:38:49 +05308
Haavard Skinnemoend25ce7d2008-05-16 11:10:33 +02009#ifndef _SPI_FLASH_H_
10#define _SPI_FLASH_H_
11
Mike Frysingere06ab652009-11-03 11:36:39 -050012#include <linux/types.h>
Vignesh Rc4e88622019-02-05 11:29:23 +053013#include <linux/mtd/spi-nor.h>
Haavard Skinnemoend25ce7d2008-05-16 11:10:33 +020014
Simon Glassa00867b2020-07-19 10:15:40 -060015struct udevice;
16
Patrick Delaunayabe66b12019-02-27 15:20:38 +010017/* by default ENV use the same parameters than SF command */
18#ifndef CONFIG_ENV_SPI_BUS
19# define CONFIG_ENV_SPI_BUS CONFIG_SF_DEFAULT_BUS
20#endif
21#ifndef CONFIG_ENV_SPI_CS
22# define CONFIG_ENV_SPI_CS CONFIG_SF_DEFAULT_CS
23#endif
24#ifndef CONFIG_ENV_SPI_MAX_HZ
25# define CONFIG_ENV_SPI_MAX_HZ CONFIG_SF_DEFAULT_SPEED
26#endif
27#ifndef CONFIG_ENV_SPI_MODE
28# define CONFIG_ENV_SPI_MODE CONFIG_SF_DEFAULT_MODE
29#endif
30
Simon Glassff0960f2014-10-13 23:42:04 -060031struct spi_slave;
Jagannadha Sutradharudu Teki33adfb52013-12-23 23:34:42 +053032
Simon Glass4c2dbef2014-10-13 23:42:06 -060033struct dm_spi_flash_ops {
34 int (*read)(struct udevice *dev, u32 offset, size_t len, void *buf);
35 int (*write)(struct udevice *dev, u32 offset, size_t len,
36 const void *buf);
37 int (*erase)(struct udevice *dev, u32 offset, size_t len);
38};
39
40/* Access the serial operations for a device */
41#define sf_get_ops(dev) ((struct dm_spi_flash_ops *)(dev)->driver->ops)
42
Lukasz Majewski56c40462020-06-04 23:11:53 +080043#if CONFIG_IS_ENABLED(DM_SPI_FLASH)
Simon Glass8d987ab2015-03-26 09:29:25 -060044/**
45 * spi_flash_read_dm() - Read data from SPI flash
46 *
47 * @dev: SPI flash device
48 * @offset: Offset into device in bytes to read from
49 * @len: Number of bytes to read
50 * @buf: Buffer to put the data that is read
51 * @return 0 if OK, -ve on error
52 */
53int spi_flash_read_dm(struct udevice *dev, u32 offset, size_t len, void *buf);
54
55/**
56 * spi_flash_write_dm() - Write data to SPI flash
57 *
58 * @dev: SPI flash device
59 * @offset: Offset into device in bytes to write to
60 * @len: Number of bytes to write
61 * @buf: Buffer containing bytes to write
62 * @return 0 if OK, -ve on error
63 */
64int spi_flash_write_dm(struct udevice *dev, u32 offset, size_t len,
65 const void *buf);
66
67/**
68 * spi_flash_erase_dm() - Erase blocks of the SPI flash
69 *
70 * Note that @len must be a muiltiple of the flash sector size.
71 *
72 * @dev: SPI flash device
73 * @offset: Offset into device in bytes to start erasing
74 * @len: Number of bytes to erase
75 * @return 0 if OK, -ve on error
76 */
77int spi_flash_erase_dm(struct udevice *dev, u32 offset, size_t len);
78
Simon Glassa58986c2018-11-06 15:21:41 -070079/**
Simon Glass4806fce2019-12-06 21:42:50 -070080 * spi_flash_std_probe() - Probe a SPI flash device
81 *
82 * This is the standard internal method for probing a SPI flash device to
83 * determine its type. It can be used in chip-specific drivers which need to
84 * do this, typically with of-platdata
85 *
86 * @dev: SPI-flash device to probe
87 * @return 0 if OK, -ve on error
88 */
89int spi_flash_std_probe(struct udevice *dev);
90
Simon Glass4c2dbef2014-10-13 23:42:06 -060091int spi_flash_probe_bus_cs(unsigned int busnum, unsigned int cs,
92 unsigned int max_hz, unsigned int spi_mode,
93 struct udevice **devp);
94
95/* Compatibility function - this is the old U-Boot API */
96struct spi_flash *spi_flash_probe(unsigned int bus, unsigned int cs,
97 unsigned int max_hz, unsigned int spi_mode);
98
99/* Compatibility function - this is the old U-Boot API */
Heinrich Schuchardtb09c74f2021-03-10 18:23:57 +0100100static inline void spi_flash_free(struct spi_flash *flash)
101{
102}
Simon Glass4c2dbef2014-10-13 23:42:06 -0600103
Simon Glass4c2dbef2014-10-13 23:42:06 -0600104static inline int spi_flash_read(struct spi_flash *flash, u32 offset,
Simon Glass8d987ab2015-03-26 09:29:25 -0600105 size_t len, void *buf)
Simon Glass4c2dbef2014-10-13 23:42:06 -0600106{
Simon Glass8d987ab2015-03-26 09:29:25 -0600107 return spi_flash_read_dm(flash->dev, offset, len, buf);
Simon Glass4c2dbef2014-10-13 23:42:06 -0600108}
109
110static inline int spi_flash_write(struct spi_flash *flash, u32 offset,
Simon Glass8d987ab2015-03-26 09:29:25 -0600111 size_t len, const void *buf)
Simon Glass4c2dbef2014-10-13 23:42:06 -0600112{
Simon Glass8d987ab2015-03-26 09:29:25 -0600113 return spi_flash_write_dm(flash->dev, offset, len, buf);
Simon Glass4c2dbef2014-10-13 23:42:06 -0600114}
115
116static inline int spi_flash_erase(struct spi_flash *flash, u32 offset,
Simon Glass8d987ab2015-03-26 09:29:25 -0600117 size_t len)
Simon Glass4c2dbef2014-10-13 23:42:06 -0600118{
Simon Glass8d987ab2015-03-26 09:29:25 -0600119 return spi_flash_erase_dm(flash->dev, offset, len);
Simon Glass4c2dbef2014-10-13 23:42:06 -0600120}
121
122struct sandbox_state;
123
124int sandbox_sf_bind_emul(struct sandbox_state *state, int busnum, int cs,
Simon Glass008dcdd2018-06-11 13:07:16 -0600125 struct udevice *bus, ofnode node, const char *spec);
Simon Glass4c2dbef2014-10-13 23:42:06 -0600126
127void sandbox_sf_unbind_emul(struct sandbox_state *state, int busnum, int cs);
128
129#else
Haavard Skinnemoend25ce7d2008-05-16 11:10:33 +0200130struct spi_flash *spi_flash_probe(unsigned int bus, unsigned int cs,
131 unsigned int max_hz, unsigned int spi_mode);
Simon Glass0efc0242013-12-03 16:43:24 -0700132
Haavard Skinnemoend25ce7d2008-05-16 11:10:33 +0200133void spi_flash_free(struct spi_flash *flash);
134
135static inline int spi_flash_read(struct spi_flash *flash, u32 offset,
136 size_t len, void *buf)
137{
Vignesh Rc4e88622019-02-05 11:29:23 +0530138 struct mtd_info *mtd = &flash->mtd;
139 size_t retlen;
140
141 return mtd->_read(mtd, offset, len, &retlen, buf);
Haavard Skinnemoend25ce7d2008-05-16 11:10:33 +0200142}
143
144static inline int spi_flash_write(struct spi_flash *flash, u32 offset,
145 size_t len, const void *buf)
146{
Vignesh Rc4e88622019-02-05 11:29:23 +0530147 struct mtd_info *mtd = &flash->mtd;
148 size_t retlen;
149
150 return mtd->_write(mtd, offset, len, &retlen, buf);
Haavard Skinnemoend25ce7d2008-05-16 11:10:33 +0200151}
152
153static inline int spi_flash_erase(struct spi_flash *flash, u32 offset,
154 size_t len)
155{
Vignesh Rc4e88622019-02-05 11:29:23 +0530156 struct mtd_info *mtd = &flash->mtd;
157 struct erase_info instr;
158
159 if (offset % mtd->erasesize || len % mtd->erasesize) {
160 printf("SF: Erase offset/length not multiple of erase size\n");
161 return -EINVAL;
162 }
163
164 memset(&instr, 0, sizeof(instr));
165 instr.addr = offset;
166 instr.len = len;
167
168 return mtd->_erase(mtd, &instr);
Haavard Skinnemoend25ce7d2008-05-16 11:10:33 +0200169}
Simon Glass4c2dbef2014-10-13 23:42:06 -0600170#endif
Haavard Skinnemoend25ce7d2008-05-16 11:10:33 +0200171
Fabio Estevamc3c016c2015-11-05 12:43:42 -0200172static inline int spi_flash_protect(struct spi_flash *flash, u32 ofs, u32 len,
173 bool prot)
174{
Bin Meng439fcb92015-11-13 02:46:26 -0800175 if (!flash->flash_lock || !flash->flash_unlock)
Fabio Estevamc3c016c2015-11-05 12:43:42 -0200176 return -EOPNOTSUPP;
177
178 if (prot)
179 return flash->flash_lock(flash, ofs, len);
180 else
181 return flash->flash_unlock(flash, ofs, len);
182}
183
Haavard Skinnemoend25ce7d2008-05-16 11:10:33 +0200184#endif /* _SPI_FLASH_H_ */