wip:
This commit is contained in:
169
zh_avr_i2c.c
169
zh_avr_i2c.c
@@ -17,16 +17,23 @@
|
||||
|
||||
typedef enum
|
||||
{
|
||||
MASTER_TX,
|
||||
MASTER_RX,
|
||||
MASTER_TX_RX,
|
||||
MASTER_PR
|
||||
MASTER_WRITE,
|
||||
MASTER_READ,
|
||||
MASTER_WRITE_REG,
|
||||
MASTER_READ_REG,
|
||||
MASTER_PROBE
|
||||
} _work_mode_t;
|
||||
|
||||
avr_err_t _zh_avr_i2c_master_start(TickType_t xTicksToWait);
|
||||
|
||||
static EventGroupHandle_t _event_group_handle = NULL;
|
||||
static uint8_t _target_i2c_address = 0;
|
||||
volatile static uint8_t _work_mode = 0;
|
||||
|
||||
volatile static uint8_t *_master_tx_data = NULL;
|
||||
volatile static uint8_t _master_tx_data_size = 0;
|
||||
volatile static uint8_t _master_tx_data_counter = 0;
|
||||
|
||||
avr_err_t zh_avr_i2c_master_init(const bool pullup)
|
||||
{
|
||||
_event_group_handle = xEventGroupCreate();
|
||||
@@ -45,8 +52,39 @@ avr_err_t zh_avr_i2c_master_init(const bool pullup)
|
||||
|
||||
avr_err_t zh_avr_i2c_master_probe(const uint8_t addr, TickType_t xTicksToWait)
|
||||
{
|
||||
_work_mode = MASTER_PR;
|
||||
_work_mode = MASTER_PROBE;
|
||||
_target_i2c_address = addr;
|
||||
return _zh_avr_i2c_master_start(xTicksToWait);
|
||||
}
|
||||
|
||||
avr_err_t zh_avr_i2c_master_transmit(const uint8_t addr, uint8_t *data, uint8_t size, TickType_t xTicksToWait)
|
||||
{
|
||||
ZH_ERROR_CHECK(data != NULL || size > 0, AVR_ERR_INVALID_ARG);
|
||||
_work_mode = MASTER_WRITE;
|
||||
_target_i2c_address = addr;
|
||||
_master_tx_data = data;
|
||||
_master_tx_data_size = size;
|
||||
_master_tx_data_counter = 0;
|
||||
return _zh_avr_i2c_master_start(xTicksToWait);
|
||||
}
|
||||
|
||||
avr_err_t zh_avr_i2c_master_receive(const uint8_t addr, uint8_t *data, uint8_t size, TickType_t xTicksToWait)
|
||||
{
|
||||
ZH_ERROR_CHECK(data != NULL || size > 0, AVR_ERR_INVALID_ARG);
|
||||
_work_mode = MASTER_READ;
|
||||
_target_i2c_address = addr;
|
||||
_master_tx_data_size = size;
|
||||
_master_tx_data_counter = 0;
|
||||
return _zh_avr_i2c_master_start(xTicksToWait);
|
||||
}
|
||||
|
||||
avr_err_t zh_avr_i2c_master_transmit_receive(const uint8_t addr, uint8_t *write_data, uint8_t write_size, uint8_t *read_data, uint8_t read_size, size_t delay)
|
||||
{
|
||||
return AVR_OK;
|
||||
}
|
||||
|
||||
avr_err_t _zh_avr_i2c_master_start(TickType_t xTicksToWait)
|
||||
{
|
||||
TWCR = I2C_START | (1 << TWSTA);
|
||||
EventBits_t bits = xEventGroupWaitBits(_event_group_handle, I2C_OK | I2C_NACK | I2C_COLLISION | I2C_BUS_FAIL, pdTRUE, pdFALSE, xTicksToWait);
|
||||
if ((bits & I2C_OK) != 0)
|
||||
@@ -63,48 +101,7 @@ avr_err_t zh_avr_i2c_master_probe(const uint8_t addr, TickType_t xTicksToWait)
|
||||
}
|
||||
}
|
||||
|
||||
avr_err_t zh_avr_i2c_master_transmit(const uint8_t addr, uint8_t *data, uint8_t size, size_t delay)
|
||||
{
|
||||
TWCR = (1 << TWINT) | (0 << TWEA) | (1 << TWSTA) | (0 << TWSTO) | (0 << TWWC) | (1 << TWEN) | (1 << TWIE);
|
||||
return AVR_OK;
|
||||
}
|
||||
|
||||
avr_err_t zh_avr_i2c_master_receive(const uint8_t addr, uint8_t *data, uint8_t size, size_t delay)
|
||||
{
|
||||
return AVR_OK;
|
||||
}
|
||||
|
||||
avr_err_t zh_avr_i2c_master_transmit_receive(const uint8_t addr, uint8_t *write_data, uint8_t write_size, uint8_t *read_data, uint8_t read_size, size_t delay)
|
||||
{
|
||||
return AVR_OK;
|
||||
}
|
||||
|
||||
void DoNothing(void);
|
||||
|
||||
uint8_t i2c_Do; // Переменная состояния передатчика IIC
|
||||
uint8_t i2c_InBuff[i2c_MasterBytesRX]; // Буфер прием при работе как Slave
|
||||
uint8_t i2c_OutBuff[i2c_MasterBytesTX]; // Буфер передачи при работе как Slave
|
||||
uint8_t i2c_SlaveIndex; // Индекс буфера Slave
|
||||
|
||||
uint8_t i2c_Buffer[i2c_MaxBuffer]; // Буфер для данных работы в режиме Master
|
||||
uint8_t i2c_index; // Индекс этого буфера
|
||||
uint8_t i2c_ByteCount; // Число байт передаваемых
|
||||
|
||||
uint8_t i2c_SlaveAddress; // Адрес подчиненного
|
||||
|
||||
uint8_t i2c_PageAddress[i2c_MaxPageAddrLgth]; // Буфер адреса страниц (для режима с sawsarp)
|
||||
uint8_t i2c_PageAddrIndex; // Индекс буфера адреса страниц
|
||||
uint8_t i2c_PageAddrCount; // Число байт в адресе страницы для текущего Slave
|
||||
|
||||
// Указатели выхода из автомата:
|
||||
IIC_F MasterOutFunc = &DoNothing; // в Master режиме
|
||||
IIC_F SlaveOutFunc = &DoNothing; // в режиме Slave
|
||||
IIC_F ErrorOutFunc = &DoNothing; // в результате ошибки в режиме Master
|
||||
|
||||
uint8_t WorkLog[100]; // Лог пишем сюда
|
||||
uint8_t WorkIndex = 0; // Индекс лога
|
||||
|
||||
ISR(TWI_vect) // Прерывание TWI Тут наше все.
|
||||
ISR(TWI_vect)
|
||||
{
|
||||
BaseType_t xHigherPriorityTaskWoken = pdFALSE;
|
||||
switch (TWSR & 0xF8) // Отсекаем биты прескалера
|
||||
@@ -118,35 +115,60 @@ ISR(TWI_vect) // Прерывание TWI Тут наше все.
|
||||
printf("08\n");
|
||||
switch (_work_mode)
|
||||
{
|
||||
case MASTER_TX:
|
||||
case MASTER_WRITE:
|
||||
TWDR = (_target_i2c_address << 1) | I2C_MASTER_WRITE;
|
||||
// TWCR = I2C_START;
|
||||
break;
|
||||
case MASTER_RX:
|
||||
break;
|
||||
case MASTER_TX_RX:
|
||||
break;
|
||||
case MASTER_PR:
|
||||
case MASTER_READ:
|
||||
TWDR = (_target_i2c_address << 1) | I2C_MASTER_READ;
|
||||
TWCR = I2C_START;
|
||||
// TWCR = I2C_START;
|
||||
break;
|
||||
case MASTER_WRITE_REG:
|
||||
break;
|
||||
case MASTER_READ_REG:
|
||||
break;
|
||||
case MASTER_PROBE:
|
||||
TWDR = (_target_i2c_address << 1) | I2C_MASTER_READ;
|
||||
// TWCR = I2C_START;
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
TWCR = I2C_START;
|
||||
break;
|
||||
case 0x10: // A repeated START condition has been transmitted.
|
||||
printf("10\n");
|
||||
TWCR = I2C_START | (1 << TWSTO);
|
||||
// TWCR = I2C_START | (1 << TWSTO);
|
||||
break;
|
||||
case 0x18: // SLA+W has been transmitted. ACK has been received.
|
||||
printf("18\n");
|
||||
TWDR = *(_master_tx_data++);
|
||||
_master_tx_data_counter++;
|
||||
TWCR = I2C_START;
|
||||
break;
|
||||
case 0x20: // SLA+W has been transmitted. NOT ACK has been received.
|
||||
printf("20\n");
|
||||
TWCR = I2C_START | (1 << TWSTO);
|
||||
xEventGroupSetBitsFromISR(_event_group_handle, I2C_NACK, &xHigherPriorityTaskWoken);
|
||||
break;
|
||||
case 0x28: // Data byte has been transmitted. ACK has been received.
|
||||
printf("28\n");
|
||||
if (_master_tx_data_counter < _master_tx_data_size)
|
||||
{
|
||||
TWDR = *(_master_tx_data++);
|
||||
_master_tx_data_counter++;
|
||||
TWCR = I2C_START;
|
||||
}
|
||||
else
|
||||
{
|
||||
TWCR = I2C_START | (1 << TWSTO);
|
||||
xEventGroupSetBitsFromISR(_event_group_handle, I2C_OK, &xHigherPriorityTaskWoken);
|
||||
}
|
||||
break;
|
||||
case 0x30: // Data byte has been transmitted. NOT ACK has been received.
|
||||
printf("30\n");
|
||||
TWCR = I2C_START | (1 << TWSTO);
|
||||
xEventGroupSetBitsFromISR(_event_group_handle, I2C_NACK, &xHigherPriorityTaskWoken);
|
||||
break;
|
||||
case 0x38: // Arbitration lost in SLA+W or data bytes. Arbitration lost in SLA+R or NOT ACK bit.
|
||||
printf("38\n");
|
||||
@@ -157,13 +179,15 @@ ISR(TWI_vect) // Прерывание TWI Тут наше все.
|
||||
printf("40\n");
|
||||
switch (_work_mode)
|
||||
{
|
||||
case MASTER_TX:
|
||||
case MASTER_WRITE:
|
||||
break;
|
||||
case MASTER_RX:
|
||||
case MASTER_READ:
|
||||
break;
|
||||
case MASTER_TX_RX:
|
||||
case MASTER_WRITE_REG:
|
||||
break;
|
||||
case MASTER_PR:
|
||||
case MASTER_READ_REG:
|
||||
break;
|
||||
case MASTER_PROBE:
|
||||
TWCR = I2C_START | (1 << TWSTO);
|
||||
xEventGroupSetBitsFromISR(_event_group_handle, I2C_OK, &xHigherPriorityTaskWoken);
|
||||
break;
|
||||
@@ -675,31 +699,4 @@ ISR(TWI_vect) // Прерывание TWI Тут наше все.
|
||||
{
|
||||
portYIELD();
|
||||
};
|
||||
}
|
||||
|
||||
void DoNothing(void) // Функция пустышка, затыкать несуществующие ссылки
|
||||
{
|
||||
}
|
||||
|
||||
// void Init_i2c(void) // Настройка режима мастера
|
||||
// {
|
||||
// i2c_PORT |= 1 << i2c_SCL | 1 << i2c_SDA; // Включим подтяжку на ноги, вдруг юзер на резисторы пожмотился
|
||||
// i2c_DDR &= ~(1 << i2c_SCL | 1 << i2c_SDA);
|
||||
|
||||
// TWBR = 0x80; // Настроим битрейт
|
||||
// TWSR = 0x00;
|
||||
// }
|
||||
|
||||
void Init_Slave_i2c(IIC_F Addr) // Настройка режима слейва (если нужно)
|
||||
{
|
||||
TWAR = i2c_MasterAddress; // Внесем в регистр свой адрес, на который будем отзываться.
|
||||
// 1 в нулевом бите означает, что мы отзываемся на широковещательные пакеты
|
||||
SlaveOutFunc = Addr; // Присвоим указателю выхода по слейву функцию выхода
|
||||
|
||||
TWCR = 0 << TWSTA |
|
||||
0 << TWSTO |
|
||||
0 << TWINT |
|
||||
1 << TWEA |
|
||||
1 << TWEN |
|
||||
1 << TWIE; // Включаем агрегат и начинаем слушать шину.
|
||||
}
|
Reference in New Issue
Block a user