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

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

3天內不再提示

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

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

上一篇介紹了虛擬字符設備的驅動,這篇就深入學習字符驅動的流程,看看字符驅動和應用層是怎么配合使用的!

1、備份原來的驅動

5cd1c440-57ef-11ed-a3b6-dac502259ad0.png

2、修改原來的驅動

在打印輸出時,[BSP]開頭表示驅動,[APP]開頭表示應用,Makefile不用修改;

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("[BSP]kernel senddata ok!
");
  }else{
    printk("[BSP]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("[BSP]kernel recevdata:%s
", writebuf);
  }else{
    printk("[BSP]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("[BSP]chrdevbase driver register failed
");
  }
  printk("[BSP]chrdevbase init!
");
  return 0;
}


/*
 * @description  : 驅動出口函數
 * @param     : 無
 * @return     : 無
 */
static void __exit chrdevbase_exit(void)
{
  /* 注銷字符設備驅動 */
  unregister_chrdev(CHRDEVBASE_MAJOR, CHRDEVBASE_NAME);
  printk("[BSP]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("[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;
  }


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


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


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


  return 0;
}

3、編譯驅動和應用

5cee7630-57ef-11ed-a3b6-dac502259ad0.png

4、復制需要的文件到根文件系統中

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

5d11d6b6-57ef-11ed-a3b6-dac502259ad0.png

5、啟動內核

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

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

6、加載設備驅動

需要進入驅動文件目錄才能加載設備驅動;

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

效果如圖:

5d26e736-57ef-11ed-a3b6-dac502259ad0.png

7、創建設備節點文件

輸入如下命令創建/dev/chrdevbase 這個設備節點文件:

mknod /dev/chrdevbase c 200 0

8、驗證讀寫

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


//可以使用下面這行輸出文件名稱,輸出/dev/chrdevbase
printf("filename:%s
",argv[1]);
//可以使用下面這行輸出參數,輸出1或者 2
printf("dat:%d
",atoi(argv[2]);

讀的流程:

5d645c60-57ef-11ed-a3b6-dac502259ad0.png

寫的流程:

5db60a2e-57ef-11ed-a3b6-dac502259ad0.png

注意事項

下面這個函數的打印輸出會印象到應用層的輸出,看到應用層輸出異常就把這個函數的輸出給屏蔽就好;

5de7e224-57ef-11ed-a3b6-dac502259ad0.png

審核編輯:湯梓紅

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

    關注

    0

    文章

    229

    瀏覽量

    24962
  • 設備驅動
    +關注

    關注

    0

    文章

    64

    瀏覽量

    10813

原文標題:i.MX6ULL|字符設備驅動流程深究

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

收藏 人收藏

    評論

    相關推薦

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

    i.MX6UL/i.MX6ULL開發常見問題》基于米爾電子 i.MX6UL/i.MX6ULL產品(V.10)2.3單獨編譯內核,uboot生成很多文件,具體用哪一個?答:內核編譯出來
    發表于 07-01 17:50

    迅為I.MX6ULL終結者開發板支持JTAG調試

    的硬件環境1、迅為-i.MX6ULL終結者開發板一塊2、JLNK V9下載器一個3、JLINK V9轉換板一個(2.54mm轉2.0mm)1.2 搭建開發環境1.2.1 安裝JLINK V9驅動首先
    發表于 05-06 14:09

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

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

    i.MX6ULL開發板硬件資源

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

    迅為i.MX6ULL終結者線程基礎線程終止與回收解析

    迅為i.MX6ULL終結者線程基礎線程終止與回收
    發表于 04-02 06:43

    i.MX6ULL核心板資源

    `STAMP-IMX6ULL-CM是浙江啟揚智能科技有限公司推出的基于 NXP i.MX6ULL 處理器的產品;i.MX 6ULL 是 NXP 推出的 ARM Cortex-A7 系列
    發表于 07-12 17:50

    stm32f0717bt6i.mx6ull啟動方式的區別是什么?

    stm32f0717bt6i.mx6ull啟動方式的區別是什么?
    發表于 11-29 07:51

    初識 i.MX6ULL 寄存器

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

    ARM裸機篇之i.MX6ULL處理器資料分享

    1、i.MX6ULL處理器啟動過程i.MX6ULL是NXP基于ARM Cortex-A7內核的單核處理器家族,主頻可以高900MHz。i.MX6ULL應用處理器包含了電源管理模塊,可以降低外部電源
    發表于 04-14 16:42

    關于i.MX6ULL配置GPIO

    Linux下一切皆文件。在Linux系統當中,有一個文件專門用于配置處理器的各個外設,包括GPIO,這個文件被稱為“設備樹” ,i.MX6ULL設備樹在內核源碼中的路徑為:arch/arm/boot
    發表于 08-05 10:37

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

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

    I.MX6ULL UART傳輸問題求解

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

    如何使用Linux版本在i.mx6ull上啟用USB網絡共享?

    我從 lf-5.10-y 分支為 i.mx6ull evk 構建了 Linux。我將我的 android 手機連接到 usb otg 端口并在我的手機上啟用網絡共享。但是我沒有看到 USB0 接口
    發表于 05-09 08:06

    如何在i.MX6ULL上為PF1510配置i2c?

    我們計劃將 PF1510 與 i.MX6ULL 處理器一起使用。我看到這個設備樹示例: https://github.com/Freescale/linux-fslc/blob
    發表于 05-17 14:02

    已知的HAB漏洞是否會影響i.MX6ULL版本1.1?

    我有一個 NXP iMX6ULL rev 1.1,發現i.MX6ULL 的 HAB 機制中存在兩個已知漏洞。但是,我所指的文檔并未指定芯片版本。因此,我不確定這些漏洞是否會影響我的設備。你能幫我澄清一下嗎?
    發表于 06-02 09:07
    亚洲欧美日韩精品久久_久久精品AⅤ无码中文_日本中文字幕有码在线播放_亚洲视频高清不卡在线观看
    <acronym id="s8ci2"><small id="s8ci2"></small></acronym>
    <rt id="s8ci2"></rt><rt id="s8ci2"><optgroup id="s8ci2"></optgroup></rt>
    <acronym id="s8ci2"></acronym>
    <acronym id="s8ci2"><center id="s8ci2"></center></acronym>