(linux自学笔记)linux驱动并发控制、阻塞/非阻塞IO、异步通知

 原文地址:

http://www.cnblogs.com/hebaichuanyeah/

 

1.关于并发控制

Linux 中多个进程对共享资源的并发访问,并发访问会导致竞态。

  在单 CPU范围内避免竞态的一种方法是在进入临界区之前屏蔽系统的中断。CPU一般都具备屏蔽中断和打开中断的功能,中断屏蔽将使得中断与进程之间的并发不再发生,由于Linux 内核的进程调度等操作都依赖中断来实现,内核抢占进程之间的并发也就得以避免了。

local_irq_disable() //屏蔽中断
...
critical section //临界区
...
local_irq_enable() //开中断

 

原子操作

原子操作指的是在执行过程中不会被别的代码路径所中断的操作。

原子操作驱动

#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <asm/uaccess.h> /* copy_to_user,copy_from_user */
#include <linux/miscdevice.h>
#include <linux/pci.h>
#include <mach/map.h>
#include <mach/regs-gpio.h>
#include <mach/gpio-bank-m.h>
#include <plat/gpio-cfg.h>

#define LED_MAJOR 240
//原子变量
atomic_t atomic_led = ATOMIC_INIT(0);


int led_open (struct inode *inode,struct file *filp)
{
    unsigned tmp;
    //减1兼测试
    if (!atomic_dec_and_test(&atomic_led))
    {
        //加1
        atomic_inc(&atomic_led);
        return -1;
    }


    tmp = readl(S3C64XX_GPMDAT);
    tmp &= 0xffff0000;
    tmp |= 0x00001111;
    writel(tmp, S3C64XX_GPMCON);
    writel(0xffffffff, S3C64XX_GPMDAT);
    printk("<0>open led\n");
    return 0;
}

ssize_t led_read (struct file *filp, char __user *buf, size_t count,loff_t *f_pos)
{
    return count;
}


ssize_t led_write (struct file *filp, const char __user *buf, size_t count,loff_t *f_pos)
{
    char wbuf[1];
    unsigned tmp;

    copy_from_user(wbuf,buf,count);

    tmp = readl(S3C64XX_GPMDAT);
    tmp &= 0xfffffff0;
    tmp |= wbuf[0];
    writel(tmp, S3C64XX_GPMDAT);

    return count;
}

int led_release (struct inode *inode, struct file *filp)
{
    atomic_inc(&atomic_led);

    return 0;
}


struct file_operations led_fops ={
    .owner = THIS_MODULE,
    .open = led_open,
    .read = led_read,
    .write = led_write,
    .release = led_release,

};

int __init led_init (void)
{
    int rc;
    printk ("<0>led init\n");
    rc = register_chrdev(LED_MAJOR,"led",&led_fops);
    if (rc <0)
    {
        printk ("<0>register %s char dev error\n","led");
        return -1;
    }

    //atomic_t atomic_led = ATOMIC_INIT(0);
    //原子变量置1
    atomic_set(&atomic_led, 1);
    printk ("<0>success\n");
    return 0;
}

void __exit led_exit (void)
{
    unregister_chrdev(LED_MAJOR,"led");
    printk ("<0>module exit\n");
    return ;
}

module_init(led_init);
module_exit(led_exit);

测试程序

#include <fcntl.h>
#include "stdio.h"
#include "pthread.h"
#include "stdlib.h"

void thread_one(void)
{
    int fd,count;
    char buf[]={~0x01,~0x02,~0x04,~0x08,0x00,0xff};


    while((fd = open("/dev/my_led",O_RDWR))<0)
    {
        printf ("pthread_one:Open /dev/my_led file error\n");
        //pthread_exit(0);
        sleep(1);
    }
    for(count=0;count<3;count++)
    {
        write(fd,&buf[0],1);
        sleep(1);
        write(fd,&buf[5],1);
        sleep(1);
    }
    close (fd);
}

