バックナンバーはこちら。
https://www.simulationroom999.com/blog/In-vehicle-network-backnumber/
はじめに
IPフラグメントの受信フレーム順違いに対応するための設計手法。
登場人物
博識フクロウのフクさん
data:image/s3,"s3://crabby-images/373a5/373a5af068fed54e9e584c2fa60af90b1d08e098" alt=""
イラストACにて公開の「kino_k」さんのイラストを使用しています。
https://www.ac-illust.com/main/profile.php?id=iKciwKA9&area=1
エンジニア歴8年の太郎くん
data:image/s3,"s3://crabby-images/ebc9c/ebc9ca6dafb1ae82606cceaa3c40e0f2d33a96fd" alt=""
イラストACにて公開の「しのみ」さんのイラストを使用しています。
https://www.ac-illust.com/main/profile.php?id=uCKphAW2&area=1
IPフラグメントの順番入れ替えの判定をどこでやっているのか?
data:image/s3,"s3://crabby-images/ebc9c/ebc9ca6dafb1ae82606cceaa3c40e0f2d33a96fd" alt="太郎くん"
じゃー、IPフラグメントの順番違いをどう対応してるのか説明よろしくー。
data:image/s3,"s3://crabby-images/e3d65/e3d65e93595459b9f1054227fd7a788442a20b1b" alt="フクさん"
はいよー。
data:image/s3,"s3://crabby-images/373a5/373a5af068fed54e9e584c2fa60af90b1d08e098" alt="フクさん"
まずは、全部のフレームが揃ったかどうかチェックしているのは
ip_reass_chain_frag_into_datagram_and_validate
って関数になる。
core/ipv4/ip4_frag.c
で定義されてる関数になるね。
data:image/s3,"s3://crabby-images/294f2/294f2f45eed21736006281f54fe58d6f81d99826" alt="太郎くん"
ip4_frag.c?
そのまんまな名前のソースコードがあったとは・・・。
ip_reass_chain_frag_into_datagram_and_validateの中の完了判定部分
data:image/s3,"s3://crabby-images/373a5/373a5af068fed54e9e584c2fa60af90b1d08e098" alt="フクさん"
さらに、ip_reass_chain_frag_into_datagram_and_validateの中で完了判定をしている箇所がある。
そこだけのコードを切り抜いてきた。
/* At this point, the validation part begins: */
/* If we already received the last fragment */
if (is_last || ((ipr->flags & IP_REASS_FLAG_LASTFRAG) != 0)) {
/* and had no holes so far */
if (valid) {
/* then check if the rest of the fragments is here */
/* Check if the queue starts with the first datagram */
if ((ipr->p == NULL) || (((struct ip_reass_helper *)ipr->p->payload)->start != 0)) {
valid = 0;
} else {
/* and check that there are no holes after this datagram */
iprh_prev = iprh;
q = iprh->next_pbuf;
while (q != NULL) {
iprh = (struct ip_reass_helper *)q->payload;
if (iprh_prev->end != iprh->start) {
valid = 0;
break;
}
iprh_prev = iprh;
q = iprh->next_pbuf;
}
/* if still valid, all fragments are received
* (because to the MF==0 already arrived */
if (valid) {
LWIP_ASSERT("sanity check", ipr->p != NULL);
LWIP_ASSERT("sanity check",
((struct ip_reass_helper *)ipr->p->payload) != iprh);
LWIP_ASSERT("validate_datagram:next_pbuf!=NULL",
iprh->next_pbuf == NULL);
}
}
}
/* If valid is 0 here, there are some fragments missing in the middle
* (since MF == 0 has already arrived). Such datagrams simply time out if
* no more fragments are received... */
return valid ? IP_REASS_VALIDATE_TELEGRAM_FINISHED : IP_REASS_VALIDATE_PBUF_QUEUED;
}
完了判定部分解説
data:image/s3,"s3://crabby-images/57173/57173c22a9926dc375e9aba2a2701ee383ee12f9" alt="太郎くん"
うーん・・・。
何やってるかわからん!!
data:image/s3,"s3://crabby-images/373a5/373a5af068fed54e9e584c2fa60af90b1d08e098" alt="フクさん"
簡単に説明すると、以下の仕組みになってる。
- 各EthernetFrameは連結リストで連結されている。
- 各EthernetFrameはIPパケット上の開始位置と終了位置を保持している。
- 各EthernetFrameは開始位置順にソートされている。
- 各EthernetFrameの終了位置は次のEthernetFrameの開始位置のはずである。
data:image/s3,"s3://crabby-images/57173/57173c22a9926dc375e9aba2a2701ee383ee12f9" alt="太郎くん"
んー?
うーん?
data:image/s3,"s3://crabby-images/e3d65/e3d65e93595459b9f1054227fd7a788442a20b1b" alt="フクさん"
(わからんか・・・。)
data:image/s3,"s3://crabby-images/373a5/373a5af068fed54e9e584c2fa60af90b1d08e098" alt="フクさん"
図解するとこうだな。
data:image/s3,"s3://crabby-images/f3dc3/f3dc3366b34d54b66c838d18b54e8581c11395ce" alt=""
data:image/s3,"s3://crabby-images/0f4a3/0f4a3412b4e6b4c485d018cd2f8825371049af7c" alt="太郎くん"
おー!
そういう関係性かー!
data:image/s3,"s3://crabby-images/8b2a5/8b2a54ca7b298ef273ed6115e3d80a8a8fb15814" alt="太郎くん"
そういえば、連結リストって前にも出てきたね。
data:image/s3,"s3://crabby-images/294f2/294f2f45eed21736006281f54fe58d6f81d99826" alt="太郎くん"
あれ?
Ethnertフレーム自体が連結リストだったと思うのだけど、
今回のEthnertフレームがさらに連結されてる???
data:image/s3,"s3://crabby-images/fbd59/fbd59e10d2d275de81bc55e14a87b4db7952116c" alt="フクさん"
そうそう。
そこが話がややこしくなってるところで、
構造上は連結リストを連結リストで連結している。
data:image/s3,"s3://crabby-images/22248/222487c6a6bec90f406f108a5f7a982a5a014d15" alt="太郎くん"
また、ややこしいことに・・・。
data:image/s3,"s3://crabby-images/e3d65/e3d65e93595459b9f1054227fd7a788442a20b1b" alt="フクさん"
まぁEthernet層の話はIP層には直接関係ないんで、
一旦気にしないって方が良いかもねー。
data:image/s3,"s3://crabby-images/ebc9c/ebc9ca6dafb1ae82606cceaa3c40e0f2d33a96fd" alt="太郎くん"
とりあえず、いろいろ工夫されてるという事実はわかった。
まとめ
data:image/s3,"s3://crabby-images/373a5/373a5af068fed54e9e584c2fa60af90b1d08e098" alt="フクさん"
まとめだよ。
- IPフラグメントの順番入れ替えの判定のソースコードと該当関数説明。
- Ethnertフレームを連結リストで管理。
- 連結は常にソートされた状態。
- 開始位置と完了位置を持ってパケットの完全性を評価している。
使用したコードはこちら。
バックナンバーはこちら。
コメント