基于MM32与TFBS4711实现IrDA红外通讯

IrDA百科

IrDA是Infrared Data Association(红外线数据标准协会)的英文缩写,IrDA红外接口是一种红外线无线传输协议以及基于该协议的无线传输接口。它是用来取代点对点的线缆连接,具有小角度(30度锥角以内)、短距离、点对点直线数据传输、保密性强的特点。在IrDA物理层中,根据数据通讯速率将其分为这几类:SIR(9.6kbps~115.2kbps)、MIR(0.576Mbps和1.152Mbps)、FIR(4Mbps)和VFIR(16Mbps)。

MM32F0140 & IrDA

官方在MM32F0140用户手册中,关于UART简介中有写到:对于使用工业标准NRZ异步串行数据格式的外设,通过异步收发器(UART)可以灵活的与之进行全双工数据交换;通过分数波特率发生器,UART还可以选择宽范围的波特率;还支持异步单向通讯和半双工通讯、调制解调器(CTS/RTS)操作、以及IrDA红外通讯功能。

MM32F0140系列芯片UART支持IrDA SIR ENDEC规范的红外功能;当UART工作在IrDA红外模式时(SIREN寄存器位使能),UART的STOP位必须配置成1个停止位;UART TX引脚默认输出高电平。结合选用的IrDA收发器TFBS4711,以及SIR ENDEC应用,UART最低传输速率需达到9.6kbps,最高不得超过115.2kbps。

SIR可以工作在正常模式,也可以工作在低功耗模式,可以通过SIRLP这个寄存器位来进行配置选择。IrDA SIR物理层规定使用反相归零调制方式(RZI),该方式用一个红外光脉冲代表逻辑0;在正常模式下,0的脉冲宽度为3/16位长度。在低功耗模式下,脉冲宽度将不再是3/16位长度,而是低功耗波特率时钟周期的3倍(SIRLP_CLK*3),低功耗模式下的频率SIRLP_CLK可以通过PSC预分频器,对UART时钟源PCLK进行分频来设置,该频率范围需要设定在1.42MHz < SIRLP_CLK < 2.12MHz之间。

 

SIR发送编码器和SIR接收译码器实现了UART比特流与红外脉冲流之间的相互转换:SIR发送编码器把逻辑0作为高电脑发送,把逻辑1作为低电平发送;而SIR接收译码器则是将接收到的IrDA信号转变成比特数据后再发送给UART。

IrDA红外通讯是一个半双工通讯协议,编解码不可以同时进行:如果发送器处于忙的状态,也就是UART TX正在传输数据给IrDA编码器,IrDA接收线上的任何数据都将被IrDA解码器忽略;如果接收器处于忙的状态下,也就是说UART RX正在接收来自于IrDA解码器的数据,UART TX到IrDA的数据将不会被IrDA进行编码和传输;所以当收发数据时应当避免同时操作,以保证数据的正确性。

原理图设计

对于硬件的功能主要是测试IrDA红外通讯功能,所以功能相对简单;在原理图设计的时候,在MM32F0140最小系统的基础上增加了一个LED灯、一个KEY按键、一个CH340用于在程序调试过程中来输出运行日志等信息、最后就是一个TFB84711;对于MCU的下载接口我们使用JTAG的SWD下载模式,具体的设计图纸如下所示:

 

PCB设计

在原理图完成之后,先根据需求检查功能是否都满足了、再检查一下原理图设计,最后将其转换成PCB进行布线;在布线完成后,检查DRC没有问题后,生成GERBER文件进行投板。完成的PCB图纸如下所示:

 

回板焊接调试

因为IrDA红外通讯是点对点的,最小需要两块PCBA来配合,所以就手工焊接了两块PCB:

 

在开始调试之前除了焊接硬件,也需要基于原理图设计来准备测试程序,这个开发的测试工程就包括了LED闪烁、按键识别和处理、通过UART1和CH340来与PC端的调试终端实现数据通讯的功能;在基础功能调试完成了,就开始来实现IrDA的底层配置、功能,并通过串口调试终端软件来实现两块板子上的IrDA红外数据的收发功能。

