找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 4769|回复: 12
收起左侧

关于单片机IO口模拟实现呼吸灯的疑问

[复制链接]
ID:258164 发表于 2018-12-18 11:34 | 显示全部楼层 |阅读模式
如何利用一个定时器和IO口实现LED有呼吸的功能呢?因为我的单片机资源有限,没有硬件的PWM,所以想使用定时器来实现调节占空比,原理我是知道的,但是逻辑感觉还是没有理清楚。

我看网上很多都是用的延时或者多个定时器,如果我只有一个定时器该怎么实现呢?我的定时周期多少合适?
回复

使用道具 举报

ID:111517 发表于 2018-12-18 13:56 | 显示全部楼层
如果用一个定时器的话你,你可以用这个定时器来做一个节拍啊,假设节拍是100us,你定时器就设置为100us,然后再定义一个计数器变量放在定时器中断里面,然后每计数到100就重置,这样的话周期就是10ms,你可以在主函数里利用这个变量来更改占空比
回复

使用道具 举报

ID:94031 发表于 2018-12-18 14:21 | 显示全部楼层
参考程序:
#include<reg51.h>
void Delay1(int del)
{
        int i,j;
        for(i=0; i<del; i++)
        for(j=0; j<1827; j++)   
        ;
}
void Delay(unsigned int t)
{
while(--t);
}
//函数声明
unsigned int z,y;
void main (void)//主函数
{
        unsigned int CYCLE=400,PWM_LOW=0;//定义周期并赋值
        while (1)         //主循环
        {

/*--------整排LED灯呼吸---------*/
               
                Delay1(100);                                                                 //加延时,可以看到熄灭的过程(下面程序同理)
                for(PWM_LOW=1;PWM_LOW<CYCLE;PWM_LOW++)           //PWM_LOW表示低
                        {
                                P1=0x00;                                       
                                Delay(PWM_LOW);
                                P1=0xff;
                                Delay(CYCLE-PWM_LOW);
                        }
                P1=0xff;                                                                                
                for(PWM_LOW=CYCLE-1;PWM_LOW>0;PWM_LOW--)         //与逐渐变亮相反的过程
                        {
                                   P1=0x00;                                                        //点亮LED
                                   Delay(PWM_LOW);
                                   P1=0xff;                                                        //熄灭LED
                                  Delay(CYCLE-PWM_LOW);                                //主循环中添加其他需要一直工作的程序,延时长度,600次循环中从599减至1
                        }

}
}
回复

使用道具 举报

ID:258164 发表于 2018-12-18 14:55 | 显示全部楼层
沧浪 发表于 2018-12-18 13:56
如果用一个定时器的话你,你可以用这个定时器来做一个节拍啊,假设节拍是100us,你定时器就设置为100us,然 ...

呼吸灯一般要求是3s的周期,也就是说要在3s的时间里实现渐亮到渐灭再到渐亮的过程。我现在的定时器周期是10ms,那么我怎么利用变量来改变占空比呢,怎么和LED的高低电平配合起来
回复

使用道具 举报

ID:258164 发表于 2018-12-18 14:55 | 显示全部楼层
xuyaqi 发表于 2018-12-18 14:21
参考程序:
#include
void Delay1(int del)

你这个是延时的办法,对于我的程序来说效率太低了,如果要使用定时器该怎么做呢?
回复

使用道具 举报

ID:94031 发表于 2018-12-18 16:50 | 显示全部楼层
HelloPan111 发表于 2018-12-18 14:55
你这个是延时的办法,对于我的程序来说效率太低了,如果要使用定时器该怎么做呢?

设定好定时器中断时间,数中断次数求得延时时间。
回复

使用道具 举报

