非常好用的上位机软件(功能强大)——匿名四轴上位机

Posted 爱写代码的猫

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了非常好用的上位机软件(功能强大)——匿名四轴上位机相关的知识,希望对你有一定的参考价值。

 目录

文章目录

前言

一、什么是上位机

二、匿名四轴上位机

1、功能

2、软件的一些简单协议

三、相关的接收发送代码

一、接收数据

2.发送数据

总结




前言

对于单片机开发者,调试工具就必不可少,有时需要显示波形、发送文本、数据和一些复杂的数据包。例如PID参数整定,然而四轴匿名上位都有这些功能。本文就介绍匿名四轴上位机怎么显示波形和调试,以及一些接收发送代码。
 


一、什么是上位机

上位机是指可以直接发出操控命令的计算机,一般是PC/host computer/master computer/upper computer,屏幕上显示各种信号变化(液压,水位,温度等)。下位机是直接控制设备获取设备状况的计算机,一般是PLC/单片机single chip microcomputer/slave computer/lower computer之类的。上位机发出的命令首先给下位机,下位机再根据此命令解释成相应时序信号直接控制相应设备。下位机不时读取设备状态数据(一般为模拟量),转换成数字信号反馈给上位机。简言之如此,实际情况千差万别,但万变不离其宗:上下位机都需要编程,都有专门的开发系统。

在概念上,控制者和提供服务者是上位机,被控制者和被服务者是下位机,也可以理解为主机和从机的关系,但上位机和下位机是可以转换的。

二、匿名四轴上位机

1、功能

这个上位机可以基本收发(类似于串口调试助手)、高级收发收码(实现比较复杂的自定义接收和发送)、同时显示20条波形图(用于调试pid等)、调试无人机可以监视无人机的各个状态以及调试pid。

四轴匿名上位机下载链接:
http://链接:https://pan.baidu.com/s/1l6kT-LQCMQRySXpp2lrh7w 提取码:C5LQ

2、软件的一些简单协议

  一:基本收发
1:收码和发码格式均可设为HEX或者CHR。
2:定时发送功能可以精确到毫秒,但是不能太快(发送为独占式,数据不发送完函数不会返回),如果上一帧
      数据还没发送完毕就发送下一帧数据会出错。
3:请使用ft232串口芯片或支持高波特率的芯片,否则波特率无法设置过高。

    二:高级收码
1:收码显示为HEX格式。
2:下位机发送自定义数据,格式为:0x88+FUN+LEN+DATA+SUM
      FUN可以是 0xA1到0xAA,共10个;LEN为DATA的长度(不包括0x88、FUN、LEN、SUM)。
      SUM是0x88一直到DATA最后一字节的和,uint8格式。
    (记得打开需要使用帧的开关,更改设置后点击保存设置使设置生效)
3:数据可以是uint8、int16、uint16、int32、float这几个常用格式,多字节数据高位在前。
4:共有20个数据存储器,每个存储器的数据可以分别设置为来自10个自定义帧的30个数据。
5:高速通讯时(2ms一帧数据或者更快),请关闭高级收码页面的数据显示按钮和基本收码,否则更新过快有可
      能会造成程序卡死。
6:飞控显示对应的帧FUN为0xAF,(帧格式:0x88+0xAF+0x1C+ACC DATA+GYRO DATA+MAG DATA+ANGLE DATA
      + 0x00 0x00 + 0x00 0x00+SUM,共32字节,ACC/GYRO/MAG/ANGLE(roll/pitch/yaw)数据为int16格式,其
      中ANGLE的roll和pitch数据为实际值乘以100以后得到的整数值,yaw为乘以10以后得到的整数值,
      上位机在显示时再 除以100和10)。
7:遥控,电机pwm,电压显示对应的帧FUN为0xAE,(帧格式:0x88+0xAE+0x12+THROT YAW ROLL PITCH 
      +AUX1 2 3 4 5 + PWM:1 2 3 4 + VOTAGE + SUM,共28字节),数据为uint16格式,遥控数据最小在1000左右,
      最大在2000左右。数据都为uint16格式,其中pwm范围1-100,votage为实际值*100。
      小技巧:如果高速通讯时是为了画波形,就只开波形显示,并只保留需要观察的波形,如果是为了观察数
      据,就关闭波形显示,只保留收码显示,这样可以加快程序响应速度。
