diff options
Diffstat (limited to 'drivers/net')
-rw-r--r-- | drivers/net/wireless/orinoco.c | 44 | ||||
-rw-r--r-- | drivers/net/wireless/orinoco.h | 14 |
2 files changed, 20 insertions, 38 deletions
diff --git a/drivers/net/wireless/orinoco.c b/drivers/net/wireless/orinoco.c index e7d06b90afe2..4d6373814b93 100644 --- a/drivers/net/wireless/orinoco.c +++ b/drivers/net/wireless/orinoco.c | |||
@@ -421,9 +421,8 @@ static int orinoco_xmit(struct sk_buff *skb, struct net_device *dev) | |||
421 | hermes_t *hw = &priv->hw; | 421 | hermes_t *hw = &priv->hw; |
422 | int err = 0; | 422 | int err = 0; |
423 | u16 txfid = priv->txfid; | 423 | u16 txfid = priv->txfid; |
424 | char *p; | ||
425 | struct ethhdr *eh; | 424 | struct ethhdr *eh; |
426 | int data_len, data_off; | 425 | int data_off; |
427 | struct hermes_tx_descriptor desc; | 426 | struct hermes_tx_descriptor desc; |
428 | unsigned long flags; | 427 | unsigned long flags; |
429 | 428 | ||
@@ -453,8 +452,7 @@ static int orinoco_xmit(struct sk_buff *skb, struct net_device *dev) | |||
453 | } | 452 | } |
454 | 453 | ||
455 | /* Check packet length */ | 454 | /* Check packet length */ |
456 | data_len = skb->len; | 455 | if (skb->len < ETH_HLEN) |
457 | if (data_len < ETH_HLEN) | ||
458 | goto drop; | 456 | goto drop; |
459 | 457 | ||
460 | eh = (struct ethhdr *)skb->data; | 458 | eh = (struct ethhdr *)skb->data; |
@@ -477,22 +475,22 @@ static int orinoco_xmit(struct sk_buff *skb, struct net_device *dev) | |||
477 | 475 | ||
478 | /* Encapsulate Ethernet-II frames */ | 476 | /* Encapsulate Ethernet-II frames */ |
479 | if (ntohs(eh->h_proto) > ETH_DATA_LEN) { /* Ethernet-II frame */ | 477 | if (ntohs(eh->h_proto) > ETH_DATA_LEN) { /* Ethernet-II frame */ |
480 | struct header_struct hdr; | 478 | struct header_struct { |
481 | data_len = skb->len - ETH_HLEN; | 479 | struct ethhdr eth; /* 802.3 header */ |
482 | data_off = HERMES_802_3_OFFSET + sizeof(hdr); | 480 | u8 encap[6]; /* 802.2 header */ |
483 | p = skb->data + ETH_HLEN; | 481 | } __attribute__ ((packed)) hdr; |
484 | 482 | ||
485 | /* 802.3 header */ | 483 | /* Strip destination and source from the data */ |
486 | memcpy(hdr.dest, eh->h_dest, ETH_ALEN); | 484 | skb_pull(skb, 2 * ETH_ALEN); |
487 | memcpy(hdr.src, eh->h_source, ETH_ALEN); | 485 | data_off = HERMES_802_2_OFFSET + sizeof(encaps_hdr); |
488 | hdr.len = htons(data_len + ENCAPS_OVERHEAD); | 486 | |
489 | 487 | /* And move them to a separate header */ | |
490 | /* 802.2 header */ | 488 | memcpy(&hdr.eth, eh, 2 * ETH_ALEN); |
491 | memcpy(&hdr.dsap, &encaps_hdr, sizeof(encaps_hdr)); | 489 | hdr.eth.h_proto = htons(sizeof(encaps_hdr) + skb->len); |
492 | 490 | memcpy(hdr.encap, encaps_hdr, sizeof(encaps_hdr)); | |
493 | hdr.ethertype = eh->h_proto; | 491 | |
494 | err = hermes_bap_pwrite(hw, USER_BAP, &hdr, sizeof(hdr), | 492 | err = hermes_bap_pwrite(hw, USER_BAP, &hdr, sizeof(hdr), |
495 | txfid, HERMES_802_3_OFFSET); | 493 | txfid, HERMES_802_3_OFFSET); |
496 | if (err) { | 494 | if (err) { |
497 | if (net_ratelimit()) | 495 | if (net_ratelimit()) |
498 | printk(KERN_ERR "%s: Error %d writing packet " | 496 | printk(KERN_ERR "%s: Error %d writing packet " |
@@ -500,12 +498,10 @@ static int orinoco_xmit(struct sk_buff *skb, struct net_device *dev) | |||
500 | goto busy; | 498 | goto busy; |
501 | } | 499 | } |
502 | } else { /* IEEE 802.3 frame */ | 500 | } else { /* IEEE 802.3 frame */ |
503 | data_len = skb->len; | ||
504 | data_off = HERMES_802_3_OFFSET; | 501 | data_off = HERMES_802_3_OFFSET; |
505 | p = skb->data; | ||
506 | } | 502 | } |
507 | 503 | ||
508 | err = hermes_bap_pwrite(hw, USER_BAP, p, data_len, | 504 | err = hermes_bap_pwrite(hw, USER_BAP, skb->data, skb->len, |
509 | txfid, data_off); | 505 | txfid, data_off); |
510 | if (err) { | 506 | if (err) { |
511 | printk(KERN_ERR "%s: Error %d writing packet to BAP\n", | 507 | printk(KERN_ERR "%s: Error %d writing packet to BAP\n", |
@@ -527,7 +523,7 @@ static int orinoco_xmit(struct sk_buff *skb, struct net_device *dev) | |||
527 | } | 523 | } |
528 | 524 | ||
529 | dev->trans_start = jiffies; | 525 | dev->trans_start = jiffies; |
530 | stats->tx_bytes += data_off + data_len; | 526 | stats->tx_bytes += data_off + skb->len; |
531 | goto ok; | 527 | goto ok; |
532 | 528 | ||
533 | drop: | 529 | drop: |
diff --git a/drivers/net/wireless/orinoco.h b/drivers/net/wireless/orinoco.h index c6922a705e3b..ca01e459a2f0 100644 --- a/drivers/net/wireless/orinoco.h +++ b/drivers/net/wireless/orinoco.h | |||
@@ -30,20 +30,6 @@ struct orinoco_key { | |||
30 | char data[ORINOCO_MAX_KEY_SIZE]; | 30 | char data[ORINOCO_MAX_KEY_SIZE]; |
31 | } __attribute__ ((packed)); | 31 | } __attribute__ ((packed)); |
32 | 32 | ||
33 | struct header_struct { | ||
34 | /* 802.3 */ | ||
35 | u8 dest[ETH_ALEN]; | ||
36 | u8 src[ETH_ALEN]; | ||
37 | __be16 len; | ||
38 | /* 802.2 */ | ||
39 | u8 dsap; | ||
40 | u8 ssap; | ||
41 | u8 ctrl; | ||
42 | /* SNAP */ | ||
43 | u8 oui[3]; | ||
44 | unsigned short ethertype; | ||
45 | } __attribute__ ((packed)); | ||
46 | |||
47 | typedef enum { | 33 | typedef enum { |
48 | FIRMWARE_TYPE_AGERE, | 34 | FIRMWARE_TYPE_AGERE, |
49 | FIRMWARE_TYPE_INTERSIL, | 35 | FIRMWARE_TYPE_INTERSIL, |