aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/net/sfc/tx.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/net/sfc/tx.c')
-rw-r--r--drivers/net/sfc/tx.c161
1 files changed, 75 insertions, 86 deletions
diff --git a/drivers/net/sfc/tx.c b/drivers/net/sfc/tx.c
index 51429b6a4dee..550856fab16c 100644
--- a/drivers/net/sfc/tx.c
+++ b/drivers/net/sfc/tx.c
@@ -540,46 +540,37 @@ void efx_remove_tx_queue(struct efx_tx_queue *tx_queue)
540 540
541/** 541/**
542 * struct tso_state - TSO state for an SKB 542 * struct tso_state - TSO state for an SKB
543 * @remaining_len: Bytes of data we've yet to segment 543 * @out_len: Remaining length in current segment
544 * @seqnum: Current sequence number 544 * @seqnum: Current sequence number
545 * @ipv4_id: Current IPv4 ID, host endian
545 * @packet_space: Remaining space in current packet 546 * @packet_space: Remaining space in current packet
546 * @ifc: Input fragment cursor. 547 * @dma_addr: DMA address of current position
547 * Where we are in the current fragment of the incoming SKB. These 548 * @in_len: Remaining length in current SKB fragment
548 * values get updated in place when we split a fragment over 549 * @unmap_len: Length of SKB fragment
549 * multiple packets. 550 * @unmap_addr: DMA address of SKB fragment
550 * @p: Parameters. 551 * @unmap_single: DMA single vs page mapping flag
551 * These values are set once at the start of the TSO send and do 552 * @header_len: Number of bytes of header
552 * not get changed as the routine progresses. 553 * @full_packet_size: Number of bytes to put in each outgoing segment
553 * 554 *
554 * The state used during segmentation. It is put into this data structure 555 * The state used during segmentation. It is put into this data structure
555 * just to make it easy to pass into inline functions. 556 * just to make it easy to pass into inline functions.
556 */ 557 */
557struct tso_state { 558struct tso_state {
558 unsigned remaining_len; 559 /* Output position */
560 unsigned out_len;
559 unsigned seqnum; 561 unsigned seqnum;
562 unsigned ipv4_id;
560 unsigned packet_space; 563 unsigned packet_space;
561 564
562 struct { 565 /* Input position */
563 /* DMA address of current position */ 566 dma_addr_t dma_addr;
564 dma_addr_t dma_addr; 567 unsigned in_len;
565 /* Remaining length */ 568 unsigned unmap_len;
566 unsigned int len; 569 dma_addr_t unmap_addr;
567 /* DMA address and length of the whole fragment */ 570 bool unmap_single;
568 unsigned int unmap_len; 571
569 dma_addr_t unmap_addr; 572 unsigned header_len;
570 bool unmap_single; 573 int full_packet_size;
571 } ifc;
572
573 struct {
574 /* The number of bytes of header */
575 unsigned int header_length;
576
577 /* The number of bytes to put in each outgoing segment. */
578 int full_packet_size;
579
580 /* Current IPv4 ID, host endian. */
581 unsigned ipv4_id;
582 } p;
583}; 574};
584 575
585 576
@@ -840,35 +831,34 @@ static inline void tso_start(struct tso_state *st, const struct sk_buff *skb)
840 /* All ethernet/IP/TCP headers combined size is TCP header size 831 /* All ethernet/IP/TCP headers combined size is TCP header size
841 * plus offset of TCP header relative to start of packet. 832 * plus offset of TCP header relative to start of packet.
842 */ 833 */
843 st->p.header_length = ((tcp_hdr(skb)->doff << 2u) 834 st->header_len = ((tcp_hdr(skb)->doff << 2u)
844 + PTR_DIFF(tcp_hdr(skb), skb->data)); 835 + PTR_DIFF(tcp_hdr(skb), skb->data));
845 st->p.full_packet_size = (st->p.header_length 836 st->full_packet_size = st->header_len + skb_shinfo(skb)->gso_size;
846 + skb_shinfo(skb)->gso_size);
847 837
848 st->p.ipv4_id = ntohs(ip_hdr(skb)->id); 838 st->ipv4_id = ntohs(ip_hdr(skb)->id);
849 st->seqnum = ntohl(tcp_hdr(skb)->seq); 839 st->seqnum = ntohl(tcp_hdr(skb)->seq);
850 840
851 EFX_BUG_ON_PARANOID(tcp_hdr(skb)->urg); 841 EFX_BUG_ON_PARANOID(tcp_hdr(skb)->urg);
852 EFX_BUG_ON_PARANOID(tcp_hdr(skb)->syn); 842 EFX_BUG_ON_PARANOID(tcp_hdr(skb)->syn);
853 EFX_BUG_ON_PARANOID(tcp_hdr(skb)->rst); 843 EFX_BUG_ON_PARANOID(tcp_hdr(skb)->rst);
854 844
855 st->packet_space = st->p.full_packet_size; 845 st->packet_space = st->full_packet_size;
856 st->remaining_len = skb->len - st->p.header_length; 846 st->out_len = skb->len - st->header_len;
857 st->ifc.unmap_len = 0; 847 st->unmap_len = 0;
858 st->ifc.unmap_single = false; 848 st->unmap_single = false;
859} 849}
860 850
861static inline int tso_get_fragment(struct tso_state *st, struct efx_nic *efx, 851static inline int tso_get_fragment(struct tso_state *st, struct efx_nic *efx,
862 skb_frag_t *frag) 852 skb_frag_t *frag)
863{ 853{
864 st->ifc.unmap_addr = pci_map_page(efx->pci_dev, frag->page, 854 st->unmap_addr = pci_map_page(efx->pci_dev, frag->page,
865 frag->page_offset, frag->size, 855 frag->page_offset, frag->size,
866 PCI_DMA_TODEVICE); 856 PCI_DMA_TODEVICE);
867 if (likely(!pci_dma_mapping_error(efx->pci_dev, st->ifc.unmap_addr))) { 857 if (likely(!pci_dma_mapping_error(efx->pci_dev, st->unmap_addr))) {
868 st->ifc.unmap_single = false; 858 st->unmap_single = false;
869 st->ifc.unmap_len = frag->size; 859 st->unmap_len = frag->size;
870 st->ifc.len = frag->size; 860 st->in_len = frag->size;
871 st->ifc.dma_addr = st->ifc.unmap_addr; 861 st->dma_addr = st->unmap_addr;
872 return 0; 862 return 0;
873 } 863 }
874 return -ENOMEM; 864 return -ENOMEM;
@@ -878,16 +868,16 @@ static inline int
878tso_get_head_fragment(struct tso_state *st, struct efx_nic *efx, 868tso_get_head_fragment(struct tso_state *st, struct efx_nic *efx,
879 const struct sk_buff *skb) 869 const struct sk_buff *skb)
880{ 870{
881 int hl = st->p.header_length; 871 int hl = st->header_len;
882 int len = skb_headlen(skb) - hl; 872 int len = skb_headlen(skb) - hl;
883 873
884 st->ifc.unmap_addr = pci_map_single(efx->pci_dev, skb->data + hl, 874 st->unmap_addr = pci_map_single(efx->pci_dev, skb->data + hl,
885 len, PCI_DMA_TODEVICE); 875 len, PCI_DMA_TODEVICE);
886 if (likely(!pci_dma_mapping_error(efx->pci_dev, st->ifc.unmap_addr))) { 876 if (likely(!pci_dma_mapping_error(efx->pci_dev, st->unmap_addr))) {
887 st->ifc.unmap_single = true; 877 st->unmap_single = true;
888 st->ifc.unmap_len = len; 878 st->unmap_len = len;
889 st->ifc.len = len; 879 st->in_len = len;
890 st->ifc.dma_addr = st->ifc.unmap_addr; 880 st->dma_addr = st->unmap_addr;
891 return 0; 881 return 0;
892 } 882 }
893 return -ENOMEM; 883 return -ENOMEM;
@@ -911,38 +901,38 @@ static inline int tso_fill_packet_with_fragment(struct efx_tx_queue *tx_queue,
911 struct efx_tx_buffer *buffer; 901 struct efx_tx_buffer *buffer;
912 int n, end_of_packet, rc; 902 int n, end_of_packet, rc;
913 903
914 if (st->ifc.len == 0) 904 if (st->in_len == 0)
915 return 0; 905 return 0;
916 if (st->packet_space == 0) 906 if (st->packet_space == 0)
917 return 0; 907 return 0;
918 908
919 EFX_BUG_ON_PARANOID(st->ifc.len <= 0); 909 EFX_BUG_ON_PARANOID(st->in_len <= 0);
920 EFX_BUG_ON_PARANOID(st->packet_space <= 0); 910 EFX_BUG_ON_PARANOID(st->packet_space <= 0);
921 911
922 n = min(st->ifc.len, st->packet_space); 912 n = min(st->in_len, st->packet_space);
923 913
924 st->packet_space -= n; 914 st->packet_space -= n;
925 st->remaining_len -= n; 915 st->out_len -= n;
926 st->ifc.len -= n; 916 st->in_len -= n;
927 917
928 rc = efx_tx_queue_insert(tx_queue, st->ifc.dma_addr, n, &buffer); 918 rc = efx_tx_queue_insert(tx_queue, st->dma_addr, n, &buffer);
929 if (likely(rc == 0)) { 919 if (likely(rc == 0)) {
930 if (st->remaining_len == 0) 920 if (st->out_len == 0)
931 /* Transfer ownership of the skb */ 921 /* Transfer ownership of the skb */
932 buffer->skb = skb; 922 buffer->skb = skb;
933 923
934 end_of_packet = st->remaining_len == 0 || st->packet_space == 0; 924 end_of_packet = st->out_len == 0 || st->packet_space == 0;
935 buffer->continuation = !end_of_packet; 925 buffer->continuation = !end_of_packet;
936 926
937 if (st->ifc.len == 0) { 927 if (st->in_len == 0) {
938 /* Transfer ownership of the pci mapping */ 928 /* Transfer ownership of the pci mapping */
939 buffer->unmap_len = st->ifc.unmap_len; 929 buffer->unmap_len = st->unmap_len;
940 buffer->unmap_single = st->ifc.unmap_single; 930 buffer->unmap_single = st->unmap_single;
941 st->ifc.unmap_len = 0; 931 st->unmap_len = 0;
942 } 932 }
943 } 933 }
944 934
945 st->ifc.dma_addr += n; 935 st->dma_addr += n;
946 return rc; 936 return rc;
947} 937}
948 938
@@ -967,7 +957,7 @@ static inline int tso_start_new_packet(struct efx_tx_queue *tx_queue,
967 u8 *header; 957 u8 *header;
968 958
969 /* Allocate a DMA-mapped header buffer. */ 959 /* Allocate a DMA-mapped header buffer. */
970 if (likely(TSOH_SIZE(st->p.header_length) <= TSOH_STD_SIZE)) { 960 if (likely(TSOH_SIZE(st->header_len) <= TSOH_STD_SIZE)) {
971 if (tx_queue->tso_headers_free == NULL) { 961 if (tx_queue->tso_headers_free == NULL) {
972 if (efx_tsoh_block_alloc(tx_queue)) 962 if (efx_tsoh_block_alloc(tx_queue))
973 return -1; 963 return -1;
@@ -978,7 +968,7 @@ static inline int tso_start_new_packet(struct efx_tx_queue *tx_queue,
978 tsoh->unmap_len = 0; 968 tsoh->unmap_len = 0;
979 } else { 969 } else {
980 tx_queue->tso_long_headers++; 970 tx_queue->tso_long_headers++;
981 tsoh = efx_tsoh_heap_alloc(tx_queue, st->p.header_length); 971 tsoh = efx_tsoh_heap_alloc(tx_queue, st->header_len);
982 if (unlikely(!tsoh)) 972 if (unlikely(!tsoh))
983 return -1; 973 return -1;
984 } 974 }
@@ -988,33 +978,32 @@ static inline int tso_start_new_packet(struct efx_tx_queue *tx_queue,
988 tsoh_iph = (struct iphdr *)(header + SKB_IPV4_OFF(skb)); 978 tsoh_iph = (struct iphdr *)(header + SKB_IPV4_OFF(skb));
989 979
990 /* Copy and update the headers. */ 980 /* Copy and update the headers. */
991 memcpy(header, skb->data, st->p.header_length); 981 memcpy(header, skb->data, st->header_len);
992 982
993 tsoh_th->seq = htonl(st->seqnum); 983 tsoh_th->seq = htonl(st->seqnum);
994 st->seqnum += skb_shinfo(skb)->gso_size; 984 st->seqnum += skb_shinfo(skb)->gso_size;
995 if (st->remaining_len > skb_shinfo(skb)->gso_size) { 985 if (st->out_len > skb_shinfo(skb)->gso_size) {
996 /* This packet will not finish the TSO burst. */ 986 /* This packet will not finish the TSO burst. */
997 ip_length = st->p.full_packet_size - ETH_HDR_LEN(skb); 987 ip_length = st->full_packet_size - ETH_HDR_LEN(skb);
998 tsoh_th->fin = 0; 988 tsoh_th->fin = 0;
999 tsoh_th->psh = 0; 989 tsoh_th->psh = 0;
1000 } else { 990 } else {
1001 /* This packet will be the last in the TSO burst. */ 991 /* This packet will be the last in the TSO burst. */
1002 ip_length = (st->p.header_length - ETH_HDR_LEN(skb) 992 ip_length = st->header_len - ETH_HDR_LEN(skb) + st->out_len;
1003 + st->remaining_len);
1004 tsoh_th->fin = tcp_hdr(skb)->fin; 993 tsoh_th->fin = tcp_hdr(skb)->fin;
1005 tsoh_th->psh = tcp_hdr(skb)->psh; 994 tsoh_th->psh = tcp_hdr(skb)->psh;
1006 } 995 }
1007 tsoh_iph->tot_len = htons(ip_length); 996 tsoh_iph->tot_len = htons(ip_length);
1008 997
1009 /* Linux leaves suitable gaps in the IP ID space for us to fill. */ 998 /* Linux leaves suitable gaps in the IP ID space for us to fill. */
1010 tsoh_iph->id = htons(st->p.ipv4_id); 999 tsoh_iph->id = htons(st->ipv4_id);
1011 st->p.ipv4_id++; 1000 st->ipv4_id++;
1012 1001
1013 st->packet_space = skb_shinfo(skb)->gso_size; 1002 st->packet_space = skb_shinfo(skb)->gso_size;
1014 ++tx_queue->tso_packets; 1003 ++tx_queue->tso_packets;
1015 1004
1016 /* Form a descriptor for this header. */ 1005 /* Form a descriptor for this header. */
1017 efx_tso_put_header(tx_queue, tsoh, st->p.header_length); 1006 efx_tso_put_header(tx_queue, tsoh, st->header_len);
1018 1007
1019 return 0; 1008 return 0;
1020} 1009}
@@ -1048,7 +1037,7 @@ static int efx_enqueue_skb_tso(struct efx_tx_queue *tx_queue,
1048 /* Assume that skb header area contains exactly the headers, and 1037 /* Assume that skb header area contains exactly the headers, and
1049 * all payload is in the frag list. 1038 * all payload is in the frag list.
1050 */ 1039 */
1051 if (skb_headlen(skb) == state.p.header_length) { 1040 if (skb_headlen(skb) == state.header_len) {
1052 /* Grab the first payload fragment. */ 1041 /* Grab the first payload fragment. */
1053 EFX_BUG_ON_PARANOID(skb_shinfo(skb)->nr_frags < 1); 1042 EFX_BUG_ON_PARANOID(skb_shinfo(skb)->nr_frags < 1);
1054 frag_i = 0; 1043 frag_i = 0;
@@ -1072,7 +1061,7 @@ static int efx_enqueue_skb_tso(struct efx_tx_queue *tx_queue,
1072 goto stop; 1061 goto stop;
1073 1062
1074 /* Move onto the next fragment? */ 1063 /* Move onto the next fragment? */
1075 if (state.ifc.len == 0) { 1064 if (state.in_len == 0) {
1076 if (++frag_i >= skb_shinfo(skb)->nr_frags) 1065 if (++frag_i >= skb_shinfo(skb)->nr_frags)
1077 /* End of payload reached. */ 1066 /* End of payload reached. */
1078 break; 1067 break;
@@ -1108,13 +1097,13 @@ static int efx_enqueue_skb_tso(struct efx_tx_queue *tx_queue,
1108 1097
1109 unwind: 1098 unwind:
1110 /* Free the DMA mapping we were in the process of writing out */ 1099 /* Free the DMA mapping we were in the process of writing out */
1111 if (state.ifc.unmap_len) { 1100 if (state.unmap_len) {
1112 if (state.ifc.unmap_single) 1101 if (state.unmap_single)
1113 pci_unmap_single(efx->pci_dev, state.ifc.unmap_addr, 1102 pci_unmap_single(efx->pci_dev, state.unmap_addr,
1114 state.ifc.unmap_len, PCI_DMA_TODEVICE); 1103 state.unmap_len, PCI_DMA_TODEVICE);
1115 else 1104 else
1116 pci_unmap_page(efx->pci_dev, state.ifc.unmap_addr, 1105 pci_unmap_page(efx->pci_dev, state.unmap_addr,
1117 state.ifc.unmap_len, PCI_DMA_TODEVICE); 1106 state.unmap_len, PCI_DMA_TODEVICE);
1118 } 1107 }
1119 1108
1120 efx_enqueue_unwind(tx_queue); 1109 efx_enqueue_unwind(tx_queue);