代码实现:LED灯

复制

/*******************************************************************************

* [url=home.php?mod=space&uid=288409]@file[/url] LED.c

* [url=home.php?mod=space&uid=187600]@author[/url] King

* [url=home.php?mod=space&uid=895143]@version[/url] V1.00

* [url=home.php?mod=space&uid=212281]@date[/url] 21-Dec-2021

* [url=home.php?mod=space&uid=247401]@brief[/url] ......

*******************************************************************************/

/* Define to prevent recursive inclusion -------------------------------------*/

#define __LED_C__

/* Includes ------------------------------------------------------------------*/

#include "LED.h"

/* Private typedef -----------------------------------------------------------*/

/* Private define ------------------------------------------------------------*/

/* Private macro -------------------------------------------------------------*/

/* Private variables ---------------------------------------------------------*/

/* Private function prototypes -----------------------------------------------*/

/* Private functions ---------------------------------------------------------*/

/* Exported variables --------------------------------------------------------*/

/* Exported function prototypes ----------------------------------------------*/

/*******************************************************************************

* [url=home.php?mod=space&uid=247401]@brief[/url]

* @param

* @retval

* [url=home.php?mod=space&uid=93590]@Attention[/url]

*******************************************************************************/

void LED_Init(void)

{

GPIO_InitTypeDef GPIO_InitStructure;

RCC_AHBPeriphClockCmd(RCC_AHBENR_GPIOA, ENABLE);

GPIO_StructInit(&GPIO_InitStructure);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;

GPIO_Init(GPIOA, &GPIO_InitStructure);

GPIO_WriteBit(GPIOA, GPIO_Pin_1, Bit_RESET);

TASK_Append(TASK_ID_LED, LED_Toggle, 250);

}

/*******************************************************************************

* [url=home.php?mod=space&uid=247401]@brief[/url]

* @param

* @retval

* [url=home.php?mod=space&uid=93590]@Attention[/url]

*******************************************************************************/

void LED_Toggle(void)

{

if(!GPIO_ReadOutputDataBit(GPIOA, GPIO_Pin_1))

{

GPIO_WriteBit(GPIOA, GPIO_Pin_1, Bit_SET);

}

else

{

GPIO_WriteBit(GPIOA, GPIO_Pin_1, Bit_RESET);

}

}

/******************* (C) COPYRIGHT 2021 *************************END OF FILE***/

代码实现:KEY按键

复制

/*******************************************************************************

* [url=home.php?mod=space&uid=288409]@file[/url] KEY.c

* [url=home.php?mod=space&uid=187600]@author[/url] King

* [url=home.php?mod=space&uid=895143]@version[/url] V1.00

* [url=home.php?mod=space&uid=212281]@date[/url] 21-Dec-2021

* @brief ......

*******************************************************************************/

/* Define to prevent recursive inclusion -------------------------------------*/

#define __KEY_C__

/* Includes ------------------------------------------------------------------*/

#include "KEY.h"

/* Private typedef -----------------------------------------------------------*/

/* Private define ------------------------------------------------------------*/

/* Private macro -------------------------------------------------------------*/

/* Private variables ---------------------------------------------------------*/

/* Private function prototypes -----------------------------------------------*/

/* Private functions ---------------------------------------------------------*/

/* Exported variables --------------------------------------------------------*/

/* Exported function prototypes ----------------------------------------------*/

/*******************************************************************************

* @brief

* @param

* @retval

* [url=home.php?mod=space&uid=93590]@Attention[/url]

*******************************************************************************/

void KEY_Init(void)

{

GPIO_InitTypeDef GPIO_InitStructure;

RCC_AHBPeriphClockCmd(RCC_AHBENR_GPIOA, ENABLE);

GPIO_StructInit(&GPIO_InitStructure);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD;

GPIO_Init(GPIOA, &GPIO_InitStructure);

TASK_Append(TASK_ID_KEY, KEY_Scan, 10);

}

