目次
vmalloc – vfree" >vmalloc – vfree
slab缓存" >slab缓存
范例" >范例
内存池" >内存池
ホームページ システムチュートリアル Linux Linux ドライバーのメモリ アプリケーション テクノロジ: 原理と方法

Linux ドライバーのメモリ アプリケーション テクノロジ: 原理と方法

Feb 12, 2024 am 09:45 AM
linux Linuxチュートリアル Linuxシステム Linuxコマンド シェルスクリプト overflow 埋め込みLinux Linux を始める Linux学習

メモリは Linux システムで最も重要なリソースの 1 つであり、データ、コード、スタックなどの保存に使用できます。メモリの適用と解放は、Linux ドライバ開発における最も基本的な操作の 1 つであり、カーネル空間とユーザー空間、静的割り当てと動的割り当て、連続メモリと不連続メモリなどの概念が含まれます。この記事では、kmalloc、vmalloc、get_free_pages、dma_alloc_coherent などの Linux ドライバーのメモリ アプリケーション テクノロジを紹介し、使用例と注意事項を説明します。

Linux ドライバーのメモリ アプリケーション テクノロジ: 原理と方法

基本的なことから始めましょう。下の図は Linux のメモリ マッピング モデルです。

  1. 各プロセスには独自のプロセス空間があります。プロセス空間の 0 ~ 3G はユーザー空間、3G ~ 4G はカーネル空間です。
  2. 各プロセスのユーザー空間は同じ物理メモリ ページ内にありませんが、すべてのプロセスのカーネル空間は同じ物理アドレス
  3. に対応します。
  4. vmalloc によって割り当てられるアドレスは、ハイエンド メモリまたはローエンド メモリです。
  5. 物理マッピング領域には、0 ~ 896MB の物理アドレスがリニアにマッピングされます。

#「

Linux ドライバーのメモリ アプリケーション テクノロジ: 原理と方法

動的メモリ アプリケーション

アプリケーション層と同様に、カーネル プログラムもメモリを動的に割り当てる必要があります。違いは、割り当てられたメモリがユーザー空間にあるかカーネル空間にあるかをカーネル プロセスが制御できることです。前者は、メモリをヒープに割り当てるために使用できます。ユーザー空間の領域、例えばユーザープロセスのユーザー空間のmallocは、最終的にシステムコールを通じてカーネル空間のメモリ割り当て関数をコールバックしますこのとき、メモリ割り当て関数はユーザープロセスに属し、ユーザー プロセスのヒープ領域に領域を割り当ててそれを返すことができ、最終的にユーザー プロセスは独自のユーザー領域でメモリ割り当てを取得します。後者はカーネル領域にのみ割り当てられるため、ユーザー プロセスはカーネル領域に直接アクセスできません。スペースなので、主にカーネル プログラム自体のメモリ ニーズを満たすために使用されます。以下は、Linux カーネル スペース アプリケーション メモリの一般的な API です。 :

kmalloc – kfree

kmalloc によって要求されるメモリは物理メモリ内で連続しており、実際の物理アドレスからの固定オフセットのみが存在するため、単純な変換関係が存在します。この API

は主に、1 ページ サイズ未満のメモリ を適用するために使用されます。 kmalloc の最下層は **__get_free_pages を呼び出す必要があります。パラメータ内のメモリ タイプを示す gtp_t フラグは、この関数の略称です。一般的に使用されるメモリ タイプには、GFP_USER、GFP_KERNEL、GFP_ATOMIC** が含まれます。

  • GFP_USER はユーザー空間ページにメモリを割り当てることを意味し、ブロックされる可能性があります;
  • GFP_KERNEL は最も一般的に使用されるフラグです。このフラグを使用してメモリを適用する場合、一時的に満たせない場合、プロセスがブロックされることに注意してください。そのため、 は、割り込み処理関数、タスクレット、およびカーネル タイマー非プロセス コンテキストで GFP_KERNEL を使用してください。 ! !
  • GFP_ATOMIC は、上記の 3 つのシナリオで使用できます。このフラグは、要求されたメモリが使用できない場合は、すぐに返されることを示します。
  • rree