ID:155507 发表于 2018-12-18 18:50 | 显示全部楼层
参考程序:

  1. /*---------------------------------------------------------------------*/
  2. /* --- STC MCU Limited ------------------------------------------------*/
  3. /* --- STC 1T Series MCU Demo Programme -------------------------------*/
  4. /* 如果要在程序中使用此代码,请在程序中注明使用了STC的资料及程序        */
  5. /*---------------------------------------------------------------------*/


  6. /*************  功能说明    **************

  7. 使用Timer0模拟16通道PWM驱动程序。

  8. 输出为 P1.0 ~ P1.7, P2.0 ~ P2.7, 对应 PWM0 ~ PWM15.

  9. 定时器中断频率一般不要超过100KHZ, 留足够的时间给别的程序运行.

  10. 本例子使用11.0592MHZ时钟, 25K的中断频率, 250级PWM, 周期为10ms.

  11. 中断里处理的时间不超过6us, 占CPU时间大约为15%.

  12. ******************************************/

  13. #include    <reg52.h>

  14. #define MAIN_Fosc       11059200UL  //定义主时钟
  15. #define Timer0_Rate     25000       //中断频率


  16. typedef     unsigned char   u8;
  17. typedef     unsigned int    u16;
  18. typedef     unsigned long   u32;

  19. sfr AUXR = 0x8E;

  20. sfr P1M1 = 0x91;    //PxM1.n,PxM0.n     =00--->Standard,    01--->push-pull
  21. sfr P1M0 = 0x92;    //                  =10--->pure input,  11--->open drain
  22. sfr P0M1 = 0x93;
  23. sfr P0M0 = 0x94;
  24. sfr P2M1 = 0x95;
  25. sfr P2M0 = 0x96;
  26. sfr P3M1 = 0xB1;
  27. sfr P3M0 = 0xB2;
  28. sfr P4M1 = 0xB3;
  29. sfr P4M0 = 0xB4;
  30. sfr P5M1 = 0xC9;
  31. sfr P5M0 = 0xCA;
  32. sfr P6M1 = 0xCB;
  33. sfr P6M0 = 0xCC;
  34. sfr P7M1 = 0xE1;
  35. sfr P7M0 = 0xE2;

  36. #define Timer0_Reload   (65536UL -(MAIN_Fosc / Timer0_Rate))        //Timer 0 重装值

  37. //************** PWM8 变量和常量以及IO口定义 ***************
  38. //********************  8通道8 bit 软PWM    ********************

  39. #define     PWM_DUTY_MAX    250 // 0~255    PWM周期, 最大255
  40. #define     PWM_ON          1   // 定义占空比的电平, 1 或 0

  41. #define     PWM_OFF         (!PWM_ON)
  42. #define     PWM_ALL_ON      (0xff * PWM_ON)

  43. u8 bdata PWM_temp1,PWM_temp2;       //影射一个RAM,可位寻址,输出时同步刷新
  44. sbit    P_PWM0  =   PWM_temp1^0;    //  定义影射RAM每位对应的IO
  45. sbit    P_PWM1  =   PWM_temp1^1;
  46. sbit    P_PWM2  =   PWM_temp1^2;
  47. sbit    P_PWM3  =   PWM_temp1^3;
  48. sbit    P_PWM4  =   PWM_temp1^4;
  49. sbit    P_PWM5  =   PWM_temp1^5;
  50. sbit    P_PWM6  =   PWM_temp1^6;
  51. sbit    P_PWM7  =   PWM_temp1^7;
  52. sbit    P_PWM8  =   PWM_temp2^0;
  53. sbit    P_PWM9  =   PWM_temp2^1;
  54. sbit    P_PWM10 =   PWM_temp2^2;
  55. sbit    P_PWM11 =   PWM_temp2^3;
  56. sbit    P_PWM12 =   PWM_temp2^4;
  57. sbit    P_PWM13 =   PWM_temp2^5;
  58. sbit    P_PWM14 =   PWM_temp2^6;
  59. sbit    P_PWM15 =   PWM_temp2^7;

  60. u8  pwm_duty;       //周期计数值
  61. u8  pwm[16];        //pwm0~pwm15 为0至15路PWM的宽度值

  62. bit     B_1ms;
  63. u8      cnt_1ms;
  64. u8      cnt_20ms;

  65. /**********************************************/
  66. void main(void)
  67. {
  68.     u8  i;
  69.    
  70.     P0M0 = 0x00;
  71.     P0M1 = 0x00;
  72.     P1M0 = 0x00;
  73.     P1M1 = 0x00;
  74.     P2M0 = 0x00;
  75.     P2M1 = 0x00;
  76.     P3M0 = 0x00;
  77.     P3M1 = 0x00;
  78.     P4M0 = 0x00;
  79.     P4M1 = 0x00;
  80.     P5M0 = 0x00;
  81.     P5M1 = 0x00;
  82.     P6M0 = 0x00;
  83.     P6M1 = 0x00;
  84.     P7M0 = 0x00;
  85.     P7M1 = 0x00;

  86.     AUXR |=  (1<<7);    // Timer0 set as 1T mode
  87.     TMOD &= ~(1<<2);    // Timer0 set as Timer
  88.     TMOD &= ~0x03;      // Timer0 set as 16 bits Auto Reload
  89.     TH0 = Timer0_Reload / 256;  //Timer0 Load
  90.     TL0 = Timer0_Reload % 256;
  91.     ET0 = 1;        //Timer0 Interrupt Enable
  92.     PT0 = 1;        //高优先级
  93.     TR0 = 1;        //Timer0 Run
  94.     EA = 1;         //打开总中断

  95.     cnt_1ms = Timer0_Rate / 1000;   //1ms计数
  96.     cnt_20ms = 20;
  97.    
  98.     for(i=0; i<16; i++)     pwm[i] = i * 15 + 15;   //给PWM一个初值
  99.    
  100.     while(1)
  101.     {
  102.         if(B_1ms)   //1ms到
  103.         {
  104.             B_1ms = 0;
  105.             if(--cnt_20ms == 0) //PWM 20ms改变一阶
  106.             {
  107.                 cnt_20ms = 20;
  108.                 for(i=0; i<16; i++) pwm[i]++;
  109.             }
  110.         }
  111.     }
  112. }


  113. /********************** Timer0 1ms中断函数 ************************/
  114. void timer0 (void) interrupt 1
  115. {
  116.     P1 = PWM_temp1;         //影射RAM输出到实际的PWM端口
  117.     P2 = PWM_temp2;
  118.         
  119.     if(++pwm_duty == PWM_DUTY_MAX)      //PWM周期结束,重新开始新的周期
  120.     {
  121.         pwm_duty = 0;
  122.         PWM_temp1 = PWM_ALL_ON;
  123.         PWM_temp2 = PWM_ALL_ON;
  124.     }
  125.     ACC = pwm_duty;
  126.     if(ACC == pwm[0])       P_PWM0  = PWM_OFF;  //判断PWM占空比是否结束
  127.     if(ACC == pwm[1])       P_PWM1  = PWM_OFF;
  128.     if(ACC == pwm[2])       P_PWM2  = PWM_OFF;
  129.     if(ACC == pwm[3])       P_PWM3  = PWM_OFF;
  130.     if(ACC == pwm[4])       P_PWM4  = PWM_OFF;
  131.     if(ACC == pwm[5])       P_PWM5  = PWM_OFF;
  132.     if(ACC == pwm[6])       P_PWM6  = PWM_OFF;
  133.     if(ACC == pwm[7])       P_PWM7  = PWM_OFF;
  134.     if(ACC == pwm[8])       P_PWM8  = PWM_OFF;
  135.     if(ACC == pwm[9])       P_PWM9  = PWM_OFF;
  136.     if(ACC == pwm[10])      P_PWM10 = PWM_OFF;
  137.     if(ACC == pwm[11])      P_PWM11 = PWM_OFF;
  138.     if(ACC == pwm[12])      P_PWM12 = PWM_OFF;
  139.     if(ACC == pwm[13])      P_PWM13 = PWM_OFF;
  140.     if(ACC == pwm[14])      P_PWM14 = PWM_OFF;
  141.     if(ACC == pwm[15])      P_PWM15 = PWM_OFF;

  142.     if(--cnt_1ms == 0)
  143.     {
  144.         cnt_1ms = Timer0_Rate / 1000;
  145.         B_1ms = 1;      // 1ms标志
  146.     }
  147. }
