历史上的今天
今天是:2025年02月02日(星期日)
2020年02月02日 | STM8L051F3 硬件I2C从机实例--新手导航
2020-02-02 来源:eefocus
这是我写的第一篇有关于技术的文章,可能写的不是很好,仅供参考。
先介绍一下背景,我是第一次接触STM的芯片,以前接触过都是基于51的芯片,算是有一点基础吧。因为公司的项目需要用到STM8L051F3的硬件I2C作为从机送数据,把自己从接触到调通遇到的问题记录一下,有需要的可以参考一下。
这篇文章是以相对新手的水平去写给新手参考的,所以我会尽量写的详细点。
1.首先我们需要查看数据手册,看下硬件I2C的接口是哪个端口,我用的是STM8L051F3这一块芯片。
这款芯片的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不稳定,不好用,不过就我目前测试来讲,还是挺稳定好用的.
史海拾趣
|
老是看到好多新同学打听这个专业,N多人还在比较电力系统和电力电子与电力传动,哪个更好?哪个更有前(钱)途?马上就过年了,今天有点空,也想冒下泡,想跟对这一方向有点兴趣的兄弟姐妹简单聊一下总体情况。我也只是一名研发工程师,说得不对不全 ...… 查看全部问答> |
|
www.kingofcoder.com 100MB 免費空間 + 100MB mysql 空間 www.kingofcoder.com 100MB 免費空間 + 100MB mysql 空間 大家快點來呀 JSP, PHP, oracle空間, 很快就會開通, 大家快點登記吧 http://www.kingofcoder.com匯集大量各種編程語言文章、提供免費asp、php、jsp空間、免費mysql、oracle數據庫空間 ...… 查看全部问答> |
|
接触LM3S快一周了,这是我第一次接触ARM微处理器,没想到很快就上手啦,激动之余觉得里面那么多的寄存器,设置起来确实是十分的复杂,虽然提供了库函数,但是里面的函数及常量实在是多,如果不用去记忆这些函数就好了,为此,我编写了一个初始化配 ...… 查看全部问答> |
|
如何设计一款半导体开关系统,从而最大限度地利用开关主机功能? 半导体特性分析实验室、技术开发实验室、建模[1]实验室和可靠性实验室经常需要各种不同的源测量仪器,并且需要连接至多个装置。开关系统[2]可以确保测试条件和测量具有极高的可靠性。如图1所示,器件引脚与仪器之间的各种连接的可能性使得矩阵成 ...… 查看全部问答> |





