久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

專注電子技術學習與研究
當前位置:單片機教程網 >> MCU設計實例 >> 瀏覽文章

Linux-2.6.38驅動的幾個結構體關系總結

作者:公平   來源:本站原創   點擊數:  更新時間:2014年03月14日   【字體:

Linux驅動程序剛接觸,雖然不是很清楚,但是總歸是慢慢學習的過程。我的環境是Fedora14虛擬機。內核版本是2.6.38.1,其中的實現過程存在很多的問題,主要是因為很多的內核函數發生了較大的差別.其中最大的可能是ioctl以及互信息量的實現。這兩個的問題也使得我們在驅動設計過程中出現很多的疑惑和問題。
 
接上一部分,繼續總結:
主要包括幾個重要的結構體、并發控制、以及ioctl的實現。在驅動的設計過程主要涉及3個重要的結構體。struct file_operations,struct inode,struct file.
struct file_operations主要是涉及一些文件操作的函數,其本質上就是一個函數指針的集合,包含了文件操作的各種函數聲明,可能與應用程序設計中的相應函數只在參數上存在一定的差別。但是在2.6.36版本以后,其中的內容發生了較大的變化,主要設計了ioctl的相關操作。

    struct file_operations {
        struct module *owner;
        loff_t (*llseek) (struct file *, loff_t, int);
        ssize_t (*read) (struct file *, char __user *, size_t, loff_t *);
        ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *);
        ssize_t (*aio_read) (struct kiocb *, const struct iovec *, unsigned long, loff_t);
        ssize_t (*aio_write) (struct kiocb *, const struct iovec *, unsigned long, loff_t);
        int (*readdir) (struct file *, void *, filldir_t);
        unsigned int (*poll) (struct file *, struct poll_table_struct *);
        /*新添加的函數,同時去掉了ioctl的函數,同時返回值也發生了變化*/
        long (*unlocked_ioctl) (struct file *, unsigned int, unsigned long);
        long (*compat_ioctl) (struct file *, unsigned int, unsigned long);
        int (*mmap) (struct file *, struct vm_area_struct *);
        int (*open) (struct inode *, struct file *);
        int (*flush) (struct file *, fl_owner_t id);
        int (*release) (struct inode *, struct file *);
        int (*fsync) (struct file *, int datasync);
        int (*aio_fsync) (struct kiocb *, int datasync);
        int (*fasync) (int, struct file *, int);
        int (*lock) (struct file *, int, struct file_lock *);
        ssize_t (*sendpage) (struct file *, struct page *, int, size_t, loff_t *, int);
        unsigned long (*get_unmapped_area)(struct file *, unsigned long, unsigned long, unsigned long, unsigned long);
        int (*check_flags)(int);
        int (*flock) (struct file *, int, struct file_lock *);
        ssize_t (*splice_write)(struct pipe_inode_info *, struct file *, loff_t *, size_t, unsigned int);
        ssize_t (*splice_read)(struct file *, loff_t *, struct pipe_inode_info *, size_t, unsigned int);
        int (*setlease)(struct file *, long, struct file_lock **);
        long (*fallocate)(struct file *file, int mode, loff_t offset,
                 loff_t len);
    };

long (*unlocked_ioctl) (struct file *, unsigned int, unsigned long);
long (*compat_ioctl) (struct file *, unsigned int, unsigned long);
是最近添加進來的函數,為了實現原來的ioctl函數,同時參數以及返回值都發生了較大的變化,這也是為什么在2.6.36版本以后的內核中使用ioctl函數會報錯的原因。unlocked_ioctl函數通常用來實現原來的ioctl函數,而compat_ioctl函數則用來實現一些兼容版本的ioctl問題。返回值由原來的int變為long型,也是需要注意的。在ioctl中,第一個參數是struct inode,2.6.36以后的版本將不能直接訪問到inode參數,只能間接的訪問,具體的訪問方法后面在總結。
在驅動實現過程中主要包括對各個需要實現函數的賦值,但是open函數不能賦值與否,都會默認打開,如果不賦值,則默認該設備一直打開。其他的函數不賦值,即表示不實現該方法。常用的復制方法如下:

    /*添加該模塊的基本文件操作支持*/
    static const struct file_operations mem_fops =
    {
            /*結尾不是分號,注意其中的差別*/
            .owner = THIS_MODULE,
            .llseek = mem_llseek,
            .read = mem_read,
            .write = mem_write,
            .open = mem_open,
            .release = mem_release,
            /*添加新的操作支持*/
            .unlocked_ioctl = mem_ioctl,
    };

需要注意的是后面不再是分號,而是逗號。其中的mem_read、mem_write等是函數的具體實現過程。.owner表示該結構體屬于那個,當然就是THIS_MODULE,表示這個模塊。
 
struct inode表示的是一個文件的索引,該結構是每一個具體的物理文件(保存在存儲器中的實體文件)的索引,一個文件對應一個唯一的struct inode,其中表明了文件的大小,文件的類型,文件的時間等參數,結構體中每一個參數都能表示某一個文件的特性,通過inode就能表示文件的所有信息。
 

    struct inode {
        /* RCU path lookup touches following: */
        umode_t            i_mode;
        /*使用者的id*/
        uid_t            i_uid;
        /*使用者的組的id*/       
        gid_t            i_gid;

        const struct inode_operations    *i_op;
        struct super_block    *i_sb;

        spinlock_t        i_lock;    /* i_blocks, i_bytes, maybe i_size */
        unsigned int        i_flags;
        struct mutex        i_mutex;
        /*狀態標志*/
        unsigned long        i_state;
        unsigned long        dirtied_when;    /* jiffies of first dirtying */

        struct hlist_node    i_hash;
        struct list_head    i_wb_list;    /* backing dev IO list */
        struct list_head    i_lru;        /* inode LRU list */
        struct list_head    i_sb_list;
      
        union {
            struct list_head    i_dentry;
            struct rcu_head        i_rcu;
        };

        unsigned long        i_ino;
      
        /*引用次數,當這個數為0時,release函數才能完成*/
        atomic_t        i_count;

        unsigned int        i_nlink;
        /*設備文件的設備號*/
        dev_t            i_rdev;

        unsigned int        i_blkbits;
        u64            i_version;
        /*文件偏移量*/
        loff_t            i_size;

    #ifdef __NEED_I_SIZE_ORDERED
        seqcount_t        i_size_seqcount;
    #endif
        /*文件的時間參數,包括三種時間*/
        struct timespec        i_atime;
        struct timespec        i_mtime;
        struct timespec        i_ctime;
        blkcnt_t        i_blocks;
        unsigned short i_bytes;
        struct rw_semaphore    i_alloc_sem;
        const struct file_operations    *i_fop;    /* former ->i_op->default_file_ops */
        struct file_lock    *i_flock;
      
        /*文件的備份地址空間*/
        struct address_space    *i_mapping;
        /*設備地址空間*/
        struct address_space    i_data;

    #ifdef CONFIG_QUOTA
        struct dquot        *i_dquot[MAXQUOTAS];
    #endif
        struct list_head    i_devices;
      
        /*說明了三種不同的驅動類型*/
        union {
            struct pipe_inode_info    *i_pipe;
            struct block_device    *i_bdev;
            struct cdev        *i_cdev;
        };

        __u32            i_generation;

    #ifdef CONFIG_FSNOTIFY
        __u32            i_fsnotify_mask; /* all events this inode cares about */
        struct hlist_head    i_fsnotify_marks;
    #endif

    #ifdef CONFIG_IMA
        /* protected by i_lock */
        unsigned int        i_readcount; /* struct files open RO */
    #endif
       /*寫者使用次數*/
        atomic_t        i_writecount;
    #ifdef CONFIG_SECURITY
        void            *i_security;
    #endif
    #ifdef CONFIG_FS_POSIX_ACL
        struct posix_acl    *i_acl;
        struct posix_acl    *i_default_acl;
    #endif
        void            *i_private; /* fs or device private pointer */
    };

驅動程序設計過程中通常采用i_rdev判斷設備文件的設備號。
 
