在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

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

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

3天內不再提示

FreeRTOS的二值信號量

汽車電子技術 ? 來源:玩轉單片機 ? 作者: Julian ? 2023-02-10 15:07 ? 次閱讀

信號

FreeRTOS中的信號量是一種任務間通信的方式,信號量包括:二值信號量、互斥信號量、計數信號量,本次實驗只使用二值信號量。信號量用于任務間的同步,FreeRTOS是多任務系統,不同任務間可能需要某種同步關系

二值信號量

可以通俗理解為0或1標志位,比如串口中斷接收完數據是一種狀態,此時就需要進行串口數據處理又是一種狀態,這時使用二值信號量就能很好達到任務同步效果

信號量的基本操作有獲取信號量和釋放信號量,例如:數據分析處理任務需要處理串口數據時,先嘗試獲取信號量,若獲取不到,也就是信號量是0,則先進入阻塞等待,等待超時可先跳出,之后繼續嘗試獲取信號量。串口空閑中斷接受完一串數據后,可執行釋放信號量操作,這時,數據分析處理任務就可以獲取到信號量,進而可以處理串口數據了,實現了串口數據接收與數據處理的同步。

API函數

創建二值信號量

SemaphoreHandle_t xSemaphoreCreateBinary( void )

返回值:

NULL:創建信號量失敗,因為FreeRTOS堆棧不足。

       其它值:信號量創建成功。這個返回值存儲著信號量句柄。

非中斷釋放二值信號量

BaseType_t xSemaphoreGive( SemaphoreHandle_t xSemaphore )

參數

xSemaphore:要釋放的信號量句柄

返回值:

釋放成功返回pdPASS,失敗返回errQUEUE_FULL

中斷釋放二值信號量

BaseType_t xSemaphoreGiveFromISR( SemaphoreHandle_t xSemaphore,
                                 BaseType_t* pxHigherPriorityTaskWoken)

參數:

xSemaphore:要釋放的信號量句柄

pxHigherPriorityTaskWoken:標記退出此函數后是否需要進行任務切換

返回值:

釋放成功返回pdPASS,失敗返回errQUEUE_FULL

獲取信號量

BaseType_t xSemaphoreTake( SemaphoreHandle_t xSemaphore,
                          TickType_t xBlockTime)

參數:

xSemaphore:要釋放的信號量句柄

    xBlockTime:阻塞時間

返回值:

獲取成功返回pdTRUE,失敗返回pdFALSE

中斷獲取信號量

BaseType_t xSemaphoreTakeFromISR( SemaphoreHandle_t xSemaphore,
                                 BaseType_t* pxHigherPriorityTaskWoken)

參數:

xSemaphore:要釋放的信號量句柄

pxHigherPriorityTaskWoken:標記退出此函數后是否需要進行任務切換

返回值:

獲取成功返回pdTRUE,失敗返回pdFALSE

實現目的

通過按鍵觸發二值信號量的釋放,獲取任務一直在等待信號量的到來,再執行相應的任務

上源碼

#include "stm32f10x.h"


#include "stm32f10x.h"
#include 
#include "FreeRTOS.h"
#include "task.h"
#include "semphr.h"


void LED_Init(void)
{

  GPIO_InitTypeDef  GPIO_InitStructure;


  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);   //使能PE端口時鐘


  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1|GPIO_Pin_5;  //端口配置
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;     //推挽輸出
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;    //IO口速度為50MHz
  GPIO_Init(GPIOC, &GPIO_InitStructure);            //推挽輸出 ,IO口速度為50MHz
  GPIO_SetBits(GPIOC,GPIO_Pin_1|GPIO_Pin_5);         //輸出高 

}


void KEY_Init(void)
{
  GPIO_InitTypeDef GPIO_InitStructure;       //定義結構體變量  
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);

  GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0;     //選擇你要設置的IO口
  GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPD; //下拉輸入  
  GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;  //設置傳輸速率
  GPIO_Init(GPIOA,&GPIO_InitStructure);       //初始化GPIO
}




void USART_init(uint32_t bound)
{
  GPIO_InitTypeDef GPIO_InitStruct;   //定義GPIO結構體變量
  USART_InitTypeDef USART_InitStruct;   //定義串口結構體變量

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_USART1,ENABLE);   //使能GPIOC的時鐘

  GPIO_InitStruct.GPIO_Pin=GPIO_Pin_9;   //配置TX引腳
  GPIO_InitStruct.GPIO_Mode=GPIO_Mode_AF_PP;   //配置PA9為復用推挽輸出
  GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz;   //配置PA9速率
  GPIO_Init(GPIOA,&GPIO_InitStruct);   //GPIO初始化函數

  GPIO_InitStruct.GPIO_Pin=GPIO_Pin_10;   //配置RX引腳
  GPIO_InitStruct.GPIO_Mode=GPIO_Mode_IN_FLOATING;   //配置PA10為浮空輸入
  GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz;   //配置PA10速率
  GPIO_Init(GPIOA,&GPIO_InitStruct);   //GPIO初始化函數


  USART_InitStruct.USART_Mode=USART_Mode_Tx|USART_Mode_Rx;   //發送接收模式
  USART_InitStruct.USART_Parity=USART_Parity_No;   //無奇偶校驗
  USART_InitStruct.USART_BaudRate=bound;   //波特率
  USART_InitStruct.USART_StopBits=USART_StopBits_1;   //停止位1位
  USART_InitStruct.USART_WordLength=USART_WordLength_8b;   //字長8位
  USART_InitStruct.USART_HardwareFlowControl=USART_HardwareFlowControl_None;   //無硬件數據流控制
  USART_Init(USART1,&USART_InitStruct);   //串口初始化函數

  USART_Cmd(USART1,ENABLE);   //使能USART1
}


int fputc(int ch,FILE *f)   //printf重定向函數
{
  USART_SendData(USART1,(uint8_t)ch);   //發送一字節數據
  while(USART_GetFlagStatus(USART1,USART_FLAG_TXE) == RESET);   //等待發送完成
  return ch;
}


//創建開始任務
#define START_TASK_PRIO 1 //任務優先級
#define START_STK_SIZE 128 //任務堆棧大小
TaskHandle_t StartTask_Handler; //任務句柄
void Start_Task(void *pvParameters); //任務函數
//釋放信號量
#define Release_TASK_PRIO 2 //任務優先級
#define Release_STK_SIZE 50 //任務堆棧大小
TaskHandle_t ReleaseTask_Handler; //任務句柄
void Release_Task(void *p_arg); //任務函數
//獲取信號量
#define Gain_TASK_PRIO 3 //任務優先級
#define Gain_STK_SIZE 50 //任務堆棧大小
TaskHandle_t GainTask_Handler; //任務句柄
void Gain_Task(void *p_arg); //任務函數


SemaphoreHandle_t KeySemaphore;//串口接收二值信號量句柄


int main( void ) 
{ 
  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);//設置系統中斷優先級分組 4
  LED_Init();     //初始化 LED
  KEY_Init();      //按鍵初始化
  USART_init(115200);  //初始化串口

  //創建開始任務
  xTaskCreate(
    (TaskFunction_t )Start_Task,     //任務函數
    (const char* )"Start_Task",     //任務名稱
    (uint16_t )START_STK_SIZE,       //任務堆棧大小
    (void* )NULL,             //傳遞給任務函數的參數
    (UBaseType_t )START_TASK_PRIO,     //任務優先級
    (TaskHandle_t* )&StartTask_Handler  //任務句柄 
  );
  vTaskStartScheduler();  //開啟調度
}


