BusLogic.c
上传用户:lgb322
上传日期:2013-02-24
资源大小:30529k
文件大小:180k
源码类别:

嵌入式Linux

开发平台:

Unix_Linux

  1. /*
  2.   Linux Driver for BusLogic MultiMaster and FlashPoint SCSI Host Adapters
  3.   Copyright 1995-1998 by Leonard N. Zubkoff <lnz@dandelion.com>
  4.   This program is free software; you may redistribute and/or modify it under
  5.   the terms of the GNU General Public License Version 2 as published by the
  6.   Free Software Foundation.
  7.   This program is distributed in the hope that it will be useful, but
  8.   WITHOUT ANY WARRANTY, without even the implied warranty of MERCHANTABILITY
  9.   or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
  10.   for complete details.
  11.   The author respectfully requests that any modifications to this software be
  12.   sent directly to him for evaluation and testing.
  13.   Special thanks to Wayne Yen, Jin-Lon Hon, and Alex Win of BusLogic, whose
  14.   advice has been invaluable, to David Gentzel, for writing the original Linux
  15.   BusLogic driver, and to Paul Gortmaker, for being such a dedicated test site.
  16.   Finally, special thanks to Mylex/BusLogic for making the FlashPoint SCCB
  17.   Manager available as freely redistributable source code.
  18. */
  19. #define BusLogic_DriverVersion "2.1.15"
  20. #define BusLogic_DriverDate "17 August 1998"
  21. #include <linux/version.h>
  22. #include <linux/module.h>
  23. #include <linux/config.h>
  24. #include <linux/init.h>
  25. #include <linux/types.h>
  26. #include <linux/blk.h>
  27. #include <linux/blkdev.h>
  28. #include <linux/delay.h>
  29. #include <linux/ioport.h>
  30. #include <linux/mm.h>
  31. #include <linux/sched.h>
  32. #include <linux/stat.h>
  33. #include <linux/pci.h>
  34. #include <linux/spinlock.h>
  35. #include <asm/dma.h>
  36. #include <asm/io.h>
  37. #include <asm/system.h>
  38. #include "scsi.h"
  39. #include "hosts.h"
  40. #include "sd.h"
  41. #include "BusLogic.h"
  42. #include "FlashPoint.c"
  43. /*
  44.   BusLogic_DriverOptionsCount is a count of the number of BusLogic Driver
  45.   Options specifications provided via the Linux Kernel Command Line or via
  46.   the Loadable Kernel Module Installation Facility.
  47. */
  48. static int
  49.   BusLogic_DriverOptionsCount;
  50. /*
  51.   BusLogic_DriverOptions is an array of Driver Options structures representing
  52.   BusLogic Driver Options specifications provided via the Linux Kernel Command
  53.   Line or via the Loadable Kernel Module Installation Facility.
  54. */
  55. static BusLogic_DriverOptions_T
  56.   BusLogic_DriverOptions[BusLogic_MaxHostAdapters];
  57. /*
  58.   BusLogic can be assigned a string by insmod.
  59. */
  60. #ifdef MODULE
  61. static char *BusLogic;
  62. MODULE_PARM(BusLogic, "s");
  63. #endif
  64. /*
  65.   BusLogic_ProbeOptions is a set of Probe Options to be applied across
  66.   all BusLogic Host Adapters.
  67. */
  68. static BusLogic_ProbeOptions_T
  69.   BusLogic_ProbeOptions;
  70. /*
  71.   BusLogic_GlobalOptions is a set of Global Options to be applied across
  72.   all BusLogic Host Adapters.
  73. */
  74. static BusLogic_GlobalOptions_T
  75.   BusLogic_GlobalOptions;
  76. /*
  77.   BusLogic_FirstRegisteredHostAdapter and BusLogic_LastRegisteredHostAdapter
  78.   are pointers to the first and last registered BusLogic Host Adapters.
  79. */
  80. static BusLogic_HostAdapter_T
  81.   *BusLogic_FirstRegisteredHostAdapter,
  82.   *BusLogic_LastRegisteredHostAdapter;
  83. /*
  84.   BusLogic_ProbeInfoCount is the number of entries in BusLogic_ProbeInfoList.
  85. */
  86. static int
  87.   BusLogic_ProbeInfoCount;
  88. /*
  89.   BusLogic_ProbeInfoList is the list of I/O Addresses and Bus Probe Information
  90.   to be checked for potential BusLogic Host Adapters.  It is initialized by
  91.   interrogating the PCI Configuration Space on PCI machines as well as from the
  92.   list of standard BusLogic I/O Addresses.
  93. */
  94. static BusLogic_ProbeInfo_T
  95.   *BusLogic_ProbeInfoList;
  96. /*
  97.   BusLogic_CommandFailureReason holds a string identifying the reason why a
  98.   call to BusLogic_Command failed.  It is only non-NULL when BusLogic_Command
  99.   returns a failure code.
  100. */
  101. static char
  102.   *BusLogic_CommandFailureReason;
  103. /*
  104.   BusLogic_AnnounceDriver announces the Driver Version and Date, Author's
  105.   Name, Copyright Notice, and Electronic Mail Address.
  106. */
  107. static void BusLogic_AnnounceDriver(BusLogic_HostAdapter_T *HostAdapter)
  108. {
  109.   BusLogic_Announce("***** BusLogic SCSI Driver Version "
  110.     BusLogic_DriverVersion " of "
  111.     BusLogic_DriverDate " *****n", HostAdapter);
  112.   BusLogic_Announce("Copyright 1995-1998 by Leonard N. Zubkoff "
  113.     "<lnz@dandelion.com>n", HostAdapter);
  114. }
  115. /*
  116.   BusLogic_DriverInfo returns the Host Adapter Name to identify this SCSI
  117.   Driver and Host Adapter.
  118. */
  119. const char *BusLogic_DriverInfo(SCSI_Host_T *Host)
  120. {
  121.   BusLogic_HostAdapter_T *HostAdapter =
  122.     (BusLogic_HostAdapter_T *) Host->hostdata;
  123.   return HostAdapter->FullModelName;
  124. }
  125. /*
  126.   BusLogic_RegisterHostAdapter adds Host Adapter to the list of registered
  127.   BusLogic Host Adapters.
  128. */
  129. static void BusLogic_RegisterHostAdapter(BusLogic_HostAdapter_T *HostAdapter)
  130. {
  131.   HostAdapter->Next = NULL;
  132.   if (BusLogic_FirstRegisteredHostAdapter == NULL)
  133.     {
  134.       BusLogic_FirstRegisteredHostAdapter = HostAdapter;
  135.       BusLogic_LastRegisteredHostAdapter = HostAdapter;
  136.     }
  137.   else
  138.     {
  139.       BusLogic_LastRegisteredHostAdapter->Next = HostAdapter;
  140.       BusLogic_LastRegisteredHostAdapter = HostAdapter;
  141.     }
  142. }
  143. /*
  144.   BusLogic_UnregisterHostAdapter removes Host Adapter from the list of
  145.   registered BusLogic Host Adapters.
  146. */
  147. static void BusLogic_UnregisterHostAdapter(BusLogic_HostAdapter_T *HostAdapter)
  148. {
  149.   if (HostAdapter == BusLogic_FirstRegisteredHostAdapter)
  150.     {
  151.       BusLogic_FirstRegisteredHostAdapter =
  152. BusLogic_FirstRegisteredHostAdapter->Next;
  153.       if (HostAdapter == BusLogic_LastRegisteredHostAdapter)
  154. BusLogic_LastRegisteredHostAdapter = NULL;
  155.     }
  156.   else
  157.     {
  158.       BusLogic_HostAdapter_T *PreviousHostAdapter =
  159. BusLogic_FirstRegisteredHostAdapter;
  160.       while (PreviousHostAdapter != NULL &&
  161.      PreviousHostAdapter->Next != HostAdapter)
  162. PreviousHostAdapter = PreviousHostAdapter->Next;
  163.       if (PreviousHostAdapter != NULL)
  164. PreviousHostAdapter->Next = HostAdapter->Next;
  165.     }
  166.   HostAdapter->Next = NULL;
  167. }
  168. /*
  169.   BusLogic_InitializeCCBs initializes a group of Command Control Blocks (CCBs)
  170.   for Host Adapter from the BlockSize bytes located at BlockPointer.  The newly
  171.   created CCBs are added to Host Adapter's free list.
  172. */
  173. static void BusLogic_InitializeCCBs(BusLogic_HostAdapter_T *HostAdapter,
  174.     void *BlockPointer, int BlockSize)
  175. {
  176.   BusLogic_CCB_T *CCB = (BusLogic_CCB_T *) BlockPointer;
  177.   memset(BlockPointer, 0, BlockSize);
  178.   CCB->AllocationGroupHead = true;
  179.   while ((BlockSize -= sizeof(BusLogic_CCB_T)) >= 0)
  180.     {
  181.       CCB->Status = BusLogic_CCB_Free;
  182.       CCB->HostAdapter = HostAdapter;
  183.       if (BusLogic_FlashPointHostAdapterP(HostAdapter))
  184. {
  185.   CCB->CallbackFunction = BusLogic_QueueCompletedCCB;
  186.   CCB->BaseAddress = HostAdapter->FlashPointInfo.BaseAddress;
  187. }
  188.       CCB->Next = HostAdapter->Free_CCBs;
  189.       CCB->NextAll = HostAdapter->All_CCBs;
  190.       HostAdapter->Free_CCBs = CCB;
  191.       HostAdapter->All_CCBs = CCB;
  192.       HostAdapter->AllocatedCCBs++;
  193.       CCB++;
  194.     }
  195. }
  196. /*
  197.   BusLogic_CreateInitialCCBs allocates the initial CCBs for Host Adapter.
  198. */
  199. static boolean BusLogic_CreateInitialCCBs(BusLogic_HostAdapter_T *HostAdapter)
  200. {
  201.   int BlockSize = BusLogic_CCB_AllocationGroupSize * sizeof(BusLogic_CCB_T);
  202.   while (HostAdapter->AllocatedCCBs < HostAdapter->InitialCCBs)
  203.     {
  204.       void *BlockPointer = kmalloc(BlockSize,
  205.    (HostAdapter->BounceBuffersRequired
  206.     ? GFP_ATOMIC | GFP_DMA
  207.     : GFP_ATOMIC));
  208.       if (BlockPointer == NULL)
  209. {
  210.   BusLogic_Error("UNABLE TO ALLOCATE CCB GROUP - DETACHINGn",
  211.  HostAdapter);
  212.   return false;
  213. }
  214.       BusLogic_InitializeCCBs(HostAdapter, BlockPointer, BlockSize);
  215.     }
  216.   return true;
  217. }
  218. /*
  219.   BusLogic_DestroyCCBs deallocates the CCBs for Host Adapter.
  220. */
  221. static void BusLogic_DestroyCCBs(BusLogic_HostAdapter_T *HostAdapter)
  222. {
  223.   BusLogic_CCB_T *NextCCB = HostAdapter->All_CCBs, *CCB;
  224.   HostAdapter->All_CCBs = NULL;
  225.   HostAdapter->Free_CCBs = NULL;
  226.   while ((CCB = NextCCB) != NULL)
  227.     {
  228.       NextCCB = CCB->NextAll;
  229.       if (CCB->AllocationGroupHead)
  230. kfree(CCB);
  231.     }
  232. }
  233. /*
  234.   BusLogic_CreateAdditionalCCBs allocates Additional CCBs for Host Adapter.  If
  235.   allocation fails and there are no remaining CCBs available, the Driver Queue
  236.   Depth is decreased to a known safe value to avoid potential deadlocks when
  237.   multiple host adapters share the same IRQ Channel.
  238. */
  239. static void BusLogic_CreateAdditionalCCBs(BusLogic_HostAdapter_T *HostAdapter,
  240.   int AdditionalCCBs,
  241.   boolean SuccessMessageP)
  242. {
  243.   int BlockSize = BusLogic_CCB_AllocationGroupSize * sizeof(BusLogic_CCB_T);
  244.   int PreviouslyAllocated = HostAdapter->AllocatedCCBs;
  245.   if (AdditionalCCBs <= 0) return;
  246.   while (HostAdapter->AllocatedCCBs - PreviouslyAllocated < AdditionalCCBs)
  247.     {
  248.       void *BlockPointer = kmalloc(BlockSize,
  249.    (HostAdapter->BounceBuffersRequired
  250.     ? GFP_ATOMIC | GFP_DMA
  251.     : GFP_ATOMIC));
  252.       if (BlockPointer == NULL) break;
  253.       BusLogic_InitializeCCBs(HostAdapter, BlockPointer, BlockSize);
  254.     }
  255.   if (HostAdapter->AllocatedCCBs > PreviouslyAllocated)
  256.     {
  257.       if (SuccessMessageP)
  258. BusLogic_Notice("Allocated %d additional CCBs (total now %d)n",
  259. HostAdapter,
  260. HostAdapter->AllocatedCCBs - PreviouslyAllocated,
  261. HostAdapter->AllocatedCCBs);
  262.       return;
  263.     }
  264.   BusLogic_Notice("Failed to allocate additional CCBsn", HostAdapter);
  265.   if (HostAdapter->DriverQueueDepth >
  266.       HostAdapter->AllocatedCCBs - HostAdapter->TargetDeviceCount)
  267.     {
  268.       HostAdapter->DriverQueueDepth =
  269. HostAdapter->AllocatedCCBs - HostAdapter->TargetDeviceCount;
  270.       HostAdapter->SCSI_Host->can_queue = HostAdapter->DriverQueueDepth;
  271.     }
  272. }
  273. /*
  274.   BusLogic_AllocateCCB allocates a CCB from Host Adapter's free list,
  275.   allocating more memory from the Kernel if necessary.  The Host Adapter's
  276.   Lock should already have been acquired by the caller.
  277. */
  278. static BusLogic_CCB_T *BusLogic_AllocateCCB(BusLogic_HostAdapter_T
  279.     *HostAdapter)
  280. {
  281.   static unsigned long SerialNumber = 0;
  282.   BusLogic_CCB_T *CCB;
  283.   CCB = HostAdapter->Free_CCBs;
  284.   if (CCB != NULL)
  285.     {
  286.       CCB->SerialNumber = ++SerialNumber;
  287.       HostAdapter->Free_CCBs = CCB->Next;
  288.       CCB->Next = NULL;
  289.       if (HostAdapter->Free_CCBs == NULL)
  290. BusLogic_CreateAdditionalCCBs(HostAdapter,
  291.       HostAdapter->IncrementalCCBs,
  292.       true);
  293.       return CCB;
  294.     }
  295.   BusLogic_CreateAdditionalCCBs(HostAdapter,
  296. HostAdapter->IncrementalCCBs,
  297. true);
  298.   CCB = HostAdapter->Free_CCBs;
  299.   if (CCB == NULL) return NULL;
  300.   CCB->SerialNumber = ++SerialNumber;
  301.   HostAdapter->Free_CCBs = CCB->Next;
  302.   CCB->Next = NULL;
  303.   return CCB;
  304. }
  305. /*
  306.   BusLogic_DeallocateCCB deallocates a CCB, returning it to the Host Adapter's
  307.   free list.  The Host Adapter's Lock should already have been acquired by the
  308.   caller.
  309. */
  310. static void BusLogic_DeallocateCCB(BusLogic_CCB_T *CCB)
  311. {
  312.   BusLogic_HostAdapter_T *HostAdapter = CCB->HostAdapter;
  313.   CCB->Command = NULL;
  314.   CCB->Status = BusLogic_CCB_Free;
  315.   CCB->Next = HostAdapter->Free_CCBs;
  316.   HostAdapter->Free_CCBs = CCB;
  317. }
  318. /*
  319.   BusLogic_Command sends the command OperationCode to HostAdapter, optionally
  320.   providing ParameterLength bytes of ParameterData and receiving at most
  321.   ReplyLength bytes of ReplyData; any excess reply data is received but
  322.   discarded.
  323.   On success, this function returns the number of reply bytes read from
  324.   the Host Adapter (including any discarded data); on failure, it returns
  325.   -1 if the command was invalid, or -2 if a timeout occurred.
  326.   BusLogic_Command is called exclusively during host adapter detection and
  327.   initialization, so performance and latency are not critical, and exclusive
  328.   access to the Host Adapter hardware is assumed.  Once the host adapter and
  329.   driver are initialized, the only Host Adapter command that is issued is the
  330.   single byte Execute Mailbox Command operation code, which does not require
  331.   waiting for the Host Adapter Ready bit to be set in the Status Register.
  332. */
  333. static int BusLogic_Command(BusLogic_HostAdapter_T *HostAdapter,
  334.     BusLogic_OperationCode_T OperationCode,
  335.     void *ParameterData,
  336.     int ParameterLength,
  337.     void *ReplyData,
  338.     int ReplyLength)
  339. {
  340.   unsigned char *ParameterPointer = (unsigned char *) ParameterData;
  341.   unsigned char *ReplyPointer = (unsigned char *) ReplyData;
  342.   BusLogic_StatusRegister_T StatusRegister;
  343.   BusLogic_InterruptRegister_T InterruptRegister;
  344.   ProcessorFlags_T ProcessorFlags = 0;
  345.   int ReplyBytes = 0, Result;
  346.   long TimeoutCounter;
  347.   /*
  348.     Clear out the Reply Data if provided.
  349.   */
  350.   if (ReplyLength > 0)
  351.     memset(ReplyData, 0, ReplyLength);
  352.   /*
  353.     If the IRQ Channel has not yet been acquired, then interrupts must be
  354.     disabled while issuing host adapter commands since a Command Complete
  355.     interrupt could occur if the IRQ Channel was previously enabled by another
  356.     BusLogic Host Adapter or another driver sharing the same IRQ Channel.
  357.   */
  358.   if (!HostAdapter->IRQ_ChannelAcquired)
  359.     {
  360.       save_flags(ProcessorFlags);
  361.       cli();
  362.     }
  363.   /*
  364.     Wait for the Host Adapter Ready bit to be set and the Command/Parameter
  365.     Register Busy bit to be reset in the Status Register.
  366.   */
  367.   TimeoutCounter = 10000;
  368.   while (--TimeoutCounter >= 0)
  369.     {
  370.       StatusRegister.All = BusLogic_ReadStatusRegister(HostAdapter);
  371.       if (StatusRegister.Bits.HostAdapterReady &&
  372.   !StatusRegister.Bits.CommandParameterRegisterBusy)
  373. break;
  374.       udelay(100);
  375.     }
  376.   if (TimeoutCounter < 0)
  377.     {
  378.       BusLogic_CommandFailureReason = "Timeout waiting for Host Adapter Ready";
  379.       Result = -2;
  380.       goto Done;
  381.     }
  382.   /*
  383.     Write the OperationCode to the Command/Parameter Register.
  384.   */
  385.   HostAdapter->HostAdapterCommandCompleted = false;
  386.   BusLogic_WriteCommandParameterRegister(HostAdapter, OperationCode);
  387.   /*
  388.     Write any additional Parameter Bytes.
  389.   */
  390.   TimeoutCounter = 10000;
  391.   while (ParameterLength > 0 && --TimeoutCounter >= 0)
  392.     {
  393.       /*
  394. Wait 100 microseconds to give the Host Adapter enough time to determine
  395. whether the last value written to the Command/Parameter Register was
  396. valid or not.  If the Command Complete bit is set in the Interrupt
  397. Register, then the Command Invalid bit in the Status Register will be
  398. reset if the Operation Code or Parameter was valid and the command
  399. has completed, or set if the Operation Code or Parameter was invalid.
  400. If the Data In Register Ready bit is set in the Status Register, then
  401. the Operation Code was valid, and data is waiting to be read back
  402. from the Host Adapter.  Otherwise, wait for the Command/Parameter
  403. Register Busy bit in the Status Register to be reset.
  404.       */
  405.       udelay(100);
  406.       InterruptRegister.All = BusLogic_ReadInterruptRegister(HostAdapter);
  407.       StatusRegister.All = BusLogic_ReadStatusRegister(HostAdapter);
  408.       if (InterruptRegister.Bits.CommandComplete) break;
  409.       if (HostAdapter->HostAdapterCommandCompleted) break;
  410.       if (StatusRegister.Bits.DataInRegisterReady) break;
  411.       if (StatusRegister.Bits.CommandParameterRegisterBusy) continue;
  412.       BusLogic_WriteCommandParameterRegister(HostAdapter, *ParameterPointer++);
  413.       ParameterLength--;
  414.     }
  415.   if (TimeoutCounter < 0)
  416.     {
  417.       BusLogic_CommandFailureReason =
  418. "Timeout waiting for Parameter Acceptance";
  419.       Result = -2;
  420.       goto Done;
  421.     }
  422.   /*
  423.     The Modify I/O Address command does not cause a Command Complete Interrupt.
  424.   */
  425.   if (OperationCode == BusLogic_ModifyIOAddress)
  426.     {
  427.       StatusRegister.All = BusLogic_ReadStatusRegister(HostAdapter);
  428.       if (StatusRegister.Bits.CommandInvalid)
  429. {
  430.   BusLogic_CommandFailureReason = "Modify I/O Address Invalid";
  431.   Result = -1;
  432.   goto Done;
  433. }
  434.       if (BusLogic_GlobalOptions.TraceConfiguration)
  435. BusLogic_Notice("BusLogic_Command(%02X) Status = %02X: "
  436. "(Modify I/O Address)n", HostAdapter,
  437. OperationCode, StatusRegister.All);
  438.       Result = 0;
  439.       goto Done;
  440.     }
  441.   /*
  442.     Select an appropriate timeout value for awaiting command completion.
  443.   */
  444.   switch (OperationCode)
  445.     {
  446.     case BusLogic_InquireInstalledDevicesID0to7:
  447.     case BusLogic_InquireInstalledDevicesID8to15:
  448.     case BusLogic_InquireTargetDevices:
  449.       /* Approximately 60 seconds. */
  450.       TimeoutCounter = 60*10000;
  451.       break;
  452.     default:
  453.       /* Approximately 1 second. */
  454.       TimeoutCounter = 10000;
  455.       break;
  456.     }
  457.   /*
  458.     Receive any Reply Bytes, waiting for either the Command Complete bit to
  459.     be set in the Interrupt Register, or for the Interrupt Handler to set the
  460.     Host Adapter Command Completed bit in the Host Adapter structure.
  461.   */
  462.   while (--TimeoutCounter >= 0)
  463.     {
  464.       InterruptRegister.All = BusLogic_ReadInterruptRegister(HostAdapter);
  465.       StatusRegister.All = BusLogic_ReadStatusRegister(HostAdapter);
  466.       if (InterruptRegister.Bits.CommandComplete) break;
  467.       if (HostAdapter->HostAdapterCommandCompleted) break;
  468.       if (StatusRegister.Bits.DataInRegisterReady)
  469. {
  470.   if (++ReplyBytes <= ReplyLength)
  471.     *ReplyPointer++ = BusLogic_ReadDataInRegister(HostAdapter);
  472.   else BusLogic_ReadDataInRegister(HostAdapter);
  473. }
  474.       if (OperationCode == BusLogic_FetchHostAdapterLocalRAM &&
  475.   StatusRegister.Bits.HostAdapterReady) break;
  476.       udelay(100);
  477.     }
  478.   if (TimeoutCounter < 0)
  479.     {
  480.       BusLogic_CommandFailureReason = "Timeout waiting for Command Complete";
  481.       Result = -2;
  482.       goto Done;
  483.     }
  484.   /*
  485.     Clear any pending Command Complete Interrupt.
  486.   */
  487.   BusLogic_InterruptReset(HostAdapter);
  488.   /*
  489.     Provide tracing information if requested.
  490.   */
  491.   if (BusLogic_GlobalOptions.TraceConfiguration)
  492.     {
  493.       int i;
  494.       BusLogic_Notice("BusLogic_Command(%02X) Status = %02X: %2d ==> %2d:",
  495.       HostAdapter, OperationCode,
  496.       StatusRegister.All, ReplyLength, ReplyBytes);
  497.       if (ReplyLength > ReplyBytes) ReplyLength = ReplyBytes;
  498.       for (i = 0; i < ReplyLength; i++)
  499. BusLogic_Notice(" %02X", HostAdapter,
  500. ((unsigned char *) ReplyData)[i]);
  501.       BusLogic_Notice("n", HostAdapter);
  502.     }
  503.   /*
  504.     Process Command Invalid conditions.
  505.   */
  506.   if (StatusRegister.Bits.CommandInvalid)
  507.     {
  508.       /*
  509. Some early BusLogic Host Adapters may not recover properly from
  510. a Command Invalid condition, so if this appears to be the case,
  511. a Soft Reset is issued to the Host Adapter.  Potentially invalid
  512. commands are never attempted after Mailbox Initialization is
  513. performed, so there should be no Host Adapter state lost by a
  514. Soft Reset in response to a Command Invalid condition.
  515.       */
  516.       udelay(1000);
  517.       StatusRegister.All = BusLogic_ReadStatusRegister(HostAdapter);
  518.       if (StatusRegister.Bits.CommandInvalid ||
  519.   StatusRegister.Bits.Reserved ||
  520.   StatusRegister.Bits.DataInRegisterReady ||
  521.   StatusRegister.Bits.CommandParameterRegisterBusy ||
  522.   !StatusRegister.Bits.HostAdapterReady ||
  523.   !StatusRegister.Bits.InitializationRequired ||
  524.   StatusRegister.Bits.DiagnosticActive ||
  525.   StatusRegister.Bits.DiagnosticFailure)
  526. {
  527.   BusLogic_SoftReset(HostAdapter);
  528.   udelay(1000);
  529. }
  530.       BusLogic_CommandFailureReason = "Command Invalid";
  531.       Result = -1;
  532.       goto Done;
  533.     }
  534.   /*
  535.     Handle Excess Parameters Supplied conditions.
  536.   */
  537.   if (ParameterLength > 0)
  538.     {
  539.       BusLogic_CommandFailureReason = "Excess Parameters Supplied";
  540.       Result = -1;
  541.       goto Done;
  542.     }
  543.   /*
  544.     Indicate the command completed successfully.
  545.   */
  546.   BusLogic_CommandFailureReason = NULL;
  547.   Result = ReplyBytes;
  548.   /*
  549.     Restore the interrupt status if necessary and return.
  550.   */
  551. Done:
  552.   if (!HostAdapter->IRQ_ChannelAcquired)
  553.     restore_flags(ProcessorFlags);
  554.   return Result;
  555. }
  556. /*
  557.   BusLogic_AppendProbeAddressISA appends a single ISA I/O Address to the list
  558.   of I/O Address and Bus Probe Information to be checked for potential BusLogic
  559.   Host Adapters.
  560. */
  561. static void BusLogic_AppendProbeAddressISA(BusLogic_IO_Address_T IO_Address)
  562. {
  563.   BusLogic_ProbeInfo_T *ProbeInfo;
  564.   if (BusLogic_ProbeInfoCount >= BusLogic_MaxHostAdapters) return;
  565.   ProbeInfo = &BusLogic_ProbeInfoList[BusLogic_ProbeInfoCount++];
  566.   ProbeInfo->HostAdapterType = BusLogic_MultiMaster;
  567.   ProbeInfo->HostAdapterBusType = BusLogic_ISA_Bus;
  568.   ProbeInfo->IO_Address = IO_Address;
  569. }
  570. /*
  571.   BusLogic_InitializeProbeInfoListISA initializes the list of I/O Address and
  572.   Bus Probe Information to be checked for potential BusLogic SCSI Host Adapters
  573.   only from the list of standard BusLogic MultiMaster ISA I/O Addresses.
  574. */
  575. static void BusLogic_InitializeProbeInfoListISA(BusLogic_HostAdapter_T
  576. *PrototypeHostAdapter)
  577. {
  578.   /*
  579.     If BusLogic Driver Options specifications requested that ISA Bus Probes
  580.     be inhibited, do not proceed further.
  581.   */
  582.   if (BusLogic_ProbeOptions.NoProbeISA) return;
  583.   /*
  584.     Append the list of standard BusLogic MultiMaster ISA I/O Addresses.
  585.   */
  586.   if (BusLogic_ProbeOptions.LimitedProbeISA
  587.       ? BusLogic_ProbeOptions.Probe330
  588.       : check_region(0x330, BusLogic_MultiMasterAddressCount) == 0)
  589.     BusLogic_AppendProbeAddressISA(0x330);
  590.   if (BusLogic_ProbeOptions.LimitedProbeISA
  591.       ? BusLogic_ProbeOptions.Probe334
  592.       : check_region(0x334, BusLogic_MultiMasterAddressCount) == 0)
  593.     BusLogic_AppendProbeAddressISA(0x334);
  594.   if (BusLogic_ProbeOptions.LimitedProbeISA
  595.       ? BusLogic_ProbeOptions.Probe230
  596.       : check_region(0x230, BusLogic_MultiMasterAddressCount) == 0)
  597.     BusLogic_AppendProbeAddressISA(0x230);
  598.   if (BusLogic_ProbeOptions.LimitedProbeISA
  599.       ? BusLogic_ProbeOptions.Probe234
  600.       : check_region(0x234, BusLogic_MultiMasterAddressCount) == 0)
  601.     BusLogic_AppendProbeAddressISA(0x234);
  602.   if (BusLogic_ProbeOptions.LimitedProbeISA
  603.       ? BusLogic_ProbeOptions.Probe130
  604.       : check_region(0x130, BusLogic_MultiMasterAddressCount) == 0)
  605.     BusLogic_AppendProbeAddressISA(0x130);
  606.   if (BusLogic_ProbeOptions.LimitedProbeISA
  607.       ? BusLogic_ProbeOptions.Probe134
  608.       : check_region(0x134, BusLogic_MultiMasterAddressCount) == 0)
  609.     BusLogic_AppendProbeAddressISA(0x134);
  610. }
  611. #ifdef CONFIG_PCI
  612. /*
  613.   BusLogic_SortProbeInfo sorts a section of BusLogic_ProbeInfoList in order
  614.   of increasing PCI Bus and Device Number.
  615. */
  616. static void BusLogic_SortProbeInfo(BusLogic_ProbeInfo_T *ProbeInfoList,
  617.    int ProbeInfoCount)
  618. {
  619.   int LastInterchange = ProbeInfoCount-1, Bound, j;
  620.   while (LastInterchange > 0)
  621.     {
  622.       Bound = LastInterchange;
  623.       LastInterchange = 0;
  624.       for (j = 0; j < Bound; j++)
  625. {
  626.   BusLogic_ProbeInfo_T *ProbeInfo1 = &ProbeInfoList[j];
  627.   BusLogic_ProbeInfo_T *ProbeInfo2 = &ProbeInfoList[j+1];
  628.   if (ProbeInfo1->Bus > ProbeInfo2->Bus ||
  629.       (ProbeInfo1->Bus == ProbeInfo2->Bus &&
  630.        (ProbeInfo1->Device > ProbeInfo2->Device)))
  631.     {
  632.       BusLogic_ProbeInfo_T TempProbeInfo;
  633.       memcpy(&TempProbeInfo, ProbeInfo1, sizeof(BusLogic_ProbeInfo_T));
  634.       memcpy(ProbeInfo1, ProbeInfo2, sizeof(BusLogic_ProbeInfo_T));
  635.       memcpy(ProbeInfo2, &TempProbeInfo, sizeof(BusLogic_ProbeInfo_T));
  636.       LastInterchange = j;
  637.     }
  638. }
  639.     }
  640. }
  641. /*
  642.   BusLogic_InitializeMultiMasterProbeInfo initializes the list of I/O Address
  643.   and Bus Probe Information to be checked for potential BusLogic MultiMaster
  644.   SCSI Host Adapters by interrogating the PCI Configuration Space on PCI
  645.   machines as well as from the list of standard BusLogic MultiMaster ISA
  646.   I/O Addresses.  It returns the number of PCI MultiMaster Host Adapters found.
  647. */
  648. static int BusLogic_InitializeMultiMasterProbeInfo(BusLogic_HostAdapter_T
  649.    *PrototypeHostAdapter)
  650. {
  651.   BusLogic_ProbeInfo_T *PrimaryProbeInfo =
  652.     &BusLogic_ProbeInfoList[BusLogic_ProbeInfoCount];
  653.   int NonPrimaryPCIMultiMasterIndex = BusLogic_ProbeInfoCount + 1;
  654.   int NonPrimaryPCIMultiMasterCount = 0, PCIMultiMasterCount = 0;
  655.   boolean ForceBusDeviceScanningOrder = false;
  656.   boolean ForceBusDeviceScanningOrderChecked = false;
  657.   boolean StandardAddressSeen[6];
  658.   PCI_Device_T *PCI_Device = NULL;
  659.   int i;
  660.   if (BusLogic_ProbeInfoCount >= BusLogic_MaxHostAdapters) return 0;
  661.   BusLogic_ProbeInfoCount++;
  662.   for (i = 0; i < 6; i++)
  663.     StandardAddressSeen[i] = false;
  664.   /*
  665.     Iterate over the MultiMaster PCI Host Adapters.  For each enumerated host
  666.     adapter, determine whether its ISA Compatible I/O Port is enabled and if
  667.     so, whether it is assigned the Primary I/O Address.  A host adapter that is
  668.     assigned the Primary I/O Address will always be the preferred boot device.
  669.     The MultiMaster BIOS will first recognize a host adapter at the Primary I/O
  670.     Address, then any other PCI host adapters, and finally any host adapters
  671.     located at the remaining standard ISA I/O Addresses.  When a PCI host
  672.     adapter is found with its ISA Compatible I/O Port enabled, a command is
  673.     issued to disable the ISA Compatible I/O Port, and it is noted that the
  674.     particular standard ISA I/O Address need not be probed.
  675.   */
  676.   PrimaryProbeInfo->IO_Address = 0;
  677.   while ((PCI_Device = pci_find_device(PCI_VENDOR_ID_BUSLOGIC,
  678.        PCI_DEVICE_ID_BUSLOGIC_MULTIMASTER,
  679.        PCI_Device)) != NULL)
  680.     {
  681.       BusLogic_HostAdapter_T *HostAdapter = PrototypeHostAdapter;
  682.       BusLogic_PCIHostAdapterInformation_T PCIHostAdapterInformation;
  683.       BusLogic_ModifyIOAddressRequest_T ModifyIOAddressRequest;
  684.       unsigned char Bus = PCI_Device->bus->number;
  685.       unsigned char Device = PCI_Device->devfn >> 3;
  686.       unsigned int IRQ_Channel;
  687.       unsigned long BaseAddress0;
  688.       unsigned long BaseAddress1;
  689.       BusLogic_IO_Address_T IO_Address;
  690.       BusLogic_PCI_Address_T PCI_Address;
  691.       if (pci_enable_device(PCI_Device))
  692.        continue;
  693.       
  694.       IRQ_Channel = PCI_Device->irq;
  695.       IO_Address  = BaseAddress0 = pci_resource_start(PCI_Device, 0);
  696.       PCI_Address = BaseAddress1 = pci_resource_start(PCI_Device, 1);
  697.       if (pci_resource_flags(PCI_Device, 0) & IORESOURCE_MEM)
  698. {
  699.   BusLogic_Error("BusLogic: Base Address0 0x%X not I/O for "
  700.  "MultiMaster Host Adaptern", NULL, BaseAddress0);
  701.   BusLogic_Error("at PCI Bus %d Device %d I/O Address 0x%Xn",
  702.  NULL, Bus, Device, IO_Address);
  703.   continue;
  704. }
  705.       if (pci_resource_flags(PCI_Device,1) & IORESOURCE_IO)
  706. {
  707.   BusLogic_Error("BusLogic: Base Address1 0x%X not Memory for "
  708.  "MultiMaster Host Adaptern", NULL, BaseAddress1);
  709.   BusLogic_Error("at PCI Bus %d Device %d PCI Address 0x%Xn",
  710.  NULL, Bus, Device, PCI_Address);
  711.   continue;
  712. }
  713.       if (IRQ_Channel == 0)
  714. {
  715.   BusLogic_Error("BusLogic: IRQ Channel %d illegal for "
  716.  "MultiMaster Host Adaptern", NULL, IRQ_Channel);
  717.   BusLogic_Error("at PCI Bus %d Device %d I/O Address 0x%Xn",
  718.  NULL, Bus, Device, IO_Address);
  719.   continue;
  720. }
  721.       if (BusLogic_GlobalOptions.TraceProbe)
  722. {
  723.   BusLogic_Notice("BusLogic: PCI MultiMaster Host Adapter "
  724.   "detected atn", NULL);
  725.   BusLogic_Notice("BusLogic: PCI Bus %d Device %d I/O Address "
  726.   "0x%X PCI Address 0x%Xn", NULL,
  727.   Bus, Device, IO_Address, PCI_Address);
  728. }
  729.       /*
  730. Issue the Inquire PCI Host Adapter Information command to determine
  731. the ISA Compatible I/O Port.  If the ISA Compatible I/O Port is
  732. known and enabled, note that the particular Standard ISA I/O
  733. Address should not be probed.
  734.       */
  735.       HostAdapter->IO_Address = IO_Address;
  736.       BusLogic_InterruptReset(HostAdapter);
  737.       if (BusLogic_Command(HostAdapter,
  738.    BusLogic_InquirePCIHostAdapterInformation,
  739.    NULL, 0, &PCIHostAdapterInformation,
  740.    sizeof(PCIHostAdapterInformation))
  741.   == sizeof(PCIHostAdapterInformation))
  742. {
  743.   if (PCIHostAdapterInformation.ISACompatibleIOPort < 6)
  744.     StandardAddressSeen[PCIHostAdapterInformation
  745. .ISACompatibleIOPort] = true;
  746. }
  747.       else PCIHostAdapterInformation.ISACompatibleIOPort =
  748.      BusLogic_IO_Disable;
  749.       /*
  750. Issue the Modify I/O Address command to disable the ISA Compatible
  751. I/O Port.
  752.       */
  753.       ModifyIOAddressRequest = BusLogic_IO_Disable;
  754.       BusLogic_Command(HostAdapter, BusLogic_ModifyIOAddress,
  755.        &ModifyIOAddressRequest,
  756.        sizeof(ModifyIOAddressRequest), NULL, 0);
  757.       /*
  758. For the first MultiMaster Host Adapter enumerated, issue the Fetch
  759. Host Adapter Local RAM command to read byte 45 of the AutoSCSI area,
  760. for the setting of the "Use Bus And Device # For PCI Scanning Seq."
  761. option.  Issue the Inquire Board ID command since this option is
  762. only valid for the BT-948/958/958D.
  763.       */
  764.       if (!ForceBusDeviceScanningOrderChecked)
  765. {
  766.   BusLogic_FetchHostAdapterLocalRAMRequest_T
  767.     FetchHostAdapterLocalRAMRequest;
  768.   BusLogic_AutoSCSIByte45_T AutoSCSIByte45;
  769.   BusLogic_BoardID_T BoardID;
  770.   FetchHostAdapterLocalRAMRequest.ByteOffset =
  771.     BusLogic_AutoSCSI_BaseOffset + 45;
  772.   FetchHostAdapterLocalRAMRequest.ByteCount =
  773.     sizeof(AutoSCSIByte45);
  774.   BusLogic_Command(HostAdapter,
  775.    BusLogic_FetchHostAdapterLocalRAM,
  776.    &FetchHostAdapterLocalRAMRequest,
  777.    sizeof(FetchHostAdapterLocalRAMRequest),
  778.    &AutoSCSIByte45, sizeof(AutoSCSIByte45));
  779.   BusLogic_Command(HostAdapter, BusLogic_InquireBoardID,
  780.    NULL, 0, &BoardID, sizeof(BoardID));
  781.   if (BoardID.FirmwareVersion1stDigit == '5')
  782.     ForceBusDeviceScanningOrder =
  783.       AutoSCSIByte45.ForceBusDeviceScanningOrder;
  784.   ForceBusDeviceScanningOrderChecked = true;
  785. }
  786.       /*
  787. Determine whether this MultiMaster Host Adapter has its ISA
  788. Compatible I/O Port enabled and is assigned the Primary I/O Address.
  789. If it does, then it is the Primary MultiMaster Host Adapter and must
  790. be recognized first.  If it does not, then it is added to the list
  791. for probing after any Primary MultiMaster Host Adapter is probed.
  792.       */
  793.       if (PCIHostAdapterInformation.ISACompatibleIOPort == BusLogic_IO_330)
  794. {
  795.   PrimaryProbeInfo->HostAdapterType = BusLogic_MultiMaster;
  796.   PrimaryProbeInfo->HostAdapterBusType = BusLogic_PCI_Bus;
  797.   PrimaryProbeInfo->IO_Address = IO_Address;
  798.   PrimaryProbeInfo->PCI_Address = PCI_Address;
  799.   PrimaryProbeInfo->Bus = Bus;
  800.   PrimaryProbeInfo->Device = Device;
  801.   PrimaryProbeInfo->IRQ_Channel = IRQ_Channel;
  802.   PCIMultiMasterCount++;
  803. }
  804.       else if (BusLogic_ProbeInfoCount < BusLogic_MaxHostAdapters)
  805. {
  806.   BusLogic_ProbeInfo_T *ProbeInfo =
  807.     &BusLogic_ProbeInfoList[BusLogic_ProbeInfoCount++];
  808.   ProbeInfo->HostAdapterType = BusLogic_MultiMaster;
  809.   ProbeInfo->HostAdapterBusType = BusLogic_PCI_Bus;
  810.   ProbeInfo->IO_Address = IO_Address;
  811.   ProbeInfo->PCI_Address = PCI_Address;
  812.   ProbeInfo->Bus = Bus;
  813.   ProbeInfo->Device = Device;
  814.   ProbeInfo->IRQ_Channel = IRQ_Channel;
  815.   NonPrimaryPCIMultiMasterCount++;
  816.   PCIMultiMasterCount++;
  817. }
  818.       else BusLogic_Warning("BusLogic: Too many Host Adapters "
  819.     "detectedn", NULL);
  820.     }
  821.   /*
  822.     If the AutoSCSI "Use Bus And Device # For PCI Scanning Seq." option is ON
  823.     for the first enumerated MultiMaster Host Adapter, and if that host adapter
  824.     is a BT-948/958/958D, then the MultiMaster BIOS will recognize MultiMaster
  825.     Host Adapters in the order of increasing PCI Bus and Device Number.  In
  826.     that case, sort the probe information into the same order the BIOS uses.
  827.     If this option is OFF, then the MultiMaster BIOS will recognize MultiMaster
  828.     Host Adapters in the order they are enumerated by the PCI BIOS, and hence
  829.     no sorting is necessary.
  830.   */
  831.   if (ForceBusDeviceScanningOrder)
  832.     BusLogic_SortProbeInfo(&BusLogic_ProbeInfoList[
  833.       NonPrimaryPCIMultiMasterIndex],
  834.    NonPrimaryPCIMultiMasterCount);
  835.   /*
  836.     If no PCI MultiMaster Host Adapter is assigned the Primary I/O Address,
  837.     then the Primary I/O Address must be probed explicitly before any PCI
  838.     host adapters are probed.
  839.   */
  840.   if (!BusLogic_ProbeOptions.NoProbeISA)
  841.     if (PrimaryProbeInfo->IO_Address == 0 &&
  842. (BusLogic_ProbeOptions.LimitedProbeISA
  843.  ? BusLogic_ProbeOptions.Probe330
  844.  : check_region(0x330, BusLogic_MultiMasterAddressCount) == 0))
  845.       {
  846. PrimaryProbeInfo->HostAdapterType = BusLogic_MultiMaster;
  847. PrimaryProbeInfo->HostAdapterBusType = BusLogic_ISA_Bus;
  848. PrimaryProbeInfo->IO_Address = 0x330;
  849.       }
  850.   /*
  851.     Append the list of standard BusLogic MultiMaster ISA I/O Addresses,
  852.     omitting the Primary I/O Address which has already been handled.
  853.   */
  854.   if (!BusLogic_ProbeOptions.NoProbeISA)
  855.     {
  856.       if (!StandardAddressSeen[1] &&
  857.   (BusLogic_ProbeOptions.LimitedProbeISA
  858.    ? BusLogic_ProbeOptions.Probe334
  859.    : check_region(0x334, BusLogic_MultiMasterAddressCount) == 0))
  860. BusLogic_AppendProbeAddressISA(0x334);
  861.       if (!StandardAddressSeen[2] &&
  862.   (BusLogic_ProbeOptions.LimitedProbeISA
  863.    ? BusLogic_ProbeOptions.Probe230
  864.    : check_region(0x230, BusLogic_MultiMasterAddressCount) == 0))
  865. BusLogic_AppendProbeAddressISA(0x230);
  866.       if (!StandardAddressSeen[3] &&
  867.   (BusLogic_ProbeOptions.LimitedProbeISA
  868.    ? BusLogic_ProbeOptions.Probe234
  869.    : check_region(0x234, BusLogic_MultiMasterAddressCount) == 0))
  870. BusLogic_AppendProbeAddressISA(0x234);
  871.       if (!StandardAddressSeen[4] &&
  872.   (BusLogic_ProbeOptions.LimitedProbeISA
  873.    ? BusLogic_ProbeOptions.Probe130
  874.    : check_region(0x130, BusLogic_MultiMasterAddressCount) == 0))
  875. BusLogic_AppendProbeAddressISA(0x130);
  876.       if (!StandardAddressSeen[5] &&
  877.   (BusLogic_ProbeOptions.LimitedProbeISA
  878.    ? BusLogic_ProbeOptions.Probe134
  879.    : check_region(0x134, BusLogic_MultiMasterAddressCount) == 0))
  880. BusLogic_AppendProbeAddressISA(0x134);
  881.     }
  882.   /*
  883.     Iterate over the older non-compliant MultiMaster PCI Host Adapters,
  884.     noting the PCI bus location and assigned IRQ Channel.
  885.   */
  886.   PCI_Device = NULL;
  887.   while ((PCI_Device = pci_find_device(PCI_VENDOR_ID_BUSLOGIC,
  888.        PCI_DEVICE_ID_BUSLOGIC_MULTIMASTER_NC,
  889.        PCI_Device)) != NULL)
  890.     {
  891.       unsigned char Bus = PCI_Device->bus->number;
  892.       unsigned char Device = PCI_Device->devfn >> 3;
  893.       unsigned int IRQ_Channel = PCI_Device->irq;
  894.       BusLogic_IO_Address_T IO_Address = pci_resource_start(PCI_Device, 0);
  895.       if (pci_enable_device(PCI_Device))
  896. continue;
  897.       if (IO_Address == 0 || IRQ_Channel == 0) continue;
  898.       for (i = 0; i < BusLogic_ProbeInfoCount; i++)
  899. {
  900.   BusLogic_ProbeInfo_T *ProbeInfo = &BusLogic_ProbeInfoList[i];
  901.   if (ProbeInfo->IO_Address == IO_Address &&
  902.       ProbeInfo->HostAdapterType == BusLogic_MultiMaster)
  903.     {
  904.       ProbeInfo->HostAdapterBusType = BusLogic_PCI_Bus;
  905.       ProbeInfo->PCI_Address = 0;
  906.       ProbeInfo->Bus = Bus;
  907.       ProbeInfo->Device = Device;
  908.       ProbeInfo->IRQ_Channel = IRQ_Channel;
  909.       break;
  910.     }
  911. }
  912.     }
  913.   return PCIMultiMasterCount;
  914. }
  915. /*
  916.   BusLogic_InitializeFlashPointProbeInfo initializes the list of I/O Address
  917.   and Bus Probe Information to be checked for potential BusLogic FlashPoint
  918.   Host Adapters by interrogating the PCI Configuration Space.  It returns the
  919.   number of FlashPoint Host Adapters found.
  920. */
  921. static int BusLogic_InitializeFlashPointProbeInfo(BusLogic_HostAdapter_T
  922.   *PrototypeHostAdapter)
  923. {
  924.   int FlashPointIndex = BusLogic_ProbeInfoCount, FlashPointCount = 0;
  925.   PCI_Device_T *PCI_Device = NULL;
  926.   /*
  927.     Interrogate PCI Configuration Space for any FlashPoint Host Adapters.
  928.   */
  929.   while ((PCI_Device = pci_find_device(PCI_VENDOR_ID_BUSLOGIC,
  930.        PCI_DEVICE_ID_BUSLOGIC_FLASHPOINT,
  931.        PCI_Device)) != NULL)
  932.     {
  933.       unsigned char Bus = PCI_Device->bus->number;
  934.       unsigned char Device = PCI_Device->devfn >> 3;
  935.       unsigned int IRQ_Channel = PCI_Device->irq;
  936.       unsigned long BaseAddress0 = pci_resource_start(PCI_Device, 0);
  937.       unsigned long BaseAddress1 = pci_resource_start(PCI_Device, 1);
  938.       BusLogic_IO_Address_T IO_Address = BaseAddress0;
  939.       BusLogic_PCI_Address_T PCI_Address = BaseAddress1;
  940.       if (pci_enable_device(PCI_Device))
  941. continue;
  942. #ifndef CONFIG_SCSI_OMIT_FLASHPOINT
  943.       if (pci_resource_flags(PCI_Device, 0) & IORESOURCE_MEM)
  944. {
  945.   BusLogic_Error("BusLogic: Base Address0 0x%X not I/O for "
  946.  "FlashPoint Host Adaptern", NULL, BaseAddress0);
  947.   BusLogic_Error("at PCI Bus %d Device %d I/O Address 0x%Xn",
  948.  NULL, Bus, Device, IO_Address);
  949.   continue;
  950. }
  951.       if (pci_resource_flags(PCI_Device, 1) & IORESOURCE_IO)
  952. {
  953.   BusLogic_Error("BusLogic: Base Address1 0x%X not Memory for "
  954.  "FlashPoint Host Adaptern", NULL, BaseAddress1);
  955.   BusLogic_Error("at PCI Bus %d Device %d PCI Address 0x%Xn",
  956.  NULL, Bus, Device, PCI_Address);
  957.   continue;
  958. }
  959.       if (IRQ_Channel == 0)
  960. {
  961.   BusLogic_Error("BusLogic: IRQ Channel %d illegal for "
  962.  "FlashPoint Host Adaptern", NULL, IRQ_Channel);
  963.   BusLogic_Error("at PCI Bus %d Device %d I/O Address 0x%Xn",
  964.  NULL, Bus, Device, IO_Address);
  965.   continue;
  966. }
  967.       if (BusLogic_GlobalOptions.TraceProbe)
  968. {
  969.   BusLogic_Notice("BusLogic: FlashPoint Host Adapter "
  970.   "detected atn", NULL);
  971.   BusLogic_Notice("BusLogic: PCI Bus %d Device %d I/O Address "
  972.   "0x%X PCI Address 0x%Xn", NULL,
  973.   Bus, Device, IO_Address, PCI_Address);
  974. }
  975.       if (BusLogic_ProbeInfoCount < BusLogic_MaxHostAdapters)
  976. {
  977.   BusLogic_ProbeInfo_T *ProbeInfo =
  978.     &BusLogic_ProbeInfoList[BusLogic_ProbeInfoCount++];
  979.   ProbeInfo->HostAdapterType = BusLogic_FlashPoint;
  980.   ProbeInfo->HostAdapterBusType = BusLogic_PCI_Bus;
  981.   ProbeInfo->IO_Address = IO_Address;
  982.   ProbeInfo->PCI_Address = PCI_Address;
  983.   ProbeInfo->Bus = Bus;
  984.   ProbeInfo->Device = Device;
  985.   ProbeInfo->IRQ_Channel = IRQ_Channel;
  986.   FlashPointCount++;
  987. }
  988.       else BusLogic_Warning("BusLogic: Too many Host Adapters "
  989.     "detectedn", NULL);
  990. #else
  991.       BusLogic_Error("BusLogic: FlashPoint Host Adapter detected at "
  992.      "PCI Bus %d Device %dn", NULL, Bus, Device);
  993.       BusLogic_Error("BusLogic: I/O Address 0x%X PCI Address 0x%X, irq %d, "
  994.      "but FlashPointn", NULL, IO_Address, PCI_Address, IRQ_Channel);
  995.       BusLogic_Error("BusLogic: support was omitted in this kernel "
  996.      "configuration.n", NULL);
  997. #endif
  998.     }
  999.   /*
  1000.     The FlashPoint BIOS will scan for FlashPoint Host Adapters in the order of
  1001.     increasing PCI Bus and Device Number, so sort the probe information into
  1002.     the same order the BIOS uses.
  1003.   */
  1004.   BusLogic_SortProbeInfo(&BusLogic_ProbeInfoList[FlashPointIndex],
  1005.  FlashPointCount);
  1006.   return FlashPointCount;
  1007. }
  1008. /*
  1009.   BusLogic_InitializeProbeInfoList initializes the list of I/O Address and Bus
  1010.   Probe Information to be checked for potential BusLogic SCSI Host Adapters by
  1011.   interrogating the PCI Configuration Space on PCI machines as well as from the
  1012.   list of standard BusLogic MultiMaster ISA I/O Addresses.  By default, if both
  1013.   FlashPoint and PCI MultiMaster Host Adapters are present, this driver will
  1014.   probe for FlashPoint Host Adapters first unless the BIOS primary disk is
  1015.   controlled by the first PCI MultiMaster Host Adapter, in which case
  1016.   MultiMaster Host Adapters will be probed first.  The BusLogic Driver Options
  1017.   specifications "MultiMasterFirst" and "FlashPointFirst" can be used to force
  1018.   a particular probe order.
  1019. */
  1020. static void BusLogic_InitializeProbeInfoList(BusLogic_HostAdapter_T
  1021.      *PrototypeHostAdapter)
  1022. {
  1023.   /*
  1024.     If a PCI BIOS is present, interrogate it for MultiMaster and FlashPoint
  1025.     Host Adapters; otherwise, default to the standard ISA MultiMaster probe.
  1026.   */
  1027.   if (!BusLogic_ProbeOptions.NoProbePCI && pci_present())
  1028.     {
  1029.       if (BusLogic_ProbeOptions.MultiMasterFirst)
  1030. {
  1031.   BusLogic_InitializeMultiMasterProbeInfo(PrototypeHostAdapter);
  1032.   BusLogic_InitializeFlashPointProbeInfo(PrototypeHostAdapter);
  1033. }
  1034.       else if (BusLogic_ProbeOptions.FlashPointFirst)
  1035. {
  1036.   BusLogic_InitializeFlashPointProbeInfo(PrototypeHostAdapter);
  1037.   BusLogic_InitializeMultiMasterProbeInfo(PrototypeHostAdapter);
  1038. }
  1039.       else
  1040. {
  1041.   int FlashPointCount =
  1042.     BusLogic_InitializeFlashPointProbeInfo(PrototypeHostAdapter);
  1043.   int PCIMultiMasterCount =
  1044.     BusLogic_InitializeMultiMasterProbeInfo(PrototypeHostAdapter);
  1045.   if (FlashPointCount > 0 && PCIMultiMasterCount > 0)
  1046.     {
  1047.       BusLogic_ProbeInfo_T *ProbeInfo =
  1048. &BusLogic_ProbeInfoList[FlashPointCount];
  1049.       BusLogic_HostAdapter_T *HostAdapter = PrototypeHostAdapter;
  1050.       BusLogic_FetchHostAdapterLocalRAMRequest_T
  1051. FetchHostAdapterLocalRAMRequest;
  1052.       BusLogic_BIOSDriveMapByte_T Drive0MapByte;
  1053.       while (ProbeInfo->HostAdapterBusType != BusLogic_PCI_Bus)
  1054. ProbeInfo++;
  1055.       HostAdapter->IO_Address = ProbeInfo->IO_Address;
  1056.       FetchHostAdapterLocalRAMRequest.ByteOffset =
  1057. BusLogic_BIOS_BaseOffset + BusLogic_BIOS_DriveMapOffset + 0;
  1058.       FetchHostAdapterLocalRAMRequest.ByteCount =
  1059. sizeof(Drive0MapByte);
  1060.       BusLogic_Command(HostAdapter,
  1061.        BusLogic_FetchHostAdapterLocalRAM,
  1062.        &FetchHostAdapterLocalRAMRequest,
  1063.        sizeof(FetchHostAdapterLocalRAMRequest),
  1064.        &Drive0MapByte, sizeof(Drive0MapByte));
  1065.       /*
  1066. If the Map Byte for BIOS Drive 0 indicates that BIOS Drive 0
  1067. is controlled by this PCI MultiMaster Host Adapter, then
  1068. reverse the probe order so that MultiMaster Host Adapters are
  1069. probed before FlashPoint Host Adapters.
  1070.       */
  1071.       if (Drive0MapByte.DiskGeometry !=
  1072.   BusLogic_BIOS_Disk_Not_Installed)
  1073. {
  1074.   BusLogic_ProbeInfo_T
  1075.     SavedProbeInfo[BusLogic_MaxHostAdapters];
  1076.   int MultiMasterCount =
  1077.     BusLogic_ProbeInfoCount - FlashPointCount;
  1078.   memcpy(SavedProbeInfo,
  1079.  BusLogic_ProbeInfoList,
  1080.  BusLogic_ProbeInfoCount
  1081.  * sizeof(BusLogic_ProbeInfo_T));
  1082.   memcpy(&BusLogic_ProbeInfoList[0],
  1083.  &SavedProbeInfo[FlashPointCount],
  1084.  MultiMasterCount * sizeof(BusLogic_ProbeInfo_T));
  1085.   memcpy(&BusLogic_ProbeInfoList[MultiMasterCount],
  1086.  &SavedProbeInfo[0],
  1087.  FlashPointCount * sizeof(BusLogic_ProbeInfo_T));
  1088. }
  1089.     }
  1090. }
  1091.     }
  1092.   else BusLogic_InitializeProbeInfoListISA(PrototypeHostAdapter);
  1093. }
  1094. #endif  /* CONFIG_PCI */
  1095. /*
  1096.   BusLogic_Failure prints a standardized error message, and then returns false.
  1097. */
  1098. static boolean BusLogic_Failure(BusLogic_HostAdapter_T *HostAdapter,
  1099. char *ErrorMessage)
  1100. {
  1101.   BusLogic_AnnounceDriver(HostAdapter);
  1102.   if (HostAdapter->HostAdapterBusType == BusLogic_PCI_Bus)
  1103.     {
  1104.       BusLogic_Error("While configuring BusLogic PCI Host Adapter atn",
  1105.      HostAdapter);
  1106.       BusLogic_Error("Bus %d Device %d I/O Address 0x%X PCI Address 0x%X:n",
  1107.      HostAdapter, HostAdapter->Bus, HostAdapter->Device,
  1108.      HostAdapter->IO_Address, HostAdapter->PCI_Address);
  1109.     }
  1110.   else BusLogic_Error("While configuring BusLogic Host Adapter at "
  1111.       "I/O Address 0x%X:n", HostAdapter,
  1112.       HostAdapter->IO_Address);
  1113.   BusLogic_Error("%s FAILED - DETACHINGn", HostAdapter, ErrorMessage);
  1114.   if (BusLogic_CommandFailureReason != NULL)
  1115.     BusLogic_Error("ADDITIONAL FAILURE INFO - %sn", HostAdapter,
  1116.    BusLogic_CommandFailureReason);
  1117.   return false;
  1118. }
  1119. /*
  1120.   BusLogic_ProbeHostAdapter probes for a BusLogic Host Adapter.
  1121. */
  1122. static boolean BusLogic_ProbeHostAdapter(BusLogic_HostAdapter_T *HostAdapter)
  1123. {
  1124.   BusLogic_StatusRegister_T StatusRegister;
  1125.   BusLogic_InterruptRegister_T InterruptRegister;
  1126.   BusLogic_GeometryRegister_T GeometryRegister;
  1127.   /*
  1128.     FlashPoint Host Adapters are Probed by the FlashPoint SCCB Manager.
  1129.   */
  1130.   if (BusLogic_FlashPointHostAdapterP(HostAdapter))
  1131.     {
  1132.       FlashPoint_Info_T *FlashPointInfo = &HostAdapter->FlashPointInfo;
  1133.       FlashPointInfo->BaseAddress =
  1134. (BusLogic_Base_Address_T) HostAdapter->IO_Address;
  1135.       FlashPointInfo->IRQ_Channel = HostAdapter->IRQ_Channel;
  1136.       FlashPointInfo->Present = false;
  1137.       if (!(FlashPoint_ProbeHostAdapter(FlashPointInfo) == 0 &&
  1138.     FlashPointInfo->Present))
  1139. {
  1140.   BusLogic_Error("BusLogic: FlashPoint Host Adapter detected at "
  1141.  "PCI Bus %d Device %dn", HostAdapter,
  1142.  HostAdapter->Bus, HostAdapter->Device);
  1143.   BusLogic_Error("BusLogic: I/O Address 0x%X PCI Address 0x%X, "
  1144.  "but FlashPointn", HostAdapter,
  1145.  HostAdapter->IO_Address, HostAdapter->PCI_Address);
  1146.   BusLogic_Error("BusLogic: Probe Function failed to validate it.n",
  1147.  HostAdapter);
  1148.   return false;
  1149. }
  1150.       if (BusLogic_GlobalOptions.TraceProbe)
  1151. BusLogic_Notice("BusLogic_Probe(0x%X): FlashPoint Foundn",
  1152. HostAdapter, HostAdapter->IO_Address);
  1153.       /*
  1154. Indicate the Host Adapter Probe completed successfully.
  1155.       */
  1156.       return true;
  1157.     }
  1158.   /*
  1159.     Read the Status, Interrupt, and Geometry Registers to test if there are I/O
  1160.     ports that respond, and to check the values to determine if they are from a
  1161.     BusLogic Host Adapter.  A nonexistent I/O port will return 0xFF, in which
  1162.     case there is definitely no BusLogic Host Adapter at this base I/O Address.
  1163.     The test here is a subset of that used by the BusLogic Host Adapter BIOS.
  1164.   */
  1165.   StatusRegister.All = BusLogic_ReadStatusRegister(HostAdapter);
  1166.   InterruptRegister.All = BusLogic_ReadInterruptRegister(HostAdapter);
  1167.   GeometryRegister.All = BusLogic_ReadGeometryRegister(HostAdapter);
  1168.   if (BusLogic_GlobalOptions.TraceProbe)
  1169.     BusLogic_Notice("BusLogic_Probe(0x%X): Status 0x%02X, Interrupt 0x%02X, "
  1170.     "Geometry 0x%02Xn", HostAdapter,
  1171.     HostAdapter->IO_Address, StatusRegister.All,
  1172.     InterruptRegister.All, GeometryRegister.All);
  1173.   if (StatusRegister.All == 0 ||
  1174.       StatusRegister.Bits.DiagnosticActive ||
  1175.       StatusRegister.Bits.CommandParameterRegisterBusy ||
  1176.       StatusRegister.Bits.Reserved ||
  1177.       StatusRegister.Bits.CommandInvalid ||
  1178.       InterruptRegister.Bits.Reserved != 0)
  1179.     return false;
  1180.   /*
  1181.     Check the undocumented Geometry Register to test if there is an I/O port
  1182.     that responded.  Adaptec Host Adapters do not implement the Geometry
  1183.     Register, so this test helps serve to avoid incorrectly recognizing an
  1184.     Adaptec 1542A or 1542B as a BusLogic.  Unfortunately, the Adaptec 1542C
  1185.     series does respond to the Geometry Register I/O port, but it will be
  1186.     rejected later when the Inquire Extended Setup Information command is
  1187.     issued in BusLogic_CheckHostAdapter.  The AMI FastDisk Host Adapter is a
  1188.     BusLogic clone that implements the same interface as earlier BusLogic
  1189.     Host Adapters, including the undocumented commands, and is therefore
  1190.     supported by this driver.  However, the AMI FastDisk always returns 0x00
  1191.     upon reading the Geometry Register, so the extended translation option
  1192.     should always be left disabled on the AMI FastDisk.
  1193.   */
  1194.   if (GeometryRegister.All == 0xFF) return false;
  1195.   /*
  1196.     Indicate the Host Adapter Probe completed successfully.
  1197.   */
  1198.   return true;
  1199. }
  1200. /*
  1201.   BusLogic_HardwareResetHostAdapter issues a Hardware Reset to the Host Adapter
  1202.   and waits for Host Adapter Diagnostics to complete.  If HardReset is true, a
  1203.   Hard Reset is performed which also initiates a SCSI Bus Reset.  Otherwise, a
  1204.   Soft Reset is performed which only resets the Host Adapter without forcing a
  1205.   SCSI Bus Reset.
  1206. */
  1207. static boolean BusLogic_HardwareResetHostAdapter(BusLogic_HostAdapter_T
  1208.    *HostAdapter,
  1209.  boolean HardReset)
  1210. {
  1211.   BusLogic_StatusRegister_T StatusRegister;
  1212.   int TimeoutCounter;
  1213.   /*
  1214.     FlashPoint Host Adapters are Hard Reset by the FlashPoint SCCB Manager.
  1215.   */
  1216.   if (BusLogic_FlashPointHostAdapterP(HostAdapter))
  1217.     {
  1218.       FlashPoint_Info_T *FlashPointInfo = &HostAdapter->FlashPointInfo;
  1219.       FlashPointInfo->HostSoftReset = !HardReset;
  1220.       FlashPointInfo->ReportDataUnderrun = true;
  1221.       HostAdapter->CardHandle =
  1222. FlashPoint_HardwareResetHostAdapter(FlashPointInfo);
  1223.       if (HostAdapter->CardHandle == FlashPoint_BadCardHandle) return false;
  1224.       /*
  1225. Indicate the Host Adapter Hard Reset completed successfully.
  1226.       */
  1227.       return true;
  1228.     }
  1229.   /*
  1230.     Issue a Hard Reset or Soft Reset Command to the Host Adapter.  The Host
  1231.     Adapter should respond by setting Diagnostic Active in the Status Register.
  1232.   */
  1233.   if (HardReset)
  1234.     BusLogic_HardReset(HostAdapter);
  1235.   else BusLogic_SoftReset(HostAdapter);
  1236.   /*
  1237.     Wait until Diagnostic Active is set in the Status Register.
  1238.   */
  1239.   TimeoutCounter = 5*10000;
  1240.   while (--TimeoutCounter >= 0)
  1241.     {
  1242.       StatusRegister.All = BusLogic_ReadStatusRegister(HostAdapter);
  1243.       if (StatusRegister.Bits.DiagnosticActive) break;
  1244.       udelay(100);
  1245.     }
  1246.   if (BusLogic_GlobalOptions.TraceHardwareReset)
  1247.     BusLogic_Notice("BusLogic_HardwareReset(0x%X): Diagnostic Active, "
  1248.     "Status 0x%02Xn", HostAdapter,
  1249.     HostAdapter->IO_Address, StatusRegister.All);
  1250.   if (TimeoutCounter < 0) return false;
  1251.   /*
  1252.     Wait 100 microseconds to allow completion of any initial diagnostic
  1253.     activity which might leave the contents of the Status Register
  1254.     unpredictable.
  1255.   */
  1256.   udelay(100);
  1257.   /*
  1258.     Wait until Diagnostic Active is reset in the Status Register.
  1259.   */
  1260.   TimeoutCounter = 10*10000;
  1261.   while (--TimeoutCounter >= 0)
  1262.     {
  1263.       StatusRegister.All = BusLogic_ReadStatusRegister(HostAdapter);
  1264.       if (!StatusRegister.Bits.DiagnosticActive) break;
  1265.       udelay(100);
  1266.     }
  1267.   if (BusLogic_GlobalOptions.TraceHardwareReset)
  1268.     BusLogic_Notice("BusLogic_HardwareReset(0x%X): Diagnostic Completed, "
  1269.     "Status 0x%02Xn", HostAdapter,
  1270.     HostAdapter->IO_Address, StatusRegister.All);
  1271.   if (TimeoutCounter < 0) return false;
  1272.   /*
  1273.     Wait until at least one of the Diagnostic Failure, Host Adapter Ready,
  1274.     or Data In Register Ready bits is set in the Status Register.
  1275.   */
  1276.   TimeoutCounter = 10000;
  1277.   while (--TimeoutCounter >= 0)
  1278.     {
  1279.       StatusRegister.All = BusLogic_ReadStatusRegister(HostAdapter);
  1280.       if (StatusRegister.Bits.DiagnosticFailure ||
  1281.   StatusRegister.Bits.HostAdapterReady ||
  1282.   StatusRegister.Bits.DataInRegisterReady)
  1283. break;
  1284.       udelay(100);
  1285.     }
  1286.   if (BusLogic_GlobalOptions.TraceHardwareReset)
  1287.     BusLogic_Notice("BusLogic_HardwareReset(0x%X): Host Adapter Ready, "
  1288.     "Status 0x%02Xn", HostAdapter,
  1289.     HostAdapter->IO_Address, StatusRegister.All);
  1290.   if (TimeoutCounter < 0) return false;
  1291.   /*
  1292.     If Diagnostic Failure is set or Host Adapter Ready is reset, then an
  1293.     error occurred during the Host Adapter diagnostics.  If Data In Register
  1294.     Ready is set, then there is an Error Code available.
  1295.   */
  1296.   if (StatusRegister.Bits.DiagnosticFailure ||
  1297.       !StatusRegister.Bits.HostAdapterReady)
  1298.     {
  1299.       BusLogic_CommandFailureReason = NULL;
  1300.       BusLogic_Failure(HostAdapter, "HARD RESET DIAGNOSTICS");
  1301.       BusLogic_Error("HOST ADAPTER STATUS REGISTER = %02Xn",
  1302.      HostAdapter, StatusRegister.All);
  1303.       if (StatusRegister.Bits.DataInRegisterReady)
  1304. {
  1305.   unsigned char ErrorCode = BusLogic_ReadDataInRegister(HostAdapter);
  1306.   BusLogic_Error("HOST ADAPTER ERROR CODE = %dn",
  1307.  HostAdapter, ErrorCode);
  1308. }
  1309.       return false;
  1310.     }
  1311.   /*
  1312.     Indicate the Host Adapter Hard Reset completed successfully.
  1313.   */
  1314.   return true;
  1315. }
  1316. /*
  1317.   BusLogic_CheckHostAdapter checks to be sure this really is a BusLogic
  1318.   Host Adapter.
  1319. */
  1320. static boolean BusLogic_CheckHostAdapter(BusLogic_HostAdapter_T *HostAdapter)
  1321. {
  1322.   BusLogic_ExtendedSetupInformation_T ExtendedSetupInformation;
  1323.   BusLogic_RequestedReplyLength_T RequestedReplyLength;
  1324.   boolean Result = true;
  1325.   /*
  1326.     FlashPoint Host Adapters do not require this protection.
  1327.   */
  1328.   if (BusLogic_FlashPointHostAdapterP(HostAdapter)) return true;
  1329.   /*
  1330.     Issue the Inquire Extended Setup Information command.  Only genuine
  1331.     BusLogic Host Adapters and true clones support this command.  Adaptec 1542C
  1332.     series Host Adapters that respond to the Geometry Register I/O port will
  1333.     fail this command.
  1334.   */
  1335.   RequestedReplyLength = sizeof(ExtendedSetupInformation);
  1336.   if (BusLogic_Command(HostAdapter,
  1337.        BusLogic_InquireExtendedSetupInformation,
  1338.        &RequestedReplyLength,
  1339.        sizeof(RequestedReplyLength),
  1340.        &ExtendedSetupInformation,
  1341.        sizeof(ExtendedSetupInformation))
  1342.       != sizeof(ExtendedSetupInformation))
  1343.     Result = false;
  1344.   /*
  1345.     Provide tracing information if requested and return.
  1346.   */
  1347.   if (BusLogic_GlobalOptions.TraceProbe)
  1348.     BusLogic_Notice("BusLogic_Check(0x%X): MultiMaster %sn", HostAdapter,
  1349.     HostAdapter->IO_Address, (Result ? "Found" : "Not Found"));
  1350.   return Result;
  1351. }
  1352. /*
  1353.   BusLogic_ReadHostAdapterConfiguration reads the Configuration Information
  1354.   from Host Adapter and initializes the Host Adapter structure.
  1355. */
  1356. static boolean BusLogic_ReadHostAdapterConfiguration(BusLogic_HostAdapter_T
  1357.      *HostAdapter)
  1358. {
  1359.   BusLogic_BoardID_T BoardID;
  1360.   BusLogic_Configuration_T Configuration;
  1361.   BusLogic_SetupInformation_T SetupInformation;
  1362.   BusLogic_ExtendedSetupInformation_T ExtendedSetupInformation;
  1363.   BusLogic_HostAdapterModelNumber_T HostAdapterModelNumber;
  1364.   BusLogic_FirmwareVersion3rdDigit_T FirmwareVersion3rdDigit;
  1365.   BusLogic_FirmwareVersionLetter_T FirmwareVersionLetter;
  1366.   BusLogic_PCIHostAdapterInformation_T PCIHostAdapterInformation;
  1367.   BusLogic_FetchHostAdapterLocalRAMRequest_T FetchHostAdapterLocalRAMRequest;
  1368.   BusLogic_AutoSCSIData_T AutoSCSIData;
  1369.   BusLogic_GeometryRegister_T GeometryRegister;
  1370.   BusLogic_RequestedReplyLength_T RequestedReplyLength;
  1371.   unsigned char *TargetPointer, Character;
  1372.   int TargetID, i;
  1373.   /*
  1374.     Configuration Information for FlashPoint Host Adapters is provided in the
  1375.     FlashPoint_Info structure by the FlashPoint SCCB Manager's Probe Function.
  1376.     Initialize fields in the Host Adapter structure from the FlashPoint_Info
  1377.     structure.
  1378.   */
  1379.   if (BusLogic_FlashPointHostAdapterP(HostAdapter))
  1380.     {
  1381.       FlashPoint_Info_T *FlashPointInfo = &HostAdapter->FlashPointInfo;
  1382.       TargetPointer = HostAdapter->ModelName;
  1383.       *TargetPointer++ = 'B';
  1384.       *TargetPointer++ = 'T';
  1385.       *TargetPointer++ = '-';
  1386.       for (i = 0; i < sizeof(FlashPointInfo->ModelNumber); i++)
  1387. *TargetPointer++ = FlashPointInfo->ModelNumber[i];
  1388.       *TargetPointer++ = '';
  1389.       strcpy(HostAdapter->FirmwareVersion, FlashPoint_FirmwareVersion);
  1390.       HostAdapter->SCSI_ID = FlashPointInfo->SCSI_ID;
  1391.       HostAdapter->ExtendedTranslationEnabled =
  1392. FlashPointInfo->ExtendedTranslationEnabled;
  1393.       HostAdapter->ParityCheckingEnabled =
  1394. FlashPointInfo->ParityCheckingEnabled;
  1395.       HostAdapter->BusResetEnabled = !FlashPointInfo->HostSoftReset;
  1396.       HostAdapter->LevelSensitiveInterrupt = true;
  1397.       HostAdapter->HostWideSCSI = FlashPointInfo->HostWideSCSI;
  1398.       HostAdapter->HostDifferentialSCSI = false;
  1399.       HostAdapter->HostSupportsSCAM = true;
  1400.       HostAdapter->HostUltraSCSI = true;
  1401.       HostAdapter->ExtendedLUNSupport = true;
  1402.       HostAdapter->TerminationInfoValid = true;
  1403.       HostAdapter->LowByteTerminated = FlashPointInfo->LowByteTerminated;
  1404.       HostAdapter->HighByteTerminated = FlashPointInfo->HighByteTerminated;
  1405.       HostAdapter->SCAM_Enabled = FlashPointInfo->SCAM_Enabled;
  1406.       HostAdapter->SCAM_Level2 = FlashPointInfo->SCAM_Level2;
  1407.       HostAdapter->DriverScatterGatherLimit = BusLogic_ScatterGatherLimit;
  1408.       HostAdapter->MaxTargetDevices = (HostAdapter->HostWideSCSI ? 16 : 8);
  1409.       HostAdapter->MaxLogicalUnits = 32;
  1410.       HostAdapter->InitialCCBs = 4 * BusLogic_CCB_AllocationGroupSize;
  1411.       HostAdapter->IncrementalCCBs = BusLogic_CCB_AllocationGroupSize;
  1412.       HostAdapter->DriverQueueDepth = 255;
  1413.       HostAdapter->HostAdapterQueueDepth = HostAdapter->DriverQueueDepth;
  1414.       HostAdapter->SynchronousPermitted = FlashPointInfo->SynchronousPermitted;
  1415.       HostAdapter->FastPermitted = FlashPointInfo->FastPermitted;
  1416.       HostAdapter->UltraPermitted = FlashPointInfo->UltraPermitted;
  1417.       HostAdapter->WidePermitted = FlashPointInfo->WidePermitted;
  1418.       HostAdapter->DisconnectPermitted = FlashPointInfo->DisconnectPermitted;
  1419.       HostAdapter->TaggedQueuingPermitted = 0xFFFF;
  1420.       goto Common;
  1421.     }
  1422.   /*
  1423.     Issue the Inquire Board ID command.
  1424.   */
  1425.   if (BusLogic_Command(HostAdapter, BusLogic_InquireBoardID, NULL, 0,
  1426.        &BoardID, sizeof(BoardID)) != sizeof(BoardID))
  1427.     return BusLogic_Failure(HostAdapter, "INQUIRE BOARD ID");
  1428.   /*
  1429.     Issue the Inquire Configuration command.
  1430.   */
  1431.   if (BusLogic_Command(HostAdapter, BusLogic_InquireConfiguration, NULL, 0,
  1432.        &Configuration, sizeof(Configuration))
  1433.       != sizeof(Configuration))
  1434.     return BusLogic_Failure(HostAdapter, "INQUIRE CONFIGURATION");
  1435.   /*
  1436.     Issue the Inquire Setup Information command.
  1437.   */
  1438.   RequestedReplyLength = sizeof(SetupInformation);
  1439.   if (BusLogic_Command(HostAdapter, BusLogic_InquireSetupInformation,
  1440.        &RequestedReplyLength, sizeof(RequestedReplyLength),
  1441.        &SetupInformation, sizeof(SetupInformation))
  1442.       != sizeof(SetupInformation))
  1443.     return BusLogic_Failure(HostAdapter, "INQUIRE SETUP INFORMATION");
  1444.   /*
  1445.     Issue the Inquire Extended Setup Information command.
  1446.   */
  1447.   RequestedReplyLength = sizeof(ExtendedSetupInformation);
  1448.   if (BusLogic_Command(HostAdapter, BusLogic_InquireExtendedSetupInformation,
  1449.        &RequestedReplyLength, sizeof(RequestedReplyLength),
  1450.        &ExtendedSetupInformation,
  1451.        sizeof(ExtendedSetupInformation))
  1452.       != sizeof(ExtendedSetupInformation))
  1453.     return BusLogic_Failure(HostAdapter, "INQUIRE EXTENDED SETUP INFORMATION");
  1454.   /*
  1455.     Issue the Inquire Firmware Version 3rd Digit command.
  1456.   */
  1457.   FirmwareVersion3rdDigit = '';
  1458.   if (BoardID.FirmwareVersion1stDigit > '0')
  1459.     if (BusLogic_Command(HostAdapter, BusLogic_InquireFirmwareVersion3rdDigit,
  1460.  NULL, 0, &FirmwareVersion3rdDigit,
  1461.  sizeof(FirmwareVersion3rdDigit))
  1462. != sizeof(FirmwareVersion3rdDigit))
  1463.       return BusLogic_Failure(HostAdapter, "INQUIRE FIRMWARE 3RD DIGIT");
  1464.   /*
  1465.     Issue the Inquire Host Adapter Model Number command.
  1466.   */
  1467.   if (ExtendedSetupInformation.BusType == 'A' &&
  1468.       BoardID.FirmwareVersion1stDigit == '2')
  1469.     /* BusLogic BT-542B ISA 2.xx */
  1470.     strcpy(HostAdapterModelNumber, "542B");
  1471.   else if (ExtendedSetupInformation.BusType == 'E' &&
  1472.    BoardID.FirmwareVersion1stDigit == '2' &&
  1473.    (BoardID.FirmwareVersion2ndDigit <= '1' ||
  1474.     (BoardID.FirmwareVersion2ndDigit == '2' &&
  1475.      FirmwareVersion3rdDigit == '0')))
  1476.     /* BusLogic BT-742A EISA 2.1x or 2.20 */
  1477.     strcpy(HostAdapterModelNumber, "742A");
  1478.   else if (ExtendedSetupInformation.BusType == 'E' &&
  1479.    BoardID.FirmwareVersion1stDigit == '0')
  1480.     /* AMI FastDisk EISA Series 441 0.x */
  1481.     strcpy(HostAdapterModelNumber, "747A");
  1482.   else
  1483.     {
  1484.       RequestedReplyLength = sizeof(HostAdapterModelNumber);
  1485.       if (BusLogic_Command(HostAdapter, BusLogic_InquireHostAdapterModelNumber,
  1486.    &RequestedReplyLength, sizeof(RequestedReplyLength),
  1487.    &HostAdapterModelNumber,
  1488.    sizeof(HostAdapterModelNumber))
  1489.   != sizeof(HostAdapterModelNumber))
  1490. return BusLogic_Failure(HostAdapter,
  1491. "INQUIRE HOST ADAPTER MODEL NUMBER");
  1492.     }
  1493.   /*
  1494.     BusLogic MultiMaster Host Adapters can be identified by their model number
  1495.     and the major version number of their firmware as follows:
  1496.     5.xx BusLogic "W" Series Host Adapters:
  1497.   BT-948/958/958D
  1498.     4.xx BusLogic "C" Series Host Adapters:
  1499.   BT-946C/956C/956CD/747C/757C/757CD/445C/545C/540CF
  1500.     3.xx BusLogic "S" Series Host Adapters:
  1501.   BT-747S/747D/757S/757D/445S/545S/542D
  1502.   BT-542B/742A (revision H)
  1503.     2.xx BusLogic "A" Series Host Adapters:
  1504.   BT-542B/742A (revision G and below)
  1505.     0.xx AMI FastDisk VLB/EISA BusLogic Clone Host Adapter
  1506.   */
  1507.   /*
  1508.     Save the Model Name and Host Adapter Name in the Host Adapter structure.
  1509.   */
  1510.   TargetPointer = HostAdapter->ModelName;
  1511.   *TargetPointer++ = 'B';
  1512.   *TargetPointer++ = 'T';
  1513.   *TargetPointer++ = '-';
  1514.   for (i = 0; i < sizeof(HostAdapterModelNumber); i++)
  1515.     {
  1516.       Character = HostAdapterModelNumber[i];
  1517.       if (Character == ' ' || Character == '') break;
  1518.       *TargetPointer++ = Character;
  1519.     }
  1520.   *TargetPointer++ = '';
  1521.   /*
  1522.     Save the Firmware Version in the Host Adapter structure.
  1523.   */
  1524.   TargetPointer = HostAdapter->FirmwareVersion;
  1525.   *TargetPointer++ = BoardID.FirmwareVersion1stDigit;
  1526.   *TargetPointer++ = '.';
  1527.   *TargetPointer++ = BoardID.FirmwareVersion2ndDigit;
  1528.   if (FirmwareVersion3rdDigit != ' ' && FirmwareVersion3rdDigit != '')
  1529.     *TargetPointer++ = FirmwareVersion3rdDigit;
  1530.   *TargetPointer = '';
  1531.   /*
  1532.     Issue the Inquire Firmware Version Letter command.
  1533.   */
  1534.   if (strcmp(HostAdapter->FirmwareVersion, "3.3") >= 0)
  1535.     {
  1536.       if (BusLogic_Command(HostAdapter, BusLogic_InquireFirmwareVersionLetter,
  1537.    NULL, 0, &FirmwareVersionLetter,
  1538.    sizeof(FirmwareVersionLetter))
  1539.   != sizeof(FirmwareVersionLetter))
  1540. return BusLogic_Failure(HostAdapter,
  1541. "INQUIRE FIRMWARE VERSION LETTER");
  1542.       if (FirmwareVersionLetter != ' ' && FirmwareVersionLetter != '')
  1543. *TargetPointer++ = FirmwareVersionLetter;
  1544.       *TargetPointer = '';
  1545.     }
  1546.   /*
  1547.     Save the Host Adapter SCSI ID in the Host Adapter structure.
  1548.   */
  1549.   HostAdapter->SCSI_ID = Configuration.HostAdapterID;
  1550.   /*
  1551.     Determine the Bus Type and save it in the Host Adapter structure, determine
  1552.     and save the IRQ Channel if necessary, and determine and save the DMA
  1553.     Channel for ISA Host Adapters.
  1554.   */
  1555.   HostAdapter->HostAdapterBusType =
  1556.     BusLogic_HostAdapterBusTypes[HostAdapter->ModelName[3] - '4'];
  1557.   if (HostAdapter->IRQ_Channel == 0)
  1558.     {
  1559.       if (Configuration.IRQ_Channel9)
  1560. HostAdapter->IRQ_Channel = 9;
  1561.       else if (Configuration.IRQ_Channel10)
  1562. HostAdapter->IRQ_Channel = 10;
  1563.       else if (Configuration.IRQ_Channel11)
  1564. HostAdapter->IRQ_Channel = 11;
  1565.       else if (Configuration.IRQ_Channel12)
  1566. HostAdapter->IRQ_Channel = 12;
  1567.       else if (Configuration.IRQ_Channel14)
  1568. HostAdapter->IRQ_Channel = 14;
  1569.       else if (Configuration.IRQ_Channel15)
  1570. HostAdapter->IRQ_Channel = 15;
  1571.     }
  1572.   if (HostAdapter->HostAdapterBusType == BusLogic_ISA_Bus)
  1573.     {
  1574.       if (Configuration.DMA_Channel5)
  1575. HostAdapter->DMA_Channel = 5;
  1576.       else if (Configuration.DMA_Channel6)
  1577. HostAdapter->DMA_Channel = 6;
  1578.       else if (Configuration.DMA_Channel7)
  1579. HostAdapter->DMA_Channel = 7;
  1580.     }
  1581.   /*
  1582.     Determine whether Extended Translation is enabled and save it in
  1583.     the Host Adapter structure.
  1584.   */
  1585.   GeometryRegister.All = BusLogic_ReadGeometryRegister(HostAdapter);
  1586.   HostAdapter->ExtendedTranslationEnabled =
  1587.     GeometryRegister.Bits.ExtendedTranslationEnabled;
  1588.   /*
  1589.     Save the Scatter Gather Limits, Level Sensitive Interrupt flag, Wide
  1590.     SCSI flag, Differential SCSI flag, SCAM Supported flag, and
  1591.     Ultra SCSI flag in the Host Adapter structure.
  1592.   */
  1593.   HostAdapter->HostAdapterScatterGatherLimit =
  1594.     ExtendedSetupInformation.ScatterGatherLimit;
  1595.   HostAdapter->DriverScatterGatherLimit =
  1596.     HostAdapter->HostAdapterScatterGatherLimit;
  1597.   if (HostAdapter->HostAdapterScatterGatherLimit > BusLogic_ScatterGatherLimit)
  1598.     HostAdapter->DriverScatterGatherLimit = BusLogic_ScatterGatherLimit;
  1599.   if (ExtendedSetupInformation.Misc.LevelSensitiveInterrupt)
  1600.     HostAdapter->LevelSensitiveInterrupt = true;
  1601.   HostAdapter->HostWideSCSI = ExtendedSetupInformation.HostWideSCSI;
  1602.   HostAdapter->HostDifferentialSCSI =
  1603.     ExtendedSetupInformation.HostDifferentialSCSI;
  1604.   HostAdapter->HostSupportsSCAM = ExtendedSetupInformation.HostSupportsSCAM;
  1605.   HostAdapter->HostUltraSCSI = ExtendedSetupInformation.HostUltraSCSI;
  1606.   /*
  1607.     Determine whether Extended LUN Format CCBs are supported and save the
  1608.     information in the Host Adapter structure.
  1609.   */
  1610.   if (HostAdapter->FirmwareVersion[0] == '5' ||
  1611.       (HostAdapter->FirmwareVersion[0] == '4' && HostAdapter->HostWideSCSI))
  1612.     HostAdapter->ExtendedLUNSupport = true;
  1613.   /*
  1614.     Issue the Inquire PCI Host Adapter Information command to read the
  1615.     Termination Information from "W" series MultiMaster Host Adapters.
  1616.   */
  1617.   if (HostAdapter->FirmwareVersion[0] == '5')
  1618.     {
  1619.       if (BusLogic_Command(HostAdapter,
  1620.    BusLogic_InquirePCIHostAdapterInformation,
  1621.    NULL, 0, &PCIHostAdapterInformation,
  1622.    sizeof(PCIHostAdapterInformation))
  1623.   != sizeof(PCIHostAdapterInformation))
  1624. return BusLogic_Failure(HostAdapter,
  1625. "INQUIRE PCI HOST ADAPTER INFORMATION");
  1626.       /*
  1627. Save the Termination Information in the Host Adapter structure.
  1628.       */
  1629.       if (PCIHostAdapterInformation.GenericInfoValid)
  1630. {
  1631.   HostAdapter->TerminationInfoValid = true;
  1632.   HostAdapter->LowByteTerminated =
  1633.     PCIHostAdapterInformation.LowByteTerminated;
  1634.   HostAdapter->HighByteTerminated =
  1635.     PCIHostAdapterInformation.HighByteTerminated;
  1636. }
  1637.     }
  1638.   /*
  1639.     Issue the Fetch Host Adapter Local RAM command to read the AutoSCSI data
  1640.     from "W" and "C" series MultiMaster Host Adapters.
  1641.   */
  1642.   if (HostAdapter->FirmwareVersion[0] >= '4')
  1643.     {
  1644.       FetchHostAdapterLocalRAMRequest.ByteOffset =
  1645. BusLogic_AutoSCSI_BaseOffset;
  1646.       FetchHostAdapterLocalRAMRequest.ByteCount = sizeof(AutoSCSIData);
  1647.       if (BusLogic_Command(HostAdapter,
  1648.    BusLogic_FetchHostAdapterLocalRAM,
  1649.    &FetchHostAdapterLocalRAMRequest,
  1650.    sizeof(FetchHostAdapterLocalRAMRequest),
  1651.    &AutoSCSIData, sizeof(AutoSCSIData))
  1652.   != sizeof(AutoSCSIData))
  1653. return BusLogic_Failure(HostAdapter, "FETCH HOST ADAPTER LOCAL RAM");
  1654.       /*
  1655. Save the Parity Checking Enabled, Bus Reset Enabled, and Termination
  1656. Information in the Host Adapter structure.
  1657.       */
  1658.       HostAdapter->ParityCheckingEnabled = AutoSCSIData.ParityCheckingEnabled;
  1659.       HostAdapter->BusResetEnabled = AutoSCSIData.BusResetEnabled;
  1660.       if (HostAdapter->FirmwareVersion[0] == '4')
  1661. {
  1662.   HostAdapter->TerminationInfoValid = true;
  1663.   HostAdapter->LowByteTerminated = AutoSCSIData.LowByteTerminated;
  1664.   HostAdapter->HighByteTerminated = AutoSCSIData.HighByteTerminated;
  1665. }
  1666.       /*
  1667. Save the Wide Permitted, Fast Permitted, Synchronous Permitted,
  1668. Disconnect Permitted, Ultra Permitted, and SCAM Information in the
  1669. Host Adapter structure.
  1670.       */
  1671.       HostAdapter->WidePermitted = AutoSCSIData.WidePermitted;
  1672.       HostAdapter->FastPermitted = AutoSCSIData.FastPermitted;
  1673.       HostAdapter->SynchronousPermitted =
  1674. AutoSCSIData.SynchronousPermitted;
  1675.       HostAdapter->DisconnectPermitted =
  1676. AutoSCSIData.DisconnectPermitted;
  1677.       if (HostAdapter->HostUltraSCSI)
  1678. HostAdapter->UltraPermitted = AutoSCSIData.UltraPermitted;
  1679.       if (HostAdapter->HostSupportsSCAM)
  1680. {
  1681.   HostAdapter->SCAM_Enabled = AutoSCSIData.SCAM_Enabled;
  1682.   HostAdapter->SCAM_Level2 = AutoSCSIData.SCAM_Level2;
  1683. }
  1684.     }
  1685.   /*
  1686.     Initialize fields in the Host Adapter structure for "S" and "A" series
  1687.     MultiMaster Host Adapters.
  1688.   */
  1689.   if (HostAdapter->FirmwareVersion[0] < '4')
  1690.     {
  1691.       if (SetupInformation.SynchronousInitiationEnabled)
  1692. {
  1693.   HostAdapter->SynchronousPermitted = 0xFF;
  1694.   if (HostAdapter->HostAdapterBusType == BusLogic_EISA_Bus)
  1695.     {
  1696.       if (ExtendedSetupInformation.Misc.FastOnEISA)
  1697. HostAdapter->FastPermitted = 0xFF;
  1698.       if (strcmp(HostAdapter->ModelName, "BT-757") == 0)
  1699. HostAdapter->WidePermitted = 0xFF;
  1700.     }
  1701. }
  1702.       HostAdapter->DisconnectPermitted = 0xFF;
  1703.       HostAdapter->ParityCheckingEnabled =
  1704. SetupInformation.ParityCheckingEnabled;
  1705.       HostAdapter->BusResetEnabled = true;
  1706.     }
  1707.   /*
  1708.     Determine the maximum number of Target IDs and Logical Units supported by
  1709.     this driver for Wide and Narrow Host Adapters.
  1710.   */
  1711.   HostAdapter->MaxTargetDevices = (HostAdapter->HostWideSCSI ? 16 : 8);
  1712.   HostAdapter->MaxLogicalUnits = (HostAdapter->ExtendedLUNSupport ? 32 : 8);
  1713.   /*
  1714.     Select appropriate values for the Mailbox Count, Driver Queue Depth,
  1715.     Initial CCBs, and Incremental CCBs variables based on whether or not Strict
  1716.     Round Robin Mode is supported.  If Strict Round Robin Mode is supported,
  1717.     then there is no performance degradation in using the maximum possible
  1718.     number of Outgoing and Incoming Mailboxes and allowing the Tagged and
  1719.     Untagged Queue Depths to determine the actual utilization.  If Strict Round
  1720.     Robin Mode is not supported, then the Host Adapter must scan all the
  1721.     Outgoing Mailboxes whenever an Outgoing Mailbox entry is made, which can
  1722.     cause a substantial performance penalty.  The host adapters actually have
  1723.     room to store the following number of CCBs internally; that is, they can
  1724.     internally queue and manage this many active commands on the SCSI bus
  1725.     simultaneously.  Performance measurements demonstrate that the Driver Queue
  1726.     Depth should be set to the Mailbox Count, rather than the Host Adapter
  1727.     Queue Depth (internal CCB capacity), as it is more efficient to have the
  1728.     queued commands waiting in Outgoing Mailboxes if necessary than to block
  1729.     the process in the higher levels of the SCSI Subsystem.
  1730. 192   BT-948/958/958D
  1731. 100   BT-946C/956C/956CD/747C/757C/757CD/445C
  1732.  50   BT-545C/540CF
  1733.  30   BT-747S/747D/757S/757D/445S/545S/542D/542B/742A
  1734.   */
  1735.   if (HostAdapter->FirmwareVersion[0] == '5')
  1736.     HostAdapter->HostAdapterQueueDepth = 192;
  1737.   else if (HostAdapter->FirmwareVersion[0] == '4')
  1738.     HostAdapter->HostAdapterQueueDepth =
  1739.       (HostAdapter->HostAdapterBusType != BusLogic_ISA_Bus ? 100 : 50);
  1740.   else HostAdapter->HostAdapterQueueDepth = 30;
  1741.   if (strcmp(HostAdapter->FirmwareVersion, "3.31") >= 0)
  1742.     {
  1743.       HostAdapter->StrictRoundRobinModeSupport = true;
  1744.       HostAdapter->MailboxCount = BusLogic_MaxMailboxes;
  1745.     }
  1746.   else
  1747.     {
  1748.       HostAdapter->StrictRoundRobinModeSupport = false;
  1749.       HostAdapter->MailboxCount = 32;
  1750.     }
  1751.   HostAdapter->DriverQueueDepth = HostAdapter->MailboxCount;
  1752.   HostAdapter->InitialCCBs = 4 * BusLogic_CCB_AllocationGroupSize;
  1753.   HostAdapter->IncrementalCCBs = BusLogic_CCB_AllocationGroupSize;
  1754.   /*
  1755.     Tagged Queuing support is available and operates properly on all "W" series
  1756.     MultiMaster Host Adapters, on "C" series MultiMaster Host Adapters with
  1757.     firmware version 4.22 and above, and on "S" series MultiMaster Host
  1758.     Adapters with firmware version 3.35 and above.
  1759.   */
  1760.   HostAdapter->TaggedQueuingPermitted = 0;
  1761.   switch (HostAdapter->FirmwareVersion[0])
  1762.     {
  1763.     case '5':
  1764.       HostAdapter->TaggedQueuingPermitted = 0xFFFF;
  1765.       break;
  1766.     case '4':
  1767.       if (strcmp(HostAdapter->FirmwareVersion, "4.22") >= 0)
  1768. HostAdapter->TaggedQueuingPermitted = 0xFFFF;
  1769.       break;
  1770.     case '3':
  1771.       if (strcmp(HostAdapter->FirmwareVersion, "3.35") >= 0)
  1772. HostAdapter->TaggedQueuingPermitted = 0xFFFF;
  1773.       break;
  1774.     }
  1775.   /*
  1776.     Determine the Host Adapter BIOS Address if the BIOS is enabled and
  1777.     save it in the Host Adapter structure.  The BIOS is disabled if the
  1778.     BIOS_Address is 0.
  1779.   */
  1780.   HostAdapter->BIOS_Address = ExtendedSetupInformation.BIOS_Address << 12;
  1781.   /*
  1782.     ISA Host Adapters require Bounce Buffers if there is more than 16MB memory.
  1783.   */
  1784.   if (HostAdapter->HostAdapterBusType == BusLogic_ISA_Bus &&
  1785.       (void *) high_memory > (void *) MAX_DMA_ADDRESS)
  1786.     HostAdapter->BounceBuffersRequired = true;
  1787.   /*
  1788.     BusLogic BT-445S Host Adapters prior to board revision E have a hardware
  1789.     bug whereby when the BIOS is enabled, transfers to/from the same address
  1790.     range the BIOS occupies modulo 16MB are handled incorrectly.  Only properly
  1791.     functioning BT-445S Host Adapters have firmware version 3.37, so require
  1792.     that ISA Bounce Buffers be used for the buggy BT-445S models if there is
  1793.     more than 16MB memory.
  1794.   */
  1795.   if (HostAdapter->BIOS_Address > 0 &&
  1796.       strcmp(HostAdapter->ModelName, "BT-445S") == 0 &&
  1797.       strcmp(HostAdapter->FirmwareVersion, "3.37") < 0 &&
  1798.       (void *) high_memory > (void *) MAX_DMA_ADDRESS)
  1799.     HostAdapter->BounceBuffersRequired = true;
  1800.   /*
  1801.     Initialize parameters common to MultiMaster and FlashPoint Host Adapters.
  1802.   */
  1803. Common:
  1804.   /*
  1805.     Initialize the Host Adapter Full Model Name from the Model Name.
  1806.   */
  1807.   strcpy(HostAdapter->FullModelName, "BusLogic ");
  1808.   strcat(HostAdapter->FullModelName, HostAdapter->ModelName);
  1809.   /*
  1810.     Select an appropriate value for the Tagged Queue Depth either from a
  1811.     BusLogic Driver Options specification, or based on whether this Host
  1812.     Adapter requires that ISA Bounce Buffers be used.  The Tagged Queue Depth
  1813.     is left at 0 for automatic determination in BusLogic_SelectQueueDepths.
  1814.     Initialize the Untagged Queue Depth.
  1815.   */
  1816.   for (TargetID = 0; TargetID < BusLogic_MaxTargetDevices; TargetID++)
  1817.     {
  1818.       unsigned char QueueDepth = 0;
  1819.       if (HostAdapter->DriverOptions != NULL &&
  1820.   HostAdapter->DriverOptions->QueueDepth[TargetID] > 0)
  1821. QueueDepth = HostAdapter->DriverOptions->QueueDepth[TargetID];
  1822.       else if (HostAdapter->BounceBuffersRequired)
  1823. QueueDepth = BusLogic_TaggedQueueDepthBB;
  1824.       HostAdapter->QueueDepth[TargetID] = QueueDepth;
  1825.     }
  1826.   if (HostAdapter->BounceBuffersRequired)
  1827.     HostAdapter->UntaggedQueueDepth = BusLogic_UntaggedQueueDepthBB;
  1828.   else HostAdapter->UntaggedQueueDepth = BusLogic_UntaggedQueueDepth;
  1829.   if (HostAdapter->DriverOptions != NULL)
  1830.     HostAdapter->CommonQueueDepth =
  1831.       HostAdapter->DriverOptions->CommonQueueDepth;
  1832.   if (HostAdapter->CommonQueueDepth > 0 &&
  1833.       HostAdapter->CommonQueueDepth < HostAdapter->UntaggedQueueDepth)
  1834.     HostAdapter->UntaggedQueueDepth = HostAdapter->CommonQueueDepth;
  1835.   /*
  1836.     Tagged Queuing is only allowed if Disconnect/Reconnect is permitted.
  1837.     Therefore, mask the Tagged Queuing Permitted Default bits with the
  1838.     Disconnect/Reconnect Permitted bits.
  1839.   */
  1840.   HostAdapter->TaggedQueuingPermitted &= HostAdapter->DisconnectPermitted;
  1841.   /*
  1842.     Combine the default Tagged Queuing Permitted bits with any BusLogic Driver
  1843.     Options Tagged Queuing specification.
  1844.   */
  1845.   if (HostAdapter->DriverOptions != NULL)
  1846.     HostAdapter->TaggedQueuingPermitted =
  1847.       (HostAdapter->DriverOptions->TaggedQueuingPermitted &
  1848.        HostAdapter->DriverOptions->TaggedQueuingPermittedMask) |
  1849.       (HostAdapter->TaggedQueuingPermitted &
  1850.        ~HostAdapter->DriverOptions->TaggedQueuingPermittedMask);
  1851.   /*
  1852.     Select appropriate values for the Error Recovery Strategy array
  1853.     either from a BusLogic Driver Options specification, or using
  1854.     BusLogic_ErrorRecovery_Default.
  1855.   */
  1856.   for (TargetID = 0; TargetID < BusLogic_MaxTargetDevices; TargetID++)
  1857.     if (HostAdapter->DriverOptions != NULL)
  1858.       HostAdapter->ErrorRecoveryStrategy[TargetID] =
  1859. HostAdapter->DriverOptions->ErrorRecoveryStrategy[TargetID];
  1860.     else HostAdapter->ErrorRecoveryStrategy[TargetID] =
  1861.    BusLogic_ErrorRecovery_Default;
  1862.   /*
  1863.     Select an appropriate value for Bus Settle Time either from a BusLogic
  1864.     Driver Options specification, or from BusLogic_DefaultBusSettleTime.
  1865.   */
  1866.   if (HostAdapter->DriverOptions != NULL &&
  1867.       HostAdapter->DriverOptions->BusSettleTime > 0)
  1868.     HostAdapter->BusSettleTime = HostAdapter->DriverOptions->BusSettleTime;
  1869.   else HostAdapter->BusSettleTime = BusLogic_DefaultBusSettleTime;
  1870.   /*
  1871.     Indicate reading the Host Adapter Configuration completed successfully.
  1872.   */
  1873.   return true;
  1874. }
  1875. /*
  1876.   BusLogic_ReportHostAdapterConfiguration reports the configuration of
  1877.   Host Adapter.
  1878. */
  1879. static boolean BusLogic_ReportHostAdapterConfiguration(BusLogic_HostAdapter_T
  1880.        *HostAdapter)
  1881. {
  1882.   unsigned short AllTargetsMask = (1 << HostAdapter->MaxTargetDevices) - 1;
  1883.   unsigned short SynchronousPermitted, FastPermitted;
  1884.   unsigned short UltraPermitted, WidePermitted;
  1885.   unsigned short DisconnectPermitted, TaggedQueuingPermitted;
  1886.   boolean CommonSynchronousNegotiation, CommonTaggedQueueDepth;
  1887.   boolean CommonErrorRecovery;
  1888.   char SynchronousString[BusLogic_MaxTargetDevices+1];
  1889.   char WideString[BusLogic_MaxTargetDevices+1];
  1890.   char DisconnectString[BusLogic_MaxTargetDevices+1];
  1891.   char TaggedQueuingString[BusLogic_MaxTargetDevices+1];
  1892.   char ErrorRecoveryString[BusLogic_MaxTargetDevices+1];
  1893.   char *SynchronousMessage = SynchronousString;
  1894.   char *WideMessage = WideString;
  1895.   char *DisconnectMessage = DisconnectString;
  1896.   char *TaggedQueuingMessage = TaggedQueuingString;
  1897.   char *ErrorRecoveryMessage = ErrorRecoveryString;
  1898.   int TargetID;
  1899.   BusLogic_Info("Configuring BusLogic Model %s %s%s%s%s SCSI Host Adaptern",
  1900. HostAdapter, HostAdapter->ModelName,
  1901. BusLogic_HostAdapterBusNames[HostAdapter->HostAdapterBusType],
  1902. (HostAdapter->HostWideSCSI ? " Wide" : ""),
  1903. (HostAdapter->HostDifferentialSCSI ? " Differential" : ""),
  1904. (HostAdapter->HostUltraSCSI ? " Ultra" : ""));
  1905.   BusLogic_Info("  Firmware Version: %s, I/O Address: 0x%X, "
  1906. "IRQ Channel: %d/%sn", HostAdapter,
  1907. HostAdapter->FirmwareVersion,
  1908. HostAdapter->IO_Address, HostAdapter->IRQ_Channel,
  1909. (HostAdapter->LevelSensitiveInterrupt ? "Level" : "Edge"));
  1910.   if (HostAdapter->HostAdapterBusType != BusLogic_PCI_Bus)
  1911.     {
  1912.       BusLogic_Info("  DMA Channel: ", HostAdapter);
  1913.       if (HostAdapter->DMA_Channel > 0)
  1914. BusLogic_Info("%d, ", HostAdapter, HostAdapter->DMA_Channel);
  1915.       else BusLogic_Info("None, ", HostAdapter);
  1916.       if (HostAdapter->BIOS_Address > 0)
  1917. BusLogic_Info("BIOS Address: 0x%X, ", HostAdapter,
  1918.       HostAdapter->BIOS_Address);
  1919.       else BusLogic_Info("BIOS Address: None, ", HostAdapter);
  1920.     }
  1921.   else
  1922.     {
  1923.       BusLogic_Info("  PCI Bus: %d, Device: %d, Address: ",
  1924.     HostAdapter, HostAdapter->Bus, HostAdapter->Device);
  1925.       if (HostAdapter->PCI_Address > 0)
  1926. BusLogic_Info("0x%X, ", HostAdapter, HostAdapter->PCI_Address);
  1927.       else BusLogic_Info("Unassigned, ", HostAdapter);
  1928.     }
  1929.   BusLogic_Info("Host Adapter SCSI ID: %dn", HostAdapter,
  1930. HostAdapter->SCSI_ID);
  1931.   BusLogic_Info("  Parity Checking: %s, Extended Translation: %sn",
  1932. HostAdapter,
  1933. (HostAdapter->ParityCheckingEnabled
  1934.  ? "Enabled" : "Disabled"),
  1935. (HostAdapter->ExtendedTranslationEnabled
  1936.  ? "Enabled" : "Disabled"));
  1937.   AllTargetsMask &= ~(1 << HostAdapter->SCSI_ID);
  1938.   SynchronousPermitted = HostAdapter->SynchronousPermitted & AllTargetsMask;
  1939.   FastPermitted = HostAdapter->FastPermitted & AllTargetsMask;
  1940.   UltraPermitted = HostAdapter->UltraPermitted & AllTargetsMask;
  1941.   if ((BusLogic_MultiMasterHostAdapterP(HostAdapter) &&
  1942.        (HostAdapter->FirmwareVersion[0] >= '4' ||
  1943. HostAdapter->HostAdapterBusType == BusLogic_EISA_Bus)) ||
  1944.       BusLogic_FlashPointHostAdapterP(HostAdapter))
  1945.     {
  1946.       CommonSynchronousNegotiation = false;
  1947.       if (SynchronousPermitted == 0)
  1948. {
  1949.   SynchronousMessage = "Disabled";
  1950.   CommonSynchronousNegotiation = true;
  1951. }
  1952.       else if (SynchronousPermitted == AllTargetsMask)
  1953. {
  1954.   if (FastPermitted == 0)
  1955.     {
  1956.       SynchronousMessage = "Slow";
  1957.       CommonSynchronousNegotiation = true;
  1958.     }
  1959.   else if (FastPermitted == AllTargetsMask)
  1960.     {
  1961.       if (UltraPermitted == 0)
  1962. {
  1963.   SynchronousMessage = "Fast";
  1964.   CommonSynchronousNegotiation = true;
  1965. }
  1966.       else if (UltraPermitted == AllTargetsMask)
  1967. {
  1968.   SynchronousMessage = "Ultra";
  1969.   CommonSynchronousNegotiation = true;
  1970. }
  1971.     }
  1972. }
  1973.       if (!CommonSynchronousNegotiation)
  1974. {
  1975.   for (TargetID = 0;
  1976.        TargetID < HostAdapter->MaxTargetDevices;
  1977.        TargetID++)
  1978.     SynchronousString[TargetID] =
  1979.       ((!(SynchronousPermitted & (1 << TargetID))) ? 'N' :
  1980.        (!(FastPermitted & (1 << TargetID)) ? 'S' :
  1981. (!(UltraPermitted & (1 << TargetID)) ? 'F' : 'U')));
  1982.   SynchronousString[HostAdapter->SCSI_ID] = '#';
  1983.   SynchronousString[HostAdapter->MaxTargetDevices] = '';
  1984. }
  1985.     }
  1986.   else SynchronousMessage =
  1987.  (SynchronousPermitted == 0 ? "Disabled" : "Enabled");
  1988.   WidePermitted = HostAdapter->WidePermitted & AllTargetsMask;
  1989.   if (WidePermitted == 0)
  1990.     WideMessage = "Disabled";
  1991.   else if (WidePermitted == AllTargetsMask)
  1992.     WideMessage = "Enabled";
  1993.   else
  1994.     {
  1995.       for (TargetID = 0; TargetID < HostAdapter->MaxTargetDevices; TargetID++)
  1996.  WideString[TargetID] =
  1997.    ((WidePermitted & (1 << TargetID)) ? 'Y' : 'N');
  1998.       WideString[HostAdapter->SCSI_ID] = '#';
  1999.       WideString[HostAdapter->MaxTargetDevices] = '';
  2000.     }
  2001.   DisconnectPermitted = HostAdapter->DisconnectPermitted & AllTargetsMask;
  2002.   if (DisconnectPermitted == 0)
  2003.     DisconnectMessage = "Disabled";
  2004.   else if (DisconnectPermitted == AllTargetsMask)
  2005.     DisconnectMessage = "Enabled";
  2006.   else
  2007.     {
  2008.       for (TargetID = 0; TargetID < HostAdapter->MaxTargetDevices; TargetID++)
  2009. DisconnectString[TargetID] =
  2010.   ((DisconnectPermitted & (1 << TargetID)) ? 'Y' : 'N');
  2011.       DisconnectString[HostAdapter->SCSI_ID] = '#';
  2012.       DisconnectString[HostAdapter->MaxTargetDevices] = '';
  2013.     }
  2014.   TaggedQueuingPermitted =
  2015.     HostAdapter->TaggedQueuingPermitted & AllTargetsMask;
  2016.   if (TaggedQueuingPermitted == 0)
  2017.     TaggedQueuingMessage = "Disabled";
  2018.   else if (TaggedQueuingPermitted == AllTargetsMask)
  2019.     TaggedQueuingMessage = "Enabled";
  2020.   else
  2021.     {
  2022.       for (TargetID = 0; TargetID < HostAdapter->MaxTargetDevices; TargetID++)
  2023. TaggedQueuingString[TargetID] =
  2024.   ((TaggedQueuingPermitted & (1 << TargetID)) ? 'Y' : 'N');
  2025.       TaggedQueuingString[HostAdapter->SCSI_ID] = '#';
  2026.       TaggedQueuingString[HostAdapter->MaxTargetDevices] = '';
  2027.     }
  2028.   BusLogic_Info("  Synchronous Negotiation: %s, Wide Negotiation: %sn",
  2029. HostAdapter, SynchronousMessage, WideMessage);
  2030.   BusLogic_Info("  Disconnect/Reconnect: %s, Tagged Queuing: %sn",
  2031. HostAdapter, DisconnectMessage, TaggedQueuingMessage);
  2032.   if (BusLogic_MultiMasterHostAdapterP(HostAdapter))
  2033.     {
  2034.       BusLogic_Info("  Scatter/Gather Limit: %d of %d segments, "
  2035.     "Mailboxes: %dn", HostAdapter,
  2036.     HostAdapter->DriverScatterGatherLimit,
  2037.     HostAdapter->HostAdapterScatterGatherLimit,
  2038.     HostAdapter->MailboxCount);
  2039.       BusLogic_Info("  Driver Queue Depth: %d, "
  2040.     "Host Adapter Queue Depth: %dn",
  2041.     HostAdapter, HostAdapter->DriverQueueDepth,
  2042.     HostAdapter->HostAdapterQueueDepth);
  2043.     }
  2044.   else BusLogic_Info("  Driver Queue Depth: %d, "
  2045.      "Scatter/Gather Limit: %d segmentsn",
  2046.      HostAdapter, HostAdapter->DriverQueueDepth,
  2047.      HostAdapter->DriverScatterGatherLimit);
  2048.   BusLogic_Info("  Tagged Queue Depth: ", HostAdapter);
  2049.   CommonTaggedQueueDepth = true;
  2050.   for (TargetID = 1; TargetID < HostAdapter->MaxTargetDevices; TargetID++)
  2051.     if (HostAdapter->QueueDepth[TargetID] != HostAdapter->QueueDepth[0])
  2052.       {
  2053. CommonTaggedQueueDepth = false;
  2054. break;
  2055.       }
  2056.   if (CommonTaggedQueueDepth)
  2057.     {
  2058.       if (HostAdapter->QueueDepth[0] > 0)
  2059. BusLogic_Info("%d", HostAdapter, HostAdapter->QueueDepth[0]);
  2060.       else BusLogic_Info("Automatic", HostAdapter);
  2061.     }
  2062.   else BusLogic_Info("Individual", HostAdapter);
  2063.   BusLogic_Info(", Untagged Queue Depth: %dn", HostAdapter,
  2064. HostAdapter->UntaggedQueueDepth);
  2065.   CommonErrorRecovery = true;
  2066.   for (TargetID = 1; TargetID < HostAdapter->MaxTargetDevices; TargetID++)
  2067.     if (HostAdapter->ErrorRecoveryStrategy[TargetID] !=
  2068. HostAdapter->ErrorRecoveryStrategy[0])
  2069.       {
  2070. CommonErrorRecovery = false;
  2071. break;
  2072.       }
  2073.   if (CommonErrorRecovery)
  2074.     ErrorRecoveryMessage =
  2075.       BusLogic_ErrorRecoveryStrategyNames[
  2076. HostAdapter->ErrorRecoveryStrategy[0]];
  2077.   else
  2078.     {
  2079.       for (TargetID = 0; TargetID < HostAdapter->MaxTargetDevices; TargetID++)
  2080. ErrorRecoveryString[TargetID] =
  2081.   BusLogic_ErrorRecoveryStrategyLetters[
  2082.     HostAdapter->ErrorRecoveryStrategy[TargetID]];
  2083.       ErrorRecoveryString[HostAdapter->SCSI_ID] = '#';
  2084.       ErrorRecoveryString[HostAdapter->MaxTargetDevices] = '';
  2085.     }
  2086.   BusLogic_Info("  Error Recovery Strategy: %s, SCSI Bus Reset: %sn",
  2087. HostAdapter, ErrorRecoveryMessage,
  2088. (HostAdapter->BusResetEnabled ? "Enabled" : "Disabled"));
  2089.   if (HostAdapter->TerminationInfoValid)
  2090.     {
  2091.       if (HostAdapter->HostWideSCSI)
  2092. BusLogic_Info("  SCSI Bus Termination: %s", HostAdapter,
  2093.       (HostAdapter->LowByteTerminated
  2094.        ? (HostAdapter->HighByteTerminated
  2095.   ? "Both Enabled" : "Low Enabled")
  2096.        : (HostAdapter->HighByteTerminated
  2097.   ? "High Enabled" : "Both Disabled")));
  2098.       else BusLogic_Info("  SCSI Bus Termination: %s", HostAdapter,
  2099.  (HostAdapter->LowByteTerminated ?
  2100.   "Enabled" : "Disabled"));
  2101.       if (HostAdapter->HostSupportsSCAM)
  2102. BusLogic_Info(", SCAM: %s", HostAdapter,
  2103.       (HostAdapter->SCAM_Enabled
  2104.        ? (HostAdapter->SCAM_Level2
  2105.   ? "Enabled, Level 2" : "Enabled, Level 1")
  2106.        : "Disabled"));
  2107.       BusLogic_Info("n", HostAdapter);
  2108.     }
  2109.   /*
  2110.     Indicate reporting the Host Adapter configuration completed successfully.
  2111.   */
  2112.   return true;
  2113. }
  2114. /*
  2115.   BusLogic_AcquireResources acquires the system resources necessary to use
  2116.   Host Adapter.
  2117. */
  2118. static boolean BusLogic_AcquireResources(BusLogic_HostAdapter_T *HostAdapter)
  2119. {
  2120.   if (HostAdapter->IRQ_Channel == 0)
  2121.     {
  2122.       BusLogic_Error("NO LEGAL INTERRUPT CHANNEL ASSIGNED - DETACHINGn",
  2123.      HostAdapter);
  2124.       return false;
  2125.     }
  2126.   /*
  2127.     Acquire shared access to the IRQ Channel.
  2128.   */
  2129.   if (request_irq(HostAdapter->IRQ_Channel, BusLogic_InterruptHandler,
  2130.   SA_SHIRQ, HostAdapter->FullModelName, HostAdapter) < 0)
  2131.     {
  2132.       BusLogic_Error("UNABLE TO ACQUIRE IRQ CHANNEL %d - DETACHINGn",
  2133.      HostAdapter, HostAdapter->IRQ_Channel);
  2134.       return false;
  2135.     }
  2136.   HostAdapter->IRQ_ChannelAcquired = true;
  2137.   /*
  2138.     Acquire exclusive access to the DMA Channel.
  2139.   */
  2140.   if (HostAdapter->DMA_Channel > 0)
  2141.     {
  2142.       if (request_dma(HostAdapter->DMA_Channel,
  2143.       HostAdapter->FullModelName) < 0)
  2144. {
  2145.   BusLogic_Error("UNABLE TO ACQUIRE DMA CHANNEL %d - DETACHINGn",
  2146.  HostAdapter, HostAdapter->DMA_Channel);
  2147.   return false;
  2148. }
  2149.       set_dma_mode(HostAdapter->DMA_Channel, DMA_MODE_CASCADE);
  2150.       enable_dma(HostAdapter->DMA_Channel);
  2151.       HostAdapter->DMA_ChannelAcquired = true;
  2152.     }
  2153.   /*
  2154.     Indicate the System Resource Acquisition completed successfully,
  2155.   */
  2156.   return true;
  2157. }
  2158. /*
  2159.   BusLogic_ReleaseResources releases any system resources previously acquired
  2160.   by BusLogic_AcquireResources.
  2161. */
  2162. static void BusLogic_ReleaseResources(BusLogic_HostAdapter_T *HostAdapter)
  2163. {
  2164.   /*
  2165.     Release shared access to the IRQ Channel.
  2166.   */
  2167.   if (HostAdapter->IRQ_ChannelAcquired)
  2168.     free_irq(HostAdapter->IRQ_Channel, HostAdapter);
  2169.   /*
  2170.     Release exclusive access to the DMA Channel.
  2171.   */
  2172.   if (HostAdapter->DMA_ChannelAcquired)
  2173.     free_dma(HostAdapter->DMA_Channel);
  2174. }
  2175. /*
  2176.   BusLogic_InitializeHostAdapter initializes Host Adapter.  This is the only
  2177.   function called during SCSI Host Adapter detection which modifies the state
  2178.   of the Host Adapter from its initial power on or hard reset state.
  2179. */
  2180. static boolean BusLogic_InitializeHostAdapter(BusLogic_HostAdapter_T
  2181.       *HostAdapter)
  2182. {
  2183.   BusLogic_ExtendedMailboxRequest_T ExtendedMailboxRequest;
  2184.   BusLogic_RoundRobinModeRequest_T RoundRobinModeRequest;
  2185.   BusLogic_SetCCBFormatRequest_T SetCCBFormatRequest;
  2186.   int TargetID;
  2187.   /*
  2188.     Initialize the pointers to the first and last CCBs that are queued for
  2189.     completion processing.
  2190.   */
  2191.   HostAdapter->FirstCompletedCCB = NULL;
  2192.   HostAdapter->LastCompletedCCB = NULL;
  2193.   /*
  2194.     Initialize the Bus Device Reset Pending CCB, Tagged Queuing Active,
  2195.     Command Successful Flag, Active Commands, and Commands Since Reset
  2196.     for each Target Device.
  2197.   */
  2198.   for (TargetID = 0; TargetID < HostAdapter->MaxTargetDevices; TargetID++)
  2199.     {
  2200.       HostAdapter->BusDeviceResetPendingCCB[TargetID] = NULL;
  2201.       HostAdapter->TargetFlags[TargetID].TaggedQueuingActive = false;
  2202.       HostAdapter->TargetFlags[TargetID].CommandSuccessfulFlag = false;
  2203.       HostAdapter->ActiveCommands[TargetID] = 0;
  2204.       HostAdapter->CommandsSinceReset[TargetID] = 0;
  2205.     }
  2206.   /*
  2207.     FlashPoint Host Adapters do not use Outgoing and Incoming Mailboxes.
  2208.   */
  2209.   if (BusLogic_FlashPointHostAdapterP(HostAdapter)) goto Done;
  2210.   /*
  2211.     Initialize the Outgoing and Incoming Mailbox pointers.
  2212.   */
  2213.   HostAdapter->FirstOutgoingMailbox =
  2214.     (BusLogic_OutgoingMailbox_T *) HostAdapter->MailboxSpace;
  2215.   HostAdapter->LastOutgoingMailbox =
  2216.     HostAdapter->FirstOutgoingMailbox + HostAdapter->MailboxCount - 1;
  2217.   HostAdapter->NextOutgoingMailbox = HostAdapter->FirstOutgoingMailbox;
  2218.   HostAdapter->FirstIncomingMailbox =
  2219.     (BusLogic_IncomingMailbox_T *) (HostAdapter->LastOutgoingMailbox + 1);
  2220.   HostAdapter->LastIncomingMailbox =
  2221.     HostAdapter->FirstIncomingMailbox + HostAdapter->MailboxCount - 1;
  2222.   HostAdapter->NextIncomingMailbox = HostAdapter->FirstIncomingMailbox;
  2223.   /*
  2224.     Initialize the Outgoing and Incoming Mailbox structures.
  2225.   */
  2226.   memset(HostAdapter->FirstOutgoingMailbox, 0,
  2227.  HostAdapter->MailboxCount * sizeof(BusLogic_OutgoingMailbox_T));
  2228.   memset(HostAdapter->FirstIncomingMailbox, 0,
  2229.  HostAdapter->MailboxCount * sizeof(BusLogic_IncomingMailbox_T));
  2230.   /*
  2231.     Initialize the Host Adapter's Pointer to the Outgoing/Incoming Mailboxes.
  2232.   */
  2233.   ExtendedMailboxRequest.MailboxCount = HostAdapter->MailboxCount;
  2234.   ExtendedMailboxRequest.BaseMailboxAddress =
  2235.     Virtual_to_Bus(HostAdapter->FirstOutgoingMailbox);
  2236.   if (BusLogic_Command(HostAdapter, BusLogic_InitializeExtendedMailbox,
  2237.        &ExtendedMailboxRequest,
  2238.        sizeof(ExtendedMailboxRequest), NULL, 0) < 0)
  2239.     return BusLogic_Failure(HostAdapter, "MAILBOX INITIALIZATION");
  2240.   /*
  2241.     Enable Strict Round Robin Mode if supported by the Host Adapter.  In
  2242.     Strict Round Robin Mode, the Host Adapter only looks at the next Outgoing
  2243.     Mailbox for each new command, rather than scanning through all the
  2244.     Outgoing Mailboxes to find any that have new commands in them.  Strict
  2245.     Round Robin Mode is significantly more efficient.
  2246.   */
  2247.   if (HostAdapter->StrictRoundRobinModeSupport)
  2248.     {
  2249.       RoundRobinModeRequest = BusLogic_StrictRoundRobinMode;
  2250.       if (BusLogic_Command(HostAdapter, BusLogic_EnableStrictRoundRobinMode,
  2251.    &RoundRobinModeRequest,
  2252.    sizeof(RoundRobinModeRequest), NULL, 0) < 0)
  2253. return BusLogic_Failure(HostAdapter, "ENABLE STRICT ROUND ROBIN MODE");
  2254.     }
  2255.   /*
  2256.     For Host Adapters that support Extended LUN Format CCBs, issue the Set CCB
  2257.     Format command to allow 32 Logical Units per Target Device.
  2258.   */
  2259.   if (HostAdapter->ExtendedLUNSupport)
  2260.     {
  2261.       SetCCBFormatRequest = BusLogic_ExtendedLUNFormatCCB;
  2262.       if (BusLogic_Command(HostAdapter, BusLogic_SetCCBFormat,
  2263.    &SetCCBFormatRequest, sizeof(SetCCBFormatRequest),
  2264.    NULL, 0) < 0)
  2265. return BusLogic_Failure(HostAdapter, "SET CCB FORMAT");
  2266.     }
  2267.   /*
  2268.     Announce Successful Initialization.
  2269.   */
  2270. Done:
  2271.   if (!HostAdapter->HostAdapterInitialized)
  2272.     {
  2273.       BusLogic_Info("*** %s Initialized Successfully ***n",
  2274.     HostAdapter, HostAdapter->FullModelName);
  2275.       BusLogic_Info("n", HostAdapter);
  2276.     }
  2277.   else BusLogic_Warning("*** %s Initialized Successfully ***n",
  2278. HostAdapter, HostAdapter->FullModelName);
  2279.   HostAdapter->HostAdapterInitialized = true;
  2280.   /*
  2281.     Indicate the Host Adapter Initialization completed successfully.
  2282.   */
  2283.   return true;
  2284. }
  2285. /*
  2286.   BusLogic_TargetDeviceInquiry inquires about the Target Devices accessible
  2287.   through Host Adapter.
  2288. */
  2289. static boolean BusLogic_TargetDeviceInquiry(BusLogic_HostAdapter_T
  2290.     *HostAdapter)
  2291. {
  2292.   BusLogic_InstalledDevices_T InstalledDevices;
  2293.   BusLogic_InstalledDevices8_T InstalledDevicesID0to7;
  2294.   BusLogic_SetupInformation_T SetupInformation;
  2295.   BusLogic_SynchronousPeriod_T SynchronousPeriod;
  2296.   BusLogic_RequestedReplyLength_T RequestedReplyLength;
  2297.   int TargetID;
  2298.   /*
  2299.     Wait a few seconds between the Host Adapter Hard Reset which initiates
  2300.     a SCSI Bus Reset and issuing any SCSI Commands.  Some SCSI devices get
  2301.     confused if they receive SCSI Commands too soon after a SCSI Bus Reset.
  2302.   */
  2303.   BusLogic_Delay(HostAdapter->BusSettleTime);
  2304.   /*
  2305.     FlashPoint Host Adapters do not provide for Target Device Inquiry.
  2306.   */
  2307.   if (BusLogic_FlashPointHostAdapterP(HostAdapter)) return true;
  2308.   /*
  2309.     Inhibit the Target Device Inquiry if requested.
  2310.   */
  2311.   if (HostAdapter->DriverOptions != NULL &&
  2312.       HostAdapter->DriverOptions->LocalOptions.InhibitTargetInquiry)
  2313.     return true;
  2314.   /*
  2315.     Issue the Inquire Target Devices command for host adapters with firmware
  2316.     version 4.25 or later, or the Inquire Installed Devices ID 0 to 7 command
  2317.     for older host adapters.  This is necessary to force Synchronous Transfer
  2318.     Negotiation so that the Inquire Setup Information and Inquire Synchronous
  2319.     Period commands will return valid data.  The Inquire Target Devices command
  2320.     is preferable to Inquire Installed Devices ID 0 to 7 since it only probes
  2321.     Logical Unit 0 of each Target Device.
  2322.   */
  2323.   if (strcmp(HostAdapter->FirmwareVersion, "4.25") >= 0)
  2324.     {
  2325.       if (BusLogic_Command(HostAdapter, BusLogic_InquireTargetDevices, NULL, 0,
  2326.    &InstalledDevices, sizeof(InstalledDevices))
  2327.   != sizeof(InstalledDevices))
  2328. return BusLogic_Failure(HostAdapter, "INQUIRE TARGET DEVICES");
  2329.       for (TargetID = 0; TargetID < HostAdapter->MaxTargetDevices; TargetID++)
  2330. HostAdapter->TargetFlags[TargetID].TargetExists =
  2331.   (InstalledDevices & (1 << TargetID) ? true : false);
  2332.     }
  2333.   else
  2334.     {
  2335.       if (BusLogic_Command(HostAdapter, BusLogic_InquireInstalledDevicesID0to7,
  2336.    NULL, 0, &InstalledDevicesID0to7,
  2337.    sizeof(InstalledDevicesID0to7))
  2338.   != sizeof(InstalledDevicesID0to7))
  2339. return BusLogic_Failure(HostAdapter,
  2340. "INQUIRE INSTALLED DEVICES ID 0 TO 7");
  2341.       for (TargetID = 0; TargetID < 8; TargetID++)
  2342. HostAdapter->TargetFlags[TargetID].TargetExists =
  2343.   (InstalledDevicesID0to7[TargetID] != 0 ? true : false);
  2344.     }
  2345.   /*
  2346.     Issue the Inquire Setup Information command.
  2347.   */
  2348.   RequestedReplyLength = sizeof(SetupInformation);
  2349.   if (BusLogic_Command(HostAdapter, BusLogic_InquireSetupInformation,
  2350.        &RequestedReplyLength, sizeof(RequestedReplyLength),
  2351.        &SetupInformation, sizeof(SetupInformation))
  2352.       != sizeof(SetupInformation))
  2353.     return BusLogic_Failure(HostAdapter, "INQUIRE SETUP INFORMATION");
  2354.   for (TargetID = 0; TargetID < HostAdapter->MaxTargetDevices; TargetID++)
  2355.       HostAdapter->SynchronousOffset[TargetID] =
  2356. (TargetID < 8
  2357.  ? SetupInformation.SynchronousValuesID0to7[TargetID].Offset
  2358.  : SetupInformation.SynchronousValuesID8to15[TargetID-8].Offset);
  2359.   if (strcmp(HostAdapter->FirmwareVersion, "5.06L") >= 0)
  2360.     for (TargetID = 0; TargetID < HostAdapter->MaxTargetDevices; TargetID++)
  2361.       HostAdapter->TargetFlags[TargetID].WideTransfersActive =
  2362. (TargetID < 8
  2363.  ? (SetupInformation.WideTransfersActiveID0to7 & (1 << TargetID)
  2364.     ? true : false)
  2365.  : (SetupInformation.WideTransfersActiveID8to15 & (1 << (TargetID-8))
  2366.     ? true : false));
  2367.   /*
  2368.     Issue the Inquire Synchronous Period command.