void thread_two(void)
{
    int fd,count;
    char buf[]={~0x01,~0x02,~0x04,~0x08,0x00,0xff};


    while ((fd = open("/dev/my_led",O_RDWR)) < 0)
    {
        printf ("pthread_two:Open /dev/my_led file error\n");
        //pthread_exit(0);
    sleep(1);
    }
    for(count=0;count<3;count++)
    {
        write(fd,&buf[1],1);
        sleep(1);
        write(fd,&buf[5],1);
        sleep(1);
    }
    close(fd);
}

main()
{
    pthread_t thread_one_id,thread_two_id;
    int resurt;
    pthread_attr_t thread_one_attr,thread_two_attr;
    int thread_one_priority,thread_two_priority;



    thread_one_priority = 11;
    thread_two_priority = 11;

    pthread_attr_init(&thread_one_attr);
    pthread_attr_init(&thread_two_attr);

    //设置为指定调度算法
    pthread_attr_setinheritsched(&thread_one_attr, PTHREAD_EXPLICIT_SCHED);
    pthread_attr_setinheritsched(&thread_two_attr, PTHREAD_EXPLICIT_SCHED);

    //指定调度算法
    pthread_attr_setschedpolicy(&thread_one_attr, SCHED_RR);
    pthread_attr_setschedpolicy(&thread_two_attr, SCHED_RR);

    pthread_attr_setschedparam(&thread_one_attr, (struct sched_param *)&thread_one_priority);

    pthread_attr_setschedparam(&thread_two_attr, (struct sched_param *)&thread_two_priority);

    pthread_attr_setdetachstate(&thread_one_attr,PTHREAD_CREATE_JOINABLE);
    //sleep(1);
    if((resurt = pthread_create(&thread_one_id, &thread_one_attr, (void *)thread_one, NULL)) == -1)
    {
        perror("ss");
        printf("thread one creat error\n %d",resurt);
        exit(1);
    }


    if((resurt = pthread_create(&thread_two_id, &thread_two_attr, (void *)thread_two, NULL))== -1)
    {
        printf("thread two creat error\n");
        exit(1);
    }



    //sleep(1);
    pthread_join(thread_one_id, NULL);
    pthread_join(thread_two_id, NULL);
}

运行结果

 

 自旋锁

  自旋锁是一种对临界资源进行互斥手访问的典型手段。为了获得一个自旋锁,在某CPU 上运行的代码需先执行一个原子操作,该操作测试并设置(test-and-set)某个内存变量,由于它是原子操作,所以在该操作完成之前其他执行单元不可能访问这个内存变量。

自旋锁保护临界段

    
    spinlock(&lock_led);
    if (count)/*已经打开*/
    {
        spin_unlock(&lock_led);
        return -1;
    }

    spin_unlock(&lock_led);

 

信号量

//获得信号量

void down(struct semaphore * sem);

//释放信号量

void up(struct semaphore * sem);

 

 

 

 

 

 

 

 

 

 

 2 .关于阻塞/非阻塞IO

 

 

阻塞和非阻塞I/O 是设备访问的两种不同模式,驱动程序可以灵活地支持用户空间对设备的这两种访问方式。

  阻塞操作是指在执行设备操作时若不能获得资源则挂起进程,直到满足可操作的条件后再进行操作。被挂起的进程进入休眠状态,被从调度器的运行队列移走,直到等待的条件被满足。而非阻塞操作的进程在不能进行设备操作时并不挂起,它或者放弃,或者不停地查询,直至可以进行操作为止。

阻塞一个线程,直至另一个线程运行完。

include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <asm/uaccess.h> /* copy_to_user,copy_from_user */
#include <linux/miscdevice.h>
#include <linux/pci.h>
#include <mach/map.h>
#include <mach/regs-gpio.h>
#include <mach/gpio-bank-m.h>
#include <plat/gpio-cfg.h>
#include <linux/wait.h>
#include <linux/sched.h>

#define LED_MAJOR 240

struct my_queue
{
    //定义等待列对头
    wait_queue_head_t queue_head;
    //定义自旋锁
    spinlock_t lock;
    //文件打开标志
    int file_flag;
}led_queue;