7:最快通讯速度测试过下位机用500K波特率,每1ms发送32字节的数据,上位机显示其中6条波形,OK!
    (有可能和电脑配置有关)

    三:波形显示
1:共有20条波形,对应20个数据存储器。
2:双击波形绘制区域,可以打开波形显示开关。
3:按住Ctrl用鼠标左键点击某一条波形,可以显示数据标签,再次点击隐藏。
4:按住鼠标左键,在绘图区域从一点向右下方拖动,然后松开,可以放大显示框住的波形区域,可以多次放
      大;
5:按住鼠标左键,在绘图区域从一点向左上方拖动,然后松开,可以将放大后的波形还原。
6:按住鼠标右键,在绘图区域上下左右拖动,可以移动波形。
7:显示波形时按F9键,可以打开波形高级设置。

    四:DEBUG功能
1:在调试过程中可以将某些标志位、寄存器、变量实时发回上位机,并在DEBUG页面显示。
2:通讯格式为:0x88 + 0xAD + len + num + DATA + SUM,  len为num与DATA的总长度,num表示要改变哪个显示
      状态,例如num=0x01即是要改变第一个LED,num=0x07即是改变第一个数字输出显示。当要改变LED时,
      DATA只需一字节,DATA=0x00表示关闭LED,大于0x00表示点亮LED;当要改变数字输出时,DATA需要两字
节,
      表示 一个uint16数字,高字节在前。SUM为从0x88开始到SUM前一字节的和校验,uint8格式。
      例如:发送 0x88 + 0xAD + 0x02 + 0x01 + 0x01 + 0x39    表示点亮第一个LED
                  发送 0x88 + 0xAD + 0x03 + 0x07 + 0x00 + 0x05 + 0x44   表示在第一个数字输出位置显示 5 。

    五:键鼠控制
1:控制数据发送格式为:0x8A + 0x8A + 0X1C + THROT YAW ROLL PITCH AUX1 AUX2 AUX3 AUX4 AUX5 + 0x00 0x00 
      0x00 0x00 0x00 0x00 0x00 0x00 0x00 0x00 + SUM,遥控数据都为int16格式,中值1500,最小最大值为1000、
      2000。
2:发送频率 50Hz。
3:鼠标上下控制油门,左右控制YAW,键盘的WASD控制ROLL/PITCH,键盘12345控制AUX12345。共9通道。

    六:飞控参数
1:点击3D模型显示右下方的校正按钮,上位机会发送0X8A 0X8B 0X1C 0XAA 0XA3 +无用数据+SUM给下位机,
      其中0X8B表示飞控参数,0XAA表示零偏,0XA3表示ACC GYRO都要校正。
2:点击飞控参数界面传感器矫正功能里面的校正按钮,分别表示ACC 和GYRO的校正,不会同时校正两个传感
器,
      上位机校正ACC发送格式为:0X8A 0X8B 0X1C 0XAA 0XA1+无用数据+SUM
      上位机校正GYRO发送格式为:0X8A 0X8B 0X1C 0XAA 0XA2+无用数据+SUM(送有数据整个长度都为32字节)
3:上位机发送微调后的offset数据(仅为ACC的 X 和 Y ),格式为0X8A 0X8B 0X1C 0XAB + offset.x + offset.y
      +无用数据 +SUM,数据为int16格式。
4:上位机发送读offset的命令格式为:0X8A 0X8B 0X1C 0XAC + 无用数据 + SUM
5:上位机发送读PID数据的命令为:0X8A 0X8B 0X1C 0XAD + 无用数据 + SUM
6:下位机发送offset数据给上位机的格式为:0X88 0XAC 0X1C 0XAC + 传感器零偏数据ACC XYZ GYRO XYZ
      +无用数据+SUM,共六个int16型数据。
