STM32无线透传、PID调试工具

这篇具有很好参考价值的文章主要介绍了STM32无线透传、PID调试工具。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

一、无线透传概述与应用
无线模块 (RF wireless module) ,是数字数传电台 (Digital radio)的模块化产品,是指借助DSP 技术和无线电技术实现的高性能专业数据传输电台模块。 无线模块延伸到很多的系统中,工业控制的开关量io设备,485无线传输设备dw-m1,模拟量采集和控制设备。从使用最早的按键电码、电报、模拟电台模块加无线MODEM,发展到数字电台模块和DSP、软件无线电;传输信号也从代码、低速数据(300~1200bps)到高速数据 (N64KNE1)。
E22-230T22S/30S/900T22S/T30S是全新一代的LoRa无线模块基于SEMTECH公司SX1262射频芯片的无线串口模块 (UART),具有多种传输方式,E22-230T22S/30S工作在(220.125~236.125MHz) 频段(默认230.125MHz),E22-900T22S/T30S工作在(850.125~930.125MHz)频段(默认868125MHz), LoRa扩频技术,TTL电平输出,兼容3.3V与5V的10口电压。E22-230T22S/30S/900T22S/T30S采用全新一代LoRa扩频技术,与传统SX1276方案相比,SX1262方案传输距离更远,可达5KM,速度更快,功耗更低,体积更小,支持空中唤醒无线配置、载波监听、自动中继、通信密钥等,功能,支持分包长度设定。
二、本文中使用的无线透传模块
LoRa模块(内置MCU)采用亿百特E22-400T22S与E22-400T30S,可实现广播监听、定点传输、搭配设计的PCB使用:
STM32无线透传、PID调试工具,stm32,嵌入式硬件,单片机,51单片机,物联网,mcu,硬件工程
STM32无线透传、PID调试工具,stm32,嵌入式硬件,单片机,51单片机,物联网,mcu,硬件工程
模块连接
STM32无线透传、PID调试工具,stm32,嵌入式硬件,单片机,51单片机,物联网,mcu,硬件工程

三、配置步骤与程序讲解
1、模块配置
配置准备:准备好 E22 无线电台,将 M0 拨码开关拨至 0,M1 拨码开关拨至1,设置为配置模式(模式 2),如下图。(注:拨码开关带标识端(ON 端)为 0);
STM32无线透传、PID调试工具,stm32,嵌入式硬件,单片机,51单片机,物联网,mcu,硬件工程
配置:打开配置软件 RF_Setting(E22-E9X(SL)) V2.8.exe,选择并打开串口,读取参数,此时会显示电台参数,修改参数后,点击写入参数即完成参数配置,如下图。(注:1、除频率信道外,其他参数均按下图配置,具体参数说明可参考产品手册;2、频率通道关系到两个电台的对频,因此需要将两个配对的电台均
设置为相同的频率通道(0~83),如图将两个电台的信道均设为5);
STM32无线透传、PID调试工具,stm32,嵌入式硬件,单片机,51单片机,物联网,mcu,硬件工程
测试准备:将 M0 拨码开关拨至 0,M1 拨码开关也拨至 0,设置为一般模式(模式 0),如下图。(注:拨码开关带标识端(ON 端)为 0)
STM32无线透传、PID调试工具,stm32,嵌入式硬件,单片机,51单片机,物联网,mcu,硬件工程
测试:两个电台均配置完成,且设为一般模式后,就能够正常地进行信号的发射和接收,通过串口调试助手 XCOM 进行测试,如图所示。
STM32无线透传、PID调试工具,stm32,嵌入式硬件,单片机,51单片机,物联网,mcu,硬件工程
2、程序部分
透传数据:
STM32无线透传、PID调试工具,stm32,嵌入式硬件,单片机,51单片机,物联网,mcu,硬件工程
初始化透传串口,采用结构体形式进行初始化:
STM32无线透传、PID调试工具,stm32,嵌入式硬件,单片机,51单片机,物联网,mcu,硬件工程
透传接收处理:

//串口接收处理函数
/**
  * @brief  接受收据数据帧解包
  * @param   R_data 逐个接收到的数据
  			pUSART: 串口号
  * @retval : 解包状态
  */
void Xbee_ReceivedByte_SUART(u8 R_data, struct USART_TypeDefStruct *pUSART)
{
//  IAP_Recieve(R_data);
	Through_RecStringCmd(pUSART,R_data);			  //串口数据透传解析
    switch (pUSART->RecvParse.R_state)				  //接收数据帧解包用的结构变量
    {
        case 0:
            {        
                if (R_data == (u8)'$')
                {
                    pUSART->RecvBuff[0] = R_data; 
					pUSART->RecvParse.R_state = 1;
					pUSART->RecvParse.sum =  R_data;
					return ;
                }		
                break;
            } 
        case 1:
            {
                pUSART->RecvBuff[1] = R_data; 		 // MASTER +SLAVER
				pUSART->RecvParse.R_state++; 
				pUSART->RecvParse.sum += R_data;
				return ;               
            }
       case 2:
            {
                pUSART->RecvBuff[2] = R_data; 		 // 功能码字节
				pUSART->RecvParse.R_state++; 
				pUSART->RecvParse.sum += R_data;
				return ;               
            }
       case 3:
            {
                pUSART->RecvBuff[3] = R_data; 	// Rec Data  Length;  字节长度
				pUSART->RecvParse.R_state++; 
				pUSART->RecvParse.sum += R_data;
				pUSART->RecvParse.p_rec_num = 4;				  //接收数据缓冲器指针
				switch(pUSART->RecvBuff[2])
				{
				case 0x02:
				  pUSART->RecvParse.lenth_rec_num = 9;   //lenth_rec_num 接收长度-1
				  break;
				case 0x03:
				case 0x07:
				  pUSART->RecvParse.lenth_rec_num = 7; 	  //lenth_rec_num 接收长度-1
				  break;
				default:
				  pUSART->RecvParse.lenth_rec_num = 7+R_data; 	  //lenth_rec_num 接收长度-1
				  break;
				}
				return ;
            }
		case 4:
            {
                pUSART->RecvBuff[pUSART->RecvParse.p_rec_num++] = R_data;
							if(pUSART->RecvParse.p_rec_num >= pUSART->RecvParse.lenth_rec_num)
							{
								pUSART->RecvParse.R_state = 5; 	
							}
							pUSART->RecvParse.sum += R_data;
                return ;
            }			
        case 5: //接收数据
            { //	RecvBuff 	 接收的字符串缓冲器
            pUSART->RecvBuff[pUSART->RecvParse.p_rec_num++] = R_data;
            if(pUSART->RecvParse.sum == R_data) // 校验和通过
						{
							ReceivedComPortDataEvent(pUSART->RecvBuff,pUSART);
						}
						pUSART->RecvParse.R_state = 0;
            return ;
            }
        default:
            {
                pUSART->RecvParse.R_state = 0;
                return ;
            }
    }
    pUSART->RecvParse.R_state = 0;
    return ;
}

透传发送任务处理:

/**
  * @brief  该函数必须1ms定时器中调用
  * @param   None
  * @retval : None
  */
void CommProtocol_task(void)
{
	if(N_1ms_EventUpData>0)
	{
	 u8 i;
	 N_1ms_EventUpData = 0;
	 for(i=0;i<SIZEREPEATARRAY;i++)
	 {
	 	if(RepeatArray[i].enable ==1)			  //发送使能
		{
			RepeatArray[i].n_1ms ++;			  //时间变量累加1ms
			if(RepeatArray[i].n_1ms>=RepeatArray[i].period_ms)	//时间到达发送定时周期
			{													
				RepeatArray[i].n_1ms = 0;						//时间变量复位,待发送数据帧进入收发缓冲池
				SetSendBuffer(RepeatArray[i].pUSART,0xA2,				//串口,功能码0XA2, 定时返回数据
						OwnAddr,RepeatArray[i].objAddr,					//发送者,接收者,发送者地址要根据程序作相应变动
						RepeatArray[i].regaddr,RepeatArray[i].length,	//基地址,字节长度
						EE_ParaData.c+RepeatArray[i].regaddr,0);		//发送的起始数据地址指针,返回标志
			}
		}
	 }	
	}
}

main函数

