aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/net/wireless/p54
diff options
context:
space:
mode:
authorChristian Lamparter <chunkeey@web.de>2008-09-01 16:48:41 -0400
committerJohn W. Linville <linville@tuxdriver.com>2008-09-05 16:17:47 -0400
commit4e416a6f49b710bfe162f0cb24bc68c74493d2a0 (patch)
tree76e4a5e4a7e1cd65b142d841e5134404f970a99d /drivers/net/wireless/p54
parent0c25970dc1b0d46f2357e7c4b267ab7b93eb7cdd (diff)
p54: enhance firmware parser to reduce memory waste
This patch greatly reduces one of biggest memory waste in the driver. The firmware headers provides the right values for extra head-/tailroom and mtu size which are usually much lower than the old hardcoded ones. Signed-off-by: Christian Lamparter <chunkeey@web.de> Signed-off-by: John W. Linville <linville@tuxdriver.com>
Diffstat (limited to 'drivers/net/wireless/p54')
-rw-r--r--drivers/net/wireless/p54/p54.h6
-rw-r--r--drivers/net/wireless/p54/p54common.c36
-rw-r--r--drivers/net/wireless/p54/p54common.h11
-rw-r--r--drivers/net/wireless/p54/p54pci.c23
-rw-r--r--drivers/net/wireless/p54/p54usb.c20
5 files changed, 69 insertions, 27 deletions
diff --git a/drivers/net/wireless/p54/p54.h b/drivers/net/wireless/p54/p54.h
index fca8762fa069..b03d13edc61f 100644
--- a/drivers/net/wireless/p54/p54.h
+++ b/drivers/net/wireless/p54/p54.h
@@ -39,7 +39,6 @@ struct p54_control_hdr {
39} __attribute__ ((packed)); 39} __attribute__ ((packed));
40 40
41#define EEPROM_READBACK_LEN (sizeof(struct p54_control_hdr) + 4 /* p54_eeprom_lm86 */) 41#define EEPROM_READBACK_LEN (sizeof(struct p54_control_hdr) + 4 /* p54_eeprom_lm86 */)
42#define MAX_RX_SIZE (IEEE80211_MAX_RTS_THRESHOLD + sizeof(struct p54_control_hdr) + 20 /* length of struct p54_rx_hdr */ + 16 )
43 42
44#define ISL38XX_DEV_FIRMWARE_ADDR 0x20000 43#define ISL38XX_DEV_FIRMWARE_ADDR 0x20000
45 44
@@ -53,6 +52,9 @@ struct p54_common {
53 void (*stop)(struct ieee80211_hw *dev); 52 void (*stop)(struct ieee80211_hw *dev);
54 int mode; 53 int mode;
55 u16 seqno; 54 u16 seqno;
55 u16 rx_mtu;
56 u8 headroom;
57 u8 tailroom;
56 struct mutex conf_mutex; 58 struct mutex conf_mutex;
57 u8 mac_addr[ETH_ALEN]; 59 u8 mac_addr[ETH_ALEN];
58 u8 bssid[ETH_ALEN]; 60 u8 bssid[ETH_ALEN];
@@ -70,7 +72,7 @@ struct p54_common {
70}; 72};
71 73
72int p54_rx(struct ieee80211_hw *dev, struct sk_buff *skb); 74int p54_rx(struct ieee80211_hw *dev, struct sk_buff *skb);
73void p54_parse_firmware(struct ieee80211_hw *dev, const struct firmware *fw); 75int p54_parse_firmware(struct ieee80211_hw *dev, const struct firmware *fw);
74int p54_parse_eeprom(struct ieee80211_hw *dev, void *eeprom, int len); 76int p54_parse_eeprom(struct ieee80211_hw *dev, void *eeprom, int len);
75void p54_fill_eeprom_readback(struct p54_control_hdr *hdr); 77void p54_fill_eeprom_readback(struct p54_control_hdr *hdr);
76struct ieee80211_hw *p54_init_common(size_t priv_data_len); 78struct ieee80211_hw *p54_init_common(size_t priv_data_len);
diff --git a/drivers/net/wireless/p54/p54common.c b/drivers/net/wireless/p54/p54common.c
index 6da98e6e6a9a..fa61749b467b 100644
--- a/drivers/net/wireless/p54/p54common.c
+++ b/drivers/net/wireless/p54/p54common.c
@@ -66,8 +66,7 @@ static struct ieee80211_supported_band band_2GHz = {
66 .n_bitrates = ARRAY_SIZE(p54_rates), 66 .n_bitrates = ARRAY_SIZE(p54_rates),
67}; 67};
68 68
69 69int p54_parse_firmware(struct ieee80211_hw *dev, const struct firmware *fw)
70void p54_parse_firmware(struct ieee80211_hw *dev, const struct firmware *fw)
71{ 70{
72 struct p54_common *priv = dev->priv; 71 struct p54_common *priv = dev->priv;
73 struct bootrec_exp_if *exp_if; 72 struct bootrec_exp_if *exp_if;
@@ -79,7 +78,7 @@ void p54_parse_firmware(struct ieee80211_hw *dev, const struct firmware *fw)
79 int i; 78 int i;
80 79
81 if (priv->rx_start) 80 if (priv->rx_start)
82 return; 81 return 0;
83 82
84 while (data < end_data && *data) 83 while (data < end_data && *data)
85 data++; 84 data++;
@@ -117,11 +116,22 @@ void p54_parse_firmware(struct ieee80211_hw *dev, const struct firmware *fw)
117 if (strnlen((unsigned char*)bootrec->data, 24) < 24) 116 if (strnlen((unsigned char*)bootrec->data, 24) < 24)
118 fw_version = (unsigned char*)bootrec->data; 117 fw_version = (unsigned char*)bootrec->data;
119 break; 118 break;
120 case BR_CODE_DESCR: 119 case BR_CODE_DESCR: {
121 priv->rx_start = le32_to_cpu(((__le32 *)bootrec->data)[1]); 120 struct bootrec_desc *desc =
121 (struct bootrec_desc *)bootrec->data;
122 priv->rx_start = le32_to_cpu(desc->rx_start);
122 /* FIXME add sanity checking */ 123 /* FIXME add sanity checking */
123 priv->rx_end = le32_to_cpu(((__le32 *)bootrec->data)[2]) - 0x3500; 124 priv->rx_end = le32_to_cpu(desc->rx_end) - 0x3500;
125 priv->headroom = desc->headroom;
126 priv->tailroom = desc->tailroom;
127 if (bootrec->len == 11)
128 priv->rx_mtu = (size_t) le16_to_cpu(
129 (__le16)bootrec->data[10]);
130 else
131 priv->rx_mtu = (size_t)
132 0x620 - priv->tx_hdr_len;
124 break; 133 break;
134 }
125 case BR_CODE_EXPOSED_IF: 135 case BR_CODE_EXPOSED_IF:
126 exp_if = (struct bootrec_exp_if *) bootrec->data; 136 exp_if = (struct bootrec_exp_if *) bootrec->data;
127 for (i = 0; i < (len * sizeof(*exp_if) / 4); i++) 137 for (i = 0; i < (len * sizeof(*exp_if) / 4); i++)
@@ -152,6 +162,8 @@ void p54_parse_firmware(struct ieee80211_hw *dev, const struct firmware *fw)
152 priv->tx_stats[7].limit = 1; 162 priv->tx_stats[7].limit = 1;
153 dev->queues = 4; 163 dev->queues = 4;
154 } 164 }
165
166 return 0;
155} 167}
156EXPORT_SYMBOL_GPL(p54_parse_firmware); 168EXPORT_SYMBOL_GPL(p54_parse_firmware);
157 169
@@ -428,7 +440,7 @@ static void p54_rx_frame_sent(struct ieee80211_hw *dev, struct sk_buff *skb)
428 struct p54_control_hdr *hdr = (struct p54_control_hdr *) skb->data; 440 struct p54_control_hdr *hdr = (struct p54_control_hdr *) skb->data;
429 struct p54_frame_sent_hdr *payload = (struct p54_frame_sent_hdr *) hdr->data; 441 struct p54_frame_sent_hdr *payload = (struct p54_frame_sent_hdr *) hdr->data;
430 struct sk_buff *entry = (struct sk_buff *) priv->tx_queue.next; 442 struct sk_buff *entry = (struct sk_buff *) priv->tx_queue.next;
431 u32 addr = le32_to_cpu(hdr->req_id) - 0x70; 443 u32 addr = le32_to_cpu(hdr->req_id) - priv->headroom;
432 struct memrecord *range = NULL; 444 struct memrecord *range = NULL;
433 u32 freed = 0; 445 u32 freed = 0;
434 u32 last_addr = priv->rx_start; 446 u32 last_addr = priv->rx_start;
@@ -550,7 +562,7 @@ static void p54_assign_address(struct ieee80211_hw *dev, struct sk_buff *skb,
550 u32 target_addr = priv->rx_start; 562 u32 target_addr = priv->rx_start;
551 unsigned long flags; 563 unsigned long flags;
552 unsigned int left; 564 unsigned int left;
553 len = (len + 0x170 + 3) & ~0x3; /* 0x70 headroom, 0x100 tailroom */ 565 len = (len + priv->headroom + priv->tailroom + 3) & ~0x3;
554 566
555 spin_lock_irqsave(&priv->tx_queue.lock, flags); 567 spin_lock_irqsave(&priv->tx_queue.lock, flags);
556 left = skb_queue_len(&priv->tx_queue); 568 left = skb_queue_len(&priv->tx_queue);
@@ -585,13 +597,14 @@ static void p54_assign_address(struct ieee80211_hw *dev, struct sk_buff *skb,
585 range->start_addr = target_addr; 597 range->start_addr = target_addr;
586 range->end_addr = target_addr + len; 598 range->end_addr = target_addr + len;
587 __skb_queue_after(&priv->tx_queue, target_skb, skb); 599 __skb_queue_after(&priv->tx_queue, target_skb, skb);
588 if (largest_hole < IEEE80211_MAX_RTS_THRESHOLD + 0x170 + 600 if (largest_hole < priv->rx_mtu + priv->headroom +
601 priv->tailroom +
589 sizeof(struct p54_control_hdr)) 602 sizeof(struct p54_control_hdr))
590 ieee80211_stop_queues(dev); 603 ieee80211_stop_queues(dev);
591 } 604 }
592 spin_unlock_irqrestore(&priv->tx_queue.lock, flags); 605 spin_unlock_irqrestore(&priv->tx_queue.lock, flags);
593 606
594 data->req_id = cpu_to_le32(target_addr + 0x70); 607 data->req_id = cpu_to_le32(target_addr + priv->headroom);
595} 608}
596 609
597static int p54_tx(struct ieee80211_hw *dev, struct sk_buff *skb) 610static int p54_tx(struct ieee80211_hw *dev, struct sk_buff *skb)
@@ -704,7 +717,7 @@ static int p54_set_filter(struct ieee80211_hw *dev, u16 filter_type,
704 filter->antenna = antenna; 717 filter->antenna = antenna;
705 filter->magic3 = cpu_to_le32(magic3); 718 filter->magic3 = cpu_to_le32(magic3);
706 filter->rx_addr = cpu_to_le32(priv->rx_end); 719 filter->rx_addr = cpu_to_le32(priv->rx_end);
707 filter->max_rx = cpu_to_le16(0x0620); /* FIXME: for usb ver 1.. maybe */ 720 filter->max_rx = cpu_to_le16(priv->rx_mtu);
708 filter->rxhw = priv->rxhw; 721 filter->rxhw = priv->rxhw;
709 filter->magic8 = cpu_to_le16(magic8); 722 filter->magic8 = cpu_to_le16(magic8);
710 filter->magic9 = cpu_to_le16(magic9); 723 filter->magic9 = cpu_to_le16(magic9);
@@ -1084,7 +1097,6 @@ struct ieee80211_hw *p54_init_common(size_t priv_data_len)
1084 priv->tx_stats[3].limit = 1; 1097 priv->tx_stats[3].limit = 1;
1085 priv->tx_stats[4].limit = 5; 1098 priv->tx_stats[4].limit = 5;
1086 dev->queues = 1; 1099 dev->queues = 1;
1087
1088 dev->extra_tx_headroom = sizeof(struct p54_control_hdr) + 4 + 1100 dev->extra_tx_headroom = sizeof(struct p54_control_hdr) + 4 +
1089 sizeof(struct p54_tx_control_allocdata); 1101 sizeof(struct p54_tx_control_allocdata);
1090 1102
diff --git a/drivers/net/wireless/p54/p54common.h b/drivers/net/wireless/p54/p54common.h
index a79c1a146917..5f2af5152505 100644
--- a/drivers/net/wireless/p54/p54common.h
+++ b/drivers/net/wireless/p54/p54common.h
@@ -29,6 +29,17 @@ struct bootrec_exp_if {
29 __le16 top_compat; 29 __le16 top_compat;
30} __attribute__((packed)); 30} __attribute__((packed));
31 31
32struct bootrec_desc {
33 __le16 modes;
34 __le16 flags;
35 __le32 rx_start;
36 __le32 rx_end;
37 u8 headroom;
38 u8 tailroom;
39 u8 unimportant[6];
40 u8 rates[16];
41} __attribute__((packed));
42
32#define BR_CODE_MIN 0x80000000 43#define BR_CODE_MIN 0x80000000
33#define BR_CODE_COMPONENT_ID 0x80000001 44#define BR_CODE_COMPONENT_ID 0x80000001
34#define BR_CODE_COMPONENT_VERSION 0x80000002 45#define BR_CODE_COMPONENT_VERSION 0x80000002
diff --git a/drivers/net/wireless/p54/p54pci.c b/drivers/net/wireless/p54/p54pci.c
index a0395af74c2d..fdfc7189f0ff 100644
--- a/drivers/net/wireless/p54/p54pci.c
+++ b/drivers/net/wireless/p54/p54pci.c
@@ -81,7 +81,11 @@ static int p54p_upload_firmware(struct ieee80211_hw *dev)
81 return err; 81 return err;
82 } 82 }
83 83
84 p54_parse_firmware(dev, fw_entry); 84 err = p54_parse_firmware(dev, fw_entry);
85 if (err) {
86 release_firmware(fw_entry);
87 return err;
88 }
85 89
86 data = (__le32 *) fw_entry->data; 90 data = (__le32 *) fw_entry->data;
87 remains = fw_entry->size; 91 remains = fw_entry->size;
@@ -258,17 +262,17 @@ static void p54p_refill_rx_ring(struct ieee80211_hw *dev,
258 if (!desc->host_addr) { 262 if (!desc->host_addr) {
259 struct sk_buff *skb; 263 struct sk_buff *skb;
260 dma_addr_t mapping; 264 dma_addr_t mapping;
261 skb = dev_alloc_skb(MAX_RX_SIZE); 265 skb = dev_alloc_skb(priv->common.rx_mtu + 32);
262 if (!skb) 266 if (!skb)
263 break; 267 break;
264 268
265 mapping = pci_map_single(priv->pdev, 269 mapping = pci_map_single(priv->pdev,
266 skb_tail_pointer(skb), 270 skb_tail_pointer(skb),
267 MAX_RX_SIZE, 271 priv->common.rx_mtu + 32,
268 PCI_DMA_FROMDEVICE); 272 PCI_DMA_FROMDEVICE);
269 desc->host_addr = cpu_to_le32(mapping); 273 desc->host_addr = cpu_to_le32(mapping);
270 desc->device_addr = 0; // FIXME: necessary? 274 desc->device_addr = 0; // FIXME: necessary?
271 desc->len = cpu_to_le16(MAX_RX_SIZE); 275 desc->len = cpu_to_le16(priv->common.rx_mtu + 32);
272 desc->flags = 0; 276 desc->flags = 0;
273 rx_buf[i] = skb; 277 rx_buf[i] = skb;
274 } 278 }
@@ -311,12 +315,13 @@ static void p54p_check_rx_ring(struct ieee80211_hw *dev, u32 *index,
311 if (p54_rx(dev, skb)) { 315 if (p54_rx(dev, skb)) {
312 pci_unmap_single(priv->pdev, 316 pci_unmap_single(priv->pdev,
313 le32_to_cpu(desc->host_addr), 317 le32_to_cpu(desc->host_addr),
314 MAX_RX_SIZE, PCI_DMA_FROMDEVICE); 318 priv->common.rx_mtu + 32,
319 PCI_DMA_FROMDEVICE);
315 rx_buf[i] = NULL; 320 rx_buf[i] = NULL;
316 desc->host_addr = 0; 321 desc->host_addr = 0;
317 } else { 322 } else {
318 skb_trim(skb, 0); 323 skb_trim(skb, 0);
319 desc->len = cpu_to_le16(MAX_RX_SIZE); 324 desc->len = cpu_to_le16(priv->common.rx_mtu + 32);
320 } 325 }
321 326
322 i++; 327 i++;
@@ -534,7 +539,8 @@ static void p54p_stop(struct ieee80211_hw *dev)
534 if (desc->host_addr) 539 if (desc->host_addr)
535 pci_unmap_single(priv->pdev, 540 pci_unmap_single(priv->pdev,
536 le32_to_cpu(desc->host_addr), 541 le32_to_cpu(desc->host_addr),
537 MAX_RX_SIZE, PCI_DMA_FROMDEVICE); 542 priv->common.rx_mtu + 32,
543 PCI_DMA_FROMDEVICE);
538 kfree_skb(priv->rx_buf_data[i]); 544 kfree_skb(priv->rx_buf_data[i]);
539 priv->rx_buf_data[i] = NULL; 545 priv->rx_buf_data[i] = NULL;
540 } 546 }
@@ -544,7 +550,8 @@ static void p54p_stop(struct ieee80211_hw *dev)
544 if (desc->host_addr) 550 if (desc->host_addr)
545 pci_unmap_single(priv->pdev, 551 pci_unmap_single(priv->pdev,
546 le32_to_cpu(desc->host_addr), 552 le32_to_cpu(desc->host_addr),
547 MAX_RX_SIZE, PCI_DMA_FROMDEVICE); 553 priv->common.rx_mtu + 32,
554 PCI_DMA_FROMDEVICE);
548 kfree_skb(priv->rx_buf_mgmt[i]); 555 kfree_skb(priv->rx_buf_mgmt[i]);
549 priv->rx_buf_mgmt[i] = NULL; 556 priv->rx_buf_mgmt[i] = NULL;
550 } 557 }
diff --git a/drivers/net/wireless/p54/p54usb.c b/drivers/net/wireless/p54/p54usb.c
index 8a420df605af..4dca209a6e07 100644
--- a/drivers/net/wireless/p54/p54usb.c
+++ b/drivers/net/wireless/p54/p54usb.c
@@ -95,7 +95,7 @@ static void p54u_rx_cb(struct urb *urb)
95 skb_pull(skb, sizeof(struct net2280_tx_hdr)); 95 skb_pull(skb, sizeof(struct net2280_tx_hdr));
96 96
97 if (p54_rx(dev, skb)) { 97 if (p54_rx(dev, skb)) {
98 skb = dev_alloc_skb(MAX_RX_SIZE); 98 skb = dev_alloc_skb(priv->common.rx_mtu + 32);
99 if (unlikely(!skb)) { 99 if (unlikely(!skb)) {
100 usb_free_urb(urb); 100 usb_free_urb(urb);
101 /* TODO check rx queue length and refill *somewhere* */ 101 /* TODO check rx queue length and refill *somewhere* */
@@ -145,7 +145,7 @@ static int p54u_init_urbs(struct ieee80211_hw *dev)
145 struct p54u_rx_info *info; 145 struct p54u_rx_info *info;
146 146
147 while (skb_queue_len(&priv->rx_queue) < 32) { 147 while (skb_queue_len(&priv->rx_queue) < 32) {
148 skb = __dev_alloc_skb(MAX_RX_SIZE, GFP_KERNEL); 148 skb = __dev_alloc_skb(priv->common.rx_mtu + 32, GFP_KERNEL);
149 if (!skb) 149 if (!skb)
150 break; 150 break;
151 entry = usb_alloc_urb(0, GFP_KERNEL); 151 entry = usb_alloc_urb(0, GFP_KERNEL);
@@ -153,7 +153,10 @@ static int p54u_init_urbs(struct ieee80211_hw *dev)
153 kfree_skb(skb); 153 kfree_skb(skb);
154 break; 154 break;
155 } 155 }
156 usb_fill_bulk_urb(entry, priv->udev, usb_rcvbulkpipe(priv->udev, P54U_PIPE_DATA), skb_tail_pointer(skb), MAX_RX_SIZE, p54u_rx_cb, skb); 156 usb_fill_bulk_urb(entry, priv->udev,
157 usb_rcvbulkpipe(priv->udev, P54U_PIPE_DATA),
158 skb_tail_pointer(skb),
159 priv->common.rx_mtu + 32, p54u_rx_cb, skb);
157 info = (struct p54u_rx_info *) skb->cb; 160 info = (struct p54u_rx_info *) skb->cb;
158 info->urb = entry; 161 info->urb = entry;
159 info->dev = dev; 162 info->dev = dev;
@@ -412,7 +415,9 @@ static int p54u_upload_firmware_3887(struct ieee80211_hw *dev)
412 goto err_req_fw_failed; 415 goto err_req_fw_failed;
413 } 416 }
414 417
415 p54_parse_firmware(dev, fw_entry); 418 err = p54_parse_firmware(dev, fw_entry);
419 if (err)
420 goto err_upload_failed;
416 421
417 left = block_size = min((size_t)P54U_FW_BLOCK, fw_entry->size); 422 left = block_size = min((size_t)P54U_FW_BLOCK, fw_entry->size);
418 strcpy(buf, start_string); 423 strcpy(buf, start_string);
@@ -549,7 +554,12 @@ static int p54u_upload_firmware_net2280(struct ieee80211_hw *dev)
549 return err; 554 return err;
550 } 555 }
551 556
552 p54_parse_firmware(dev, fw_entry); 557 err = p54_parse_firmware(dev, fw_entry);
558 if (err) {
559 kfree(buf);
560 release_firmware(fw_entry);
561 return err;
562 }
553 563
554#define P54U_WRITE(type, addr, data) \ 564#define P54U_WRITE(type, addr, data) \
555 do {\ 565 do {\