/*******************************************************************************

* @brief

* @param

* @retval

* @attention

*******************************************************************************/

void KEY_SubScan(uint8_t *State, uint8_t *Count, uint8_t Value, char *Name)

{

if(*State == 0)

{

if(Value != Bit_RESET) *Count += 1;

else *Count = 0;

if(*Count > 5)

{

*Count = 0; *State = 1;

printf("\r\n%s Pressed", Name);

}

}

else

{

if(Value == Bit_RESET) *Count += 1;

else *Count = 0;

if(*Count > 5)

{

*Count = 0; *State = 0;

printf("\r\n%s Release", Name);

}

}

}

/*******************************************************************************

* @brief

* @param

* @retval

* @attention

*******************************************************************************/

void KEY_Scan(void)

{

static uint8_t KeyState = 0;

static uint8_t KeyCount = 0;

KEY_SubScan(&KeyState, &KeyCount, GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_0), "KEY");

}

/******************* (C) COPYRIGHT 2021 *************************END OF FILE***/

代码实现:UART2 & SHELL接口移植

复制

/*******************************************************************************

* [url=home.php?mod=space&uid=288409]@file[/url] shell_port.c

* [url=home.php?mod=space&uid=187600]@author[/url] King

* [url=home.php?mod=space&uid=895143]@version[/url] V1.00

* [url=home.php?mod=space&uid=212281]@date[/url] 21-Dec-2021

* @brief ......

*******************************************************************************/

/* Define to prevent recursive inclusion -------------------------------------*/

#define __SHELL_PORT_C__

/* Includes ------------------------------------------------------------------*/

#include "shell_port.h"

/* Private typedef -----------------------------------------------------------*/

/* Private define ------------------------------------------------------------*/

/* Private macro -------------------------------------------------------------*/

/* Private variables ---------------------------------------------------------*/

SHELL_TypeDef shell;

/* Private function prototypes -----------------------------------------------*/

/* Private functions ---------------------------------------------------------*/

/* Exported variables --------------------------------------------------------*/

/* Exported function prototypes ----------------------------------------------*/

/*******************************************************************************

* @brief

* @param

* @retval

* @attention

*******************************************************************************/

void shellPortWrite(const char ch)

{

UART_SendData(UART2, (uint8_t)ch);

while(UART_GetFlagStatus(UART2, UART_IT_TXIEN) == RESET);

}

/*******************************************************************************

* @brief

* @param

* @retval

* @attention

*******************************************************************************/

void shellPortInit(void)

{

GPIO_InitTypeDef GPIO_InitStructure;

NVIC_InitTypeDef NVIC_InitStructure;

UART_InitTypeDef UART_InitStructure;

RCC_AHBPeriphClockCmd(RCC_AHBENR_GPIOA, ENABLE);

RCC_APB1PeriphClockCmd(RCC_APB1ENR_UART2, ENABLE);

NVIC_InitStructure.NVIC_IRQChannel = UART2_IRQn;

NVIC_InitStructure.NVIC_IRQChannelPriority = 1;

NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

NVIC_Init(&NVIC_InitStructure);

UART_StructInit(&UART_InitStructure);

UART_InitStructure.UART_BaudRate = 115200;

UART_InitStructure.UART_WordLength = UART_WordLength_8b;

UART_InitStructure.UART_StopBits = UART_StopBits_1;

UART_InitStructure.UART_Parity = UART_Parity_No;

UART_InitStructure.UART_HardwareFlowControl = UART_HardwareFlowControl_None;

UART_InitStructure.UART_Mode = UART_Mode_Rx | UART_Mode_Tx;

UART_Init(UART2, &UART_InitStructure);

UART_ITConfig(UART2, UART_IT_RXIEN, ENABLE);

UART_Cmd(UART2, ENABLE);

GPIO_PinAFConfig(GPIOA, GPIO_PinSource2, GPIO_AF_1);

GPIO_PinAFConfig(GPIOA, GPIO_PinSource3, GPIO_AF_1);

GPIO_StructInit(&GPIO_InitStructure);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;

GPIO_Init(GPIOA, &GPIO_InitStructure);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;

GPIO_Init(GPIOA, &GPIO_InitStructure);

shell.write = shellPortWrite;

shellInit(&shell);

}