int main(void)
{	  
	LED_Init();		  				//初始化与LED连接的硬件接口
	CommProtocol_init();		    //通信协议缓冲队列初始化,只是将数据帧缓冲队列中的发送使能标志清零
	XBeeUsart_init();				//初始化电台的串口通信实例   
	UsartShow_init();				//GPS串口初始化
	SystemIni();
	SysTick_Config(SystemCoreClock / 1000);	//1ms
	SetContiuneDefaultData();  		// 初始化向地面站连续发送数据的地址空间
	while(1)
	{			   
		CommProtocol_task();
		USART_Send_Task(&XBee_Usart);			   	//向XBee串口发送任务	
		if(N_1ms_GPS_Watch>10)
		{
			N_1ms_GPS_Watch=0;
			unPkg_GPS_task();						//解析GPS数据任务
		}
		EE_ParaData.name.Ki = rand()%100+1;
	}
}

程序兼容STM32F1系列单片机,移植或直接使用工程即可,F4系列只需修改串口部分。以上程序只是主要的处理过程,有一些函数未给出,都包含在工程中。
3、上位机
安装:安装很简单,按照资料的安装步骤一步步安装即可,安装成功后打开我们的文件夹,如图箭头1所指JS-GCS.exe就是我们的上位机程序。
STM32无线透传、PID调试工具,stm32,嵌入式硬件,单片机,51单片机,物联网,mcu,硬件工程
数据添加:将程序中的需要透传到上位机的数据添加到上图箭头2的飞鸿地址表中,如图,注意地址与H地址的填写,是根据字节长度来决定的。
STM32无线透传、PID调试工具,stm32,嵌入式硬件,单片机,51单片机,物联网,mcu,硬件工程
添加完数据保存关闭,打开上位机,就可以看到我们所需要的数据,如图
STM32无线透传、PID调试工具,stm32,嵌入式硬件,单片机,51单片机,物联网,mcu,硬件工程
上位机连接透传模块前,需要把T30S用数据线连接到电脑,T22S与单片机串口连接,然后如图选择对应的串口号,无线电变为绿色表示连接成功。
STM32无线透传、PID调试工具,stm32,嵌入式硬件,单片机,51单片机,物联网,mcu,硬件工程

数据上传:这里用一个GPS来测试,把GPS数据发送到上位机观察,GPS解析如下,解析后赋值给透传的结构体,另外把生成一个0-100的随机数赋值给Ki:

/************************
* 解包GPS数据
*
*************************/
void unPkg_GPS_task(void)
{
	char strtemp[50];
	static u8 t = 0;
	u8 i;
	u8 length;
	short tmp;

	if(New_GPS_Packet==1)
	{
		New_GPS_Packet = 0;
		N_1ms_GPS_Watch = 0;
	 //$GPGGA,050029.00,3202.2762,N,11848.9585,E,1,04,1.8,12.42,M,1.92,M,,*67
		if((gps_head[0]=='G')&&(gps_head[1]=='G')&&(gps_head[2]=='A')) //GGA
		{
			p_stdio = 0;
			if(getstr(Gpsstr)>0)
			{
				//EE_ParaData.name.GpsSol_iTow =Gpsstr[5]-'0'; 	
			}					
			if(getstr(strtemp)>0)
			{
				GPS_Latitude = getGPS_degree(strtemp);	
				EE_ParaData.name.GPS_avail = 1;
			}		
			else
			{
				EE_ParaData.name.GPS_avail =0;
			}		
			EE_ParaData.name.GPS_Latitude = GPS_Latitude;
			getstr(strtemp); // 'N'	            
			if(getstr(strtemp)>0)
				GPS_Longitude = getGPS_degree(strtemp);  
			EE_ParaData.name.GPS_Longitude = GPS_Longitude;
			getstr(strtemp); //'w'               
			if(getstr(strtemp)>0)
				GPS_avail = atoi(strtemp);     
			if(getstr(strtemp)>0)
				GPS_starnum = atoi(strtemp);  //12
			EE_ParaData.name.GPS_StarNum = GPS_starnum;
			if(getstr(strtemp)>0)
				GPSHdop = atof(strtemp);    //15	
			if(getstr(strtemp)>0)
				GPS_Altitude = atof(strtemp);  //13
			EE_ParaData.name.GPS_Altitude = GPS_Altitude;
		}			
		if((gps_head[0]=='G')&&(gps_head[1]=='S')&&(gps_head[2]=='A')) //GSA
		{
			p_stdio = 0;
			getstr(strtemp); // 模式
			getstr(strtemp); // 定位类型
			
			for(i=0;i<12;i++)
			{
				StarID[i] = 0xfe;
				if(getstr(strtemp)>0)
				StarID[i] = atoi(strtemp);
			}
			if(getstr(strtemp)>0)
				Pdop = atof(strtemp);    	
			if(getstr(strtemp)>0)
				Hdop = atof(strtemp);    	
			if(getstr(strtemp)>0)
				Vdop = atof(strtemp); 
		    EE_ParaData.name.GPS_Pdop = 	Pdop;
		}	
//#BESTVELA,COM1,0,75.0,FINESTEERING,729,18045.000,80000000,b748,1250;SOL_COMPUTED,DOPPLER_VELOCITY,0.150,0.000,0.0562,316.726716,0.1090,0.0*48f920de
		if((gps_head[0]=='S')&&(gps_head[1]=='T')&&(gps_head[2]=='V')&(gps_head[3]=='E')&&(gps_head[4]=='L'))
		{
			p_stdio = 0;
			t++;
			if(t>=5)
			{
			//	Beep();
				t = 0;
			}
			while(str[p_stdio++] !=';')	  // 不处理分号前数据
			{
				if(p_stdio>=150) return;
			}
			getstr(strtemp); // get 	"SOL_COMPUTED"
			getstr(strtemp); // get 	DOPPLER_VELOCITY
			getstr(strtemp);	// get latency
			getstr(strtemp); // get age

			if(getstr(strtemp)>0)
				GPSSpeed[0] = atof(strtemp);
			if(getstr(strtemp)>0)
				GPSSpeed[1] = atof(strtemp);
			if(getstr(strtemp)>0)
				GPSSpeed[2] = atof(strtemp); 
			tmp= (short)(100*GPSSpeed[0]*cos(GPSSpeed[1]*deg_rad));
		}
	}
	length = length;
}