复制代码



  1. /*---------------------------------------------------------------------*/
  2. /* --- STC MCU Limited ------------------------------------------------*/
  3. /* --- STC15F4K60S4 系列 定时器软件模拟PWM举例-------------------------*/
  4. /* --- Mobile: (86)13922805190 ----------------------------------------*/
  5. /* --- Fax: 86-0513-55012956,55012947,55012969 ------------------------*/
  6. /* --- Tel: 86-0513-55012928,55012929,55012966-------------------------*/
  7. /* --- Web: www.STCMCU.com --------------------------------------------*/
  8. /* --- Web: www.GXWMCU.com --------------------------------------------*/
  9. /* 如果要在程序中使用此代码,请在程序中注明使用了STC的资料及程序        */
  10. /* 如果要在文章中应用此代码,请在文章中注明使用了STC的资料及程序        */
  11. /*---------------------------------------------------------------------*/

  12. //本示例在Keil开发环境下请选择Intel的8058芯片型号进行编译
  13. //若无特别说明,工作频率一般为11.0592MHz


  14. #include "reg51.h"

  15. //#define PWM6BIT   64              //6-bit PWM 周期数
  16. #define PWM8BIT     256             //8-bit PWM 周期数
  17. //#define PWM10BIT  1024            //10-bit PWM 周期数
  18. //#define PWM16BIT  65536           //16-bit PWM 周期数

  19. #define HIGHDUTY 64                 //高电平周期数(占空比64/256=25%)
  20. #define LOWDUTY  (PWM8BIT-HIGHDUTY) //低电平周期数

  21. sfr P0M1 = 0x93;
  22. sfr P0M0 = 0x94;
  23. sfr P1M1 = 0x91;
  24. sfr P1M0 = 0x92;
  25. sfr P2M1 = 0x95;
  26. sfr P2M0 = 0x96;
  27. sfr P3M1 = 0xb1;
  28. sfr P3M0 = 0xb2;
  29. sfr P4M1 = 0xb3;
  30. sfr P4M0 = 0xb4;
  31. sfr P5M1 = 0xC9;
  32. sfr P5M0 = 0xCA;
  33. sfr P6M1 = 0xCB;
  34. sfr P6M0 = 0xCC;
  35. sfr P7M1 = 0xE1;
  36. sfr P7M0 = 0xE2;

  37. sfr AUXR      = 0x8e;               //辅助寄存器
  38. sfr INT_CLKO  = 0x8f;               //时钟输出控制寄存器
  39. sbit T0CLKO   = P3^5;               //定时器0的时钟输出口

  40. bit flag;

  41. //定时器0中断服务程序
  42. void tm0() interrupt 1
  43. {
  44.     flag = !flag;                   //反转PWM的输出标志
  45.     if (flag)
  46.     {
  47.         TL0 = (65536-HIGHDUTY);     //准备高电平的重载值
  48.         TH0 = (65536-HIGHDUTY) >> 8;
  49.     }
  50.     else
  51.     {
  52.         TL0 = (65536-LOWDUTY);      //准备低电平的重载值
  53.         TH0 = (65536-LOWDUTY) >> 8;
  54.     }
  55. }

  56. void main()
  57. {
  58.     P0M0 = 0x00;
  59.     P0M1 = 0x00;
  60.     P1M0 = 0x00;
  61.     P1M1 = 0x00;
  62.     P2M0 = 0x00;
  63.     P2M1 = 0x00;
  64.     P3M0 = 0x00;
  65.     P3M1 = 0x00;
  66.     P4M0 = 0x00;
  67.     P4M1 = 0x00;
  68.     P5M0 = 0x00;
  69.     P5M1 = 0x00;
  70.     P6M0 = 0x00;
  71.     P6M1 = 0x00;
  72.     P7M0 = 0x00;
  73.     P7M1 = 0x00;

  74.     AUXR = 0x80;                    //定时器0为1T模式
  75.     INT_CLKO = 0x01;                //使能定时器0的时钟输出功能
  76.     TMOD &= 0xf0;                   //设置定时器0为模式0(16位自动重装载)
  77.     TL0 = (65536-LOWDUTY);          //初始化定时器初值和重装值
  78.     TH0 = (65536-LOWDUTY) >> 8;
  79.     T0CLKO = 1;                     //初始化时钟输出脚(软PWM口)
  80.     flag = 0;                       //初始化标志位
  81.     TR0 = 1;                        //定时器0开始计时
  82.     ET0 = 1;                        //使能定时器0中断
  83.     EA = 1;
  84.     while (1);
  85. }
