一区二区三区三上|欧美在线视频五区|国产午夜无码在线观看视频|亚洲国产裸体网站|无码成年人影视|亚洲AV亚洲AV|成人开心激情五月|欧美性爱内射视频|超碰人人干人人上|一区二区无码三区亚洲人区久久精品

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

STM32串口接收不定長數(shù)據(jù):采用標志位(比如0X0D,0X0A)結束法

RXL111 ? 來源:RXL111 ? 作者:RXL111 ? 2022-09-23 14:06 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

缺點:有些情況下會導致數(shù)據(jù)丟失(可能返回數(shù)據(jù)中0x0d、0a本身為有效數(shù)據(jù))
適用:約定協(xié)議的數(shù)據(jù)幀(發(fā)送數(shù)據(jù)的設備必須以相應的約定字節(jié)作為一次數(shù)據(jù)結束)


void USART1_IRQHandler(void)                	//串口中斷服務程序(函數(shù))
	{
	u8 Res;	//定義Res,用于Res =USART_ReceiveData(USART1);中存儲串口1發(fā)送的數(shù)據(jù)(這里的數(shù)據(jù)按位發(fā)送)
#if SYSTEM_SUPPORT_OS  //如果SYSTEM_SUPPORT_OS為真,則需要支持OS		
	OSIntEnter();    
#endif
	if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)  //接收中斷(接收到的數(shù)據(jù)必須是0x0d 0x0a結尾)
		{
		Res =USART_ReceiveData(USART1);	//讀取接收到的數(shù)據(jù)
		
		if((USART_RX_STA&0x8000)==0)//接收未完成    1000 0000 0000 0000
//判斷USART_RX_STA的第一位是否為0,這時因為USART_RX_STA的初始值為0,所以我們進入if(USART_RX_STA&0x4000)。
			{
			if(USART_RX_STA&0x4000)//接收到了0x0d	0100 0000 0000 0000
//判斷USART_RX_STA的第二位是否為1,所以我們進入else //還沒收到0X0D。
				{
				if(Res!=0x0a)USART_RX_STA=0;//接收錯誤,重新開始
				else USART_RX_STA|=0x8000;	//接收完成了 
				}
			else //還沒收到0X0D
				{	
				if(Res==0x0d)USART_RX_STA|=0x4000;//再次判斷這次接收到的是不是0x0d,判斷了Res是否0x0d, 即Res是否為回車,這里如果串口有輸入數(shù)據(jù)的話明顯可以判斷的,所以我們進入下面的else.
				else
					{
					USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;  // 0011 1111 1111 1111
					USART_RX_STA++;
					if(USART_RX_STA>(USART_REC_LEN-1))USART_RX_STA=0;//接收數(shù)據(jù)錯誤,重新開始接收	  
					}		 
				}
			}   		 
     } 
#if SYSTEM_SUPPORT_OS 	
	OSIntExit();  											 
#endif
}

OSIntEnter()和OSIntExit()兩者必須成對出現(xiàn)。
進入中斷時調用OSIntEnter(),退出中斷時調用OSIntExit()。
OSIntEnter 是進?中斷服務函數(shù),?來記錄中斷嵌套層數(shù)(OSIntNesting增加 1);
OSIntEnter()應該在中斷關閉后調用,所以函數(shù)里面沒有使用OS_ENTER_CRITICAL() and OS_EXIT_CRITICAL(),如此在調用OSIntEnter()前需關閉中斷。
OSIntExit():所有中斷結束后進行任務調度,使系統(tǒng)更加實時。
OSIntExit 是退出中斷服務函數(shù),該函數(shù)可能觸發(fā)?次任務切換(當 OSIntNesting==0&&調度器未上鎖&&就緒表最?優(yōu)先級任務 != 被中斷的任務優(yōu)先級時),否則繼續(xù)返回原來的任務執(zhí)?代碼(如果 OSIntNesting 不為 0,則減 1)。
OS_Sched():uCOS進行任務調度,不在中斷調用。
OSIntNesting:統(tǒng)計中斷嵌套數(shù),最多255。在OSIntExit()和OS_Sched()中都有判別。
OS_ENTER_CRITICAL():保存中斷狀態(tài),關中斷。uCOS將無法再執(zhí)行任務調度,硬件中斷也被屏蔽。

void  OSIntEnter (void)

{

if (OSRunning == OS_TRUE) {

if (OSIntNesting < 255u) {

OSIntNesting++;                      /* Increment ISR nesting level                        */

}

}

}

這個函數(shù)的作用是對全局變量OSIntNesting增1,OSIntNesting為中斷嵌套深度。

void  OSIntExit (void)