数据观测与上位机使用,可以看到GPS数据已经上传到上位机,数据区与数据观测区是一个实时动态的数据,非常便于观测
STM32无线透传、PID调试工具,stm32,嵌入式硬件,单片机,51单片机,物联网,mcu,硬件工程

数据修改,点击1位置弹出edit修改框,在修改值中输入我们需要的数值,例如这里输入5,然后点击保存
STM32无线透传、PID调试工具,stm32,嵌入式硬件,单片机,51单片机,物联网,mcu,硬件工程
Kp变为5说明写入成功
STM32无线透传、PID调试工具,stm32,嵌入式硬件,单片机,51单片机,物联网,mcu,硬件工程
我们到单片机上看仿真的数据也已经被修改
STM32无线透传、PID调试工具,stm32,嵌入式硬件,单片机,51单片机,物联网,mcu,硬件工程
四、IAP功能
IAP,全名为in applacation programming,即在应用编程,与之相对应的叫做isp,in system programming,在系统编程,两者的不同是isp需要依靠烧写器在单片机复位离线的情况下编程,需要人工的干预,而iap则是用户自己的程序在运行过程中对User Flash 的部分区域进行烧写,目的是为了在产品发布后可以方便地通过预留的通信口对产品中的固件程序进行更新升级。在工程应用中经常会出现我们的产品被安装在某个特定的机械结构中,更新程序的时候拆机很不方便,使用iap技术能很好地降低工作量。不懂原理的小伙伴自行搜索哟。如图依次点击,选择bin文件,最后点击编程即可烧录到单片机。这部分程序后续会上传,BOOT和APP文件。
STM32无线透传、PID调试工具,stm32,嵌入式硬件,单片机,51单片机,物联网,mcu,硬件工程

五、总结
该上位机配合E22透传模块使用,传输距离远,数据修改和观测数据非常方便,对无人车、无人机PID调试可以大大提高效率,PID调试时可以实时观测实际值与目标值的变化来快速整定PID参数,当然对其他的调试如需远程观察数据的项目来说也很方便。文章来源地址https://www.toymoban.com/news/detail-848073.html

