今天要介绍的就是该模块,该模块是16路pwm模块,使用I2C总线可以控制16路舵机(led)。

接线OE空着就可以,其他VCC是芯片供电+5,SCL时钟线,SDA信号线,GND地线。

芯片介绍可以看:https://blog.csdn.net/asmallwhite/article/details/83048091 不过“默认情况下,若将A0-A5全部接地,则其器件地址为:0x40。” 错了,地址是 0x80

51单片机的代码在:http://www.51hei.com/bbs/blog-228471-7372.html    (很感谢,放上就跑,很好的代码,但是ACK函数里有个地方有问题 116行“while((sda=1)&&(i<255)) ” 应该改成"while((sda==1)&&(i<255)) "

不修改也能跑通,但是觉得应该这样才对。

再次感谢,代码来自:西瓜太狼 http://www.51hei.com/bbs/blog-228471-7372.html

  1. /**************************************************************************
  2. PCA9685模块简单应用
  3. 平台:89C52,晶振:11.0592
  4. ***************************************************************************/
  5. #include"stc89.h"
  6. #include <intrins.h>
  7. #include <stdio.h>
  8. #include <math.h>
  9. typedef unsigned char uchar;
  10. typedef unsigned int uint;
  11.  
  12. sbit scl=P2^1; //时钟输入线
  13. sbit sda=P2^0; //数据输入/输出端
  14.  
  15. #define PCA9685_adrr 0x80// 1+A5+A4+A3+A2+A1+A0+w/r
  16. //片选地址,将焊接点置1可改变地址,
  17. // 当IIC总 呱嫌 多片PCA9685或相同地址时才需焊接
  18. #define PCA9685_SUBADR1 0x2
  19. #define PCA9685_SUBADR2 0x3
  20. #define PCA9685_SUBADR3 0x4
  21.  
  22. #define PCA9685_MODE1 0x0
  23. #define PCA9685_PRESCALE 0xFE
  24.  
  25. #define LED0_ON_L 0x6
  26. #define LED0_ON_H 0x7
  27. #define LED0_OFF_L 0x8
  28. #define LED0_OFF_H 0x9
  29.  
  30. #define ALLLED_ON_L 0xFA
  31. #define ALLLED_ON_H 0xFB
  32. #define ALLLED_OFF_L 0xFC
  33. #define ALLLED_OFF_H 0xFD
  34.  
  35. #define SERVOMIN 90 // this is the 'minimum' pulse length count (out of 4096)
  36. #define SERVOMAX 700 // this is the 'maximum' pulse length count (out of 4096)
  37. #define SERVO000 130 //0度对应4096的脉宽计数值
  38. #define SERVO180 520 //180度对应4096的脉宽计算值,四个值可根据不同舵机修改
  39.  
  40. /**********************函数的声明*********************************/
  41. /*---------------------------------------------------------------
  42. 毫秒延时函数
  43. ----------------------------------------------------------------*/
  44. void delayms(uint z)
  45. {
  46. uint x,y;
  47. for(x=z;x>0;x--)
  48. for(y=148;y>0;y--);
  49. }
  50. /*---------------------------------------------------------------
  51. IIC总线所需的通用函数
  52. ----------------------------------------------------------------*/
  53. /*---------------------------------------------------------------
  54. 微妙级别延时函数 大于4.7us
  55. ----------------------------------------------------------------*/
  56. void delayus()
  57. {
  58. _nop_(); //在intrins.h文件里
  59. _nop_();
  60. _nop_();
  61. _nop_();
  62. _nop_();
  63.  
  64. }
  65. /*---------------------------------------------------------------
  66. IIC总线初始化函数
  67. ----------------------------------------------------------------*/
  68. void init()
  69. {
  70. sda=1; //sda scl使用前总是被拉高
  71. delayus();
  72. scl=1;
  73. delayus();
  74. }
  75. /*---------------------------------------------------------------
  76. IIC总线启动信号函数
  77. ----------------------------------------------------------------*/
  78. void start()
  79. {
  80. sda=1;
  81. delayus();
  82. scl=1; //scl拉高时 sda突然来个低电平 就启动了IIC总线
  83. delayus();
  84. sda=0;
  85. delayus();
  86. scl=0;
  87. delayus();
  88. }
  89. /*---------------------------------------------------------------
  90. IIC总线停止信号函数
  91. ----------------------------------------------------------------*/
  92. void stop()
  93. {
  94. sda=0;
  95. delayus();
  96. scl=1; //scl拉高时 sda突然来个高电平 就停止了IIC总线
  97. delayus();
  98. sda=1;
  99. delayus();
  100. }
  101. /*---------------------------------------------------------------
  102. IIC总线应答信号函数
  103. ----------------------------------------------------------------*/
  104. void ACK()
  105. {
  106. uchar i;
  107. scl=1;
  108. delayus();
  109. while((sda==1)&&(i<255)) //原来这里是sda=1这里应该等待从设备拉低总线
  110. i++;
  111. scl=0;
  112. delayus();
  113. }
  114. /*---------------------------------------------------------------
  115. 写一个字节,无返回值,需输入一个字节值
  116. ----------------------------------------------------------------*/
  117. void write_byte(uchar byte)
  118. {
  119. uchar i,temp;
  120. temp=byte;
  121. for(i=0;i<8;i++)
  122. {
  123. temp=temp<<1;
  124. scl=0;
  125. delayus();
  126. sda=CY;
  127. delayus();
  128. scl=1;
  129. delayus();
  130. }
  131. scl=0;
  132. delayus();
  133. sda=1;
  134. delayus();
  135. }
  136. /*---------------------------------------------------------------
  137. 读一个字节函数,有返回值
  138. ----------------------------------------------------------------*/
  139. uchar read_byte()
  140. {
  141. uchar i,j,k;
  142. scl=0;
  143. delayus();
  144. sda=1;
  145. delayus();
  146. for(i=0;i<8;i++)
  147. {
  148. delayus();
  149. scl=1;
  150. delayus();
  151. if(sda==1)
  152. {
  153. j=1;
  154. }
  155. else j=0;
  156. k=(k<< 1)|j;
  157. scl=0;
  158. }
  159. delayus();
  160. return k;
  161. }
  162. /*---------------------------------------------------------------
  163. 有关PCA9685模块的函数
  164. ----------------------------------------------------------------*/
  165. /*---------------------------------------------------------------
  166. 向PCA9685里写地址,数据
  167. ----------------------------------------------------------------*/
  168. void PCA9685_write(uchar address,uchar date)
  169. {
  170. start();
  171. write_byte(PCA9685_adrr); //PCA9685的片选地址
  172. ACK();
  173. write_byte(address); //写地址控制字节
  174. ACK();
  175. write_byte(date); //写数据
  176. ACK();
  177. stop();
  178. }
  179. /*---------------------------------------------------------------
  180. 从PCA9685里的地址值中读数据(有返回值)
  181. ----------------------------------------------------------------*/
  182. uchar PCA9685_read(uchar address)
  183. {
  184. uchar date;
  185. start();
  186. write_byte(PCA9685_adrr); //PCA9685的片选地址
  187. ACK();
  188. write_byte(address);
  189. ACK();
  190. start();
  191. write_byte(PCA9685_adrr|0x01); //地址的第八位控制数据流方向,就是写或读
  192. ACK();
  193. date=read_byte();
  194. stop();
  195. return date;
  196. }
  197. /*---------------------------------------------------------------
  198. PCA9685复位
  199. ----------------------------------------------------------------*/
  200. void reset(void)
  201. {
  202. PCA9685_write(PCA9685_MODE1,0x0);
  203. }
  204.  
  205. void begin(void)
  206. {
  207. reset();
  208. }
  209. /*---------------------------------------------------------------
  210. PCA9685修改频率函数
  211. ----------------------------------------------------------------*/
  212. void setPWMFreq(float freq)
  213. {
  214. uint prescale,oldmode,newmode;
  215. float prescaleval;
  216. freq *= 0.92; // Correct for overshoot in the frequency setting
  217. prescaleval = 25000000;
  218. prescaleval /= 4096;
  219. prescaleval /= freq;
  220. prescaleval -= 1;
  221. prescale = floor(prescaleval + 0.5);
  222.  
  223. oldmode = PCA9685_read(PCA9685_MODE1);
  224. newmode = (oldmode&0x7F) | 0x10; // sleep
  225. PCA9685_write(PCA9685_MODE1, newmode); // go to sleep
  226. PCA9685_write(PCA9685_PRESCALE, prescale); // set the prescaler
  227. PCA9685_write(PCA9685_MODE1, oldmode);
  228. delayms(2);
  229. PCA9685_write(PCA9685_MODE1, oldmode | 0xa1);
  230. }
  231. /*---------------------------------------------------------------
  232. PCA9685修改角度函数
  233. num:舵机PWM输出引脚0~15,on:PWM上升计数值0~4096,off:PWM下降计数值0~4096
  234. 一个PWM周期分成4096份,由0开始+1计数,计到on时跳变为高电平,继续计数到off时
  235. 跳变为低电平,直到计满4096重新开始。所以当on不等于0时可作延时,当on等于0时,
  236. off/4096的值就是PWM的占空比。
  237. ----------------------------------------------------------------*/
  238. void setPWM(uint num, uint on, uint off)
  239. {
  240. PCA9685_write(LED0_ON_L+4*num,on);
  241. PCA9685_write(LED0_ON_H+4*num,on>>8);
  242. PCA9685_write(LED0_OFF_L+4*num,off);
  243. PCA9685_write(LED0_OFF_H+4*num,off>>8);
  244. }
  245.  
  246. /*---------------------------------------------------------------
  247. 主函数
  248. ----------------------------------------------------------------*/
  249. void main()
  250. {
  251. begin();
  252. setPWMFreq(50);
  253. //例如要求舵机转到60度,这么算,
  254. //60度对应的脉宽=0.5ms+(60/180)*(2.5ms-0.5ms)=1.1666ms
  255. //利用占空比=1.1666ms/20ms=off/4096,off=239,50hz对应周期20ms
  256. //setPWM(num,0,239);;;;当然也可以利用SERVO000和SERVO180计算
  257. while(1)
  258. {
  259. setPWM(0, 0, SERVOMIN);//第0路舵机转到最小角度
  260. setPWM(1, 0, SERVO000);//第1路舵机转到0角度
  261. setPWM(15, 0, 3000);
  262. delayms(1500);
  263. setPWM(0, 0, SERVOMAX);
  264. // setPWM(1, 0, SERVO180);
  265. delayms(1500);
  266. }
  267. }

