chinese直男口爆体育生外卖, 99久久er热在这里只有精品99, 又色又爽又黄18禁美女裸身无遮挡, gogogo高清免费观看日本电视,私密按摩师高清版在线,人妻视频毛茸茸,91论坛 兴趣闲谈,欧美 亚洲 精品 8区,国产精品久久久久精品免费

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

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

Free RTOS的互斥信號量

汽車電子技術(shù) ? 來源:玩轉(zhuǎn)單片機(jī) ? 作者:Julian ? 2023-02-10 15:36 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

二進(jìn)制信號量和互斥量非常相似,但確實(shí)有一些細(xì)微的區(qū)別?;コ怏w包含優(yōu)先級繼承機(jī)制,而二進(jìn)制信號量沒有。這使得二進(jìn)制信號量成為實(shí)現(xiàn)同步(任務(wù)之間或任務(wù)與中斷之間)的更好選擇,互斥體成為實(shí)現(xiàn)簡單互斥的更好選擇。

使用互斥信號量時(shí),需要在FreeRTOSConfig.h中加入配置代碼

//使用互斥信號量
#define configUSE_MUTEXES          1

創(chuàng)建互斥信號量

SemaphoreHandle_t xSemaphoreCreateMutex( void );

返回值:

NULL:創(chuàng)建信號量失敗,因?yàn)镕reeRTOS堆棧不足。

其它值:信號量創(chuàng)建成功。這個(gè)返回值存儲著信號量句柄。

釋放和獲取API函數(shù)請看二值信號量那篇推文

實(shí)驗(yàn)小例程

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

//毫秒級的延時(shí)
void Delay_Ms(u16 time)
{    
   u16 i=0;  
   while(time--)
   {
      i=12000;  //自己定義
      while(i--) ;    
   }
}

void LED_Init(void)
{
  GPIO_InitTypeDef GPIO_InitStructure;        //定義結(jié)構(gòu)體變量
  
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE);  //開啟時(shí)鐘
  
  GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0;            //選擇你要設(shè)置的IO口
  GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP;      //設(shè)置推挽輸出模式
  GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;     //設(shè)置傳輸速率
  GPIO_Init(GPIOC,&GPIO_InitStructure);                //初始化GPIO
  
  GPIO_SetBits(GPIOC,GPIO_Pin_0);             //將LED端口拉高,熄滅LED
}

void KEY_Init(void)
{
  GPIO_InitTypeDef GPIO_InitStructure; //定義結(jié)構(gòu)體變量  
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOE,ENABLE);
  
  GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0;     //選擇你要設(shè)置的IO口
  GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPD;//下拉輸入  
  GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;     //設(shè)置傳輸速率
  GPIO_Init(GPIOA,&GPIO_InitStructure);      /* 初始化GPIO */
  
  GPIO_InitStructure.GPIO_Pin=GPIO_Pin_3|GPIO_Pin_2|GPIO_Pin_4;
  GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPU;  //上拉輸入
  GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
  GPIO_Init(GPIOE,&GPIO_InitStructure);
}


void USART_init(uint32_t bound)
{
  GPIO_InitTypeDef GPIO_InitStruct;   //定義GPIO結(jié)構(gòu)體變量
  USART_InitTypeDef USART_InitStruct;   //定義串口結(jié)構(gòu)體變量
  
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_USART1,ENABLE);   //使能GPIOC的時(shí)鐘
  
  GPIO_InitStruct.GPIO_Pin=GPIO_Pin_9;   //配置TX引腳
  GPIO_InitStruct.GPIO_Mode=GPIO_Mode_AF_PP;   //配置PA9為復(fù)用推挽輸出
  GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz;   //配置PA9速率
  GPIO_Init(GPIOA,&GPIO_InitStruct);   //GPIO初始化函數(shù)
  
  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初始化函數(shù)
  
  
  USART_InitStruct.USART_Mode=USART_Mode_Tx|USART_Mode_Rx;   //發(fā)送接收模式
  USART_InitStruct.USART_Parity=USART_Parity_No;   //無奇偶校驗(yàn)
  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;   //無硬件數(shù)據(jù)流控制
  USART_Init(USART1,&USART_InitStruct);   //串口初始化函數(shù)
  
  USART_Cmd(USART1,ENABLE);   //使能USART1
}

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


#define START_TASK_PRIO 5      //任務(wù)優(yōu)先級
#define START_STK_SIZE 128      //任務(wù)堆棧大小
TaskHandle_t StartTask_Handler;   //任務(wù)句柄
void Start_Task(void *pvParameters);//任務(wù)函數(shù)

#define Low_TASK_PRIO 2       //任務(wù)優(yōu)先級
#define Low_STK_SIZE 50       //任務(wù)堆棧大小
TaskHandle_t LowTask_Handler;     //任務(wù)句柄
void Low_Task(void *p_arg);     //任務(wù)函數(shù)

