標題:
ADXL345測傾斜角(程序)
[打印本頁]
作者:
grh
時間:
2017-7-26 20:09
標題:
ADXL345測傾斜角(程序)
ADXL345測傾斜角
單片機源程序如下:
#include <REG51.H>
#include <math.h> //Keil library
#include <stdio.h> //Keil library
#include <INTRINS.H>
#define uchar unsigned char
#define uint unsigned int
#define DataPort P0 //LCD1602數(shù)據(jù)端口
sfr TUXR=0x8E;
sfr CMOD=0xD9;
sfr CCON=0xD8;
sfr CL=0xE9;
sfr CH=0xF9;
sfr CCAPM0=0xDA;
sfr CCAPM1=0xDB;
sfr CCAP0L=0xEA;
sfr CCAP0H=0xFA;
sfr CCAP1L=0xEB;
sfr CCAP1H=0xFB;
sbit CR=CCON^6;
sbit SCL=P2^0; //IIC時鐘引腳定義
sbit SDA=P2^1; //IIC數(shù)據(jù)引腳定義
sbit p22=P2^2;
sbit p31=P3^1;
sbit p33=P3^3;
sbit LCM_RS=P2^6; //LCD1602命令端口
sbit LCM_RW=P2^5; //LCD1602命令端口
sbit LCM_EN=P2^7; //LCD1602命令端口
#define SlaveAddress 0xA6 //定義器件在IIC總線中的從地址,根據(jù)ALT ADDRESS地址引腳不同修改
//ALT ADDRESS引腳接地時地址為0xA6,接電源時地址為0x3A
typedef unsigned char BYTE;
typedef unsigned short WORD;
BYTE BUF[8]; //接收數(shù)據(jù)緩存區(qū)
uchar ge,shi,bai,qian,wan; //顯示變量 個十百千萬
int dis_data; //變量
float jd;
uint PWMH;
void InitLcd(); //初始化lcd1602
void Init_ADXL345(void); //初始化ADXL345
void WriteDataLCM(uchar dataW); // LCD1602相關(guān)函數(shù)
void WriteCommandLCM(uchar CMD,uchar Attribc); //
void DisplayOneChar(uchar X,uchar Y,uchar DData); //
void conversion(uint temp_data); //數(shù)據(jù)處理
void Single_Write_ADXL345(uchar REG_Address,uchar REG_data); //單個寫入數(shù)據(jù)
uchar Single_Read_ADXL345(uchar REG_Address); //單個讀取內(nèi)部寄存器數(shù)據(jù)
void Multiple_Read_ADXL345(); //連續(xù)的讀取內(nèi)部寄存器數(shù)據(jù)
//------------------------------------
void Delay5us();
void Delay5ms();
void ADXL345_Start(); //與iic有關(guān)函數(shù)
void ADXL345_Stop();
void ADXL345_SendACK(bit ack); //傳送應(yīng)答
bit ADXL345_RecvACK(); //接收應(yīng)答
void ADXL345_SendByte(BYTE dat);
BYTE ADXL345_RecvByte();
void ADXL345_ReadPage();
void ADXL345_WritePage();
/*******************************/
void WaitForEnable(void)
{
DataPort=0xff;
LCM_RS=0;
LCM_RW=1;
_nop_(); _nop_();
LCM_EN=1;
_nop_();_nop_(); _nop_();_nop_();
while(DataPort&0x80);
LCM_EN=0;
}
/*******************************/
void WriteCommandLCM(uchar CMD,uchar Attribc)
{
if(Attribc)
WaitForEnable();
LCM_RS=0;
LCM_RW=0;
_nop_();_nop_();
DataPort=CMD;
_nop_();_nop_();
LCM_EN=1;
nop_();_nop_();nop_();_nop_();
LCM_EN=0;
}
/*******************************/
void WriteDataLCM(uchar dataW)
{
WaitForEnable();
LCM_RS=1;LCM_RW=0;
_nop_();_nop_();
DataPort=dataW;_nop_();
LCM_EN=1;
_nop_();_nop_();_nop_();_nop_();
LCM_EN=0;
}
/***********************************/
void InitLcd()
{
WriteCommandLCM(0x38,1);
WriteCommandLCM(0x08,1);
WriteCommandLCM(0x01,1);
WriteCommandLCM(0x06,1);
WriteCommandLCM(0x0c,1);
}
/***********************************/
void DisplayOneChar(uchar X,uchar Y,uchar DData) //X,Y為字符坐標
{
Y&=0x01;
X&=0x0f;
if(Y)
X|=0x40; //8+4=c
X|=0x80;
WriteCommandLCM(X,1);
WriteDataLCM(DData);
}
void Delay5us() //@12.000MHz
{
unsigned char i;
i = 12;
while (--i);
}
void Delay5ms() //@12.000MHz
{
unsigned char i, j;
_nop_();
_nop_();
i = 59;
j = 89;
do
{
while (--j);
} while (--i);
}
/**************************************
起始信號
**************************************/
void ADXL345_Start()
{
SDA = 1; //拉高數(shù)據(jù)線
SCL = 1; //拉高時鐘線
Delay5us(); //延時
SDA = 0; //產(chǎn)生下降沿
Delay5us(); //延時
SCL = 0; //拉低時鐘線
}
/**************************************
停止信號
**************************************/
void ADXL345_Stop()
{
SDA = 0; //拉低數(shù)據(jù)線
SCL = 1; //拉高時鐘線
Delay5us(); //延時
SDA = 1; //產(chǎn)生上升沿
Delay5us(); //延時
}
/**************************************
發(fā)送應(yīng)答信號
入口參數(shù):ack (0:ACK 1:NAK)
**************************************/
void ADXL345_SendACK(bit ack)
{
SDA = ack; //寫應(yīng)答信號
SCL = 1; //拉高時鐘線
Delay5us(); //延時
SCL = 0; //拉低時鐘線
Delay5us(); //延時
}
/**************************************
接收應(yīng)答信號
**************************************/
bit ADXL345_RecvACK()
{
SCL = 1; //拉高時鐘線
Delay5us(); //延時
CY = SDA; //讀應(yīng)答信號
SCL = 0; //拉低時鐘線
Delay5us(); //延時
return CY;
}
/**************************************
向IIC總線發(fā)送一個字節(jié)數(shù)據(jù)
**************************************/
void ADXL345_SendByte(BYTE dat)
{
BYTE i;
for (i=0; i<8; i++) //8位計數(shù)器
{
dat <<= 1; //移出數(shù)據(jù)的最高位
SDA = CY; //送數(shù)據(jù)口
SCL = 1; //拉高時鐘線
Delay5us(); //延時
SCL = 0; //拉低時鐘線
Delay5us(); //延時
}
ADXL345_RecvACK();
}
/**************************************
從IIC總線接收一個字節(jié)數(shù)據(jù)
**************************************/
BYTE ADXL345_RecvByte()
{
BYTE i;
BYTE dat = 0;
SDA = 1; //使能內(nèi)部上拉,準備讀取數(shù)據(jù),
for (i=0; i<8; i++) //8位計數(shù)器
{
dat <<= 1;
SCL = 1; //拉高時鐘線
Delay5us(); //延時
dat |= SDA; //讀數(shù)據(jù)
SCL = 0; //拉低時鐘線
Delay5us(); //延時
}
return dat;
}
//******單字節(jié)寫入*******************************************
//用于ADXL345初始化
void Single_Write_ADXL345(uchar REG_Address,uchar REG_data)
{
ADXL345_Start(); //起始信號
ADXL345_SendByte(SlaveAddress); //發(fā)送設(shè)備地址+寫信號
ADXL345_SendByte(REG_Address); //內(nèi)部寄存器地址,請參考中文pdf22頁
ADXL345_SendByte(REG_data); //內(nèi)部寄存器數(shù)據(jù),請參考中文pdf22頁
//相當(dāng)于 向相應(yīng)的地址寫入命令字
ADXL345_Stop(); //發(fā)送停止信號
}
//********單字節(jié)讀取*****************************************
uchar Single_Read_ADXL345(uchar REG_Address)
{
uchar REG_data;
ADXL345_Start(); //起始信號
ADXL345_SendByte(SlaveAddress); //發(fā)送設(shè)備地址+寫信號,//0xA6寫入
ADXL345_SendByte(REG_Address); //發(fā)送存儲單元地址,從0開始
ADXL345_Start(); //起始信號
ADXL345_SendByte(SlaveAddress+1); //發(fā)送設(shè)備地址+讀信號,//0xA7讀取
REG_data=ADXL345_RecvByte(); //讀出寄存器數(shù)據(jù)
ADXL345_SendACK(1);
ADXL345_Stop(); //停止信號
return REG_data;
}
//*********************************************************
//
//連續(xù)讀出ADXL345內(nèi)部加速度數(shù)據(jù),地址范圍0x32~0x37
//
//*********************************************************
void Multiple_read_ADXL345(void)
{
uchar i;
ADXL345_Start(); //起始信號
ADXL345_SendByte(SlaveAddress); //發(fā)送設(shè)備地址+寫信號
ADXL345_SendByte(0x32); //發(fā)送存儲單元地址,從0x32開始
ADXL345_Start(); //起始信號
ADXL345_SendByte(SlaveAddress+1); //發(fā)送設(shè)備地址+讀信號
for (i=0; i<6; i++) //連續(xù)讀取6個地址數(shù)據(jù),存儲中BUF
{
BUF[i] = ADXL345_RecvByte(); //BUF[0]存儲0x32地址中的數(shù)據(jù)
if (i == 5)
{
ADXL345_SendACK(1); //最后一個數(shù)據(jù)需要回NOACK
}
else
{
ADXL345_SendACK(0); //回應(yīng)ACK
}
}
ADXL345_Stop(); //停止信號
Delay5ms();
}
//*****************************************************************
//初始化ADXL345,根據(jù)需要請參考pdf進行修改************************
void Init_ADXL345()
{
Single_Write_ADXL345(0x31,0x0B); //測量范圍,正負16g,13位模式
Single_Write_ADXL345(0x2C,0x08); //速率設(shè)定為12.5 參考pdf13頁
Single_Write_ADXL345(0x2D,0x08); //選擇電源模式 參考pdf24頁
Single_Write_ADXL345(0x2E,0x80); //使能 DATA_READY 中斷
Single_Write_ADXL345(0x1E,0x00); //X 偏移量 根據(jù)測試傳感器的狀態(tài)寫入,pdf29頁//自己調(diào)試,得出偏移量
Single_Write_ADXL345(0x1F,0x00); //Y 偏移量 根據(jù)測試傳感器的狀態(tài)寫入,pdf29頁
Single_Write_ADXL345(0x20,0x05); //Z 偏移量 根據(jù)測試傳感器的狀態(tài)寫入,pdf29頁
}
//*********************************************************
void conversion(uint temp_data)
{
wan=temp_data/10000+0x30 ; //0x30='0'
temp_data=temp_data%10000; //取余運算
qian=temp_data/1000+0x30 ;
temp_data=temp_data%1000; //取余運算
bai=temp_data/100+0x30 ;
temp_data=temp_data%100; //取余運算
shi=temp_data/10+0x30 ;
temp_data=temp_data%10; //取余運算
ge=temp_data+0x30;
}
//顯示x軸傾斜角,即x軸與垂線所成角度
void display_jd_x()
{
bit bj=0;//標記
int dis_data_x,dis_data_z; //x,y軸加速度的原始數(shù)據(jù),用補碼形式表示
float mg_x,mg_z; //角度,加速度
dis_data_x=(BUF[1]<<8)+BUF[0]; //合成數(shù)據(jù)
dis_data_z=(BUF[5]<<8)+BUF[4]; //合成數(shù)據(jù)
if((dis_data_x<=0)&&(dis_data_z>0)) //第二象限
{
dis_data_x=-dis_data_x;
DisplayOneChar(8,0,' ');
bj=1;
}
else if((dis_data_z<0)&&(dis_data_x>=0)) //第四象限
{
dis_data_z=-dis_data_z;
DisplayOneChar(8,0,'-'); //顯示正負符號位
}
else if((dis_data_z<=0)&&(dis_data_x<0)) //第三象限
{
bj=1;
dis_data_z=-dis_data_z;
dis_data_x=-dis_data_x;
DisplayOneChar(8,0,'-'); //顯示正負符號位
}
else if((dis_data_z>=0)&&(dis_data_x>0))
DisplayOneChar(8,0,' '); //第一象限
mg_x=(float)dis_data_x*3.9; //計算數(shù)據(jù)和顯示,查看ADXL345快速入門第4頁,1LSB=3.9mg(毫g)
mg_z=(float)dis_data_z*3.9; //強制類型轉(zhuǎn)換,dis_data_z的類型和值不會發(fā)生改變
jd=atan2(mg_z,mg_x)*(180/3.14159);
if(bj==1)jd=180-jd;
//jd=jd*10; 消除小數(shù)點,便于轉(zhuǎn)換數(shù)據(jù)和顯示
conversion(jd*10); //轉(zhuǎn)換出顯示需要的數(shù)據(jù) //
DisplayOneChar(6,0,'X');
DisplayOneChar(7,0,':');
DisplayOneChar(9,0,qian);
DisplayOneChar(10,0,bai);
DisplayOneChar(11,0,shi);
DisplayOneChar(12,0,'.');
DisplayOneChar(13,0,ge);
DisplayOneChar(14,0,0xdf); //顯示 °
DisplayOneChar(15,0,' ');
}
void PCAinit() //PCA模塊初始化
{
CCON=0;
CL=0;
CH=0;
CMOD=0x00; //sysclk/12
CCAPM0=0x42;//8位PWM,無中斷
CR=1;
CCAP0H=CCAP0L=256-240;
}
void init_t0() //定時器0初始化
{
TUXR&=0x7f; //12T
TMOD=0x01;
TH0=(65536-50000)/256;
TL0=(65536-50000)%256;
TR0=1;
EA=1;
ET0=1;
}
//*********************************************************
//******主程序********
//*********************************************************
void main()
{
char H;
InitLcd(); //液晶初始化ADXL345
Delay5ms();
Init_ADXL345(); //初始化ADXL345
PCAinit();
init_t0();
H=CCAP0H;
while(1)
{
if(p33==0)
{
Delay5ms();
if(p33==0)
{
H=H+5;
if(H>255)H=255;
CCAP0H=H;
CCAP0L=H;
while(p33==0);
}
}
if(p31==0)
{
Delay5ms();
if(p31==0)
{
H=H-5;
if(H<0)H=0;
……………………
…………限于本文篇幅 余下代碼請從51黑下載附件…………
復(fù)制代碼
所有資料51hei提供下載:
ADXL345測傾斜角.zip
(69.11 KB, 下載次數(shù): 351)
2017-7-26 20:07 上傳
點擊文件名下載附件
下載積分: 黑幣 -5
作者:
我們失戀在夏季
時間:
2017-8-1 20:10
謝謝 大神解答
作者:
djk0125
時間:
2017-12-21 15:47
感謝分享!。
作者:
lgp214
時間:
2018-4-23 21:05
非常感謝大神的程序,Y軸的怎么寫麻煩大神指導(dǎo)下?
作者:
eassion_1
時間:
2018-8-17 11:33
感謝樓主感謝感謝
作者:
chengweiquan
時間:
2018-8-22 15:31
感謝樓主分享
作者:
376262974
時間:
2018-11-21 11:00
學(xué)習(xí)一下
作者:
376262974
時間:
2018-11-21 11:03
代碼寫得好亂,不好分析。
作者:
jerryyaozl
時間:
2019-2-28 16:34
感謝大神分享。
作者:
qingxie
時間:
2019-3-30 23:17
感謝!!
作者:
zhangchaoliang
時間:
2019-4-26 22:32
好的,后面的程序。
作者:
lyt87336
時間:
2019-7-6 12:00
sbit p22=P2^2;
sbit p31=P3^1;
sbit p33=P3^3;這連接的是什么
作者:
GJ2020
時間:
2020-7-28 14:48
不錯,有用的數(shù)據(jù)
作者:
ganganoo
時間:
2020-11-16 21:49
有接法嗎
作者:
qewnja
時間:
2020-11-22 20:23
感謝分享,最近正在研究這個
作者:
fjhcpu
時間:
2021-9-11 17:13
精度如何?
歡迎光臨 (http://www.torrancerestoration.com/bbs/)
Powered by Discuz! X3.1