16路PWM输出的pca9685模块的更多相关文章

  1. 4412 4路pwm输出

    一.4412 xpwmTOUT1 这是4412的GPD0_1路,itop中被使用为LCD的背光电路的pwm功能.因此如果使用教程中的代码,同样操作GPD0_1是行不通的. 会出现错误,所以需要解除在内 ...

  2. J20航模遥控器开源项目系列教程(五)| 制作STM32F0接收机,8路PWM输出,SBUS输出,PPM输出 | 加密狗无线化,畅玩飞行模拟器

    我们的开源宗旨:自由 协调 开放 合作 共享 拥抱开源,丰富国内开源生态,开展多人运动,欢迎加入我们哈~ 和一群志同道合的人,做自己所热爱的事! 项目开源地址:https://github.com/J ...

  3. STM32 PWM输出(映射)

    STM32 的定时器除了 TIM6 和 7.其他的定时器都可以用来产生 PWM 输出.其中高级定时器 TIM1 和 TIM8 可以同时产生多达 7 路的 PWM 输出.而通用定时器也能同时产生多达 4 ...

  4. 定时器同步+触发三ADC采样+输出6路PWM波

    为了熟悉定时器定时器和ADC 用STM32F407DIS做了一个简单的工程: 通过高级定时器TIM1溢出更新时间作为触发输出信号(TRGO),触发TIM8开始计数: 同时TIM1的通道1.2.3以及分 ...

  5. 95-基于FMC接口的2路CameraLink Base输出子卡模块

    基于FMC接口的2路CameraLink Base输出子卡模块 1.板卡概述 FMC连接器是一种高速多pin的互连器件,广泛应用于板卡对接的设备中,特别是在xilinx公司的所有开发板中都使用.该Ca ...

  6. STM32F103ZET6 PWM输出

    1.通用定时器的PWM功能 STM32F103ZET6有4个通用定时器,分别是TIM2.TIM3.TIM4.TIM5. 通用定时器由一个可编程预分频器驱动的16位自动装载计数器构成. 通用定时器的很多 ...

  7. Arduino学习经验(一)之解决舵机库和pwm输出冲突

    一.前言 最近在公司学习Arduino uno ,用它实现小车超声波避障功能.实现的功能很简单,就是在小车前方挂一个超声波模块,当碰到障碍物时,会通过舵机进行摆头,判断两边的距离,进行左右转弯.但是碰 ...

  8. (五)转载:通用定时器PWM输出

    1.     TIMER输出PWM基本概念 脉冲宽度调制(PWM),是英文“Pulse Width Modulation”的缩写,简称脉宽调制,是利用微处理器的数字输出来对模拟电路进行控制的一种非常有 ...

  9. TIMER门控模式控制PWM输出长度

    TIMER门控模式控制PWM输出长度 参照一些网友代码做了些修改,由TIM4来控制TIM2的PWM输出长度, 采用主从的门控模式,即TIM4输出高时候TIM2使能输出 //TIM2 PWM输出,由TI ...