#define Med_TASK_PRIO 3       //任務(wù)優(yōu)先級
#define Med_STK_SIZE 50       //任務(wù)堆棧大小
TaskHandle_t MedTask_Handler;     //任務(wù)句柄
void Med_Task(void *p_arg);     //任務(wù)函數(shù)

#define High_TASK_PRIO 4       //任務(wù)優(yōu)先級
#define High_STK_SIZE 50       //任務(wù)堆棧大小
TaskHandle_t HighTask_Handler;     //任務(wù)句柄
void High_Task(void *p_arg);     //任務(wù)函數(shù)

SemaphoreHandle_t Mutex_Handle =NULL;  //二值信號量句柄

int main( void ) 
{
  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);//設(shè)置系統(tǒng)中斷優(yōu)先級分組 4
  
  LED_Init(); //初始化 LED
  KEY_Init();
  USART_init(9600);
  
  //創(chuàng)建開始任務(wù)
  xTaskCreate(
    (TaskFunction_t )Start_Task,     //任務(wù)函數(shù)
    (const char* )"Start_Task",     //任務(wù)名稱
    (uint16_t )START_STK_SIZE,       //任務(wù)堆棧大小
    (void* )NULL,             //傳遞給任務(wù)函數(shù)的參數(shù)
    (UBaseType_t )START_TASK_PRIO,     //任務(wù)優(yōu)先級
    (TaskHandle_t* )&StartTask_Handler  //任務(wù)句柄 
  );
  vTaskStartScheduler();  //開啟調(diào)度
}

//開始任務(wù)函數(shù)
void Start_Task(void *pvParameters)
{
  taskENTER_CRITICAL();   //進(jìn)入臨界區(qū)
   /* 創(chuàng)建Test_Queue */
  Mutex_Handle = xSemaphoreCreateMutex();
  if(Mutex_Handle != NULL)
  {
    xSemaphoreGive(Mutex_Handle);//釋放信號量
  }
  //創(chuàng)建 Low 任務(wù)
  xTaskCreate(
    (TaskFunction_t )Low_Task, 
    (const char* )"Low_Task", 
    (uint16_t )Low_STK_SIZE, 
    (void* )NULL,
    (UBaseType_t )Low_TASK_PRIO,
    (TaskHandle_t* )&LowTask_Handler
  );
  //創(chuàng)建 Med 任務(wù)
  xTaskCreate(
    (TaskFunction_t )Med_Task, 
    (const char* )"Med_Task", 
    (uint16_t )Med_STK_SIZE, 
    (void* )NULL,
    (UBaseType_t )Med_TASK_PRIO,
    (TaskHandle_t* )&MedTask_Handler
  );
  //創(chuàng)建 High 任務(wù)
  xTaskCreate(
    (TaskFunction_t )High_Task, 
    (const char* )"High_Task", 
    (uint16_t )High_STK_SIZE, 
    (void* )NULL,
    (UBaseType_t )High_TASK_PRIO,
    (TaskHandle_t* )&HighTask_Handler
  );
  vTaskDelete(StartTask_Handler); //刪除開始任務(wù)
  taskEXIT_CRITICAL();   //退出臨界區(qū)
}


void Low_Task(void *pvParameters)
{
  int count = 0;
  while(1)
  {
    printf("Low正在等待n");
    xSemaphoreTake(Mutex_Handle,portMAX_DELAY);
    printf("Low獲取成功%d次n",++count);
    Delay_Ms(5000);
    xSemaphoreGive(Mutex_Handle);//釋放信號量
    vTaskDelay(50);
  }
}

void Med_Task(void *pvParameters)
{
  //BaseType_t xReturn = NULL;
  while(1)
  {
    printf("正在運(yùn)行n");
    vTaskDelay(50);
  }
}

void High_Task(void *pvParameters)
{
  int count = 0;
  while(1)
  {
    printf("High正在等待n");
    xSemaphoreTake(Mutex_Handle,portMAX_DELAY);
    printf("High獲取成功%d次n",++count);
    xSemaphoreGive(Mutex_Handle);//釋放信號量
    vTaskDelay(50);
  }
}

實(shí)驗(yàn)現(xiàn)象


poYBAGPl88aAQierAADP34qVgyk228.png

解讀:互斥信號量,其實(shí)就是將Low任務(wù)的優(yōu)先級和High任務(wù)的優(yōu)先級變成了一樣的優(yōu)先級(短暫拉高最低優(yōu)先級任務(wù)),從而解決優(yōu)先級翻轉(zhuǎn)問題

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

    關(guān)注

    2

    文章

    807

    瀏覽量

    42309
  • 繼承機(jī)制
    +關(guān)注

    關(guān)注

    0

    文章

    2

    瀏覽量

    5725
  • 信號量
    +關(guān)注

    關(guān)注

    0

    文章

    53

    瀏覽量

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

掃碼添加小助手