/*******************************************************************************

* @brief

* @param

* @retval

* @attention

*******************************************************************************/

void UART2_IRQHandler(void)

{

if(UART_GetITStatus(UART2, UART_IT_RXIEN) != RESET)

{

shellHandler(&shell, UART_ReceiveData(UART2));

UART_ClearITPendingBit(UART2, UART_IT_RXIEN);

}

}

/*******************************************************************************

* @brief

* @param

* @retval

* @attention

*******************************************************************************/

int fputc(int ch, FILE *f)

{

UART_SendData(UART2, (uint8_t)ch);

while(UART_GetFlagStatus(UART2, UART_IT_TXIEN) == RESET);

return ch;

}

/******************* (C) COPYRIGHT 2021 *************************END OF FILE***/

代码实现:UART1 & IrDA实现

复制

/*******************************************************************************

* @file IrDA.c

* @author King

* @version V1.00

* @date 21-Dec-2021

* @brief ......

*******************************************************************************/

/* Define to prevent recursive inclusion -------------------------------------*/

#define __IrDA_C__

/* Includes ------------------------------------------------------------------*/

#include "IrDA.h"

/* Private typedef -----------------------------------------------------------*/

/* Private define ------------------------------------------------------------*/

/* Private macro -------------------------------------------------------------*/

/* Private variables ---------------------------------------------------------*/

/* Private function prototypes -----------------------------------------------*/

/* Private functions ---------------------------------------------------------*/

/* Exported variables --------------------------------------------------------*/

/* Exported function prototypes ----------------------------------------------*/

/*******************************************************************************

* @brief

* @param

* @retval

* @attention

*******************************************************************************/

void IrDA_Init(void)

{

GPIO_InitTypeDef GPIO_InitStructure;

NVIC_InitTypeDef NVIC_InitStructure;

UART_InitTypeDef UART_InitStructure;

RCC_AHBPeriphClockCmd(RCC_AHBENR_GPIOA, ENABLE);

RCC_APB2PeriphClockCmd(RCC_APB2ENR_UART1, ENABLE);

NVIC_InitStructure.NVIC_IRQChannel = UART1_IRQn;

NVIC_InitStructure.NVIC_IRQChannelPriority = 1;

NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

NVIC_Init(&NVIC_InitStructure);

UART_StructInit(&UART_InitStructure);

UART_InitStructure.UART_BaudRate = 115200;

UART_InitStructure.UART_WordLength = UART_WordLength_8b;

UART_InitStructure.UART_StopBits = UART_StopBits_1;

UART_InitStructure.UART_Parity = UART_Parity_No;

UART_InitStructure.UART_HardwareFlowControl = UART_HardwareFlowControl_None;

UART_InitStructure.UART_Mode = UART_Mode_Rx | UART_Mode_Tx;

UART_Init(UART1, &UART_InitStructure);

UART_ClearITPendingBit(UART1, UART_IT_RXIEN);

UART_ITConfig(UART1, UART_IT_RXIEN, ENABLE);

UART_Cmd(UART1, ENABLE);

GPIO_PinAFConfig(GPIOA, GPIO_PinSource9, GPIO_AF_1);

GPIO_PinAFConfig(GPIOA, GPIO_PinSource10, GPIO_AF_1);

GPIO_StructInit(&GPIO_InitStructure);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;

GPIO_Init(GPIOA, &GPIO_InitStructure);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;

GPIO_Init(GPIOA, &GPIO_InitStructure);

GPIO_StructInit(&GPIO_InitStructure);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;

GPIO_Init(GPIOA, &GPIO_InitStructure);

GPIO_WriteBit(GPIOA, GPIO_Pin_11, Bit_RESET);

/* SIRLPP_CLK = PCLK2 / 0x28 = 72000000 / 40 = 1.8MHz */

UART1->IRDA = 0x00002801;

}