随机推荐

  1. 关于ijkplayer下载的demo不能运行,这是因为FFmpeg

    前提是你在Mac上已经配置了 homebrew 包管理工具 关于ijkPlayer的demo和framework的使用,也许当直接下载下来不能使用,这时候你需要再你下载的当前目录下运行,你看下自己的目 ...

  2. java.sql.SQLException: Value '0000-00-00' can not be represented as java.sql.Timestamp

    java.sql.SQLException: Value '0000-00-00' can not be represented as java.sql.Timestamp 错误是因为时间类型出现了0 ...

  3. C# 向程序新建的窗体中添加控件,控件需要先实例化,然后用controls.add添加到新的窗体中去

    C# 向程序新建的窗体中添加控件,控件需要先实例化,然后用controls.add添加到新的窗体中去 Form settingForm = new Form(); setForm deviceSet ...

  4. python ---split()函数讲解

    python ---split()函数讲解 split中文翻译为分裂. 在python用于分割字符串使用. split()就是将一个字符串分裂成多个字符串组成的列表. split()可以传入参数,也可 ...

  5. Linux中彻底删除Google-Chrome浏览器

    sudo apt-get autoremove --purge google-chrome-stable   卸载chrome后, 删除-/.config/google-chrome,重新安装.

  6. 2019-04-26-day041-数据库的索引

    内容回顾 多表查询 联表查 内连接 左右两表中能连上的行才被保留 表1 inner join 表2 on 表1.字段1=表2.字段2 外连接 左外连接 表1中所有的项都会被保留,而表2中只有匹配上表1 ...

  7. Delphi编程之爬取贴吧图片最终版

    接着前面两篇文章的内容,我们今天把这个贴吧爬取图片的程序完善,让它具有可以下载贴吧多页和帖子多页图片的能力. 主界面设计如下,包含3个labelededit,3个button,1个memo,1个str ...

  8. mysql navcate longblob 查询结果导出倒入

    由于之前项目为了查询方便,且不受权限控制.所以把image储存在了数据库.但是也遇到了很多瓶颈问题. 1,丢图从日志上查询获知,丢图95%以上是由于mysql的timeout时间失效引起的.(由于另一 ...

  9. Netty 服务端启动过程

    在 Netty 中创建 1 个 NioServerSocketChannel 在指定的端口监听客户端连接,这个过程主要有以下  个步骤: 创建 NioServerSocketChannel 初始化并注 ...

  10. 前端axios下载excel(二进制)

    需求:通过后端接口下载excel文件,后端没有文件地址,返回二进制流文件 实现:axios(ajax类似) 主要代码: axios:设置返回数据格式为blob或者arraybuffer 如: var ...