ips.c
上传用户:jlfgdled
上传日期:2013-04-10
资源大小:33168k
文件大小:274k
源码类别:

Linux/Unix编程

开发平台:

Unix_Linux

  1.       scb->cmd.dcdb.dcdb_address = cpu_to_le32(VIRT_TO_BUS(&scb->dcdb));
  2.    if (pt->CmdBSize) {
  3.       if (scb->cmd.dcdb.op_code == IPS_CMD_DCDB)
  4.          scb->dcdb.buffer_pointer = cpu_to_le32(scb->data_busaddr);
  5.       else
  6.          scb->cmd.basic_io.sg_addr = cpu_to_le32(scb->data_busaddr);
  7.    }
  8.    /* set timeouts */
  9.    if (pt->TimeOut) {
  10.       scb->timeout = pt->TimeOut;
  11.       if (pt->TimeOut <= 10)
  12.          scb->dcdb.cmd_attribute |= IPS_TIMEOUT10;
  13.       else if (pt->TimeOut <= 60)
  14.          scb->dcdb.cmd_attribute |= IPS_TIMEOUT60;
  15.       else
  16.          scb->dcdb.cmd_attribute |= IPS_TIMEOUT20M;
  17.    }
  18.    /* assume success */
  19.    scb->scsi_cmd->result = DID_OK << 16;
  20.    /* success */
  21.    return (1);
  22. }
  23. /****************************************************************************/
  24. /*                                                                          */
  25. /* Routine Name: ips_cleanup_passthru                                       */
  26. /*                                                                          */
  27. /* Routine Description:                                                     */
  28. /*                                                                          */
  29. /*   Cleanup after a passthru command                                       */
  30. /*                                                                          */
  31. /****************************************************************************/
  32. static void
  33. ips_cleanup_passthru(ips_ha_t *ha, ips_scb_t *scb) {
  34.    ips_passthru_t *pt;
  35.    METHOD_TRACE("ips_cleanup_passthru", 1);
  36.    if ((!scb) || (!scb->scsi_cmd) || (!scb->scsi_cmd->request_buffer)) {
  37.       DEBUG_VAR(1, "(%s%d) couldn't cleanup after passthru",
  38.                 ips_name, ha->host_num);
  39.       return ;
  40.    }
  41.    if(!scb->scsi_cmd->use_sg)
  42.       pt = (ips_passthru_t *) scb->scsi_cmd->request_buffer;
  43.    else
  44.       pt = (ips_passthru_t *) ha->ioctl_data;
  45.    /* Copy data back to the user */
  46.    if (scb->cmd.dcdb.op_code == IPS_CMD_DCDB)        /* Copy DCDB Back to Caller's Area */
  47.       memcpy(&pt->CoppCP.dcdb, &scb->dcdb, sizeof(IPS_DCDB_TABLE));
  48.    
  49.    pt->BasicStatus = scb->basic_status;
  50.    pt->ExtendedStatus = scb->extended_status;
  51.    pt->AdapterType = ha->ad_type;
  52.    if(ha->device_id == IPS_DEVICEID_COPPERHEAD && 
  53.      (scb->cmd.flashfw.op_code == IPS_CMD_DOWNLOAD || 
  54.      scb->cmd.flashfw.op_code == IPS_CMD_RW_BIOSFW))
  55.       ips_free_flash_copperhead(ha);
  56.    if(scb->scsi_cmd->use_sg){
  57.       int i, length = 0;
  58.       struct scatterlist *sg = scb->scsi_cmd->request_buffer;
  59.       for(i = 0; i < scb->scsi_cmd->use_sg; i++){
  60.          memcpy(sg[i].address, &ha->ioctl_data[length], sg[i].length);
  61.          length += sg[i].length;
  62.       }
  63.    }
  64. }
  65. /****************************************************************************/
  66. /*                                                                          */
  67. /* Routine Name: ips_host_info                                              */
  68. /*                                                                          */
  69. /* Routine Description:                                                     */
  70. /*                                                                          */
  71. /*   The passthru interface for the driver                                  */
  72. /*                                                                          */
  73. /****************************************************************************/
  74. static int
  75. ips_host_info(ips_ha_t *ha, char *ptr, off_t offset, int len) {
  76.    IPS_INFOSTR info;
  77.    METHOD_TRACE("ips_host_info", 1);
  78.    info.buffer = ptr;
  79.    info.length = len;
  80.    info.offset = offset;
  81.    info.pos = 0;
  82.    info.localpos = 0;
  83.    copy_info(&info, "nIBM ServeRAID General Information:nn");
  84.    if ((le32_to_cpu(ha->nvram->signature) == IPS_NVRAM_P5_SIG) &&
  85.        (le16_to_cpu(ha->nvram->adapter_type) != 0))
  86.       copy_info(&info, "tController Type                   : %sn", ips_adapter_name[ha->ad_type-1]);
  87.    else
  88.       copy_info(&info, "tController Type                   : Unknownn");
  89.    if (ha->io_addr)
  90.       copy_info(&info, "tIO region                         : 0x%lx (%d bytes)n",
  91.                 ha->io_addr, ha->io_len);
  92.    if (ha->mem_addr) {
  93.       copy_info(&info, "tMemory region                     : 0x%lx (%d bytes)n",
  94.                 ha->mem_addr, ha->mem_len);
  95.       copy_info(&info, "tShared memory address             : 0x%lxn", ha->mem_ptr);
  96.    }
  97.    copy_info(&info, "tIRQ number                        : %dn", ha->irq);
  98.    if (le32_to_cpu(ha->nvram->signature) == IPS_NVRAM_P5_SIG)
  99.       copy_info(&info, "tBIOS Version                      : %c%c%c%c%c%c%c%cn",
  100.                 ha->nvram->bios_high[0], ha->nvram->bios_high[1],
  101.                 ha->nvram->bios_high[2], ha->nvram->bios_high[3],
  102.                 ha->nvram->bios_low[0], ha->nvram->bios_low[1],
  103.                 ha->nvram->bios_low[2], ha->nvram->bios_low[3]);
  104.    copy_info(&info, "tFirmware Version                  : %c%c%c%c%c%c%c%cn",
  105.              ha->enq->CodeBlkVersion[0], ha->enq->CodeBlkVersion[1],
  106.              ha->enq->CodeBlkVersion[2], ha->enq->CodeBlkVersion[3],
  107.              ha->enq->CodeBlkVersion[4], ha->enq->CodeBlkVersion[5],
  108.              ha->enq->CodeBlkVersion[6], ha->enq->CodeBlkVersion[7]);
  109.    copy_info(&info, "tBoot Block Version                : %c%c%c%c%c%c%c%cn",
  110.              ha->enq->BootBlkVersion[0], ha->enq->BootBlkVersion[1],
  111.              ha->enq->BootBlkVersion[2], ha->enq->BootBlkVersion[3],
  112.              ha->enq->BootBlkVersion[4], ha->enq->BootBlkVersion[5],
  113.              ha->enq->BootBlkVersion[6], ha->enq->BootBlkVersion[7]);
  114.    copy_info(&info, "tDriver Version                    : %s%sn",
  115.              IPS_VERSION_HIGH, IPS_VERSION_LOW);
  116.    copy_info(&info, "tMax Physical Devices              : %dn",
  117.              ha->enq->ucMaxPhysicalDevices);
  118.    copy_info(&info, "tMax Active Commands               : %dn",
  119.              ha->max_cmds);
  120.    copy_info(&info, "tCurrent Queued Commands           : %dn",
  121.              ha->scb_waitlist.count);
  122.    copy_info(&info, "tCurrent Active Commands           : %dn",
  123.              ha->scb_activelist.count - ha->num_ioctl);
  124.    copy_info(&info, "tCurrent Queued PT Commands        : %dn",
  125.              ha->copp_waitlist.count);
  126.    copy_info(&info, "tCurrent Active PT Commands        : %dn",
  127.              ha->num_ioctl);
  128.    copy_info(&info, "n");
  129.    return (info.localpos);
  130. }
  131. /****************************************************************************/
  132. /*                                                                          */
  133. /* Routine Name: copy_mem_info                                              */
  134. /*                                                                          */
  135. /* Routine Description:                                                     */
  136. /*                                                                          */
  137. /*   Copy data into an IPS_INFOSTR structure                                */
  138. /*                                                                          */
  139. /****************************************************************************/
  140. static void
  141. copy_mem_info(IPS_INFOSTR *info, char *data, int len) {
  142.    METHOD_TRACE("copy_mem_info", 1);
  143.    if (info->pos + len < info->offset) {
  144.       info->pos += len;
  145.       return;
  146.    }
  147.    if (info->pos < info->offset) {
  148.       data += (info->offset - info->pos);
  149.       len -= (info->offset - info->pos);
  150.       info->pos += (info->offset - info->pos);
  151.    }
  152.    if (info->localpos + len > info->length)
  153.       len = info->length - info->localpos;
  154.    if (len > 0) {
  155.       memcpy(info->buffer + info->localpos, data, len);
  156.       info->pos += len;
  157.       info->localpos += len;
  158.    }
  159. }
  160. /****************************************************************************/
  161. /*                                                                          */
  162. /* Routine Name: copy_info                                                  */
  163. /*                                                                          */
  164. /* Routine Description:                                                     */
  165. /*                                                                          */
  166. /*   printf style wrapper for an info structure                             */
  167. /*                                                                          */
  168. /****************************************************************************/
  169. static int
  170. copy_info(IPS_INFOSTR *info, char *fmt, ...) {
  171.    va_list args;
  172.    char buf[128];
  173.    int len;
  174.    METHOD_TRACE("copy_info", 1);
  175.    va_start(args, fmt);
  176.    len = vsprintf(buf, fmt, args);
  177.    va_end(args);
  178.    copy_mem_info(info, buf, len);
  179.    return (len);
  180. }
  181. /****************************************************************************/
  182. /*                                                                          */
  183. /* Routine Name: ips_identify_controller                                    */
  184. /*                                                                          */
  185. /* Routine Description:                                                     */
  186. /*                                                                          */
  187. /*   Identify this controller                                               */
  188. /*                                                                          */
  189. /****************************************************************************/
  190. static void
  191. ips_identify_controller(ips_ha_t *ha) {
  192.    METHOD_TRACE("ips_identify_controller", 1);
  193.    switch (ha->device_id) {
  194.    case IPS_DEVICEID_COPPERHEAD:
  195.       if (ha->revision_id <= IPS_REVID_SERVERAID) {
  196.          ha->ad_type = IPS_ADTYPE_SERVERAID;
  197.       } else if (ha->revision_id == IPS_REVID_SERVERAID2) {
  198.          ha->ad_type = IPS_ADTYPE_SERVERAID2;
  199.       } else if (ha->revision_id == IPS_REVID_NAVAJO) {
  200.          ha->ad_type = IPS_ADTYPE_NAVAJO;
  201.       } else if ((ha->revision_id == IPS_REVID_SERVERAID2) && (ha->slot_num == 0)) {
  202.          ha->ad_type = IPS_ADTYPE_KIOWA;
  203.       } else if ((ha->revision_id >= IPS_REVID_CLARINETP1) &&
  204.                  (ha->revision_id <= IPS_REVID_CLARINETP3)) {
  205.          if (ha->enq->ucMaxPhysicalDevices == 15)
  206.             ha->ad_type = IPS_ADTYPE_SERVERAID3L;
  207.          else
  208.             ha->ad_type = IPS_ADTYPE_SERVERAID3;
  209.       } else if ((ha->revision_id >= IPS_REVID_TROMBONE32) &&
  210.                  (ha->revision_id <= IPS_REVID_TROMBONE64)) {
  211.          ha->ad_type = IPS_ADTYPE_SERVERAID4H;
  212.       }
  213.       break;
  214.    case IPS_DEVICEID_MORPHEUS:
  215.       switch (ha->subdevice_id) {
  216.       case IPS_SUBDEVICEID_4L:
  217.          ha->ad_type = IPS_ADTYPE_SERVERAID4L;
  218.          break;
  219.       case IPS_SUBDEVICEID_4M:
  220.          ha->ad_type = IPS_ADTYPE_SERVERAID4M;
  221.          break;
  222.       case IPS_SUBDEVICEID_4MX:
  223.          ha->ad_type = IPS_ADTYPE_SERVERAID4MX;
  224.          break;
  225.       case IPS_SUBDEVICEID_4LX:
  226.          ha->ad_type = IPS_ADTYPE_SERVERAID4LX;
  227.          break;
  228.       case IPS_SUBDEVICEID_5I2:
  229.          ha->ad_type = IPS_ADTYPE_SERVERAID5I2;
  230.          break;
  231.       case IPS_SUBDEVICEID_5I1:
  232.          ha->ad_type = IPS_ADTYPE_SERVERAID5I1;
  233.          break;
  234.       }
  235.       break;
  236.    }
  237. }
  238. /****************************************************************************/
  239. /*                                                                          */
  240. /* Routine Name: ips_get_bios_version                                       */
  241. /*                                                                          */
  242. /* Routine Description:                                                     */
  243. /*                                                                          */
  244. /*   Get the BIOS revision number                                           */
  245. /*                                                                          */
  246. /****************************************************************************/
  247. static void
  248. ips_get_bios_version(ips_ha_t *ha, int intr) {
  249.    ips_scb_t *scb;
  250.    int        ret;
  251.    uint8_t   major;
  252.    uint8_t   minor;
  253.    uint8_t   subminor;
  254.    uint8_t  *buffer;
  255.    char       hexDigits[] = {'0','1','2','3','4','5','6','7','8','9','A','B','C','D','E','F'};
  256.    METHOD_TRACE("ips_get_bios_version", 1);
  257.    major = 0;
  258.    minor = 0;
  259.    strncpy(ha->bios_version, "       ?", 8);
  260.    if (ha->device_id == IPS_DEVICEID_COPPERHEAD) {
  261.       if (IPS_USE_MEMIO(ha)) {
  262.          /* Memory Mapped I/O */
  263.          /* test 1st byte */
  264.          writel(0, ha->mem_ptr + IPS_REG_FLAP);
  265.          if (ha->revision_id == IPS_REVID_TROMBONE64)
  266.             udelay(25); /* 25 us */
  267.          if (readb(ha->mem_ptr + IPS_REG_FLDP) != 0x55)
  268.             return;
  269.          writel(1, ha->mem_ptr + IPS_REG_FLAP);
  270.          if (ha->revision_id == IPS_REVID_TROMBONE64)
  271.             udelay(25); /* 25 us */
  272.          if (readb(ha->mem_ptr + IPS_REG_FLDP) != 0xAA)
  273.             return;
  274.          /* Get Major version */
  275.          writel(0x1FF, ha->mem_ptr + IPS_REG_FLAP);
  276.          if (ha->revision_id == IPS_REVID_TROMBONE64)
  277.             udelay(25); /* 25 us */
  278.          major = readb(ha->mem_ptr + IPS_REG_FLDP);
  279.          /* Get Minor version */
  280.          writel(0x1FE, ha->mem_ptr + IPS_REG_FLAP);
  281.          if (ha->revision_id == IPS_REVID_TROMBONE64)
  282.             udelay(25); /* 25 us */
  283.          minor = readb(ha->mem_ptr + IPS_REG_FLDP);
  284.          /* Get SubMinor version */
  285.          writel(0x1FD, ha->mem_ptr + IPS_REG_FLAP);
  286.          if (ha->revision_id == IPS_REVID_TROMBONE64)
  287.             udelay(25); /* 25 us */
  288.          subminor = readb(ha->mem_ptr + IPS_REG_FLDP);
  289.       } else {
  290.          /* Programmed I/O */
  291.          /* test 1st byte */
  292.          outl(0, ha->io_addr + IPS_REG_FLAP);
  293.          if (ha->revision_id == IPS_REVID_TROMBONE64)
  294.             udelay(25); /* 25 us */
  295.          if (inb(ha->io_addr + IPS_REG_FLDP) != 0x55)
  296.             return ;
  297.          outl(cpu_to_le32(1), ha->io_addr + IPS_REG_FLAP);
  298.          if (ha->revision_id == IPS_REVID_TROMBONE64)
  299.             udelay(25); /* 25 us */
  300.          if (inb(ha->io_addr + IPS_REG_FLDP) != 0xAA)
  301.             return ;
  302.          /* Get Major version */
  303.          outl(cpu_to_le32(0x1FF), ha->io_addr + IPS_REG_FLAP);
  304.          if (ha->revision_id == IPS_REVID_TROMBONE64)
  305.             udelay(25); /* 25 us */
  306.          major = inb(ha->io_addr + IPS_REG_FLDP);
  307.          /* Get Minor version */
  308.          outl(cpu_to_le32(0x1FE), ha->io_addr + IPS_REG_FLAP);
  309.          if (ha->revision_id == IPS_REVID_TROMBONE64)
  310.             udelay(25); /* 25 us */
  311.          minor = inb(ha->io_addr + IPS_REG_FLDP);
  312.          /* Get SubMinor version */
  313.          outl(cpu_to_le32(0x1FD), ha->io_addr + IPS_REG_FLAP);
  314.          if (ha->revision_id == IPS_REVID_TROMBONE64)
  315.             udelay(25); /* 25 us */
  316.          subminor = inb(ha->io_addr + IPS_REG_FLDP);
  317.       }
  318.    } else {
  319.       /* Morpheus Family - Send Command to the card */
  320.       buffer = kmalloc(0x1000, GFP_ATOMIC);
  321.       if (!buffer)
  322.          return;
  323.       memset(buffer, 0, 0x1000);
  324.       scb = &ha->scbs[ha->max_cmds-1];
  325.       ips_init_scb(ha, scb);
  326.       scb->timeout = ips_cmd_timeout;
  327.       scb->cdb[0] = IPS_CMD_RW_BIOSFW;
  328.       scb->cmd.flashfw.op_code = IPS_CMD_RW_BIOSFW;
  329.       scb->cmd.flashfw.command_id = IPS_COMMAND_ID(ha, scb);
  330.       scb->cmd.flashfw.type = 1;
  331.       scb->cmd.flashfw.direction = 0;
  332.       scb->cmd.flashfw.count = cpu_to_le32(0x800);
  333.       scb->cmd.flashfw.buffer_addr = cpu_to_le32(VIRT_TO_BUS(buffer));
  334.       scb->cmd.flashfw.total_packets = 1;
  335.       scb->cmd.flashfw.packet_num = 0;
  336.       /* issue the command */
  337.       if (((ret = ips_send_wait(ha, scb, ips_cmd_timeout, intr)) == IPS_FAILURE) ||
  338.           (ret == IPS_SUCCESS_IMM) ||
  339.           ((scb->basic_status & IPS_GSC_STATUS_MASK) > 1)) {
  340.          /* Error occurred */
  341.          kfree(buffer);
  342.          return;
  343.       }
  344.       if ((buffer[0xC0] == 0x55) && (buffer[0xC1] == 0xAA)) {
  345.          major    = buffer[0x1ff + 0xC0];  /* Offset 0x1ff after the header (0xc0) */
  346.          minor    = buffer[0x1fe + 0xC0];  /* Offset 0x1fe after the header (0xc0) */
  347.          subminor = buffer[0x1fd + 0xC0];  /* Offset 0x1fd after the header (0xc0) */
  348.       } else {
  349.          return;
  350.       }
  351.       kfree(buffer);
  352.    }
  353.    ha->bios_version[0] = hexDigits[(major & 0xF0) >> 4];
  354.    ha->bios_version[1] = '.';
  355.    ha->bios_version[2] = hexDigits[major & 0x0F];
  356.    ha->bios_version[3] = hexDigits[subminor];
  357.    ha->bios_version[4] = '.';
  358.    ha->bios_version[5] = hexDigits[(minor & 0xF0) >> 4];
  359.    ha->bios_version[6] = hexDigits[minor & 0x0F];
  360.    ha->bios_version[7] = 0;
  361. }
  362. /****************************************************************************/
  363. /*                                                                          */
  364. /* Routine Name: ips_hainit                                                 */
  365. /*                                                                          */
  366. /* Routine Description:                                                     */
  367. /*                                                                          */
  368. /*   Initialize the controller                                              */
  369. /*                                                                          */
  370. /* NOTE: Assumes to be called from with a lock                              */
  371. /*                                                                          */
  372. /****************************************************************************/
  373. static int
  374. ips_hainit(ips_ha_t *ha) {
  375.    int            i;
  376.    struct timeval tv;
  377.    METHOD_TRACE("ips_hainit", 1);
  378.    if (!ha)
  379.       return (0);
  380.    if (ha->func.statinit)
  381.       (*ha->func.statinit)(ha);
  382.    if (ha->func.enableint)
  383.       (*ha->func.enableint)(ha);
  384.    /* Send FFDC */
  385.    ha->reset_count = 1;
  386.    do_gettimeofday(&tv);
  387.    ha->last_ffdc = tv.tv_sec;
  388.    ips_ffdc_reset(ha, IPS_INTR_IORL);
  389.    if (!ips_read_config(ha, IPS_INTR_IORL)) {
  390.       printk(KERN_WARNING "(%s%d) unable to read config from controller.n",
  391.              ips_name, ha->host_num);
  392.       return (0);
  393.    } /* end if */
  394.    if (!ips_read_adapter_status(ha, IPS_INTR_IORL)) {
  395.       printk(KERN_WARNING "(%s%d) unable to read controller status.n",
  396.              ips_name, ha->host_num);
  397.       return (0);
  398.    }
  399.    /* Identify this controller */
  400.    ips_identify_controller(ha);
  401.    if (!ips_read_subsystem_parameters(ha, IPS_INTR_IORL)) {
  402.       printk(KERN_WARNING "(%s%d) unable to read subsystem parameters.n",
  403.              ips_name, ha->host_num);
  404.       return (0);
  405.    }
  406.    /* write nvram user page 5 */
  407.    if (!ips_write_driver_status(ha, IPS_INTR_IORL)) {
  408.       printk(KERN_WARNING "(%s%d) unable to write driver info to controller.n",
  409.              ips_name, ha->host_num);
  410.       return (0);
  411.    }
  412.    /* set limits on SID, LUN, BUS */
  413.    ha->ntargets = IPS_MAX_TARGETS + 1;
  414.    ha->nlun = 1;
  415.    ha->nbus = (ha->enq->ucMaxPhysicalDevices / IPS_MAX_TARGETS) + 1;
  416.    switch (ha->conf->logical_drive[0].ucStripeSize) {
  417.    case 4:
  418.       ha->max_xfer = 0x10000;
  419.       break;
  420.    case 5:
  421.       ha->max_xfer = 0x20000;
  422.       break;
  423.    case 6:
  424.       ha->max_xfer = 0x40000;
  425.       break;
  426.    case 7:
  427.    default:
  428.       ha->max_xfer = 0x80000;
  429.       break;
  430.    }
  431.    /* setup max concurrent commands */
  432.    if (le32_to_cpu(ha->subsys->param[4]) & 0x1) {
  433.       /* Use the new method */
  434.       ha->max_cmds = ha->enq->ucConcurrentCmdCount;
  435.    } else {
  436.       /* use the old method */
  437.       switch (ha->conf->logical_drive[0].ucStripeSize) {
  438.       case 4:
  439.          ha->max_cmds = 32;
  440.          break;
  441.       case 5:
  442.          ha->max_cmds = 16;
  443.          break;
  444.       case 6:
  445.          ha->max_cmds = 8;
  446.          break;
  447.       case 7:
  448.       default:
  449.          ha->max_cmds = 4;
  450.          break;
  451.       }
  452.    }
  453.    /* Limit the Active Commands on a Lite Adapter */
  454.    if ((ha->ad_type == IPS_ADTYPE_SERVERAID3L) ||
  455.        (ha->ad_type == IPS_ADTYPE_SERVERAID4L) ||
  456.        (ha->ad_type == IPS_ADTYPE_SERVERAID4LX)) {
  457.       if ((ha->max_cmds > MaxLiteCmds) && (MaxLiteCmds)) 
  458.          ha->max_cmds = MaxLiteCmds;
  459.    }
  460.    /* set controller IDs */
  461.    ha->ha_id[0] = IPS_ADAPTER_ID;
  462.    for (i = 1; i < ha->nbus; i++) {
  463.       ha->ha_id[i] = ha->conf->init_id[i-1] & 0x1f;
  464.       ha->dcdb_active[i-1] = 0;
  465.    }
  466.    return (1);
  467. }
  468. /****************************************************************************/
  469. /*                                                                          */
  470. /* Routine Name: ips_next                                                   */
  471. /*                                                                          */
  472. /* Routine Description:                                                     */
  473. /*                                                                          */
  474. /*   Take the next command off the queue and send it to the controller      */
  475. /*                                                                          */
  476. /****************************************************************************/
  477. static void
  478. ips_next(ips_ha_t *ha, int intr) {
  479.    ips_scb_t            *scb;
  480.    Scsi_Cmnd            *SC;
  481.    Scsi_Cmnd            *p;
  482.    Scsi_Cmnd            *q;
  483.    ips_copp_wait_item_t *item;
  484.    int                   ret;
  485.    unsigned long         cpu_flags;
  486.    unsigned long         cpu_flags2 = 0;
  487.    METHOD_TRACE("ips_next", 1);
  488.    if (!ha)
  489.       return ;
  490.    /*
  491.     * Block access to the queue function so
  492.     * this command won't time out
  493.     */
  494.    if (intr == IPS_INTR_ON) 
  495.        spin_lock_irqsave(&io_request_lock, cpu_flags2);
  496.    
  497.    if ((ha->subsys->param[3] & 0x300000)  && ( ha->scb_activelist.count == 0 )) {
  498.       struct timeval tv;
  499.       do_gettimeofday(&tv);
  500.       IPS_HA_LOCK(cpu_flags);
  501.       if (tv.tv_sec - ha->last_ffdc > IPS_SECS_8HOURS) {
  502.          ha->last_ffdc = tv.tv_sec;
  503.          IPS_HA_UNLOCK(cpu_flags);
  504.          ips_ffdc_time(ha);
  505.       } else {
  506.          IPS_HA_UNLOCK(cpu_flags);
  507.       }
  508.    }
  509.    if (intr == IPS_INTR_ON)
  510.        spin_unlock_irqrestore(&io_request_lock, cpu_flags2);
  511.    /*
  512.     * Send passthru commands
  513.     * These have priority over normal I/O
  514.     * but shouldn't affect performance too much
  515.     * since we limit the number that can be active
  516.     * on the card at any one time
  517.     */
  518.    IPS_HA_LOCK(cpu_flags);
  519.    IPS_QUEUE_LOCK(&ha->copp_waitlist);
  520.    while ((ha->num_ioctl < IPS_MAX_IOCTL) &&
  521.           (ha->copp_waitlist.head) &&
  522.           (scb = ips_getscb(ha))) {
  523.       IPS_QUEUE_UNLOCK(&ha->copp_waitlist);
  524.       item = ips_removeq_copp_head(&ha->copp_waitlist);
  525.       ha->num_ioctl++;
  526.       IPS_HA_UNLOCK(cpu_flags);
  527.       scb->scsi_cmd = item->scsi_cmd;
  528.       scb->sem = item->sem;
  529.       kfree(item);
  530.       ret = ips_make_passthru(ha, scb->scsi_cmd, scb, intr);
  531.       switch (ret) {
  532.       case IPS_FAILURE:
  533.          if (scb->scsi_cmd) {
  534.             scb->scsi_cmd->result = DID_ERROR << 16;
  535.             scb->scsi_cmd->scsi_done(scb->scsi_cmd);
  536.          }
  537.          ips_freescb(ha, scb);
  538.          break;
  539.       case IPS_SUCCESS_IMM:
  540.          if (scb->scsi_cmd) {
  541.             scb->scsi_cmd->result = DID_OK << 16;
  542.             scb->scsi_cmd->scsi_done(scb->scsi_cmd);
  543.          }
  544.          ips_freescb(ha, scb);
  545.          break;
  546.       default:
  547.          break;
  548.       } /* end case */
  549.       if (ret != IPS_SUCCESS) {
  550.          IPS_HA_LOCK(cpu_flags);
  551.          IPS_QUEUE_LOCK(&ha->copp_waitlist);
  552.          ha->num_ioctl--;
  553.          continue;
  554.      }
  555.       ret = ips_send_cmd(ha, scb);
  556.       if (ret == IPS_SUCCESS) 
  557.          ips_putq_scb_head(&ha->scb_activelist, scb);
  558.       else
  559.          ha->num_ioctl--;
  560.       switch(ret) {
  561.       case IPS_FAILURE:
  562.          if (scb->scsi_cmd) {
  563.             scb->scsi_cmd->result = DID_ERROR << 16;
  564.          }
  565.          ips_freescb(ha, scb);
  566.          break;
  567.       case IPS_SUCCESS_IMM:
  568.          ips_freescb(ha, scb);
  569.          break;
  570.       default:
  571.          break;
  572.       } /* end case */
  573.       IPS_HA_LOCK(cpu_flags);
  574.       IPS_QUEUE_LOCK(&ha->copp_waitlist);
  575.    }
  576.    IPS_QUEUE_UNLOCK(&ha->copp_waitlist);
  577.    IPS_HA_UNLOCK(cpu_flags);
  578.    /*
  579.     * Send "Normal" I/O commands
  580.     */
  581.    IPS_HA_LOCK(cpu_flags);
  582.    IPS_QUEUE_LOCK(&ha->scb_waitlist);
  583.    p = ha->scb_waitlist.head;
  584.    IPS_QUEUE_UNLOCK(&ha->scb_waitlist);
  585.    while ((p) && (scb = ips_getscb(ha))) {
  586.       if ((p->channel > 0) && (ha->dcdb_active[p->channel-1] & (1 << p->target))) {
  587.          ips_freescb(ha, scb);
  588.          p = (Scsi_Cmnd *) p->host_scribble;
  589.          continue;
  590.       }
  591.       q = p;
  592.       SC = ips_removeq_wait(&ha->scb_waitlist, q);
  593.       if (SC == NULL)                    /* Should never happen, but good to check anyway */
  594.          continue;
  595.       IPS_HA_UNLOCK(cpu_flags);          /* Unlock HA after command is taken off queue */
  596.       SC->result = DID_OK;
  597.       SC->host_scribble = NULL;
  598.       memset(SC->sense_buffer, 0, sizeof(SC->sense_buffer));
  599.       scb->target_id = SC->target;
  600.       scb->lun = SC->lun;
  601.       scb->bus = SC->channel;
  602.       scb->scsi_cmd = SC;
  603.       scb->breakup = 0;
  604.       scb->data_len = 0;
  605.       scb->callback = ipsintr_done;
  606.       scb->timeout = ips_cmd_timeout;
  607.       memset(&scb->cmd, 0, 16);
  608.       /* copy in the CDB */
  609.       memcpy(scb->cdb, SC->cmnd, SC->cmd_len);
  610.       /* Now handle the data buffer */
  611.       if (SC->use_sg) {
  612.          struct scatterlist *sg;
  613.          int                 i;
  614.          sg = SC->request_buffer;
  615.          scb->sg_count = pci_map_sg(ha->pcidev, sg, SC->use_sg, PCI_DMA_BIDIRECTIONAL);
  616.          scb->flags |= IPS_SCB_MAP_SG;
  617.          if (scb->sg_count == 1) {
  618.             if (sg_dma_len(sg) > ha->max_xfer) {
  619.              scb->breakup = 1;
  620.                scb->data_len = ha->max_xfer;
  621.             } else
  622.                scb->data_len = sg_dma_len(sg);
  623.             scb->dcdb.transfer_length = scb->data_len;
  624.             scb->data_busaddr = sg_dma_address(sg);
  625.             scb->sg_len = 0;
  626.          } else {
  627.             /* Check for the first Element being bigger than MAX_XFER */
  628.             if (sg_dma_len(&sg[0]) > ha->max_xfer) {
  629.                scb->sg_list[0].address = cpu_to_le32(sg_dma_address(&sg[0]));
  630.                scb->sg_list[0].length = ha->max_xfer;
  631.                scb->data_len = ha->max_xfer;
  632.                scb->breakup = 0; 
  633.                scb->sg_break=1;  
  634.                scb->sg_len = 1;
  635.             } else {
  636.                for (i = 0; i < scb->sg_count; i++) {
  637.                   scb->sg_list[i].address = cpu_to_le32(sg_dma_address(&sg[i]));
  638.                   scb->sg_list[i].length = cpu_to_le32(sg_dma_len(&sg[i]));
  639.             
  640.                   if (scb->data_len + sg_dma_len(&sg[i]) > ha->max_xfer) {
  641.                      /*
  642.                       * Data Breakup required
  643.                       */
  644.                      scb->breakup = i;
  645.                      break;
  646.                   }
  647.                
  648.                   scb->data_len += sg_dma_len(&sg[i]);
  649.                }
  650.                if (!scb->breakup)
  651.                   scb->sg_len = scb->sg_count;
  652.                else
  653.                   scb->sg_len = scb->breakup;
  654.             }
  655.             scb->dcdb.transfer_length = scb->data_len;
  656.             scb->data_busaddr = scb->sg_busaddr;
  657.          }
  658.       } else {
  659.          if (SC->request_bufflen) {
  660.             if (SC->request_bufflen > ha->max_xfer) {
  661.                /*
  662.                 * Data breakup required
  663.                 */
  664.                scb->breakup = 1;
  665.                scb->data_len = ha->max_xfer;
  666.             } else {
  667.                scb->data_len = SC->request_bufflen;
  668.             }
  669.             scb->dcdb.transfer_length = scb->data_len;
  670.             scb->data_busaddr = pci_map_single(ha->pcidev, SC->request_buffer,
  671.                                                scb->data_len, PCI_DMA_BIDIRECTIONAL);
  672.             scb->flags |= IPS_SCB_MAP_SINGLE;
  673.             scb->sg_len = 0;
  674.          } else {
  675.             scb->data_busaddr = 0L;
  676.             scb->sg_len = 0;
  677.             scb->data_len = 0;
  678.             scb->dcdb.transfer_length = 0;
  679.          }
  680.       }
  681.       scb->dcdb.cmd_attribute = ips_command_direction[scb->scsi_cmd->cmnd[0]];
  682.       if (!scb->dcdb.cmd_attribute & 0x3)
  683.          scb->dcdb.transfer_length = 0;
  684.       if (scb->data_len >= IPS_MAX_XFER) {
  685.          scb->dcdb.cmd_attribute |= IPS_TRANSFER64K;
  686.          scb->dcdb.transfer_length = 0;
  687.       }
  688.       ret = ips_send_cmd(ha, scb);
  689.       if (ret == IPS_SUCCESS)
  690.          ips_putq_scb_head(&ha->scb_activelist, scb);
  691.       switch(ret) {
  692.       case IPS_FAILURE:
  693.          if (scb->scsi_cmd) {
  694.             scb->scsi_cmd->result = DID_ERROR << 16;
  695.             scb->scsi_cmd->scsi_done(scb->scsi_cmd);
  696.          }
  697.          if (scb->bus)
  698.             ha->dcdb_active[scb->bus-1] &= ~(1 << scb->target_id);
  699.          ips_freescb(ha, scb);
  700.          break;
  701.       case IPS_SUCCESS_IMM:
  702.          if (scb->scsi_cmd)
  703.             scb->scsi_cmd->scsi_done(scb->scsi_cmd);
  704.          if (scb->bus)
  705.             ha->dcdb_active[scb->bus-1] &= ~(1 << scb->target_id);
  706.          ips_freescb(ha, scb);
  707.          break;
  708.       default:
  709.          break;
  710.       } /* end case */
  711.       p = (Scsi_Cmnd *) p->host_scribble;
  712.       IPS_HA_LOCK(cpu_flags);
  713.    } /* end while */
  714.    IPS_HA_UNLOCK(cpu_flags);
  715. }
  716. /****************************************************************************/
  717. /*                                                                          */
  718. /* Routine Name: ips_putq_scb_head                                          */
  719. /*                                                                          */
  720. /* Routine Description:                                                     */
  721. /*                                                                          */
  722. /*   Add an item to the head of the queue                                   */
  723. /*                                                                          */
  724. /* ASSUMED to be called from within a lock                                  */
  725. /*                                                                          */
  726. /****************************************************************************/
  727. static inline void
  728. ips_putq_scb_head(ips_scb_queue_t *queue, ips_scb_t *item) {
  729.    METHOD_TRACE("ips_putq_scb_head", 1);
  730.    if (!item)
  731.       return ;
  732.    IPS_QUEUE_LOCK(queue);
  733.    item->q_next = queue->head;
  734.    queue->head = item;
  735.    if (!queue->tail)
  736.       queue->tail = item;
  737.    queue->count++;
  738.    IPS_QUEUE_UNLOCK(queue);
  739. }
  740. /****************************************************************************/
  741. /*                                                                          */
  742. /* Routine Name: ips_putq_scb_tail                                          */
  743. /*                                                                          */
  744. /* Routine Description:                                                     */
  745. /*                                                                          */
  746. /*   Add an item to the tail of the queue                                   */
  747. /*                                                                          */
  748. /* ASSUMED to be called from within a lock                                  */
  749. /*                                                                          */
  750. /****************************************************************************/
  751. static inline void
  752. ips_putq_scb_tail(ips_scb_queue_t *queue, ips_scb_t *item) {
  753.    METHOD_TRACE("ips_putq_scb_tail", 1);
  754.    if (!item)
  755.       return ;
  756.    IPS_QUEUE_LOCK(queue);
  757.    item->q_next = NULL;
  758.    if (queue->tail)
  759.       queue->tail->q_next = item;
  760.    queue->tail = item;
  761.    if (!queue->head)
  762.       queue->head = item;
  763.    queue->count++;
  764.    IPS_QUEUE_UNLOCK(queue);
  765. }
  766. /****************************************************************************/
  767. /*                                                                          */
  768. /* Routine Name: ips_removeq_scb_head                                       */
  769. /*                                                                          */
  770. /* Routine Description:                                                     */
  771. /*                                                                          */
  772. /*   Remove the head of the queue                                           */
  773. /*                                                                          */
  774. /* ASSUMED to be called from within a lock                                  */
  775. /*                                                                          */
  776. /****************************************************************************/
  777. static inline ips_scb_t *
  778. ips_removeq_scb_head(ips_scb_queue_t *queue) {
  779.    ips_scb_t  *item;
  780.    METHOD_TRACE("ips_removeq_scb_head", 1);
  781.    IPS_QUEUE_LOCK(queue);
  782.    item = queue->head;
  783.    if (!item) {
  784.       IPS_QUEUE_UNLOCK(queue);
  785.       return (NULL);
  786.    }
  787.    queue->head = item->q_next;
  788.    item->q_next = NULL;
  789.    if (queue->tail == item)
  790.       queue->tail = NULL;
  791.    queue->count--;
  792.    IPS_QUEUE_UNLOCK(queue);
  793.    return (item);
  794. }
  795. /****************************************************************************/
  796. /*                                                                          */
  797. /* Routine Name: ips_removeq_scb                                            */
  798. /*                                                                          */
  799. /* Routine Description:                                                     */
  800. /*                                                                          */
  801. /*   Remove an item from a queue                                            */
  802. /*                                                                          */
  803. /* ASSUMED to be called from within a lock                                  */
  804. /*                                                                          */
  805. /****************************************************************************/
  806. static inline ips_scb_t *
  807. ips_removeq_scb(ips_scb_queue_t *queue, ips_scb_t *item) {
  808.    ips_scb_t  *p;
  809.    METHOD_TRACE("ips_removeq_scb", 1);
  810.    if (!item)
  811.       return (NULL);
  812.    IPS_QUEUE_LOCK(queue);
  813.    if (item == queue->head) {
  814.       IPS_QUEUE_UNLOCK(queue);
  815.       return (ips_removeq_scb_head(queue));
  816.    }
  817.    p = queue->head;
  818.    while ((p) && (item != p->q_next))
  819.       p = p->q_next;
  820.    if (p) {
  821.       /* found a match */
  822.       p->q_next = item->q_next;
  823.       if (!item->q_next)
  824.          queue->tail = p;
  825.       item->q_next = NULL;
  826.       queue->count--;
  827.       IPS_QUEUE_UNLOCK(queue);
  828.       return (item);
  829.    }
  830.    IPS_QUEUE_UNLOCK(queue);
  831.    return (NULL);
  832. }
  833. /****************************************************************************/
  834. /*                                                                          */
  835. /* Routine Name: ips_putq_wait_head                                         */
  836. /*                                                                          */
  837. /* Routine Description:                                                     */
  838. /*                                                                          */
  839. /*   Add an item to the head of the queue                                   */
  840. /*                                                                          */
  841. /* ASSUMED to be called from within a lock                                  */
  842. /*                                                                          */
  843. /****************************************************************************/
  844. static inline void
  845. ips_putq_wait_head(ips_wait_queue_t *queue, Scsi_Cmnd *item) {
  846.    METHOD_TRACE("ips_putq_wait_head", 1);
  847.    if (!item)
  848.       return ;
  849.    IPS_QUEUE_LOCK(queue);
  850.    item->host_scribble = (char *) queue->head;
  851.    queue->head = item;
  852.    if (!queue->tail)
  853.       queue->tail = item;
  854.    queue->count++;
  855.    IPS_QUEUE_UNLOCK(queue);
  856. }
  857. /****************************************************************************/
  858. /*                                                                          */
  859. /* Routine Name: ips_putq_wait_tail                                         */
  860. /*                                                                          */
  861. /* Routine Description:                                                     */
  862. /*                                                                          */
  863. /*   Add an item to the tail of the queue                                   */
  864. /*                                                                          */
  865. /* ASSUMED to be called from within a lock                                  */
  866. /*                                                                          */
  867. /****************************************************************************/
  868. static inline void
  869. ips_putq_wait_tail(ips_wait_queue_t *queue, Scsi_Cmnd *item) {
  870.    METHOD_TRACE("ips_putq_wait_tail", 1);
  871.    if (!item)
  872.       return ;
  873.    IPS_QUEUE_LOCK(queue);
  874.    item->host_scribble = NULL;
  875.    if (queue->tail)
  876.       queue->tail->host_scribble = (char *)item;
  877.    queue->tail = item;
  878.    if (!queue->head)
  879.       queue->head = item;
  880.    queue->count++;
  881.    IPS_QUEUE_UNLOCK(queue);
  882. }
  883. /****************************************************************************/
  884. /*                                                                          */
  885. /* Routine Name: ips_removeq_wait_head                                      */
  886. /*                                                                          */
  887. /* Routine Description:                                                     */
  888. /*                                                                          */
  889. /*   Remove the head of the queue                                           */
  890. /*                                                                          */
  891. /* ASSUMED to be called from within a lock                                  */
  892. /*                                                                          */
  893. /****************************************************************************/
  894. static inline Scsi_Cmnd *
  895. ips_removeq_wait_head(ips_wait_queue_t *queue) {
  896.    Scsi_Cmnd  *item;
  897.    METHOD_TRACE("ips_removeq_wait_head", 1);
  898.    IPS_QUEUE_LOCK(queue);
  899.    item = queue->head;
  900.    if (!item) {
  901.       IPS_QUEUE_UNLOCK(queue);
  902.       return (NULL);
  903.    }
  904.    queue->head = (Scsi_Cmnd *) item->host_scribble;
  905.    item->host_scribble = NULL;
  906.    if (queue->tail == item)
  907.       queue->tail = NULL;
  908.    queue->count--;
  909.    IPS_QUEUE_UNLOCK(queue);
  910.    return (item);
  911. }
  912. /****************************************************************************/
  913. /*                                                                          */
  914. /* Routine Name: ips_removeq_wait                                           */
  915. /*                                                                          */
  916. /* Routine Description:                                                     */
  917. /*                                                                          */
  918. /*   Remove an item from a queue                                            */
  919. /*                                                                          */
  920. /* ASSUMED to be called from within a lock                                  */
  921. /*                                                                          */
  922. /****************************************************************************/
  923. static inline Scsi_Cmnd *
  924. ips_removeq_wait(ips_wait_queue_t *queue, Scsi_Cmnd *item) {
  925.    Scsi_Cmnd  *p;
  926.    METHOD_TRACE("ips_removeq_wait", 1);
  927.    if (!item)
  928.       return (NULL);
  929.    IPS_QUEUE_LOCK(queue);
  930.    if (item == queue->head) {
  931.       IPS_QUEUE_UNLOCK(queue);
  932.       return (ips_removeq_wait_head(queue));
  933.    }
  934.    p = queue->head;
  935.    while ((p) && (item != (Scsi_Cmnd *) p->host_scribble))
  936.       p = (Scsi_Cmnd *) p->host_scribble;
  937.    if (p) {
  938.       /* found a match */
  939.       p->host_scribble = item->host_scribble;
  940.       if (!item->host_scribble)
  941.          queue->tail = p;
  942.       item->host_scribble = NULL;
  943.       queue->count--;
  944.       IPS_QUEUE_UNLOCK(queue);
  945.       return (item);
  946.    }
  947.    IPS_QUEUE_UNLOCK(queue);
  948.    return (NULL);
  949. }
  950. /****************************************************************************/
  951. /*                                                                          */
  952. /* Routine Name: ips_putq_copp_head                                         */
  953. /*                                                                          */
  954. /* Routine Description:                                                     */
  955. /*                                                                          */
  956. /*   Add an item to the head of the queue                                   */
  957. /*                                                                          */
  958. /* ASSUMED to be called from within a lock                                  */
  959. /*                                                                          */
  960. /****************************************************************************/
  961. static inline void
  962. ips_putq_copp_head(ips_copp_queue_t *queue, ips_copp_wait_item_t *item) {
  963.    METHOD_TRACE("ips_putq_copp_head", 1);
  964.    if (!item)
  965.       return ;
  966.    IPS_QUEUE_LOCK(queue);
  967.    item->next = queue->head;
  968.    queue->head = item;
  969.    if (!queue->tail)
  970.       queue->tail = item;
  971.    queue->count++;
  972.    IPS_QUEUE_UNLOCK(queue);
  973. }
  974. /****************************************************************************/
  975. /*                                                                          */
  976. /* Routine Name: ips_putq_copp_tail                                         */
  977. /*                                                                          */
  978. /* Routine Description:                                                     */
  979. /*                                                                          */
  980. /*   Add an item to the tail of the queue                                   */
  981. /*                                                                          */
  982. /* ASSUMED to be called from within a lock                                  */
  983. /*                                                                          */
  984. /****************************************************************************/
  985. static inline void
  986. ips_putq_copp_tail(ips_copp_queue_t *queue, ips_copp_wait_item_t *item) {
  987.    METHOD_TRACE("ips_putq_copp_tail", 1);
  988.    if (!item)
  989.       return ;
  990.    IPS_QUEUE_LOCK(queue);
  991.    item->next = NULL;
  992.    if (queue->tail)
  993.       queue->tail->next = item;
  994.    queue->tail = item;
  995.    if (!queue->head)
  996.       queue->head = item;
  997.    queue->count++;
  998.    IPS_QUEUE_UNLOCK(queue);
  999. }
  1000. /****************************************************************************/
  1001. /*                                                                          */
  1002. /* Routine Name: ips_removeq_copp_head                                      */
  1003. /*                                                                          */
  1004. /* Routine Description:                                                     */
  1005. /*                                                                          */
  1006. /*   Remove the head of the queue                                           */
  1007. /*                                                                          */
  1008. /* ASSUMED to be called from within a lock                                  */
  1009. /*                                                                          */
  1010. /****************************************************************************/
  1011. static inline ips_copp_wait_item_t *
  1012. ips_removeq_copp_head(ips_copp_queue_t *queue) {
  1013.    ips_copp_wait_item_t *item;
  1014.    METHOD_TRACE("ips_removeq_copp_head", 1);
  1015.    IPS_QUEUE_LOCK(queue);
  1016.    item = queue->head;
  1017.    if (!item) {
  1018.       IPS_QUEUE_UNLOCK(queue);
  1019.       return (NULL);
  1020.    }
  1021.    queue->head = item->next;
  1022.    item->next = NULL;
  1023.    if (queue->tail == item)
  1024.       queue->tail = NULL;
  1025.    queue->count--;
  1026.    IPS_QUEUE_UNLOCK(queue);
  1027.    return (item);
  1028. }
  1029. /****************************************************************************/
  1030. /*                                                                          */
  1031. /* Routine Name: ips_removeq_copp                                           */
  1032. /*                                                                          */
  1033. /* Routine Description:                                                     */
  1034. /*                                                                          */
  1035. /*   Remove an item from a queue                                            */
  1036. /*                                                                          */
  1037. /* ASSUMED to be called from within a lock                                  */
  1038. /*                                                                          */
  1039. /****************************************************************************/
  1040. static inline ips_copp_wait_item_t *
  1041. ips_removeq_copp(ips_copp_queue_t *queue, ips_copp_wait_item_t *item) {
  1042.    ips_copp_wait_item_t *p;
  1043.    METHOD_TRACE("ips_removeq_copp", 1);
  1044.    if (!item)
  1045.       return (NULL);
  1046.    IPS_QUEUE_LOCK(queue);
  1047.    if (item == queue->head) {
  1048.       IPS_QUEUE_UNLOCK(queue);
  1049.       return (ips_removeq_copp_head(queue));
  1050.    }
  1051.    p = queue->head;
  1052.    while ((p) && (item != p->next))
  1053.       p = p->next;
  1054.    if (p) {
  1055.       /* found a match */
  1056.       p->next = item->next;
  1057.       if (!item->next)
  1058.          queue->tail = p;
  1059.       item->next = NULL;
  1060.       queue->count--;
  1061.       IPS_QUEUE_UNLOCK(queue);
  1062.       return (item);
  1063.    }
  1064.    IPS_QUEUE_UNLOCK(queue);
  1065.    return (NULL);
  1066. }
  1067. /****************************************************************************/
  1068. /*                                                                          */
  1069. /* Routine Name: ipsintr_blocking                                           */
  1070. /*                                                                          */
  1071. /* Routine Description:                                                     */
  1072. /*                                                                          */
  1073. /*   Finalize an interrupt for internal commands                            */
  1074. /*                                                                          */
  1075. /****************************************************************************/
  1076. static void
  1077. ipsintr_blocking(ips_ha_t *ha, ips_scb_t *scb) {
  1078.    METHOD_TRACE("ipsintr_blocking", 2);
  1079.    if ((ha->waitflag == TRUE) &&
  1080.        (ha->cmd_in_progress == scb->cdb[0])) {
  1081.       ha->waitflag = FALSE;
  1082.       return ;
  1083.    }
  1084. }
  1085. /****************************************************************************/
  1086. /*                                                                          */
  1087. /* Routine Name: ipsintr_done                                               */
  1088. /*                                                                          */
  1089. /* Routine Description:                                                     */
  1090. /*                                                                          */
  1091. /*   Finalize an interrupt for non-internal commands                        */
  1092. /*                                                                          */
  1093. /****************************************************************************/
  1094. static void
  1095. ipsintr_done(ips_ha_t *ha, ips_scb_t *scb) {
  1096.    METHOD_TRACE("ipsintr_done", 2);
  1097.    if (!scb) {
  1098.       printk(KERN_WARNING "(%s%d) Spurious interrupt; scb NULL.n",
  1099.              ips_name, ha->host_num);
  1100.       return ;
  1101.    }
  1102.    if (scb->scsi_cmd == NULL) {
  1103.       /* unexpected interrupt */
  1104.       printk(KERN_WARNING "(%s%d) Spurious interrupt; scsi_cmd not set.n",
  1105.              ips_name, ha->host_num);
  1106.       return;
  1107.    }
  1108.    ips_done(ha, scb);
  1109. }
  1110. /****************************************************************************/
  1111. /*                                                                          */
  1112. /* Routine Name: ips_done                                                   */
  1113. /*                                                                          */
  1114. /* Routine Description:                                                     */
  1115. /*                                                                          */
  1116. /*   Do housekeeping on completed commands                                  */
  1117. /*                                                                          */
  1118. /****************************************************************************/
  1119. static void
  1120. ips_done(ips_ha_t *ha, ips_scb_t *scb) {
  1121.    int ret;
  1122.    unsigned long cpu_flags;
  1123.    METHOD_TRACE("ips_done", 1);
  1124.    if (!scb)
  1125.       return ;
  1126.    if ((scb->scsi_cmd) && (ips_is_passthru(scb->scsi_cmd))) {
  1127.       ips_cleanup_passthru(ha, scb);
  1128.       IPS_HA_LOCK(cpu_flags);
  1129.       ha->num_ioctl--;
  1130.       IPS_HA_UNLOCK(cpu_flags);
  1131.    } else {
  1132.       /*
  1133.        * Check to see if this command had too much
  1134.        * data and had to be broke up.  If so, queue
  1135.        * the rest of the data and continue.
  1136.        */
  1137.       if ((scb->breakup) || (scb->sg_break)) {
  1138.          /* we had a data breakup */
  1139.          uint8_t bk_save;
  1140.          bk_save = scb->breakup;
  1141.          scb->breakup = 0;
  1142.          mod_timer(&scb->scsi_cmd->eh_timeout, jiffies + 120 * HZ);
  1143.          if (scb->sg_count) {
  1144.             /* S/G request */
  1145.             struct scatterlist *sg;
  1146.             int                 i;
  1147.             sg = scb->scsi_cmd->request_buffer;
  1148.             if (scb->sg_count == 1) {
  1149.                if (sg_dma_len(sg) - (bk_save * ha->max_xfer) > ha->max_xfer) {
  1150.                   /* Further breakup required */
  1151.                   scb->data_len = ha->max_xfer;
  1152.                   scb->data_busaddr = sg_dma_address(sg) + (bk_save * ha->max_xfer);
  1153.                   scb->breakup = bk_save + 1;
  1154.                } else {
  1155.                   scb->data_len = sg_dma_len(sg) - (bk_save * ha->max_xfer);
  1156.                   scb->data_busaddr = sg_dma_address(sg) + (bk_save * ha->max_xfer);
  1157.                }
  1158.                scb->dcdb.transfer_length = scb->data_len;
  1159.                scb->sg_len = 0;
  1160.             } else {
  1161.                /* We're here because there was MORE than one s/g unit. */
  1162.             /* bk_save points to which sg unit to look at           */
  1163.             /* sg_break points to how far through this unit we are  */
  1164.             /* NOTE: We will not move from one sg to another here,  */
  1165.                /*    just finish the one we are in.  Not the most      */
  1166.                /*    efficient, but it keeps it from getting too hacky */
  1167.          /* IF sg_break is non-zero, then just work on this current sg piece, */
  1168.                /* pointed to by bk_save                                             */
  1169.                if (scb->sg_break) {
  1170.                   scb->sg_len = 1;
  1171.                   scb->sg_list[0].address = sg_dma_address(&sg[bk_save])
  1172.                                             + ha->max_xfer*scb->sg_break;
  1173.                   if (ha->max_xfer > sg_dma_len(&sg[bk_save]) - ha->max_xfer * scb->sg_break) 
  1174.                      scb->sg_list[0].length = sg_dma_len(&sg[bk_save]) - ha->max_xfer * scb->sg_break;
  1175.                   else 
  1176.                      scb->sg_list[0].length = ha->max_xfer;
  1177.                   scb->sg_break++;              /* MUST GO HERE for math below to work */
  1178.                   scb->data_len = scb->sg_list[0].length;;
  1179.                   if (sg_dma_len(&sg[bk_save]) <= ha->max_xfer * scb->sg_break ) {
  1180.                      scb->sg_break = 0;         /* No more work in this unit */
  1181.                      if (( bk_save + 1 ) >= scb->sg_count) 
  1182.                         scb->breakup = 0;
  1183.                      else
  1184.                         scb->breakup = bk_save + 1;
  1185.                   }
  1186.                } else {
  1187.          /* ( sg_break == 0 ), so this is our first look at a new sg piece */
  1188.                   if (sg_dma_len(&sg[bk_save]) > ha->max_xfer) {
  1189.                      scb->sg_list[0].address = sg_dma_address(&sg[bk_save]);
  1190.                      scb->sg_list[0].length = ha->max_xfer;
  1191.                      scb->breakup = bk_save;
  1192.                      scb->sg_break = 1;
  1193.                      scb->data_len = ha->max_xfer;
  1194.                      scb->sg_len = 1;
  1195.             } else {
  1196.          /* OK, the next sg is a short one, so loop until full */
  1197.                      scb->data_len = 0;
  1198.                      scb->sg_len = 0;
  1199.                      scb->sg_break = 0;
  1200.                      /*   We're only doing full units here */
  1201.                      for (i = bk_save; i < scb->sg_count; i++) {
  1202.                         scb->sg_list[i - bk_save].address = sg_dma_address(&sg[i]);
  1203.                         scb->sg_list[i - bk_save].length = cpu_to_le32(sg_dma_len(&sg[i]));
  1204.                         if (scb->data_len + sg_dma_len(&sg[i]) > ha->max_xfer) {
  1205.                            scb->breakup = i;  /* sneaky, if not more work, than breakup is 0 */
  1206.                            break;
  1207.                         }
  1208.                         scb->data_len += sg_dma_len(&sg[i]);
  1209.                         scb->sg_len++;           /* only if we didn't get too big */
  1210.             }
  1211.          }
  1212.          }
  1213.                /* Also, we need to be sure we don't queue work ( breakup != 0 )
  1214.                   if no more sg units for next time */
  1215.                scb->dcdb.transfer_length = scb->data_len;
  1216.                scb->data_busaddr = scb->sg_busaddr;
  1217.             }
  1218.                                               
  1219.          } else {
  1220.             /* Non S/G Request */
  1221.             pci_unmap_single(ha->pcidev, scb->data_busaddr, scb->data_len,
  1222.                              PCI_DMA_BIDIRECTIONAL);
  1223.             if ((scb->scsi_cmd->request_bufflen - (bk_save * ha->max_xfer)) > ha->max_xfer) {
  1224.                /* Further breakup required */
  1225.                scb->data_len = ha->max_xfer;
  1226.                scb->data_busaddr = pci_map_single(ha->pcidev, scb->scsi_cmd->request_buffer + (bk_save * ha->max_xfer), scb->data_len, PCI_DMA_BIDIRECTIONAL);
  1227.                scb->breakup = bk_save + 1;
  1228.             } else {
  1229.                scb->data_len = scb->scsi_cmd->request_bufflen - (bk_save * ha->max_xfer);
  1230.                scb->data_busaddr = pci_map_single(ha->pcidev, scb->scsi_cmd->request_buffer + (bk_save * ha->max_xfer), scb->data_len, PCI_DMA_BIDIRECTIONAL);
  1231.     }
  1232.             scb->dcdb.transfer_length = scb->data_len;
  1233.             scb->sg_len = 0;
  1234.          }
  1235.          scb->dcdb.cmd_attribute |= ips_command_direction[scb->scsi_cmd->cmnd[0]];
  1236.          if (!scb->dcdb.cmd_attribute & 0x3)
  1237.             scb->dcdb.transfer_length = 0;
  1238.          if (scb->data_len >= IPS_MAX_XFER) {
  1239.             scb->dcdb.cmd_attribute |= IPS_TRANSFER64K;
  1240.             scb->dcdb.transfer_length = 0;
  1241.          }
  1242.          ret = ips_send_cmd(ha, scb);
  1243.          switch(ret) {
  1244.          case IPS_FAILURE:
  1245.             if (scb->scsi_cmd) {
  1246.                scb->scsi_cmd->result = DID_ERROR << 16;
  1247.                scb->scsi_cmd->scsi_done(scb->scsi_cmd);
  1248.             }
  1249.             ips_freescb(ha, scb);
  1250.             break;
  1251.          case IPS_SUCCESS_IMM:
  1252.             if (scb->scsi_cmd) {
  1253.                scb->scsi_cmd->result = DID_ERROR << 16;
  1254.                scb->scsi_cmd->scsi_done(scb->scsi_cmd);
  1255.             }
  1256.             ips_freescb(ha, scb);
  1257.             break;
  1258.          default:
  1259.             break;
  1260.          } /* end case */
  1261.          return ;
  1262.       }
  1263.    } /* end if passthru */
  1264.    if (scb->bus) {
  1265.       IPS_HA_LOCK(cpu_flags);
  1266.       ha->dcdb_active[scb->bus-1] &= ~(1 << scb->target_id);
  1267.       IPS_HA_UNLOCK(cpu_flags);
  1268.    }
  1269.    scb->scsi_cmd->scsi_done(scb->scsi_cmd);
  1270.    ips_freescb(ha, scb);
  1271. }
  1272. /****************************************************************************/
  1273. /*                                                                          */
  1274. /* Routine Name: ips_map_status                                             */
  1275. /*                                                                          */
  1276. /* Routine Description:                                                     */
  1277. /*                                                                          */
  1278. /*   Map ServeRAID error codes to Linux Error Codes                         */
  1279. /*                                                                          */
  1280. /****************************************************************************/
  1281. static int
  1282. ips_map_status(ips_ha_t *ha, ips_scb_t *scb, ips_stat_t *sp) {
  1283.    int       errcode;
  1284.    int       device_error;
  1285.    uint32_t  transfer_len;
  1286.    IPS_DCDB_TABLE_TAPE *tapeDCDB;
  1287.    METHOD_TRACE("ips_map_status", 1);
  1288.    if (scb->bus) {
  1289.       DEBUG_VAR(2, "(%s%d) Physical device error (%d %d %d): %x %x, Sense Key: %x, ASC: %x, ASCQ: %x",
  1290.                 ips_name,
  1291.                 ha->host_num,
  1292.                 scb->scsi_cmd->channel,
  1293.                 scb->scsi_cmd->target,
  1294.                 scb->scsi_cmd->lun,
  1295.                 scb->basic_status,
  1296.                 scb->extended_status,
  1297.                 scb->extended_status == IPS_ERR_CKCOND ? scb->dcdb.sense_info[2] & 0xf : 0,
  1298.                 scb->extended_status == IPS_ERR_CKCOND ? scb->dcdb.sense_info[12] : 0,
  1299.                 scb->extended_status == IPS_ERR_CKCOND ? scb->dcdb.sense_info[13] : 0);
  1300.    }
  1301.    /* default driver error */
  1302.    errcode = DID_ERROR;
  1303.    device_error = 0;
  1304.    switch (scb->basic_status & IPS_GSC_STATUS_MASK) {
  1305.    case IPS_CMD_TIMEOUT:
  1306.       errcode = DID_TIME_OUT;
  1307.       break;
  1308.    case IPS_INVAL_OPCO:
  1309.    case IPS_INVAL_CMD_BLK:
  1310.    case IPS_INVAL_PARM_BLK:
  1311.    case IPS_LD_ERROR:
  1312.    case IPS_CMD_CMPLT_WERROR:
  1313.       break;
  1314.    case IPS_PHYS_DRV_ERROR:
  1315.       switch (scb->extended_status) {
  1316.       case IPS_ERR_SEL_TO:
  1317.          if (scb->bus)
  1318.             errcode = DID_NO_CONNECT;
  1319.          break;
  1320.       case IPS_ERR_OU_RUN:   
  1321.          if ( ( scb->cmd.dcdb.op_code == IPS_CMD_EXTENDED_DCDB ) ||
  1322.               ( scb->cmd.dcdb.op_code == IPS_CMD_EXTENDED_DCDB_SG ) ) {
  1323.              tapeDCDB = ( IPS_DCDB_TABLE_TAPE * ) &scb->dcdb;
  1324.              transfer_len = tapeDCDB->transfer_length;
  1325.          } else {
  1326.              transfer_len = ( uint32_t ) scb->dcdb.transfer_length;
  1327.          }
  1328.         if ((scb->bus) && (transfer_len < scb->data_len)) {
  1329.             /* Underrun - set default to no error */
  1330.             errcode = DID_OK;
  1331.             /* Restrict access to physical DASD */
  1332.             if ((scb->scsi_cmd->cmnd[0] == INQUIRY) &&
  1333.                 ((((char *) scb->scsi_cmd->buffer)[0] & 0x1f) == TYPE_DISK)) {
  1334.                /* underflow -- no error               */
  1335.                /* restrict access to physical DASD    */
  1336.                errcode = DID_TIME_OUT;
  1337.                break;
  1338.             }
  1339.          } else
  1340.             errcode = DID_ERROR;
  1341.          break;
  1342.       case IPS_ERR_RECOVERY:
  1343.          /* don't fail recovered errors */
  1344.          if (scb->bus)
  1345.             errcode = DID_OK;
  1346.          break;
  1347.       case IPS_ERR_HOST_RESET:
  1348.       case IPS_ERR_DEV_RESET:
  1349.          errcode = DID_RESET;
  1350.          break;
  1351.       case IPS_ERR_CKCOND:
  1352.          if (scb->bus) {
  1353.             if ((scb->cmd.dcdb.op_code == IPS_CMD_EXTENDED_DCDB) ||
  1354.                 (scb->cmd.dcdb.op_code == IPS_CMD_EXTENDED_DCDB_SG)) {
  1355.                tapeDCDB = (IPS_DCDB_TABLE_TAPE *) &scb->dcdb;
  1356.                memcpy(scb->scsi_cmd->sense_buffer, tapeDCDB->sense_info,
  1357.                       sizeof(scb->scsi_cmd->sense_buffer));
  1358.             } else {
  1359.                memcpy(scb->scsi_cmd->sense_buffer, scb->dcdb.sense_info,
  1360.                       sizeof(scb->scsi_cmd->sense_buffer));
  1361.             }
  1362.             device_error = 2; /* check condition */
  1363.          }
  1364.          errcode = DID_OK;
  1365.          break;
  1366.       default:
  1367.          errcode = DID_ERROR;
  1368.          break;
  1369.       } /* end switch */
  1370.    } /* end switch */
  1371.    scb->scsi_cmd->result = device_error | (errcode << 16);
  1372.    return (1);
  1373. }
  1374. /****************************************************************************/
  1375. /*                                                                          */
  1376. /* Routine Name: ips_send_wait                                              */
  1377. /*                                                                          */
  1378. /* Routine Description:                                                     */
  1379. /*                                                                          */
  1380. /*   Send a command to the controller and wait for it to return             */
  1381. /*                                                                          */
  1382. /*   The FFDC Time Stamp use this function for the callback, but doesn't    */
  1383. /*   actually need to wait.                                                 */
  1384. /****************************************************************************/
  1385. static int
  1386. ips_send_wait(ips_ha_t *ha, ips_scb_t *scb, int timeout, int intr) {
  1387.    int       ret;
  1388.    METHOD_TRACE("ips_send_wait", 1);
  1389.    if (intr != IPS_FFDC) {      /* Won't be Waiting if this is a Time Stamp */
  1390.       ha->waitflag = TRUE;
  1391.       ha->cmd_in_progress = scb->cdb[0];
  1392.    }
  1393.    scb->callback = ipsintr_blocking;
  1394.    ret = ips_send_cmd(ha, scb);
  1395.    if ((ret == IPS_FAILURE) || (ret == IPS_SUCCESS_IMM))
  1396.       return (ret);
  1397.    if (intr != IPS_FFDC)       /* Don't Wait around if this is a Time Stamp */
  1398.       ret = ips_wait(ha, timeout, intr);
  1399.    return (ret);
  1400. }
  1401. /****************************************************************************/
  1402. /*                                                                          */
  1403. /* Routine Name: ips_send_cmd                                               */
  1404. /*                                                                          */
  1405. /* Routine Description:                                                     */
  1406. /*                                                                          */
  1407. /*   Map SCSI commands to ServeRAID commands for logical drives             */
  1408. /*                                                                          */
  1409. /****************************************************************************/
  1410. static int
  1411. ips_send_cmd(ips_ha_t *ha, ips_scb_t *scb) {
  1412.    int       ret;
  1413.    char     *sp;
  1414.    int       device_error;
  1415.    IPS_DCDB_TABLE_TAPE *tapeDCDB;
  1416.    METHOD_TRACE("ips_send_cmd", 1);
  1417.    ret = IPS_SUCCESS;
  1418.    if (!scb->scsi_cmd) {
  1419.       /* internal command */
  1420.       if (scb->bus > 0) {
  1421.          /* ServeRAID commands can't be issued */
  1422.          /* to real devices -- fail them       */
  1423.          if ((ha->waitflag == TRUE) &&
  1424.              (ha->cmd_in_progress == scb->cdb[0])) {
  1425.             ha->waitflag = FALSE;
  1426.          }
  1427.          return (1);
  1428.       }
  1429.    } else if ((scb->bus == 0) && (!ips_is_passthru(scb->scsi_cmd))) {
  1430.       /* command to logical bus -- interpret */
  1431.       ret = IPS_SUCCESS_IMM;
  1432.       switch (scb->scsi_cmd->cmnd[0]) {
  1433.       case ALLOW_MEDIUM_REMOVAL:
  1434.       case REZERO_UNIT:
  1435.       case ERASE:
  1436.       case WRITE_FILEMARKS:
  1437.       case SPACE:
  1438.          scb->scsi_cmd->result = DID_ERROR << 16;
  1439.          break;
  1440.       case START_STOP:
  1441.          scb->scsi_cmd->result = DID_OK << 16;
  1442.       case TEST_UNIT_READY:
  1443.       case INQUIRY:
  1444.          if (scb->target_id == IPS_ADAPTER_ID) {
  1445.             /*
  1446.              * Either we have a TUR
  1447.              * or we have a SCSI inquiry
  1448.              */
  1449.             if (scb->scsi_cmd->cmnd[0] == TEST_UNIT_READY)
  1450.                scb->scsi_cmd->result = DID_OK << 16;
  1451.             if (scb->scsi_cmd->cmnd[0] == INQUIRY) {
  1452.                IPS_SCSI_INQ_DATA inquiry;
  1453.                memset(&inquiry, 0, sizeof(IPS_SCSI_INQ_DATA));
  1454.                inquiry.DeviceType = IPS_SCSI_INQ_TYPE_PROCESSOR;
  1455.                inquiry.DeviceTypeQualifier = IPS_SCSI_INQ_LU_CONNECTED;
  1456.                inquiry.Version = IPS_SCSI_INQ_REV2;
  1457.                inquiry.ResponseDataFormat = IPS_SCSI_INQ_RD_REV2;
  1458.                inquiry.AdditionalLength = 31;
  1459.                inquiry.Flags[0] = IPS_SCSI_INQ_Address16;
  1460.                inquiry.Flags[1] = IPS_SCSI_INQ_WBus16 | IPS_SCSI_INQ_Sync;
  1461.                strncpy(inquiry.VendorId, "IBM     ", 8);
  1462.                strncpy(inquiry.ProductId, "SERVERAID       ", 16);
  1463.                strncpy(inquiry.ProductRevisionLevel, "1.00", 4);
  1464.                memcpy(scb->scsi_cmd->request_buffer, &inquiry, scb->scsi_cmd->request_bufflen);
  1465.                scb->scsi_cmd->result = DID_OK << 16;
  1466.             }
  1467.          } else {
  1468.             scb->cmd.logical_info.op_code = IPS_CMD_GET_LD_INFO;
  1469.             scb->cmd.logical_info.command_id = IPS_COMMAND_ID(ha, scb);
  1470.             scb->cmd.logical_info.buffer_addr = cpu_to_le32(VIRT_TO_BUS(&ha->adapt->logical_drive_info));
  1471.             scb->cmd.logical_info.reserved = 0;
  1472.             scb->cmd.logical_info.reserved2 = 0;
  1473.             ret = IPS_SUCCESS;
  1474.          }
  1475.          break;
  1476.       case REQUEST_SENSE:
  1477.          ips_reqsen(ha, scb);
  1478.          scb->scsi_cmd->result = DID_OK << 16;
  1479.          break;
  1480.       case READ_6:
  1481.       case WRITE_6:
  1482.          if (!scb->sg_len) {
  1483.             scb->cmd.basic_io.op_code =
  1484.             (scb->scsi_cmd->cmnd[0] == READ_6) ? IPS_CMD_READ : IPS_CMD_WRITE;
  1485.          } else {
  1486.             scb->cmd.basic_io.op_code =
  1487.             (scb->scsi_cmd->cmnd[0] == READ_6) ? IPS_CMD_READ_SG : IPS_CMD_WRITE_SG;
  1488.          }
  1489.          scb->cmd.basic_io.command_id = IPS_COMMAND_ID(ha, scb);
  1490.          scb->cmd.basic_io.log_drv = scb->target_id;
  1491.          scb->cmd.basic_io.sg_count = scb->sg_len;
  1492.          scb->cmd.basic_io.sg_addr = cpu_to_le32(scb->data_busaddr);
  1493.          if (scb->cmd.basic_io.lba)
  1494.             scb->cmd.basic_io.lba = cpu_to_le32(le32_to_cpu(scb->cmd.basic_io.lba) +
  1495.                                                 le16_to_cpu(scb->cmd.basic_io.sector_count));
  1496.          else
  1497.             scb->cmd.basic_io.lba = (((scb->scsi_cmd->cmnd[1] & 0x1f) << 16) |
  1498.                                      (scb->scsi_cmd->cmnd[2] << 8) |
  1499.                                      (scb->scsi_cmd->cmnd[3]));
  1500.          scb->cmd.basic_io.sector_count = cpu_to_le16(scb->data_len / IPS_BLKSIZE);
  1501.          if (le16_to_cpu(scb->cmd.basic_io.sector_count) == 0)
  1502.             scb->cmd.basic_io.sector_count = cpu_to_le16(256);
  1503.          scb->cmd.basic_io.reserved = 0;
  1504.          ret = IPS_SUCCESS;
  1505.          break;
  1506.       case READ_10:
  1507.       case WRITE_10:
  1508.          if (!scb->sg_len) {
  1509.             scb->cmd.basic_io.op_code =
  1510.             (scb->scsi_cmd->cmnd[0] == READ_10) ? IPS_CMD_READ : IPS_CMD_WRITE;
  1511.          } else {
  1512.             scb->cmd.basic_io.op_code =
  1513.             (scb->scsi_cmd->cmnd[0] == READ_10) ? IPS_CMD_READ_SG : IPS_CMD_WRITE_SG;
  1514.          }
  1515.          scb->cmd.basic_io.command_id = IPS_COMMAND_ID(ha, scb);
  1516.          scb->cmd.basic_io.log_drv = scb->target_id;
  1517.          scb->cmd.basic_io.sg_count = scb->sg_len;
  1518.          scb->cmd.basic_io.sg_addr = cpu_to_le32(scb->data_busaddr);
  1519.          if (scb->cmd.basic_io.lba)
  1520.             scb->cmd.basic_io.lba = cpu_to_le32(le32_to_cpu(scb->cmd.basic_io.lba) +
  1521.                                                 le16_to_cpu(scb->cmd.basic_io.sector_count));
  1522.          else
  1523.             scb->cmd.basic_io.lba = ((scb->scsi_cmd->cmnd[2] << 24) |
  1524.                                      (scb->scsi_cmd->cmnd[3] << 16) |
  1525.                                      (scb->scsi_cmd->cmnd[4] << 8) |
  1526.                                      scb->scsi_cmd->cmnd[5]);
  1527.          scb->cmd.basic_io.sector_count = cpu_to_le16(scb->data_len / IPS_BLKSIZE);
  1528.          scb->cmd.basic_io.reserved = 0;
  1529.          if (cpu_to_le16(scb->cmd.basic_io.sector_count) == 0) {
  1530.             /*
  1531.              * This is a null condition
  1532.              * we don't have to do anything
  1533.              * so just return
  1534.              */
  1535.             scb->scsi_cmd->result = DID_OK << 16;
  1536.          } else
  1537.             ret = IPS_SUCCESS;
  1538.          break;
  1539.       case RESERVE:
  1540.       case RELEASE:
  1541.          scb->scsi_cmd->result = DID_OK << 16;
  1542.          break;
  1543.       case MODE_SENSE:
  1544.          scb->cmd.basic_io.op_code = IPS_CMD_ENQUIRY;
  1545.          scb->cmd.basic_io.command_id = IPS_COMMAND_ID(ha, scb);
  1546.          scb->cmd.basic_io.sg_addr = cpu_to_le32(VIRT_TO_BUS(ha->enq));
  1547.          ret = IPS_SUCCESS;
  1548.          break;
  1549.       case READ_CAPACITY:
  1550.          scb->cmd.logical_info.op_code = IPS_CMD_GET_LD_INFO;
  1551.          scb->cmd.logical_info.command_id = IPS_COMMAND_ID(ha, scb);
  1552.          scb->cmd.logical_info.buffer_addr = cpu_to_le32(VIRT_TO_BUS(&ha->adapt->logical_drive_info));
  1553.          scb->cmd.logical_info.reserved = 0;
  1554.          scb->cmd.logical_info.reserved2 = 0;
  1555.          scb->cmd.logical_info.reserved3 = 0;
  1556.          ret = IPS_SUCCESS;
  1557.          break;
  1558.       case SEND_DIAGNOSTIC:
  1559.       case REASSIGN_BLOCKS:
  1560.       case FORMAT_UNIT:
  1561.       case SEEK_10:
  1562.       case VERIFY:
  1563.       case READ_DEFECT_DATA:
  1564.       case READ_BUFFER:
  1565.       case WRITE_BUFFER:
  1566.          scb->scsi_cmd->result = DID_OK << 16;
  1567.          break;
  1568.       default:
  1569.          /* Set the Return Info to appear like the Command was */
  1570.          /* attempted, a Check Condition occurred, and Sense   */
  1571.          /* Data indicating an Invalid CDB OpCode is returned. */
  1572.          sp = (char *) scb->scsi_cmd->sense_buffer;
  1573.          memset(sp, 0, sizeof(scb->scsi_cmd->sense_buffer));
  1574.   
  1575.          sp[0] = 0x70;             /* Error Code               */ 
  1576.          sp[2] = ILLEGAL_REQUEST;  /* Sense Key 5 Illegal Req. */
  1577.          sp[7] = 0x0A;             /* Additional Sense Length  */
  1578.          sp[12] = 0x20;            /* ASC = Invalid OpCode     */
  1579.          sp[13] = 0x00;            /* ASCQ                     */
  1580.    
  1581.          device_error = 2;         /* Indicate Check Condition */
  1582.          scb->scsi_cmd->result = device_error | (DID_OK << 16);
  1583.          break;
  1584.       } /* end switch */
  1585.    } /* end if */
  1586.    if (ret == IPS_SUCCESS_IMM)
  1587.       return (ret);
  1588.    /* setup DCDB */
  1589.    if (scb->bus > 0) {
  1590.       if (!scb->sg_len)
  1591.          scb->cmd.dcdb.op_code = IPS_CMD_DCDB;
  1592.       else
  1593.          scb->cmd.dcdb.op_code = IPS_CMD_DCDB_SG;
  1594.       /* If we already know the Device is Not there, no need to attempt a Command   */
  1595.       /* This also protects an NT FailOver Controller from getting CDB's sent to it */
  1596.       if ( ha->conf->dev[scb->bus-1][scb->target_id].ucState == 0 ) {
  1597.          scb->scsi_cmd->result = DID_NO_CONNECT << 16;
  1598.          return (IPS_SUCCESS_IMM); 
  1599.       }
  1600.       ha->dcdb_active[scb->bus-1] |= (1 << scb->target_id);
  1601.       scb->cmd.dcdb.command_id = IPS_COMMAND_ID(ha, scb);
  1602.       scb->cmd.dcdb.dcdb_address = cpu_to_le32(VIRT_TO_BUS(&scb->dcdb));
  1603.       scb->cmd.dcdb.reserved = 0;
  1604.       scb->cmd.dcdb.reserved2 = 0;
  1605.       scb->cmd.dcdb.reserved3 = 0;
  1606.       if (ha->subsys->param[4] & 0x00100000) {          /* If NEW Tape DCDB is Supported */
  1607.          if (!scb->sg_len)
  1608.             scb->cmd.dcdb.op_code = IPS_CMD_EXTENDED_DCDB;
  1609.          else
  1610.             scb->cmd.dcdb.op_code = IPS_CMD_EXTENDED_DCDB_SG;
  1611.  
  1612.          tapeDCDB = (IPS_DCDB_TABLE_TAPE *) &scb->dcdb; /* Use Same Data Area as Old DCDB Struct */
  1613.          tapeDCDB->device_address = ((scb->bus - 1) << 4) | scb->target_id;
  1614.          tapeDCDB->cmd_attribute |= IPS_DISCONNECT_ALLOWED;
  1615.          tapeDCDB->cmd_attribute &= ~IPS_TRANSFER64K;   /* Always Turn OFF 64K Size Flag */
  1616.          if (scb->timeout) {
  1617.            if (scb->timeout <= 10)
  1618.                tapeDCDB->cmd_attribute |= IPS_TIMEOUT10;
  1619.             else if (scb->timeout <= 60)
  1620.                tapeDCDB->cmd_attribute |= IPS_TIMEOUT60;
  1621.             else
  1622.                tapeDCDB->cmd_attribute |= IPS_TIMEOUT20M;
  1623.          }
  1624.          if (!(tapeDCDB->cmd_attribute & IPS_TIMEOUT20M))
  1625.             tapeDCDB->cmd_attribute |= IPS_TIMEOUT20M;
  1626.          tapeDCDB->sense_length = sizeof(tapeDCDB->sense_info);
  1627.          tapeDCDB->transfer_length = scb->data_len;
  1628.          tapeDCDB->buffer_pointer = cpu_to_le32(scb->data_busaddr);
  1629.          tapeDCDB->sg_count = scb->sg_len;
  1630.          tapeDCDB->cdb_length = scb->scsi_cmd->cmd_len;
  1631.          tapeDCDB->reserved_for_LUN = 0;
  1632.          tapeDCDB->reserved = 0;
  1633.          tapeDCDB->scsi_status = 0;
  1634.          memcpy(tapeDCDB->scsi_cdb, scb->scsi_cmd->cmnd, scb->scsi_cmd->cmd_len);
  1635.       } else {
  1636.          scb->dcdb.device_address = ((scb->bus - 1) << 4) | scb->target_id;
  1637.          scb->dcdb.cmd_attribute |= IPS_DISCONNECT_ALLOWED;
  1638.          if (scb->timeout) {
  1639.            if (scb->timeout <= 10)
  1640.                scb->dcdb.cmd_attribute |= IPS_TIMEOUT10;
  1641.             else if (scb->timeout <= 60)
  1642.                scb->dcdb.cmd_attribute |= IPS_TIMEOUT60;
  1643.             else
  1644.                scb->dcdb.cmd_attribute |= IPS_TIMEOUT20M;
  1645.          }
  1646.          if (!(scb->dcdb.cmd_attribute & IPS_TIMEOUT20M))
  1647.             scb->dcdb.cmd_attribute |= IPS_TIMEOUT20M;
  1648.          scb->dcdb.sense_length = sizeof(scb->dcdb.sense_info);
  1649.          scb->dcdb.transfer_length = scb->data_len;
  1650.          if ( scb->dcdb.cmd_attribute & IPS_TRANSFER64K ) 
  1651.              scb->dcdb.transfer_length = 0;
  1652.          scb->dcdb.buffer_pointer = cpu_to_le32(scb->data_busaddr);
  1653.          scb->dcdb.sg_count = scb->sg_len;
  1654.          scb->dcdb.cdb_length = scb->scsi_cmd->cmd_len;
  1655.          memcpy(scb->dcdb.scsi_cdb, scb->scsi_cmd->cmnd, scb->scsi_cmd->cmd_len);
  1656.       }
  1657.    }
  1658.    return ((*ha->func.issue)(ha, scb));
  1659. }
  1660. /****************************************************************************/
  1661. /*                                                                          */
  1662. /* Routine Name: ips_chk_status                                             */
  1663. /*                                                                          */
  1664. /* Routine Description:                                                     */
  1665. /*                                                                          */
  1666. /*   Check the status of commands to logical drives                         */
  1667. /*                                                                          */
  1668. /****************************************************************************/
  1669. static void
  1670. ips_chkstatus(ips_ha_t *ha, IPS_STATUS *pstatus) {
  1671.    ips_scb_t  *scb;
  1672.    ips_stat_t *sp;
  1673.    uint8_t     basic_status;
  1674.    uint8_t     ext_status;
  1675.    int         errcode;
  1676.    METHOD_TRACE("ips_chkstatus", 1);
  1677.    scb = &ha->scbs[pstatus->fields.command_id];
  1678.    scb->basic_status = basic_status = pstatus->fields.basic_status & IPS_BASIC_STATUS_MASK;
  1679.    scb->extended_status = ext_status = pstatus->fields.extended_status;
  1680.    sp = &ha->sp;
  1681.    sp->residue_len = 0;
  1682.    sp->scb_addr = (void *) scb;
  1683.    /* Remove the item from the active queue */
  1684.    ips_removeq_scb(&ha->scb_activelist, scb);
  1685.    if (!scb->scsi_cmd)
  1686.       /* internal commands are handled in do_ipsintr */
  1687.       return ;
  1688.    DEBUG_VAR(2, "(%s%d) ips_chkstatus: cmd 0x%X id %d (%d %d %d)",
  1689.              ips_name,
  1690.              ha->host_num,
  1691.              scb->cdb[0],
  1692.              scb->cmd.basic_io.command_id,
  1693.              scb->bus,
  1694.              scb->target_id,
  1695.              scb->lun);
  1696.    if ((scb->scsi_cmd) && (ips_is_passthru(scb->scsi_cmd)))
  1697.       /* passthru - just returns the raw result */
  1698.       return ;
  1699.    errcode = DID_OK;
  1700.    if (((basic_status & IPS_GSC_STATUS_MASK) == IPS_CMD_SUCCESS) ||
  1701.        ((basic_status & IPS_GSC_STATUS_MASK) == IPS_CMD_RECOVERED_ERROR)) {
  1702.       if (scb->bus == 0) {
  1703.          if ((basic_status & IPS_GSC_STATUS_MASK) == IPS_CMD_RECOVERED_ERROR) {
  1704.             DEBUG_VAR(1, "(%s%d) Recovered Logical Drive Error OpCode: %x, BSB: %x, ESB: %x",
  1705.                       ips_name, ha->host_num,
  1706.                       scb->cmd.basic_io.op_code, basic_status, ext_status);
  1707.          }
  1708.          switch (scb->scsi_cmd->cmnd[0]) {
  1709.          case ALLOW_MEDIUM_REMOVAL:
  1710.          case REZERO_UNIT:
  1711.          case ERASE:
  1712.          case WRITE_FILEMARKS:
  1713.          case SPACE:
  1714.             errcode = DID_ERROR;
  1715.             break;
  1716.          case START_STOP:
  1717.             break;
  1718.          case TEST_UNIT_READY:
  1719.             if (!ips_online(ha, scb)) {
  1720.                errcode = DID_TIME_OUT;
  1721.             }
  1722.             break;
  1723.          case INQUIRY:
  1724.             if (ips_online(ha, scb)) {
  1725.                ips_inquiry(ha, scb);
  1726.             } else {
  1727.                errcode = DID_TIME_OUT;
  1728.             }
  1729.             break;
  1730.          case REQUEST_SENSE:
  1731.             ips_reqsen(ha, scb);
  1732.             break;
  1733.          case READ_6:
  1734.          case WRITE_6:
  1735.          case READ_10:
  1736.          case WRITE_10:
  1737.          case RESERVE:
  1738.          case RELEASE:
  1739.             break;
  1740.          case MODE_SENSE:
  1741.             if (!ips_online(ha, scb) || !ips_msense(ha, scb)) {
  1742.                errcode = DID_ERROR;
  1743.             }
  1744.             break;
  1745.          case READ_CAPACITY:
  1746.             if (ips_online(ha, scb))
  1747.                ips_rdcap(ha, scb);
  1748.             else {
  1749.                errcode = DID_TIME_OUT;
  1750.             }
  1751.             break;
  1752.          case SEND_DIAGNOSTIC:
  1753.          case REASSIGN_BLOCKS:
  1754.             break;
  1755.          case FORMAT_UNIT:
  1756.             errcode = DID_ERROR;
  1757.             break;
  1758.          case SEEK_10:
  1759.          case VERIFY:
  1760.          case READ_DEFECT_DATA:
  1761.          case READ_BUFFER:
  1762.          case WRITE_BUFFER:
  1763.             break;
  1764.          default:
  1765.             errcode = DID_ERROR;
  1766.          } /* end switch */
  1767.          scb->scsi_cmd->result = errcode << 16;
  1768.       } else { /* bus == 0 */
  1769.          /* restrict access to physical drives */
  1770.          if ((scb->scsi_cmd->cmnd[0] == INQUIRY) &&
  1771.              ((((char *) scb->scsi_cmd->buffer)[0] & 0x1f) == TYPE_DISK)) {
  1772.             scb->scsi_cmd->result = DID_TIME_OUT << 16;
  1773.          }
  1774.       } /* else */
  1775.    } else { /* recovered error / success */
  1776.       if (scb->bus == 0) {
  1777.          DEBUG_VAR(1, "(%s%d) Unrecovered Logical Drive Error OpCode: %x, BSB: %x, ESB: %x",
  1778.                    ips_name, ha->host_num,
  1779.                    scb->cmd.basic_io.op_code, basic_status, ext_status);
  1780.       }
  1781.       ips_map_status(ha, scb, sp);
  1782.    } /* else */
  1783. }
  1784. /****************************************************************************/
  1785. /*                                                                          */
  1786. /* Routine Name: ips_online                                                 */
  1787. /*                                                                          */
  1788. /* Routine Description:                                                     */
  1789. /*                                                                          */
  1790. /*   Determine if a logical drive is online                                 */
  1791. /*                                                                          */
  1792. /****************************************************************************/
  1793. static int
  1794. ips_online(ips_ha_t *ha, ips_scb_t *scb) {
  1795.    METHOD_TRACE("ips_online", 1);
  1796.    if (scb->target_id >= IPS_MAX_LD)
  1797.       return (0);
  1798.    if ((scb->basic_status & IPS_GSC_STATUS_MASK) > 1) {
  1799.       memset(&ha->adapt->logical_drive_info, 0, sizeof(ha->adapt->logical_drive_info));
  1800.       return (0);
  1801.    }
  1802.    if (ha->adapt->logical_drive_info.drive_info[scb->target_id].state != IPS_LD_OFFLINE &&
  1803.        ha->adapt->logical_drive_info.drive_info[scb->target_id].state != IPS_LD_FREE &&
  1804.        ha->adapt->logical_drive_info.drive_info[scb->target_id].state != IPS_LD_CRS &&
  1805.        ha->adapt->logical_drive_info.drive_info[scb->target_id].state != IPS_LD_SYS)
  1806.       return (1);
  1807.    else
  1808.       return (0);
  1809. }
  1810. /****************************************************************************/
  1811. /*                                                                          */
  1812. /* Routine Name: ips_inquiry                                                */
  1813. /*                                                                          */
  1814. /* Routine Description:                                                     */
  1815. /*                                                                          */
  1816. /*   Simulate an inquiry command to a logical drive                         */
  1817. /*                                                                          */
  1818. /****************************************************************************/
  1819. static int
  1820. ips_inquiry(ips_ha_t *ha, ips_scb_t *scb) {
  1821.    IPS_SCSI_INQ_DATA inquiry;
  1822.    METHOD_TRACE("ips_inquiry", 1);
  1823.    memset(&inquiry, 0, sizeof(IPS_SCSI_INQ_DATA));
  1824.    inquiry.DeviceType = IPS_SCSI_INQ_TYPE_DASD;
  1825.    inquiry.DeviceTypeQualifier = IPS_SCSI_INQ_LU_CONNECTED;
  1826.    inquiry.Version = IPS_SCSI_INQ_REV2;
  1827.    inquiry.ResponseDataFormat = IPS_SCSI_INQ_RD_REV2;
  1828.    inquiry.AdditionalLength = 31;
  1829.    inquiry.Flags[0] = IPS_SCSI_INQ_Address16;
  1830.    inquiry.Flags[1] = IPS_SCSI_INQ_WBus16 | IPS_SCSI_INQ_Sync;
  1831.    strncpy(inquiry.VendorId, "IBM     ", 8);
  1832.    strncpy(inquiry.ProductId, "SERVERAID       ", 16);
  1833.    strncpy(inquiry.ProductRevisionLevel, "1.00", 4);
  1834.    memcpy(scb->scsi_cmd->request_buffer, &inquiry, scb->scsi_cmd->request_bufflen);
  1835.    return (1);
  1836. }
  1837. /****************************************************************************/
  1838. /*                                                                          */
  1839. /* Routine Name: ips_rdcap                                                  */
  1840. /*                                                                          */
  1841. /* Routine Description:                                                     */
  1842. /*                                                                          */
  1843. /*   Simulate a read capacity command to a logical drive                    */
  1844. /*                                                                          */
  1845. /****************************************************************************/
  1846. static int
  1847. ips_rdcap(ips_ha_t *ha, ips_scb_t *scb) {
  1848.    IPS_SCSI_CAPACITY *cap;
  1849.    METHOD_TRACE("ips_rdcap", 1);
  1850.    if (scb->scsi_cmd->bufflen < 8)
  1851.       return (0);
  1852.    cap = (IPS_SCSI_CAPACITY *) scb->scsi_cmd->request_buffer;
  1853.    cap->lba = cpu_to_be32(le32_to_cpu(ha->adapt->logical_drive_info.drive_info[scb->target_id].sector_count) - 1);
  1854.    cap->len = cpu_to_be32((uint32_t) IPS_BLKSIZE);
  1855.    return (1);
  1856. }
  1857. /****************************************************************************/
  1858. /*                                                                          */
  1859. /* Routine Name: ips_msense                                                 */
  1860. /*                                                                          */
  1861. /* Routine Description:                                                     */
  1862. /*                                                                          */
  1863. /*   Simulate a mode sense command to a logical drive                       */
  1864. /*                                                                          */
  1865. /****************************************************************************/
  1866. static int
  1867. ips_msense(ips_ha_t *ha, ips_scb_t *scb) {
  1868.    uint16_t               heads;
  1869.    uint16_t               sectors;
  1870.    uint32_t               cylinders;
  1871.    IPS_SCSI_MODE_PAGE_DATA mdata;
  1872.    METHOD_TRACE("ips_msense", 1);
  1873.    if (le32_to_cpu(ha->enq->ulDriveSize[scb->target_id]) > 0x400000 &&
  1874.        (ha->enq->ucMiscFlag & 0x8) == 0) {
  1875.       heads = IPS_NORM_HEADS;
  1876.       sectors = IPS_NORM_SECTORS;
  1877.    } else {
  1878.       heads = IPS_COMP_HEADS;
  1879.       sectors = IPS_COMP_SECTORS;
  1880.    }
  1881.    cylinders = (le32_to_cpu(ha->enq->ulDriveSize[scb->target_id]) - 1) / (heads * sectors);
  1882.    memset(&mdata, 0, sizeof(IPS_SCSI_MODE_PAGE_DATA));
  1883.    mdata.hdr.BlockDescLength = 8;
  1884.    switch (scb->scsi_cmd->cmnd[2] & 0x3f) {
  1885.    case 0x03: /* page 3 */
  1886.       mdata.pdata.pg3.PageCode = 3;
  1887.       mdata.pdata.pg3.PageLength = sizeof(IPS_SCSI_MODE_PAGE3);
  1888.       mdata.hdr.DataLength = 3 + mdata.hdr.BlockDescLength + mdata.pdata.pg3.PageLength;
  1889.       mdata.pdata.pg3.TracksPerZone = 0;
  1890.       mdata.pdata.pg3.AltSectorsPerZone = 0;
  1891.       mdata.pdata.pg3.AltTracksPerZone = 0;
  1892.       mdata.pdata.pg3.AltTracksPerVolume = 0;
  1893.       mdata.pdata.pg3.SectorsPerTrack = cpu_to_be16(sectors);
  1894.       mdata.pdata.pg3.BytesPerSector = cpu_to_be16(IPS_BLKSIZE);
  1895.       mdata.pdata.pg3.Interleave = cpu_to_be16(1);
  1896.       mdata.pdata.pg3.TrackSkew = 0;
  1897.       mdata.pdata.pg3.CylinderSkew = 0;
  1898.       mdata.pdata.pg3.flags = IPS_SCSI_MP3_SoftSector;
  1899.       break;
  1900.    case 0x4:
  1901.       mdata.pdata.pg4.PageCode = 4;
  1902.       mdata.pdata.pg4.PageLength = sizeof(IPS_SCSI_MODE_PAGE4);
  1903.       mdata.hdr.DataLength = 3 + mdata.hdr.BlockDescLength + mdata.pdata.pg4.PageLength;
  1904.       mdata.pdata.pg4.CylindersHigh = cpu_to_be16((cylinders >> 8) & 0xFFFF);
  1905.       mdata.pdata.pg4.CylindersLow = (cylinders & 0xFF);
  1906.       mdata.pdata.pg4.Heads = heads;
  1907.       mdata.pdata.pg4.WritePrecompHigh = 0;
  1908.       mdata.pdata.pg4.WritePrecompLow = 0;
  1909.       mdata.pdata.pg4.ReducedWriteCurrentHigh = 0;
  1910.       mdata.pdata.pg4.ReducedWriteCurrentLow = 0;
  1911.       mdata.pdata.pg4.StepRate = cpu_to_be16(1);
  1912.       mdata.pdata.pg4.LandingZoneHigh = 0;
  1913.       mdata.pdata.pg4.LandingZoneLow = 0;
  1914.       mdata.pdata.pg4.flags = 0;
  1915.       mdata.pdata.pg4.RotationalOffset = 0;
  1916.       mdata.pdata.pg4.MediumRotationRate = 0;
  1917.       break;
  1918.    default:
  1919.       return (0);
  1920.    } /* end switch */
  1921.    memcpy(scb->scsi_cmd->request_buffer, &mdata, scb->scsi_cmd->request_bufflen);
  1922.    return (1);
  1923. }
  1924. /****************************************************************************/
  1925. /*                                                                          */
  1926. /* Routine Name: ips_reqsen                                                 */
  1927. /*                                                                          */
  1928. /* Routine Description:                                                     */
  1929. /*                                                                          */
  1930. /*   Simulate a request sense command to a logical drive                    */
  1931. /*                                                                          */
  1932. /****************************************************************************/
  1933. static int
  1934. ips_reqsen(ips_ha_t *ha, ips_scb_t *scb) {
  1935.    IPS_SCSI_REQSEN reqsen;
  1936.    METHOD_TRACE("ips_reqsen", 1);
  1937.    memset(&reqsen, 0, sizeof(IPS_SCSI_REQSEN));
  1938.    reqsen.ResponseCode = IPS_SCSI_REQSEN_VALID | IPS_SCSI_REQSEN_CURRENT_ERR;
  1939.    reqsen.AdditionalLength = 10;
  1940.    reqsen.AdditionalSenseCode = IPS_SCSI_REQSEN_NO_SENSE;
  1941.    reqsen.AdditionalSenseCodeQual = IPS_SCSI_REQSEN_NO_SENSE;
  1942.    memcpy(scb->scsi_cmd->request_buffer, &reqsen, scb->scsi_cmd->request_bufflen);
  1943.    return (1);
  1944. }
  1945. /****************************************************************************/
  1946. /*                                                                          */
  1947. /* Routine Name: ips_free                                                   */
  1948. /*                                                                          */
  1949. /* Routine Description:                                                     */
  1950. /*                                                                          */
  1951. /*   Free any allocated space for this controller                           */
  1952. /*                                                                          */
  1953. /****************************************************************************/
  1954. static void
  1955. ips_free(ips_ha_t *ha) {
  1956.    METHOD_TRACE("ips_free", 1);
  1957.    if (ha) {
  1958.       if (ha->enq) {
  1959.          kfree(ha->enq);
  1960.          ha->enq = NULL;
  1961.       }
  1962.       if (ha->conf) {
  1963.          kfree(ha->conf);
  1964.          ha->conf = NULL;
  1965.       }
  1966.       if (ha->adapt) {
  1967.          kfree(ha->adapt);
  1968.          ha->adapt = NULL;
  1969.       }
  1970.       if (ha->nvram) {
  1971.          kfree(ha->nvram);
  1972.          ha->nvram = NULL;
  1973.       }
  1974.       if (ha->subsys) {
  1975.          kfree(ha->subsys);
  1976.          ha->subsys = NULL;
  1977.       }
  1978.       if (ha->dummy) {
  1979.          kfree(ha->dummy);
  1980.          ha->dummy = NULL;
  1981.       }
  1982.       if (ha->ioctl_data) {
  1983.          free_pages((unsigned long) ha->ioctl_data, ha->ioctl_order);
  1984.          ha->ioctl_data = NULL;
  1985.          ha->ioctl_datasize = 0;
  1986.          ha->ioctl_order = 0;
  1987.       }
  1988.       ips_deallocatescbs(ha, ha->max_cmds);
  1989.       /* free memory mapped (if applicable) */
  1990.       if (ha->mem_ptr) {
  1991.          iounmap(ha->ioremap_ptr);
  1992.          ha->ioremap_ptr = NULL;
  1993.          ha->mem_ptr = NULL;
  1994.       }
  1995. #if LINUX_VERSION_CODE >= LinuxVersionCode(2,4,0)
  1996.       if (ha->mem_addr) 
  1997.           release_mem_region(ha->mem_addr, ha->mem_len);
  1998. #endif
  1999.       ha->mem_addr = 0;
  2000.    }
  2001. }
  2002. /****************************************************************************/
  2003. /*                                                                          */
  2004. /* Routine Name: ips_deallocatescbs                                         */
  2005. /*                                                                          */
  2006. /* Routine Description:                                                     */
  2007. /*                                                                          */
  2008. /*   Free the command blocks                                                */
  2009. /*                                                                          */
  2010. /****************************************************************************/
  2011. static int
  2012. ips_deallocatescbs(ips_ha_t *ha, int cmds) {
  2013.    if (ha->scbs) {
  2014.       pci_free_consistent(ha->pcidev,sizeof(IPS_SG_LIST) * IPS_MAX_SG *
  2015.                           cmds, ha->scbs->sg_list, ha->scbs->sg_busaddr);
  2016.       pci_free_consistent(ha->pcidev, sizeof(ips_scb_t) * cmds,
  2017.                           ha->scbs, ha->scbs->scb_busaddr);
  2018.       ha->scbs = NULL;
  2019.    } /* end if */
  2020. return 1;
  2021. }
  2022. /****************************************************************************/
  2023. /*                                                                          */
  2024. /* Routine Name: ips_allocatescbs                                           */
  2025. /*                                                                          */
  2026. /* Routine Description:                                                     */
  2027. /*                                                                          */
  2028. /*   Allocate the command blocks                                            */
  2029. /*                                                                          */
  2030. /****************************************************************************/
  2031. static int
  2032. ips_allocatescbs(ips_ha_t *ha) {
  2033.    ips_scb_t *scb_p;
  2034.    IPS_SG_LIST* ips_sg;
  2035.    int        i;
  2036.    dma_addr_t command_dma, sg_dma;
  2037.    
  2038.    METHOD_TRACE("ips_allocatescbs", 1);
  2039.    /* Allocate memory for the SCBs */
  2040.    ha->scbs = pci_alloc_consistent(ha->pcidev, ha->max_cmds * sizeof(ips_scb_t),
  2041.                            &command_dma);
  2042.    if (ha->scbs == NULL)
  2043.       return 0;
  2044.    ips_sg = pci_alloc_consistent(ha->pcidev, sizeof(IPS_SG_LIST) * IPS_MAX_SG * 
  2045.                          ha->max_cmds, &sg_dma);
  2046.    if(ips_sg == NULL){
  2047.       pci_free_consistent(ha->pcidev,ha->max_cmds * sizeof(ips_scb_t),ha->scbs, command_dma);
  2048.       return 0;
  2049.    }
  2050.    memset(ha->scbs, 0, ha->max_cmds * sizeof(ips_scb_t));
  2051.    for (i = 0; i < ha->max_cmds; i++) {
  2052.       scb_p = &ha->scbs[i];
  2053.       scb_p->scb_busaddr = command_dma + sizeof(ips_scb_t) * i;
  2054.       /* set up S/G list */
  2055.       scb_p->sg_list = ips_sg + i * IPS_MAX_SG;
  2056.       scb_p->sg_busaddr = sg_dma + sizeof(IPS_SG_LIST) * IPS_MAX_SG * i;
  2057.       /* add to the free list */
  2058.       if (i < ha->max_cmds - 1) {
  2059.          scb_p->q_next = ha->scb_freelist;
  2060.          ha->scb_freelist = scb_p;
  2061.       }
  2062.    }
  2063.    /* success */
  2064.    return (1);
  2065. }
  2066. /****************************************************************************/
  2067. /*                                                                          */
  2068. /* Routine Name: ips_init_scb                                               */
  2069. /*                                                                          */
  2070. /* Routine Description:                                                     */
  2071. /*                                                                          */
  2072. /*   Initialize a CCB to default values                                     */
  2073. /*                                                                          */
  2074. /****************************************************************************/
  2075. static void
  2076. ips_init_scb(ips_ha_t *ha, ips_scb_t *scb) {
  2077.    IPS_SG_LIST *sg_list;
  2078.    uint32_t cmd_busaddr, sg_busaddr;
  2079.    METHOD_TRACE("ips_init_scb", 1);
  2080.    if (scb == NULL)
  2081.       return ;
  2082.    sg_list = scb->sg_list;
  2083.    cmd_busaddr = scb->scb_busaddr;
  2084.    sg_busaddr = scb->sg_busaddr;
  2085.    /* zero fill */
  2086.    memset(scb, 0, sizeof(ips_scb_t));
  2087.    memset(ha->dummy, 0, sizeof(IPS_IO_CMD));
  2088.    /* Initialize dummy command bucket */
  2089.    ha->dummy->op_code = 0xFF;
  2090.    ha->dummy->ccsar = cpu_to_le32(VIRT_TO_BUS(ha->dummy));
  2091.    ha->dummy->command_id = IPS_MAX_CMDS;
  2092.    /* set bus address of scb */
  2093.    scb->scb_busaddr = cmd_busaddr;
  2094.    scb->sg_busaddr = sg_busaddr;
  2095.    scb->sg_list = sg_list;
  2096.    /* Neptune Fix */
  2097.    scb->cmd.basic_io.cccr = cpu_to_le32((uint32_t) IPS_BIT_ILE);
  2098.    scb->cmd.basic_io.ccsar = cpu_to_le32(VIRT_TO_BUS(ha->dummy));
  2099. }
  2100. /****************************************************************************/
  2101. /*                                                                          */
  2102. /* Routine Name: ips_get_scb                                                */
  2103. /*                                                                          */
  2104. /* Routine Description:                                                     */
  2105. /*                                                                          */
  2106. /*   Initialize a CCB to default values                                     */
  2107. /*                                                                          */
  2108. /* ASSUMED to be callled from within a lock                                 */
  2109. /*                                                                          */
  2110. /****************************************************************************/
  2111. static ips_scb_t *
  2112. ips_getscb(ips_ha_t *ha) {
  2113.    ips_scb_t     *scb;
  2114.    unsigned long  cpu_flags;
  2115.    METHOD_TRACE("ips_getscb", 1);
  2116.    IPS_SCB_LOCK(cpu_flags);
  2117.    if ((scb = ha->scb_freelist) == NULL) {
  2118.       IPS_SCB_UNLOCK(cpu_flags);
  2119.       return (NULL);
  2120.    }
  2121.    ha->scb_freelist = scb->q_next;
  2122.    scb->q_next = NULL;
  2123.    IPS_SCB_UNLOCK(cpu_flags);
  2124.    ips_init_scb(ha, scb);
  2125.    return (scb);