7:下位机发送PID数据给上位机的格式为:0X88 0XAC 0X1C 0XAD + PID数据 + 无用数据 + SUM
      PID数据为rol_p,rol_i,rol_d,pit_p..i..d,yaw_p,,i,,d,共9个uint16型数据。
8:上位机发送PID数据给下位机的格式为:0X8A 0X8B 0X1C 0XAE +PID数据+ 无用数据 + SUM
      PID数据格式和下位机发送给上位机的格式一样。
9:点击飞控解锁按钮,上位机会发送0X8A 0X8B 0X1C 0XA1+无用数据+SUM给下位机,如果下位机已经解锁,
      点击此按钮会发送0X8A 0X8B 0X1C 0XA0+无用数据+SUM给下位机,锁定飞控。

三、相关的接收发送代码


一、接收数据

1.接收数据流

若接收到的数据包是按协议发送的,则将数据存入RX_upcomputer数组中。

//数据的接收
#define E_START                    0   //准备成功
#define E_OK                       1   //成功
#define E_FRAME_HEADER_ERROR       2   //帧头错误
#define E_FRAME_RTAIL_ERROR        3   //帧尾错误
#define LINE_LIN_up                   32  //数据长度
uint8_t uart_flag_up;                     //接收标志

vu8 RX_upcomputer[33];
void get_upcomputer_data(uint8_t data)
{
	static uint8_t uart_num=0;
	vu8 sum=0;
    vu8 i=0;
    RX_upcomputer[uart_num++]=data;
    if(1==uart_num)
    {
        //接收到的第一个字节不为0X8A,帧头错误
        if(0X8A!=RX_upcomputer[0])
        {
            uart_num=0;
            uart_flag_up=E_FRAME_HEADER_ERROR;
        }
    }
		if(2==uart_num)
    {
        //接收到的第二个字节不为0x8B,帧头错误
        if(0X8B!=RX_upcomputer[1])
        {
            uart_num=0;
            uart_flag_up=E_FRAME_HEADER_ERROR;
        }
    }
		if(3==uart_num)
    {
        //接收到的第三个字节不为0X1C,帧头错误
        if(0X1C!=RX_upcomputer[2])
        {
            uart_num=0;
            uart_flag_up=E_FRAME_HEADER_ERROR;
        }
    }
		
    if(LINE_LIN_up==uart_num)
    {
        uart_flag_up=E_OK;
        //接收到的最后一个字节为校验位
			for(i=0;i<LINE_LIN_up-1;i++)
			sum += RX_upcomputer[i];        //校验位

        if(sum==RX_upcomputer[LINE_LIN_up-1])
            uart_flag_up=E_OK;
        else                                //接收的最后一个字节校验位错误,帧尾错误
            uart_flag_up=E_FRAME_RTAIL_ERROR;
        uart_num=0;
    }
	
}

2.解析数据

解析存在 RX_upcomputer数组中的数据。

extern float Out_XP,Out_XI,Out_XD,In_XP,In_XI,In_XD,Out_YP,Out_YI,Out_YD,In_YP,In_YI,In_YD,ZP,ZI,ZD;
extern float out_zp;
//解析上位机发送的pid数据
void upcomputer_receive1(void)
{
	
	In_XP=((int)RX_upcomputer[4]<<8)|RX_upcomputer[5];
	In_XP=In_XP/100;
	In_XI=((int)RX_upcomputer[6]<<8)|RX_upcomputer[7];
	In_XI=In_XI/100;
	In_XD=((int)RX_upcomputer[8]<<8)|RX_upcomputer[9];
	In_XD=In_XD/100;
	out_zp=((int)RX_upcomputer[10]<<8)|RX_upcomputer[11];
	out_zp=out_zp/100;
	In_YI=((int)RX_upcomputer[12]<<8)|RX_upcomputer[13];
	In_YI=In_YI/100;
	In_YD=((int)RX_upcomputer[14]<<8)|RX_upcomputer[15];
	In_YD=In_YD/100;
	ZP=((int)RX_upcomputer[16]<<8)|RX_upcomputer[17];
	ZP=ZP/100;
	ZI=((int)RX_upcomputer[18]<<8)|RX_upcomputer[19];
	ZI=ZI/100;
	ZD=((int)RX_upcomputer[20]<<8)|RX_upcomputer[21];
	ZD=ZD/100;
}