{

#if OS_CRITICAL_METHOD == 3u                               /* Allocate storage for CPU status register */

OS_CPU_SR  cpu_sr = 0u;

#endif

if (OSRunning == OS_TRUE) {

OS_ENTER_CRITICAL();

if (OSIntNesting > 0u) {                           /* Prevent OSIntNesting from wrapping       */

OSIntNesting--;

}

if (OSIntNesting == 0u) {                          /* Reschedule only if all ISRs complete ... */

if (OSLockNesting == 0u) {                     /* ... and not locked.                      */

OS_SchedNew();

OSTCBHighRdy = OSTCBPrioTbl[OSPrioHighRdy];

if (OSPrioHighRdy != OSPrioCur) {          /* No Ctx Sw if current task is highest rdy */

#if OS_TASK_PROFILE_EN > 0u

OSTCBHighRdy->OSTCBCtxSwCtr++;         /* Inc. # of context switches to this task  */

#endif

OSCtxSwCtr++;                          /* Keep track of the number of ctx switches */

OSIntCtxSw();                          /* Perform interrupt level ctx switch       */

}

}

}

OS_EXIT_CRITICAL();

}

}

函數(shù)的前面部分對OSIntNesting減1,剛好與OSIntEnter() 相對應;后面部分則進行任務調度。

總結:任何中斷服務函數(shù),我們都應該加上 OSIntEnter 和 OSIntExit 函數(shù),UCOSII 是?個可剝奪型的內核,中斷服務?程序運?之后,系統(tǒng)會根據(jù)情況進??次任務調度去運?優(yōu)先級別最?的就緒任務,?并不?定接著運?被中斷的任務!


#if...#endif是C++中的條件編譯預處理命令 有兩種格式:

1:#ifdef 標示符

程序段1

#else

程序段2

#endif

表示:如果標示符已經(jīng)被#define命令定義過,則編譯程序段1,否則編譯程序段2。期中else部分可以沒有。

2:#if 表達式

程序段1

#else

程序段2

#endif

表示:如果表達式為真,則編譯程序段1,否則編譯程序段2.
if((USART_RX_STA&0x8000)==0) //0x8000,即二進制1000 0000 0000 0000,與變量USART_RX_STA,按位與(&),并與0比較,作用是判斷USART_RX_STA數(shù)值第16位是否為0。
USART_RX_STA&0x8000有兩種可能:
第一種1××× ×××× ×××× ××××&1000 0000 0000 0000=1000 0000 0000 0000
第二種0××× ×××× ×××× ××××&1000 0000 0000 0000=0000 0000 0000 0000
由此可以判斷USART_RX_STA第16位是否為0


USART_RX_STA的作用,USART_RX_STA一共有16位,前兩位為標記位,后14位記錄了串口發(fā)送的數(shù)的位數(shù)。第一位標記位標記了Res是否為0x0a,第二位標記位標記了Res是否為0x0d。
知識點:0x0d是回車的ASCLL碼,0x0a是換行的ASCLL碼

USART_RX_BUF這個是用來保存接收到的數(shù)據(jù)的可以看到每次結束判斷會有
USART_RX_BUF[USART_RX_STA&0X3FFF]=Res ;
USART_RX_STA=0; //接收狀態(tài)標記
USART_RX_STA的作用就是在全部函數(shù)之間實現(xiàn)一個消息傳遞,自己設置,自己管理,自己識別。
bit15 bit14 bit13~0
接收完成標志0x0a 接收到0X0d標志 接收到的有效數(shù)據(jù)個數(shù)

USART_RX_STA|=0x4000;將第二位狀態(tài)標志位置為1;在倒數(shù)第1次循環(huán)中使用USART_RX_STA|=0x8000;將第一位狀態(tài)標志位也置為1,;而后串口數(shù)據(jù)接收結束,所有從串口接收的數(shù)據(jù)保存在USART_RX_BUF[ ]數(shù)組中,串口所發(fā)送的數(shù)據(jù)長度保存在USART_RX_STA的后14位中。

審核編輯:湯梓紅

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權轉載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • STM32
    +關注

    關注

    2293

    文章

    11032

    瀏覽量

    364996
  • 串口
    +關注

    關注

    15

    文章

    1588

    瀏覽量

    79946