复制代码
回复

使用道具 举报

ID:258164 发表于 2018-12-18 20:32 | 显示全部楼层
xuyaqi 发表于 2018-12-18 16:50
设定好定时器中断时间,数中断次数求得延时时间。

这样还是不行啊,我的目标是最好在中断里实现呼吸灯的效果,如果是在主循环里的话,对我的程序效率来说太差了。
回复

使用道具 举报

ID:258164 发表于 2018-12-18 20:34 | 显示全部楼层

谢谢拉,不过你给的这个程序看着有点乱啊,我看了还是没有太理解思路是什么
回复

使用道具 举报

ID:401564 发表于 2018-12-19 22:46 | 显示全部楼层
楼上的程序太恐怖了
给你一个算法,自己去写代码
模拟PWM的频率设定为1KHZ,如果你要PWM的分辨率达到1%,那你就得把寄存器中断设定为10uS一次了,10uS进入一次中断,然后,再在寄存器的中断中设定占空比,但我觉得这不太现实,10uS都没有能执行几条指令,还不如用两个定时器,C不会,帮不上你
但PWM的频率设定为1KHZ是对的,我测试过很多的LED手电,呼吸灯的频率都是1024uS的
但程序不会很大的,肯定是没有上面的那么多代码的,我就是用汇编也不会有那么代码的,更何况是C呢

回复

使用道具 举报

ID:258164 发表于 2018-12-20 17:55 | 显示全部楼层
Y_G_G 发表于 2018-12-19 22:46
楼上的程序太恐怖了
给你一个算法,自己去写代码
模拟PWM的频率设定为1KHZ,如果你要PWM的分辨率达到1%,那 ...

我其实 不需要呼吸灯的效果有多好,只要肉眼看到的效果是呼吸的就行。
另外如果用两个定时器的话,程序逻辑该是怎么样的呢
回复

使用道具 举报

ID:401564 发表于 2018-12-20 23:08 | 显示全部楼层
完全的算法不好说,大概就是进入一个定时器就累加一次变量,再把这个变量当成另一个定时器的中断时间,然后关闭当前定时器,再打开另一个定时器,中断返回
总的来说就是进入中断就关闭当前定时器,由另一个定时器来打开
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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