Linux カーネルプログラミングにおけるコンテナの of() 関数の紹介

Linux カーネルプログラミングにおけるコンテナの of() 関数の紹介

序文

Linux カーネルプログラミングでは、マクロ関数 container_of(ptr, type, member) をよく見かけますが、ソースコードを辿っていくと、私たちのような一般人は絶望してしまいます (これは一体何なのか? 関数はこのように定義できるのか??? なぜ 0 があるのか​​??? ああ、忘れて、諦めよう...)。 カーネル マスターが強力になるのは、この部分です。たった 2 行のコードで、私たちは自分の人生を疑うようになります。すべてにはプロセスが必要なので、ゆっくりと進めてください。

実際、原理は非常に単純です。構造体型のメンバー member のアドレス ptr を指定して、構造体型の開始アドレスを見つけます。

type = ptr - size の開始アドレス (ここではバイト単位であるため、char * に変換する必要があります)。

ここまでで機能の説明は終わりです。簡単ですよね? ? ? 実はそうではありません。ここではサイズの計算方法については何も触れられておらず、それが私たちを混乱させています。

さて、関数プロトタイプのコンテナから始めましょう。

#define container_of(ptr, type, member) ({ \         
const typeof( ((type *)0)->member ) *__mptr = (ptr); \         
(type *)( (char *)__mptr - offsetof(type,member) );})

次は offserof 関数のプロトタイプです。

#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)

どうですか、かっこいいでしょう? さて、発表を始めましょう:

1. 0ポインタの使用(勝手に名付けたので、問題があるかは分かりません)

事実がすべてを物語っています:

#include <stdio.h>
 
構造体テスト
{
	文字 i ;
	整数j;
	文字k;
};
 
int メイン()
{
	構造体テスト temp;
	printf("&temp = %p\n",&temp);   
	printf("&temp.k = %p\n",&temp.k);
	printf("&((struct test *)0)->k = %d\n",((int)&((struct test *)0)->k));
 
}

コンパイルして実行すると、次の結果が得られます。

&temp = 0xbf9815b4
&temp.k = 0xbf9815bc
&((構造体テスト*)0)->k = 8

私の言いたいことはおわかりでしょう。カスタム構造には i、j、k という 3 つの変数があります。 バイト アラインメント要件のため、構造体のサイズは 4 バイト * 3 = 12 バイトです。&((struct test *)0)->k の機能は、k から構造体 temp の開始アドレスまでのバイト数 (つまり、サイズ) を見つけることです。ここで 0 は struct test * 型に強制的に変換され、その役割は構造体の開始アドレスへのポインターとして機能することであり、& ((struct test *) 0) -> k の役割は k から開始ポインターまでのバイト数を計算することです。 。 。実際、相対アドレスを見つけるためです。開始アドレスは 0 なので、&k の値はサイズです (注: 印刷時に整数が必要なので、int への強制変換があります)。これにより、必要なサイズを見つけることができます。 さて、私は誤って offsetof() 関数の機能の説明を終えてしまいました:

#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)

今度は見た目が良くなりましたね?(なぜ最下層でこうなっているのかはまだわかりません…ただ、このように動作するということはわかります)、offsetof() の機能は、私たちが夢見ているサイズを見つけて、それを size_t の形式で返すことです(size_t: 符号なし整数)。

2. カーネルプログラミングの厳密さ

#define container_of(ptr, type, member) ({ \         
const typeof( ((type *)0)->member ) *__mptr = (ptr); \         
(type *)( (char *)__mptr - offsetof(type,member) );})

ここでは 2 行目だけを見ます。

const typeof( ((type *)0)->member ) *__mptr = (ptr);  

その機能は何ですか? 実際のところ、あまり効果はありません (批判しないでください、私の話を最後まで言わせてください) が、形式的には _mptr = ptr なので、なぜ同じ変数を定義する必要があるのでしょうか? ? ? 実はこれがカーネル開発者のすごいところです。使用時に開発者が入力したパラメータに問題がある場合、つまり ptr とメンバーの型が一致しない場合、コンパイル時に警告が出ますが、改行を削除すると警告は出なくなり、この警告はまさに必要です (エラーがどこにあるか分からないままエラーを防ぐため)。 。 。これは厳密ですよね?