//解析上位机微调后的offset
int offset_x,offset_y;
void upcomputer_receive2(void)
{
	offset_x=((int)RX_upcomputer[4]<<8)|RX_upcomputer[5];
	
	offset_y=((int)RX_upcomputer[6]<<8)|RX_upcomputer[7];
	
}


2.发送数据

void usart1_SendByte(u8 data)
{
    /* 发送字节数据到UART */
    USART_SendData(USART1,data);

    /*等待发送寄存器为空 */
    while (USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET);
}

/****************** 发送八位数组************************/
void usart1_SendArray(vu8 *array, uint16_t num)
{
    vu8 i;

  for(i=0; i<num; i++)
  {
        /* ·发送一个字节数据到USART */
     usart1_SendByte(array[i]);
  }
    /* 等待发送完成 */
  while(USART_GetFlagStatus(USART1,USART_FLAG_TC)==RESET);
}


vu8 testdatatosend[50];
/*波形显示函数*/
void Test_Send_User(uint16_t data1, uint16_t data2, uint16_t data3,uint16_t data4,uint16_t data5,uint16_t data6,uint16_t data7,uint16_t data8,uint16_t data9,uint16_t data10)
{
    vu8 _cnt=0;
    vu8 sum=0;
    vu8 i=0;

    testdatatosend[_cnt++]=0x88;
    testdatatosend[_cnt++]=0xA1;
    testdatatosend[_cnt++]=0;

    testdatatosend[_cnt++]=data1>>8;     
    testdatatosend[_cnt++]=data1&0xff;   

    testdatatosend[_cnt++]=data2>>8;
    testdatatosend[_cnt++]=data2&0xff;

    testdatatosend[_cnt++]=data3>>8;
    testdatatosend[_cnt++]=data3&0xff;

    testdatatosend[_cnt++]=data4>>8;
    testdatatosend[_cnt++]=data4&0xff;

    testdatatosend[_cnt++]=data5>>8;
    testdatatosend[_cnt++]=data5&0xff;

    testdatatosend[_cnt++]=data6>>8;
    testdatatosend[_cnt++]=data6&0xff;

    testdatatosend[_cnt++]=data7>>8;
    testdatatosend[_cnt++]=data7&0xff;

    testdatatosend[_cnt++]=data8>>8;
    testdatatosend[_cnt++]=data8&0xff;

    testdatatosend[_cnt++]=data9>>8;
    testdatatosend[_cnt++]=data9&0xff;

    testdatatosend[_cnt++]=data10>>8;
    testdatatosend[_cnt++]=data10&0xff;



    testdatatosend[2] = _cnt-3;

    for(i=0;i<_cnt;i++)
        sum += testdatatosend[i];

    testdatatosend[_cnt++]=sum;

    usart1_SendArray(testdatatosend,_cnt);
}





/*****************************************************************************************
*下面的函数主要用于无人机调试,也可用于其他pid调试
*
***************************************************************************************/

/*发送陀螺仪数据给上位机*/
void Test_Send_User1(int16_t acc_x_,int16_t acc_y_,int16_t acc_z_,int16_t gyro_x_,int16_t gyro_y_,int16_t gyro_z_,int16_t roll_,int16_t pitch_,int16_t yaw_)
{
	 vu8 _cnt=0;
    vu8 sum=0;
    vu8 i=0;

    testdatatosend[_cnt++]=0x88;
    testdatatosend[_cnt++]=0xAF;
    testdatatosend[_cnt++]=0x1C;

    testdatatosend[_cnt++]=acc_x_>>8;     //
    testdatatosend[_cnt++]=acc_x_&0xff;   //

    testdatatosend[_cnt++]=acc_y_>>8;
    testdatatosend[_cnt++]=acc_y_&0xff;

    testdatatosend[_cnt++]=acc_z_>>8;
    testdatatosend[_cnt++]=acc_z_&0xff;

    testdatatosend[_cnt++]=gyro_x_>>8;
    testdatatosend[_cnt++]=gyro_x_&0xff;

    testdatatosend[_cnt++]=gyro_y_>>8;
    testdatatosend[_cnt++]=gyro_y_&0xff;

    testdatatosend[_cnt++]=gyro_z_>>8;
    testdatatosend[_cnt++]=gyro_z_&0xff;
		
		testdatatosend[_cnt++]=0;
		testdatatosend[_cnt++]=0;
		
		testdatatosend[_cnt++]=0;
		testdatatosend[_cnt++]=0;
		
		testdatatosend[_cnt++]=0;
		testdatatosend[_cnt++]=0;

    testdatatosend[_cnt++]=roll_>>8;
    testdatatosend[_cnt++]=roll_&0xff;

    testdatatosend[_cnt++]=pitch_>>8;
    testdatatosend[_cnt++]=pitch_&0xff;

    testdatatosend[_cnt++]=yaw_>>8;
    testdatatosend[_cnt++]=yaw_&0xff;

    testdatatosend[_cnt++]=0x00;
    testdatatosend[_cnt++]=0x00;
		testdatatosend[_cnt++]=0x00;
    testdatatosend[_cnt++]=0x00;

    testdatatosend[2] = _cnt-3;

    for(i=0;i<_cnt;i++)
        sum += testdatatosend[i];

    testdatatosend[_cnt++]=sum;

    usart1_SendArray(testdatatosend,_cnt);
	
}




/*发送电机pwm和飞控电池电压给上位机*/
void Test_Send_User2(uint16_t throt,uint16_t yaw_2,uint16_t roll_2,uint16_t pitch_2,uint16_t aux_1,uint16_t aux_2,uint16_t aux_3,uint16_t aux_4,uint16_t aux_5,uint16_t pwm1,uint16_t pwm2,uint16_t pwm3,uint16_t pwm4,uint16_t votage)
{
	
	vu8 _cnt=0;
    vu8 sum=0;
    vu8 i=0;

    testdatatosend[_cnt++]=0x88;
    testdatatosend[_cnt++]=0xAE;
    testdatatosend[_cnt++]=0x12;
	
	  testdatatosend[_cnt++]=throt>>8;
    testdatatosend[_cnt++]=throt&0xff;

    testdatatosend[_cnt++]=yaw_2>>8;     //
    testdatatosend[_cnt++]=yaw_2&0xff;   //

    testdatatosend[_cnt++]=roll_2>>8;
    testdatatosend[_cnt++]=roll_2&0xff;

    testdatatosend[_cnt++]=pitch_2>>8;
    testdatatosend[_cnt++]=pitch_2&0xff;

    testdatatosend[_cnt++]=aux_1>>8;
    testdatatosend[_cnt++]=aux_1&0xff;

    testdatatosend[_cnt++]=aux_2>>8;
    testdatatosend[_cnt++]=aux_2&0xff;

    testdatatosend[_cnt++]=aux_3>>8;
    testdatatosend[_cnt++]=aux_3&0xff;

    testdatatosend[_cnt++]=aux_4>>8;
    testdatatosend[_cnt++]=aux_4&0xff;

    testdatatosend[_cnt++]=aux_5>>8;
    testdatatosend[_cnt++]=aux_5&0xff;

    testdatatosend[_cnt++]=pwm1>>8;
		testdatatosend[_cnt++]=pwm1&0xff;

    testdatatosend[_cnt++]=pwm2>>8;
		testdatatosend[_cnt++]=pwm2&0xff;
		
    testdatatosend[_cnt++]=pwm3>>8;
		testdatatosend[_cnt++]=pwm3&0xff;

    testdatatosend[_cnt++]=pwm4>>8;
		testdatatosend[_cnt++]=pwm4&0xff;
		
		testdatatosend[_cnt++]=votage>>8;
    testdatatosend[_cnt++]=votage&0xff;

    testdatatosend[2] = _cnt-3;

    for(i=0;i<_cnt;i++)
        sum += testdatatosend[i];

    testdatatosend[_cnt++]=sum;

    usart1_SendArray(testdatatosend,_cnt);
	
}



