【连载】从单片机到操作系统④——FreeRTOS创建任务&开启调度详解

本文是杰杰原创文章,如需转载请说明出处:

【连载】从单片机到操作系统④——FreeRTOS创建任务&开启调度详解

创客的兄弟姐妹们大家好,我是杰杰。又到了更新的时候了。

开始今天的内容之前,先补充一下上篇文章【连载】从单片机到操作系统③——走进FreeRTOS的一点点遗漏的知识点。

 1BaseType_t xTaskCreate(       TaskFunction_t pvTaskCode,
2                              const char * const pcName,
3                              uint16_t usStackDepth,
4                              void *pvParameters,
5                              UBaseType_t uxPriority,
6                              TaskHandle_t *pvCreatedTask
7                          )
;
8创建任务中的堆栈大小问题,在task.h中有这样子的描述:
9/**
10* @param usStackDepth The size of the task stack specified as the number of variables the stack * can hold - not the number of bytes.  For example, if the stack is 16 bits wide and  
11* usStackDepth is defined as 100, 200 byteswill be allocated for stack storage.
12*/

代码可左右滑动


  当任务创建时,内核会分为每个任务分配属于任务自己的唯一堆栈usStackDepth 值用于告诉内核为它应该分配多大的栈空间。

这个值指定的是栈空间可以保存多少个(word) ,而不是多少个字节(byte)

文档也有说明,如果是16位宽度的话,假如usStackDepth = 100;那么就是200个字节(byte)。

当然,我用的是stm3232位宽度的, usStackDepth=100;那么就是400个字节(byte)。

 

  好啦,补充完毕。下面正式开始我们今天的主题。



  我自己学的是应用层的东西,很多底层的东西我也不懂,水平有限,出错了还请多多包涵。

  其实我自己写文章的时候也去跟着火哥的书看着底层的东西啦,但是本身自己也是不懂,不敢乱写。所以,这个《从单片机到操作系统》系列的文章,我会讲一点底层,更多的是应用层,主要是用的方面。

 

 按照一般的写代码的习惯,在main函数里面各类初始化完毕了,并且创建任务成功了,那么,可以开启任务调度了。

 1int main(void)
2
{
3    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);//设置系统中断优先级分组4    
4    Delay_Init();                       //延时函数初始化    
5    Uart_Init(115200);                  //初始化串口
6    LED_Init();                     //初始化LED
7    KEY_Init();
8    //创建开始任务
9    xTaskCreate((TaskFunction_t )start_task,            //任务函数
10                (const char*    )"start_task",          //任务名称
11                (uint16_t       )START_STK_SIZE,        //任务堆栈大小
12                (void*          )NULL,                  //传递给任务函数的参数
13                (UBaseType_t    )START_TASK_PRIO,       //任务优先级
14                (TaskHandle_t*  )&StartTask_Handler);   //任务句柄              
15    vTaskStartScheduler();          //开启任务调度
16}


  来大概看看分析一下创建任务的过程,虽然说会用就行,但是也是要知道了解一下的。

注意:下面说的创建任务均为xTaskCreate动态创建)而非静态创建。

 1pxStack = ( StackType_t * ) pvPortMalloc( ( ( ( size_t ) usStackDepth ) * sizeof( StackType_t ) ) ); 
