找回密码
 立即注册

QQ登录

只需一步,快速开始

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

stm8L152R8T6串口UART调试,波特率计算与程序源码

[复制链接]
跳转到指定楼层
楼主
先总结一些调试串口遇到的问题
1、串口调试接收开始多了一个0x00,或结尾多了一个0x00,前者我遇到的是串口寄存器(USART1_CR2)配置问题,后者是串口发送中断的计数值问题。USART1_DR =  ui8TX_buffer[cnts++];    if(cnts>=ui8TX_number)

2、还有波特率的计算。
USART1_BRR2 [ 15:12    3:0 ]
USART1_BRR1 [ 11 : 4 ]
主时钟频率/分频系数/波特率=装载值
16000000Hz/1/9600=1666.66=1667转换为十六进制0x683
  USART1_BRR2 =0x03;
  USART1_BRR1 =0x68;


单片机源程序如下:
  1. /* MAIN.C file
  2. *
  3. * Copyright (c) 2002-2005 STMicroelectronics
  4. */

  5. #include "STM8L152R.h"
  6. #include "led.h"
  7. #include "MATH.h"
  8. /*
  9. PC2 USART1_RX ---->>> CH340 TX
  10. PC3 USART1_TX ---->>> CH340 RX
  11. */

  12. #define TXBUF_SIZE 16
  13. #define RXBUF_SIZE 16

  14. _Bool bTX_finished=0;
  15. _Bool bRX_finished=0;//串口接收完一帧数据标志位

  16. unsigned int ui8TX_number = 16;

  17. unsigned int ui8RX_counter=16;
  18. //66 64 00 03 04 05 06 07 08 09 0A 0B 0C 0D 64 66
  19. /*
  20.                          0x66 0x64 0x00                                                      0x64 0x66
  21. 发送  |数据包头| 指令| AD 电压 | 气压   |排气调整值|进气调整值|   3byte 空 |数据包尾| //发送信息 指令0x00
  22. 接收  |数据包头| 指令|目标气压 |进气P|I | D | 排气P|  I | D   |   3byte 空 |数据包尾| //接收信息 指令0x01 修改目标气压
  23. 接收  |数据包头| 指令|目标气压 |进气P|I | D | 排气P|  I | D   |   3byte 空 |数据包尾| //接收信息 指令0x02 修改PID参数
  24. */
  25. /***************************************|数据包头| 指令| AD 电压 | 气压   |排气调整值|进气调整值|   3byte 空 |数据包尾|*/
  26. unsigned char ui8TX_buffer[TXBUF_SIZE]={0x66,0x64,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x64,0x66};
  27. unsigned char ui8RX_buffer[RXBUF_SIZE];

  28. char RX_OK=0;
  29. extern unsigned char LED_RAM[3];

  30. #define USART1_SR_TC    USART1_SR|(~0x40)//USART1_SR_TC 为0发送未完成   为1发送完成         
  31. #define USART1_SR_RXNE  USART1_SR|(~0x20)//USART1_SR_RXNE 为1表示接收到了数据  为0没收到数据
  32. void delay_100ms(unsigned int x_ms)
  33. {

  34.   unsigned int i,j;

  35.   for(i=x_ms;i>0;i--)//2*255个指令周期

  36.     for(j=0;j<110;j++);//2*255个指令周期

  37.   //delay_10ms共消耗 x_ms*2*255+2*x_ms个指令周期  

  38.   //255*2*255+2*255=130610us=130ms

  39.   //此延时函数,延时时间为130ms

  40.   //16M/8/2=1M 一个指令周期为1us

  41. }

  42. void MENU_print_Px(int P)
  43. {
  44.          if(P>999)
  45.      {
  46.                         LED_TOP_putc(2,((P)%10)+'0',0);
  47.                   P/=10;
  48.                   LED_TOP_putc(1,((P)%10)+'0',0);
  49.       P/=10;
  50.                   LED_TOP_putc(0,((P)%10)+'A',0);
  51.      }
  52.    else                 
  53.                  {LED_TOP_putc(2,((P)%10)+'0',0);
  54.                   P/=10;
  55.                   LED_TOP_putc(1,((P)%10)+'0',0);
  56.       P/=10;
  57.                   LED_TOP_putc(0,((P)%10)+'0',0);
  58.                  }       
  59. }
  60. void UART_Init(void)
  61. {
  62.     /*********************************/

  63.     PC_DDR = 0<<2;//输入

  64.     PC_CR1 = 1<<2;//上拉输入

  65.    
  66.     PC_ODR = 1<<3;

  67.     PC_CR1 = 1<<3;//推挽输出

  68.     PC_CR2 = 1<<3;//输出摆率10M   

  69.     PC_DDR = 1<<3;//输出高电平,TX空闲状态为高电平,如果不设置,会莫名奇妙的发送0x00


  70.     CLK_PCKENR1=1<<5;//开启USART1外设时钟


  71.     //开启引脚的UART功能

  72.     //00: USART1_TX on PC3 and USART1_RX on PC2

  73.     //01: USART1_TX on PA2 and USART1_RX on PA3

  74.     //10: USART1_TX on PC5 and USART1_RX on PC6

  75. //   SYSCFG_RMPCR1 &= (~0x30);//PC2,PC3 //默认

  76.     //如果需要使用其他引脚作为串口发送接收引脚,请在这里选择

  77.    

  78.     //设置串口工作方式

  79.     USART1_CR1 &= 0x00;//~0X3E;//1 start bit,8 data bit,n stop bit

  80.     USART1_CR3 &= 0x00;//1 stop bit

  81.     //设置波特率

  82.     //波特率设置为9600

  83.     // 2000000/9600=208    //2000000是此时系统的主时钟频率,默认8分频,16M/8=2M

  84.     //208(DEC)=00D0(HEX)

  85.     USART1_BRR2 =0x03;//the BRR2 should be programmed before BRR1

  86.     USART1_BRR1 =0x68;//9600 683


  87.   //  USART1_CR2 = 1<<3;//使能发送

  88.    // USART1_CR2 = 1<<7;//打开发送中断

  89.          // USART1_CR2 = 1<<2;//使能接收
  90.       USART1_CR2=0x24;//打开接收
  91.    // USART1_CR2 = 1<<5;//打开接收中断

  92. }
  93. /*
  94. 发送输入采集电压数据到串口 2byte
  95. 低位在前,高位在后
  96. */
  97. void recover_command(void)
  98. {
  99.         //指令为0  只设置气压参数
  100.         //指令为1  只设定PID参数
  101.         //发送返回为  0 告诉当前是通讯控制
  102.         //发送返回为  1 告诉当前是输入控制
  103.         char SET_Pressure_Pid=0;
  104.        
  105.         SET_Pressure_Pid = ui8TX_buffer[2];  
  106.        
  107. }
  108. void send_ADC_data(int IN_voltage)
  109. {
  110.         ui8TX_buffer[3]=        IN_voltage   ;
  111.         ui8TX_buffer[4]=        IN_voltage>>8;
  112. }
  113. void send_DAC_data(int IN_pressure)
  114. {
  115.         ui8TX_buffer[5]=        IN_pressure   ;
  116.         ui8TX_buffer[6]=        IN_pressure>>8;
  117. }
  118. void main(void)
  119. {

  120.         int  count=0;
  121.         int  num_test=0;
  122.         int  DIR=0;
  123.         char RX_OK_flag=0;
  124.         int  AD_TEST=4096;
  125.         char AD_TEST_DIR=0;

  126.         _asm("sim");  /* Disable interrupts */
  127.         CLK_CKDIVR=0x00;//内部16MHz  不分频
  128.         UART_Init();
  129.         LED_init();
  130.         _asm("rim");
  131.         while(1)
  132.   {
  133.                 count++;
  134.                  if(count % 100 == 0)
  135.                 {
  136.                         if(AD_TEST_DIR == 0)
  137.                         {  
  138.                                 AD_TEST+=100;
  139.                                 if(AD_TEST>16383)
  140.                                 {
  141.                                         AD_TEST_DIR=1;
  142.                                 }
  143.                         }
  144.                         else
  145.                         {
  146.                                 AD_TEST-=100;
  147.                                 if(AD_TEST<0)
  148.                                 {
  149.                                         AD_TEST_DIR=0;
  150.                                 }
  151.                         }
  152.                   send_DAC_data(AD_TEST);  
  153.                   USART1_CR2 |=0x88;//打开发送  
  154.            delay_100ms(200);  
  155.                 }
  156.                 if(count % 10 == 0)
  157.                             {
  158.                                                 if(bRX_finished == 1)//接收完成
  159.                                                 {
  160.                                                  if(ui8RX_buffer[2] == 0x00)//分析接收的命令  修改执行变量
  161.                                                         RX_OK_flag = 1;
  162.                                                         else
  163.                                                         RX_OK_flag = 0;
  164.                                                          bRX_finished = 0; //清除接收完成标志,准备下一次接收
  165.                                                 }
  166.                                          if(RX_OK_flag == 1)//响应接收到的指令,并执行相应操作
  167.                                          {  
  168.                                                         send_ADC_data(num_test);  
  169.                                                         USART1_CR2 |=0x88;//打开发送  返回想要的结果
  170.                                                         RX_OK_flag = 0;//执行操作完成,修改回变量值
  171.                                                           if(DIR == 0)
  172.                                                                  {
  173.                                                                                  num_test+=1;
  174.                                                                                  if(num_test>=8192)//10922
  175.                                                                                  DIR=1;
  176.                                                                  }
  177.                                                                  else
  178.                                                                  {
  179.                                                                                 num_test-=1;
  180.                                                                                 if(num_test<=4096)//5461
  181.                                                                                 DIR=0;
  182.                                                                  }
  183.                                                 }
  184.                                         }
  185.                 delay_100ms(1);
  186.                 if(count>=10000)
  187.                                         count=0;                 
  188.   }
  189. }



  190. /*************************************************************************************************** ***

  191. * 名 称: USART_TX_ISR()

  192. * 功 能: UART的Tx事件处理函数

  193. * 入口参数:无

  194. * 出口参数:无

  195. * 说 明:发送ui8TX_buffer数组中的数据,发送完成后,置位bTX_finished标志位,下一次进入中断后关闭发送

  196. * 范 例:无

  197. ******************************************************************************************************/

  198. @far @interrupt void USART_TX_ISR (void)
  199. {
  200.   static unsigned int cnts=0;

  201.         if(USART1_SR_TC != 1)//不等于1 表示发送未完成
  202.   {
  203.                 USART1_DR =  ui8TX_buffer[cnts++];
  204.     if(cnts>=ui8TX_number)
  205.     {
  206.       cnts=0;
  207.       bTX_finished=1;
  208.                         USART1_CR2 &= ~0x80;//关闭发送中断  
  209.     }  
  210.   }
  211.         else cnts=0;
  212. }

  213. /******************************************************************************************************

  214. * 名               称: USART_RX_ISR()

  215. * 功               能: UART的Tx事件处理函数

  216. * 入口             参数:无

  217. * 出口             参数:无

  218. * 说               明:

  219. * 范               例:无

  220. ******************************************************************************************************/

  221. @far @interrupt void USART_RX_ISR (void)
  222. {
  223.   static unsigned int ui8RX_temp=0;
  224.        
  225.   if(USART1_SR_RXNE)
  226.   {
  227.     ui8RX_buffer[ui8RX_temp++] = USART1_DR;
  228.                 if(ui8RX_temp>=ui8RX_counter)
  229.     {
  230.       ui8RX_temp=0;//接收到正确包头和包尾 共4byte
  231.                         if(ui8RX_buffer[0] == 0x66 && ui8RX_buffer[1] == 0x64 && ui8RX_buffer[14] == 0x64 && ui8RX_buffer[15] == 0x66)
  232.       bRX_finished=1;
  233.                         else
  234.                   bRX_finished=0;  
  235.     }
  236.   }
  237. }
复制代码

所有资料51hei提供下载:
UART.rar (92.02 KB, 下载次数: 41)


评分

参与人数 1黑币 +50 收起 理由
admin + 50 共享资料的黑币奖励!

查看全部评分

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

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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