訂閱
糾錯
加入自媒體

函數(shù)指針編譯時出錯怎么辦?

2021-06-16 15:41
一口Linux
關注

2)  分步解析

有的C語言基礎不是很好的朋友,可能無法一眼看出來這個定義,為了讓新手更容易看懂,我們來看一下下面一個遞進式的定義:

int fun;

這是一個整型變量fun;

int fun();

這是一個函數(shù)fun,參數(shù)    :空返回值:int型

int fun(struct touch_message *);

這是一個函數(shù)fun,參數(shù)    :struct touch_message *的一個指針返回值:int型

上述的變化都好理解,下面我們將fun做如下修改:

int (*fun)(struct touch_message *);

括號的優(yōu)先級最高,(fun)一旦如此定義,那么fun就要先和結合,所以fun變成了一個指針,

那么該指針指向什么呢?就需要看外面是如何定義的,右邊是(struct touch_message * ),左邊是int,所以說明指針指向的是一個函數(shù),

參數(shù)    :struct touch_message *的一個指針返回值:int型

舉例:將函數(shù)my_fun賦值給函數(shù)指針fun。int my_fun(struct touch_message *){}int (*fun)(struct touch_message *);fun = my_fun;

這里有一個隱藏的知識點,函數(shù)名其實也是一個地址,而且賦值的時候函數(shù)類型必須和函數(shù)指針類型一致。

typedef int (*fun)(struct touch_message *);

如果左邊再加上typedef呢?相當于是設置fun為新的類型,我們可以用fun來定義一個函數(shù)指針,該函數(shù)類型同上。

舉例用新的類型定義一個函數(shù)指針變量,并給他賦值。typedef int (*fun)(struct touch_message *);int my_fun(struct touch_message *){}fun fun_ptr;fun_ptr = my_fun;

然后將參數(shù)修改為,touch_message_t,就得到了粉絲的源碼中的樣子,

typedef int (*fun)(touch_message_t);

但是粉絲的源碼中定義的函數(shù)類型缺少了對函數(shù)返回值的描述,所以左側增加一個int或者其他類型即可即可。

3. 函數(shù)指針

函數(shù)指針在linux內核中使用非常頻繁,

比如字符設備,內核給多有的字符設備提供了一個統(tǒng)一的接口,我們對設備的所有操作被抽象成read、write、open、close等,并封裝到結構體struct file_operations 中:

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 (*iterate) (struct file *, struct dir_context *);
unsigned int (*poll) (struct file *, struct poll_table_struct *);
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 *, loff_t, loff_t, 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);
int (*show_fdinfo)(struct seq_file *m, struct file *f);
};

那么我們應該如何定義該結構體變量并初始化呢?

static struct file_operations hello_ops =

.open = hello_open,
.release = hello_release,
.read = hello_read,
.write = hello_write,
};

函數(shù)定義如下:

static int hello_open (struct inode *inode, struct file *filep)

return 0;

static int hello_release (struct inode *inode, struct file *filep)

return 0;

static ssize_t hello_read (struct file *filep, char __user *buf, size_t size, loff_t *pos)

return size;

static ssize_t hello_write (struct file *filep, const char __user *buf, size_t size, loff_t *pos)

return size;

注意,函數(shù)的參數(shù)和返回值,必須嚴格按照結構體struct file_operations中的類型定義。


<上一頁  1  2  
聲明: 本文由入駐維科號的作者撰寫,觀點僅代表作者本人,不代表OFweek立場。如有侵權或其他問題,請聯(lián)系舉報。

發(fā)表評論

0條評論,0人參與

請輸入評論內容...

請輸入評論/評論長度6~500個字

您提交的評論過于頻繁,請輸入驗證碼繼續(xù)

暫無評論

暫無評論

    掃碼關注公眾號
    OFweek人工智能網(wǎng)
    獲取更多精彩內容
    文章糾錯
    x
    *文字標題:
    *糾錯內容:
    聯(lián)系郵箱:
    *驗 證 碼:

    粵公網(wǎng)安備 44030502002758號