2/*lint !e961 MISRA exception as the casts are only redundant for some ports. */
3            if( pxStack != NULL )
4            {
5                /* Allocate space for the TCB. */
6                pxNewTCB = ( TCB_t * ) pvPortMalloc( sizeof( TCB_t ) );
7                /*lint !e961 MISRA exception as the casts are only redundant for some paths. */
8                if( pxNewTCB != NULL )
9                {
10                    /* Store the stack location in the TCB. */
11                    pxNewTCB->pxStack = pxStack;
12                }
13                else
14                {
15                    /* The stack cannot be used as the TCB was not created.  Free
16                    it again. */

17                    vPortFree( pxStack );
18                }
19            }
20            else
21            {
22                pxNewTCB = NULL;
23            }
24        }

  首先是利用pvPortMalloc给任务的堆栈分配空间if( pxStack != NULL )如果内存申请成功就接着给任务控制块申请内存pxNewTCB = ( TCB_t * ) pvPortMalloc( sizeof( TCB_t ) );同样是使用pvPortMalloc();如果任务控制块内存申请失败则释放 之前已经申请成功的任务堆栈的内存vPortFree( pxStack );

  然后就初始化任务相关的东西,并且将新初始化的任务控制块添加到列表中prvAddNewTaskToReadyList( pxNewTCB );

  最后返回任务的状态,如果是成功了就是pdPASS,假如失败了就是返回errCOULD_NOT_ALLOCATE_REQUIRED_MEMORY;

 

 1prvInitialiseNewTask(     pxTaskCode, 
2                          pcName,
3                         ( uint32_t ) usStackDepth,
4                          pvParameters,
5                          uxPriority,
6                         pxCreatedTask,
7                          pxNewTCB,
8                         NULL );
9            prvAddNewTaskToReadyList( pxNewTCB );
10            xReturn = pdPASS;
11        }
12        else
13        {
14            xReturn = errCOULD_NOT_ALLOCATE_REQUIRED_MEMORY;
15        }
16        return xReturn;
17    }
18// 相关宏定义
19#define pdPASS            ( pdTRUE )
20#define pdTRUE            ( ( BaseType_t ) 1 )
21/* FreeRTOS error definitions. */
22#define errCOULD_NOT_ALLOCATE_REQUIRED_MEMORY    ( -1 )


  具体的static void prvInitialiseNewTask(()实现请参考FreeRTOStasks.c文件的767行代码。具体的static void prvAddNewTaskToReadyList( TCB_t *pxNewTCB )实现请参考FreeRTOStasks.c文件的963行代码。


  因为这些是tasks.c中的静态的函数,仅供xTaskCreate创建任务内部调用的,我们无需理会这些函数的实现过程,当然如果需要请自行了解。


创建完任务就开启任务调度了:

1vTaskStartScheduler();          //开启任务调度


在任务调度里面,会创建一个空闲任务(我们将的都是动态创建任务,静态创建其实一样的)

 1xReturn = xTaskCreate(    prvIdleTask,
2                          "IDLE", configMINIMAL_STACK_SIZE,
3                          ( void * ) NULL,
4                          ( tskIDLE_PRIORITY | portPRIVILEGE_BIT ),
5                          &xIdleTaskHandle );
6/*lint !e961 MISRA exception, justified as it is not a redundant explicit cast to all supported compilers. */
7    }
8相关宏定义:
9#define tskIDLE_PRIORITY            ( ( UBaseType_t ) 0U )
10#ifndef portPRIVILEGE_BIT
11    #define portPRIVILEGE_BIT ( ( UBaseType_t ) 0x00 )
12#endif
13#define configUSE_TIMERS                        1                              
14 //为1时启用软件定时器

从上面的代码我们可以看出,空闲任务的优先级tskIDLE_PRIORITY0,也就是说空闲任务的优先级最低。当CPU没事干的时候才执行空闲任务,以待随时切换优先级更高的任务。

如果使用了软件定时器的话,我们还需要创建定时器任务,创建的函数是:

1#if ( configUSE_TIMERS == 1 )
2    BaseType_t xTimerCreateTimerTask( void )
3

然后还要把中断关一下

1portDISABLE_INTERRUPTS();

至于为什么关中断,也有说明:

 1/* Interrupts are turned off here, toensure a tick does not occur
2before or during the call toxPortStartScheduler().  The stacks of
3the created tasks contain a status wordwith interrupts switched on
4so interrupts will automatically getre-enabled when the first task
5starts to run. */

6/ *中断在这里被关闭,以确保不会发生滴答
7在调用xPortStartScheduler()之前或期间。堆栈
8创建的任务包含一个打开中断的状态字
9因此中断将在第一个任务时自动重新启用
10开始运行。*/


那么如何打开中断呢????这是个很重要的问题

别担心,我们在SVC中断服务函数里面就会打开中断的

看代码:

 1__asm void vPortSVCHandler( void )
2
{
3         PRESERVE8
4         ldr    r3, =pxCurrentTCB  /* Restore the context. */
5         ldrr1, [r3]                            /* UsepxCurrentTCBConst to get the pxCurrentTCB address. */
6         ldrr0, [r1]                            /* Thefirst item in pxCurrentTCB is the task top of stack. */
7         ldmiar0!, {r4-r11}             /* Pop theregisters that are not automatically saved on exception entry and the criticalnesting count. */
8         msrpsp, r0                                   /*Restore the task stack pointer. */
9         isb
10         movr0, #0
11         msr  basepri, r0
12         orrr14, #0xd
13         bxr14
14}

 

1msr  basepri, r0

  就是它把中断打开的。看不懂没所谓,我也不懂汇编,看得懂知道就好啦。

 

1xSchedulerRunning = pdTRUE;

任务调度开始运行

 

1/* If configGENERATE_RUN_TIME_STATS isdefined then the following
2macro must be defined to configure thetimer/counter used to generate
3the run time counter time base. */

4portCONFIGURE_TIMER_FOR_RUN_TIME_STATS();

 

如果configGENERATE_RUN_TIME_STATS使用时间统计功能,这个宏为1,那么用户必须实现一个宏portCONFIGURE_TIMER_FOR_RUN_TIME_STATS();用来配置一个定时器或者计数器。

 