加入工程師交流群

    評論

    相關(guān)推薦
    熱點(diǎn)推薦

    實(shí)時(shí)操作系統(tǒng)FreeRTOS信號量應(yīng)用

    二值信號量通常用于互斥訪問或同步,二值信號量互斥信號量非常相似,但還是有細(xì)微差別,互斥
    的頭像 發(fā)表于 06-08 09:24 ?4100次閱讀
    實(shí)時(shí)操作系統(tǒng)FreeRTOS<b class='flag-5'>信號量</b>應(yīng)用

    FreeRTOS信號量使用教程

    信號量是操作系統(tǒng)中重要的一部分,信號量一般用來進(jìn)行資源管理和任務(wù)同步, FreeRTOS中信號量又分為二值信號量、 計(jì)數(shù)型信號量
    的頭像 發(fā)表于 12-19 09:22 ?3831次閱讀
    FreeRTOS<b class='flag-5'>信號量</b>使用教程

    轉(zhuǎn):第23章 FreeRTOS互斥信號量

    本章節(jié)講解FreeRTOS重要的資源共享機(jī)制---互斥信號量(Mutex,即MutualExclusion的縮寫)。注意,建議初學(xué)者學(xué)習(xí)完前兩個(gè)章節(jié)的信號量后再學(xué)習(xí)本章節(jié)的互斥
    發(fā)表于 09-06 14:58

    第15章 互斥信號量

    15.1 互斥信號量15.1.1互斥信號量的概念及其作用 互斥信號量就是
    發(fā)表于 10-06 16:40

    信號量互斥信號量該怎么選擇?

    既然說信號量可能會導(dǎo)致優(yōu)先級反轉(zhuǎn),那全都在工程里使用互斥信號不就行了?還要信號量干啥?大家一起用互斥信號
    發(fā)表于 08-26 03:14

    關(guān)于UCOSIII的信號量互斥信號量的理解?

    信號量。如果其它任務(wù)中有請求信號量,且該任務(wù)優(yōu)先級高于當(dāng)前任務(wù)優(yōu)先級,進(jìn)行任務(wù)切換;如果其它任務(wù)中當(dāng)前沒有請求此信號量,或該任務(wù)優(yōu)先級低于當(dāng)前任務(wù)優(yōu)先級,不進(jìn)行任務(wù)切換?請求互斥
    發(fā)表于 03-13 00:11

    例程使用互斥信號量初始化如何設(shè)置?

    OS_MUTEXTEST_MUTEX; //定義一個(gè)互斥信號量//創(chuàng)建一個(gè)互斥信號量OSMutexCreate((OS_MUTEX*)&TEST_MUTEX, (CPU_CHAR
    發(fā)表于 06-02 16:22

    UCOS擴(kuò)展例程-UCOSIII互斥信號量

    UCOS擴(kuò)展例程-UCOSIII互斥信號量
    發(fā)表于 12-14 17:24 ?27次下載

    信號量互斥鎖的區(qū)別

    互斥用于線程的互斥,信號線用于線程的同步。這是互斥信號
    發(fā)表于 11-13 17:43 ?1.3w次閱讀
    <b class='flag-5'>信號量</b>和<b class='flag-5'>互斥</b>鎖的區(qū)別

    詳解互斥信號量的概念和運(yùn)行

    1 、互 斥 信 號 1.1 互斥信號量的概念及其作用 互斥信號量的主要作用是對資源實(shí)現(xiàn)互斥
    的頭像 發(fā)表于 10-22 11:57 ?1.2w次閱讀
    詳解<b class='flag-5'>互斥</b><b class='flag-5'>信號量</b>的概念和運(yùn)行

    FreeRTOS 隊(duì)列 信號量 互斥

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

    FreeRTOS系列第20篇---FreeRTOS信號量API函數(shù)

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

    FreeRTOS的二值信號量

    FreeRTOS中的信號量是一種任務(wù)間通信的方式,信號量包括:二值信號量、互斥信號量、計(jì)數(shù)信號量
    的頭像 發(fā)表于 02-10 15:07 ?1776次閱讀

    Free RTOS的計(jì)數(shù)型信號量

    上篇講解了二值信號量,二值信號量只能判斷有無,而不能確定事件發(fā)生的次數(shù),因此我們?yōu)榱舜_定事件的次數(shù)引入了計(jì)數(shù)型信號量
    的頭像 發(fā)表于 02-10 15:29 ?1311次閱讀
    <b class='flag-5'>Free</b> <b class='flag-5'>RTOS</b>的計(jì)數(shù)型<b class='flag-5'>信號量</b>

    使用Linux信號量實(shí)現(xiàn)互斥點(diǎn)燈

    信號量常用于控制對共享資源的訪問,有計(jì)數(shù)型信號量和二值信號量之分。初始化時(shí)信號量值大于1的,就是計(jì)數(shù)型信號量,計(jì)數(shù)型
    的頭像 發(fā)表于 04-13 15:12 ?1057次閱讀
    使用Linux<b class='flag-5'>信號量</b>實(shí)現(xiàn)<b class='flag-5'>互斥</b>點(diǎn)燈