/*******************************************************************************

* @brief

* @param

* @retval

* @attention

*******************************************************************************/

void UART1_IRQHandler(void)

{

if(UART_GetITStatus(UART1, UART_IT_RXIEN) != RESET)

{

QUEUE_WRITE(QUEUE_IrDA_RX_IDX, UART_ReceiveData(UART1));

UART_ClearITPendingBit(UART1, UART_IT_RXIEN);

}

}

/*******************************************************************************

* @brief

* @param

* @retval

* @attention

*******************************************************************************/

void IrDA_SendData(uint8_t Data)

{

UART_SendData(UART1, Data);

while(UART_GetFlagStatus(UART1, UART_IT_TXIEN) == RESET);

}

/*******************************************************************************

* @brief

* @param

* @retval

* @attention

*******************************************************************************/

void IrDA_SHELL_Handler(uint8_t Data)

{

printf("\r\nIrDA TX : ");

for(uint8_t i = 0; i < 10; i++)

{

IrDA_SendData(Data+i);

printf("0x%02x ", Data+i);

}

}

SHELL_EXPORT_CMD(IrDA, IrDA_SHELL_Handler, IrDA Send Data);

/******************* (C) COPYRIGHT 2021 *************************END OF FILE***/

测试运行结果

两块硬件单板,通过SHELL命令的调用方式,错开时间相互发送数据:

后续

MM32F0140系列MCU的IrDA红外通讯功能,它遵循的是IrDA SIR ENDEC规范;而NEC编码是红外遥控器编码方式的一种,它是一个满足特定时序和频率的载波通讯协议,它们唯一的共性就是使用红外进行通讯,所以刚到很多网友在问如何通过MCU的IrDA来实现对于遥控器的解码,这个这么说吧,我现在还没想到怎么来实现。

---------------------

作者:xld0932

链接:https://bbs.21ic.com/icview-3197654-1-1.html

来源:21ic.com

此文章已获得原创/原创奖标签,著作权归21ic所有,任何人未经允许禁止转载。

  • 1
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
以下是一个基于MM32F5270E芯片编写的红外循迹小车实现识别转弯的简单程序示例: ```c #include "mm32f5270.h" #define LEFT_SENSOR_PIN GPIO_Pin_0 #define RIGHT_SENSOR_PIN GPIO_Pin_1 void GPIO_Configuration(void) { RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); GPIO_InitTypeDef GPIO_InitStructure; GPIO_InitStructure.GPIO_Pin = LEFT_SENSOR_PIN | RIGHT_SENSOR_PIN; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; GPIO_Init(GPIOA, &GPIO_InitStructure); } void Motor_Stop(void) { // 停止电机动作 // 通过设置电机控制引脚的状态 } void Motor_Turn_Left(void) { // 向左转 // 通过设置电机控制引脚的状态 } void Motor_Turn_Right(void) { // 向右转 // 通过设置电机控制引脚的状态 } int main(void) { GPIO_Configuration(); while (1) { if (GPIO_ReadInputDataBit(GPIOA, LEFT_SENSOR_PIN) && GPIO_ReadInputDataBit(GPIOA, RIGHT_SENSOR_PIN)) { // 两侧都检测到黑线,直行 // 执行前进的动作 } else if (GPIO_ReadInputDataBit(GPIOA, LEFT_SENSOR_PIN)) { // 左侧检测到黑线,向右转 Motor_Turn_Right(); } else if (GPIO_ReadInputDataBit(GPIOA, RIGHT_SENSOR_PIN)) { // 右侧检测到黑线,向左转 Motor_Turn_Left(); } else { // 没有检测到黑线,停止 Motor_Stop(); } } } ``` 以上代码仅为示例,具体的电机控制、引脚配置和传感器读取可能需要根据你的硬件连接和具体需求进行调整。请确保在编写代码时参考芯片的数据手册和相关资料,以确保正确配置和操作。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值