找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 1559|回复: 2
打印 上一主题 下一主题
收起左侧

stm8l152c6单片机adc1采集数据在1.0伏进位跳跃变成1.99伏

[复制链接]
跳转到指定楼层
楼主
ID:466148 发表于 2019-4-1 22:56 来自手机 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
用网上茗风的adc测stm8l内部参考电压程序跑adc1,用的也是他下面的lcd显示程序,配置如下
开发环境:IAR for stm8 v6.5.3

lixianshen 地板
2019-3-29 19:28
大师,我用你的lcd程序和你的adc程序调试stm8ldiscover套件时发现,stm8l152c 6t6的adc测电压时,输入电压从0伏调到2.99伏,lcd输出显示也从0到2.99伏,输入电压从2.99到1伏时,输出显示2.99到1伏。但是输入电压从1伏到0.99伏时,显示从1伏一下跳到1.99伏,输入电压从1伏调到0伏,显示就从1伏跳到1.99,显示跟着从1.99伏逐渐降到1.0,1.0伏就是对应输入电压的0伏。就可以要多显示1伏,如果现在输入又从0伏调大到1伏,显示就从1增大到1.99伏,然后进位又变成1伏,显示过了1伏后继续增加输入电压到2.99伏,显示也从1伏到2.99。而在电脑上的软件调试程序窗口中显示的电压,与输入电压是同步增减的,几乎相等。而单片机输入电压是用万用表和示波器同步看的,输入调压是用精密绕线电位器调压,输入电压没有跳跃。接收adc高位数据,从1到15增加没有问题,从15降到1之后,又跳到19,而adc的低位数据从255的0,然后从255到0,基本都很有规律。如果用lcd显示电压比如说1.95伏不正确时,按重启键,然后lcd显示就变为0.95伏,就正常显示,此时调低输入电压,显示也跟着正常变低,一旦调高电压超过1伏,显示就成0.99跳到1.99。adc程序从寄存器换成了库函数,毛病一样。输入从pc7换成pf0,毛病一样。程序触发等等都试过,情况没有变化,不知道是那里的问题?请板主和各位大师解答一下。

  1. *硬件平台:STM8L-DISCOVERY

  2. *功能说明:使用STM8L-DISCOVERY液晶屏显示一串6位数字

  3. *作    者:茗风

  4. #include"iostm8l152c6.h"

  5. #include"stm8l15x.h”
  6. #define a 0x01

  7. #define b 0x02

  8. #define c 0x04

  9. #define d 0x08

  10. #define e 0x10

  11. #define f 0x20

  12. #define g 0x40

  13. #define m 0x80


  14. const uint8_t LCD_Tab[10] = {

  15.                 a + b + c + d + e + f,                        // Displays "0"

  16.                 b + c,                                        // Displays "1"

  17.                 a + b + m + g + e + d,                        // Displays "2"

  18.                 a + b + m + g + c + d,                        // Displays "3"

  19.                 f + g + m + b + c,                        // Displays "4"

  20.                 a + f + g + m + c +d,                        // Displays "5"

  21.                 a + f + e + d + c + g + m ,                // Displays "6"

  22.                 a + b + c,                                // Displays "7"

  23.                 a + b + c + d + e + f + g + m,                // Displays "8"

  24.                 a + b + c + d + f + g + m                // Displays "9"


  25. };


  26. /******************************************************************************************************

  27. *  名    称:void LCD_Config(void)

  28. *  功    能:配置DAC,禁用DMA,不使用TIM4触发,也不用软件触发,写入DHR的数据被立即送入DOR寄存器,

  29. *            立即输出对应电压

  30. *  入口参数:无

  31. *  出口参数:无

  32. *  说    明:STM8L152C6属于中等容量MCU,只有一路DAC输出,输出引脚为PF0

  33. *  范    例:无

  34. ******************************************************************************************************/

  35. void LCD_Config(void)

  36. {

  37. //------打开LCD/RTC时钟------

  38.   CLK_PCKENR2_PCKEN22=1;//打开RTC时钟,LCD刷新频率与此时钟有关

  39.   CLK_PCKENR2_PCKEN23=1;//打开LCD时钟,读写LCD寄存器用到此时钟

  40.   

  41. //---选择LSE作为RTC时钟---

  42.   CLK_CRTCR_RTCSEL0=0;

  43.   CLK_CRTCR_RTCSEL1=0;

  44.   CLK_CRTCR_RTCSEL2=0;

  45.   CLK_CRTCR_RTCSEL3=1;

  46. /* 0000: No clock selected

  47.    0001: HSI clock used as RTC clock source

  48.    0010: LSI clock used as RTC clock source

  49.    0100: HSE clock used as RTC clock source

  50.    1000: LSE clock used as RTC clock sourc*/

  51.   

  52. //----设置RTC时钟分频值----

  53.   CLK_CRTCR_RTCDIV0=0;

  54.   CLK_CRTCR_RTCDIV1=0;

  55.   CLK_CRTCR_RTCDIV2=0;

  56. /*000: RTC clock source/1

  57.   001: RTC clock source /2

  58.   010: RTC clock source /4

  59.   011: RTC clock source /8

  60.   100: RTC clock source /16

  61.   101: RTC clock source /32

  62.   110: RTC clock source /64

  63.   111: RTC clock source /128*/

  64.   

  65. //----设置LCD预分频值----  

  66.   LCD_FRQ_PS0=0;// 2^PS[3:0]

  67.   LCD_FRQ_PS1=0;//分频值为1

  68.   LCD_FRQ_PS2=0;

  69.   LCD_FRQ_PS3=0;

  70.   

  71. //----设置LCD分频值----  

  72.   LCD_FRQ_DIV0=1;//DIV[3:0]+16

  73.   LCD_FRQ_DIV1=1;//分频值为15+16=31

  74.   LCD_FRQ_DIV2=1;

  75.   LCD_FRQ_DIV3=1;  



  76. //以上分频值的设置,最为了得到适合的LCD的刷新频率,如果增大分频值,会导致

  77. //LCD刷新频率变低,会看到LCD显示出现闪烁

  78. //比如,我们将PS[3:0]设置为0011,会看到液晶闪烁  

  79.   

  80. //----1/4 duty----  

  81.   LCD_CR1_DUTY0=1;//1/4 duty

  82.   LCD_CR1_DUTY1=1;

  83. /* Duty ratio selection

  84.    00: Static duty

  85.    01: 1/2 duty

  86.    10: 1/3 duty

  87.    11: 1/4 duty      */


  88. //----1/3 bias----     

  89.   LCD_CR1_B2=0;//1/3 bias

  90. /* 0: 1/3 bias

  91.    1: 1/2 bias  */


  92. //----内部电压源----   

  93.   LCD_CR2_VSEL=0;

  94.   

  95. //----打开引脚的SEG功能----     

  96. // LCD_PM0=0xFF;//头文件这个地方定义错误,无法直接向LCD_PM0写入数据

  97. // LCD_PM1=0xFF;//PM0寄存器定义错误,导致PM1也无法直接写入

  98. // LCD_PM2=0xFF;//PM0寄存器定义错误,导致PM2也无法直接写入

  99.   *((uint8_t *)0x5404)=0xFF;//直接向LCD_PM0寄存器的地址写入数据

  100.   *((uint8_t *)0x5405)=0xFF;//直接向LCD_PM1寄存器的地址写入数据

  101.   *((uint8_t *)0x5406)=0xFF;//直接向LCD_PM2寄存器的地址写入数据

  102.   

  103. //----To set contrast to mean value----   

  104.   LCD_CR2_CC0=0;//对比度

  105.   LCD_CR2_CC1=1;

  106.   LCD_CR2_CC2=0;

  107. /*  000: VLCD0  2.6V

  108.     001: VLCD1  2.7V

  109.     010: VLCD2  2.8V

  110.     011: VLCD3  2.9V

  111.     100: VLCD4  3.0V

  112.     101: VLCD5  3.1V

  113.     110: VLCD6  3.2V

  114.     111: VLCD7    */

  115.         

  116. //----Dead time 0----         

  117.   LCD_CR3_DEAD0=0;//no dead time

  118.   LCD_CR3_DEAD1=0;  

  119.   LCD_CR3_DEAD2=0;

  120. //----LCD_PulseOnDuration_1----

  121.   LCD_CR2_PON0=1;

  122.   LCD_CR2_PON1=0;  

  123.   LCD_CR2_PON2=0;         

  124. /*  000: 0 CLKps pulses

  125.     001: 1 CLKps pulses

  126.     010: 2 CLKps pulses

  127.     011: 3 CLKps pulses

  128.     100: 4 CLKps pulses

  129.     101: 5 CLKps pulses

  130.     110: 6 CLKps pulses

  131.     111: 7 CLKps pulses  */

  132.         

  133. //----Enable LCD peripheral----        

  134.   LCD_CR3_LCDEN=1;

  135. }


  136. /******************************************************************************************************

  137. *  名          称:LCD_DisplayNum(uint8_t number)

  138. *  功            能:控制段式液晶屏的数字显示部分

  139. *  入口参数:number:要显示的数字

  140. *  出口参数:无

  141. *  说          明:根据数字的长度,判断要显示的长度,长度大于6位,只显示后六位

  142. *  范          例:无

  143. ******************************************************************************************************/

  144. void LCD_DisplayNum(uint32_t number)

  145. {

  146.   uint8_t cnts=0,tmp=0;

  147.   

  148.   if(number<10)cnts=1;

  149.   else if(number<100)cnts=2;

  150.   else if(number<1000)cnts=3;

  151.   else if(number<10000)cnts=4;

  152.   else if(number<100000)cnts=5;

  153.   else if(number<(uint32_t) 1000000)cnts=6;

  154.   else cnts=6;


  155.   //判断需要显示数字的长度,确定在LCD屏上需要的位数

  156.   switch(cnts)

  157.   {

  158.     case 6:

  159.            tmp = LCD_Tab[number%1000000/100000];

  160.            ((tmp&m)==0)?(LCD_RAM0&=~0x02):(LCD_RAM0 |=0x02) ;

  161.            ((tmp&e)==0)?(LCD_RAM0&=~0x01):(LCD_RAM0 |=0x01) ;

  162.            ((tmp&g)==0)?(LCD_RAM2&=~0x80):(LCD_RAM2 |=0x80) ;

  163.            ((tmp&b)==0)?(LCD_RAM2&=~0x40):(LCD_RAM2 |=0x40) ;

  164.            ((tmp&f)==0)?(LCD_RAM6&=~0x08):(LCD_RAM6 |=0x08) ;

  165.            ((tmp&a)==0)?(LCD_RAM6&=~0x04):(LCD_RAM6 |=0x04) ;

  166.            ((tmp&c)==0)?(LCD_RAM3&=~0x20):(LCD_RAM3 |=0x20) ;

  167.            ((tmp&d)==0)?(LCD_RAM3&=~0x10):(LCD_RAM3 |=0x10) ;                     

  168.     case 5:

  169.            tmp = LCD_Tab[number%100000/10000];

  170.            ((tmp&m)==0)?(LCD_RAM0&=~0x08):(LCD_RAM0 |=0x08) ;

  171.            ((tmp&e)==0)?(LCD_RAM0&=~0x04):(LCD_RAM0 |=0x04) ;

  172.            ((tmp&g)==0)?(LCD_RAM2&=~0x20):(LCD_RAM2 |=0x20) ;

  173.            ((tmp&b)==0)?(LCD_RAM2&=~0x10):(LCD_RAM2 |=0x10) ;

  174.            ((tmp&f)==0)?(LCD_RAM6&=~0x02):(LCD_RAM6 |=0x02) ;

  175.            ((tmp&a)==0)?(LCD_RAM6&=~0x01):(LCD_RAM6 |=0x01) ;

  176.            ((tmp&c)==0)?(LCD_RAM3&=~0x80):(LCD_RAM3 |=0x80) ;

  177.            ((tmp&d)==0)?(LCD_RAM3&=~0x40):(LCD_RAM3 |=0x40) ;           

  178.            

  179.     case 4:

  180.            tmp = LCD_Tab[number%10000/1000];


  181.            ((tmp&m)==0)?(LCD_RAM0&=~0x20):(LCD_RAM0 |=0x20) ;

  182.            ((tmp&e)==0)?(LCD_RAM0&=~0x10):(LCD_RAM0 |=0x10) ;

  183.            ((tmp&g)==0)?(LCD_RAM2&=~0x08):(LCD_RAM2 |=0x08) ;

  184.            ((tmp&b)==0)?(LCD_RAM2&=~0x04):(LCD_RAM2 |=0x04) ;

  185.            ((tmp&f)==0)?(LCD_RAM5&=~0x80):(LCD_RAM5 |=0x80) ;

  186.            ((tmp&a)==0)?(LCD_RAM5&=~0x40):(LCD_RAM5 |=0x40) ;

  187.            ((tmp&c)==0)?(LCD_RAM4&=~0x02):(LCD_RAM4 |=0x02) ;

  188.            ((tmp&d)==0)?(LCD_RAM4&=~0x01):(LCD_RAM4 |=0x01) ;           

  189.     case 3:

  190.            tmp = LCD_Tab[number%1000/100];

  191.            ((tmp&m)==0)?(LCD_RAM0&=~0x80):(LCD_RAM0 |=0x80) ;

  192.            ((tmp&e)==0)?(LCD_RAM0&=~0x40):(LCD_RAM0 |=0x40) ;

  193.            ((tmp&g)==0)?(LCD_RAM2&=~0x02):(LCD_RAM2 |=0x02) ;

  194.            ((tmp&b)==0)?(LCD_RAM2&=~0x01):(LCD_RAM2 |=0x01) ;

  195.            ((tmp&f)==0)?(LCD_RAM5&=~0x20):(LCD_RAM5 |=0x20) ;

  196.            ((tmp&a)==0)?(LCD_RAM5&=~0x10):(LCD_RAM5 |=0x10) ;

  197.            ((tmp&c)==0)?(LCD_RAM4&=~0x08):(LCD_RAM4 |=0x08) ;

  198.            ((tmp&d)==0)?(LCD_RAM4&=~0x04):(LCD_RAM4 |=0x04) ;


  199.    case 2:

  200.            tmp = LCD_Tab[number%100/10];

  201.            ((tmp&m)==0)?(LCD_RAM1&=~0x02):(LCD_RAM1 |=0x02) ;

  202.            ((tmp&e)==0)?(LCD_RAM1&=~0x01):(LCD_RAM1 |=0x01) ;

  203.            ((tmp&g)==0)?(LCD_RAM1&=~0x80):(LCD_RAM1 |=0x80) ;

  204.            ((tmp&b)==0)?(LCD_RAM1&=~0x40):(LCD_RAM1 |=0x40) ;

  205.            ((tmp&f)==0)?(LCD_RAM5&=~0x08):(LCD_RAM5 |=0x08) ;

  206.            ((tmp&a)==0)?(LCD_RAM5&=~0x04):(LCD_RAM5 |=0x04) ;

  207.            ((tmp&c)==0)?(LCD_RAM4&=~0x20):(LCD_RAM4 |=0x20) ;

  208.            ((tmp&d)==0)?(LCD_RAM4&=~0x10):(LCD_RAM4 |=0x10) ;


  209.    case 1:

  210.            tmp = LCD_Tab[number%10];

  211.            ((tmp&m)==0)?(LCD_RAM1&=~0x08):(LCD_RAM1 |=0x08) ;

  212.            ((tmp&e)==0)?(LCD_RAM1&=~0x04):(LCD_RAM1 |=0x04) ;

  213.            ((tmp&g)==0)?(LCD_RAM1&=~0x20):(LCD_RAM1 |=0x20) ;

  214.            ((tmp&b)==0)?(LCD_RAM1&=~0x10):(LCD_RAM1 |=0x10) ;

  215.            ((tmp&f)==0)?(LCD_RAM5&=~0x02):(LCD_RAM5 |=0x02) ;

  216.            ((tmp&a)==0)?(LCD_RAM5&=~0x01):(LCD_RAM5 |=0x01) ;

  217.            ((tmp&c)==0)?(LCD_RAM4&=~0x80):(LCD_RAM4 |=0x80) ;

  218.            ((tmp&d)==0)?(LCD_RAM4&=~0x40):(LCD_RAM4 |=0x40) ;

  219.     break;


  220.     default:break;

  221.   }

  222. }

  223. void main(void)

  224. {

  225.   LCD_Config();

  226.   LCD_DisplayNum(201609);

  227. //  asm("rim");               //enable interrupts

  228.   while(1)

  229.   {

  230.     asm("wfi");

  231.   }

  232. }
复制代码

分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏 分享淘帖 顶 踩
回复

使用道具 举报

沙发
ID:466148 发表于 2019-4-2 10:02 来自手机 | 只看该作者
今天早上看了一下,感觉是lcd显示程序有问题,调用lcdconfig()在程序中放的位置不同,就会有退位后已经是0伏,仍然还显示1,另外两位也会出现同样问题。
回复

使用道具 举报

板凳
ID:466148 发表于 2019-4-10 07:59 来自手机 | 只看该作者
问题已经解决了,主要是原程序中没有清屏语句,在写入数据到显示缓存之前加上lcd_clear()函数就没有问题了。
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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