diff options
author | James Bottomley <James.Bottomley@steeleye.com> | 2005-05-24 18:15:43 -0400 |
---|---|---|
committer | James Bottomley <jejb@mulgrave.(none)> | 2005-05-26 14:28:31 -0400 |
commit | b1abb4d67f2a706f52a95064001e0c55d9be2d26 (patch) | |
tree | 4095e216afd32cc1681718fa41ad8ca4e37b77fb /drivers/scsi/aic7xxx/aic7xxx_osm.c | |
parent | 153b1e1fd957861e2c185473dd3c3d93561066e4 (diff) |
[SCSI] aic7xxx: remove separate target and device allocations
Since the aic driver is now taught to speak in terms of the generic
linux devices, we can now also dispense with the transport class get
routines (since we update the parameters when the driver sees they
change) and also plumb it into the spi transport transfer agreement
reporting infrastructure.
Signed-off-by: James Bottomley <James.Bottomley@SteelEye.com>
Diffstat (limited to 'drivers/scsi/aic7xxx/aic7xxx_osm.c')
-rw-r--r-- | drivers/scsi/aic7xxx/aic7xxx_osm.c | 575 |
1 files changed, 201 insertions, 374 deletions
diff --git a/drivers/scsi/aic7xxx/aic7xxx_osm.c b/drivers/scsi/aic7xxx/aic7xxx_osm.c index c13e56320010..57d22c4f0081 100644 --- a/drivers/scsi/aic7xxx/aic7xxx_osm.c +++ b/drivers/scsi/aic7xxx/aic7xxx_osm.c | |||
@@ -122,8 +122,6 @@ | |||
122 | #include "aic7xxx_osm.h" | 122 | #include "aic7xxx_osm.h" |
123 | #include "aic7xxx_inline.h" | 123 | #include "aic7xxx_inline.h" |
124 | #include <scsi/scsicam.h> | 124 | #include <scsi/scsicam.h> |
125 | #include <scsi/scsi_transport.h> | ||
126 | #include <scsi/scsi_transport_spi.h> | ||
127 | 125 | ||
128 | static struct scsi_transport_template *ahc_linux_transport_template = NULL; | 126 | static struct scsi_transport_template *ahc_linux_transport_template = NULL; |
129 | 127 | ||
@@ -423,7 +421,7 @@ MODULE_PARM_DESC(aic7xxx, | |||
423 | ); | 421 | ); |
424 | 422 | ||
425 | static void ahc_linux_handle_scsi_status(struct ahc_softc *, | 423 | static void ahc_linux_handle_scsi_status(struct ahc_softc *, |
426 | struct ahc_linux_device *, | 424 | struct scsi_device *, |
427 | struct scb *); | 425 | struct scb *); |
428 | static void ahc_linux_queue_cmd_complete(struct ahc_softc *ahc, | 426 | static void ahc_linux_queue_cmd_complete(struct ahc_softc *ahc, |
429 | struct scsi_cmnd *cmd); | 427 | struct scsi_cmnd *cmd); |
@@ -434,17 +432,7 @@ static int ahc_linux_queue_recovery_cmd(struct scsi_cmnd *cmd, scb_flag flag); | |||
434 | static void ahc_linux_initialize_scsi_bus(struct ahc_softc *ahc); | 432 | static void ahc_linux_initialize_scsi_bus(struct ahc_softc *ahc); |
435 | static u_int ahc_linux_user_tagdepth(struct ahc_softc *ahc, | 433 | static u_int ahc_linux_user_tagdepth(struct ahc_softc *ahc, |
436 | struct ahc_devinfo *devinfo); | 434 | struct ahc_devinfo *devinfo); |
437 | static void ahc_linux_device_queue_depth(struct ahc_softc *ahc, | 435 | static void ahc_linux_device_queue_depth(struct scsi_device *); |
438 | struct ahc_linux_device *dev); | ||
439 | static struct ahc_linux_target* ahc_linux_alloc_target(struct ahc_softc*, | ||
440 | u_int, u_int); | ||
441 | static void ahc_linux_free_target(struct ahc_softc*, | ||
442 | struct ahc_linux_target*); | ||
443 | static struct ahc_linux_device* ahc_linux_alloc_device(struct ahc_softc*, | ||
444 | struct ahc_linux_target*, | ||
445 | u_int); | ||
446 | static void ahc_linux_free_device(struct ahc_softc*, | ||
447 | struct ahc_linux_device*); | ||
448 | static int ahc_linux_run_command(struct ahc_softc*, | 436 | static int ahc_linux_run_command(struct ahc_softc*, |
449 | struct ahc_linux_device *, | 437 | struct ahc_linux_device *, |
450 | struct scsi_cmnd *); | 438 | struct scsi_cmnd *); |
@@ -454,32 +442,12 @@ static int aic7xxx_setup(char *s); | |||
454 | static int ahc_linux_next_unit(void); | 442 | static int ahc_linux_next_unit(void); |
455 | 443 | ||
456 | /********************************* Inlines ************************************/ | 444 | /********************************* Inlines ************************************/ |
457 | static __inline struct ahc_linux_device* | ||
458 | ahc_linux_get_device(struct ahc_softc *ahc, u_int channel, | ||
459 | u_int target, u_int lun); | ||
460 | static __inline void ahc_linux_unmap_scb(struct ahc_softc*, struct scb*); | 445 | static __inline void ahc_linux_unmap_scb(struct ahc_softc*, struct scb*); |
461 | 446 | ||
462 | static __inline int ahc_linux_map_seg(struct ahc_softc *ahc, struct scb *scb, | 447 | static __inline int ahc_linux_map_seg(struct ahc_softc *ahc, struct scb *scb, |
463 | struct ahc_dma_seg *sg, | 448 | struct ahc_dma_seg *sg, |
464 | dma_addr_t addr, bus_size_t len); | 449 | dma_addr_t addr, bus_size_t len); |
465 | 450 | ||
466 | static __inline struct ahc_linux_device* | ||
467 | ahc_linux_get_device(struct ahc_softc *ahc, u_int channel, u_int target, | ||
468 | u_int lun) | ||
469 | { | ||
470 | struct ahc_linux_target *targ; | ||
471 | struct ahc_linux_device *dev; | ||
472 | u_int target_offset; | ||
473 | |||
474 | target_offset = target; | ||
475 | if (channel != 0) | ||
476 | target_offset += 8; | ||
477 | targ = ahc->platform_data->targets[target_offset]; | ||
478 | BUG_ON(targ == NULL); | ||
479 | dev = targ->devices[lun]; | ||
480 | return dev; | ||
481 | } | ||
482 | |||
483 | static __inline void | 451 | static __inline void |
484 | ahc_linux_unmap_scb(struct ahc_softc *ahc, struct scb *scb) | 452 | ahc_linux_unmap_scb(struct ahc_softc *ahc, struct scb *scb) |
485 | { | 453 | { |
@@ -611,7 +579,7 @@ static int | |||
611 | ahc_linux_queue(struct scsi_cmnd * cmd, void (*scsi_done) (struct scsi_cmnd *)) | 579 | ahc_linux_queue(struct scsi_cmnd * cmd, void (*scsi_done) (struct scsi_cmnd *)) |
612 | { | 580 | { |
613 | struct ahc_softc *ahc; | 581 | struct ahc_softc *ahc; |
614 | struct ahc_linux_device *dev; | 582 | struct ahc_linux_device *dev = scsi_transport_device_data(cmd->device); |
615 | 583 | ||
616 | ahc = *(struct ahc_softc **)cmd->device->host->hostdata; | 584 | ahc = *(struct ahc_softc **)cmd->device->host->hostdata; |
617 | 585 | ||
@@ -629,132 +597,162 @@ ahc_linux_queue(struct scsi_cmnd * cmd, void (*scsi_done) (struct scsi_cmnd *)) | |||
629 | if (ahc->platform_data->qfrozen != 0) | 597 | if (ahc->platform_data->qfrozen != 0) |
630 | return SCSI_MLQUEUE_HOST_BUSY; | 598 | return SCSI_MLQUEUE_HOST_BUSY; |
631 | 599 | ||
632 | dev = ahc_linux_get_device(ahc, cmd->device->channel, cmd->device->id, | ||
633 | cmd->device->lun); | ||
634 | BUG_ON(dev == NULL); | ||
635 | |||
636 | cmd->result = CAM_REQ_INPROG << 16; | 600 | cmd->result = CAM_REQ_INPROG << 16; |
637 | 601 | ||
638 | return ahc_linux_run_command(ahc, dev, cmd); | 602 | return ahc_linux_run_command(ahc, dev, cmd); |
639 | } | 603 | } |
640 | 604 | ||
641 | static int | 605 | static inline struct scsi_target ** |
642 | ahc_linux_slave_alloc(struct scsi_device *device) | 606 | ahc_linux_target_in_softc(struct scsi_target *starget) |
643 | { | 607 | { |
644 | struct ahc_softc *ahc; | 608 | struct ahc_softc *ahc = |
645 | struct ahc_linux_target *targ; | 609 | *((struct ahc_softc **)dev_to_shost(&starget->dev)->hostdata); |
646 | struct scsi_target *starget = device->sdev_target; | ||
647 | struct ahc_linux_device *dev; | ||
648 | unsigned int target_offset; | 610 | unsigned int target_offset; |
611 | |||
612 | target_offset = starget->id; | ||
613 | if (starget->channel != 0) | ||
614 | target_offset += 8; | ||
615 | |||
616 | return &ahc->platform_data->starget[target_offset]; | ||
617 | } | ||
618 | |||
619 | static int | ||
620 | ahc_linux_target_alloc(struct scsi_target *starget) | ||
621 | { | ||
622 | struct ahc_softc *ahc = | ||
623 | *((struct ahc_softc **)dev_to_shost(&starget->dev)->hostdata); | ||
624 | struct seeprom_config *sc = ahc->seep_config; | ||
649 | unsigned long flags; | 625 | unsigned long flags; |
650 | int retval = -ENOMEM; | 626 | struct scsi_target **ahc_targp = ahc_linux_target_in_softc(starget); |
627 | struct ahc_linux_target *targ = scsi_transport_target_data(starget); | ||
628 | unsigned short scsirate; | ||
629 | struct ahc_devinfo devinfo; | ||
630 | struct ahc_initiator_tinfo *tinfo; | ||
631 | struct ahc_tmode_tstate *tstate; | ||
632 | char channel = starget->channel + 'A'; | ||
633 | unsigned int our_id = ahc->our_id; | ||
634 | unsigned int target_offset; | ||
651 | 635 | ||
652 | target_offset = starget->id; | 636 | target_offset = starget->id; |
653 | if (starget->channel != 0) | 637 | if (starget->channel != 0) |
654 | target_offset += 8; | 638 | target_offset += 8; |
639 | |||
640 | if (starget->channel) | ||
641 | our_id = ahc->our_id_b; | ||
655 | 642 | ||
656 | ahc = *((struct ahc_softc **)device->host->hostdata); | ||
657 | if (bootverbose) | ||
658 | printf("%s: Slave Alloc %d\n", ahc_name(ahc), device->id); | ||
659 | ahc_lock(ahc, &flags); | 643 | ahc_lock(ahc, &flags); |
660 | targ = ahc->platform_data->targets[target_offset]; | ||
661 | if (targ == NULL) { | ||
662 | struct seeprom_config *sc; | ||
663 | |||
664 | targ = ahc_linux_alloc_target(ahc, starget->channel, | ||
665 | starget->id); | ||
666 | sc = ahc->seep_config; | ||
667 | if (targ == NULL) | ||
668 | goto out; | ||
669 | 644 | ||
670 | if (sc) { | 645 | BUG_ON(*ahc_targp != NULL); |
671 | unsigned short scsirate; | ||
672 | struct ahc_devinfo devinfo; | ||
673 | struct ahc_initiator_tinfo *tinfo; | ||
674 | struct ahc_tmode_tstate *tstate; | ||
675 | char channel = starget->channel + 'A'; | ||
676 | unsigned int our_id = ahc->our_id; | ||
677 | 646 | ||
678 | if (starget->channel) | 647 | *ahc_targp = starget; |
679 | our_id = ahc->our_id_b; | 648 | memset(targ, 0, sizeof(*targ)); |
680 | 649 | ||
681 | if ((ahc->features & AHC_ULTRA2) != 0) { | 650 | if (sc) { |
682 | scsirate = sc->device_flags[target_offset] & CFXFER; | 651 | if ((ahc->features & AHC_ULTRA2) != 0) { |
683 | } else { | 652 | scsirate = sc->device_flags[target_offset] & CFXFER; |
684 | scsirate = (sc->device_flags[target_offset] & CFXFER) << 4; | 653 | } else { |
685 | if (sc->device_flags[target_offset] & CFSYNCH) | 654 | scsirate = (sc->device_flags[target_offset] & CFXFER) << 4; |
686 | scsirate |= SOFS; | 655 | if (sc->device_flags[target_offset] & CFSYNCH) |
687 | } | 656 | scsirate |= SOFS; |
688 | if (sc->device_flags[target_offset] & CFWIDEB) { | ||
689 | scsirate |= WIDEXFER; | ||
690 | spi_max_width(starget) = 1; | ||
691 | } else | ||
692 | spi_max_width(starget) = 0; | ||
693 | spi_min_period(starget) = | ||
694 | ahc_find_period(ahc, scsirate, AHC_SYNCRATE_DT); | ||
695 | tinfo = ahc_fetch_transinfo(ahc, channel, ahc->our_id, | ||
696 | targ->target, &tstate); | ||
697 | ahc_compile_devinfo(&devinfo, our_id, targ->target, | ||
698 | CAM_LUN_WILDCARD, channel, | ||
699 | ROLE_INITIATOR); | ||
700 | ahc_set_syncrate(ahc, &devinfo, NULL, 0, 0, 0, | ||
701 | AHC_TRANS_GOAL, /*paused*/FALSE); | ||
702 | ahc_set_width(ahc, &devinfo, MSG_EXT_WDTR_BUS_8_BIT, | ||
703 | AHC_TRANS_GOAL, /*paused*/FALSE); | ||
704 | } | 657 | } |
705 | 658 | if (sc->device_flags[target_offset] & CFWIDEB) { | |
706 | } | 659 | scsirate |= WIDEXFER; |
707 | dev = targ->devices[device->lun]; | 660 | spi_max_width(starget) = 1; |
708 | if (dev == NULL) { | 661 | } else |
709 | dev = ahc_linux_alloc_device(ahc, targ, device->lun); | 662 | spi_max_width(starget) = 0; |
710 | if (dev == NULL) | 663 | spi_min_period(starget) = |
711 | goto out; | 664 | ahc_find_period(ahc, scsirate, AHC_SYNCRATE_DT); |
665 | tinfo = ahc_fetch_transinfo(ahc, channel, ahc->our_id, | ||
666 | starget->id, &tstate); | ||
712 | } | 667 | } |
713 | retval = 0; | 668 | ahc_compile_devinfo(&devinfo, our_id, starget->id, |
714 | 669 | CAM_LUN_WILDCARD, channel, | |
715 | out: | 670 | ROLE_INITIATOR); |
671 | ahc_set_syncrate(ahc, &devinfo, NULL, 0, 0, 0, | ||
672 | AHC_TRANS_GOAL, /*paused*/FALSE); | ||
673 | ahc_set_width(ahc, &devinfo, MSG_EXT_WDTR_BUS_8_BIT, | ||
674 | AHC_TRANS_GOAL, /*paused*/FALSE); | ||
716 | ahc_unlock(ahc, &flags); | 675 | ahc_unlock(ahc, &flags); |
717 | return retval; | 676 | |
677 | return 0; | ||
678 | } | ||
679 | |||
680 | static void | ||
681 | ahc_linux_target_destroy(struct scsi_target *starget) | ||
682 | { | ||
683 | struct scsi_target **ahc_targp = ahc_linux_target_in_softc(starget); | ||
684 | |||
685 | *ahc_targp = NULL; | ||
686 | } | ||
687 | |||
688 | static int | ||
689 | ahc_linux_slave_alloc(struct scsi_device *sdev) | ||
690 | { | ||
691 | struct ahc_softc *ahc = | ||
692 | *((struct ahc_softc **)sdev->host->hostdata); | ||
693 | struct scsi_target *starget = sdev->sdev_target; | ||
694 | struct ahc_linux_target *targ = scsi_transport_target_data(starget); | ||
695 | struct ahc_linux_device *dev; | ||
696 | |||
697 | if (bootverbose) | ||
698 | printf("%s: Slave Alloc %d\n", ahc_name(ahc), sdev->id); | ||
699 | |||
700 | BUG_ON(targ->sdev[sdev->lun] != NULL); | ||
701 | |||
702 | dev = scsi_transport_device_data(sdev); | ||
703 | memset(dev, 0, sizeof(*dev)); | ||
704 | |||
705 | /* | ||
706 | * We start out life using untagged | ||
707 | * transactions of which we allow one. | ||
708 | */ | ||
709 | dev->openings = 1; | ||
710 | |||
711 | /* | ||
712 | * Set maxtags to 0. This will be changed if we | ||
713 | * later determine that we are dealing with | ||
714 | * a tagged queuing capable device. | ||
715 | */ | ||
716 | dev->maxtags = 0; | ||
717 | |||
718 | targ->sdev[sdev->lun] = sdev; | ||
719 | |||
720 | return 0; | ||
718 | } | 721 | } |
719 | 722 | ||
720 | static int | 723 | static int |
721 | ahc_linux_slave_configure(struct scsi_device *device) | 724 | ahc_linux_slave_configure(struct scsi_device *sdev) |
722 | { | 725 | { |
723 | struct ahc_softc *ahc; | 726 | struct ahc_softc *ahc; |
724 | struct ahc_linux_device *dev; | ||
725 | 727 | ||
726 | ahc = *((struct ahc_softc **)device->host->hostdata); | 728 | ahc = *((struct ahc_softc **)sdev->host->hostdata); |
727 | 729 | ||
728 | if (bootverbose) | 730 | if (bootverbose) |
729 | printf("%s: Slave Configure %d\n", ahc_name(ahc), device->id); | 731 | printf("%s: Slave Configure %d\n", ahc_name(ahc), sdev->id); |
730 | 732 | ||
731 | dev = ahc_linux_get_device(ahc, device->channel, device->id, | 733 | ahc_linux_device_queue_depth(sdev); |
732 | device->lun); | ||
733 | dev->scsi_device = device; | ||
734 | ahc_linux_device_queue_depth(ahc, dev); | ||
735 | 734 | ||
736 | /* Initial Domain Validation */ | 735 | /* Initial Domain Validation */ |
737 | if (!spi_initial_dv(device->sdev_target)) | 736 | if (!spi_initial_dv(sdev->sdev_target)) |
738 | spi_dv_device(device); | 737 | spi_dv_device(sdev); |
739 | 738 | ||
740 | return 0; | 739 | return 0; |
741 | } | 740 | } |
742 | 741 | ||
743 | static void | 742 | static void |
744 | ahc_linux_slave_destroy(struct scsi_device *device) | 743 | ahc_linux_slave_destroy(struct scsi_device *sdev) |
745 | { | 744 | { |
746 | struct ahc_softc *ahc; | 745 | struct ahc_softc *ahc; |
747 | struct ahc_linux_device *dev; | 746 | struct ahc_linux_device *dev = scsi_transport_device_data(sdev); |
747 | struct ahc_linux_target *targ = scsi_transport_target_data(sdev->sdev_target); | ||
748 | 748 | ||
749 | ahc = *((struct ahc_softc **)device->host->hostdata); | 749 | ahc = *((struct ahc_softc **)sdev->host->hostdata); |
750 | if (bootverbose) | 750 | if (bootverbose) |
751 | printf("%s: Slave Destroy %d\n", ahc_name(ahc), device->id); | 751 | printf("%s: Slave Destroy %d\n", ahc_name(ahc), sdev->id); |
752 | dev = ahc_linux_get_device(ahc, device->channel, | ||
753 | device->id, device->lun); | ||
754 | 752 | ||
755 | BUG_ON(dev->active); | 753 | BUG_ON(dev->active); |
756 | 754 | ||
757 | ahc_linux_free_device(ahc, dev); | 755 | targ->sdev[sdev->lun] = NULL; |
758 | } | 756 | } |
759 | 757 | ||
760 | #if defined(__i386__) | 758 | #if defined(__i386__) |
@@ -874,6 +872,8 @@ struct scsi_host_template aic7xxx_driver_template = { | |||
874 | .slave_alloc = ahc_linux_slave_alloc, | 872 | .slave_alloc = ahc_linux_slave_alloc, |
875 | .slave_configure = ahc_linux_slave_configure, | 873 | .slave_configure = ahc_linux_slave_configure, |
876 | .slave_destroy = ahc_linux_slave_destroy, | 874 | .slave_destroy = ahc_linux_slave_destroy, |
875 | .target_alloc = ahc_linux_target_alloc, | ||
876 | .target_destroy = ahc_linux_target_destroy, | ||
877 | }; | 877 | }; |
878 | 878 | ||
879 | /**************************** Tasklet Handler *********************************/ | 879 | /**************************** Tasklet Handler *********************************/ |
@@ -1335,8 +1335,7 @@ ahc_platform_alloc(struct ahc_softc *ahc, void *platform_arg) | |||
1335 | void | 1335 | void |
1336 | ahc_platform_free(struct ahc_softc *ahc) | 1336 | ahc_platform_free(struct ahc_softc *ahc) |
1337 | { | 1337 | { |
1338 | struct ahc_linux_target *targ; | 1338 | struct scsi_target *starget; |
1339 | struct ahc_linux_device *dev; | ||
1340 | int i, j; | 1339 | int i, j; |
1341 | 1340 | ||
1342 | if (ahc->platform_data != NULL) { | 1341 | if (ahc->platform_data != NULL) { |
@@ -1347,22 +1346,17 @@ ahc_platform_free(struct ahc_softc *ahc) | |||
1347 | 1346 | ||
1348 | /* destroy all of the device and target objects */ | 1347 | /* destroy all of the device and target objects */ |
1349 | for (i = 0; i < AHC_NUM_TARGETS; i++) { | 1348 | for (i = 0; i < AHC_NUM_TARGETS; i++) { |
1350 | targ = ahc->platform_data->targets[i]; | 1349 | starget = ahc->platform_data->starget[i]; |
1351 | if (targ != NULL) { | 1350 | if (starget != NULL) { |
1352 | /* Keep target around through the loop. */ | ||
1353 | targ->refcount++; | ||
1354 | for (j = 0; j < AHC_NUM_LUNS; j++) { | 1351 | for (j = 0; j < AHC_NUM_LUNS; j++) { |
1352 | struct ahc_linux_target *targ = | ||
1353 | scsi_transport_target_data(starget); | ||
1355 | 1354 | ||
1356 | if (targ->devices[j] == NULL) | 1355 | if (targ->sdev[j] == NULL) |
1357 | continue; | 1356 | continue; |
1358 | dev = targ->devices[j]; | 1357 | targ->sdev[j] = NULL; |
1359 | ahc_linux_free_device(ahc, dev); | ||
1360 | } | 1358 | } |
1361 | /* | 1359 | ahc->platform_data->starget[i] = NULL; |
1362 | * Forcibly free the target now that | ||
1363 | * all devices are gone. | ||
1364 | */ | ||
1365 | ahc_linux_free_target(ahc, targ); | ||
1366 | } | 1360 | } |
1367 | } | 1361 | } |
1368 | 1362 | ||
@@ -1395,15 +1389,25 @@ void | |||
1395 | ahc_platform_set_tags(struct ahc_softc *ahc, struct ahc_devinfo *devinfo, | 1389 | ahc_platform_set_tags(struct ahc_softc *ahc, struct ahc_devinfo *devinfo, |
1396 | ahc_queue_alg alg) | 1390 | ahc_queue_alg alg) |
1397 | { | 1391 | { |
1392 | struct scsi_target *starget; | ||
1393 | struct ahc_linux_target *targ; | ||
1398 | struct ahc_linux_device *dev; | 1394 | struct ahc_linux_device *dev; |
1395 | struct scsi_device *sdev; | ||
1396 | u_int target_offset; | ||
1399 | int was_queuing; | 1397 | int was_queuing; |
1400 | int now_queuing; | 1398 | int now_queuing; |
1401 | 1399 | ||
1402 | dev = ahc_linux_get_device(ahc, devinfo->channel - 'A', | 1400 | target_offset = devinfo->target; |
1403 | devinfo->target, | 1401 | if (devinfo->channel != 'A') |
1404 | devinfo->lun); | 1402 | target_offset += 8; |
1405 | if (dev == NULL) | 1403 | starget = ahc->platform_data->starget[target_offset]; |
1404 | targ = scsi_transport_target_data(starget); | ||
1405 | BUG_ON(targ == NULL); | ||
1406 | sdev = targ->sdev[devinfo->lun]; | ||
1407 | if (sdev == NULL) | ||
1406 | return; | 1408 | return; |
1409 | dev = scsi_transport_device_data(sdev); | ||
1410 | |||
1407 | was_queuing = dev->flags & (AHC_DEV_Q_BASIC|AHC_DEV_Q_TAGGED); | 1411 | was_queuing = dev->flags & (AHC_DEV_Q_BASIC|AHC_DEV_Q_TAGGED); |
1408 | switch (alg) { | 1412 | switch (alg) { |
1409 | default: | 1413 | default: |
@@ -1454,30 +1458,28 @@ ahc_platform_set_tags(struct ahc_softc *ahc, struct ahc_devinfo *devinfo, | |||
1454 | dev->maxtags = 0; | 1458 | dev->maxtags = 0; |
1455 | dev->openings = 1 - dev->active; | 1459 | dev->openings = 1 - dev->active; |
1456 | } | 1460 | } |
1457 | if (dev->scsi_device != NULL) { | 1461 | switch ((dev->flags & (AHC_DEV_Q_BASIC|AHC_DEV_Q_TAGGED))) { |
1458 | switch ((dev->flags & (AHC_DEV_Q_BASIC|AHC_DEV_Q_TAGGED))) { | 1462 | case AHC_DEV_Q_BASIC: |
1459 | case AHC_DEV_Q_BASIC: | 1463 | scsi_adjust_queue_depth(sdev, |
1460 | scsi_adjust_queue_depth(dev->scsi_device, | 1464 | MSG_SIMPLE_TASK, |
1461 | MSG_SIMPLE_TASK, | 1465 | dev->openings + dev->active); |
1462 | dev->openings + dev->active); | 1466 | break; |
1463 | break; | 1467 | case AHC_DEV_Q_TAGGED: |
1464 | case AHC_DEV_Q_TAGGED: | 1468 | scsi_adjust_queue_depth(sdev, |
1465 | scsi_adjust_queue_depth(dev->scsi_device, | 1469 | MSG_ORDERED_TASK, |
1466 | MSG_ORDERED_TASK, | 1470 | dev->openings + dev->active); |
1467 | dev->openings + dev->active); | 1471 | break; |
1468 | break; | 1472 | default: |
1469 | default: | 1473 | /* |
1470 | /* | 1474 | * We allow the OS to queue 2 untagged transactions to |
1471 | * We allow the OS to queue 2 untagged transactions to | 1475 | * us at any time even though we can only execute them |
1472 | * us at any time even though we can only execute them | 1476 | * serially on the controller/device. This should |
1473 | * serially on the controller/device. This should | 1477 | * remove some latency. |
1474 | * remove some latency. | 1478 | */ |
1475 | */ | 1479 | scsi_adjust_queue_depth(sdev, |
1476 | scsi_adjust_queue_depth(dev->scsi_device, | 1480 | /*NON-TAGGED*/0, |
1477 | /*NON-TAGGED*/0, | 1481 | /*queue depth*/2); |
1478 | /*queue depth*/2); | 1482 | break; |
1479 | break; | ||
1480 | } | ||
1481 | } | 1483 | } |
1482 | } | 1484 | } |
1483 | 1485 | ||
@@ -1523,22 +1525,20 @@ ahc_linux_user_tagdepth(struct ahc_softc *ahc, struct ahc_devinfo *devinfo) | |||
1523 | * Determines the queue depth for a given device. | 1525 | * Determines the queue depth for a given device. |
1524 | */ | 1526 | */ |
1525 | static void | 1527 | static void |
1526 | ahc_linux_device_queue_depth(struct ahc_softc *ahc, | 1528 | ahc_linux_device_queue_depth(struct scsi_device *sdev) |
1527 | struct ahc_linux_device *dev) | ||
1528 | { | 1529 | { |
1529 | struct ahc_devinfo devinfo; | 1530 | struct ahc_devinfo devinfo; |
1530 | u_int tags; | 1531 | u_int tags; |
1532 | struct ahc_softc *ahc = *((struct ahc_softc **)sdev->host->hostdata); | ||
1531 | 1533 | ||
1532 | ahc_compile_devinfo(&devinfo, | 1534 | ahc_compile_devinfo(&devinfo, |
1533 | dev->target->channel == 0 | 1535 | sdev->sdev_target->channel == 0 |
1534 | ? ahc->our_id : ahc->our_id_b, | 1536 | ? ahc->our_id : ahc->our_id_b, |
1535 | dev->target->target, dev->lun, | 1537 | sdev->sdev_target->id, sdev->lun, |
1536 | dev->target->channel == 0 ? 'A' : 'B', | 1538 | sdev->sdev_target->channel == 0 ? 'A' : 'B', |
1537 | ROLE_INITIATOR); | 1539 | ROLE_INITIATOR); |
1538 | tags = ahc_linux_user_tagdepth(ahc, &devinfo); | 1540 | tags = ahc_linux_user_tagdepth(ahc, &devinfo); |
1539 | if (tags != 0 | 1541 | if (tags != 0 && sdev->tagged_supported != 0) { |
1540 | && dev->scsi_device != NULL | ||
1541 | && dev->scsi_device->tagged_supported != 0) { | ||
1542 | 1542 | ||
1543 | ahc_set_tags(ahc, &devinfo, AHC_QUEUE_TAGGED); | 1543 | ahc_set_tags(ahc, &devinfo, AHC_QUEUE_TAGGED); |
1544 | ahc_print_devinfo(ahc, &devinfo); | 1544 | ahc_print_devinfo(ahc, &devinfo); |
@@ -1767,106 +1767,6 @@ ahc_platform_flushwork(struct ahc_softc *ahc) | |||
1767 | 1767 | ||
1768 | } | 1768 | } |
1769 | 1769 | ||
1770 | static struct ahc_linux_target* | ||
1771 | ahc_linux_alloc_target(struct ahc_softc *ahc, u_int channel, u_int target) | ||
1772 | { | ||
1773 | struct ahc_linux_target *targ; | ||
1774 | u_int target_offset; | ||
1775 | |||
1776 | target_offset = target; | ||
1777 | if (channel != 0) | ||
1778 | target_offset += 8; | ||
1779 | |||
1780 | targ = malloc(sizeof(*targ), M_DEVBUG, M_NOWAIT); | ||
1781 | if (targ == NULL) | ||
1782 | return (NULL); | ||
1783 | memset(targ, 0, sizeof(*targ)); | ||
1784 | targ->channel = channel; | ||
1785 | targ->target = target; | ||
1786 | targ->ahc = ahc; | ||
1787 | ahc->platform_data->targets[target_offset] = targ; | ||
1788 | return (targ); | ||
1789 | } | ||
1790 | |||
1791 | static void | ||
1792 | ahc_linux_free_target(struct ahc_softc *ahc, struct ahc_linux_target *targ) | ||
1793 | { | ||
1794 | struct ahc_devinfo devinfo; | ||
1795 | struct ahc_initiator_tinfo *tinfo; | ||
1796 | struct ahc_tmode_tstate *tstate; | ||
1797 | u_int our_id; | ||
1798 | u_int target_offset; | ||
1799 | char channel; | ||
1800 | |||
1801 | /* | ||
1802 | * Force a negotiation to async/narrow on any | ||
1803 | * future command to this device unless a bus | ||
1804 | * reset occurs between now and that command. | ||
1805 | */ | ||
1806 | channel = 'A' + targ->channel; | ||
1807 | our_id = ahc->our_id; | ||
1808 | target_offset = targ->target; | ||
1809 | if (targ->channel != 0) { | ||
1810 | target_offset += 8; | ||
1811 | our_id = ahc->our_id_b; | ||
1812 | } | ||
1813 | tinfo = ahc_fetch_transinfo(ahc, channel, our_id, | ||
1814 | targ->target, &tstate); | ||
1815 | ahc_compile_devinfo(&devinfo, our_id, targ->target, CAM_LUN_WILDCARD, | ||
1816 | channel, ROLE_INITIATOR); | ||
1817 | ahc_set_syncrate(ahc, &devinfo, NULL, 0, 0, 0, | ||
1818 | AHC_TRANS_GOAL, /*paused*/FALSE); | ||
1819 | ahc_set_width(ahc, &devinfo, MSG_EXT_WDTR_BUS_8_BIT, | ||
1820 | AHC_TRANS_GOAL, /*paused*/FALSE); | ||
1821 | ahc_update_neg_request(ahc, &devinfo, tstate, tinfo, AHC_NEG_ALWAYS); | ||
1822 | ahc->platform_data->targets[target_offset] = NULL; | ||
1823 | free(targ, M_DEVBUF); | ||
1824 | } | ||
1825 | |||
1826 | static struct ahc_linux_device* | ||
1827 | ahc_linux_alloc_device(struct ahc_softc *ahc, | ||
1828 | struct ahc_linux_target *targ, u_int lun) | ||
1829 | { | ||
1830 | struct ahc_linux_device *dev; | ||
1831 | |||
1832 | dev = malloc(sizeof(*dev), M_DEVBUG, M_NOWAIT); | ||
1833 | if (dev == NULL) | ||
1834 | return (NULL); | ||
1835 | memset(dev, 0, sizeof(*dev)); | ||
1836 | dev->lun = lun; | ||
1837 | dev->target = targ; | ||
1838 | |||
1839 | /* | ||
1840 | * We start out life using untagged | ||
1841 | * transactions of which we allow one. | ||
1842 | */ | ||
1843 | dev->openings = 1; | ||
1844 | |||
1845 | /* | ||
1846 | * Set maxtags to 0. This will be changed if we | ||
1847 | * later determine that we are dealing with | ||
1848 | * a tagged queuing capable device. | ||
1849 | */ | ||
1850 | dev->maxtags = 0; | ||
1851 | |||
1852 | targ->refcount++; | ||
1853 | targ->devices[lun] = dev; | ||
1854 | return (dev); | ||
1855 | } | ||
1856 | |||
1857 | static void | ||
1858 | ahc_linux_free_device(struct ahc_softc *ahc, struct ahc_linux_device *dev) | ||
1859 | { | ||
1860 | struct ahc_linux_target *targ; | ||
1861 | |||
1862 | targ = dev->target; | ||
1863 | targ->devices[dev->lun] = NULL; | ||
1864 | free(dev, M_DEVBUF); | ||
1865 | targ->refcount--; | ||
1866 | if (targ->refcount == 0) | ||
1867 | ahc_linux_free_target(ahc, targ); | ||
1868 | } | ||
1869 | |||
1870 | void | 1770 | void |
1871 | ahc_send_async(struct ahc_softc *ahc, char channel, | 1771 | ahc_send_async(struct ahc_softc *ahc, char channel, |
1872 | u_int target, u_int lun, ac_code code, void *arg) | 1772 | u_int target, u_int lun, ac_code code, void *arg) |
@@ -1875,11 +1775,15 @@ ahc_send_async(struct ahc_softc *ahc, char channel, | |||
1875 | case AC_TRANSFER_NEG: | 1775 | case AC_TRANSFER_NEG: |
1876 | { | 1776 | { |
1877 | char buf[80]; | 1777 | char buf[80]; |
1778 | struct scsi_target *starget; | ||
1878 | struct ahc_linux_target *targ; | 1779 | struct ahc_linux_target *targ; |
1879 | struct info_str info; | 1780 | struct info_str info; |
1880 | struct ahc_initiator_tinfo *tinfo; | 1781 | struct ahc_initiator_tinfo *tinfo; |
1881 | struct ahc_tmode_tstate *tstate; | 1782 | struct ahc_tmode_tstate *tstate; |
1882 | int target_offset; | 1783 | int target_offset; |
1784 | unsigned int target_ppr_options; | ||
1785 | |||
1786 | BUG_ON(target == CAM_TARGET_WILDCARD); | ||
1883 | 1787 | ||
1884 | info.buffer = buf; | 1788 | info.buffer = buf; |
1885 | info.length = sizeof(buf); | 1789 | info.length = sizeof(buf); |
@@ -1908,32 +1812,30 @@ ahc_send_async(struct ahc_softc *ahc, char channel, | |||
1908 | target_offset = target; | 1812 | target_offset = target; |
1909 | if (channel == 'B') | 1813 | if (channel == 'B') |
1910 | target_offset += 8; | 1814 | target_offset += 8; |
1911 | targ = ahc->platform_data->targets[target_offset]; | 1815 | starget = ahc->platform_data->starget[target_offset]; |
1816 | targ = scsi_transport_target_data(starget); | ||
1912 | if (targ == NULL) | 1817 | if (targ == NULL) |
1913 | break; | 1818 | break; |
1914 | if (tinfo->curr.period == targ->last_tinfo.period | 1819 | |
1915 | && tinfo->curr.width == targ->last_tinfo.width | 1820 | target_ppr_options = |
1916 | && tinfo->curr.offset == targ->last_tinfo.offset | 1821 | (spi_dt(starget) ? MSG_EXT_PPR_DT_REQ : 0) |
1917 | && tinfo->curr.ppr_options == targ->last_tinfo.ppr_options) | 1822 | + (spi_qas(starget) ? MSG_EXT_PPR_QAS_REQ : 0) |
1823 | + (spi_iu(starget) ? MSG_EXT_PPR_IU_REQ : 0); | ||
1824 | |||
1825 | if (tinfo->curr.period == spi_period(starget) | ||
1826 | && tinfo->curr.width == spi_width(starget) | ||
1827 | && tinfo->curr.offset == spi_offset(starget) | ||
1828 | && tinfo->curr.ppr_options == target_ppr_options) | ||
1918 | if (bootverbose == 0) | 1829 | if (bootverbose == 0) |
1919 | break; | 1830 | break; |
1920 | 1831 | ||
1921 | targ->last_tinfo.period = tinfo->curr.period; | 1832 | spi_period(starget) = tinfo->curr.period; |
1922 | targ->last_tinfo.width = tinfo->curr.width; | 1833 | spi_width(starget) = tinfo->curr.width; |
1923 | targ->last_tinfo.offset = tinfo->curr.offset; | 1834 | spi_offset(starget) = tinfo->curr.offset; |
1924 | targ->last_tinfo.ppr_options = tinfo->curr.ppr_options; | 1835 | spi_dt(starget) = tinfo->curr.ppr_options & MSG_EXT_PPR_DT_REQ; |
1925 | 1836 | spi_qas(starget) = tinfo->curr.ppr_options & MSG_EXT_PPR_QAS_REQ; | |
1926 | printf("(%s:%c:", ahc_name(ahc), channel); | 1837 | spi_iu(starget) = tinfo->curr.ppr_options & MSG_EXT_PPR_IU_REQ; |
1927 | if (target == CAM_TARGET_WILDCARD) | 1838 | spi_display_xfer_agreement(starget); |
1928 | printf("*): "); | ||
1929 | else | ||
1930 | printf("%d): ", target); | ||
1931 | ahc_format_transinfo(&info, &tinfo->curr); | ||
1932 | if (info.pos < info.length) | ||
1933 | *info.buffer = '\0'; | ||
1934 | else | ||
1935 | buf[info.length - 1] = '\0'; | ||
1936 | printf("%s", buf); | ||
1937 | break; | 1839 | break; |
1938 | } | 1840 | } |
1939 | case AC_SENT_BDR: | 1841 | case AC_SENT_BDR: |
@@ -2038,7 +1940,7 @@ ahc_done(struct ahc_softc *ahc, struct scb *scb) | |||
2038 | ahc_set_transaction_status(scb, CAM_REQ_CMP); | 1940 | ahc_set_transaction_status(scb, CAM_REQ_CMP); |
2039 | } | 1941 | } |
2040 | } else if (ahc_get_transaction_status(scb) == CAM_SCSI_STATUS_ERROR) { | 1942 | } else if (ahc_get_transaction_status(scb) == CAM_SCSI_STATUS_ERROR) { |
2041 | ahc_linux_handle_scsi_status(ahc, dev, scb); | 1943 | ahc_linux_handle_scsi_status(ahc, cmd->device, scb); |
2042 | } | 1944 | } |
2043 | 1945 | ||
2044 | if (dev->openings == 1 | 1946 | if (dev->openings == 1 |
@@ -2077,14 +1979,15 @@ ahc_done(struct ahc_softc *ahc, struct scb *scb) | |||
2077 | 1979 | ||
2078 | static void | 1980 | static void |
2079 | ahc_linux_handle_scsi_status(struct ahc_softc *ahc, | 1981 | ahc_linux_handle_scsi_status(struct ahc_softc *ahc, |
2080 | struct ahc_linux_device *dev, struct scb *scb) | 1982 | struct scsi_device *sdev, struct scb *scb) |
2081 | { | 1983 | { |
2082 | struct ahc_devinfo devinfo; | 1984 | struct ahc_devinfo devinfo; |
1985 | struct ahc_linux_device *dev = scsi_transport_device_data(sdev); | ||
2083 | 1986 | ||
2084 | ahc_compile_devinfo(&devinfo, | 1987 | ahc_compile_devinfo(&devinfo, |
2085 | ahc->our_id, | 1988 | ahc->our_id, |
2086 | dev->target->target, dev->lun, | 1989 | sdev->sdev_target->id, sdev->lun, |
2087 | dev->target->channel == 0 ? 'A' : 'B', | 1990 | sdev->sdev_target->channel == 0 ? 'A' : 'B', |
2088 | ROLE_INITIATOR); | 1991 | ROLE_INITIATOR); |
2089 | 1992 | ||
2090 | /* | 1993 | /* |
@@ -2368,8 +2271,7 @@ ahc_linux_queue_recovery_cmd(struct scsi_cmnd *cmd, scb_flag flag) | |||
2368 | * at all, and the system wanted us to just abort the | 2271 | * at all, and the system wanted us to just abort the |
2369 | * command, return success. | 2272 | * command, return success. |
2370 | */ | 2273 | */ |
2371 | dev = ahc_linux_get_device(ahc, cmd->device->channel, cmd->device->id, | 2274 | dev = scsi_transport_device_data(cmd->device); |
2372 | cmd->device->lun); | ||
2373 | 2275 | ||
2374 | if (dev == NULL) { | 2276 | if (dev == NULL) { |
2375 | /* | 2277 | /* |
@@ -2626,18 +2528,6 @@ ahc_platform_dump_card_state(struct ahc_softc *ahc) | |||
2626 | 2528 | ||
2627 | static void ahc_linux_exit(void); | 2529 | static void ahc_linux_exit(void); |
2628 | 2530 | ||
2629 | static void ahc_linux_get_width(struct scsi_target *starget) | ||
2630 | { | ||
2631 | struct Scsi_Host *shost = dev_to_shost(starget->dev.parent); | ||
2632 | struct ahc_softc *ahc = *((struct ahc_softc **)shost->hostdata); | ||
2633 | struct ahc_tmode_tstate *tstate; | ||
2634 | struct ahc_initiator_tinfo *tinfo | ||
2635 | = ahc_fetch_transinfo(ahc, | ||
2636 | starget->channel + 'A', | ||
2637 | shost->this_id, starget->id, &tstate); | ||
2638 | spi_width(starget) = tinfo->curr.width; | ||
2639 | } | ||
2640 | |||
2641 | static void ahc_linux_set_width(struct scsi_target *starget, int width) | 2531 | static void ahc_linux_set_width(struct scsi_target *starget, int width) |
2642 | { | 2532 | { |
2643 | struct Scsi_Host *shost = dev_to_shost(starget->dev.parent); | 2533 | struct Scsi_Host *shost = dev_to_shost(starget->dev.parent); |
@@ -2652,18 +2542,6 @@ static void ahc_linux_set_width(struct scsi_target *starget, int width) | |||
2652 | ahc_unlock(ahc, &flags); | 2542 | ahc_unlock(ahc, &flags); |
2653 | } | 2543 | } |
2654 | 2544 | ||
2655 | static void ahc_linux_get_period(struct scsi_target *starget) | ||
2656 | { | ||
2657 | struct Scsi_Host *shost = dev_to_shost(starget->dev.parent); | ||
2658 | struct ahc_softc *ahc = *((struct ahc_softc **)shost->hostdata); | ||
2659 | struct ahc_tmode_tstate *tstate; | ||
2660 | struct ahc_initiator_tinfo *tinfo | ||
2661 | = ahc_fetch_transinfo(ahc, | ||
2662 | starget->channel + 'A', | ||
2663 | shost->this_id, starget->id, &tstate); | ||
2664 | spi_period(starget) = tinfo->curr.period; | ||
2665 | } | ||
2666 | |||
2667 | static void ahc_linux_set_period(struct scsi_target *starget, int period) | 2545 | static void ahc_linux_set_period(struct scsi_target *starget, int period) |
2668 | { | 2546 | { |
2669 | struct Scsi_Host *shost = dev_to_shost(starget->dev.parent); | 2547 | struct Scsi_Host *shost = dev_to_shost(starget->dev.parent); |
@@ -2692,7 +2570,6 @@ static void ahc_linux_set_period(struct scsi_target *starget, int period) | |||
2692 | 2570 | ||
2693 | /* all PPR requests apart from QAS require wide transfers */ | 2571 | /* all PPR requests apart from QAS require wide transfers */ |
2694 | if (ppr_options & ~MSG_EXT_PPR_QAS_REQ) { | 2572 | if (ppr_options & ~MSG_EXT_PPR_QAS_REQ) { |
2695 | ahc_linux_get_width(starget); | ||
2696 | if (spi_width(starget) == 0) | 2573 | if (spi_width(starget) == 0) |
2697 | ppr_options &= MSG_EXT_PPR_QAS_REQ; | 2574 | ppr_options &= MSG_EXT_PPR_QAS_REQ; |
2698 | } | 2575 | } |
@@ -2704,18 +2581,6 @@ static void ahc_linux_set_period(struct scsi_target *starget, int period) | |||
2704 | ahc_unlock(ahc, &flags); | 2581 | ahc_unlock(ahc, &flags); |
2705 | } | 2582 | } |
2706 | 2583 | ||
2707 | static void ahc_linux_get_offset(struct scsi_target *starget) | ||
2708 | { | ||
2709 | struct Scsi_Host *shost = dev_to_shost(starget->dev.parent); | ||
2710 | struct ahc_softc *ahc = *((struct ahc_softc **)shost->hostdata); | ||
2711 | struct ahc_tmode_tstate *tstate; | ||
2712 | struct ahc_initiator_tinfo *tinfo | ||
2713 | = ahc_fetch_transinfo(ahc, | ||
2714 | starget->channel + 'A', | ||
2715 | shost->this_id, starget->id, &tstate); | ||
2716 | spi_offset(starget) = tinfo->curr.offset; | ||
2717 | } | ||
2718 | |||
2719 | static void ahc_linux_set_offset(struct scsi_target *starget, int offset) | 2584 | static void ahc_linux_set_offset(struct scsi_target *starget, int offset) |
2720 | { | 2585 | { |
2721 | struct Scsi_Host *shost = dev_to_shost(starget->dev.parent); | 2586 | struct Scsi_Host *shost = dev_to_shost(starget->dev.parent); |
@@ -2744,18 +2609,6 @@ static void ahc_linux_set_offset(struct scsi_target *starget, int offset) | |||
2744 | ahc_unlock(ahc, &flags); | 2609 | ahc_unlock(ahc, &flags); |
2745 | } | 2610 | } |
2746 | 2611 | ||
2747 | static void ahc_linux_get_dt(struct scsi_target *starget) | ||
2748 | { | ||
2749 | struct Scsi_Host *shost = dev_to_shost(starget->dev.parent); | ||
2750 | struct ahc_softc *ahc = *((struct ahc_softc **)shost->hostdata); | ||
2751 | struct ahc_tmode_tstate *tstate; | ||
2752 | struct ahc_initiator_tinfo *tinfo | ||
2753 | = ahc_fetch_transinfo(ahc, | ||
2754 | starget->channel + 'A', | ||
2755 | shost->this_id, starget->id, &tstate); | ||
2756 | spi_dt(starget) = tinfo->curr.ppr_options & MSG_EXT_PPR_DT_REQ; | ||
2757 | } | ||
2758 | |||
2759 | static void ahc_linux_set_dt(struct scsi_target *starget, int dt) | 2612 | static void ahc_linux_set_dt(struct scsi_target *starget, int dt) |
2760 | { | 2613 | { |
2761 | struct Scsi_Host *shost = dev_to_shost(starget->dev.parent); | 2614 | struct Scsi_Host *shost = dev_to_shost(starget->dev.parent); |
@@ -2787,18 +2640,6 @@ static void ahc_linux_set_dt(struct scsi_target *starget, int dt) | |||
2787 | ahc_unlock(ahc, &flags); | 2640 | ahc_unlock(ahc, &flags); |
2788 | } | 2641 | } |
2789 | 2642 | ||
2790 | static void ahc_linux_get_qas(struct scsi_target *starget) | ||
2791 | { | ||
2792 | struct Scsi_Host *shost = dev_to_shost(starget->dev.parent); | ||
2793 | struct ahc_softc *ahc = *((struct ahc_softc **)shost->hostdata); | ||
2794 | struct ahc_tmode_tstate *tstate; | ||
2795 | struct ahc_initiator_tinfo *tinfo | ||
2796 | = ahc_fetch_transinfo(ahc, | ||
2797 | starget->channel + 'A', | ||
2798 | shost->this_id, starget->id, &tstate); | ||
2799 | spi_dt(starget) = tinfo->curr.ppr_options & MSG_EXT_PPR_QAS_REQ; | ||
2800 | } | ||
2801 | |||
2802 | static void ahc_linux_set_qas(struct scsi_target *starget, int qas) | 2643 | static void ahc_linux_set_qas(struct scsi_target *starget, int qas) |
2803 | { | 2644 | { |
2804 | struct Scsi_Host *shost = dev_to_shost(starget->dev.parent); | 2645 | struct Scsi_Host *shost = dev_to_shost(starget->dev.parent); |
@@ -2827,18 +2668,6 @@ static void ahc_linux_set_qas(struct scsi_target *starget, int qas) | |||
2827 | ahc_unlock(ahc, &flags); | 2668 | ahc_unlock(ahc, &flags); |
2828 | } | 2669 | } |
2829 | 2670 | ||
2830 | static void ahc_linux_get_iu(struct scsi_target *starget) | ||
2831 | { | ||
2832 | struct Scsi_Host *shost = dev_to_shost(starget->dev.parent); | ||
2833 | struct ahc_softc *ahc = *((struct ahc_softc **)shost->hostdata); | ||
2834 | struct ahc_tmode_tstate *tstate; | ||
2835 | struct ahc_initiator_tinfo *tinfo | ||
2836 | = ahc_fetch_transinfo(ahc, | ||
2837 | starget->channel + 'A', | ||
2838 | shost->this_id, starget->id, &tstate); | ||
2839 | spi_dt(starget) = tinfo->curr.ppr_options & MSG_EXT_PPR_IU_REQ; | ||
2840 | } | ||
2841 | |||
2842 | static void ahc_linux_set_iu(struct scsi_target *starget, int iu) | 2671 | static void ahc_linux_set_iu(struct scsi_target *starget, int iu) |
2843 | { | 2672 | { |
2844 | struct Scsi_Host *shost = dev_to_shost(starget->dev.parent); | 2673 | struct Scsi_Host *shost = dev_to_shost(starget->dev.parent); |
@@ -2868,22 +2697,16 @@ static void ahc_linux_set_iu(struct scsi_target *starget, int iu) | |||
2868 | } | 2697 | } |
2869 | 2698 | ||
2870 | static struct spi_function_template ahc_linux_transport_functions = { | 2699 | static struct spi_function_template ahc_linux_transport_functions = { |
2871 | .get_offset = ahc_linux_get_offset, | ||
2872 | .set_offset = ahc_linux_set_offset, | 2700 | .set_offset = ahc_linux_set_offset, |
2873 | .show_offset = 1, | 2701 | .show_offset = 1, |
2874 | .get_period = ahc_linux_get_period, | ||
2875 | .set_period = ahc_linux_set_period, | 2702 | .set_period = ahc_linux_set_period, |
2876 | .show_period = 1, | 2703 | .show_period = 1, |
2877 | .get_width = ahc_linux_get_width, | ||
2878 | .set_width = ahc_linux_set_width, | 2704 | .set_width = ahc_linux_set_width, |
2879 | .show_width = 1, | 2705 | .show_width = 1, |
2880 | .get_dt = ahc_linux_get_dt, | ||
2881 | .set_dt = ahc_linux_set_dt, | 2706 | .set_dt = ahc_linux_set_dt, |
2882 | .show_dt = 1, | 2707 | .show_dt = 1, |
2883 | .get_iu = ahc_linux_get_iu, | ||
2884 | .set_iu = ahc_linux_set_iu, | 2708 | .set_iu = ahc_linux_set_iu, |
2885 | .show_iu = 1, | 2709 | .show_iu = 1, |
2886 | .get_qas = ahc_linux_get_qas, | ||
2887 | .set_qas = ahc_linux_set_qas, | 2710 | .set_qas = ahc_linux_set_qas, |
2888 | .show_qas = 1, | 2711 | .show_qas = 1, |
2889 | }; | 2712 | }; |
@@ -2896,6 +2719,10 @@ ahc_linux_init(void) | |||
2896 | ahc_linux_transport_template = spi_attach_transport(&ahc_linux_transport_functions); | 2719 | ahc_linux_transport_template = spi_attach_transport(&ahc_linux_transport_functions); |
2897 | if (!ahc_linux_transport_template) | 2720 | if (!ahc_linux_transport_template) |
2898 | return -ENODEV; | 2721 | return -ENODEV; |
2722 | scsi_transport_reserve_target(ahc_linux_transport_template, | ||
2723 | sizeof(struct ahc_linux_target)); | ||
2724 | scsi_transport_reserve_device(ahc_linux_transport_template, | ||
2725 | sizeof(struct ahc_linux_device)); | ||
2899 | if (ahc_linux_detect(&aic7xxx_driver_template)) | 2726 | if (ahc_linux_detect(&aic7xxx_driver_template)) |
2900 | return 0; | 2727 | return 0; |
2901 | spi_release_transport(ahc_linux_transport_template); | 2728 | spi_release_transport(ahc_linux_transport_template); |