/* * TINET (TCP/IP Protocol Stack) * * Copyright (C) 2001-2017 by Dep. of Computer Science and Engineering * Tomakomai National College of Technology, JAPAN * * 上記著作権者は,以下の (1)~(4) の条件か,Free Software Foundation * によって公表されている GNU General Public License の Version 2 に記 * 述されている条件を満たす場合に限り,本ソフトウェア(本ソフトウェア * を改変したものを含む.以下同じ)を使用・複製・改変・再配布(以下, * 利用と呼ぶ)することを無償で許諾する. * (1) 本ソフトウェアをソースコードの形で利用する場合には,上記の著作 * 権表示,この利用条件および下記の無保証規定が,そのままの形でソー * スコード中に含まれていること. * (2) 本ソフトウェアを,ライブラリ形式など,他のソフトウェア開発に使 * 用できる形で再配布する場合には,再配布に伴うドキュメント(利用 * 者マニュアルなど)に,上記の著作権表示,この利用条件および下記 * の無保証規定を掲載すること. * (3) 本ソフトウェアを,機器に組み込むなど,他のソフトウェア開発に使 * 用できない形で再配布する場合には,次の条件を満たすこと. * (a) 再配布に伴うドキュメント(利用者マニュアルなど)に,上記の著 * 作権表示,この利用条件および下記の無保証規定を掲載すること. * (4) 本ソフトウェアの利用により直接的または間接的に生じるいかなる損 * 害からも,上記著作権者およびTOPPERSプロジェクトを免責すること. * * 本ソフトウェアは,無保証で提供されているものである.上記著作権者お * よびTOPPERSプロジェクトは,本ソフトウェアに関して,その適用可能性も * 含めて,いかなる保証も行わない.また,本ソフトウェアの利用により直 * 接的または間接的に生じたいかなる損害に関しても,その責任を負わない. * * @(#) $Id$ */ /* * Copyright (c) 1982, 1986, 1988, 1993 * The Regents of the University of California. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. All advertising materials mentioning features or use of this software * must display the following acknowledgement: * This product includes software developed by the University of * California, Berkeley and its contributors. * 4. Neither the name of the University nor the names of its contributors * may be used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * * @(#)ip_icmp.c 8.2 (Berkeley) 1/4/94 * $FreeBSD: src/sys/netinet/ip_icmp.c,v 1.33.2.3 1999/10/14 11:49:38 des Exp $ */ #include #ifdef TARGET_KERNEL_ASP #include #include #include #include "kernel_cfg.h" #include "tinet_cfg.h" #endif /* of #ifdef TARGET_KERNEL_ASP */ #ifdef TARGET_KERNEL_JSP #include #include #include "tinet_id.h" #endif /* of #ifdef TARGET_KERNEL_JSP */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #if defined(_IP4_CFG) /* * 変数 */ #ifdef SUPPORT_MIB /* * SNMP の 管理情報ベース (MIB) */ T_ICMP_STATS icmp_stats; #endif /* of #ifdef SUPPORT_MIB */ /* * 関数 */ static void icmp_echo (T_NET_BUF *input, uint_t ihoff); static void icmp_unreach (T_NET_BUF *input, uint_t ihoff); #if defined(NUM_REDIRECT_ROUTE_ENTRY) #if NUM_REDIRECT_ROUTE_ENTRY > 0 static void icmp_redirect (T_NET_BUF *input, uint_t ihoff); #endif /* of #if NUM_REDIRECT_ROUTE_ENTRY > 0 */ #endif /* of #if defined(NUM_REDIRECT_ROUTE_ENTRY) */ /* * icmp_echo -- エコー要求を受信したときの処理 * * input には IF ヘッダと IP ヘッダも含まれている。 */ static void icmp_echo (T_NET_BUF *input, uint_t ihoff) { T_IP4_HDR *ip4h; T_ICMP4_HDR *icmp4h; T_IFNET *ifp = IF_GET_IFNET(); NET_COUNT_MIB(icmp_stats.icmpInEchos, 1); /* IPv4 アドレスが未定義の時は応答しない。*/ if (!IN4_IS_ADDR_ANY(&ifp->in4_ifaddr.addr)) { /* * メッセージの型をエコー要求 (8) から * エコー応答 (0) に変更して送り返す。 */ icmp4h = GET_ICMP4_HDR(input, ihoff); icmp4h->type = ICMP4_ECHO_REPLY; /* * 宛先アドレスは受信したメッセージの送信元アドレス。 * 送信元アドレスは自 IPv4 アドレス。 */ ip4h = GET_IP4_HDR(input); ip4h->dst = ip4h->src; ip4h->src = htonl(ifp->in4_ifaddr.addr); /* チェックサムを計算する。*/ icmp4h->sum = 0; icmp4h->sum = in_cksum(icmp4h, (uint_t)(((input->len - GET_IF_IP4_HDR_SIZE(input)) + 3) >> 2 << 2)); /* 送信する。*/ NET_COUNT_ICMP4(net_count_icmp4.out_octets, input->len - GET_IF_IP4_HDR_SIZE(input)); NET_COUNT_ICMP4(net_count_icmp4.out_packets, 1); NET_COUNT_MIB(icmp_stats.icmpOutMsgs, 1); NET_COUNT_MIB(icmp_stats.icmpOutEchoReps, 1); ip_output(input, TMO_ICMP_OUTPUT); } } /* * icmp_unreach -- ICMP4_UNREACH を受信したときの処理 */ static const int8_t code2error[] = { EV_HURCH, /* ICMP4_UNREACH_NET 0 */ EV_HURCH, /* ICMP4_UNREACH_HOST 1 */ EV_CNNRF, /* ICMP4_UNREACH_PROTOCOL 2 */ EV_CNNRF, /* ICMP4_UNREACH_PORT 3 */ EV_MSIZE, /* ICMP4_UNREACH_FLAG 4 */ EV_HURCH, /* ICMP4_UNREACH_SRC_FAIL 5 */ EV_HURCH, /* ICMP4_UNREACH_NET_UNKNOWN 6 */ EV_HURCH, /* ICMP4_UNREACH_HOST_UNKNOWN 7 */ EV_HURCH, /* ICMP4_UNREACH_ISOLATED 8 */ EV_HURCH, /* ICMP4_UNREACH_NET_PROHIB 9 */ EV_HURCH, /* ICMP4_UNREACH_HOST_PROHIB 10 */ EV_HURCH, /* ICMP4_UNREACH_TOS_NET 11 */ EV_HURCH, /* ICMP4_UNREACH_TOS_HOST 12 */ EV_HURCH, /* ICMP4_UNREACH_FILTER_PROHIB 13 */ EV_HURCH, /* ICMP4_UNREACH_HOST_PRECEDENCE 14 */ EV_HURCH, /* ICMP4_UNREACH_PRECEDENCE_CUTOFF 15 */ }; static void icmp_unreach (T_NET_BUF *input, uint_t ihoff) { T_IP4_HDR *ip4h; uint8_t code; ER error; NET_COUNT_MIB(icmp_stats.icmpInDestUnreachs, 1); ip4h = (T_IP4_HDR*)GET_ICMP4_SDU(input, ihoff); code = GET_ICMP4_HDR(input, ihoff)->code; error = code2error[code]; /* 最終ヘッダが TCP/UDP のみ対応する。*/ if (ip4h->proto == IPPROTO_TCP || ip4h->proto == IPPROTO_UDP) { memcpy(GET_IP4_HDR(input), ip4h, input->len - (IP4_HDR_SIZE + ICMP4_HDR_SIZE)); input->len -= IP4_HDR_SIZE + ICMP4_HDR_SIZE; #if defined(SUPPORT_TCP) if (ip4h->proto == IPPROTO_TCP) tcp_notify(input, error); #endif /* of #if defined(SUPPORT_TCP) */ #if defined(SUPPORT_UDP) && TNUM_UDP4_CEPID > 0 if (ip4h->proto == IPPROTO_UDP) udp4_notify(input, error); #endif /* of #if defined(SUPPORT_UDP) && TNUM_UDP4_CEPID > 0 */ } else syslog(LOG_NOTICE, "[ICMP] error, code: %d.", code); } #if defined(NUM_REDIRECT_ROUTE_ENTRY) #if NUM_REDIRECT_ROUTE_ENTRY > 0 /* * icmp_redirect -- 向け直しメッセージを受信したときの処理 * * input には IF ヘッダと IP ヘッダも含まれている。 */ static void icmp_redirect (T_NET_BUF *input, uint_t ihoff) { T_IP4_HDR *ip4h, *sip4h; T_ICMP4_HDR *icmp4h; ip4h = GET_IP4_HDR(input); icmp4h = GET_ICMP4_HDR(input, ihoff); sip4h = (T_IP4_HDR *)GET_ICMP4_SDU(input, ihoff); /* * 内容チェック、以下の場合はエラー * * ・ICMP コードが、ICMP4_REDIRECT_PORT_AND_HOST (3) 以上 * ・データグラム長が、ICMP の最小長より短い (IP ヘッダ + ICMP ヘッダ + IP ヘッダ + 8) * ・データグラム長が、ICMP の最大長より短い * ・ */ if (icmp4h->code > ICMP4_REDIRECT_PORT_AND_HOST || ip4h->len < GET_IP4_ICMP4_HDR_SIZE(input) + 8 + IP4_HDR_SIZE || ip4h->len < GET_IP4_ICMP4_HDR_SIZE(input) + 8 + (IP4_VHL_HL(sip4h->vhl) << 2)) { NET_COUNT_ICMP4(net_count_icmp4.in_err_packets, 1); return; } in4_rtredirect(icmp4h->data.addr, sip4h->dst, IN_RTF_REDIRECT, TMO_IN_REDIRECT); } #endif /* of #if NUM_REDIRECT_ROUTE_ENTRY > 0 */ #endif /* of #if defined(NUM_REDIRECT_ROUTE_ENTRY) */ /* * icmp_input -- ICMP の入力関数 * * input には IF ヘッダと IP ヘッダも含まれている。 */ uint_t icmp_input (T_NET_BUF **inputp, uint_t *offp, uint_t *nextp) { T_NET_BUF *input = *inputp; T_ICMP4_HDR *icmp4h; T_IN4_ADDR addr; uint_t len, align; NET_COUNT_ICMP4(net_count_icmp4.in_octets, input->len - GET_IF_IP4_HDR_SIZE(input)); NET_COUNT_ICMP4(net_count_icmp4.in_packets, 1); /* ICMP ヘッダの長さをチェックする。*/ if (input->len < IF_IP4_ICMP4_HDR_SIZE) { NET_COUNT_ICMP4(net_count_icmp4.in_err_packets, 1); NET_COUNT_MIB(icmp_stats.icmpInErrors, 1); goto buf_rel; } icmp4h = (T_ICMP4_HDR *)(input->buf + *offp); /* 4 オクテット境界のデータ長 */ len = input->len - *offp; align = (len + 3) >> 2 << 2; /* 4 オクテット境界までパディングで埋める。*/ if (align > len) memset((uint8_t*)input->buf + input->len, 0, (size_t)(align - len)); /* チェックサムを計算する。*/ if (in_cksum(icmp4h, align) != 0) { NET_COUNT_ICMP4(net_count_icmp4.in_err_packets, 1); goto buf_rel; } /* メッセージの型により分岐する。*/ switch (icmp4h->type) { case ICMP4_ECHO_REQUEST: icmp_echo(input, *offp); return IPPROTO_DONE; break; case ICMP4_ECHO_REPLY: #if defined(SUPPORT_INET4) && defined(ICMP_CFG_CALLBACK_ECHO_REPLY) icmp_echo_reply(input, *offp); #endif /* of #if defined(SUPPORT_INET4) && defined(ICMP_CFG_CALLBACK_ECHO_REPLY) */ break; case ICMP4_UNREACH: icmp_unreach(input, *offp); break; case ICMP4_REDIRECT: #if defined(NUM_REDIRECT_ROUTE_ENTRY) #if NUM_REDIRECT_ROUTE_ENTRY > 0 addr = ntohl(icmp4h->data.addr); syslog(LOG_INFO, "[ICMP] redirect, addr: %s.", ip2str(NULL, &addr)); icmp_redirect(input, *offp); #else /* of #if NUM_REDIRECT_ROUTE_ENTRY > 0 */ addr = ntohl(icmp4h->data.addr); syslog(LOG_INFO, "[ICMP] redirect ignored, addr: %s.", ip2str(NULL, &addr)); #endif /* of #if NUM_REDIRECT_ROUTE_ENTRY > 0 */ #else /* of #if defined(NUM_REDIRECT_ROUTE_ENTRY) */ addr = ntohl(icmp4h->data.addr); syslog(LOG_INFO, "[ICMP] redirect ignored, addr: %s.", ip2str(NULL, &addr)); #endif /* of #if defined(NUM_REDIRECT_ROUTE_ENTRY) */ break; default: syslog(LOG_INFO, "[ICMP] unknown type: %d.", icmp4h->type); NET_COUNT_ICMP4(net_count_icmp4.in_err_packets, 1); break; } buf_rel: syscall(rel_net_buf(input)); return IPPROTO_DONE; } #ifdef ICMP_REPLY_ERROR /* * icmp_error -- ICMP エラー送信関数。 * * input には IF ヘッダと IP ヘッダも含まれている。 * icmp_error では、ネットワークバッファ input を返却しないので、 * 呼出し側で、開放してから終了する。 */ void icmp_error (uint8_t code, T_NET_BUF *input) { T_IP4_HDR *ip4h; T_ICMP4_HDR *icmp4h; T_NET_BUF *output; T_IN4_ADDR saddr; uint_t len, ip4hl, align; ip4h = GET_IP4_HDR(input); ip4hl = GET_IP4_HDR_SIZE(input); /* 送信用の IP データグラムを獲得する。*/ if (input->len - ip4hl < 8) len = input->len - ip4hl; else len = 8; saddr = ntohl(ip4h->src); if (in4_get_datagram(&output, (uint_t)(ICMP4_HDR_SIZE + ip4hl + len), 0, &saddr, NULL, IPPROTO_ICMP, IP4_DEFTTL, NBA_SEARCH_ASCENT, TMO_ICMP_OUTPUT) != E_OK) return; /* ICMP ヘッダを設定する。*/ icmp4h = GET_ICMP4_HDR(output, IF_IP4_ICMP4_HDR_OFFSET); icmp4h->type = ICMP4_UNREACH; icmp4h->code = code; icmp4h->data.addr= 0; /* エラーが発生した IP ヘッダと データ 8 オクテットをコピーする。*/ memcpy(GET_ICMP4_SDU(output, IF_IP4_ICMP4_HDR_OFFSET), GET_IP4_HDR(input), (size_t)(ip4hl + len)); /* 4 オクテット境界のデータ長 */ align = (len + 3) >> 2 << 2; /* 4 オクテット境界までパディングで埋める。*/ if (align > len) memset((uint8_t*)GET_ICMP4_SDU(output, IF_IP4_ICMP4_HDR_OFFSET) + ip4hl + len, 0, (size_t)(align - len)); /* チェックサムを計算する。*/ icmp4h->sum = 0; icmp4h->sum = in_cksum(icmp4h, (uint_t)(ICMP4_HDR_SIZE + ip4hl + align)); /* 送信する。*/ NET_COUNT_ICMP4(net_count_icmp4.out_octets, output->len - GET_IF_IP4_HDR_SIZE(output)); NET_COUNT_ICMP4(net_count_icmp4.out_packets, 1); NET_COUNT_MIB(icmp_stats.icmpOutMsgs, 1); NET_COUNT_MIB(icmp_stats.icmpOutDestUnreachs, 1); ip_output(output, TMO_ICMP_OUTPUT); } #endif /* of #ifdef ICMP_REPLY_ERROR */ #endif /* of #if defined(_IP4_CFG) */