0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內(nèi)不再提示

基于i.MX6ULL的新字符設備驅動模板

玩轉單片機 ? 來源:玩轉單片機 ? 2023-03-06 09:11 ? 次閱讀

這個系列好久沒寫了,本來寫了一千來字的,后來直接全刪了,看到的資料基本都是局部到整體,給人一種牽制的感覺,現(xiàn)在就換個寫法,從整體到局部的反向學習,畢竟本人也是初學者,適當根據(jù)實際切換學習方法很重要!

|照蘆花飄

對于新學的知識啥也不懂,先復制過來試著跑起來看看效果!

1、復制一份上次字符驅動的文件,然后使用VSCode打開,把原子公開資料中的.vscode文件夾復制過來!

1973d70c-bb71-11ed-bfe3-dac502259ad0.png

2、修改一下.vscode中的json文件!

19920ae2-bb71-11ed-bfe3-dac502259ad0.png

3、編寫驅動代碼和應用代碼!

chrdevbase.c文件

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


#define CHRDEVBASE_CNT      1    /* 設備號個數(shù) */
#define CHRDEVBASE_NAME   "chrdevbase"  /* 名字 */


/* chrdevbase 設備結構體 */
struct newchr_dev{
  dev_t devid;       /* 設備號 */
  struct cdev cdev;     /* cdev */
  struct class *class;   /* 類 */
  struct device *device;   /* 設備 */
  int major;         /* 主設備號 */
  int minor;         /* 次設備號 */
};


struct newchr_dev chrdevbase;/* 自定義字符設備 */


/*
 * @description    : 打開設備
 * @param - inode   : 傳遞給驅動的inode
 * @param - filp   : 設備文件,file結構體有個叫做private_data的成員變量
 *             一般在open的時候將private_data指向設備結構體。
 * @return       : 0 成功;其他 失敗
 */
