基于STM32F103 LL库的I2C通信原理与实现

引言

STM32F103系列微控制器是STMicroelectronics推出的一款高性能、低功耗的32位ARM Cortex-M3处理器。本文将介绍如何基于STM32F103的LL库(Low-Level库)来进行I2C通信的操作,通过配置寄存器实现I2C主设备的数据读写。

I2C基础原理

I2C(Inter-Integrated Circuit)是一种串行通信协议,用于连接微控制器与外围设备,例如传感器、存储器等。I2C通信包括两根线路:时钟线(SCL)和数据线(SDA)。通信的设备分为主设备(Master)和从设备(Slave),主设备负责发起通信,而从设备负责响应。I2C通信按字节传输,每个字节由8个位组成,其中7个位用于数据,最后一个位是应答位(ACK)。

STM32F103 LL库的I2C配置

首先,我们需要在STM32CubeMX中进行引脚配置和LL库的初始化。

  1. 打开STM32CubeMX,选择相应的STM32F103型号。
  2. 在 “Pinout & Configuration” 选项卡中,配置I2C的SCL和SDA引脚。
  3. 在 “Configuration” 选项卡中,启用 “I2C1” 外设,选择 “I2C1 Mode” 为 “I2C”.
  4. 配置 I2C1 的时钟频率和模式。
  5. 生成代码并打开STM32CubeIDE。

STM32F103 LL库的I2C编程

以下是一个简单的例子,演示如何在STM32F103上使用LL库实现I2C通信。

#include "stm32f1xx_ll_bus.h"
#include "stm32f1xx_ll_gpio.h"
#include "stm32f1xx_ll_i2c.h"
#include "stm32f1xx_ll_rcc.h"
#include "stm32f1xx_ll_usart.h"

#define I2C_SCL_PIN LL_GPIO_PIN_6
#define I2C_SDA_PIN LL_GPIO_PIN_7

I2C_HandleTypeDef hi2c1;

void I2C_Init() {
    LL_GPIO_InitTypeDef GPIO_InitStruct = {0};

    // 使能I2C1和GPIOB时钟
    LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_I2C1);
    LL_APB2_GRP1_EnableClock(LL_APB2_GRP1_PERIPH_GPIOB);

    // 配置SCL和SDA引脚
    GPIO_InitStruct.Pin = I2C_SCL_PIN | I2C_SDA_PIN;
    GPIO_InitStruct.Mode = LL_GPIO_MODE_ALTERNATE;
    GPIO_InitStruct.Speed = LL_GPIO_SPEED_FREQ_HIGH;
    GPIO_InitStruct.OutputType = LL_GPIO_OUTPUT_OPENDRAIN;
    LL_GPIO_Init(GPIOB, &GPIO_InitStruct);

    // 配置I2C1外设
    hi2c1.Instance = I2C1;
    hi2c1.Init.ClockSpeed = 100000;
    hi2c1.Init.DutyCycle = I2C_DUTYCYCLE_2;
    hi2c1.Init.OwnAddress1 = 0;
    hi2c1.Init.AddressingMode = I2C_ADDRESSINGMODE_7BIT;
    hi2c1.Init.DualAddressMode = I2C_DUALADDRESS_DISABLE;
    hi2c1.Init.OwnAddress2 = 0;
    hi2c1.Init.GeneralCallMode = I2C_GENERALCALL_DISABLE;
    hi2c1.Init.NoStretchMode = I2C_NOSTRETCH_DISABLE;
    if (LL_I2C_Init(&hi2c1) != HAL_OK) {
        // 处理初始化失败的情况
    }
}

void I2C_WriteData(uint8_t devAddr, uint8_t regAddr, uint8_t data) {
    while (LL_I2C_IsActiveFlag_BUSY(I2C1))
        ;

    // 发送起始信号
    LL_I2C_GenerateStartCondition(I2C1);
    while (!LL_I2C_IsActiveFlag_SB(I2C1))
        ;

    // 发送设备地址和写标志
    LL_I2C_TransmitData8(I2C1, devAddr << 1);
    while (!LL_I2C_IsActiveFlag_ADDR(I2C1))
        ;
    LL_I2C_ClearFlag_ADDR(I2C1);

    // 发送寄存器地址
    LL_I2C_TransmitData8(I2C1, regAddr);
    while (!LL_I2C_IsActiveFlag_TXE(I2C1))
        ;

    // 发送数据
    LL_I2C_TransmitData8(I2C1, data);
    while (!LL_I2C_IsActiveFlag_TXE(I2C1))
        ;

    // 发送停止信号
    LL_I2C_GenerateStopCondition(I2C1);
}