到了这里,关于STM32无线透传、PID调试工具的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【嵌入式学习笔记】嵌入式基础9——STM32启动过程

    程序段交叉引用关系(Section Cross References):描述各文件之间函数调用关系 删除映像未使用的程序段(Removing Unused input sections from the image):描述工程中未用到被删除的冗余程序段(函数/数据) 映像符号表(Image Symbol Table):描述各符号(程序段/数据)在存储器中的地址、类

    2024年02月15日
    浏览(87)
  • 嵌入式 STM32 通讯协议--MODBUS

    目录 一、自定义通信协议 1、协议介绍 2、网络协议 3、自定义的通信协议  二、MODBUS通信协议 1、概述 2、MODBUS帧结构  协议描述 3、MODBUS数据模型   4、MODBUS事务处理的定义 5、MODBUS功能码  6、功能码定义   7、MODBUS数据链路层 8、MODBUS地址规则  9、MODBUS帧描述 10、MODBUS两种

    2024年02月11日
    浏览(66)
  • STM32的时钟系统(嵌入式学习)

    时钟是指用于计量和同步时间的装置或系统。时钟是嵌入式系统的脉搏,处理器内核在时钟驱动下完成指令执行,状态变换等动作,外设部件在时钟的驱动下完成各种工作,例如:串口数据的发送、AD转换、定时器计数等。因此时钟对于计算机系统是至关重要的,通常时钟系

    2024年02月16日
    浏览(50)
  • 嵌入式——新建STM32工程(标准库)

    目录 一、初识标准库 1.CMSIS标准及库层级关系 2.库文件介绍 (1)Libraries文件夹 ①CMSIS文件夹 ②STM32F10x_Std_Periph_Driver文件夹 ③ 在用库建立一个完整的工程时,还需要添加stm32f10x_it.c、 stm32f10x_conf.h 和 system_stm32f10x.c文件 (2)Project文件夹 (3)Utilities文件夹 3.库各文件之间的关

    2024年01月23日
    浏览(59)
  • STM32串口通信详解(嵌入式学习)

    时钟信号在电子领域中是指用于同步和定时电路操作的周期性信号。它在数字系统和通信系统中起着至关重要的作用,用于协调各个组件之间的数据传输和操作。 时钟信号有以下几个重要的方面: 频率:时钟信号的频率是指单位时间内信号周期的数量。它通常以赫兹(Hz)为

    2024年02月09日
    浏览(69)
  • stm32毕设分享 Stm32酒驾检查系统 - 单片机 嵌入式 物联网

    🔥 这两年开始毕业设计和毕业答辩的要求和难度不断提升,传统的毕设题目缺少创新和亮点,往往达不到毕业答辩的要求,这两年不断有学弟学妹告诉学长自己做的项目系统达不到老师的要求。 为了大家能够顺利以及最少的精力通过毕设,学长分享优质毕业设计项目,今天

    2024年01月23日
    浏览(76)
  • stm32毕设分享 stm32实现车牌识别系统 -物联网 嵌入式 单片机

    🔥 这两年开始毕业设计和毕业答辩的要求和难度不断提升,传统的毕设题目缺少创新和亮点,往往达不到毕业答辩的要求,这两年不断有学弟学妹告诉学长自己做的项目系统达不到老师的要求。 为了大家能够顺利以及最少的精力通过毕设,学长分享优质毕业设计项目,今天

    2024年02月20日
    浏览(63)
  • 嵌入式学习笔记——STM32的时钟树

    在之前的所有代码编程的过程中,似乎每次都绕不开一个叫做时钟使能的东西,当时我们是在数据手册上直接看其挂接在那条时钟线上的,那么STM32内部的时钟到底是怎么一个构型呢,本文来对此做一个介绍。 老规矩,一个新的名词出现,首先需要搞清楚它是个啥,下图中对

    2024年02月02日
    浏览(58)
  • STM32的中断系统详解(嵌入式学习)

    中断是处理器中的一种机制,用于响应和处理突发事件或紧急事件。当发生中断时,当前正在执行的程序会被暂时中止,处理器会跳转到中断处理程序(也称为中断服务例程),对中断事件进行处理。处理完中断后,处理器再返回到被中断的程序继续执行。 中断可以分为内部

    2024年02月12日
    浏览(75)
  • 嵌入式C语言基础(STM32)

    前言:一条混迹嵌入式3年的老咸鱼,想到自己第一次接触到stm32的库函数时,c语言稀碎,痛不欲生的场景,该文章为萌新指条明路。 位操作在嵌入式中常用于直接对芯片的寄存器进行操作,当时作为初学者的我看着一脸懵逼,至于为什么这样修改,下面好好分析一下。  一

    2024年02月02日
    浏览(59)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包