来到我们的重点了,开启任务调度,那么任务到这了就不会返回了。

1if( xPortStartScheduler() != pdFALSE )
2                   {
3                            /*Should not reach here as if the scheduler is running the
4                            functionwill not return. */
5                   }

然后就能开启第一个任务了,感觉好难是吧,我一开始也是觉得的,但是写了这篇文章,觉得还行吧,也不算太难,可能也是在查看代码跟别人的书籍吧,写东西其实还是蛮好的,能加深理解,写过文章的人就知道,懂了不一定能写出来,所以,我还是很希望朋友们能投稿的。杰杰随时欢迎。。。

 

开始任务就按照套路模板添加自己的代码就好啦,很简单的。

创建任务

 1 xTaskCreate((TaskFunction_t )led0_task,    
2                (const char*    )"led0_task",  
3                (uint16_t       )LED0_STK_SIZE,
4                (void*          )NULL,                                    
5                (UBaseType_t    )LED0_TASK_PRIO,  
6                (TaskHandle_t*  )&LED0Task_Handler);  
7   //创建LED1任务
8   xTaskCreate((TaskFunction_t )led1_task,    
9                (const char*    )"led1_task",  
10                (uint16_t       )LED1_STK_SIZE,
11                (void*          )NULL,
12                (UBaseType_t    )LED1_TASK_PRIO,
13                (TaskHandle_t*  )&LED1Task_Handler);      

 

创建完任务就开启任务调度

1vTaskStartScheduler();          //开启任务调度


然后具体实现任务函数

 

 1//LED0任务函数
2void led0_task(void *pvParameters)
3
{
4   while(1)
5    {
6       LED0=~LED0;
7       vTaskDelay(500);
8    }
9}  
10//LED1任务函数
11void led1_task(void *pvParameters)
12
{
13   while(1)
14    {
15       LED1=0;
16       vTaskDelay(200);
17       LED1=1;
18       vTaskDelay(800);
19    }
20}


好啦,今天的介绍到这了为止,后面还会持续更新,敬请期待哦~


欢迎大家一起来讨论操作系统的知识

我们的群号是:783234154



【连载】从单片机到操作系统①

【连载】从单片机到操作系统②

【连载】从单片机到操作系统③——走进FreeRTOS


创客:

创客飞梦空间是开源公众号

欢迎大家分享出去

也欢迎大家投稿




  • 9
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
FreeRTOS 中,任务调度是由内核的调度器负责的以下是 FreeRTOS任务调度的详细步骤: 1. 初始化:在系统启动时,FreeRTOS 内核初始化,并创建一个空闲任务(Idle Task),该任务在没有其他任务可运行时被调度执行。 2. 任务创建:应用程序可以使用 FreeRTOS 提供的 API 函数,如 `xTaskCreate` 来创建任务。在创建任务时,需要指定任务的优先级、堆栈大小和任务函数等参数。 3. 任务就绪:一旦任务创建,它就处于就绪状态,表示任务已准备好运行。但此时任务还没有被调度执行。 4. 调度器启动:在系统初始化完成后,调度器开始工作。调度器会选择一个具有最高优先级的就绪任务,并将其设置为当前运行任务。 5. 任务切换:一旦当前运行任务被确定,调度器会切换到该任务的上下文,并开始执行该任务的代码。任务可以执行一段时间,直到它自愿放弃 CPU 控制权,或者因为时间片用完、等待事件发生等原因而被剥夺 CPU 控制权。 6. 任务阻塞:任务可以通过调用 FreeRTOS 提供的阻塞函数(如 `vTaskDelay`、`xQueueReceive` 等)主动放弃 CPU 控制权,并进入阻塞状态,等待某个事件的发生。在任务阻塞期间,调度器会选择下一个就绪任务运行。 7. 中断处理:当系统中断发生时,中断服务程序被执行。如果中断服务程序需要唤醒一个任务,可以使用 FreeRTOS 的中断安全机制,如 `xSemaphoreGiveFromISR` 或 `xTaskResumeFromISR`。 8. 任务唤醒:当一个任务等待的事件发生时(如消息队列有数据可用、定时器到期等),该任务会由阻塞状态切换到就绪状态,并参与到任务调度中。 9. 任务优先级调度FreeRTOS 使用优先级抢占调度算法。当两个或多个任务具有相同优先级时,调度器会按照时间片轮转的方式进行调度。如果某个任务的优先级高于当前正在运行的任务调度器会立即切换到该任务执行。 10. 任务删除:当一个任务完成了它的工作或不再需要执行时,可以通过调用 `vTaskDelete` 函数来删除该任务。被删除的任务将不再参与任务调度。 以上是 FreeRTOS任务调度的一般步骤。通过这种方式,FreeRTOS 能够高效地管理和调度多个任务,并提供实时性能保证。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值