2019-12-19 11:55:58 +01:00
// SPDX-License-Identifier: GPL-2.0-or-later
2008-07-14 22:38:35 +02:00
/*
* at24 . c - handle most I2C EEPROMs
*
* Copyright ( C ) 2005 - 2007 David Brownell
* Copyright ( C ) 2008 Wolfram Sang , Pengutronix
*/
2018-03-23 17:15:02 +01:00
2008-07-14 22:38:35 +02:00
# include <linux/kernel.h>
# include <linux/init.h>
# include <linux/module.h>
2017-10-01 12:49:48 +02:00
# include <linux/of_device.h>
2008-07-14 22:38:35 +02:00
# include <linux/slab.h>
# include <linux/delay.h>
# include <linux/mutex.h>
# include <linux/mod_devicetable.h>
# include <linux/bitops.h>
# include <linux/jiffies.h>
misc: eeprom: at24: use device_property_*() functions instead of of_get_property()
Allow the at24 driver to get configuration information from both OF and
ACPI by using the more generic device_property functions.
This change was inspired by the at25.c driver.
I have a custom board with a ST M24C02 EEPROM attached to an I2C bus.
With the following ACPI construct, this patch instantiates a working
instance of the driver.
Device (EEP0) {
Name (_HID, "PRP0001")
Name (_DSD, Package () {
ToUUID("daffd814-6eba-4d8c-8a91-bc9bbf4aa301"),
Package () {
Package () {"compatible", Package () {"st,24c02"}},
Package () {"pagesize", 16},
},
})
Name (_CRS, ResourceTemplate () {
I2cSerialBus (
0x0057, ControllerInitiated, 400000,
AddressingMode7Bit, "\\_SB.PCI0.I2C3", 0x00,
ResourceConsumer,,)
})
}
Signed-off-by: Ben Gardner <gardner.ben@gmail.com>
Reviewed-by: Andy Shevchenko <andy.shevchenko@gmail.com>
Signed-off-by: Wolfram Sang <wsa@the-dreams.de>
2017-02-09 11:36:08 -06:00
# include <linux/property.h>
2015-10-23 12:16:44 +03:00
# include <linux/acpi.h>
2008-07-14 22:38:35 +02:00
# include <linux/i2c.h>
2016-02-26 20:59:20 +01:00
# include <linux/nvmem-provider.h>
2017-11-28 21:51:40 +01:00
# include <linux/regmap.h>
2017-10-10 11:30:37 +05:30
# include <linux/pm_runtime.h>
2017-12-19 11:28:54 +01:00
# include <linux/gpio/consumer.h>
2008-07-14 22:38:35 +02:00
2018-08-07 12:07:43 +02:00
/* Address pointer is 16 bit. */
# define AT24_FLAG_ADDR16 BIT(7)
/* sysfs-entry will be read-only. */
# define AT24_FLAG_READONLY BIT(6)
/* sysfs-entry will be world-readable. */
# define AT24_FLAG_IRUGO BIT(5)
/* Take always 8 addresses (24c00). */
# define AT24_FLAG_TAKE8ADDR BIT(4)
/* Factory-programmed serial number. */
# define AT24_FLAG_SERIAL BIT(3)
/* Factory-programmed mac address. */
# define AT24_FLAG_MAC BIT(2)
/* Does not auto-rollover reads to the next slave address. */
# define AT24_FLAG_NO_RDROL BIT(1)
2008-07-14 22:38:35 +02:00
/*
* I2C EEPROMs from most vendors are inexpensive and mostly interchangeable .
* Differences between different vendor product lines ( like Atmel AT24C or
* MicroChip 24L C , etc ) won ' t much matter for typical read / write access .
* There are also I2C RAM chips , likewise interchangeable . One example
* would be the PCF8570 , which acts like a 24 c02 EEPROM ( 256 bytes ) .
*
* However , misconfiguration can lose data . " Set 16-bit memory address "
* to a part with 8 - bit addressing will overwrite data . Writing with too
* big a page size also loses data . And it ' s not safe to assume that the
* conventional addresses 0x50 . .0 x57 only hold eeproms ; a PCF8563 RTC
* uses 0x51 , for just one example .
*
* Accordingly , explicit board - specific configuration data should be used
* in almost all cases . ( One partial exception is an SMBus used to access
* " SPD " data for DRAM sticks . Those only use 24 c02 EEPROMs . )
*
* So this driver uses " new style " I2C driver binding , expecting to be
* told what devices exist . That may be in arch / X / mach - Y / board - Z . c or
* similar kernel - resident tables ; or , configuration data coming from
* a bootloader .
*
* Other than binding model , current differences from " eeprom " driver are
* that this one handles write access and isn ' t restricted to 24 c02 devices .
* It also handles larger devices ( 32 kbit and up ) with two - byte addresses ,
* which won ' t work on pure SMBus systems .
*/
2017-11-28 21:51:40 +01:00
struct at24_client {
struct i2c_client * client ;
struct regmap * regmap ;
} ;
2008-07-14 22:38:35 +02:00
struct at24_data {
/*
* Lock protects against activities from other Linux tasks ,
* but not from changes by other I2C masters .
*/
struct mutex lock ;
2017-12-13 11:56:23 +01:00
unsigned int write_max ;
unsigned int num_addresses ;
2017-11-28 21:51:50 +01:00
unsigned int offset_adj ;
2008-07-14 22:38:35 +02:00
2018-03-19 10:17:17 +01:00
u32 byte_len ;
u16 page_size ;
u8 flags ;
2016-02-26 20:59:20 +01:00
struct nvmem_device * nvmem ;
2017-12-19 11:28:54 +01:00
struct gpio_desc * wp_gpio ;
2008-07-14 22:38:35 +02:00
/*
* Some chips tie up multiple I2C addresses ; dummy devices reserve
* them for us , and we ' ll use them with SMBus calls .
*/
2017-11-28 21:51:40 +01:00
struct at24_client client [ ] ;
2008-07-14 22:38:35 +02:00
} ;
/*
* This parameter is to help this driver avoid blocking other drivers out
* of I2C for potentially troublesome amounts of time . With a 100 kHz I2C
* clock , one 256 byte read takes about 1 / 43 second which is excessive ;
* but the 1 / 170 second it takes at 400 kHz may be quite reasonable ; and
* at 1 MHz ( Fm + ) a 1 / 430 second delay could easily be invisible .
*
* This value is forced to be a power of two so that writes align on pages .
*/
2017-12-18 18:16:46 +01:00
static unsigned int at24_io_limit = 128 ;
module_param_named ( io_limit , at24_io_limit , uint , 0 ) ;
MODULE_PARM_DESC ( at24_io_limit , " Maximum bytes per I/O (default 128) " ) ;
2008-07-14 22:38:35 +02:00
/*
* Specs often allow 5 msec for a page write , sometimes 20 msec ;
* it ' s important to recover from write timeouts .
*/
2017-12-18 18:16:46 +01:00
static unsigned int at24_write_timeout = 25 ;
module_param_named ( write_timeout , at24_write_timeout , uint , 0 ) ;
MODULE_PARM_DESC ( at24_write_timeout , " Time (in ms) to try writes (default 25) " ) ;
2008-07-14 22:38:35 +02:00
2017-12-20 11:48:56 -05:00
struct at24_chip_data {
u32 byte_len ;
u8 flags ;
} ;
# define AT24_CHIP_DATA(_name, _len, _flags) \
static const struct at24_chip_data _name = { \
. byte_len = _len , . flags = _flags , \
}
/* needs 8 addresses as A0-A2 are ignored */
AT24_CHIP_DATA ( at24_data_24c00 , 128 / 8 , AT24_FLAG_TAKE8ADDR ) ;
/* old variants can't be handled with this generic entry! */
AT24_CHIP_DATA ( at24_data_24c01 , 1024 / 8 , 0 ) ;
AT24_CHIP_DATA ( at24_data_24cs01 , 16 ,
AT24_FLAG_SERIAL | AT24_FLAG_READONLY ) ;
AT24_CHIP_DATA ( at24_data_24c02 , 2048 / 8 , 0 ) ;
AT24_CHIP_DATA ( at24_data_24cs02 , 16 ,
AT24_FLAG_SERIAL | AT24_FLAG_READONLY ) ;
AT24_CHIP_DATA ( at24_data_24mac402 , 48 / 8 ,
AT24_FLAG_MAC | AT24_FLAG_READONLY ) ;
AT24_CHIP_DATA ( at24_data_24mac602 , 64 / 8 ,
AT24_FLAG_MAC | AT24_FLAG_READONLY ) ;
/* spd is a 24c02 in memory DIMMs */
AT24_CHIP_DATA ( at24_data_spd , 2048 / 8 ,
AT24_FLAG_READONLY | AT24_FLAG_IRUGO ) ;
AT24_CHIP_DATA ( at24_data_24c04 , 4096 / 8 , 0 ) ;
AT24_CHIP_DATA ( at24_data_24cs04 , 16 ,
AT24_FLAG_SERIAL | AT24_FLAG_READONLY ) ;
/* 24rf08 quirk is handled at i2c-core */
AT24_CHIP_DATA ( at24_data_24c08 , 8192 / 8 , 0 ) ;
AT24_CHIP_DATA ( at24_data_24cs08 , 16 ,
AT24_FLAG_SERIAL | AT24_FLAG_READONLY ) ;
AT24_CHIP_DATA ( at24_data_24c16 , 16384 / 8 , 0 ) ;
AT24_CHIP_DATA ( at24_data_24cs16 , 16 ,
AT24_FLAG_SERIAL | AT24_FLAG_READONLY ) ;
AT24_CHIP_DATA ( at24_data_24c32 , 32768 / 8 , AT24_FLAG_ADDR16 ) ;
AT24_CHIP_DATA ( at24_data_24cs32 , 16 ,
AT24_FLAG_ADDR16 | AT24_FLAG_SERIAL | AT24_FLAG_READONLY ) ;
AT24_CHIP_DATA ( at24_data_24c64 , 65536 / 8 , AT24_FLAG_ADDR16 ) ;
AT24_CHIP_DATA ( at24_data_24cs64 , 16 ,
AT24_FLAG_ADDR16 | AT24_FLAG_SERIAL | AT24_FLAG_READONLY ) ;
AT24_CHIP_DATA ( at24_data_24c128 , 131072 / 8 , AT24_FLAG_ADDR16 ) ;
AT24_CHIP_DATA ( at24_data_24c256 , 262144 / 8 , AT24_FLAG_ADDR16 ) ;
AT24_CHIP_DATA ( at24_data_24c512 , 524288 / 8 , AT24_FLAG_ADDR16 ) ;
AT24_CHIP_DATA ( at24_data_24c1024 , 1048576 / 8 , AT24_FLAG_ADDR16 ) ;
2018-11-29 21:58:58 +02:00
AT24_CHIP_DATA ( at24_data_24c2048 , 2097152 / 8 , AT24_FLAG_ADDR16 ) ;
2017-12-20 11:48:56 -05:00
/* identical to 24c08 ? */
AT24_CHIP_DATA ( at24_data_INT3499 , 8192 / 8 , 0 ) ;
2008-07-14 22:38:35 +02:00
static const struct i2c_device_id at24_ids [ ] = {
2017-12-20 11:48:56 -05:00
{ " 24c00 " , ( kernel_ulong_t ) & at24_data_24c00 } ,
{ " 24c01 " , ( kernel_ulong_t ) & at24_data_24c01 } ,
{ " 24cs01 " , ( kernel_ulong_t ) & at24_data_24cs01 } ,
{ " 24c02 " , ( kernel_ulong_t ) & at24_data_24c02 } ,
{ " 24cs02 " , ( kernel_ulong_t ) & at24_data_24cs02 } ,
{ " 24mac402 " , ( kernel_ulong_t ) & at24_data_24mac402 } ,
{ " 24mac602 " , ( kernel_ulong_t ) & at24_data_24mac602 } ,
{ " spd " , ( kernel_ulong_t ) & at24_data_spd } ,
{ " 24c04 " , ( kernel_ulong_t ) & at24_data_24c04 } ,
{ " 24cs04 " , ( kernel_ulong_t ) & at24_data_24cs04 } ,
{ " 24c08 " , ( kernel_ulong_t ) & at24_data_24c08 } ,
{ " 24cs08 " , ( kernel_ulong_t ) & at24_data_24cs08 } ,
{ " 24c16 " , ( kernel_ulong_t ) & at24_data_24c16 } ,
{ " 24cs16 " , ( kernel_ulong_t ) & at24_data_24cs16 } ,
{ " 24c32 " , ( kernel_ulong_t ) & at24_data_24c32 } ,
{ " 24cs32 " , ( kernel_ulong_t ) & at24_data_24cs32 } ,
{ " 24c64 " , ( kernel_ulong_t ) & at24_data_24c64 } ,
{ " 24cs64 " , ( kernel_ulong_t ) & at24_data_24cs64 } ,
{ " 24c128 " , ( kernel_ulong_t ) & at24_data_24c128 } ,
{ " 24c256 " , ( kernel_ulong_t ) & at24_data_24c256 } ,
{ " 24c512 " , ( kernel_ulong_t ) & at24_data_24c512 } ,
{ " 24c1024 " , ( kernel_ulong_t ) & at24_data_24c1024 } ,
2018-11-29 21:58:58 +02:00
{ " 24c2048 " , ( kernel_ulong_t ) & at24_data_24c2048 } ,
2017-12-20 11:48:56 -05:00
{ " at24 " , 0 } ,
2008-07-14 22:38:35 +02:00
{ /* END OF LIST */ }
} ;
MODULE_DEVICE_TABLE ( i2c , at24_ids ) ;
2017-10-01 12:49:48 +02:00
static const struct of_device_id at24_of_match [ ] = {
2017-12-20 11:48:56 -05:00
{ . compatible = " atmel,24c00 " , . data = & at24_data_24c00 } ,
{ . compatible = " atmel,24c01 " , . data = & at24_data_24c01 } ,
2017-12-28 11:49:13 +01:00
{ . compatible = " atmel,24cs01 " , . data = & at24_data_24cs01 } ,
2017-12-20 11:48:56 -05:00
{ . compatible = " atmel,24c02 " , . data = & at24_data_24c02 } ,
2017-12-28 11:49:13 +01:00
{ . compatible = " atmel,24cs02 " , . data = & at24_data_24cs02 } ,
{ . compatible = " atmel,24mac402 " , . data = & at24_data_24mac402 } ,
{ . compatible = " atmel,24mac602 " , . data = & at24_data_24mac602 } ,
2017-12-20 11:48:56 -05:00
{ . compatible = " atmel,spd " , . data = & at24_data_spd } ,
{ . compatible = " atmel,24c04 " , . data = & at24_data_24c04 } ,
2017-12-28 11:49:13 +01:00
{ . compatible = " atmel,24cs04 " , . data = & at24_data_24cs04 } ,
2017-12-20 11:48:56 -05:00
{ . compatible = " atmel,24c08 " , . data = & at24_data_24c08 } ,
2017-12-28 11:49:13 +01:00
{ . compatible = " atmel,24cs08 " , . data = & at24_data_24cs08 } ,
2017-12-20 11:48:56 -05:00
{ . compatible = " atmel,24c16 " , . data = & at24_data_24c16 } ,
2017-12-28 11:49:13 +01:00
{ . compatible = " atmel,24cs16 " , . data = & at24_data_24cs16 } ,
2017-12-20 11:48:56 -05:00
{ . compatible = " atmel,24c32 " , . data = & at24_data_24c32 } ,
2017-12-28 11:49:13 +01:00
{ . compatible = " atmel,24cs32 " , . data = & at24_data_24cs32 } ,
2017-12-20 11:48:56 -05:00
{ . compatible = " atmel,24c64 " , . data = & at24_data_24c64 } ,
2017-12-28 11:49:13 +01:00
{ . compatible = " atmel,24cs64 " , . data = & at24_data_24cs64 } ,
2017-12-20 11:48:56 -05:00
{ . compatible = " atmel,24c128 " , . data = & at24_data_24c128 } ,
{ . compatible = " atmel,24c256 " , . data = & at24_data_24c256 } ,
{ . compatible = " atmel,24c512 " , . data = & at24_data_24c512 } ,
{ . compatible = " atmel,24c1024 " , . data = & at24_data_24c1024 } ,
2018-11-29 21:58:58 +02:00
{ . compatible = " atmel,24c2048 " , . data = & at24_data_24c2048 } ,
2017-12-20 11:48:56 -05:00
{ /* END OF LIST */ } ,
2017-10-01 12:49:48 +02:00
} ;
MODULE_DEVICE_TABLE ( of , at24_of_match ) ;
2015-10-23 12:16:44 +03:00
static const struct acpi_device_id at24_acpi_ids [ ] = {
2017-12-20 11:48:56 -05:00
{ " INT3499 " , ( kernel_ulong_t ) & at24_data_INT3499 } ,
{ /* END OF LIST */ }
2015-10-23 12:16:44 +03:00
} ;
MODULE_DEVICE_TABLE ( acpi , at24_acpi_ids ) ;
2008-07-14 22:38:35 +02:00
/*
* This routine supports chips which consume multiple I2C addresses . It
* computes the addressing information to be used for a given r / w request .
* Assumes that sanity checks for offset happened at sysfs - layer .
2016-06-06 10:48:48 +02:00
*
* Slave address and byte offset derive from the offset . Always
* set the byte address ; on a multi - master board , another master
* may have changed the chip ' s " current " address pointer .
2008-07-14 22:38:35 +02:00
*/
2017-11-28 21:51:42 +01:00
static struct at24_client * at24_translate_offset ( struct at24_data * at24 ,
unsigned int * offset )
2008-07-14 22:38:35 +02:00
{
2017-12-13 11:56:23 +01:00
unsigned int i ;
2008-07-14 22:38:35 +02:00
2018-03-19 10:17:17 +01:00
if ( at24 - > flags & AT24_FLAG_ADDR16 ) {
2008-07-14 22:38:35 +02:00
i = * offset > > 16 ;
* offset & = 0xffff ;
} else {
i = * offset > > 8 ;
* offset & = 0xff ;
}
2017-11-28 21:51:42 +01:00
return & at24 - > client [ i ] ;
2008-07-14 22:38:35 +02:00
}
2018-03-19 10:17:14 +01:00
static struct device * at24_base_client_dev ( struct at24_data * at24 )
{
return & at24 - > client [ 0 ] . client - > dev ;
}
2017-12-08 11:28:30 -05:00
static size_t at24_adjust_read_count ( struct at24_data * at24 ,
unsigned int offset , size_t count )
{
unsigned int bits ;
size_t remainder ;
/*
* In case of multi - address chips that don ' t rollover reads to
* the next slave address : truncate the count to the slave boundary ,
* so that the read never straddles slaves .
*/
2018-03-19 10:17:17 +01:00
if ( at24 - > flags & AT24_FLAG_NO_RDROL ) {
bits = ( at24 - > flags & AT24_FLAG_ADDR16 ) ? 16 : 8 ;
2017-12-08 11:28:30 -05:00
remainder = BIT ( bits ) - offset ;
if ( count > remainder )
count = remainder ;
}
2017-12-18 18:16:46 +01:00
if ( count > at24_io_limit )
count = at24_io_limit ;
2017-12-08 11:28:30 -05:00
return count ;
}
2017-11-28 21:51:50 +01:00
static ssize_t at24_regmap_read ( struct at24_data * at24 , char * buf ,
unsigned int offset , size_t count )
{
unsigned long timeout , read_time ;
struct at24_client * at24_client ;
struct i2c_client * client ;
struct regmap * regmap ;
int ret ;
at24_client = at24_translate_offset ( at24 , & offset ) ;
regmap = at24_client - > regmap ;
client = at24_client - > client ;
2017-12-08 11:28:30 -05:00
count = at24_adjust_read_count ( at24 , offset , count ) ;
2017-11-28 21:51:50 +01:00
/* adjust offset for mac and serial read ops */
offset + = at24 - > offset_adj ;
2018-08-16 19:45:34 +02:00
timeout = jiffies + msecs_to_jiffies ( at24_write_timeout ) ;
do {
/*
* The timestamp shall be taken before the actual operation
* to avoid a premature timeout in case of high CPU load .
*/
read_time = jiffies ;
2017-11-28 21:51:50 +01:00
ret = regmap_bulk_read ( regmap , offset , buf , count ) ;
dev_dbg ( & client - > dev , " read %zu@%d --> %d (%ld) \n " ,
count , offset , ret , jiffies ) ;
if ( ! ret )
return count ;
2018-08-16 19:45:34 +02:00
usleep_range ( 1000 , 1500 ) ;
} while ( time_before ( read_time , timeout ) ) ;
2017-11-28 21:51:50 +01:00
return - ETIMEDOUT ;
}
2008-07-14 22:38:35 +02:00
/*
* Note that if the hardware write - protect pin is pulled high , the whole
* chip is normally write protected . But there are plenty of product
* variants here , including OTP fuses and partial chip protect .
*
2016-06-06 10:48:49 +02:00
* We only use page mode writes ; the alternative is sloooow . These routines
* write at most one page .
2008-07-14 22:38:35 +02:00
*/
2016-06-06 10:48:49 +02:00
static size_t at24_adjust_write_count ( struct at24_data * at24 ,
unsigned int offset , size_t count )
2008-07-14 22:38:35 +02:00
{
2017-12-13 11:56:23 +01:00
unsigned int next_page ;
2008-07-14 22:38:35 +02:00
/* write_max is at most a page */
if ( count > at24 - > write_max )
count = at24 - > write_max ;
/* Never roll over backwards, to the start of this page */
2018-03-19 10:17:17 +01:00
next_page = roundup ( offset + 1 , at24 - > page_size ) ;
2008-07-14 22:38:35 +02:00
if ( offset + count > next_page )
count = next_page - offset ;
2016-06-06 10:48:49 +02:00
return count ;
}
2017-11-28 21:51:45 +01:00
static ssize_t at24_regmap_write ( struct at24_data * at24 , const char * buf ,
unsigned int offset , size_t count )
{
unsigned long timeout , write_time ;
struct at24_client * at24_client ;
struct i2c_client * client ;
struct regmap * regmap ;
int ret ;
at24_client = at24_translate_offset ( at24 , & offset ) ;
regmap = at24_client - > regmap ;
client = at24_client - > client ;
count = at24_adjust_write_count ( at24 , offset , count ) ;
2018-08-16 19:45:34 +02:00
timeout = jiffies + msecs_to_jiffies ( at24_write_timeout ) ;
do {
/*
* The timestamp shall be taken before the actual operation
* to avoid a premature timeout in case of high CPU load .
*/
write_time = jiffies ;
2017-11-28 21:51:45 +01:00
ret = regmap_bulk_write ( regmap , offset , buf , count ) ;
dev_dbg ( & client - > dev , " write %zu@%d --> %d (%ld) \n " ,
count , offset , ret , jiffies ) ;
if ( ! ret )
return count ;
2018-08-16 19:45:34 +02:00
usleep_range ( 1000 , 1500 ) ;
} while ( time_before ( write_time , timeout ) ) ;
2017-11-28 21:51:45 +01:00
return - ETIMEDOUT ;
}
2016-06-06 10:48:44 +02:00
static int at24_read ( void * priv , unsigned int off , void * val , size_t count )
{
2018-03-19 10:17:03 +01:00
struct at24_data * at24 ;
struct device * dev ;
2016-06-06 10:48:44 +02:00
char * buf = val ;
2017-10-10 11:30:37 +05:30
int ret ;
2016-06-06 10:48:44 +02:00
2018-03-19 10:17:03 +01:00
at24 = priv ;
2018-03-19 10:17:14 +01:00
dev = at24_base_client_dev ( at24 ) ;
2018-03-19 10:17:03 +01:00
2016-06-06 10:48:44 +02:00
if ( unlikely ( ! count ) )
return count ;
2018-03-19 10:17:17 +01:00
if ( off + count > at24 - > byte_len )
2017-11-24 07:47:50 +01:00
return - EINVAL ;
2017-12-01 13:37:12 -05:00
ret = pm_runtime_get_sync ( dev ) ;
2017-10-10 11:30:37 +05:30
if ( ret < 0 ) {
2017-12-01 13:37:12 -05:00
pm_runtime_put_noidle ( dev ) ;
2017-10-10 11:30:37 +05:30
return ret ;
}
2016-06-06 10:48:44 +02:00
/*
* Read data from chip , protecting against concurrent updates
* from this host , but not from other I2C masters .
*/
mutex_lock ( & at24 - > lock ) ;
while ( count ) {
2018-03-19 10:17:06 +01:00
ret = at24_regmap_read ( at24 , buf , off , count ) ;
if ( ret < 0 ) {
2016-06-06 10:48:44 +02:00
mutex_unlock ( & at24 - > lock ) ;
2017-12-01 13:37:12 -05:00
pm_runtime_put ( dev ) ;
2018-03-19 10:17:06 +01:00
return ret ;
2016-06-06 10:48:44 +02:00
}
2018-03-19 10:17:06 +01:00
buf + = ret ;
off + = ret ;
count - = ret ;
2016-06-06 10:48:44 +02:00
}
mutex_unlock ( & at24 - > lock ) ;
2017-12-01 13:37:12 -05:00
pm_runtime_put ( dev ) ;
2017-10-10 11:30:37 +05:30
2016-06-06 10:48:44 +02:00
return 0 ;
}
2016-04-24 20:28:06 +01:00
static int at24_write ( void * priv , unsigned int off , void * val , size_t count )
2008-07-14 22:38:35 +02:00
{
2018-03-19 10:17:03 +01:00
struct at24_data * at24 ;
struct device * dev ;
2016-04-24 20:28:06 +01:00
char * buf = val ;
2017-10-10 11:30:37 +05:30
int ret ;
2008-07-14 22:38:35 +02:00
2018-03-19 10:17:03 +01:00
at24 = priv ;
2018-03-19 10:17:14 +01:00
dev = at24_base_client_dev ( at24 ) ;
2018-03-19 10:17:03 +01:00
2008-07-14 22:38:35 +02:00
if ( unlikely ( ! count ) )
2016-04-24 20:28:06 +01:00
return - EINVAL ;
2008-07-14 22:38:35 +02:00
2018-03-19 10:17:17 +01:00
if ( off + count > at24 - > byte_len )
2017-11-24 07:47:50 +01:00
return - EINVAL ;
2017-12-01 13:37:12 -05:00
ret = pm_runtime_get_sync ( dev ) ;
2017-10-10 11:30:37 +05:30
if ( ret < 0 ) {
2017-12-01 13:37:12 -05:00
pm_runtime_put_noidle ( dev ) ;
2017-10-10 11:30:37 +05:30
return ret ;
}
2008-07-14 22:38:35 +02:00
/*
* Write data to chip , protecting against concurrent updates
* from this host , but not from other I2C masters .
*/
mutex_lock ( & at24 - > lock ) ;
2017-12-19 11:28:54 +01:00
gpiod_set_value_cansleep ( at24 - > wp_gpio , 0 ) ;
2008-07-14 22:38:35 +02:00
while ( count ) {
2018-03-19 10:17:07 +01:00
ret = at24_regmap_write ( at24 , buf , off , count ) ;
if ( ret < 0 ) {
2017-12-19 11:28:54 +01:00
gpiod_set_value_cansleep ( at24 - > wp_gpio , 1 ) ;
2016-04-24 20:28:06 +01:00
mutex_unlock ( & at24 - > lock ) ;
2017-12-01 13:37:12 -05:00
pm_runtime_put ( dev ) ;
2018-03-19 10:17:07 +01:00
return ret ;
2008-07-14 22:38:35 +02:00
}
2018-03-19 10:17:07 +01:00
buf + = ret ;
off + = ret ;
count - = ret ;
2008-07-14 22:38:35 +02:00
}
2017-12-19 11:28:54 +01:00
gpiod_set_value_cansleep ( at24 - > wp_gpio , 1 ) ;
2008-07-14 22:38:35 +02:00
mutex_unlock ( & at24 - > lock ) ;
2017-12-01 13:37:12 -05:00
pm_runtime_put ( dev ) ;
2017-10-10 11:30:37 +05:30
2016-02-26 20:59:20 +01:00
return 0 ;
}
2018-08-07 12:07:43 +02:00
static const struct at24_chip_data * at24_get_chip_data ( struct device * dev )
2018-03-19 10:17:16 +01:00
{
struct device_node * of_node = dev - > of_node ;
const struct at24_chip_data * cdata ;
const struct i2c_device_id * id ;
id = i2c_match_id ( at24_ids , to_i2c_client ( dev ) ) ;
/*
* The I2C core allows OF nodes compatibles to match against the
* I2C device ID table as a fallback , so check not only if an OF
* node is present but also if it matches an OF device ID entry .
*/
if ( of_node & & of_match_device ( at24_of_match , dev ) )
cdata = of_device_get_match_data ( dev ) ;
else if ( id )
2018-05-07 12:08:37 +02:00
cdata = ( void * ) id - > driver_data ;
2018-03-19 10:17:16 +01:00
else
cdata = acpi_device_get_match_data ( dev ) ;
if ( ! cdata )
2018-08-07 12:07:43 +02:00
return ERR_PTR ( - ENODEV ) ;
2018-03-19 10:17:16 +01:00
2018-08-07 12:07:43 +02:00
return cdata ;
2018-03-19 10:17:16 +01:00
}
2018-03-23 09:05:07 +01:00
static int at24_make_dummy_client ( struct at24_data * at24 , unsigned int index ,
struct regmap_config * regmap_config )
{
struct i2c_client * base_client , * dummy_client ;
struct regmap * regmap ;
struct device * dev ;
base_client = at24 - > client [ 0 ] . client ;
dev = & base_client - > dev ;
2019-05-20 09:10:41 +02:00
dummy_client = devm_i2c_new_dummy_device ( dev , base_client - > adapter ,
base_client - > addr + index ) ;
if ( IS_ERR ( dummy_client ) )
return PTR_ERR ( dummy_client ) ;
2018-03-23 09:05:07 +01:00
regmap = devm_regmap_init_i2c ( dummy_client , regmap_config ) ;
2019-05-20 09:10:41 +02:00
if ( IS_ERR ( regmap ) )
2018-03-23 09:05:07 +01:00
return PTR_ERR ( regmap ) ;
at24 - > client [ index ] . client = dummy_client ;
at24 - > client [ index ] . regmap = regmap ;
return 0 ;
}
2017-11-28 21:51:50 +01:00
static unsigned int at24_get_offset_adj ( u8 flags , unsigned int byte_len )
{
if ( flags & AT24_FLAG_MAC ) {
/* EUI-48 starts from 0x9a, EUI-64 from 0x98 */
return 0xa0 - byte_len ;
} else if ( flags & AT24_FLAG_SERIAL & & flags & AT24_FLAG_ADDR16 ) {
/*
* For 16 bit address pointers , the word address must contain
* a ' 10 ' sequence in bits 11 and 10 regardless of the
* intended position of the address pointer .
*/
return 0x0800 ;
} else if ( flags & AT24_FLAG_SERIAL ) {
/*
* Otherwise the word address must begin with a ' 10 ' sequence ,
* regardless of the intended address .
*/
return 0x0080 ;
} else {
return 0 ;
}
}
2018-03-19 10:17:15 +01:00
static int at24_probe ( struct i2c_client * client )
2008-07-14 22:38:35 +02:00
{
2017-12-18 18:24:43 +01:00
struct regmap_config regmap_config = { } ;
2018-03-19 10:17:02 +01:00
struct nvmem_config nvmem_config = { } ;
2018-08-07 12:07:43 +02:00
u32 byte_len , page_size , flags , addrw ;
const struct at24_chip_data * cdata ;
2018-03-19 10:17:12 +01:00
struct device * dev = & client - > dev ;
2018-03-19 10:17:21 +01:00
bool i2c_fn_i2c , i2c_fn_block ;
2018-03-19 10:17:03 +01:00
unsigned int i , num_addresses ;
struct at24_data * at24 ;
2018-03-19 10:17:18 +01:00
struct regmap * regmap ;
2018-03-19 10:17:03 +01:00
bool writable ;
2016-08-12 13:32:57 +02:00
u8 test_byte ;
2018-03-19 10:17:03 +01:00
int err ;
2008-07-14 22:38:35 +02:00
2018-03-19 10:17:21 +01:00
i2c_fn_i2c = i2c_check_functionality ( client - > adapter , I2C_FUNC_I2C ) ;
i2c_fn_block = i2c_check_functionality ( client - > adapter ,
I2C_FUNC_SMBUS_WRITE_I2C_BLOCK ) ;
2018-08-07 12:07:43 +02:00
cdata = at24_get_chip_data ( dev ) ;
if ( IS_ERR ( cdata ) )
return PTR_ERR ( cdata ) ;
err = device_property_read_u32 ( dev , " pagesize " , & page_size ) ;
2018-03-19 10:17:16 +01:00
if ( err )
2018-08-07 12:07:43 +02:00
/*
* This is slow , but we can ' t know all eeproms , so we better
2019-06-25 09:55:44 +02:00
* play safe . Specifying custom eeprom - types via device tree
* or properties is recommended anyhow .
2018-08-07 12:07:43 +02:00
*/
page_size = 1 ;
flags = cdata - > flags ;
if ( device_property_present ( dev , " read-only " ) )
flags | = AT24_FLAG_READONLY ;
if ( device_property_present ( dev , " no-read-rollover " ) )
flags | = AT24_FLAG_NO_RDROL ;
err = device_property_read_u32 ( dev , " address-width " , & addrw ) ;
if ( ! err ) {
switch ( addrw ) {
case 8 :
if ( flags & AT24_FLAG_ADDR16 )
dev_warn ( dev ,
" Override address width to be 8, while default is 16 \n " ) ;
flags & = ~ AT24_FLAG_ADDR16 ;
break ;
case 16 :
flags | = AT24_FLAG_ADDR16 ;
break ;
default :
dev_warn ( dev , " Bad \" address-width \" property: %u \n " ,
addrw ) ;
}
}
err = device_property_read_u32 ( dev , " size " , & byte_len ) ;
if ( err )
byte_len = cdata - > byte_len ;
2008-07-14 22:38:35 +02:00
2018-03-19 10:17:21 +01:00
if ( ! i2c_fn_i2c & & ! i2c_fn_block )
2018-08-07 12:07:43 +02:00
page_size = 1 ;
2018-03-19 10:17:18 +01:00
2018-08-07 12:07:43 +02:00
if ( ! page_size ) {
2018-03-19 10:17:12 +01:00
dev_err ( dev , " page_size must not be 0! \n " ) ;
2013-05-28 13:00:20 -07:00
return - EINVAL ;
2010-11-17 13:00:49 +01:00
}
2018-03-19 10:17:19 +01:00
2018-08-07 12:07:43 +02:00
if ( ! is_power_of_2 ( page_size ) )
2018-03-19 10:17:12 +01:00
dev_warn ( dev , " page_size looks suspicious (no power of 2)! \n " ) ;
2008-07-14 22:38:35 +02:00
2019-02-05 14:18:25 +01:00
err = device_property_read_u32 ( dev , " num-addresses " , & num_addresses ) ;
if ( err ) {
if ( flags & AT24_FLAG_TAKE8ADDR )
num_addresses = 8 ;
else
num_addresses = DIV_ROUND_UP ( byte_len ,
( flags & AT24_FLAG_ADDR16 ) ? 65536 : 256 ) ;
}
2008-07-14 22:38:35 +02:00
2018-08-07 12:07:43 +02:00
if ( ( flags & AT24_FLAG_SERIAL ) & & ( flags & AT24_FLAG_MAC ) ) {
2018-03-19 10:17:18 +01:00
dev_err ( dev ,
" invalid device data - cannot have both AT24_FLAG_SERIAL & AT24_FLAG_MAC. " ) ;
return - EINVAL ;
}
2017-12-18 18:24:43 +01:00
regmap_config . val_bits = 8 ;
2018-08-07 12:07:43 +02:00
regmap_config . reg_bits = ( flags & AT24_FLAG_ADDR16 ) ? 16 : 8 ;
2018-03-19 10:17:01 +01:00
regmap_config . disable_locking = true ;
2017-11-28 21:51:40 +01:00
2018-03-19 10:17:18 +01:00
regmap = devm_regmap_init_i2c ( client , & regmap_config ) ;
if ( IS_ERR ( regmap ) )
return PTR_ERR ( regmap ) ;
eeprom: at24: use struct_size() in devm_kzalloc()
One of the more common cases of allocation size calculations is finding
the size of a structure that has a zero-sized array at the end, along
with memory for some number of elements for that array. For example:
struct foo {
int stuff;
struct boo entry[];
};
size = sizeof(struct foo) + count * sizeof(struct boo);
instance = devm_kzalloc(dev, size, GFP_KERNEL);
Instead of leaving these open-coded and prone to type mistakes, we can
now use the new struct_size() helper:
instance = devm_kzalloc(dev, struct_size(instance, entry, count), GFP_KERNEL);
Notice that, in this case, variable at24_size is not necessary, hence it
is removed.
This code was detected with the help of Coccinelle.
Signed-off-by: Gustavo A. R. Silva <gustavo@embeddedor.com>
Signed-off-by: Bartosz Golaszewski <bgolaszewski@baylibre.com>
2019-05-29 11:30:52 -05:00
at24 = devm_kzalloc ( dev , struct_size ( at24 , client , num_addresses ) ,
GFP_KERNEL ) ;
2013-05-28 13:00:20 -07:00
if ( ! at24 )
return - ENOMEM ;
2008-07-14 22:38:35 +02:00
mutex_init ( & at24 - > lock ) ;
2018-08-07 12:07:43 +02:00
at24 - > byte_len = byte_len ;
at24 - > page_size = page_size ;
at24 - > flags = flags ;
2008-07-14 22:38:35 +02:00
at24 - > num_addresses = num_addresses ;
2018-08-07 12:07:43 +02:00
at24 - > offset_adj = at24_get_offset_adj ( flags , byte_len ) ;
2018-03-19 10:17:18 +01:00
at24 - > client [ 0 ] . client = client ;
at24 - > client [ 0 ] . regmap = regmap ;
2008-07-14 22:38:35 +02:00
2018-03-19 10:17:12 +01:00
at24 - > wp_gpio = devm_gpiod_get_optional ( dev , " wp " , GPIOD_OUT_HIGH ) ;
2017-12-19 11:28:54 +01:00
if ( IS_ERR ( at24 - > wp_gpio ) )
return PTR_ERR ( at24 - > wp_gpio ) ;
2018-08-07 12:07:43 +02:00
writable = ! ( flags & AT24_FLAG_READONLY ) ;
2008-07-14 22:38:35 +02:00
if ( writable ) {
2017-12-18 18:16:46 +01:00
at24 - > write_max = min_t ( unsigned int ,
2018-08-07 12:07:43 +02:00
page_size , at24_io_limit ) ;
2018-03-19 10:17:21 +01:00
if ( ! i2c_fn_i2c & & at24 - > write_max > I2C_SMBUS_BLOCK_MAX )
2017-11-28 21:51:54 +01:00
at24 - > write_max = I2C_SMBUS_BLOCK_MAX ;
2008-07-14 22:38:35 +02:00
}
/* use dummy devices for multiple-address chips */
for ( i = 1 ; i < num_addresses ; i + + ) {
2018-03-23 09:05:07 +01:00
err = at24_make_dummy_client ( at24 , i , & regmap_config ) ;
2019-05-20 09:10:41 +02:00
if ( err )
2018-03-23 09:05:07 +01:00
return err ;
2008-07-14 22:38:35 +02:00
}
2018-03-19 10:17:12 +01:00
nvmem_config . name = dev_name ( dev ) ;
nvmem_config . dev = dev ;
2018-03-19 10:17:02 +01:00
nvmem_config . read_only = ! writable ;
2019-07-28 18:41:38 +02:00
nvmem_config . root_only = ! ( flags & AT24_FLAG_IRUGO ) ;
2018-03-19 10:17:02 +01:00
nvmem_config . owner = THIS_MODULE ;
nvmem_config . compat = true ;
2018-03-19 10:17:12 +01:00
nvmem_config . base_dev = dev ;
2018-03-19 10:17:02 +01:00
nvmem_config . reg_read = at24_read ;
nvmem_config . reg_write = at24_write ;
nvmem_config . priv = at24 ;
nvmem_config . stride = 1 ;
nvmem_config . word_size = 1 ;
2018-08-07 12:07:43 +02:00
nvmem_config . size = byte_len ;
2018-03-19 10:17:02 +01:00
2018-04-05 09:49:20 +02:00
at24 - > nvmem = devm_nvmem_register ( dev , & nvmem_config ) ;
2019-05-20 09:10:42 +02:00
if ( IS_ERR ( at24 - > nvmem ) )
return PTR_ERR ( at24 - > nvmem ) ;
i2c_set_clientdata ( client , at24 ) ;
/* enable runtime pm */
pm_runtime_set_active ( dev ) ;
pm_runtime_enable ( dev ) ;
/*
* Perform a one - byte test read to verify that the
* chip is functional .
*/
err = at24_read ( at24 , 0 , & test_byte , 1 ) ;
pm_runtime_idle ( dev ) ;
if ( err ) {
pm_runtime_disable ( dev ) ;
return - ENODEV ;
2016-02-26 20:59:20 +01:00
}
2008-07-14 22:38:35 +02:00
2019-10-04 10:37:48 +02:00
if ( writable )
dev_info ( dev , " %u byte %s EEPROM, writable, %u bytes/write \n " ,
byte_len , client - > name , at24 - > write_max ) ;
else
dev_info ( dev , " %u byte %s EEPROM, read-only \n " ,
byte_len , client - > name ) ;
2008-07-14 22:38:35 +02:00
return 0 ;
}
2012-11-19 13:26:02 -05:00
static int at24_remove ( struct i2c_client * client )
2008-07-14 22:38:35 +02:00
{
2017-10-10 11:30:37 +05:30
pm_runtime_disable ( & client - > dev ) ;
pm_runtime_set_suspended ( & client - > dev ) ;
2008-07-14 22:38:35 +02:00
return 0 ;
}
static struct i2c_driver at24_driver = {
. driver = {
. name = " at24 " ,
2017-10-01 12:49:48 +02:00
. of_match_table = at24_of_match ,
2015-10-23 12:16:44 +03:00
. acpi_match_table = ACPI_PTR ( at24_acpi_ids ) ,
2008-07-14 22:38:35 +02:00
} ,
2018-03-19 10:17:15 +01:00
. probe_new = at24_probe ,
2012-11-19 13:21:23 -05:00
. remove = at24_remove ,
2008-07-14 22:38:35 +02:00
. id_table = at24_ids ,
} ;
static int __init at24_init ( void )
{
2017-12-18 18:16:46 +01:00
if ( ! at24_io_limit ) {
pr_err ( " at24: at24_io_limit must not be 0! \n " ) ;
2010-11-17 13:00:49 +01:00
return - EINVAL ;
}
2017-12-18 18:16:46 +01:00
at24_io_limit = rounddown_pow_of_two ( at24_io_limit ) ;
2008-07-14 22:38:35 +02:00
return i2c_add_driver ( & at24_driver ) ;
}
module_init ( at24_init ) ;
static void __exit at24_exit ( void )
{
i2c_del_driver ( & at24_driver ) ;
}
module_exit ( at24_exit ) ;
MODULE_DESCRIPTION ( " Driver for most I2C EEPROMs " ) ;
MODULE_AUTHOR ( " David Brownell and Wolfram Sang " ) ;
MODULE_LICENSE ( " GPL " ) ;