2017-05-22 15:08:04 +02:00
|
|
|
/**
|
|
|
|
******************************************************************************
|
|
|
|
* @addtogroup PIOS PIOS Core hardware abstraction layer
|
|
|
|
* @{
|
|
|
|
* @addtogroup PIOS_I2C I2C Functions
|
2017-05-26 01:06:31 +02:00
|
|
|
* @brief STM32F30x Hardware dependent I2C functionality
|
2017-05-22 15:08:04 +02:00
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file pios_i2c.c
|
2017-05-26 01:06:31 +02:00
|
|
|
* @author The LibrePilot Project, http://www.librepilot.org Copyright (C) 2017.
|
|
|
|
* dRonin, http://dronin.org, Copyright (C) 2016
|
|
|
|
* Tau Labs, http://taulabs.org, Copyright (C) 2012-2014
|
|
|
|
* The OpenPilot Team, http://www.openpilot.org Copyright (C) 2012.
|
2017-05-22 15:08:04 +02:00
|
|
|
* @brief I2C Enable/Disable routines
|
|
|
|
* @see The GNU Public License (GPL) Version 3
|
|
|
|
*
|
|
|
|
*****************************************************************************/
|
|
|
|
/*
|
|
|
|
* 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, write to the Free Software Foundation, Inc.,
|
|
|
|
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Project Includes */
|
|
|
|
#include "pios.h"
|
|
|
|
|
|
|
|
#if defined(PIOS_INCLUDE_I2C)
|
|
|
|
|
|
|
|
#include <pios_i2c_priv.h>
|
|
|
|
|
|
|
|
enum i2c_adapter_state {
|
|
|
|
I2C_STATE_FSM_FAULT = 0, /* Must be zero so undefined transitions land here */
|
|
|
|
|
|
|
|
I2C_STATE_BUS_ERROR,
|
|
|
|
|
|
|
|
I2C_STATE_STOPPED,
|
|
|
|
I2C_STATE_STOPPING,
|
|
|
|
I2C_STATE_STARTING,
|
|
|
|
|
|
|
|
I2C_STATE_R_MORE_TXN_ADDR,
|
|
|
|
I2C_STATE_R_MORE_TXN_PRE_ONE,
|
|
|
|
I2C_STATE_R_MORE_TXN_PRE_FIRST,
|
|
|
|
I2C_STATE_R_MORE_TXN_PRE_MIDDLE,
|
|
|
|
I2C_STATE_R_MORE_TXN_PRE_LAST,
|
|
|
|
I2C_STATE_R_MORE_TXN_POST_LAST,
|
|
|
|
|
|
|
|
I2C_STATE_R_LAST_TXN_ADDR,
|
|
|
|
I2C_STATE_R_LAST_TXN_PRE_ONE,
|
|
|
|
I2C_STATE_R_LAST_TXN_PRE_FIRST,
|
|
|
|
I2C_STATE_R_LAST_TXN_PRE_MIDDLE,
|
|
|
|
I2C_STATE_R_LAST_TXN_PRE_LAST,
|
|
|
|
I2C_STATE_R_LAST_TXN_POST_LAST,
|
|
|
|
|
|
|
|
I2C_STATE_W_MORE_TXN_ADDR,
|
|
|
|
I2C_STATE_W_MORE_TXN_MIDDLE,
|
|
|
|
I2C_STATE_W_MORE_TXN_LAST,
|
|
|
|
|
|
|
|
I2C_STATE_W_LAST_TXN_ADDR,
|
|
|
|
I2C_STATE_W_LAST_TXN_MIDDLE,
|
|
|
|
I2C_STATE_W_LAST_TXN_LAST,
|
|
|
|
|
|
|
|
I2C_STATE_NACK,
|
|
|
|
|
|
|
|
I2C_STATE_NUM_STATES /* Must be last */
|
|
|
|
};
|
|
|
|
|
|
|
|
enum i2c_adapter_event {
|
|
|
|
I2C_EVENT_START,
|
|
|
|
I2C_EVENT_RECEIVER_BUFFER_NOT_EMPTY,
|
|
|
|
I2C_EVENT_TRANSMIT_BUFFER_EMPTY,
|
|
|
|
I2C_EVENT_TRANSFER_COMPLETE,
|
|
|
|
I2C_EVENT_STOP,
|
|
|
|
I2C_EVENT_NACK,
|
|
|
|
I2C_EVENT_BUS_ERROR,
|
|
|
|
I2C_EVENT_STOPPED,
|
|
|
|
I2C_EVENT_AUTO,
|
|
|
|
|
|
|
|
I2C_EVENT_NUM_EVENTS /* Must be last */
|
|
|
|
};
|
|
|
|
|
|
|
|
/* dirty hack to keep compatible with enum in pios_i2c_priv.h */
|
|
|
|
#define I2C_STATE_WRITE_BYTE I2C_STATE_W_MORE_TXN_ADDR
|
|
|
|
#define I2C_STATE_READ_BYTE I2C_STATE_W_MORE_TXN_MIDDLE
|
|
|
|
#define I2C_STATE_TRANSFER_COMPLETE I2C_STATE_W_MORE_TXN_LAST
|
|
|
|
|
|
|
|
// #define I2C_HALT_ON_ERRORS
|
|
|
|
|
|
|
|
#if defined(PIOS_I2C_DIAGNOSTICS)
|
|
|
|
static struct pios_i2c_fault_history i2c_adapter_fault_history;
|
|
|
|
|
|
|
|
static volatile uint32_t i2c_evirq_history[I2C_LOG_DEPTH];
|
|
|
|
static volatile uint8_t i2c_evirq_history_pointer;
|
|
|
|
|
|
|
|
static volatile uint32_t i2c_erirq_history[I2C_LOG_DEPTH];
|
|
|
|
static volatile uint8_t i2c_erirq_history_pointer;
|
|
|
|
|
|
|
|
static volatile uint8_t i2c_state_history[I2C_LOG_DEPTH];
|
|
|
|
static volatile uint8_t i2c_state_history_pointer;
|
|
|
|
|
|
|
|
static volatile uint8_t i2c_state_event_history[I2C_LOG_DEPTH];
|
|
|
|
static volatile uint8_t i2c_state_event_history_pointer;
|
|
|
|
|
|
|
|
static uint32_t i2c_fsm_fault_count;
|
|
|
|
static uint32_t i2c_bad_event_counter;
|
|
|
|
static uint32_t i2c_error_interrupt_counter;
|
|
|
|
static uint32_t i2c_nack_counter;
|
|
|
|
static uint32_t i2c_timeout_counter;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static void go_fsm_fault(struct pios_i2c_adapter *i2c_adapter, bool *woken);
|
|
|
|
static void go_bus_error(struct pios_i2c_adapter *i2c_adapter, bool *woken);
|
|
|
|
static void go_stopped(struct pios_i2c_adapter *i2c_adapter, bool *woken);
|
|
|
|
static void go_starting(struct pios_i2c_adapter *i2c_adapter, bool *woken);
|
|
|
|
static void go_write_byte(struct pios_i2c_adapter *i2c_adapter, bool *woken);
|
|
|
|
static void go_read_byte(struct pios_i2c_adapter *i2c_adapter, bool *woken);
|
|
|
|
static void go_transfer_complete(struct pios_i2c_adapter *i2c_adapter, bool *woken);
|
|
|
|
static void go_nack(struct pios_i2c_adapter *i2c_adapter, bool *woken);
|
|
|
|
|
|
|
|
struct i2c_adapter_transition {
|
|
|
|
void (*entry_fn)(struct pios_i2c_adapter *i2c_adapter, bool *woken);
|
|
|
|
enum i2c_adapter_state next_state[I2C_EVENT_NUM_EVENTS];
|
|
|
|
};
|
|
|
|
|
|
|
|
static void i2c_adapter_process_auto(struct pios_i2c_adapter *i2c_adapter, bool *woken);
|
|
|
|
static void i2c_adapter_inject_event(struct pios_i2c_adapter *i2c_adapter, enum i2c_adapter_event event, bool *woken);
|
|
|
|
static void i2c_adapter_fsm_init(struct pios_i2c_adapter *i2c_adapter);
|
|
|
|
static void i2c_adapter_reset_bus(struct pios_i2c_adapter *i2c_adapter);
|
|
|
|
static bool i2c_adapter_callback_handler(struct pios_i2c_adapter *i2c_adapter, bool *woken);
|
|
|
|
|
|
|
|
#if defined(PIOS_I2C_DIAGNOSTICS)
|
|
|
|
static void i2c_adapter_log_fault(struct pios_i2c_adapter *i2c_adapter, enum pios_i2c_error_type type);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static const struct i2c_adapter_transition i2c_adapter_transitions[I2C_STATE_NUM_STATES] = {
|
|
|
|
[I2C_STATE_FSM_FAULT] = {
|
|
|
|
.entry_fn = go_fsm_fault,
|
|
|
|
.next_state = {
|
|
|
|
[I2C_EVENT_AUTO] = I2C_STATE_STOPPED,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
[I2C_STATE_BUS_ERROR] = {
|
|
|
|
.entry_fn = go_bus_error,
|
|
|
|
.next_state = {
|
|
|
|
[I2C_EVENT_AUTO] = I2C_STATE_STOPPED,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
[I2C_STATE_STOPPED] = {
|
|
|
|
.entry_fn = go_stopped,
|
|
|
|
.next_state = {
|
|
|
|
[I2C_EVENT_START] = I2C_STATE_STARTING,
|
|
|
|
[I2C_EVENT_BUS_ERROR] = I2C_STATE_BUS_ERROR,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
[I2C_STATE_STARTING] = {
|
|
|
|
.entry_fn = go_starting,
|
|
|
|
.next_state = {
|
|
|
|
[I2C_EVENT_TRANSMIT_BUFFER_EMPTY] = I2C_STATE_WRITE_BYTE,
|
|
|
|
[I2C_EVENT_RECEIVER_BUFFER_NOT_EMPTY] = I2C_STATE_READ_BYTE,
|
|
|
|
[I2C_EVENT_NACK] = I2C_STATE_NACK,
|
|
|
|
[I2C_EVENT_BUS_ERROR] = I2C_STATE_BUS_ERROR,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
[I2C_STATE_WRITE_BYTE] = {
|
|
|
|
.entry_fn = go_write_byte,
|
|
|
|
.next_state = {
|
|
|
|
[I2C_EVENT_TRANSMIT_BUFFER_EMPTY] = I2C_STATE_WRITE_BYTE,
|
|
|
|
[I2C_EVENT_RECEIVER_BUFFER_NOT_EMPTY] = I2C_STATE_READ_BYTE,
|
|
|
|
[I2C_EVENT_TRANSFER_COMPLETE] = I2C_STATE_TRANSFER_COMPLETE,
|
|
|
|
[I2C_EVENT_STOP] = I2C_STATE_STOPPED,
|
|
|
|
[I2C_EVENT_BUS_ERROR] = I2C_STATE_BUS_ERROR,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
[I2C_STATE_READ_BYTE] = {
|
|
|
|
.entry_fn = go_read_byte,
|
|
|
|
.next_state = {
|
|
|
|
[I2C_EVENT_TRANSMIT_BUFFER_EMPTY] = I2C_STATE_WRITE_BYTE,
|
|
|
|
[I2C_EVENT_RECEIVER_BUFFER_NOT_EMPTY] = I2C_STATE_READ_BYTE,
|
|
|
|
[I2C_EVENT_TRANSFER_COMPLETE] = I2C_STATE_TRANSFER_COMPLETE,
|
|
|
|
[I2C_EVENT_STOP] = I2C_STATE_STOPPED,
|
|
|
|
[I2C_EVENT_BUS_ERROR] = I2C_STATE_BUS_ERROR,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
[I2C_STATE_TRANSFER_COMPLETE] = {
|
|
|
|
.entry_fn = go_transfer_complete,
|
|
|
|
.next_state = {
|
|
|
|
[I2C_EVENT_AUTO] = I2C_STATE_STARTING,
|
|
|
|
[I2C_EVENT_BUS_ERROR] = I2C_STATE_BUS_ERROR,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
[I2C_STATE_NACK] = {
|
|
|
|
.entry_fn = go_nack,
|
|
|
|
.next_state = {
|
|
|
|
[I2C_EVENT_AUTO] = I2C_STATE_STOPPED,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
static void go_fsm_fault(struct pios_i2c_adapter *i2c_adapter, __attribute__((unused)) bool *woken)
|
|
|
|
{
|
|
|
|
#if defined(I2C_HALT_ON_ERRORS)
|
|
|
|
PIOS_DEBUG_Assert(0);
|
|
|
|
#endif
|
|
|
|
/* Note that this transfer has hit a bus error */
|
|
|
|
i2c_adapter->bus_error = true;
|
|
|
|
|
|
|
|
i2c_adapter_reset_bus(i2c_adapter);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void go_bus_error(struct pios_i2c_adapter *i2c_adapter, __attribute__((unused)) bool *woken)
|
|
|
|
{
|
|
|
|
/* Note that this transfer has hit a bus error */
|
|
|
|
i2c_adapter->bus_error = true;
|
|
|
|
|
|
|
|
i2c_adapter_reset_bus(i2c_adapter);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void go_stopped(struct pios_i2c_adapter *i2c_adapter, bool *woken)
|
|
|
|
{
|
|
|
|
I2C_ITConfig(i2c_adapter->cfg->regs, I2C_IT_ERRI | I2C_IT_TCI | I2C_IT_NACKI | I2C_IT_RXI | I2C_IT_STOPI | I2C_IT_TXI, DISABLE);
|
|
|
|
|
|
|
|
if (i2c_adapter->callback) {
|
|
|
|
i2c_adapter_callback_handler(i2c_adapter, woken);
|
|
|
|
} else {
|
2017-05-30 16:31:39 +02:00
|
|
|
#ifdef PIOS_INCLUDE_FREERTOS
|
2017-05-22 15:08:04 +02:00
|
|
|
signed portBASE_TYPE pxHigherPriorityTaskWoken = pdFALSE;
|
|
|
|
if (xSemaphoreGiveFromISR(i2c_adapter->sem_ready, &pxHigherPriorityTaskWoken) != pdTRUE) {
|
|
|
|
#if defined(I2C_HALT_ON_ERRORS)
|
|
|
|
PIOS_DEBUG_Assert(0);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
if (pxHigherPriorityTaskWoken == pdTRUE) {
|
|
|
|
*woken = true;
|
|
|
|
}
|
2017-05-30 16:31:39 +02:00
|
|
|
#endif /* PIOS_INCLUDE_FREERTOS */
|
2017-05-22 15:08:04 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void go_starting(struct pios_i2c_adapter *i2c_adapter, __attribute__((unused)) bool *woken)
|
|
|
|
{
|
|
|
|
PIOS_DEBUG_Assert(i2c_adapter->active_txn);
|
|
|
|
PIOS_DEBUG_Assert(i2c_adapter->active_txn <= i2c_adapter->last_txn);
|
|
|
|
PIOS_DEBUG_Assert(i2c_adapter->active_txn->len <= 255); // FIXME: implement this using TCR
|
|
|
|
|
|
|
|
i2c_adapter->active_byte = &(i2c_adapter->active_txn->buf[0]);
|
|
|
|
i2c_adapter->last_byte = &(i2c_adapter->active_txn->buf[i2c_adapter->active_txn->len - 1]);
|
|
|
|
|
|
|
|
I2C_ITConfig(i2c_adapter->cfg->regs, I2C_IT_ERRI | I2C_IT_TCI | I2C_IT_NACKI | I2C_IT_RXI | I2C_IT_STOPI | I2C_IT_TXI, ENABLE);
|
|
|
|
|
|
|
|
I2C_TransferHandling(
|
|
|
|
i2c_adapter->cfg->regs,
|
|
|
|
(i2c_adapter->active_txn->addr << 1),
|
|
|
|
i2c_adapter->active_txn->len,
|
|
|
|
i2c_adapter->active_txn == i2c_adapter->last_txn ? I2C_AutoEnd_Mode : I2C_SoftEnd_Mode,
|
|
|
|
i2c_adapter->active_txn->rw == PIOS_I2C_TXN_WRITE ? I2C_Generate_Start_Write : I2C_Generate_Start_Read);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void go_write_byte(struct pios_i2c_adapter *i2c_adapter, __attribute__((unused)) bool *woken)
|
|
|
|
{
|
|
|
|
PIOS_DEBUG_Assert(i2c_adapter->active_txn);
|
|
|
|
PIOS_DEBUG_Assert(i2c_adapter->active_txn <= i2c_adapter->last_txn);
|
|
|
|
|
|
|
|
I2C_SendData(i2c_adapter->cfg->regs, *(i2c_adapter->active_byte));
|
|
|
|
|
|
|
|
/* Move to the next byte */
|
|
|
|
i2c_adapter->active_byte++;
|
|
|
|
PIOS_DEBUG_Assert(i2c_adapter->active_byte <= i2c_adapter->last_byte);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void go_read_byte(struct pios_i2c_adapter *i2c_adapter, __attribute__((unused)) bool *woken)
|
|
|
|
{
|
|
|
|
PIOS_DEBUG_Assert(i2c_adapter->active_txn);
|
|
|
|
PIOS_DEBUG_Assert(i2c_adapter->active_txn <= i2c_adapter->last_txn);
|
|
|
|
|
|
|
|
*(i2c_adapter->active_byte) = I2C_ReceiveData(i2c_adapter->cfg->regs);
|
|
|
|
|
|
|
|
/* Move to the next byte */
|
|
|
|
i2c_adapter->active_byte++;
|
|
|
|
PIOS_DEBUG_Assert(i2c_adapter->active_byte <= i2c_adapter->last_byte);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void go_transfer_complete(struct pios_i2c_adapter *i2c_adapter, __attribute__((unused)) bool *woken)
|
|
|
|
{
|
|
|
|
/* Move to the next transaction */
|
|
|
|
i2c_adapter->active_txn++;
|
|
|
|
PIOS_DEBUG_Assert(i2c_adapter->active_txn <= i2c_adapter->last_txn);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void go_nack(struct pios_i2c_adapter *i2c_adapter, __attribute__((unused)) bool *woken)
|
|
|
|
{
|
|
|
|
i2c_adapter->nack = true;
|
|
|
|
I2C_ITConfig(i2c_adapter->cfg->regs, I2C_IT_ERRI | I2C_IT_TCI | I2C_IT_NACKI | I2C_IT_RXI | I2C_IT_STOPI | I2C_IT_TXI, DISABLE);
|
|
|
|
I2C_AcknowledgeConfig(i2c_adapter->cfg->regs, DISABLE);
|
|
|
|
I2C_GenerateSTOP(i2c_adapter->cfg->regs, ENABLE);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void i2c_adapter_inject_event(struct pios_i2c_adapter *i2c_adapter, enum i2c_adapter_event event, bool *woken)
|
|
|
|
{
|
|
|
|
#if defined(PIOS_I2C_DIAGNOSTICS)
|
|
|
|
i2c_state_event_history[i2c_state_event_history_pointer] = event;
|
|
|
|
i2c_state_event_history_pointer = (i2c_state_event_history_pointer + 1) % I2C_LOG_DEPTH;
|
|
|
|
|
|
|
|
i2c_state_history[i2c_state_history_pointer] = i2c_adapter->curr_state;
|
|
|
|
i2c_state_history_pointer = (i2c_state_history_pointer + 1) % I2C_LOG_DEPTH;
|
|
|
|
|
|
|
|
if (i2c_adapter_transitions[i2c_adapter->curr_state].next_state[event] == I2C_STATE_FSM_FAULT) {
|
|
|
|
i2c_adapter_log_fault(i2c_adapter, PIOS_I2C_ERROR_FSM);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
/*
|
|
|
|
* Move to the next state
|
|
|
|
*
|
|
|
|
* This is done prior to calling the new state's entry function to
|
|
|
|
* guarantee that the entry function never depends on the previous
|
|
|
|
* state. This way, it cannot ever know what the previous state was.
|
|
|
|
*/
|
|
|
|
enum i2c_adapter_state prev_state = i2c_adapter->curr_state;
|
|
|
|
if (prev_state) {
|
|
|
|
;
|
|
|
|
}
|
|
|
|
|
|
|
|
i2c_adapter->curr_state = i2c_adapter_transitions[i2c_adapter->curr_state].next_state[event];
|
|
|
|
|
|
|
|
/* Call the entry function (if any) for the next state. */
|
|
|
|
if (i2c_adapter_transitions[i2c_adapter->curr_state].entry_fn) {
|
|
|
|
i2c_adapter_transitions[i2c_adapter->curr_state].entry_fn(i2c_adapter, woken);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Process any AUTO transitions in the FSM */
|
|
|
|
i2c_adapter_process_auto(i2c_adapter, woken);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void i2c_adapter_process_auto(struct pios_i2c_adapter *i2c_adapter, bool *woken)
|
|
|
|
{
|
|
|
|
enum i2c_adapter_state prev_state = i2c_adapter->curr_state;
|
|
|
|
|
|
|
|
if (prev_state) {
|
|
|
|
;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (i2c_adapter_transitions[i2c_adapter->curr_state].next_state[I2C_EVENT_AUTO]) {
|
|
|
|
i2c_adapter->curr_state = i2c_adapter_transitions[i2c_adapter->curr_state].next_state[I2C_EVENT_AUTO];
|
|
|
|
|
|
|
|
/* Call the entry function (if any) for the next state. */
|
|
|
|
if (i2c_adapter_transitions[i2c_adapter->curr_state].entry_fn) {
|
|
|
|
i2c_adapter_transitions[i2c_adapter->curr_state].entry_fn(i2c_adapter, woken);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void i2c_adapter_fsm_init(struct pios_i2c_adapter *i2c_adapter)
|
|
|
|
{
|
|
|
|
i2c_adapter_reset_bus(i2c_adapter);
|
|
|
|
i2c_adapter->curr_state = I2C_STATE_STOPPED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void i2c_adapter_reset_bus(struct pios_i2c_adapter *i2c_adapter)
|
|
|
|
{
|
|
|
|
uint8_t retry_count = 0;
|
|
|
|
uint8_t retry_count_clk = 0;
|
|
|
|
static const uint8_t MAX_I2C_RETRY_COUNT = 10;
|
|
|
|
|
|
|
|
/* Reset the I2C block */
|
|
|
|
I2C_DeInit(i2c_adapter->cfg->regs);
|
|
|
|
|
|
|
|
/* Make sure the bus is free by clocking it until any slaves release the bus. */
|
|
|
|
GPIO_InitTypeDef scl_gpio_init;
|
|
|
|
scl_gpio_init = i2c_adapter->cfg->scl.init;
|
|
|
|
scl_gpio_init.GPIO_Mode = GPIO_Mode_OUT;
|
|
|
|
GPIO_SetBits(i2c_adapter->cfg->scl.gpio, i2c_adapter->cfg->scl.init.GPIO_Pin);
|
|
|
|
GPIO_Init(i2c_adapter->cfg->scl.gpio, &scl_gpio_init);
|
|
|
|
|
|
|
|
GPIO_InitTypeDef sda_gpio_init;
|
|
|
|
sda_gpio_init = i2c_adapter->cfg->sda.init;
|
|
|
|
sda_gpio_init.GPIO_Mode = GPIO_Mode_OUT;
|
|
|
|
GPIO_SetBits(i2c_adapter->cfg->sda.gpio, i2c_adapter->cfg->sda.init.GPIO_Pin);
|
|
|
|
GPIO_Init(i2c_adapter->cfg->sda.gpio, &sda_gpio_init);
|
|
|
|
|
|
|
|
/* Check SDA line to determine if slave is asserting bus and clock out if so, this may */
|
|
|
|
/* have to be repeated (due to further bus errors) but better than clocking 0xFF into an */
|
|
|
|
/* ESC */
|
|
|
|
|
|
|
|
retry_count_clk = 0;
|
|
|
|
while (GPIO_ReadInputDataBit(i2c_adapter->cfg->sda.gpio, i2c_adapter->cfg->sda.init.GPIO_Pin) == Bit_RESET && (retry_count_clk++ < MAX_I2C_RETRY_COUNT)) {
|
|
|
|
retry_count = 0;
|
|
|
|
/* Set clock high and wait for any clock stretching to finish. */
|
|
|
|
GPIO_SetBits(i2c_adapter->cfg->scl.gpio, i2c_adapter->cfg->scl.init.GPIO_Pin);
|
|
|
|
while (GPIO_ReadInputDataBit(i2c_adapter->cfg->scl.gpio, i2c_adapter->cfg->scl.init.GPIO_Pin) == Bit_RESET && (retry_count++ < MAX_I2C_RETRY_COUNT)) {
|
|
|
|
PIOS_DELAY_WaituS(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
PIOS_DELAY_WaituS(2);
|
|
|
|
|
|
|
|
/* Set clock low */
|
|
|
|
GPIO_ResetBits(i2c_adapter->cfg->scl.gpio, i2c_adapter->cfg->scl.init.GPIO_Pin);
|
|
|
|
PIOS_DELAY_WaituS(2);
|
|
|
|
|
|
|
|
/* Clock high again */
|
|
|
|
GPIO_SetBits(i2c_adapter->cfg->scl.gpio, i2c_adapter->cfg->scl.init.GPIO_Pin);
|
|
|
|
PIOS_DELAY_WaituS(2);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Generate a start then stop condition */
|
|
|
|
GPIO_SetBits(i2c_adapter->cfg->scl.gpio, i2c_adapter->cfg->scl.init.GPIO_Pin);
|
|
|
|
PIOS_DELAY_WaituS(2);
|
|
|
|
GPIO_ResetBits(i2c_adapter->cfg->sda.gpio, i2c_adapter->cfg->sda.init.GPIO_Pin);
|
|
|
|
PIOS_DELAY_WaituS(2);
|
|
|
|
GPIO_ResetBits(i2c_adapter->cfg->sda.gpio, i2c_adapter->cfg->sda.init.GPIO_Pin);
|
|
|
|
PIOS_DELAY_WaituS(2);
|
|
|
|
|
|
|
|
/* Set data and clock high and wait for any clock stretching to finish. */
|
|
|
|
GPIO_SetBits(i2c_adapter->cfg->sda.gpio, i2c_adapter->cfg->sda.init.GPIO_Pin);
|
|
|
|
GPIO_SetBits(i2c_adapter->cfg->scl.gpio, i2c_adapter->cfg->scl.init.GPIO_Pin);
|
|
|
|
|
|
|
|
retry_count = 0;
|
|
|
|
while (GPIO_ReadInputDataBit(i2c_adapter->cfg->scl.gpio, i2c_adapter->cfg->scl.init.GPIO_Pin) == Bit_RESET && (retry_count++ < MAX_I2C_RETRY_COUNT)) {
|
|
|
|
PIOS_DELAY_WaituS(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Wait for data to be high */
|
|
|
|
retry_count = 0;
|
|
|
|
while (GPIO_ReadInputDataBit(i2c_adapter->cfg->sda.gpio, i2c_adapter->cfg->sda.init.GPIO_Pin) != Bit_SET && (retry_count++ < MAX_I2C_RETRY_COUNT)) {
|
|
|
|
PIOS_DELAY_WaituS(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Bus signals are guaranteed to be high (ie. free) after this point */
|
|
|
|
/* Initialize the GPIO pins to the peripheral function */
|
|
|
|
if (i2c_adapter->cfg->remapSCL) {
|
|
|
|
GPIO_PinAFConfig(i2c_adapter->cfg->scl.gpio,
|
|
|
|
__builtin_ctz(i2c_adapter->cfg->scl.init.GPIO_Pin),
|
|
|
|
i2c_adapter->cfg->remapSCL);
|
|
|
|
}
|
|
|
|
if (i2c_adapter->cfg->remapSDA) {
|
|
|
|
GPIO_PinAFConfig(i2c_adapter->cfg->sda.gpio,
|
|
|
|
__builtin_ctz(i2c_adapter->cfg->sda.init.GPIO_Pin),
|
|
|
|
i2c_adapter->cfg->remapSDA);
|
|
|
|
}
|
|
|
|
GPIO_Init(i2c_adapter->cfg->scl.gpio, (GPIO_InitTypeDef *)&(i2c_adapter->cfg->scl.init)); // Struct is const, function signature not
|
|
|
|
GPIO_Init(i2c_adapter->cfg->sda.gpio, (GPIO_InitTypeDef *)&(i2c_adapter->cfg->sda.init));
|
|
|
|
|
|
|
|
RCC_I2CCLKConfig(i2c_adapter->cfg->regs == I2C2 ? RCC_I2C2CLK_SYSCLK : RCC_I2C1CLK_SYSCLK);
|
|
|
|
|
|
|
|
/* Reset the I2C block */
|
|
|
|
I2C_DeInit(i2c_adapter->cfg->regs);
|
|
|
|
|
|
|
|
/* Initialize the I2C block */
|
|
|
|
I2C_Init(i2c_adapter->cfg->regs, (I2C_InitTypeDef *)&(i2c_adapter->cfg->init));
|
|
|
|
|
|
|
|
/* Enable the I2C block */
|
|
|
|
I2C_Cmd(i2c_adapter->cfg->regs, ENABLE);
|
|
|
|
|
|
|
|
if (I2C_GetFlagStatus(i2c_adapter->cfg->regs, I2C_FLAG_BUSY)) {
|
|
|
|
I2C_SoftwareResetCmd(i2c_adapter->cfg->regs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Logs the last N state transitions and N IRQ events due to
|
|
|
|
* an error condition
|
|
|
|
* \param[in] i2c the adapter number to log an event for
|
|
|
|
*/
|
|
|
|
#if defined(PIOS_I2C_DIAGNOSTICS)
|
|
|
|
void i2c_adapter_log_fault(__attribute__((unused)) struct pios_i2c_adapter *i2c_adapter, enum pios_i2c_error_type type)
|
|
|
|
{
|
|
|
|
i2c_adapter_fault_history.type = type;
|
|
|
|
for (uint8_t i = 0; i < I2C_LOG_DEPTH; i++) {
|
|
|
|
i2c_adapter_fault_history.evirq[i] =
|
|
|
|
i2c_evirq_history[(I2C_LOG_DEPTH + i2c_evirq_history_pointer - 1 - i) % I2C_LOG_DEPTH];
|
|
|
|
i2c_adapter_fault_history.erirq[i] =
|
|
|
|
i2c_erirq_history[(I2C_LOG_DEPTH + i2c_erirq_history_pointer - 1 - i) % I2C_LOG_DEPTH];
|
|
|
|
i2c_adapter_fault_history.event[i] =
|
|
|
|
i2c_state_event_history[(I2C_LOG_DEPTH + i2c_state_event_history_pointer - 1 - i) % I2C_LOG_DEPTH];
|
|
|
|
i2c_adapter_fault_history.state[i] =
|
|
|
|
i2c_state_history[(I2C_LOG_DEPTH + i2c_state_history_pointer - 1 - i) % I2C_LOG_DEPTH];
|
|
|
|
}
|
|
|
|
switch (type) {
|
|
|
|
case PIOS_I2C_ERROR_EVENT:
|
|
|
|
i2c_bad_event_counter++;
|
|
|
|
break;
|
|
|
|
case PIOS_I2C_ERROR_FSM:
|
|
|
|
i2c_fsm_fault_count++;
|
|
|
|
break;
|
|
|
|
case PIOS_I2C_ERROR_INTERRUPT:
|
|
|
|
i2c_error_interrupt_counter++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif /* if defined(PIOS_I2C_DIAGNOSTICS) */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Logs the last N state transitions and N IRQ events due to
|
|
|
|
* an error condition
|
|
|
|
* \param[out] data address where to copy the pios_i2c_fault_history structure to
|
|
|
|
* \param[out] counts three uint16 that receive the bad event, fsm, and error irq
|
|
|
|
* counts
|
|
|
|
*/
|
|
|
|
void PIOS_I2C_GetDiagnostics(struct pios_i2c_fault_history *data, uint8_t *counts)
|
|
|
|
{
|
|
|
|
#if defined(PIOS_I2C_DIAGNOSTICS)
|
|
|
|
memcpy(data, &i2c_adapter_fault_history, sizeof(i2c_adapter_fault_history));
|
|
|
|
counts[PIOS_I2C_BAD_EVENT_COUNTER] = i2c_bad_event_counter;
|
|
|
|
counts[PIOS_I2C_FSM_FAULT_COUNT] = i2c_fsm_fault_count;
|
|
|
|
counts[PIOS_I2C_ERROR_INTERRUPT_COUNTER] = i2c_error_interrupt_counter;
|
|
|
|
counts[PIOS_I2C_NACK_COUNTER] = i2c_nack_counter;
|
|
|
|
counts[PIOS_I2C_TIMEOUT_COUNTER] = i2c_timeout_counter;
|
|
|
|
#else
|
|
|
|
struct pios_i2c_fault_history i2c_adapter_fault_history;
|
|
|
|
i2c_adapter_fault_history.type = PIOS_I2C_ERROR_EVENT;
|
|
|
|
|
|
|
|
memcpy(data, &i2c_adapter_fault_history, sizeof(i2c_adapter_fault_history));
|
|
|
|
memset(counts, 0, sizeof(*counts) * PIOS_I2C_ERROR_COUNT_NUMELEM);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool PIOS_I2C_validate(struct pios_i2c_adapter *i2c_adapter)
|
|
|
|
{
|
|
|
|
return i2c_adapter->magic == PIOS_I2C_DEV_MAGIC;
|
|
|
|
}
|
|
|
|
|
2017-05-30 16:31:39 +02:00
|
|
|
#if defined(PIOS_INCLUDE_FREERTOS)
|
|
|
|
static struct pios_i2c_adapter *PIOS_I2C_alloc(void)
|
2017-05-22 15:08:04 +02:00
|
|
|
{
|
2017-05-30 16:31:39 +02:00
|
|
|
struct pios_i2c_adapter *i2c_adapter;
|
2017-05-22 15:08:04 +02:00
|
|
|
|
2017-05-30 16:31:39 +02:00
|
|
|
i2c_adapter = (struct pios_i2c_adapter *)pios_malloc(sizeof(*i2c_adapter));
|
2017-05-22 15:08:04 +02:00
|
|
|
if (!i2c_adapter) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
i2c_adapter->magic = PIOS_I2C_DEV_MAGIC;
|
|
|
|
return i2c_adapter;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
static struct pios_i2c_adapter pios_i2c_adapters[PIOS_I2C_MAX_DEVS];
|
|
|
|
static uint8_t pios_i2c_num_adapters;
|
|
|
|
static struct pios_i2c_adapter *PIOS_I2C_alloc(void)
|
|
|
|
{
|
|
|
|
struct pios_i2c_adapter *i2c_adapter;
|
|
|
|
|
|
|
|
if (pios_i2c_num_adapters >= PIOS_I2C_MAX_DEVS) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
i2c_adapter = &pios_i2c_adapters[pios_i2c_num_adapters++];
|
|
|
|
i2c_adapter->magic = PIOS_I2C_DEV_MAGIC;
|
|
|
|
|
|
|
|
return i2c_adapter;
|
|
|
|
}
|
|
|
|
#endif /* if defined(PIOS_INCLUDE_CHIBIOS) && 0 */
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Initializes IIC driver
|
|
|
|
* \param[in] mode currently only mode 0 supported
|
|
|
|
* \return < 0 if initialisation failed
|
|
|
|
*/
|
|
|
|
int32_t PIOS_I2C_Init(uint32_t *i2c_id, const struct pios_i2c_adapter_cfg *cfg)
|
|
|
|
{
|
|
|
|
PIOS_DEBUG_Assert(i2c_id);
|
|
|
|
PIOS_DEBUG_Assert(cfg);
|
|
|
|
|
|
|
|
struct pios_i2c_adapter *i2c_adapter;
|
|
|
|
|
|
|
|
i2c_adapter = (struct pios_i2c_adapter *)PIOS_I2C_alloc();
|
|
|
|
if (!i2c_adapter) {
|
|
|
|
goto out_fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Bind the configuration to the device instance */
|
|
|
|
i2c_adapter->cfg = cfg;
|
|
|
|
|
|
|
|
vSemaphoreCreateBinary(i2c_adapter->sem_ready);
|
|
|
|
i2c_adapter->sem_busy = xSemaphoreCreateMutex();
|
|
|
|
|
|
|
|
/* Initialize the state machine */
|
|
|
|
i2c_adapter_fsm_init(i2c_adapter);
|
|
|
|
|
|
|
|
*i2c_id = (uint32_t)i2c_adapter;
|
|
|
|
|
|
|
|
/* Configure and enable I2C interrupts */
|
|
|
|
NVIC_Init((NVIC_InitTypeDef *)&(i2c_adapter->cfg->event.init));
|
|
|
|
NVIC_Init((NVIC_InitTypeDef *)&(i2c_adapter->cfg->error.init));
|
|
|
|
|
|
|
|
/* No error */
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
out_fail:
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t PIOS_I2C_Transfer(uint32_t i2c_id, const struct pios_i2c_txn txn_list[], uint32_t num_txns)
|
|
|
|
{
|
|
|
|
// FIXME: only supports transfer sizes up to 255 bytes
|
|
|
|
struct pios_i2c_adapter *i2c_adapter = (struct pios_i2c_adapter *)i2c_id;
|
|
|
|
|
|
|
|
bool valid = PIOS_I2C_validate(i2c_adapter);
|
|
|
|
|
|
|
|
PIOS_Assert(valid)
|
|
|
|
|
|
|
|
PIOS_DEBUG_Assert(txn_list);
|
|
|
|
PIOS_DEBUG_Assert(num_txns);
|
|
|
|
|
|
|
|
bool semaphore_success = true;
|
|
|
|
|
2017-05-30 16:31:39 +02:00
|
|
|
#ifdef PIOS_INCLUDE_FREERTOS
|
2017-05-22 15:08:04 +02:00
|
|
|
/* Lock the bus */
|
|
|
|
portTickType timeout;
|
|
|
|
timeout = i2c_adapter->cfg->transfer_timeout_ms / portTICK_RATE_MS;
|
|
|
|
if (xSemaphoreTake(i2c_adapter->sem_busy, timeout) == pdFALSE) {
|
|
|
|
return -2;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
PIOS_IRQ_Disable();
|
|
|
|
if (i2c_adapter->busy) {
|
|
|
|
PIOS_IRQ_Enable();
|
|
|
|
return -2;
|
|
|
|
}
|
|
|
|
i2c_adapter->busy = 1;
|
|
|
|
PIOS_IRQ_Enable();
|
2017-05-30 16:31:39 +02:00
|
|
|
#endif /* PIOS_INCLUDE_FREERTOS */
|
2017-05-22 15:08:04 +02:00
|
|
|
|
|
|
|
PIOS_DEBUG_Assert(i2c_adapter->curr_state == I2C_STATE_STOPPED);
|
|
|
|
|
|
|
|
i2c_adapter->last_txn = &txn_list[num_txns - 1];
|
|
|
|
i2c_adapter->active_txn = &txn_list[0];
|
|
|
|
i2c_adapter->bus_error = false;
|
|
|
|
i2c_adapter->nack = false;
|
|
|
|
|
2017-05-30 16:31:39 +02:00
|
|
|
#ifdef PIOS_INCLUDE_FREERTOS
|
2017-05-22 15:08:04 +02:00
|
|
|
/* Make sure the done/ready semaphore is consumed before we start */
|
|
|
|
semaphore_success &= (xSemaphoreTake(i2c_adapter->sem_ready, timeout) == pdTRUE);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
i2c_adapter->callback = NULL;
|
|
|
|
|
|
|
|
bool dummy = false;
|
|
|
|
i2c_adapter_inject_event(i2c_adapter, I2C_EVENT_START, &dummy);
|
|
|
|
|
|
|
|
/* Wait for the transfer to complete */
|
2017-05-30 16:31:39 +02:00
|
|
|
#ifdef PIOS_INCLUDE_FREERTOS
|
2017-05-22 15:08:04 +02:00
|
|
|
semaphore_success &= (xSemaphoreTake(i2c_adapter->sem_ready, timeout) == pdTRUE);
|
|
|
|
xSemaphoreGive(i2c_adapter->sem_ready);
|
2017-05-30 16:31:39 +02:00
|
|
|
#endif /* PIOS_INCLUDE_FREERTOS */
|
2017-05-22 15:08:04 +02:00
|
|
|
|
2017-05-30 16:31:39 +02:00
|
|
|
#ifdef PIOS_INCLUDE_FREERTOS
|
2017-05-22 15:08:04 +02:00
|
|
|
/* Unlock the bus */
|
|
|
|
xSemaphoreGive(i2c_adapter->sem_busy);
|
|
|
|
#if defined(PIOS_I2C_DIAGNOSTICS)
|
|
|
|
if (!semaphore_success) {
|
|
|
|
i2c_timeout_counter++;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
PIOS_IRQ_Disable();
|
|
|
|
i2c_adapter->busy = 0;
|
|
|
|
PIOS_IRQ_Enable();
|
2017-05-30 16:31:39 +02:00
|
|
|
#endif /* PIOS_INCLUDE_FREERTOS */
|
2017-05-22 15:08:04 +02:00
|
|
|
|
|
|
|
return !semaphore_success ? -2 :
|
|
|
|
i2c_adapter->bus_error ? -1 :
|
|
|
|
i2c_adapter->nack ? -3 :
|
|
|
|
0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int32_t PIOS_I2C_Transfer_Callback_Internal(struct pios_i2c_adapter *i2c_adapter, const struct pios_i2c_txn txn_list[], uint32_t num_txns, pios_i2c_callback callback)
|
|
|
|
{
|
|
|
|
PIOS_DEBUG_Assert(i2c_adapter->curr_state == I2C_STATE_STOPPED);
|
|
|
|
|
|
|
|
i2c_adapter->last_txn = &txn_list[num_txns - 1];
|
|
|
|
i2c_adapter->active_txn = &txn_list[0];
|
|
|
|
i2c_adapter->bus_error = false;
|
|
|
|
i2c_adapter->nack = false;
|
|
|
|
i2c_adapter->callback = callback;
|
|
|
|
|
|
|
|
// Estimate bytes of transmission. Per txns: 1 adress byte + length
|
|
|
|
i2c_adapter->transfer_timeout_ticks = num_txns;
|
|
|
|
for (uint32_t i = 0; i < num_txns; i++) {
|
|
|
|
i2c_adapter->transfer_timeout_ticks += txn_list[i].len;
|
|
|
|
}
|
|
|
|
// timeout if it takes eight times the expected time
|
|
|
|
i2c_adapter->transfer_timeout_ticks <<= 3;
|
|
|
|
|
|
|
|
bool dummy = false;
|
|
|
|
i2c_adapter_inject_event(i2c_adapter, I2C_EVENT_START, &dummy);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t PIOS_I2C_Transfer_Callback(uint32_t i2c_id, const struct pios_i2c_txn txn_list[], uint32_t num_txns, pios_i2c_callback callback)
|
|
|
|
{
|
|
|
|
// FIXME: only supports transfer sizes up to 255 bytes
|
|
|
|
struct pios_i2c_adapter *i2c_adapter = (struct pios_i2c_adapter *)i2c_id;
|
|
|
|
|
|
|
|
bool valid = PIOS_I2C_validate(i2c_adapter);
|
|
|
|
|
|
|
|
PIOS_Assert(valid)
|
|
|
|
|
|
|
|
PIOS_DEBUG_Assert(txn_list);
|
|
|
|
PIOS_DEBUG_Assert(num_txns);
|
|
|
|
|
2017-05-30 16:31:39 +02:00
|
|
|
#ifdef PIOS_INCLUDE_FREERTOS
|
2017-05-22 15:08:04 +02:00
|
|
|
/* Lock the bus */
|
|
|
|
portTickType timeout;
|
|
|
|
timeout = i2c_adapter->cfg->transfer_timeout_ms / portTICK_RATE_MS;
|
|
|
|
if (xSemaphoreTake(i2c_adapter->sem_busy, timeout) == pdFALSE) {
|
|
|
|
return -2;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
PIOS_IRQ_Disable();
|
|
|
|
if (i2c_adapter->busy) {
|
|
|
|
PIOS_IRQ_Enable();
|
|
|
|
return -2;
|
|
|
|
}
|
|
|
|
i2c_adapter->busy = 1;
|
|
|
|
PIOS_IRQ_Enable();
|
2017-05-30 16:31:39 +02:00
|
|
|
#endif /* PIOS_INCLUDE_FREERTOS */
|
2017-05-22 15:08:04 +02:00
|
|
|
|
|
|
|
return PIOS_I2C_Transfer_Callback_Internal(i2c_adapter, txn_list, num_txns, callback);
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t PIOS_I2C_Transfer_CallbackFromISR(uint32_t i2c_id, const struct pios_i2c_txn txn_list[], uint32_t num_txns, pios_i2c_callback callback, bool *woken)
|
|
|
|
{
|
|
|
|
// FIXME: only supports transfer sizes up to 255 bytes
|
|
|
|
struct pios_i2c_adapter *i2c_adapter = (struct pios_i2c_adapter *)i2c_id;
|
|
|
|
|
|
|
|
bool valid = PIOS_I2C_validate(i2c_adapter);
|
|
|
|
|
|
|
|
PIOS_Assert(valid)
|
|
|
|
|
|
|
|
PIOS_DEBUG_Assert(txn_list);
|
|
|
|
PIOS_DEBUG_Assert(num_txns);
|
|
|
|
|
2017-05-30 16:31:39 +02:00
|
|
|
#ifdef PIOS_INCLUDE_FREERTOS
|
2017-05-22 15:08:04 +02:00
|
|
|
signed portBASE_TYPE xHigherPriorityTaskWoken = pdFALSE;
|
|
|
|
|
|
|
|
/* Lock the bus */
|
|
|
|
bool locked = xSemaphoreTakeFromISR(i2c_adapter->sem_busy, &xHigherPriorityTaskWoken) == pdTRUE;
|
|
|
|
|
|
|
|
if (xHigherPriorityTaskWoken == pdTRUE) {
|
|
|
|
*woken = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!locked) {
|
|
|
|
return -2;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
PIOS_IRQ_Disable();
|
|
|
|
if (i2c_adapter->busy) {
|
|
|
|
PIOS_IRQ_Enable();
|
|
|
|
return -2;
|
|
|
|
}
|
|
|
|
i2c_adapter->busy = 1;
|
|
|
|
PIOS_IRQ_Enable();
|
2017-05-30 16:31:39 +02:00
|
|
|
#endif /* PIOS_INCLUDE_FREERTOS */
|
2017-05-22 15:08:04 +02:00
|
|
|
|
|
|
|
return PIOS_I2C_Transfer_Callback_Internal(i2c_adapter, txn_list, num_txns, callback);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static bool i2c_adapter_callback_handler(struct pios_i2c_adapter *i2c_adapter, bool *woken)
|
|
|
|
{
|
2017-05-30 16:31:39 +02:00
|
|
|
#ifdef PIOS_INCLUDE_FREERTOS
|
2017-05-22 15:08:04 +02:00
|
|
|
/* Unlock the bus */
|
|
|
|
signed portBASE_TYPE xHigherPriorityTaskWoken = pdFALSE;
|
|
|
|
xSemaphoreGiveFromISR(i2c_adapter->sem_busy, &xHigherPriorityTaskWoken);
|
|
|
|
if (xHigherPriorityTaskWoken == pdTRUE) {
|
|
|
|
*woken = true;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
PIOS_IRQ_Disable();
|
|
|
|
i2c_adapter->busy = 0;
|
|
|
|
PIOS_IRQ_Enable();
|
2017-05-30 16:31:39 +02:00
|
|
|
#endif /* PIOS_INCLUDE_FREERTOS */
|
2017-05-22 15:08:04 +02:00
|
|
|
|
|
|
|
|
|
|
|
// Execute user supplied function
|
|
|
|
|
|
|
|
if (i2c_adapter->callback()) {
|
|
|
|
*woken = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return !i2c_adapter->bus_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void PIOS_I2C_EV_IRQ_Handler(uint32_t i2c_id)
|
|
|
|
{
|
|
|
|
struct pios_i2c_adapter *i2c_adapter = (struct pios_i2c_adapter *)i2c_id;
|
|
|
|
|
|
|
|
bool valid = PIOS_I2C_validate(i2c_adapter);
|
|
|
|
|
|
|
|
PIOS_Assert(valid)
|
|
|
|
|
|
|
|
bool woken = false;
|
|
|
|
|
|
|
|
if (I2C_GetFlagStatus(i2c_adapter->cfg->regs, I2C_FLAG_RXNE)) {
|
|
|
|
// flag will be cleared by event
|
|
|
|
#if defined(PIOS_I2C_DIAGNOSTICS)
|
|
|
|
i2c_evirq_history[i2c_evirq_history_pointer] = I2C_FLAG_RXNE;
|
|
|
|
i2c_evirq_history_pointer = (i2c_evirq_history_pointer + 1) % I2C_LOG_DEPTH;
|
|
|
|
#endif
|
|
|
|
i2c_adapter_inject_event(i2c_adapter, I2C_EVENT_RECEIVER_BUFFER_NOT_EMPTY, &woken);
|
|
|
|
} else if (I2C_GetFlagStatus(i2c_adapter->cfg->regs, I2C_FLAG_TXIS)) {
|
|
|
|
// flag will be cleared by event
|
|
|
|
#if defined(PIOS_I2C_DIAGNOSTICS)
|
|
|
|
i2c_evirq_history[i2c_evirq_history_pointer] = I2C_FLAG_TXIS;
|
|
|
|
i2c_evirq_history_pointer = (i2c_evirq_history_pointer + 1) % I2C_LOG_DEPTH;
|
|
|
|
#endif
|
|
|
|
i2c_adapter_inject_event(i2c_adapter, I2C_EVENT_TRANSMIT_BUFFER_EMPTY, &woken);
|
|
|
|
} else if (I2C_GetFlagStatus(i2c_adapter->cfg->regs, I2C_FLAG_NACKF)) {
|
|
|
|
I2C_ClearFlag(i2c_adapter->cfg->regs, I2C_FLAG_NACKF);
|
|
|
|
#if defined(PIOS_I2C_DIAGNOSTICS)
|
|
|
|
i2c_evirq_history[i2c_evirq_history_pointer] = I2C_FLAG_NACKF;
|
|
|
|
i2c_evirq_history_pointer = (i2c_evirq_history_pointer + 1) % I2C_LOG_DEPTH;
|
|
|
|
++i2c_nack_counter;
|
|
|
|
#endif
|
|
|
|
i2c_adapter_inject_event(i2c_adapter, I2C_EVENT_NACK, &woken);
|
|
|
|
} else if (I2C_GetFlagStatus(i2c_adapter->cfg->regs, I2C_FLAG_TC)) {
|
|
|
|
I2C_ClearFlag(i2c_adapter->cfg->regs, I2C_FLAG_TC);
|
|
|
|
#if defined(PIOS_I2C_DIAGNOSTICS)
|
|
|
|
i2c_evirq_history[i2c_evirq_history_pointer] = I2C_FLAG_TC;
|
|
|
|
i2c_evirq_history_pointer = (i2c_evirq_history_pointer + 1) % I2C_LOG_DEPTH;
|
|
|
|
#endif
|
|
|
|
i2c_adapter_inject_event(i2c_adapter, I2C_EVENT_TRANSFER_COMPLETE, &woken);
|
|
|
|
} else if (I2C_GetFlagStatus(i2c_adapter->cfg->regs, I2C_FLAG_STOPF)) {
|
|
|
|
I2C_ClearFlag(i2c_adapter->cfg->regs, I2C_FLAG_STOPF);
|
|
|
|
#if defined(PIOS_I2C_DIAGNOSTICS)
|
|
|
|
i2c_evirq_history[i2c_evirq_history_pointer] = I2C_FLAG_STOPF;
|
|
|
|
i2c_evirq_history_pointer = (i2c_evirq_history_pointer + 1) % I2C_LOG_DEPTH;
|
|
|
|
#endif
|
|
|
|
i2c_adapter_inject_event(i2c_adapter, I2C_EVENT_STOP, &woken);
|
|
|
|
}
|
|
|
|
|
|
|
|
portEND_SWITCHING_ISR(woken ? pdTRUE : pdFALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void PIOS_I2C_ER_IRQ_Handler(uint32_t i2c_id)
|
|
|
|
{
|
|
|
|
struct pios_i2c_adapter *i2c_adapter = (struct pios_i2c_adapter *)i2c_id;
|
|
|
|
|
|
|
|
bool valid = PIOS_I2C_validate(i2c_adapter);
|
|
|
|
|
|
|
|
PIOS_Assert(valid)
|
|
|
|
|
|
|
|
else if (I2C_GetFlagStatus(i2c_adapter->cfg->regs, I2C_FLAG_BERR)) {
|
|
|
|
I2C_ClearFlag(i2c_adapter->cfg->regs, I2C_FLAG_BERR);
|
|
|
|
#if defined(PIOS_I2C_DIAGNOSTICS)
|
|
|
|
i2c_erirq_history[i2c_erirq_history_pointer] = I2C_FLAG_BERR;
|
|
|
|
i2c_erirq_history_pointer = (i2c_erirq_history_pointer + 1) % I2C_LOG_DEPTH;
|
|
|
|
#endif
|
|
|
|
} else if (I2C_GetFlagStatus(i2c_adapter->cfg->regs, I2C_FLAG_ARLO)) {
|
|
|
|
I2C_ClearFlag(i2c_adapter->cfg->regs, I2C_FLAG_ARLO);
|
|
|
|
#if defined(PIOS_I2C_DIAGNOSTICS)
|
|
|
|
i2c_erirq_history[i2c_erirq_history_pointer] = I2C_FLAG_ARLO;
|
|
|
|
i2c_erirq_history_pointer = (i2c_erirq_history_pointer + 1) % I2C_LOG_DEPTH;
|
|
|
|
#endif
|
|
|
|
} else if (I2C_GetFlagStatus(i2c_adapter->cfg->regs, I2C_FLAG_OVR)) {
|
|
|
|
I2C_ClearFlag(i2c_adapter->cfg->regs, I2C_FLAG_OVR);
|
|
|
|
#if defined(PIOS_I2C_DIAGNOSTICS)
|
|
|
|
i2c_erirq_history[i2c_erirq_history_pointer] = I2C_FLAG_OVR;
|
|
|
|
i2c_erirq_history_pointer = (i2c_erirq_history_pointer + 1) % I2C_LOG_DEPTH;
|
|
|
|
#endif
|
|
|
|
} else if (I2C_GetFlagStatus(i2c_adapter->cfg->regs, I2C_FLAG_PECERR)) {
|
|
|
|
I2C_ClearFlag(i2c_adapter->cfg->regs, I2C_FLAG_PECERR);
|
|
|
|
#if defined(PIOS_I2C_DIAGNOSTICS)
|
|
|
|
i2c_erirq_history[i2c_erirq_history_pointer] = I2C_FLAG_PECERR;
|
|
|
|
i2c_erirq_history_pointer = (i2c_erirq_history_pointer + 1) % I2C_LOG_DEPTH;
|
|
|
|
#endif
|
|
|
|
} else if (I2C_GetFlagStatus(i2c_adapter->cfg->regs, I2C_FLAG_TIMEOUT)) {
|
|
|
|
I2C_ClearFlag(i2c_adapter->cfg->regs, I2C_FLAG_TIMEOUT);
|
|
|
|
#if defined(PIOS_I2C_DIAGNOSTICS)
|
|
|
|
i2c_erirq_history[i2c_erirq_history_pointer] = I2C_FLAG_TIMEOUT;
|
|
|
|
i2c_erirq_history_pointer = (i2c_erirq_history_pointer + 1) % I2C_LOG_DEPTH;
|
|
|
|
#endif
|
|
|
|
} else if (I2C_GetFlagStatus(i2c_adapter->cfg->regs, I2C_FLAG_ALERT)) {
|
|
|
|
I2C_ClearFlag(i2c_adapter->cfg->regs, I2C_FLAG_ALERT);
|
|
|
|
#if defined(PIOS_I2C_DIAGNOSTICS)
|
|
|
|
i2c_erirq_history[i2c_erirq_history_pointer] = I2C_FLAG_ALERT;
|
|
|
|
i2c_erirq_history_pointer = (i2c_erirq_history_pointer + 1) % I2C_LOG_DEPTH;
|
|
|
|
#endif
|
|
|
|
} else if (I2C_GetFlagStatus(i2c_adapter->cfg->regs, I2C_FLAG_BUSY)) {
|
|
|
|
I2C_ClearFlag(i2c_adapter->cfg->regs, I2C_FLAG_BUSY);
|
|
|
|
#if defined(PIOS_I2C_DIAGNOSTICS)
|
|
|
|
i2c_erirq_history[i2c_erirq_history_pointer] = I2C_FLAG_BUSY;
|
|
|
|
i2c_erirq_history_pointer = (i2c_erirq_history_pointer + 1) % I2C_LOG_DEPTH;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined(PIOS_I2C_DIAGNOSTICS)
|
|
|
|
i2c_adapter_log_fault(i2c_adapter, PIOS_I2C_ERROR_INTERRUPT);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Fail hard on any errors for now */
|
|
|
|
bool woken = false;
|
|
|
|
i2c_adapter_inject_event(i2c_adapter, I2C_EVENT_BUS_ERROR, &woken);
|
|
|
|
|
|
|
|
portEND_SWITCHING_ISR(woken ? pdTRUE : pdFALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* if defined(PIOS_INCLUDE_I2C) */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @}
|
|
|
|
* @}
|
|
|
|
*/
|