2018-03-18 22:43:59 +00:00
|
|
|
#include "eeprom.h"
|
|
|
|
|
|
|
|
volatile uint32_t *EEPROMSTick, EEPROMSTickCur;
|
|
|
|
|
|
|
|
uint8_t load_byte(uint16_t eeprom_addr)
|
|
|
|
{
|
|
|
|
uint8_t data;
|
2018-04-03 15:18:28 +00:00
|
|
|
// Send address to read from
|
2018-03-18 22:43:59 +00:00
|
|
|
I2C_GenerateSTART(I2C1, ENABLE);
|
|
|
|
while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT));
|
|
|
|
I2C_Send7bitAddress(I2C1, EEPROM_ADDR, I2C_Direction_Transmitter);
|
|
|
|
while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED));
|
|
|
|
I2C_SendData(I2C1, (eeprom_addr & 0xFF00) >> 8);
|
|
|
|
while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED));
|
|
|
|
I2C_SendData(I2C1, (eeprom_addr & 0x00FF));
|
|
|
|
while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED));
|
|
|
|
|
2018-04-03 15:18:28 +00:00
|
|
|
// Switch to Rx mode
|
2018-03-18 22:43:59 +00:00
|
|
|
I2C_GenerateSTART(I2C1, ENABLE);
|
|
|
|
while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT));
|
|
|
|
I2C_Send7bitAddress(I2C1, EEPROM_ADDR, I2C_Direction_Receiver);
|
|
|
|
while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_RECEIVER_MODE_SELECTED));
|
|
|
|
|
2018-04-03 15:18:28 +00:00
|
|
|
// Load byte
|
2018-03-18 22:43:59 +00:00
|
|
|
while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_RECEIVED));
|
|
|
|
data = I2C_ReceiveData(I2C1);
|
|
|
|
I2C_GenerateSTOP(I2C1, ENABLE);
|
|
|
|
|
2018-04-03 15:18:28 +00:00
|
|
|
// Return data byte
|
2018-03-18 22:43:59 +00:00
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
|
|
|
void write_byte(uint16_t eeprom_addr, uint8_t data)
|
|
|
|
{
|
2018-04-03 15:18:28 +00:00
|
|
|
// Send address to write to
|
2018-03-18 22:43:59 +00:00
|
|
|
I2C_GenerateSTART(I2C1, ENABLE);
|
|
|
|
while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT));
|
|
|
|
I2C_Send7bitAddress(I2C1, EEPROM_ADDR, I2C_Direction_Transmitter);
|
|
|
|
while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED));
|
|
|
|
I2C_SendData(I2C1, (eeprom_addr & 0xFF00) >> 8);
|
|
|
|
while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED));
|
|
|
|
I2C_SendData(I2C1, (eeprom_addr & 0x00FF));
|
|
|
|
while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED));
|
|
|
|
|
2018-04-03 15:18:28 +00:00
|
|
|
// Send data byte
|
2018-03-18 22:43:59 +00:00
|
|
|
I2C_SendData(I2C1, data);
|
|
|
|
while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED));
|
|
|
|
I2C_GenerateSTOP(I2C1, ENABLE);
|
|
|
|
|
2018-04-03 15:18:28 +00:00
|
|
|
// Wait 5 ms to satisfy the write cycle time of the EEPROM
|
2018-03-18 22:43:59 +00:00
|
|
|
EEPROMSTickCur = *EEPROMSTick;
|
|
|
|
while((*EEPROMSTick - EEPROMSTickCur) <= 50); // 5ms write cycle, see datasheet param 17
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
void init_eeprom(volatile uint32_t *SysTickCnt)
|
|
|
|
{
|
2018-04-03 15:18:28 +00:00
|
|
|
// Enable GPIOB and I2C1 cloc
|
2018-03-18 22:43:59 +00:00
|
|
|
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
|
|
|
|
RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C1, ENABLE);
|
|
|
|
|
2018-04-03 15:18:28 +00:00
|
|
|
// Create gpio struct and fill it with default values
|
2018-03-18 22:43:59 +00:00
|
|
|
GPIO_InitTypeDef gpio;
|
|
|
|
GPIO_StructInit(&gpio);
|
|
|
|
|
2018-04-03 15:18:28 +00:00
|
|
|
// Set PB6 to alternate function push pull (SCL)
|
2018-03-18 22:43:59 +00:00
|
|
|
gpio.GPIO_Mode = GPIO_Mode_AF_PP;
|
|
|
|
gpio.GPIO_Pin = GPIO_Pin_6;
|
|
|
|
GPIO_Init(GPIOB, &gpio);
|
|
|
|
|
2018-04-03 15:18:28 +00:00
|
|
|
// Set PB7 to alternate function open drain (SDA)
|
2018-03-18 22:43:59 +00:00
|
|
|
gpio.GPIO_Mode = GPIO_Mode_AF_OD;
|
|
|
|
gpio.GPIO_Pin = GPIO_Pin_7;
|
|
|
|
GPIO_Init(GPIOB, &gpio);
|
|
|
|
|
2018-04-03 15:18:28 +00:00
|
|
|
// Set I2C clock to 400 kHz
|
2018-03-18 22:43:59 +00:00
|
|
|
I2C_InitTypeDef i2c;
|
|
|
|
I2C_StructInit(&i2c);
|
|
|
|
i2c.I2C_ClockSpeed = 400000;
|
|
|
|
I2C_Init(I2C1, &i2c);
|
|
|
|
|
2018-04-03 15:18:28 +00:00
|
|
|
// Enable I2C1 and save SysTick pointer
|
2018-03-18 22:43:59 +00:00
|
|
|
I2C_Cmd(I2C1, ENABLE);
|
|
|
|
EEPROMSTick = SysTickCnt;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
void run_eeprom(uint8_t *success)
|
|
|
|
{
|
2018-04-03 15:18:28 +00:00
|
|
|
// Write some data to some addresses and check if the date matches after a read, if not, return
|
2018-03-18 22:43:59 +00:00
|
|
|
uint8_t set, read;
|
|
|
|
*success = 0;
|
|
|
|
set = 0xAA;
|
|
|
|
write_byte(0x0000, set);
|
|
|
|
read = load_byte(0x0000);
|
|
|
|
if (set != read) return;
|
|
|
|
set = 0xBA;
|
|
|
|
write_byte(0x0010, set);
|
|
|
|
read = load_byte(0x0010);
|
|
|
|
if (set != read) return;
|
|
|
|
set = 0xAD;
|
|
|
|
write_byte(0x0001, set);
|
|
|
|
read = load_byte(0x0001);
|
|
|
|
if (set != read) return;
|
|
|
|
set = 0x00;
|
|
|
|
write_byte(0x0000, set);
|
|
|
|
read = load_byte(0x0000);
|
|
|
|
if (set != read) return;
|
|
|
|
set = 0x88;
|
|
|
|
write_byte(0x0002, set);
|
|
|
|
read = load_byte(0x0002);
|
|
|
|
if (set != read) return;
|
|
|
|
set = 0x01;
|
|
|
|
write_byte(0x0000, set);
|
|
|
|
read = load_byte(0x0000);
|
|
|
|
if (set != read) return;
|
|
|
|
set = 0x55;
|
|
|
|
write_byte(0x00005, set);
|
|
|
|
read = load_byte(0x0005);
|
|
|
|
if (set != read) return;
|
|
|
|
set = 0xAA;
|
|
|
|
write_byte(0x0005, set);
|
|
|
|
read = load_byte(0x0005);
|
|
|
|
if (set != read) return;
|
2018-04-03 15:18:28 +00:00
|
|
|
// If everything matches, set success to true and then return
|
2018-03-18 22:43:59 +00:00
|
|
|
*success = 1;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
void deinit_eeprom(void)
|
|
|
|
{
|
2018-04-03 15:18:28 +00:00
|
|
|
// Disable I2C1
|
2018-03-18 22:43:59 +00:00
|
|
|
I2C_Cmd(I2C1, DISABLE);
|
|
|
|
I2C_DeInit(I2C1);
|
|
|
|
return;
|
|
|
|
}
|