关于STM32 RS485控制I/O口不能正常输出高低电平的解决方法

一、问题:

问题现象:在进行RS485操作时,发现接收时而进时而不进中断:

将485的AB输出脚直接与串口的TX,RX对接发现串口定时发数据能进入中断,但是有点不受控制,

接着测量MAX485各个引脚的脚,发现MAX485的RE\DE引脚一直处于高电平

根据RS485工作原理:

  • 当单片机要发送数据的时候,控制CTRL为高电平,数据通过TXD发送出去。
  • 当单片机要接收数据的时候,控制CTRL为低电平,数据通过RXD接收回来。

普通的485电路,除了“用RXD连接485芯片的RO引脚、用TXD连接485芯片的DI引脚”,还会用一个单片机的普通IO引脚连接到RE、DE引脚上,来进行RS485收发控制。

 检测:

通过万用表测量控制RE / DE的引脚一直处于高电平,即使main函数就单独写将该IO引脚为低电平,测量出来还是高电平。

 

二、处理方法

       根据手册查找,了解到 I/O口不能正常输出一般都是端口被复用了造成的,除了检查程序中是否有把端口复用的程序外,还应注意:有些端口在单片机上电时默认就是复用的,如与JTAG相关的PA13,PA14,PA15,PB3,PB4引脚,禁用JTAG或SWD可以释放其中的一些引脚,见表格:

 

RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO ,ENABLE);//重映射需要先使能AFIO时钟
GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable,ENABLE);//只关闭JTAG而保留SWD

 RS485控制收发的引脚正是 ——A15

修改初始化函数: 

void RS485_Config(void)
{
  GPIO_InitTypeDef GPIO_InitStructure;
  USART_InitTypeDef USART_InitStructure;
	NVIC_InitTypeDef NVIC_InitStructure;
	
/* config USART clock */
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOC, ENABLE);
  RCC_APB1PeriphClockCmd(RS485_USART_CLK, ENABLE);
	
  //处理复位问题
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO ,ENABLE);  
  GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable,ENABLE);
 
	/* TX GPIO */
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
  GPIO_Init(GPIOC, &GPIO_InitStructure);

 /*RX */
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; 
  GPIO_Init(GPIOC, &GPIO_InitStructure);

 /* 485收发控制引脚 */ 
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_15 ;
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  GPIO_Init(GPIOA, &GPIO_InitStructure);


/* USART */
  USART_InitStructure.USART_BaudRate = RS485_USART_BAUDRATE;
  USART_InitStructure.USART_WordLength = USART_WordLength_8b;
  USART_InitStructure.USART_StopBits = USART_StopBits_1;
  USART_InitStructure.USART_Parity = USART_Parity_No ;
  USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
  USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
  USART_Init(RS485_USART, &USART_InitStructure);

#ifdef EN_UART4_RX   	//接收使能
/*NVIC*/
  NVIC_InitStructure.NVIC_IRQChannel = UART4_IRQn; 
  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;
  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2;
  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; 
  NVIC_Init(&NVIC_InitStructure); 

  USART_ITConfig(RS485_USART, USART_IT_IDLE, ENABLE);
#endif
  USART_Cmd(RS485_USART, ENABLE);

  _uart4_dma_configuration();
	
  USART_DMACmd(UART4, USART_DMAReq_Tx, ENABLE);  			
  USART_DMACmd(UART4, USART_DMAReq_Rx, ENABLE);  			

  USART_GetITStatus(UART4, USART_FLAG_TC);
  RS485_TX_EN =0;   // 设置485 mode控制: 0 :RX,1:TX
}

三、结果

万用表测量引脚可以引脚变低电平了,测试成功!!

 

参考:STM32 I/O口不能正常输出高低电平问题的解决方案_奇葩猴的博客-CSDN博客

 

### STM32 输出低电平不足的原因分析 STM32输出低电平不足可能由多个因素引起。硬件连接不当是一个常见原因,例如外部电路设计不合理或者元件特性不匹配可能导致实际电压偏离预期值[^1]。 另外,在某些情况下,如果存在光耦合器如6N137这样的器件,则需注意其对逻辑电平的影响。因为这类隔离组件可能会反转输入到输出之间的信号状态,即原本期望的低电平经过转换后变成了电平,反之亦然。因此当遇到类似情况时应考虑调整MCU发出指令的方向来补偿这种变化[^2]。 对于特定引脚始终呈现某一固定电平的现象,可能是由于这些引脚被内部或外部拉电阻强制保持在一个稳定的状态上。比如一些通信接RS485)中的RE/DE管脚会受到内置上下拉结构影响而无法自由切换高低电平。此时应当检查相关设置并确认是否有其他外设干扰到了目标引脚的功能实现[^4]。 最后,软件层面也可能存在问题。初始化配置错误或是程序运行过程中出现了未预见的情况都可能导致指定引脚的行为不符合预期。确保正确设置了GPIO的工作模式以及相应的寄存器参数是非常重要的一步。此外,还需留意是否存在中断服务例程或其他后台进程意外改变了所关心引脚的状态[^3]。 针对上述提到的各种可能性,可以采取如下措施: - **验证硬件连接**:仔细审查PCB布局图和原理图文件,确保所有连线准确无误,并且使用的元器件规格满足需求。 - **处理信号反相问题**:如果是由于光电隔离等原因造成的电平翻转现象,可以在编写代码时加入必要的取反操作以恢复正常工作秩序。 - **排查引脚冲突**:利用调试工具检可疑线路两端的实际波形特征,排除潜在的竞争条件带来的负面影响。 - **优化编程逻辑**:按照官方文档指导完成全面细致的环境搭建过程,同时遵循最佳实践原则构建稳健可靠的嵌入式应用程序框架。 ```c // 设置 GPIO 为推挽输出模式 __HAL_RCC_GPIOA_CLK_ENABLE(); GPIO_InitTypeDef GPIO_InitStruct = {0}; GPIO_InitStruct.Pin = GPIO_PIN_0; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); // 清除 PA0 上的任何先前输出 HAL_GPIO_WritePin(GPIOA, GPIO_PIN_0, GPIO_PIN_RESET); ```
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值