同じシリーズの API には、

もあります。 リーリー

__get_free_pages – free_pages

__get_free_pages() は kmalloc() と同様、物理的に連続したメモリです。基礎となるバディ アルゴリズムが **(2^n )×PAGE_SIZE

はメモリを管理するため、常にページ単位でメモリを割り当てます** リーリー

同じシリーズの API には、

もあります。

unsigned long __get_free_page(gfp_t gfp)        
unsigned long get_zeroed_page(gfp_t gfp_mask)    
struct page *alloc_pages(gfp_t gfp_mask, unsigned int order)
void free_page(unsigned long addr)  

vmalloc – vfree

vmalloc在虚拟内存空间给出一块连续的内存区,实质上,这片连续的虚拟内存在物理内存中并不一定连续,所以vmalloc申请的虚拟内存和物理内存之间也就没有简单的换算关系,正因如此,vmalloc()通常用于分配远大于__get_free_pages()的内存空间,它的实现需要建立新的页表,此外还会调用使用GFP_KERN的kmalloc,so,一定不要在中断处理函数,tasklet和内核定时器等非进程上下文中使用vmalloc!

/**     
 * vmalloc  -  allocate virtually contiguous memory
 * @size:          allocation size
 * Allocate enough pages to cover @size from the page level allocator 
and map them into contiguous kernel virtual space.
 */
void *vmalloc(unsigned long size)   

/**
 *      vfree  -  release memory allocated by vmalloc()
 *      @addr:          memory base address
 */
void vfree(const void *addr)  

同系列的API还有

/**
 * vmalloc_32  -  allocate virtually contiguous memory (32bit addressable)
 * @size:          allocation size
 * Allocate enough 32bit PA addressable pages to cover @size from the 
page level allocator and map them into contiguous kernel virtual space.
 */
void *vmalloc_32(unsigned long size) 

slab缓存

我们知道,页是内存映射的基本单位,但内核中很多频繁创建的对象所需内存都不到一页,此时如果仍然按照页映射的方式,频繁的进行分配和释放就会造成资源的浪费,同时也会降低系统性能。为了解决的这样的问题,内核引入了slab机制,使对象在前后两次被使用时被分配在同一块内存或同一类内存空间,且保留了基本的数据结构,就可以大大提高效率。kmalloc的底层即是使用slab算法管理分配的内存的。注意,slab依然是以页为单位进行映射,只是映射之后分割这些页为相同的更小的单元,从而节省了内存。slab分配的单元不能小于32B或大于128K。

/**
 * kmem_cache_create - 创建slab缓存对象
 * @name:slab缓存区名字,
 * @size:slab分配的缓存区的每一个单元的大小
 * @align:缓存区内存的对齐方式,一般给0
 * @flags:控制分配的位掩码,
 * %SLAB_POISON - Poison the slab with a known test pattern (a5a5a5a5) to catch references 
to uninitialised memory.
 * %SLAB_RED_ZONE - Insert `Red' zones around the allocated memory to check for buffer 
overruns.
 * %SLAB_HWCACHE_ALIGN - Align the objects in this cache to a hardware cacheline.  
This can be beneficial if you're counting cycles as closely as davem.
 * %SLAB_CACHE_DMA - Use GFP_DMA memory
 * %SLAB_STORE_USER - Store the last owner for bug hunting
 *define SLAB_PANIC - Panic if kmem_cache_create() fails 
 */
struct kmem_cache *kmem_cache_create(const char *name, size_t size,
 size_t align,unsigned long flags, void (*ctor)(void *))


/**
 * kmem_cache_alloc - Allocate an object from this cache. 
 * @cachep: The cache to allocate from.
 * @flags: See kmalloc().
 * The flags are only relevant if the cache has no available objects.
 */
