496 lines
11 KiB
C++
Raw Normal View History

2021-02-03 17:20:20 +01:00
//
// FILE: I2C_24LC1025.cpp
// AUTHOR: Rob Tillaart
2023-09-11 10:03:32 +02:00
// VERSION: 0.2.5
2021-02-03 17:20:20 +01:00
// PURPOSE: I2C_24LC1025 library for Arduino with EEPROM I2C_24LC1025 et al.
2022-06-08 10:45:22 +02:00
// URL: https://github.com/RobTillaart/I2C_24LC1025
2021-02-03 17:20:20 +01:00
#include "I2C_24LC1025.h"
2022-06-12 17:10:15 +02:00
// I2C buffer needs max 2 bytes for EEPROM address
// 1 byte for EEPROM register address is available in transmit buffer
2023-05-11 12:39:35 +02:00
#if defined(ESP32) || defined(ESP8266) || defined(PICO_RP2040)
2021-02-03 17:20:20 +01:00
#define I2C_BUFFERSIZE 128
#else
2022-06-12 17:10:15 +02:00
#define I2C_BUFFERSIZE 30 // AVR, STM
2021-02-03 17:20:20 +01:00
#endif
////////////////////////////////////////////////////////////////////
//
// PUBLIC FUNCTIONS
//
I2C_24LC1025::I2C_24LC1025(uint8_t deviceAddress, TwoWire * wire)
{
_deviceAddress = deviceAddress;
2021-09-02 07:36:22 +02:00
_deviceSize = I2C_DEVICESIZE_24LC1025;
_pageSize = I2C_PAGESIZE_24LC1025;
2021-02-03 17:20:20 +01:00
_wire = wire;
}
2023-05-02 13:37:13 +02:00
#if defined(ESP8266) || defined(ESP32)
2023-05-11 12:39:35 +02:00
2023-09-11 10:03:32 +02:00
bool I2C_24LC1025::begin(uint8_t sda, uint8_t scl, int8_t writeProtectPin)
2021-02-03 17:20:20 +01:00
{
if ((sda < 255) && (scl < 255))
{
_wire->begin(sda, scl);
}
else
{
_wire->begin();
}
_lastWrite = 0;
2023-09-11 10:03:32 +02:00
_writeProtectPin = writeProtectPin;
if (_writeProtectPin >= 0)
{
pinMode(_writeProtectPin, OUTPUT);
preventWrite();
}
2021-02-03 17:20:20 +01:00
return isConnected();
}
2023-05-11 12:39:35 +02:00
#elif defined(ARDUINO_ARCH_RP2040) && !defined(__MBED__)
2021-02-03 17:20:20 +01:00
2023-09-11 10:03:32 +02:00
bool I2C_24LC1025::begin(uint8_t sda, uint8_t scl, int8_t writeProtectPin)
2023-05-02 13:37:13 +02:00
{
if ((sda < 255) && (scl < 255))
{
_wire->setSCL(scl);
_wire->setSDA(sda);
_wire->begin();
}
_lastWrite = 0;
2023-09-11 10:03:32 +02:00
_writeProtectPin = writeProtectPin;
if (_writeProtectPin >= 0)
{
pinMode(_writeProtectPin, OUTPUT);
preventWrite();
}
2023-05-02 13:37:13 +02:00
return isConnected();
}
2023-09-11 10:03:32 +02:00
2023-05-02 13:37:13 +02:00
#endif
2023-05-11 12:39:35 +02:00
2023-09-11 10:03:32 +02:00
bool I2C_24LC1025::begin(int8_t writeProtectPin)
2021-02-03 17:20:20 +01:00
{
_wire->begin();
_lastWrite = 0;
2023-09-11 10:03:32 +02:00
_writeProtectPin = writeProtectPin;
if (_writeProtectPin >= 0)
{
pinMode(_writeProtectPin, OUTPUT);
preventWrite();
}
2021-02-03 17:20:20 +01:00
return isConnected();
}
bool I2C_24LC1025::isConnected()
{
_wire->beginTransmission(_deviceAddress);
return (_wire->endTransmission() == 0);
}
2022-06-08 10:45:22 +02:00
/////////////////////////////////////////////////////////////
//
// WRITE SECTION
//
// returns I2C status, 0 = OK
int I2C_24LC1025::writeByte(const uint32_t memoryAddress, const uint8_t data)
2021-02-03 17:20:20 +01:00
{
2022-06-08 10:45:22 +02:00
int rv = _WriteBlock(memoryAddress, &data, 1);
2021-02-03 17:20:20 +01:00
return rv;
}
2022-06-08 10:45:22 +02:00
// returns I2C status, 0 = OK
2021-02-03 17:20:20 +01:00
int I2C_24LC1025::setBlock(const uint32_t memoryAddress, const uint8_t data, const uint32_t length)
{
uint8_t buffer[I2C_BUFFERSIZE];
for (uint8_t i = 0; i < I2C_BUFFERSIZE; i++)
{
buffer[i] = data;
}
int rv = _pageBlock(memoryAddress, buffer, length, false);
return rv;
}
2022-06-08 10:45:22 +02:00
// returns I2C status, 0 = OK
2022-06-12 17:10:15 +02:00
int I2C_24LC1025::writeBlock(const uint32_t memoryAddress, const uint8_t * buffer, const uint32_t length)
2021-02-03 17:20:20 +01:00
{
int rv = _pageBlock(memoryAddress, buffer, length, true);
return rv;
}
2022-06-08 10:45:22 +02:00
/////////////////////////////////////////////////////////////
//
// READ SECTION
//
// returns the value stored in memoryAddress
2021-02-03 17:20:20 +01:00
uint8_t I2C_24LC1025::readByte(const uint32_t memoryAddress)
{
uint8_t rdata;
_ReadBlock(memoryAddress, &rdata, 1);
return rdata;
}
2022-06-12 17:10:15 +02:00
2022-06-08 10:45:22 +02:00
// returns bytes read.
2022-06-12 17:10:15 +02:00
uint32_t I2C_24LC1025::readBlock(const uint32_t memoryAddress, uint8_t * buffer, const uint32_t length)
2021-02-03 17:20:20 +01:00
{
uint32_t addr = memoryAddress;
uint32_t len = length;
uint32_t rv = 0;
if ((addr < 0x10000) && ((addr + len) > 0x10000))
{
uint32_t sublen = 0x10000 - addr;
rv = readBlock(addr, (uint8_t *) buffer, sublen);
rv += readBlock(0x10000, (uint8_t *) &buffer[sublen], len - sublen);
return rv;
}
while (len > 0)
{
uint8_t cnt = I2C_BUFFERSIZE;
if (cnt > len) cnt = len;
rv += _ReadBlock(addr, buffer, cnt);
addr += cnt;
buffer += cnt;
len -= cnt;
}
return rv;
}
2022-06-08 10:45:22 +02:00
/////////////////////////////////////////////////////////////
//
// UPDATE SECTION
//
// returns 0 == OK
2021-02-03 17:20:20 +01:00
int I2C_24LC1025::updateByte(const uint32_t memoryAddress, const uint8_t data)
{
if (data == readByte(memoryAddress)) return 0;
return writeByte(memoryAddress, data);
}
2022-06-08 10:45:22 +02:00
// returns bytes written.
2022-06-12 17:10:15 +02:00
uint32_t I2C_24LC1025::updateBlock(const uint32_t memoryAddress, const uint8_t * buffer, const uint32_t length)
2021-02-03 17:20:20 +01:00
{
uint32_t addr = memoryAddress;
uint32_t len = length;
uint32_t rv = 0;
while (len > 0)
{
uint8_t buf[I2C_BUFFERSIZE];
uint8_t cnt = I2C_BUFFERSIZE;
2022-06-12 17:10:15 +02:00
2021-02-03 17:20:20 +01:00
if (cnt > len) cnt = len;
rv += _ReadBlock(addr, buf, cnt);
if (memcmp(buffer, buf, cnt) != 0)
{
_pageBlock(addr, buffer, cnt, true);
}
addr += cnt;
buffer += cnt;
len -= cnt;
}
return rv;
}
2022-06-08 10:45:22 +02:00
/////////////////////////////////////////////////////////////
//
// VERIFY SECTION
//
// return false if write or verify failed.
bool I2C_24LC1025::writeByteVerify(const uint32_t memoryAddress, const uint8_t value)
{
if (writeByte(memoryAddress, value) != 0 ) return false;
uint8_t data = readByte(memoryAddress);
return (data == value);
}
// return false if write or verify failed.
2022-06-12 17:10:15 +02:00
bool I2C_24LC1025::writeBlockVerify(const uint32_t memoryAddress, const uint8_t * buffer, const uint32_t length)
2022-06-08 10:45:22 +02:00
{
if (writeBlock(memoryAddress, buffer, length) != 0) return false;
uint8_t data[length];
if (readBlock(memoryAddress, data, length) != length) return false;
return memcmp(data, buffer, length) == 0;
}
// return false if write or verify failed.
bool I2C_24LC1025::setBlockVerify(const uint32_t memoryAddress, const uint8_t value, const uint32_t length)
{
if (setBlock(memoryAddress, value, length) != 0) return false;
uint8_t data[length];
if (readBlock(memoryAddress, data, length) != length) return false;
for (uint32_t i = 0; i < length; i++)
{
if (data[i] != value) return false;
}
return true;
}
// return false if write or verify failed.
bool I2C_24LC1025::updateByteVerify(const uint32_t memoryAddress, const uint8_t value)
{
if (updateByte(memoryAddress, value) != 0 ) return false;
uint8_t data = readByte(memoryAddress);
return (data == value);
}
// return false if write or verify failed.
2022-06-12 17:10:15 +02:00
bool I2C_24LC1025::updateBlockVerify(const uint32_t memoryAddress, const uint8_t * buffer, const uint32_t length)
2022-06-08 10:45:22 +02:00
{
if (updateBlock(memoryAddress, buffer, length) != length) return false;
uint8_t data[length];
if (readBlock(memoryAddress, data, length) != length) return false;
return memcmp(data, buffer, length) == 0;
}
2021-02-03 17:20:20 +01:00
2023-09-11 10:03:32 +02:00
/////////////////////////////////////////////////////////////
2023-05-02 13:37:13 +02:00
//
// METADATA SECTION
//
uint32_t I2C_24LC1025::getDeviceSize()
{
return _deviceSize;
}
uint8_t I2C_24LC1025::getPageSize()
{
return _pageSize;
}
uint32_t I2C_24LC1025::getLastWrite()
{
return _lastWrite;
}
void I2C_24LC1025::setExtraWriteCycleTime(uint8_t ms)
{
_extraTWR = ms;
}
uint8_t I2C_24LC1025::getExtraWriteCycleTime()
{
return _extraTWR;
}
2023-09-11 10:03:32 +02:00
//
// WRITEPROTECT
//
bool I2C_24LC1025::hasWriteProtectPin()
{
return (_writeProtectPin >= 0);
}
void I2C_24LC1025::allowWrite()
{
if (hasWriteProtectPin())
{
digitalWrite(_writeProtectPin, LOW);
}
}
void I2C_24LC1025::preventWrite()
{
if (hasWriteProtectPin())
{
digitalWrite(_writeProtectPin, HIGH);
}
}
void I2C_24LC1025::setAutoWriteProtect(bool b)
{
if (hasWriteProtectPin())
{
_autoWriteProtect = b;
}
}
bool I2C_24LC1025::getAutoWriteProtect()
{
return _autoWriteProtect;
}
2023-05-02 13:37:13 +02:00
2021-02-03 17:20:20 +01:00
////////////////////////////////////////////////////////////////////
//
2023-09-11 10:03:32 +02:00
// PRIVATE
2021-02-03 17:20:20 +01:00
//
2022-06-12 17:10:15 +02:00
// _pageBlock aligns buffer to page boundaries for writing.
// and to I2C buffer size
// returns 0 = OK otherwise error
2021-02-03 17:20:20 +01:00
int I2C_24LC1025::_pageBlock(uint32_t memoryAddress, const uint8_t * buffer, const uint16_t length, const bool incrBuffer)
{
uint32_t addr = memoryAddress;
uint16_t len = length;
while (len > 0)
{
uint8_t bytesUntilPageBoundary = this->_pageSize - addr % this->_pageSize;
uint8_t cnt = I2C_BUFFERSIZE;
if (cnt > len) cnt = len;
if (cnt > bytesUntilPageBoundary) cnt = bytesUntilPageBoundary;
int rv = _WriteBlock(addr, buffer, cnt);
if (rv != 0) return rv;
addr += cnt;
if (incrBuffer) buffer += cnt;
len -= cnt;
}
return 0;
}
void I2C_24LC1025::_beginTransmission(uint32_t memoryAddress)
{
// chapter 5+6 - datasheet - need three bytes for address
_actualAddress = _deviceAddress;
if (memoryAddress >= 0x10000) _actualAddress |= 0x04; // addresbit 16
2022-06-12 17:10:15 +02:00
// Wait until EEPROM gives ACK again.
2023-05-02 13:37:13 +02:00
// this is a bit faster than the hardcoded 5 milliSeconds // chapter 7
2022-06-12 17:10:15 +02:00
// TWR = WriteCycleTime
uint32_t waitTime = I2C_WRITEDELAY + _extraTWR * 1000UL; // do the math once.
while ((micros() - _lastWrite) <= waitTime)
2021-02-03 17:20:20 +01:00
{
_wire->beginTransmission(_actualAddress);
if (_wire->endTransmission() == 0) break;
yield();
delayMicroseconds(50);
}
2022-06-12 17:10:15 +02:00
2021-02-03 17:20:20 +01:00
uint16_t memAddr = (memoryAddress & 0xFFFF);
2022-06-12 17:10:15 +02:00
_wire->beginTransmission(_actualAddress); // device address + bit 16
_wire->write((memAddr >> 8) & 0xFF); // highByte
_wire->write(memAddr & 0xFF); // lowByte
2021-02-03 17:20:20 +01:00
}
2022-06-12 17:10:15 +02:00
// pre: length <= this->_pageSize && length <= I2C_BUFFERSIZE;
// returns 0 = OK otherwise error
int I2C_24LC1025::_WriteBlock(uint32_t memoryAddress, const uint8_t * buffer, const uint8_t length)
2021-02-03 17:20:20 +01:00
{
2022-06-12 17:10:15 +02:00
_waitEEReady();
2023-09-11 10:03:32 +02:00
if (_autoWriteProtect)
{
digitalWrite(_writeProtectPin, LOW);
}
2021-02-03 17:20:20 +01:00
this->_beginTransmission(memoryAddress);
_wire->write(buffer, length);
int rv = _wire->endTransmission();
2023-09-11 10:03:32 +02:00
if (_autoWriteProtect)
{
digitalWrite(_writeProtectPin, HIGH);
}
2021-02-03 17:20:20 +01:00
_lastWrite = micros();
2022-06-12 17:10:15 +02:00
2023-09-11 10:03:32 +02:00
yield(); // For OS scheduling
2022-06-12 17:10:15 +02:00
// if (rv != 0)
// {
// if (_debug)
// {
// Serial.print("mem addr w: ");
// Serial.print(memoryAddress, HEX);
// Serial.print("\t");
// Serial.println(rv);
// }
// return -(abs(rv)); // error
// }
2021-02-03 17:20:20 +01:00
return rv;
}
2022-06-12 17:10:15 +02:00
// pre: buffer is large enough to hold length bytes
// returns bytes read
uint8_t I2C_24LC1025::_ReadBlock(uint32_t memoryAddress, uint8_t * buffer, const uint8_t length)
2021-02-03 17:20:20 +01:00
{
2022-06-12 17:10:15 +02:00
_waitEEReady();
2021-02-03 17:20:20 +01:00
2022-06-12 17:10:15 +02:00
// _beginTransmissinon sets actual address !!!
2021-02-03 17:20:20 +01:00
this->_beginTransmission(memoryAddress);
int rv = _wire->endTransmission();
if (rv != 0)
{
2022-06-12 17:10:15 +02:00
// if (_debug)
// {
// Serial.print("mem addr r: ");
// Serial.print(memoryAddress, HEX);
// Serial.print("\t");
// Serial.println(rv);
// }
2023-09-11 10:03:32 +02:00
return 0; // error
2021-02-03 17:20:20 +01:00
}
2022-06-12 17:10:15 +02:00
// readBytes will always be equal or smaller to length
2021-02-03 17:20:20 +01:00
uint8_t readBytes = _wire->requestFrom(_actualAddress, length);
2023-09-11 10:03:32 +02:00
yield(); // For OS scheduling
2021-02-03 17:20:20 +01:00
uint8_t cnt = 0;
while (cnt < readBytes)
{
buffer[cnt++] = _wire->read();
}
return readBytes;
}
2022-06-12 17:10:15 +02:00
void I2C_24LC1025::_waitEEReady()
{
// Wait until EEPROM gives ACK again.
// this is a bit faster than the hardcoded 5 milliSeconds
// TWR = WriteCycleTime
2023-05-02 13:37:13 +02:00
uint32_t waitTime = I2C_WRITEDELAY + _extraTWR * 1000UL;
2022-06-12 17:10:15 +02:00
while ((micros() - _lastWrite) <= waitTime)
{
2023-09-11 10:03:32 +02:00
if (isConnected()) return;
// TODO remove previous code
// _wire->beginTransmission(_deviceAddress);
// int x = _wire->endTransmission();
// if (x == 0) return;
yield(); // For OS scheduling
2022-06-12 17:10:15 +02:00
}
return;
}
2023-09-11 10:03:32 +02:00
// -- END OF FILE --
2021-12-19 17:08:13 +01:00