[RTT例程练习] 2.3 信号量-检测按键(同步) -(信号量互斥)

这里,使用信号量来同步两个线程。

简单来说,线程每隔一定的时间扫描按键。一旦其检测到有效的按键,便release 信号量,使得线程2 从挂起进入 运行态,向串口打印按键信息。

这样,便实现了两个线程的同步。

程序:

#include <rtthread.h>

#include "stm32f10x.h"

#define KEY_PORT    GPIOB
#define KEY_PIN     (GPIO_Pin_0 | GPIO_Pin_1)
#define KEY_CLOCK   RCC_AHB1Periph_GPIOB

void rt_init_thread_entry(void *parameter)
{
	GPIO_InitTypeDef GPIO_InitStructure;
	
	//开启PB0时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
	
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_10MHz;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
	
	GPIO_Init(GPIOB,&GPIO_InitStructure);
}   

static rt_uint8_t thread1_stack[1024];
struct rt_thread thread1;

static int key;
static struct rt_semaphore sem;

static void rt_thread_entry1(void *parameter)
{
    int temp;
    
    while (1)
    {
        key = GPIO_ReadInputData(KEY_PORT);
        
        if (key & KEY_PIN)
        {
            temp = key;
            rt_thread_delay(RT_TICK_PER_SECOND / 50);
            key = GPIO_ReadInputData(KEY_PORT);
            if (key == temp);
                rt_sem_release(&sem);
        }
        
        rt_thread_delay(RT_TICK_PER_SECOND / 10);
    }
}

static rt_uint8_t thread2_stack[1024];
struct rt_thread thread2;

//static int key;
static void rt_thread_entry2(void *parameter)
{
    while (1)
    {
        rt_sem_take(&sem, RT_WAITING_FOREVER);
        
        if (key & KEY_PIN)
        {
            rt_kprintf("some keys has been pressed: %x\n", key);
        }
    }
}

int rt_application_init()
{
    rt_thread_t init_thread;
    rt_err_t result;
    
    result = rt_sem_init(&sem, "sem",
        0, RT_IPC_FLAG_FIFO);
    if (result != RT_EOK)
    {
        rt_kprintf("error, init sem failed!\n");
        return 0;
    }
    
    init_thread = rt_thread_create("init", 
        rt_init_thread_entry,
        RT_NULL,
        2048, 8, 20);
    if (init_thread != RT_NULL)
        rt_thread_startup(init_thread);
    
    /* init thread1 */
    rt_thread_init(&thread1, 
        "keyp", // producer
        rt_thread_entry1, 
        RT_NULL, 
        &thread1_stack[0], sizeof(thread1_stack),
        11, 25);
    rt_thread_startup(&thread1);

    rt_thread_init(&thread2,
        "keyc", // consumer
        rt_thread_entry2, 
        RT_NULL,
        &thread2_stack[0], sizeof(thread2_stack),
        11, 24);
    rt_thread_startup(&thread2);
    
    return 0;
}

结果为:

some keys has been pressed : 40
some keys has been pressed : 80


posted @ 2013-02-25 21:31  lyyyuna  阅读(344)  评论(0编辑  收藏  举报