void *kmem_cache_alloc(struct kmem_cache *cachep, gfp_t flags)  


/**
 * kmem_cache_free - Deallocate an object
 * @cachep: The cache the allocation was from.
 * @objp: The previously allocated object.
 * Free an object which was previously allocated from this cache.
 */
void kmem_cache_free(struct kmem_cache *cachep, void *objp)  


void kmem_cache_destroy(struct kmem_cache *s)  

范例

//创建slab对象
struct kmem_cache_t *xj_sbcache;
xj_sbcache = kmem_cache_create("xjslab",sizeof(struct xj_unit_t),0,SLAB_CACHE_DMA|SLAB_PANIC,NULL,NULL);

//分配slab缓存
struct xj_unit_t *xj_unit;
xj_unit = kmem_cache_alloc(xj_sbcache,GFP_KERNEL);

/* 使用slab缓存 */

/* 释放slab缓存 */
kmem_cache_free(xj_sbcache, xj_unit);

/* 销毁slab缓存 */
kmem_cache_destroy(xj_sbcache);

内存池

除了slab机制,内核还提供了传统的内存池机制来管理小块内存的分配。内存池主要是用来解决可能出现的内存不足的情况,因为一个内存池在创建的时候就已经分配好了一内存,当我们用mempool_alloc向一个已经创建好的内存池申请申请内存时,该函数首先会尝试回调内存池创建时的分配内存函数,如果已经没有内存可以分配,他就会使用内存池创建时预先分配的内存,这样就可以避免因为无内存分配而陷入休眠,当然,如果预分配的内存也已经使用完毕,还是会陷入休眠。slab机制的目的是提高内存使用率以及内存管理效率,内存池的目的是避免内存的分配失败。下面是内核中提供的关于内存池的API

/**     
 * mempool_create - create a memory pool
 * @min_nr:    the minimum number of elements guaranteed to be  allocated for this pool.
 * @alloc_fn:  user-defined element-allocation function.
 * @free_fn:   user-defined element-freeing function.
 * @pool_data: optional private data available to the user-defined functions.
 *              
 * this function creates and allocates a guaranteed size, preallocated memory pool. 
The pool can be used from the mempool_alloc() and mempool_free() functions. 
 * This function might sleep. Both the alloc_fn() and the free_fn() functions
 might sleep - as long as the mempool_alloc() function is not called from IRQ contexts.
 */
mempool_t *mempool_create(int min_nr, mempool_alloc_t *alloc_fn, mempool_free_t 
*free_fn, void *pool_data)

/**     
 * mempool_alloc - allocate an element from a specific memory pool
 * @pool:      pointer to the memory pool which was allocated via mempool_create().
 * @gfp_mask:  the usual allocation bitmask.
 * this function only sleeps if the alloc_fn() function sleeps or returns NULL.
 Note that due to preallocation, this function never* fails when called from process
 contexts. (it might fail if called from an IRQ context.)
 */     
void * mempool_alloc(mempool_t *pool, gfp_t gfp_mask)    

/**
 * mempool_free - return an element to the pool.
 * @element:   pool element pointer.
 * @pool:      pointer to the memory pool which was allocated via mempool_create().
 *
 * this function only sleeps if the free_fn() function sleeps.
 */     
void mempool_free(void *element, mempool_t *pool)    

/**
 * mempool_destroy - deallocate a memory pool
 * @pool:      pointer to the memory pool which was allocated via mempool_create().
 *
 * Free all reserved elements in @pool and @pool itself. 
 This function only sleeps if the free_fn() function sleeps.
 */     
void mempool_destroy(mempool_t *pool)  

通过本文,我们了解了Linux驱动中的内存申请技术,它们各有优缺点和适用场景。我们应该根据实际需求选择合适的函数,并遵循一些基本原则,如匹配申请和释放函数,检查返回值是否为空,避免内存泄漏等。内存申请技术是Linux驱动开发中不可或缺的一部分,它可以保证驱动程序的正常运行和数据交换,也可以提升驱动程序的性能和稳定性。希望本文能够对你有所帮助和启发。

