1 /* $NetBSD: in_offload.c,v 1.1 2006/11/25 18:41:36 yamt Exp $ */
4 * Copyright (c)2005, 2006 YAMAMOTO Takashi,
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
16 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
17 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
20 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
29 #include <sys/cdefs.h>
30 __KERNEL_RCSID(0, "$NetBSD: in_offload.c,v 1.1 2006/11/25 18:41:36 yamt Exp $");
32 #include <sys/param.h>
37 #include <netinet/in.h>
38 #include <netinet/in_systm.h>
39 #include <netinet/ip.h>
40 #include <netinet/tcp.h>
41 #include <netinet/in_offload.h>
43 struct ip_tso_output_args
{
45 const struct sockaddr
*sa
;
49 static int ip_tso_output_callback(void *, struct mbuf
*);
52 ip_tso_output_callback(void *vp
, struct mbuf
*m
)
54 struct ip_tso_output_args
*args
= vp
;
55 struct ifnet
*ifp
= args
->ifp
;
57 return (*ifp
->if_output
)(ifp
, m
, args
->sa
, args
->rt
);
61 ip_tso_output(struct ifnet
*ifp
, struct mbuf
*m
, const struct sockaddr
*sa
,
64 struct ip_tso_output_args args
;
70 return tcp4_segment(m
, ip_tso_output_callback
, &args
);
74 * tcp4_segment: handle M_CSUM_TSOv4 by software.
76 * => always consume m.
77 * => call output_func with output_arg for each segments.
81 tcp4_segment(struct mbuf
*m
, int (*output_func
)(void *, struct mbuf
*),
93 struct mbuf
*hdr
= NULL
;
97 KASSERT((m
->m_flags
& M_PKTHDR
) != 0);
98 KASSERT((m
->m_pkthdr
.csum_flags
& M_CSUM_TSOv4
) != 0);
100 m
->m_pkthdr
.csum_flags
= 0;
102 len
= m
->m_pkthdr
.len
;
103 KASSERT(len
>= sizeof(*iph
) + sizeof(*th
));
105 if (m
->m_len
< sizeof(*iph
)) {
106 m
= m_pullup(m
, sizeof(*iph
));
112 iph
= mtod(m
, struct ip
*);
113 iphlen
= iph
->ip_hl
* 4;
114 KASSERT(iph
->ip_v
== IPVERSION
);
115 KASSERT(iphlen
>= sizeof(*iph
));
116 KASSERT(iph
->ip_p
== IPPROTO_TCP
);
117 ipid
= ntohs(iph
->ip_id
);
119 hlen
= iphlen
+ sizeof(*th
);
120 if (m
->m_len
< hlen
) {
121 m
= m_pullup(m
, hlen
);
127 th
= (void *)(mtod(m
, char *) + iphlen
);
128 tcpseq
= ntohl(th
->th_seq
);
129 thlen
= th
->th_off
* 4;
130 hlen
= iphlen
+ thlen
;
132 mss
= m
->m_pkthdr
.segsz
;
136 t
= m_split(m
, hlen
, M_NOWAIT
);
144 KASSERT(len
% mss
== 0);
148 n
= m_dup(hdr
, 0, hlen
, M_NOWAIT
);
153 KASSERT(n
->m_len
== hlen
); /* XXX */
155 t
= m_split(m
, mss
, M_NOWAIT
);
164 KASSERT(n
->m_len
>= hlen
); /* XXX */
166 n
->m_pkthdr
.len
= hlen
+ mss
;
167 iph
= mtod(n
, struct ip
*);
168 KASSERT(iph
->ip_v
== IPVERSION
);
169 iph
->ip_len
= htons(n
->m_pkthdr
.len
);
170 iph
->ip_id
= htons(ipid
);
171 th
= (void *)(mtod(n
, char *) + iphlen
);
172 th
->th_seq
= htonl(tcpseq
);
174 iph
->ip_sum
= in_cksum(n
, iphlen
);
176 th
->th_sum
= in4_cksum(n
, IPPROTO_TCP
, iphlen
, thlen
+ mss
);
178 error
= (*output_func
)(output_arg
, n
);