找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 4158|回复: 2
收起左侧

PCA9685模块简单应用单片机源码

[复制链接]
ID:228471 发表于 2017-8-20 19:44 | 显示全部楼层 |阅读模式
  1. /**************************************************************************
  2.                         PCA9685模块简单应用
  3.                 平台:89C52,晶振:11.0592
  4. ***************************************************************************/
  5. #include<reg52.h>           
  6. #include <intrins.h>  
  7. #include <stdio.h>
  8. #include <math.h>
  9. typedef  unsigned char  uchar;        
  10. typedef  unsigned int   uint;        


  11. sbit scl=P3^6;                        //时钟输入线
  12. sbit sda=P3^7;                   //数据输入/输出端


  13. #define PCA9685_adrr 0x80//  1+A5+A4+A3+A2+A1+A0+w/r
  14.                         //片选地址,将焊接点置1可改变地址,
  15.                         // 当IIC总 呱嫌 多片PCA9685或相同地址时才需焊接
  16. #define PCA9685_SUBADR1 0x2
  17. #define PCA9685_SUBADR2 0x3
  18. #define PCA9685_SUBADR3 0x4


  19. #define PCA9685_MODE1 0x0
  20. #define PCA9685_PRESCALE 0xFE


  21. #define LED0_ON_L 0x6
  22. #define LED0_ON_H 0x7
  23. #define LED0_OFF_L 0x8
  24. #define LED0_OFF_H 0x9


  25. #define ALLLED_ON_L 0xFA
  26. #define ALLLED_ON_H 0xFB
  27. #define ALLLED_OFF_L 0xFC
  28. #define ALLLED_OFF_H 0xFD


  29. #define SERVOMIN  115 // this is the 'minimum' pulse length count (out of 4096)
  30. #define SERVOMAX  590 // this is the 'maximum' pulse length count (out of 4096)
  31. #define SERVO000  130 //0度对应4096的脉宽计数值
  32. #define SERVO180  520 //180度对应4096的脉宽计算值,四个值可根据不同舵机修改


  33. /**********************函数的声明*********************************/
  34. /*---------------------------------------------------------------
  35.                   毫秒延时函数
  36. ----------------------------------------------------------------*/
  37. void delayms(uint z)
  38. {
  39.   uint x,y;
  40.   for(x=z;x>0;x--)
  41.       for(y=148;y>0;y--);
  42. }
  43. /*---------------------------------------------------------------
  44.                                                                         IIC总线所需的通用函数
  45. ----------------------------------------------------------------*/
  46. /*---------------------------------------------------------------
  47.                  微妙级别延时函数 大于4.7us
  48. ----------------------------------------------------------------*/
  49. void delayus()
  50. {
  51.           _nop_();          //在intrins.h文件里
  52.                 _nop_();
  53.                 _nop_();
  54.                 _nop_();
  55.                 _nop_();

  56. }
  57. /*---------------------------------------------------------------
  58.                  IIC总线初始化函数
  59. ----------------------------------------------------------------*/
  60. void init()
  61. {
  62.     sda=1;                //sda scl使用前总是被拉高
  63.     delayus();
  64.     scl=1;
  65.     delayus();
  66. }
  67. /*---------------------------------------------------------------
  68.                  IIC总线启动信号函数
  69. ----------------------------------------------------------------*/
  70. void start()
  71. {
  72.     sda=1;
  73.     delayus();
  74.     scl=1;                        //scl拉高时 sda突然来个低电平 就启动了IIC总线
  75.     delayus();
  76.     sda=0;
  77.     delayus();
  78.     scl=0;
  79.     delayus();
  80. }
  81. /*---------------------------------------------------------------
  82.                  IIC总线停止信号函数
  83. ----------------------------------------------------------------*/
  84. void stop()
  85. {
  86.     sda=0;
  87.     delayus();
  88.     scl=1;                         //scl拉高时 sda突然来个高电平 就停止了IIC总线
  89.     delayus();
  90.     sda=1;                  
  91.     delayus();
  92. }
  93. /*---------------------------------------------------------------
  94.                  IIC总线应答信号函数
  95. ----------------------------------------------------------------*/
  96. void ACK()
  97. {
  98.     uchar i;
  99.     scl=1;
  100.     delayus();
  101.     while((sda=1)&&(i<255))         
  102.                 i++;                                       
  103.     scl=0;                                 
  104.     delayus();
  105. }
  106. /*---------------------------------------------------------------
  107.                  写一个字节,无返回值,需输入一个字节值
  108. ----------------------------------------------------------------*/
  109. void write_byte(uchar byte)
  110. {
  111.     uchar i,temp;
  112.     temp=byte;
  113.     for(i=0;i<8;i++)
  114.     {
  115.         temp=temp<<1;  
  116.         scl=0;                  
  117.                                 delayus();
  118.                                 sda=CY;                 
  119.                                 delayus();
  120.                                 scl=1;           
  121.                                 delayus();
  122.     }
  123.     scl=0;                  
  124.     delayus();
  125.     sda=1;                 
  126.     delayus();
  127. }
  128. /*---------------------------------------------------------------
  129.                  读一个字节函数,有返回值
  130. ----------------------------------------------------------------*/
  131. uchar read_byte()
  132. {
  133.                 uchar i,j,k;
  134.                 scl=0;
  135.                 delayus();
  136.                 sda=1;
  137.                 delayus();
  138.                 for(i=0;i<8;i++)        
  139.                 {
  140.                                 delayus();
  141.                                 scl=1;
  142.                 delayus();
  143.                 if(sda==1)
  144.                 {
  145.                                 j=1;
  146.                 }
  147.                 else j=0;
  148.                 k=(k<< 1)|j;  
  149.                 scl=0;            
  150.                 }
  151.                 delayus();
  152.                 return k;
  153. }
  154. /*---------------------------------------------------------------
  155.                 有关PCA9685模块的函数
  156. ----------------------------------------------------------------*/
  157. /*---------------------------------------------------------------
  158.                 向PCA9685里写地址,数据
  159. ----------------------------------------------------------------*/
  160. void PCA9685_write(uchar address,uchar date)
  161. {
  162.                 start();
  163.                 write_byte(PCA9685_adrr);        //PCA9685的片选地址
  164.                 ACK();                          
  165.                 write_byte(address);  //写地址控制字节
  166.                 ACK();
  167.                 write_byte(date);          //写数据
  168.                 ACK();
  169.                 stop();
  170. }
  171. /*---------------------------------------------------------------
  172.             从PCA9685里的地址值中读数据(有返回值)
  173. ----------------------------------------------------------------*/
  174. uchar PCA9685_read(uchar address)
  175. {
  176.                 uchar date;
  177.                 start();
  178.                 write_byte(PCA9685_adrr); //PCA9685的片选地址
  179.                 ACK();
  180.                 write_byte(address);
  181.                 ACK();
  182.                 start();
  183.                 write_byte(PCA9685_adrr|0x01);        //地址的第八位控制数据流方向,就是写或读
  184.                 ACK();
  185.                 date=read_byte();
  186.                 stop();
  187.                 return date;
  188. }
  189. /*---------------------------------------------------------------
  190.                         PCA9685复位
  191. ----------------------------------------------------------------*/
  192. void reset(void)
  193. {
  194.                 PCA9685_write(PCA9685_MODE1,0x0);
  195. }


  196. void begin(void)
  197. {
  198.                 reset();
  199. }
  200. /*---------------------------------------------------------------
  201.                                         PCA9685修改频率函数
  202. ----------------------------------------------------------------*/
  203. void setPWMFreq(float freq)
  204. {
  205.                 uint prescale,oldmode,newmode;
  206.                 float prescaleval;
  207.                 freq *= 0.92;  // Correct for overshoot in the frequency setting
  208.                 prescaleval = 25000000;
  209.                 prescaleval /= 4096;
  210.                 prescaleval /= freq;
  211.                 prescaleval -= 1;
  212.                 prescale = floor(prescaleval + 0.5);

  213.                 oldmode = PCA9685_read(PCA9685_MODE1);
  214.                 newmode = (oldmode&0x7F) | 0x10; // sleep
  215.                 PCA9685_write(PCA9685_MODE1, newmode); // go to sleep
  216.                 PCA9685_write(PCA9685_PRESCALE, prescale); // set the prescaler
  217.                 PCA9685_write(PCA9685_MODE1, oldmode);
  218.                 delayms(2);
  219.                 PCA9685_write(PCA9685_MODE1, oldmode | 0xa1);
  220. }
  221. /*---------------------------------------------------------------
  222.                                 PCA9685修改角度函数
  223. num:舵机PWM输出引脚0~15,on:PWM上升计数值0~4096,off:PWM下降计数值0~4096
  224. 一个PWM周期分成4096份,由0开始+1计数,计到on时跳变为高电平,继续计数到off时
  225. 跳变为低电平,直到计满4096重新开始。所以当on不等于0时可作延时,当on等于0时,
  226. off/4096的值就是PWM的占空比。
  227. ----------------------------------------------------------------*/
  228. void setPWM(uint num, uint on, uint off)
  229. {
  230.                 PCA9685_write(LED0_ON_L+4*num,on);
  231.                 PCA9685_write(LED0_ON_H+4*num,on>>8);
  232.                 PCA9685_write(LED0_OFF_L+4*num,off);
  233.                 PCA9685_write(LED0_OFF_H+4*num,off>>8);
  234.         }


  235. /*---------------------------------------------------------------
  236.                       主函数
  237. ----------------------------------------------------------------*/
  238. void main()
  239. {
  240.                 begin();
  241.                 setPWMFreq(50);  
  242.                 //例如要求舵机转到60度,这么算,
  243.                 //60度对应的脉宽=0.5ms+(60/180)*(2.5ms-0.5ms)=1.1666ms
  244.                 //利用占空比=1.1666ms/20ms=off/4096,off=239,50hz对应周期20ms
  245.                 //setPWM(num,0,239);;;;当然也可以利用SERVO000和SERVO180计算
  246.                 while(1)
  247.                 {
  248.                                 setPWM(0, 0, SERVOMIN);//第0路舵机转到最小角度
  249.                                 setPWM(1, 0, SERVO000);//第1路舵机转到0角度
  250.                                                                 setPWM(15, 0, 3000);
  251.                                 delayms(1500);
  252. //                                setPWM(0, 0, SERVOMAX);
  253. //                                setPWM(1, 0, SERVO180);
  254. //                                delayms(1500);
  255.                 }               
  256. }

复制代码


回复

使用道具 举报

ID:271114 发表于 2018-1-3 22:54 | 显示全部楼层
请问这个 prescale = floor(prescaleval + 0.5);是啥意思,没看明白
回复

使用道具 举报

ID:492269 发表于 2019-3-16 21:36 | 显示全部楼层
谢谢楼主分享
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表