以上がLinux ドライバーのメモリ アプリケーション テクノロジ: 原理と方法の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。

ホットAIツール

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Clothoff.io

Clothoff.io

AI衣類リムーバー

Video Face Swap

Video Face Swap

完全無料の AI 顔交換ツールを使用して、あらゆるビデオの顔を簡単に交換できます。

ホットツール

メモ帳++7.3.1

メモ帳++7.3.1

使いやすく無料のコードエディター

SublimeText3 中国語版

SublimeText3 中国語版

中国語版、とても使いやすい

ゼンドスタジオ 13.0.1

ゼンドスタジオ 13.0.1

強力な PHP 統合開発環境

ドリームウィーバー CS6

ドリームウィーバー CS6

ビジュアル Web 開発ツール

SublimeText3 Mac版

SublimeText3 Mac版

神レベルのコード編集ソフト(SublimeText3)

VSCODEオフラインテクノロジー交換活動に参加した経験 VSCODEオフラインテクノロジー交換活動に参加した経験 May 29, 2025 pm 10:00 PM

私はVSCodeオフラインテクノロジー交換活動に参加した経験があります。私の主な利益には、プラグイン開発の共有、実用的なデモンストレーション、他の開発者とのコミュニケーションが含まれます。 1.プラグイン開発の共有:VSCODEのプラグインAPIを使用して、自動フォーマットや静的分析プラグインなどの開発効率を改善する方法を学びました。 2。実践的なデモンストレーション:リモート開発にVSCodeを使用する方法を学び、その柔軟性とスケーラビリティを実現しました。 3。開発者との通信:起動時にロードされたプラグインの数を減らしたり、プラグインロード順序を管理するなど、VSCODEスタートアップ速度を最適化するスキルを取得しました。要するに、このイベントは私に大きな恩恵をもたらし、VSCODEに興味がある人に参加することを強くお勧めします。

Linuxのユーザーリソースを制限する方法は? ulimitを構成する方法は? Linuxのユーザーリソースを制限する方法は? ulimitを構成する方法は? May 29, 2025 pm 11:09 PM

Linuxシステムは、リソースの過度の使用を防ぐために、ULIMITコマンドを介してユーザーリソースを制限します。 1.ulimitは、ファイル記述子(-n)、メモリサイズ(-v)、スレッドカウント(-u)などの数を制限できるビルトインシェルコマンドであり、ソフト制限(現在の有効値)とハードリミット(最大上限)に分割されます。 2。Ulimit-N2048などの一時的な変更には、Ulimitコマンドを直接使用しますが、現在のセッションでのみ有効です。 3.永続的な効果を得るには、/etc/security/limits.confを変更し、PAM構成ファイルを変更し、SessionRequiredPam_limits.soを追加する必要があります。 4. SystemDサービスは、ユニットファイルにLIMを設定する必要があります

LinuxのInformixとMySQLの比較 LinuxのInformixとMySQLの比較 May 29, 2025 pm 11:21 PM

InformixとMySQLはどちらも人気のあるリレーショナルデータベース管理システムです。どちらもLinux環境でうまく機能し、広く使用されています。以下は、Linuxプラットフォーム上の2つの比較と分析です。Informixのインストールと構成:LinuxにInformixを展開するには、対応するインストールファイルをダウンロードし、公式ドキュメントに従ってインストールと構成プロセスを完了する必要があります。 MySQL:MySQLのインストールプロセスは比較的簡単で、システムパッケージ管理ツール(APTやYumなど)を介して簡単にインストールできます。また、参照用のネットワークには多数のチュートリアルとコミュニティサポートがあります。パフォーマンスInformix:Informixには優れたパフォーマンスがあります

