Implement raw EEPROM read and write functionality

This commit is contained in:
2024-09-20 16:56:29 +02:00
parent 171e827ab5
commit 29b0ace897
4 changed files with 93 additions and 21 deletions

View File

@@ -4,7 +4,7 @@
#include "common/math.h" #include "common/math.h"
#include "common/types.h" #include "common/types.h"
#include "common/watchdog.h" #include "common/watchdog.h"
#include "common/eeprom.h" #include "common/memory.h"
#include <stddef.h> #include <stddef.h>
#include <stdarg.h> #include <stdarg.h>

View File

@@ -1,17 +0,0 @@
#include "common.h"
// TODO: Implement EEPROM storage with wear leveling
int MEM_Read(mem_data_t *out)
{
UNUSED(out);
return 0;
}
int MEM_Write(mem_data_t *in)
{
UNUSED(in);
return 0;
}

89
src/common/memory.c Normal file
View File

@@ -0,0 +1,89 @@
#include "common.h"
#include <util/atomic.h>
static int WriteRaw(word addr, byte data);
static byte ReadRaw(word addr);
int MEM_Read(mem_data_t *out)
{
// TODO
UNUSED(out);
UNUSED(ReadRaw);
return 0;
}
int MEM_Write(mem_data_t *in)
{
// TODO
UNUSED(in);
UNUSED(WriteRaw);
return 0;
}
static int WriteRaw(word addr, byte data)
{
// The EEMWE bit determines whether setting EEWE to
// one causes the EEPROM to be written. When EEMWE
// is set, setting EEWE within four clock cycles
// will write data to the EEPROM at the selected
// address.
// If EEMWE is zero, setting EEWE will have no
// effect. When EEMWE has been written to one by
// software, hardware clears the bit to zero after
// four clock cycles.
// Caution: An interrupt between the last two steps
// will make the write cycle fail, since the EEPROM
// Master Write Enable will time-out.
// If an interrupt routine accessing the EEPROM is
// interrupting another EEPROM Access, the EEAR or
// EEDR reGister will be modified, causing the
// interrupted EEPROM Access to fail.
// It is recommended to have the Global Interrupt
// Flag cleared during all the steps to avoid these
// problems.
// Wait until ready
while (EECR & BIT(EEWE));
// No interrupts during EEPROM write
ATOMIC_BLOCK(ATOMIC_RESTORESTATE) {
EEAR = addr;
EEDR = data;
// Write to address
EECR |= BIT(EEMWE);
EECR |= BIT(EEWE);
}
return 0;
}
static byte ReadRaw(word addr)
{
// Wait until ready
while (EECR & BIT(EEWE));
EEAR = addr;
// The EEPROM Read Enable Signal EERE is the read
// strobe to the EEPROM. When the correct address
// is set up in the EEAR Register, the EERE bit
// must be written to a logic one to trigger the
// EEPROM read.
// Read from address
EECR |= BIT(EERE);
// The EEPROM read access takes one instruction,
// and the requested data is available immediately.
// When the EEPROM is read, the CPU is halted for
// four cycles before the next instruction is
// executed.
return EEDR;
}

View File

@@ -1,5 +1,5 @@
#ifndef MAD_CORE_COMMON_EEPROM_H #ifndef MAD_CORE_COMMON_MEMORY_H
#define MAD_CORE_COMMON_EEPROM_H #define MAD_CORE_COMMON_MEMORY_H
typedef struct mem_data_s mem_data_t; typedef struct mem_data_s mem_data_t;
@@ -11,4 +11,4 @@ struct mem_data_s {
int MEM_Read(mem_data_t *out); int MEM_Read(mem_data_t *out);
int MEM_Write(mem_data_t *in); int MEM_Write(mem_data_t *in);
#endif // MAD_CORE_COMMON_EEPROM_H #endif // MAD_CORE_COMMON_MEMORY_H