STM32读取LX-224总线舵机信息

一、舵机指令包格式

帧头: 连续收到两个 0x55 ,表示有数据包到达。
ID: 每个舵机都有一个 ID 号。ID 号范围 0~253,转换为十六进制 0x00~0xFD。广播 ID: ID 号 254(0xFE) 为广播 ID,若控制器发出的 ID 号为 254(0xFE),所有的舵机均接收指令,但都不返回应答信息,(读取舵机 ID 号除外,具体说明参见下面指令介绍)以防总线冲突。
数据长度: 等于待发送的数据(包含本身一个字节)长度,即数据长度 Length加 3 等于这一包指令的长度,从帧头到校验和。
指令: 控制舵机的各种指令,如位置、速度控制等。
参数: 除指令外需要补充的控制信息。
校验和: 校验和 Checksum,计算方法如下:
Checksum = ~ (ID + Length + Cmd+ Prm1 + … Prm N)若括号内的计算和超出 255,则取最低的一个字节,“~”表示取反。

 二、74HC126D芯片介绍

1、定义:

        74HC126D是一款四缓冲器/线路驱动器集成电路(IC),属于74系列芯片。该芯片具有四个独立的缓冲器/线路驱动器,可以同时处理四个信号输入,并将其转换为输出信号。采用SOP-14封装,具有较小的体积和较低的功耗,适合高密度集成和小型化设计。此外,该芯片具有高速度、低噪声和低功耗等优良性能,可以保证信号传输的稳定性和可靠性。因此,74HC126D芯片在各种电子设备和系统中得到广泛应用。

控制 TX_CON(控制发送):

  1. 将 TX_CON 拉低

    • 使用微控制器的GPIO输出功能,将一个引脚输出低电平(例如0V或接地)。
    • 这将导致74HC126D芯片上对应的 TX_CON 输入端被拉低,使得输出端 Y1 处于激活状态。
    • 微控制器通过串口 TX 发送数据时,数据信号将被74HC126D内部的放大器放大并传输到 Y1 输出端,驱动外部设备或传输线。

控制 RX_CON(控制接收):

  1. 将 RX_CON 拉高
    • 使用微控制器的GPIO输出功能,将一个引脚输出高电平(例如Vcc或逻辑高电平)。
    • 这将使得74HC126D芯片上对应的 RX_CON 输入端被拉高,导致输出端 Y2 进入高阻态。
    • 当外部设备通过串口 RX 发送数据时,数据信号能够直接进入微控制器的串口接收引脚,而不受74HC126D的影响或干扰。

硬件原理图:

2、74HC126D

74HC126D介绍:
功能描述:

可以看出,当OE输出高电平时 输入是高电平那么输出就是高电平,输入是低电平输出就是低电平。

当OE为低电平时,不管输入状态是什么,输出都是高阻抗关断状态(抽象理解为悬空)。

高阻输出一般是指数字电路输出时不为高电平或低电平,而是相当于断开的一种状态,输出点的电位由后面的电路决定。

这个芯片的作用就是,当需要写入的时候,拉低TX_CON,这样,串口TX发送什么,输出就是什么。拉高RX_CON,这样,串口接收RX就相当于悬空,什么也不干。接收数据也是如此.

三、程序解读

 STM32串口1发送指令,另一个串口3打印接收的数据

串口1发送指令:

void Serial_SendArray(uint8_t *Array, uint16_t Length)
{
	uint16_t i;
	GPIO_ResetBits(GPIOA, GPIO_Pin_11);      //TX_COn  
	GPIO_SetBits(GPIOA, GPIO_Pin_12);		 //RX_COn	74HC126D接收状态
	for (i = 0; i < Length; i ++)
	{
		Serial_SendByte(Array[i]);
	}
	if(i == Length)
	{
		GPIO_SetBits(GPIOA, GPIO_Pin_11);    //74HC126D发送状态
		GPIO_SetBits(GPIOA, GPIO_Pin_12);
		
	}
}
//读取舵机信息
uint16_t LobotSerialServoRead(uint8_t id, uint8_t value)
{
  uint16_t ret;
 

  buf[0] = buf[1] = LOBOT_SERVO_FRAME_HEADER;
  buf[2] = id;
  buf[3] = 3;
  buf[4] = value;
  buf[5] = LobotCheckSum(buf);
	
  LobotSerialWrite(buf, 6);  //发送获取位置的命令
	
  ret = LobotSerialMsgHandle(); //获取接收到的数据
	
//	USART2_Send_Data(ret);
//	USART2_Send_Data(LobotRxBuf[LobotRxBuf[3]+2]);

  return ret;
}

接收中断:

#define BUFFER_SIZE 20  // 假设缓冲区大小为20字节
#define FRAME_START_BYTE 0x55 // 帧起始字节