void Start_Task(void *pvParameters)
{
  taskENTER_CRITICAL();   //進入臨界區
  //創建二值信號量
  KeySemaphore = xSemaphoreCreateBinary();
  //創建 釋放信號量 任務
  xTaskCreate(
    (TaskFunction_t )Release_Task, 
    (const char* )"Release_Task", 
    (uint16_t )Release_STK_SIZE, 
    (void* )NULL,
    (UBaseType_t )Release_TASK_PRIO,
    (TaskHandle_t* )&ReleaseTask_Handler
  ); 
  //創建 獲取信號量 任務
  xTaskCreate(
    (TaskFunction_t )Gain_Task, 
    (const char* )"Gain_Task", 
    (uint16_t )Gain_STK_SIZE, 
    (void* )NULL,
    (UBaseType_t )Gain_TASK_PRIO,
    (TaskHandle_t* )&GainTask_Handler
  ); 
  vTaskDelete(StartTask_Handler); //刪除開始任務
  taskEXIT_CRITICAL();   //退出臨界區
}
//釋放信號量 任務函數
void Release_Task(void *pvParameters)
{
  BaseType_t xReturn = NULL;
  while(1)
  {
    if(GPIO_ReadInputDataBit( GPIOA, GPIO_Pin_0))
    {
      vTaskDelay(10);
      if(GPIO_ReadInputDataBit( GPIOA, GPIO_Pin_0))
      {
        xReturn = xSemaphoreGive(KeySemaphore);
        if(xReturn == pdPASS)
          printf("釋放成功\\n");
      }
    }

    vTaskDelay(10);
  }
}


//獲取信號量 任務函數
void Gain_Task(void *pvParameters)
{
  BaseType_t xReturn = NULL;
  while(1)
  {
    xReturn = xSemaphoreTake(KeySemaphore,portMAX_DELAY);//一直阻塞獲取
    if(xReturn == pdPASS)
      printf("獲取成功\\n");
    vTaskDelay(10);
  }
}
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • FreeRTOS
    +關注

    關注

    12

    文章

    484

    瀏覽量

    62414
  • 任務系統
    +關注

    關注

    0

    文章

    4

    瀏覽量

    6136
  • 信號量
    +關注

    關注

    0

    文章

    53

    瀏覽量

    8378