int led_open (struct inode *inode,struct file *filp)
{
    unsigned tmp;
    struct my_queue * dev = &led_queue;
    //自旋锁 临界段
    spin_lock(&dev->lock);

    //定义等待列对
    DECLARE_WAITQUEUE(wait,current);
    //进入等待列对头
    add_wait_queue(&dev->queue_head, &wait);
    dev->file_flag--;
    spin_unlock(&dev->lock);
    while(dev->file_flag < 0)
    {
        //改变进程状态
         __set_current_state(TASK_INTERRUPTIBLE);
        //
        schedule();
    }

    spin_lock(&dev->lock);
    dev->file_flag = 0;
    spin_unlock(&dev->lock);


    tmp = readl(S3C64XX_GPMDAT);
    tmp &= 0xffff0000;
    tmp |= 0x00001111;
    writel(tmp, S3C64XX_GPMCON);
    writel(0xffffffff, S3C64XX_GPMDAT);
    printk("<0>open led\n");
    //移除
    remove_wait_queue(&dev->queue_head, &wait);
    return 0;
}

ssize_t led_read (struct file *filp, char __user *buf, size_t count,loff_t *f_pos)
{
    return count;
}


ssize_t led_write (struct file *filp, const char __user *buf, size_t count,loff_t *f_pos)
{
    char wbuf[1];
    unsigned tmp;

    copy_from_user(wbuf,buf,count);

    tmp = readl(S3C64XX_GPMDAT);
    tmp &= 0xfffffff0;
    tmp |= wbuf[0];
    writel(tmp, S3C64XX_GPMDAT);
    return count;
}

int led_release (struct inode *inode, struct file *filp)
{
    struct my_queue * dev = &led_queue;
    spin_lock(&dev->lock);
    dev->file_flag = 1;
    spin_unlock(&dev->lock);

    wake_up_interruptible(&dev->queue_head);

    return 0;
}


struct file_operations led_fops ={
    .owner = THIS_MODULE,
    .open = led_open,
    .read = led_read,
    .write = led_write,
    .release = led_release,

};

int __init led_init (void)
{
    int rc;
    struct my_queue * dev = &led_queue;

    printk ("<0>led init\n");
    rc = register_chrdev(LED_MAJOR,"led",&led_fops);
    if (rc <0)
    {
        printk ("<0>register %s char dev error\n","led");
        return -1;
    }
    //初始化列对头
    init_waitqueue_head(&dev->queue_head);
    //初始化自旋锁
    spin_lock_init(&dev->lock);
    //文件打开标志
    dev->file_flag = 1;
    printk ("<0>success\n");
    return 0;
}

void __exit led_exit (void)
{
    unregister_chrdev(LED_MAJOR,"led");
    printk ("<0>module exit\n");
    return ;
}

module_init(led_init);
module_exit(led_exit);

 

 

非阻塞,poll轮循

#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <asm/uaccess.h> /* copy_to_user,copy_from_user */
#include <linux/miscdevice.h>
#include <linux/pci.h>
#include <mach/map.h>
#include <mach/regs-gpio.h>
#include <mach/gpio-bank-m.h>
#include <plat/gpio-cfg.h>
#include <linux/wait.h>
#include <linux/sched.h>
#include <linux/poll.h>

#define LED_MAJOR 240

struct drive_str
{
    //定义等待列对头
    wait_queue_head_t queue_head;
    //定义自旋锁
    spinlock_t lock;
    //文件打开标志
    int file_flag;
}led_driver;


int led_open (struct inode *inode,struct file *filp)
{
    unsigned tmp;


    tmp = readl(S3C64XX_GPMDAT);
    tmp &= 0xffff0000;
    tmp |= 0x00001111;
    writel(tmp, S3C64XX_GPMCON);
    writel(0xffffffff, S3C64XX_GPMDAT);
    printk("<0>open led\n");
    //移除
    return 0;
}

