色yeye在线视频观看_亚洲人亚洲精品成人网站_一级毛片免费播放_91精品一区二区中文字幕_一区二区三区日本视频_成人性生交大免费看


嵌入式linux下LED驅動教程

分享到:
           

    在嵌入式系統的設計中,LED一般直接由CPU的GPIO(通用可編程I/O口)控制。GPIO一般由兩組寄存器控制,即一組控制寄存器和一組數據寄存器。控制寄存器可設置GPIO口的工作方式為輸入或是輸出。當引腳被設置為輸出時,向數據寄存器的對應位寫入1和0會分別在引腳上產生高電平和低電平;當引腳設置為輸入時,讀取數據寄存器的對應位可獲得引腳上的電平為高或低。

    LED設備介于字符設備和塊設備之間我們稱之為misc(雜設備),其實按linux下的驅動位置來講還是一種字符設備。代碼清單1.1給出了Linux下LED的驅動。

代碼清單1.1 Linux操作系統下LED的驅動

    1 #include .../*包含內核中的多個頭文件*/

    2 /*設備結構體*/
    3 struct light_dev {
    4     struct cdev cdev; /*字符設備cdev結構體*/
    5     unsigned char vaule; /*LED亮時為1,熄滅時為0,用戶可讀寫此值*/
    6 };

    7 struct light_dev *light_devp;
    8 int light_major = LIGHT_MAJOR;

    9 MODULE_AUTHOR("Barry Song <21cnbao@gmail.com>");
    10 MODULE_LICENSE("Dual BSD/GPL");
    11 /*打開和關閉函數*/
    12 int light_open(struct inode *inode, struct file *filp)
    13 {
    14     struct light_dev *dev;
    15     /* 獲得設備結構體指針 */
    16     dev = container_of(inode->i_cdev, struct light_dev, cdev);
    17     /* 讓設備結構體作為設備的私有信息 */
    18     filp->private_data = dev;
    19     return 0;
    20 }

    21 int light_release(struct inode *inode, struct file *filp)
    22 {
    23     return 0;
    24 }

    25 /*讀寫設備:可以不需要 */
    26 ssize_t light_read(struct file *filp, char __user *buf, size_t count,
    27     loff_t *f_pos)
    28 {
    29     struct light_dev *dev = filp->private_data; /*獲得設備結構體 */
    30     if (copy_to_user(buf, &(dev->value), 1))
    31         return -EFAULT;

    32     return 1;
    33 }

    34 ssize_t light_write(struct file *filp, const char __user *buf, size_t count,
    35 loff_t *f_pos)
    36 {
    37     struct light_dev *dev = filp->private_data;

    38     if (copy_from_user(&(dev->value), buf, 1))
    39         return -EFAULT;

    40     /*根據寫入的值點亮和熄滅LED*/
    41     if (dev->value == 1)
    42         light_on();
    43     else
    44         light_off();

    45     return 1;
    46 }

    47 /* ioctl函數 */
    48 int light_ioctl(struct inode *inode, struct file *filp, unsigned int cmd,
    49     unsigned long arg)
    50 {
    51     struct light_dev *dev = filp->private_data;

    52     switch (cmd) {
    53     case LIGHT_ON:
    54         dev->value = 1;
    55         light_on();
    56         break;
    57     case LIGHT_OFF:
    58         dev->value = 0;
    59         light_off();
    60         break;
    61     default:
    62         /* 不能支持的命令 */
    63         return -ENOTTY;
    64     }

    65     return 0;
    66 }

    67 struct file_operations light_fops = {
    68     .owner = THIS_MODULE,
    69     .read = light_read,
    70     .write = light_write,
    71     .ioctl = light_ioctl,
    72     .open = light_open,
    73     .release = light_release,
    74 };

    75 /*設置字符設備cdev結構體*/
    76 static void light_setup_cdev(struct light_dev *dev, int index)
    77 {
    78     int err, devno = MKDEV(light_major, index);
    79     cdev_init(&dev->cdev, &light_fops);
    80     dev->cdev.owner = THIS_MODULE;
    81     dev->cdev.ops = &light_fops;
    82     err = cdev_add(&dev->cdev, devno, 1);
    83     if (err)
    84         printk(KERN_NOTICE "Error %d adding LED%d", err, index);
    85 }

    86 /*模塊加載函數*/
    87 int light_init(void)
    88 {
    89     int result;
    90     dev_t dev = MKDEV(light_major, 0);
    91     /* 申請字符設備號*/
    92     if (light_major)
    93         result = register_chrdev_region(dev, 1, "LED");
    94     else {
    95         result = alloc_chrdev_region(&dev, 0, 1, "LED");
    96         light_major = MAJOR(dev);
    97     }
    98     if (result < 0)
    99         return result;

    100     /* 分配設備結構體的內存 */
    101     light_devp = kmalloc(sizeof(struct light_dev), GFP_KERNEL);
    102     if (!light_devp) {
    103         result = -ENOMEM;
    104         goto fail_malloc;
    105     }
    106     memset(light_devp, 0, sizeof(struct light_dev));
    107     light_setup_cdev(light_devp, 0);
    108     light_gpio_init();
    109     return 0;

    110 fail_malloc:
    111     unregister_chrdev_region(dev, light_devp);
    112     return result;
    113 }

    114 /*模塊卸載函數*/
    115 void light_cleanup(void)
    116 {
    117     cdev_del(&light_devp->cdev); /*刪除字符設備結構體*/
    118     kfree(light_devp); /*釋放在light_init中分配的內存*/
    119     unregister_chrdev_region(MKDEV(light_major, 0), 1); /*刪除字符設備*/
    120 }

    121 module_init(light_init);
    122 module_exit(light_cleanup);

   熱點鏈接:

   1、嵌入式linux驅動要學哪些
   2、典型嵌入式Linux系統設置

更多新聞>> 

主站蜘蛛池模板: 免费无码黄动漫在线观看 | 三个男吃我奶头一边一个视频 | 亚洲国产精品久久久天堂不卡海量 | 中文字幕在线无码手机一区 | 一本色综合亚洲精品 | 客厅享受丝袜人妻张雅婷 | 49vv婷婷网| 黄色免费看片网站 | 极品美女扒开粉嫩小泬 | 亚洲成a人片在线不卡一二三区 | 久久久久无码专区亚洲AV | 国内自拍小视频 | 97SE亚洲国产综合自在线 | 亚洲AV日韩AV永久无码PPT | 一面亲着一面膜下奶韩剧放下 | 正在播放av | 欧美日本一二三区 | 亚洲精品久久久久私欲 | 亚洲av不卡一区二区三区 | 韩国美女一级片 | 免费人成在线观看成人片 | 婷婷精品国产亚洲AV麻豆不片 | 少妇被粗大的猛烈进出va视频 | 人妻VA精品VA欧美VA | 免费大黄网站 | 日韩久久无码免费毛片软件 | 亚洲国产精品无码7777一线 | 99久久无码一区人妻A片红豆 | 在线观看片免费人成视频无码 | 亚洲色偷偷av| 精品国模一区二区三区 | 国产精品成人AV在线观看春天 | 性色aⅤ极品无码专区亚洲 国产精品久久久久久久久无码日本蜜乳 | 精品国产人妻一区二区三区 | 人妻夜夜爽天天爽一区 | 成年人免费黄色片 | 老师课后辅导乳揉搓H在线观看 | 亚洲熟女乱综合一区二区 | 国产精品人妻99一区二区三区 | 少妇中文字幕乱码亚洲影视 | 中文字幕精品亚洲无线码二区 |