2014-05-26 13:22:46 +04:00
# include <linux/export.h>
2015-10-26 11:31:29 +03:00
# include <linux/if_vlan.h>
2014-05-19 20:59:52 +04:00
# include <net/ip.h>
# include <net/tso.h>
2014-10-21 18:06:05 +04:00
# include <asm/unaligned.h>
2014-05-19 20:59:52 +04:00
/* Calculate expected number of TX descriptors */
int tso_count_descs ( struct sk_buff * skb )
{
/* The Marvell Way */
return skb_shinfo ( skb ) - > gso_segs * 2 + skb_shinfo ( skb ) - > nr_frags ;
}
2014-05-26 13:22:46 +04:00
EXPORT_SYMBOL ( tso_count_descs ) ;
2014-05-19 20:59:52 +04:00
void tso_build_hdr ( struct sk_buff * skb , char * hdr , struct tso_t * tso ,
int size , bool is_last )
{
struct tcphdr * tcph ;
int hdr_len = skb_transport_offset ( skb ) + tcp_hdrlen ( skb ) ;
int mac_hdr_len = skb_network_offset ( skb ) ;
memcpy ( hdr , skb - > data , hdr_len ) ;
2015-10-26 11:31:29 +03:00
if ( ! tso - > ipv6 ) {
struct iphdr * iph = ( void * ) ( hdr + mac_hdr_len ) ;
iph - > id = htons ( tso - > ip_id ) ;
iph - > tot_len = htons ( size + hdr_len - mac_hdr_len ) ;
tso - > ip_id + + ;
} else {
struct ipv6hdr * iph = ( void * ) ( hdr + mac_hdr_len ) ;
iph - > payload_len = htons ( size + tcp_hdrlen ( skb ) ) ;
}
2014-05-19 20:59:52 +04:00
tcph = ( struct tcphdr * ) ( hdr + skb_transport_offset ( skb ) ) ;
2014-10-21 18:06:05 +04:00
put_unaligned_be32 ( tso - > tcp_seq , & tcph - > seq ) ;
2014-05-19 20:59:52 +04:00
if ( ! is_last ) {
/* Clear all special flags for not last packet */
tcph - > psh = 0 ;
tcph - > fin = 0 ;
tcph - > rst = 0 ;
}
}
2014-05-26 13:22:46 +04:00
EXPORT_SYMBOL ( tso_build_hdr ) ;
2014-05-19 20:59:52 +04:00
void tso_build_data ( struct sk_buff * skb , struct tso_t * tso , int size )
{
tso - > tcp_seq + = size ;
tso - > size - = size ;
tso - > data + = size ;
if ( ( tso - > size = = 0 ) & &
( tso - > next_frag_idx < skb_shinfo ( skb ) - > nr_frags ) ) {
skb_frag_t * frag = & skb_shinfo ( skb ) - > frags [ tso - > next_frag_idx ] ;
/* Move to next segment */
tso - > size = frag - > size ;
tso - > data = page_address ( frag - > page . p ) + frag - > page_offset ;
tso - > next_frag_idx + + ;
}
}
2014-05-26 13:22:46 +04:00
EXPORT_SYMBOL ( tso_build_data ) ;
2014-05-19 20:59:52 +04:00
void tso_start ( struct sk_buff * skb , struct tso_t * tso )
{
int hdr_len = skb_transport_offset ( skb ) + tcp_hdrlen ( skb ) ;
tso - > ip_id = ntohs ( ip_hdr ( skb ) - > id ) ;
tso - > tcp_seq = ntohl ( tcp_hdr ( skb ) - > seq ) ;
tso - > next_frag_idx = 0 ;
2015-10-26 11:31:29 +03:00
tso - > ipv6 = vlan_get_protocol ( skb ) = = htons ( ETH_P_IPV6 ) ;
2014-05-19 20:59:52 +04:00
/* Build first data */
tso - > size = skb_headlen ( skb ) - hdr_len ;
tso - > data = skb - > data + hdr_len ;
if ( ( tso - > size = = 0 ) & &
( tso - > next_frag_idx < skb_shinfo ( skb ) - > nr_frags ) ) {
skb_frag_t * frag = & skb_shinfo ( skb ) - > frags [ tso - > next_frag_idx ] ;
/* Move to next segment */
tso - > size = frag - > size ;
tso - > data = page_address ( frag - > page . p ) + frag - > page_offset ;
tso - > next_frag_idx + + ;
}
}
2014-05-26 13:22:46 +04:00
EXPORT_SYMBOL ( tso_start ) ;