diff options
Diffstat (limited to 'drivers/scsi/aic7xxx/aic79xx_osm.c')
-rw-r--r-- | drivers/scsi/aic7xxx/aic79xx_osm.c | 108 |
1 files changed, 20 insertions, 88 deletions
diff --git a/drivers/scsi/aic7xxx/aic79xx_osm.c b/drivers/scsi/aic7xxx/aic79xx_osm.c index 2f158624c5d2..3feb739cd554 100644 --- a/drivers/scsi/aic7xxx/aic79xx_osm.c +++ b/drivers/scsi/aic7xxx/aic79xx_osm.c | |||
@@ -60,11 +60,6 @@ static struct scsi_transport_template *ahd_linux_transport_template = NULL; | |||
60 | #include <linux/delay.h> /* For ssleep/msleep */ | 60 | #include <linux/delay.h> /* For ssleep/msleep */ |
61 | 61 | ||
62 | /* | 62 | /* |
63 | * Lock protecting manipulation of the ahd softc list. | ||
64 | */ | ||
65 | spinlock_t ahd_list_spinlock; | ||
66 | |||
67 | /* | ||
68 | * Bucket size for counting good commands in between bad ones. | 63 | * Bucket size for counting good commands in between bad ones. |
69 | */ | 64 | */ |
70 | #define AHD_LINUX_ERR_THRESH 1000 | 65 | #define AHD_LINUX_ERR_THRESH 1000 |
@@ -303,13 +298,6 @@ static uint32_t aic79xx_pci_parity = ~0; | |||
303 | uint32_t aic79xx_allow_memio = ~0; | 298 | uint32_t aic79xx_allow_memio = ~0; |
304 | 299 | ||
305 | /* | 300 | /* |
306 | * aic79xx_detect() has been run, so register all device arrivals | ||
307 | * immediately with the system rather than deferring to the sorted | ||
308 | * attachment performed by aic79xx_detect(). | ||
309 | */ | ||
310 | int aic79xx_detect_complete; | ||
311 | |||
312 | /* | ||
313 | * So that we can set how long each device is given as a selection timeout. | 301 | * So that we can set how long each device is given as a selection timeout. |
314 | * The table of values goes like this: | 302 | * The table of values goes like this: |
315 | * 0 - 256ms | 303 | * 0 - 256ms |
@@ -387,7 +375,9 @@ static void ahd_linux_setup_tag_info_global(char *p); | |||
387 | static aic_option_callback_t ahd_linux_setup_tag_info; | 375 | static aic_option_callback_t ahd_linux_setup_tag_info; |
388 | static aic_option_callback_t ahd_linux_setup_iocell_info; | 376 | static aic_option_callback_t ahd_linux_setup_iocell_info; |
389 | static int aic79xx_setup(char *c); | 377 | static int aic79xx_setup(char *c); |
390 | static int ahd_linux_next_unit(void); | 378 | |
379 | static int ahd_linux_unit; | ||
380 | |||
391 | 381 | ||
392 | /****************************** Inlines ***************************************/ | 382 | /****************************** Inlines ***************************************/ |
393 | static __inline void ahd_linux_unmap_scb(struct ahd_softc*, struct scb*); | 383 | static __inline void ahd_linux_unmap_scb(struct ahd_softc*, struct scb*); |
@@ -418,50 +408,6 @@ ahd_linux_unmap_scb(struct ahd_softc *ahd, struct scb *scb) | |||
418 | ((((cmd)->device->id << TID_SHIFT) & TID) | (ahd)->our_id) | 408 | ((((cmd)->device->id << TID_SHIFT) & TID) | (ahd)->our_id) |
419 | 409 | ||
420 | /* | 410 | /* |
421 | * Try to detect an Adaptec 79XX controller. | ||
422 | */ | ||
423 | static int | ||
424 | ahd_linux_detect(struct scsi_host_template *template) | ||
425 | { | ||
426 | struct ahd_softc *ahd; | ||
427 | int found; | ||
428 | int error = 0; | ||
429 | |||
430 | /* | ||
431 | * If we've been passed any parameters, process them now. | ||
432 | */ | ||
433 | if (aic79xx) | ||
434 | aic79xx_setup(aic79xx); | ||
435 | |||
436 | template->proc_name = "aic79xx"; | ||
437 | |||
438 | /* | ||
439 | * Initialize our softc list lock prior to | ||
440 | * probing for any adapters. | ||
441 | */ | ||
442 | ahd_list_lockinit(); | ||
443 | |||
444 | #ifdef CONFIG_PCI | ||
445 | error = ahd_linux_pci_init(); | ||
446 | if (error) | ||
447 | return error; | ||
448 | #endif | ||
449 | |||
450 | /* | ||
451 | * Register with the SCSI layer all | ||
452 | * controllers we've found. | ||
453 | */ | ||
454 | found = 0; | ||
455 | TAILQ_FOREACH(ahd, &ahd_tailq, links) { | ||
456 | |||
457 | if (ahd_linux_register_host(ahd, template) == 0) | ||
458 | found++; | ||
459 | } | ||
460 | aic79xx_detect_complete++; | ||
461 | return found; | ||
462 | } | ||
463 | |||
464 | /* | ||
465 | * Return a string describing the driver. | 411 | * Return a string describing the driver. |
466 | */ | 412 | */ |
467 | static const char * | 413 | static const char * |
@@ -760,6 +706,7 @@ ahd_linux_bus_reset(struct scsi_cmnd *cmd) | |||
760 | struct scsi_host_template aic79xx_driver_template = { | 706 | struct scsi_host_template aic79xx_driver_template = { |
761 | .module = THIS_MODULE, | 707 | .module = THIS_MODULE, |
762 | .name = "aic79xx", | 708 | .name = "aic79xx", |
709 | .proc_name = "aic79xx", | ||
763 | .proc_info = ahd_linux_proc_info, | 710 | .proc_info = ahd_linux_proc_info, |
764 | .info = ahd_linux_info, | 711 | .info = ahd_linux_info, |
765 | .queuecommand = ahd_linux_queue, | 712 | .queuecommand = ahd_linux_queue, |
@@ -1072,7 +1019,7 @@ ahd_linux_register_host(struct ahd_softc *ahd, struct scsi_host_template *templa | |||
1072 | host->max_lun = AHD_NUM_LUNS; | 1019 | host->max_lun = AHD_NUM_LUNS; |
1073 | host->max_channel = 0; | 1020 | host->max_channel = 0; |
1074 | host->sg_tablesize = AHD_NSEG; | 1021 | host->sg_tablesize = AHD_NSEG; |
1075 | ahd_set_unit(ahd, ahd_linux_next_unit()); | 1022 | ahd_set_unit(ahd, ahd_linux_unit++); |
1076 | sprintf(buf, "scsi%d", host->host_no); | 1023 | sprintf(buf, "scsi%d", host->host_no); |
1077 | new_name = malloc(strlen(buf) + 1, M_DEVBUF, M_NOWAIT); | 1024 | new_name = malloc(strlen(buf) + 1, M_DEVBUF, M_NOWAIT); |
1078 | if (new_name != NULL) { | 1025 | if (new_name != NULL) { |
@@ -1101,29 +1048,6 @@ ahd_linux_get_memsize(void) | |||
1101 | } | 1048 | } |
1102 | 1049 | ||
1103 | /* | 1050 | /* |
1104 | * Find the smallest available unit number to use | ||
1105 | * for a new device. We don't just use a static | ||
1106 | * count to handle the "repeated hot-(un)plug" | ||
1107 | * scenario. | ||
1108 | */ | ||
1109 | static int | ||
1110 | ahd_linux_next_unit(void) | ||
1111 | { | ||
1112 | struct ahd_softc *ahd; | ||
1113 | int unit; | ||
1114 | |||
1115 | unit = 0; | ||
1116 | retry: | ||
1117 | TAILQ_FOREACH(ahd, &ahd_tailq, links) { | ||
1118 | if (ahd->unit == unit) { | ||
1119 | unit++; | ||
1120 | goto retry; | ||
1121 | } | ||
1122 | } | ||
1123 | return (unit); | ||
1124 | } | ||
1125 | |||
1126 | /* | ||
1127 | * Place the SCSI bus into a known state by either resetting it, | 1051 | * Place the SCSI bus into a known state by either resetting it, |
1128 | * or forcing transfer negotiations on the next command to any | 1052 | * or forcing transfer negotiations on the next command to any |
1129 | * target. | 1053 | * target. |
@@ -2755,23 +2679,31 @@ static struct spi_function_template ahd_linux_transport_functions = { | |||
2755 | .show_hold_mcs = 1, | 2679 | .show_hold_mcs = 1, |
2756 | }; | 2680 | }; |
2757 | 2681 | ||
2758 | |||
2759 | |||
2760 | static int __init | 2682 | static int __init |
2761 | ahd_linux_init(void) | 2683 | ahd_linux_init(void) |
2762 | { | 2684 | { |
2763 | ahd_linux_transport_template = spi_attach_transport(&ahd_linux_transport_functions); | 2685 | int error = 0; |
2686 | |||
2687 | /* | ||
2688 | * If we've been passed any parameters, process them now. | ||
2689 | */ | ||
2690 | if (aic79xx) | ||
2691 | aic79xx_setup(aic79xx); | ||
2692 | |||
2693 | ahd_linux_transport_template = | ||
2694 | spi_attach_transport(&ahd_linux_transport_functions); | ||
2764 | if (!ahd_linux_transport_template) | 2695 | if (!ahd_linux_transport_template) |
2765 | return -ENODEV; | 2696 | return -ENODEV; |
2697 | |||
2766 | scsi_transport_reserve_target(ahd_linux_transport_template, | 2698 | scsi_transport_reserve_target(ahd_linux_transport_template, |
2767 | sizeof(struct ahd_linux_target)); | 2699 | sizeof(struct ahd_linux_target)); |
2768 | scsi_transport_reserve_device(ahd_linux_transport_template, | 2700 | scsi_transport_reserve_device(ahd_linux_transport_template, |
2769 | sizeof(struct ahd_linux_device)); | 2701 | sizeof(struct ahd_linux_device)); |
2770 | if (ahd_linux_detect(&aic79xx_driver_template) > 0) | ||
2771 | return 0; | ||
2772 | spi_release_transport(ahd_linux_transport_template); | ||
2773 | 2702 | ||
2774 | return -ENODEV; | 2703 | error = ahd_linux_pci_init(); |
2704 | if (error) | ||
2705 | spi_release_transport(ahd_linux_transport_template); | ||
2706 | return error; | ||
2775 | } | 2707 | } |
2776 | 2708 | ||
2777 | static void __exit | 2709 | static void __exit |