打印
[其它]

IO口模拟串口

[复制链接]
775|63
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
mmbs|  楼主 | 2023-12-17 21:46 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式

在做IO口模拟串口试验之前要搞懂串口通信的原理。一般的串行通信是以10位数据为一帧传送数据,包括一位起始位,8位数据位,一位停止位。
在传送数据之前,引脚一直抱保持高电平,起始位是低电平,接着跟8位数据,最后将电平拉高。
那么在模拟发送数据时,首先将IO口的电平置一。
void RCC_Configuration(void)
{
SystemInit();
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
}//初始化时钟
void GPIO_Configuration(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_Init(GPIOA, &GPIO_InitStructure);
}//配置IO口

// 发送函数
void SendIO(uint8_t send)
{
uint8_t i = 0;
GPIO_ResetBits(GPIOA,GPIO_Pin_9); //起始位 低电平
Delay(0x2E5); //发送完起始位,延时 1000000 / 9600 = 104 432 是试出来的延时
for(i=0; i<8;i++) //8位数据位 一位停止位
{
if(send & 0x01)
{
GPIO_SetBits(GPIOA,GPIO_Pin_9);//当前数据位是1 ,就拉高
}
else
{
GPIO_ResetBits(GPIOA,GPIO_Pin_9);//当前数据位是0,就置零
}
send >>= 1 ;
Delay(0x33A); // 发送每一位数据位延时
}
//Delay(0x212); // 发送完数据位延时
GPIO_SetBits(GPIOA,GPIO_Pin_9); //停止位
}
延时函数:
void Delay(__IO uint32_t nCount)
{
for(; nCount != 0; nCount–);
}
发送函数中用到的延时是经过我测量后得到的,用逻辑分析仪分析发送数据的过程时,每发送一位,延时约104us,Delay(0x33A)大约是104us 。 这个发送函数直接在main函数中调用就可以了。
接收函数的编写。
接收函数主要是考虑到一个下降沿信号,当使用串口调试助手向STM32 发送数据时,调试助手发送的是一帧数据,包括起始位,数据位,停止位。那么首先就应该检测到这个起始位,也就是这个下降沿信号,外部中断正好能检测到下降沿。所以应用外部中断。

void GPIO_Configuration(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
EXTI_InitTypeDef EXTI_InitStruct;
NVIC_InitTypeDef NVIC_InitStructure;

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; //上拉输入 ,IO口默认是高电平
GPIO_Init(GPIOA, &GPIO_InitStructure);
GPIO_SetBits(GPIOA,GPIO_Pin_10);//保持高电平

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_Init(GPIOA, &GPIO_InitStructure);

EXTI_InitStruct.EXTI_Line=EXTI_Line10;
EXTI_InitStruct.EXTI_Mode=EXTI_Mode_Interrupt;
EXTI_InitStruct.EXTI_Trigger=EXTI_Trigger_Falling; //下降沿中断
EXTI_InitStruct.EXTI_LineCmd=ENABLE;
EXTI_Init(&EXTI_InitStruct);

NVIC_InitStructure.NVIC_IRQChannel=EXTI15_10_IRQn; //外部中断,边沿触发
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2;
NVIC_InitStructure.NVIC_IRQChannelCmd=ENABLE;
NVIC_Init(&NVIC_InitStructure);

}//配置外部中断

这里定义一个枚举类型,表示接收到哪一位。
enum stat{
COM_START_BIT, //起始位
COM_D0_BIT, //bit0
COM_D1_BIT, //bit1
COM_D2_BIT, //bit2
COM_D3_BIT, //bit3
COM_D4_BIT, //bit4
COM_D5_BIT, //bit5
COM_D6_BIT, //bit6
COM_D7_BIT, //bit7
COM_STOP_BIT, //bit8
};

定义8位接收数据
u8 recvData ; //接收的数据

写中断函数:
void EXTI15_10_IRQHandler(void)
{
enum stat recvStat = COM_STOP_BIT; //定义初始为停止位
if(EXTI_GetITStatus(EXTI_Line10)!=RESET) //检查外部中断是否产生了
{
if(!GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_10)) //检测引脚高低电平,如果是低电平,则说明检测到起始位
{
GPIO_SetBits(GPIOA,GPIO_Pin_2);//LED 代表检测到了起始位
Delay(0x33A); //延时 0x432 约等于104us
if(recvStat == COM_STOP_BIT)
{
recvStat = COM_START_BIT; //此时的状态是起始
while(recvStat!= COM_STOP_BIT) // 循环到停止位
{
recvStat++; // 改变状态
if(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_10)) //‘1’
{
recvData |= (1 <<(recvStat -1));
}
else
{
recvData &= ~(1 << (recvStat -1));
}
Delay(0x33A);
}
}
}
EXTI_ClearITPendingBit(EXTI_Line10); //清除EXTI_Line10中断挂起标志位

}
  • 1

}

