本帖最后由 xld0932 于 2022-2-13 00:01 编辑
#申请原创# @21小跑堂
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来实现对于遥控器的解码,这个这么说吧,我现在还没想到怎么来实现。
软件工程源代码
Template.zip
(716 KB)
|
lihai