2024-08-28 13:10:57 +10:00
|
|
|
|
|
|
|
#include <Arduino.h>
|
|
|
|
#include "eeprom.hpp"
|
2024-08-31 01:53:25 +10:00
|
|
|
#include "util.hpp"
|
2024-08-28 13:10:57 +10:00
|
|
|
|
2024-08-31 01:53:25 +10:00
|
|
|
using namespace eeprom;
|
|
|
|
|
|
|
|
constexpr uint8_t WRITE_MS = 3;
|
2024-08-28 13:10:57 +10:00
|
|
|
constexpr uint8_t ADDR = 0x50;
|
|
|
|
|
2024-09-30 23:25:16 +10:00
|
|
|
static uint32_t at = 0;
|
|
|
|
static uint32_t last_block_end = 0;
|
2024-08-28 13:10:57 +10:00
|
|
|
|
2024-08-31 01:53:25 +10:00
|
|
|
static bool is_responsive(uint8_t i2c_addr) {
|
|
|
|
bool s = mem.start_write(i2c_addr);
|
|
|
|
return s;
|
|
|
|
}
|
2024-08-28 13:10:57 +10:00
|
|
|
|
2024-08-31 01:53:25 +10:00
|
|
|
static void wait_for_responsive(uint8_t i2c_addr) {
|
|
|
|
while(!is_responsive(i2c_addr)) {
|
|
|
|
delay(1);
|
|
|
|
}
|
2024-08-28 13:10:57 +10:00
|
|
|
}
|
|
|
|
|
2024-08-31 01:53:25 +10:00
|
|
|
static uint8_t get_i2c_addr(uint32_t at) {
|
2024-09-29 18:07:58 +10:00
|
|
|
return ADDR ^ (at >> 16);
|
2024-08-28 13:10:57 +10:00
|
|
|
}
|
|
|
|
|
2024-08-31 01:53:25 +10:00
|
|
|
static void set_address(uint8_t i2c_addr, uint16_t at) {
|
|
|
|
char addr_bytes[2] = {at >> 8, at};
|
|
|
|
mem.start_write(i2c_addr);
|
|
|
|
mem.write(at >> 8);
|
|
|
|
mem.write(at);
|
2024-08-28 13:10:57 +10:00
|
|
|
}
|
|
|
|
|
2024-08-31 01:53:25 +10:00
|
|
|
void eeprom::jump(uint32_t p_at) {
|
2024-09-30 23:25:16 +10:00
|
|
|
last_block_end = 0;
|
2024-08-31 02:06:10 +10:00
|
|
|
at = p_at;
|
2024-08-31 01:53:25 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
void eeprom::read(char* data, uint16_t data_size) {
|
|
|
|
uint16_t recorded = 0;
|
|
|
|
|
|
|
|
while(recorded < data_size && at < LENGTH) {
|
2024-09-30 23:25:16 +10:00
|
|
|
|
2024-08-31 01:53:25 +10:00
|
|
|
uint32_t block_end = at / BLOCK_SIZE * BLOCK_SIZE + BLOCK_SIZE;
|
|
|
|
uint8_t read_len = min(min(block_end - at, data_size - recorded), 255);
|
|
|
|
uint8_t i2c_addr = get_i2c_addr(at);
|
|
|
|
|
2024-09-30 23:25:16 +10:00
|
|
|
if(block_end != last_block_end) {
|
|
|
|
set_address(i2c_addr, at);
|
|
|
|
last_block_end = block_end;
|
|
|
|
}
|
|
|
|
|
2024-08-31 01:53:25 +10:00
|
|
|
mem.start_read(i2c_addr);
|
|
|
|
mem.read(data + recorded, read_len, true);
|
|
|
|
recorded += read_len;
|
|
|
|
at += read_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
mem.end();
|
|
|
|
|
|
|
|
for(uint16_t i = recorded; i < data_size; i++) {
|
|
|
|
data[i] = 0;
|
|
|
|
}
|
2024-08-28 13:10:57 +10:00
|
|
|
}
|
|
|
|
|
2024-08-31 01:53:25 +10:00
|
|
|
void eeprom::page_write(uint32_t at, const char* data) {
|
2024-08-31 02:06:10 +10:00
|
|
|
if(at >= LENGTH) return;
|
2024-08-31 01:53:25 +10:00
|
|
|
uint8_t i2c_addr = get_i2c_addr(at);
|
|
|
|
wait_for_responsive(i2c_addr);
|
|
|
|
mem.start_write(i2c_addr);
|
|
|
|
mem.write(at >> 8);
|
|
|
|
mem.write(at);
|
|
|
|
mem.write(data, PAGE_SIZE - (at % PAGE_SIZE));
|
|
|
|
mem.end();
|
|
|
|
delay(WRITE_MS);
|
2024-09-30 23:25:16 +10:00
|
|
|
last_block_end = 0;
|
2024-08-28 13:10:57 +10:00
|
|
|
}
|
|
|
|
|
2024-09-29 18:07:58 +10:00
|
|
|
uint32_t eeprom::get_addr() {
|
|
|
|
return at;
|
|
|
|
}
|
|
|
|
|