static int chrdevbase_open(struct inode *inode, struct file *filp)
{
  printk("[BSP]chrdevbase open!
");
  filp->private_data = &chrdevbase; /* 設置私有數(shù)據(jù) */
  return 0;
}


/*
 * @description    : 從設備讀取數(shù)據(jù) 
 * @param - filp   : 要打開的設備文件(文件描述符)
 * @param - buf   : 返回給用戶空間的數(shù)據(jù)緩沖區(qū)
 * @param - cnt   : 要讀取的數(shù)據(jù)長度
 * @param - offt   : 相對于文件首地址的偏移
 * @return       : 讀取的字節(jié)數(shù),如果為負值,表示讀取失敗
 */
static ssize_t chrdevbase_read(struct file *filp, char __user *buf, size_t cnt, loff_t *offt)
{
  printk("chrdevbase read!
");
  return 0;
}


/*
 * @description    : 向設備寫數(shù)據(jù) 
 * @param - filp   : 設備文件,表示打開的文件描述符
 * @param - buf   : 要寫給設備寫入的數(shù)據(jù)
 * @param - cnt   : 要寫入的數(shù)據(jù)長度
 * @param - offt   : 相對于文件首地址的偏移
 * @return       : 寫入的字節(jié)數(shù),如果為負值,表示寫入失敗
 */
static ssize_t chrdevbase_write(struct file *filp, const char __user *buf, size_t cnt, loff_t *offt)
{
  printk("chrdevbase write!
");
  return 0;
}


/*
 * @description    : 關閉/釋放設備
 * @param - filp   : 要關閉的設備文件(文件描述符)
 * @return       : 0 成功;其他 失敗
 */
static int chrdevbase_release(struct inode *inode, struct file *filp)
{
  printk("[BSP]release!
");
  return 0;
}


/*
 * 設備操作函數(shù)結構體
 */
static struct file_operations chrdevbase_fops = {
  .owner = THIS_MODULE,  
  .open = chrdevbase_open,
  .read = chrdevbase_read,
  .write = chrdevbase_write,
  .release = chrdevbase_release,
};


/*
 * @description  : 驅動入口函數(shù) 
 * @param     : 無
 * @return     : 0 成功;其他 失敗
 */
static int __init chrdevbase_init(void)
{
  /* 注冊字符設備驅動 */
  /* 1、創(chuàng)建設備號 */
  if (chrdevbase.major) { /* 定義了設備號 */
    chrdevbase.devid = MKDEV(chrdevbase.major, 0);
    register_chrdev_region(chrdevbase.devid, CHRDEVBASE_CNT, CHRDEVBASE_NAME);
  } else { /* 沒有定義設備號 */
    alloc_chrdev_region(&chrdevbase.devid, 0, CHRDEVBASE_CNT,CHRDEVBASE_NAME); /* 申請設備號 */
    chrdevbase.major = MAJOR(chrdevbase.devid); /* 獲取主設備號 */
    chrdevbase.minor = MINOR(chrdevbase.devid); /* 獲取次設備號 */
  }
  printk("newcheled major=%d,minor=%d
",chrdevbase.major,chrdevbase.minor);


  /* 2、初始化 cdev */
  chrdevbase.cdev.owner = THIS_MODULE;
  cdev_init(&chrdevbase.cdev, &chrdevbase_fops);


  /* 3、添加一個 cdev */
  cdev_add(&chrdevbase.cdev, chrdevbase.devid, CHRDEVBASE_CNT);


  /* 4、創(chuàng)建類 */
  chrdevbase.class = class_create(THIS_MODULE, CHRDEVBASE_NAME);
  if (IS_ERR(chrdevbase.class)) {
    return PTR_ERR(chrdevbase.class);
  }


  /* 5、創(chuàng)建設備 */
  chrdevbase.device = device_create(chrdevbase.class, NULL,chrdevbase.devid, NULL, CHRDEVBASE_NAME);
  if (IS_ERR(chrdevbase.device)) {
    return PTR_ERR(chrdevbase.device);
  }


  return 0;
}


/*
 * @description  : 驅動出口函數(shù)
 * @param     : 無
 * @return     : 無
 */
static void __exit chrdevbase_exit(void)
{
  /* 注銷字符設備 */
  cdev_del(&chrdevbase.cdev);/* 刪除 cdev */
  unregister_chrdev_region(chrdevbase.devid, CHRDEVBASE_CNT);/* 注銷設備號 */


  device_destroy(chrdevbase.class, chrdevbase.devid);/* 銷毀設備 */
  class_destroy(chrdevbase.class);/* 銷毀類 */


  printk("[BSP]chrdevbase exit!
");
}


/* 
 * 將上面兩個函數(shù)指定為驅動的入口和出口函數(shù) 
 */
module_init(chrdevbase_init);
module_exit(chrdevbase_exit);


/* 
 * LICENSE和作者信息
 */
MODULE_LICENSE("GPL");
MODULE_AUTHOR("zuozhongkai");

chrdevbaseApp.c文件

#include "stdio.h"
#include "unistd.h"
#include "sys/types.h"
#include "sys/stat.h"
#include "fcntl.h"
#include "stdlib.h"
#include "string.h"


/*
 * @description    : main主程序
 * @param - argc   : argv數(shù)組元素個數(shù)
 * @param - argv   : 具體參數(shù)
 * @return       : 0 成功;其他 失敗
 */
int main(int argc, char *argv[])
{
  int fd, retvalue;
  char *filename;


  if(argc != 3){
    printf("[APP]Error Usage!
");
    return -1;
  }


  filename = argv[1];


  /* 打開驅動文件 */
  fd  = open(filename, O_RDWR);
  if(fd < 0){
    printf("[APP]Can't open file %s
", filename);
    return -1;
  }


  /* 關閉設備 */
  retvalue = close(fd);
  if(retvalue < 0){
    printf("[APP]Can't close file %s
", filename);
    return -1;
  }


  return 0;
}

4、編譯驅動和應用!

//編譯驅動
make
// 編譯應用
arm-linux-gnueabihf-gcc chrdevbaseApp.c -o chrdevbaseApp

19c18e34-bb71-11ed-bfe3-dac502259ad0.png

5、把編譯后的文件復制到根文件中!

19edc526-bb71-11ed-bfe3-dac502259ad0.png

6、加載驅動!

// 加載驅動
insmod chrdevbase.ko

1a20e3de-bb71-11ed-bfe3-dac502259ad0.png

7、測試驅動!

//讀
./chrdevbaseApp /dev/chrdevbase 1
// 寫
./chrdevbaseApp /dev/chrdevbase 2

1a423688-bb71-11ed-bfe3-dac502259ad0.png

| 細節(jié)剖析

新字符驅動和舊字符驅動最大的區(qū)別不知道各位有沒看出來,就是不用創(chuàng)建設備節(jié)點文件和指定設備號,通過系統(tǒng)自動分配就不容易發(fā)生沖突,所以學習新字符驅動的第一步先看創(chuàng)建設備號的方式:

1a5f462e-bb71-11ed-bfe3-dac502259ad0.png

相關函數(shù)API

// 如果沒有指定設備號的話就使用如下函數(shù)來申請設備號:
int alloc_chrdev_region(dev_t *dev, unsigned baseminor, unsigned count, const char *name)
// 如果給定了設備的主設備號和次設備號就使用如下所示函數(shù)來注冊設備號即可:
int register_chrdev_region(dev_t from, unsigned count, const char *name)
//不管是通過alloc_chrdev_region函數(shù)還是register_chrdev_region函數(shù)申請的設備號,統(tǒng)一使用如下釋放函數(shù):
void unregister_chrdev_region(dev_t from, unsigned count)
參數(shù) from 是要申請的起始設備號,也就是給定的設備號;參數(shù) count 是要申請的數(shù)量,一般都是一個;參數(shù) name 是設備名字。

第二步再看字符設備結構

1a917aae-bb71-11ed-bfe3-dac502259ad0.png

在 Linux 中使用 cdev 結構體表示一個字符設備,cdev 結構體在 include/linux/cdev.h 文件中的定義如下:

struct cdev {
  struct kobject kobj;
  struct module *owner;
  const struct file_operations *ops;
  struct list_head list;
  dev_t dev;
  unsigned int count;
};
在 cdev 中有兩個重要的成員變量:ops 和 dev,這兩個就是字符設備文件操作函數(shù)集合file_operations 以及設備號 dev_t。編寫字符設備驅動之前需要定義一個 cdev 結構體變量,這個變量就表示一個字符設備,如上圖第29行;

1abd0480-bb71-11ed-bfe3-dac502259ad0.png

定義好 cdev 變量以后就要使用 cdev_init 函數(shù)對其進行初始化,cdev_init 函數(shù)原型如下:

void cdev_init(struct cdev *cdev, const struct file_operations *fops)
參數(shù) cdev 就是要初始化的 cdev 結構體變量,參數(shù) fops 就是字符設備文件操作函數(shù)集合。

cdev_add 函數(shù)用于向 Linux 系統(tǒng)添加字符設備(cdev 結構體變量),首先使用 cdev_init 函數(shù)完成對 cdev 結構體變量的初始化,然后使用 cdev_add 函數(shù)向 Linux 系統(tǒng)添加這個字符設備。cdev_add 函數(shù)原型如下:

int cdev_add(struct cdev *p, dev_t dev, unsigned count)

參數(shù) p 指向要添加的字符設備(cdev 結構體變量),參數(shù) dev 就是設備所使用的設備號,參數(shù) count 是要添加的設備數(shù)量。

卸載驅動的時候一定要使用 cdev_del 函數(shù)從 Linux 內(nèi)核中刪除相應的字符設備,cdev_del函數(shù)原型如下:

void cdev_del(struct cdev *p)
參數(shù) p 就是要刪除的字符設備。 mdev 機制
在Linux下通過udev來實現(xiàn)設備文件的自動創(chuàng)建與刪除。使用busybox構建根文件系統(tǒng)的時候,busybox會創(chuàng)建一個udev的簡化版本mdev。


所以,在嵌入式開發(fā)中使用mdev來實現(xiàn)設備節(jié)點文件的自動創(chuàng)建與刪除。Linux系統(tǒng)中的熱插拔事件也由mdev 管理,在/etc/init.d/rcS 文件中如下語句:


echo /sbin/mdev > /proc/sys/kernel/hotplug

第三步看類

1ae27e54-bb71-11ed-bfe3-dac502259ad0.png

自動創(chuàng)建設備節(jié)點的工作是在驅動程序的入口函數(shù)中完成的,一般在 cdev_add 函數(shù)后面添加自動創(chuàng)建設備節(jié)點相關代碼。首先要創(chuàng)建一個 class 類,class 是個結構體,定義在文件include/linux/device.h 里面。class_create 是類創(chuàng)建函數(shù),class_create 是個宏定義,內(nèi)容如下:

#define class_create(owner, name) 
({ 
  static struct lock_class_key __key; 
  __class_create(owner, name, &__key); 
})


structclass*__class_create(structmodule*owner,constchar*name,structlock_class_key*key)
將宏 class_create 展開以后內(nèi)容如下:
struct class *class_create (struct module *owner, const char *name)
class_create 一共有兩個參數(shù),參數(shù) owner 一般為 THIS_MODULE,參數(shù) name 是類名字。返回值是個指向結構體 class 的指針,也就是創(chuàng)建的類。 卸載驅動程序的時候需要刪除掉類,類刪除函數(shù)為 class_destroy,函數(shù)原型如下:
void class_destroy(struct class *cls);
參數(shù) cls 就是要刪除的類。 第四步看創(chuàng)建設備

1b289d80-bb71-11ed-bfe3-dac502259ad0.png

節(jié)創(chuàng)建好類以后還不能實現(xiàn)自動創(chuàng)建設備節(jié)點,還需要在這個類下創(chuàng)建一個設備。使用 device_create 函數(shù)在類下面創(chuàng)建設備,device_create 函數(shù)原型如下:

struct device *device_create(struct class *class, 
  struct device *parent,
  dev_t devt, 
  void *drvdata, 
  const char *fmt, ...)
device_create 是個可變參數(shù)函數(shù),參數(shù) class 就是設備要創(chuàng)建哪個類下面;參數(shù) parent 是父設備,一般為 NULL,也就是沒有父設備;參數(shù) devt 是設備號;參數(shù) drvdata 是設備可能會使用的一些數(shù)據(jù),一般為 NULL;參數(shù) fmt 是設備名字,如果設置 fmt=xxx 的話,就會生成/dev/xxx這個設備文件。返回值就是創(chuàng)建好的設備。 卸載驅動的時候需要刪除掉創(chuàng)建的設備,設備刪除函數(shù)為 device_destroy,函數(shù)原型如下:
void device_destroy(struct class *class, dev_t devt)
參數(shù) class 是要刪除的設備所處的類,參數(shù) devt 是要刪除的設備號。 第五步看注銷

1b5d2a64-bb71-11ed-bfe3-dac502259ad0.png

初始化的時候創(chuàng)建了啥,注銷設備的時候就需要銷毀初始化創(chuàng)建的東西,注意注銷的順序是有要求的,不能誰便顛倒順序! | 特別說明 每個硬件設備都有一些屬性,比如主設備號(dev_t),類(class)、設備(device)、開關狀態(tài)(state)等等,在編寫驅動的時候你可以將這些屬性全部寫成變量的形式不是很合理的,這時候就可以設置文件私有數(shù)據(jù)!

1b9759b4-bb71-11ed-bfe3-dac502259ad0.png

審核編輯:湯梓紅
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權轉載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內(nèi)容侵權或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 驅動
    +關注

    關注

    12

    文章

    1840

    瀏覽量

    85291
  • 文件
    +關注

    關注

    1

    文章

    566

    瀏覽量

    24744
  • 字符
    +關注

    關注

    0

    文章

    233

    瀏覽量

    25208
  • IMX6ULL
    +關注

    關注

    3

    文章

    16

    瀏覽量

    4042
  • vscode
    +關注

    關注

    1

    文章

    155

    瀏覽量

    7712

原文標題:i.MX6ULL|新字符設備驅動模板

文章出處:【微信號:玩轉單片機,微信公眾號:玩轉單片機】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    i.MX6ULL 驅動開發(fā)7—按鍵輸入捕獲與GPIO輸入配置與高低電平讀取

    本篇主要介紹了i.MX6ULL的按鍵檢測的使用,主要的知識點是設備樹的修改,以及GPIO的輸入配置與高低電平的讀取。
    的頭像 發(fā)表于 05-24 09:11 ?6244次閱讀
    <b class='flag-5'>i.MX6ULL</b> <b class='flag-5'>驅動</b>開發(fā)7—按鍵輸入捕獲與GPIO輸入配置與高低電平讀取

    i.MX6ULL驅動開發(fā)2—新字符設備開發(fā)模板

    上篇文章介紹了字符設備的開發(fā)模板,但那是一種舊版本的驅動開發(fā)模式,設備驅動需要手動分配
    的頭像 發(fā)表于 03-17 09:11 ?3180次閱讀
    <b class='flag-5'>i.MX6ULL</b><b class='flag-5'>驅動</b>開發(fā)2—<b class='flag-5'>新字符</b><b class='flag-5'>設備</b>開發(fā)<b class='flag-5'>模板</b>

    移植NXP官方linux 5.4內(nèi)核到i.MX6ULL開發(fā)板

    本文描述移植NXP官方 linux 5.4 內(nèi)核到i.MX6ULL開發(fā)板。
    發(fā)表于 12-19 11:10 ?2048次閱讀

    i.MX6ULL開發(fā)板硬件資源

    迅為i.MX6ULL 終結者開發(fā)板硬件資源非常豐富,幾乎將 i.MX6ULL 芯片的所有資源都擴展引出到底板上了,底板提供了豐富的外設接口,開發(fā)板的尺寸是 190mm*125mm,充分考慮了人性化設計,整體顯得十分大。
    發(fā)表于 12-29 06:18

    初識 i.MX6ULL 寄存器

    裸機開發(fā)_L1_匯編LED實驗0. 本節(jié)目標1. 硬件層電路2. 初識 i.MX6ULL 寄存器2.1 i.MX6ULL 時鐘控制寄存器2.2 i.MX6ULL IO復用寄存器2.3
    發(fā)表于 12-20 07:13

    I.MX6ULL無法枚舉USB2514是為什么?

    你好目前,I.MX6ULL開發(fā)存在一些問題。其中之一是OTG USB2無法正常掛載USB2514,無法正確枚舉下游設備,只顯示設備id。usb設計要注意什么。
    發(fā)表于 04-03 06:55

    I.MX6ULL UART傳輸問題求解

    I.MX6ULL UART傳輸問題
    發(fā)表于 04-21 08:09

    珠海明遠智??萍悸?lián)合NXP強勢推出i.MX6ull核心板

    接口,用于連接外圍設備,如WLAN、Bluetooth?、GPS、顯示器和攝像頭傳感器。?為了加速基于NXP i.MX6ULLi.MX6UL芯片的產(chǎn)品設計,珠海明遠智睿科技聯(lián)合恩智浦推出了高質(zhì)量
    發(fā)表于 04-24 14:10 ?560次閱讀

    飛凌i.MX6ULL開發(fā)板的評測,再次進階擁有更高的性價比

    處理器MCIMX6Y2開發(fā)設計,采用先進的ARMCortex-A7內(nèi)核,運行速度高達800MHz。i.MX6ULL應用處理器包括一個集成的電源管理模塊,降低了外接電源的復雜性,并簡化了上電時序。 i.MX6ULL
    發(fā)表于 10-27 11:55 ?1488次閱讀
    飛凌<b class='flag-5'>i.MX6ULL</b>開發(fā)板的評測,再次進階擁有更高的性價比

    基于NXP i.MX6ULL處理器的FETMX6ULL-C核心板

    “性價比高,功能接口豐富,資料齊全,穩(wěn)定性強”這是許多用戶對飛凌FETMX6ULL-S核心板的評價。作為NXP公司一顆經(jīng)典的MPU,i.MX6ULL的市場認可度無需多言。而作為NXP公司的金牌
    發(fā)表于 04-11 15:05 ?1156次閱讀
    基于NXP <b class='flag-5'>i.MX6ULL</b>處理器的FETMX<b class='flag-5'>6ULL</b>-C核心板

    i.MX6ULL驅動開發(fā)4——點亮LED(寄存器版)

    本篇主要介紹了如何通過操作寄存器來點亮i.MX6ULL開發(fā)板上的led,通過編寫LED對應的驅動程序和應用程序,實現(xiàn)程序設計的分層。
    的頭像 發(fā)表于 05-21 21:26 ?2981次閱讀
    【<b class='flag-5'>i.MX6ULL</b>】<b class='flag-5'>驅動</b>開發(fā)4——點亮LED(寄存器版)

    i.MX6ULL|字符設備驅動流程深究

    上一篇介紹了虛擬字符設備驅動,這篇就深入學習字符驅動的流程,看看字符
    的頭像 發(fā)表于 10-31 10:14 ?734次閱讀

    【北京迅為】i.MX6ULL開發(fā)板移植 Debian 文件系統(tǒng)

    【北京迅為】i.MX6ULL開發(fā)板移植 Debian 文件系統(tǒng)
    的頭像 發(fā)表于 02-10 15:34 ?1151次閱讀
    【北京迅為】<b class='flag-5'>i.MX6ULL</b>開發(fā)板移植 Debian 文件系統(tǒng)

    基于i.MX6ULL的掉電檢測設計與軟件測試

    基于i.MX6ULL的掉電檢測設計與軟件測試基于i.MX6ULL平臺設計實現(xiàn)掉電檢測功能,首先選擇一路IO,利用IO電平變化觸發(fā)中斷,在編寫驅動時捕獲該路GPIO的中斷,然后在中斷響應函數(shù)中發(fā)
    的頭像 發(fā)表于 11-09 10:40 ?850次閱讀
    基于<b class='flag-5'>i.MX6ULL</b>的掉電檢測設計與軟件測試

    【迅為電子】i.MX6UL和i.MX6ULL芯片區(qū)別與開發(fā)板對比

    【迅為電子】i.MX6UL和i.MX6ULL芯片區(qū)別與開發(fā)板對比
    的頭像 發(fā)表于 11-28 14:31 ?386次閱讀
    【迅為電子】<b class='flag-5'>i.MX6</b>UL和<b class='flag-5'>i.MX6ULL</b>芯片區(qū)別與開發(fā)板對比