147 lines
5.0 KiB
C
Raw Normal View History

2020-11-27 11:16:22 +01:00
#pragma once
//
2013-09-30 16:38:08 +02:00
// FILE: I2C_eeprom.h
// AUTHOR: Rob Tillaart
2022-10-31 16:53:19 +01:00
// VERSION: 1.6.2
2020-11-27 11:16:22 +01:00
// PURPOSE: Arduino Library for external I2C EEPROM 24LC256 et al.
// URL: https://github.com/RobTillaart/I2C_EEPROM.git
2013-09-30 16:38:08 +02:00
//
2020-11-27 11:16:22 +01:00
// HISTORY: See I2C_eeprom.cpp
2013-09-30 16:38:08 +02:00
#include "Arduino.h"
2021-01-29 12:31:58 +01:00
#include "Wire.h"
2022-10-31 16:53:19 +01:00
#define I2C_EEPROM_VERSION (F("1.6.2"))
2021-01-29 12:31:58 +01:00
#define I2C_DEVICESIZE_24LC512 65536
#define I2C_DEVICESIZE_24LC256 32768
#define I2C_DEVICESIZE_24LC128 16384
#define I2C_DEVICESIZE_24LC64 8192
#define I2C_DEVICESIZE_24LC32 4096
#define I2C_DEVICESIZE_24LC16 2048
#define I2C_DEVICESIZE_24LC08 1024
#define I2C_DEVICESIZE_24LC04 512
#define I2C_DEVICESIZE_24LC02 256
#define I2C_DEVICESIZE_24LC01 128
#ifndef UNIT_TEST_FRIEND
#define UNIT_TEST_FRIEND
#endif
class I2C_eeprom
2013-09-30 16:38:08 +02:00
{
public:
2019-09-03 11:41:28 +02:00
/**
2021-01-29 12:31:58 +01:00
* Initializes the EEPROM with a default deviceSize of I2C_DEVICESIZE_24LC256 (32K EEPROM)
2019-09-03 11:41:28 +02:00
*/
2021-01-29 12:31:58 +01:00
I2C_eeprom(const uint8_t deviceAddress, TwoWire *wire = &Wire);
2019-09-03 11:41:28 +02:00
/**
* Initializes the EEPROM for the given device address.
*
* It will try to guess page size and address word size based on the size of the device.
2020-11-27 11:16:22 +01:00
*
2019-09-03 11:41:28 +02:00
* @param deviceAddress Byte address of the device.
* @param deviceSize Max size in bytes of the device (divide your device size in Kbits by 8)
2021-01-29 12:31:58 +01:00
* @param wire Select alternative Wire interface
2019-09-03 11:41:28 +02:00
*/
2021-01-29 12:31:58 +01:00
I2C_eeprom(const uint8_t deviceAddress, const uint32_t deviceSize, TwoWire *wire = &Wire);
2019-09-03 11:41:28 +02:00
2020-11-27 11:16:22 +01:00
#if defined (ESP8266) || defined(ESP32)
2022-10-31 16:53:19 +01:00
// set the I2C pins explicitly (overrule)
2021-01-29 12:31:58 +01:00
bool begin(uint8_t sda, uint8_t scl);
2020-11-27 11:16:22 +01:00
#endif
2022-10-31 16:53:19 +01:00
// use default I2C pins.
2021-01-29 12:31:58 +01:00
bool begin();
bool isConnected();
2020-11-27 11:16:22 +01:00
2022-06-12 12:47:07 +02:00
2021-12-19 20:05:24 +01:00
// writes a byte to memoryAddress
2022-06-12 12:47:07 +02:00
// returns I2C status, 0 = OK
2021-01-29 12:31:58 +01:00
int writeByte(const uint16_t memoryAddress, const uint8_t value);
2021-12-19 20:05:24 +01:00
// writes length bytes from buffer to EEPROM
2022-06-12 12:47:07 +02:00
// returns I2C status, 0 = OK
2022-06-07 17:43:11 +02:00
int writeBlock(const uint16_t memoryAddress, const uint8_t * buffer, const uint16_t length);
2021-12-19 20:05:24 +01:00
// set length bytes in the EEPROM to the same value.
2022-06-12 12:47:07 +02:00
// returns I2C status, 0 = OK
2021-01-29 12:31:58 +01:00
int setBlock(const uint16_t memoryAddress, const uint8_t value, const uint16_t length);
2019-09-03 11:41:28 +02:00
2021-12-19 20:05:24 +01:00
// returns the value stored in memoryAddress
2020-11-27 11:16:22 +01:00
uint8_t readByte(const uint16_t memoryAddress);
2021-12-19 20:05:24 +01:00
// reads length bytes into buffer
2022-06-12 12:47:07 +02:00
// returns bytes read.
2022-06-07 17:43:11 +02:00
uint16_t readBlock(const uint16_t memoryAddress, uint8_t * buffer, const uint16_t length);
2020-11-27 11:16:22 +01:00
2021-12-19 20:05:24 +01:00
// updates a byte at memoryAddress, writes only if there is a new value.
// return 0 if data is same or written OK, error code otherwise.
2021-01-29 12:31:58 +01:00
int updateByte(const uint16_t memoryAddress, const uint8_t value);
2021-12-19 20:05:24 +01:00
// updates a block in memory, writes only if there is a new value.
// only to be used when you expect to write same buffer multiple times.
// test your performance gains!
2022-06-12 12:47:07 +02:00
// returns bytes written.
uint16_t updateBlock(const uint16_t memoryAddress, const uint8_t * buffer, const uint16_t length);
2021-01-29 12:31:58 +01:00
2022-06-07 17:43:11 +02:00
// same functions as above but with verify
2022-06-12 12:47:07 +02:00
// return false if write or verify failed.
2022-06-07 17:43:11 +02:00
bool writeByteVerify(const uint16_t memoryAddress, const uint8_t value);
bool writeBlockVerify(const uint16_t memoryAddress, const uint8_t * buffer, const uint16_t length);
bool setBlockVerify(const uint16_t memoryAddress, const uint8_t value, const uint16_t length);
bool updateByteVerify(const uint16_t memoryAddress, const uint8_t value);
bool updateBlockVerify(const uint16_t memoryAddress, const uint8_t * buffer, const uint16_t length);
2022-06-12 12:47:07 +02:00
2022-06-07 17:43:11 +02:00
// Meta data functions
2021-01-29 12:31:58 +01:00
uint32_t determineSize(const bool debug = false);
uint32_t getDeviceSize() { return _deviceSize; };
uint8_t getPageSize() { return _pageSize; };
uint8_t getPageSize(uint32_t deviceSize);
uint32_t getLastWrite() { return _lastWrite; };
2021-11-08 13:26:03 +01:00
// TWR = WriteCycleTime
// 5 ms is minimum, one can add extra ms here to adjust timing of both read() and write()
void setExtraWriteCycleTime(uint8_t ms) { _extraTWR = ms; };
uint8_t getExtraWriteCycleTime() { return _extraTWR; };
2022-06-12 12:47:07 +02:00
private:
2020-11-27 11:16:22 +01:00
uint8_t _deviceAddress;
2022-06-12 12:47:07 +02:00
uint32_t _lastWrite = 0; // for waitEEReady
2021-01-29 12:31:58 +01:00
uint32_t _deviceSize;
2020-11-27 11:16:22 +01:00
uint8_t _pageSize;
2021-11-08 13:26:03 +01:00
uint8_t _extraTWR = 0; // milliseconds
2021-12-19 20:05:24 +01:00
// 24LC32..24LC512 use two bytes for memory address
// 24LC01..24LC16 use one-byte addresses + part of device address
2020-11-27 11:16:22 +01:00
bool _isAddressSizeTwoWords;
2020-11-27 11:16:22 +01:00
void _beginTransmission(const uint16_t memoryAddress);
2022-06-12 12:47:07 +02:00
// returns I2C status, 0 = OK
int _pageBlock(const uint16_t memoryAddress, const uint8_t * buffer, const uint16_t length, const bool incrBuffer);
// returns I2C status, 0 = OK
int _WriteBlock(const uint16_t memoryAddress, const uint8_t * buffer, const uint8_t length);
// returns bytes read.
uint8_t _ReadBlock(const uint16_t memoryAddress, uint8_t * buffer, const uint8_t length);
2021-12-19 20:05:24 +01:00
// to optimize the write latency of the EEPROM
2020-11-27 11:16:22 +01:00
void _waitEEReady();
2021-01-29 12:31:58 +01:00
TwoWire * _wire;
2022-06-12 12:47:07 +02:00
bool _debug = false;
2021-01-29 12:31:58 +01:00
UNIT_TEST_FRIEND;
2013-09-30 16:38:08 +02:00
};
2021-12-19 20:05:24 +01:00
2020-11-27 11:16:22 +01:00
// -- END OF FILE --
2021-12-19 20:05:24 +01:00