亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

全志A33 linux led驅動編程(附實測參考代碼)

發布時間:2020-03-03 02:52:36 來源:網絡 閱讀:910 作者:sinlinx123 欄目:系統運維

開發平臺

全志A33 linux led驅動編程(附實測參考代碼)

開發平臺

*  芯靈思SinlinxA33開發板

淘寶店鋪: https://sinlinx.taobao.com/

全志A33 linux led驅動編程(附實測參考代碼)

嵌入式linux 開發板交流 QQ:641395230

#實驗原理
全志A33 linux led驅動編程(附實測參考代碼)

在芯靈思開發板上,沒有led燈模塊,只能通過引腳電平觀察: 這里我選擇LS-INT引腳。

全志A33一共有10組IO口,每組IO有9個相關功能控制器,LS-INT屬于PB7,相關寄存器如圖

本次實驗只用到這兩個寄存器,在程序中命名為gpio_con,gpio_dat ,設置為輸出引腳。

  • 1)注冊 class_register(class) 將class注冊到內核中。調用前,必須手動分配class內存;調用后必須設置class的name等參數注冊 class_create(owner,name) 創建class并將class注冊到內核中。返回值為class結構體指針。注銷 void class_unregister(struct class cls) 注銷class,與class_register()配對使用。注銷void class_destroy(struct class cls) 注銷class,與class_create()配對使用內核中定義了struct class結構體顧名思義,一個struct class結構體類型變量對應一個類,內核同時提供了class_create(…)函數,可以用它來創建一個類,這個類存放于sysfs下面,一旦創建好了這個類,再調用device_create(…)函數來在/dev目錄下創建相應的設備節點。這樣,加載模塊的時候,用戶空間中的udev會自動響應device_create(…)函數,去/sysfs下尋找對應的類從而創建設備節點.
  • 2)void ioremap(unsigned long phys_addr , unsigned long size , unsigned long flags)用mmap映射一個設備意味著使用戶空間的一段地址關聯到設備內存上,這使得只要程序在分配的地址范圍內進行讀取或寫入實際上就是對設備的訪問。解除映射void iounmap(void addr)//取消ioremap所映射的IO地址
  • 3)register_chrdev(unsigned int major, const char name,const struct file_operations fops);
    但其實這個函數是linux版本2.4之前的注冊方式,它的原理是:
    • (1)確定一個主設備號,如果major=0,則會自動分配設備號
    • (2)構造一個file_operations結構體, 然后放在chrdevs數組中
    • (3)注冊:register_chrdev,cat /proc/devices查看內核中已經注冊過的字符設備驅動(和塊設備驅動),注意這里并不是驅動文件設備節點!
  • 4) Linux使用file_operations結構訪問驅動程序的函數,這個結構的每一個成員的名字都對應著一個調用
  • 5) class_device_create() 調用class_create為該設備創建一個class,再為每個設備調用 class_device_create創建對應的設備。大致用法如下:struct class *myclass = class_create(THIS_MODULE, “my_device_driver”);class_device_create(myclass, NULL, MKDEV(major_num, 0), NULL, “my_device”);這樣的module被加載時,udev daemon就會自動在/dev下創建my_device設備件。

總體代碼框架

1)先要有file_operations先要有引腳初始化函數myled_init(void),在myled_init里面注冊class并將class類注冊到內核中
創建設備節點,初始化引腳已經將寄存器地址映射到虛擬內存中,最后調用module_init(myled_init)驅動的加載就靠它
2)創建這個file_operations結構體
    static struct file_operations myled_oprs = {
    .owner = THIS_MODULE,
    .open  = led_open,
    .write = led_write,
    .release = led_release,
    }; 
下面就圍繞這個結構體寫函數led_write() led_open() led_release()
3)最后要注銷設備

附實測代碼,參考下

LED驅動代碼:

#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 <linux/of.h>
#include <linux/of_device.h>
#include <linux/of_platform.h>
static int major;
static struct class *led_class;
volatile unsigned long *gpio_con = NULL;
volatile unsigned long *gpio_dat = NULL;
static int led_open (struct inode *node, struct file *filp)
{
    /* PB7 - 0x01C20824 */
   if (gpio_con) {
        printk("ioremap  0x%x\n", gpio_con);
        }
        else {
            return -EINVAL;
        }
    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);
        }

        return 1; 
}

static int led_release (struct inode *node, struct file *filp)
{
    printk("iounmap(0x%x)\n", gpio_con);
    iounmap(gpio_con);
    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;
}

APP代碼:

#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 <linux/of.h>
#include <linux/of_device.h>
#include <linux/of_platform.h>
static int major;
static struct class *led_class;
volatile unsigned long *gpio_con = NULL;
volatile unsigned long *gpio_dat = NULL;
static int led_open (struct inode *node, struct file *filp)
{
    /* PB7 - 0x01C20824 */
   if (gpio_con) {
             printk("ioremap  0x%x\n", gpio_con);
        }
        else {
             return -EINVAL;
        }
    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);
        }

        return 1; 
}

static int led_release (struct inode *node, struct file *filp)
{
    printk("iounmap(0x%x)\n", gpio_con);
    iounmap(gpio_con);
    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;
}

Makefile代碼:


#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;
}
向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

海宁市| 辽中县| 于田县| 利川市| 恩平市| 临潭县| 林周县| 绥江县| 阳新县| 武穴市| 长兴县| 黄陵县| 凤凰县| 印江| 电白县| 广宗县| 内黄县| 积石山| 正阳县| 元氏县| 仁化县| 五华县| 久治县| 扎鲁特旗| 平果县| 保靖县| 天峨县| 双桥区| 广元市| 连州市| 绥宁县| 绵竹市| 浏阳市| 泸州市| 永嘉县| 诸暨市| 民和| 明水县| 琼中| 南丹县| 东丰县|