From d96149a9d1aa53368b54e64aedb91436ffc85bce Mon Sep 17 00:00:00 2001 From: patacongo <patacongo@42af7a65-404d-4744-a932-0658087f49c3> Date: Thu, 8 Sep 2011 17:56:08 +0000 Subject: [PATCH] Fix I2C/FSMC conflict for STM32; Fix STM32 clock setup git-svn-id: svn://svn.code.sf.net/p/nuttx/code/trunk@3942 42af7a65-404d-4744-a932-0658087f49c3 --- ChangeLog | 4 + arch/arm/src/stm32/chip/stm32_i2c.h | 4 +- arch/arm/src/stm32/chip/stm32_rcc.h | 2 +- arch/arm/src/stm32/stm32_i2c.c | 469 ++++++++++++++------- configs/stm3210e-eval/README.txt | 6 + configs/stm3210e-eval/RIDE/defconfig | 5 + configs/stm3210e-eval/buttons/defconfig | 5 + configs/stm3210e-eval/nsh/defconfig | 5 + configs/stm3210e-eval/nsh2/defconfig | 5 + configs/stm3210e-eval/nx/defconfig | 5 + configs/stm3210e-eval/nxlines/defconfig | 5 + configs/stm3210e-eval/nxtext/defconfig | 5 + configs/stm3210e-eval/ostest/defconfig | 5 + configs/stm3210e-eval/src/up_selectlcd.c | 2 +- configs/stm3210e-eval/usbserial/defconfig | 5 + configs/stm3210e-eval/usbstorage/defconfig | 5 + 16 files changed, 387 insertions(+), 150 deletions(-) diff --git a/ChangeLog b/ChangeLog index 97b90f6087..0e168cb6f6 100644 --- a/ChangeLog +++ b/ChangeLog @@ -2060,3 +2060,7 @@ Li Zhuoyi (Lzyy). * arch/arm/src/lpc17xx/lpc17_i2c.c: I2C driver for the NXP LPC17xx family submitted by Li Zhuoyi (Lzyy) + * arch/arm/src/stm32_i2c.c: Correct two issues with the STM32 I2C driver: + (1) Clocking needs to be based on PCLK1, not HCLK and fast speed settings + need some additional bits; and (2) Correct a hang that will occur on + I2C1 if FSMC is also enabled. diff --git a/arch/arm/src/stm32/chip/stm32_i2c.h b/arch/arm/src/stm32/chip/stm32_i2c.h index 8e40a3598d..2049ac1f46 100644 --- a/arch/arm/src/stm32/chip/stm32_i2c.h +++ b/arch/arm/src/stm32/chip/stm32_i2c.h @@ -82,7 +82,7 @@ /* Control register 1 */ -#define I2C_CR1_PE (1 << 0) /* Bit 0: Peripheral Enable*/ +#define I2C_CR1_PE (1 << 0) /* Bit 0: Peripheral Enable */ #define I2C_CR1_SMBUS (1 << 1) /* Bit 1: SMBus Mode */ #define I2C_CR1_SMBTYPE (1 << 3) /* Bit 3: SMBus Type */ #define I2C_CR1_ENARP (1 << 4) /* Bit 4: ARP Enable */ @@ -167,7 +167,7 @@ #define I2C_CCR_CCR_SHIFT (0) /* Bits 11-0: Clock Control Register in Fast/Standard mode (Master mode) */ #define I2C_CCR_CCR_MASK (0x0fff << I2C_CCR_CCR_SHIFT) #define I2C_CCR_DUTY (1 << 14) /* Bit 14: Fast Mode Duty Cycle */ -#define I2C_CCR_FS (1 << 15) /* Bit 15: I2C Master Mode Selection */ +#define I2C_CCR_FS (1 << 15) /* Bit 15: Fast Mode Selection */ /* TRISE Register */ diff --git a/arch/arm/src/stm32/chip/stm32_rcc.h b/arch/arm/src/stm32/chip/stm32_rcc.h index 0f0cc9c27b..1ed298d208 100644 --- a/arch/arm/src/stm32/chip/stm32_rcc.h +++ b/arch/arm/src/stm32/chip/stm32_rcc.h @@ -176,7 +176,7 @@ /* APB2 Peripheral reset register */ -#define RCC_APB2RSTR_AFIORST (1 << 0) /* Bit 0: Alternate Function I/O reset */ +#define RCC_APB2RSTR_AFIORST (1 << 0) /* Bit 0: Alternate Function I/O reset */ #define RCC_APB2RSTR_IOPARST (1 << 2) /* Bit 2: I/O port A reset */ #define RCC_APB2RSTR_IOPBRST (1 << 3) /* Bit 3: IO port B reset */ #define RCC_APB2RSTR_IOPCRST (1 << 4) /* Bit 4: IO port C reset */ diff --git a/arch/arm/src/stm32/stm32_i2c.c b/arch/arm/src/stm32/stm32_i2c.c index 09f61c05cf..fb553c7cd9 100644 --- a/arch/arm/src/stm32/stm32_i2c.c +++ b/arch/arm/src/stm32/stm32_i2c.c @@ -94,8 +94,9 @@ * Pre-processor Definitions ************************************************************************************/ /* Configuration ********************************************************************/ -/* Interrupt wait timeout in milliseconds */ +/* Interrupt wait timeout in seconds and milliseconds */ +#undef CONFIG_STM32_I2CTIMEOSEC #ifndef CONFIG_STM32_I2CTIMEOMS # define CONFIG_STM32_I2CTIMEOMS 50 #endif @@ -104,61 +105,65 @@ * Private Types ************************************************************************************/ -/** I2C Device Private Data - */ -struct stm32_i2c_priv_s { - uint32_t base; - int refs; - sem_t sem_excl; - sem_t sem_isr; +/* I2C Device Private Data */ + +struct stm32_i2c_priv_s +{ + uint32_t base; + int refs; + sem_t sem_excl; + sem_t sem_isr; - uint8_t msgc; - struct i2c_msg_s *msgv; - uint8_t * ptr; - int dcnt; - uint16_t flags; + uint8_t msgc; + struct i2c_msg_s *msgv; + uint8_t *ptr; + int dcnt; + uint16_t flags; - uint32_t status; + uint32_t status; }; -/** I2C Device, Instance - */ -struct stm32_i2c_inst_s { - struct i2c_ops_s * ops; - struct stm32_i2c_priv_s * priv; +/* I2C Device, Instance */ + +struct stm32_i2c_inst_s +{ + struct i2c_ops_s *ops; + struct stm32_i2c_priv_s *priv; - uint32_t frequency; - int address; - uint16_t flags; + uint32_t frequency; + int address; + uint16_t flags; }; /************************************************************************************ * Private Data ************************************************************************************/ -#if CONFIG_STM32_I2C1 -struct stm32_i2c_priv_s stm32_i2c1_priv = { - .base = STM32_I2C1_BASE, - .refs = 0, - .msgc = 0, - .msgv = NULL, - .ptr = NULL, - .dcnt = 0, - .flags = 0, - .status = 0 +#ifdef CONFIG_STM32_I2C1 +struct stm32_i2c_priv_s stm32_i2c1_priv = +{ + .base = STM32_I2C1_BASE, + .refs = 0, + .msgc = 0, + .msgv = NULL, + .ptr = NULL, + .dcnt = 0, + .flags = 0, + .status = 0 }; #endif -#if CONFIG_STM32_I2C2 -struct stm32_i2c_priv_s stm32_i2c2_priv = { - .base = STM32_I2C2_BASE, - .refs = 0, - .msgc = 0, - .msgv = NULL, - .ptr = NULL, - .dcnt = 0, - .flags = 0, - .status = 0 +#ifdef CONFIG_STM32_I2C2 +struct stm32_i2c_priv_s stm32_i2c2_priv = +{ + .base = STM32_I2C2_BASE, + .refs = 0, + .msgc = 0, + .msgv = NULL, + .ptr = NULL, + .dcnt = 0, + .flags = 0, + .status = 0 }; #endif @@ -200,13 +205,24 @@ int inline stm32_i2c_sem_waitisr(FAR struct i2c_dev_s *dev) flags = irqsave(); do { + /* Get the current time */ + (void)clock_gettime(CLOCK_REALTIME, &abstime); + + /* Calculate a time in the future */ + +#if defined(CONFIG_STM32_I2CTIMEOSEC) && CONFIG_STM32_I2CTIMEOSEC > 0 + abstime.tv_sec += CONFIG_STM32_I2CTIMEOSEC; +#endif +#if defined(CONFIG_STM32_I2CTIMEOMS) && CONFIG_STM32_I2CTIMEOMS > 0 abstime.tv_nsec += CONFIG_STM32_I2CTIMEOMS * 1000 * 1000; if (abstime.tv_nsec > 1000 * 1000 * 1000) { abstime.tv_sec++; abstime.tv_nsec -= 1000 * 1000 * 1000; } +#endif + /* Wait until either the transfer is complete or the timeout expires */ ret = sem_timedwait(&((struct stm32_i2c_inst_s *)dev)->priv->sem_isr, &abstime); } @@ -235,58 +251,166 @@ void inline stm32_i2c_sem_destroy(FAR struct i2c_dev_s *dev) static void stm32_i2c_setclock(FAR struct stm32_i2c_priv_s *priv, uint32_t frequency) { - /* Disable Peripheral if rising time is to be changed, - * and restore state on return. */ + uint16_t cr1; + uint16_t ccr; + uint16_t trise; + uint16_t freqmhz; + uint16_t speed; - uint16_t cr1 = stm32_i2c_getreg(priv, STM32_I2C_CR1_OFFSET); - - if (cr1 & I2C_CR1_PE) - stm32_i2c_putreg(priv, STM32_I2C_CR1_OFFSET, cr1 ^ I2C_CR1_PE); - - /* Update timing and control registers */ - - if (frequency < 400000) { - - /* Speed: 100 kHz - * Risetime: 1000 ns - * Duty: t_low / t_high = 1 - */ - stm32_i2c_putreg(priv, STM32_I2C_CCR_OFFSET, STM32_BOARD_HCLK/200000); - stm32_i2c_putreg(priv, STM32_I2C_TRISE_OFFSET, STM32_BOARD_HCLK/1000000 + 1); + /* Disable the selected I2C peripheral to configure TRISE */ + + cr1 = stm32_i2c_getreg(priv, STM32_I2C_CR1_OFFSET); + stm32_i2c_putreg(priv, STM32_I2C_CR1_OFFSET, cr1 & ~I2C_CR1_PE); + + /* Update timing and control registers */ + + freqmhz = (uint16_t)(STM32_PCLK1_FREQUENCY / 1000000); + ccr = 0; + + /* Configure speed in standard mode */ + + if (frequency <= 100000) + { + /* Standard mode speed calculation */ + + speed = (uint16_t)(STM32_PCLK1_FREQUENCY / (frequency << 1)); + + /* The CCR fault must be >= 4 */ + + if (speed < 4) + { + /* Set the minimum allowed value */ + + speed = 4; + } + ccr |= speed; + + /* Set Maximum Rise Time for standard mode */ + + trise = freqmhz + 1; } - else { - - /* Speed: 400 kHz - * Duty: t_low / t_high = 2 - */ - stm32_i2c_putreg(priv, STM32_I2C_CCR_OFFSET, STM32_BOARD_HCLK/1200000); - stm32_i2c_putreg(priv, STM32_I2C_TRISE_OFFSET, 300*(STM32_BOARD_HCLK / 1000000)/1000 + 1); + + /* Configure speed in fast mode */ + + else /* (frequency <= 400000) */ + { + /* Fast mode speed calculation with Tlow/Thigh = 16/9 */ + +#ifdef CONFIG_I2C_DUTY16_9 + speed = (uint16_t)(STM32_PCLK1_FREQUENCY / (frequency * 25)); + + /* Set DUTY and fast speed bits */ + + ccr |= (I2C_CCR_DUTY|I2C_CCR_FS); +#else + /* Fast mode speed calculation with Tlow/Thigh = 2 */ + + speed = (uint16_t)(STM32_PCLK1_FREQUENCY / (frequency * 3)); + + /* Set fast speed bit */ + + ccr |= I2C_CCR_FS; +#endif + + /* Verify that the CCR speed value is nonzero */ + + if (speed < 1) + { + /* Set the minimum allowed value */ + + speed = 1; + } + ccr |= speed; + + /* Set Maximum Rise Time for fast mode */ + + trise = (uint16_t)(((freqmhz * 300) / 1000) + 1); } - - /* Restore state */ - - if (cr1 & I2C_CR1_PE) - stm32_i2c_putreg(priv, STM32_I2C_CR1_OFFSET, cr1); + + /* Write the new values of the CCR and TRISE registers */ + + stm32_i2c_putreg(priv, STM32_I2C_CCR_OFFSET, ccr); + stm32_i2c_putreg(priv, STM32_I2C_TRISE_OFFSET, trise); + + /* Bit 14 of OAR1 must be configured and kept at 1 */ + + stm32_i2c_putreg(priv, STM32_I2C_OAR1_OFFSET, I2C_OAR1_ONE); + + /* Re-enable the peripheral (or not) */ + + stm32_i2c_putreg(priv, STM32_I2C_CR1_OFFSET, cr1); } static inline void stm32_i2c_sendstart(FAR struct stm32_i2c_priv_s *priv) { - /* Disable ACK on receive by default and generate START */ - stm32_i2c_modifyreg(priv, STM32_I2C_CR1_OFFSET, I2C_CR1_ACK, I2C_CR1_START); + /* Disable ACK on receive by default and generate START */ + + stm32_i2c_modifyreg(priv, STM32_I2C_CR1_OFFSET, I2C_CR1_ACK, I2C_CR1_START); +} + +static inline void stm32_i2c_clrstart(FAR struct stm32_i2c_priv_s *priv) +{ + /* "This [START] bit is set and cleared by software and cleared by hardware + * when start is sent or PE=0." The bit must be cleared by software if the + * START is never sent. + */ + + stm32_i2c_modifyreg(priv, STM32_I2C_CR1_OFFSET, I2C_CR1_START, 0); } static inline void stm32_i2c_sendstop(FAR struct stm32_i2c_priv_s *priv) { - stm32_i2c_modifyreg(priv, STM32_I2C_CR1_OFFSET, I2C_CR1_ACK, I2C_CR1_STOP); + stm32_i2c_modifyreg(priv, STM32_I2C_CR1_OFFSET, I2C_CR1_ACK, I2C_CR1_STOP); } static inline uint32_t stm32_i2c_getstatus(FAR struct stm32_i2c_priv_s *priv) { - uint32_t status = stm32_i2c_getreg(priv, STM32_I2C_SR1_OFFSET); - status |= (stm32_i2c_getreg(priv, STM32_I2C_SR2_OFFSET) << 16); - return status; + uint32_t status = stm32_i2c_getreg(priv, STM32_I2C_SR1_OFFSET); + status |= (stm32_i2c_getreg(priv, STM32_I2C_SR2_OFFSET) << 16); + return status; } +/* FSMC must be disable while accessing I2C1 because it uses a common resource (LBAR) */ + +#if defined(CONFIG_STM32_FSMC) && defined (CONFIG_STM32_I2C1) +static inline uint32_t stm32_i2c_disablefsmc(FAR struct stm32_i2c_priv_s *priv) +{ + uint32_t ret = 0; + uint32_t regval; + + /* Is this I2C1 */ + +#ifdef CONFIG_STM32_I2C2 + if (priv->base == STM32_I2C1_BASE) +#endif + { + /* Disable FSMC unconditionally */ + + ret = getreg32( STM32_RCC_AHBENR); + regval = ret & ~RCC_AHBENR_FSMCEN; + putreg32(regval, STM32_RCC_AHBENR); + } + return regval; +} + +static inline void stm32_i2c_enablefsmc(uint32_t ahbenr) +{ + uint32_t regval; + + /* Enable AHB clocking to the FSMC only if it was previously enabled. */ + + if ((ahbenr & RCC_AHBENR_FSMCEN) != 0) + { + regval = getreg32( STM32_RCC_AHBENR); + regval |= RCC_AHBENR_FSMCEN; + putreg32(regval, STM32_RCC_AHBENR); + } +} +#else +# define stm32_i2c_disablefsmc() (0) +# define stm32_i2c_enablefsmc(ahbenr) +#endif + /************************************************************************************ * Interrupt Service Routines ************************************************************************************/ @@ -456,14 +580,14 @@ static int stm32_i2c_isr(struct stm32_i2c_priv_s * priv) /* Decode ***************************************************************************/ -#if CONFIG_STM32_I2C1 +#ifdef CONFIG_STM32_I2C1 static int stm32_i2c1_isr(int irq, void *context) { return stm32_i2c_isr(&stm32_i2c1_priv); } #endif -#if CONFIG_STM32_I2C2 +#ifdef CONFIG_STM32_I2C2 static int stm32_i2c2_isr(int irq, void *context) { return stm32_i2c_isr(&stm32_i2c2_priv); @@ -481,16 +605,16 @@ static int stm32_i2c_init(FAR struct stm32_i2c_priv_s *priv) switch( priv->base ) { -#if CONFIG_STM32_I2C1 +#ifdef CONFIG_STM32_I2C1 case STM32_I2C1_BASE: /* enable power and reset the peripheral */ - - modifyreg32(STM32_RCC_APB1ENR, 0, RCC_APB1ENR_I2C1EN); + + modifyreg32(STM32_RCC_APB1ENR, 0, RCC_APB1ENR_I2C1EN); modifyreg32(STM32_RCC_APB1RSTR, 0, RCC_APB1RSTR_I2C1RST); modifyreg32(STM32_RCC_APB1RSTR, RCC_APB1RSTR_I2C1RST, 0); - + /* configure pins */ if (stm32_configgpio(GPIO_I2C1_SCL)==ERROR) return ERROR; @@ -498,7 +622,7 @@ static int stm32_i2c_init(FAR struct stm32_i2c_priv_s *priv) stm32_unconfiggpio(GPIO_I2C1_SCL); return ERROR; } - + /* attach ISRs */ irq_attach(STM32_IRQ_I2C1EV, stm32_i2c1_isr); @@ -508,7 +632,7 @@ static int stm32_i2c_init(FAR struct stm32_i2c_priv_s *priv) break; #endif -#if CONFIG_STM32_I2C2 +#ifdef CONFIG_STM32_I2C2 case STM32_I2C2_BASE: /* enable power and reset the peripheral */ @@ -546,7 +670,7 @@ static int stm32_i2c_init(FAR struct stm32_i2c_priv_s *priv) #ifndef NON_ISR I2C_CR2_ITERREN | I2C_CR2_ITEVFEN | // I2C_CR2_ITBUFEN | #endif - (STM32_BOARD_HCLK / 1000000) + (STM32_PCLK1_FREQUENCY / 1000000) ); stm32_i2c_setclock(priv, 100000); @@ -567,7 +691,7 @@ static int stm32_i2c_deinit(FAR struct stm32_i2c_priv_s *priv) switch( priv->base ) { -#if CONFIG_STM32_I2C1 +#ifdef CONFIG_STM32_I2C1 case STM32_I2C1_BASE: stm32_unconfiggpio(GPIO_I2C1_SCL); stm32_unconfiggpio(GPIO_I2C1_SDA); @@ -581,7 +705,7 @@ static int stm32_i2c_deinit(FAR struct stm32_i2c_priv_s *priv) break; #endif -#if CONFIG_STM32_I2C2 +#ifdef CONFIG_STM32_I2C2 case STM32_I2C2_BASE: stm32_unconfiggpio(GPIO_I2C2_SCL); stm32_unconfiggpio(GPIO_I2C2_SDA); @@ -609,7 +733,7 @@ uint32_t stm32_i2c_setfrequency(FAR struct i2c_dev_s *dev, uint32_t frequency) { stm32_i2c_sem_wait(dev); -#if STM32_BOARD_HCLK < 4000000 +#if STM32_PCLK1_FREQUENCY < 4000000 ((struct stm32_i2c_inst_s *)dev)->frequency = 100000; #else ((struct stm32_i2c_inst_s *)dev)->frequency = frequency; @@ -632,82 +756,135 @@ int stm32_i2c_setaddress(FAR struct i2c_dev_s *dev, int addr, int nbits) int stm32_i2c_process(FAR struct i2c_dev_s *dev, FAR struct i2c_msg_s *msgs, int count) { - struct stm32_i2c_inst_s *inst = (struct stm32_i2c_inst_s *)dev; - uint32_t status = 0; - int status_errno = 0; - - ASSERT(count); - - /* wait as stop might still be in progress - * - * \todo GET RID OF THIS PERFORMANCE LOSS and for() loop - */ - for (; stm32_i2c_getreg(inst->priv, STM32_I2C_CR1_OFFSET) & I2C_CR1_STOP; ) up_waste(); - - /* Old transfers are done */ - inst->priv->msgv = msgs; - inst->priv->msgc = count; - - /* Set clock (on change it toggles I2C_CR1_PE !) */ - stm32_i2c_setclock(inst->priv, inst->frequency); + struct stm32_i2c_inst_s *inst = (struct stm32_i2c_inst_s *)dev; + uint32_t status = 0; + uint32_t ahbenr; + int status_errno = 0; - /* Trigger start condition, then the process moves into the ISR */ - stm32_i2c_sendstart(inst->priv); + ASSERT(count); + + /* Disable FSMC that shares a pin with I2C1 (LBAR) */ + + ahbenr = stm32_i2c_disablefsmc(inst->priv); + + /* wait as stop might still be in progress + * + * \todo GET RID OF THIS PERFORMANCE LOSS and for() loop + */ + + for (; stm32_i2c_getreg(inst->priv, STM32_I2C_CR1_OFFSET) & I2C_CR1_STOP; ) + { + up_waste(); + } + /* Old transfers are done */ + + inst->priv->msgv = msgs; + inst->priv->msgc = count; + /* Set clock (on change it toggles I2C_CR1_PE !) */ + + stm32_i2c_setclock(inst->priv, inst->frequency); + + /* Trigger start condition, then the process moves into the ISR */ + + stm32_i2c_sendstart(inst->priv); + #ifdef NON_ISR - do { - do { + do + { + do + { stm32_i2c_isr(&stm32_i2c1_priv); status = inst->priv->status; - } while( status & (I2C_SR2_BUSY<<16) ); + } + while (status & (I2C_SR2_BUSY << 16)); } - while( sem_trywait( &((struct stm32_i2c_inst_s *)dev)->priv->sem_isr ) != 0 ); + while( sem_trywait( &((struct stm32_i2c_inst_s *)dev)->priv->sem_isr ) != 0 ); #else #if 1 - /* Wait for an ISR, if there was a timeout, fetch latest status to get the BUSY flag */ + /* Wait for an ISR, if there was a timeout, fetch latest status to get + * the BUSY flag. + */ - if (stm32_i2c_sem_waitisr(dev) == ERROR) { - status = stm32_i2c_getstatus(inst->priv); - status_errno = ETIMEDOUT; + if (stm32_i2c_sem_waitisr(dev) == ERROR) + { + status = stm32_i2c_getstatus(inst->priv); + status_errno = ETIMEDOUT; + + /* " Note: When the STOP, START or PEC bit is set, the software must + * not perform any write access to I2C_CR1 before this bit is + * cleared by hardware. Otherwise there is a risk of setting a + * second STOP, START or PEC request." + */ + + stm32_i2c_clrstart(inst->priv); + } + else + { + /* clear SR2 (BUSY flag) as we've done successfully */ + + status = inst->priv->status & 0xffff; } - else status = inst->priv->status & 0xFFFF; /* clear SR2 (BUSY flag) as we've done successfully */ #else - do { - printf("%x, %d\n", inst->priv->status, isr_count ); + do + { + printf("%x, %d\n", inst->priv->status, isr_count ); } - while( sem_trywait( &inst->priv->sem_isr ) != 0 ); + while( sem_trywait( &inst->priv->sem_isr ) != 0 ); #endif #endif - if (status & I2C_SR1_BERR) { /* Bus Error */ - status_errno = EIO; + if (status & I2C_SR1_BERR) + { + /* Bus Error */ + + status_errno = EIO; } - else if (status & I2C_SR1_ARLO) { /* Arbitration Lost (master mode) */ - status_errno = EAGAIN; + else if (status & I2C_SR1_ARLO) + { + /* Arbitration Lost (master mode) */ + + status_errno = EAGAIN; } - else if (status & I2C_SR1_AF) { /* Acknowledge Failure */ - status_errno = ENXIO; + else if (status & I2C_SR1_AF) + { + /* Acknowledge Failure */ + + status_errno = ENXIO; } - else if (status & I2C_SR1_OVR) { /* Overrun/Underrun */ - status_errno = EIO; + else if (status & I2C_SR1_OVR) + { + /* Overrun/Underrun */ + + status_errno = EIO; } - else if (status & I2C_SR1_PECERR) { /* PEC Error in reception */ - status_errno = EPROTO; + else if (status & I2C_SR1_PECERR) + { + /* PEC Error in reception */ + + status_errno = EPROTO; } - else if (status & I2C_SR1_TIMEOUT) {/* Timeout or Tlow Error */ - status_errno = ETIME; + else if (status & I2C_SR1_TIMEOUT) + { + /* Timeout or Tlow Error */ + + status_errno = ETIME; } - else if (status & (I2C_SR2_BUSY<<16) ) { /* I2C Bus is for some reason busy */ - status_errno = EBUSY; + else if (status & (I2C_SR2_BUSY << 16)) + { + /* I2C Bus is for some reason busy */ + + status_errno = EBUSY; } - -// printf("end_count = %d, dcnt=%d\n", isr_count, inst->priv->dcnt); fflush(stdout); - - stm32_i2c_sem_post(dev); - - errno = status_errno; - return -status_errno; + + /* Re-enable the FSMC */ + + stm32_i2c_enablefsmc(ahbenr); + stm32_i2c_sem_post(dev); + + errno = status_errno; + return -status_errno; } int stm32_i2c_write(FAR struct i2c_dev_s *dev, const uint8_t *buffer, int buflen) @@ -802,11 +979,11 @@ FAR struct i2c_dev_s * up_i2cinitialize(int port) struct stm32_i2c_inst_s * inst = NULL; /* device, single instance */ int irqs; -#if STM32_BOARD_HCLK < 4000000 +#if STM32_PCLK1_FREQUENCY < 4000000 # warning STM32_I2C_INIT: Peripheral clock must be at least 4 MHz to support 400 kHz operation. #endif -#if STM32_BOARD_HCLK < 2000000 +#if STM32_PCLK1_FREQUENCY < 2000000 # warning STM32_I2C_INIT: Peripheral clock must be at least 2 MHz to support 100 kHz operation. return NULL; #endif @@ -814,10 +991,10 @@ FAR struct i2c_dev_s * up_i2cinitialize(int port) /* Get I2C private structure */ switch(port) { -#if CONFIG_STM32_I2C1 +#ifdef CONFIG_STM32_I2C1 case 1: priv = (struct stm32_i2c_priv_s *)&stm32_i2c1_priv; break; #endif -#if CONFIG_STM32_I2C2 +#ifdef CONFIG_STM32_I2C2 case 2: priv = (struct stm32_i2c_priv_s *)&stm32_i2c2_priv; break; #endif default: return NULL; diff --git a/configs/stm3210e-eval/README.txt b/configs/stm3210e-eval/README.txt index 0224f65a29..772020e960 100755 --- a/configs/stm3210e-eval/README.txt +++ b/configs/stm3210e-eval/README.txt @@ -360,6 +360,12 @@ STM3210E-EVAL-specific Configuration Options CONFIG_STM32_USART1 CONFIG_STM32_ADC3 + Timer and I2C devices may need to the following to force power to be applied + unconditionally at power up. (Otherwise, the device is powered when it is + initialized). + + CONFIG_STM32_FORCEPOWER + Alternate pin mappings (should not be used with the STM3210E-EVAL board): CONFIG_STM32_TIM1_FULL_REMAP diff --git a/configs/stm3210e-eval/RIDE/defconfig b/configs/stm3210e-eval/RIDE/defconfig index 9e31d2bd67..4733a438dc 100755 --- a/configs/stm3210e-eval/RIDE/defconfig +++ b/configs/stm3210e-eval/RIDE/defconfig @@ -127,6 +127,11 @@ CONFIG_STM32_TIM8=n CONFIG_STM32_USART1=y CONFIG_STM32_ADC3=n +# +# Timer and I2C devices may need to the following to force power to be applied: +# +#CONFIG_STM32_FORCEPOWER=y + # # STM32F103Z specific serial device driver settings # diff --git a/configs/stm3210e-eval/buttons/defconfig b/configs/stm3210e-eval/buttons/defconfig index bc1c291d29..7a9e636d94 100644 --- a/configs/stm3210e-eval/buttons/defconfig +++ b/configs/stm3210e-eval/buttons/defconfig @@ -141,6 +141,11 @@ CONFIG_STM32_TIM8=n CONFIG_STM32_USART1=y CONFIG_STM32_ADC3=n +# +# Timer and I2C devices may need to the following to force power to be applied: +# +#CONFIG_STM32_FORCEPOWER=y + # # STM32F103Z specific serial device driver settings # diff --git a/configs/stm3210e-eval/nsh/defconfig b/configs/stm3210e-eval/nsh/defconfig index 908aff7f15..97b155ff8d 100755 --- a/configs/stm3210e-eval/nsh/defconfig +++ b/configs/stm3210e-eval/nsh/defconfig @@ -137,6 +137,11 @@ CONFIG_STM32_TIM8=n CONFIG_STM32_USART1=y CONFIG_STM32_ADC3=n +# +# Timer and I2C devices may need to the following to force power to be applied: +# +#CONFIG_STM32_FORCEPOWER=y + # # STM32F103Z specific serial device driver settings # diff --git a/configs/stm3210e-eval/nsh2/defconfig b/configs/stm3210e-eval/nsh2/defconfig index d496698868..d92fd37738 100644 --- a/configs/stm3210e-eval/nsh2/defconfig +++ b/configs/stm3210e-eval/nsh2/defconfig @@ -137,6 +137,11 @@ CONFIG_STM32_TIM8=n CONFIG_STM32_USART1=y CONFIG_STM32_ADC3=n +# +# Timer and I2C devices may need to the following to force power to be applied: +# +#CONFIG_STM32_FORCEPOWER=y + # # STM32F103Z specific serial device driver settings # diff --git a/configs/stm3210e-eval/nx/defconfig b/configs/stm3210e-eval/nx/defconfig index 894f569f5e..0824528814 100644 --- a/configs/stm3210e-eval/nx/defconfig +++ b/configs/stm3210e-eval/nx/defconfig @@ -137,6 +137,11 @@ CONFIG_STM32_TIM8=n CONFIG_STM32_USART1=y CONFIG_STM32_ADC3=n +# +# Timer and I2C devices may need to the following to force power to be applied: +# +#CONFIG_STM32_FORCEPOWER=y + # # STM32F103Z specific serial device driver settings # diff --git a/configs/stm3210e-eval/nxlines/defconfig b/configs/stm3210e-eval/nxlines/defconfig index 6a9ea3979a..d41e604152 100644 --- a/configs/stm3210e-eval/nxlines/defconfig +++ b/configs/stm3210e-eval/nxlines/defconfig @@ -137,6 +137,11 @@ CONFIG_STM32_TIM8=n CONFIG_STM32_USART1=y CONFIG_STM32_ADC3=n +# +# Timer and I2C devices may need to the following to force power to be applied: +# +#CONFIG_STM32_FORCEPOWER=y + # # STM32F103Z specific serial device driver settings # diff --git a/configs/stm3210e-eval/nxtext/defconfig b/configs/stm3210e-eval/nxtext/defconfig index d48637a1b6..9193dd084b 100644 --- a/configs/stm3210e-eval/nxtext/defconfig +++ b/configs/stm3210e-eval/nxtext/defconfig @@ -137,6 +137,11 @@ CONFIG_STM32_TIM8=n CONFIG_STM32_USART1=y CONFIG_STM32_ADC3=n +# +# Timer and I2C devices may need to the following to force power to be applied: +# +#CONFIG_STM32_FORCEPOWER=y + # # STM32F103Z specific serial device driver settings # diff --git a/configs/stm3210e-eval/ostest/defconfig b/configs/stm3210e-eval/ostest/defconfig index 3fec688713..41aeb41d8c 100755 --- a/configs/stm3210e-eval/ostest/defconfig +++ b/configs/stm3210e-eval/ostest/defconfig @@ -139,6 +139,11 @@ CONFIG_STM32_TIM8=n CONFIG_STM32_USART1=y CONFIG_STM32_ADC3=n +# +# Timer and I2C devices may need to the following to force power to be applied: +# +#CONFIG_STM32_FORCEPOWER=y + # # STM32F103Z specific serial device driver settings # diff --git a/configs/stm3210e-eval/src/up_selectlcd.c b/configs/stm3210e-eval/src/up_selectlcd.c index d2b189a09d..221834ff58 100644 --- a/configs/stm3210e-eval/src/up_selectlcd.c +++ b/configs/stm3210e-eval/src/up_selectlcd.c @@ -88,7 +88,7 @@ * *JP7 will switch to PD6 */ -/* GPIO configurations unique to SRAM */ +/* GPIO configurations unique to the LCD */ static const uint16_t g_lcdconfig[] = { diff --git a/configs/stm3210e-eval/usbserial/defconfig b/configs/stm3210e-eval/usbserial/defconfig index 5680fc74a1..362951cd37 100755 --- a/configs/stm3210e-eval/usbserial/defconfig +++ b/configs/stm3210e-eval/usbserial/defconfig @@ -139,6 +139,11 @@ CONFIG_STM32_TIM8=n CONFIG_STM32_USART1=y CONFIG_STM32_ADC3=n +# +# Timer and I2C devices may need to the following to force power to be applied: +# +#CONFIG_STM32_FORCEPOWER=y + # # STM32F103Z specific serial device driver settings # diff --git a/configs/stm3210e-eval/usbstorage/defconfig b/configs/stm3210e-eval/usbstorage/defconfig index da24167c74..5ad7be2ebc 100755 --- a/configs/stm3210e-eval/usbstorage/defconfig +++ b/configs/stm3210e-eval/usbstorage/defconfig @@ -137,6 +137,11 @@ CONFIG_STM32_TIM8=n CONFIG_STM32_USART1=y CONFIG_STM32_ADC3=n +# +# Timer and I2C devices may need to the following to force power to be applied: +# +#CONFIG_STM32_FORCEPOWER=y + # # STM32F103Z specific serial device driver settings #