uint8_t I2C_ReadData(uint8_t devAddr, uint8_t regAddr) {
    uint8_t data;

    while (LL_I2C_IsActiveFlag_BUSY(I2C1))
        ;

    // 发送起始信号
    LL_I2C_GenerateStartCondition(I2C1);
    while (!LL_I2C_IsActiveFlag_SB(I2C1))
        ;

    // 发送设备地址和写标志
    LL_I2C_TransmitData8(I2C1, devAddr << 1);
    while (!LL_I2C_IsActiveFlag_ADDR(I2C1))
        ;
    LL_I2C_ClearFlag_ADDR(I2C1);

    // 发送寄存器地址
    LL_I2C_TransmitData8(I2C1, regAddr);
    while (!LL_I2C_IsActiveFlag_TXE(I2C1))
        ;

    // 重新启动
    LL_I2C_GenerateStartCondition(I2C1);
    while (!LL_I2C_IsActiveFlag_SB(I2C1))
        ;

    // 发送设备地址和读标志
    LL_I2C_TransmitData8(I2C1, (devAddr << 1) | 1);
    while (!LL_I2C_IsActiveFlag_ADDR(I2C1))
        ;
    LL_I2C_ClearFlag_ADDR(I2C1);

    // 开启接收模式
    LL_I2C_AcknowledgeNextData(I2C1, LL_I2C_ACK);

    // 发送停止信号
    LL_I2C_GenerateStopCondition(I2C1);

    // 等待数据接收
    while (!LL_I2C_IsActiveFlag_RXNE(I2C1))
        ;
    data = LL_I2C_ReceiveData8(I2C1);

    return data;
}

int main(void) {
    I2C_Init();

    uint8_t deviceAddress = 0x50;
    uint8_t registerAddress = 0x00;
    uint8_t dataToWrite = 0xAA;

    // 写入数据
    I2C_WriteData(deviceAddress, registerAddress, dataToWrite);

    // 读取数据
    uint8_t readData = I2C_ReadData(deviceAddress, registerAddress);

    // 此处可进行处理读取到的数据

    while (1) {
    }
}

代码解释

  1. I2C_Init(): 初始化I2C外设和引脚。
  2. I2C_WriteData(uint8_t devAddr, uint8_t regAddr, uint8_t data): 向指定设备的指定寄存器写入数据。
  3. I2C_ReadData(uint8_t devAddr, uint8_t regAddr): 从指定设备的指定寄存器读取数据。

结论

通过以上配置和代码,我们演示了如何基于STM32F103的LL库实现I2C通信。LL库提供了对底层硬件的直接控制,使得我们可以更加灵活地进行配置和操作。在实际应用中,可以根据需要扩展和改进这个基础框架,以适应更复杂的I2C通信场景。

  • 9
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
STM32是由意法半导体(STMicroelectronics)公司推出的基于ARM Cortex-M内核的高性能、低功耗、高性价比32位微控制器系列。自其面世以来,STM32凭借其广泛的适用性和卓越的特性,已成为嵌入式系统设计领域的主流选择之一,广泛应用于工业控制、消费电子、物联网、汽车电子、医疗设备、智能家居等多个领域。 内核与架构 STM32产品线采用了不同版本的ARM Cortex-M内核,包括M0、M0+、M3、M4、M7等,分别对应不同级别的性能需求。这些内核提供单周期乘法、硬件除法、DSP指令集、浮点单元(FPU)等功能,以满足不同应用场景中的计算密集型任务需求。处理器架构遵循哈佛结构,具有独立的指令总线和数据总线,确保高效的代码执行和数据访问。 丰富的外设与接口 STM32微控制器集成了丰富的外设资源,以适应各种复杂系统设计。这些外设包括但不限于: 通信接口:如USART、UART、SPI、I2C、CAN、USB(全速/高速)、Ethernet、无线连接模块(如BLE、Wi-Fi)等,用于实现设备间的串行通信和网络连接。 定时器:多种通用定时器、高级定时器、基本定时器以及PWM输出,支持定时、计数、脉冲捕获、电机控制等多种功能。 模拟外设:高精度ADC(模数转换器)、DAC(数模转换器)、比较器、温度传感器等,用于采集和处理模拟信号。 存储器:内置Flash和SRAM,容量从几KB到几MB不等,满足不同应用的数据存储和运行空间需求。部分型号还支持外部存储器接口(如FSMC、Octo-SPI)以扩展存储能力。 安全与保护机制:如加密加速器、安全单元、内存保护单元(MPU)、看门狗定时器、时钟安全系统(CSS)等,保障系统安全稳定运行。 开发环境与生态系统 STM32拥有强大的软件支持和生态系统,简化开发流程并加速产品上市时间: 开发工具:官方提供STM32CubeMX初始化配置工具,帮助开发者快速进行项目设置、外设配置及代码生成。此外,还有STM32CubeIDE集成开发环境,集成了编译器、调试器和仿真器支持。 软件STM32Cube软件包包含HAL(硬件抽象层)LL(低层),前者提供跨平台、跨系列的统一API接口,后者直接面向寄存器提供高效访问。同时,还提供各类外设驱动、中间件组件(如FreeRTOS、FatFS、LwIP等)以及特定应用框架(如STM32Cube.AI for AI推理)。 社区与资源:ST官方社区、论坛、博客、技术文档、培训材料、应用笔记、用户案例等资源丰富,为开发者提供全方位的技术支持和交流平台。 产品线与封装 STM32产品线按性能、功耗、外设组合等特性划分为多个子系列,如STM32F、STM32L、STM32G、STM32H等,每个子系列下又包含多种型号,以适应不同成本、性能、尺寸和功耗要求。封装形式多样,从小型QFN、LQFP到大型BGA,满足不同应用场景的封装密度和散热需求。 综上所述,STM32微控制器以其强大的内核性能、丰富的外设集成、完善的开发支持和广泛的市场应用,为嵌入式系统设计提供了高度灵活且极具竞争力的解决方案。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值