收藏 人收藏

    評論

    相關推薦

    實時操作系統FreeRTOS信號量應用

    信號量通常用于互斥訪問或同步,信號量和互斥信號量
    的頭像 發表于 06-08 09:24 ?3771次閱讀
    實時操作系統<b class='flag-5'>FreeRTOS</b><b class='flag-5'>信號量</b>應用

    FreeRTOS串口中斷接收不定長的數據與信號量的使用

    FreeRTOS例程,使用串口中斷接收不定長的數據,以及信號量的使用
    的頭像 發表于 09-26 09:02 ?4421次閱讀
    <b class='flag-5'>FreeRTOS</b>串口中斷接收不定長的數據與<b class='flag-5'>二</b><b class='flag-5'>值</b><b class='flag-5'>信號量</b>的使用

    FreeRTOS信號量使用教程

    信號量是操作系統中重要的一部分,信號量一般用來進行資源管理和任務同步, FreeRTOS信號量又分為
    的頭像 發表于 12-19 09:22 ?3295次閱讀
    <b class='flag-5'>FreeRTOS</b><b class='flag-5'>信號量</b>使用教程

    轉:第22章 FreeRTOS信號量

    本章節講解FreeRTOS任務間的同步和資源共享機制,信號量。
    發表于 09-06 10:02

    轉:第25章 FreeRTOS任務信號量

    本章節為大家講解FreeRTOS信號量的另一種實現方式----基于任務通知(Task Notifications)的
    發表于 09-07 10:58

    FreeRTOS信號量卡住出不來怎么辦?

    串口中斷函數在調用xSemaphoreGiveFromISR( xBinarySemaphore, &xHigherPriorityTaskWoken ),給出信號量時,卡在了
    發表于 06-10 09:25

    FreeRTOS信號量介紹

    FreeRTOS信號量 & ESP32實戰閱讀建議:有一定操作系統基礎知識。FreeRTOS信號量1.
    發表于 01-27 07:28

    怎樣去使用FreeRTOS信號量

    怎樣去使用FreeRTOS信號量呢?怎樣通過按鍵中斷來控制LED燈的亮滅從而實現任務與中斷之間的同步呢?
    發表于 02-28 07:23

    大部分國產低端MCU沒有空閑中斷怎么辦?

    FreeRTOS信號量 Timer UART 空閑中斷的原理: IDLE中斷叫空閑中斷,不叫幀中斷。那么什么叫空閑,怎么定義空閑呢?在實際發送數據的時候,比如一串字符串,其實
    的頭像 發表于 03-17 10:12 ?3478次閱讀

    FreeRTOS信號量 & ESP32實戰

    FreeRTOS信號量 & ESP32實戰閱讀建議:有一定操作系統基礎知識。FreeRTOS信號量1.
    發表于 12-03 18:06 ?1次下載
    <b class='flag-5'>FreeRTOS</b><b class='flag-5'>信號量</b> & ESP32實戰

    FreeRTOS 隊列 信號量 互斥

    文章目錄前言Queue 隊列semaphore 信號量Mutex 互斥微信公眾號前言FreeRTOS STM32CubeMX配置 內存管理 任務管理上節介紹了用STM32CubeMX生成帶
    發表于 12-09 09:51 ?0次下載
    <b class='flag-5'>FreeRTOS</b> 隊列 <b class='flag-5'>信號量</b> 互斥<b class='flag-5'>量</b>

    STM32WB55XX freertos 信號量+dma+idle 不定長串口接收 + dma傳輸完成中斷

    2、開啟全局中斷低功耗串口1 : 添加 DMA時鐘源選擇1.2 freertos 配置添加串口任務創建信號量、用戶代碼如果使用信號量同步,把使能空閑中斷放在默認任務開頭可以避免一
    發表于 12-24 19:13 ?5次下載
    STM32WB55XX  <b class='flag-5'>freertos</b> <b class='flag-5'>二</b><b class='flag-5'>值</b><b class='flag-5'>信號量</b>+dma+idle 不定長串口接收 + dma傳輸完成中斷

    FreeRTOS高級篇6---FreeRTOS信號量分析

    FreeRTOS信號量包括進制信號量、計數信號量、互斥信號量(以后簡稱互斥
    發表于 01-26 17:39 ?7次下載
    <b class='flag-5'>FreeRTOS</b>高級篇6---<b class='flag-5'>FreeRTOS</b><b class='flag-5'>信號量</b>分析

    FreeRTOS系列第20篇---FreeRTOS信號量API函數

    FreeRTOS信號量包括進制信號量、計數信號量、互斥信號量(以后簡稱互斥
    發表于 01-26 17:44 ?4次下載
    <b class='flag-5'>FreeRTOS</b>系列第20篇---<b class='flag-5'>FreeRTOS</b><b class='flag-5'>信號量</b>API函數

    FreeRTOS四種信號量詳細介紹

    1、信號量 信號量通常用于互斥訪問或同步,
    的頭像 發表于 07-06 17:14 ?2759次閱讀
    主站蜘蛛池模板: 日本在线www | 天天躁夜夜躁狠狠躁2024 | 3344免费播放观看视频 | 亚洲五月激情综合图片区 | 天天摸日日 | 五月婷婷综合在线 | 日韩一卡 二卡 三卡 四卡 免费视频 | 一级毛片子 | 一级片视频播放 | 福利久久| 色综合欧美| 天天色天天色 | 国产精品手机在线 | 456成人网 | 9久热久re爱免费精品视频 | 天天爱夜夜 | 亚洲伊人久久大香线蕉结合 | 天天摸天天做天天爽水多 | 久久久久久久国产 | 亚洲欧美强伦一区二区另类 | 国产精品高清一区二区三区不卡 | 五等分的新娘免费漫画 | 久久久香蕉视频 | 51精品国产 | 高清人人天天夜夜曰狠狠狠狠 | 午夜精品久久久久久久第一页 | 伊人欧美在线 | 永久在线免费观看 | 88av视频在线| 色y情视频免费看 | 激情婷婷色 | 成人网中文字幕色 | 成年男人永久免费看片 | 中文三级视频 | 国产精品久久自在自2021 | 亚洲韩国日本欧美一区二区三区 | 亚洲第一免费播放区 | 大黄蕉| 日韩色影视| 女同毛片免费网站 | 开心色99×xxxx |