/*
	Demo program for the 2017/18 open door day at HTL Hollabrunn
    Copyright (C) 2018  Andreas Mieke

    This program is free software: you can redistribute it and/or modify
    it under the terms of the GNU General Public License as published by
    the Free Software Foundation, either version 3 of the License, or
    (at your option) any later version.

    This program is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    GNU General Public License for more details.

    You should have received a copy of the GNU General Public License
    along with this program.  If not, see <https://www.gnu.org/licenses/>.
*/

#include "eeprom.h"

extern volatile uint32_t SysTickCnt;

// EEPROM addresse
#define EEPROM_ADDR (uint8_t)0xA0				// 0b10100000
												//   ----			Vendor address part
												//       ---  User address part
												//          - Keep free for R/W bit (set by I2C_Send7bitAddress())

uint32_t last_write_tick = 0;

void eeprom_init(void)
{
	// Init I2C1
	i2c1_init();
}

void eeprom_read(uint16_t address, uint8_t *data, uint16_t length)
{
	uint16_t cur_pos;
	// Send address of EEPROM and start address on EEPROM
	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, (address & 0xFF00) >> 8);
	while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED));
	I2C_SendData(I2C1, (address & 0x00FF));
	while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED));
	
	// Switch to receive mode
	I2C_GenerateSTART(I2C1, ENABLE);
	I2C_AcknowledgeConfig(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));
	
	// Read all bytes and disable ACK on last byte
	for(cur_pos = 0; cur_pos < length; cur_pos++) {
		if(cur_pos == length - 1) {
			I2C_AcknowledgeConfig(I2C1, DISABLE);
		}
		while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_RECEIVED));
		data[cur_pos] = I2C_ReceiveData(I2C1);
	}
	
	// Send stop condition
	I2C_GenerateSTOP(I2C1, ENABLE);
}

void eeprom_write(uint16_t address, uint8_t *data, uint16_t length)
{
	uint8_t cur_page = 0;
	uint16_t cur_pos = 0;
	address = address & 0xFFA0;
	
	// If more than one page is needed, cycle over the pages
	for(cur_page = 0; cur_page <= ((length-1)/64); cur_page++) {
		// Wait 5 ms for the write cycle (see datasheet)
		while((SysTickCnt - last_write_tick) <= 5);
		// Send start condition
		I2C_GenerateSTART(I2C1, ENABLE);
		// Send EEPROM address and start address of data to send
		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, (address & 0xFF00) >> 8);
		while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED));
		I2C_SendData(I2C1, (address & 0x00FF));
		while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED));
		
		// Send max 64 bytes (1 page) of data
		for(; (cur_pos < length) && (cur_pos%64 <= 63); cur_pos++) {
			I2C_SendData(I2C1, data[cur_pos]);
			while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED));
		}
		
		// Generate stop condition and calculate address of next page (if needed)
		I2C_GenerateSTOP(I2C1, ENABLE);
		address += 0x0040;
		last_write_tick = SysTickCnt;
	}
}