diff options
author | David Kilroy <kilroyd@googlemail.com> | 2009-02-04 18:05:52 -0500 |
---|---|---|
committer | John W. Linville <linville@tuxdriver.com> | 2009-02-13 13:44:26 -0500 |
commit | 37a2e566f82de9a88fe119479162f9984af2180d (patch) | |
tree | 4d2e99a9b9bfdc4dfa57262a0965afc0c540653d /drivers/net/wireless/orinoco/main.c | |
parent | 4adb474b6b7e26e1318acab5e98864aa78f9b233 (diff) |
orinoco: Move firmware handling into a separate file
No functional change.
Signed-off-by: David Kilroy <kilroyd@googlemail.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
Diffstat (limited to 'drivers/net/wireless/orinoco/main.c')
-rw-r--r-- | drivers/net/wireless/orinoco/main.c | 335 |
1 files changed, 1 insertions, 334 deletions
diff --git a/drivers/net/wireless/orinoco/main.c b/drivers/net/wireless/orinoco/main.c index f0454440dd7b..f49cabd70877 100644 --- a/drivers/net/wireless/orinoco/main.c +++ b/drivers/net/wireless/orinoco/main.c | |||
@@ -83,7 +83,6 @@ | |||
83 | #include <linux/netdevice.h> | 83 | #include <linux/netdevice.h> |
84 | #include <linux/etherdevice.h> | 84 | #include <linux/etherdevice.h> |
85 | #include <linux/ethtool.h> | 85 | #include <linux/ethtool.h> |
86 | #include <linux/firmware.h> | ||
87 | #include <linux/suspend.h> | 86 | #include <linux/suspend.h> |
88 | #include <linux/if_arp.h> | 87 | #include <linux/if_arp.h> |
89 | #include <linux/wireless.h> | 88 | #include <linux/wireless.h> |
@@ -94,6 +93,7 @@ | |||
94 | #include "hermes_dld.h" | 93 | #include "hermes_dld.h" |
95 | #include "scan.h" | 94 | #include "scan.h" |
96 | #include "mic.h" | 95 | #include "mic.h" |
96 | #include "fw.h" | ||
97 | 97 | ||
98 | #include "orinoco.h" | 98 | #include "orinoco.h" |
99 | 99 | ||
@@ -310,339 +310,6 @@ static inline u8 *orinoco_get_wpa_ie(u8 *data, size_t len) | |||
310 | 310 | ||
311 | 311 | ||
312 | /********************************************************************/ | 312 | /********************************************************************/ |
313 | /* Download functionality */ | ||
314 | /********************************************************************/ | ||
315 | |||
316 | struct fw_info { | ||
317 | char *pri_fw; | ||
318 | char *sta_fw; | ||
319 | char *ap_fw; | ||
320 | u32 pda_addr; | ||
321 | u16 pda_size; | ||
322 | }; | ||
323 | |||
324 | const static struct fw_info orinoco_fw[] = { | ||
325 | { NULL, "agere_sta_fw.bin", "agere_ap_fw.bin", 0x00390000, 1000 }, | ||
326 | { NULL, "prism_sta_fw.bin", "prism_ap_fw.bin", 0, 1024 }, | ||
327 | { "symbol_sp24t_prim_fw", "symbol_sp24t_sec_fw", NULL, 0x00003100, 512 } | ||
328 | }; | ||
329 | |||
330 | /* Structure used to access fields in FW | ||
331 | * Make sure LE decoding macros are used | ||
332 | */ | ||
333 | struct orinoco_fw_header { | ||
334 | char hdr_vers[6]; /* ASCII string for header version */ | ||
335 | __le16 headersize; /* Total length of header */ | ||
336 | __le32 entry_point; /* NIC entry point */ | ||
337 | __le32 blocks; /* Number of blocks to program */ | ||
338 | __le32 block_offset; /* Offset of block data from eof header */ | ||
339 | __le32 pdr_offset; /* Offset to PDR data from eof header */ | ||
340 | __le32 pri_offset; /* Offset to primary plug data */ | ||
341 | __le32 compat_offset; /* Offset to compatibility data*/ | ||
342 | char signature[0]; /* FW signature length headersize-20 */ | ||
343 | } __attribute__ ((packed)); | ||
344 | |||
345 | /* Download either STA or AP firmware into the card. */ | ||
346 | static int | ||
347 | orinoco_dl_firmware(struct orinoco_private *priv, | ||
348 | const struct fw_info *fw, | ||
349 | int ap) | ||
350 | { | ||
351 | /* Plug Data Area (PDA) */ | ||
352 | __le16 *pda; | ||
353 | |||
354 | hermes_t *hw = &priv->hw; | ||
355 | const struct firmware *fw_entry; | ||
356 | const struct orinoco_fw_header *hdr; | ||
357 | const unsigned char *first_block; | ||
358 | const unsigned char *end; | ||
359 | const char *firmware; | ||
360 | struct net_device *dev = priv->ndev; | ||
361 | int err = 0; | ||
362 | |||
363 | pda = kzalloc(fw->pda_size, GFP_KERNEL); | ||
364 | if (!pda) | ||
365 | return -ENOMEM; | ||
366 | |||
367 | if (ap) | ||
368 | firmware = fw->ap_fw; | ||
369 | else | ||
370 | firmware = fw->sta_fw; | ||
371 | |||
372 | printk(KERN_DEBUG "%s: Attempting to download firmware %s\n", | ||
373 | dev->name, firmware); | ||
374 | |||
375 | /* Read current plug data */ | ||
376 | err = hermes_read_pda(hw, pda, fw->pda_addr, fw->pda_size, 0); | ||
377 | printk(KERN_DEBUG "%s: Read PDA returned %d\n", dev->name, err); | ||
378 | if (err) | ||
379 | goto free; | ||
380 | |||
381 | if (!priv->cached_fw) { | ||
382 | err = request_firmware(&fw_entry, firmware, priv->dev); | ||
383 | |||
384 | if (err) { | ||
385 | printk(KERN_ERR "%s: Cannot find firmware %s\n", | ||
386 | dev->name, firmware); | ||
387 | err = -ENOENT; | ||
388 | goto free; | ||
389 | } | ||
390 | } else | ||
391 | fw_entry = priv->cached_fw; | ||
392 | |||
393 | hdr = (const struct orinoco_fw_header *) fw_entry->data; | ||
394 | |||
395 | /* Enable aux port to allow programming */ | ||
396 | err = hermesi_program_init(hw, le32_to_cpu(hdr->entry_point)); | ||
397 | printk(KERN_DEBUG "%s: Program init returned %d\n", dev->name, err); | ||
398 | if (err != 0) | ||
399 | goto abort; | ||
400 | |||
401 | /* Program data */ | ||
402 | first_block = (fw_entry->data + | ||
403 | le16_to_cpu(hdr->headersize) + | ||
404 | le32_to_cpu(hdr->block_offset)); | ||
405 | end = fw_entry->data + fw_entry->size; | ||
406 | |||
407 | err = hermes_program(hw, first_block, end); | ||
408 | printk(KERN_DEBUG "%s: Program returned %d\n", dev->name, err); | ||
409 | if (err != 0) | ||
410 | goto abort; | ||
411 | |||
412 | /* Update production data */ | ||
413 | first_block = (fw_entry->data + | ||
414 | le16_to_cpu(hdr->headersize) + | ||
415 | le32_to_cpu(hdr->pdr_offset)); | ||
416 | |||
417 | err = hermes_apply_pda_with_defaults(hw, first_block, pda); | ||
418 | printk(KERN_DEBUG "%s: Apply PDA returned %d\n", dev->name, err); | ||
419 | if (err) | ||
420 | goto abort; | ||
421 | |||
422 | /* Tell card we've finished */ | ||
423 | err = hermesi_program_end(hw); | ||
424 | printk(KERN_DEBUG "%s: Program end returned %d\n", dev->name, err); | ||
425 | if (err != 0) | ||
426 | goto abort; | ||
427 | |||
428 | /* Check if we're running */ | ||
429 | printk(KERN_DEBUG "%s: hermes_present returned %d\n", | ||
430 | dev->name, hermes_present(hw)); | ||
431 | |||
432 | abort: | ||
433 | /* If we requested the firmware, release it. */ | ||
434 | if (!priv->cached_fw) | ||
435 | release_firmware(fw_entry); | ||
436 | |||
437 | free: | ||
438 | kfree(pda); | ||
439 | return err; | ||
440 | } | ||
441 | |||
442 | /* End markers */ | ||
443 | #define TEXT_END 0x1A /* End of text header */ | ||
444 | |||
445 | /* | ||
446 | * Process a firmware image - stop the card, load the firmware, reset | ||
447 | * the card and make sure it responds. For the secondary firmware take | ||
448 | * care of the PDA - read it and then write it on top of the firmware. | ||
449 | */ | ||
450 | static int | ||
451 | symbol_dl_image(struct orinoco_private *priv, const struct fw_info *fw, | ||
452 | const unsigned char *image, const unsigned char *end, | ||
453 | int secondary) | ||
454 | { | ||
455 | hermes_t *hw = &priv->hw; | ||
456 | int ret = 0; | ||
457 | const unsigned char *ptr; | ||
458 | const unsigned char *first_block; | ||
459 | |||
460 | /* Plug Data Area (PDA) */ | ||
461 | __le16 *pda = NULL; | ||
462 | |||
463 | /* Binary block begins after the 0x1A marker */ | ||
464 | ptr = image; | ||
465 | while (*ptr++ != TEXT_END); | ||
466 | first_block = ptr; | ||
467 | |||
468 | /* Read the PDA from EEPROM */ | ||
469 | if (secondary) { | ||
470 | pda = kzalloc(fw->pda_size, GFP_KERNEL); | ||
471 | if (!pda) | ||
472 | return -ENOMEM; | ||
473 | |||
474 | ret = hermes_read_pda(hw, pda, fw->pda_addr, fw->pda_size, 1); | ||
475 | if (ret) | ||
476 | goto free; | ||
477 | } | ||
478 | |||
479 | /* Stop the firmware, so that it can be safely rewritten */ | ||
480 | if (priv->stop_fw) { | ||
481 | ret = priv->stop_fw(priv, 1); | ||
482 | if (ret) | ||
483 | goto free; | ||
484 | } | ||
485 | |||
486 | /* Program the adapter with new firmware */ | ||
487 | ret = hermes_program(hw, first_block, end); | ||
488 | if (ret) | ||
489 | goto free; | ||
490 | |||
491 | /* Write the PDA to the adapter */ | ||
492 | if (secondary) { | ||
493 | size_t len = hermes_blocks_length(first_block); | ||
494 | ptr = first_block + len; | ||
495 | ret = hermes_apply_pda(hw, ptr, pda); | ||
496 | kfree(pda); | ||
497 | if (ret) | ||
498 | return ret; | ||
499 | } | ||
500 | |||
501 | /* Run the firmware */ | ||
502 | if (priv->stop_fw) { | ||
503 | ret = priv->stop_fw(priv, 0); | ||
504 | if (ret) | ||
505 | return ret; | ||
506 | } | ||
507 | |||
508 | /* Reset hermes chip and make sure it responds */ | ||
509 | ret = hermes_init(hw); | ||
510 | |||
511 | /* hermes_reset() should return 0 with the secondary firmware */ | ||
512 | if (secondary && ret != 0) | ||
513 | return -ENODEV; | ||
514 | |||
515 | /* And this should work with any firmware */ | ||
516 | if (!hermes_present(hw)) | ||
517 | return -ENODEV; | ||
518 | |||
519 | return 0; | ||
520 | |||
521 | free: | ||
522 | kfree(pda); | ||
523 | return ret; | ||
524 | } | ||
525 | |||
526 | |||
527 | /* | ||
528 | * Download the firmware into the card, this also does a PCMCIA soft | ||
529 | * reset on the card, to make sure it's in a sane state. | ||
530 | */ | ||
531 | static int | ||
532 | symbol_dl_firmware(struct orinoco_private *priv, | ||
533 | const struct fw_info *fw) | ||
534 | { | ||
535 | struct net_device *dev = priv->ndev; | ||
536 | int ret; | ||
537 | const struct firmware *fw_entry; | ||
538 | |||
539 | if (!priv->cached_pri_fw) { | ||
540 | if (request_firmware(&fw_entry, fw->pri_fw, priv->dev) != 0) { | ||
541 | printk(KERN_ERR "%s: Cannot find firmware: %s\n", | ||
542 | dev->name, fw->pri_fw); | ||
543 | return -ENOENT; | ||
544 | } | ||
545 | } else | ||
546 | fw_entry = priv->cached_pri_fw; | ||
547 | |||
548 | /* Load primary firmware */ | ||
549 | ret = symbol_dl_image(priv, fw, fw_entry->data, | ||
550 | fw_entry->data + fw_entry->size, 0); | ||
551 | |||
552 | if (!priv->cached_pri_fw) | ||
553 | release_firmware(fw_entry); | ||
554 | if (ret) { | ||
555 | printk(KERN_ERR "%s: Primary firmware download failed\n", | ||
556 | dev->name); | ||
557 | return ret; | ||
558 | } | ||
559 | |||
560 | if (!priv->cached_fw) { | ||
561 | if (request_firmware(&fw_entry, fw->sta_fw, priv->dev) != 0) { | ||
562 | printk(KERN_ERR "%s: Cannot find firmware: %s\n", | ||
563 | dev->name, fw->sta_fw); | ||
564 | return -ENOENT; | ||
565 | } | ||
566 | } else | ||
567 | fw_entry = priv->cached_fw; | ||
568 | |||
569 | /* Load secondary firmware */ | ||
570 | ret = symbol_dl_image(priv, fw, fw_entry->data, | ||
571 | fw_entry->data + fw_entry->size, 1); | ||
572 | if (!priv->cached_fw) | ||
573 | release_firmware(fw_entry); | ||
574 | if (ret) { | ||
575 | printk(KERN_ERR "%s: Secondary firmware download failed\n", | ||
576 | dev->name); | ||
577 | } | ||
578 | |||
579 | return ret; | ||
580 | } | ||
581 | |||
582 | static int orinoco_download(struct orinoco_private *priv) | ||
583 | { | ||
584 | int err = 0; | ||
585 | /* Reload firmware */ | ||
586 | switch (priv->firmware_type) { | ||
587 | case FIRMWARE_TYPE_AGERE: | ||
588 | /* case FIRMWARE_TYPE_INTERSIL: */ | ||
589 | err = orinoco_dl_firmware(priv, | ||
590 | &orinoco_fw[priv->firmware_type], 0); | ||
591 | break; | ||
592 | |||
593 | case FIRMWARE_TYPE_SYMBOL: | ||
594 | err = symbol_dl_firmware(priv, | ||
595 | &orinoco_fw[priv->firmware_type]); | ||
596 | break; | ||
597 | case FIRMWARE_TYPE_INTERSIL: | ||
598 | break; | ||
599 | } | ||
600 | /* TODO: if we fail we probably need to reinitialise | ||
601 | * the driver */ | ||
602 | |||
603 | return err; | ||
604 | } | ||
605 | |||
606 | #if defined(CONFIG_HERMES_CACHE_FW_ON_INIT) || defined(CONFIG_PM_SLEEP) | ||
607 | static void orinoco_cache_fw(struct orinoco_private *priv, int ap) | ||
608 | { | ||
609 | const struct firmware *fw_entry = NULL; | ||
610 | const char *pri_fw; | ||
611 | const char *fw; | ||
612 | |||
613 | pri_fw = orinoco_fw[priv->firmware_type].pri_fw; | ||
614 | if (ap) | ||
615 | fw = orinoco_fw[priv->firmware_type].ap_fw; | ||
616 | else | ||
617 | fw = orinoco_fw[priv->firmware_type].sta_fw; | ||
618 | |||
619 | if (pri_fw) { | ||
620 | if (request_firmware(&fw_entry, pri_fw, priv->dev) == 0) | ||
621 | priv->cached_pri_fw = fw_entry; | ||
622 | } | ||
623 | |||
624 | if (fw) { | ||
625 | if (request_firmware(&fw_entry, fw, priv->dev) == 0) | ||
626 | priv->cached_fw = fw_entry; | ||
627 | } | ||
628 | } | ||
629 | |||
630 | static void orinoco_uncache_fw(struct orinoco_private *priv) | ||
631 | { | ||
632 | if (priv->cached_pri_fw) | ||
633 | release_firmware(priv->cached_pri_fw); | ||
634 | if (priv->cached_fw) | ||
635 | release_firmware(priv->cached_fw); | ||
636 | |||
637 | priv->cached_pri_fw = NULL; | ||
638 | priv->cached_fw = NULL; | ||
639 | } | ||
640 | #else | ||
641 | #define orinoco_cache_fw(priv, ap) | ||
642 | #define orinoco_uncache_fw(priv) | ||
643 | #endif | ||
644 | |||
645 | /********************************************************************/ | ||
646 | /* Device methods */ | 313 | /* Device methods */ |
647 | /********************************************************************/ | 314 | /********************************************************************/ |
648 | 315 | ||