ssize_t led_read (struct file *filp, char __user *buf, size_t count,loff_t *f_pos)
{
    struct drive_str * dev = &led_driver;
    //自旋锁 临界段
    spin_lock(&dev->lock);
    if(dev->file_flag == 0)
        return (0);

    dev->file_flag = 0;
    spin_unlock(&dev->lock);
    return (1);
}


ssize_t led_write (struct file *filp, const char __user *buf, size_t count,loff_t *f_pos)
{
    char wbuf[1];
    unsigned tmp;



    copy_from_user(wbuf,buf,count);

    tmp = readl(S3C64XX_GPMDAT);
    tmp &= 0xfffffff0;
    tmp |= wbuf[0];
    writel(tmp, S3C64XX_GPMDAT);
    return count;
}

int led_release (struct inode *inode, struct file *filp)
{
    struct drive_str * dev = &led_driver;

    spin_lock(&dev->lock);
    dev->file_flag = 1;
    spin_unlock(&dev->lock);

    return 0;
}

static unsigned int led_poll(struct file * filp,poll_table *wait)
{
    unsigned int mask = 0;
    struct drive_str *dev = &led_driver;

    poll_wait(filp, &dev->queue_head,wait);

    if(dev->file_flag)
    {
        mask = 1;
    }

    return (mask);
}


struct file_operations led_fops ={
    .owner = THIS_MODULE,
    .open = led_open,
    .read = led_read,
    .write = led_write,
    .release = led_release,
    .poll = led_poll,

};

int __init led_init (void)
{
    int rc;
    struct drive_str * dev = &led_driver;

    printk ("<0>led init\n");
    rc = register_chrdev(LED_MAJOR,"led",&led_fops);
    if (rc <0)
    {
        printk ("<0>register %s char dev error\n","led");
        return -1;
    }
    //初始化列对头
    init_waitqueue_head(&dev->queue_head);
    //初始化自旋锁
    spin_lock_init(&dev->lock);
    //文件打开标志
    dev->file_flag = 1;
    printk ("<0>success\n");
    return 0;
}

void __exit led_exit (void)
{
    unregister_chrdev(LED_MAJOR,"led");
    printk ("<0>module exit\n");
    return ;
}

module_init(led_init);
module_exit(led_exit);

测试程序

#include "stdio.h"
#include "pthread.h"
#include "stdlib.h"
#include <fcntl.h>
#include "poll.h"

void thread_one(void)
{
    int fd,count,ret;
    char buf[]={~0x01,~0x02,~0x04,~0x08,0x00,0xff};
    struct pollfd fds[1];

    fd = open("/dev/my_led",O_RDWR);
    fds[0].fd = fd;
    fds[0].events = POLLIN;


    if (fd < 0)
    {
        printf ("Open /dev/my_led file error\n");
        pthread_exit(0);
    }
    if((ret = read(fd,&buf[1],1)) == 0)
    {
        ret = poll(fds,1,10000);
        if(ret == 0)
        {
            printf("time out\n");
        printf("%u\n",ret);
            pthread_exit(0);

        }

    }

    for(count=0;count<3;count++)
    {
        write(fd,&buf[0],1);
        sleep(1);
        write(fd,&buf[5],1);
        sleep(1);
    }
    close (fd);
}

void thread_two(void)
{
    int fd,count,ret;
    char buf[]={~0x01,~0x02,~0x04,~0x08,0x00,0xff};
    struct pollfd fds[1];



    fd = open("/dev/my_led",O_RDWR);
    fds[0].fd = fd;
    fds[0].events = POLLIN;
    if (fd == 1)
    {
        printf ("open /dev/my_led file error\n");
        pthread_exit(0);
    }

    if((ret = read(fd,&buf[1],1)) == 0)
    {
        ret = poll(fds,1,10000);
        if(ret == 0)
        {
            printf("time out\n");
        printf("%u\n",ret);
            pthread_exit(0);

        }

    }
    for(count=0;count<3;count++)
    {
        write(fd,&buf[1],1);
        sleep(1);
        write(fd,&buf[5],1);
        sleep(1);
    }
    close(fd);
}

