source: azure_iot_hub/trunk/asp3_dcre/tinet/netinet6/nd6.c@ 388

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

Azure IoT Hub Device C SDK を使ったサンプルの追加

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