精品国产人成在线_亚洲高清无码在线观看_国产在线视频国产永久2021_国产AV综合第一页一个的一区免费影院黑人_最近中文字幕MV高清在线视频

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

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

3天內不再提示

i.MX6ULL|字符設備驅動開發實踐

玩轉單片機 ? 來源:玩轉單片機 ? 作者:玩轉單片機 ? 2022-10-31 11:27 ? 次閱讀

字符設備驅動開發的基本步驟可以看上一篇,本節就以 chrdevbase 這個虛擬設備為例,完整的編寫一個字符設備驅動模塊。chrdevbase 不是實際存在的一個設備,方便講解字符設備的開發而引入的一個虛擬設備。chrdevbase 設備有兩個緩沖區,一個為讀緩沖區,一個為寫緩沖區,這兩個緩沖區的大小都為 100 字節。在應用程序中可以向 chrdevbase 設備的寫緩沖區中寫入數據,從讀緩沖區中讀取數據。chrdevbase 這個虛擬設備的功能很簡單,但是它包含了字符設備的最基本功能。

|需求目標

應用程序調用 open 函數打開 chrdevbase 這個設備,打開以后可以使用 write 函數向chrdevbase 的寫緩沖區 writebuf 中寫入數據(不超過 100 個字節),也可以使用 read 函數讀取讀緩沖區 readbuf 中的數據操作,操作完成以后應用程序使用 close 函數關閉 chrdevbase 設備。

|實現過程

1.新建文件一個nxp文件夾,然后把原廠內核文件復制過來

dd73bc60-56e2-11ed-a3b6-dac502259ad0.png

2、創建Linux_Drivers用于存放驅動文件,創建01_chrdevbase用于存放chrdevbase實驗文件,chrdevbase.c是底層驅動代碼,chrdevbaseApp.c是應用代碼,Makefile編譯底層驅動;

dd9605fe-56e2-11ed-a3b6-dac502259ad0.png

3、編寫代碼

chrdevbase.c文件

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


#define CHRDEVBASE_MAJOR  200        /* 主設備號 */
#define CHRDEVBASE_NAME    "chrdevbase"   /* 設備名     */


static char readbuf[100];    /* 讀緩沖區 */
static char writebuf[100];    /* 寫緩沖區 */
static char kerneldata[] = {"kernel data!"};


/*
 * @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("chrdevbase open!
");
  return 0;
}


/*
 * @description    : 從設備讀取數據 
 * @param - filp   : 要打開的設備文件(文件描述符)
 * @param - buf   : 返回給用戶空間的數據緩沖區
 * @param - cnt   : 要讀取的數據長度
 * @param - offt   : 相對于文件首地址的偏移
 * @return       : 讀取的字節數,如果為負值,表示讀取失敗
 */
