Candidate fix for STM32 F4 I2C
git-svn-id: svn://svn.code.sf.net/p/nuttx/code/trunk@4537 42af7a65-404d-4744-a932-0658087f49c3
This commit is contained in:
parent
31bc31d063
commit
e13b101db6
@ -1,12 +1,13 @@
|
|||||||
/************************************************************************************
|
/************************************************************************************
|
||||||
* arch/arm/src/stm32/stm32_i2c.c
|
* arch/arm/src/stm32/stm32_i2c.c
|
||||||
|
* STM32 I2C Hardware Layer - Device Driver
|
||||||
*
|
*
|
||||||
* Copyright (C) 2011 Uros Platise. All rights reserved.
|
* Copyright (C) 2011 Uros Platise. All rights reserved.
|
||||||
* Author: Uros Platise <uros.platise@isotel.eu>
|
* Author: Uros Platise <uros.platise@isotel.eu>
|
||||||
*
|
*
|
||||||
* With extensions, modifications by:
|
* With extensions, modifications by:
|
||||||
*
|
*
|
||||||
* Copyright (C) 2011 Gregory Nutt. All rights reserved.
|
* Copyright (C) 2011-2012 Gregory Nutt. All rights reserved.
|
||||||
* Author: Gregroy Nutt <gnutt@nuttx.org>
|
* Author: Gregroy Nutt <gnutt@nuttx.org>
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
@ -38,11 +39,7 @@
|
|||||||
*
|
*
|
||||||
************************************************************************************/
|
************************************************************************************/
|
||||||
|
|
||||||
/* \file
|
/* Supports:
|
||||||
* \author Uros Platise
|
|
||||||
* \brief STM32 I2C Hardware Layer - Device Driver
|
|
||||||
*
|
|
||||||
* Supports:
|
|
||||||
* - Master operation, 100 kHz (standard) and 400 kHz (full speed)
|
* - Master operation, 100 kHz (standard) and 400 kHz (full speed)
|
||||||
* - Multiple instances (shared bus)
|
* - Multiple instances (shared bus)
|
||||||
* - Interrupt based operation
|
* - Interrupt based operation
|
||||||
@ -55,7 +52,7 @@
|
|||||||
* its own private data, as frequency, address, mode of operation (in the future)
|
* its own private data, as frequency, address, mode of operation (in the future)
|
||||||
* - Private: Private data of an I2C Hardware
|
* - Private: Private data of an I2C Hardware
|
||||||
*
|
*
|
||||||
* \todo
|
* TODO
|
||||||
* - Check for all possible deadlocks (as BUSY='1' I2C needs to be reset in HW using the I2C_CR1_SWRST)
|
* - Check for all possible deadlocks (as BUSY='1' I2C needs to be reset in HW using the I2C_CR1_SWRST)
|
||||||
* - SMBus support (hardware layer timings are already supported) and add SMBA gpio pin
|
* - SMBus support (hardware layer timings are already supported) and add SMBA gpio pin
|
||||||
* - Slave support with multiple addresses (on multiple instances):
|
* - Slave support with multiple addresses (on multiple instances):
|
||||||
@ -65,7 +62,7 @@
|
|||||||
* - Multi-master support
|
* - Multi-master support
|
||||||
* - DMA (to get rid of too many CPU wake-ups and interventions)
|
* - DMA (to get rid of too many CPU wake-ups and interventions)
|
||||||
* - Be ready for IPMI
|
* - Be ready for IPMI
|
||||||
**/
|
*/
|
||||||
|
|
||||||
/************************************************************************************
|
/************************************************************************************
|
||||||
* Included Files
|
* Included Files
|
||||||
@ -156,7 +153,7 @@
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifndef CONFIG_I2C_NTRACE
|
#ifndef CONFIG_I2C_NTRACE
|
||||||
# define CONFIG_I2C_NTRACE 20
|
# define CONFIG_I2C_NTRACE 32
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/************************************************************************************
|
/************************************************************************************
|
||||||
@ -176,14 +173,14 @@ enum stm32_intstate_e
|
|||||||
enum stm32_trace_e
|
enum stm32_trace_e
|
||||||
{
|
{
|
||||||
I2CEVENT_NONE = 0, /* No events have occurred with this status */
|
I2CEVENT_NONE = 0, /* No events have occurred with this status */
|
||||||
I2CEVENT_SB, /* Start/Master, param = msgc */
|
I2CEVENT_SENDADDR, /* Start/Master bit set and address sent, param = msgc */
|
||||||
I2CEVENT_SENDBYTE, /* Send byte, param = byte sent */
|
I2CEVENT_SENDBYTE, /* Send byte, param = dcnt */
|
||||||
I2CEVENT_READ, /* Read data, param = dcnt */
|
|
||||||
I2CEVENT_ITBUFEN, /* Enable buffer interrupts, param = 0 */
|
I2CEVENT_ITBUFEN, /* Enable buffer interrupts, param = 0 */
|
||||||
I2CEVENT_RXNE, /* Read more dta, param = dcnt */
|
I2CEVENT_RCVBYTE, /* Read more dta, param = dcnt */
|
||||||
I2CEVENT_REITBUFEN, /* Re-enable buffer interrupts, param = 0 */
|
I2CEVENT_REITBUFEN, /* Re-enable buffer interrupts, param = 0 */
|
||||||
I2CEVENT_DISITBUFEN, /* Disable buffer interrupts, param = 0 */
|
I2CEVENT_DISITBUFEN, /* Disable buffer interrupts, param = 0 */
|
||||||
I2CEVENT_BTFSTART, /* Last byte sent, re-starting, param = msgc */
|
I2CEVENT_BTFNOSTART, /* BTF on last byte with no restart, param = msgc */
|
||||||
|
I2CEVENT_BTFRESTART, /* Last byte sent, re-starting, param = msgc */
|
||||||
I2CEVENT_BTFSTOP, /* Last byte sten, send stop, param = 0 */
|
I2CEVENT_BTFSTOP, /* Last byte sten, send stop, param = 0 */
|
||||||
I2CEVENT_ERROR /* Error occurred, param = 0 */
|
I2CEVENT_ERROR /* Error occurred, param = 0 */
|
||||||
};
|
};
|
||||||
@ -196,6 +193,7 @@ struct stm32_trace_s
|
|||||||
uint32_t count; /* Interrupt count when status change */
|
uint32_t count; /* Interrupt count when status change */
|
||||||
enum stm32_intstate_e event; /* Last event that occurred with this status */
|
enum stm32_intstate_e event; /* Last event that occurred with this status */
|
||||||
uint32_t parm; /* Parameter associated with the event */
|
uint32_t parm; /* Parameter associated with the event */
|
||||||
|
uint32_t time; /* First of event or first status */
|
||||||
};
|
};
|
||||||
|
|
||||||
/* I2C Device Private Data */
|
/* I2C Device Private Data */
|
||||||
@ -220,8 +218,7 @@ struct stm32_i2c_priv_s
|
|||||||
|
|
||||||
#ifdef CONFIG_I2C_TRACE
|
#ifdef CONFIG_I2C_TRACE
|
||||||
int tndx; /* Trace array index */
|
int tndx; /* Trace array index */
|
||||||
uint32_t isr_count; /* Count of ISRs processed */
|
uint32_t start_time; /* Time when the trace was started */
|
||||||
uint32_t old_status; /* Last 32-bit status value */
|
|
||||||
|
|
||||||
/* The actual trace data */
|
/* The actual trace data */
|
||||||
|
|
||||||
@ -674,64 +671,104 @@ static inline void stm32_i2c_sem_destroy(FAR struct i2c_dev_s *dev)
|
|||||||
************************************************************************************/
|
************************************************************************************/
|
||||||
|
|
||||||
#ifdef CONFIG_I2C_TRACE
|
#ifdef CONFIG_I2C_TRACE
|
||||||
|
static void stm32_i2c_traceclear(FAR struct stm32_i2c_priv_s *priv)
|
||||||
|
{
|
||||||
|
struct stm32_trace_s *trace = &priv->trace[priv->tndx];
|
||||||
|
|
||||||
|
trace->status = 0; /* I2C 32-bit SR2|SR1 status */
|
||||||
|
trace->count = 0; /* Interrupt count when status change */
|
||||||
|
trace->event = I2CEVENT_NONE; /* Last event that occurred with this status */
|
||||||
|
trace->parm = 0; /* Parameter associated with the event */
|
||||||
|
trace->time = 0; /* Time of first status or event */
|
||||||
|
}
|
||||||
|
|
||||||
static void stm32_i2c_tracereset(FAR struct stm32_i2c_priv_s *priv)
|
static void stm32_i2c_tracereset(FAR struct stm32_i2c_priv_s *priv)
|
||||||
{
|
{
|
||||||
/* Reset the trace info for a new data collection */
|
/* Reset the trace info for a new data collection */
|
||||||
|
|
||||||
priv->isr_count = 0;
|
priv->tndx = 0;
|
||||||
priv->old_status = 0xffffffff;
|
priv->start_time = clock_systimer();
|
||||||
priv->tndx = -1;
|
stm32_i2c_traceclear(priv);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void stm32_i2c_tracenew(FAR struct stm32_i2c_priv_s *priv, uint32_t status)
|
static void stm32_i2c_tracenew(FAR struct stm32_i2c_priv_s *priv, uint32_t status)
|
||||||
{
|
{
|
||||||
/* Increment the cout of interrupts received */
|
struct stm32_trace_s *trace = &priv->trace[priv->tndx];
|
||||||
|
|
||||||
priv->isr_count++;
|
/* Is the current entry uninitialized? Has the status changed? */
|
||||||
|
|
||||||
/* Has the status changed from the last interrupt */
|
if (trace->count == 0 || status != trace->status)
|
||||||
|
{
|
||||||
|
/* Yes.. Was it the status changed? */
|
||||||
|
|
||||||
if (status != priv->old_status)
|
if (trace->count != 0)
|
||||||
{
|
{
|
||||||
/* Yes.. bump up the trace index (unless we are out of trace entries) */
|
/* Yes.. bump up the trace index (unless we are out of trace entries) */
|
||||||
|
|
||||||
if (priv->tndx < (CONFIG_I2C_NTRACE-1))
|
if (priv->tndx >= (CONFIG_I2C_NTRACE-1))
|
||||||
{
|
{
|
||||||
|
i2cdbg("Trace table overflow\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
priv->tndx++;
|
priv->tndx++;
|
||||||
|
trace = &priv->trace[priv->tndx];
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Initialize the new trace entry */
|
/* Initialize the new trace entry */
|
||||||
|
|
||||||
priv->trace[priv->tndx].status = status;
|
stm32_i2c_traceclear(priv);
|
||||||
priv->trace[priv->tndx].count = priv->isr_count;
|
trace->status = status;
|
||||||
priv->trace[priv->tndx].event = I2CEVENT_NONE;
|
trace->count = 1;
|
||||||
priv->trace[priv->tndx].parm = 0;
|
trace->time = clock_systimer();
|
||||||
priv->old_status = status;
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* Just increment the count of times that we have seen this status */
|
||||||
|
|
||||||
|
trace->count++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void stm32_i2c_traceevent(FAR struct stm32_i2c_priv_s *priv,
|
static void stm32_i2c_traceevent(FAR struct stm32_i2c_priv_s *priv,
|
||||||
enum stm32_trace_e event, uint32_t parm)
|
enum stm32_trace_e event, uint32_t parm)
|
||||||
{
|
{
|
||||||
/* Add the event to the trace entry (possibly overwriting a previous trace
|
struct stm32_trace_s *trace;
|
||||||
* event.
|
|
||||||
*/
|
|
||||||
|
|
||||||
priv->trace[priv->tndx].event = event;
|
if (event != I2CEVENT_NONE)
|
||||||
priv->trace[priv->tndx].parm = parm;
|
{
|
||||||
|
trace = &priv->trace[priv->tndx];
|
||||||
|
|
||||||
|
/* Initialize the new trace entry */
|
||||||
|
|
||||||
|
trace->event = event;
|
||||||
|
trace->parm = parm;
|
||||||
|
|
||||||
|
/* Bump up the trace index (unless we are out of trace entries) */
|
||||||
|
|
||||||
|
if (priv->tndx >= (CONFIG_I2C_NTRACE-1))
|
||||||
|
{
|
||||||
|
i2cdbg("Trace table overflow\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
priv->tndx++;
|
||||||
|
stm32_i2c_traceclear(priv);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void stm32_i2c_tracedump(FAR struct stm32_i2c_priv_s *priv)
|
static void stm32_i2c_tracedump(FAR struct stm32_i2c_priv_s *priv)
|
||||||
{
|
{
|
||||||
|
struct stm32_trace_s *trace;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
/* Dump all of the buffered trace entries */
|
lib_rawprintf("Elapsed time: %d\n", clock_systimer() - priv->start_time);
|
||||||
|
|
||||||
for (i = 0; i <= priv->tndx; i++)
|
for (i = 0; i <= priv->tndx; i++)
|
||||||
{
|
{
|
||||||
lib_rawprintf("%2d. STATUS: %08x COUNT: %3d EVENT: %2d PARM: %08x\n", i,
|
trace = &priv->trace[i];
|
||||||
priv->trace[i].status, priv->trace[i].count,
|
lib_rawprintf("%2d. STATUS: %08x COUNT: %3d EVENT: %2d PARM: %08x TIME: %d\n",
|
||||||
priv->trace[i].event, priv->trace[i].parm);
|
i+1, trace->status, trace->count, trace->event, trace->parm,
|
||||||
|
trace->time - priv->start_time);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif /* CONFIG_I2C_TRACE */
|
#endif /* CONFIG_I2C_TRACE */
|
||||||
@ -989,7 +1026,7 @@ static int stm32_i2c_isr(struct stm32_i2c_priv_s *priv)
|
|||||||
|
|
||||||
if ((status & I2C_SR1_SB) != 0)
|
if ((status & I2C_SR1_SB) != 0)
|
||||||
{
|
{
|
||||||
stm32_i2c_traceevent(priv, I2CEVENT_SB, priv->msgc);
|
stm32_i2c_traceevent(priv, I2CEVENT_SENDADDR, priv->msgc);
|
||||||
|
|
||||||
/* Get run-time data */
|
/* Get run-time data */
|
||||||
|
|
||||||
@ -1020,7 +1057,7 @@ static int stm32_i2c_isr(struct stm32_i2c_priv_s *priv)
|
|||||||
|
|
||||||
else if ((status & I2C_SR1_ADD10) != 0)
|
else if ((status & I2C_SR1_ADD10) != 0)
|
||||||
{
|
{
|
||||||
/* \todo Finish 10-bit mode addressing */
|
/* TODO: Finish 10-bit mode addressing */
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Was address sent, continue with either sending or reading data */
|
/* Was address sent, continue with either sending or reading data */
|
||||||
@ -1029,11 +1066,9 @@ static int stm32_i2c_isr(struct stm32_i2c_priv_s *priv)
|
|||||||
{
|
{
|
||||||
if (priv->dcnt > 0)
|
if (priv->dcnt > 0)
|
||||||
{
|
{
|
||||||
stm32_i2c_traceevent(priv, I2CEVENT_READ, priv->dcnt);
|
|
||||||
|
|
||||||
/* Send a byte */
|
/* Send a byte */
|
||||||
|
|
||||||
stm32_i2c_traceevent(priv, I2CEVENT_SENDBYTE, *priv->ptr);
|
stm32_i2c_traceevent(priv, I2CEVENT_SENDBYTE, priv->dcnt);
|
||||||
stm32_i2c_putreg(priv, STM32_I2C_DR_OFFSET, *priv->ptr++);
|
stm32_i2c_putreg(priv, STM32_I2C_DR_OFFSET, *priv->ptr++);
|
||||||
priv->dcnt--;
|
priv->dcnt--;
|
||||||
}
|
}
|
||||||
@ -1057,7 +1092,7 @@ static int stm32_i2c_isr(struct stm32_i2c_priv_s *priv)
|
|||||||
|
|
||||||
if (priv->dcnt > 0)
|
if (priv->dcnt > 0)
|
||||||
{
|
{
|
||||||
stm32_i2c_traceevent(priv, I2CEVENT_RXNE, priv->dcnt);
|
stm32_i2c_traceevent(priv, I2CEVENT_RCVBYTE, priv->dcnt);
|
||||||
|
|
||||||
/* Receive a byte */
|
/* Receive a byte */
|
||||||
|
|
||||||
@ -1090,9 +1125,15 @@ static int stm32_i2c_isr(struct stm32_i2c_priv_s *priv)
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* Was last byte received or sent? */
|
/* Was last byte received or sent? Hmmm... the F4 seems to differ from
|
||||||
|
* the F1 in that BTF is not set after data is received (only RXNE).
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef CONFIG_STM32_STM32F40XX
|
||||||
|
if (priv->dcnt <= 0 && (status & (I2C_SR1_BTF|I2C_SR1_RXNE)) != 0)
|
||||||
|
#else
|
||||||
if (priv->dcnt <= 0 && (status & I2C_SR1_BTF) != 0)
|
if (priv->dcnt <= 0 && (status & I2C_SR1_BTF) != 0)
|
||||||
|
#endif
|
||||||
{
|
{
|
||||||
stm32_i2c_getreg(priv, STM32_I2C_DR_OFFSET); /* ACK ISR */
|
stm32_i2c_getreg(priv, STM32_I2C_DR_OFFSET); /* ACK ISR */
|
||||||
|
|
||||||
@ -1106,9 +1147,9 @@ static int stm32_i2c_isr(struct stm32_i2c_priv_s *priv)
|
|||||||
|
|
||||||
if (priv->msgc > 0)
|
if (priv->msgc > 0)
|
||||||
{
|
{
|
||||||
stm32_i2c_traceevent(priv, I2CEVENT_BTFSTART, priv->msgc);
|
|
||||||
if (priv->msgv->flags & I2C_M_NORESTART)
|
if (priv->msgv->flags & I2C_M_NORESTART)
|
||||||
{
|
{
|
||||||
|
stm32_i2c_traceevent(priv, I2CEVENT_BTFNOSTART, priv->msgc);
|
||||||
priv->ptr = priv->msgv->buffer;
|
priv->ptr = priv->msgv->buffer;
|
||||||
priv->dcnt = priv->msgv->length;
|
priv->dcnt = priv->msgv->length;
|
||||||
priv->flags = priv->msgv->flags;
|
priv->flags = priv->msgv->flags;
|
||||||
@ -1123,6 +1164,7 @@ static int stm32_i2c_isr(struct stm32_i2c_priv_s *priv)
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
stm32_i2c_traceevent(priv, I2CEVENT_BTFRESTART, priv->msgc);
|
||||||
stm32_i2c_sendstart(priv);
|
stm32_i2c_sendstart(priv);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user