VirtualBox

source: vbox/trunk/include/VBox/vmm/pdmnetinline.h@ 36121

Last change on this file since 36121 was 35361, checked in by vboxsync, 14 years ago

fix OSE

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 25.7 KB
Line 
1/** @file
2 * PDM - Networking Helpers, Inlined Code. (DEV,++)
3 *
4 * This is all inlined because it's too tedious to create 2-3 libraries to
5 * contain it all (same bad excuse as for intnetinline.h).
6 */
7
8/*
9 * Copyright (C) 2010 Oracle Corporation
10 *
11 * This file is part of VirtualBox Open Source Edition (OSE), as
12 * available from http://www.virtualbox.org. This file is free software;
13 * you can redistribute it and/or modify it under the terms of the GNU
14 * General Public License (GPL) as published by the Free Software
15 * Foundation, in version 2 as it comes in the "COPYING" file of the
16 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
17 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
18 *
19 * The contents of this file may alternatively be used under the terms
20 * of the Common Development and Distribution License Version 1.0
21 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
22 * VirtualBox OSE distribution, in which case the provisions of the
23 * CDDL are applicable instead of those of the GPL.
24 *
25 * You may elect to license modified versions of this file under the
26 * terms and conditions of either the GPL or the CDDL or both.
27 */
28
29
30/*******************************************************************************
31* Header Files *
32*******************************************************************************/
33#include <VBox/log.h>
34#include <VBox/types.h>
35#include <iprt/asm.h>
36#include <iprt/assert.h>
37#include <iprt/net.h>
38#include <iprt/string.h>
39
40
41/**
42 * Checksum type.
43 */
44typedef enum PDMNETCSUMTYPE
45{
46 /** No checksum. */
47 PDMNETCSUMTYPE_NONE = 0,
48 /** Normal TCP checksum. */
49 PDMNETCSUMTYPE_COMPLETE,
50 /** Checksum on pseudo header (used with GSO). */
51 PDMNETCSUMTYPE_PSEUDO,
52 /** The usual 32-bit hack. */
53 PDMNETCSUMTYPE_32_BIT_HACK = 0x7fffffff
54} PDMNETCSUMTYPE;
55
56
57/**
58 * Validates the GSO context.
59 *
60 * @returns true if valid, false if not (not asserted or logged).
61 * @param pGso The GSO context.
62 * @param cbGsoMax The max size of the GSO context.
63 * @param cbFrame The max size of the GSO frame (use to validate
64 * the MSS).
65 */
66DECLINLINE(bool) PDMNetGsoIsValid(PCPDMNETWORKGSO pGso, size_t cbGsoMax, size_t cbFrame)
67{
68 PDMNETWORKGSOTYPE enmType;
69
70 if (RT_UNLIKELY(cbGsoMax < sizeof(*pGso)))
71 return false;
72
73 enmType = (PDMNETWORKGSOTYPE)pGso->u8Type;
74 if (RT_UNLIKELY( enmType <= PDMNETWORKGSOTYPE_INVALID || enmType >= PDMNETWORKGSOTYPE_END ))
75 return false;
76
77 /* all types requires both headers. */
78 if (RT_UNLIKELY( pGso->offHdr1 < sizeof(RTNETETHERHDR) ))
79 return false;
80 if (RT_UNLIKELY( pGso->offHdr2 <= pGso->offHdr1 ))
81 return false;
82 if (RT_UNLIKELY( pGso->cbHdrs <= pGso->offHdr2 ))
83 return false;
84
85 /* min size of the 1st header(s). */
86 switch (enmType)
87 {
88 case PDMNETWORKGSOTYPE_IPV4_TCP:
89 case PDMNETWORKGSOTYPE_IPV4_UDP:
90 if (RT_UNLIKELY( (unsigned)pGso->offHdr2 - pGso->offHdr1 < RTNETIPV4_MIN_LEN ))
91 return false;
92 break;
93 case PDMNETWORKGSOTYPE_IPV6_TCP:
94 case PDMNETWORKGSOTYPE_IPV6_UDP:
95 if (RT_UNLIKELY( (unsigned)pGso->offHdr2 - pGso->offHdr1 < RTNETIPV6_MIN_LEN ))
96 return false;
97 break;
98 case PDMNETWORKGSOTYPE_IPV4_IPV6_TCP:
99 case PDMNETWORKGSOTYPE_IPV4_IPV6_UDP:
100 if (RT_UNLIKELY( (unsigned)pGso->offHdr2 - pGso->offHdr1 < RTNETIPV4_MIN_LEN + RTNETIPV6_MIN_LEN ))
101 return false;
102 break;
103 case PDMNETWORKGSOTYPE_INVALID:
104 case PDMNETWORKGSOTYPE_END:
105 break;
106 /* no default case! want gcc warnings. */
107 }
108
109 /* min size of the 2nd header. */
110 switch (enmType)
111 {
112 case PDMNETWORKGSOTYPE_IPV4_TCP:
113 case PDMNETWORKGSOTYPE_IPV6_TCP:
114 case PDMNETWORKGSOTYPE_IPV4_IPV6_TCP:
115 if (RT_UNLIKELY( (unsigned)pGso->cbHdrs - pGso->offHdr2 < RTNETTCP_MIN_LEN ))
116 return false;
117 break;
118 case PDMNETWORKGSOTYPE_IPV4_UDP:
119 case PDMNETWORKGSOTYPE_IPV6_UDP:
120 case PDMNETWORKGSOTYPE_IPV4_IPV6_UDP:
121 if (RT_UNLIKELY( (unsigned)pGso->cbHdrs - pGso->offHdr2 < RTNETUDP_MIN_LEN ))
122 return false;
123 break;
124 case PDMNETWORKGSOTYPE_INVALID:
125 case PDMNETWORKGSOTYPE_END:
126 break;
127 /* no default case! want gcc warnings. */
128 }
129
130 /* There must be at more than one segment. */
131 if (RT_UNLIKELY( cbFrame <= pGso->cbHdrs ))
132 return false;
133 if (RT_UNLIKELY( cbFrame - pGso->cbHdrs < pGso->cbMaxSeg ))
134 return false;
135
136 return true;
137}
138
139
140/**
141 * Calculates the number of segments a GSO frame will be segmented into.
142 *
143 * @returns Segment count.
144 * @param pGso The GSO context.
145 * @param cbFrame The GSO frame size (header proto + payload).
146 */
147DECLINLINE(uint32_t) PDMNetGsoCalcSegmentCount(PCPDMNETWORKGSO pGso, size_t cbFrame)
148{
149 size_t cbPayload;
150 Assert(PDMNetGsoIsValid(pGso, sizeof(*pGso), cbFrame));
151 cbPayload = cbFrame - pGso->cbHdrs;
152 return (uint32_t)((cbPayload + pGso->cbMaxSeg - 1) / pGso->cbMaxSeg);
153}
154
155
156/**
157 * Used to find the IPv6 header when handling 4to6 tunneling.
158 *
159 * @returns Offset of the IPv6 header.
160 * @param pbSegHdrs The headers / frame start.
161 * @param offIpHdr The offset of the IPv4 header.
162 */
163DECLINLINE(uint8_t) pgmNetGsoCalcIpv6Offset(uint8_t *pbSegHdrs, uint8_t offIPv4Hdr)
164{
165 PCRTNETIPV4 pIPv4Hdr = (PCRTNETIPV4)&pbSegHdrs[offIPv4Hdr];
166 return offIPv4Hdr + pIPv4Hdr->ip_hl * 4;
167}
168
169
170/**
171 * Update an UDP header after carving out a segment
172 *
173 * @param u32PseudoSum The pseudo checksum.
174 * @param pbSegHdrs Pointer to the header bytes / frame start.
175 * @param offUdpHdr The offset into @a pbSegHdrs of the UDP header.
176 * @param pbPayload Pointer to the payload bytes.
177 * @param cbPayload The amount of payload.
178 * @param cbHdrs The size of all the headers.
179 * @param enmCsumType Whether to checksum the payload, the pseudo
180 * header or nothing.
181 * @internal
182 */
183DECLINLINE(void) pdmNetGsoUpdateUdpHdr(uint32_t u32PseudoSum, uint8_t *pbSegHdrs, uint8_t offUdpHdr,
184 uint8_t const *pbPayload, uint32_t cbPayload, uint8_t cbHdrs,
185 PDMNETCSUMTYPE enmCsumType)
186{
187 PRTNETUDP pUdpHdr = (PRTNETUDP)&pbSegHdrs[offUdpHdr];
188 pUdpHdr->uh_ulen = cbPayload + cbHdrs - offUdpHdr;
189 switch (enmCsumType)
190 {
191 case PDMNETCSUMTYPE_NONE:
192 pUdpHdr->uh_sum = 0;
193 break;
194 case PDMNETCSUMTYPE_COMPLETE:
195 pUdpHdr->uh_sum = RTNetUDPChecksum(u32PseudoSum, pUdpHdr);
196 break;
197 case PDMNETCSUMTYPE_PSEUDO:
198 pUdpHdr->uh_sum = ~RTNetIPv4FinalizeChecksum(u32PseudoSum);
199 break;
200 default:
201 AssertFailed();
202 break;
203 }
204}
205
206
207/**
208 * Update a TCP header after carving out a segment.
209 *
210 * @param u32PseudoSum The pseudo checksum.
211 * @param pbSegHdrs Pointer to the header bytes / frame start.
212 * @param offTcpHdr The offset into @a pbSegHdrs of the TCP header.
213 * @param pbPayload Pointer to the payload bytes.
214 * @param cbPayload The amount of payload.
215 * @param offPayload The offset into the payload that we're splitting
216 * up. We're ASSUMING that the payload follows
217 * immediately after the TCP header w/ options.
218 * @param cbHdrs The size of all the headers.
219 * @param fLastSeg Set if this is the last segment.
220 * @param enmCsumType Whether to checksum the payload, the pseudo
221 * header or nothing.
222 * @internal
223 */
224DECLINLINE(void) pdmNetGsoUpdateTcpHdr(uint32_t u32PseudoSum, uint8_t *pbSegHdrs, uint8_t offTcpHdr,
225 uint8_t const *pbPayload, uint32_t cbPayload, uint32_t offPayload, uint8_t cbHdrs,
226 bool fLastSeg, PDMNETCSUMTYPE enmCsumType)
227{
228 PRTNETTCP pTcpHdr = (PRTNETTCP)&pbSegHdrs[offTcpHdr];
229 pTcpHdr->th_seq = RT_H2N_U32(RT_N2H_U32(pTcpHdr->th_seq) + offPayload);
230 if (!fLastSeg)
231 pTcpHdr->th_flags &= ~(RTNETTCP_F_FIN | RTNETTCP_F_PSH);
232 switch (enmCsumType)
233 {
234 case PDMNETCSUMTYPE_NONE:
235 pTcpHdr->th_sum = 0;
236 break;
237 case PDMNETCSUMTYPE_COMPLETE:
238 pTcpHdr->th_sum = RTNetTCPChecksum(u32PseudoSum, pTcpHdr, pbPayload, cbPayload);
239 break;
240 case PDMNETCSUMTYPE_PSEUDO:
241 pTcpHdr->th_sum = ~RTNetIPv4FinalizeChecksum(u32PseudoSum);
242 break;
243 default:
244 AssertFailed();
245 break;
246 }
247}
248
249
250/**
251 * Updates a IPv6 header after carving out a segment.
252 *
253 * @returns 32-bit intermediary checksum value for the pseudo header.
254 * @param pbSegHdrs Pointer to the header bytes.
255 * @param offIpHdr The offset into @a pbSegHdrs of the IP header.
256 * @param cbSegPayload The amount of segmented payload. Not to be
257 * confused with the IP payload.
258 * @param cbHdrs The size of all the headers.
259 * @param offPktHdr Offset of the protocol packet header. For the
260 * pseudo header checksum calulation.
261 * @param bProtocol The protocol type. For the pseudo header.
262 * @internal
263 */
264DECLINLINE(uint32_t) pdmNetGsoUpdateIPv6Hdr(uint8_t *pbSegHdrs, uint8_t offIpHdr, uint32_t cbSegPayload, uint8_t cbHdrs,
265 uint8_t offPktHdr, uint8_t bProtocol)
266{
267 PRTNETIPV6 pIpHdr = (PRTNETIPV6)&pbSegHdrs[offIpHdr];
268 uint16_t cbPayload = (uint16_t)(cbHdrs - (offIpHdr + sizeof(RTNETIPV6)) + cbSegPayload);
269 pIpHdr->ip6_plen = RT_H2N_U16(cbPayload);
270 return RTNetIPv6PseudoChecksumEx(pIpHdr, bProtocol, (uint16_t)(cbHdrs - offPktHdr + cbSegPayload));
271}
272
273
274/**
275 * Updates a IPv4 header after carving out a segment.
276 *
277 * @returns 32-bit intermediary checksum value for the pseudo header.
278 * @param pbSegHdrs Pointer to the header bytes.
279 * @param offIpHdr The offset into @a pbSegHdrs of the IP header.
280 * @param cbSegPayload The amount of segmented payload.
281 * @param iSeg The segment index.
282 * @param cbHdrs The size of all the headers.
283 * @internal
284 */
285DECLINLINE(uint32_t) pdmNetGsoUpdateIPv4Hdr(uint8_t *pbSegHdrs, uint8_t offIpHdr, uint32_t cbSegPayload,
286 uint32_t iSeg, uint8_t cbHdrs)
287{
288 PRTNETIPV4 pIpHdr = (PRTNETIPV4)&pbSegHdrs[offIpHdr];
289 pIpHdr->ip_len = RT_H2N_U16(cbHdrs - offIpHdr + cbSegPayload);
290 pIpHdr->ip_id = RT_H2N_U16(RT_N2H_U16(pIpHdr->ip_id) + iSeg);
291 pIpHdr->ip_sum = RTNetIPv4HdrChecksum(pIpHdr);
292 return RTNetIPv4PseudoChecksum(pIpHdr);
293}
294
295
296/**
297 * Carves out the specified segment in a destructive manner.
298 *
299 * This is for sequentially carving out segments and pushing them along for
300 * processing or sending. To avoid allocating a temporary buffer for
301 * constructing the segment in, we trash the previous frame by putting the
302 * header at the end of it.
303 *
304 * @returns Pointer to the segment frame that we've carved out.
305 * @param pGso The GSO context data.
306 * @param pbFrame Pointer to the GSO frame.
307 * @param cbFrame The size of the GSO frame.
308 * @param pbHdrScatch Pointer to a pGso->cbHdrs sized area where we
309 * can save the original header prototypes on the
310 * first call (@a iSeg is 0) and retrieve it on
311 * susequent calls. (Just use a 256 bytes
312 * buffer to make life easy.)
313 * @param iSeg The segment that we're carving out (0-based).
314 * @param cSegs The number of segments in the GSO frame. Use
315 * PDMNetGsoCalcSegmentCount to find this.
316 * @param pcbSegFrame Where to return the size of the returned segment
317 * frame.
318 */
319DECLINLINE(void *) PDMNetGsoCarveSegmentQD(PCPDMNETWORKGSO pGso, uint8_t *pbFrame, size_t cbFrame, uint8_t *pbHdrScatch,
320 uint32_t iSeg, uint32_t cSegs, uint32_t *pcbSegFrame)
321{
322 /*
323 * Figure out where the payload is and where the header starts before we
324 * do the protocol specific carving.
325 */
326 uint8_t * const pbSegHdrs = pbFrame + pGso->cbMaxSeg * iSeg;
327 uint8_t * const pbSegPayload = pbSegHdrs + pGso->cbHdrs;
328 uint32_t const cbSegPayload = iSeg + 1 != cSegs
329 ? pGso->cbMaxSeg
330 : (uint32_t)(cbFrame - iSeg * pGso->cbMaxSeg - pGso->cbHdrs);
331 uint32_t const cbSegFrame = cbSegPayload + pGso->cbHdrs;
332
333 /*
334 * Check assumptions (doing it after declaring the variables because of C).
335 */
336 Assert(iSeg < cSegs);
337 Assert(cSegs == PDMNetGsoCalcSegmentCount(pGso, cbFrame));
338 Assert(PDMNetGsoIsValid(pGso, sizeof(*pGso), cbFrame));
339
340 /*
341 * Copy the header and do the protocol specific massaging of it.
342 */
343 if (iSeg != 0)
344 memcpy(pbSegHdrs, pbHdrScatch, pGso->cbHdrs);
345 else
346 memcpy(pbHdrScatch, pbSegHdrs, pGso->cbHdrs);
347
348 switch ((PDMNETWORKGSOTYPE)pGso->u8Type)
349 {
350 case PDMNETWORKGSOTYPE_IPV4_TCP:
351 pdmNetGsoUpdateTcpHdr(pdmNetGsoUpdateIPv4Hdr(pbSegHdrs, pGso->offHdr1, cbSegPayload, iSeg, pGso->cbHdrs),
352 pbSegHdrs, pGso->offHdr2, pbSegPayload, cbSegPayload, iSeg * pGso->cbMaxSeg,
353 pGso->cbHdrs, iSeg + 1 == cSegs, PDMNETCSUMTYPE_COMPLETE);
354 break;
355 case PDMNETWORKGSOTYPE_IPV4_UDP:
356 pdmNetGsoUpdateUdpHdr(pdmNetGsoUpdateIPv4Hdr(pbSegHdrs, pGso->offHdr1, cbSegPayload, iSeg, pGso->cbHdrs),
357 pbSegHdrs, pGso->offHdr2, pbSegPayload, cbSegPayload, pGso->cbHdrs, PDMNETCSUMTYPE_COMPLETE);
358 break;
359 case PDMNETWORKGSOTYPE_IPV6_TCP:
360 pdmNetGsoUpdateTcpHdr(pdmNetGsoUpdateIPv6Hdr(pbSegHdrs, pGso->offHdr1, cbSegPayload, pGso->cbHdrs,
361 pGso->offHdr2, RTNETIPV4_PROT_TCP),
362 pbSegHdrs, pGso->offHdr2, pbSegPayload, cbSegPayload, iSeg * pGso->cbMaxSeg,
363 pGso->cbHdrs, iSeg + 1 == cSegs, PDMNETCSUMTYPE_COMPLETE);
364 break;
365 case PDMNETWORKGSOTYPE_IPV6_UDP:
366 pdmNetGsoUpdateUdpHdr(pdmNetGsoUpdateIPv6Hdr(pbSegHdrs, pGso->offHdr1, cbSegPayload, pGso->cbHdrs,
367 pGso->offHdr2, RTNETIPV4_PROT_UDP),
368 pbSegHdrs, pGso->offHdr2, pbSegPayload, cbSegPayload, pGso->cbHdrs, PDMNETCSUMTYPE_COMPLETE);
369 break;
370 case PDMNETWORKGSOTYPE_IPV4_IPV6_TCP:
371 pdmNetGsoUpdateIPv4Hdr(pbSegHdrs, pGso->offHdr1, cbSegPayload, iSeg, pGso->cbHdrs);
372 pdmNetGsoUpdateTcpHdr(pdmNetGsoUpdateIPv6Hdr(pbSegHdrs, pgmNetGsoCalcIpv6Offset(pbSegHdrs, pGso->offHdr1),
373 cbSegPayload, pGso->cbHdrs, pGso->offHdr2, RTNETIPV4_PROT_TCP),
374 pbSegHdrs, pGso->offHdr2, pbSegPayload, cbSegPayload, iSeg * pGso->cbMaxSeg,
375 pGso->cbHdrs, iSeg + 1 == cSegs, PDMNETCSUMTYPE_COMPLETE);
376 break;
377 case PDMNETWORKGSOTYPE_IPV4_IPV6_UDP:
378 pdmNetGsoUpdateIPv4Hdr(pbSegHdrs, pGso->offHdr1, cbSegPayload, iSeg, pGso->cbHdrs);
379 pdmNetGsoUpdateUdpHdr(pdmNetGsoUpdateIPv6Hdr(pbSegHdrs, pgmNetGsoCalcIpv6Offset(pbSegHdrs, pGso->offHdr1),
380 cbSegPayload, pGso->cbHdrs, pGso->offHdr2, RTNETIPV4_PROT_UDP),
381 pbSegHdrs, pGso->offHdr2, pbSegPayload, cbSegPayload, pGso->cbHdrs, PDMNETCSUMTYPE_COMPLETE);
382 break;
383 case PDMNETWORKGSOTYPE_INVALID:
384 case PDMNETWORKGSOTYPE_END:
385 /* no default! wnat gcc warnings. */
386 break;
387 }
388
389 *pcbSegFrame = cbSegFrame;
390 return pbSegHdrs;
391}
392
393
394/**
395 * Carves out the specified segment in a non-destructive manner.
396 *
397 * The segment headers and segment payload is kept separate here. The GSO frame
398 * is still expected to be one linear chunk of data, but we don't modify any of
399 * it.
400 *
401 * @returns The offset into the GSO frame of the payload.
402 * @param pGso The GSO context data.
403 * @param pbFrame Pointer to the GSO frame. Used for retrieving
404 * the header prototype and for checksumming the
405 * payload. The buffer is not modified.
406 * @param cbFrame The size of the GSO frame.
407 * @param iSeg The segment that we're carving out (0-based).
408 * @param cSegs The number of segments in the GSO frame. Use
409 * PDMNetGsoCalcSegmentCount to find this.
410 * @param pbSegHdrs Where to return the headers for the segment
411 * that's been carved out. The buffer must be at
412 * least pGso->cbHdrs in size, using a 256 byte
413 * buffer is a recommended simplification.
414 * @param pcbSegPayload Where to return the size of the returned
415 * segment payload.
416 */
417DECLINLINE(uint32_t) PDMNetGsoCarveSegment(PCPDMNETWORKGSO pGso, const uint8_t *pbFrame, size_t cbFrame,
418 uint32_t iSeg, uint32_t cSegs, uint8_t *pbSegHdrs, uint32_t *pcbSegPayload)
419{
420 /*
421 * Figure out where the payload is and where the header starts before we
422 * do the protocol specific carving.
423 */
424 uint8_t const * const pbSegPayload = pbFrame + pGso->cbHdrs + iSeg * pGso->cbMaxSeg;
425 uint32_t const cbSegPayload = iSeg + 1 != cSegs
426 ? pGso->cbMaxSeg
427 : (uint32_t)(cbFrame - iSeg * pGso->cbMaxSeg - pGso->cbHdrs);
428
429 /*
430 * Check assumptions (doing it after declaring the variables because of C).
431 */
432 Assert(iSeg < cSegs);
433 Assert(cSegs == PDMNetGsoCalcSegmentCount(pGso, cbFrame));
434 Assert(PDMNetGsoIsValid(pGso, sizeof(*pGso), cbFrame));
435
436 /*
437 * Copy the header and do the protocol specific massaging of it.
438 */
439 memcpy(pbSegHdrs, pbFrame, pGso->cbHdrs);
440
441 switch ((PDMNETWORKGSOTYPE)pGso->u8Type)
442 {
443 case PDMNETWORKGSOTYPE_IPV4_TCP:
444 pdmNetGsoUpdateTcpHdr(pdmNetGsoUpdateIPv4Hdr(pbSegHdrs, pGso->offHdr1, cbSegPayload, iSeg, pGso->cbHdrs),
445 pbSegHdrs, pGso->offHdr2, pbSegPayload, cbSegPayload, iSeg * pGso->cbMaxSeg,
446 pGso->cbHdrs, iSeg + 1 == cSegs, PDMNETCSUMTYPE_COMPLETE);
447 break;
448 case PDMNETWORKGSOTYPE_IPV4_UDP:
449 pdmNetGsoUpdateUdpHdr(pdmNetGsoUpdateIPv4Hdr(pbSegHdrs, pGso->offHdr1, cbSegPayload, iSeg, pGso->cbHdrs),
450 pbSegHdrs, pGso->offHdr2, pbSegPayload, cbSegPayload, pGso->cbHdrs, PDMNETCSUMTYPE_COMPLETE);
451 break;
452 case PDMNETWORKGSOTYPE_IPV6_TCP:
453 pdmNetGsoUpdateTcpHdr(pdmNetGsoUpdateIPv6Hdr(pbSegHdrs, pGso->offHdr1, cbSegPayload, pGso->cbHdrs,
454 pGso->offHdr2, RTNETIPV4_PROT_TCP),
455 pbSegHdrs, pGso->offHdr2, pbSegPayload, cbSegPayload, iSeg * pGso->cbMaxSeg,
456 pGso->cbHdrs, iSeg + 1 == cSegs, PDMNETCSUMTYPE_COMPLETE);
457 break;
458 case PDMNETWORKGSOTYPE_IPV6_UDP:
459 pdmNetGsoUpdateUdpHdr(pdmNetGsoUpdateIPv6Hdr(pbSegHdrs, pGso->offHdr1, cbSegPayload, pGso->cbHdrs,
460 pGso->offHdr2, RTNETIPV4_PROT_UDP),
461 pbSegHdrs, pGso->offHdr2, pbSegPayload, cbSegPayload, pGso->cbHdrs, PDMNETCSUMTYPE_COMPLETE);
462 break;
463 case PDMNETWORKGSOTYPE_IPV4_IPV6_TCP:
464 pdmNetGsoUpdateIPv4Hdr(pbSegHdrs, pGso->offHdr1, cbSegPayload, iSeg, pGso->cbHdrs);
465 pdmNetGsoUpdateTcpHdr(pdmNetGsoUpdateIPv6Hdr(pbSegHdrs, pgmNetGsoCalcIpv6Offset(pbSegHdrs, pGso->offHdr1),
466 cbSegPayload, pGso->cbHdrs, pGso->offHdr2, RTNETIPV4_PROT_TCP),
467 pbSegHdrs, pGso->offHdr2, pbSegPayload, cbSegPayload, iSeg * pGso->cbMaxSeg,
468 pGso->cbHdrs, iSeg + 1 == cSegs, PDMNETCSUMTYPE_COMPLETE);
469 break;
470 case PDMNETWORKGSOTYPE_IPV4_IPV6_UDP:
471 pdmNetGsoUpdateIPv4Hdr(pbSegHdrs, pGso->offHdr1, cbSegPayload, iSeg, pGso->cbHdrs);
472 pdmNetGsoUpdateUdpHdr(pdmNetGsoUpdateIPv6Hdr(pbSegHdrs, pgmNetGsoCalcIpv6Offset(pbSegHdrs, pGso->offHdr1),
473 cbSegPayload, pGso->cbHdrs, pGso->offHdr2, RTNETIPV4_PROT_UDP),
474 pbSegHdrs, pGso->offHdr2, pbSegPayload, cbSegPayload, pGso->cbHdrs, PDMNETCSUMTYPE_COMPLETE);
475 break;
476 case PDMNETWORKGSOTYPE_INVALID:
477 case PDMNETWORKGSOTYPE_END:
478 /* no default! wnat gcc warnings. */
479 break;
480 }
481
482 *pcbSegPayload = cbSegPayload;
483 return pGso->cbHdrs + iSeg * pGso->cbMaxSeg;
484}
485
486
487/**
488 * Prepares the GSO frame for direct use without any segmenting.
489 *
490 * @param pGso The GSO context.
491 * @param pvFrame The frame to prepare.
492 * @param cbFrame The frame size.
493 * @param enmCsumType Whether to checksum the payload, the pseudo
494 * header or nothing.
495 */
496DECLINLINE(void) PDMNetGsoPrepForDirectUse(PCPDMNETWORKGSO pGso, void *pvFrame, size_t cbFrame, PDMNETCSUMTYPE enmCsumType)
497{
498 /*
499 * Figure out where the payload is and where the header starts before we
500 * do the protocol bits.
501 */
502 uint8_t * const pbHdrs = (uint8_t *)pvFrame;
503 uint8_t * const pbPayload = pbHdrs + pGso->cbHdrs;
504 uint32_t const cbFrame32 = (uint32_t)cbFrame;
505 uint32_t const cbPayload = cbFrame32 - pGso->cbHdrs;
506
507 /*
508 * Check assumptions (doing it after declaring the variables because of C).
509 */
510 Assert(PDMNetGsoIsValid(pGso, sizeof(*pGso), cbFrame));
511
512 /*
513 * Get down to busienss.
514 */
515 switch ((PDMNETWORKGSOTYPE)pGso->u8Type)
516 {
517 case PDMNETWORKGSOTYPE_IPV4_TCP:
518 pdmNetGsoUpdateTcpHdr(pdmNetGsoUpdateIPv4Hdr(pbHdrs, pGso->offHdr1, cbFrame32 - pGso->cbHdrs, 0, pGso->cbHdrs),
519 pbHdrs, pGso->offHdr2, pbPayload, cbPayload, 0, pGso->cbHdrs, true, enmCsumType);
520 break;
521 case PDMNETWORKGSOTYPE_IPV4_UDP:
522 pdmNetGsoUpdateUdpHdr(pdmNetGsoUpdateIPv4Hdr(pbHdrs, pGso->offHdr1, cbFrame32 - pGso->cbHdrs, 0, pGso->cbHdrs),
523 pbHdrs, pGso->offHdr2, pbPayload, cbPayload, pGso->cbHdrs, enmCsumType);
524 break;
525 case PDMNETWORKGSOTYPE_IPV6_TCP:
526 pdmNetGsoUpdateTcpHdr(pdmNetGsoUpdateIPv6Hdr(pbHdrs, pGso->offHdr1, cbPayload, pGso->cbHdrs,
527 pGso->offHdr2, RTNETIPV4_PROT_TCP),
528 pbHdrs, pGso->offHdr2, pbPayload, cbPayload, 0, pGso->cbHdrs, true, enmCsumType);
529 break;
530 case PDMNETWORKGSOTYPE_IPV6_UDP:
531 pdmNetGsoUpdateUdpHdr(pdmNetGsoUpdateIPv6Hdr(pbHdrs, pGso->offHdr1, cbPayload, pGso->cbHdrs,
532 pGso->offHdr2, RTNETIPV4_PROT_UDP),
533 pbHdrs, pGso->offHdr2, pbPayload, cbPayload, pGso->cbHdrs, enmCsumType);
534 break;
535 case PDMNETWORKGSOTYPE_IPV4_IPV6_TCP:
536 pdmNetGsoUpdateIPv4Hdr(pbHdrs, pGso->offHdr1, cbPayload, 0, pGso->cbHdrs);
537 pdmNetGsoUpdateTcpHdr(pdmNetGsoUpdateIPv6Hdr(pbHdrs, pgmNetGsoCalcIpv6Offset(pbHdrs, pGso->offHdr1),
538 cbPayload, pGso->cbHdrs, pGso->offHdr2, RTNETIPV4_PROT_TCP),
539 pbHdrs, pGso->offHdr2, pbPayload, cbPayload, 0, pGso->cbHdrs, true, enmCsumType);
540 break;
541 case PDMNETWORKGSOTYPE_IPV4_IPV6_UDP:
542 pdmNetGsoUpdateIPv4Hdr(pbHdrs, pGso->offHdr1, cbPayload, 0, pGso->cbHdrs);
543 pdmNetGsoUpdateUdpHdr(pdmNetGsoUpdateIPv6Hdr(pbHdrs, pgmNetGsoCalcIpv6Offset(pbHdrs, pGso->offHdr1),
544 cbPayload, pGso->cbHdrs, pGso->offHdr2, RTNETIPV4_PROT_UDP),
545 pbHdrs, pGso->offHdr2, pbPayload, cbPayload, pGso->cbHdrs, enmCsumType);
546 break;
547 case PDMNETWORKGSOTYPE_INVALID:
548 case PDMNETWORKGSOTYPE_END:
549 /* no default! wnat gcc warnings. */
550 break;
551 }
552}
553
554
555/**
556 * Gets the GSO type name string.
557 *
558 * @returns Pointer to read only name string.
559 * @param enmType The type.
560 */
561DECLINLINE(const char *) PDMNetGsoTypeName(PDMNETWORKGSOTYPE enmType)
562{
563 switch (enmType)
564 {
565 case PDMNETWORKGSOTYPE_IPV4_TCP: return "TCPv4";
566 case PDMNETWORKGSOTYPE_IPV6_TCP: return "TCPv6";
567 case PDMNETWORKGSOTYPE_IPV4_UDP: return "UDPv4";
568 case PDMNETWORKGSOTYPE_IPV6_UDP: return "UDPv6";
569 case PDMNETWORKGSOTYPE_IPV4_IPV6_TCP: return "4to6TCP";
570 case PDMNETWORKGSOTYPE_IPV4_IPV6_UDP: return "4to6UDP";
571 case PDMNETWORKGSOTYPE_INVALID: return "invalid";
572 case PDMNETWORKGSOTYPE_END: return "end";
573 }
574 return "bad-gso-type";
575}
576
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