久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 6794|回復: 5
收起左側

STC15W4K32S4單片機ADC讀取 串口收發 設定定時器的代碼與Proteus仿真圖

[復制鏈接]
ID:867327 發表于 2021-1-3 13:46 | 顯示全部樓層 |閱讀模式
文件夾 e:\dpj\20210101
目標 STC15W4K32S4
proteus 新建單片機 串口 分壓調節電壓源 更改單片機主頻為11059200
虛擬串口軟件安裝及配置
單片機使用COM4 串口調試器使用COM3 波特率 115200
ADC讀取電壓數據正常 P1.0 P1.1
串口通訊測試正常 ADC 通過可變電阻調節電壓后 模擬量變化發送串口
串口通訊測試正常 中斷接收<>之間字符,回送  因地址空間,字符個數<10
程序運行時間以protues左下角的運行時間為準  見截圖
定時器設置及中斷處理 以1ms為周期,每3000ms在串口發送字符
串口命令設置定時器發送數據到串口的時間間隔 <后的第一個字符為A 第二個為數字 表示間隔秒數
仿真原理圖如下(STC15W4K32S4單片機proteus仿真工程文件可到本帖附件中下載)
運行畫面.jpg

單片機源程序如下:
  1. /*---------------------------------------------------------------------*/
  2. /* --- STC MCU Limited ------------------------------------------------*/
  3. /* --- STC15F4K60S4 系列 定時器1用作串口1的波特率發生器舉例------------*/
  4. /* 如果要在程序中使用此代碼,請在程序中注明使用了STC的資料及程序        */
  5. /* 如果要在文章中應用此代碼,請在文章中注明使用了STC的資料及程序        */
  6. /*---------------------------------------------------------------------*/

  7. //本示例在Keil開發環境下請選擇Intel的8058芯片型號進行編譯
  8. //若無特別說明,工作頻率一般為11.0592MHz

  9. #include        "STC15Fxxxx.H"
  10. //#include "reg51.h"
  11. #include <stdio.h>

  12. typedef unsigned char BYTE;
  13. typedef unsigned int WORD;
  14. //typedef unsigned int u16;  //對數據類型進行聲明定義
  15. //typedef unsigned char u8;

  16. #define FOSC 11059200L          //系統頻率
  17. #define BAUD 115200             //串口波特率

  18. #define NONE_PARITY     0       //無校驗
  19. #define ODD_PARITY      1       //奇校驗
  20. #define EVEN_PARITY     2       //偶校驗
  21. #define MARK_PARITY     3       //標記校驗
  22. #define SPACE_PARITY    4       //空白校驗

  23. #define PARITYBIT NONE_PARITY   //定義校驗位
  24. /* STC15Fxxxx.H中已定義
  25. //sfr P0M1 = 0x93;
  26. //sfr P0M0 = 0x94;
  27. //sfr P1M1 = 0x91;
  28. //sfr P1M0 = 0x92;
  29. //sfr P2M1 = 0x95;
  30. //sfr P2M0 = 0x96;
  31. //sfr P3M1 = 0xb1;
  32. //sfr P3M0 = 0xb2;
  33. //sfr P4M1 = 0xb3;
  34. //sfr P4M0 = 0xb4;
  35. //sfr P5M1 = 0xC9;
  36. //sfr P5M0 = 0xCA;
  37. //sfr P6M1 = 0xCB;
  38. //sfr P6M0 = 0xCC;
  39. //sfr P7M1 = 0xE1;
  40. //sfr P7M0 = 0xE2;

  41. //sfr AUXR  = 0x8e;               //輔助寄存器

  42. //sfr P_SW1   = 0xA2;             //外設功能切換寄存器1
  43. */
  44. #define S1_S0 0x40              //P_SW1.6
  45. #define S1_S1 0x80              //P_SW1.7

  46. #define ADC_POWER   0x80            //ADC電源控制位
  47. #define ADC_FLAG    0x10            //ADC完成標志
  48. #define ADC_START   0x08            //ADC起始控制位
  49. #define ADC_SPEEDLL 0x00            //540個時鐘
  50. #define ADC_SPEEDL  0x20            //360個時鐘
  51. #define ADC_SPEEDH  0x40            //180個時鐘
  52. #define ADC_SPEEDHH 0x60            //90個時鐘

  53. //sbit P22 = P2^2;
  54. unsigned int ch;a,b;
  55. unsigned int dd=0;
  56. unsigned int cc=0;
  57. unsigned int cc0=0;
  58. unsigned int cc1=0;
  59. unsigned int cc00=0;
  60. unsigned int cc01=0;
  61. unsigned int cc10=0;
  62. unsigned int cc11=0;
  63. unsigned int time0count0 = 0 ;
  64. unsigned int time0count1 = 0 ;
  65. unsigned int time0count = 3000 ;
  66. bit busy;
  67. char receivechar = 0;
  68. int receive_count = 0;
  69. int received = 0;
  70. WORD receivestr[10];

  71. void InitUart();
  72. void InitADC();
  73. void Timer0Init();
  74. void SendData(BYTE dat);
  75. void SendString(char *s);
  76. void serial_put_uint(unsigned int number);
  77. void Delay(WORD n);
  78. //void ShowResult(BYTE ch);
  79. unsigned int GetADCResultint(BYTE ch);
  80. void main()
  81. {
  82.         int initreceivestrcount=0;
  83.         unsigned int tt0=0,tt1=0;
  84. //                unsigned int d1,d2,d3,i;
  85.         unsigned int i=0;
  86.         if (i == 0 )                //初始化tt0 tt1 P0 -P7 數據
  87.         {
  88.                 /*
  89.                 例如將P1M0=0X60和P1M1=0X00譯成bai二進制代碼是
  90.                 P1M0=0110 0000,duP1M1=0000 0000.
  91.                 按照I/O口工作模式設置zhi規定
  92.                 都為0時為準雙向口(dao傳統8051單片機zhuanI/O口模式)
  93.                 為0,1時為推挽(shu強上拉輸出),
  94.                 為1,0時為高阻(僅輸入),
  95.                 為1,1時為開漏(內部上拉斷開,須外加上拉電阻)
  96.                 你選擇的是P1M0和P1M1所以針對的是P1口的操作,
  97.                 將兩個二進制代碼一位位的比較可知:P1.0~P1.4及P1.7都為0,
  98.                 故這些I/O口設置的是標準狀態;P1.5和P1.6為高阻態
  99.                 (P1M0上的第六位和第七位都為1,P1M1上的第六位和第七位都是0。1,0時為高阻)
  100.                 */
  101.                 tt0=16640;
  102.                 tt1=tt0/256;
  103.                 P0M0 = 0x00;
  104.                 P0M1 = 0x00;
  105.                 P1M0 = 0x00;
  106.                 P1M1 = 0x03;
  107.                 P2M0 = 0x00;
  108.                 P2M1 = 0x00;
  109.                 P3M0 = 0x00;
  110.                 P3M1 = 0x00;
  111.                 P4M0 = 0x00;
  112.                 P4M1 = 0x00;
  113.                 P5M0 = 0x00;
  114.                 P5M1 = 0x00;
  115.                 P6M0 = 0x00;
  116.                 P6M1 = 0x00;
  117.                 P7M0 = 0x00;
  118.                 P7M1 = 0x00;
  119.         }
  120.         InitUart();                     //初始化串口
  121.         //printf ("20200102 第一次調試  a  1 取左字節   a %d A/256=A %d   C51\n" , tt0,tt1 ); /* ?? "Hello C51 " */
  122.         SendString("20200102 第二次調試 \r\n");
  123.         initreceivestrcount=0;
  124.         while (initreceivestrcount<10)                //初始化 receivestr
  125.         {
  126.                 receivestr[initreceivestrcount]=0;
  127.                 initreceivestrcount = initreceivestrcount + 1 ;
  128.         }
  129.         //printf ("Hello");  //printf 不能使用 編譯通過 不能仿真運行
  130.         InitADC();                      //初始化ADC
  131.         Timer0Init();  //定時器0初始化
  132.         /* 除法測試
  133.                 //printf ("Hello  ADC 第y次調試 a  1 加代碼a %d A/256=A %d   C51\n" , tt0,tt1 );  
  134. //                sprintf(str," receive %d \r\n",receivechar);
  135. //                SendString(str);
  136. //                sprintf(str," d1 %d a \r\n",tt0);
  137. //                SendString(str);
  138. //                sprintf(str,"  d2 %d  a \r\n",tt1);
  139. //                SendString(str);
  140. //                sprintf(str,"  d3 %d a \r\n",d3);
  141. //                SendString(str);
  142.         */
  143.     SendString("c STC15W4kS32\r\n Uart Test !\r\n");
  144.     while(1)
  145.                 {

  146.                         if (received == 1)                // 在串口接收預定義字符后回送  20210103 增加控制定時顯示功能
  147.                         {
  148.                                 /*  除法測試及sprintf
  149. //                                SendString("\r\n");
  150. //                                d1 = receivechar;
  151. //                                d2 = d1 ;
  152. //                                d3=d2*256;
  153. //                                d3=d1-d3;
  154. //                                sprintf(str," receive %d a\r\n",receivechar);
  155. //                                SendString(str);
  156. //                                sprintf(str," d1 %d b \r\n",d1);
  157. //                                SendString(str);
  158. //                                sprintf(str,"  d2 %d  c \r\n",d2);
  159. //                                SendString(str);
  160. //                                sprintf(str,"  d3 %d d \r\n",d3);
  161. //                                SendString(str);
  162. //                                SendString(receivestr);receivechar
  163.                                 */
  164.                                 serial_put_uint(receivechar);
  165.                                 SendString(" start ");
  166.                                 //serial_put_uint(11);
  167.                                 for (i=0;i<receive_count;i++)
  168.                                 {
  169.                                         SendData(receivestr[i]);
  170.                                 }
  171.                                 SendString(" COUNT= ");
  172.                                 serial_put_uint(receive_count);
  173.                                 receive_count = 0 ;
  174.                                 if (receivestr[1] == 65 )
  175.                                 {
  176.                                         time0count = ( receivestr[2] - 48 ) * 1000 ;
  177.                                         SendString(" \r\n  串口設定 time0   ");                                                                                                
  178.                                         serial_put_uint(time0count/1000);
  179.                                         SendString(" 秒定時  ");        
  180.                                 }                                       
  181.                                 SendString(" \r\n");
  182.                                 received = 0 ;
  183.                         }
  184.                         if (time0count0 != time0count1)                         // 串口設定的秒定時
  185.                         {
  186.                                 SendString(" time0   ");                                                                                                
  187.                                 serial_put_uint(time0count/1000);
  188.                                 SendString(" 秒定時 執行 ");                                                                                                
  189.                                 serial_put_uint(time0count0);
  190.                                 SendString(" \r\n ");               
  191.                                 time0count1 = time0count0;
  192.                         }                                
  193.                         b=b+1;
  194.                         dd=0;
  195.                         cc00 = GetADCResultint(dd);                 //讀取ADC 0 P1.0 數據
  196.                         if (cc01 != cc00)                         // ADC 0 P1.0 數據變化發送串口
  197.                         {
  198.                                 SendString(" ADC ");                                                                                                
  199. //                                serial_put_uint(b);
  200.                                 SendString(" ch ");                                                                                                
  201.                                 serial_put_uint(dd);
  202.                                 SendString(" new ");                                                                                                
  203.                                 serial_put_uint(cc00);
  204.                                 SendString("  old ");                                                                                                
  205.                                 serial_put_uint(cc01);
  206.                                 SendString(" end \r\n");                                                                                                
  207.                                 //printf ("  ADC %d  ch %d :0x  %x 0d %d  \n", b,dd,cc00,cc01 );
  208.                                 cc01 = cc00;
  209.                         }
  210.                         dd=1;
  211.                         cc10 = GetADCResultint(dd);                        //讀取ADC 1 P1.1 數據
  212.                         if (cc11 != cc10)                                // ADC 1 P1.1 數據變化發送串口
  213.                         {
  214.                                 SendString(" ADC ");                                                                                                
  215. //                                serial_put_uint(b);
  216.                                 SendString(" ch ");                                                                                                
  217.                                 serial_put_uint(dd);
  218.                                 SendString(" new ");                                                                                                
  219.                                 serial_put_uint(cc10);
  220.                                 SendString("  old ");                                                                                                
  221.                                 serial_put_uint(cc11);
  222.                                 SendString(" end \r\n");                                                                                                
  223.                                 //printf ("  ADC  %d ch %d :0x  %x 0d00 %d  \n", b,dd,cc10,cc11 );
  224.                                 cc11=cc10;
  225.                         }
  226.                         dd=0;
  227.                         if  (b>10000) b=0;

  228.                 }
  229. }

  230. /*******************************************************************************
  231. * 函 數 名         : Timer0Init
  232. * 函數功能                   : 定時器0初始化
  233. * 輸    入         : 無
  234. * 輸    出         : 無
  235. *******************************************************************************/
  236. void Timer0Init()
  237. {
  238.         TMOD|=0X01;//選擇為定時器0模式,工作方式1,僅用TR0打開啟動。

  239.         TH0=0XFC;        //給定時器賦初值,定時1ms
  240.         TL0=0X18;        
  241.         ET0=1;//打開定時器0中斷允許
  242.         EA=1;//打開總中斷
  243.         TR0=1;//打開定時器                        
  244. }
  245. /*----------------------------
  246. 初始化串口
  247. ----------------------------*/
  248. void InitUart()
  249. {
  250. //    ACC = P_SW1;
  251. //    ACC &= ~(S1_S0 | S1_S1);    //S1_S0=0 S1_S1=0
  252. //    P_SW1 = ACC;                //(P3.0/RxD, P3.1/TxD)
  253.    
  254. //  ACC = P_SW1;
  255. //  ACC &= ~(S1_S0 | S1_S1);    //S1_S0=1 S1_S1=0
  256. //  ACC |= S1_S0;               //(P3.6/RxD_2, P3.7/TxD_2)
  257. //  P_SW1 = ACC;  
  258. //  
  259. //  ACC = P_SW1;
  260. //  ACC &= ~(S1_S0 | S1_S1);    //S1_S0=0 S1_S1=1
  261. //  ACC |= S1_S1;               //(P1.6/RxD_3, P1.7/TxD_3)
  262. //  P_SW1 = ACC;  

  263. #if (PARITYBIT == NONE_PARITY)
  264.     SCON = 0x50;                //8位可變波特率
  265. #elif (PARITYBIT == ODD_PARITY) || (PARITYBIT == EVEN_PARITY) || (PARITYBIT == MARK_PARITY)
  266.     SCON = 0xda;                //9位可變波特率,校驗位初始為1
  267. #elif (PARITYBIT == SPACE_PARITY)
  268.     SCON = 0xd2;                //9位可變波特率,校驗位初始為0
  269. #endif

  270.     AUXR = 0x40;                //定時器1為1T模式
  271.     TMOD = 0x00;                //定時器1為模式0(16位自動重載)
  272.     TL1 = (65536 - (FOSC/4/BAUD));   //設置波特率重裝值
  273.     TH1 = (65536 - (FOSC/4/BAUD))>>8;
  274.     TR1 = 1;                    //定時器1開始啟動
  275.     ES = 1;                     //使能串口中斷
  276.     EA = 1;
  277.         
  278.         
  279.         
  280. //    SCON = 0x5a;                //設置串口為8位可變波特率
  281. //#if URMD == 0
  282. //    T2L = (65536 - (FOSC/4/BAUD));
  283. //    T2H = (65536 - (FOSC/4/BAUD)) >> 8;
  284. //    AUXR = 0x14;                //T2為1T模式, 并啟動定時器2
  285. //    AUXR |= 0x01;               //選擇定時器2為串口1的波特率發生器
  286. //#elif URMD == 1
  287. //    AUXR = 0x40;                //定時器1為1T模式
  288. //    TMOD = 0x00;                //定時器1為模式0(16位自動重載)
  289. //    TL1 = (65536 - (FOSC/4/BAUD));
  290. //    TH1 = (65536 - (FOSC/4/BAUD)) >> 8;
  291. //    TR1 = 1;                    //定時器1開始啟動
  292. //#else
  293. //    TMOD = 0x20;                //設置定時器1為8位自動重裝載模式
  294. //    AUXR = 0x40;                //定時器1為1T模式
  295. //    TH1 = TL1 = (256 - (FOSC/32/BAUD));
  296. //    TR1 = 1;
  297. //#endif
  298. }

  299. /*----------------------------
  300. 初始化ADC
  301. ----------------------------*/
  302. void InitADC()
  303. {
  304.                 P1M0 =1;
  305.                 P1M1 =0;
  306.                 P1 = 0Xff;
  307.     P1ASF = 0xff;                   //設置P1口為AD口
  308.     ADC_RES = 0;                    //清除結果寄存器
  309.     ADC_RESL = 0;                    //清除結果寄存器
  310.     ADC_CONTR = ADC_POWER | ADC_SPEEDLL;
  311.     Delay(2);                       //ADC上電并延時
  312. }

  313. /*----------------------------
  314. 讀取ADC結果 20210101 注釋GetADCResult 用GetADCResultint代替
  315. ----------------------------*/

  316. unsigned int GetADCResultint(BYTE ch)
  317. {
  318.                 unsigned int ret=0;
  319.     ADC_CONTR = ADC_POWER | ADC_SPEEDLL | ch | ADC_START;
  320.     _nop_();                        //等待4個NOP
  321.     _nop_();
  322.     _nop_();
  323.     _nop_();
  324.     while (!(ADC_CONTR & ADC_FLAG));//等待ADC轉換完成
  325.     ADC_CONTR &= ~ADC_FLAG;         //Close ADC
  326.                 ret = ADC_RES*4 + ADC_RESL;
  327.     return ret;                 //返回ADC結果
  328. }


  329. /*----------------------------
  330. UART 中斷服務程序
  331. -----------------------------*/
  332. void Uart() interrupt 4
  333. {

  334.     if (RI)
  335.     {
  336.         RI = 0;                 //清除RI位
  337.         receivechar = SBUF;              //P0顯示串口數據
  338.                         if  ( receivechar == 60 ) receive_count = 0 ;
  339.                         if ((receive_count<9) && ( receivechar != 62 )&&(received == 0 ))
  340.                         {
  341.                                 receivestr[receive_count] = receivechar;
  342.                                 receivestr[receive_count+1] = 0;
  343.                                 receive_count = receive_count + 1 ;
  344.                                 
  345.                         }
  346.                         else
  347.                         {
  348.                                 received = 1;
  349. //                                while (initreceivestrcount<10)
  350. //                                {
  351. //                                        receivestr[initreceivestrcount]=0;
  352. //                                        initreceivestrcount = initreceivestrcount + 1 ;
  353. //                                }
  354.                                 
  355.                         }
  356. //                        SendData(receivechar);
  357. //        P22 = RB8;              //P2.2顯示校驗位
  358.     }
  359.                         
  360.     if (TI)
  361.     {
  362.         TI = 0;                 //清除TI位
  363.         busy = 0;               //清忙標志
  364.     }
  365. }

  366. void Timer0() interrupt 1
  367. {
  368.         static u16 itime0;
  369.         TH0=0XFC;        //給定時器賦初值,定時1ms
  370.         TL0=0X18;
  371.         itime0++;
  372.         if(itime0==time0count)
  373.         {
  374.                 itime0=0;
  375.                 time0count0 = time0count0 + 1;        
  376.                 if ( time0count0 > 20 ) time0count0 = 0;
  377.         }        
  378. }

  379. /*----------------------------
  380. 發送串口數據
  381. ----------------------------*/
  382. void SendData(BYTE dat)
  383. {
  384.     while (busy);               //等待前面的數據發送完成
  385.     ACC = dat;                  //獲取校驗位P (PSW.0)
  386. //    if (P)                      //根據P來設置校驗位
  387. //    {
  388. //#if (PARITYBIT == ODD_PARITY)
  389. //        TB8 = 0;                //設置校驗位為0
  390. //#elif (PARITYBIT == EVEN_PARITY)
  391. //        TB8 = 1;                //設置校驗位為1
  392. //#endif
  393. //    }
  394. //    else
  395. //    {
  396. //#if (PARITYBIT == ODD_PARITY)
  397. //        TB8 = 1;                //設置校驗位為1
  398. //#elif (PARITYBIT == EVEN_PARITY)
  399. //        TB8 = 0;                //設置校驗位為0
  400. //#endif
  401. //    }
  402.     busy = 1;
  403.     SBUF = ACC;                 //寫數據到UART數據寄存器
  404. }

  405. /*----------------------------
  406. 發送字符串
  407. ----------------------------*/
  408. void SendString(char *s)
  409. {
  410. //        EA = 0 ;
  411. //        ES = 0 ;
  412.         while (*s)                  //檢測字符串結束標志
  413.         {
  414.                         SendData(*s++);         //發送當前字符
  415.         }
  416. //        EA = 1 ;
  417. //        ES = 1 ;
  418. }
  419. void serial_put_uint(unsigned int number)
  420. {
  421.       unsigned int i;
  422.       unsigned int nTmp;
  423. //        *s++ =num/10 +'0';
  424. //*s++=num%10 +'0';
  425. //*s =0;
  426.         
  427.      //感覺這寫法怪怪的嗎?干嘛不for(i=1;i<5;i++) 因為這樣不需要使用庫函數pow()
  428.      for(i=10000;i>=1;i=i/10)
  429.       {
  430.             nTmp = number/i;
  431.                                                 if(nTmp >=1 ) SendData((unsigned char)(nTmp%10 + 0x30));
  432.       }
  433. }

  434. /*----------------------------
  435. 軟件延時
  436. ----------------------------*/
  437. void Delay(WORD n)
  438. {
  439.     WORD x;

  440.     while (n--)
  441.     {
  442.         x = 5000;
  443.         while (x--);
  444.     }
  445. }
復制代碼
51hei.png
所有資料51hei提供下載:
20210103a.rar (580.92 KB, 下載次數: 243)

評分

參與人數 1黑幣 +50 收起 理由
admin + 50 共享資料的黑幣獎勵!

查看全部評分

回復

使用道具 舉報

ID:328014 發表于 2021-1-3 14:23 | 顯示全部樓層
好東東,可惜我用Proteus8.8居然打不開這個工程文件,樓主是用的8.9嗎?
回復

使用道具 舉報

ID:867327 發表于 2021-1-3 16:48 | 顯示全部樓層
Proteus8.10
回復

使用道具 舉報

ID:996527 發表于 2022-3-25 12:50 | 顯示全部樓層
厲害,可惜我版本太低打不開
回復

使用道具 舉報

ID:192215 發表于 2022-5-16 10:30 | 顯示全部樓層
樓主遇沒遇到過打開后LCD白屏問題
回復

使用道具 舉報

ID:1058975 發表于 2023-1-30 18:49 | 顯示全部樓層
不錯,參考一下 STC 定時器
回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

手機版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 久久免费福利 | 在线成人| 欧美日韩综合一区 | 欧美色性 | 免费视频一区二区 | 久久久久久免费毛片精品 | 国产精品日日夜夜 | 99久久99 | 亚洲欧美日韩精品久久亚洲区 | 午夜免费视频 | 日韩一区二区在线播放 | 国产乱码久久久 | 国产情侣久久 | 色综合久 | 精品国产欧美日韩不卡在线观看 | av一区二区三区四区 | 一区精品在线观看 | 欧美电影免费观看 | 亚洲毛片在线 | 久久久精品网站 | 四虎最新地址 | 91精品国产综合久久婷婷香蕉 | 日韩av一区二区在线观看 | 久久久久一区 | 亚洲精品免费在线 | 国产精品欧美日韩 | 国产精品1 | 天天成人综合网 | 懂色av一区二区三区在线播放 | 永久精品 | 久久久九九 | 中文字幕一级 | 韩日一区二区三区 | 亚洲va欧美va人人爽午夜 | 久久久av中文字幕 | 亚洲欧美日韩久久 | 九九久久久 | 国产亚洲一区二区三区在线观看 | 精品国产一区二区 | 三区在线观看 | 看片91|