Skip to content

Commit 120d75e

Browse files
madalinbucurdavem330
authored andcommitted
dpaa_eth: fix SG mapping
An issue in the code mapping the skb fragments into scatter-gather frames was evidentiated by netperf TCP_SENDFILE tests. The size was set wrong for all fragments but the first, affecting the transmission of any skb with more than one fragment. Signed-off-by: Madalin Bucur <[email protected]> Signed-off-by: David S. Miller <[email protected]>
1 parent afdd6ae commit 120d75e

File tree

1 file changed

+13
-10
lines changed

1 file changed

+13
-10
lines changed

drivers/net/ethernet/freescale/dpaa/dpaa_eth.c

Lines changed: 13 additions & 10 deletions
Original file line numberDiff line numberDiff line change
@@ -1916,8 +1916,10 @@ static int skb_to_sg_fd(struct dpaa_priv *priv,
19161916
goto csum_failed;
19171917
}
19181918

1919+
/* SGT[0] is used by the linear part */
19191920
sgt = (struct qm_sg_entry *)(sgt_buf + priv->tx_headroom);
1920-
qm_sg_entry_set_len(&sgt[0], skb_headlen(skb));
1921+
frag_len = skb_headlen(skb);
1922+
qm_sg_entry_set_len(&sgt[0], frag_len);
19211923
sgt[0].bpid = FSL_DPAA_BPID_INV;
19221924
sgt[0].offset = 0;
19231925
addr = dma_map_single(dev, skb->data,
@@ -1930,9 +1932,9 @@ static int skb_to_sg_fd(struct dpaa_priv *priv,
19301932
qm_sg_entry_set64(&sgt[0], addr);
19311933

19321934
/* populate the rest of SGT entries */
1933-
frag = &skb_shinfo(skb)->frags[0];
1934-
frag_len = frag->size;
1935-
for (i = 1; i <= nr_frags; i++, frag++) {
1935+
for (i = 0; i < nr_frags; i++) {
1936+
frag = &skb_shinfo(skb)->frags[i];
1937+
frag_len = frag->size;
19361938
WARN_ON(!skb_frag_page(frag));
19371939
addr = skb_frag_dma_map(dev, frag, 0,
19381940
frag_len, dma_dir);
@@ -1942,15 +1944,16 @@ static int skb_to_sg_fd(struct dpaa_priv *priv,
19421944
goto sg_map_failed;
19431945
}
19441946

1945-
qm_sg_entry_set_len(&sgt[i], frag_len);
1946-
sgt[i].bpid = FSL_DPAA_BPID_INV;
1947-
sgt[i].offset = 0;
1947+
qm_sg_entry_set_len(&sgt[i + 1], frag_len);
1948+
sgt[i + 1].bpid = FSL_DPAA_BPID_INV;
1949+
sgt[i + 1].offset = 0;
19481950

19491951
/* keep the offset in the address */
1950-
qm_sg_entry_set64(&sgt[i], addr);
1951-
frag_len = frag->size;
1952+
qm_sg_entry_set64(&sgt[i + 1], addr);
19521953
}
1953-
qm_sg_entry_set_f(&sgt[i - 1], frag_len);
1954+
1955+
/* Set the final bit in the last used entry of the SGT */
1956+
qm_sg_entry_set_f(&sgt[nr_frags], frag_len);
19541957

19551958
qm_fd_set_sg(fd, priv->tx_headroom, skb->len);
19561959

0 commit comments

Comments
 (0)