嵌入式工程师经常会遇到系统资源不够的情况,比如我就遇到了串口不够,但IO引脚和定时器还富余,这时就可以通过模拟方式来扩展出一个串口。
这个方法适用于所有嵌入式系统,但是要注意,对于速度不要设置得太高,尽量不要高于9600。否则,当串口所用到的资源优先级不够高,而系统又很繁忙的时候,容易丢失数据或出错。
软件IIC是同样的道理,用软件模拟相关时序,但是IIC对时序要求不高,相对容易
我做的是单总线SWI,即同一根线完成收和发,半双工方式运行。接收数据和发送数据不能同时。
串口时序
先说一下串口的时序,用得最多的就是9600,8,N,1,解释一下:
起始位1,在配置描述中中省略,低电平。
9600是波特率,即每秒产生的位数。也有些设备是9600或更低。
8:数据位是8位
N:无
1:1个停止位,高电平
以9600,8,N,1为例,整个时序如下:
串口空闲时默认为高电平,当有一个低电平到来时,表示收到起始位,然后依次是8个数据位,接着收到1个高电平,每个位的宽度为104微秒。
串口的波特率误差容忍为5%以内,为什么呢?其实很简单,串口的数据采样是在每个数据位的中心位置,即50%处采样,而一个数据帧有10位(1起始,8数据,1停止,无校验),即每个数据位只能允许5%的误差。如果你的软件工作时经常被中断打断,而中断执行的时间又比较长,那么只能选择更低的波特率,或者干脆放弃软件模拟这种方式。
发送数据
发送数据很简单,按照上图时序依次发出就行。以发送一个字节为例,u8 data = 0xa3
1、将这个IO口(以PA0为例),配置为开漏输出,上拉,输出为1。
2、配置定时器的溢出周期为104微秒,9600的倒数。
3、启动定时器
4、在定时器的中断中执行以下内容:
- PA = 0x01&data;
- data >>= 1; //共右移8次
- 第1次进中断:PA0 = 0;
- 第2~9次进中断:
- 第10次进中断:PA0 = 1;
- 第11次进中断:关定时器
- 发送完成
接收数据
接收数据要麻烦一些,不过也不难,按上图来。
1、将这个IO口(以PA0为例),配置为输入EXTI模式,上拉,输出为1,这样不影响输入。
2、配置定时器的溢出周期为52微秒。即为输出时定时器周期的一半时间。
3、PA0为低时,触发中断,此时EXTI中断,并启动定时器
4、定时器中断中执行如下操作
- 偶数次中断直接退出,因为此时对应的是电平变化的时刻,无法读取电平
- 第1次进中断:多次读取PA0的电平状态,均为0时表示正确收到低电平,可以执行后续数据读取操作,否则收到是毛刺信号,退出即可。重置为初始接收状态。
- 第3~17次奇数中断,多次读取PA0的电平,多数值为1,即认为收到高电平,否则认为收到低电平。如果是高电平:buf = (buf|0x80)>>1,低电平时,buf >>=1。 =注意,移位操作需要进行7次,而不是8次= 。
- 第19次进中断:多次读取PA0的电平,
- 多数值为1,此时关定时器,并开启EXTI中断,接收正常完成。
- 多数值为0,表示帧结束错误,此时关定时器,并开启EXTI中断,进行错误处理流程。这一步也可以不处理,认为收到的数据正确即可。
- 接收完成