Использую плату STM32L Discovery с STM32L152RCT6, среда программирования Keil uVision 5, набор библиотек StdPeriph последней версии.
Успешно запустил GPIO и несколько USART (то есть контроллер исправен), а вот с I2C происходит нечто странное, уже 2 дня убил и решения не нашел.
Использую I2C1 ножки PB8 и PB9, на шине акселерометр MMA7660 и резисторы подтяжки 10К, контроллер тактуется от внутреннего MSI на 2 МГЦ.
Мой недописанный код:
Код
#include "i2c_routines.h"
#include "STM32L1xx.h"
void i2c1_init(void)
{
GPIO_InitTypeDef PORT;
I2C_InitTypeDef I2C_INIT;
RCC_AHBPeriphClockCmd(RCC_AHBENR_GPIOBEN, ENABLE);
RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C1, ENABLE);
I2C_INIT.I2C_Ack = I2C_Ack_Disable;
I2C_INIT.I2C_AcknowledgedAddress = I2C_AcknowledgedAddress_7bit;
I2C_INIT.I2C_ClockSpeed = 10000;
I2C_INIT.I2C_DutyCycle = I2C_DutyCycle_2;
I2C_INIT.I2C_Mode = I2C_Mode_I2C;
I2C_INIT.I2C_OwnAddress1 = 0x11;
I2C_DeInit(I2C1);
PORT.GPIO_Mode = GPIO_Mode_AF;
PORT.GPIO_OType = GPIO_OType_OD;
PORT.GPIO_PuPd = GPIO_PuPd_UP;
PORT.GPIO_Speed = GPIO_Speed_40MHz;
PORT.GPIO_Pin = GPIO_Pin_8 | GPIO_Pin_9;
GPIO_Init(GPIOB, &PORT);
GPIO_PinAFConfig(GPIOB, GPIO_PinSource8, GPIO_AF_I2C1);
GPIO_PinAFConfig(GPIOB, GPIO_PinSource9, GPIO_AF_I2C1);
I2C_Init(I2C1, &I2C_INIT);
I2C_Cmd(I2C1, ENABLE);
}
void I2C1_single_write(uint8_t HW_address, uint8_t addr, uint8_t data)
{
I2C_GenerateSTART(I2C1, ENABLE);
while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT));
I2C_Send7bitAddress(I2C1, HW_address, I2C_Direction_Transmitter);
while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED));
}
#include "STM32L1xx.h"
void i2c1_init(void)
{
GPIO_InitTypeDef PORT;
I2C_InitTypeDef I2C_INIT;
RCC_AHBPeriphClockCmd(RCC_AHBENR_GPIOBEN, ENABLE);
RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C1, ENABLE);
I2C_INIT.I2C_Ack = I2C_Ack_Disable;
I2C_INIT.I2C_AcknowledgedAddress = I2C_AcknowledgedAddress_7bit;
I2C_INIT.I2C_ClockSpeed = 10000;
I2C_INIT.I2C_DutyCycle = I2C_DutyCycle_2;
I2C_INIT.I2C_Mode = I2C_Mode_I2C;
I2C_INIT.I2C_OwnAddress1 = 0x11;
I2C_DeInit(I2C1);
PORT.GPIO_Mode = GPIO_Mode_AF;
PORT.GPIO_OType = GPIO_OType_OD;
PORT.GPIO_PuPd = GPIO_PuPd_UP;
PORT.GPIO_Speed = GPIO_Speed_40MHz;
PORT.GPIO_Pin = GPIO_Pin_8 | GPIO_Pin_9;
GPIO_Init(GPIOB, &PORT);
GPIO_PinAFConfig(GPIOB, GPIO_PinSource8, GPIO_AF_I2C1);
GPIO_PinAFConfig(GPIOB, GPIO_PinSource9, GPIO_AF_I2C1);
I2C_Init(I2C1, &I2C_INIT);
I2C_Cmd(I2C1, ENABLE);
}
void I2C1_single_write(uint8_t HW_address, uint8_t addr, uint8_t data)
{
I2C_GenerateSTART(I2C1, ENABLE);
while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT));
I2C_Send7bitAddress(I2C1, HW_address, I2C_Direction_Transmitter);
while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED));
}
Проблемы начинаются уже на этом этапе. Инициализация проходит успешно, но после вызова I2C_GenerateSTART(I2C1, ENABLE); контроллер кладет линию SCL (PB8) на землю и не отпускает её. При этом статусный бит I2C1.SR2 BUSY установлен в 1, т.е. шина занята. То же самое происходит если после инициализации I2C при помощи отладчика записать 1 в бит START регистра I2C1.CR1, а если записать 1 в бит STOP, то контроллер отпускает линию SCL. Пробовал убрать с шины акселерометр и снять установленный на плату LCD - безрезультатно.
С помощью Google находил похожие проблемы, однако там всё решалось исправлением опечаток или же ошибок подключения.
Прошу помощи в решении данной проблемы.