VirtualBox

source: vbox/trunk/src/VBox/Devices/Network/slirp/ip_output.c@ 19748

Last change on this file since 19748 was 19525, checked in by vboxsync, 16 years ago

NAT: Correct ethernet package check

  • Property svn:eol-style set to native
File size: 7.1 KB
Line 
1/*
2 * Copyright (c) 1982, 1986, 1988, 1990, 1993
3 * The Regents of the University of California. All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
13 * 3. All advertising materials mentioning features or use of this software
14 * must display the following acknowledgement:
15 * This product includes software developed by the University of
16 * California, Berkeley and its contributors.
17 * 4. Neither the name of the University nor the names of its contributors
18 * may be used to endorse or promote products derived from this software
19 * without specific prior written permission.
20 *
21 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
22 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
23 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
24 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
25 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
26 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
27 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
28 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
29 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
30 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
31 * SUCH DAMAGE.
32 *
33 * @(#)ip_output.c 8.3 (Berkeley) 1/21/94
34 * ip_output.c,v 1.9 1994/11/16 10:17:10 jkh Exp
35 */
36
37/*
38 * Changes and additions relating to SLiRP are
39 * Copyright (c) 1995 Danny Gasparovski.
40 *
41 * Please read the file COPYRIGHT for the
42 * terms and conditions of the copyright.
43 */
44
45#include <slirp.h>
46
47
48/*
49 * IP output. The packet in mbuf chain m contains a skeletal IP
50 * header (with len, off, ttl, proto, tos, src, dst).
51 * The mbuf chain containing the packet will be freed.
52 * The mbuf opt, if present, will not be freed.
53 */
54int
55ip_output(PNATState pData, struct socket *so, struct mbuf *m0)
56{
57 register struct ip *ip;
58 register struct mbuf *m = m0;
59 register int hlen = sizeof(struct ip );
60 int len, off, error = 0;
61
62 DEBUG_CALL("ip_output");
63 DEBUG_ARG("so = %lx", (long)so);
64 DEBUG_ARG("m0 = %lx", (long)m0);
65 if(m->m_data != (MBUF_HEAD(m) + if_maxlinkhdr))
66 {
67 LogRel(("NAT: ethernet detects corruption of the packet"));
68 AssertMsgFailed(("!!Ethernet frame corrupted!!"));
69 }
70
71#if 0 /* We do no options */
72 if (opt)
73 {
74 m = ip_insertoptions(m, opt, &len);
75 hlen = len;
76 }
77#endif
78 ip = mtod(m, struct ip *);
79 /*
80 * Fill in IP header.
81 */
82 ip->ip_v = IPVERSION;
83 ip->ip_off &= IP_DF;
84 ip->ip_id = htons(ip_currid++);
85 ip->ip_hl = hlen >> 2;
86 ipstat.ips_localout++;
87
88 /*
89 * Verify that we have any chance at all of being able to queue
90 * the packet or packet fragments
91 */
92#if 0 /* XXX Hmmm... */
93 if (if_queued > if_thresh && towrite <= 0)
94 {
95 error = ENOBUFS;
96 goto bad;
97 }
98#endif
99
100 /*
101 * If small enough for interface, can just send directly.
102 */
103 if ((u_int16_t)ip->ip_len <= if_mtu)
104 {
105 ip->ip_len = htons((u_int16_t)ip->ip_len);
106 ip->ip_off = htons((u_int16_t)ip->ip_off);
107 ip->ip_sum = 0;
108 ip->ip_sum = cksum(m, hlen);
109
110#ifdef VBOX_WITH_NAT_SERVICE
111 {
112 struct ethhdr *eh, *eh0;
113 eh = (struct ethhdr *)m->m_dat;
114 eh0 = (struct ethhdr *)m0->m_dat;
115 memcpy(eh->h_source, eh0->h_source, ETH_ALEN);
116 }
117#endif
118 if_output(pData, so, m);
119 goto done;
120 }
121
122 /*
123 * Too large for interface; fragment if possible.
124 * Must be able to put at least 8 bytes per fragment.
125 */
126 if (ip->ip_off & IP_DF)
127 {
128 error = -1;
129 ipstat.ips_cantfrag++;
130 goto bad;
131 }
132
133 len = (if_mtu - hlen) &~ 7; /* ip databytes per packet */
134 if (len < 8)
135 {
136 error = -1;
137 goto bad;
138 }
139
140 {
141 int mhlen, firstlen = len;
142 struct mbuf **mnext = &m->m_nextpkt;
143
144 /*
145 * Loop through length of segment after first fragment,
146 * make new header and copy data of each part and link onto chain.
147 */
148 m0 = m;
149 mhlen = sizeof (struct ip);
150 for (off = hlen + len; off < (u_int16_t)ip->ip_len; off += len)
151 {
152#ifdef VBOX_WITH_NAT_SERVICE
153 struct ethhdr *eh0;
154 struct ethhdr *eh;
155#endif
156 register struct ip *mhip;
157 m = m_get(pData);
158 if (m == 0)
159 {
160 error = -1;
161 ipstat.ips_odropped++;
162 goto sendorfree;
163 }
164#ifdef VBOX_WITH_NAT_SERVICE
165 eh0 = (struct ethhdr *)m0->m_dat;
166 eh = (struct ethhdr *)m->m_dat;
167 memcpy(eh->h_source, eh0->h_source, ETH_ALEN);
168#endif
169 m->m_data += if_maxlinkhdr;
170 mhip = mtod(m, struct ip *);
171 *mhip = *ip;
172
173#if 0 /* No options */
174 if (hlen > sizeof (struct ip))
175 {
176 mhlen = ip_optcopy(ip, mhip) + sizeof (struct ip);
177 mhip->ip_hl = mhlen >> 2;
178 }
179#endif
180 m->m_len = mhlen;
181 mhip->ip_off = ((off - hlen) >> 3) + (ip->ip_off & ~IP_MF);
182 if (ip->ip_off & IP_MF)
183 mhip->ip_off |= IP_MF;
184 if (off + len >= (u_int16_t)ip->ip_len)
185 len = (u_int16_t)ip->ip_len - off;
186 else
187 mhip->ip_off |= IP_MF;
188 mhip->ip_len = htons((u_int16_t)(len + mhlen));
189
190 if (m_copy(m, m0, off, len) < 0)
191 {
192 error = -1;
193 goto sendorfree;
194 }
195
196 mhip->ip_off = htons((u_int16_t)mhip->ip_off);
197 mhip->ip_sum = 0;
198 mhip->ip_sum = cksum(m, mhlen);
199 *mnext = m;
200 mnext = &m->m_nextpkt;
201 ipstat.ips_ofragments++;
202 }
203 /*
204 * Update first fragment by trimming what's been copied out
205 * and updating header, then send each fragment (in order).
206 */
207 m = m0;
208 m_adj(m, hlen + firstlen - (u_int16_t)ip->ip_len);
209 ip->ip_len = htons((u_int16_t)m->m_len);
210 ip->ip_off = htons((u_int16_t)(ip->ip_off | IP_MF));
211 ip->ip_sum = 0;
212 ip->ip_sum = cksum(m, hlen);
213
214sendorfree:
215 for (m = m0; m; m = m0)
216 {
217 m0 = m->m_nextpkt;
218 m->m_nextpkt = 0;
219 if (error == 0)
220 if_output(pData, so, m);
221 else
222 m_freem(pData, m);
223 }
224
225 if (error == 0)
226 ipstat.ips_fragmented++;
227 }
228
229done:
230 return (error);
231
232bad:
233 m_freem(pData, m0);
234 goto done;
235}
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette