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

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 11776|回復: 0
打印 上一主題 下一主題
收起左側

MSP430單片機串口通訊程序庫(精簡版)

[復制鏈接]
跳轉到指定樓層
樓主
ID:108615 發表于 2016-3-14 20:05 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式
本程序是《MSP430系列單片機系統工程設計與實踐》書里面的源碼,(包含工程文件(例4.1.6) )完整例程下載:http://www.zg4o1577.cn/bbs/dpj-46245-1.html


關于本程序的詳細解說大家可以點擊上圖下載電子書

這個串口庫文件的調用方法可先下載完整代碼然后找到(例4.1.6)這一節,里面有調用方法
uart.c :
  1. /*
  2.                
  3.                  MSP430F4XX單片機串口通訊程序庫(精簡版)

  4.   說明:該程序庫包括串口初始化、串口單字節收/發函數,以及串口終端設備接口
  5.         函數。可以作為各種程序的底層驅動使用。
  6.             要使用該庫函數,需要將本文件(UART.c)添加進工程,并在需要調用
  7.         串口函數的文件開頭處包含"UART.h"
  8.             本程序庫中的串口初始化函數會根據波特率為串口收發模塊自動選擇最
  9.         合適的時鐘源,自動配置波特率分頻系數,用戶不需要了解串口底層寄存器。
  10.         對于4800bps及以下的波特率,會選擇ACLK作為時鐘;對于4800bps以上設置,
  11.         會選擇SMCLK作時鐘。若串口一直打開,為了降低功耗,盡量使用4800以下波
  12.         特率;若使用4800以上波特率,推薦間歇使用串口。使用38400bps以上波特率
  13.         最好適當提高CPU主頻(2~4MHz)。
  14.             串口收發的等待過程中會將CPU置于休眠模式(LPM0或LPM3),以降低耗電。
  15.         但要注意某些應用中可能不允許關閉時鐘(比如TimerA用SMCLK作時鐘,若串
  16.         口等待過程中進入LPM3,關閉了SMCLK,造成計時錯誤),遇到這類情況請修
  17.         改UART_LPM()函數,刪除休眠語句或降低休眠深度,但功耗會有所增加。
  18.             程序后半部分是標準終端輸入輸出函數,對標準終端設備輸入的數據流進
  19.         行解析。通過串口與PC機相連;在PC機運行超級終端程序,從而將PC的屏幕和鍵
  20.         盤映射成單片機的輸入/輸出終端。這樣在單片機程序中可以使用格式化輸入/
  21.         輸出函數,如printf/scanf等。詳細的原理請參考《超級終端人機對話范例》
  22.         程序。終端輸入函數帶有退格功能,因此有輸入緩沖區。請根據需要自行修改
  23.         輸入行緩沖區大小(LINE_LENGTH),該值限定了終端設備每行最大能輸入的字符數。
  24.      
  25.   (C)西安電子科技大學 測控技術與儀器教研中心 編寫:謝楷 2008/02/21
  26.   
  27. */
  28. //******************************************************************************
  29. //                        
  30. //
  31. //                MSP430FE425
  32. //           +-----------------+
  33. //        /|\|              XIN|-
  34. //         | |                 | 32kHz
  35. //         --|RST          XOUT|-
  36. //           |                 |
  37. //           |        (TXD)P2.4|----------->  //  ----> RXD(2)
  38. //           |                 | 300~115200 bps            
  39. //           |        (RXD)P2.5|<-----------  //  <---- TXD(3)  PC(DB9)
  40. //           |                 |
  41. //           |             GND |------------------------GND(5)
  42. //
  43. //******************************************************************************
  44. #include "msp430x42x.h"
  45. #define F_ACLK  (32768)  /*ACLK對應晶體頻率,勿修改*/

  46. char TxFlag=1;
  47. char RxFlag=0;
  48. /****************************************************************************
  49. * 名    稱:UART_Init()
  50. * 功    能:初始化串口。設置其工作模式及波特率。
  51. * 入口參數:
  52. *           Baud         波特率    (300~115200)
  53.             Parity       奇偶校驗位('n'=無校驗  'p'=偶校驗  'o'=奇校驗)
  54.             DatsBits     數據位位數    (7或8)
  55.             StopBits     停止位位數    (1或2)
  56. * 出口參數:返回值為1時表示初化成功,為0表示參數出錯
  57. * 范    例: UART_Init(9600,'n',8,1) //設成9600bps,無校驗,8位數據,1位停止位
  58.             UART_Init(2400,'p',7,2) //設成2400bps,偶校驗,7位數據,2位停止位
  59. ****************************************************************************/
  60. char UART_Init(long int Baud,char Parity,char DataBits,char StopBits)
  61. {
  62.     unsigned long int BRCLK;       //波特率發生器時鐘頻率
  63.     int FreqMul,FLLDx,BRDIV,BRMOD; //倍頻系數、DCO倍頻、波特率分頻系數、分頻尾數
  64.     int i;
  65.     unsigned char const ModTable[8]={0x00,0x08,0x88,0x2A,0x55,0x6B,0xdd,0xef};
  66.              //分頻尾數所對應的調制系數(將0~7個"1"均勻分布在一個字節的8bit中)

  67.     //-------------設置波特率發生器時鐘源,并計算波特率時鐘頻率--------------
  68.     UTCTL0 &=~(SSEL0+SSEL1); //清除之前的時鐘設置
  69.     if(Baud<=4800)      
  70.     {
  71.       UTCTL0 |= SSEL0;       //低于4800的波特率,用ACLK,降低功耗
  72.       BRCLK=F_ACLK;          //波特率發生器時鐘頻率=ACLK
  73.     }
  74.     else
  75.     {
  76.       UTCTL0 |= SSEL1;            //高于4800的波特率,用SMCLK,保證速度
  77.       FreqMul=(SCFQCTL&0x7F)+1;   //獲得倍頻系數
  78.       FLLDx=((SCFI0&0xC0)>>6)+1;  //獲得DCO倍頻系數(DCOPLUS所帶來的額外倍頻)
  79.       BRCLK=F_ACLK*FreqMul;       //計算波特率發生器時鐘頻率=ACLK*倍頻系數
  80.       if(FLL_CTL0&DCOPLUS) BRCLK*=FLLDx; //若開啟了DCOPLUS,還要計算額外倍頻
  81.     }
  82. //------------------------設置波特率-------------------------   
  83.   if((Baud<300)||(Baud>115200))  return(0);  //波特率范圍300-115200bps
  84.   BRDIV=BRCLK/Baud;                          //計算波特率分頻系數(整數部分)
  85.   BRMOD=((BRCLK*8)/Baud)%8;                  //計算波特率分頻尾數(除不盡的余數)
  86.   UBR00 = BRDIV%256;           
  87.   UBR10 = BRDIV/256;                         //整數部分系數
  88.   UMCTL0 = ModTable[BRMOD];                  //余數部分系數
  89. //------------------------設置校驗位-------------------------  
  90.   switch(Parity)
  91.   {
  92.    case 'n':case'N': U0CTL&=~PENA;     break;        //無校驗
  93.    case 'p':case'P': U0CTL|= PENA+PEV ;break;        //偶校驗
  94.    case 'o':case'O': U0CTL|= PENA; U0CTL&=~PEV;break;//奇校驗  
  95.   default : return(0);               //參數錯誤
  96.   }
  97. //------------------------設置數據位-------------------------   
  98.   switch(DataBits)
  99.   {
  100.    case 7:case'7': U0CTL&=~CHAR; break;   //7位數據
  101.    case 8:case'8': U0CTL|= CHAR; break;   //8位數據
  102.   default : return(0);      //參數錯誤
  103.   }
  104. //------------------------設置停止位-------------------------   
  105.   switch(StopBits)
  106.   {
  107.    case 1:case'1': U0CTL&=~SPB; break;  //1位停止位
  108.    case 2:case'2': U0CTL|= SPB; break;  //2位停止位
  109.   default : return(0);     //參數錯誤
  110.   }
  111.   P2SEL |= 0x30;              // P2.4,5 = USART0 TXD/RXD
  112.   ME1 |= UTXE0 + URXE0;       // Enable USART0 TXD/RXD
  113.   UCTL0 &= ~SWRST;            // Initialize USART state machine
  114.   IE1 |= URXIE0+UTXIE0;       // Enable USART0 RX interrupt
  115.   _EINT();
  116.   for(i=0;i<4000;i++);        //略延遲,等待波特率分頻穩定
  117.   return(1); //設置成功
  118. }
  119. /****************************************************************************
  120. * 名    稱:UART_LPM()
  121. * 功    能:串口收/發等待過程中,將CPU及時鐘系統關閉,休眠省電
  122. * 入口參數:無
  123. * 出口參數:無
  124. * 說    明: 若與其他外設的時鐘沖突,可注釋掉該函數,但會增加功耗。
  125. ****************************************************************************/
  126. void UART_LPM()
  127. {
  128.   if(UTCTL0&SSEL0) LPM3;  //若以ACLK 作時鐘,進入LPM3休眠(僅打開ACLK)
  129.   else             LPM0;  //若以SMCLK作時鐘,進入LPM0休眠(不關閉SMCLK)
  130. }
  131. /****************************************************************************
  132. * 名    稱:UART_PutChar()
  133. * 功    能:從串口發送1字節數據
  134. * 入口參數:Chr:  待發送的一字節數據
  135. * 出口參數:無
  136. * 說    明: 在等待發送完畢的過程中,CPU會休眠
  137. ****************************************************************************/
  138. void UART_PutChar(char Chr)
  139. {
  140.   while (TxFlag==0) UART_LPM(); // 等待上一字節發完,并休眠
  141.   TxFlag=0; //小技巧:等上一字節發完,再發本字節,把字節間隔等待的時間讓給計算
  142.   TXBUF0=Chr;
  143. }
  144. /****************************************************************************
  145. * 名    稱:UART_GetChar()
  146. * 功    能:從串口接收1字節數據
  147. * 入口參數:無  
  148. * 出口參數:收到的一字節數據
  149. * 說    明: 如果串口沒有數據,會一直等待。等待過程中,CPU會休眠
  150. ****************************************************************************/
  151. char UART_GetChar(void)
  152. {
  153.   while (RxFlag==0) UART_LPM(); // 收到一字節?
  154.   RxFlag=0;
  155.   return(RXBUF0);
  156. }
  157. /****************************************************************************
  158. * 名    稱:UART_IsRcvChar()
  159. * 功    能:判斷串口是否收到1字節數據
  160. * 入口參數:無  
  161. * 出口參數:1表示有數據,0表示無數據
  162. * 說    明: 配合UART_GetChar()函數使用,先檢查到有數據再接收,可以消除阻塞
  163. ****************************************************************************/
  164. char UART_IsRcvChar()
  165. {
  166.   return(RxFlag);
  167. }

  168. /****************************************************************************
  169. * 名    稱:UART_RX()
  170. * 功    能:串口接收中斷,每接收到1字節會發生一次中斷
  171. ****************************************************************************/
  172. #pragma vector=UART0RX_VECTOR
  173. __interrupt void UART_RX (void)
  174. {
  175.   RxFlag=1;
  176.   /*在這里添加用戶中斷服務程序代碼,如將數據壓入接收緩沖等*/
  177.   __low_power_mode_off_on_exit();
  178. }
  179. /****************************************************************************
  180. * 名    稱:UART_TX()
  181. * 功    能:串口發送中斷,每發完1字節會發生一次中斷
  182. ****************************************************************************/
  183. #pragma vector=UART0TX_VECTOR
  184. __interrupt void UART_TX (void)
  185. {
  186.   TxFlag=1;
  187.   /*在這里添加用戶中斷服務程序代碼,如將數據從緩沖取出等*/  
  188.   __low_power_mode_off_on_exit();
  189. }

  190. /*===================以下是串口終端設備接口函數庫==========================*/

  191. #define LINE_LENGTH 20          /* 行緩沖區大小,決定每行最多輸入的字符數*/

  192. /*標準終端設備中,特殊ASCII碼定義,請勿修改*/
  193. #define In_BACKSP 0x08          /* ASCII  <--  (退格鍵)  */
  194. #define In_DELETE 0x7F          /* ASCII <DEL> (DEL 鍵)  */
  195. #define In_EOL '\r'             /* ASCII <CR>  (回車鍵)  */
  196. #define In_SKIP '\3'            /* ASCII control-C */
  197. #define In_EOF '\x1A'           /* ASCII control-Z */

  198. #define Out_DELETE "\x8 \x8"    /* VT100 backspace and clear */
  199. #define Out_SKIP "^C\n"         /* ^C and new line */
  200. #define Out_EOF "^Z"            /* ^Z and return EOF */
  201. #include "stdio.h"

  202. /****************************************************************************
  203. * 名    稱:putchar()
  204. * 功    能:向標準終端設備發送一字節數據(1個字符)
  205. * 入口參數:ch: 待發送的字符  
  206. * 出口參數:發出的字符
  207. * 說    明: printf函數會調用該函數作為底層輸出。這里從串口輸出字符到PC機的超
  208.             級終端軟件上,printf的結果將打印到超級終端上。若修改該函數,將字
  209.             符以其他方式輸出,如顯示到LCD上,printf的結果將顯示在LCD上。
  210. ****************************************************************************/
  211. int putchar(int ch)
  212. {
  213.   if (ch == '\n')        //  '\n'(回車)擴展成 '\n''\r' (回車+換行)
  214.   {
  215.     UART_PutChar(0x0d) ; //'\r'
  216.   }
  217.   UART_PutChar(ch);      //從串口發出數據  
  218.   return (ch);
  219. }
  220. /****************************************************************************
  221. * 名    稱:put_message()
  222. * 功    能:向標準終端設備發送一個字符串
  223. * 入口參數:*s: 字符串(數組)頭指針(數組名)
  224. * 出口參數:無
  225. ****************************************************************************/
  226. static void put_message(char *s)
  227. {
  228.   while (*s)        //當前字符不為空 (字符串以0x00結尾)
  229.     putchar(*s++);  //輸出一個字符,指針指向下一字符
  230. }

  231. /****************************************************************************
  232. * 名    稱:getchar()
  233. * 功    能:從標準終端設備接收一字節數據(1個字符)
  234. * 入口參數:無
  235. * 出口參數:收到的字符
  236. * 說    明: scanf函數會調用該函數作為底層輸入。這里從PC機鍵盤借助超級終端軟
  237.             件通過串口輸入字符到單片機上。scanf函數的輸入即源自PC機鍵盤。若
  238.             修改該函數,將字符以其他方式輸入,如單片機IO口,可用按鈕向scanf
  239.             函數輸入數據。本函數帶有緩存,能夠處理退格等刪除操作。若不需要刪
  240.             除操作,可直接調用UART_GetChar()函數。
  241. ****************************************************************************/
  242. int getchar(void)
  243. {
  244.   static char io_buffer[LINE_LENGTH + 2];     /* Where to put chars */
  245.   static int ptr;                             /* Pointer in buffer */
  246.   char c;

  247.   for (;;)
  248.   {
  249.     if (io_buffer[ptr])                 //如果緩沖區有字符
  250.       return (io_buffer[ptr++]);        //則逐個返回字符
  251.     ptr = 0;                            //直到發送完畢為止,清空緩沖區指針
  252.   while(1)                                //緩沖區沒有字符,才會執行到這里,開始等待字符輸入
  253.     { c = UART_GetChar();                //等待串口接收一個字符
  254.       if (c == In_EOF && !ptr)                //----EOF鍵(Ctrl+Z)----
  255.       {                                        //EOF符只能在未輸入其他字符時才有效
  256.         put_message(Out_EOF);                //讓終端顯示EOF符
  257.         return EOF;                        //返回EOF符
  258.       }
  259.       if ((c == In_DELETE)||(c==In_BACKSP)) //----退格或DEL鍵----
  260.       {
  261.         if (ptr)
  262.         {
  263.           ptr--;                                //從緩沖區刪掉一個字符
  264.           put_message(Out_DELETE);                //讓終端顯示也刪掉一個字符
  265.         }
  266.       }
  267.       else if (c == In_SKIP)                        //-----取消鍵 Ctrl + C----
  268.       {
  269.         put_message(Out_SKIP);                        //讓終端顯示放棄本行跳到下一行
  270.         ptr = LINE_LENGTH + 1;                  /* 這里永遠是0(結束符) */
  271.         break;
  272.       }
  273.       else if (c == In_EOL)                        //--------遇到回車鍵------
  274.       {
  275.         putchar(io_buffer[ptr++] = '\n');        //讓終端顯示換行
  276.         io_buffer[ptr] = 0;                        //末尾增添字符串結束字符NULL
  277.         ptr = 0;                                //指針清空
  278.         break;                                        //跳出后開始返回數據
  279.       }
  280.       else if (ptr < LINE_LENGTH)                //----正常ASCII碼字符----
  281.       {
  282.         if (c >= ' ')                                //刪除0x20以下的其他ASCII碼
  283.         {
  284.           putchar(io_buffer[ptr++] = c);        //存入緩沖區
  285.         }
  286.       }
  287.       else                                        //--------沖區已滿--------
  288.       {
  289.         putchar('\7');                //向終端發送鳴響符,PC會響一聲,提示已滿
  290.       }
  291.     }
  292.   }
  293. }