struct file是指文件對象,表示進程中打開的文件,一個物理文件只有一個inode,但是可以被打開很多次,因此可以存在很多struct file結構體。

    struct file {
        /*
         * fu_list becomes invalid after file_free is called and queued via
         * fu_rcuhead for RCU freeing
         */
        union {
            struct list_head    fu_list;
            struct rcu_head     fu_rcuhead;
        } f_u;
      
        /*文件的路徑*/
        struct path        f_path;
    #define f_dentry    f_path.dentry
    #define f_vfsmnt    f_path.mnt
        /*該文件支持的操作集合*/
        const struct file_operations    *f_op;
        spinlock_t        f_lock; /* f_ep_links, f_flags, no IRQ */
    #ifdef CONFIG_SMP
        int            f_sb_list_cpu;
    #endif
        /*文件對象的使用次數*/
        atomic_long_t        f_count;

        unsigned int         f_flags;
        fmode_t            f_mode;
        loff_t            f_pos;
        struct fown_struct    f_owner;
        const struct cred    *f_cred;
        struct file_ra_state    f_ra;

        u64            f_version;
    #ifdef CONFIG_SECURITY
        void            *f_security;
    #endif

        /* needed for tty driver, and maybe others */
        /*通常用來指向具體的數據或者設備文件,實現操作*/
        void            *private_data;

    #ifdef CONFIG_EPOLL
        /* Used by fs/eventpoll.c to link all the hooks to this file */
        struct list_head    f_ep_links;
    #endif /* #ifdef CONFIG_EPOLL */
        struct address_space    *f_mapping;
    #ifdef CONFIG_DEBUG_WRITECOUNT
        unsigned long f_mnt_write_state;
    #endif
    };

     訪問inode主要是通過這兩個機構體之間的管理型。
    struct path {
     struct vfsmount *mnt;
     struct dentry *dentry;
    };
    struct dentry {
        /* RCU lookup touched fields */
        unsigned int d_flags;        /* protected by d_lock */
        seqcount_t d_seq;        /* per dentry seqlock */
        struct hlist_bl_node d_hash;    /* lookup hash list */
        struct dentry *d_parent;    /* parent directory */
        struct qstr d_name;
        struct inode *d_inode;        /* Where the name belongs to - NULL is
                         * negative */
        unsigned char d_iname[DNAME_INLINE_LEN];    /* small names */

        /* Ref lookup also touches following */
        unsigned int d_count;        /* protected by d_lock */
        spinlock_t d_lock;        /* per dentry lock */
        const struct dentry_operations *d_op;
        struct super_block *d_sb;    /* The root of the dentry tree */
        unsigned long d_time;        /* used by d_revalidate */
        void *d_fsdata;            /* fs-specific data */

        struct list_head d_lru;        /* LRU list */
        /*
         * d_child and d_rcu can share memory
         */
        union {
            struct list_head d_child;    /* child of parent list */
             struct rcu_head d_rcu;
        } d_u;
        struct list_head d_subdirs;    /* our children */
        struct list_head d_alias;    /* inode alias list */
    };

其中的可以通過struct file間接的訪問物理文件的struct inode,具體的實現是filp->f_path.entry->d_inode,這個過程也就實現了將inode和file結構體之間的聯系。
上面的幾個主要的結構體是驅動實現過程中最重要的幾個。具體的意義還要聯系起來分析。

關閉窗口

相關文章

主站蜘蛛池模板: 五月婷婷激情 | 农村妇女毛片精品久久久 | 久久成人国产精品 | 乱一性一乱一交一视频a∨ 色爱av | 成人国产一区二区三区精品麻豆 | 97人澡人人添人人爽欧美 | 亚洲欧美激情四射 | 91精品久久久久久久久中文字幕 | 国产一区2区 | 永久免费在线观看 | 一本一道久久a久久精品综合 | 国产精品久久久久久久久久久久冷 | 天天操夜夜看 | 国产精品视频免费观看 | 国产高清精品一区二区三区 | 精品欧美一区免费观看α√ | 欧美激情精品久久久久 | 午夜影院在线观看 | 狠狠操狠狠干 | 亚洲视频国产视频 | 一级毛片,一级毛片 | 亚洲欧美日韩在线一区二区 | 久久综合欧美 | 好姑娘影视在线观看高清 | 久久国内精品 | 欧美一区二区三区视频 | 美日韩精品| 国产精品三级 | 国产亚洲欧美另类一区二区三区 | 亚洲精品国产成人 | com.国产| 精品一二三区在线观看 | 欧美精品乱码久久久久久按摩 | 欧美在线看片 | 精品国产乱码久久久久久蜜柚 | 成人免费福利 | 中文字幕久久精品 | 国产午夜精品一区二区三区四区 | 日韩欧美理论片 | 国产高清在线精品一区二区三区 | 91激情视频 |