//本人不太懂C語言,所以麻煩大神,就告訴我把獨立按鍵的程序放在哪,或者直接把程序給我,萬分感謝嘍。想要的是KEY1鍵控制一個舵機,按鍵后旋轉一個角度,旋轉到那個角度后停止2-3秒,然后自動回到原始位置,剩下的三個舵機也是。
#include<reg52.h>
#include <intrins.h>
#include <stdio.h>
#include <math.h>
typedef unsigned char uchar;
typedef unsigned int uint;
sbit scl=P3^6; //時鐘輸入線
sbit sda=P3^7; //數據輸入/輸出端
#define PCA9685_adrr 0x80// 1+A5+A4+A3+A2+A1+A0+w/r
//片選地址,將焊接點置1可改變地址,
// 當IIC總線上有多片PCA9685或相同地址時才需焊接
#define PCA9685_SUBADR1 0x2
#define PCA9685_SUBADR2 0x3
#define PCA9685_SUBADR3 0x4
#define PCA9685_MODE1 0x0
#define PCA9685_PRESCALE 0xFE
#define LED0_ON_L 0x6
#define LED0_ON_H 0x7
#define LED0_OFF_L 0x8
#define LED0_OFF_H 0x9
#define ALLLED_ON_L 0xFA
#define ALLLED_ON_H 0xFB
#define ALLLED_OFF_L 0xFC
#define ALLLED_OFF_H 0xFD
#define SERVOMIN 115 // this is the 'minimum' pulse length count (out of 4096)
#define SERVOMAX 590 // this is the 'maximum' pulse length count (out of 4096)
#define SERVO000 130 //0度對應4096的脈寬計數值
#define SERVO180 520 //180度對應4096的脈寬計算值,四個值可根據不同舵機修改
/**********************函數的聲明*********************************/
/*---------------------------------------------------------------
毫秒延時函數
----------------------------------------------------------------*/
void delayms(uint z)
{
uint x,y;
for(x=z;x>0;x--)
for(y=148;y>0;y--);
}
/*---------------------------------------------------------------
IIC總線所需的通用函數
----------------------------------------------------------------*/
/*---------------------------------------------------------------
微妙級別延時函數 大于4.7us
----------------------------------------------------------------*/
void delayus()
{
_nop_(); //在intrins.h文件里
_nop_();
_nop_();
_nop_();
_nop_();
}
/*---------------------------------------------------------------
IIC總線初始化函數
----------------------------------------------------------------*/
void init()
{
sda=1; //sda scl使用前總是被拉高
delayus();
scl=1;
delayus();
}
/*---------------------------------------------------------------
IIC總線啟動信號函數
----------------------------------------------------------------*/
void start()
{
sda=1;
delayus();
scl=1; //scl拉高時 sda突然來個低電平 就啟動了IIC總線
delayus();
sda=0;
delayus();
scl=0;
delayus();
}
/*---------------------------------------------------------------
IIC總線停止信號函數
----------------------------------------------------------------*/
void stop()
{
sda=0;
delayus();
scl=1; //scl拉高時 sda突然來個高電平 就停止了IIC總線
delayus();
sda=1;
delayus();
}
/*---------------------------------------------------------------
IIC總線應答信號函數
----------------------------------------------------------------*/
void ACK()
{
uchar i;
scl=1;
delayus();
while((sda=1)&&(i<255))
i++;
scl=0;
delayus();
}
/*---------------------------------------------------------------
寫一個字節,無返回值,需輸入一個字節值
----------------------------------------------------------------*/
void write_byte(uchar byte)
{
uchar i,temp;
temp=byte;
for(i=0;i<8;i++)
{
temp=temp<<1;
scl=0;
delayus();
sda=CY;
delayus();
scl=1;
delayus();
}
scl=0;
delayus();
sda=1;
delayus();
}
/*---------------------------------------------------------------
讀一個字節函數,有返回值
----------------------------------------------------------------*/
uchar read_byte()
{
uchar i,j,k;
scl=0;
delayus();
sda=1;
delayus();
for(i=0;i<8;i++)
{
delayus();
scl=1;
delayus();
if(sda==1)
{
j=1;
}
else j=0;
k=(k<< 1)|j;
scl=0;
}
delayus();
return k;
}
/*---------------------------------------------------------------
有關PCA9685模塊的函數
----------------------------------------------------------------*/
/*---------------------------------------------------------------
向PCA9685里寫地址,數據
----------------------------------------------------------------*/
void PCA9685_write(uchar address,uchar date)
{
start();
write_byte(PCA9685_adrr); //PCA9685的片選地址
ACK();
write_byte(address); //寫地址控制字節
ACK();
write_byte(date); //寫數據
ACK();
stop();
}
/*---------------------------------------------------------------
從PCA9685里的地址值中讀數據(有返回值)
----------------------------------------------------------------*/
uchar PCA9685_read(uchar address)
{
uchar date;
start();
write_byte(PCA9685_adrr); //PCA9685的片選地址
ACK();
write_byte(address);
ACK();
start();
write_byte(PCA9685_adrr|0x01); //地址的第八位控制數據流方向,就是寫或讀
ACK();
date=read_byte();
stop();
return date;
}
/*---------------------------------------------------------------
PCA9685復位
----------------------------------------------------------------*/
void reset(void)
{
PCA9685_write(PCA9685_MODE1,0x0);
}
void begin(void)
{
reset();
}
/*---------------------------------------------------------------
PCA9685修改頻率函數
----------------------------------------------------------------*/
void setPWMFreq(float freq)
{
uint prescale,oldmode,newmode;
float prescaleval;
freq *= 0.92; // Correct for overshoot in the frequency setting
prescaleval = 25000000;
prescaleval /= 4096;
prescaleval /= freq;
prescaleval -= 1;
prescale = floor(prescaleval + 0.5);
oldmode = PCA9685_read(PCA9685_MODE1);
newmode = (oldmode&0x7F) | 0x10; // sleep
PCA9685_write(PCA9685_MODE1, newmode); // go to sleep
PCA9685_write(PCA9685_PRESCALE, prescale); // set the prescaler
PCA9685_write(PCA9685_MODE1, oldmode);
delayms(2);
PCA9685_write(PCA9685_MODE1, oldmode | 0xa1);
}
/*---------------------------------------------------------------
PCA9685修改角度函數
num:舵機PWM輸出引腳0~15,on:PWM上升計數值0~4096,off:PWM下降計數值0~4096
一個PWM周期分成4096份,由0開始+1計數,計到on時跳變為高電平,繼續計數到off時
跳變為低電平,直到計滿4096重新開始。所以當on不等于0時可作延時,當on等于0時,
off/4096的值就是PWM的占空比。
----------------------------------------------------------------*/
void setPWM(uint num, uint on, uint off)
{
PCA9685_write(LED0_ON_L+4*num,on);
PCA9685_write(LED0_ON_H+4*num,on>>8);
PCA9685_write(LED0_OFF_L+4*num,off);
PCA9685_write(LED0_OFF_H+4*num,off>>8);
}
/*---------------------------------------------------------------
主函數
----------------------------------------------------------------*/
void main()
{
begin();
setPWMFreq(50);
//例如要求舵機轉到60度,這么算,
//60度對應的脈寬=0.5ms+(60/180)*(2.5ms-0.5ms)=1.1666ms
//利用占空比=1.1666ms/20ms=off/4096,off=239,50hz對應周期20ms
//setPWM(num,0,239);;;;當然也可以利用SERVO000和SERVO180計算
while(1)
{
setPWM(0, 0, SERVOMIN);//第0路舵機轉到最小角度
setPWM(1, 0, SERVO000);
setPWM(2, 0, SERVO000);//第1路舵機轉到0角度
setPWM(3, 0, SERVO000);
delayms(1500);
setPWM(0, 0, SERVOMAX);
setPWM(1, 0, SERVO180);
setPWM(2, 0, SERVO180);
setPWM(3, 0, SERVO180);
delayms(1500);
}
}
/*---------------------------------------------------------------
END
----------------------------------------------------------------*/
|