日本黄色一级经典视频|伊人久久精品视频|亚洲黄色色周成人视频九九九|av免费网址黄色小短片|黄色Av无码亚洲成年人|亚洲1区2区3区无码|真人黄片免费观看|无码一级小说欧美日免费三级|日韩中文字幕91在线看|精品久久久无码中文字幕边打电话

當前位置:首頁 > > 充電吧
[導讀]一般驅動都分為兩個部分: 一:寫出框架 二:硬件操作 框架搭建: ①頭文件的拷貝 #include #include #include #include #i

一般驅動都分為兩個部分: 一:寫出框架 二:硬件操作 框架搭建: ①頭文件的拷貝
#include   
#include   
#include   
#include   
#include   
#include   
#include   
#include   
#include   
#include 
②將open、read驅動函數框架寫出
static int second_drv_open(struct inode *inode,struct file *file)
{
    return 0;
}
③定義fileoperation
static struct file_operations second_drv_fops = {
    .owner = THIS_MODULE,   /* 這是一個宏,推向編譯模塊時自動創(chuàng)建的__this_module變量 */
    .open  = second_drv_open,
    .read = second_drv_read,
};
④ 入口函數編寫(注冊驅動程序)
int major; 

static int second_drv_init(void)
{
        major = register_chrdev(0,"second_drv",&second_drv_fops);

        return 0;
}
出口函數編寫(卸載驅動程序)
static int second_drv_exit(void)
{
        unregister_chrdev(major,"second_drv");
        return 0;
}

入口函數和出口函數也只是普通的函數,何以達到入口出口的功能!需要通過module_來修飾。

module_init(second_drv_init);
module_exit(second_drv_exit);
⑤給sysfs提供更多信息,利用udev機制創(chuàng)建設備節(jié)點 Ⅰ定義兩個結構體
static struct class *seconddrv_class;
static struct class_device *seconddrv_class_dev;
Ⅱ 入口函數中創(chuàng)建一個類=>類下面創(chuàng)建一個設備
int major;
static int second_drv_init(void)
{
        major = register_chrdev(0,"second_drv",&second_drv_fops);
        seconddrv_class = class_create(THIS_MODULE,"seconddrv");
        seconddrv_class_dev = class_device_create(seconddrv_class,NULL,MKDEV(major,0),NULL,"buttons");
}

補: mdev是udev的簡化版本,mdev應用程序會被內核來調用,根據⑤提供的信息來創(chuàng)建/dev/buttons 設備節(jié)點。

Ⅲ 出口函數中卸載類
static int second_drv_exit(void)
{
        unregister_chrdev(major,"second_drv");
        class_device_unregister(seconddrv_class_dev);
        class_destroy(seconddrv_class);
        return 0;
}
MODULE_LICENSE("GPL");

通過 “GPL” 指明 這是GNU General Public License的任意版本

測試:

修改Makefile
cp second_drv.ko /work/nfs_root/czg
insmod ./second_drv.ko
lsmod
cat /proc/devices
ls /dev/buttons -l

硬件操作 ① 看原理圖-確定哪些引腳 ② 看2440手冊-確定引腳狀態(tài) ③ 編程 Ⅰ單片機:物理地址 Ⅱ 驅動:虛擬地址

VA = ioremap(PA,size)

原理圖:

2440手冊:


編程: ① 定義寄存器
volatile unsigned long *gpfcon;
volatile unsigned long *gpfdat;
volatile unsigned long *gpgcon;
volatile unsigned long *gpgdat;
② 地址映射建立和解除 入口函數中建立:
static int second_drv_init(void)
{
        major = register_chrdev(0,"second_drv",&second_drv_fops);
        seconddrv_class = class_create(THIS_MODULE,"seconddrv");
        seconddrv_class_dev = class_device_create(seconddrv_class,NULL,MKDEV(major,0),NULL,"buttons");
        gpfcon = (volatile unsigned long *)ioremap(0x56000050,16);
        gpfdat = gpfcon + 1;
        gpgcon = (volatile unsigned long *)ioremap(0x56000060,16);
        gpgdat = gpgcon + 1;
        return 0;
}
出口函數中解除:
static int second_drv_exit(void)
{
        unregister_chrdev(major,"second_drv");
        class_device_unregister(seconddrv_class_dev);
        class_destroy(seconddrv_class);
        iounmap(gpfcon);
        iounmap(gpgcon);
        return 0;
}
③ open函數中配置引腳
static int second_drv_open(struct inode *inode,struct file *file)
{
    /* 配置GPF0,2為輸入引腳 */
    *gpfcon &= ~((0x3<<(0*2)) | (0x3<<(2*2)));

    /* 配置GPG3為輸入引腳 */
    *gpgcon &= ~((0x3<<(3*2)));

    return 0;
}
④ read函數中返回引腳狀態(tài)
static ssize_t second_drv_read (struct file *file, char __user *buf, size_t count, loff_t *ppos)
{
    /* 返回4個引腳的電平狀態(tài) */
    unsigned char key_vals[3];
    int regval;
    //看用戶需要讀取的空間,和這里的是否相同
    if(count != sizeof(key_vals))
        return -EINVAL;
    /* 讀GPF0,2 */
    regval = *gpfdat;
    key_vals[0] = (regval & (1<<0)) ? 1 : 0;
    key_vals[1] = (regval & (1<<2)) ? 1 : 0;
    /* 讀GPF3 */
    regval = *gpgdat;
    key_vals[2] = (regval & (1<<3)) ? 1 : 0;

    //將值返回給用戶程序
    copy_to_user(buf,key_vals,sizeof(key_vals));
    return sizeof(key_vals);
}
完整代碼:

Makefile

KERN_DIR = /work/system/linux-2.6.22.6

all:
    make -C $(KERN_DIR) M=`pwd` modules 

clean:
    make -C $(KERN_DIR) M=`pwd` modules clean
    rm -rf modules.order

obj-m += second_drv.o

驅動程序:second_drv.c

#include   
#include   
#include   
#include   
#include   
#include   
#include   
#include   
#include   
#include 

static struct class *seconddrv_class;
static struct class_device *seconddrv_class_dev;

volatile unsigned long *gpfcon;
volatile unsigned long *gpfdat;
volatile unsigned long *gpgcon;
volatile unsigned long *gpgdat;

static int second_drv_open(struct inode *inode,struct file *file)
{
    /* 配置GPF0,2為輸入引腳 */
    *gpfcon &= ~((0x3<<(0*2)) | (0x3<<(2*2)));

    /* 配置GPG3為輸入引腳 */
    *gpgcon &= ~((0x3<<(3*2)));

    return 0;
}

static ssize_t second_drv_read (struct file *file, char __user *buf, size_t count, loff_t *ppos)
{
    /* 返回4個引腳的電平狀態(tài) */
    unsigned char key_vals[3];
    int regval;
    //看用戶需要讀取的空間,和這里的是否相同
    if(count != sizeof(key_vals))
        return -EINVAL;
    /* 讀GPF0,2 */
    regval = *gpfdat;
    key_vals[0] = (regval & (1<<0)) ? 1 : 0;
    key_vals[1] = (regval & (1<<2)) ? 1 : 0;
    /* 讀GPF3 */
    regval = *gpgdat;
    key_vals[2] = (regval & (1<<3)) ? 1 : 0;

    //將值返回給用戶程序
    copy_to_user(buf,key_vals,sizeof(key_vals));
    return sizeof(key_vals);
}

static struct file_operations second_drv_fops = {
    .owner = THIS_MODULE,   /* 這是一個宏,推向編譯模塊時自動創(chuàng)建的__this_module變量 */
    .open  = second_drv_open,
    .read = second_drv_read,
};

int major; 

static int second_drv_init(void)
{
        major = register_chrdev(0,"second_drv",&second_drv_fops);
        seconddrv_class = class_create(THIS_MODULE,"seconddrv");
        seconddrv_class_dev = class_device_create(seconddrv_class,NULL,MKDEV(major,0),NULL,"buttons");
        gpfcon = (volatile unsigned long *)ioremap(0x56000050,16);
        gpfdat = gpfcon + 1;
        gpgcon = (volatile unsigned long *)ioremap(0x56000060,16);
        gpgdat = gpgcon + 1;
        return 0;
}

static int second_drv_exit(void)
{
        unregister_chrdev(major,"second_drv");
        class_device_unregister(seconddrv_class_dev);
        class_destroy(seconddrv_class);
        iounmap(gpfcon);
        iounmap(gpgcon);
        return 0;
}

