全志A33 lichee Linux内核原子操作(附实测代码)

全志A33 lichee Linux内核原子操作(附实测代码)


image

开发平台

*  芯灵思SinlinxA33开发板

淘宝店铺: https://sinlinx.taobao.com/

image

嵌入式linux 开发板交流 641395230

原子操作是指不会被线程调度机制打断的操作;这种操作一旦开始,就一直运行到结束,中间不会有任何线程切换。

原子操作是不可分割的,在执行完毕之前不会被任何其它任务或事件中断。在单处理器系统(UniProcessor)中,能够在单条指令中完成的操作都可以认为是" 原子操作",因为中断只能发生于指令之间。这也是某些CPU指令系统中引入了test_and_set、test_and_clear等指令用于临界资源互斥的原因。但是,在对称多处理器(Symmetric Multi-Processor)结构中就不同了,由于系统中有多个处理器在独立地运行,即使能在单条指令中完成的操作也有可能受到干扰。我们以decl (递减指令)为例,这是一个典型的"读-改-写"过程,涉及两次内存访问。设想在不同CPU运行的两个进程都在递减某个计数值,可能发生的情况是:

  • ⒈ CPU A(CPU A上所运行的进程,以下同)从内存单元把当前计数值⑵装载进它的寄存器中;
  • ⒉ CPU B从内存单元把当前计数值⑵装载进它的寄存器中。
  • ⒊ CPU A在它的寄存器中将计数值递减为1;
  • ⒋ CPU B在它的寄存器中将计数值递减为1;
  • ⒌ CPU A把修改后的计数值⑴写回内存单元。
  • ⒍ CPU B把修改后的计数值⑴写回内存单元。

我们看到,内存里的计数值应该是0,然而它却是1。如果该计数值是一个共享资源的引用计数,每个进程都在递减后把该值与0进行比较,从而确定是否需要释放该共享资源。这时,两个进程都去掉了对该共享资源的引用,但没有一个进程能够释放它--两个进程都推断出:计数值是1,共享资源仍然在被使用。

Linux原子操作大部分使用汇编语言实现,因为c语言并不能实现这样的操作。
原子操作需要硬件的支持,因此是架构相关的,其API和原子类型的定义都定义在内核源码树的 include/asm/atomic.h 文件中

原子操作相关API

atomic.h 这个文件中包含了和具体芯片架构相关的原子操作头文件arch\arm\include\asm\atomic.h

ATOMIC_INIT(v);
作用: 初始化一个个原子变量,一般比较少用。

atomic_read(atomic_t * v);
作用: 读取原子变量中的值

atomic_set(atomic_t * v, int i);
作用: 设置原子变量值为i

`void atomic_add(int i, atomic_t *v)
作用: 把原子变量值加上i

void atomic_sub(int i, atomic_t *v)
作用: 把原子变量值减去i

atomic_sub_and_test(i, v)
作用: 把原子变量v的值减去i,判断相减后的原子变量值是否为0,如果为0返回真

atomic_inc(v);
作用: 把原子变量v加上1

atomic_dec(v)
作用: 把原子变量v减去1

atomic_dec_and_test(v)
作用: 把原子变量v的值减去1,判断相减后的原子变量值是否为0,如果为0返回真

atomic_inc_and_test(v)
作用: 把原子变量v的值加1,判断相加后的原子变量值是否为0,如果为0返回真

atomic_add_negative(i,v)
作用: 把原子变量v的值加i,判断相加后的原子变量值是否为负数,如果为负数返回真

int atomic_add_return(int i, atomic_t *v)
作用: 把原子变量v的值加i,返回相加后原子变量的结果

int atomic_sub_return(int i, atomic_t *v)
作用: 把原子变量v的值减i,返回相减后原子变量的结果

atomic_inc_return(v)
作用: 把原子变量v的值加1后,返回结果

atomic_dec_return(v)·
作用: 把原子变量v的值减1后返回结果

实验现象:当多个APP调用同一个驱动时,不会发生混乱,依次执行

image

未实现原子操作,所有进程会都执行


image

驱动代码:

#include <linux/module.h>  
#include <linux/kernel.h>  
#include <linux/fs.h>  
#include <linux/init.h>  
#include <linux/delay.h>  
#include <linux/uaccess.h>  
#include <asm/irq.h>  
#include <asm/io.h>  
#include <asm/types.h>  
#include <linux/of.h>  
#include <linux/of_device.h>  
#include <linux/of_platform.h>  
#include <linux/atomic.h>  
static int major;  
static struct class *led_class;  
volatile unsigned long *gpio_con = NULL;  
volatile unsigned long *gpio_dat = NULL;  
//定义原子变量  ,初始化值为1  
atomic_t  atomic_v = ATOMIC_INIT(1);  
static int led_open (struct inode *node, struct file *filp)  
{  
    // atomic_dec_and_test(v),判断减1结果是否0,为0返回真。  
    if( !atomic_dec_and_test(&atomic_v) ){  
      printk("done done done \n");  
        return -1;  
    }  
  
    /* PB7 - 0x01C20824 */  
    if (gpio_con) {  
        printk("ioremap  0x%x\n", gpio_con);  
    }  
    else {  
        return -EINVAL;  
    }  
    printk(" open open open  \n");  
    return 0;  
}  
  
static ssize_t led_write (struct file *filp, const char __user *buf, size_t size, loff_t *off)  
{  
    unsigned char val;          
    copy_from_user(&val, buf, 1);  
  
    if (val)  
    {  
        *gpio_dat |= (1<<7);  
    }  
    else  
    {  
        *gpio_dat &= ~(1<<7);  
    }  
    printk(" write write write  \n");  
    return 1;   
}  
  
static int led_release (struct inode *node, struct file *filp)  
{  
    //释放信号量  
    atomic_set(&atomic_v,1);   
    printk("iounmap(0x%x)\n", gpio_con);  
    iounmap(gpio_con);  
    printk(" release release release  \n");  
    return 0;  
}  
  
  
static struct file_operations myled_oprs = {  
    .owner = THIS_MODULE,  
    .open  = led_open,  
    .write = led_write,  
    .release = led_release,  
};  
static int myled_init(void)  
{  
   major = register_chrdev(0, "myled", &myled_oprs);  
   led_class = class_create(THIS_MODULE, "myled");  
   device_create(led_class, NULL, MKDEV(major, 0), NULL, "ledzzzzzzzz");   
   gpio_con = (volatile unsigned long *)ioremap(0x01C20824, 1);   //0x01C20824  
   gpio_dat = gpio_con + 4;     //0x01C20834          
   *gpio_con &= ~(7<<28);  
   *gpio_con |=  (1<<28);  
   *gpio_dat &= ~(1<<7);  
   return 0;  
}  
module_init(myled_init);  
module_exit(led_release);  
MODULE_LICENSE("GPL");

APP代码:

#include <sys/stat.h>  
#include <fcntl.h>  
#include <stdio.h>  
/* ledtest on 
 *  *   * ledtest off 
 *   *     */  
int main(int argc, char **argv)  
{  
    int fd;  
    unsigned char val = 1;  
  
        fd = open("/dev/ledzzzzzzzz", O_RDWR);  
        if (fd < 0)  
        {  
            printf("can't open!\n");  
        }  
        if (argc != 2)  
        {  
            printf("Usage :\n");  
            printf("%s <on|off>\n", argv[0]);  
            return 0;  
        }  
  
        if (strcmp(argv[1], "on") == 0)  
        {  
            val  = 1;  
        }  
        else  
        {  
            val = 0;  
        }  
        write(fd, &val, 1);  
  
    return 0;  
}  
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容