source: asp3_tinet_ecnl_arm/trunk/asp3_dcre/tinet/netinet6/nd6.c@ 364

Last change on this file since 364 was 364, checked in by coas-nagasima, 5 years ago

TINETとSocket APIなどを更新

  • Property svn:eol-style set to native
  • Property svn:mime-type set to text/x-csrc;charset=UTF-8
File size: 19.6 KB
Line 
1/*
2 * TINET (TCP/IP Protocol Stack)
3 *
4 * Copyright (C) 2001-2017 by Dep. of Computer Science and Engineering
5 * Tomakomai National College of Technology, JAPAN
6 *
7 * 上記著作権者は,以下の (1)~(4) の条件か,Free Software Foundation
8 * によって公表されている GNU General Public License の Version 2 に記
9 * 述されている条件を満たす場合に限り,本ソフトウェア(本ソフトウェア
10 * を改変したものを含む.以下同じ)を使用・複製・改変・再配布(以下,
11 * 利用と呼ぶ)することを無償で許諾する.
12 * (1) 本ソフトウェアをソースコードの形で利用する場合には,上記の著作
13 * 権表示,この利用条件および下記の無保証規定が,そのままの形でソー
14 * スコード中に含まれていること.
15 * (2) 本ソフトウェアを,ライブラリ形式など,他のソフトウェア開発に使
16 * 用できる形で再配布する場合には,再配布に伴うドキュメント(利用
17 * 者マニュアルなど)に,上記の著作権表示,この利用条件および下記
18 * の無保証規定を掲載すること.
19 * (3) 本ソフトウェアを,機器に組み込むなど,他のソフトウェア開発に使
20 * 用できない形で再配布する場合には,次の条件を満たすこと.
21 * (a) 再配布に伴うドキュメント(利用者マニュアルなど)に,上記の著
22 * 作権表示,この利用条件および下記の無保証規定を掲載すること.
23 * (4) 本ソフトウェアの利用により直接的または間接的に生じるいかなる損
24 * 害からも,上記著作権者およびTOPPERSプロジェクトを免責すること.
25 *
26 * 本ソフトウェアは,無保証で提供されているものである.上記著作権者お
27 * よびTOPPERSプロジェクトは,本ソフトウェアに関して,その適用可能性も
28 * 含めて,いかなる保証も行わない.また,本ソフトウェアの利用により直
29 * 接的または間接的に生じたいかなる損害に関しても,その責任を負わない.
30 *
31 * @(#) $Id$
32 */
33
34/* $FreeBSD: src/sys/netinet6/nd6.c,v 1.20 2002/08/02 20:49:14 rwatson Exp $ */
35/* $KAME: nd6.c,v 1.144 2001/05/24 07:44:00 itojun Exp $ */
36
37/*
38 * Copyright (C) 1995, 1996, 1997, and 1998 WIDE Project.
39 * All rights reserved.
40 *
41 * Redistribution and use in source and binary forms, with or without
42 * modification, are permitted provided that the following conditions
43 * are met:
44 * 1. Redistributions of source code must retain the above copyright
45 * notice, this list of conditions and the following disclaimer.
46 * 2. Redistributions in binary form must reproduce the above copyright
47 * notice, this list of conditions and the following disclaimer in the
48 * documentation and/or other materials provided with the distribution.
49 * 3. Neither the name of the project nor the names of its contributors
50 * may be used to endorse or promote products derived from this software
51 * without specific prior written permission.
52 *
53 * THIS SOFTWARE IS PROVIDED BY THE PROJECT AND CONTRIBUTORS ``AS IS'' AND
54 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
55 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
56 * ARE DISCLAIMED. IN NO EVENT SHALL THE PROJECT OR CONTRIBUTORS BE LIABLE
57 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
58 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
59 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
60 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
61 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
62 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
63 * SUCH DAMAGE.
64 */
65
66/*
67 * XXX
68 * KAME 970409 note:
69 * BSD/OS version heavily modifies this code, related to llinfo.
70 * Since we don't have BSD/OS version of net/route.c in our hand,
71 * I left the code mostly as it was in 970310. -- itojun
72 */
73
74#include <string.h>
75
76#ifdef TARGET_KERNEL_ASP
77
78#include <kernel.h>
79#include <sil.h>
80#include "kernel_cfg.h"
81
82#endif /* of #ifdef TARGET_KERNEL_ASP */
83
84#ifdef TARGET_KERNEL_JSP
85
86#include <s_services.h>
87#include <t_services.h>
88#include "kernel_id.h"
89
90#endif /* of #ifdef TARGET_KERNEL_JSP */
91
92#include <tinet_defs.h>
93#include <tinet_config.h>
94
95#include <net/if.h>
96#include <net/if_loop.h>
97#include <net/if_ppp.h>
98#include <net/ethernet.h>
99#include <net/if_arp.h>
100#include <net/ppp.h>
101#include <net/ppp_ipcp.h>
102#include <net/net.h>
103#include <net/net_endian.h>
104#include <net/net_var.h>
105#include <net/net_buf.h>
106#include <net/net_timer.h>
107#include <net/net_count.h>
108
109#include <netinet/in.h>
110#include <netinet/in_var.h>
111#include <netinet/ip.h>
112#include <netinet/ip_var.h>
113#include <netinet/ip_icmp.h>
114
115#include <netinet6/nd6.h>
116
117#include <net/if_var.h>
118
119#ifdef _IP6_CFG
120
121/*
122 * 近隣キャッシュ
123 */
124
125static T_LLINFO_ND6 nd6_cache[NUM_ND6_CACHE_ENTRY];
126
127/*
128 * nd6_free -- ルータを削除する。
129 */
130
131static void
132nd6_free (T_LLINFO_ND6 *ln)
133{
134 T_DEF_ROUTER *dr;
135
136 dr = nd6_defrtrlist_lookup(&ln->addr);
137 if (dr != NULL) {
138 ln->state = ND6_LLINFO_INCOMPLETE;
139
140 /* プレフィックス・リストからルータを削除する。*/
141 nd6_defrtrlist_del(dr);
142 }
143 }
144
145/*
146 * nd6_timer -- 近隣探索管理タイマー
147 */
148
149void
150nd6_timer (void)
151{
152 T_IFNET *ifp = IF_GET_IFNET();
153 T_LLINFO_ND6 *ln;
154 SYSTIM now;
155 int_t ix;
156
157 /* 近隣キャッシュの処理 */
158 syscall(wai_sem(SEM_ND6_CACHE));
159 syscall(get_tim(&now));
160 for (ix = NUM_ND6_CACHE_ENTRY; ix -- > 0; ) {
161 ln = &nd6_cache[ix];
162 if (ln->state >= ND6_LLINFO_NO_STATE && (int32_t)(ln->expire - now) < 0) {
163 /* ln->expire < now */
164
165 switch (ln->state) {
166
167 case ND6_LLINFO_INCOMPLETE: /* 未解決 */
168 if (ln->asked < ND6_MAX_MCAST_QUERY) {
169 /*
170 * マルチキャストアドレス要請の再送回数以下の場合の処理
171 */
172 ln->asked ++;
173 ln->expire = now + ND6_RETRANS_TIME; /* 近隣探索送信間隔 */
174 nd6_ns_output(ifp, &in6_addr_unspecified, &ln->addr, ln, false);
175 }
176 else {
177 if (ln->hold != NULL) {
178 /*
179 * 最終的には、保留されているネットワークバッファが
180 * あれば、ICMP6 ERROR を自分自身に返したあと、
181 * ネットワークバッファを開放する。
182 * しかし、現在、icmp6_error の実装が不完全のため、
183 * ネットワークバッファの開放のみ行う。
184 */
185#if 0 /* 保留 */
186 icmp6_error(ln->hold, ICMP6_DST_UNREACH,
187 ICMP6_DST_UNREACH_ADDR, 0);
188#endif /* #if 0 */
189 syscall(rel_net_buf(ln->hold));
190 ln->hold = NULL;
191 }
192 nd6_free(ln);
193 memset((void*)ln, 0, sizeof(*ln));
194 }
195 break;
196
197 case ND6_LLINFO_REACHABLE: /* 到達可能 */
198 if (ln->expire) {
199 ln->expire = now + ND6_GCOLLECTION_TIME;
200 ln->state = ND6_LLINFO_STALE;
201 }
202 break;
203
204 case ND6_LLINFO_STALE: /* 到達可能性は未確認 */
205 if (ln->expire) {
206 if (ln->hold != NULL) {
207 syscall(rel_net_buf(ln->hold));
208 ln->hold = NULL;
209 }
210 nd6_free(ln);
211 memset((void*)ln, 0, sizeof(*ln));
212 }
213 break;
214
215 case ND6_LLINFO_DELAY: /* 到達可能性の確認待機 */
216 ln->asked = 1;
217 ln->expire = now + ND6_RETRANS_TIME; /* 近隣探索送信間隔 */
218 ln->state = ND6_LLINFO_PROBE;
219 nd6_ns_output(ifp, &ln->addr, &ln->addr, ln, false);
220 break;
221
222 case ND6_LLINFO_PROBE: /* 到達可能性を確認中 */
223 if (ln->asked < ND6_MAX_UCAST_QUERY) {
224 /*
225 * ユニキャストアドレス要請の再送回数以下の場合の処理
226 */
227 ln->asked ++;
228 ln->expire += ND6_RETRANS_TIME; /* 近隣探索送信間隔 */
229 nd6_ns_output(ifp, &ln->addr, &ln->addr, ln, false);
230 }
231 else {
232 if (ln->hold != NULL) {
233 syscall(rel_net_buf(ln->hold));
234 ln->hold = NULL;
235 }
236 nd6_free(ln);
237 memset((void*)ln, 0, sizeof(*ln));
238 }
239 break;
240
241 }
242 }
243 }
244 syscall(sig_sem(SEM_ND6_CACHE));
245
246 /* ディフォルトルータ・リストの処理 */
247 nd6_defrtrlist_timer();
248
249 /* プレフィックス・リストの処理 */
250 nd6_prelist_timer();
251
252 /* ネットワークインタフェースのアドレスの処理 */
253 in6_ifaddr_timer(ifp);
254
255 /*
256 * 到達可能状態の有効時間の更新
257 */
258 if ((int32_t)(nd6_recalc_reachtm_interval - now) < 0) {
259 /* nd6_recalc_reachtm_interval < now */
260 nd6_recalc_reachtm_interval = now + ND6_RECALC_REACHTM_INTERVAL;
261 nd6_reachable_time = ND6_CALC_REACH_TIME(nd6_base_reachable_time);
262 }
263 timeout((callout_func)nd6_timer, NULL, ND6_TIMER_TMO);
264 }
265
266/*
267 * nd6_get_cache -- 近隣キャッシュを獲得する。
268 */
269
270const T_LLINFO_ND6 *
271nd6_get_cache (void)
272{
273 return nd6_cache;
274 }
275
276/*
277 * nd6_ifattach -- 近隣探索の初期設定を行う。
278 */
279
280void
281nd6_ifattach (T_IFNET *ifp)
282{
283 }
284
285/*
286 * nd6_output_hold -- 近隣探索キャッシュに保留されている
287 * アドレス解決待ちのデータグラムがあれば送信する。
288 *
289 * 注意: セマフォ SEM_ND6_CACHE によりロック状態で呼出すこと。
290 */
291
292ER
293nd6_output_hold (T_IFNET *ifp, T_LLINFO_ND6 *ln)
294{
295 SYSTIM now;
296
297 /* アドレス解決待ちのデータグラムがあれば送信する。*/
298 if (ln->hold) {
299
300 /* タイムアウトの処理 */
301 if (ln->tmout != TMO_FEVR) {
302
303 /*
304 * タイムアウトの指定が永久でなく、時間を過ぎていれば、
305 * 出力しないで終了する。
306 */
307 syscall(get_tim(&now));
308 if ((int32_t)(ln->tmout - now) < 0) {
309 /* ln->tmout < now */
310 syscall(rel_net_buf(ln->hold));
311 ln->hold = NULL;
312 return E_TMOUT;
313 }
314 }
315
316 nd6_output(ifp, ln->hold, &ln->addr, ln, TMO_FEVR);
317 ln->hold = NULL;
318 }
319 return E_OK;
320 }
321
322/*
323 * nd6_lookup -- 近隣探索キャッシュを探索する。
324 *
325 * 注意: セマフォ SEM_ND6_CACHE によりロック状態で呼出すこと。
326 *
327 */
328
329T_LLINFO_ND6 *
330nd6_lookup (const T_IN6_ADDR *addr, bool_t create)
331{
332 SYSTIM min = 0xffffffff;
333 int_t ix, fix = -1, mix = -1;
334
335 for (ix = NUM_ND6_CACHE_ENTRY; ix -- > 0; ) {
336 if (nd6_cache[ix].state == 0) {
337 /* 未使用エントリのインデックスを記録する。*/
338 if (fix == -1)
339 fix = ix;
340 }
341 else if (IN6_ARE_ADDR_EQUAL(addr, &nd6_cache[ix].addr)) {
342 return &nd6_cache[ix];
343 }
344 else if ((int32_t)(nd6_cache[ix].expire - min) < 0) { /* nd6_cache[ix].expire < min */
345 /* 有効期限が最短エントリのインデックスを記録する。*/
346 min = nd6_cache[ix].expire;
347 mix = ix;
348 }
349 }
350
351 if (create) {
352 if (fix == -1) {
353 if (nd6_cache[mix].hold != NULL) {
354 syscall(rel_net_buf(nd6_cache[mix].hold));
355 nd6_cache[mix].hold = NULL;
356 }
357 fix = mix;
358 }
359 syscall(get_tim(&nd6_cache[fix].expire));
360 nd6_cache[fix].addr = *addr;
361 nd6_cache[fix].state = ND6_LLINFO_NO_STATE;
362 return &nd6_cache[fix];
363 }
364 else
365 return NULL;
366 }
367
368/*
369 * nd6_cache_lladdr -- 近隣探索キャッシュに登録する。
370 */
371
372T_LLINFO_ND6 *
373nd6_cache_lladdr (T_IFNET *ifp, const T_IN6_ADDR *from,
374 T_IF_ADDR *lladdr, uint8_t type, uint8_t code)
375{
376 T_LLINFO_ND6 *ln;
377 SYSTIM now;
378 bool_t newentry, olladdr, llchange, doupdate;
379 int_t newstate = ND6_LLINFO_INCOMPLETE;
380
381 /* アドレスが無指定なら何もしない。*/
382 if (IN6_IS_ADDR_UNSPECIFIED(from))
383 return NULL;
384
385 /* 近隣キャッシュに登録する。*/
386 syscall(wai_sem(SEM_ND6_CACHE));
387 if ((ln = nd6_lookup(from, false)) == NULL) {
388 ln = nd6_lookup(from, true);
389 newentry = true;
390 }
391 else
392 newentry = false;
393
394 olladdr = ln->state > ND6_LLINFO_INCOMPLETE;
395 if (!olladdr || lladdr == NULL)
396 llchange = false;
397 else if (memcmp(lladdr, &ln->ifaddr, sizeof(T_IF_ADDR)))
398 llchange = true;
399 else
400 llchange = false;
401
402 /*
403 * 状態遷移表
404 *
405 * nd_na_flags
406 * newnetry olladdr lladdr llchange 処理 (L: lladdr を登録する)
407 *
408 * F F N - (1)
409 * F T N - (2)
410 * F F Y - (3) L * -> STALE
411 * F T Y F (4) L
412 * F T Y T (5) L * -> STALE
413 * T - N - (6) * -> NO_STATE
414 * T - Y - (7) L * -> STALE
415 */
416
417 /* データリンク層のアドレスが通知されていれば更新する。*/
418 if (lladdr != NULL) {
419 ln->ifaddr = *(T_IF_ADDR *)lladdr;
420 }
421
422 if (newentry) {
423 if (lladdr != NULL) /* (7) */
424 newstate = ND6_LLINFO_STALE;
425 else /* (6) */
426 newstate = ND6_LLINFO_NO_STATE;
427 doupdate = true;
428 }
429 else {
430 if ((!olladdr && lladdr != NULL) || /* (3) */
431 ( olladdr && lladdr != NULL && llchange)) { /* (5) */
432 newstate = ND6_LLINFO_STALE;
433 doupdate = true;
434 }
435 else /* (1),(2),(4) */
436 doupdate = false;
437 }
438
439 /* 近隣キャッシュエントリを更新する。*/
440 if (doupdate) {
441 ln->state = newstate;
442
443 syscall(get_tim(&now));
444 if (ln->state == ND6_LLINFO_STALE) {
445 ln->expire = now + ND6_GCOLLECTION_TIME;
446
447 /* アドレス解決待ちのデータグラムがあれば送信する。*/
448 nd6_output_hold(ifp, ln);
449 }
450 else if (ln->state == ND6_LLINFO_INCOMPLETE)
451 ln->expire = now;
452 }
453 syscall(sig_sem(SEM_ND6_CACHE));
454
455
456 /* ルータ通知の処理 */
457 switch (type) {
458
459 case ND_NEIGHBOR_SOLICIT:
460 if (newentry)
461 ln->flags &= ~ND6_LLIF_ROUTER;
462 break;
463
464 case ND_REDIRECT:
465 if (code == ND_REDIRECT_ROUTER)
466 ln->flags |= ND6_LLIF_ROUTER;
467 else if (newentry)
468 ln->flags &= ~ND6_LLIF_ROUTER;
469 break;
470
471 case ND_ROUTER_SOLICIT:
472 ln->flags &= ~ND6_LLIF_ROUTER;
473 break;
474
475 case ND_ROUTER_ADVERT:
476 if ((!newentry && (olladdr || lladdr != NULL)) ||
477 ( newentry && lladdr))
478 ln->flags |= ND6_LLIF_ROUTER;
479 break;
480 }
481
482 return ln;
483 }
484
485/*
486 * nd6_is_addr_neighbor -- 指定されたアドレスのノードが同一リンクにあるかチェックする。
487 */
488
489bool_t
490nd6_is_addr_neighbor (T_IFNET *ifp, const T_IN6_ADDR *addr)
491{
492 if (IN6_IS_ADDR_LINKLOCAL(addr))
493 return true;
494
495 if (nd6_onlink_prefix_lookup(addr) != NULL)
496 return true;
497
498 if (nd6_lookup(addr, false) != NULL)
499 return true;
500
501 return false;
502 }
503
504/*
505 * nd6_output -- 近隣探索の出力関数
506 *
507 * 注意: セマフォ SEM_ND6_CACHE によりロック状態で呼出すこと。
508 */
509
510ER
511nd6_output (T_IFNET *ifp, T_NET_BUF *output, const T_IN6_ADDR *dst, T_LLINFO_ND6 *ln, TMO tmout)
512{
513 ER error = E_OK;
514 SYSTIM now;
515
516 /*
517 * あて先アドレスがマルチキャストか、近隣キャッシュが
518 * 不要の場合は直ちに送信する。
519 */
520 if (!(IN6_IS_ADDR_MULTICAST(dst) || !IF_IN6_NEED_CACHE(ifp))) {
521 if (ln == NULL) {
522 /* 近隣キャッシュが未定義の場合 */
523 if (nd6_is_addr_neighbor(ifp, dst)) {
524 /*
525 * 指定されたアドレスのノードが同一リンクにある時は、
526 * 近隣キャッシュを探索し、未定義の場合は登録する。
527 */
528 ln = nd6_lookup(dst, true);
529 }
530 else {
531 error = E_CLS;
532 goto err_ret;
533 }
534 }
535
536 /*
537 * 近隣キャッシュの状態が、データリンク層のアドレスが分かっていても、
538 * 到達可能性が確認されていないときは、到達不能通知を開始する。
539 */
540 if (ln->state == ND6_LLINFO_STALE) {
541 syscall(get_tim(&now));
542 ln->expire = now + ND6_DELAY_TIME;
543 ln->asked = 0;
544 ln->state = ND6_LLINFO_DELAY;
545 }
546
547 /*
548 * 近隣キャッシュの状態が、データリンク層のアドレス未解決の場合は
549 * 近隣探索を行う。
550 */
551 if (ln->state <= ND6_LLINFO_INCOMPLETE) {
552
553 /* タイムアウトの指定がポーリングならタイムアウトで終了する。*/
554 if (tmout == TMO_POL) {
555 error = E_TMOUT;
556 goto err_ret;
557 }
558
559 if (ln->state == ND6_LLINFO_NO_STATE)
560 ln->state = ND6_LLINFO_INCOMPLETE;
561
562 /* すでに、未解決のネットワークバッファがあれば、開放する。*/
563 if (ln->hold != NULL) {
564 syscall(rel_net_buf(ln->hold));
565 ln->hold = NULL;
566 }
567
568 /*
569 * 送信をペンディングする。
570 * IF でネットワークバッファを開放しないフラグが設定されているときは、
571 * 送信をペンディングしない。
572 */
573 if ((output->flags & NB_FLG_NOREL_IFOUT) == 0) {
574 ln->hold = output;
575
576 /* タイムアウトの設定 */
577 if (tmout == TMO_FEVR)
578 ln->tmout = tmout;
579 else {
580 syscall(get_tim(&now));
581 ln->tmout = now + tmout;
582 }
583 }
584 else {
585 output->flags &= ~NB_FLG_NOREL_IFOUT;
586 syscall(get_tim(&now));
587 ln->tmout = now + tmout;
588 }
589
590 if (ln->expire) {
591 if (ln->asked < ND6_MAX_MCAST_QUERY) {
592 syscall(get_tim(&now));
593 if ((int32_t)(ln->expire - now) < 0) {
594 /* ln->expire < now */
595 ln->expire += ND6_RETRANS_TIME; /* 近隣探索送信間隔 */
596 ln->asked ++;
597 nd6_ns_output(ifp, &in6_addr_unspecified, dst, ln, false);
598 }
599 }
600 }
601 return error;
602 }
603 }
604
605 /*
606 * 重複が検出されたインタフェースには出力しない。
607 */
608 if ((ifp->flags & ND6_IFF_IFDISABLED) != 0) {
609 return E_OK;
610 }
611
612 IF_SET_PROTO(output, IF_PROTO_IPV6);
613 error = IF_OUTPUT(output, dst, ln ? & ln->ifaddr : NULL, tmout);
614 return error;
615
616err_ret:
617 syscall(rel_net_buf(output));
618 return error;
619 }
620
621/*
622 * nd6_storelladdr -- 出力するネットワークインタフェースのアドレスを返す。
623 */
624
625ER
626nd6_storelladdr (T_IF_ADDR *out, const T_IN6_ADDR *dst, T_IF_ADDR *ifa)
627{
628 if (IN6_IS_ADDR_MULTICAST(dst)) {
629 IF_IN6_RESOLVE_MULTICAST(out, dst);
630 }
631 else {
632 *out = *ifa;
633 }
634
635 return E_OK;
636 }
637
638/*
639 * nd6_option -- 近隣探索オプションの処理
640 *
641 * 注意: オプションオフセット配列には、オフセット + 8 が設定されている。
642 */
643
644ER
645nd6_options (uint8_t *opt, void *nh, uint_t len)
646{
647 T_ND_OPT_HDR *opth;
648 ER error = E_OK;
649 int_t pi_count = 0; /* プレフィックス情報の個数 */
650
651 opth = (T_ND_OPT_HDR *)nh;
652 memset(opt, 0, ND_OPT_OFF_ARRAY_SIZE);
653
654 while (error == E_OK && ((uint8_t *)opth - (uint8_t *)nh) < (256 - 8) && len > 0) {
655
656 /* オプション長が 0 の時は、不正なオプションとして処理する。*/
657 if (opth->len == 0) {
658 error = E_PAR;
659 break;
660 }
661
662 /* オプションのタイプにより分岐する。*/
663 switch (opth->type) {
664 case ND_OPT_SOURCE_LINKADDR: /* 送信先ネットワークインタフェースのアドレス */
665 case ND_OPT_TARGET_LINKADDR: /* 対象ネットワークインタフェースのアドレス */
666 case ND_OPT_REDIRECTED_HEADER: /* リダイレクト・ヘッダ */
667 case ND_OPT_MTU: /* MTU */
668 if (opt[ND_OPT_OFF_ARRAY_IX(opth->type)]) {
669 /* プレフィックス情報以外の同じオプションが複数指定されている。*/
670 error = E_PAR;
671 }
672 else {
673 opt[ND_OPT_OFF_ARRAY_IX(opth->type)] = (uint8_t *)opth - (uint8_t *)nh + 8;
674 }
675 break;
676 case ND_OPT_PREFIX_INFORMATION: /* プレフィックス情報 */
677 if (!opt[ND_OPT_OFF_ARRAY_IX(ND_OPT_PREFIX_INFORMATION)]) {
678 /*
679 * プレフィックス情報は最初に現れたオフセットのみ記録する。
680 */
681 opt[ND_OPT_OFF_ARRAY_IX(opth->type)] = (uint8_t *)opth - (uint8_t *)nh + 8;
682 }
683 pi_count ++;
684 break;
685 default:
686 /*
687 * 将来定義されるプロトコルへの拡張に対応するため、
688 * 認識できないオプションは単に無視し、処理を続行する。
689 */
690 break;
691 }
692 len -= opth->len << 3; /* オプション長の単位は 8 バイト */
693 opth = (T_ND_OPT_HDR *)((uint8_t *)opth + (opth->len << 3));
694 }
695
696 opt[ND_OPT_OFF_ARRAY_IX(ND_OPT_PREFIX_INFO_COUNT)] = pi_count;
697 return error;
698 }
699
700#endif /* of #ifdef _IP6_CFG */
Note: See TracBrowser for help on using the repository browser.