typeof( ((type *)0)->メンバー )

その機能はメンバーのタイプを取得することだけであり、それ以上のものではありません。これで基本的に終わりです

3. 結論

container_of(ptr, type, member) 関数の実装は、次の 2 つの部分で構成されます。

1. ptrとメンバーが同じ型であるかどうかを判定する

2. サイズを計算します。構造体の開始アドレス = (type *)((char *)ptr - size) (注: 構造体ポインタへの強制変換)

これで、container_of() の関数は、構造体変数のメンバーのアドレスを通じて構造体変数の最初のアドレスを見つけることだということがわかりました。

container_of(ptr,type,member)、ここで、ptr、type、member はそれぞれポインター、型、メンバーを表します。

Linux カーネル プログラミング コンテナの of() 関数に関するこの記事はこれで終わりです。Linux コンテナの of() 関数に関するその他の関連コンテンツについては、123WORDPRESS.COM の以前の記事を検索するか、次の関連記事を引き続き参照してください。今後とも 123WORDPRESS.COM を応援していただければ幸いです。

以下もご興味があるかもしれません:
  • LinuxカーネルマクロContainer_Ofの詳細な説明
  • C言語コンテナof()関数のケース詳細説明
  • Linuxカーネルマクロcontainer_ofの詳細な分析
  • Linuxカーネルのcontainer_of関数の詳細な説明
  • C言語マクロ関数コンテナof()の紹介

<<:  最適なウェブページ幅とその互換性のある実装方法

>>:  html-cssタグのスタイル設定が機能しない2つの理由

推薦する

複数のサーバーにNginxリバースプロキシを実装する方法

Nginx は複数のサーバーをリバース プロキシします。つまり、nginx に異なるリクエストを送信...

jQueryで大画面スクロール再生効果を実現

この記事では、大画面スクロール効果を実現するためのjQueryの具体的なコードを参考までに紹介します...

Alibaba Cloud Ubuntu 16.04 が IPSec サービスを構築

IPSec の概要IPSec (インターネット プロトコル セキュリティ): ネットワーク層と適用さ...

LinuxでMySQLのリモートアクセス権を有効にし、ファイアウォールでポート3306を開きます。

mysqlのリモートアクセス権を有効にするデフォルトでは、MySQL ユーザーにはリモート アクセ...

CSS @font-face パフォーマンス最適化の詳細な理解

この記事では主に、フォント読み込みの最適化に関する一般的な戦略を紹介します。内容の大部分は参考資料と...

Alibaba Cloud SSHリモート接続がしばらくすると切断される問題を解決

問題の再現Alibaba Cloud Server は、Finalshell リモート接続を使用して...

JavaScript ループトラバーサルの 24 種類のメソッドをすべてご存知ですか?

目次序文1. 配列走査法1. 各() 2. マップ() 3. 〜のために4. フィルター() 5. ...

Web フォントの読み込みを最適化する方法をご存知ですか?

タイトル通りです!一般的に使用される font-family はブラウザの組み込みフォントを読み込み...

有名なウェブサイトのロゴに使われている25種類のフォントのコレクション

この記事では、25 の有名な Web サイト (Google、Yahoo、Twitter、Digg ...

データバインディングとリストデータの表示にはVue3を使用する

目次1. Vue2との比較1. Vue3の新機能2. Vue2とVue3の応答原理の比較3. 配列の...

時間範囲効果を実現するためのJavaScript

この記事では、時間範囲効果を実現するためのJavaScriptの具体的なコードを参考までに紹介します...

Ubuntu でホームディレクトリを新しいパーティションに移行する詳細なチュートリアル

ユーザーのホーム ディレクトリがどんどん大きくなってきたら、ホーム ディレクトリを新しいパーティショ...

Win7 システムでの MySQL 5.7.11 の詳細なインストール チュートリアル

オペレーティング システム: Win7 64 ビット Ultimate Edition MySQL ...

画像を読み込むための JavaScript キャンバス

この記事では、画像を読み込むためのJavaScriptキャンバスの具体的なコードを参考までに紹介しま...

Reactで例外を適切にキャプチャする方法

目次序文エラー境界エラー境界を超えてトライ/キャッチwindow.onerror、エラーイベント未処...