main()
{
    pthread_t thread_one_id,thread_two_id;
    int resurt;
    pthread_attr_t thread_one_attr,thread_two_attr;
    int thread_one_priority,thread_two_priority;



    thread_one_priority = 11;
    thread_two_priority = 11;

    pthread_attr_init(&thread_one_attr);
    pthread_attr_init(&thread_two_attr);

    //设置为指定调度算法
    pthread_attr_setinheritsched(&thread_one_attr, PTHREAD_EXPLICIT_SCHED);
    pthread_attr_setinheritsched(&thread_two_attr, PTHREAD_EXPLICIT_SCHED);

    //指定调度算法
    pthread_attr_setschedpolicy(&thread_one_attr, SCHED_RR);
    pthread_attr_setschedpolicy(&thread_two_attr, SCHED_RR);

    pthread_attr_setschedparam(&thread_one_attr, (struct sched_param *)&thread_one_priority);

    pthread_attr_setschedparam(&thread_two_attr, (struct sched_param *)&thread_two_priority);

    pthread_attr_setdetachstate(&thread_one_attr,PTHREAD_CREATE_JOINABLE);
    //sleep(1);
    if((resurt = pthread_create(&thread_one_id, &thread_one_attr, (void *)thread_one, NULL)) == -1)
    {
        printf("thread one creat error\n %d",resurt);
        exit(1);
    }


    if((resurt = pthread_create(&thread_two_id, &thread_two_attr, (void *)thread_two, NULL))== -1)
    {
        printf("thread two creat error\n");
        exit(1);
    }



    //sleep(1);
    pthread_join(thread_one_id, NULL);
    pthread_join(thread_two_id, NULL);
}

 

 

 

 

 

 

 

 

 

 

 

3.异步通知

 

 

使用信号进行进程间通信是UNIX 系统中的一种传统机制。在Linux系统中,异步通知使用信号来实现。

 

使用异步通知,一个进程结束后发信号给另一个进程

驱动程序:

include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <asm/uaccess.h> /* copy_to_user,copy_from_user */
#include <linux/miscdevice.h>
#include <linux/pci.h>
#include <mach/map.h>
#include <mach/regs-gpio.h>
#include <mach/gpio-bank-m.h>
#include <plat/gpio-cfg.h>
#include <linux/wait.h>
#include <linux/sched.h>
#include <linux/poll.h>
#include <asm/signal.h>
#include <asm-generic/siginfo.h>

#define LED_MAJOR 240

struct drive_str
{
    struct fasync_struct * async
}led_driver;



static int led_open (struct inode *inode,struct file *filp)
{
    unsigned tmp;


    tmp = readl(S3C64XX_GPMDAT);
    tmp &= 0xffff0000;
    tmp |= 0x00001111;
    writel(tmp, S3C64XX_GPMCON);
    writel(0xffffffff, S3C64XX_GPMDAT);
    printk("<0>open led\n");

    return 0;
}

static ssize_t led_read (struct file *filp, char __user *buf, size_t count,loff_t *f_pos)
{

    return (count);
}


static ssize_t led_write (struct file *filp, const char __user *buf, size_t count,loff_t *f_pos)
{
    char wbuf[1];
    unsigned tmp;

    copy_from_user(wbuf,buf,count);

    tmp = readl(S3C64XX_GPMDAT);
    tmp &= 0xfffffff0;
    tmp |= wbuf[0];
    writel(tmp, S3C64XX_GPMDAT);
    return count;
}

static int led_fasync(int fd,struct file * filp,int mode)
{
    struct drive_str * dev = &led_driver;
    return (fasync_helper(fd,filp,mode,&dev->async));
}

static int led_release (struct inode *inode, struct file *filp)
{
    struct drive_str * dev = &led_driver;


    kill_fasync(&dev->async,SIGIO, POLL_IN);

    return (0);

}



struct file_operations led_fops ={
    .owner = THIS_MODULE,
    .open = led_open,
    .read = led_read,
    .write = led_write,
    .release = led_release,
    .fasync = led_fasync,

};