/*发送OFFSET给上位机*/
void Test_Send_User3(int16_t acc_x_3,int16_t acc_y_3,int16_t acc_z_3,int16_t gyro_x_3,int16_t gyro_y_3,int16_t gyro_z_3)
{
	vu8 _cnt=0;
    vu8 sum=0;
    vu8 i=0;

    testdatatosend[_cnt++]=0x88;
    testdatatosend[_cnt++]=0xAC;
    testdatatosend[_cnt++]=0x1C;
	  testdatatosend[_cnt++]=0xAC;
	
	  testdatatosend[_cnt++]=acc_x_3>>8;
    testdatatosend[_cnt++]=acc_x_3&0xff;

    testdatatosend[_cnt++]=acc_y_3>>8;     //
    testdatatosend[_cnt++]=acc_y_3&0xff;   //

    testdatatosend[_cnt++]=acc_z_3>>8;
    testdatatosend[_cnt++]=acc_z_3&0xff;

    testdatatosend[_cnt++]=gyro_x_3>>8;
    testdatatosend[_cnt++]=gyro_x_3&0xff;

    testdatatosend[_cnt++]=gyro_y_3>>8;
    testdatatosend[_cnt++]=gyro_y_3&0xff;

    testdatatosend[_cnt++]=gyro_z_3>>8;
    testdatatosend[_cnt++]=gyro_z_3&0xff;

    //testdatatosend[2] = _cnt-3;

    for(i=0;i<_cnt;i++)
        sum += testdatatosend[i];

    testdatatosend[31]=sum;

    usart1_SendArray(testdatatosend,32);
	
	
}


//给上位机发送PID参数*/
void Test_Send_User4(uint16_t rol_p,uint16_t rol_i,uint16_t rol_d,uint16_t pit_p,uint16_t pit_i,uint16_t pit_d,uint16_t yaw_p,uint16_t yaw_i,uint16_t yaw_d)
{
	vu8 _cnt=0;
    vu8 sum=0;
    vu8 i=0;

    testdatatosend[_cnt++]=0x88;
    testdatatosend[_cnt++]=0xAC;
    testdatatosend[_cnt++]=0x1C;
	  testdatatosend[_cnt++]=0xAD;
	
	  testdatatosend[_cnt++]=rol_p>>8;
    testdatatosend[_cnt++]=rol_p&0xff;

    testdatatosend[_cnt++]=rol_i>>8;     //
    testdatatosend[_cnt++]=rol_i&0xff;   //

    testdatatosend[_cnt++]=rol_d>>8;
    testdatatosend[_cnt++]=rol_d&0xff;

    testdatatosend[_cnt++]=pit_p>>8;
    testdatatosend[_cnt++]=pit_p&0xff;

    testdatatosend[_cnt++]=pit_i>>8;
    testdatatosend[_cnt++]=pit_i&0xff;

    testdatatosend[_cnt++]=pit_d>>8;
    testdatatosend[_cnt++]=pit_d&0xff;

    testdatatosend[_cnt++]=yaw_p>>8;
    testdatatosend[_cnt++]=yaw_p&0xff;

    testdatatosend[_cnt++]=yaw_i>>8;
    testdatatosend[_cnt++]=yaw_i&0xff;

    testdatatosend[_cnt++]=yaw_d>>8;
    testdatatosend[_cnt++]=yaw_d&0xff;

   // testdatatosend[2] = _cnt-3;

    for(i=0;i<_cnt;i++)
        sum += testdatatosend[i];

    testdatatosend[31]=sum;

    usart1_SendArray(testdatatosend,32);
	
	
}



总结

以上就是今天讲的内容,四轴匿名上位机的使用、协议和一些接收发送代码。

这是我的学习总结,若有不当之处,欢迎指出。

以上是关于非常好用的上位机软件(功能强大)——匿名四轴上位机的主要内容,如果未能解决你的问题,请参考以下文章

基于Arduino平台dmp库+匿名四轴上位机测试MPU6050

匿名上位机的使用(51版)

单片机的上位机开发用啥软件好?

怎样用labview编写上位机软件?

怎样用vc编写上位机软件

匿名上位机v2.6和V7自定义帧代码和飞控姿态代码