主函数:
void RCC_Configuration(void);
void GPIO_Configuration(void);
void SendIO(uint8_t send);
void ReceiveIO(void);
extern u8 recvData ;
void Delay(__IO uint32_t nCount);

int main(void)
{
RCC_Configuration();
GPIO_Configuration();
GPIO_SetBits(GPIOA,GPIO_Pin_9);//先拉高 ,第一次数据没有之前没有停止位
while(1)
{
SendIO(recvData);
Delay(0xFFFFF);
}
}


使用特权

评论回复
沙发
tpgf| | 2024-1-3 12:04 | 只看该作者
使用io口都可以模拟哪些类型的串口通讯呢

使用特权

评论回复
板凳
nawu| | 2024-1-3 12:13 | 只看该作者
可以使用不同的io口模拟多个不同的串口通讯吗

使用特权

评论回复
地板
zljiu| | 2024-1-3 16:15 | 只看该作者
这个跟模拟的串口通讯的种类也有关系

使用特权

评论回复
5
yangxiaor520| | 2024-1-3 19:24 | 只看该作者
串口协议不负杂,IO模拟容易实现。

使用特权

评论回复
6
tfqi| | 2024-1-3 22:31 | 只看该作者
如果发送接收端的电平不匹配的话  如何处理呢

使用特权

评论回复
7
gwsan| | 2024-1-3 23:09 | 只看该作者
如果模拟的话   还需要注意硬件上加上拉电阻

使用特权

评论回复
8
aoyi| | 2024-1-3 23:38 | 只看该作者
nawu 发表于 2024-1-3 12:13
可以使用不同的io口模拟多个不同的串口通讯吗

如果使用模拟的话 估计软件会比较不好做

使用特权

评论回复
9
tabmone| | 2024-1-4 10:58 | 只看该作者
选择一个IO口作为发送端,另一个IO口作为接收端。这两个IO口可以是任何两个空闲的IO口。

使用特权

评论回复
10
yeates333| | 2024-1-4 11:05 | 只看该作者
这种方式的实现需要先了解串口的时序图,并使用两个IO引脚分别作为收发引脚。

使用特权

评论回复
11
pentruman| | 2024-1-4 11:35 | 只看该作者
需要通过改进硬件设计、优化软件程序等

使用特权

评论回复
12
updownq| | 2024-1-4 12:04 | 只看该作者
可以在系统资源有限的情况下,通过模拟方式实现串行通信,扩展系统的功能。

使用特权

评论回复
13
cashrwood| | 2024-1-4 12:34 | 只看该作者
如果使用中断来处理串口通信,确保编写中断服务程序来处理发送和接收中断,以及处理任何可能发生的错误。

使用特权

评论回复
14
loutin| | 2024-1-4 13:18 | 只看该作者
根据通信协议要求,设计相应的硬件电路。这包括配置IO端口、电平转换器、波特率发生器等部分。

使用特权

评论回复
15
primojones| | 2024-1-4 16:23 | 只看该作者
模拟串口的实现可能会占用大量的CPU资源,因此在资源受限的单片机中需要谨慎使用。

使用特权

评论回复
16
uiint| | 2024-1-4 16:53 | 只看该作者
每当检测到电平变化时,就记录一个时钟周期的时间戳。然后,根据这些时间戳和预先设定的时钟频率,就可以解码出原始的数据位。

使用特权

评论回复
17
uptown| | 2024-1-4 17:24 | 只看该作者
在某些应用场景下,当嵌入式系统中的串口资源不足,或者需要节省硬件成本时,可以采用这种方法。

IO口模拟串口的基本原理是使用两个GPIO引脚(一个用于发送,一个用于接收)和一个定时器来模拟串行通信的时序。

使用特权

评论回复
18
xiaoyaodz| | 2024-1-4 17:54 | 只看该作者
IO口模拟串口是一种技术,通过普通IO口和定时器中断,可以实现串口通信的功能。

使用特权

评论回复
19
minzisc| | 2024-1-4 18:26 | 只看该作者
根据所需的波特率,配置定时器的溢出周期和中断频率。例如,如果波特率为9600,则定时器的溢出周期为104微秒。

使用特权

评论回复
20
lihuami| | 2024-1-5 09:35 | 只看该作者
IO口模拟串口是一种灵活且实用的串口通信方式,但在使用过程中需要注意正确配置相关参数并根据具体的协议进行编程。

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

179

主题

6359

帖子

3

粉丝