STM32CubeMX

年末年始休暇を利用して、STM32F103C8でI2CデバイスのMCP23017をPB6,PB7(I2C1 SCL,SDA)に接続した回路をブレッドボードに実装してみた。
そして、STM32CubeMX(CubeF1)にて初期化部分のコードを生成したところバグがあることが分かった。
開発環境の各バージョンは次の通りである。
・STM32FCubeMX Ver5.0.0
・STM32CubeF1 Ver1.7.0
・IAR社 EWARM 7.50.3

STM32CubeMXで自動生成された下記コードでは、I2C SCLにクロックが出力されない事が分かった。

void HAL_I2C_MspInit(I2C_HandleTypeDef* hi2c)
{

  GPIO_InitTypeDef GPIO_InitStruct = {0};
  if(hi2c->Instance==I2C1)
  {
  /* USER CODE BEGIN I2C1_MspInit 0 */

  /* USER CODE END I2C1_MspInit 0 */

    __HAL_RCC_GPIOB_CLK_ENABLE();
    /**I2C1 GPIO Configuration    
    PB6     ------> I2C1_SCL
    PB7     ------> I2C1_SDA 
    */
    GPIO_InitStruct.Pin = GPIO_PIN_6|GPIO_PIN_7;
    GPIO_InitStruct.Mode = GPIO_MODE_AF_OD;
    GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;
    HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);

    /* Peripheral clock enable */   ★★I2CペリフェラルへのクロックEnableは、
    __HAL_RCC_I2C1_CLK_ENABLE();    ★★HAL_GPIO_Init()より前に行わないとI2Cが正常に動作しない
  /* USER CODE BEGIN I2C1_MspInit 1 */

  /* USER CODE END I2C1_MspInit 1 */
  }

}

私は、GPIOBのクロックの直後に移動した。

__HAL_RCC_GPIOB_CLK_ENABLE();
__HAL_RCC_I2C1_CLK_ENABLE();

ちなみに、MCP23017のdevice addressは0x20であるが、I2C APIのパラメータに渡すときは、左にシフトして(0x40)渡すことが、API仕様に書いてあります。

#include "stm32f1xx.h"

/** i2cスレーブアドレス */
#define SLAVE_ADDR  0x40
/** i2cデバイスハンドル */
I2C_HandleTypeDef   MCP23017_handle;

void MCP23017_init(void)
{
    uint8_t rw_buff[2];

    /* Initialize for I2C */
    MCP23017_handle.Instance = I2C1;
    MCP23017_handle.Init.ClockSpeed = 100000;
    MCP23017_handle.Init.DutyCycle = I2C_DUTYCYCLE_2;
    MCP23017_handle.Init.OwnAddress1 = 0;
    MCP23017_handle.Init.AddressingMode = I2C_ADDRESSINGMODE_7BIT;
    MCP23017_handle.Init.DualAddressMode = I2C_DUALADDRESS_DISABLE;
    MCP23017_handle.Init.OwnAddress2 = 0;
    MCP23017_handle.Init.GeneralCallMode = I2C_GENERALCALL_DISABLE;
    MCP23017_handle.Init.NoStretchMode = I2C_NOSTRETCH_DISABLE;
    if( HAL_I2C_Init( &MCP23017_handle ) != HAL_OK )
    {
        Error_Handler();
    }

    /** I2Cデバイスからの読み出し(デバッグ用) */
    rw_buff[0] = 0x05;
    if( HAL_I2C_Master_Receive( &MCP23017_handle, SLAVE_ADDR, rw_buff, 1, HAL_MAX_DELAY ) != HAL_OK ){
//  if( HAL_I2C_Master_Transmit( &MCP23017_handle, SLAVE_ADDR, rw_buff, 1, HAL_MAX_DELAY ) != HAL_OK ){
//  if( HAL_I2C_Master_Transmit_IT( &MCP23017_handle, SLAVE_ADDR, rw_buff, 1 ) != HAL_OK ){
        Error_Handler();
    }
}