Debianの下にFileBeatとElasticSearchを統合する方法 Debianの下にFileBeatとElasticSearchを統合する方法 May 28, 2025 pm 05:09 PM

Debianオペレーティングシステムでは、FileBeatとElasticSearchの統合により、ログデータの収集、送信、およびストレージを簡素化できます。以下は特定の実装手順です。ステップ1:Elasticsearchを展開する最初のタスクは、DebianシステムにElasticsearchのインストールを完了することです。 Elastic Softwareパッケージの対応するバージョンをElastic公式Webサイトからダウンロードし、公式ガイダンスに従ってインストールプロセスを完了することができます。 elasticsearchwgethttps://artifacts.elastic.co/downloads/elasticseをダウンロードしてインストールします

vscodeプラグインの更新後のエディタークラッシュの理由とソリューション vscodeプラグインの更新後のエディタークラッシュの理由とソリューション May 29, 2025 pm 10:03 PM

VSCODEプラグインが更新された後にエディターがクラッシュする理由は、VSCODEまたは他のプラグインの既存のバージョンを備えたプラグインに互換性のある問題があるためです。ソリューションには以下が含まれます。1。プラグインを無効にして、問題を1つずつトラブルシューティングします。 2。問題プラグインを以前のバージョンにダウングレードします。 3.代替プラグインを見つけます。 4. VSCODEとプラグインを更新し、十分なテストを実施します。 5.データの損失を防ぐために、自動バックアップ機能を設定します。

MySQLを中国のインターフェイスに調整する方法は? MySQLの中国語環境を簡単に設定できます MySQLを中国のインターフェイスに調整する方法は? MySQLの中国語環境を簡単に設定できます Jun 04, 2025 pm 06:36 PM

MySQLを中国のインターフェイスにチューニングするために、MySQLWorkBenchまたはコマンドラインツールを使用して実装できます。 1)mysqlworkbenchで、「設定」を開き、「外観」タブを選択し、「言語」ドロップダウンメニューで「中国語(簡素化)」を選択して再起動します。 2)コマンドラインツールを使用する場合、LinuxまたはMacOSで「exportlang = zh_cn.utf-8」を使用するなど、オペレーティングシステムのロケール変数を設定し、MySQLクライアントを実行します。

Debian Notepadを他の編集者と比較する方法 Debian Notepadを他の編集者と比較する方法 May 29, 2025 pm 10:42 PM

Debian Text Editorは、主に毎日の単純なテキスト編集作業に使用される基本的なテキスト編集ツールです。他の主流の編集者と比較して、パフォーマンスとユーザーエクスペリエンスに特定の制限があります。 Debian Text Editorsと比較した他のいくつかの編集者の利点と機能は次のとおりです。メモ帳の読み込み速度速度:Notepadは、わずか8秒で1GBサイズのSQLファイルなどの大きなファイルをすばやくロードできます。コード着色機能:約80のプログラミング言語でコード着色をサポートしているため、コーディング効率の向上に役立ちます。バッチ操作機能:列編集モードがあります。これは、財務検証や操作、メンテナンスログなどのバッチ処理タスクを実行するのに便利です。拡張プラグインブランチ

DebianでSFTPサービスを開始する方法 DebianでSFTPサービスを開始する方法 May 29, 2025 pm 10:51 PM

DebianシステムでSFTPサービスを開始するには、通常、OpenSSHサーバーの助けが必要です。以下は特定の手順です。1。最初にOpenSSHサーバーをインストールし、OpenSSHサーバーがDebianシステムにインストールされていることを確認します。インストールされていない場合は、次のコマンド:sudoaptupdatesudoaptinInstallopenssh-server2でインストールを完了できます。 OpenSSHサーバーのインストールが完了した後、OpenSSHサーバーは通常自動的に開始されます。次のコマンドを介してその実行ステータスを確認できます。SudosystemCtlStatussshサービスが実行されていない場合、次のコマンドで開始できます。

See all articles