单片机
返回首页

STM8L051F3 硬件I2C从机实例--新手导航

2020-02-02 来源:eefocus

这是我写的第一篇有关于技术的文章,可能写的不是很好,仅供参考。


先介绍一下背景,我是第一次接触STM的芯片,以前接触过都是基于51的芯片,算是有一点基础吧。因为公司的项目需要用到STM8L051F3的硬件I2C作为从机送数据,把自己从接触到调通遇到的问题记录一下,有需要的可以参考一下。


这篇文章是以相对新手的水平去写给新手参考的,所以我会尽量写的详细点。


1.首先我们需要查看数据手册,看下硬件I2C的接口是哪个端口,我用的是STM8L051F3这一块芯片。

通过手册可以知道,这款芯片的I2C端口为C0跟C1(一开始忘记去查手册,傻呼呼的用别的端口调试了很久。。。)

这款芯片的I2C应该是不能复用别的端口的,反正手册我没查到,只能用C0跟C1。


2.接下来开始初始化I2C需要的设置。(这里我只讲7位地址的。)

void I2C_Init(void)

{

  CLK_PeripheralClockConfig(CLK_Peripheral_I2C1, ENABLE); //初始化I2C时钟

  CLK->SWR=CLK_ICKCR_HSION;//时钟选择为HSI

  CLK->CKDIVR = 0;   //时钟不分频

  CLK->PCKENR1 = 0x08; // 使能I2C时钟

  


/* Init GPIO for I2C use */ //初始化端口C0和C1

GPIOC->CR1 |= 0x03;

GPIOC->DDR &= ~0x03;

GPIOC->CR2 &= ~0x03;


//初始化I2C寄存器

I2C1->CR1 |= 0x01;         // Enable I2C peripheral

I2C1->CR2 = 0x04;       // Enable I2C acknowledgement

I2C1->FREQR = 16;       // Set I2C Freq value (16MHz)

//下面这里要重点说明一下,STM8L051F3的硬件I2C作为从机是可以具备2个地址的。(用不到的话等下不要使能地址2即可)

I2C1->OARL = (0x44<< 1) ; // 地址1 = 0x44.第0位是10位地址的0位,7位地址要左移1位。

I2C1->OARH = 0x40;       // 此位需要置1,看手册。

I2C1->OAR2 = (0x46 << 1)|0x01; //这里是地址2 = 0x46的寄存器,第0位置1是使能2个地址,如果用不到直接屏蔽此语句即可。

I2C1->ITR = 0x07;       // all I2C interrupt enable  使能I2C中断


}


3.记得开启中断(enableInterrupts();),下面看中断内容。

中断里面我们需要2组数组来储存收发的数据,这个自己在主函数里定义2个全局变量数组,大小为你自己想需要收发的数据看有多少了。


#define MAX_Id 10

u8 Slave_Buffer_Tx[MAX_Id];

u8 Slave_Buffer_Rx[MAX_Id];


下面是一个广播地址的

INTERRUPT_HANDLER(I2C_IRQHandler,29)

{

unsigned char Add;

unsigned char Nuse;

static unsigned char RX_Cnt=0;


if(I2C1->SR2&0X0F) //I2C 出现错误

{

I2C1->SR2&=0xf0;

//I2C->CR2|=1<<7;

//I2C->CR2&=~(1<<7);

RX_Cnt=0;

I2C_Tx_Idx=0;

I2C1->SR2 |=0x02;

}


if(I2C1->SR1&0X02) //地址匹配

{

(void)(I2C1->SR3); //先读I2C_SR1,再读I2C_SR3,就可以清除ADDR

// I2C->DR = 0X00;

I2C_Tx_Idx=0;

I2C_Rx_Idx=0;

I2C1->DR = Slave_Buffer_Tx[I2C_Tx_Idx++];

}

if(I2C1->SR1&0X10) //停止条件

{

Nuse = I2C1->CR2; //清除停止位(SR1第4位),先读SR1,再写CR2

nop();

I2C1->CR2 = Nuse;

}

if(I2C1->SR1&0X04) //BTF位,先读SR1,再读或者写DR寄存器清除

{

Nuse=I2C1->DR;

// I2C->DR=Nuse;

}

if(I2C1->SR1 & 0x40) //RXNE,数据寄存器是否为空,0空1非

{

//将收到的数据储存到RX数组中。虽然我们是从机,用不到这里的数据,但是这里必须读DR寄存器收取数据.不然会出错

Slave_Buffer_Rx[I2C_Rx_Idx++]=I2C1->DR;

if(I2C_Rx_Idx>=MAX_Id)

{

I2C_Rx_Idx=0;

}

}

if(I2C1->SR1 & 0x80) //将发送的数据放入DR寄存器,清除TXE

{

I2C1->DR = Slave_Buffer_Tx[I2C_Tx_Idx++];

if(I2C_Tx_Idx>=MAX_Id)I2C_Tx_Idx=0;

}


}