static int __init led_init (void)
{
    int rc;
    struct drive_str * dev = &led_driver;

    printk ("<0>led init\n");
    rc = register_chrdev(LED_MAJOR,"led",&led_fops);
    if (rc <0)
    {
        printk ("<0>register %s char dev error\n","led");
        return -1;
    }

    dev->async = NULL;
    printk ("<0>success\n");
    return 0;
}

static void __exit led_exit (void)
{
    unregister_chrdev(LED_MAJOR,"led");
    printk ("<0>module exit\n");
    return ;
}

module_init(led_init);
module_exit(led_exit);

 

测试程序:

#include "stdio.h"
#include "pthread.h"
#include "stdlib.h"
#include <fcntl.h>
#include <signal.h>
#include <unistd.h>
int thread_two_fd;

void sign_handler(int signum);

void thread_one(void)
{
    int fd,count;
    char buf[]={~0x01,~0x02,~0x04,~0x08,0x00,0xff};


    fd = open("/dev/my_led",O_RDWR);
    if (fd < 0)
    {
        printf ("Open /dev/my_led file error\n");
        pthread_exit(0);
    }
    for(count=0;count<3;count++)
    {
        write(fd,&buf[0],1);
        sleep(1);
        write(fd,&buf[5],1);
        sleep(1);
    }
    close (fd);
}

void thread_two(void)
{
    int fd,oflags;
    char buf[]={~0x01,~0x02,~0x04,~0x08,0x00,0xff};


    fd = open("/dev/my_led",O_RDWR);
    if (fd < 0)
    {
        printf ("open /dev/my_led file error\n");
        pthread_exit(0);
    }
    signal(SIGIO,sign_handler);
    //改变已打开文件属性
    fcntl(fd, F_SETOWN, getpid());
    oflags = fcntl(fd, F_GETFL);
    fcntl(fd, F_SETFL, oflags | FASYNC);
    thread_two_fd = fd;
    pause();

    close(fd);
}

main()
{
    pthread_t thread_one_id,thread_two_id;
    int resurt;
    pthread_attr_t thread_one_attr,thread_two_attr;
    int thread_one_priority,thread_two_priority;



    thread_one_priority = 11;
    thread_two_priority = 11;

    pthread_attr_init(&thread_one_attr);
    pthread_attr_init(&thread_two_attr);

    //设置为指定调度算法
    pthread_attr_setinheritsched(&thread_one_attr, PTHREAD_EXPLICIT_SCHED);
    pthread_attr_setinheritsched(&thread_two_attr, PTHREAD_EXPLICIT_SCHED);

    //指定调度算法
    pthread_attr_setschedpolicy(&thread_one_attr, SCHED_RR);
    pthread_attr_setschedpolicy(&thread_two_attr, SCHED_RR);

    pthread_attr_setschedparam(&thread_one_attr, (struct sched_param *)&thread_one_priority);

    pthread_attr_setschedparam(&thread_two_attr, (struct sched_param *)&thread_two_priority);

    pthread_attr_setdetachstate(&thread_one_attr,PTHREAD_CREATE_JOINABLE);
    //sleep(1);
    if((resurt = pthread_create(&thread_one_id, &thread_one_attr, (void *)thread_one, NULL)) == -1)
    {
        printf("thread one creat error\n %d",resurt);
        exit(1);
    }


    if((resurt = pthread_create(&thread_two_id, &thread_two_attr, (void *)thread_two, NULL))== -1)
    {
        printf("thread two creat error\n");
        exit(1);
    }



    //sleep(1);
    pthread_join(thread_one_id, NULL);
    pthread_join(thread_two_id, NULL);
}
void sign_handler(int signum)
{
    int count;
    char buf[]={~0x01,~0x02,~0x04,~0x08,0x00,0xff};

    printf("sign_handle\n");
    for(count=0;count<3;count++)
    {

        write(thread_two_fd ,&buf[1],1);
        sleep(1);
        write(thread_two_fd ,&buf[5],1);
        sleep(1);
    }
 
}

 

posted @ 2014-01-20 07:05  默默地EEer  阅读(731)  评论(0编辑  收藏  举报