您的位置:首页 > 其它

keil MDK 之RTX信号量的API函数

2015-12-20 23:30 411 查看
一、RTX信号量的API函数总共有4个,如下图



下面我们对这四个函数进行解析

1、os_sem_init函数原型

void os_sem_init (
OS_ID semaphore,        /* The semaphore object to initialize */
U16   token_count );    /* Initial number of tokens */

函数描述:该函数用于信号量的初始化并设置信号量的值

第一个参数为信号量的ID

第二个参数为信号量的初始值

使用举例:

__task void task1 (void) {
..
os_sem_init (&semaphore1, 0);
os_sem_send (&semaphore1);
..
}

2、os_sem_send函数原型

OS_RESULT os_sem_send (
OS_ID semaphore );    /* The semaphore whose token count is incremented */

函数描述:

该函数用于释放信号量,调用后信号量加1

第一个参数是信号量的ID

使用举例:

__task void task1 (void) {
..
os_sem_init (&semaphore1, 0);
os_sem_send (&semaphore1);
..
}

3、isr_sem_send函数原型

void isr_sem_send (
OS_ID semaphore );    /* The semaphore whose token count is incremented */

函数描述:

该函数用于释放信号量,只能在中断中使用,调用后信号量加1

第一个参数是信号量的ID

使用举例:

void timer1 (void) __irq {
..
isr_sem_send (&semaphore1);
..
}

4、os_sem_wait函数原型

OS_RESULT os_sem_wait (
OS_ID semaphore,    /* The semaphore to get the token from */
U16   timeout );    /* Length of time to wait for the token */

函数描述:

该函数用于获取信号量,如果信号量大于1,就减1,否则该任务挂起,等待有信号量

第一个参数是信号量的ID

第二个参数是等待信号的延时时间  范围1 --- 0xFFFF 如果参数是0xFFFF就无线等待

使用举例:

__task void task1 (void) {
..
os_sem_wait (&semaphore1, 0xffff);
..
}

实验目的:

学习RTX的信号量

实验内容:

1、k1按下发送信号量

2、AppTaskRun任务实现LED闪烁

3、AppTaskLED等待信号量 实现LED反转

4、AppTaskUserKey任务实现按键键值的处理

下面是完整的代码

#include "bsp.h" /* 底层硬件驱动 */

#include <RTL.h>

static uint64_t AppTaskLEDStk[256/8];/*任务栈*/

static uint64_t AppTaskStartStk[512/8];/*任务栈*/

static uint64_t AppTaskUserKeyStk[512/8];/*任务栈*/

static uint64_t AppTaskRunStk[512/8];/*任务栈*/

/*任务句柄*/

OS_TID HandleTaskLED = NULL;

OS_TID HandleTaskKey = NULL;

OS_TID HandleTaskRun = NULL;

/*函数声明*/

static void AppTaskCreate(void);

__task void AppTaskLED(void);

__task void AppTaskStart(void);

__task void AppTaskKey(void);

__task void AppTaskRun(void);

/*信号量的ID*/

OS_SEM sem;

/*

*********************************************************************************************************

* 函 数 名: main

* 功能说明: c程序入口

* 形    参:无

* 返 回 值: 错误代码(无需处理)

*********************************************************************************************************

*/

int main(void)

{

/*
ST固件库中的启动文件已经执行了 SystemInit() 函数,该函数在 system_stm32f4xx.c 文件,主要功能是
配置CPU系统的时钟,内部Flash访问时序,配置FSMC用于外部SRAM
*/
bsp_Init();/**/
os_sys_init_user(AppTaskStart,            /*任务函数*/
2,                       /*任务优先级*/
&AppTaskStartStk,        /*任务栈*/
sizeof(AppTaskStartStk));/*任务栈大小*/
/* 进入主程序循环体 */
while (1)
{
;
}

}

/*

*********************************************************************************************************

* 函 数 名: AppTaskLED

* 功能说明: LED闪烁的任务

* 形    参:无

* 返 回 值: 无

*********************************************************************************************************

*/

__task void AppTaskLED(void)

{
static uint8_t i = 0;
OS_RESULT Resurt;
while(1)
{
if(i % 2 == 0)
{
GPIO_ResetBits(GPIOI,GPIO_Pin_10);/*点亮LED*/
}
else
{
GPIO_SetBits(GPIOI,GPIO_Pin_10);/*熄灭LED*/
}
i++;
Resurt = os_sem_wait(sem,0xFFFF);

switch(Resurt)
{
case OS_R_SEM:
printf("收到信号量\r\n");
break;
case OS_R_OK:
printf("无需等待,立即获得信号量\r\n");
break;
case OS_R_TMO:
printf("超时\r\n");
break;
default:
break;
}
os_dly_wait(10);/*系统延时函数 因为时钟节拍为1000 所以这里是延时800ms,也就是使AppTaskLED任务挂起800MS*/
}

}

/*

*********************************************************************************************************

* 函 数 名: AppTaskRun

* 功能说明: LED闪烁的任务

* 形    参:无

* 返 回 值: 无

*********************************************************************************************************

*/

__task void AppTaskRun(void)

{
static uint8_t i = 0;
while(1)
{
if(i % 2 == 0)
{
GPIO_ResetBits(GPIOC,GPIO_Pin_2);
}
else
{
GPIO_SetBits(GPIOC,GPIO_Pin_2);
}
i++;
os_dly_wait(200);
}

}

/*

*********************************************************************************************************

* 函 数 名: AppTaskKey

* 功能说明: 键值处理函数

* 形    参:无

* 返 回 值: 无

*********************************************************************************************************

*/

__task void AppTaskKey(void)

{
uint8_t ucKeyCode = 0;
while(1)
{
ucKeyCode = bsp_GetKey();
switch(ucKeyCode)
{
case KEY_DOWN_K1:
printf("发送信号量\r\n");
os_sem_send(sem);
break;
case KEY_DOWN_K2:
break;
case KEY_DOWN_K3:
break;
default:
break;
}
os_dly_wait(20);
}

}

/*

*********************************************************************************************************

* 函 数 名:AppTaskCreate

* 功能说明: 任务创建

* 形    参:无

* 返 回 值: 无

*********************************************************************************************************

*/

static void AppTaskCreate(void)

{
HandleTaskLED = os_tsk_create_user(AppTaskLED,            /*任务函数*/
1,                     /*优先级 注意RTX的数字越小,优先级越低*/
&AppTaskLEDStk,        /*任务栈起始地址*/
sizeof(AppTaskLEDStk));/*任务栈大小*/
HandleTaskKey = os_tsk_create_user(AppTaskKey,            /*任务函数*/
 3,                     /*优先级 注意RTX的数字越小,优先级越低*/
&AppTaskUserKeyStk,    /*任务栈起始地址*/
sizeof(AppTaskUserKeyStk));/*任务栈大小*/
HandleTaskRun = os_tsk_create_user(AppTaskRun,
 2,
&AppTaskRunStk,
sizeof(AppTaskRunStk));

}

/*

*********************************************************************************************************

* 函 数 名:AppTaskStart

* 功能说明: 开始任务

* 形    参:无

* 返 回 值: 无

*********************************************************************************************************

*/

__task void AppTaskStart(void)

{

AppTaskCreate();/*创建AppTaskLED任务*/
os_sem_init(&sem,0);
while(1)
{
os_dly_wait(10);/*系统延时函数*/
bsp_KeyScan();
}

}

实验现象:

按下K1 LED反转,同时串口调试助手可以看到相关信息
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: