驅動08.塊設備驅動程式

来源:http://www.cnblogs.com/Lwd-linux/archive/2017/01/13/6280459.html
-Advertisement-
Play Games

1 塊設備的概述 linux支持的兩種重要的設備類型分別是字元設備和塊設備,塊設備可以隨機地以固定大小的塊傳送數據。與字元設備相比,塊設備有以下幾個特殊之處: (1)塊設備可以從數據的任何位置進行訪問 (2)塊數據總是以固定長度進行傳輸,即便請求的這是一個位元組 (3)對塊設備的訪問有大量的緩存。當進 ...


1 塊設備的概述

  linux支持的兩種重要的設備類型分別是字元設備和塊設備,塊設備可以隨機地以固定大小的塊傳送數據。與字元設備相比,塊設備有以下幾個特殊之處:

  (1)塊設備可以從數據的任何位置進行訪問

  (2)塊數據總是以固定長度進行傳輸,即便請求的這是一個位元組

  (3)對塊設備的訪問有大量的緩存。當進行讀時,如果已經緩存了,就直接使用緩存中的數據,而不再讀設備,對於寫也通過緩存來進行延遲處理。

  在塊系統中,數據塊指的是固定大小的數據,該固定大小由內核規定,通常是4096個位元組。與數據塊對應的是扇區,它是由設備硬體所決定的一個塊,其大小取決於硬體,常見的硬體的扇區大多都是512個位元組。數據塊的大小都是扇區大小的整數倍。

2 對比之前的字元設備驅動程式的區別;

app:open  read   write

塊: drv_open drv_read drv_write

       硬體

如果按照操作字元設備的那套方法來操作塊設備,效率低,且容易損壞儀器。

塊設備的框架:

app:      open,read,write "1.txt"
---------------------------------------------  文件的讀寫
文件系統: vfat, ext2, ext3, yaffs2, jffs2      (把文件的讀寫轉換為扇區的讀寫)
-----------------ll_rw_block-----------------  扇區的讀寫
                       1. 把"讀寫"放入隊列
                       2. 調用隊列的處理函數(優化/調順序/合併)
            塊設備驅動程式     
---------------------------------------------
硬體:        硬碟,flash

3 分析ll_rw_block函數

    ll_rw_block
        for (i = 0; i < nr; i++) {
            struct buffer_head *bh = bhs[i];
            submit_bh(rw, bh);
                struct bio *bio; // 使用bh來構造bio (block input/output)
                submit_bio(rw, bio);
                    // 通用的構造請求: 使用bio來構造請求(request)
                    generic_make_request(bio);
                        __generic_make_request(bio);
                            request_queue_t *q = bdev_get_queue(bio->bi_bdev); // 找到隊列  
                            
                            // 調用隊列的"構造請求函數"
                            ret = q->make_request_fn(q, bio);
                                    // 預設的函數是__make_request
                                    __make_request
                                        // 先嘗試合併
                                        elv_merge(q, &req, bio);
                                        
                                        // 如果合併不成,使用bio構造請求
                                        init_request_from_bio(req, bio);
                                        
                                        // 把請求放入隊列
                                        add_request(q, req);
                                        
                                        // 執行隊列
                                        __generic_unplug_device(q);
                                                // 調用隊列的"處理函數"
                                                q->request_fn(q);

4 怎麼寫塊設備驅動程式呢?
  1. 分配gendisk: alloc_disk
  2. 設置
  2.1 分配/設置隊列: request_queue_t  // 它提供讀寫能力
    blk_init_queue
  2.2 設置gendisk其他信息             // 它提供屬性: 比如容量
  3. 註冊: add_disk
參考:drivers\block\xd.c和drivers\block\z2ram.c
--------------------------------------------------------------編輯於2017-01-13 01:03:00

 

  1 /*參考:drivers\block\z2ram.c*/
  2 
  3 
  4 #include <linux/module.h>
  5 #include <linux/errno.h>
  6 #include <linux/interrupt.h>
  7 #include <linux/mm.h>
  8 #include <linux/fs.h>
  9 #include <linux/kernel.h>
 10 #include <linux/timer.h>
 11 #include <linux/genhd.h>
 12 #include <linux/hdreg.h>
 13 #include <linux/ioport.h>
 14 #include <linux/init.h>
 15 #include <linux/wait.h>
 16 #include <linux/blkdev.h>
 17 #include <linux/blkpg.h>
 18 #include <linux/delay.h>
 19 #include <linux/io.h>
 20 
 21 #include <asm/system.h>
 22 #include <asm/uaccess.h>
 23 #include <asm/dma.h>
 24 
 25 
 26 #define RAMBLOCK_SIZE (1024*1024)
 27 #define DEVICE_NAME "ramblock"
 28 
 29 static DEFINE_SPINLOCK(ramblock_lock);
 30 
 31 static struct gendisk *ramblock_gendisk;
 32 static int major;
 33 static struct request_queue *ramblock_queue;
 34 static unsigned char *ramblock_buf;
 35 
 36 static int ramblock_getgeo(struct block_device *bdev, struct hd_geometry *geo)
 37 {
 38     geo->heads = 2;
 39     geo->cylinders = 32;
 40     geo->sectors = RAMBLOCK_SIZE/2/32/512;
 41     return 0;
 42 }
 43 
 44 
 45 static struct block_device_operations ramblock_fops = {
 46     .owner    = THIS_MODULE,
 47     .getgeo    = ramblock_getgeo,
 48 };
 49 
 50 static void do_ramblock_request(request_queue_t *q)
 51 {    
 52     struct request *req;
 53     if((req = elv_next_request(q)) != NULL)
 54     {
 55         unsigned long offset = req->sector <<9;
 56         unsigned long len = req->current_nr_sectors << 9;
 57 
 58         if (rq_data_dir(req) == READ)
 59         {
 60             memcpy(req->buffer, ramblock_buf+offset, len);
 61         }
 62         else
 63         {
 64             memcpy(ramblock_buf+offset, req->buffer, len);
 65         }
 66         end_request(req, 1);
 67             
 68     }
 69 
 70 
 71 }
 72 
 73 static int ramblock_init(void)
 74 {    
 75     ramblock_gendisk = alloc_disk(16);
 76 
 77     ramblock_queue = blk_init_queue(do_ramblock_request, &ramblock_lock);
 78        ramblock_gendisk->queue = ramblock_queue;
 79 
 80 
 81     major = register_blkdev(0, DEVICE_NAME);
 82         ramblock_gendisk->major = major;
 83         ramblock_gendisk->first_minor = 0;
 84         ramblock_gendisk->fops = &ramblock_fops;
 85         sprintf(ramblock_gendisk->disk_name, "ramblock");
 86     set_capacity(ramblock_gendisk, RAMBLOCK_SIZE / 512);
 87 
 88     ramblock_buf = kzalloc(RAMBLOCK_SIZE, GFP_KERNEL);
 89         add_disk(ramblock_gendisk);
 90 
 91 
 92     return 0;
 93 }
 94 
 95 static void ramblock_exit(void)
 96 {
 97     unregister_blkdev(major, DEVICE_NAME);
 98     del_gendisk(ramblock_gendisk);
 99     put_disk(ramblock_gendisk);
100     blk_cleanup_queue(ramblock_queue);
101     
102     kfree(ramblock_buf);
103     
104     
105 }
106 
107 module_init(ramblock_init);
108 module_exit(ramblock_exit);
109 MODULE_LICENSE("GPL");
ramblock.c

 

--------------------------------------------------------------編輯於2017-01-13 14:34:44

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 這篇文章將對linux下udp socket編程重要知識點進行總結,無論是開發人員應知應會的,還是說udp socket的一些偏僻知識點,本文都會講到。儘可能做到,讀了一篇文章之後,大家對udp socket有一個比較全面的認識。本文分為兩個專題,第一個是常用的upd socket框架,第二個是一些 ...
  • redhat 的更新包只對註冊的用戶生效,所以我們需要自己手動更改成CentOS 的更新包,CentOS幾乎和redhat是一樣的,所以無需擔心軟體包是否可安裝,安裝之後是否有問題。 ...
  • linux教學輔助訓練(第二階段) 標簽(空格分隔):Linux輔助訓練 [更多資料點我查看][1] 提示 :本階段性練習題是對《實戰教學筆記》相應章節知識的歸納與擴展部分,必須要會,是面試前必須重溫的一套基礎練習。(第一階段即為實戰教學筆記第四節 Linux命令基礎)。 一,問答考試 1.1 解釋 ...
  • 第十節 正則表達式 標簽(空格分隔):Linux實戰教學筆記 [更多資料點我查看][1] 第1章 什麼是正則表達式 1. 正則表達式就是為了處理大量的文本|字元串而定義的一套規則和方法 2. 通過定義的這些特殊符號的輔助,系統管理員就可以快速過濾,替換或輸出需要的字元串。Linux正則表達式一般以行 ...
  • 這裡說的“後門”並不是教你做壞事,而是讓你做好事,搭建自己的調試工具更好地進行調試開發。我們都知道,當程式發生異常錯誤時,我們需要定位到錯誤,有時我們還想,我們在不修改程式的前提下,就能通過log來定位錯誤呢?有人會說,我在我的程式裡加多點列印就好了,程式每做一步我就加一行列印,到時一查log就知道 ...
  • 1 比較nor/nand flash NOR NAND介面: RAM-Like,引腳多 引腳少,復用容量: 小 1M 2M 3M 大:128M 256M G讀: 簡單 複雜寫: 發出特定命令 慢 發出特定命令 快價格: 貴 便宜特點: 無位反轉、壞塊 位反轉、壞塊 關鍵重要的程式 大數據、容忍可以出 ...
  • 1 nand flash的操作 目的:讀地址A的數據,把數據B寫到地址A。 問1. 原理圖上NAND FLASH和S3C2440之間只有數據線,怎麼傳輸地址?答1.在DATA0~DATA7上既傳輸數據,又傳輸地址,當ALE為高電平時傳輸的是地址。 問2. 從NAND FLASH晶元手冊可知,要操作N ...
  • 1.桌面右擊新建txt文件複製下麵兩行代碼,修改文件尾碼名為bat保存文件 netsh wlan set hostednetwork mode=allow ssid=zhangxh key=xiaoheng123netsh wlan start hostednetwork 2.右擊bat文件以管理員 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...