下面是2个广播地址的

//这款芯片的中断文件在这里文件里面(stm8l15x_it.c),I2C的中断向量就是这个。

INTERRUPT_HANDLER(I2C_IRQHandler,29)

{

unsigned char Add;

unsigned char Nuse;

static unsigned char RX_Cnt=0;

B_I2C *base;


if(I2C1->SR2&0X0F) //I2C 出现错误中断的处理。没特别的中断需要处理的话,这里直接复制就好。

{

I2C1->SR2&=0xf0;

RX_Cnt=0;

I2C_Tx_Idx=0;

I2C1->SR2 |=0x02;

}


//第七位判断是从机地址1还是地址2,选择等下接收到的数据要放在哪个结构体里面(或者你们自己替换成变量也可以的,只是把收到的数据存起来而已,之后我们才可以去调用)

if(I2C1->SR3&0x80)

{

base = &B_I2C_p1;//add1

}

else

{

base = &B_I2C_p2_0x46;//add2

}


if(I2C1->SR1&0X02) //SR1的第一位会检测广播地址是否匹配

{

(void)(I2C1->SR3); //先读I2C_SR1,再读I2C_SR3,就可以清除ADDR

// I2C->DR = 0X00;

I2C_Tx_Idx=0;

I2C_Rx_Idx=0;

I2C1->DR = Slave_Buffer_Tx[I2C_Tx_Idx++];

}

if(I2C1->SR1&0X10) //停止条件

{

Nuse = I2C1->CR2; //清除停止位(SR1第4位),先读SR1,再写CR2

nop();

I2C1->CR2 = Nuse;

}

if(I2C1->SR1&0X04) //BTF位,先读SR1,再读或者写DR寄存器清除

{

Nuse=I2C1->DR;

// I2C->DR=Nuse;

}

if(I2C1->SR1 & 0x40) //RXNE,数据寄存器是否为空,0空1非

{

//将收到的数据储存到RX数组中。虽然我们是从机,用不到这里的数据,但是这里必须读DR寄存器收取数据.不然会出错

Slave_Buffer_Rx[I2C_Rx_Idx++]=I2C1->DR;

if(I2C_Rx_Idx>=MAX_Id)

{

I2C_Rx_Idx=0;

}

}

if(I2C1->SR1 & 0x80) //将发送的数据放入DR寄存器,清除TXE

{

I2C1->DR = Slave_Buffer_Tx[I2C_Tx_Idx++];

if(I2C_Tx_Idx>=MAX_Id)I2C_Tx_Idx=0;

}


}


很多人都说硬件I2C不稳定,不好用,不过就我目前测试来讲,还是挺稳定好用的.

进入单片机查看更多内容>>
相关视频
  • RISC-V嵌入式系统开发

  • SOC系统级芯片设计实验

  • 云龙51单片机实训视频教程(王云,字幕版)

  • 2022 Digi-Key KOL 系列: 你见过1GHz主频的单片机吗?Teensy 4.1开发板介绍

  • TI 新一代 C2000™ 微控制器:全方位助力伺服及马达驱动应用

  • MSP430电容触摸技术 - 防水Demo演示

最新器件
精选电路图
  • CCD图像传感器在微光电视系统中的应用

  • 光控音效发生器电路

  • 非常简单的150W功放电路图

  • 分享一个电网倾角计电路

  • 电谐波图形均衡器示意图

  • 一种构建12V和230V双直流电源的简单方法

    相关电子头条文章