mv643xx_eth: enable hardware TX checksumming with vlan tags
Although mv643xx_eth has no hardware support for inserting a vlan tag by twiddling some bits in the TX descriptor, it does support hardware TX checksumming on packets where the IP header starts {a limited set of values other than 14} bytes into the packet. This patch sets mv643xx_eth's ->vlan_features to NETIF_F_SG | NETIF_F_IP_CSUM, which prevents the stack from checksumming vlan'ed packets in software, and if vlan tags are present on a transmitted packet, notifies the hardware of this fact by toggling the right bits in the TX descriptor. Signed-off-by: Lennert Buytenhek <buytenh@marvell.com>
This commit is contained in:
parent
2f7eb47a7b
commit
e32b661750
1 changed files with 28 additions and 1 deletions
|
@ -235,6 +235,8 @@ struct tx_desc {
|
||||||
#define GEN_IP_V4_CHECKSUM 0x00040000
|
#define GEN_IP_V4_CHECKSUM 0x00040000
|
||||||
#define GEN_TCP_UDP_CHECKSUM 0x00020000
|
#define GEN_TCP_UDP_CHECKSUM 0x00020000
|
||||||
#define UDP_FRAME 0x00010000
|
#define UDP_FRAME 0x00010000
|
||||||
|
#define MAC_HDR_EXTRA_4_BYTES 0x00008000
|
||||||
|
#define MAC_HDR_EXTRA_8_BYTES 0x00000200
|
||||||
|
|
||||||
#define TX_IHL_SHIFT 11
|
#define TX_IHL_SHIFT 11
|
||||||
|
|
||||||
|
@ -757,12 +759,36 @@ static void txq_submit_skb(struct tx_queue *txq, struct sk_buff *skb)
|
||||||
desc->buf_ptr = dma_map_single(NULL, skb->data, length, DMA_TO_DEVICE);
|
desc->buf_ptr = dma_map_single(NULL, skb->data, length, DMA_TO_DEVICE);
|
||||||
|
|
||||||
if (skb->ip_summed == CHECKSUM_PARTIAL) {
|
if (skb->ip_summed == CHECKSUM_PARTIAL) {
|
||||||
BUG_ON(skb->protocol != htons(ETH_P_IP));
|
int mac_hdr_len;
|
||||||
|
|
||||||
|
BUG_ON(skb->protocol != htons(ETH_P_IP) &&
|
||||||
|
skb->protocol != htons(ETH_P_8021Q));
|
||||||
|
|
||||||
cmd_sts |= GEN_TCP_UDP_CHECKSUM |
|
cmd_sts |= GEN_TCP_UDP_CHECKSUM |
|
||||||
GEN_IP_V4_CHECKSUM |
|
GEN_IP_V4_CHECKSUM |
|
||||||
ip_hdr(skb)->ihl << TX_IHL_SHIFT;
|
ip_hdr(skb)->ihl << TX_IHL_SHIFT;
|
||||||
|
|
||||||
|
mac_hdr_len = (void *)ip_hdr(skb) - (void *)skb->data;
|
||||||
|
switch (mac_hdr_len - ETH_HLEN) {
|
||||||
|
case 0:
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
cmd_sts |= MAC_HDR_EXTRA_4_BYTES;
|
||||||
|
break;
|
||||||
|
case 8:
|
||||||
|
cmd_sts |= MAC_HDR_EXTRA_8_BYTES;
|
||||||
|
break;
|
||||||
|
case 12:
|
||||||
|
cmd_sts |= MAC_HDR_EXTRA_4_BYTES;
|
||||||
|
cmd_sts |= MAC_HDR_EXTRA_8_BYTES;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
if (net_ratelimit())
|
||||||
|
dev_printk(KERN_ERR, &txq_to_mp(txq)->dev->dev,
|
||||||
|
"mac header length is %d?!\n", mac_hdr_len);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
switch (ip_hdr(skb)->protocol) {
|
switch (ip_hdr(skb)->protocol) {
|
||||||
case IPPROTO_UDP:
|
case IPPROTO_UDP:
|
||||||
cmd_sts |= UDP_FRAME;
|
cmd_sts |= UDP_FRAME;
|
||||||
|
@ -2565,6 +2591,7 @@ static int mv643xx_eth_probe(struct platform_device *pdev)
|
||||||
* have to map the buffers to ISA memory which is only 16 MB
|
* have to map the buffers to ISA memory which is only 16 MB
|
||||||
*/
|
*/
|
||||||
dev->features = NETIF_F_SG | NETIF_F_IP_CSUM;
|
dev->features = NETIF_F_SG | NETIF_F_IP_CSUM;
|
||||||
|
dev->vlan_features = NETIF_F_SG | NETIF_F_IP_CSUM;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
SET_NETDEV_DEV(dev, &pdev->dev);
|
SET_NETDEV_DEV(dev, &pdev->dev);
|
||||||
|
|
Loading…
Reference in a new issue