復制代碼



分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏 分享淘帖 頂 踩
回復

使用道具 舉報

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

本版積分規則

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

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 在线观看亚洲精品视频 | 欧美视频在线免费 | 成人av一区二区三区 | 日韩欧美一区二区三区在线播放 | 69性欧美高清影院 | 黄色在线免费观看 | 日韩精品中文字幕一区二区三区 | 91高清免费 | 在线视频 中文字幕 | 亚洲成人日韩 | 欧美中文一区 | 欧美大片在线观看 | 亚洲不卡在线观看 | 国产精品精品久久久久久 | 自拍偷拍中文字幕 | 日本特黄a级高清免费大片 成年人黄色小视频 | 亚洲精品456| 欧美vide | 成人在线国产 | 天天拍天天射 | 一级在线观看 | 亚洲精品一区中文字幕乱码 | 中文字幕成人在线 | 国产一区亚洲二区三区 | 成人av电影在线 | 黄色免费在线观看网址 | 日韩欧美一区二区三区 | 欧美精品一区二区免费 | 91久久久久久 | 色综合久久久 | 毛片在线免费播放 | 999久久久| 欧美一区二区三区视频 | 久久综合一区 | 在线观看中文字幕视频 | 干干干操操操 | 精品一区二区三区免费视频 | www四虎影视 | 国产高清精品一区二区三区 | 欧美日韩亚洲国产综合 | 天天操夜夜操 |