uint8_t rxBuf;;  // 定义接收数据的缓冲变量
static uint8_t frameHeaderCount = 0; 
static uint8_t dataLength = 0;  // 数据长度,默认为2
static uint8_t dataCount = 0;  // 数据计数器
static uint8_t RxState = 0;
static uint8_t pRxPacket = 0;
uint8_t frameHeaderBit = 0;

uint8_t rxBuffer[BUFFER_SIZE];
int rxIndex = 0;  // 缓冲区索引


void USART1_IRQHandler(void)
{
	if (USART_GetITStatus(USART1, USART_IT_RXNE) == SET)
	{
		USART_ClearITPendingBit(USART1, USART_IT_RXNE);

		rxBuf = USART_ReceiveData(USART1);

		if (!frameHeaderBit) {                    //判断连续两个帧头是否为0x55
			if (rxBuf == FRAME_START_BYTE) {
				frameHeaderCount++;
				rxBuffer[rxIndex++] = rxBuf;
				if(frameHeaderCount == 2){
				frameHeaderCount = 0;
				frameHeaderBit = 1;  //接收到完整的帧头
					
				rxIndex = 2;    //从第三个开始存数据
				}

			}
    }else{
        // 已经处于帧内
        rxBuffer[rxIndex++] = rxBuf;  //从索引3开始

        // 假设帧长度固定为7字节
        if (rxIndex >= 7) {
            // 检查帧头是否正确
            if (rxBuffer[0] == FRAME_START_BYTE && rxBuffer[1] == FRAME_START_BYTE) {
                // 打印整个接收到的数据帧
                printf("Received frame: ");
				
                for (int i = 0; i < rxIndex; i++) {
                    printf("%02X ", rxBuffer[i]);   //十六进制打印
                }
                printf("\n");


                // 复位缓冲区和状态标志,准备接收下一个帧
				
                rxIndex = 0;
                frameHeaderBit = 0;
				frameHeaderCount=0;
            } else {
                // 如果帧头不正确,丢弃整个缓冲区
                rxIndex = 0;
                frameHeaderBit = 0;
				frameHeaderCount=0;
            }
        }	
	}
  }
}

 校验和计算:

  1. 循环计算累加和:从buf数组的第三个元素开始累加,累加的长度由buf[3]决定,直到累加到指定长度。

  2. 取反操作:对累加和进行按位取反。

  3. 类型转换:将取反后的结果转换为uint8_t类型。

  4. 返回校验和:将转换后的结果作为函数的返回值,即为计算得到的校验和。

  5. 延时操作:调用延时函数Delay_ms(10),用于稳定处理过程。

//校验和
uint8_t checksum;

//缓冲区
extern uint8_t rxBuffer[BUFFER_SIZE];

 uint8_t buf[6];
//校验计算
uint8_t LobotCheckSum(uint8_t buf[])
{
  uint8_t i;
  uint8_t temp = 0;
  for (i = 2; i < buf[3] + 2; i++) 
	{
    temp += buf[i];
  }
  temp = ~temp;
  i = (uint8_t)temp;
  Delay_ms(10);
  return i;
}

调用校验和计算函数,获取校验和。通过判断校验和是否与 rxBuffer[3] + 2 位置的数据是否相等,来进行下一步的命令判断。

命令cmd位于 rxBuffer数组的索引5

uint16_t LobotSerialMsgHandle(void)
{
	uint8_t cmd;
	uint16_t ret;

	checksum = LobotCheckSum(rxBuffer);
	// 校验和验证
	if (checksum == rxBuffer[rxBuffer[3] + 2]) {
		uint8_t cmd;
		uint16_t ret;
//		printf("Checksum valid:%d\n\r",checksum);
		Delay_ms(1000);
		// 进一步处理数据
	cmd = rxBuffer[4]; //14
//	printf("cmd :%d\n",cmd);
	switch(cmd)
	{
		break;
		case LOBOT_SERVO_ID_READ:
			ret = rxBuffer[2];			//读取ID   
			return ret;
		break;
		default:
			break;
	}
	} 
	else {
		printf("Checksum invalid\n\r");
	}
	return 0;
}

串口打印情况:

前面两个0x55是帧头,第一个和第二个01都是舵机的id,04是数据的长度(从第一个01到第二个01,总共4个数据),OE是命令指令的十六进制(十进制为14),EB是校验和。

计算过程:

十六进制转十进制

01  ---- >  01

 04 ---- >  04

OE ---- >  14

EB ----->  235

从第三位累加到倒数第二位,即第一个01  --- > 第二个01

校验和 = ~(1+ 4 + 14 + 1 )=~(20)= 255 - 20 =235

由此可以验证数据正确 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值