收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評論

    相關推薦
    熱點推薦

    Labview讀取二進制文件(有的時候會自動添加0X0D0x0A前面)

    我是利用Labview開發(fā)TCP/IP通信的,需要把接收的網(wǎng)絡數(shù)據(jù)保存為二進制文件。用UltraEdit查看保存的文件里面有好多二進制值,0x00~0xFF,里面會有
    發(fā)表于 05-02 14:48

    stm32串口怎么用DMA接收不定長數(shù)據(jù)?

    stm32串口怎么用DMA接收不定長數(shù)據(jù),求大神點播1. 網(wǎng)上查到,空閑中斷+DMA可以接收
    發(fā)表于 08-21 09:21

    串口接收0x0a0x0d表示的什么?

    void USART1_IRQHandler(void){ u8 res; if(USART1->SR&(163)USART_RX_STA=0;//接收數(shù)據(jù)錯誤,重新開始接收 }}}
    發(fā)表于 08-19 03:09

    stm32識別TC35短信有兩個“0x0d 0x0a”怎么解決?

    原子哥的串口代碼里,遇到0x0d 0x0a就停止接收,但是從TC35模塊讀取短信時,前面的短信息說明,會有兩個“0x0d
    發(fā)表于 08-28 08:00

    Labview串口時候數(shù)據(jù)中的0x0A與換行符沖突怎么解決?

    串口數(shù)據(jù)傳輸時候,含有0x0A與換行符沖突,如果關閉終止符,那樣就無法識別串口數(shù)據(jù)結束,無法識別
    發(fā)表于 12-06 21:33

    STM32串口接收不定長數(shù)據(jù)原理是什么

    STM32串口接收不定長數(shù)據(jù)原理是什么
    發(fā)表于 12-08 07:29

    STM32串口接收不定長數(shù)據(jù)的實現(xiàn)方法

    STM32串口接收不定長數(shù)據(jù)的實現(xiàn)方法
    發(fā)表于 12-09 06:17

    針對接收一幀含有多個字節(jié)的不定長數(shù)據(jù)接收方式進行討論

    (比如0X0D,0X0A)結束 非常常見的一種接收方式,正點原子的例程便是
    發(fā)表于 12-09 07:16

    如何利用STM32串口去解析json命令呢

    STM32串口解析json命令(使用HAL庫)串口接收中斷處理(0x0D 0x0A作為結尾)測試
    發(fā)表于 01-21 10:07

    STM32串口接收不定長數(shù)據(jù)的程序免費下載

    本文檔的主要內容詳細介紹的是STM32串口接收不定長數(shù)據(jù)的程序免費下載。
    發(fā)表于 08-26 08:00 ?62次下載
    <b class='flag-5'>STM32</b><b class='flag-5'>串口</b><b class='flag-5'>接收</b><b class='flag-5'>不定長</b><b class='flag-5'>數(shù)據(jù)</b>的程序免費下載

    STM32串口通信 (采用鏈表接收不定長數(shù)據(jù)幀)

    STM32串口接收不定長數(shù)據(jù)幀->鏈表數(shù)據(jù)幀說明二級目錄三級目錄
    發(fā)表于 11-23 18:07 ?31次下載
    <b class='flag-5'>STM32</b><b class='flag-5'>串口</b>通信 (<b class='flag-5'>采用</b>鏈表<b class='flag-5'>接收</b><b class='flag-5'>不定長</b><b class='flag-5'>數(shù)據(jù)</b>幀)

    STM32串口接收不定長數(shù)據(jù)的幾種方法

    (比如0X0D,0X0A)結束 非常常見的一種接收方式,正點原子的例程便是
    發(fā)表于 11-26 13:21 ?17次下載
    <b class='flag-5'>STM32</b><b class='flag-5'>串口</b><b class='flag-5'>接收</b><b class='flag-5'>不定長</b><b class='flag-5'>數(shù)據(jù)</b>的幾種方法

    stm32 串口接收不定長度數(shù)據(jù)及黏包處理 + 串口DMA接收

    ,那么stm32串口是如何實現(xiàn)接收不定長度數(shù)據(jù)的呢? 串口接收
    發(fā)表于 12-23 19:09 ?27次下載
    <b class='flag-5'>stm32</b> <b class='flag-5'>串口</b><b class='flag-5'>接收</b><b class='flag-5'>不定長度數(shù)據(jù)</b>及黏包處理 + <b class='flag-5'>串口</b>DMA<b class='flag-5'>接收</b>

    STM32 DMA串口接收不定長數(shù)據(jù)

    STM32 DMA串口接收不定長數(shù)據(jù)
    發(fā)表于 12-24 18:50 ?41次下載
    <b class='flag-5'>STM32</b>  DMA<b class='flag-5'>串口</b><b class='flag-5'>接收</b><b class='flag-5'>不定長</b><b class='flag-5'>數(shù)據(jù)</b>

    RM0451_超低功耗 STM32L0x0 先進的基于 Arm? 的 32 MCU

    RM0451_超低功耗 STM32L0x0 先進的基于 Arm? 的 32 MCU
    發(fā)表于 11-23 08:20 ?2次下載
    RM0451_超低功耗 <b class='flag-5'>STM32L0x0</b> 先進的基于 Arm? 的 32 <b class='flag-5'>位</b> MCU