一、AS5600介绍
AS5600是一个易于编程的磁性旋转位置传感器,具有高分辨率的12位模拟或PWM输出。这种非接触式系统测量一个直径磁化的轴上磁铁的绝对角度。
引脚如下图
他有两种供电模式:5V和3.3V
我们为了和stm32F103C8T6单片机的电压一致,也使用3.3V供电,然后开始画PCB。
二、pcb设计
使用嘉立创EDA画图
三、实物照片
已经把as5600贴到了电机后面,暂时没有稍微长一点的螺丝,不然用四个螺丝固定更好。
四、程序代码
代码实现的功能:
1、可以实时显示当前绝对位置的角度(0~360)
2、可以实时显示增量角度,比如正转了2转,显示为720度,又在此基础上反转了3转,显示为-360度。
3、一上电当前位置即可自动设置为初始零位。
3、也可以测实时的速度(暂时没有写这部分功能,实现也简单,两次位置差除以时间即可)
难点:as5600接在了电机尾部,步进电机的轴转5.18圈,减速器输出的轴转1圈,as5600直接读的是步进电机轴的位置(我这个步进电机带了减速器,减速比为5.18)所以还需要一些转换。
程序大概思路:
设置3个变量temp0(记录一上电之后电机轴的初始位置,在程序中只需运行一次即可),temp1(用于记录本次电机轴的位置),temp2(用于记录下一次电机轴的位置)。temp1和temp2主要用于过360度判断,即如果temp-temp2>180度(在很短的时间内),那么肯定是过了360度了,即下一转开始了,因为电机不可能在很短的时间转过这么多度。而且从程序初始化以后就要一直拿比较temp1和temp2,每次都不能少,比较完了就把temp2赋值给temp1,然后下次再获取最新的temp2以后,继续和temp1比较。就是要一直和上一次的数据比较,中间不能又一次断开,这样可以保证不漏过那个变化的点,每次过360度的时候,进行一个计数(程序中用过零点标记sign_angle计数),然后只要知道初始角度(temp0)、过零点的次数(sign_angle),以及当前点的角度(temp2),就可以算出角度增量。
程序代码如下:
下面是as5600.h文件
#ifndef __AS5600__
#define __AS5600__
#include "sys.h"
#include "stm32f10x.h"
#define Slave_Addr 0x36 //设备从地址
#define Write_Bit 0 //写标记
#define Read_Bit 1 //读标记
#define Angle_Hight_Register_Addr 0x0C //寄存器高位地址
#define Angle_Low_Register_Addr 0x0D //寄存器低位地址
#define Jian_Su_Bi 5.18 //步进电机减速比
void AS5600_Init ( void ); //初始化
u16 AS5600_Read_Len ( u8 addr, u8 reg, u8 len, u8 *buf );//从AS5600读取一次数据
void Get_Ini_Val(void); //得到上电后角度初始值
void Get_Temp_Add(void); //等到角度增量(原始值表示的)
void Change_angle(void); //将原始增量数据转化为角度
void Get_Num_sign(void); //用于过零点计数
下面是as5600.c文件
#include "as5600.h"
#include "i2c.h"
#include "delay.h"
u32 angle_ini = 0; //初始角度值
u32 temp0 = 0; //初始角度原始输出值
u32 temp1 = 0; //上次角度原始输出值
u32 temp2 = 0; //这次角度原始输出值
u32 temp_add = 0; //从初始角度开始的累计角度原始值
u8 buf[2] = {0}; //用于oled显示存放变量,和本程序关系不大
int sign_angle = 0; //过零点标记(即从0度转到360度之后继续转又回零的那个点,正向经过一次加一,反向经过一次减一)
double True_Angle = 0.0; //真实角度(累计角度)
int dir = 0; //0正向 1反向 //方向
double Current_Angle = 0; //当前角度(就是绝对位置角度,小于360度)
void AS5600_Init ( void ) {
IIC_Init();
}
u16 AS5600_Read_Len ( u8 addr, u8 reg, u8 len, u8 *buf ) {
IIC_Start();
IIC_Send_Byte ( ( addr << 1 ) | Write_Bit );
if ( IIC_Wait_Ack() ) {
IIC_Stop();
return 1;
}
IIC_Send_Byte ( reg );
IIC_Wait_Ack();
IIC_Start();
IIC_Send_Byte ( ( addr << 1 ) | Read_Bit ); // 发送器件地址 + 读命令
IIC_Wait_Ack(); // 等待应答
while ( len ) {
if ( len == 1 ) {
*buf = IIC_Read_Byte ( 0 ); // 读数据,发送nACK
} else {
*buf = IIC_Read_Byte ( 1 ) & 0x000f; // 读数据,发送ACK 原始1f,改为0f
}
len--;
buf++;
}
IIC_Stop();
return 0;
}
void Get_Ini_Val(void)//获得初始角度
{
u8 i = 0;
u32 transfer = 0;
for ( i = 0; i < 20; i++ ) { // 刚开始数据可能不稳定,直接丢掉
AS5600_Read_Len ( Slave_Addr, Angle_Hight_Register_Addr, 2, buf );
delay_ms ( 5 );
}
for ( i = 0; i < 20; i++ ) { // 软件滤波
AS5600_Read_Len ( Slave_Addr, Angle_Hight_Register_Addr, 2, buf );
transfer += ( ( buf[0] << 8 ) | buf[1] );
delay_ms ( 5 );
}
temp0 = transfer / 20;
temp2 = temp0;//给temp2初始化为初始值
temp1 = temp0;//给temp1初始化为初始值
}
void Get_Temp_Add(void)//计算角度增量
{
if(sign_angle == 0) //当从没经过零点时
{
if(temp2 >= temp0) //正转
{ temp_add = temp2 - temp0;
dir = 0;}
else //反转
{ temp_add = temp0 - temp2;
dir = 1;}
}
else if(sign_angle > 0)//经过一次及以上零点位置后,分两种情况,正向经过与反向经过,需分开讨论
{
temp_add = 4096 + temp2 - temp0 + ( sign_angle - 1)*4096; //正向经过
dir = 0;
}
else
{
temp_add =4096 + temp0 - 4096*(sign_angle+1) - temp2; //反向经过
dir = 1;
}
}
void Change_angle(void)//考虑减速比,计算增量实际角度
{
True_Angle = (temp_add/(Jian_Su_Bi*4096))*360; //4096代表as5600原始数据最大值(0~4096)
}
void Get_Current_angle(void)//获取360度角度,即计算绝对位置,小于等于360度
{
u32 x0 = 0;
x0 = (temp_add*100)%((u32)(Jian_Su_Bi*100)*4096);
Current_Angle = (x0/(Jian_Su_Bi*4096))*360/100;
}
void Get_Num_sign(void) //计算过零点次数,这个函数也可以用定时器中断来调用,效果更好
{
u32 x;
AS5600_Read_Len ( Slave_Addr, Angle_Hight_Register_Addr, 2, buf );
temp2 = ( ( buf[0] << 8 ) | buf[1] );
if(temp1 >= temp2)
{
x = temp1 - temp2;
if(x>2048)//正转通过零点
{
sign_angle++;
}
}
else
{
x = temp2 -temp1;
if(x>2048)//反转通过零点
{
sign_angle--;
}
}
temp1 = temp2;//每次都把temp2赋给temp1
}
下面是main函数
#include "stm32f10x.h"
#include "oled.h"
#include "delay.h"
#include "timer.h"
#include "key.h"
#include "exti.h"
#include "as5600.h"
#include "stdio.h"
#include "sys.h"
#include "key.h"
#include "mtor.h"
#include "exti.h"
#include "oledfont.h"
extern u8 TIM2_Pulse_TIM3_Counter_OK;
uint16_t pwm = 7199;
u32 pulsecnt = 200;
extern double Current_Angle;
extern double True_Angle;
extern int dir;
extern u32 temp_add;
extern u32 temp2;
extern u32 temp0;
uint8_t sign = 0;
extern int sign_angle;
int main(void)
{
char strff[21];
char strff2[21];
static uint8_t x = 1;
delay_init();
OLED_Init();
led_Init();
KEY_Init();
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //设置NVIC中断分组2:2位抢占优先级,2位响应优先级
TIM3_PWM_Init(pwm,19);//控制频率 72000000/(7199+1)/(19+1)=500HZ
TIM_SetCompare2(TIM3,10);//控制占空比,原始49,改为2999
motor_init();
TIM2_Init(pulsecnt);
EXTIX_Init();
AS5600_Init();
memset(strff,0,sizeof(strff));
memset(strff2,0,sizeof(strff2));
Get_Ini_Val();//获得初始角度
while(1)
{
Get_Temp_Add();
Change_angle();
Get_Current_angle();
Get_Num_sign();
sprintf(strff,"%.2f",True_Angle);//%0.2f angle
//sprintf(strff2,"%.2f",Current_Angle);//%0.2f angle
sprintf(strff2,"%.2f",Current_Angle);
if(dir == 0)
{
OLED_ShowChar(0,0,'+');
OLED_ShowChar(0,4,'+');
}
else
{
OLED_ShowChar(0,0,'-');
OLED_ShowChar(0,4,'-');
}
OLED_ShowNum(6,0,True_Angle,5,12);
OLED_ShowString(0,2,strff);
OLED_ShowNum(6,4,Current_Angle,5,12);
OLED_ShowString(0,6,strff2);
//OLED_ShowNum(54,6,3600000/pwm,5,12);
Get_Num_sign();
switch(sign)
{
case 1:
if(x==1)
{/*TIM_Cmd(TIM3,ENABLE);
led_On_Off();
x++;*/
setDir();
x++;
}
else if(x==2)
{
/*setDir();
x++;*/
TIM_Cmd(TIM3,ENABLE);
led_On_Off();
x++;
}
else if(x==3)
{ TIM_Cmd(TIM3,DISABLE);
led_On_Off();
x=1;}
sign = 0;
break;
case 2:
if(pwm>0)
{
pwm-=100;
}else
{
pwm=7199;
}
sign = 0;
TIM_SetAutoreload(TIM3,pwm);
break;
case 3:
if(pwm<15000)
{
pwm+=100;
}else
{
pwm=7199;
}
sign = 0;
TIM_SetAutoreload(TIM3,pwm);
break;
case 4:
//正转1/4,反转1/4,正转半圈,反转半圈
Pulse_output(1036);
/*setDir();
Pulse_output(1036);
while(TIM2_Pulse_TIM3_Counter_OK!=1);
TIM2_Pulse_TIM3_Counter_OK=0;*/
//setDir();
/*Pulse_output(100);
while(TIM2_Pulse_TIM3_Counter_OK!=1);
TIM2_Pulse_TIM3_Counter_OK=0;
setDir();
Pulse_output(100);
while(TIM2_Pulse_TIM3_Counter_OK!=1);
TIM2_Pulse_TIM3_Counter_OK=0;
*/
sign = 0;
break;
default:
break;
}
if(TIM2_Pulse_TIM3_Counter_OK==1)
{
TIM2_Pulse_TIM3_Counter_OK=0;
}
}
}
注意switch后面的语句是用来按钮控制步进电机运动的,和as5600本身无关。
五、实物演示
图中一个是基于STM32F103C8T6的控制器,一个是TB6600步进电机驱动板。
实测,代码正常运行,功能都可以实现,连续正转10圈误差在3.6度以内,再反转10圈回零点误差在0.15度以内,数据也基本稳定。
测试数据记录
测试的时候每次给半圈的脉冲,所以一转是两个数据,0->180.21->0.103->180.63->0.239…依次测量得到。
最后,附上程序代码以及AS5600的berger制板文件的下载地址
链接:https://pan.baidu.com/s/1zxl2O3oEvmDIDNtbDGoTPg
提取码:m5k5
演示效果观看地址:
https://www.bilibili.com/video/BV1hP411N7PJ/?vd_source=81fb9332eb85b94d92d4e3884ff48c6a