source: EcnlProtoTool/trunk/asp3_dcre/tinet/netinet6/ip6_output.c@ 270

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

mruby版ECNLプロトタイピング・ツールを追加

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
  • Property svn:mime-type set to text/x-csrc
File size: 12.7 KB
Line 
1/*
2 * TINET (TCP/IP Protocol Stack)
3 *
4 * Copyright (C) 2001-2009 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: ip6_output.c 270 2017-02-09 04:03:47Z coas-nagasima $
44 */
45
46/* $FreeBSD: src/sys/netinet6/ip6_output.c,v 1.43 2002/10/31 19:45:48 ume Exp $ */
47/* $KAME: ip6_output.c,v 1.279 2002/01/26 06:12:30 jinmei 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 * Copyright (c) 1982, 1986, 1988, 1990, 1993
80 * The Regents of the University of California. All rights reserved.
81 *
82 * Redistribution and use in source and binary forms, with or without
83 * modification, are permitted provided that the following conditions
84 * are met:
85 * 1. Redistributions of source code must retain the above copyright
86 * notice, this list of conditions and the following disclaimer.
87 * 2. Redistributions in binary form must reproduce the above copyright
88 * notice, this list of conditions and the following disclaimer in the
89 * documentation and/or other materials provided with the distribution.
90 * 3. All advertising materials mentioning features or use of this software
91 * must display the following acknowledgement:
92 * This product includes software developed by the University of
93 * California, Berkeley and its contributors.
94 * 4. Neither the name of the University nor the names of its contributors
95 * may be used to endorse or promote products derived from this software
96 * without specific prior written permission.
97 *
98 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
99 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
100 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
101 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
102 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
103 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
104 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
105 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
106 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
107 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
108 * SUCH DAMAGE.
109 *
110 * @(#)ip_output.c 8.3 (Berkeley) 1/21/94
111 */
112
113#include <string.h>
114
115#ifdef TARGET_KERNEL_ASP
116
117#include <kernel.h>
118#include <sil.h>
119
120#endif /* of #ifdef TARGET_KERNEL_ASP */
121
122#ifdef TARGET_KERNEL_JSP
123
124#include <s_services.h>
125#include <t_services.h>
126
127#endif /* of #ifdef TARGET_KERNEL_JSP */
128
129#include <tinet_defs.h>
130#include <tinet_config.h>
131
132#include <net/if.h>
133#include <net/if_loop.h>
134#include <net/if_ppp.h>
135#include <net/ethernet.h>
136#include <net/if_arp.h>
137#include <net/ppp_ipcp.h>
138#include <net/net.h>
139#include <net/net_buf.h>
140#include <net/net_count.h>
141
142#include <netinet/in.h>
143#include <netinet6/in6.h>
144#include <netinet6/in6_var.h>
145#include <netinet6/nd6.h>
146#include <netinet/ip6.h>
147#include <netinet6/ip6_var.h>
148#include <netinet/icmp6.h>
149
150#include <net/if6_var.h>
151
152#ifdef SUPPORT_INET6
153
154/*
155 * å…
156¨åŸŸå¤‰æ•°
157 */
158
159/* リンク MTU */
160
161uint32_t linkmtu = IF_MTU;
162
163/*
164 * 局所変数
165 */
166
167#ifdef IP6_CFG_FRAGMENT
168
169static uint32_t ip6_id;
170
171#endif /* of #ifdef IP6_CFG_FRAGMENT */
172
173/*
174 * ip6_output -- IPv6 の出力関数
175 */
176
177ER
178ip6_output (T_NET_BUF *output, uint16_t flags, TMO tmout)
179{
180 T_IP6_HDR *ip6h;
181 T_IN6_ADDR *gw;
182 T_IFNET *ifp = IF_GET_IFNET();
183 ER error = E_OK;
184
185#ifdef IP6_CFG_FRAGMENT
186
187 T_IP6_HDR *fip6h;
188 T_IP6_FRAG_HDR *fip6f;
189 T_NET_BUF *frag;
190 uint32_t mtu, id;
191 uint16_t plen, foff, flen;
192 uint8_t nextproto;
193 bool_t alwaysfrag = false;
194
195#endif /* of #ifdef IP6_CFG_FRAGMENT */
196
197 ip6h = GET_IP6_HDR(output);
198
199 NET_COUNT_IP6(net_count_ip6[NC_IP6_OUT_OCTETS],
200 GET_IP_HDR_SIZE(ip6h) + GET_IP_SDU_SIZE(ip6h));
201 NET_COUNT_IP6(net_count_ip6[NC_IP6_OUT_PACKETS], 1);
202 NET_COUNT_MIB(in6_ifstat.ipv6IfStatsOutRequests, 1);
203
204 /*
205 * 重複アドレス検出要請以外には、送信å…
206ƒã‚¢ãƒ‰ãƒ¬ã‚¹ã«
207 * 無指定を指定できない。
208 */
209 if (IN6_IS_ADDR_UNSPECIFIED(&ip6h->src) && (flags & IPV6_OUT_FLG_DAD) == 0) {
210 error = E_PAR;
211 goto drop;
212 }
213
214 /* 送信å…
215ƒã‚¢ãƒ‰ãƒ¬ã‚¹ã«ã¯ãƒžãƒ«ãƒã‚­ãƒ£ã‚¹ãƒˆã‚¢ãƒ‰ãƒ¬ã‚¹ã‚’指定できない。*/
216 if (IN6_IS_ADDR_MULTICAST(&ip6h->src)) {
217 error = E_PAR;
218 goto drop;
219 }
220
221 /*
222 * 経路選択を行う。
223 */
224 if ((gw = in6_rtalloc(ifp, &ip6h->dst)) == NULL) {
225 error = E_PAR;
226 goto drop;
227 }
228
229 if (!IN6_IS_ADDR_MULTICAST(&ip6h->dst)) {
230 /* ユニキャストアドレスの処理 */
231 }
232 else {
233 /* マルチキャストアドレスの処理 */
234 NET_COUNT_MIB(in6_ifstat.ipv6IfStatsOutMcastPkts, 1);
235 if (flags & IPV6_OUT_FLG_HOP_LIMIT)
236 ip6h->hlim = IPV6_OUT_GET_HOP_LIMIT(flags);
237 else
238 ip6h->hlim = IPV6_DEFAULT_MULTICAST_HOPS;
239 }
240
241 /* Traffic Class と Flow Label をサポートしないので 0 に設定する。*/
242 ip6h->vcf = htonl(IP6_MAKE_VCF(IPV6_VERSION, 0));
243
244#ifdef IP6_CFG_FRAGMENT
245
246
247#if NUM_IN6_HOSTCACHE_ENTRY > 0
248
249 mtu = in6_hostcache_getmtu(&ip6h->dst);
250
251 if (0 < mtu && mtu <= IPV6_MMTU) {
252
253 /*
254 * Path MTU が登録されている場合、
255 * MTU が IPv6 MMTU(1280 オクテット)以下の場合は、
256 * MTU を IPv6 MMTU に設定し分割して送信する。
257 */
258 mtu = IPV6_MMTU;
259 alwaysfrag = true;
260 }
261 else if (mtu == 0)
262 mtu = linkmtu;
263
264#else /* of #if NUM_IN6_HOSTCACHE_ENTRY > 0 */
265
266 mtu = linkmtu;
267
268#endif /* of #if NUM_IN6_HOSTCACHE_ENTRY > 0 */
269
270 plen = ntohs(ip6h->plen);
271 if (plen + IP6_HDR_SIZE <= mtu && !alwaysfrag) {
272
273 /* 分割する必
274要がないときの処理 */
275 if ((error = nd6_output(ifp, output, gw, NULL, tmout)) != E_OK) {
276 NET_COUNT_IP6(net_count_ip6[NC_IP6_OUT_ERR_PACKETS], 1);
277 NET_COUNT_MIB(in6_ifstat.ipv6IfStatsOutDiscards, 1);
278 }
279 }
280 else {
281
282 /* 断片 ID を設定する。*/
283 id = ip6_id ++;
284
285 /* MTU が IPv6 の最大パケットサイズを超
286えないようにする。*/
287 if (mtu > IPV6_MAXPACKET)
288 mtu = IPV6_MAXPACKET;
289
290 /*
291 * MTU から、分割不能部分(TINET-1.4 では、分割ヘッダ以外の
292 * 拡張ヘッダの出力は未実装
293のため 0 オクテット)
294 * と断片ヘッダのサイズを引いたとき、8 オクテット以上なければ
295 * 分割して送信できないためエラーにする。
296 */
297 if (((mtu - sizeof(T_IP6_FRAG_HDR)) & ~7) < 8) {
298 error = E_PAR;
299 goto drop;
300 }
301
302 /*
303 * IPv6 ヘッダの next フィールドに設定する値は、断片ヘッダ。
304 * 断片ヘッダの next フィールドに設定する値は、
305 * å…
306ƒã®ãƒ‡ãƒ¼ã‚¿ã‚°ãƒ©ãƒ ã® next フィールドの値。
307 */
308 nextproto = ip6h->next;
309 ip6h->next = IPPROTO_FRAGMENT;
310
311 /* 分割して送信する。*/
312 NET_COUNT_IP6(net_count_ip6[NC_IP6_FRAG_OUT], 1);
313 for (foff = 0; foff < plen; foff += flen) {
314
315 /* 断片の長さを計算し、8 オクテット境界に調整する。*/
316 if (foff + (mtu - (IP6_HDR_SIZE + sizeof(T_IP6_FRAG_HDR))) < plen)
317 flen = (mtu - (IP6_HDR_SIZE + sizeof(T_IP6_FRAG_HDR))) >> 3 << 3;
318 else
319 flen = plen - foff;
320
321 /* ネットワークバッファを獲得する。*/
322 if (tget_net_buf(&frag, flen + IF_IP6_HDR_SIZE + sizeof(T_IP6_FRAG_HDR), TMO_IP6_FRAG_GET_NET_BUF) == E_OK) {
323
324 /* IPv6 ヘッダをコピーする。*/
325 fip6h = GET_IP6_HDR(frag);
326 *fip6h = *ip6h;
327
328 /* IPv6 ヘッダを埋める。*/
329 fip6h->plen = htons(flen + sizeof(T_IP6_FRAG_HDR));
330
331 /* 断片ヘッダを埋める。*/
332 fip6f = (T_IP6_FRAG_HDR *)GET_IP6_NEXT_HDR(frag);
333 fip6f->off_flag = htons((uint16_t)(foff & ~7));
334 if (foff + (mtu - (IP6_HDR_SIZE + sizeof(T_IP6_FRAG_HDR))) < plen)
335 fip6f->off_flag |= IP6F_MORE_FRAG;
336 fip6f->reserved = 0;
337 fip6f->ident = htonl(id);
338 fip6f->next = nextproto;
339
340 /* SDU をコピーする。*/
341 memcpy((uint8_t*)(fip6f + 1), GET_IP6_NEXT_HDR(output) + foff, flen);
342 NET_COUNT_IP6(net_count_ip6[NC_IP6_FRAG_OUT_FRAGS], 1);
343 NET_COUNT_MIB(in6_ifstat.ipv6IfStatsOutFragCreates, 1);
344
345 /* 送信する。*/
346 if ((error = nd6_output(ifp, frag, gw, NULL, tmout)) != E_OK) {
347 NET_COUNT_MIB(in6_ifstat.ipv6IfStatsOutFragFails, 1);
348 goto drop;
349 }
350 }
351 else {
352 /* ネットワークバッファが獲得できなければ、送信をあきらめる。*/
353 error = E_NOMEM;
354 goto drop;
355 }
356 }
357
358 /* IF でネットワークバッファを開放しないフラグをチェックする。*/
359 if ((output->flags & NB_FLG_NOREL_IFOUT) == 0) {
360 syscall(rel_net_buf(output));
361 }
362 else
363 output->flags &= (uint8_t)~NB_FLG_NOREL_IFOUT;
364
365 NET_COUNT_MIB(in6_ifstat.ipv6IfStatsOutFragOKs, 1);
366 }
367
368#else /* #ifdef IP6_CFG_FRAGMENT */
369
370 if ((error = nd6_output(ifp, output, gw, NULL, tmout)) != E_OK) {
371 NET_COUNT_IP6(net_count_ip6[NC_IP6_OUT_ERR_PACKETS], 1);
372 NET_COUNT_MIB(in6_ifstat.ipv6IfStatsOutDiscards, 1);
373 }
374
375#endif /* #ifdef IP6_CFG_FRAGMENT */
376
377 return error;
378
379drop:
380 /* IF でネットワークバッファを開放しないフラグをチェックする。*/
381 if ((output->flags & NB_FLG_NOREL_IFOUT) == 0) {
382 syscall(rel_net_buf(output));
383 }
384 else
385 output->flags &= (uint8_t)~NB_FLG_NOREL_IFOUT;
386
387 NET_COUNT_IP6(net_count_ip6[NC_IP6_OUT_ERR_PACKETS], 1);
388 NET_COUNT_MIB(in6_ifstat.ipv6IfStatsOutDiscards, 1);
389 return error;
390 }
391
392#endif /* of #ifdef SUPPORT_INET6 */
Note: See TracBrowser for help on using the repository browser.