static ssize_t chrdevbase_read(struct file *filp, char __user *buf, size_t cnt, loff_t *offt)
{
  int retvalue = 0;
  
  /* 向用戶空間發送數據 */
  memcpy(readbuf, kerneldata, sizeof(kerneldata));
  retvalue = copy_to_user(buf, readbuf, cnt);
  if(retvalue == 0){
    printk("kernel senddata ok!
");
  }else{
    printk("kernel senddata failed!
");
  }
  
  //printk("chrdevbase read!
");
  return 0;
}


/*
 * @description    : 向設備寫數據 
 * @param - filp   : 設備文件,表示打開的文件描述符
 * @param - buf   : 要寫給設備寫入的數據
 * @param - cnt   : 要寫入的數據長度
 * @param - offt   : 相對于文件首地址的偏移
 * @return       : 寫入的字節數,如果為負值,表示寫入失敗
 */
static ssize_t chrdevbase_write(struct file *filp, const char __user *buf, size_t cnt, loff_t *offt)
{
  int retvalue = 0;
  /* 接收用戶空間傳遞給內核的數據并且打印出來 */
  retvalue = copy_from_user(writebuf, buf, cnt);
  if(retvalue == 0){
    printk("kernel recevdata:%s
", writebuf);
  }else{
    printk("kernel recevdata failed!
");
  }
  
  //printk("chrdevbase write!
");
  return 0;
}


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


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


/*
 * @description  : 驅動入口函數 
 * @param     : 無
 * @return     : 0 成功;其他 失敗
 */
static int __init chrdevbase_init(void)
{
  int retvalue = 0;


  /* 注冊字符設備驅動 */
  retvalue = register_chrdev(CHRDEVBASE_MAJOR, CHRDEVBASE_NAME, &chrdevbase_fops);
  if(retvalue < 0){
    printk("chrdevbase driver register failed
");
  }
  printk("chrdevbase init!
");
  return 0;
}


/*
 * @description  : 驅動出口函數
 * @param     : 無
 * @return     : 無
 */
static void __exit chrdevbase_exit(void)
{
  /* 注銷字符設備驅動 */
  unregister_chrdev(CHRDEVBASE_MAJOR, CHRDEVBASE_NAME);
  printk("chrdevbase exit!
");
}


/* 
 * 將上面兩個函數指定為驅動的入口和出口函數 
 */
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"


static char usrdata[] = {"usr data!"};


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


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


  filename = argv[1];


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


  if(atoi(argv[2]) == 1){ 
    /* 從驅動文件讀取數據 */
    retvalue = read(fd, readbuf, 50);
    if(retvalue < 0){
      printf("read file %s failed!
", filename);
    }else{
      /*  讀取成功,打印出讀取成功的數據 */
      printf("read data:%s
",readbuf);
    }
  }


  if(atoi(argv[2]) == 2){
    /* 向設備驅動寫數據 */
    memcpy(writebuf, usrdata, sizeof(usrdata));
    retvalue = write(fd, writebuf, 50);
    if(retvalue < 0){
      printf("write file %s failed!
", filename);
    }
  }


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


  return 0;
}

Makefile文件

KERNELDIR := /home/noah/linux/nxp/linux-imx-rel_imx_4.1.15_2.1.0_ga
CURRENT_PATH := $(shell pwd)
obj-m := chrdevbase.o


build: kernel_modules


kernel_modules:
  $(MAKE) -C $(KERNELDIR) M=$(CURRENT_PATH) modules
clean:
  $(MAKE) -C $(KERNELDIR) M=$(CURRENT_PATH) clean

相關解析:

第 1 行,KERNELDIR 表示開發板所使用的 Linux 內核源碼目錄,使用絕對路徑,大家根據自己的實際情況填寫即可。
第2行,CURRENT_PATH表示當前路徑,直接通過運行“pwd”命令來獲取當前所處路徑。
第 3 行,obj-m 表示將 chrdevbase.c 這個文件編譯為 chrdevbase.ko 模塊。
第 8 行,具體的編譯命令,后面的 modules 表示編譯模塊,-C 表示將當前的工作目錄切換到指定目錄中,也就是 KERNERLDIR 目錄。M 表示模塊源碼目錄,“make modules”命令中加入 M=dir 以后程序會自動到指定的 dir 目錄中讀取模塊的源碼并將其編譯為.ko 文件。

4、編譯驅動

直接使用make命令會報錯的,因為kernel中沒有指定編譯器和架構,使用了默認的x86平臺編譯報錯。

dda9deee-56e2-11ed-a3b6-dac502259ad0.png

5、修改內核的Makefile文件

直接定義ARCH和CROSS_COMPILE 這兩個的變量值為 arm 和 arm-linux-gnueabihf-

ddcd3074-56e2-11ed-a3b6-dac502259ad0.png

ddf54550-56e2-11ed-a3b6-dac502259ad0.png

6、再次編譯驅動

編譯通過,會生成不少編譯文件;

de235ab2-56e2-11ed-a3b6-dac502259ad0.png

7、編譯應用程序

應用程序只有一個文件,在ubuntu對應文件夾,直接輸入指令進行編譯:

arm-linux-gnueabihf-gcc chrdevbaseApp.c -o chrdevbaseApp

8、SD卡啟動系統

插入SD卡,把撥碼開關調到SD卡啟動,然后ping一下網關,確認網絡通暢才能繼續的進行,如果ping不通就請看看前幾節;

de5d4ae2-56e2-11ed-a3b6-dac502259ad0.png

9、啟動內核

在uboot界面輸入下面指令啟動系統,

tftp80800000zImage
tftp 83000000 imx6ull-14x14-evk.dtb
bootz 80800000 - 83000000

10、創建目錄并復制驅動文件

檢查開發板根文件系統中有沒有“/lib/modules/4.1.15”這個目錄,如果沒有的話自行創建。注意,“/lib/modules/4.1.15”這個目錄用來存放驅動模塊,使用modprobe 命令加載驅動模塊的時候,驅動模塊要存放在此目錄下?!?lib/modules”是通用的,不管你用的什么板子、什么內核,這部分是一樣的。

將 chrdevbase.ko 和 chrdevbaseAPP 復制到 rootfs/lib/modules/4.1.15 目錄中:

de71797c-56e2-11ed-a3b6-dac502259ad0.png

11、加載設備驅動

自制的根文件系統,有些命令是不支持的;

//加載驅動
insmodchrdevbase.ko
// 查看驅動
lsmod
// 指令查看devices信息
cat/proc/devices

效果如下圖:

de90d736-56e2-11ed-a3b6-dac502259ad0.png

12、創建設備節點文件

驅動加載成功需要在/dev 目錄下創建一個與之對應的設備節點文件,應用程序就是通過操作這個設備節點文件來完成對具體設備的操作。輸入如下命令創建/dev/chrdevbase 這個設備節點文件:

mknod /dev/chrdevbase c 200 0
其中“mknod”是創建節點命令,“/dev/chrdevbase”是要創建的節點文件,“c”表示這是個字符設備,“200”是設備的主設備號,“0”是設備的次設備號。創建完成以后就會存在/dev/chrdevbase 這個文件,可以使用“ls /dev/chrdevbase -l”命令查看,結果如圖:

decc0f18-56e2-11ed-a3b6-dac502259ad0.png

13、驗證讀寫

使用 chrdevbaseApp 軟件操作 chrdevbase 這個設備,看看讀寫是否正常,首先進行讀操作,輸入如下命令:

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

相關解析:

三個參數“./chrdevbaseApp”、“/dev/chrdevbase”和“1”,這三個參數分別對應 argv[0]、argv[1]和 argv[2]。

第一個參數表示運行 chrdevbaseAPP 這個軟件,

第二個參數表示測試APP要打開/dev/chrdevbase這個設備。

第三個參數就是要執行的操作,1表示從chrdevbase中讀取數據,2 表示向 chrdevbase 寫數據。

deee90ec-56e2-11ed-a3b6-dac502259ad0.png

效果如下:

df11f686-56e2-11ed-a3b6-dac502259ad0.png

14、卸載驅動模塊

如果不再使用某個設備的話可以將其驅動卸載掉,命令如下:

// 卸載chrdevbase.ko
rmmod chrdevbase.ko
// 查看驅動
lsmod

至此,chrdevbase 這個設備的整個驅動就驗證完成了,驅動工作正常。

審核編輯:湯梓紅

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • Linux
    +關注

    關注

    87

    文章

    11227

    瀏覽量

    208925
  • 函數
    +關注

    關注

    3

    文章

    4305

    瀏覽量

    62430
  • 驅動開發
    +關注

    關注

    0

    文章

    130

    瀏覽量

    12062

原文標題:i.MX6ULL|字符設備驅動開發實踐

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

收藏 人收藏

    評論

    相關推薦

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

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

    i.MX6ULL嵌入式Linux開發1-uboot移植初探

    本系列教程以i.MX6ULL處理器的ARM開發板為實驗基礎,學習記錄嵌入式Linux開發的各種知識與經驗,主要內容包括嵌入式Linux移植,嵌入式Linux驅動
    的頭像 發表于 03-07 08:57 ?3845次閱讀
    <b class='flag-5'>i.MX6ULL</b>嵌入式Linux<b class='flag-5'>開發</b>1-uboot移植初探

    使用i.MX6ULL開發板進行Linux根文件系統的完善

    上一篇推文講了怎么移植根文件系統,并在i.MX6ULL開發板中運行起來,但是會出現一些提示,現在來進行根文件的完善。
    發表于 10-17 11:13 ?765次閱讀

    移植NXP官方linux 5.4內核到i.MX6ULL開發

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

    i.MX6UL/i.MX6ULL開發常見問題】單獨編譯內核,uboot生成很多文件,具體用哪一個?

    i.MX6UL/i.MX6ULL開發常見問題》基于米爾電子 i.MX6UL/i.MX6ULL產品(V.10)2.3單獨編譯內核,uboot
    發表于 07-01 17:50

    I.MX6ULL終結者開發板裸機仿真jlink調試

    I.MX6ULL‘終結者’開發板預留了JTAG仿真接口,并給出了開發文檔,可以實現在JLINK仿真器條件下的單步跟蹤、斷點調試等功能,使得開發研究i
    發表于 07-07 10:56

    i.MX6ULL開發板硬件資源

    迅為i.MX6ULL 終結者開發板硬件資源非常豐富,幾乎將 i.MX6ULL 芯片的所有資源都擴展引出到底板上了,底板提供了豐富的外設接口,開發板的尺寸是 190mm*125mm,充分
    發表于 12-29 06:18

    初識 i.MX6ULL 寄存器

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

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

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

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

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

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

    合作伙伴,飛凌不負美譽,基于i.MX6ULL匠心打造的FETMX6ULL-S核心板一經問世便好評不斷,且已有數百家來自工業、醫療、電力、物聯網等行業的用戶采用此款核心板快速完成了整機產品的開發上市。
    發表于 04-11 15:05 ?1139次閱讀
    基于NXP <b class='flag-5'>i.MX6ULL</b>處理器的FETMX<b class='flag-5'>6ULL</b>-C核心板

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

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

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

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

    【北京迅為】i.MX6ULL開發板移植 Debian 文件系統

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

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

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