module_init(second_drv_init);
module_exit(second_drv_exit);
MODULE_LICENSE("GPL");

驅動測試程序:seconddrvtest.c

#include   
#include   
#include   
#include   

/* 
 * firstdrvtest
 */

int main(int argc, char **argv)
{
    int fd;
    unsigned char key_vals[3];
    int cnt = 0;
    fd = open("/dev/buttons",O_RDWR);
    if(fd < 0)
    {
        printf("can't open!n");
    }
    while(1)
    {
        read(fd,key_vals,sizeof(key_vals));
        if(!key_vals[0] || !key_vals[1] || !key_vals[2])
        {
            printf("%04d key pressed: %d %d %d.n",cnt++,key_vals[0],key_vals[1],key_vals[2]);
        }

    }

    return 0;
}
測試:

后臺運行:./seconddrvtest &
查看后臺程序: top
放在前臺:fg

insmod second_drv.ko
lsmod
cat /proc/devices
ls /dev/buttons -l
./seconddrvtest

本站聲明: 本文章由作者或相關機構授權發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內容真實性等。需要轉載請聯系該專欄作者,如若文章內容侵犯您的權益,請及時聯系本站刪除。
換一批
延伸閱讀

LED驅動電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關鍵字: 驅動電源

在工業(yè)自動化蓬勃發(fā)展的當下,工業(yè)電機作為核心動力設備,其驅動電源的性能直接關系到整個系統的穩(wěn)定性和可靠性。其中,反電動勢抑制與過流保護是驅動電源設計中至關重要的兩個環(huán)節(jié),集成化方案的設計成為提升電機驅動性能的關鍵。

關鍵字: 工業(yè)電機 驅動電源

LED 驅動電源作為 LED 照明系統的 “心臟”,其穩(wěn)定性直接決定了整個照明設備的使用壽命。然而,在實際應用中,LED 驅動電源易損壞的問題卻十分常見,不僅增加了維護成本,還影響了用戶體驗。要解決這一問題,需從設計、生...

關鍵字: 驅動電源 照明系統 散熱

根據LED驅動電源的公式,電感內電流波動大小和電感值成反比,輸出紋波和輸出電容值成反比。所以加大電感值和輸出電容值可以減小紋波。

關鍵字: LED 設計 驅動電源

電動汽車(EV)作為新能源汽車的重要代表,正逐漸成為全球汽車產業(yè)的重要發(fā)展方向。電動汽車的核心技術之一是電機驅動控制系統,而絕緣柵雙極型晶體管(IGBT)作為電機驅動系統中的關鍵元件,其性能直接影響到電動汽車的動力性能和...

關鍵字: 電動汽車 新能源 驅動電源

在現代城市建設中,街道及停車場照明作為基礎設施的重要組成部分,其質量和效率直接關系到城市的公共安全、居民生活質量和能源利用效率。隨著科技的進步,高亮度白光發(fā)光二極管(LED)因其獨特的優(yōu)勢逐漸取代傳統光源,成為大功率區(qū)域...

關鍵字: 發(fā)光二極管 驅動電源 LED

LED通用照明設計工程師會遇到許多挑戰(zhàn),如功率密度、功率因數校正(PFC)、空間受限和可靠性等。

關鍵字: LED 驅動電源 功率因數校正

在LED照明技術日益普及的今天,LED驅動電源的電磁干擾(EMI)問題成為了一個不可忽視的挑戰(zhàn)。電磁干擾不僅會影響LED燈具的正常工作,還可能對周圍電子設備造成不利影響,甚至引發(fā)系統故障。因此,采取有效的硬件措施來解決L...

關鍵字: LED照明技術 電磁干擾 驅動電源

開關電源具有效率高的特性,而且開關電源的變壓器體積比串聯穩(wěn)壓型電源的要小得多,電源電路比較整潔,整機重量也有所下降,所以,現在的LED驅動電源

關鍵字: LED 驅動電源 開關電源

LED驅動電源是把電源供應轉換為特定的電壓電流以驅動LED發(fā)光的電壓轉換器,通常情況下:LED驅動電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關鍵字: LED 隧道燈 驅動電源
關閉