blob: 749c95bb7df7c0ea34d720584d8feef4b98b6cce [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*****************************************************************************/
2/* ips.c -- driver for the Adaptec / IBM ServeRAID controller */
3/* */
4/* Written By: Keith Mitchell, IBM Corporation */
5/* Jack Hammer, Adaptec, Inc. */
6/* David Jeffery, Adaptec, Inc. */
7/* */
8/* Copyright (C) 2000 IBM Corporation */
9/* Copyright (C) 2002,2003 Adaptec, Inc. */
10/* */
11/* This program is free software; you can redistribute it and/or modify */
12/* it under the terms of the GNU General Public License as published by */
13/* the Free Software Foundation; either version 2 of the License, or */
14/* (at your option) any later version. */
15/* */
16/* This program is distributed in the hope that it will be useful, */
17/* but WITHOUT ANY WARRANTY; without even the implied warranty of */
18/* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the */
19/* GNU General Public License for more details. */
20/* */
21/* NO WARRANTY */
22/* THE PROGRAM IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OR */
23/* CONDITIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED INCLUDING, WITHOUT */
24/* LIMITATION, ANY WARRANTIES OR CONDITIONS OF TITLE, NON-INFRINGEMENT, */
25/* MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE. Each Recipient is */
26/* solely responsible for determining the appropriateness of using and */
27/* distributing the Program and assumes all risks associated with its */
28/* exercise of rights under this Agreement, including but not limited to */
29/* the risks and costs of program errors, damage to or loss of data, */
30/* programs or equipment, and unavailability or interruption of operations. */
31/* */
32/* DISCLAIMER OF LIABILITY */
33/* NEITHER RECIPIENT NOR ANY CONTRIBUTORS SHALL HAVE ANY LIABILITY FOR ANY */
34/* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL */
35/* DAMAGES (INCLUDING WITHOUT LIMITATION LOST PROFITS), HOWEVER CAUSED AND */
36/* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR */
37/* TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE */
38/* USE OR DISTRIBUTION OF THE PROGRAM OR THE EXERCISE OF ANY RIGHTS GRANTED */
39/* HEREUNDER, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGES */
40/* */
41/* You should have received a copy of the GNU General Public License */
42/* along with this program; if not, write to the Free Software */
43/* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
44/* */
45/* Bugs/Comments/Suggestions about this driver should be mailed to: */
46/* ipslinux@adaptec.com */
47/* */
48/* For system support issues, contact your local IBM Customer support. */
49/* Directions to find IBM Customer Support for each country can be found at: */
50/* http://www.ibm.com/planetwide/ */
51/* */
52/*****************************************************************************/
53
54/*****************************************************************************/
55/* Change Log */
56/* */
57/* 0.99.02 - Breakup commands that are bigger than 8 * the stripe size */
58/* 0.99.03 - Make interrupt routine handle all completed request on the */
59/* adapter not just the first one */
60/* - Make sure passthru commands get woken up if we run out of */
61/* SCBs */
62/* - Send all of the commands on the queue at once rather than */
63/* one at a time since the card will support it. */
64/* 0.99.04 - Fix race condition in the passthru mechanism -- this required */
65/* the interface to the utilities to change */
66/* - Fix error recovery code */
67/* 0.99.05 - Fix an oops when we get certain passthru commands */
68/* 1.00.00 - Initial Public Release */
69/* Functionally equivalent to 0.99.05 */
70/* 3.60.00 - Bump max commands to 128 for use with firmware 3.60 */
71/* - Change version to 3.60 to coincide with release numbering. */
72/* 3.60.01 - Remove bogus error check in passthru routine */
73/* 3.60.02 - Make DCDB direction based on lookup table */
74/* - Only allow one DCDB command to a SCSI ID at a time */
75/* 4.00.00 - Add support for ServeRAID 4 */
76/* 4.00.01 - Add support for First Failure Data Capture */
77/* 4.00.02 - Fix problem with PT DCDB with no buffer */
78/* 4.00.03 - Add alternative passthru interface */
79/* - Add ability to flash BIOS */
80/* 4.00.04 - Rename structures/constants to be prefixed with IPS_ */
81/* 4.00.05 - Remove wish_block from init routine */
82/* - Use linux/spinlock.h instead of asm/spinlock.h for kernels */
83/* 2.3.18 and later */
84/* - Sync with other changes from the 2.3 kernels */
85/* 4.00.06 - Fix timeout with initial FFDC command */
86/* 4.00.06a - Port to 2.4 (trivial) -- Christoph Hellwig <hch@infradead.org> */
87/* 4.10.00 - Add support for ServeRAID 4M/4L */
88/* 4.10.13 - Fix for dynamic unload and proc file system */
89/* 4.20.03 - Rename version to coincide with new release schedules */
90/* Performance fixes */
91/* Fix truncation of /proc files with cat */
92/* Merge in changes through kernel 2.4.0test1ac21 */
93/* 4.20.13 - Fix some failure cases / reset code */
94/* - Hook into the reboot_notifier to flush the controller cache */
95/* 4.50.01 - Fix problem when there is a hole in logical drive numbering */
96/* 4.70.09 - Use a Common ( Large Buffer ) for Flashing from the JCRM CD */
97/* - Add IPSSEND Flash Support */
98/* - Set Sense Data for Unknown SCSI Command */
99/* - Use Slot Number from NVRAM Page 5 */
100/* - Restore caller's DCDB Structure */
101/* 4.70.12 - Corrective actions for bad controller ( during initialization )*/
102/* 4.70.13 - Don't Send CDB's if we already know the device is not present */
103/* - Don't release HA Lock in ips_next() until SC taken off queue */
104/* - Unregister SCSI device in ips_release() */
105/* 4.70.15 - Fix Breakup for very large ( non-SG ) requests in ips_done() */
106/* 4.71.00 - Change all memory allocations to not use GFP_DMA flag */
107/* Code Clean-Up for 2.4.x kernel */
108/* 4.72.00 - Allow for a Scatter-Gather Element to exceed MAX_XFER Size */
109/* 4.72.01 - I/O Mapped Memory release ( so "insmod ips" does not Fail ) */
110/* - Don't Issue Internal FFDC Command if there are Active Commands */
111/* - Close Window for getting too many IOCTL's active */
112/* 4.80.00 - Make ia64 Safe */
113/* 4.80.04 - Eliminate calls to strtok() if 2.4.x or greater */
114/* - Adjustments to Device Queue Depth */
115/* 4.80.14 - Take all semaphores off stack */
116/* - Clean Up New_IOCTL path */
117/* 4.80.20 - Set max_sectors in Scsi_Host structure ( if >= 2.4.7 kernel ) */
118/* - 5 second delay needed after resetting an i960 adapter */
119/* 4.80.26 - Clean up potential code problems ( Arjan's recommendations ) */
120/* 4.90.01 - Version Matching for FirmWare, BIOS, and Driver */
121/* 4.90.05 - Use New PCI Architecture to facilitate Hot Plug Development */
122/* 4.90.08 - Increase Delays in Flashing ( Trombone Only - 4H ) */
123/* 4.90.08 - Data Corruption if First Scatter Gather Element is > 64K */
124/* 4.90.11 - Don't actually RESET unless it's physically required */
125/* - Remove unused compile options */
126/* 5.00.01 - Sarasota ( 5i ) adapters must always be scanned first */
127/* - Get rid on IOCTL_NEW_COMMAND code */
128/* - Add Extended DCDB Commands for Tape Support in 5I */
129/* 5.10.12 - use pci_dma interfaces, update for 2.5 kernel changes */
130/* 5.10.15 - remove unused code (sem, macros, etc.) */
131/* 5.30.00 - use __devexit_p() */
132/* 6.00.00 - Add 6x Adapters and Battery Flash */
133/* 6.10.00 - Remove 1G Addressing Limitations */
134/* 6.11.xx - Get VersionInfo buffer off the stack ! DDTS 60401 */
135/* 6.11.xx - Make Logical Drive Info structure safe for DMA DDTS 60639 */
Jack Hammerc1a15462005-07-26 10:20:33 -0400136/* 7.10.18 - Add highmem_io flag in SCSI Templete for 2.4 kernels */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700137/* - Fix path/name for scsi_hosts.h include for 2.6 kernels */
138/* - Fix sort order of 7k */
139/* - Remove 3 unused "inline" functions */
Jack Hammerc1a15462005-07-26 10:20:33 -0400140/* 7.12.xx - Use STATIC functions whereever possible */
141/* - Clean up deprecated MODULE_PARM calls */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700142/*****************************************************************************/
143
144/*
145 * Conditional Compilation directives for this driver:
146 *
147 * IPS_DEBUG - Turn on debugging info
148 *
149 * Parameters:
150 *
151 * debug:<number> - Set debug level to <number>
152 * NOTE: only works when IPS_DEBUG compile directive is used.
153 * 1 - Normal debug messages
154 * 2 - Verbose debug messages
155 * 11 - Method trace (non interrupt)
156 * 12 - Method trace (includes interrupt)
157 *
158 * noi2o - Don't use I2O Queues (ServeRAID 4 only)
159 * nommap - Don't use memory mapped I/O
160 * ioctlsize - Initial size of the IOCTL buffer
161 */
162
163#include <asm/io.h>
164#include <asm/byteorder.h>
165#include <asm/page.h>
166#include <linux/stddef.h>
167#include <linux/version.h>
168#include <linux/string.h>
169#include <linux/errno.h>
170#include <linux/kernel.h>
171#include <linux/ioport.h>
172#include <linux/slab.h>
173#include <linux/delay.h>
174#include <linux/pci.h>
175#include <linux/proc_fs.h>
176#include <linux/reboot.h>
177#include <linux/interrupt.h>
178
179#include <linux/blkdev.h>
180#include <linux/types.h>
181
182#include <scsi/sg.h>
183
184#include "scsi.h"
185
186#if LINUX_VERSION_CODE <= KERNEL_VERSION(2,5,0)
187#include "hosts.h"
188#else
189#include <scsi/scsi_host.h>
190#endif
191
192#include "ips.h"
193
194#include <linux/module.h>
195
196#include <linux/stat.h>
197#include <linux/config.h>
198
199#include <linux/spinlock.h>
200#include <linux/init.h>
201
202#include <linux/smp.h>
203
204#ifdef MODULE
205static char *ips = NULL;
206module_param(ips, charp, 0);
207#endif
208
209/*
210 * DRIVER_VER
211 */
Jack Hammerc1a15462005-07-26 10:20:33 -0400212#define IPS_VERSION_HIGH "7.12"
213#define IPS_VERSION_LOW ".02 "
Linus Torvalds1da177e2005-04-16 15:20:36 -0700214
215#if !defined(__i386__) && !defined(__ia64__) && !defined(__x86_64__)
216#warning "This driver has only been tested on the x86/ia64/x86_64 platforms"
217#endif
218
219#if LINUX_VERSION_CODE <= KERNEL_VERSION(2,5,0)
220#include <linux/blk.h>
221#include "sd.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700222#define IPS_LOCK_SAVE(lock,flags) spin_lock_irqsave(&io_request_lock,flags)
223#define IPS_UNLOCK_RESTORE(lock,flags) spin_unlock_irqrestore(&io_request_lock,flags)
224#ifndef __devexit_p
225#define __devexit_p(x) x
226#endif
227#else
Linus Torvalds1da177e2005-04-16 15:20:36 -0700228#define IPS_LOCK_SAVE(lock,flags) do{spin_lock(lock);(void)flags;}while(0)
229#define IPS_UNLOCK_RESTORE(lock,flags) do{spin_unlock(lock);(void)flags;}while(0)
230#endif
231
232#define IPS_DMA_DIR(scb) ((!scb->scsi_cmd || ips_is_passthru(scb->scsi_cmd) || \
be7db052005-04-17 15:26:13 -0500233 DMA_NONE == scb->scsi_cmd->sc_data_direction) ? \
Linus Torvalds1da177e2005-04-16 15:20:36 -0700234 PCI_DMA_BIDIRECTIONAL : \
be7db052005-04-17 15:26:13 -0500235 scb->scsi_cmd->sc_data_direction)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700236
237#ifdef IPS_DEBUG
238#define METHOD_TRACE(s, i) if (ips_debug >= (i+10)) printk(KERN_NOTICE s "\n");
239#define DEBUG(i, s) if (ips_debug >= i) printk(KERN_NOTICE s "\n");
240#define DEBUG_VAR(i, s, v...) if (ips_debug >= i) printk(KERN_NOTICE s "\n", v);
241#else
242#define METHOD_TRACE(s, i)
243#define DEBUG(i, s)
244#define DEBUG_VAR(i, s, v...)
245#endif
246
247/*
248 * Function prototypes
249 */
250static int ips_detect(Scsi_Host_Template *);
251static int ips_release(struct Scsi_Host *);
252static int ips_eh_abort(Scsi_Cmnd *);
253static int ips_eh_reset(Scsi_Cmnd *);
254static int ips_queue(Scsi_Cmnd *, void (*)(Scsi_Cmnd *));
255static const char *ips_info(struct Scsi_Host *);
256static irqreturn_t do_ipsintr(int, void *, struct pt_regs *);
257static int ips_hainit(ips_ha_t *);
258static int ips_map_status(ips_ha_t *, ips_scb_t *, ips_stat_t *);
259static int ips_send_wait(ips_ha_t *, ips_scb_t *, int, int);
260static int ips_send_cmd(ips_ha_t *, ips_scb_t *);
261static int ips_online(ips_ha_t *, ips_scb_t *);
262static int ips_inquiry(ips_ha_t *, ips_scb_t *);
263static int ips_rdcap(ips_ha_t *, ips_scb_t *);
264static int ips_msense(ips_ha_t *, ips_scb_t *);
265static int ips_reqsen(ips_ha_t *, ips_scb_t *);
266static int ips_deallocatescbs(ips_ha_t *, int);
267static int ips_allocatescbs(ips_ha_t *);
268static int ips_reset_copperhead(ips_ha_t *);
269static int ips_reset_copperhead_memio(ips_ha_t *);
270static int ips_reset_morpheus(ips_ha_t *);
271static int ips_issue_copperhead(ips_ha_t *, ips_scb_t *);
272static int ips_issue_copperhead_memio(ips_ha_t *, ips_scb_t *);
273static int ips_issue_i2o(ips_ha_t *, ips_scb_t *);
274static int ips_issue_i2o_memio(ips_ha_t *, ips_scb_t *);
275static int ips_isintr_copperhead(ips_ha_t *);
276static int ips_isintr_copperhead_memio(ips_ha_t *);
277static int ips_isintr_morpheus(ips_ha_t *);
278static int ips_wait(ips_ha_t *, int, int);
279static int ips_write_driver_status(ips_ha_t *, int);
280static int ips_read_adapter_status(ips_ha_t *, int);
281static int ips_read_subsystem_parameters(ips_ha_t *, int);
282static int ips_read_config(ips_ha_t *, int);
283static int ips_clear_adapter(ips_ha_t *, int);
284static int ips_readwrite_page5(ips_ha_t *, int, int);
285static int ips_init_copperhead(ips_ha_t *);
286static int ips_init_copperhead_memio(ips_ha_t *);
287static int ips_init_morpheus(ips_ha_t *);
288static int ips_isinit_copperhead(ips_ha_t *);
289static int ips_isinit_copperhead_memio(ips_ha_t *);
290static int ips_isinit_morpheus(ips_ha_t *);
291static int ips_erase_bios(ips_ha_t *);
292static int ips_program_bios(ips_ha_t *, char *, uint32_t, uint32_t);
293static int ips_verify_bios(ips_ha_t *, char *, uint32_t, uint32_t);
294static int ips_erase_bios_memio(ips_ha_t *);
295static int ips_program_bios_memio(ips_ha_t *, char *, uint32_t, uint32_t);
296static int ips_verify_bios_memio(ips_ha_t *, char *, uint32_t, uint32_t);
297static int ips_flash_copperhead(ips_ha_t *, ips_passthru_t *, ips_scb_t *);
298static int ips_flash_bios(ips_ha_t *, ips_passthru_t *, ips_scb_t *);
299static int ips_flash_firmware(ips_ha_t *, ips_passthru_t *, ips_scb_t *);
300static void ips_free_flash_copperhead(ips_ha_t * ha);
301static void ips_get_bios_version(ips_ha_t *, int);
302static void ips_identify_controller(ips_ha_t *);
303static void ips_chkstatus(ips_ha_t *, IPS_STATUS *);
304static void ips_enable_int_copperhead(ips_ha_t *);
305static void ips_enable_int_copperhead_memio(ips_ha_t *);
306static void ips_enable_int_morpheus(ips_ha_t *);
307static int ips_intr_copperhead(ips_ha_t *);
308static int ips_intr_morpheus(ips_ha_t *);
309static void ips_next(ips_ha_t *, int);
310static void ipsintr_blocking(ips_ha_t *, struct ips_scb *);
311static void ipsintr_done(ips_ha_t *, struct ips_scb *);
312static void ips_done(ips_ha_t *, ips_scb_t *);
313static void ips_free(ips_ha_t *);
314static void ips_init_scb(ips_ha_t *, ips_scb_t *);
315static void ips_freescb(ips_ha_t *, ips_scb_t *);
316static void ips_setup_funclist(ips_ha_t *);
317static void ips_statinit(ips_ha_t *);
318static void ips_statinit_memio(ips_ha_t *);
319static void ips_fix_ffdc_time(ips_ha_t *, ips_scb_t *, time_t);
320static void ips_ffdc_reset(ips_ha_t *, int);
321static void ips_ffdc_time(ips_ha_t *);
322static uint32_t ips_statupd_copperhead(ips_ha_t *);
323static uint32_t ips_statupd_copperhead_memio(ips_ha_t *);
324static uint32_t ips_statupd_morpheus(ips_ha_t *);
325static ips_scb_t *ips_getscb(ips_ha_t *);
326static void ips_putq_scb_head(ips_scb_queue_t *, ips_scb_t *);
327static void ips_putq_wait_tail(ips_wait_queue_t *, Scsi_Cmnd *);
328static void ips_putq_copp_tail(ips_copp_queue_t *,
329 ips_copp_wait_item_t *);
330static ips_scb_t *ips_removeq_scb_head(ips_scb_queue_t *);
331static ips_scb_t *ips_removeq_scb(ips_scb_queue_t *, ips_scb_t *);
332static Scsi_Cmnd *ips_removeq_wait_head(ips_wait_queue_t *);
333static Scsi_Cmnd *ips_removeq_wait(ips_wait_queue_t *, Scsi_Cmnd *);
334static ips_copp_wait_item_t *ips_removeq_copp(ips_copp_queue_t *,
335 ips_copp_wait_item_t *);
336static ips_copp_wait_item_t *ips_removeq_copp_head(ips_copp_queue_t *);
337
338static int ips_is_passthru(Scsi_Cmnd *);
339static int ips_make_passthru(ips_ha_t *, Scsi_Cmnd *, ips_scb_t *, int);
340static int ips_usrcmd(ips_ha_t *, ips_passthru_t *, ips_scb_t *);
341static void ips_cleanup_passthru(ips_ha_t *, ips_scb_t *);
342static void ips_scmd_buf_write(Scsi_Cmnd * scmd, void *data,
343 unsigned int count);
344static void ips_scmd_buf_read(Scsi_Cmnd * scmd, void *data, unsigned int count);
345
346static int ips_proc_info(struct Scsi_Host *, char *, char **, off_t, int, int);
347static int ips_host_info(ips_ha_t *, char *, off_t, int);
348static void copy_mem_info(IPS_INFOSTR *, char *, int);
349static int copy_info(IPS_INFOSTR *, char *, ...);
350static int ips_get_version_info(ips_ha_t * ha, dma_addr_t, int intr);
351static void ips_version_check(ips_ha_t * ha, int intr);
352static int ips_abort_init(ips_ha_t * ha, int index);
353static int ips_init_phase2(int index);
354
355static int ips_init_phase1(struct pci_dev *pci_dev, int *indexPtr);
356static int ips_register_scsi(int index);
357
Jack Hammeree807c22005-08-29 10:44:34 -0400358static int ips_poll_for_flush_complete(ips_ha_t * ha);
359static void ips_flush_and_reset(ips_ha_t *ha);
360
Linus Torvalds1da177e2005-04-16 15:20:36 -0700361/*
362 * global variables
363 */
364static const char ips_name[] = "ips";
365static struct Scsi_Host *ips_sh[IPS_MAX_ADAPTERS]; /* Array of host controller structures */
366static ips_ha_t *ips_ha[IPS_MAX_ADAPTERS]; /* Array of HA structures */
367static unsigned int ips_next_controller;
368static unsigned int ips_num_controllers;
369static unsigned int ips_released_controllers;
370static int ips_hotplug;
371static int ips_cmd_timeout = 60;
372static int ips_reset_timeout = 60 * 5;
373static int ips_force_memio = 1; /* Always use Memory Mapped I/O */
374static int ips_force_i2o = 1; /* Always use I2O command delivery */
375static int ips_ioctlsize = IPS_IOCTL_SIZE; /* Size of the ioctl buffer */
376static int ips_cd_boot; /* Booting from Manager CD */
377static char *ips_FlashData = NULL; /* CD Boot - Flash Data Buffer */
378static dma_addr_t ips_flashbusaddr;
379static long ips_FlashDataInUse; /* CD Boot - Flash Data In Use Flag */
380static uint32_t MaxLiteCmds = 32; /* Max Active Cmds for a Lite Adapter */
381static Scsi_Host_Template ips_driver_template = {
382 .detect = ips_detect,
383 .release = ips_release,
384 .info = ips_info,
385 .queuecommand = ips_queue,
386 .eh_abort_handler = ips_eh_abort,
387 .eh_host_reset_handler = ips_eh_reset,
388 .proc_name = "ips",
389#if LINUX_VERSION_CODE > KERNEL_VERSION(2,5,0)
390 .proc_info = ips_proc_info,
391 .slave_configure = ips_slave_configure,
392#else
393 .proc_info = ips_proc24_info,
394 .select_queue_depths = ips_select_queue_depth,
395#endif
396 .bios_param = ips_biosparam,
397 .this_id = -1,
398 .sg_tablesize = IPS_MAX_SG,
399 .cmd_per_lun = 3,
400 .use_clustering = ENABLE_CLUSTERING,
401#if LINUX_VERSION_CODE < KERNEL_VERSION(2,5,0)
402 .use_new_eh_code = 1,
403#endif
404#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,4,20) && LINUX_VERSION_CODE < KERNEL_VERSION(2,5,0)
405 .highmem_io = 1,
406#endif
407};
408
409static IPS_DEFINE_COMPAT_TABLE( Compatable ); /* Version Compatability Table */
410
411
412/* This table describes all ServeRAID Adapters */
413static struct pci_device_id ips_pci_table[] = {
414 { 0x1014, 0x002E, PCI_ANY_ID, PCI_ANY_ID, 0, 0 },
415 { 0x1014, 0x01BD, PCI_ANY_ID, PCI_ANY_ID, 0, 0 },
416 { 0x9005, 0x0250, PCI_ANY_ID, PCI_ANY_ID, 0, 0 },
417 { 0, }
418};
419
420MODULE_DEVICE_TABLE( pci, ips_pci_table );
421
422static char ips_hot_plug_name[] = "ips";
423
424static int __devinit ips_insert_device(struct pci_dev *pci_dev, const struct pci_device_id *ent);
425static void __devexit ips_remove_device(struct pci_dev *pci_dev);
426
427static struct pci_driver ips_pci_driver = {
428 .name = ips_hot_plug_name,
429 .id_table = ips_pci_table,
430 .probe = ips_insert_device,
431 .remove = __devexit_p(ips_remove_device),
432};
433
434
435/*
436 * Necessary forward function protoypes
437 */
438static int ips_halt(struct notifier_block *nb, ulong event, void *buf);
439
440#define MAX_ADAPTER_NAME 15
441
442static char ips_adapter_name[][30] = {
443 "ServeRAID",
444 "ServeRAID II",
445 "ServeRAID on motherboard",
446 "ServeRAID on motherboard",
447 "ServeRAID 3H",
448 "ServeRAID 3L",
449 "ServeRAID 4H",
450 "ServeRAID 4M",
451 "ServeRAID 4L",
452 "ServeRAID 4Mx",
453 "ServeRAID 4Lx",
454 "ServeRAID 5i",
455 "ServeRAID 5i",
456 "ServeRAID 6M",
457 "ServeRAID 6i",
458 "ServeRAID 7t",
459 "ServeRAID 7k",
460 "ServeRAID 7M"
461};
462
463static struct notifier_block ips_notifier = {
464 ips_halt, NULL, 0
465};
466
467/*
468 * Direction table
469 */
470static char ips_command_direction[] = {
471 IPS_DATA_NONE, IPS_DATA_NONE, IPS_DATA_IN, IPS_DATA_IN, IPS_DATA_OUT,
472 IPS_DATA_IN, IPS_DATA_IN, IPS_DATA_OUT, IPS_DATA_IN, IPS_DATA_UNK,
473 IPS_DATA_OUT, IPS_DATA_OUT, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
474 IPS_DATA_IN, IPS_DATA_NONE, IPS_DATA_NONE, IPS_DATA_IN, IPS_DATA_OUT,
475 IPS_DATA_IN, IPS_DATA_OUT, IPS_DATA_NONE, IPS_DATA_NONE, IPS_DATA_OUT,
476 IPS_DATA_NONE, IPS_DATA_IN, IPS_DATA_NONE, IPS_DATA_IN, IPS_DATA_OUT,
477 IPS_DATA_NONE, IPS_DATA_UNK, IPS_DATA_IN, IPS_DATA_UNK, IPS_DATA_IN,
478 IPS_DATA_UNK, IPS_DATA_OUT, IPS_DATA_IN, IPS_DATA_UNK, IPS_DATA_UNK,
479 IPS_DATA_IN, IPS_DATA_IN, IPS_DATA_OUT, IPS_DATA_NONE, IPS_DATA_UNK,
480 IPS_DATA_IN, IPS_DATA_OUT, IPS_DATA_OUT, IPS_DATA_OUT, IPS_DATA_OUT,
481 IPS_DATA_OUT, IPS_DATA_NONE, IPS_DATA_IN, IPS_DATA_NONE, IPS_DATA_NONE,
482 IPS_DATA_IN, IPS_DATA_OUT, IPS_DATA_OUT, IPS_DATA_OUT, IPS_DATA_OUT,
483 IPS_DATA_IN, IPS_DATA_OUT, IPS_DATA_IN, IPS_DATA_OUT, IPS_DATA_OUT,
484 IPS_DATA_OUT, IPS_DATA_IN, IPS_DATA_IN, IPS_DATA_IN, IPS_DATA_NONE,
485 IPS_DATA_UNK, IPS_DATA_NONE, IPS_DATA_NONE, IPS_DATA_NONE, IPS_DATA_UNK,
486 IPS_DATA_NONE, IPS_DATA_OUT, IPS_DATA_IN, IPS_DATA_UNK, IPS_DATA_UNK,
487 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
488 IPS_DATA_OUT, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
489 IPS_DATA_IN, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
490 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
491 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
492 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
493 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
494 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
495 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
496 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
497 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
498 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
499 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
500 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
501 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
502 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
503 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
504 IPS_DATA_NONE, IPS_DATA_NONE, IPS_DATA_UNK, IPS_DATA_IN, IPS_DATA_NONE,
505 IPS_DATA_OUT, IPS_DATA_UNK, IPS_DATA_NONE, IPS_DATA_UNK, IPS_DATA_OUT,
506 IPS_DATA_OUT, IPS_DATA_OUT, IPS_DATA_OUT, IPS_DATA_OUT, IPS_DATA_NONE,
507 IPS_DATA_UNK, IPS_DATA_IN, IPS_DATA_OUT, IPS_DATA_IN, IPS_DATA_IN,
508 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
509 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
510 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
511 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
512 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
513 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
514 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
515 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
516 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
517 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_OUT,
518 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
519 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
520 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK,
521 IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK, IPS_DATA_UNK
522};
523
524
525/****************************************************************************/
526/* */
527/* Routine Name: ips_setup */
528/* */
529/* Routine Description: */
530/* */
531/* setup parameters to the driver */
532/* */
533/****************************************************************************/
534static int
535ips_setup(char *ips_str)
536{
537
538 int i;
539 char *key;
540 char *value;
541 IPS_OPTION options[] = {
542 {"noi2o", &ips_force_i2o, 0},
543 {"nommap", &ips_force_memio, 0},
544 {"ioctlsize", &ips_ioctlsize, IPS_IOCTL_SIZE},
545 {"cdboot", &ips_cd_boot, 0},
546 {"maxcmds", &MaxLiteCmds, 32},
547 };
548
549 /* Don't use strtok() anymore ( if 2.4 Kernel or beyond ) */
550 /* Search for value */
551 while ((key = strsep(&ips_str, ",."))) {
552 if (!*key)
553 continue;
554 value = strchr(key, ':');
555 if (value)
556 *value++ = '\0';
557 /*
558 * We now have key/value pairs.
559 * Update the variables
560 */
561 for (i = 0; i < (sizeof (options) / sizeof (options[0])); i++) {
562 if (strnicmp
563 (key, options[i].option_name,
564 strlen(options[i].option_name)) == 0) {
565 if (value)
566 *options[i].option_flag =
567 simple_strtoul(value, NULL, 0);
568 else
569 *options[i].option_flag =
570 options[i].option_value;
571 break;
572 }
573 }
574 }
575
576 return (1);
577}
578
579__setup("ips=", ips_setup);
580
581/****************************************************************************/
582/* */
583/* Routine Name: ips_detect */
584/* */
585/* Routine Description: */
586/* */
587/* Detect and initialize the driver */
588/* */
589/* NOTE: this routine is called under the io_request_lock spinlock */
590/* */
591/****************************************************************************/
592static int
593ips_detect(Scsi_Host_Template * SHT)
594{
595 int i;
596
597 METHOD_TRACE("ips_detect", 1);
598
599#ifdef MODULE
600 if (ips)
601 ips_setup(ips);
602#endif
603
604 for (i = 0; i < ips_num_controllers; i++) {
605 if (ips_register_scsi(i))
606 ips_free(ips_ha[i]);
607 ips_released_controllers++;
608 }
609 ips_hotplug = 1;
610 return (ips_num_controllers);
611}
612
613/****************************************************************************/
614/* configure the function pointers to use the functions that will work */
615/* with the found version of the adapter */
616/****************************************************************************/
617static void
618ips_setup_funclist(ips_ha_t * ha)
619{
620
621 /*
622 * Setup Functions
623 */
624 if (IPS_IS_MORPHEUS(ha) || IPS_IS_MARCO(ha)) {
625 /* morpheus / marco / sebring */
626 ha->func.isintr = ips_isintr_morpheus;
627 ha->func.isinit = ips_isinit_morpheus;
628 ha->func.issue = ips_issue_i2o_memio;
629 ha->func.init = ips_init_morpheus;
630 ha->func.statupd = ips_statupd_morpheus;
631 ha->func.reset = ips_reset_morpheus;
632 ha->func.intr = ips_intr_morpheus;
633 ha->func.enableint = ips_enable_int_morpheus;
634 } else if (IPS_USE_MEMIO(ha)) {
635 /* copperhead w/MEMIO */
636 ha->func.isintr = ips_isintr_copperhead_memio;
637 ha->func.isinit = ips_isinit_copperhead_memio;
638 ha->func.init = ips_init_copperhead_memio;
639 ha->func.statupd = ips_statupd_copperhead_memio;
640 ha->func.statinit = ips_statinit_memio;
641 ha->func.reset = ips_reset_copperhead_memio;
642 ha->func.intr = ips_intr_copperhead;
643 ha->func.erasebios = ips_erase_bios_memio;
644 ha->func.programbios = ips_program_bios_memio;
645 ha->func.verifybios = ips_verify_bios_memio;
646 ha->func.enableint = ips_enable_int_copperhead_memio;
647 if (IPS_USE_I2O_DELIVER(ha))
648 ha->func.issue = ips_issue_i2o_memio;
649 else
650 ha->func.issue = ips_issue_copperhead_memio;
651 } else {
652 /* copperhead */
653 ha->func.isintr = ips_isintr_copperhead;
654 ha->func.isinit = ips_isinit_copperhead;
655 ha->func.init = ips_init_copperhead;
656 ha->func.statupd = ips_statupd_copperhead;
657 ha->func.statinit = ips_statinit;
658 ha->func.reset = ips_reset_copperhead;
659 ha->func.intr = ips_intr_copperhead;
660 ha->func.erasebios = ips_erase_bios;
661 ha->func.programbios = ips_program_bios;
662 ha->func.verifybios = ips_verify_bios;
663 ha->func.enableint = ips_enable_int_copperhead;
664
665 if (IPS_USE_I2O_DELIVER(ha))
666 ha->func.issue = ips_issue_i2o;
667 else
668 ha->func.issue = ips_issue_copperhead;
669 }
670}
671
672/****************************************************************************/
673/* */
674/* Routine Name: ips_release */
675/* */
676/* Routine Description: */
677/* */
678/* Remove a driver */
679/* */
680/****************************************************************************/
681static int
682ips_release(struct Scsi_Host *sh)
683{
684 ips_scb_t *scb;
685 ips_ha_t *ha;
686 int i;
687
688 METHOD_TRACE("ips_release", 1);
689
690 for (i = 0; i < IPS_MAX_ADAPTERS && ips_sh[i] != sh; i++) ;
691
692 if (i == IPS_MAX_ADAPTERS) {
693 printk(KERN_WARNING
694 "(%s) release, invalid Scsi_Host pointer.\n", ips_name);
695 BUG();
696 return (FALSE);
697 }
698
699 ha = IPS_HA(sh);
700
701 if (!ha)
702 return (FALSE);
703
704 /* flush the cache on the controller */
705 scb = &ha->scbs[ha->max_cmds - 1];
706
707 ips_init_scb(ha, scb);
708
709 scb->timeout = ips_cmd_timeout;
710 scb->cdb[0] = IPS_CMD_FLUSH;
711
712 scb->cmd.flush_cache.op_code = IPS_CMD_FLUSH;
713 scb->cmd.flush_cache.command_id = IPS_COMMAND_ID(ha, scb);
714 scb->cmd.flush_cache.state = IPS_NORM_STATE;
715 scb->cmd.flush_cache.reserved = 0;
716 scb->cmd.flush_cache.reserved2 = 0;
717 scb->cmd.flush_cache.reserved3 = 0;
718 scb->cmd.flush_cache.reserved4 = 0;
719
720 IPS_PRINTK(KERN_WARNING, ha->pcidev, "Flushing Cache.\n");
721
722 /* send command */
723 if (ips_send_wait(ha, scb, ips_cmd_timeout, IPS_INTR_ON) == IPS_FAILURE)
724 IPS_PRINTK(KERN_WARNING, ha->pcidev, "Incomplete Flush.\n");
725
726 IPS_PRINTK(KERN_WARNING, ha->pcidev, "Flushing Complete.\n");
727
728 ips_sh[i] = NULL;
729 ips_ha[i] = NULL;
730
731 /* free extra memory */
732 ips_free(ha);
733
734 /* Free I/O Region */
735 if (ha->io_addr)
736 release_region(ha->io_addr, ha->io_len);
737
738 /* free IRQ */
739 free_irq(ha->irq, ha);
740
741 IPS_REMOVE_HOST(sh);
742 scsi_host_put(sh);
743
744 ips_released_controllers++;
745
746 return (FALSE);
747}
748
749/****************************************************************************/
750/* */
751/* Routine Name: ips_halt */
752/* */
753/* Routine Description: */
754/* */
755/* Perform cleanup when the system reboots */
756/* */
757/****************************************************************************/
758static int
759ips_halt(struct notifier_block *nb, ulong event, void *buf)
760{
761 ips_scb_t *scb;
762 ips_ha_t *ha;
763 int i;
764
765 if ((event != SYS_RESTART) && (event != SYS_HALT) &&
766 (event != SYS_POWER_OFF))
767 return (NOTIFY_DONE);
768
769 for (i = 0; i < ips_next_controller; i++) {
770 ha = (ips_ha_t *) ips_ha[i];
771
772 if (!ha)
773 continue;
774
775 if (!ha->active)
776 continue;
777
778 /* flush the cache on the controller */
779 scb = &ha->scbs[ha->max_cmds - 1];
780
781 ips_init_scb(ha, scb);
782
783 scb->timeout = ips_cmd_timeout;
784 scb->cdb[0] = IPS_CMD_FLUSH;
785
786 scb->cmd.flush_cache.op_code = IPS_CMD_FLUSH;
787 scb->cmd.flush_cache.command_id = IPS_COMMAND_ID(ha, scb);
788 scb->cmd.flush_cache.state = IPS_NORM_STATE;
789 scb->cmd.flush_cache.reserved = 0;
790 scb->cmd.flush_cache.reserved2 = 0;
791 scb->cmd.flush_cache.reserved3 = 0;
792 scb->cmd.flush_cache.reserved4 = 0;
793
794 IPS_PRINTK(KERN_WARNING, ha->pcidev, "Flushing Cache.\n");
795
796 /* send command */
797 if (ips_send_wait(ha, scb, ips_cmd_timeout, IPS_INTR_ON) ==
798 IPS_FAILURE)
799 IPS_PRINTK(KERN_WARNING, ha->pcidev,
800 "Incomplete Flush.\n");
801 else
802 IPS_PRINTK(KERN_WARNING, ha->pcidev,
803 "Flushing Complete.\n");
804 }
805
806 return (NOTIFY_OK);
807}
808
809/****************************************************************************/
810/* */
811/* Routine Name: ips_eh_abort */
812/* */
813/* Routine Description: */
814/* */
815/* Abort a command (using the new error code stuff) */
816/* Note: this routine is called under the io_request_lock */
817/****************************************************************************/
818int
819ips_eh_abort(Scsi_Cmnd * SC)
820{
821 ips_ha_t *ha;
822 ips_copp_wait_item_t *item;
823 int ret;
Jeff Garzik 8fa728a2005-05-28 07:54:40 -0400824 unsigned long cpu_flags;
825 struct Scsi_Host *host;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700826
827 METHOD_TRACE("ips_eh_abort", 1);
828
829 if (!SC)
830 return (FAILED);
831
Jeff Garzik 8fa728a2005-05-28 07:54:40 -0400832 host = SC->device->host;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700833 ha = (ips_ha_t *) SC->device->host->hostdata;
834
835 if (!ha)
836 return (FAILED);
837
838 if (!ha->active)
839 return (FAILED);
840
Jeff Garzik 8fa728a2005-05-28 07:54:40 -0400841 IPS_LOCK_SAVE(host->host_lock, cpu_flags);
842
Linus Torvalds1da177e2005-04-16 15:20:36 -0700843 /* See if the command is on the copp queue */
844 item = ha->copp_waitlist.head;
845 while ((item) && (item->scsi_cmd != SC))
846 item = item->next;
847
848 if (item) {
849 /* Found it */
850 ips_removeq_copp(&ha->copp_waitlist, item);
851 ret = (SUCCESS);
852
853 /* See if the command is on the wait queue */
854 } else if (ips_removeq_wait(&ha->scb_waitlist, SC)) {
855 /* command not sent yet */
856 ret = (SUCCESS);
857 } else {
858 /* command must have already been sent */
859 ret = (FAILED);
860 }
Jeff Garzik 8fa728a2005-05-28 07:54:40 -0400861
862 IPS_UNLOCK_RESTORE(host->host_lock, cpu_flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700863 return ret;
864}
865
866/****************************************************************************/
867/* */
868/* Routine Name: ips_eh_reset */
869/* */
870/* Routine Description: */
871/* */
872/* Reset the controller (with new eh error code) */
873/* */
874/* NOTE: this routine is called under the io_request_lock spinlock */
875/* */
876/****************************************************************************/
877static int
Jeff Garzik df0ae242005-05-28 07:57:14 -0400878__ips_eh_reset(Scsi_Cmnd * SC)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700879{
880 int ret;
881 int i;
882 ips_ha_t *ha;
883 ips_scb_t *scb;
884 ips_copp_wait_item_t *item;
885
886 METHOD_TRACE("ips_eh_reset", 1);
887
888#ifdef NO_IPS_RESET
889 return (FAILED);
890#else
891
892 if (!SC) {
893 DEBUG(1, "Reset called with NULL scsi command");
894
895 return (FAILED);
896 }
897
898 ha = (ips_ha_t *) SC->device->host->hostdata;
899
900 if (!ha) {
901 DEBUG(1, "Reset called with NULL ha struct");
902
903 return (FAILED);
904 }
905
906 if (!ha->active)
907 return (FAILED);
908
909 /* See if the command is on the copp queue */
910 item = ha->copp_waitlist.head;
911 while ((item) && (item->scsi_cmd != SC))
912 item = item->next;
913
914 if (item) {
915 /* Found it */
916 ips_removeq_copp(&ha->copp_waitlist, item);
917 return (SUCCESS);
918 }
919
920 /* See if the command is on the wait queue */
921 if (ips_removeq_wait(&ha->scb_waitlist, SC)) {
922 /* command not sent yet */
923 return (SUCCESS);
924 }
925
926 /* An explanation for the casual observer: */
927 /* Part of the function of a RAID controller is automatic error */
928 /* detection and recovery. As such, the only problem that physically */
929 /* resetting an adapter will ever fix is when, for some reason, */
930 /* the driver is not successfully communicating with the adapter. */
931 /* Therefore, we will attempt to flush this adapter. If that succeeds, */
932 /* then there's no real purpose in a physical reset. This will complete */
933 /* much faster and avoids any problems that might be caused by a */
934 /* physical reset ( such as having to fail all the outstanding I/O's ). */
935
936 if (ha->ioctl_reset == 0) { /* IF Not an IOCTL Requested Reset */
937 scb = &ha->scbs[ha->max_cmds - 1];
938
939 ips_init_scb(ha, scb);
940
941 scb->timeout = ips_cmd_timeout;
942 scb->cdb[0] = IPS_CMD_FLUSH;
943
944 scb->cmd.flush_cache.op_code = IPS_CMD_FLUSH;
945 scb->cmd.flush_cache.command_id = IPS_COMMAND_ID(ha, scb);
946 scb->cmd.flush_cache.state = IPS_NORM_STATE;
947 scb->cmd.flush_cache.reserved = 0;
948 scb->cmd.flush_cache.reserved2 = 0;
949 scb->cmd.flush_cache.reserved3 = 0;
950 scb->cmd.flush_cache.reserved4 = 0;
951
952 /* Attempt the flush command */
953 ret = ips_send_wait(ha, scb, ips_cmd_timeout, IPS_INTR_IORL);
954 if (ret == IPS_SUCCESS) {
955 IPS_PRINTK(KERN_NOTICE, ha->pcidev,
956 "Reset Request - Flushed Cache\n");
957 return (SUCCESS);
958 }
959 }
960
961 /* Either we can't communicate with the adapter or it's an IOCTL request */
962 /* from a utility. A physical reset is needed at this point. */
963
964 ha->ioctl_reset = 0; /* Reset the IOCTL Requested Reset Flag */
965
966 /*
967 * command must have already been sent
968 * reset the controller
969 */
970 IPS_PRINTK(KERN_NOTICE, ha->pcidev, "Resetting controller.\n");
971 ret = (*ha->func.reset) (ha);
972
973 if (!ret) {
974 Scsi_Cmnd *scsi_cmd;
975
976 IPS_PRINTK(KERN_NOTICE, ha->pcidev,
977 "Controller reset failed - controller now offline.\n");
978
979 /* Now fail all of the active commands */
980 DEBUG_VAR(1, "(%s%d) Failing active commands",
981 ips_name, ha->host_num);
982
983 while ((scb = ips_removeq_scb_head(&ha->scb_activelist))) {
984 scb->scsi_cmd->result = DID_ERROR << 16;
985 scb->scsi_cmd->scsi_done(scb->scsi_cmd);
986 ips_freescb(ha, scb);
987 }
988
989 /* Now fail all of the pending commands */
990 DEBUG_VAR(1, "(%s%d) Failing pending commands",
991 ips_name, ha->host_num);
992
993 while ((scsi_cmd = ips_removeq_wait_head(&ha->scb_waitlist))) {
994 scsi_cmd->result = DID_ERROR;
995 scsi_cmd->scsi_done(scsi_cmd);
996 }
997
998 ha->active = FALSE;
999 return (FAILED);
1000 }
1001
1002 if (!ips_clear_adapter(ha, IPS_INTR_IORL)) {
1003 Scsi_Cmnd *scsi_cmd;
1004
1005 IPS_PRINTK(KERN_NOTICE, ha->pcidev,
1006 "Controller reset failed - controller now offline.\n");
1007
1008 /* Now fail all of the active commands */
1009 DEBUG_VAR(1, "(%s%d) Failing active commands",
1010 ips_name, ha->host_num);
1011
1012 while ((scb = ips_removeq_scb_head(&ha->scb_activelist))) {
1013 scb->scsi_cmd->result = DID_ERROR << 16;
1014 scb->scsi_cmd->scsi_done(scb->scsi_cmd);
1015 ips_freescb(ha, scb);
1016 }
1017
1018 /* Now fail all of the pending commands */
1019 DEBUG_VAR(1, "(%s%d) Failing pending commands",
1020 ips_name, ha->host_num);
1021
1022 while ((scsi_cmd = ips_removeq_wait_head(&ha->scb_waitlist))) {
1023 scsi_cmd->result = DID_ERROR << 16;
1024 scsi_cmd->scsi_done(scsi_cmd);
1025 }
1026
1027 ha->active = FALSE;
1028 return (FAILED);
1029 }
1030
1031 /* FFDC */
1032 if (le32_to_cpu(ha->subsys->param[3]) & 0x300000) {
1033 struct timeval tv;
1034
1035 do_gettimeofday(&tv);
1036 ha->last_ffdc = tv.tv_sec;
1037 ha->reset_count++;
1038 ips_ffdc_reset(ha, IPS_INTR_IORL);
1039 }
1040
1041 /* Now fail all of the active commands */
1042 DEBUG_VAR(1, "(%s%d) Failing active commands", ips_name, ha->host_num);
1043
1044 while ((scb = ips_removeq_scb_head(&ha->scb_activelist))) {
1045 scb->scsi_cmd->result =
1046 (DID_RESET << 16) | (SUGGEST_RETRY << 24);
1047 scb->scsi_cmd->scsi_done(scb->scsi_cmd);
1048 ips_freescb(ha, scb);
1049 }
1050
1051 /* Reset DCDB active command bits */
1052 for (i = 1; i < ha->nbus; i++)
1053 ha->dcdb_active[i - 1] = 0;
1054
1055 /* Reset the number of active IOCTLs */
1056 ha->num_ioctl = 0;
1057
1058 ips_next(ha, IPS_INTR_IORL);
1059
1060 return (SUCCESS);
1061#endif /* NO_IPS_RESET */
1062
1063}
1064
Jeff Garzik df0ae242005-05-28 07:57:14 -04001065static int
1066ips_eh_reset(Scsi_Cmnd * SC)
1067{
1068 int rc;
1069
1070 spin_lock_irq(SC->device->host->host_lock);
1071 rc = __ips_eh_reset(SC);
1072 spin_unlock_irq(SC->device->host->host_lock);
1073
1074 return rc;
1075}
1076
Linus Torvalds1da177e2005-04-16 15:20:36 -07001077/****************************************************************************/
1078/* */
1079/* Routine Name: ips_queue */
1080/* */
1081/* Routine Description: */
1082/* */
1083/* Send a command to the controller */
1084/* */
1085/* NOTE: */
1086/* Linux obtains io_request_lock before calling this function */
1087/* */
1088/****************************************************************************/
1089static int
1090ips_queue(Scsi_Cmnd * SC, void (*done) (Scsi_Cmnd *))
1091{
1092 ips_ha_t *ha;
1093 ips_passthru_t *pt;
1094
1095 METHOD_TRACE("ips_queue", 1);
1096
1097 ha = (ips_ha_t *) SC->device->host->hostdata;
1098
1099 if (!ha)
1100 return (1);
1101
1102 if (!ha->active)
1103 return (DID_ERROR);
1104
1105 if (ips_is_passthru(SC)) {
1106 if (ha->copp_waitlist.count == IPS_MAX_IOCTL_QUEUE) {
1107 SC->result = DID_BUS_BUSY << 16;
1108 done(SC);
1109
1110 return (0);
1111 }
1112 } else if (ha->scb_waitlist.count == IPS_MAX_QUEUE) {
1113 SC->result = DID_BUS_BUSY << 16;
1114 done(SC);
1115
1116 return (0);
1117 }
1118
1119 SC->scsi_done = done;
1120
1121 DEBUG_VAR(2, "(%s%d): ips_queue: cmd 0x%X (%d %d %d)",
1122 ips_name,
1123 ha->host_num,
1124 SC->cmnd[0],
1125 SC->device->channel, SC->device->id, SC->device->lun);
1126
1127 /* Check for command to initiator IDs */
Jeff Garzik422c0d62005-10-24 18:05:09 -04001128 if ((scmd_channel(SC) > 0)
1129 && (scmd_id(SC) == ha->ha_id[scmd_channel(SC)])) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001130 SC->result = DID_NO_CONNECT << 16;
1131 done(SC);
1132
1133 return (0);
1134 }
1135
1136 if (ips_is_passthru(SC)) {
1137
1138 ips_copp_wait_item_t *scratch;
1139
1140 /* A Reset IOCTL is only sent by the boot CD in extreme cases. */
1141 /* There can never be any system activity ( network or disk ), but check */
1142 /* anyway just as a good practice. */
1143 pt = (ips_passthru_t *) SC->request_buffer;
1144 if ((pt->CoppCP.cmd.reset.op_code == IPS_CMD_RESET_CHANNEL) &&
1145 (pt->CoppCP.cmd.reset.adapter_flag == 1)) {
1146 if (ha->scb_activelist.count != 0) {
1147 SC->result = DID_BUS_BUSY << 16;
1148 done(SC);
1149 return (0);
1150 }
1151 ha->ioctl_reset = 1; /* This reset request is from an IOCTL */
1152 ips_eh_reset(SC);
1153 SC->result = DID_OK << 16;
1154 SC->scsi_done(SC);
1155 return (0);
1156 }
1157
1158 /* allocate space for the scribble */
1159 scratch = kmalloc(sizeof (ips_copp_wait_item_t), GFP_ATOMIC);
1160
1161 if (!scratch) {
1162 SC->result = DID_ERROR << 16;
1163 done(SC);
1164
1165 return (0);
1166 }
1167
1168 scratch->scsi_cmd = SC;
1169 scratch->next = NULL;
1170
1171 ips_putq_copp_tail(&ha->copp_waitlist, scratch);
1172 } else {
1173 ips_putq_wait_tail(&ha->scb_waitlist, SC);
1174 }
1175
1176 ips_next(ha, IPS_INTR_IORL);
1177
1178 return (0);
1179}
1180
1181/****************************************************************************/
1182/* */
1183/* Routine Name: ips_biosparam */
1184/* */
1185/* Routine Description: */
1186/* */
1187/* Set bios geometry for the controller */
1188/* */
1189/****************************************************************************/
1190static int
1191#if LINUX_VERSION_CODE < KERNEL_VERSION(2,5,0)
1192ips_biosparam(Disk * disk, kdev_t dev, int geom[])
1193{
1194 ips_ha_t *ha = (ips_ha_t *) disk->device->host->hostdata;
1195 unsigned long capacity = disk->capacity;
1196#else
1197ips_biosparam(struct scsi_device *sdev, struct block_device *bdev,
1198 sector_t capacity, int geom[])
1199{
1200 ips_ha_t *ha = (ips_ha_t *) sdev->host->hostdata;
1201#endif
1202 int heads;
1203 int sectors;
1204 int cylinders;
1205
1206 METHOD_TRACE("ips_biosparam", 1);
1207
1208 if (!ha)
1209 /* ?!?! host adater info invalid */
1210 return (0);
1211
1212 if (!ha->active)
1213 return (0);
1214
1215 if (!ips_read_adapter_status(ha, IPS_INTR_ON))
1216 /* ?!?! Enquiry command failed */
1217 return (0);
1218
1219 if ((capacity > 0x400000) && ((ha->enq->ucMiscFlag & 0x8) == 0)) {
1220 heads = IPS_NORM_HEADS;
1221 sectors = IPS_NORM_SECTORS;
1222 } else {
1223 heads = IPS_COMP_HEADS;
1224 sectors = IPS_COMP_SECTORS;
1225 }
1226
1227 cylinders = (unsigned long) capacity / (heads * sectors);
1228
1229 DEBUG_VAR(2, "Geometry: heads: %d, sectors: %d, cylinders: %d",
1230 heads, sectors, cylinders);
1231
1232 geom[0] = heads;
1233 geom[1] = sectors;
1234 geom[2] = cylinders;
1235
1236 return (0);
1237}
1238
1239#if LINUX_VERSION_CODE < KERNEL_VERSION(2,5,0)
1240
1241/* ips_proc24_info is a wrapper around ips_proc_info *
1242 * for compatibility with the 2.4 scsi parameters */
1243static int
1244ips_proc24_info(char *buffer, char **start, off_t offset, int length,
1245 int hostno, int func)
1246{
1247 int i;
1248
1249 for (i = 0; i < ips_next_controller; i++) {
1250 if (ips_sh[i] && ips_sh[i]->host_no == hostno) {
1251 return ips_proc_info(ips_sh[i], buffer, start,
1252 offset, length, func);
1253 }
1254 }
1255 return -EINVAL;
1256}
1257
1258/****************************************************************************/
1259/* */
1260/* Routine Name: ips_select_queue_depth */
1261/* */
1262/* Routine Description: */
1263/* */
1264/* Select queue depths for the devices on the contoller */
1265/* */
1266/****************************************************************************/
1267static void
1268ips_select_queue_depth(struct Scsi_Host *host, Scsi_Device * scsi_devs)
1269{
1270 Scsi_Device *device;
1271 ips_ha_t *ha;
1272 int count = 0;
1273 int min;
1274
1275 ha = IPS_HA(host);
1276 min = ha->max_cmds / 4;
1277
1278 for (device = scsi_devs; device; device = device->next) {
1279 if (device->host == host) {
1280 if ((device->channel == 0) && (device->type == 0))
1281 count++;
1282 }
1283 }
1284
1285 for (device = scsi_devs; device; device = device->next) {
1286 if (device->host == host) {
1287 if ((device->channel == 0) && (device->type == 0)) {
1288 device->queue_depth =
1289 (ha->max_cmds - 1) / count;
1290 if (device->queue_depth < min)
1291 device->queue_depth = min;
1292 } else {
1293 device->queue_depth = 2;
1294 }
1295
1296 if (device->queue_depth < 2)
1297 device->queue_depth = 2;
1298 }
1299 }
1300}
1301
1302#else
1303/****************************************************************************/
1304/* */
1305/* Routine Name: ips_slave_configure */
1306/* */
1307/* Routine Description: */
1308/* */
1309/* Set queue depths on devices once scan is complete */
1310/* */
1311/****************************************************************************/
1312static int
1313ips_slave_configure(Scsi_Device * SDptr)
1314{
1315 ips_ha_t *ha;
1316 int min;
1317
1318 ha = IPS_HA(SDptr->host);
1319 if (SDptr->tagged_supported && SDptr->type == TYPE_DISK) {
1320 min = ha->max_cmds / 2;
1321 if (ha->enq->ucLogDriveCount <= 2)
1322 min = ha->max_cmds - 1;
1323 scsi_adjust_queue_depth(SDptr, MSG_ORDERED_TAG, min);
1324 }
1325 return 0;
1326}
1327#endif
1328
1329/****************************************************************************/
1330/* */
1331/* Routine Name: do_ipsintr */
1332/* */
1333/* Routine Description: */
1334/* */
1335/* Wrapper for the interrupt handler */
1336/* */
1337/****************************************************************************/
1338static irqreturn_t
1339do_ipsintr(int irq, void *dev_id, struct pt_regs * regs)
1340{
1341 ips_ha_t *ha;
1342 unsigned long cpu_flags;
1343 struct Scsi_Host *host;
1344 int irqstatus;
1345
1346 METHOD_TRACE("do_ipsintr", 2);
1347
1348 ha = (ips_ha_t *) dev_id;
1349 if (!ha)
1350 return IRQ_NONE;
1351 host = ips_sh[ha->host_num];
1352 /* interrupt during initialization */
1353 if (!host) {
1354 (*ha->func.intr) (ha);
1355 return IRQ_HANDLED;
1356 }
1357
1358 IPS_LOCK_SAVE(host->host_lock, cpu_flags);
1359
1360 if (!ha->active) {
1361 IPS_UNLOCK_RESTORE(host->host_lock, cpu_flags);
1362 return IRQ_HANDLED;
1363 }
1364
1365 irqstatus = (*ha->func.intr) (ha);
1366
1367 IPS_UNLOCK_RESTORE(host->host_lock, cpu_flags);
1368
1369 /* start the next command */
1370 ips_next(ha, IPS_INTR_ON);
1371 return IRQ_RETVAL(irqstatus);
1372}
1373
1374/****************************************************************************/
1375/* */
1376/* Routine Name: ips_intr_copperhead */
1377/* */
1378/* Routine Description: */
1379/* */
1380/* Polling interrupt handler */
1381/* */
1382/* ASSUMES interrupts are disabled */
1383/* */
1384/****************************************************************************/
1385int
1386ips_intr_copperhead(ips_ha_t * ha)
1387{
1388 ips_stat_t *sp;
1389 ips_scb_t *scb;
1390 IPS_STATUS cstatus;
1391 int intrstatus;
1392
1393 METHOD_TRACE("ips_intr", 2);
1394
1395 if (!ha)
1396 return 0;
1397
1398 if (!ha->active)
1399 return 0;
1400
1401 intrstatus = (*ha->func.isintr) (ha);
1402
1403 if (!intrstatus) {
1404 /*
1405 * Unexpected/Shared interrupt
1406 */
1407
1408 return 0;
1409 }
1410
1411 while (TRUE) {
1412 sp = &ha->sp;
1413
1414 intrstatus = (*ha->func.isintr) (ha);
1415
1416 if (!intrstatus)
1417 break;
1418 else
1419 cstatus.value = (*ha->func.statupd) (ha);
1420
1421 if (cstatus.fields.command_id > (IPS_MAX_CMDS - 1)) {
1422 /* Spurious Interupt ? */
1423 continue;
1424 }
1425
1426 ips_chkstatus(ha, &cstatus);
1427 scb = (ips_scb_t *) sp->scb_addr;
1428
1429 /*
1430 * use the callback function to finish things up
1431 * NOTE: interrupts are OFF for this
1432 */
1433 (*scb->callback) (ha, scb);
1434 } /* end while */
1435 return 1;
1436}
1437
1438/****************************************************************************/
1439/* */
1440/* Routine Name: ips_intr_morpheus */
1441/* */
1442/* Routine Description: */
1443/* */
1444/* Polling interrupt handler */
1445/* */
1446/* ASSUMES interrupts are disabled */
1447/* */
1448/****************************************************************************/
1449int
1450ips_intr_morpheus(ips_ha_t * ha)
1451{
1452 ips_stat_t *sp;
1453 ips_scb_t *scb;
1454 IPS_STATUS cstatus;
1455 int intrstatus;
1456
1457 METHOD_TRACE("ips_intr_morpheus", 2);
1458
1459 if (!ha)
1460 return 0;
1461
1462 if (!ha->active)
1463 return 0;
1464
1465 intrstatus = (*ha->func.isintr) (ha);
1466
1467 if (!intrstatus) {
1468 /*
1469 * Unexpected/Shared interrupt
1470 */
1471
1472 return 0;
1473 }
1474
1475 while (TRUE) {
1476 sp = &ha->sp;
1477
1478 intrstatus = (*ha->func.isintr) (ha);
1479
1480 if (!intrstatus)
1481 break;
1482 else
1483 cstatus.value = (*ha->func.statupd) (ha);
1484
1485 if (cstatus.value == 0xffffffff)
1486 /* No more to process */
1487 break;
1488
1489 if (cstatus.fields.command_id > (IPS_MAX_CMDS - 1)) {
1490 IPS_PRINTK(KERN_WARNING, ha->pcidev,
1491 "Spurious interrupt; no ccb.\n");
1492
1493 continue;
1494 }
1495
1496 ips_chkstatus(ha, &cstatus);
1497 scb = (ips_scb_t *) sp->scb_addr;
1498
1499 /*
1500 * use the callback function to finish things up
1501 * NOTE: interrupts are OFF for this
1502 */
1503 (*scb->callback) (ha, scb);
1504 } /* end while */
1505 return 1;
1506}
1507
1508/****************************************************************************/
1509/* */
1510/* Routine Name: ips_info */
1511/* */
1512/* Routine Description: */
1513/* */
1514/* Return info about the driver */
1515/* */
1516/****************************************************************************/
1517static const char *
1518ips_info(struct Scsi_Host *SH)
1519{
1520 static char buffer[256];
1521 char *bp;
1522 ips_ha_t *ha;
1523
1524 METHOD_TRACE("ips_info", 1);
1525
1526 ha = IPS_HA(SH);
1527
1528 if (!ha)
1529 return (NULL);
1530
1531 bp = &buffer[0];
1532 memset(bp, 0, sizeof (buffer));
1533
1534 sprintf(bp, "%s%s%s Build %d", "IBM PCI ServeRAID ",
1535 IPS_VERSION_HIGH, IPS_VERSION_LOW, IPS_BUILD_IDENT);
1536
1537 if (ha->ad_type > 0 && ha->ad_type <= MAX_ADAPTER_NAME) {
1538 strcat(bp, " <");
1539 strcat(bp, ips_adapter_name[ha->ad_type - 1]);
1540 strcat(bp, ">");
1541 }
1542
1543 return (bp);
1544}
1545
1546/****************************************************************************/
1547/* */
1548/* Routine Name: ips_proc_info */
1549/* */
1550/* Routine Description: */
1551/* */
1552/* The passthru interface for the driver */
1553/* */
1554/****************************************************************************/
1555static int
1556ips_proc_info(struct Scsi_Host *host, char *buffer, char **start, off_t offset,
1557 int length, int func)
1558{
1559 int i;
1560 int ret;
1561 ips_ha_t *ha = NULL;
1562
1563 METHOD_TRACE("ips_proc_info", 1);
1564
1565 /* Find our host structure */
1566 for (i = 0; i < ips_next_controller; i++) {
1567 if (ips_sh[i]) {
1568 if (ips_sh[i] == host) {
1569 ha = (ips_ha_t *) ips_sh[i]->hostdata;
1570 break;
1571 }
1572 }
1573 }
1574
1575 if (!ha)
1576 return (-EINVAL);
1577
1578 if (func) {
1579 /* write */
1580 return (0);
1581 } else {
1582 /* read */
1583 if (start)
1584 *start = buffer;
1585
1586 ret = ips_host_info(ha, buffer, offset, length);
1587
1588 return (ret);
1589 }
1590}
1591
1592/*--------------------------------------------------------------------------*/
1593/* Helper Functions */
1594/*--------------------------------------------------------------------------*/
1595
1596/****************************************************************************/
1597/* */
1598/* Routine Name: ips_is_passthru */
1599/* */
1600/* Routine Description: */
1601/* */
1602/* Determine if the specified SCSI command is really a passthru command */
1603/* */
1604/****************************************************************************/
1605static int
1606ips_is_passthru(Scsi_Cmnd * SC)
1607{
Jack Hammera3632fa2005-10-25 14:13:03 -04001608 unsigned long flags;
1609
Linus Torvalds1da177e2005-04-16 15:20:36 -07001610 METHOD_TRACE("ips_is_passthru", 1);
1611
1612 if (!SC)
1613 return (0);
1614
1615 if ((SC->cmnd[0] == IPS_IOCTL_COMMAND) &&
1616 (SC->device->channel == 0) &&
1617 (SC->device->id == IPS_ADAPTER_ID) &&
1618 (SC->device->lun == 0) && SC->request_buffer) {
1619 if ((!SC->use_sg) && SC->request_bufflen &&
1620 (((char *) SC->request_buffer)[0] == 'C') &&
1621 (((char *) SC->request_buffer)[1] == 'O') &&
1622 (((char *) SC->request_buffer)[2] == 'P') &&
1623 (((char *) SC->request_buffer)[3] == 'P'))
1624 return 1;
1625 else if (SC->use_sg) {
1626 struct scatterlist *sg = SC->request_buffer;
Jack Hammera3632fa2005-10-25 14:13:03 -04001627 char *buffer;
1628
1629 /* kmap_atomic() ensures addressability of the user buffer.*/
1630 /* local_irq_save() protects the KM_IRQ0 address slot. */
1631 local_irq_save(flags);
1632 buffer = kmap_atomic(sg->page, KM_IRQ0) + sg->offset;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001633 if (buffer && buffer[0] == 'C' && buffer[1] == 'O' &&
Jack Hammera3632fa2005-10-25 14:13:03 -04001634 buffer[2] == 'P' && buffer[3] == 'P') {
1635 kunmap_atomic(buffer - sg->offset, KM_IRQ0);
1636 local_irq_restore(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001637 return 1;
Jack Hammera3632fa2005-10-25 14:13:03 -04001638 }
1639 kunmap_atomic(buffer - sg->offset, KM_IRQ0);
1640 local_irq_restore(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001641 }
1642 }
1643 return 0;
1644}
1645
1646/****************************************************************************/
1647/* */
1648/* Routine Name: ips_alloc_passthru_buffer */
1649/* */
1650/* Routine Description: */
1651/* allocate a buffer large enough for the ioctl data if the ioctl buffer */
1652/* is too small or doesn't exist */
1653/****************************************************************************/
1654static int
1655ips_alloc_passthru_buffer(ips_ha_t * ha, int length)
1656{
1657 void *bigger_buf;
1658 dma_addr_t dma_busaddr;
1659
1660 if (ha->ioctl_data && length <= ha->ioctl_len)
1661 return 0;
1662 /* there is no buffer or it's not big enough, allocate a new one */
1663 bigger_buf = pci_alloc_consistent(ha->pcidev, length, &dma_busaddr);
1664 if (bigger_buf) {
1665 /* free the old memory */
1666 pci_free_consistent(ha->pcidev, ha->ioctl_len, ha->ioctl_data,
1667 ha->ioctl_busaddr);
1668 /* use the new memory */
1669 ha->ioctl_data = (char *) bigger_buf;
1670 ha->ioctl_len = length;
1671 ha->ioctl_busaddr = dma_busaddr;
1672 } else {
1673 return -1;
1674 }
1675 return 0;
1676}
1677
1678/****************************************************************************/
1679/* */
1680/* Routine Name: ips_make_passthru */
1681/* */
1682/* Routine Description: */
1683/* */
1684/* Make a passthru command out of the info in the Scsi block */
1685/* */
1686/****************************************************************************/
1687static int
1688ips_make_passthru(ips_ha_t * ha, Scsi_Cmnd * SC, ips_scb_t * scb, int intr)
1689{
1690 ips_passthru_t *pt;
1691 int length = 0;
1692 int ret;
1693
1694 METHOD_TRACE("ips_make_passthru", 1);
1695
1696 if (!SC->use_sg) {
1697 length = SC->request_bufflen;
1698 } else {
1699 struct scatterlist *sg = SC->request_buffer;
1700 int i;
1701 for (i = 0; i < SC->use_sg; i++)
1702 length += sg[i].length;
1703 }
1704 if (length < sizeof (ips_passthru_t)) {
1705 /* wrong size */
1706 DEBUG_VAR(1, "(%s%d) Passthru structure wrong size",
1707 ips_name, ha->host_num);
1708 return (IPS_FAILURE);
1709 }
1710 if (ips_alloc_passthru_buffer(ha, length)) {
1711 /* allocation failure! If ha->ioctl_data exists, use it to return
1712 some error codes. Return a failed command to the scsi layer. */
1713 if (ha->ioctl_data) {
1714 pt = (ips_passthru_t *) ha->ioctl_data;
1715 ips_scmd_buf_read(SC, pt, sizeof (ips_passthru_t));
1716 pt->BasicStatus = 0x0B;
1717 pt->ExtendedStatus = 0x00;
1718 ips_scmd_buf_write(SC, pt, sizeof (ips_passthru_t));
1719 }
1720 return IPS_FAILURE;
1721 }
1722 ha->ioctl_datasize = length;
1723
1724 ips_scmd_buf_read(SC, ha->ioctl_data, ha->ioctl_datasize);
1725 pt = (ips_passthru_t *) ha->ioctl_data;
1726
1727 /*
1728 * Some notes about the passthru interface used
1729 *
1730 * IF the scsi op_code == 0x0d then we assume
1731 * that the data came along with/goes with the
1732 * packet we received from the sg driver. In this
1733 * case the CmdBSize field of the pt structure is
1734 * used for the size of the buffer.
1735 */
1736
1737 switch (pt->CoppCmd) {
1738 case IPS_NUMCTRLS:
1739 memcpy(ha->ioctl_data + sizeof (ips_passthru_t),
1740 &ips_num_controllers, sizeof (int));
1741 ips_scmd_buf_write(SC, ha->ioctl_data,
1742 sizeof (ips_passthru_t) + sizeof (int));
1743 SC->result = DID_OK << 16;
1744
1745 return (IPS_SUCCESS_IMM);
1746
1747 case IPS_COPPUSRCMD:
1748 case IPS_COPPIOCCMD:
1749 if (SC->cmnd[0] == IPS_IOCTL_COMMAND) {
1750 if (length < (sizeof (ips_passthru_t) + pt->CmdBSize)) {
1751 /* wrong size */
1752 DEBUG_VAR(1,
1753 "(%s%d) Passthru structure wrong size",
1754 ips_name, ha->host_num);
1755
1756 return (IPS_FAILURE);
1757 }
1758
1759 if (ha->device_id == IPS_DEVICEID_COPPERHEAD &&
1760 pt->CoppCP.cmd.flashfw.op_code ==
1761 IPS_CMD_RW_BIOSFW) {
1762 ret = ips_flash_copperhead(ha, pt, scb);
1763 ips_scmd_buf_write(SC, ha->ioctl_data,
1764 sizeof (ips_passthru_t));
1765 return ret;
1766 }
1767 if (ips_usrcmd(ha, pt, scb))
1768 return (IPS_SUCCESS);
1769 else
1770 return (IPS_FAILURE);
1771 }
1772
1773 break;
1774
1775 } /* end switch */
1776
1777 return (IPS_FAILURE);
1778}
1779
1780/****************************************************************************/
1781/* Routine Name: ips_flash_copperhead */
1782/* Routine Description: */
1783/* Flash the BIOS/FW on a Copperhead style controller */
1784/****************************************************************************/
1785static int
1786ips_flash_copperhead(ips_ha_t * ha, ips_passthru_t * pt, ips_scb_t * scb)
1787{
1788 int datasize;
1789
1790 /* Trombone is the only copperhead that can do packet flash, but only
1791 * for firmware. No one said it had to make sence. */
1792 if (IPS_IS_TROMBONE(ha) && pt->CoppCP.cmd.flashfw.type == IPS_FW_IMAGE) {
1793 if (ips_usrcmd(ha, pt, scb))
1794 return IPS_SUCCESS;
1795 else
1796 return IPS_FAILURE;
1797 }
1798 pt->BasicStatus = 0x0B;
1799 pt->ExtendedStatus = 0;
1800 scb->scsi_cmd->result = DID_OK << 16;
1801 /* IF it's OK to Use the "CD BOOT" Flash Buffer, then you can */
1802 /* avoid allocating a huge buffer per adapter ( which can fail ). */
1803 if (pt->CoppCP.cmd.flashfw.type == IPS_BIOS_IMAGE &&
1804 pt->CoppCP.cmd.flashfw.direction == IPS_ERASE_BIOS) {
1805 pt->BasicStatus = 0;
1806 return ips_flash_bios(ha, pt, scb);
1807 } else if (pt->CoppCP.cmd.flashfw.packet_num == 0) {
1808 if (ips_FlashData && !test_and_set_bit(0, &ips_FlashDataInUse)){
1809 ha->flash_data = ips_FlashData;
1810 ha->flash_busaddr = ips_flashbusaddr;
1811 ha->flash_len = PAGE_SIZE << 7;
1812 ha->flash_datasize = 0;
1813 } else if (!ha->flash_data) {
1814 datasize = pt->CoppCP.cmd.flashfw.total_packets *
1815 pt->CoppCP.cmd.flashfw.count;
1816 ha->flash_data = pci_alloc_consistent(ha->pcidev,
1817 datasize,
1818 &ha->flash_busaddr);
1819 if (!ha->flash_data){
1820 printk(KERN_WARNING "Unable to allocate a flash buffer\n");
1821 return IPS_FAILURE;
1822 }
1823 ha->flash_datasize = 0;
1824 ha->flash_len = datasize;
1825 } else
1826 return IPS_FAILURE;
1827 } else {
1828 if (pt->CoppCP.cmd.flashfw.count + ha->flash_datasize >
1829 ha->flash_len) {
1830 ips_free_flash_copperhead(ha);
1831 IPS_PRINTK(KERN_WARNING, ha->pcidev,
1832 "failed size sanity check\n");
1833 return IPS_FAILURE;
1834 }
1835 }
1836 if (!ha->flash_data)
1837 return IPS_FAILURE;
1838 pt->BasicStatus = 0;
1839 memcpy(&ha->flash_data[ha->flash_datasize], pt + 1,
1840 pt->CoppCP.cmd.flashfw.count);
1841 ha->flash_datasize += pt->CoppCP.cmd.flashfw.count;
1842 if (pt->CoppCP.cmd.flashfw.packet_num ==
1843 pt->CoppCP.cmd.flashfw.total_packets - 1) {
1844 if (pt->CoppCP.cmd.flashfw.type == IPS_BIOS_IMAGE)
1845 return ips_flash_bios(ha, pt, scb);
1846 else if (pt->CoppCP.cmd.flashfw.type == IPS_FW_IMAGE)
1847 return ips_flash_firmware(ha, pt, scb);
1848 }
1849 return IPS_SUCCESS_IMM;
1850}
1851
1852/****************************************************************************/
1853/* Routine Name: ips_flash_bios */
1854/* Routine Description: */
1855/* flashes the bios of a copperhead adapter */
1856/****************************************************************************/
1857static int
1858ips_flash_bios(ips_ha_t * ha, ips_passthru_t * pt, ips_scb_t * scb)
1859{
1860
1861 if (pt->CoppCP.cmd.flashfw.type == IPS_BIOS_IMAGE &&
1862 pt->CoppCP.cmd.flashfw.direction == IPS_WRITE_BIOS) {
1863 if ((!ha->func.programbios) || (!ha->func.erasebios) ||
1864 (!ha->func.verifybios))
1865 goto error;
1866 if ((*ha->func.erasebios) (ha)) {
1867 DEBUG_VAR(1,
1868 "(%s%d) flash bios failed - unable to erase flash",
1869 ips_name, ha->host_num);
1870 goto error;
1871 } else
1872 if ((*ha->func.programbios) (ha,
1873 ha->flash_data +
1874 IPS_BIOS_HEADER,
1875 ha->flash_datasize -
1876 IPS_BIOS_HEADER, 0)) {
1877 DEBUG_VAR(1,
1878 "(%s%d) flash bios failed - unable to flash",
1879 ips_name, ha->host_num);
1880 goto error;
1881 } else
1882 if ((*ha->func.verifybios) (ha,
1883 ha->flash_data +
1884 IPS_BIOS_HEADER,
1885 ha->flash_datasize -
1886 IPS_BIOS_HEADER, 0)) {
1887 DEBUG_VAR(1,
1888 "(%s%d) flash bios failed - unable to verify flash",
1889 ips_name, ha->host_num);
1890 goto error;
1891 }
1892 ips_free_flash_copperhead(ha);
1893 return IPS_SUCCESS_IMM;
1894 } else if (pt->CoppCP.cmd.flashfw.type == IPS_BIOS_IMAGE &&
1895 pt->CoppCP.cmd.flashfw.direction == IPS_ERASE_BIOS) {
1896 if (!ha->func.erasebios)
1897 goto error;
1898 if ((*ha->func.erasebios) (ha)) {
1899 DEBUG_VAR(1,
1900 "(%s%d) flash bios failed - unable to erase flash",
1901 ips_name, ha->host_num);
1902 goto error;
1903 }
1904 return IPS_SUCCESS_IMM;
1905 }
1906 error:
1907 pt->BasicStatus = 0x0B;
1908 pt->ExtendedStatus = 0x00;
1909 ips_free_flash_copperhead(ha);
1910 return IPS_FAILURE;
1911}
1912
1913/****************************************************************************/
1914/* */
1915/* Routine Name: ips_fill_scb_sg_single */
1916/* */
1917/* Routine Description: */
1918/* Fill in a single scb sg_list element from an address */
1919/* return a -1 if a breakup occurred */
1920/****************************************************************************/
1921static int
1922ips_fill_scb_sg_single(ips_ha_t * ha, dma_addr_t busaddr,
1923 ips_scb_t * scb, int indx, unsigned int e_len)
1924{
1925
1926 int ret_val = 0;
1927
1928 if ((scb->data_len + e_len) > ha->max_xfer) {
1929 e_len = ha->max_xfer - scb->data_len;
1930 scb->breakup = indx;
1931 ++scb->sg_break;
1932 ret_val = -1;
1933 } else {
1934 scb->breakup = 0;
1935 scb->sg_break = 0;
1936 }
1937 if (IPS_USE_ENH_SGLIST(ha)) {
1938 scb->sg_list.enh_list[indx].address_lo =
1939 cpu_to_le32(pci_dma_lo32(busaddr));
1940 scb->sg_list.enh_list[indx].address_hi =
1941 cpu_to_le32(pci_dma_hi32(busaddr));
1942 scb->sg_list.enh_list[indx].length = cpu_to_le32(e_len);
1943 } else {
1944 scb->sg_list.std_list[indx].address =
1945 cpu_to_le32(pci_dma_lo32(busaddr));
1946 scb->sg_list.std_list[indx].length = cpu_to_le32(e_len);
1947 }
1948
1949 ++scb->sg_len;
1950 scb->data_len += e_len;
1951 return ret_val;
1952}
1953
1954/****************************************************************************/
1955/* Routine Name: ips_flash_firmware */
1956/* Routine Description: */
1957/* flashes the firmware of a copperhead adapter */
1958/****************************************************************************/
1959static int
1960ips_flash_firmware(ips_ha_t * ha, ips_passthru_t * pt, ips_scb_t * scb)
1961{
1962 IPS_SG_LIST sg_list;
1963 uint32_t cmd_busaddr;
1964
1965 if (pt->CoppCP.cmd.flashfw.type == IPS_FW_IMAGE &&
1966 pt->CoppCP.cmd.flashfw.direction == IPS_WRITE_FW) {
1967 memset(&pt->CoppCP.cmd, 0, sizeof (IPS_HOST_COMMAND));
1968 pt->CoppCP.cmd.flashfw.op_code = IPS_CMD_DOWNLOAD;
1969 pt->CoppCP.cmd.flashfw.count = cpu_to_le32(ha->flash_datasize);
1970 } else {
1971 pt->BasicStatus = 0x0B;
1972 pt->ExtendedStatus = 0x00;
1973 ips_free_flash_copperhead(ha);
1974 return IPS_FAILURE;
1975 }
1976 /* Save the S/G list pointer so it doesn't get clobbered */
1977 sg_list.list = scb->sg_list.list;
1978 cmd_busaddr = scb->scb_busaddr;
1979 /* copy in the CP */
1980 memcpy(&scb->cmd, &pt->CoppCP.cmd, sizeof (IPS_IOCTL_CMD));
1981 /* FIX stuff that might be wrong */
1982 scb->sg_list.list = sg_list.list;
1983 scb->scb_busaddr = cmd_busaddr;
1984 scb->bus = scb->scsi_cmd->device->channel;
1985 scb->target_id = scb->scsi_cmd->device->id;
1986 scb->lun = scb->scsi_cmd->device->lun;
1987 scb->sg_len = 0;
1988 scb->data_len = 0;
1989 scb->flags = 0;
1990 scb->op_code = 0;
1991 scb->callback = ipsintr_done;
1992 scb->timeout = ips_cmd_timeout;
1993
1994 scb->data_len = ha->flash_datasize;
1995 scb->data_busaddr =
1996 pci_map_single(ha->pcidev, ha->flash_data, scb->data_len,
1997 IPS_DMA_DIR(scb));
1998 scb->flags |= IPS_SCB_MAP_SINGLE;
1999 scb->cmd.flashfw.command_id = IPS_COMMAND_ID(ha, scb);
2000 scb->cmd.flashfw.buffer_addr = cpu_to_le32(scb->data_busaddr);
2001 if (pt->TimeOut)
2002 scb->timeout = pt->TimeOut;
2003 scb->scsi_cmd->result = DID_OK << 16;
2004 return IPS_SUCCESS;
2005}
2006
2007/****************************************************************************/
2008/* Routine Name: ips_free_flash_copperhead */
2009/* Routine Description: */
2010/* release the memory resources used to hold the flash image */
2011/****************************************************************************/
2012static void
2013ips_free_flash_copperhead(ips_ha_t * ha)
2014{
2015 if (ha->flash_data == ips_FlashData)
2016 test_and_clear_bit(0, &ips_FlashDataInUse);
2017 else if (ha->flash_data)
2018 pci_free_consistent(ha->pcidev, ha->flash_len, ha->flash_data,
2019 ha->flash_busaddr);
2020 ha->flash_data = NULL;
2021}
2022
2023/****************************************************************************/
2024/* */
2025/* Routine Name: ips_usrcmd */
2026/* */
2027/* Routine Description: */
2028/* */
2029/* Process a user command and make it ready to send */
2030/* */
2031/****************************************************************************/
2032static int
2033ips_usrcmd(ips_ha_t * ha, ips_passthru_t * pt, ips_scb_t * scb)
2034{
2035 IPS_SG_LIST sg_list;
2036 uint32_t cmd_busaddr;
2037
2038 METHOD_TRACE("ips_usrcmd", 1);
2039
2040 if ((!scb) || (!pt) || (!ha))
2041 return (0);
2042
2043 /* Save the S/G list pointer so it doesn't get clobbered */
2044 sg_list.list = scb->sg_list.list;
2045 cmd_busaddr = scb->scb_busaddr;
2046 /* copy in the CP */
2047 memcpy(&scb->cmd, &pt->CoppCP.cmd, sizeof (IPS_IOCTL_CMD));
2048 memcpy(&scb->dcdb, &pt->CoppCP.dcdb, sizeof (IPS_DCDB_TABLE));
2049
2050 /* FIX stuff that might be wrong */
2051 scb->sg_list.list = sg_list.list;
2052 scb->scb_busaddr = cmd_busaddr;
2053 scb->bus = scb->scsi_cmd->device->channel;
2054 scb->target_id = scb->scsi_cmd->device->id;
2055 scb->lun = scb->scsi_cmd->device->lun;
2056 scb->sg_len = 0;
2057 scb->data_len = 0;
2058 scb->flags = 0;
2059 scb->op_code = 0;
2060 scb->callback = ipsintr_done;
2061 scb->timeout = ips_cmd_timeout;
2062 scb->cmd.basic_io.command_id = IPS_COMMAND_ID(ha, scb);
2063
2064 /* we don't support DCDB/READ/WRITE Scatter Gather */
2065 if ((scb->cmd.basic_io.op_code == IPS_CMD_READ_SG) ||
2066 (scb->cmd.basic_io.op_code == IPS_CMD_WRITE_SG) ||
2067 (scb->cmd.basic_io.op_code == IPS_CMD_DCDB_SG))
2068 return (0);
2069
2070 if (pt->CmdBSize) {
2071 scb->data_len = pt->CmdBSize;
2072 scb->data_busaddr = ha->ioctl_busaddr + sizeof (ips_passthru_t);
2073 } else {
2074 scb->data_busaddr = 0L;
2075 }
2076
2077 if (scb->cmd.dcdb.op_code == IPS_CMD_DCDB)
2078 scb->cmd.dcdb.dcdb_address = cpu_to_le32(scb->scb_busaddr +
2079 (unsigned long) &scb->
2080 dcdb -
2081 (unsigned long) scb);
2082
2083 if (pt->CmdBSize) {
2084 if (scb->cmd.dcdb.op_code == IPS_CMD_DCDB)
2085 scb->dcdb.buffer_pointer =
2086 cpu_to_le32(scb->data_busaddr);
2087 else
2088 scb->cmd.basic_io.sg_addr =
2089 cpu_to_le32(scb->data_busaddr);
2090 }
2091
2092 /* set timeouts */
2093 if (pt->TimeOut) {
2094 scb->timeout = pt->TimeOut;
2095
2096 if (pt->TimeOut <= 10)
2097 scb->dcdb.cmd_attribute |= IPS_TIMEOUT10;
2098 else if (pt->TimeOut <= 60)
2099 scb->dcdb.cmd_attribute |= IPS_TIMEOUT60;
2100 else
2101 scb->dcdb.cmd_attribute |= IPS_TIMEOUT20M;
2102 }
2103
2104 /* assume success */
2105 scb->scsi_cmd->result = DID_OK << 16;
2106
2107 /* success */
2108 return (1);
2109}
2110
2111/****************************************************************************/
2112/* */
2113/* Routine Name: ips_cleanup_passthru */
2114/* */
2115/* Routine Description: */
2116/* */
2117/* Cleanup after a passthru command */
2118/* */
2119/****************************************************************************/
2120static void
2121ips_cleanup_passthru(ips_ha_t * ha, ips_scb_t * scb)
2122{
2123 ips_passthru_t *pt;
2124
2125 METHOD_TRACE("ips_cleanup_passthru", 1);
2126
2127 if ((!scb) || (!scb->scsi_cmd) || (!scb->scsi_cmd->request_buffer)) {
2128 DEBUG_VAR(1, "(%s%d) couldn't cleanup after passthru",
2129 ips_name, ha->host_num);
2130
2131 return;
2132 }
2133 pt = (ips_passthru_t *) ha->ioctl_data;
2134
2135 /* Copy data back to the user */
2136 if (scb->cmd.dcdb.op_code == IPS_CMD_DCDB) /* Copy DCDB Back to Caller's Area */
2137 memcpy(&pt->CoppCP.dcdb, &scb->dcdb, sizeof (IPS_DCDB_TABLE));
2138
2139 pt->BasicStatus = scb->basic_status;
2140 pt->ExtendedStatus = scb->extended_status;
2141 pt->AdapterType = ha->ad_type;
2142
2143 if (ha->device_id == IPS_DEVICEID_COPPERHEAD &&
2144 (scb->cmd.flashfw.op_code == IPS_CMD_DOWNLOAD ||
2145 scb->cmd.flashfw.op_code == IPS_CMD_RW_BIOSFW))
2146 ips_free_flash_copperhead(ha);
2147
2148 ips_scmd_buf_write(scb->scsi_cmd, ha->ioctl_data, ha->ioctl_datasize);
2149}
2150
2151/****************************************************************************/
2152/* */
2153/* Routine Name: ips_host_info */
2154/* */
2155/* Routine Description: */
2156/* */
2157/* The passthru interface for the driver */
2158/* */
2159/****************************************************************************/
2160static int
2161ips_host_info(ips_ha_t * ha, char *ptr, off_t offset, int len)
2162{
2163 IPS_INFOSTR info;
2164
2165 METHOD_TRACE("ips_host_info", 1);
2166
2167 info.buffer = ptr;
2168 info.length = len;
2169 info.offset = offset;
2170 info.pos = 0;
2171 info.localpos = 0;
2172
2173 copy_info(&info, "\nIBM ServeRAID General Information:\n\n");
2174
2175 if ((le32_to_cpu(ha->nvram->signature) == IPS_NVRAM_P5_SIG) &&
2176 (le16_to_cpu(ha->nvram->adapter_type) != 0))
2177 copy_info(&info, "\tController Type : %s\n",
2178 ips_adapter_name[ha->ad_type - 1]);
2179 else
2180 copy_info(&info,
2181 "\tController Type : Unknown\n");
2182
2183 if (ha->io_addr)
2184 copy_info(&info,
2185 "\tIO region : 0x%lx (%d bytes)\n",
2186 ha->io_addr, ha->io_len);
2187
2188 if (ha->mem_addr) {
2189 copy_info(&info,
2190 "\tMemory region : 0x%lx (%d bytes)\n",
2191 ha->mem_addr, ha->mem_len);
2192 copy_info(&info,
2193 "\tShared memory address : 0x%lx\n",
2194 ha->mem_ptr);
2195 }
2196
2197 copy_info(&info, "\tIRQ number : %d\n", ha->irq);
2198
2199 /* For the Next 3 lines Check for Binary 0 at the end and don't include it if it's there. */
2200 /* That keeps everything happy for "text" operations on the proc file. */
2201
2202 if (le32_to_cpu(ha->nvram->signature) == IPS_NVRAM_P5_SIG) {
2203 if (ha->nvram->bios_low[3] == 0) {
2204 copy_info(&info,
2205 "\tBIOS Version : %c%c%c%c%c%c%c\n",
2206 ha->nvram->bios_high[0], ha->nvram->bios_high[1],
2207 ha->nvram->bios_high[2], ha->nvram->bios_high[3],
2208 ha->nvram->bios_low[0], ha->nvram->bios_low[1],
2209 ha->nvram->bios_low[2]);
2210
2211 } else {
2212 copy_info(&info,
2213 "\tBIOS Version : %c%c%c%c%c%c%c%c\n",
2214 ha->nvram->bios_high[0], ha->nvram->bios_high[1],
2215 ha->nvram->bios_high[2], ha->nvram->bios_high[3],
2216 ha->nvram->bios_low[0], ha->nvram->bios_low[1],
2217 ha->nvram->bios_low[2], ha->nvram->bios_low[3]);
2218 }
2219
2220 }
2221
2222 if (ha->enq->CodeBlkVersion[7] == 0) {
2223 copy_info(&info,
2224 "\tFirmware Version : %c%c%c%c%c%c%c\n",
2225 ha->enq->CodeBlkVersion[0], ha->enq->CodeBlkVersion[1],
2226 ha->enq->CodeBlkVersion[2], ha->enq->CodeBlkVersion[3],
2227 ha->enq->CodeBlkVersion[4], ha->enq->CodeBlkVersion[5],
2228 ha->enq->CodeBlkVersion[6]);
2229 } else {
2230 copy_info(&info,
2231 "\tFirmware Version : %c%c%c%c%c%c%c%c\n",
2232 ha->enq->CodeBlkVersion[0], ha->enq->CodeBlkVersion[1],
2233 ha->enq->CodeBlkVersion[2], ha->enq->CodeBlkVersion[3],
2234 ha->enq->CodeBlkVersion[4], ha->enq->CodeBlkVersion[5],
2235 ha->enq->CodeBlkVersion[6], ha->enq->CodeBlkVersion[7]);
2236 }
2237
2238 if (ha->enq->BootBlkVersion[7] == 0) {
2239 copy_info(&info,
2240 "\tBoot Block Version : %c%c%c%c%c%c%c\n",
2241 ha->enq->BootBlkVersion[0], ha->enq->BootBlkVersion[1],
2242 ha->enq->BootBlkVersion[2], ha->enq->BootBlkVersion[3],
2243 ha->enq->BootBlkVersion[4], ha->enq->BootBlkVersion[5],
2244 ha->enq->BootBlkVersion[6]);
2245 } else {
2246 copy_info(&info,
2247 "\tBoot Block Version : %c%c%c%c%c%c%c%c\n",
2248 ha->enq->BootBlkVersion[0], ha->enq->BootBlkVersion[1],
2249 ha->enq->BootBlkVersion[2], ha->enq->BootBlkVersion[3],
2250 ha->enq->BootBlkVersion[4], ha->enq->BootBlkVersion[5],
2251 ha->enq->BootBlkVersion[6], ha->enq->BootBlkVersion[7]);
2252 }
2253
2254 copy_info(&info, "\tDriver Version : %s%s\n",
2255 IPS_VERSION_HIGH, IPS_VERSION_LOW);
2256
2257 copy_info(&info, "\tDriver Build : %d\n",
2258 IPS_BUILD_IDENT);
2259
2260 copy_info(&info, "\tMax Physical Devices : %d\n",
2261 ha->enq->ucMaxPhysicalDevices);
2262 copy_info(&info, "\tMax Active Commands : %d\n",
2263 ha->max_cmds);
2264 copy_info(&info, "\tCurrent Queued Commands : %d\n",
2265 ha->scb_waitlist.count);
2266 copy_info(&info, "\tCurrent Active Commands : %d\n",
2267 ha->scb_activelist.count - ha->num_ioctl);
2268 copy_info(&info, "\tCurrent Queued PT Commands : %d\n",
2269 ha->copp_waitlist.count);
2270 copy_info(&info, "\tCurrent Active PT Commands : %d\n",
2271 ha->num_ioctl);
2272
2273 copy_info(&info, "\n");
2274
2275 return (info.localpos);
2276}
2277
2278/****************************************************************************/
2279/* */
2280/* Routine Name: copy_mem_info */
2281/* */
2282/* Routine Description: */
2283/* */
2284/* Copy data into an IPS_INFOSTR structure */
2285/* */
2286/****************************************************************************/
2287static void
2288copy_mem_info(IPS_INFOSTR * info, char *data, int len)
2289{
2290 METHOD_TRACE("copy_mem_info", 1);
2291
2292 if (info->pos + len < info->offset) {
2293 info->pos += len;
2294 return;
2295 }
2296
2297 if (info->pos < info->offset) {
2298 data += (info->offset - info->pos);
2299 len -= (info->offset - info->pos);
2300 info->pos += (info->offset - info->pos);
2301 }
2302
2303 if (info->localpos + len > info->length)
2304 len = info->length - info->localpos;
2305
2306 if (len > 0) {
2307 memcpy(info->buffer + info->localpos, data, len);
2308 info->pos += len;
2309 info->localpos += len;
2310 }
2311}
2312
2313/****************************************************************************/
2314/* */
2315/* Routine Name: copy_info */
2316/* */
2317/* Routine Description: */
2318/* */
2319/* printf style wrapper for an info structure */
2320/* */
2321/****************************************************************************/
2322static int
2323copy_info(IPS_INFOSTR * info, char *fmt, ...)
2324{
2325 va_list args;
2326 char buf[128];
2327 int len;
2328
2329 METHOD_TRACE("copy_info", 1);
2330
2331 va_start(args, fmt);
2332 len = vsprintf(buf, fmt, args);
2333 va_end(args);
2334
2335 copy_mem_info(info, buf, len);
2336
2337 return (len);
2338}
2339
2340/****************************************************************************/
2341/* */
2342/* Routine Name: ips_identify_controller */
2343/* */
2344/* Routine Description: */
2345/* */
2346/* Identify this controller */
2347/* */
2348/****************************************************************************/
2349static void
2350ips_identify_controller(ips_ha_t * ha)
2351{
2352 METHOD_TRACE("ips_identify_controller", 1);
2353
2354 switch (ha->device_id) {
2355 case IPS_DEVICEID_COPPERHEAD:
2356 if (ha->revision_id <= IPS_REVID_SERVERAID) {
2357 ha->ad_type = IPS_ADTYPE_SERVERAID;
2358 } else if (ha->revision_id == IPS_REVID_SERVERAID2) {
2359 ha->ad_type = IPS_ADTYPE_SERVERAID2;
2360 } else if (ha->revision_id == IPS_REVID_NAVAJO) {
2361 ha->ad_type = IPS_ADTYPE_NAVAJO;
2362 } else if ((ha->revision_id == IPS_REVID_SERVERAID2)
2363 && (ha->slot_num == 0)) {
2364 ha->ad_type = IPS_ADTYPE_KIOWA;
2365 } else if ((ha->revision_id >= IPS_REVID_CLARINETP1) &&
2366 (ha->revision_id <= IPS_REVID_CLARINETP3)) {
2367 if (ha->enq->ucMaxPhysicalDevices == 15)
2368 ha->ad_type = IPS_ADTYPE_SERVERAID3L;
2369 else
2370 ha->ad_type = IPS_ADTYPE_SERVERAID3;
2371 } else if ((ha->revision_id >= IPS_REVID_TROMBONE32) &&
2372 (ha->revision_id <= IPS_REVID_TROMBONE64)) {
2373 ha->ad_type = IPS_ADTYPE_SERVERAID4H;
2374 }
2375 break;
2376
2377 case IPS_DEVICEID_MORPHEUS:
2378 switch (ha->subdevice_id) {
2379 case IPS_SUBDEVICEID_4L:
2380 ha->ad_type = IPS_ADTYPE_SERVERAID4L;
2381 break;
2382
2383 case IPS_SUBDEVICEID_4M:
2384 ha->ad_type = IPS_ADTYPE_SERVERAID4M;
2385 break;
2386
2387 case IPS_SUBDEVICEID_4MX:
2388 ha->ad_type = IPS_ADTYPE_SERVERAID4MX;
2389 break;
2390
2391 case IPS_SUBDEVICEID_4LX:
2392 ha->ad_type = IPS_ADTYPE_SERVERAID4LX;
2393 break;
2394
2395 case IPS_SUBDEVICEID_5I2:
2396 ha->ad_type = IPS_ADTYPE_SERVERAID5I2;
2397 break;
2398
2399 case IPS_SUBDEVICEID_5I1:
2400 ha->ad_type = IPS_ADTYPE_SERVERAID5I1;
2401 break;
2402 }
2403
2404 break;
2405
2406 case IPS_DEVICEID_MARCO:
2407 switch (ha->subdevice_id) {
2408 case IPS_SUBDEVICEID_6M:
2409 ha->ad_type = IPS_ADTYPE_SERVERAID6M;
2410 break;
2411 case IPS_SUBDEVICEID_6I:
2412 ha->ad_type = IPS_ADTYPE_SERVERAID6I;
2413 break;
2414 case IPS_SUBDEVICEID_7k:
2415 ha->ad_type = IPS_ADTYPE_SERVERAID7k;
2416 break;
2417 case IPS_SUBDEVICEID_7M:
2418 ha->ad_type = IPS_ADTYPE_SERVERAID7M;
2419 break;
2420 }
2421 break;
2422 }
2423}
2424
2425/****************************************************************************/
2426/* */
2427/* Routine Name: ips_get_bios_version */
2428/* */
2429/* Routine Description: */
2430/* */
2431/* Get the BIOS revision number */
2432/* */
2433/****************************************************************************/
2434static void
2435ips_get_bios_version(ips_ha_t * ha, int intr)
2436{
2437 ips_scb_t *scb;
2438 int ret;
2439 uint8_t major;
2440 uint8_t minor;
2441 uint8_t subminor;
2442 uint8_t *buffer;
2443 char hexDigits[] =
2444 { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C',
2445 'D', 'E', 'F' };
2446
2447 METHOD_TRACE("ips_get_bios_version", 1);
2448
2449 major = 0;
2450 minor = 0;
2451
2452 strncpy(ha->bios_version, " ?", 8);
2453
2454 if (ha->device_id == IPS_DEVICEID_COPPERHEAD) {
2455 if (IPS_USE_MEMIO(ha)) {
2456 /* Memory Mapped I/O */
2457
2458 /* test 1st byte */
2459 writel(0, ha->mem_ptr + IPS_REG_FLAP);
2460 if (ha->revision_id == IPS_REVID_TROMBONE64)
2461 udelay(25); /* 25 us */
2462
2463 if (readb(ha->mem_ptr + IPS_REG_FLDP) != 0x55)
2464 return;
2465
2466 writel(1, ha->mem_ptr + IPS_REG_FLAP);
2467 if (ha->revision_id == IPS_REVID_TROMBONE64)
2468 udelay(25); /* 25 us */
2469
2470 if (readb(ha->mem_ptr + IPS_REG_FLDP) != 0xAA)
2471 return;
2472
2473 /* Get Major version */
2474 writel(0x1FF, ha->mem_ptr + IPS_REG_FLAP);
2475 if (ha->revision_id == IPS_REVID_TROMBONE64)
2476 udelay(25); /* 25 us */
2477
2478 major = readb(ha->mem_ptr + IPS_REG_FLDP);
2479
2480 /* Get Minor version */
2481 writel(0x1FE, ha->mem_ptr + IPS_REG_FLAP);
2482 if (ha->revision_id == IPS_REVID_TROMBONE64)
2483 udelay(25); /* 25 us */
2484 minor = readb(ha->mem_ptr + IPS_REG_FLDP);
2485
2486 /* Get SubMinor version */
2487 writel(0x1FD, ha->mem_ptr + IPS_REG_FLAP);
2488 if (ha->revision_id == IPS_REVID_TROMBONE64)
2489 udelay(25); /* 25 us */
2490 subminor = readb(ha->mem_ptr + IPS_REG_FLDP);
2491
2492 } else {
2493 /* Programmed I/O */
2494
2495 /* test 1st byte */
2496 outl(0, ha->io_addr + IPS_REG_FLAP);
2497 if (ha->revision_id == IPS_REVID_TROMBONE64)
2498 udelay(25); /* 25 us */
2499
2500 if (inb(ha->io_addr + IPS_REG_FLDP) != 0x55)
2501 return;
2502
2503 outl(cpu_to_le32(1), ha->io_addr + IPS_REG_FLAP);
2504 if (ha->revision_id == IPS_REVID_TROMBONE64)
2505 udelay(25); /* 25 us */
2506
2507 if (inb(ha->io_addr + IPS_REG_FLDP) != 0xAA)
2508 return;
2509
2510 /* Get Major version */
2511 outl(cpu_to_le32(0x1FF), ha->io_addr + IPS_REG_FLAP);
2512 if (ha->revision_id == IPS_REVID_TROMBONE64)
2513 udelay(25); /* 25 us */
2514
2515 major = inb(ha->io_addr + IPS_REG_FLDP);
2516
2517 /* Get Minor version */
2518 outl(cpu_to_le32(0x1FE), ha->io_addr + IPS_REG_FLAP);
2519 if (ha->revision_id == IPS_REVID_TROMBONE64)
2520 udelay(25); /* 25 us */
2521
2522 minor = inb(ha->io_addr + IPS_REG_FLDP);
2523
2524 /* Get SubMinor version */
2525 outl(cpu_to_le32(0x1FD), ha->io_addr + IPS_REG_FLAP);
2526 if (ha->revision_id == IPS_REVID_TROMBONE64)
2527 udelay(25); /* 25 us */
2528
2529 subminor = inb(ha->io_addr + IPS_REG_FLDP);
2530
2531 }
2532 } else {
2533 /* Morpheus Family - Send Command to the card */
2534
2535 buffer = ha->ioctl_data;
2536
2537 memset(buffer, 0, 0x1000);
2538
2539 scb = &ha->scbs[ha->max_cmds - 1];
2540
2541 ips_init_scb(ha, scb);
2542
2543 scb->timeout = ips_cmd_timeout;
2544 scb->cdb[0] = IPS_CMD_RW_BIOSFW;
2545
2546 scb->cmd.flashfw.op_code = IPS_CMD_RW_BIOSFW;
2547 scb->cmd.flashfw.command_id = IPS_COMMAND_ID(ha, scb);
2548 scb->cmd.flashfw.type = 1;
2549 scb->cmd.flashfw.direction = 0;
2550 scb->cmd.flashfw.count = cpu_to_le32(0x800);
2551 scb->cmd.flashfw.total_packets = 1;
2552 scb->cmd.flashfw.packet_num = 0;
2553 scb->data_len = 0x1000;
2554 scb->cmd.flashfw.buffer_addr = ha->ioctl_busaddr;
2555
2556 /* issue the command */
2557 if (((ret =
2558 ips_send_wait(ha, scb, ips_cmd_timeout,
2559 intr)) == IPS_FAILURE)
2560 || (ret == IPS_SUCCESS_IMM)
2561 || ((scb->basic_status & IPS_GSC_STATUS_MASK) > 1)) {
2562 /* Error occurred */
2563
2564 return;
2565 }
2566
2567 if ((buffer[0xC0] == 0x55) && (buffer[0xC1] == 0xAA)) {
2568 major = buffer[0x1ff + 0xC0]; /* Offset 0x1ff after the header (0xc0) */
2569 minor = buffer[0x1fe + 0xC0]; /* Offset 0x1fe after the header (0xc0) */
2570 subminor = buffer[0x1fd + 0xC0]; /* Offset 0x1fd after the header (0xc0) */
2571 } else {
2572 return;
2573 }
2574 }
2575
2576 ha->bios_version[0] = hexDigits[(major & 0xF0) >> 4];
2577 ha->bios_version[1] = '.';
2578 ha->bios_version[2] = hexDigits[major & 0x0F];
2579 ha->bios_version[3] = hexDigits[subminor];
2580 ha->bios_version[4] = '.';
2581 ha->bios_version[5] = hexDigits[(minor & 0xF0) >> 4];
2582 ha->bios_version[6] = hexDigits[minor & 0x0F];
2583 ha->bios_version[7] = 0;
2584}
2585
2586/****************************************************************************/
2587/* */
2588/* Routine Name: ips_hainit */
2589/* */
2590/* Routine Description: */
2591/* */
2592/* Initialize the controller */
2593/* */
2594/* NOTE: Assumes to be called from with a lock */
2595/* */
2596/****************************************************************************/
2597static int
2598ips_hainit(ips_ha_t * ha)
2599{
2600 int i;
2601 struct timeval tv;
2602
2603 METHOD_TRACE("ips_hainit", 1);
2604
2605 if (!ha)
2606 return (0);
2607
2608 if (ha->func.statinit)
2609 (*ha->func.statinit) (ha);
2610
2611 if (ha->func.enableint)
2612 (*ha->func.enableint) (ha);
2613
2614 /* Send FFDC */
2615 ha->reset_count = 1;
2616 do_gettimeofday(&tv);
2617 ha->last_ffdc = tv.tv_sec;
2618 ips_ffdc_reset(ha, IPS_INTR_IORL);
2619
2620 if (!ips_read_config(ha, IPS_INTR_IORL)) {
2621 IPS_PRINTK(KERN_WARNING, ha->pcidev,
2622 "unable to read config from controller.\n");
2623
2624 return (0);
2625 }
2626 /* end if */
2627 if (!ips_read_adapter_status(ha, IPS_INTR_IORL)) {
2628 IPS_PRINTK(KERN_WARNING, ha->pcidev,
2629 "unable to read controller status.\n");
2630
2631 return (0);
2632 }
2633
2634 /* Identify this controller */
2635 ips_identify_controller(ha);
2636
2637 if (!ips_read_subsystem_parameters(ha, IPS_INTR_IORL)) {
2638 IPS_PRINTK(KERN_WARNING, ha->pcidev,
2639 "unable to read subsystem parameters.\n");
2640
2641 return (0);
2642 }
2643
2644 /* write nvram user page 5 */
2645 if (!ips_write_driver_status(ha, IPS_INTR_IORL)) {
2646 IPS_PRINTK(KERN_WARNING, ha->pcidev,
2647 "unable to write driver info to controller.\n");
2648
2649 return (0);
2650 }
2651
2652 /* If there are Logical Drives and a Reset Occurred, then an EraseStripeLock is Needed */
2653 if ((ha->conf->ucLogDriveCount > 0) && (ha->requires_esl == 1))
2654 ips_clear_adapter(ha, IPS_INTR_IORL);
2655
2656 /* set limits on SID, LUN, BUS */
2657 ha->ntargets = IPS_MAX_TARGETS + 1;
2658 ha->nlun = 1;
2659 ha->nbus = (ha->enq->ucMaxPhysicalDevices / IPS_MAX_TARGETS) + 1;
2660
2661 switch (ha->conf->logical_drive[0].ucStripeSize) {
2662 case 4:
2663 ha->max_xfer = 0x10000;
2664 break;
2665
2666 case 5:
2667 ha->max_xfer = 0x20000;
2668 break;
2669
2670 case 6:
2671 ha->max_xfer = 0x40000;
2672 break;
2673
2674 case 7:
2675 default:
2676 ha->max_xfer = 0x80000;
2677 break;
2678 }
2679
2680 /* setup max concurrent commands */
2681 if (le32_to_cpu(ha->subsys->param[4]) & 0x1) {
2682 /* Use the new method */
2683 ha->max_cmds = ha->enq->ucConcurrentCmdCount;
2684 } else {
2685 /* use the old method */
2686 switch (ha->conf->logical_drive[0].ucStripeSize) {
2687 case 4:
2688 ha->max_cmds = 32;
2689 break;
2690
2691 case 5:
2692 ha->max_cmds = 16;
2693 break;
2694
2695 case 6:
2696 ha->max_cmds = 8;
2697 break;
2698
2699 case 7:
2700 default:
2701 ha->max_cmds = 4;
2702 break;
2703 }
2704 }
2705
2706 /* Limit the Active Commands on a Lite Adapter */
2707 if ((ha->ad_type == IPS_ADTYPE_SERVERAID3L) ||
2708 (ha->ad_type == IPS_ADTYPE_SERVERAID4L) ||
2709 (ha->ad_type == IPS_ADTYPE_SERVERAID4LX)) {
2710 if ((ha->max_cmds > MaxLiteCmds) && (MaxLiteCmds))
2711 ha->max_cmds = MaxLiteCmds;
2712 }
2713
2714 /* set controller IDs */
2715 ha->ha_id[0] = IPS_ADAPTER_ID;
2716 for (i = 1; i < ha->nbus; i++) {
2717 ha->ha_id[i] = ha->conf->init_id[i - 1] & 0x1f;
2718 ha->dcdb_active[i - 1] = 0;
2719 }
2720
2721 return (1);
2722}
2723
2724/****************************************************************************/
2725/* */
2726/* Routine Name: ips_next */
2727/* */
2728/* Routine Description: */
2729/* */
2730/* Take the next command off the queue and send it to the controller */
2731/* */
2732/****************************************************************************/
2733static void
2734ips_next(ips_ha_t * ha, int intr)
2735{
2736 ips_scb_t *scb;
2737 Scsi_Cmnd *SC;
2738 Scsi_Cmnd *p;
2739 Scsi_Cmnd *q;
2740 ips_copp_wait_item_t *item;
2741 int ret;
2742 unsigned long cpu_flags = 0;
2743 struct Scsi_Host *host;
2744 METHOD_TRACE("ips_next", 1);
2745
2746 if (!ha)
2747 return;
2748 host = ips_sh[ha->host_num];
2749 /*
2750 * Block access to the queue function so
2751 * this command won't time out
2752 */
2753 if (intr == IPS_INTR_ON)
2754 IPS_LOCK_SAVE(host->host_lock, cpu_flags);
2755
2756 if ((ha->subsys->param[3] & 0x300000)
2757 && (ha->scb_activelist.count == 0)) {
2758 struct timeval tv;
2759
2760 do_gettimeofday(&tv);
2761
2762 if (tv.tv_sec - ha->last_ffdc > IPS_SECS_8HOURS) {
2763 ha->last_ffdc = tv.tv_sec;
2764 ips_ffdc_time(ha);
2765 }
2766 }
2767
2768 /*
2769 * Send passthru commands
2770 * These have priority over normal I/O
2771 * but shouldn't affect performance too much
2772 * since we limit the number that can be active
2773 * on the card at any one time
2774 */
2775 while ((ha->num_ioctl < IPS_MAX_IOCTL) &&
2776 (ha->copp_waitlist.head) && (scb = ips_getscb(ha))) {
2777
2778 item = ips_removeq_copp_head(&ha->copp_waitlist);
2779 ha->num_ioctl++;
2780 if (intr == IPS_INTR_ON)
2781 IPS_UNLOCK_RESTORE(host->host_lock, cpu_flags);
2782 scb->scsi_cmd = item->scsi_cmd;
2783 kfree(item);
2784
2785 ret = ips_make_passthru(ha, scb->scsi_cmd, scb, intr);
2786
2787 if (intr == IPS_INTR_ON)
2788 IPS_LOCK_SAVE(host->host_lock, cpu_flags);
2789 switch (ret) {
2790 case IPS_FAILURE:
2791 if (scb->scsi_cmd) {
2792 scb->scsi_cmd->result = DID_ERROR << 16;
2793 scb->scsi_cmd->scsi_done(scb->scsi_cmd);
2794 }
2795
2796 ips_freescb(ha, scb);
2797 break;
2798 case IPS_SUCCESS_IMM:
2799 if (scb->scsi_cmd) {
2800 scb->scsi_cmd->result = DID_OK << 16;
2801 scb->scsi_cmd->scsi_done(scb->scsi_cmd);
2802 }
2803
2804 ips_freescb(ha, scb);
2805 break;
2806 default:
2807 break;
2808 } /* end case */
2809
2810 if (ret != IPS_SUCCESS) {
2811 ha->num_ioctl--;
2812 continue;
2813 }
2814
2815 ret = ips_send_cmd(ha, scb);
2816
2817 if (ret == IPS_SUCCESS)
2818 ips_putq_scb_head(&ha->scb_activelist, scb);
2819 else
2820 ha->num_ioctl--;
2821
2822 switch (ret) {
2823 case IPS_FAILURE:
2824 if (scb->scsi_cmd) {
2825 scb->scsi_cmd->result = DID_ERROR << 16;
2826 }
2827
2828 ips_freescb(ha, scb);
2829 break;
2830 case IPS_SUCCESS_IMM:
2831 ips_freescb(ha, scb);
2832 break;
2833 default:
2834 break;
2835 } /* end case */
2836
2837 }
2838
2839 /*
2840 * Send "Normal" I/O commands
2841 */
2842
2843 p = ha->scb_waitlist.head;
2844 while ((p) && (scb = ips_getscb(ha))) {
Jeff Garzik422c0d62005-10-24 18:05:09 -04002845 if ((scmd_channel(p) > 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002846 && (ha->
Jeff Garzik422c0d62005-10-24 18:05:09 -04002847 dcdb_active[scmd_channel(p) -
2848 1] & (1 << scmd_id(p)))) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002849 ips_freescb(ha, scb);
2850 p = (Scsi_Cmnd *) p->host_scribble;
2851 continue;
2852 }
2853
2854 q = p;
2855 SC = ips_removeq_wait(&ha->scb_waitlist, q);
2856
2857 if (intr == IPS_INTR_ON)
2858 IPS_UNLOCK_RESTORE(host->host_lock, cpu_flags); /* Unlock HA after command is taken off queue */
2859
2860 SC->result = DID_OK;
2861 SC->host_scribble = NULL;
2862
2863 memset(SC->sense_buffer, 0, sizeof (SC->sense_buffer));
2864
2865 scb->target_id = SC->device->id;
2866 scb->lun = SC->device->lun;
2867 scb->bus = SC->device->channel;
2868 scb->scsi_cmd = SC;
2869 scb->breakup = 0;
2870 scb->data_len = 0;
2871 scb->callback = ipsintr_done;
2872 scb->timeout = ips_cmd_timeout;
2873 memset(&scb->cmd, 0, 16);
2874
2875 /* copy in the CDB */
2876 memcpy(scb->cdb, SC->cmnd, SC->cmd_len);
2877
2878 /* Now handle the data buffer */
2879 if (SC->use_sg) {
2880 struct scatterlist *sg;
2881 int i;
2882
2883 sg = SC->request_buffer;
2884 scb->sg_count = pci_map_sg(ha->pcidev, sg, SC->use_sg,
be7db052005-04-17 15:26:13 -05002885 SC->sc_data_direction);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002886 scb->flags |= IPS_SCB_MAP_SG;
2887 for (i = 0; i < scb->sg_count; i++) {
2888 if (ips_fill_scb_sg_single
2889 (ha, sg_dma_address(&sg[i]), scb, i,
2890 sg_dma_len(&sg[i])) < 0)
2891 break;
2892 }
2893 scb->dcdb.transfer_length = scb->data_len;
2894 } else {
2895 if (SC->request_bufflen) {
2896 scb->data_busaddr =
2897 pci_map_single(ha->pcidev,
2898 SC->request_buffer,
2899 SC->request_bufflen,
be7db052005-04-17 15:26:13 -05002900 SC->sc_data_direction);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002901 scb->flags |= IPS_SCB_MAP_SINGLE;
2902 ips_fill_scb_sg_single(ha, scb->data_busaddr,
2903 scb, 0,
2904 SC->request_bufflen);
2905 scb->dcdb.transfer_length = scb->data_len;
2906 } else {
2907 scb->data_busaddr = 0L;
2908 scb->sg_len = 0;
2909 scb->data_len = 0;
2910 scb->dcdb.transfer_length = 0;
2911 }
2912
2913 }
2914
2915 scb->dcdb.cmd_attribute =
2916 ips_command_direction[scb->scsi_cmd->cmnd[0]];
2917
2918 /* Allow a WRITE BUFFER Command to Have no Data */
2919 /* This is Used by Tape Flash Utilites */
2920 if ((scb->scsi_cmd->cmnd[0] == WRITE_BUFFER) && (scb->data_len == 0))
2921 scb->dcdb.cmd_attribute = 0;
2922
2923 if (!(scb->dcdb.cmd_attribute & 0x3))
2924 scb->dcdb.transfer_length = 0;
2925
2926 if (scb->data_len >= IPS_MAX_XFER) {
2927 scb->dcdb.cmd_attribute |= IPS_TRANSFER64K;
2928 scb->dcdb.transfer_length = 0;
2929 }
2930 if (intr == IPS_INTR_ON)
2931 IPS_LOCK_SAVE(host->host_lock, cpu_flags);
2932
2933 ret = ips_send_cmd(ha, scb);
2934
2935 switch (ret) {
2936 case IPS_SUCCESS:
2937 ips_putq_scb_head(&ha->scb_activelist, scb);
2938 break;
2939 case IPS_FAILURE:
2940 if (scb->scsi_cmd) {
2941 scb->scsi_cmd->result = DID_ERROR << 16;
2942 scb->scsi_cmd->scsi_done(scb->scsi_cmd);
2943 }
2944
2945 if (scb->bus)
2946 ha->dcdb_active[scb->bus - 1] &=
2947 ~(1 << scb->target_id);
2948
2949 ips_freescb(ha, scb);
2950 break;
2951 case IPS_SUCCESS_IMM:
2952 if (scb->scsi_cmd)
2953 scb->scsi_cmd->scsi_done(scb->scsi_cmd);
2954
2955 if (scb->bus)
2956 ha->dcdb_active[scb->bus - 1] &=
2957 ~(1 << scb->target_id);
2958
2959 ips_freescb(ha, scb);
2960 break;
2961 default:
2962 break;
2963 } /* end case */
2964
2965 p = (Scsi_Cmnd *) p->host_scribble;
2966
2967 } /* end while */
2968
2969 if (intr == IPS_INTR_ON)
2970 IPS_UNLOCK_RESTORE(host->host_lock, cpu_flags);
2971}
2972
2973/****************************************************************************/
2974/* */
2975/* Routine Name: ips_putq_scb_head */
2976/* */
2977/* Routine Description: */
2978/* */
2979/* Add an item to the head of the queue */
2980/* */
2981/* ASSUMED to be called from within the HA lock */
2982/* */
2983/****************************************************************************/
2984static void
2985ips_putq_scb_head(ips_scb_queue_t * queue, ips_scb_t * item)
2986{
2987 METHOD_TRACE("ips_putq_scb_head", 1);
2988
2989 if (!item)
2990 return;
2991
2992 item->q_next = queue->head;
2993 queue->head = item;
2994
2995 if (!queue->tail)
2996 queue->tail = item;
2997
2998 queue->count++;
2999}
3000
3001/****************************************************************************/
3002/* */
3003/* Routine Name: ips_removeq_scb_head */
3004/* */
3005/* Routine Description: */
3006/* */
3007/* Remove the head of the queue */
3008/* */
3009/* ASSUMED to be called from within the HA lock */
3010/* */
3011/****************************************************************************/
3012static ips_scb_t *
3013ips_removeq_scb_head(ips_scb_queue_t * queue)
3014{
3015 ips_scb_t *item;
3016
3017 METHOD_TRACE("ips_removeq_scb_head", 1);
3018
3019 item = queue->head;
3020
3021 if (!item) {
3022 return (NULL);
3023 }
3024
3025 queue->head = item->q_next;
3026 item->q_next = NULL;
3027
3028 if (queue->tail == item)
3029 queue->tail = NULL;
3030
3031 queue->count--;
3032
3033 return (item);
3034}
3035
3036/****************************************************************************/
3037/* */
3038/* Routine Name: ips_removeq_scb */
3039/* */
3040/* Routine Description: */
3041/* */
3042/* Remove an item from a queue */
3043/* */
3044/* ASSUMED to be called from within the HA lock */
3045/* */
3046/****************************************************************************/
3047static ips_scb_t *
3048ips_removeq_scb(ips_scb_queue_t * queue, ips_scb_t * item)
3049{
3050 ips_scb_t *p;
3051
3052 METHOD_TRACE("ips_removeq_scb", 1);
3053
3054 if (!item)
3055 return (NULL);
3056
3057 if (item == queue->head) {
3058 return (ips_removeq_scb_head(queue));
3059 }
3060
3061 p = queue->head;
3062
3063 while ((p) && (item != p->q_next))
3064 p = p->q_next;
3065
3066 if (p) {
3067 /* found a match */
3068 p->q_next = item->q_next;
3069
3070 if (!item->q_next)
3071 queue->tail = p;
3072
3073 item->q_next = NULL;
3074 queue->count--;
3075
3076 return (item);
3077 }
3078
3079 return (NULL);
3080}
3081
3082/****************************************************************************/
3083/* */
3084/* Routine Name: ips_putq_wait_tail */
3085/* */
3086/* Routine Description: */
3087/* */
3088/* Add an item to the tail of the queue */
3089/* */
3090/* ASSUMED to be called from within the HA lock */
3091/* */
3092/****************************************************************************/
3093static void
3094ips_putq_wait_tail(ips_wait_queue_t * queue, Scsi_Cmnd * item)
3095{
3096 METHOD_TRACE("ips_putq_wait_tail", 1);
3097
3098 if (!item)
3099 return;
3100
3101 item->host_scribble = NULL;
3102
3103 if (queue->tail)
3104 queue->tail->host_scribble = (char *) item;
3105
3106 queue->tail = item;
3107
3108 if (!queue->head)
3109 queue->head = item;
3110
3111 queue->count++;
3112}
3113
3114/****************************************************************************/
3115/* */
3116/* Routine Name: ips_removeq_wait_head */
3117/* */
3118/* Routine Description: */
3119/* */
3120/* Remove the head of the queue */
3121/* */
3122/* ASSUMED to be called from within the HA lock */
3123/* */
3124/****************************************************************************/
3125static Scsi_Cmnd *
3126ips_removeq_wait_head(ips_wait_queue_t * queue)
3127{
3128 Scsi_Cmnd *item;
3129
3130 METHOD_TRACE("ips_removeq_wait_head", 1);
3131
3132 item = queue->head;
3133
3134 if (!item) {
3135 return (NULL);
3136 }
3137
3138 queue->head = (Scsi_Cmnd *) item->host_scribble;
3139 item->host_scribble = NULL;
3140
3141 if (queue->tail == item)
3142 queue->tail = NULL;
3143
3144 queue->count--;
3145
3146 return (item);
3147}
3148
3149/****************************************************************************/
3150/* */
3151/* Routine Name: ips_removeq_wait */
3152/* */
3153/* Routine Description: */
3154/* */
3155/* Remove an item from a queue */
3156/* */
3157/* ASSUMED to be called from within the HA lock */
3158/* */
3159/****************************************************************************/
3160static Scsi_Cmnd *
3161ips_removeq_wait(ips_wait_queue_t * queue, Scsi_Cmnd * item)
3162{
3163 Scsi_Cmnd *p;
3164
3165 METHOD_TRACE("ips_removeq_wait", 1);
3166
3167 if (!item)
3168 return (NULL);
3169
3170 if (item == queue->head) {
3171 return (ips_removeq_wait_head(queue));
3172 }
3173
3174 p = queue->head;
3175
3176 while ((p) && (item != (Scsi_Cmnd *) p->host_scribble))
3177 p = (Scsi_Cmnd *) p->host_scribble;
3178
3179 if (p) {
3180 /* found a match */
3181 p->host_scribble = item->host_scribble;
3182
3183 if (!item->host_scribble)
3184 queue->tail = p;
3185
3186 item->host_scribble = NULL;
3187 queue->count--;
3188
3189 return (item);
3190 }
3191
3192 return (NULL);
3193}
3194
3195/****************************************************************************/
3196/* */
3197/* Routine Name: ips_putq_copp_tail */
3198/* */
3199/* Routine Description: */
3200/* */
3201/* Add an item to the tail of the queue */
3202/* */
3203/* ASSUMED to be called from within the HA lock */
3204/* */
3205/****************************************************************************/
3206static void
3207ips_putq_copp_tail(ips_copp_queue_t * queue, ips_copp_wait_item_t * item)
3208{
3209 METHOD_TRACE("ips_putq_copp_tail", 1);
3210
3211 if (!item)
3212 return;
3213
3214 item->next = NULL;
3215
3216 if (queue->tail)
3217 queue->tail->next = item;
3218
3219 queue->tail = item;
3220
3221 if (!queue->head)
3222 queue->head = item;
3223
3224 queue->count++;
3225}
3226
3227/****************************************************************************/
3228/* */
3229/* Routine Name: ips_removeq_copp_head */
3230/* */
3231/* Routine Description: */
3232/* */
3233/* Remove the head of the queue */
3234/* */
3235/* ASSUMED to be called from within the HA lock */
3236/* */
3237/****************************************************************************/
3238static ips_copp_wait_item_t *
3239ips_removeq_copp_head(ips_copp_queue_t * queue)
3240{
3241 ips_copp_wait_item_t *item;
3242
3243 METHOD_TRACE("ips_removeq_copp_head", 1);
3244
3245 item = queue->head;
3246
3247 if (!item) {
3248 return (NULL);
3249 }
3250
3251 queue->head = item->next;
3252 item->next = NULL;
3253
3254 if (queue->tail == item)
3255 queue->tail = NULL;
3256
3257 queue->count--;
3258
3259 return (item);
3260}
3261
3262/****************************************************************************/
3263/* */
3264/* Routine Name: ips_removeq_copp */
3265/* */
3266/* Routine Description: */
3267/* */
3268/* Remove an item from a queue */
3269/* */
3270/* ASSUMED to be called from within the HA lock */
3271/* */
3272/****************************************************************************/
3273static ips_copp_wait_item_t *
3274ips_removeq_copp(ips_copp_queue_t * queue, ips_copp_wait_item_t * item)
3275{
3276 ips_copp_wait_item_t *p;
3277
3278 METHOD_TRACE("ips_removeq_copp", 1);
3279
3280 if (!item)
3281 return (NULL);
3282
3283 if (item == queue->head) {
3284 return (ips_removeq_copp_head(queue));
3285 }
3286
3287 p = queue->head;
3288
3289 while ((p) && (item != p->next))
3290 p = p->next;
3291
3292 if (p) {
3293 /* found a match */
3294 p->next = item->next;
3295
3296 if (!item->next)
3297 queue->tail = p;
3298
3299 item->next = NULL;
3300 queue->count--;
3301
3302 return (item);
3303 }
3304
3305 return (NULL);
3306}
3307
3308/****************************************************************************/
3309/* */
3310/* Routine Name: ipsintr_blocking */
3311/* */
3312/* Routine Description: */
3313/* */
3314/* Finalize an interrupt for internal commands */
3315/* */
3316/****************************************************************************/
3317static void
3318ipsintr_blocking(ips_ha_t * ha, ips_scb_t * scb)
3319{
3320 METHOD_TRACE("ipsintr_blocking", 2);
3321
3322 ips_freescb(ha, scb);
3323 if ((ha->waitflag == TRUE) && (ha->cmd_in_progress == scb->cdb[0])) {
3324 ha->waitflag = FALSE;
3325
3326 return;
3327 }
3328}
3329
3330/****************************************************************************/
3331/* */
3332/* Routine Name: ipsintr_done */
3333/* */
3334/* Routine Description: */
3335/* */
3336/* Finalize an interrupt for non-internal commands */
3337/* */
3338/****************************************************************************/
3339static void
3340ipsintr_done(ips_ha_t * ha, ips_scb_t * scb)
3341{
3342 METHOD_TRACE("ipsintr_done", 2);
3343
3344 if (!scb) {
3345 IPS_PRINTK(KERN_WARNING, ha->pcidev,
3346 "Spurious interrupt; scb NULL.\n");
3347
3348 return;
3349 }
3350
3351 if (scb->scsi_cmd == NULL) {
3352 /* unexpected interrupt */
3353 IPS_PRINTK(KERN_WARNING, ha->pcidev,
3354 "Spurious interrupt; scsi_cmd not set.\n");
3355
3356 return;
3357 }
3358
3359 ips_done(ha, scb);
3360}
3361
3362/****************************************************************************/
3363/* */
3364/* Routine Name: ips_done */
3365/* */
3366/* Routine Description: */
3367/* */
3368/* Do housekeeping on completed commands */
3369/* ASSUMED to be called form within the request lock */
3370/****************************************************************************/
3371static void
3372ips_done(ips_ha_t * ha, ips_scb_t * scb)
3373{
3374 int ret;
3375
3376 METHOD_TRACE("ips_done", 1);
3377
3378 if (!scb)
3379 return;
3380
3381 if ((scb->scsi_cmd) && (ips_is_passthru(scb->scsi_cmd))) {
3382 ips_cleanup_passthru(ha, scb);
3383 ha->num_ioctl--;
3384 } else {
3385 /*
3386 * Check to see if this command had too much
3387 * data and had to be broke up. If so, queue
3388 * the rest of the data and continue.
3389 */
3390 if ((scb->breakup) || (scb->sg_break)) {
3391 /* we had a data breakup */
3392 scb->data_len = 0;
3393
3394 if (scb->sg_count) {
3395 /* S/G request */
3396 struct scatterlist *sg;
3397 int ips_sg_index = 0;
3398 int sg_dma_index;
3399
3400 sg = scb->scsi_cmd->request_buffer;
3401
3402 /* Spin forward to last dma chunk */
3403 sg_dma_index = scb->breakup;
3404
3405 /* Take care of possible partial on last chunk */
3406 ips_fill_scb_sg_single(ha,
3407 sg_dma_address(&sg
3408 [sg_dma_index]),
3409 scb, ips_sg_index++,
3410 sg_dma_len(&sg
3411 [sg_dma_index]));
3412
3413 for (; sg_dma_index < scb->sg_count;
3414 sg_dma_index++) {
3415 if (ips_fill_scb_sg_single
3416 (ha,
3417 sg_dma_address(&sg[sg_dma_index]),
3418 scb, ips_sg_index++,
3419 sg_dma_len(&sg[sg_dma_index])) < 0)
3420 break;
3421
3422 }
3423
3424 } else {
3425 /* Non S/G Request */
3426 (void) ips_fill_scb_sg_single(ha,
3427 scb->
3428 data_busaddr +
3429 (scb->sg_break *
3430 ha->max_xfer),
3431 scb, 0,
3432 scb->scsi_cmd->
3433 request_bufflen -
3434 (scb->sg_break *
3435 ha->max_xfer));
3436 }
3437
3438 scb->dcdb.transfer_length = scb->data_len;
3439 scb->dcdb.cmd_attribute |=
3440 ips_command_direction[scb->scsi_cmd->cmnd[0]];
3441
3442 if (!(scb->dcdb.cmd_attribute & 0x3))
3443 scb->dcdb.transfer_length = 0;
3444
3445 if (scb->data_len >= IPS_MAX_XFER) {
3446 scb->dcdb.cmd_attribute |= IPS_TRANSFER64K;
3447 scb->dcdb.transfer_length = 0;
3448 }
3449
3450 ret = ips_send_cmd(ha, scb);
3451
3452 switch (ret) {
3453 case IPS_FAILURE:
3454 if (scb->scsi_cmd) {
3455 scb->scsi_cmd->result = DID_ERROR << 16;
3456 scb->scsi_cmd->scsi_done(scb->scsi_cmd);
3457 }
3458
3459 ips_freescb(ha, scb);
3460 break;
3461 case IPS_SUCCESS_IMM:
3462 if (scb->scsi_cmd) {
3463 scb->scsi_cmd->result = DID_ERROR << 16;
3464 scb->scsi_cmd->scsi_done(scb->scsi_cmd);
3465 }
3466
3467 ips_freescb(ha, scb);
3468 break;
3469 default:
3470 break;
3471 } /* end case */
3472
3473 return;
3474 }
3475 } /* end if passthru */
3476
3477 if (scb->bus) {
3478 ha->dcdb_active[scb->bus - 1] &= ~(1 << scb->target_id);
3479 }
3480
3481 scb->scsi_cmd->scsi_done(scb->scsi_cmd);
3482
3483 ips_freescb(ha, scb);
3484}
3485
3486/****************************************************************************/
3487/* */
3488/* Routine Name: ips_map_status */
3489/* */
3490/* Routine Description: */
3491/* */
3492/* Map Controller Error codes to Linux Error Codes */
3493/* */
3494/****************************************************************************/
3495static int
3496ips_map_status(ips_ha_t * ha, ips_scb_t * scb, ips_stat_t * sp)
3497{
3498 int errcode;
3499 int device_error;
3500 uint32_t transfer_len;
3501 IPS_DCDB_TABLE_TAPE *tapeDCDB;
3502
3503 METHOD_TRACE("ips_map_status", 1);
3504
3505 if (scb->bus) {
3506 DEBUG_VAR(2,
3507 "(%s%d) Physical device error (%d %d %d): %x %x, Sense Key: %x, ASC: %x, ASCQ: %x",
3508 ips_name, ha->host_num,
3509 scb->scsi_cmd->device->channel,
3510 scb->scsi_cmd->device->id, scb->scsi_cmd->device->lun,
3511 scb->basic_status, scb->extended_status,
3512 scb->extended_status ==
3513 IPS_ERR_CKCOND ? scb->dcdb.sense_info[2] & 0xf : 0,
3514 scb->extended_status ==
3515 IPS_ERR_CKCOND ? scb->dcdb.sense_info[12] : 0,
3516 scb->extended_status ==
3517 IPS_ERR_CKCOND ? scb->dcdb.sense_info[13] : 0);
3518 }
3519
3520 /* default driver error */
3521 errcode = DID_ERROR;
3522 device_error = 0;
3523
3524 switch (scb->basic_status & IPS_GSC_STATUS_MASK) {
3525 case IPS_CMD_TIMEOUT:
3526 errcode = DID_TIME_OUT;
3527 break;
3528
3529 case IPS_INVAL_OPCO:
3530 case IPS_INVAL_CMD_BLK:
3531 case IPS_INVAL_PARM_BLK:
3532 case IPS_LD_ERROR:
3533 case IPS_CMD_CMPLT_WERROR:
3534 break;
3535
3536 case IPS_PHYS_DRV_ERROR:
3537 switch (scb->extended_status) {
3538 case IPS_ERR_SEL_TO:
3539 if (scb->bus)
3540 errcode = DID_NO_CONNECT;
3541
3542 break;
3543
3544 case IPS_ERR_OU_RUN:
3545 if ((scb->cmd.dcdb.op_code == IPS_CMD_EXTENDED_DCDB) ||
3546 (scb->cmd.dcdb.op_code ==
3547 IPS_CMD_EXTENDED_DCDB_SG)) {
3548 tapeDCDB = (IPS_DCDB_TABLE_TAPE *) & scb->dcdb;
3549 transfer_len = tapeDCDB->transfer_length;
3550 } else {
3551 transfer_len =
3552 (uint32_t) scb->dcdb.transfer_length;
3553 }
3554
3555 if ((scb->bus) && (transfer_len < scb->data_len)) {
3556 /* Underrun - set default to no error */
3557 errcode = DID_OK;
3558
3559 /* Restrict access to physical DASD */
3560 if ((scb->scsi_cmd->cmnd[0] == INQUIRY) &&
3561 ((((char *) scb->scsi_cmd->
3562 buffer)[0] & 0x1f) == TYPE_DISK)) {
3563 /* underflow -- no error */
3564 /* restrict access to physical DASD */
3565 errcode = DID_TIME_OUT;
3566 break;
3567 }
3568 } else
3569 errcode = DID_ERROR;
3570
3571 break;
3572
3573 case IPS_ERR_RECOVERY:
3574 /* don't fail recovered errors */
3575 if (scb->bus)
3576 errcode = DID_OK;
3577
3578 break;
3579
3580 case IPS_ERR_HOST_RESET:
3581 case IPS_ERR_DEV_RESET:
3582 errcode = DID_RESET;
3583 break;
3584
3585 case IPS_ERR_CKCOND:
3586 if (scb->bus) {
3587 if ((scb->cmd.dcdb.op_code ==
3588 IPS_CMD_EXTENDED_DCDB)
3589 || (scb->cmd.dcdb.op_code ==
3590 IPS_CMD_EXTENDED_DCDB_SG)) {
3591 tapeDCDB =
3592 (IPS_DCDB_TABLE_TAPE *) & scb->dcdb;
3593 memcpy(scb->scsi_cmd->sense_buffer,
3594 tapeDCDB->sense_info,
3595 sizeof (scb->scsi_cmd->
3596 sense_buffer));
3597 } else {
3598 memcpy(scb->scsi_cmd->sense_buffer,
3599 scb->dcdb.sense_info,
3600 sizeof (scb->scsi_cmd->
3601 sense_buffer));
3602 }
3603 device_error = 2; /* check condition */
3604 }
3605
3606 errcode = DID_OK;
3607
3608 break;
3609
3610 default:
3611 errcode = DID_ERROR;
3612 break;
3613
3614 } /* end switch */
3615 } /* end switch */
3616
3617 scb->scsi_cmd->result = device_error | (errcode << 16);
3618
3619 return (1);
3620}
3621
3622/****************************************************************************/
3623/* */
3624/* Routine Name: ips_send_wait */
3625/* */
3626/* Routine Description: */
3627/* */
3628/* Send a command to the controller and wait for it to return */
3629/* */
3630/* The FFDC Time Stamp use this function for the callback, but doesn't */
3631/* actually need to wait. */
3632/****************************************************************************/
3633static int
3634ips_send_wait(ips_ha_t * ha, ips_scb_t * scb, int timeout, int intr)
3635{
3636 int ret;
3637
3638 METHOD_TRACE("ips_send_wait", 1);
3639
3640 if (intr != IPS_FFDC) { /* Won't be Waiting if this is a Time Stamp */
3641 ha->waitflag = TRUE;
3642 ha->cmd_in_progress = scb->cdb[0];
3643 }
3644 scb->callback = ipsintr_blocking;
3645 ret = ips_send_cmd(ha, scb);
3646
3647 if ((ret == IPS_FAILURE) || (ret == IPS_SUCCESS_IMM))
3648 return (ret);
3649
3650 if (intr != IPS_FFDC) /* Don't Wait around if this is a Time Stamp */
3651 ret = ips_wait(ha, timeout, intr);
3652
3653 return (ret);
3654}
3655
3656/****************************************************************************/
3657/* */
3658/* Routine Name: ips_scmd_buf_write */
3659/* */
3660/* Routine Description: */
3661/* Write data to Scsi_Cmnd request_buffer at proper offsets */
3662/****************************************************************************/
3663static void
3664ips_scmd_buf_write(Scsi_Cmnd * scmd, void *data, unsigned
3665 int count)
3666{
3667 if (scmd->use_sg) {
3668 int i;
3669 unsigned int min_cnt, xfer_cnt;
3670 char *cdata = (char *) data;
Jack Hammera3632fa2005-10-25 14:13:03 -04003671 unsigned char *buffer;
3672 unsigned long flags;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003673 struct scatterlist *sg = scmd->request_buffer;
3674 for (i = 0, xfer_cnt = 0;
3675 (i < scmd->use_sg) && (xfer_cnt < count); i++) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07003676 min_cnt = min(count - xfer_cnt, sg[i].length);
Jack Hammera3632fa2005-10-25 14:13:03 -04003677
3678 /* kmap_atomic() ensures addressability of the data buffer.*/
3679 /* local_irq_save() protects the KM_IRQ0 address slot. */
3680 local_irq_save(flags);
3681 buffer = kmap_atomic(sg[i].page, KM_IRQ0) + sg[i].offset;
3682 memcpy(buffer, &cdata[xfer_cnt], min_cnt);
3683 kunmap_atomic(buffer - sg[i].offset, KM_IRQ0);
3684 local_irq_restore(flags);
3685
Linus Torvalds1da177e2005-04-16 15:20:36 -07003686 xfer_cnt += min_cnt;
3687 }
3688
3689 } else {
3690 unsigned int min_cnt = min(count, scmd->request_bufflen);
3691 memcpy(scmd->request_buffer, data, min_cnt);
3692 }
3693}
3694
3695/****************************************************************************/
3696/* */
3697/* Routine Name: ips_scmd_buf_read */
3698/* */
3699/* Routine Description: */
3700/* Copy data from a Scsi_Cmnd to a new, linear buffer */
3701/****************************************************************************/
3702static void
3703ips_scmd_buf_read(Scsi_Cmnd * scmd, void *data, unsigned
3704 int count)
3705{
3706 if (scmd->use_sg) {
3707 int i;
3708 unsigned int min_cnt, xfer_cnt;
3709 char *cdata = (char *) data;
Jack Hammera3632fa2005-10-25 14:13:03 -04003710 unsigned char *buffer;
3711 unsigned long flags;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003712 struct scatterlist *sg = scmd->request_buffer;
3713 for (i = 0, xfer_cnt = 0;
3714 (i < scmd->use_sg) && (xfer_cnt < count); i++) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07003715 min_cnt = min(count - xfer_cnt, sg[i].length);
Jack Hammera3632fa2005-10-25 14:13:03 -04003716
3717 /* kmap_atomic() ensures addressability of the data buffer.*/
3718 /* local_irq_save() protects the KM_IRQ0 address slot. */
3719 local_irq_save(flags);
3720 buffer = kmap_atomic(sg[i].page, KM_IRQ0) + sg[i].offset;
3721 memcpy(&cdata[xfer_cnt], buffer, min_cnt);
3722 kunmap_atomic(buffer - sg[i].offset, KM_IRQ0);
3723 local_irq_restore(flags);
3724
Linus Torvalds1da177e2005-04-16 15:20:36 -07003725 xfer_cnt += min_cnt;
3726 }
3727
3728 } else {
3729 unsigned int min_cnt = min(count, scmd->request_bufflen);
3730 memcpy(data, scmd->request_buffer, min_cnt);
3731 }
3732}
3733
3734/****************************************************************************/
3735/* */
3736/* Routine Name: ips_send_cmd */
3737/* */
3738/* Routine Description: */
3739/* */
3740/* Map SCSI commands to ServeRAID commands for logical drives */
3741/* */
3742/****************************************************************************/
3743static int
3744ips_send_cmd(ips_ha_t * ha, ips_scb_t * scb)
3745{
3746 int ret;
3747 char *sp;
3748 int device_error;
3749 IPS_DCDB_TABLE_TAPE *tapeDCDB;
3750 int TimeOut;
3751
3752 METHOD_TRACE("ips_send_cmd", 1);
3753
3754 ret = IPS_SUCCESS;
3755
3756 if (!scb->scsi_cmd) {
3757 /* internal command */
3758
3759 if (scb->bus > 0) {
3760 /* Controller commands can't be issued */
3761 /* to real devices -- fail them */
3762 if ((ha->waitflag == TRUE) &&
3763 (ha->cmd_in_progress == scb->cdb[0])) {
3764 ha->waitflag = FALSE;
3765 }
3766
3767 return (1);
3768 }
3769 } else if ((scb->bus == 0) && (!ips_is_passthru(scb->scsi_cmd))) {
3770 /* command to logical bus -- interpret */
3771 ret = IPS_SUCCESS_IMM;
3772
3773 switch (scb->scsi_cmd->cmnd[0]) {
3774 case ALLOW_MEDIUM_REMOVAL:
3775 case REZERO_UNIT:
3776 case ERASE:
3777 case WRITE_FILEMARKS:
3778 case SPACE:
3779 scb->scsi_cmd->result = DID_ERROR << 16;
3780 break;
3781
3782 case START_STOP:
3783 scb->scsi_cmd->result = DID_OK << 16;
3784
3785 case TEST_UNIT_READY:
3786 case INQUIRY:
3787 if (scb->target_id == IPS_ADAPTER_ID) {
3788 /*
3789 * Either we have a TUR
3790 * or we have a SCSI inquiry
3791 */
3792 if (scb->scsi_cmd->cmnd[0] == TEST_UNIT_READY)
3793 scb->scsi_cmd->result = DID_OK << 16;
3794
3795 if (scb->scsi_cmd->cmnd[0] == INQUIRY) {
3796 IPS_SCSI_INQ_DATA inquiry;
3797
3798 memset(&inquiry, 0,
3799 sizeof (IPS_SCSI_INQ_DATA));
3800
3801 inquiry.DeviceType =
3802 IPS_SCSI_INQ_TYPE_PROCESSOR;
3803 inquiry.DeviceTypeQualifier =
3804 IPS_SCSI_INQ_LU_CONNECTED;
3805 inquiry.Version = IPS_SCSI_INQ_REV2;
3806 inquiry.ResponseDataFormat =
3807 IPS_SCSI_INQ_RD_REV2;
3808 inquiry.AdditionalLength = 31;
3809 inquiry.Flags[0] =
3810 IPS_SCSI_INQ_Address16;
3811 inquiry.Flags[1] =
3812 IPS_SCSI_INQ_WBus16 |
3813 IPS_SCSI_INQ_Sync;
3814 strncpy(inquiry.VendorId, "IBM ",
3815 8);
3816 strncpy(inquiry.ProductId,
3817 "SERVERAID ", 16);
3818 strncpy(inquiry.ProductRevisionLevel,
3819 "1.00", 4);
3820
3821 ips_scmd_buf_write(scb->scsi_cmd,
3822 &inquiry,
3823 sizeof (inquiry));
3824
3825 scb->scsi_cmd->result = DID_OK << 16;
3826 }
3827 } else {
3828 scb->cmd.logical_info.op_code = IPS_CMD_GET_LD_INFO;
3829 scb->cmd.logical_info.command_id = IPS_COMMAND_ID(ha, scb);
3830 scb->cmd.logical_info.reserved = 0;
3831 scb->cmd.logical_info.reserved2 = 0;
3832 scb->data_len = sizeof (IPS_LD_INFO);
3833 scb->data_busaddr = ha->logical_drive_info_dma_addr;
3834 scb->flags = 0;
3835 scb->cmd.logical_info.buffer_addr = scb->data_busaddr;
3836 ret = IPS_SUCCESS;
3837 }
3838
3839 break;
3840
3841 case REQUEST_SENSE:
3842 ips_reqsen(ha, scb);
3843 scb->scsi_cmd->result = DID_OK << 16;
3844 break;
3845
3846 case READ_6:
3847 case WRITE_6:
3848 if (!scb->sg_len) {
3849 scb->cmd.basic_io.op_code =
3850 (scb->scsi_cmd->cmnd[0] ==
3851 READ_6) ? IPS_CMD_READ : IPS_CMD_WRITE;
3852 scb->cmd.basic_io.enhanced_sg = 0;
3853 scb->cmd.basic_io.sg_addr =
3854 cpu_to_le32(scb->data_busaddr);
3855 } else {
3856 scb->cmd.basic_io.op_code =
3857 (scb->scsi_cmd->cmnd[0] ==
3858 READ_6) ? IPS_CMD_READ_SG :
3859 IPS_CMD_WRITE_SG;
3860 scb->cmd.basic_io.enhanced_sg =
3861 IPS_USE_ENH_SGLIST(ha) ? 0xFF : 0;
3862 scb->cmd.basic_io.sg_addr =
3863 cpu_to_le32(scb->sg_busaddr);
3864 }
3865
3866 scb->cmd.basic_io.segment_4G = 0;
3867 scb->cmd.basic_io.command_id = IPS_COMMAND_ID(ha, scb);
3868 scb->cmd.basic_io.log_drv = scb->target_id;
3869 scb->cmd.basic_io.sg_count = scb->sg_len;
3870
3871 if (scb->cmd.basic_io.lba)
3872 scb->cmd.basic_io.lba =
3873 cpu_to_le32(le32_to_cpu
3874 (scb->cmd.basic_io.lba) +
3875 le16_to_cpu(scb->cmd.basic_io.
3876 sector_count));
3877 else
3878 scb->cmd.basic_io.lba =
3879 (((scb->scsi_cmd->
3880 cmnd[1] & 0x1f) << 16) | (scb->scsi_cmd->
3881 cmnd[2] << 8) |
3882 (scb->scsi_cmd->cmnd[3]));
3883
3884 scb->cmd.basic_io.sector_count =
3885 cpu_to_le16(scb->data_len / IPS_BLKSIZE);
3886
3887 if (le16_to_cpu(scb->cmd.basic_io.sector_count) == 0)
3888 scb->cmd.basic_io.sector_count =
3889 cpu_to_le16(256);
3890
3891 ret = IPS_SUCCESS;
3892 break;
3893
3894 case READ_10:
3895 case WRITE_10:
3896 if (!scb->sg_len) {
3897 scb->cmd.basic_io.op_code =
3898 (scb->scsi_cmd->cmnd[0] ==
3899 READ_10) ? IPS_CMD_READ : IPS_CMD_WRITE;
3900 scb->cmd.basic_io.enhanced_sg = 0;
3901 scb->cmd.basic_io.sg_addr =
3902 cpu_to_le32(scb->data_busaddr);
3903 } else {
3904 scb->cmd.basic_io.op_code =
3905 (scb->scsi_cmd->cmnd[0] ==
3906 READ_10) ? IPS_CMD_READ_SG :
3907 IPS_CMD_WRITE_SG;
3908 scb->cmd.basic_io.enhanced_sg =
3909 IPS_USE_ENH_SGLIST(ha) ? 0xFF : 0;
3910 scb->cmd.basic_io.sg_addr =
3911 cpu_to_le32(scb->sg_busaddr);
3912 }
3913
3914 scb->cmd.basic_io.segment_4G = 0;
3915 scb->cmd.basic_io.command_id = IPS_COMMAND_ID(ha, scb);
3916 scb->cmd.basic_io.log_drv = scb->target_id;
3917 scb->cmd.basic_io.sg_count = scb->sg_len;
3918
3919 if (scb->cmd.basic_io.lba)
3920 scb->cmd.basic_io.lba =
3921 cpu_to_le32(le32_to_cpu
3922 (scb->cmd.basic_io.lba) +
3923 le16_to_cpu(scb->cmd.basic_io.
3924 sector_count));
3925 else
3926 scb->cmd.basic_io.lba =
3927 ((scb->scsi_cmd->cmnd[2] << 24) | (scb->
3928 scsi_cmd->
3929 cmnd[3]
3930 << 16) |
3931 (scb->scsi_cmd->cmnd[4] << 8) | scb->
3932 scsi_cmd->cmnd[5]);
3933
3934 scb->cmd.basic_io.sector_count =
3935 cpu_to_le16(scb->data_len / IPS_BLKSIZE);
3936
3937 if (cpu_to_le16(scb->cmd.basic_io.sector_count) == 0) {
3938 /*
3939 * This is a null condition
3940 * we don't have to do anything
3941 * so just return
3942 */
3943 scb->scsi_cmd->result = DID_OK << 16;
3944 } else
3945 ret = IPS_SUCCESS;
3946
3947 break;
3948
3949 case RESERVE:
3950 case RELEASE:
3951 scb->scsi_cmd->result = DID_OK << 16;
3952 break;
3953
3954 case MODE_SENSE:
3955 scb->cmd.basic_io.op_code = IPS_CMD_ENQUIRY;
3956 scb->cmd.basic_io.command_id = IPS_COMMAND_ID(ha, scb);
3957 scb->cmd.basic_io.segment_4G = 0;
3958 scb->cmd.basic_io.enhanced_sg = 0;
3959 scb->data_len = sizeof (*ha->enq);
3960 scb->cmd.basic_io.sg_addr = ha->enq_busaddr;
3961 ret = IPS_SUCCESS;
3962 break;
3963
3964 case READ_CAPACITY:
3965 scb->cmd.logical_info.op_code = IPS_CMD_GET_LD_INFO;
3966 scb->cmd.logical_info.command_id = IPS_COMMAND_ID(ha, scb);
3967 scb->cmd.logical_info.reserved = 0;
3968 scb->cmd.logical_info.reserved2 = 0;
3969 scb->cmd.logical_info.reserved3 = 0;
3970 scb->data_len = sizeof (IPS_LD_INFO);
3971 scb->data_busaddr = ha->logical_drive_info_dma_addr;
3972 scb->flags = 0;
3973 scb->cmd.logical_info.buffer_addr = scb->data_busaddr;
3974 ret = IPS_SUCCESS;
3975 break;
3976
3977 case SEND_DIAGNOSTIC:
3978 case REASSIGN_BLOCKS:
3979 case FORMAT_UNIT:
3980 case SEEK_10:
3981 case VERIFY:
3982 case READ_DEFECT_DATA:
3983 case READ_BUFFER:
3984 case WRITE_BUFFER:
3985 scb->scsi_cmd->result = DID_OK << 16;
3986 break;
3987
3988 default:
3989 /* Set the Return Info to appear like the Command was */
3990 /* attempted, a Check Condition occurred, and Sense */
3991 /* Data indicating an Invalid CDB OpCode is returned. */
3992 sp = (char *) scb->scsi_cmd->sense_buffer;
3993 memset(sp, 0, sizeof (scb->scsi_cmd->sense_buffer));
3994
3995 sp[0] = 0x70; /* Error Code */
3996 sp[2] = ILLEGAL_REQUEST; /* Sense Key 5 Illegal Req. */
3997 sp[7] = 0x0A; /* Additional Sense Length */
3998 sp[12] = 0x20; /* ASC = Invalid OpCode */
3999 sp[13] = 0x00; /* ASCQ */
4000
4001 device_error = 2; /* Indicate Check Condition */
4002 scb->scsi_cmd->result = device_error | (DID_OK << 16);
4003 break;
4004 } /* end switch */
4005 }
4006 /* end if */
4007 if (ret == IPS_SUCCESS_IMM)
4008 return (ret);
4009
4010 /* setup DCDB */
4011 if (scb->bus > 0) {
4012
4013 /* If we already know the Device is Not there, no need to attempt a Command */
4014 /* This also protects an NT FailOver Controller from getting CDB's sent to it */
4015 if (ha->conf->dev[scb->bus - 1][scb->target_id].ucState == 0) {
4016 scb->scsi_cmd->result = DID_NO_CONNECT << 16;
4017 return (IPS_SUCCESS_IMM);
4018 }
4019
4020 ha->dcdb_active[scb->bus - 1] |= (1 << scb->target_id);
4021 scb->cmd.dcdb.command_id = IPS_COMMAND_ID(ha, scb);
4022 scb->cmd.dcdb.dcdb_address = cpu_to_le32(scb->scb_busaddr +
4023 (unsigned long) &scb->
4024 dcdb -
4025 (unsigned long) scb);
4026 scb->cmd.dcdb.reserved = 0;
4027 scb->cmd.dcdb.reserved2 = 0;
4028 scb->cmd.dcdb.reserved3 = 0;
4029 scb->cmd.dcdb.segment_4G = 0;
4030 scb->cmd.dcdb.enhanced_sg = 0;
4031
4032 TimeOut = scb->scsi_cmd->timeout_per_command;
4033
4034 if (ha->subsys->param[4] & 0x00100000) { /* If NEW Tape DCDB is Supported */
4035 if (!scb->sg_len) {
4036 scb->cmd.dcdb.op_code = IPS_CMD_EXTENDED_DCDB;
4037 } else {
4038 scb->cmd.dcdb.op_code =
4039 IPS_CMD_EXTENDED_DCDB_SG;
4040 scb->cmd.dcdb.enhanced_sg =
4041 IPS_USE_ENH_SGLIST(ha) ? 0xFF : 0;
4042 }
4043
4044 tapeDCDB = (IPS_DCDB_TABLE_TAPE *) & scb->dcdb; /* Use Same Data Area as Old DCDB Struct */
4045 tapeDCDB->device_address =
4046 ((scb->bus - 1) << 4) | scb->target_id;
4047 tapeDCDB->cmd_attribute |= IPS_DISCONNECT_ALLOWED;
4048 tapeDCDB->cmd_attribute &= ~IPS_TRANSFER64K; /* Always Turn OFF 64K Size Flag */
4049
4050 if (TimeOut) {
4051 if (TimeOut < (10 * HZ))
4052 tapeDCDB->cmd_attribute |= IPS_TIMEOUT10; /* TimeOut is 10 Seconds */
4053 else if (TimeOut < (60 * HZ))
4054 tapeDCDB->cmd_attribute |= IPS_TIMEOUT60; /* TimeOut is 60 Seconds */
4055 else if (TimeOut < (1200 * HZ))
4056 tapeDCDB->cmd_attribute |= IPS_TIMEOUT20M; /* TimeOut is 20 Minutes */
4057 }
4058
4059 tapeDCDB->cdb_length = scb->scsi_cmd->cmd_len;
4060 tapeDCDB->reserved_for_LUN = 0;
4061 tapeDCDB->transfer_length = scb->data_len;
4062 if (scb->cmd.dcdb.op_code == IPS_CMD_EXTENDED_DCDB_SG)
4063 tapeDCDB->buffer_pointer =
4064 cpu_to_le32(scb->sg_busaddr);
4065 else
4066 tapeDCDB->buffer_pointer =
4067 cpu_to_le32(scb->data_busaddr);
4068 tapeDCDB->sg_count = scb->sg_len;
4069 tapeDCDB->sense_length = sizeof (tapeDCDB->sense_info);
4070 tapeDCDB->scsi_status = 0;
4071 tapeDCDB->reserved = 0;
4072 memcpy(tapeDCDB->scsi_cdb, scb->scsi_cmd->cmnd,
4073 scb->scsi_cmd->cmd_len);
4074 } else {
4075 if (!scb->sg_len) {
4076 scb->cmd.dcdb.op_code = IPS_CMD_DCDB;
4077 } else {
4078 scb->cmd.dcdb.op_code = IPS_CMD_DCDB_SG;
4079 scb->cmd.dcdb.enhanced_sg =
4080 IPS_USE_ENH_SGLIST(ha) ? 0xFF : 0;
4081 }
4082
4083 scb->dcdb.device_address =
4084 ((scb->bus - 1) << 4) | scb->target_id;
4085 scb->dcdb.cmd_attribute |= IPS_DISCONNECT_ALLOWED;
4086
4087 if (TimeOut) {
4088 if (TimeOut < (10 * HZ))
4089 scb->dcdb.cmd_attribute |= IPS_TIMEOUT10; /* TimeOut is 10 Seconds */
4090 else if (TimeOut < (60 * HZ))
4091 scb->dcdb.cmd_attribute |= IPS_TIMEOUT60; /* TimeOut is 60 Seconds */
4092 else if (TimeOut < (1200 * HZ))
4093 scb->dcdb.cmd_attribute |= IPS_TIMEOUT20M; /* TimeOut is 20 Minutes */
4094 }
4095
4096 scb->dcdb.transfer_length = scb->data_len;
4097 if (scb->dcdb.cmd_attribute & IPS_TRANSFER64K)
4098 scb->dcdb.transfer_length = 0;
4099 if (scb->cmd.dcdb.op_code == IPS_CMD_DCDB_SG)
4100 scb->dcdb.buffer_pointer =
4101 cpu_to_le32(scb->sg_busaddr);
4102 else
4103 scb->dcdb.buffer_pointer =
4104 cpu_to_le32(scb->data_busaddr);
4105 scb->dcdb.cdb_length = scb->scsi_cmd->cmd_len;
4106 scb->dcdb.sense_length = sizeof (scb->dcdb.sense_info);
4107 scb->dcdb.sg_count = scb->sg_len;
4108 scb->dcdb.reserved = 0;
4109 memcpy(scb->dcdb.scsi_cdb, scb->scsi_cmd->cmnd,
4110 scb->scsi_cmd->cmd_len);
4111 scb->dcdb.scsi_status = 0;
4112 scb->dcdb.reserved2[0] = 0;
4113 scb->dcdb.reserved2[1] = 0;
4114 scb->dcdb.reserved2[2] = 0;
4115 }
4116 }
4117
4118 return ((*ha->func.issue) (ha, scb));
4119}
4120
4121/****************************************************************************/
4122/* */
4123/* Routine Name: ips_chk_status */
4124/* */
4125/* Routine Description: */
4126/* */
4127/* Check the status of commands to logical drives */
4128/* Assumed to be called with the HA lock */
4129/****************************************************************************/
4130static void
4131ips_chkstatus(ips_ha_t * ha, IPS_STATUS * pstatus)
4132{
4133 ips_scb_t *scb;
4134 ips_stat_t *sp;
4135 uint8_t basic_status;
4136 uint8_t ext_status;
4137 int errcode;
4138
4139 METHOD_TRACE("ips_chkstatus", 1);
4140
4141 scb = &ha->scbs[pstatus->fields.command_id];
4142 scb->basic_status = basic_status =
4143 pstatus->fields.basic_status & IPS_BASIC_STATUS_MASK;
4144 scb->extended_status = ext_status = pstatus->fields.extended_status;
4145
4146 sp = &ha->sp;
4147 sp->residue_len = 0;
4148 sp->scb_addr = (void *) scb;
4149
4150 /* Remove the item from the active queue */
4151 ips_removeq_scb(&ha->scb_activelist, scb);
4152
4153 if (!scb->scsi_cmd)
4154 /* internal commands are handled in do_ipsintr */
4155 return;
4156
4157 DEBUG_VAR(2, "(%s%d) ips_chkstatus: cmd 0x%X id %d (%d %d %d)",
4158 ips_name,
4159 ha->host_num,
4160 scb->cdb[0],
4161 scb->cmd.basic_io.command_id,
4162 scb->bus, scb->target_id, scb->lun);
4163
4164 if ((scb->scsi_cmd) && (ips_is_passthru(scb->scsi_cmd)))
4165 /* passthru - just returns the raw result */
4166 return;
4167
4168 errcode = DID_OK;
4169
4170 if (((basic_status & IPS_GSC_STATUS_MASK) == IPS_CMD_SUCCESS) ||
4171 ((basic_status & IPS_GSC_STATUS_MASK) == IPS_CMD_RECOVERED_ERROR)) {
4172
4173 if (scb->bus == 0) {
4174 if ((basic_status & IPS_GSC_STATUS_MASK) ==
4175 IPS_CMD_RECOVERED_ERROR) {
4176 DEBUG_VAR(1,
4177 "(%s%d) Recovered Logical Drive Error OpCode: %x, BSB: %x, ESB: %x",
4178 ips_name, ha->host_num,
4179 scb->cmd.basic_io.op_code,
4180 basic_status, ext_status);
4181 }
4182
4183 switch (scb->scsi_cmd->cmnd[0]) {
4184 case ALLOW_MEDIUM_REMOVAL:
4185 case REZERO_UNIT:
4186 case ERASE:
4187 case WRITE_FILEMARKS:
4188 case SPACE:
4189 errcode = DID_ERROR;
4190 break;
4191
4192 case START_STOP:
4193 break;
4194
4195 case TEST_UNIT_READY:
4196 if (!ips_online(ha, scb)) {
4197 errcode = DID_TIME_OUT;
4198 }
4199 break;
4200
4201 case INQUIRY:
4202 if (ips_online(ha, scb)) {
4203 ips_inquiry(ha, scb);
4204 } else {
4205 errcode = DID_TIME_OUT;
4206 }
4207 break;
4208
4209 case REQUEST_SENSE:
4210 ips_reqsen(ha, scb);
4211 break;
4212
4213 case READ_6:
4214 case WRITE_6:
4215 case READ_10:
4216 case WRITE_10:
4217 case RESERVE:
4218 case RELEASE:
4219 break;
4220
4221 case MODE_SENSE:
4222 if (!ips_online(ha, scb)
4223 || !ips_msense(ha, scb)) {
4224 errcode = DID_ERROR;
4225 }
4226 break;
4227
4228 case READ_CAPACITY:
4229 if (ips_online(ha, scb))
4230 ips_rdcap(ha, scb);
4231 else {
4232 errcode = DID_TIME_OUT;
4233 }
4234 break;
4235
4236 case SEND_DIAGNOSTIC:
4237 case REASSIGN_BLOCKS:
4238 break;
4239
4240 case FORMAT_UNIT:
4241 errcode = DID_ERROR;
4242 break;
4243
4244 case SEEK_10:
4245 case VERIFY:
4246 case READ_DEFECT_DATA:
4247 case READ_BUFFER:
4248 case WRITE_BUFFER:
4249 break;
4250
4251 default:
4252 errcode = DID_ERROR;
4253 } /* end switch */
4254
4255 scb->scsi_cmd->result = errcode << 16;
4256 } else { /* bus == 0 */
4257 /* restrict access to physical drives */
4258 if ((scb->scsi_cmd->cmnd[0] == INQUIRY) &&
4259 ((((char *) scb->scsi_cmd->buffer)[0] & 0x1f) ==
4260 TYPE_DISK)) {
4261
4262 scb->scsi_cmd->result = DID_TIME_OUT << 16;
4263 }
4264 } /* else */
4265 } else { /* recovered error / success */
4266 if (scb->bus == 0) {
4267 DEBUG_VAR(1,
4268 "(%s%d) Unrecovered Logical Drive Error OpCode: %x, BSB: %x, ESB: %x",
4269 ips_name, ha->host_num,
4270 scb->cmd.basic_io.op_code, basic_status,
4271 ext_status);
4272 }
4273
4274 ips_map_status(ha, scb, sp);
4275 } /* else */
4276}
4277
4278/****************************************************************************/
4279/* */
4280/* Routine Name: ips_online */
4281/* */
4282/* Routine Description: */
4283/* */
4284/* Determine if a logical drive is online */
4285/* */
4286/****************************************************************************/
4287static int
4288ips_online(ips_ha_t * ha, ips_scb_t * scb)
4289{
4290 METHOD_TRACE("ips_online", 1);
4291
4292 if (scb->target_id >= IPS_MAX_LD)
4293 return (0);
4294
4295 if ((scb->basic_status & IPS_GSC_STATUS_MASK) > 1) {
4296 memset(ha->logical_drive_info, 0, sizeof (IPS_LD_INFO));
4297 return (0);
4298 }
4299
4300 if (ha->logical_drive_info->drive_info[scb->target_id].state !=
4301 IPS_LD_OFFLINE
4302 && ha->logical_drive_info->drive_info[scb->target_id].state !=
4303 IPS_LD_FREE
4304 && ha->logical_drive_info->drive_info[scb->target_id].state !=
4305 IPS_LD_CRS
4306 && ha->logical_drive_info->drive_info[scb->target_id].state !=
4307 IPS_LD_SYS)
4308 return (1);
4309 else
4310 return (0);
4311}
4312
4313/****************************************************************************/
4314/* */
4315/* Routine Name: ips_inquiry */
4316/* */
4317/* Routine Description: */
4318/* */
4319/* Simulate an inquiry command to a logical drive */
4320/* */
4321/****************************************************************************/
4322static int
4323ips_inquiry(ips_ha_t * ha, ips_scb_t * scb)
4324{
4325 IPS_SCSI_INQ_DATA inquiry;
4326
4327 METHOD_TRACE("ips_inquiry", 1);
4328
4329 memset(&inquiry, 0, sizeof (IPS_SCSI_INQ_DATA));
4330
4331 inquiry.DeviceType = IPS_SCSI_INQ_TYPE_DASD;
4332 inquiry.DeviceTypeQualifier = IPS_SCSI_INQ_LU_CONNECTED;
4333 inquiry.Version = IPS_SCSI_INQ_REV2;
4334 inquiry.ResponseDataFormat = IPS_SCSI_INQ_RD_REV2;
4335 inquiry.AdditionalLength = 31;
4336 inquiry.Flags[0] = IPS_SCSI_INQ_Address16;
4337 inquiry.Flags[1] =
4338 IPS_SCSI_INQ_WBus16 | IPS_SCSI_INQ_Sync | IPS_SCSI_INQ_CmdQue;
4339 strncpy(inquiry.VendorId, "IBM ", 8);
4340 strncpy(inquiry.ProductId, "SERVERAID ", 16);
4341 strncpy(inquiry.ProductRevisionLevel, "1.00", 4);
4342
4343 ips_scmd_buf_write(scb->scsi_cmd, &inquiry, sizeof (inquiry));
4344
4345 return (1);
4346}
4347
4348/****************************************************************************/
4349/* */
4350/* Routine Name: ips_rdcap */
4351/* */
4352/* Routine Description: */
4353/* */
4354/* Simulate a read capacity command to a logical drive */
4355/* */
4356/****************************************************************************/
4357static int
4358ips_rdcap(ips_ha_t * ha, ips_scb_t * scb)
4359{
4360 IPS_SCSI_CAPACITY cap;
4361
4362 METHOD_TRACE("ips_rdcap", 1);
4363
4364 if (scb->scsi_cmd->bufflen < 8)
4365 return (0);
4366
4367 cap.lba =
4368 cpu_to_be32(le32_to_cpu
4369 (ha->logical_drive_info->
4370 drive_info[scb->target_id].sector_count) - 1);
4371 cap.len = cpu_to_be32((uint32_t) IPS_BLKSIZE);
4372
4373 ips_scmd_buf_write(scb->scsi_cmd, &cap, sizeof (cap));
4374
4375 return (1);
4376}
4377
4378/****************************************************************************/
4379/* */
4380/* Routine Name: ips_msense */
4381/* */
4382/* Routine Description: */
4383/* */
4384/* Simulate a mode sense command to a logical drive */
4385/* */
4386/****************************************************************************/
4387static int
4388ips_msense(ips_ha_t * ha, ips_scb_t * scb)
4389{
4390 uint16_t heads;
4391 uint16_t sectors;
4392 uint32_t cylinders;
4393 IPS_SCSI_MODE_PAGE_DATA mdata;
4394
4395 METHOD_TRACE("ips_msense", 1);
4396
4397 if (le32_to_cpu(ha->enq->ulDriveSize[scb->target_id]) > 0x400000 &&
4398 (ha->enq->ucMiscFlag & 0x8) == 0) {
4399 heads = IPS_NORM_HEADS;
4400 sectors = IPS_NORM_SECTORS;
4401 } else {
4402 heads = IPS_COMP_HEADS;
4403 sectors = IPS_COMP_SECTORS;
4404 }
4405
4406 cylinders =
4407 (le32_to_cpu(ha->enq->ulDriveSize[scb->target_id]) -
4408 1) / (heads * sectors);
4409
4410 memset(&mdata, 0, sizeof (IPS_SCSI_MODE_PAGE_DATA));
4411
4412 mdata.hdr.BlockDescLength = 8;
4413
4414 switch (scb->scsi_cmd->cmnd[2] & 0x3f) {
4415 case 0x03: /* page 3 */
4416 mdata.pdata.pg3.PageCode = 3;
4417 mdata.pdata.pg3.PageLength = sizeof (IPS_SCSI_MODE_PAGE3);
4418 mdata.hdr.DataLength =
4419 3 + mdata.hdr.BlockDescLength + mdata.pdata.pg3.PageLength;
4420 mdata.pdata.pg3.TracksPerZone = 0;
4421 mdata.pdata.pg3.AltSectorsPerZone = 0;
4422 mdata.pdata.pg3.AltTracksPerZone = 0;
4423 mdata.pdata.pg3.AltTracksPerVolume = 0;
4424 mdata.pdata.pg3.SectorsPerTrack = cpu_to_be16(sectors);
4425 mdata.pdata.pg3.BytesPerSector = cpu_to_be16(IPS_BLKSIZE);
4426 mdata.pdata.pg3.Interleave = cpu_to_be16(1);
4427 mdata.pdata.pg3.TrackSkew = 0;
4428 mdata.pdata.pg3.CylinderSkew = 0;
4429 mdata.pdata.pg3.flags = IPS_SCSI_MP3_SoftSector;
4430 break;
4431
4432 case 0x4:
4433 mdata.pdata.pg4.PageCode = 4;
4434 mdata.pdata.pg4.PageLength = sizeof (IPS_SCSI_MODE_PAGE4);
4435 mdata.hdr.DataLength =
4436 3 + mdata.hdr.BlockDescLength + mdata.pdata.pg4.PageLength;
4437 mdata.pdata.pg4.CylindersHigh =
4438 cpu_to_be16((cylinders >> 8) & 0xFFFF);
4439 mdata.pdata.pg4.CylindersLow = (cylinders & 0xFF);
4440 mdata.pdata.pg4.Heads = heads;
4441 mdata.pdata.pg4.WritePrecompHigh = 0;
4442 mdata.pdata.pg4.WritePrecompLow = 0;
4443 mdata.pdata.pg4.ReducedWriteCurrentHigh = 0;
4444 mdata.pdata.pg4.ReducedWriteCurrentLow = 0;
4445 mdata.pdata.pg4.StepRate = cpu_to_be16(1);
4446 mdata.pdata.pg4.LandingZoneHigh = 0;
4447 mdata.pdata.pg4.LandingZoneLow = 0;
4448 mdata.pdata.pg4.flags = 0;
4449 mdata.pdata.pg4.RotationalOffset = 0;
4450 mdata.pdata.pg4.MediumRotationRate = 0;
4451 break;
4452 case 0x8:
4453 mdata.pdata.pg8.PageCode = 8;
4454 mdata.pdata.pg8.PageLength = sizeof (IPS_SCSI_MODE_PAGE8);
4455 mdata.hdr.DataLength =
4456 3 + mdata.hdr.BlockDescLength + mdata.pdata.pg8.PageLength;
4457 /* everything else is left set to 0 */
4458 break;
4459
4460 default:
4461 return (0);
4462 } /* end switch */
4463
4464 ips_scmd_buf_write(scb->scsi_cmd, &mdata, sizeof (mdata));
4465
4466 return (1);
4467}
4468
4469/****************************************************************************/
4470/* */
4471/* Routine Name: ips_reqsen */
4472/* */
4473/* Routine Description: */
4474/* */
4475/* Simulate a request sense command to a logical drive */
4476/* */
4477/****************************************************************************/
4478static int
4479ips_reqsen(ips_ha_t * ha, ips_scb_t * scb)
4480{
4481 IPS_SCSI_REQSEN reqsen;
4482
4483 METHOD_TRACE("ips_reqsen", 1);
4484
4485 memset(&reqsen, 0, sizeof (IPS_SCSI_REQSEN));
4486
4487 reqsen.ResponseCode =
4488 IPS_SCSI_REQSEN_VALID | IPS_SCSI_REQSEN_CURRENT_ERR;
4489 reqsen.AdditionalLength = 10;
4490 reqsen.AdditionalSenseCode = IPS_SCSI_REQSEN_NO_SENSE;
4491 reqsen.AdditionalSenseCodeQual = IPS_SCSI_REQSEN_NO_SENSE;
4492
4493 ips_scmd_buf_write(scb->scsi_cmd, &reqsen, sizeof (reqsen));
4494
4495 return (1);
4496}
4497
4498/****************************************************************************/
4499/* */
4500/* Routine Name: ips_free */
4501/* */
4502/* Routine Description: */
4503/* */
4504/* Free any allocated space for this controller */
4505/* */
4506/****************************************************************************/
4507static void
4508ips_free(ips_ha_t * ha)
4509{
4510
4511 METHOD_TRACE("ips_free", 1);
4512
4513 if (ha) {
4514 if (ha->enq) {
4515 pci_free_consistent(ha->pcidev, sizeof(IPS_ENQ),
4516 ha->enq, ha->enq_busaddr);
4517 ha->enq = NULL;
4518 }
4519
4520 if (ha->conf) {
4521 kfree(ha->conf);
4522 ha->conf = NULL;
4523 }
4524
4525 if (ha->adapt) {
4526 pci_free_consistent(ha->pcidev,
4527 sizeof (IPS_ADAPTER) +
4528 sizeof (IPS_IO_CMD), ha->adapt,
4529 ha->adapt->hw_status_start);
4530 ha->adapt = NULL;
4531 }
4532
4533 if (ha->logical_drive_info) {
4534 pci_free_consistent(ha->pcidev,
4535 sizeof (IPS_LD_INFO),
4536 ha->logical_drive_info,
4537 ha->logical_drive_info_dma_addr);
4538 ha->logical_drive_info = NULL;
4539 }
4540
4541 if (ha->nvram) {
4542 kfree(ha->nvram);
4543 ha->nvram = NULL;
4544 }
4545
4546 if (ha->subsys) {
4547 kfree(ha->subsys);
4548 ha->subsys = NULL;
4549 }
4550
4551 if (ha->ioctl_data) {
4552 pci_free_consistent(ha->pcidev, ha->ioctl_len,
4553 ha->ioctl_data, ha->ioctl_busaddr);
4554 ha->ioctl_data = NULL;
4555 ha->ioctl_datasize = 0;
4556 ha->ioctl_len = 0;
4557 }
4558 ips_deallocatescbs(ha, ha->max_cmds);
4559
4560 /* free memory mapped (if applicable) */
4561 if (ha->mem_ptr) {
4562 iounmap(ha->ioremap_ptr);
4563 ha->ioremap_ptr = NULL;
4564 ha->mem_ptr = NULL;
4565 }
4566
4567 if (ha->mem_addr)
4568 release_mem_region(ha->mem_addr, ha->mem_len);
4569 ha->mem_addr = 0;
4570
4571 }
4572}
4573
4574/****************************************************************************/
4575/* */
4576/* Routine Name: ips_deallocatescbs */
4577/* */
4578/* Routine Description: */
4579/* */
4580/* Free the command blocks */
4581/* */
4582/****************************************************************************/
4583static int
4584ips_deallocatescbs(ips_ha_t * ha, int cmds)
4585{
4586 if (ha->scbs) {
4587 pci_free_consistent(ha->pcidev,
4588 IPS_SGLIST_SIZE(ha) * IPS_MAX_SG * cmds,
4589 ha->scbs->sg_list.list,
4590 ha->scbs->sg_busaddr);
4591 pci_free_consistent(ha->pcidev, sizeof (ips_scb_t) * cmds,
4592 ha->scbs, ha->scbs->scb_busaddr);
4593 ha->scbs = NULL;
4594 } /* end if */
4595 return 1;
4596}
4597
4598/****************************************************************************/
4599/* */
4600/* Routine Name: ips_allocatescbs */
4601/* */
4602/* Routine Description: */
4603/* */
4604/* Allocate the command blocks */
4605/* */
4606/****************************************************************************/
4607static int
4608ips_allocatescbs(ips_ha_t * ha)
4609{
4610 ips_scb_t *scb_p;
4611 IPS_SG_LIST ips_sg;
4612 int i;
4613 dma_addr_t command_dma, sg_dma;
4614
4615 METHOD_TRACE("ips_allocatescbs", 1);
4616
4617 /* Allocate memory for the SCBs */
4618 ha->scbs =
4619 pci_alloc_consistent(ha->pcidev, ha->max_cmds * sizeof (ips_scb_t),
4620 &command_dma);
4621 if (ha->scbs == NULL)
4622 return 0;
4623 ips_sg.list =
4624 pci_alloc_consistent(ha->pcidev,
4625 IPS_SGLIST_SIZE(ha) * IPS_MAX_SG *
4626 ha->max_cmds, &sg_dma);
4627 if (ips_sg.list == NULL) {
4628 pci_free_consistent(ha->pcidev,
4629 ha->max_cmds * sizeof (ips_scb_t), ha->scbs,
4630 command_dma);
4631 return 0;
4632 }
4633
4634 memset(ha->scbs, 0, ha->max_cmds * sizeof (ips_scb_t));
4635
4636 for (i = 0; i < ha->max_cmds; i++) {
4637 scb_p = &ha->scbs[i];
4638 scb_p->scb_busaddr = command_dma + sizeof (ips_scb_t) * i;
4639 /* set up S/G list */
4640 if (IPS_USE_ENH_SGLIST(ha)) {
4641 scb_p->sg_list.enh_list =
4642 ips_sg.enh_list + i * IPS_MAX_SG;
4643 scb_p->sg_busaddr =
4644 sg_dma + IPS_SGLIST_SIZE(ha) * IPS_MAX_SG * i;
4645 } else {
4646 scb_p->sg_list.std_list =
4647 ips_sg.std_list + i * IPS_MAX_SG;
4648 scb_p->sg_busaddr =
4649 sg_dma + IPS_SGLIST_SIZE(ha) * IPS_MAX_SG * i;
4650 }
4651
4652 /* add to the free list */
4653 if (i < ha->max_cmds - 1) {
4654 scb_p->q_next = ha->scb_freelist;
4655 ha->scb_freelist = scb_p;
4656 }
4657 }
4658
4659 /* success */
4660 return (1);
4661}
4662
4663/****************************************************************************/
4664/* */
4665/* Routine Name: ips_init_scb */
4666/* */
4667/* Routine Description: */
4668/* */
4669/* Initialize a CCB to default values */
4670/* */
4671/****************************************************************************/
4672static void
4673ips_init_scb(ips_ha_t * ha, ips_scb_t * scb)
4674{
4675 IPS_SG_LIST sg_list;
4676 uint32_t cmd_busaddr, sg_busaddr;
4677 METHOD_TRACE("ips_init_scb", 1);
4678
4679 if (scb == NULL)
4680 return;
4681
4682 sg_list.list = scb->sg_list.list;
4683 cmd_busaddr = scb->scb_busaddr;
4684 sg_busaddr = scb->sg_busaddr;
4685 /* zero fill */
4686 memset(scb, 0, sizeof (ips_scb_t));
4687 memset(ha->dummy, 0, sizeof (IPS_IO_CMD));
4688
4689 /* Initialize dummy command bucket */
4690 ha->dummy->op_code = 0xFF;
4691 ha->dummy->ccsar = cpu_to_le32(ha->adapt->hw_status_start
4692 + sizeof (IPS_ADAPTER));
4693 ha->dummy->command_id = IPS_MAX_CMDS;
4694
4695 /* set bus address of scb */
4696 scb->scb_busaddr = cmd_busaddr;
4697 scb->sg_busaddr = sg_busaddr;
4698 scb->sg_list.list = sg_list.list;
4699
4700 /* Neptune Fix */
4701 scb->cmd.basic_io.cccr = cpu_to_le32((uint32_t) IPS_BIT_ILE);
4702 scb->cmd.basic_io.ccsar = cpu_to_le32(ha->adapt->hw_status_start
4703 + sizeof (IPS_ADAPTER));
4704}
4705
4706/****************************************************************************/
4707/* */
4708/* Routine Name: ips_get_scb */
4709/* */
4710/* Routine Description: */
4711/* */
4712/* Initialize a CCB to default values */
4713/* */
4714/* ASSUMED to be callled from within a lock */
4715/* */
4716/****************************************************************************/
4717static ips_scb_t *
4718ips_getscb(ips_ha_t * ha)
4719{
4720 ips_scb_t *scb;
4721
4722 METHOD_TRACE("ips_getscb", 1);
4723
4724 if ((scb = ha->scb_freelist) == NULL) {
4725
4726 return (NULL);
4727 }
4728
4729 ha->scb_freelist = scb->q_next;
4730 scb->flags = 0;
4731 scb->q_next = NULL;
4732
4733 ips_init_scb(ha, scb);
4734
4735 return (scb);
4736}
4737
4738/****************************************************************************/
4739/* */
4740/* Routine Name: ips_free_scb */
4741/* */
4742/* Routine Description: */
4743/* */
4744/* Return an unused CCB back to the free list */
4745/* */
4746/* ASSUMED to be called from within a lock */
4747/* */
4748/****************************************************************************/
4749static void
4750ips_freescb(ips_ha_t * ha, ips_scb_t * scb)
4751{
4752
4753 METHOD_TRACE("ips_freescb", 1);
4754 if (scb->flags & IPS_SCB_MAP_SG)
4755 pci_unmap_sg(ha->pcidev, scb->scsi_cmd->request_buffer,
4756 scb->scsi_cmd->use_sg, IPS_DMA_DIR(scb));
4757 else if (scb->flags & IPS_SCB_MAP_SINGLE)
4758 pci_unmap_single(ha->pcidev, scb->data_busaddr, scb->data_len,
4759 IPS_DMA_DIR(scb));
4760
4761 /* check to make sure this is not our "special" scb */
4762 if (IPS_COMMAND_ID(ha, scb) < (ha->max_cmds - 1)) {
4763 scb->q_next = ha->scb_freelist;
4764 ha->scb_freelist = scb;
4765 }
4766}
4767
4768/****************************************************************************/
4769/* */
4770/* Routine Name: ips_isinit_copperhead */
4771/* */
4772/* Routine Description: */
4773/* */
4774/* Is controller initialized ? */
4775/* */
4776/****************************************************************************/
4777static int
4778ips_isinit_copperhead(ips_ha_t * ha)
4779{
4780 uint8_t scpr;
4781 uint8_t isr;
4782
4783 METHOD_TRACE("ips_isinit_copperhead", 1);
4784
4785 isr = inb(ha->io_addr + IPS_REG_HISR);
4786 scpr = inb(ha->io_addr + IPS_REG_SCPR);
4787
4788 if (((isr & IPS_BIT_EI) == 0) && ((scpr & IPS_BIT_EBM) == 0))
4789 return (0);
4790 else
4791 return (1);
4792}
4793
4794/****************************************************************************/
4795/* */
4796/* Routine Name: ips_isinit_copperhead_memio */
4797/* */
4798/* Routine Description: */
4799/* */
4800/* Is controller initialized ? */
4801/* */
4802/****************************************************************************/
4803static int
4804ips_isinit_copperhead_memio(ips_ha_t * ha)
4805{
4806 uint8_t isr = 0;
4807 uint8_t scpr;
4808
4809 METHOD_TRACE("ips_is_init_copperhead_memio", 1);
4810
4811 isr = readb(ha->mem_ptr + IPS_REG_HISR);
4812 scpr = readb(ha->mem_ptr + IPS_REG_SCPR);
4813
4814 if (((isr & IPS_BIT_EI) == 0) && ((scpr & IPS_BIT_EBM) == 0))
4815 return (0);
4816 else
4817 return (1);
4818}
4819
4820/****************************************************************************/
4821/* */
4822/* Routine Name: ips_isinit_morpheus */
4823/* */
4824/* Routine Description: */
4825/* */
4826/* Is controller initialized ? */
4827/* */
4828/****************************************************************************/
4829static int
4830ips_isinit_morpheus(ips_ha_t * ha)
4831{
4832 uint32_t post;
4833 uint32_t bits;
4834
4835 METHOD_TRACE("ips_is_init_morpheus", 1);
Jack Hammeree807c22005-08-29 10:44:34 -04004836
4837 if (ips_isintr_morpheus(ha))
4838 ips_flush_and_reset(ha);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004839
4840 post = readl(ha->mem_ptr + IPS_REG_I960_MSG0);
4841 bits = readl(ha->mem_ptr + IPS_REG_I2O_HIR);
4842
4843 if (post == 0)
4844 return (0);
4845 else if (bits & 0x3)
4846 return (0);
4847 else
4848 return (1);
4849}
4850
4851/****************************************************************************/
4852/* */
Jack Hammeree807c22005-08-29 10:44:34 -04004853/* Routine Name: ips_flush_and_reset */
4854/* */
4855/* Routine Description: */
4856/* */
4857/* Perform cleanup ( FLUSH and RESET ) when the adapter is in an unknown */
4858/* state ( was trying to INIT and an interrupt was already pending ) ... */
4859/* */
4860/****************************************************************************/
4861static void
4862ips_flush_and_reset(ips_ha_t *ha)
4863{
4864 ips_scb_t *scb;
4865 int ret;
4866 int time;
4867 int done;
4868 dma_addr_t command_dma;
4869
4870 /* Create a usuable SCB */
4871 scb = pci_alloc_consistent(ha->pcidev, sizeof(ips_scb_t), &command_dma);
4872 if (scb) {
4873 memset(scb, 0, sizeof(ips_scb_t));
4874 ips_init_scb(ha, scb);
4875 scb->scb_busaddr = command_dma;
4876
4877 scb->timeout = ips_cmd_timeout;
4878 scb->cdb[0] = IPS_CMD_FLUSH;
4879
4880 scb->cmd.flush_cache.op_code = IPS_CMD_FLUSH;
4881 scb->cmd.flush_cache.command_id = IPS_MAX_CMDS; /* Use an ID that would otherwise not exist */
4882 scb->cmd.flush_cache.state = IPS_NORM_STATE;
4883 scb->cmd.flush_cache.reserved = 0;
4884 scb->cmd.flush_cache.reserved2 = 0;
4885 scb->cmd.flush_cache.reserved3 = 0;
4886 scb->cmd.flush_cache.reserved4 = 0;
4887
4888 ret = ips_send_cmd(ha, scb); /* Send the Flush Command */
4889
4890 if (ret == IPS_SUCCESS) {
4891 time = 60 * IPS_ONE_SEC; /* Max Wait time is 60 seconds */
4892 done = 0;
4893
4894 while ((time > 0) && (!done)) {
4895 done = ips_poll_for_flush_complete(ha);
4896 /* This may look evil, but it's only done during extremely rare start-up conditions ! */
4897 udelay(1000);
4898 time--;
4899 }
4900 }
4901 }
4902
4903 /* Now RESET and INIT the adapter */
4904 (*ha->func.reset) (ha);
4905
4906 pci_free_consistent(ha->pcidev, sizeof(ips_scb_t), scb, command_dma);
4907 return;
4908}
4909
4910/****************************************************************************/
4911/* */
4912/* Routine Name: ips_poll_for_flush_complete */
4913/* */
4914/* Routine Description: */
4915/* */
4916/* Poll for the Flush Command issued by ips_flush_and_reset() to complete */
4917/* All other responses are just taken off the queue and ignored */
4918/* */
4919/****************************************************************************/
4920static int
4921ips_poll_for_flush_complete(ips_ha_t * ha)
4922{
4923 IPS_STATUS cstatus;
4924
4925 while (TRUE) {
4926 cstatus.value = (*ha->func.statupd) (ha);
4927
4928 if (cstatus.value == 0xffffffff) /* If No Interrupt to process */
4929 break;
4930
4931 /* Success is when we see the Flush Command ID */
4932 if (cstatus.fields.command_id == IPS_MAX_CMDS )
4933 return 1;
4934 }
4935
4936 return 0;
4937
4938/****************************************************************************/
4939/* */
Linus Torvalds1da177e2005-04-16 15:20:36 -07004940/* Routine Name: ips_enable_int_copperhead */
4941/* */
4942/* Routine Description: */
4943/* Turn on interrupts */
4944/* */
4945/****************************************************************************/
4946static void
4947ips_enable_int_copperhead(ips_ha_t * ha)
4948{
4949 METHOD_TRACE("ips_enable_int_copperhead", 1);
4950
4951 outb(ha->io_addr + IPS_REG_HISR, IPS_BIT_EI);
4952 inb(ha->io_addr + IPS_REG_HISR); /*Ensure PCI Posting Completes*/
4953}
4954
4955/****************************************************************************/
4956/* */
4957/* Routine Name: ips_enable_int_copperhead_memio */
4958/* */
4959/* Routine Description: */
4960/* Turn on interrupts */
4961/* */
4962/****************************************************************************/
4963static void
4964ips_enable_int_copperhead_memio(ips_ha_t * ha)
4965{
4966 METHOD_TRACE("ips_enable_int_copperhead_memio", 1);
4967
4968 writeb(IPS_BIT_EI, ha->mem_ptr + IPS_REG_HISR);
4969 readb(ha->mem_ptr + IPS_REG_HISR); /*Ensure PCI Posting Completes*/
4970}
4971
4972/****************************************************************************/
4973/* */
4974/* Routine Name: ips_enable_int_morpheus */
4975/* */
4976/* Routine Description: */
4977/* Turn on interrupts */
4978/* */
4979/****************************************************************************/
4980static void
4981ips_enable_int_morpheus(ips_ha_t * ha)
4982{
4983 uint32_t Oimr;
4984
4985 METHOD_TRACE("ips_enable_int_morpheus", 1);
4986
4987 Oimr = readl(ha->mem_ptr + IPS_REG_I960_OIMR);
4988 Oimr &= ~0x08;
4989 writel(Oimr, ha->mem_ptr + IPS_REG_I960_OIMR);
4990 readl(ha->mem_ptr + IPS_REG_I960_OIMR); /*Ensure PCI Posting Completes*/
4991}
4992
4993/****************************************************************************/
4994/* */
4995/* Routine Name: ips_init_copperhead */
4996/* */
4997/* Routine Description: */
4998/* */
4999/* Initialize a copperhead controller */
5000/* */
5001/****************************************************************************/
5002static int
5003ips_init_copperhead(ips_ha_t * ha)
5004{
5005 uint8_t Isr;
5006 uint8_t Cbsp;
5007 uint8_t PostByte[IPS_MAX_POST_BYTES];
5008 uint8_t ConfigByte[IPS_MAX_CONFIG_BYTES];
5009 int i, j;
5010
5011 METHOD_TRACE("ips_init_copperhead", 1);
5012
5013 for (i = 0; i < IPS_MAX_POST_BYTES; i++) {
5014 for (j = 0; j < 45; j++) {
5015 Isr = inb(ha->io_addr + IPS_REG_HISR);
5016 if (Isr & IPS_BIT_GHI)
5017 break;
5018
5019 /* Delay for 1 Second */
5020 MDELAY(IPS_ONE_SEC);
5021 }
5022
5023 if (j >= 45)
5024 /* error occurred */
5025 return (0);
5026
5027 PostByte[i] = inb(ha->io_addr + IPS_REG_ISPR);
5028 outb(Isr, ha->io_addr + IPS_REG_HISR);
5029 }
5030
5031 if (PostByte[0] < IPS_GOOD_POST_STATUS) {
5032 IPS_PRINTK(KERN_WARNING, ha->pcidev,
5033 "reset controller fails (post status %x %x).\n",
5034 PostByte[0], PostByte[1]);
5035
5036 return (0);
5037 }
5038
5039 for (i = 0; i < IPS_MAX_CONFIG_BYTES; i++) {
5040 for (j = 0; j < 240; j++) {
5041 Isr = inb(ha->io_addr + IPS_REG_HISR);
5042 if (Isr & IPS_BIT_GHI)
5043 break;
5044
5045 /* Delay for 1 Second */
5046 MDELAY(IPS_ONE_SEC);
5047 }
5048
5049 if (j >= 240)
5050 /* error occurred */
5051 return (0);
5052
5053 ConfigByte[i] = inb(ha->io_addr + IPS_REG_ISPR);
5054 outb(Isr, ha->io_addr + IPS_REG_HISR);
5055 }
5056
5057 for (i = 0; i < 240; i++) {
5058 Cbsp = inb(ha->io_addr + IPS_REG_CBSP);
5059
5060 if ((Cbsp & IPS_BIT_OP) == 0)
5061 break;
5062
5063 /* Delay for 1 Second */
5064 MDELAY(IPS_ONE_SEC);
5065 }
5066
5067 if (i >= 240)
5068 /* reset failed */
5069 return (0);
5070
5071 /* setup CCCR */
5072 outl(cpu_to_le32(0x1010), ha->io_addr + IPS_REG_CCCR);
5073
5074 /* Enable busmastering */
5075 outb(IPS_BIT_EBM, ha->io_addr + IPS_REG_SCPR);
5076
5077 if (ha->revision_id == IPS_REVID_TROMBONE64)
5078 /* fix for anaconda64 */
5079 outl(0, ha->io_addr + IPS_REG_NDAE);
5080
5081 /* Enable interrupts */
5082 outb(IPS_BIT_EI, ha->io_addr + IPS_REG_HISR);
5083
5084 return (1);
5085}
5086
5087/****************************************************************************/
5088/* */
5089/* Routine Name: ips_init_copperhead_memio */
5090/* */
5091/* Routine Description: */
5092/* */
5093/* Initialize a copperhead controller with memory mapped I/O */
5094/* */
5095/****************************************************************************/
5096static int
5097ips_init_copperhead_memio(ips_ha_t * ha)
5098{
5099 uint8_t Isr = 0;
5100 uint8_t Cbsp;
5101 uint8_t PostByte[IPS_MAX_POST_BYTES];
5102 uint8_t ConfigByte[IPS_MAX_CONFIG_BYTES];
5103 int i, j;
5104
5105 METHOD_TRACE("ips_init_copperhead_memio", 1);
5106
5107 for (i = 0; i < IPS_MAX_POST_BYTES; i++) {
5108 for (j = 0; j < 45; j++) {
5109 Isr = readb(ha->mem_ptr + IPS_REG_HISR);
5110 if (Isr & IPS_BIT_GHI)
5111 break;
5112
5113 /* Delay for 1 Second */
5114 MDELAY(IPS_ONE_SEC);
5115 }
5116
5117 if (j >= 45)
5118 /* error occurred */
5119 return (0);
5120
5121 PostByte[i] = readb(ha->mem_ptr + IPS_REG_ISPR);
5122 writeb(Isr, ha->mem_ptr + IPS_REG_HISR);
5123 }
5124
5125 if (PostByte[0] < IPS_GOOD_POST_STATUS) {
5126 IPS_PRINTK(KERN_WARNING, ha->pcidev,
5127 "reset controller fails (post status %x %x).\n",
5128 PostByte[0], PostByte[1]);
5129
5130 return (0);
5131 }
5132
5133 for (i = 0; i < IPS_MAX_CONFIG_BYTES; i++) {
5134 for (j = 0; j < 240; j++) {
5135 Isr = readb(ha->mem_ptr + IPS_REG_HISR);
5136 if (Isr & IPS_BIT_GHI)
5137 break;
5138
5139 /* Delay for 1 Second */
5140 MDELAY(IPS_ONE_SEC);
5141 }
5142
5143 if (j >= 240)
5144 /* error occurred */
5145 return (0);
5146
5147 ConfigByte[i] = readb(ha->mem_ptr + IPS_REG_ISPR);
5148 writeb(Isr, ha->mem_ptr + IPS_REG_HISR);
5149 }
5150
5151 for (i = 0; i < 240; i++) {
5152 Cbsp = readb(ha->mem_ptr + IPS_REG_CBSP);
5153
5154 if ((Cbsp & IPS_BIT_OP) == 0)
5155 break;
5156
5157 /* Delay for 1 Second */
5158 MDELAY(IPS_ONE_SEC);
5159 }
5160
5161 if (i >= 240)
5162 /* error occurred */
5163 return (0);
5164
5165 /* setup CCCR */
5166 writel(0x1010, ha->mem_ptr + IPS_REG_CCCR);
5167
5168 /* Enable busmastering */
5169 writeb(IPS_BIT_EBM, ha->mem_ptr + IPS_REG_SCPR);
5170
5171 if (ha->revision_id == IPS_REVID_TROMBONE64)
5172 /* fix for anaconda64 */
5173 writel(0, ha->mem_ptr + IPS_REG_NDAE);
5174
5175 /* Enable interrupts */
5176 writeb(IPS_BIT_EI, ha->mem_ptr + IPS_REG_HISR);
5177
5178 /* if we get here then everything went OK */
5179 return (1);
5180}
5181
5182/****************************************************************************/
5183/* */
5184/* Routine Name: ips_init_morpheus */
5185/* */
5186/* Routine Description: */
5187/* */
5188/* Initialize a morpheus controller */
5189/* */
5190/****************************************************************************/
5191static int
5192ips_init_morpheus(ips_ha_t * ha)
5193{
5194 uint32_t Post;
5195 uint32_t Config;
5196 uint32_t Isr;
5197 uint32_t Oimr;
5198 int i;
5199
5200 METHOD_TRACE("ips_init_morpheus", 1);
5201
5202 /* Wait up to 45 secs for Post */
5203 for (i = 0; i < 45; i++) {
5204 Isr = readl(ha->mem_ptr + IPS_REG_I2O_HIR);
5205
5206 if (Isr & IPS_BIT_I960_MSG0I)
5207 break;
5208
5209 /* Delay for 1 Second */
5210 MDELAY(IPS_ONE_SEC);
5211 }
5212
5213 if (i >= 45) {
5214 /* error occurred */
5215 IPS_PRINTK(KERN_WARNING, ha->pcidev,
5216 "timeout waiting for post.\n");
5217
5218 return (0);
5219 }
5220
5221 Post = readl(ha->mem_ptr + IPS_REG_I960_MSG0);
5222
5223 if (Post == 0x4F00) { /* If Flashing the Battery PIC */
5224 IPS_PRINTK(KERN_WARNING, ha->pcidev,
5225 "Flashing Battery PIC, Please wait ...\n");
5226
5227 /* Clear the interrupt bit */
5228 Isr = (uint32_t) IPS_BIT_I960_MSG0I;
5229 writel(Isr, ha->mem_ptr + IPS_REG_I2O_HIR);
5230
5231 for (i = 0; i < 120; i++) { /* Wait Up to 2 Min. for Completion */
5232 Post = readl(ha->mem_ptr + IPS_REG_I960_MSG0);
5233 if (Post != 0x4F00)
5234 break;
5235 /* Delay for 1 Second */
5236 MDELAY(IPS_ONE_SEC);
5237 }
5238
5239 if (i >= 120) {
5240 IPS_PRINTK(KERN_WARNING, ha->pcidev,
5241 "timeout waiting for Battery PIC Flash\n");
5242 return (0);
5243 }
5244
5245 }
5246
5247 /* Clear the interrupt bit */
5248 Isr = (uint32_t) IPS_BIT_I960_MSG0I;
5249 writel(Isr, ha->mem_ptr + IPS_REG_I2O_HIR);
5250
5251 if (Post < (IPS_GOOD_POST_STATUS << 8)) {
5252 IPS_PRINTK(KERN_WARNING, ha->pcidev,
5253 "reset controller fails (post status %x).\n", Post);
5254
5255 return (0);
5256 }
5257
5258 /* Wait up to 240 secs for config bytes */
5259 for (i = 0; i < 240; i++) {
5260 Isr = readl(ha->mem_ptr + IPS_REG_I2O_HIR);
5261
5262 if (Isr & IPS_BIT_I960_MSG1I)
5263 break;
5264
5265 /* Delay for 1 Second */
5266 MDELAY(IPS_ONE_SEC);
5267 }
5268
5269 if (i >= 240) {
5270 /* error occurred */
5271 IPS_PRINTK(KERN_WARNING, ha->pcidev,
5272 "timeout waiting for config.\n");
5273
5274 return (0);
5275 }
5276
5277 Config = readl(ha->mem_ptr + IPS_REG_I960_MSG1);
5278
5279 /* Clear interrupt bit */
5280 Isr = (uint32_t) IPS_BIT_I960_MSG1I;
5281 writel(Isr, ha->mem_ptr + IPS_REG_I2O_HIR);
5282
5283 /* Turn on the interrupts */
5284 Oimr = readl(ha->mem_ptr + IPS_REG_I960_OIMR);
5285 Oimr &= ~0x8;
5286 writel(Oimr, ha->mem_ptr + IPS_REG_I960_OIMR);
5287
5288 /* if we get here then everything went OK */
5289
5290 /* Since we did a RESET, an EraseStripeLock may be needed */
5291 if (Post == 0xEF10) {
5292 if ((Config == 0x000F) || (Config == 0x0009))
5293 ha->requires_esl = 1;
5294 }
5295
5296 return (1);
5297}
5298
5299/****************************************************************************/
5300/* */
5301/* Routine Name: ips_reset_copperhead */
5302/* */
5303/* Routine Description: */
5304/* */
5305/* Reset the controller */
5306/* */
5307/****************************************************************************/
5308static int
5309ips_reset_copperhead(ips_ha_t * ha)
5310{
5311 int reset_counter;
5312
5313 METHOD_TRACE("ips_reset_copperhead", 1);
5314
5315 DEBUG_VAR(1, "(%s%d) ips_reset_copperhead: io addr: %x, irq: %d",
5316 ips_name, ha->host_num, ha->io_addr, ha->irq);
5317
5318 reset_counter = 0;
5319
5320 while (reset_counter < 2) {
5321 reset_counter++;
5322
5323 outb(IPS_BIT_RST, ha->io_addr + IPS_REG_SCPR);
5324
5325 /* Delay for 1 Second */
5326 MDELAY(IPS_ONE_SEC);
5327
5328 outb(0, ha->io_addr + IPS_REG_SCPR);
5329
5330 /* Delay for 1 Second */
5331 MDELAY(IPS_ONE_SEC);
5332
5333 if ((*ha->func.init) (ha))
5334 break;
5335 else if (reset_counter >= 2) {
5336
5337 return (0);
5338 }
5339 }
5340
5341 return (1);
5342}
5343
5344/****************************************************************************/
5345/* */
5346/* Routine Name: ips_reset_copperhead_memio */
5347/* */
5348/* Routine Description: */
5349/* */
5350/* Reset the controller */
5351/* */
5352/****************************************************************************/
5353static int
5354ips_reset_copperhead_memio(ips_ha_t * ha)
5355{
5356 int reset_counter;
5357
5358 METHOD_TRACE("ips_reset_copperhead_memio", 1);
5359
5360 DEBUG_VAR(1, "(%s%d) ips_reset_copperhead_memio: mem addr: %x, irq: %d",
5361 ips_name, ha->host_num, ha->mem_addr, ha->irq);
5362
5363 reset_counter = 0;
5364
5365 while (reset_counter < 2) {
5366 reset_counter++;
5367
5368 writeb(IPS_BIT_RST, ha->mem_ptr + IPS_REG_SCPR);
5369
5370 /* Delay for 1 Second */
5371 MDELAY(IPS_ONE_SEC);
5372
5373 writeb(0, ha->mem_ptr + IPS_REG_SCPR);
5374
5375 /* Delay for 1 Second */
5376 MDELAY(IPS_ONE_SEC);
5377
5378 if ((*ha->func.init) (ha))
5379 break;
5380 else if (reset_counter >= 2) {
5381
5382 return (0);
5383 }
5384 }
5385
5386 return (1);
5387}
5388
5389/****************************************************************************/
5390/* */
5391/* Routine Name: ips_reset_morpheus */
5392/* */
5393/* Routine Description: */
5394/* */
5395/* Reset the controller */
5396/* */
5397/****************************************************************************/
5398static int
5399ips_reset_morpheus(ips_ha_t * ha)
5400{
5401 int reset_counter;
5402 uint8_t junk;
5403
5404 METHOD_TRACE("ips_reset_morpheus", 1);
5405
5406 DEBUG_VAR(1, "(%s%d) ips_reset_morpheus: mem addr: %x, irq: %d",
5407 ips_name, ha->host_num, ha->mem_addr, ha->irq);
5408
5409 reset_counter = 0;
5410
5411 while (reset_counter < 2) {
5412 reset_counter++;
5413
5414 writel(0x80000000, ha->mem_ptr + IPS_REG_I960_IDR);
5415
5416 /* Delay for 5 Seconds */
5417 MDELAY(5 * IPS_ONE_SEC);
5418
5419 /* Do a PCI config read to wait for adapter */
5420 pci_read_config_byte(ha->pcidev, 4, &junk);
5421
5422 if ((*ha->func.init) (ha))
5423 break;
5424 else if (reset_counter >= 2) {
5425
5426 return (0);
5427 }
5428 }
5429
5430 return (1);
5431}
5432
5433/****************************************************************************/
5434/* */
5435/* Routine Name: ips_statinit */
5436/* */
5437/* Routine Description: */
5438/* */
5439/* Initialize the status queues on the controller */
5440/* */
5441/****************************************************************************/
5442static void
5443ips_statinit(ips_ha_t * ha)
5444{
5445 uint32_t phys_status_start;
5446
5447 METHOD_TRACE("ips_statinit", 1);
5448
5449 ha->adapt->p_status_start = ha->adapt->status;
5450 ha->adapt->p_status_end = ha->adapt->status + IPS_MAX_CMDS;
5451 ha->adapt->p_status_tail = ha->adapt->status;
5452
5453 phys_status_start = ha->adapt->hw_status_start;
5454 outl(cpu_to_le32(phys_status_start), ha->io_addr + IPS_REG_SQSR);
5455 outl(cpu_to_le32(phys_status_start + IPS_STATUS_Q_SIZE),
5456 ha->io_addr + IPS_REG_SQER);
5457 outl(cpu_to_le32(phys_status_start + IPS_STATUS_SIZE),
5458 ha->io_addr + IPS_REG_SQHR);
5459 outl(cpu_to_le32(phys_status_start), ha->io_addr + IPS_REG_SQTR);
5460
5461 ha->adapt->hw_status_tail = phys_status_start;
5462}
5463
5464/****************************************************************************/
5465/* */
5466/* Routine Name: ips_statinit_memio */
5467/* */
5468/* Routine Description: */
5469/* */
5470/* Initialize the status queues on the controller */
5471/* */
5472/****************************************************************************/
5473static void
5474ips_statinit_memio(ips_ha_t * ha)
5475{
5476 uint32_t phys_status_start;
5477
5478 METHOD_TRACE("ips_statinit_memio", 1);
5479
5480 ha->adapt->p_status_start = ha->adapt->status;
5481 ha->adapt->p_status_end = ha->adapt->status + IPS_MAX_CMDS;
5482 ha->adapt->p_status_tail = ha->adapt->status;
5483
5484 phys_status_start = ha->adapt->hw_status_start;
5485 writel(phys_status_start, ha->mem_ptr + IPS_REG_SQSR);
5486 writel(phys_status_start + IPS_STATUS_Q_SIZE,
5487 ha->mem_ptr + IPS_REG_SQER);
5488 writel(phys_status_start + IPS_STATUS_SIZE, ha->mem_ptr + IPS_REG_SQHR);
5489 writel(phys_status_start, ha->mem_ptr + IPS_REG_SQTR);
5490
5491 ha->adapt->hw_status_tail = phys_status_start;
5492}
5493
5494/****************************************************************************/
5495/* */
5496/* Routine Name: ips_statupd_copperhead */
5497/* */
5498/* Routine Description: */
5499/* */
5500/* Remove an element from the status queue */
5501/* */
5502/****************************************************************************/
5503static uint32_t
5504ips_statupd_copperhead(ips_ha_t * ha)
5505{
5506 METHOD_TRACE("ips_statupd_copperhead", 1);
5507
5508 if (ha->adapt->p_status_tail != ha->adapt->p_status_end) {
5509 ha->adapt->p_status_tail++;
5510 ha->adapt->hw_status_tail += sizeof (IPS_STATUS);
5511 } else {
5512 ha->adapt->p_status_tail = ha->adapt->p_status_start;
5513 ha->adapt->hw_status_tail = ha->adapt->hw_status_start;
5514 }
5515
5516 outl(cpu_to_le32(ha->adapt->hw_status_tail),
5517 ha->io_addr + IPS_REG_SQTR);
5518
5519 return (ha->adapt->p_status_tail->value);
5520}
5521
5522/****************************************************************************/
5523/* */
5524/* Routine Name: ips_statupd_copperhead_memio */
5525/* */
5526/* Routine Description: */
5527/* */
5528/* Remove an element from the status queue */
5529/* */
5530/****************************************************************************/
5531static uint32_t
5532ips_statupd_copperhead_memio(ips_ha_t * ha)
5533{
5534 METHOD_TRACE("ips_statupd_copperhead_memio", 1);
5535
5536 if (ha->adapt->p_status_tail != ha->adapt->p_status_end) {
5537 ha->adapt->p_status_tail++;
5538 ha->adapt->hw_status_tail += sizeof (IPS_STATUS);
5539 } else {
5540 ha->adapt->p_status_tail = ha->adapt->p_status_start;
5541 ha->adapt->hw_status_tail = ha->adapt->hw_status_start;
5542 }
5543
5544 writel(ha->adapt->hw_status_tail, ha->mem_ptr + IPS_REG_SQTR);
5545
5546 return (ha->adapt->p_status_tail->value);
5547}
5548
5549/****************************************************************************/
5550/* */
5551/* Routine Name: ips_statupd_morpheus */
5552/* */
5553/* Routine Description: */
5554/* */
5555/* Remove an element from the status queue */
5556/* */
5557/****************************************************************************/
5558static uint32_t
5559ips_statupd_morpheus(ips_ha_t * ha)
5560{
5561 uint32_t val;
5562
5563 METHOD_TRACE("ips_statupd_morpheus", 1);
5564
5565 val = readl(ha->mem_ptr + IPS_REG_I2O_OUTMSGQ);
5566
5567 return (val);
5568}
5569
5570/****************************************************************************/
5571/* */
5572/* Routine Name: ips_issue_copperhead */
5573/* */
5574/* Routine Description: */
5575/* */
5576/* Send a command down to the controller */
5577/* */
5578/****************************************************************************/
5579static int
5580ips_issue_copperhead(ips_ha_t * ha, ips_scb_t * scb)
5581{
5582 uint32_t TimeOut;
5583 uint32_t val;
5584
5585 METHOD_TRACE("ips_issue_copperhead", 1);
5586
5587 if (scb->scsi_cmd) {
5588 DEBUG_VAR(2, "(%s%d) ips_issue: cmd 0x%X id %d (%d %d %d)",
5589 ips_name,
5590 ha->host_num,
5591 scb->cdb[0],
5592 scb->cmd.basic_io.command_id,
5593 scb->bus, scb->target_id, scb->lun);
5594 } else {
5595 DEBUG_VAR(2, KERN_NOTICE "(%s%d) ips_issue: logical cmd id %d",
5596 ips_name, ha->host_num, scb->cmd.basic_io.command_id);
5597 }
5598
5599 TimeOut = 0;
5600
5601 while ((val =
5602 le32_to_cpu(inl(ha->io_addr + IPS_REG_CCCR))) & IPS_BIT_SEM) {
5603 udelay(1000);
5604
5605 if (++TimeOut >= IPS_SEM_TIMEOUT) {
5606 if (!(val & IPS_BIT_START_STOP))
5607 break;
5608
5609 IPS_PRINTK(KERN_WARNING, ha->pcidev,
5610 "ips_issue val [0x%x].\n", val);
5611 IPS_PRINTK(KERN_WARNING, ha->pcidev,
5612 "ips_issue semaphore chk timeout.\n");
5613
5614 return (IPS_FAILURE);
5615 } /* end if */
5616 } /* end while */
5617
5618 outl(cpu_to_le32(scb->scb_busaddr), ha->io_addr + IPS_REG_CCSAR);
5619 outw(cpu_to_le32(IPS_BIT_START_CMD), ha->io_addr + IPS_REG_CCCR);
5620
5621 return (IPS_SUCCESS);
5622}
5623
5624/****************************************************************************/
5625/* */
5626/* Routine Name: ips_issue_copperhead_memio */
5627/* */
5628/* Routine Description: */
5629/* */
5630/* Send a command down to the controller */
5631/* */
5632/****************************************************************************/
5633static int
5634ips_issue_copperhead_memio(ips_ha_t * ha, ips_scb_t * scb)
5635{
5636 uint32_t TimeOut;
5637 uint32_t val;
5638
5639 METHOD_TRACE("ips_issue_copperhead_memio", 1);
5640
5641 if (scb->scsi_cmd) {
5642 DEBUG_VAR(2, "(%s%d) ips_issue: cmd 0x%X id %d (%d %d %d)",
5643 ips_name,
5644 ha->host_num,
5645 scb->cdb[0],
5646 scb->cmd.basic_io.command_id,
5647 scb->bus, scb->target_id, scb->lun);
5648 } else {
5649 DEBUG_VAR(2, "(%s%d) ips_issue: logical cmd id %d",
5650 ips_name, ha->host_num, scb->cmd.basic_io.command_id);
5651 }
5652
5653 TimeOut = 0;
5654
5655 while ((val = readl(ha->mem_ptr + IPS_REG_CCCR)) & IPS_BIT_SEM) {
5656 udelay(1000);
5657
5658 if (++TimeOut >= IPS_SEM_TIMEOUT) {
5659 if (!(val & IPS_BIT_START_STOP))
5660 break;
5661
5662 IPS_PRINTK(KERN_WARNING, ha->pcidev,
5663 "ips_issue val [0x%x].\n", val);
5664 IPS_PRINTK(KERN_WARNING, ha->pcidev,
5665 "ips_issue semaphore chk timeout.\n");
5666
5667 return (IPS_FAILURE);
5668 } /* end if */
5669 } /* end while */
5670
5671 writel(scb->scb_busaddr, ha->mem_ptr + IPS_REG_CCSAR);
5672 writel(IPS_BIT_START_CMD, ha->mem_ptr + IPS_REG_CCCR);
5673
5674 return (IPS_SUCCESS);
5675}
5676
5677/****************************************************************************/
5678/* */
5679/* Routine Name: ips_issue_i2o */
5680/* */
5681/* Routine Description: */
5682/* */
5683/* Send a command down to the controller */
5684/* */
5685/****************************************************************************/
5686static int
5687ips_issue_i2o(ips_ha_t * ha, ips_scb_t * scb)
5688{
5689
5690 METHOD_TRACE("ips_issue_i2o", 1);
5691
5692 if (scb->scsi_cmd) {
5693 DEBUG_VAR(2, "(%s%d) ips_issue: cmd 0x%X id %d (%d %d %d)",
5694 ips_name,
5695 ha->host_num,
5696 scb->cdb[0],
5697 scb->cmd.basic_io.command_id,
5698 scb->bus, scb->target_id, scb->lun);
5699 } else {
5700 DEBUG_VAR(2, "(%s%d) ips_issue: logical cmd id %d",
5701 ips_name, ha->host_num, scb->cmd.basic_io.command_id);
5702 }
5703
5704 outl(cpu_to_le32(scb->scb_busaddr), ha->io_addr + IPS_REG_I2O_INMSGQ);
5705
5706 return (IPS_SUCCESS);
5707}
5708
5709/****************************************************************************/
5710/* */
5711/* Routine Name: ips_issue_i2o_memio */
5712/* */
5713/* Routine Description: */
5714/* */
5715/* Send a command down to the controller */
5716/* */
5717/****************************************************************************/
5718static int
5719ips_issue_i2o_memio(ips_ha_t * ha, ips_scb_t * scb)
5720{
5721
5722 METHOD_TRACE("ips_issue_i2o_memio", 1);
5723
5724 if (scb->scsi_cmd) {
5725 DEBUG_VAR(2, "(%s%d) ips_issue: cmd 0x%X id %d (%d %d %d)",
5726 ips_name,
5727 ha->host_num,
5728 scb->cdb[0],
5729 scb->cmd.basic_io.command_id,
5730 scb->bus, scb->target_id, scb->lun);
5731 } else {
5732 DEBUG_VAR(2, "(%s%d) ips_issue: logical cmd id %d",
5733 ips_name, ha->host_num, scb->cmd.basic_io.command_id);
5734 }
5735
5736 writel(scb->scb_busaddr, ha->mem_ptr + IPS_REG_I2O_INMSGQ);
5737
5738 return (IPS_SUCCESS);
5739}
5740
5741/****************************************************************************/
5742/* */
5743/* Routine Name: ips_isintr_copperhead */
5744/* */
5745/* Routine Description: */
5746/* */
5747/* Test to see if an interrupt is for us */
5748/* */
5749/****************************************************************************/
5750static int
5751ips_isintr_copperhead(ips_ha_t * ha)
5752{
5753 uint8_t Isr;
5754
5755 METHOD_TRACE("ips_isintr_copperhead", 2);
5756
5757 Isr = inb(ha->io_addr + IPS_REG_HISR);
5758
5759 if (Isr == 0xFF)
5760 /* ?!?! Nothing really there */
5761 return (0);
5762
5763 if (Isr & IPS_BIT_SCE)
5764 return (1);
5765 else if (Isr & (IPS_BIT_SQO | IPS_BIT_GHI)) {
5766 /* status queue overflow or GHI */
5767 /* just clear the interrupt */
5768 outb(Isr, ha->io_addr + IPS_REG_HISR);
5769 }
5770
5771 return (0);
5772}
5773
5774/****************************************************************************/
5775/* */
5776/* Routine Name: ips_isintr_copperhead_memio */
5777/* */
5778/* Routine Description: */
5779/* */
5780/* Test to see if an interrupt is for us */
5781/* */
5782/****************************************************************************/
5783static int
5784ips_isintr_copperhead_memio(ips_ha_t * ha)
5785{
5786 uint8_t Isr;
5787
5788 METHOD_TRACE("ips_isintr_memio", 2);
5789
5790 Isr = readb(ha->mem_ptr + IPS_REG_HISR);
5791
5792 if (Isr == 0xFF)
5793 /* ?!?! Nothing really there */
5794 return (0);
5795
5796 if (Isr & IPS_BIT_SCE)
5797 return (1);
5798 else if (Isr & (IPS_BIT_SQO | IPS_BIT_GHI)) {
5799 /* status queue overflow or GHI */
5800 /* just clear the interrupt */
5801 writeb(Isr, ha->mem_ptr + IPS_REG_HISR);
5802 }
5803
5804 return (0);
5805}
5806
5807/****************************************************************************/
5808/* */
5809/* Routine Name: ips_isintr_morpheus */
5810/* */
5811/* Routine Description: */
5812/* */
5813/* Test to see if an interrupt is for us */
5814/* */
5815/****************************************************************************/
5816static int
5817ips_isintr_morpheus(ips_ha_t * ha)
5818{
5819 uint32_t Isr;
5820
5821 METHOD_TRACE("ips_isintr_morpheus", 2);
5822
5823 Isr = readl(ha->mem_ptr + IPS_REG_I2O_HIR);
5824
5825 if (Isr & IPS_BIT_I2O_OPQI)
5826 return (1);
5827 else
5828 return (0);
5829}
5830
5831/****************************************************************************/
5832/* */
5833/* Routine Name: ips_wait */
5834/* */
5835/* Routine Description: */
5836/* */
5837/* Wait for a command to complete */
5838/* */
5839/****************************************************************************/
5840static int
5841ips_wait(ips_ha_t * ha, int time, int intr)
5842{
5843 int ret;
5844 int done;
5845
5846 METHOD_TRACE("ips_wait", 1);
5847
5848 ret = IPS_FAILURE;
5849 done = FALSE;
5850
5851 time *= IPS_ONE_SEC; /* convert seconds */
5852
5853 while ((time > 0) && (!done)) {
5854 if (intr == IPS_INTR_ON) {
5855 if (ha->waitflag == FALSE) {
5856 ret = IPS_SUCCESS;
5857 done = TRUE;
5858 break;
5859 }
5860 } else if (intr == IPS_INTR_IORL) {
5861 if (ha->waitflag == FALSE) {
5862 /*
5863 * controller generated an interrupt to
5864 * acknowledge completion of the command
5865 * and ips_intr() has serviced the interrupt.
5866 */
5867 ret = IPS_SUCCESS;
5868 done = TRUE;
5869 break;
5870 }
5871
5872 /*
5873 * NOTE: we already have the io_request_lock so
5874 * even if we get an interrupt it won't get serviced
5875 * until after we finish.
5876 */
5877
5878 (*ha->func.intr) (ha);
5879 }
5880
5881 /* This looks like a very evil loop, but it only does this during start-up */
5882 udelay(1000);
5883 time--;
5884 }
5885
5886 return (ret);
5887}
5888
5889/****************************************************************************/
5890/* */
5891/* Routine Name: ips_write_driver_status */
5892/* */
5893/* Routine Description: */
5894/* */
5895/* Write OS/Driver version to Page 5 of the nvram on the controller */
5896/* */
5897/****************************************************************************/
5898static int
5899ips_write_driver_status(ips_ha_t * ha, int intr)
5900{
5901 METHOD_TRACE("ips_write_driver_status", 1);
5902
5903 if (!ips_readwrite_page5(ha, FALSE, intr)) {
5904 IPS_PRINTK(KERN_WARNING, ha->pcidev,
5905 "unable to read NVRAM page 5.\n");
5906
5907 return (0);
5908 }
5909
5910 /* check to make sure the page has a valid */
5911 /* signature */
5912 if (le32_to_cpu(ha->nvram->signature) != IPS_NVRAM_P5_SIG) {
5913 DEBUG_VAR(1,
5914 "(%s%d) NVRAM page 5 has an invalid signature: %X.",
5915 ips_name, ha->host_num, ha->nvram->signature);
5916 ha->nvram->signature = IPS_NVRAM_P5_SIG;
5917 }
5918
5919 DEBUG_VAR(2,
5920 "(%s%d) Ad Type: %d, Ad Slot: %d, BIOS: %c%c%c%c %c%c%c%c.",
5921 ips_name, ha->host_num, le16_to_cpu(ha->nvram->adapter_type),
5922 ha->nvram->adapter_slot, ha->nvram->bios_high[0],
5923 ha->nvram->bios_high[1], ha->nvram->bios_high[2],
5924 ha->nvram->bios_high[3], ha->nvram->bios_low[0],
5925 ha->nvram->bios_low[1], ha->nvram->bios_low[2],
5926 ha->nvram->bios_low[3]);
5927
5928 ips_get_bios_version(ha, intr);
5929
5930 /* change values (as needed) */
5931 ha->nvram->operating_system = IPS_OS_LINUX;
5932 ha->nvram->adapter_type = ha->ad_type;
5933 strncpy((char *) ha->nvram->driver_high, IPS_VERSION_HIGH, 4);
5934 strncpy((char *) ha->nvram->driver_low, IPS_VERSION_LOW, 4);
5935 strncpy((char *) ha->nvram->bios_high, ha->bios_version, 4);
5936 strncpy((char *) ha->nvram->bios_low, ha->bios_version + 4, 4);
5937
5938 ips_version_check(ha, intr); /* Check BIOS/FW/Driver Versions */
5939
5940 /* now update the page */
5941 if (!ips_readwrite_page5(ha, TRUE, intr)) {
5942 IPS_PRINTK(KERN_WARNING, ha->pcidev,
5943 "unable to write NVRAM page 5.\n");
5944
5945 return (0);
5946 }
5947
5948 /* IF NVRAM Page 5 is OK, Use it for Slot Number Info Because Linux Doesn't Do Slots */
5949 ha->slot_num = ha->nvram->adapter_slot;
5950
5951 return (1);
5952}
5953
5954/****************************************************************************/
5955/* */
5956/* Routine Name: ips_read_adapter_status */
5957/* */
5958/* Routine Description: */
5959/* */
5960/* Do an Inquiry command to the adapter */
5961/* */
5962/****************************************************************************/
5963static int
5964ips_read_adapter_status(ips_ha_t * ha, int intr)
5965{
5966 ips_scb_t *scb;
5967 int ret;
5968
5969 METHOD_TRACE("ips_read_adapter_status", 1);
5970
5971 scb = &ha->scbs[ha->max_cmds - 1];
5972
5973 ips_init_scb(ha, scb);
5974
5975 scb->timeout = ips_cmd_timeout;
5976 scb->cdb[0] = IPS_CMD_ENQUIRY;
5977
5978 scb->cmd.basic_io.op_code = IPS_CMD_ENQUIRY;
5979 scb->cmd.basic_io.command_id = IPS_COMMAND_ID(ha, scb);
5980 scb->cmd.basic_io.sg_count = 0;
5981 scb->cmd.basic_io.lba = 0;
5982 scb->cmd.basic_io.sector_count = 0;
5983 scb->cmd.basic_io.log_drv = 0;
5984 scb->data_len = sizeof (*ha->enq);
5985 scb->cmd.basic_io.sg_addr = ha->enq_busaddr;
5986
5987 /* send command */
5988 if (((ret =
5989 ips_send_wait(ha, scb, ips_cmd_timeout, intr)) == IPS_FAILURE)
5990 || (ret == IPS_SUCCESS_IMM)
5991 || ((scb->basic_status & IPS_GSC_STATUS_MASK) > 1))
5992 return (0);
5993
5994 return (1);
5995}
5996
5997/****************************************************************************/
5998/* */
5999/* Routine Name: ips_read_subsystem_parameters */
6000/* */
6001/* Routine Description: */
6002/* */
6003/* Read subsystem parameters from the adapter */
6004/* */
6005/****************************************************************************/
6006static int
6007ips_read_subsystem_parameters(ips_ha_t * ha, int intr)
6008{
6009 ips_scb_t *scb;
6010 int ret;
6011
6012 METHOD_TRACE("ips_read_subsystem_parameters", 1);
6013
6014 scb = &ha->scbs[ha->max_cmds - 1];
6015
6016 ips_init_scb(ha, scb);
6017
6018 scb->timeout = ips_cmd_timeout;
6019 scb->cdb[0] = IPS_CMD_GET_SUBSYS;
6020
6021 scb->cmd.basic_io.op_code = IPS_CMD_GET_SUBSYS;
6022 scb->cmd.basic_io.command_id = IPS_COMMAND_ID(ha, scb);
6023 scb->cmd.basic_io.sg_count = 0;
6024 scb->cmd.basic_io.lba = 0;
6025 scb->cmd.basic_io.sector_count = 0;
6026 scb->cmd.basic_io.log_drv = 0;
6027 scb->data_len = sizeof (*ha->subsys);
6028 scb->cmd.basic_io.sg_addr = ha->ioctl_busaddr;
6029
6030 /* send command */
6031 if (((ret =
6032 ips_send_wait(ha, scb, ips_cmd_timeout, intr)) == IPS_FAILURE)
6033 || (ret == IPS_SUCCESS_IMM)
6034 || ((scb->basic_status & IPS_GSC_STATUS_MASK) > 1))
6035 return (0);
6036
6037 memcpy(ha->subsys, ha->ioctl_data, sizeof(*ha->subsys));
6038 return (1);
6039}
6040
6041/****************************************************************************/
6042/* */
6043/* Routine Name: ips_read_config */
6044/* */
6045/* Routine Description: */
6046/* */
6047/* Read the configuration on the adapter */
6048/* */
6049/****************************************************************************/
6050static int
6051ips_read_config(ips_ha_t * ha, int intr)
6052{
6053 ips_scb_t *scb;
6054 int i;
6055 int ret;
6056
6057 METHOD_TRACE("ips_read_config", 1);
6058
6059 /* set defaults for initiator IDs */
6060 for (i = 0; i < 4; i++)
6061 ha->conf->init_id[i] = 7;
6062
6063 scb = &ha->scbs[ha->max_cmds - 1];
6064
6065 ips_init_scb(ha, scb);
6066
6067 scb->timeout = ips_cmd_timeout;
6068 scb->cdb[0] = IPS_CMD_READ_CONF;
6069
6070 scb->cmd.basic_io.op_code = IPS_CMD_READ_CONF;
6071 scb->cmd.basic_io.command_id = IPS_COMMAND_ID(ha, scb);
6072 scb->data_len = sizeof (*ha->conf);
6073 scb->cmd.basic_io.sg_addr = ha->ioctl_busaddr;
6074
6075 /* send command */
6076 if (((ret =
6077 ips_send_wait(ha, scb, ips_cmd_timeout, intr)) == IPS_FAILURE)
6078 || (ret == IPS_SUCCESS_IMM)
6079 || ((scb->basic_status & IPS_GSC_STATUS_MASK) > 1)) {
6080
6081 memset(ha->conf, 0, sizeof (IPS_CONF));
6082
6083 /* reset initiator IDs */
6084 for (i = 0; i < 4; i++)
6085 ha->conf->init_id[i] = 7;
6086
6087 /* Allow Completed with Errors, so JCRM can access the Adapter to fix the problems */
6088 if ((scb->basic_status & IPS_GSC_STATUS_MASK) ==
6089 IPS_CMD_CMPLT_WERROR)
6090 return (1);
6091
6092 return (0);
6093 }
6094
6095 memcpy(ha->conf, ha->ioctl_data, sizeof(*ha->conf));
6096 return (1);
6097}
6098
6099/****************************************************************************/
6100/* */
6101/* Routine Name: ips_readwrite_page5 */
6102/* */
6103/* Routine Description: */
6104/* */
6105/* Read nvram page 5 from the adapter */
6106/* */
6107/****************************************************************************/
6108static int
6109ips_readwrite_page5(ips_ha_t * ha, int write, int intr)
6110{
6111 ips_scb_t *scb;
6112 int ret;
6113
6114 METHOD_TRACE("ips_readwrite_page5", 1);
6115
6116 scb = &ha->scbs[ha->max_cmds - 1];
6117
6118 ips_init_scb(ha, scb);
6119
6120 scb->timeout = ips_cmd_timeout;
6121 scb->cdb[0] = IPS_CMD_RW_NVRAM_PAGE;
6122
6123 scb->cmd.nvram.op_code = IPS_CMD_RW_NVRAM_PAGE;
6124 scb->cmd.nvram.command_id = IPS_COMMAND_ID(ha, scb);
6125 scb->cmd.nvram.page = 5;
6126 scb->cmd.nvram.write = write;
6127 scb->cmd.nvram.reserved = 0;
6128 scb->cmd.nvram.reserved2 = 0;
6129 scb->data_len = sizeof (*ha->nvram);
6130 scb->cmd.nvram.buffer_addr = ha->ioctl_busaddr;
6131 if (write)
6132 memcpy(ha->ioctl_data, ha->nvram, sizeof(*ha->nvram));
6133
6134 /* issue the command */
6135 if (((ret =
6136 ips_send_wait(ha, scb, ips_cmd_timeout, intr)) == IPS_FAILURE)
6137 || (ret == IPS_SUCCESS_IMM)
6138 || ((scb->basic_status & IPS_GSC_STATUS_MASK) > 1)) {
6139
6140 memset(ha->nvram, 0, sizeof (IPS_NVRAM_P5));
6141
6142 return (0);
6143 }
6144 if (!write)
6145 memcpy(ha->nvram, ha->ioctl_data, sizeof(*ha->nvram));
6146 return (1);
6147}
6148
6149/****************************************************************************/
6150/* */
6151/* Routine Name: ips_clear_adapter */
6152/* */
6153/* Routine Description: */
6154/* */
6155/* Clear the stripe lock tables */
6156/* */
6157/****************************************************************************/
6158static int
6159ips_clear_adapter(ips_ha_t * ha, int intr)
6160{
6161 ips_scb_t *scb;
6162 int ret;
6163
6164 METHOD_TRACE("ips_clear_adapter", 1);
6165
6166 scb = &ha->scbs[ha->max_cmds - 1];
6167
6168 ips_init_scb(ha, scb);
6169
6170 scb->timeout = ips_reset_timeout;
6171 scb->cdb[0] = IPS_CMD_CONFIG_SYNC;
6172
6173 scb->cmd.config_sync.op_code = IPS_CMD_CONFIG_SYNC;
6174 scb->cmd.config_sync.command_id = IPS_COMMAND_ID(ha, scb);
6175 scb->cmd.config_sync.channel = 0;
6176 scb->cmd.config_sync.source_target = IPS_POCL;
6177 scb->cmd.config_sync.reserved = 0;
6178 scb->cmd.config_sync.reserved2 = 0;
6179 scb->cmd.config_sync.reserved3 = 0;
6180
6181 /* issue command */
6182 if (((ret =
6183 ips_send_wait(ha, scb, ips_reset_timeout, intr)) == IPS_FAILURE)
6184 || (ret == IPS_SUCCESS_IMM)
6185 || ((scb->basic_status & IPS_GSC_STATUS_MASK) > 1))
6186 return (0);
6187
6188 /* send unlock stripe command */
6189 ips_init_scb(ha, scb);
6190
6191 scb->cdb[0] = IPS_CMD_ERROR_TABLE;
6192 scb->timeout = ips_reset_timeout;
6193
6194 scb->cmd.unlock_stripe.op_code = IPS_CMD_ERROR_TABLE;
6195 scb->cmd.unlock_stripe.command_id = IPS_COMMAND_ID(ha, scb);
6196 scb->cmd.unlock_stripe.log_drv = 0;
6197 scb->cmd.unlock_stripe.control = IPS_CSL;
6198 scb->cmd.unlock_stripe.reserved = 0;
6199 scb->cmd.unlock_stripe.reserved2 = 0;
6200 scb->cmd.unlock_stripe.reserved3 = 0;
6201
6202 /* issue command */
6203 if (((ret =
6204 ips_send_wait(ha, scb, ips_cmd_timeout, intr)) == IPS_FAILURE)
6205 || (ret == IPS_SUCCESS_IMM)
6206 || ((scb->basic_status & IPS_GSC_STATUS_MASK) > 1))
6207 return (0);
6208
6209 return (1);
6210}
6211
6212/****************************************************************************/
6213/* */
6214/* Routine Name: ips_ffdc_reset */
6215/* */
6216/* Routine Description: */
6217/* */
6218/* FFDC: write reset info */
6219/* */
6220/****************************************************************************/
6221static void
6222ips_ffdc_reset(ips_ha_t * ha, int intr)
6223{
6224 ips_scb_t *scb;
6225
6226 METHOD_TRACE("ips_ffdc_reset", 1);
6227
6228 scb = &ha->scbs[ha->max_cmds - 1];
6229
6230 ips_init_scb(ha, scb);
6231
6232 scb->timeout = ips_cmd_timeout;
6233 scb->cdb[0] = IPS_CMD_FFDC;
6234 scb->cmd.ffdc.op_code = IPS_CMD_FFDC;
6235 scb->cmd.ffdc.command_id = IPS_COMMAND_ID(ha, scb);
6236 scb->cmd.ffdc.reset_count = ha->reset_count;
6237 scb->cmd.ffdc.reset_type = 0x80;
6238
6239 /* convert time to what the card wants */
6240 ips_fix_ffdc_time(ha, scb, ha->last_ffdc);
6241
6242 /* issue command */
6243 ips_send_wait(ha, scb, ips_cmd_timeout, intr);
6244}
6245
6246/****************************************************************************/
6247/* */
6248/* Routine Name: ips_ffdc_time */
6249/* */
6250/* Routine Description: */
6251/* */
6252/* FFDC: write time info */
6253/* */
6254/****************************************************************************/
6255static void
6256ips_ffdc_time(ips_ha_t * ha)
6257{
6258 ips_scb_t *scb;
6259
6260 METHOD_TRACE("ips_ffdc_time", 1);
6261
6262 DEBUG_VAR(1, "(%s%d) Sending time update.", ips_name, ha->host_num);
6263
6264 scb = &ha->scbs[ha->max_cmds - 1];
6265
6266 ips_init_scb(ha, scb);
6267
6268 scb->timeout = ips_cmd_timeout;
6269 scb->cdb[0] = IPS_CMD_FFDC;
6270 scb->cmd.ffdc.op_code = IPS_CMD_FFDC;
6271 scb->cmd.ffdc.command_id = IPS_COMMAND_ID(ha, scb);
6272 scb->cmd.ffdc.reset_count = 0;
6273 scb->cmd.ffdc.reset_type = 0;
6274
6275 /* convert time to what the card wants */
6276 ips_fix_ffdc_time(ha, scb, ha->last_ffdc);
6277
6278 /* issue command */
6279 ips_send_wait(ha, scb, ips_cmd_timeout, IPS_FFDC);
6280}
6281
6282/****************************************************************************/
6283/* */
6284/* Routine Name: ips_fix_ffdc_time */
6285/* */
6286/* Routine Description: */
6287/* Adjust time_t to what the card wants */
6288/* */
6289/****************************************************************************/
6290static void
6291ips_fix_ffdc_time(ips_ha_t * ha, ips_scb_t * scb, time_t current_time)
6292{
6293 long days;
6294 long rem;
6295 int i;
6296 int year;
6297 int yleap;
6298 int year_lengths[2] = { IPS_DAYS_NORMAL_YEAR, IPS_DAYS_LEAP_YEAR };
6299 int month_lengths[12][2] = { {31, 31},
6300 {28, 29},
6301 {31, 31},
6302 {30, 30},
6303 {31, 31},
6304 {30, 30},
6305 {31, 31},
6306 {31, 31},
6307 {30, 30},
6308 {31, 31},
6309 {30, 30},
6310 {31, 31}
6311 };
6312
6313 METHOD_TRACE("ips_fix_ffdc_time", 1);
6314
6315 days = current_time / IPS_SECS_DAY;
6316 rem = current_time % IPS_SECS_DAY;
6317
6318 scb->cmd.ffdc.hour = (rem / IPS_SECS_HOUR);
6319 rem = rem % IPS_SECS_HOUR;
6320 scb->cmd.ffdc.minute = (rem / IPS_SECS_MIN);
6321 scb->cmd.ffdc.second = (rem % IPS_SECS_MIN);
6322
6323 year = IPS_EPOCH_YEAR;
6324 while (days < 0 || days >= year_lengths[yleap = IPS_IS_LEAP_YEAR(year)]) {
6325 int newy;
6326
6327 newy = year + (days / IPS_DAYS_NORMAL_YEAR);
6328 if (days < 0)
6329 --newy;
6330 days -= (newy - year) * IPS_DAYS_NORMAL_YEAR +
6331 IPS_NUM_LEAP_YEARS_THROUGH(newy - 1) -
6332 IPS_NUM_LEAP_YEARS_THROUGH(year - 1);
6333 year = newy;
6334 }
6335
6336 scb->cmd.ffdc.yearH = year / 100;
6337 scb->cmd.ffdc.yearL = year % 100;
6338
6339 for (i = 0; days >= month_lengths[i][yleap]; ++i)
6340 days -= month_lengths[i][yleap];
6341
6342 scb->cmd.ffdc.month = i + 1;
6343 scb->cmd.ffdc.day = days + 1;
6344}
6345
6346/****************************************************************************
6347 * BIOS Flash Routines *
6348 ****************************************************************************/
6349
6350/****************************************************************************/
6351/* */
6352/* Routine Name: ips_erase_bios */
6353/* */
6354/* Routine Description: */
6355/* Erase the BIOS on the adapter */
6356/* */
6357/****************************************************************************/
6358static int
6359ips_erase_bios(ips_ha_t * ha)
6360{
6361 int timeout;
6362 uint8_t status = 0;
6363
6364 METHOD_TRACE("ips_erase_bios", 1);
6365
6366 status = 0;
6367
6368 /* Clear the status register */
6369 outl(0, ha->io_addr + IPS_REG_FLAP);
6370 if (ha->revision_id == IPS_REVID_TROMBONE64)
6371 udelay(25); /* 25 us */
6372
6373 outb(0x50, ha->io_addr + IPS_REG_FLDP);
6374 if (ha->revision_id == IPS_REVID_TROMBONE64)
6375 udelay(25); /* 25 us */
6376
6377 /* Erase Setup */
6378 outb(0x20, ha->io_addr + IPS_REG_FLDP);
6379 if (ha->revision_id == IPS_REVID_TROMBONE64)
6380 udelay(25); /* 25 us */
6381
6382 /* Erase Confirm */
6383 outb(0xD0, ha->io_addr + IPS_REG_FLDP);
6384 if (ha->revision_id == IPS_REVID_TROMBONE64)
6385 udelay(25); /* 25 us */
6386
6387 /* Erase Status */
6388 outb(0x70, ha->io_addr + IPS_REG_FLDP);
6389 if (ha->revision_id == IPS_REVID_TROMBONE64)
6390 udelay(25); /* 25 us */
6391
6392 timeout = 80000; /* 80 seconds */
6393
6394 while (timeout > 0) {
6395 if (ha->revision_id == IPS_REVID_TROMBONE64) {
6396 outl(0, ha->io_addr + IPS_REG_FLAP);
6397 udelay(25); /* 25 us */
6398 }
6399
6400 status = inb(ha->io_addr + IPS_REG_FLDP);
6401
6402 if (status & 0x80)
6403 break;
6404
6405 MDELAY(1);
6406 timeout--;
6407 }
6408
6409 /* check for timeout */
6410 if (timeout <= 0) {
6411 /* timeout */
6412
6413 /* try to suspend the erase */
6414 outb(0xB0, ha->io_addr + IPS_REG_FLDP);
6415 if (ha->revision_id == IPS_REVID_TROMBONE64)
6416 udelay(25); /* 25 us */
6417
6418 /* wait for 10 seconds */
6419 timeout = 10000;
6420 while (timeout > 0) {
6421 if (ha->revision_id == IPS_REVID_TROMBONE64) {
6422 outl(0, ha->io_addr + IPS_REG_FLAP);
6423 udelay(25); /* 25 us */
6424 }
6425
6426 status = inb(ha->io_addr + IPS_REG_FLDP);
6427
6428 if (status & 0xC0)
6429 break;
6430
6431 MDELAY(1);
6432 timeout--;
6433 }
6434
6435 return (1);
6436 }
6437
6438 /* check for valid VPP */
6439 if (status & 0x08)
6440 /* VPP failure */
6441 return (1);
6442
6443 /* check for succesful flash */
6444 if (status & 0x30)
6445 /* sequence error */
6446 return (1);
6447
6448 /* Otherwise, we were successful */
6449 /* clear status */
6450 outb(0x50, ha->io_addr + IPS_REG_FLDP);
6451 if (ha->revision_id == IPS_REVID_TROMBONE64)
6452 udelay(25); /* 25 us */
6453
6454 /* enable reads */
6455 outb(0xFF, ha->io_addr + IPS_REG_FLDP);
6456 if (ha->revision_id == IPS_REVID_TROMBONE64)
6457 udelay(25); /* 25 us */
6458
6459 return (0);
6460}
6461
6462/****************************************************************************/
6463/* */
6464/* Routine Name: ips_erase_bios_memio */
6465/* */
6466/* Routine Description: */
6467/* Erase the BIOS on the adapter */
6468/* */
6469/****************************************************************************/
6470static int
6471ips_erase_bios_memio(ips_ha_t * ha)
6472{
6473 int timeout;
6474 uint8_t status;
6475
6476 METHOD_TRACE("ips_erase_bios_memio", 1);
6477
6478 status = 0;
6479
6480 /* Clear the status register */
6481 writel(0, ha->mem_ptr + IPS_REG_FLAP);
6482 if (ha->revision_id == IPS_REVID_TROMBONE64)
6483 udelay(25); /* 25 us */
6484
6485 writeb(0x50, ha->mem_ptr + IPS_REG_FLDP);
6486 if (ha->revision_id == IPS_REVID_TROMBONE64)
6487 udelay(25); /* 25 us */
6488
6489 /* Erase Setup */
6490 writeb(0x20, ha->mem_ptr + IPS_REG_FLDP);
6491 if (ha->revision_id == IPS_REVID_TROMBONE64)
6492 udelay(25); /* 25 us */
6493
6494 /* Erase Confirm */
6495 writeb(0xD0, ha->mem_ptr + IPS_REG_FLDP);
6496 if (ha->revision_id == IPS_REVID_TROMBONE64)
6497 udelay(25); /* 25 us */
6498
6499 /* Erase Status */
6500 writeb(0x70, ha->mem_ptr + IPS_REG_FLDP);
6501 if (ha->revision_id == IPS_REVID_TROMBONE64)
6502 udelay(25); /* 25 us */
6503
6504 timeout = 80000; /* 80 seconds */
6505
6506 while (timeout > 0) {
6507 if (ha->revision_id == IPS_REVID_TROMBONE64) {
6508 writel(0, ha->mem_ptr + IPS_REG_FLAP);
6509 udelay(25); /* 25 us */
6510 }
6511
6512 status = readb(ha->mem_ptr + IPS_REG_FLDP);
6513
6514 if (status & 0x80)
6515 break;
6516
6517 MDELAY(1);
6518 timeout--;
6519 }
6520
6521 /* check for timeout */
6522 if (timeout <= 0) {
6523 /* timeout */
6524
6525 /* try to suspend the erase */
6526 writeb(0xB0, ha->mem_ptr + IPS_REG_FLDP);
6527 if (ha->revision_id == IPS_REVID_TROMBONE64)
6528 udelay(25); /* 25 us */
6529
6530 /* wait for 10 seconds */
6531 timeout = 10000;
6532 while (timeout > 0) {
6533 if (ha->revision_id == IPS_REVID_TROMBONE64) {
6534 writel(0, ha->mem_ptr + IPS_REG_FLAP);
6535 udelay(25); /* 25 us */
6536 }
6537
6538 status = readb(ha->mem_ptr + IPS_REG_FLDP);
6539
6540 if (status & 0xC0)
6541 break;
6542
6543 MDELAY(1);
6544 timeout--;
6545 }
6546
6547 return (1);
6548 }
6549
6550 /* check for valid VPP */
6551 if (status & 0x08)
6552 /* VPP failure */
6553 return (1);
6554
6555 /* check for succesful flash */
6556 if (status & 0x30)
6557 /* sequence error */
6558 return (1);
6559
6560 /* Otherwise, we were successful */
6561 /* clear status */
6562 writeb(0x50, ha->mem_ptr + IPS_REG_FLDP);
6563 if (ha->revision_id == IPS_REVID_TROMBONE64)
6564 udelay(25); /* 25 us */
6565
6566 /* enable reads */
6567 writeb(0xFF, ha->mem_ptr + IPS_REG_FLDP);
6568 if (ha->revision_id == IPS_REVID_TROMBONE64)
6569 udelay(25); /* 25 us */
6570
6571 return (0);
6572}
6573
6574/****************************************************************************/
6575/* */
6576/* Routine Name: ips_program_bios */
6577/* */
6578/* Routine Description: */
6579/* Program the BIOS on the adapter */
6580/* */
6581/****************************************************************************/
6582static int
6583ips_program_bios(ips_ha_t * ha, char *buffer, uint32_t buffersize,
6584 uint32_t offset)
6585{
6586 int i;
6587 int timeout;
6588 uint8_t status = 0;
6589
6590 METHOD_TRACE("ips_program_bios", 1);
6591
6592 status = 0;
6593
6594 for (i = 0; i < buffersize; i++) {
6595 /* write a byte */
6596 outl(cpu_to_le32(i + offset), ha->io_addr + IPS_REG_FLAP);
6597 if (ha->revision_id == IPS_REVID_TROMBONE64)
6598 udelay(25); /* 25 us */
6599
6600 outb(0x40, ha->io_addr + IPS_REG_FLDP);
6601 if (ha->revision_id == IPS_REVID_TROMBONE64)
6602 udelay(25); /* 25 us */
6603
6604 outb(buffer[i], ha->io_addr + IPS_REG_FLDP);
6605 if (ha->revision_id == IPS_REVID_TROMBONE64)
6606 udelay(25); /* 25 us */
6607
6608 /* wait up to one second */
6609 timeout = 1000;
6610 while (timeout > 0) {
6611 if (ha->revision_id == IPS_REVID_TROMBONE64) {
6612 outl(0, ha->io_addr + IPS_REG_FLAP);
6613 udelay(25); /* 25 us */
6614 }
6615
6616 status = inb(ha->io_addr + IPS_REG_FLDP);
6617
6618 if (status & 0x80)
6619 break;
6620
6621 MDELAY(1);
6622 timeout--;
6623 }
6624
6625 if (timeout == 0) {
6626 /* timeout error */
6627 outl(0, ha->io_addr + IPS_REG_FLAP);
6628 if (ha->revision_id == IPS_REVID_TROMBONE64)
6629 udelay(25); /* 25 us */
6630
6631 outb(0xFF, ha->io_addr + IPS_REG_FLDP);
6632 if (ha->revision_id == IPS_REVID_TROMBONE64)
6633 udelay(25); /* 25 us */
6634
6635 return (1);
6636 }
6637
6638 /* check the status */
6639 if (status & 0x18) {
6640 /* programming error */
6641 outl(0, ha->io_addr + IPS_REG_FLAP);
6642 if (ha->revision_id == IPS_REVID_TROMBONE64)
6643 udelay(25); /* 25 us */
6644
6645 outb(0xFF, ha->io_addr + IPS_REG_FLDP);
6646 if (ha->revision_id == IPS_REVID_TROMBONE64)
6647 udelay(25); /* 25 us */
6648
6649 return (1);
6650 }
6651 } /* end for */
6652
6653 /* Enable reading */
6654 outl(0, ha->io_addr + IPS_REG_FLAP);
6655 if (ha->revision_id == IPS_REVID_TROMBONE64)
6656 udelay(25); /* 25 us */
6657
6658 outb(0xFF, ha->io_addr + IPS_REG_FLDP);
6659 if (ha->revision_id == IPS_REVID_TROMBONE64)
6660 udelay(25); /* 25 us */
6661
6662 return (0);
6663}
6664
6665/****************************************************************************/
6666/* */
6667/* Routine Name: ips_program_bios_memio */
6668/* */
6669/* Routine Description: */
6670/* Program the BIOS on the adapter */
6671/* */
6672/****************************************************************************/
6673static int
6674ips_program_bios_memio(ips_ha_t * ha, char *buffer, uint32_t buffersize,
6675 uint32_t offset)
6676{
6677 int i;
6678 int timeout;
6679 uint8_t status = 0;
6680
6681 METHOD_TRACE("ips_program_bios_memio", 1);
6682
6683 status = 0;
6684
6685 for (i = 0; i < buffersize; i++) {
6686 /* write a byte */
6687 writel(i + offset, ha->mem_ptr + IPS_REG_FLAP);
6688 if (ha->revision_id == IPS_REVID_TROMBONE64)
6689 udelay(25); /* 25 us */
6690
6691 writeb(0x40, ha->mem_ptr + IPS_REG_FLDP);
6692 if (ha->revision_id == IPS_REVID_TROMBONE64)
6693 udelay(25); /* 25 us */
6694
6695 writeb(buffer[i], ha->mem_ptr + IPS_REG_FLDP);
6696 if (ha->revision_id == IPS_REVID_TROMBONE64)
6697 udelay(25); /* 25 us */
6698
6699 /* wait up to one second */
6700 timeout = 1000;
6701 while (timeout > 0) {
6702 if (ha->revision_id == IPS_REVID_TROMBONE64) {
6703 writel(0, ha->mem_ptr + IPS_REG_FLAP);
6704 udelay(25); /* 25 us */
6705 }
6706
6707 status = readb(ha->mem_ptr + IPS_REG_FLDP);
6708
6709 if (status & 0x80)
6710 break;
6711
6712 MDELAY(1);
6713 timeout--;
6714 }
6715
6716 if (timeout == 0) {
6717 /* timeout error */
6718 writel(0, ha->mem_ptr + IPS_REG_FLAP);
6719 if (ha->revision_id == IPS_REVID_TROMBONE64)
6720 udelay(25); /* 25 us */
6721
6722 writeb(0xFF, ha->mem_ptr + IPS_REG_FLDP);
6723 if (ha->revision_id == IPS_REVID_TROMBONE64)
6724 udelay(25); /* 25 us */
6725
6726 return (1);
6727 }
6728
6729 /* check the status */
6730 if (status & 0x18) {
6731 /* programming error */
6732 writel(0, ha->mem_ptr + IPS_REG_FLAP);
6733 if (ha->revision_id == IPS_REVID_TROMBONE64)
6734 udelay(25); /* 25 us */
6735
6736 writeb(0xFF, ha->mem_ptr + IPS_REG_FLDP);
6737 if (ha->revision_id == IPS_REVID_TROMBONE64)
6738 udelay(25); /* 25 us */
6739
6740 return (1);
6741 }
6742 } /* end for */
6743
6744 /* Enable reading */
6745 writel(0, ha->mem_ptr + IPS_REG_FLAP);
6746 if (ha->revision_id == IPS_REVID_TROMBONE64)
6747 udelay(25); /* 25 us */
6748
6749 writeb(0xFF, ha->mem_ptr + IPS_REG_FLDP);
6750 if (ha->revision_id == IPS_REVID_TROMBONE64)
6751 udelay(25); /* 25 us */
6752
6753 return (0);
6754}
6755
6756/****************************************************************************/
6757/* */
6758/* Routine Name: ips_verify_bios */
6759/* */
6760/* Routine Description: */
6761/* Verify the BIOS on the adapter */
6762/* */
6763/****************************************************************************/
6764static int
6765ips_verify_bios(ips_ha_t * ha, char *buffer, uint32_t buffersize,
6766 uint32_t offset)
6767{
6768 uint8_t checksum;
6769 int i;
6770
6771 METHOD_TRACE("ips_verify_bios", 1);
6772
6773 /* test 1st byte */
6774 outl(0, ha->io_addr + IPS_REG_FLAP);
6775 if (ha->revision_id == IPS_REVID_TROMBONE64)
6776 udelay(25); /* 25 us */
6777
6778 if (inb(ha->io_addr + IPS_REG_FLDP) != 0x55)
6779 return (1);
6780
6781 outl(cpu_to_le32(1), ha->io_addr + IPS_REG_FLAP);
6782 if (ha->revision_id == IPS_REVID_TROMBONE64)
6783 udelay(25); /* 25 us */
6784 if (inb(ha->io_addr + IPS_REG_FLDP) != 0xAA)
6785 return (1);
6786
6787 checksum = 0xff;
6788 for (i = 2; i < buffersize; i++) {
6789
6790 outl(cpu_to_le32(i + offset), ha->io_addr + IPS_REG_FLAP);
6791 if (ha->revision_id == IPS_REVID_TROMBONE64)
6792 udelay(25); /* 25 us */
6793
6794 checksum = (uint8_t) checksum + inb(ha->io_addr + IPS_REG_FLDP);
6795 }
6796
6797 if (checksum != 0)
6798 /* failure */
6799 return (1);
6800 else
6801 /* success */
6802 return (0);
6803}
6804
6805/****************************************************************************/
6806/* */
6807/* Routine Name: ips_verify_bios_memio */
6808/* */
6809/* Routine Description: */
6810/* Verify the BIOS on the adapter */
6811/* */
6812/****************************************************************************/
6813static int
6814ips_verify_bios_memio(ips_ha_t * ha, char *buffer, uint32_t buffersize,
6815 uint32_t offset)
6816{
6817 uint8_t checksum;
6818 int i;
6819
6820 METHOD_TRACE("ips_verify_bios_memio", 1);
6821
6822 /* test 1st byte */
6823 writel(0, ha->mem_ptr + IPS_REG_FLAP);
6824 if (ha->revision_id == IPS_REVID_TROMBONE64)
6825 udelay(25); /* 25 us */
6826
6827 if (readb(ha->mem_ptr + IPS_REG_FLDP) != 0x55)
6828 return (1);
6829
6830 writel(1, ha->mem_ptr + IPS_REG_FLAP);
6831 if (ha->revision_id == IPS_REVID_TROMBONE64)
6832 udelay(25); /* 25 us */
6833 if (readb(ha->mem_ptr + IPS_REG_FLDP) != 0xAA)
6834 return (1);
6835
6836 checksum = 0xff;
6837 for (i = 2; i < buffersize; i++) {
6838
6839 writel(i + offset, ha->mem_ptr + IPS_REG_FLAP);
6840 if (ha->revision_id == IPS_REVID_TROMBONE64)
6841 udelay(25); /* 25 us */
6842
6843 checksum =
6844 (uint8_t) checksum + readb(ha->mem_ptr + IPS_REG_FLDP);
6845 }
6846
6847 if (checksum != 0)
6848 /* failure */
6849 return (1);
6850 else
6851 /* success */
6852 return (0);
6853}
6854
6855/*---------------------------------------------------------------------------*/
6856/* Routine Name: ips_version_check */
6857/* */
6858/* Dependencies: */
6859/* Assumes that ips_read_adapter_status() is called first filling in */
6860/* the data for SubSystem Parameters. */
6861/* Called from ips_write_driver_status() so it also assumes NVRAM Page 5 */
6862/* Data is available. */
6863/* */
6864/*---------------------------------------------------------------------------*/
6865static void
6866ips_version_check(ips_ha_t * ha, int intr)
6867{
6868 IPS_VERSION_DATA *VersionInfo;
6869 uint8_t FirmwareVersion[IPS_COMPAT_ID_LENGTH + 1];
6870 uint8_t BiosVersion[IPS_COMPAT_ID_LENGTH + 1];
6871 int MatchError;
6872 int rc;
6873 char BiosString[10];
6874 char FirmwareString[10];
6875
6876 METHOD_TRACE("ips_version_check", 1);
6877
6878 VersionInfo = ( IPS_VERSION_DATA * ) ha->ioctl_data;
6879
6880 memset(FirmwareVersion, 0, IPS_COMPAT_ID_LENGTH + 1);
6881 memset(BiosVersion, 0, IPS_COMPAT_ID_LENGTH + 1);
6882
6883 /* Get the Compatible BIOS Version from NVRAM Page 5 */
6884 memcpy(BiosVersion, ha->nvram->BiosCompatibilityID,
6885 IPS_COMPAT_ID_LENGTH);
6886
6887 rc = IPS_FAILURE;
6888 if (ha->subsys->param[4] & IPS_GET_VERSION_SUPPORT) { /* If Versioning is Supported */
6889 /* Get the Version Info with a Get Version Command */
6890 memset( VersionInfo, 0, sizeof (IPS_VERSION_DATA));
6891 rc = ips_get_version_info(ha, ha->ioctl_busaddr, intr);
6892 if (rc == IPS_SUCCESS)
6893 memcpy(FirmwareVersion, VersionInfo->compatibilityId,
6894 IPS_COMPAT_ID_LENGTH);
6895 }
6896
6897 if (rc != IPS_SUCCESS) { /* If Data Not Obtainable from a GetVersion Command */
6898 /* Get the Firmware Version from Enquiry Data */
6899 memcpy(FirmwareVersion, ha->enq->CodeBlkVersion,
6900 IPS_COMPAT_ID_LENGTH);
6901 }
6902
6903 /* printk(KERN_WARNING "Adapter's BIOS Version = %s\n", BiosVersion); */
6904 /* printk(KERN_WARNING "BIOS Compatible Version = %s\n", IPS_COMPAT_BIOS); */
6905 /* printk(KERN_WARNING "Adapter's Firmware Version = %s\n", FirmwareVersion); */
6906 /* printk(KERN_WARNING "Firmware Compatible Version = %s \n", Compatable[ ha->nvram->adapter_type ]); */
6907
6908 MatchError = 0;
6909
6910 if (strncmp
6911 (FirmwareVersion, Compatable[ha->nvram->adapter_type],
6912 IPS_COMPAT_ID_LENGTH) != 0)
6913 MatchError = 1;
6914
6915 if (strncmp(BiosVersion, IPS_COMPAT_BIOS, IPS_COMPAT_ID_LENGTH) != 0)
6916 MatchError = 1;
6917
6918 ha->nvram->versioning = 1; /* Indicate the Driver Supports Versioning */
6919
6920 if (MatchError) {
6921 ha->nvram->version_mismatch = 1;
6922 if (ips_cd_boot == 0) {
6923 strncpy(&BiosString[0], ha->nvram->bios_high, 4);
6924 strncpy(&BiosString[4], ha->nvram->bios_low, 4);
6925 BiosString[8] = 0;
6926
6927 strncpy(&FirmwareString[0], ha->enq->CodeBlkVersion, 8);
6928 FirmwareString[8] = 0;
6929
6930 IPS_PRINTK(KERN_WARNING, ha->pcidev,
6931 "Warning ! ! ! ServeRAID Version Mismatch\n");
6932 IPS_PRINTK(KERN_WARNING, ha->pcidev,
6933 "Bios = %s, Firmware = %s, Device Driver = %s%s\n",
6934 BiosString, FirmwareString, IPS_VERSION_HIGH,
6935 IPS_VERSION_LOW);
6936 IPS_PRINTK(KERN_WARNING, ha->pcidev,
6937 "These levels should match to avoid possible compatibility problems.\n");
6938 }
6939 } else {
6940 ha->nvram->version_mismatch = 0;
6941 }
6942
6943 return;
6944}
6945
6946/*---------------------------------------------------------------------------*/
6947/* Routine Name: ips_get_version_info */
6948/* */
6949/* Routine Description: */
6950/* Issue an internal GETVERSION Command */
6951/* */
6952/* Return Value: */
6953/* 0 if Successful, else non-zero */
6954/*---------------------------------------------------------------------------*/
6955static int
6956ips_get_version_info(ips_ha_t * ha, dma_addr_t Buffer, int intr)
6957{
6958 ips_scb_t *scb;
6959 int rc;
6960
6961 METHOD_TRACE("ips_get_version_info", 1);
6962
6963 scb = &ha->scbs[ha->max_cmds - 1];
6964
6965 ips_init_scb(ha, scb);
6966
6967 scb->timeout = ips_cmd_timeout;
6968 scb->cdb[0] = IPS_CMD_GET_VERSION_INFO;
6969 scb->cmd.version_info.op_code = IPS_CMD_GET_VERSION_INFO;
6970 scb->cmd.version_info.command_id = IPS_COMMAND_ID(ha, scb);
6971 scb->cmd.version_info.reserved = 0;
6972 scb->cmd.version_info.count = sizeof (IPS_VERSION_DATA);
6973 scb->cmd.version_info.reserved2 = 0;
6974 scb->data_len = sizeof (IPS_VERSION_DATA);
6975 scb->data_busaddr = Buffer;
6976 scb->cmd.version_info.buffer_addr = Buffer;
6977 scb->flags = 0;
6978
6979 /* issue command */
6980 rc = ips_send_wait(ha, scb, ips_cmd_timeout, intr);
6981 return (rc);
6982}
6983
6984/****************************************************************************/
6985/* */
6986/* Routine Name: ips_abort_init */
6987/* */
6988/* Routine Description: */
6989/* cleanup routine for a failed adapter initialization */
6990/****************************************************************************/
6991static int
6992ips_abort_init(ips_ha_t * ha, int index)
6993{
6994 ha->active = 0;
6995 ips_free(ha);
6996 ips_ha[index] = NULL;
6997 ips_sh[index] = NULL;
6998 return -1;
6999}
7000
7001/****************************************************************************/
7002/* */
7003/* Routine Name: ips_shift_controllers */
7004/* */
7005/* Routine Description: */
7006/* helper function for ordering adapters */
7007/****************************************************************************/
7008static void
7009ips_shift_controllers(int lowindex, int highindex)
7010{
7011 ips_ha_t *ha_sav = ips_ha[highindex];
7012 struct Scsi_Host *sh_sav = ips_sh[highindex];
7013 int i;
7014
7015 for (i = highindex; i > lowindex; i--) {
7016 ips_ha[i] = ips_ha[i - 1];
7017 ips_sh[i] = ips_sh[i - 1];
7018 ips_ha[i]->host_num = i;
7019 }
7020 ha_sav->host_num = lowindex;
7021 ips_ha[lowindex] = ha_sav;
7022 ips_sh[lowindex] = sh_sav;
7023}
7024
7025/****************************************************************************/
7026/* */
7027/* Routine Name: ips_order_controllers */
7028/* */
7029/* Routine Description: */
7030/* place controllers is the "proper" boot order */
7031/****************************************************************************/
7032static void
7033ips_order_controllers(void)
7034{
7035 int i, j, tmp, position = 0;
7036 IPS_NVRAM_P5 *nvram;
7037 if (!ips_ha[0])
7038 return;
7039 nvram = ips_ha[0]->nvram;
7040
7041 if (nvram->adapter_order[0]) {
7042 for (i = 1; i <= nvram->adapter_order[0]; i++) {
7043 for (j = position; j < ips_num_controllers; j++) {
7044 switch (ips_ha[j]->ad_type) {
7045 case IPS_ADTYPE_SERVERAID6M:
7046 case IPS_ADTYPE_SERVERAID7M:
7047 if (nvram->adapter_order[i] == 'M') {
7048 ips_shift_controllers(position,
7049 j);
7050 position++;
7051 }
7052 break;
7053 case IPS_ADTYPE_SERVERAID4L:
7054 case IPS_ADTYPE_SERVERAID4M:
7055 case IPS_ADTYPE_SERVERAID4MX:
7056 case IPS_ADTYPE_SERVERAID4LX:
7057 if (nvram->adapter_order[i] == 'N') {
7058 ips_shift_controllers(position,
7059 j);
7060 position++;
7061 }
7062 break;
7063 case IPS_ADTYPE_SERVERAID6I:
7064 case IPS_ADTYPE_SERVERAID5I2:
7065 case IPS_ADTYPE_SERVERAID5I1:
7066 case IPS_ADTYPE_SERVERAID7k:
7067 if (nvram->adapter_order[i] == 'S') {
7068 ips_shift_controllers(position,
7069 j);
7070 position++;
7071 }
7072 break;
7073 case IPS_ADTYPE_SERVERAID:
7074 case IPS_ADTYPE_SERVERAID2:
7075 case IPS_ADTYPE_NAVAJO:
7076 case IPS_ADTYPE_KIOWA:
7077 case IPS_ADTYPE_SERVERAID3L:
7078 case IPS_ADTYPE_SERVERAID3:
7079 case IPS_ADTYPE_SERVERAID4H:
7080 if (nvram->adapter_order[i] == 'A') {
7081 ips_shift_controllers(position,
7082 j);
7083 position++;
7084 }
7085 break;
7086 default:
7087 break;
7088 }
7089 }
7090 }
7091 /* if adapter_order[0], then ordering is complete */
7092 return;
7093 }
7094 /* old bios, use older ordering */
7095 tmp = 0;
7096 for (i = position; i < ips_num_controllers; i++) {
7097 if (ips_ha[i]->ad_type == IPS_ADTYPE_SERVERAID5I2 ||
7098 ips_ha[i]->ad_type == IPS_ADTYPE_SERVERAID5I1) {
7099 ips_shift_controllers(position, i);
7100 position++;
7101 tmp = 1;
7102 }
7103 }
7104 /* if there were no 5I cards, then don't do any extra ordering */
7105 if (!tmp)
7106 return;
7107 for (i = position; i < ips_num_controllers; i++) {
7108 if (ips_ha[i]->ad_type == IPS_ADTYPE_SERVERAID4L ||
7109 ips_ha[i]->ad_type == IPS_ADTYPE_SERVERAID4M ||
7110 ips_ha[i]->ad_type == IPS_ADTYPE_SERVERAID4LX ||
7111 ips_ha[i]->ad_type == IPS_ADTYPE_SERVERAID4MX) {
7112 ips_shift_controllers(position, i);
7113 position++;
7114 }
7115 }
7116
7117 return;
7118}
7119
7120/****************************************************************************/
7121/* */
7122/* Routine Name: ips_register_scsi */
7123/* */
7124/* Routine Description: */
7125/* perform any registration and setup with the scsi layer */
7126/****************************************************************************/
7127static int
7128ips_register_scsi(int index)
7129{
7130 struct Scsi_Host *sh;
7131 ips_ha_t *ha, *oldha = ips_ha[index];
7132 sh = scsi_host_alloc(&ips_driver_template, sizeof (ips_ha_t));
7133 if (!sh) {
7134 IPS_PRINTK(KERN_WARNING, oldha->pcidev,
7135 "Unable to register controller with SCSI subsystem\n");
7136 return -1;
7137 }
7138 ha = IPS_HA(sh);
7139 memcpy(ha, oldha, sizeof (ips_ha_t));
7140 free_irq(oldha->irq, oldha);
7141 /* Install the interrupt handler with the new ha */
7142 if (request_irq(ha->irq, do_ipsintr, SA_SHIRQ, ips_name, ha)) {
7143 IPS_PRINTK(KERN_WARNING, ha->pcidev,
7144 "Unable to install interrupt handler\n");
7145 scsi_host_put(sh);
7146 return -1;
7147 }
7148
7149 kfree(oldha);
7150 ips_sh[index] = sh;
7151 ips_ha[index] = ha;
7152 IPS_SCSI_SET_DEVICE(sh, ha);
7153
7154 /* Store away needed values for later use */
7155 sh->io_port = ha->io_addr;
7156 sh->n_io_port = ha->io_addr ? 255 : 0;
7157 sh->unique_id = (ha->io_addr) ? ha->io_addr : ha->mem_addr;
7158 sh->irq = ha->irq;
7159 sh->sg_tablesize = sh->hostt->sg_tablesize;
7160 sh->can_queue = sh->hostt->can_queue;
7161 sh->cmd_per_lun = sh->hostt->cmd_per_lun;
7162 sh->unchecked_isa_dma = sh->hostt->unchecked_isa_dma;
7163 sh->use_clustering = sh->hostt->use_clustering;
7164
7165#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,4,7)
7166 sh->max_sectors = 128;
7167#endif
7168
7169 sh->max_id = ha->ntargets;
7170 sh->max_lun = ha->nlun;
7171 sh->max_channel = ha->nbus - 1;
7172 sh->can_queue = ha->max_cmds - 1;
7173
7174 IPS_ADD_HOST(sh, NULL);
7175 return 0;
7176}
7177
7178/*---------------------------------------------------------------------------*/
7179/* Routine Name: ips_remove_device */
7180/* */
7181/* Routine Description: */
7182/* Remove one Adapter ( Hot Plugging ) */
7183/*---------------------------------------------------------------------------*/
7184static void __devexit
7185ips_remove_device(struct pci_dev *pci_dev)
7186{
7187 int i;
7188 struct Scsi_Host *sh;
7189 ips_ha_t *ha;
7190
7191 for (i = 0; i < IPS_MAX_ADAPTERS; i++) {
7192 ha = ips_ha[i];
7193 if (ha) {
7194 if ((pci_dev->bus->number == ha->pcidev->bus->number) &&
7195 (pci_dev->devfn == ha->pcidev->devfn)) {
7196 sh = ips_sh[i];
7197 ips_release(sh);
7198 }
7199 }
7200 }
7201}
7202
7203/****************************************************************************/
7204/* */
7205/* Routine Name: ips_module_init */
7206/* */
7207/* Routine Description: */
7208/* function called on module load */
7209/****************************************************************************/
7210static int __init
7211ips_module_init(void)
7212{
7213 if (pci_module_init(&ips_pci_driver) < 0)
7214 return -ENODEV;
7215 ips_driver_template.module = THIS_MODULE;
7216 ips_order_controllers();
7217 if (IPS_REGISTER_HOSTS(&ips_driver_template)) {
7218 pci_unregister_driver(&ips_pci_driver);
7219 return -ENODEV;
7220 }
7221 register_reboot_notifier(&ips_notifier);
7222 return 0;
7223}
7224
7225/****************************************************************************/
7226/* */
7227/* Routine Name: ips_module_exit */
7228/* */
7229/* Routine Description: */
7230/* function called on module unload */
7231/****************************************************************************/
7232static void __exit
7233ips_module_exit(void)
7234{
7235 IPS_UNREGISTER_HOSTS(&ips_driver_template);
7236 pci_unregister_driver(&ips_pci_driver);
7237 unregister_reboot_notifier(&ips_notifier);
7238}
7239
7240module_init(ips_module_init);
7241module_exit(ips_module_exit);
7242
7243/*---------------------------------------------------------------------------*/
7244/* Routine Name: ips_insert_device */
7245/* */
7246/* Routine Description: */
7247/* Add One Adapter ( Hot Plug ) */
7248/* */
7249/* Return Value: */
7250/* 0 if Successful, else non-zero */
7251/*---------------------------------------------------------------------------*/
7252static int __devinit
7253ips_insert_device(struct pci_dev *pci_dev, const struct pci_device_id *ent)
7254{
7255 int index;
7256 int rc;
7257
7258 METHOD_TRACE("ips_insert_device", 1);
7259 if (pci_enable_device(pci_dev))
7260 return -1;
7261
7262 rc = ips_init_phase1(pci_dev, &index);
7263 if (rc == SUCCESS)
7264 rc = ips_init_phase2(index);
7265
7266 if (ips_hotplug)
7267 if (ips_register_scsi(index)) {
7268 ips_free(ips_ha[index]);
7269 rc = -1;
7270 }
7271
7272 if (rc == SUCCESS)
7273 ips_num_controllers++;
7274
7275 ips_next_controller = ips_num_controllers;
7276 return rc;
7277}
7278
7279/*---------------------------------------------------------------------------*/
7280/* Routine Name: ips_init_phase1 */
7281/* */
7282/* Routine Description: */
7283/* Adapter Initialization */
7284/* */
7285/* Return Value: */
7286/* 0 if Successful, else non-zero */
7287/*---------------------------------------------------------------------------*/
7288static int
7289ips_init_phase1(struct pci_dev *pci_dev, int *indexPtr)
7290{
7291 ips_ha_t *ha;
7292 uint32_t io_addr;
7293 uint32_t mem_addr;
7294 uint32_t io_len;
7295 uint32_t mem_len;
7296 uint8_t revision_id;
7297 uint8_t bus;
7298 uint8_t func;
7299 uint8_t irq;
7300 uint16_t subdevice_id;
7301 int j;
7302 int index;
7303 dma_addr_t dma_address;
7304 char __iomem *ioremap_ptr;
7305 char __iomem *mem_ptr;
7306 uint32_t IsDead;
7307
7308 METHOD_TRACE("ips_init_phase1", 1);
7309 index = IPS_MAX_ADAPTERS;
7310 for (j = 0; j < IPS_MAX_ADAPTERS; j++) {
7311 if (ips_ha[j] == 0) {
7312 index = j;
7313 break;
7314 }
7315 }
7316
7317 if (index >= IPS_MAX_ADAPTERS)
7318 return -1;
7319
7320 /* stuff that we get in dev */
7321 irq = pci_dev->irq;
7322 bus = pci_dev->bus->number;
7323 func = pci_dev->devfn;
7324
7325 /* Init MEM/IO addresses to 0 */
7326 mem_addr = 0;
7327 io_addr = 0;
7328 mem_len = 0;
7329 io_len = 0;
7330
7331 for (j = 0; j < 2; j++) {
7332 if (!pci_resource_start(pci_dev, j))
7333 break;
7334
7335 if (pci_resource_flags(pci_dev, j) & IORESOURCE_IO) {
7336 io_addr = pci_resource_start(pci_dev, j);
7337 io_len = pci_resource_len(pci_dev, j);
7338 } else {
7339 mem_addr = pci_resource_start(pci_dev, j);
7340 mem_len = pci_resource_len(pci_dev, j);
7341 }
7342 }
7343
7344 /* setup memory mapped area (if applicable) */
7345 if (mem_addr) {
7346 uint32_t base;
7347 uint32_t offs;
7348
7349 if (!request_mem_region(mem_addr, mem_len, "ips")) {
7350 IPS_PRINTK(KERN_WARNING, pci_dev,
7351 "Couldn't allocate IO Memory space %x len %d.\n",
7352 mem_addr, mem_len);
7353 return -1;
7354 }
7355
7356 base = mem_addr & PAGE_MASK;
7357 offs = mem_addr - base;
7358 ioremap_ptr = ioremap(base, PAGE_SIZE);
7359 mem_ptr = ioremap_ptr + offs;
7360 } else {
7361 ioremap_ptr = NULL;
7362 mem_ptr = NULL;
7363 }
7364
7365 /* setup I/O mapped area (if applicable) */
7366 if (io_addr) {
7367 if (!request_region(io_addr, io_len, "ips")) {
7368 IPS_PRINTK(KERN_WARNING, pci_dev,
7369 "Couldn't allocate IO space %x len %d.\n",
7370 io_addr, io_len);
7371 return -1;
7372 }
7373 }
7374
7375 /* get the revision ID */
7376 if (pci_read_config_byte(pci_dev, PCI_REVISION_ID, &revision_id)) {
7377 IPS_PRINTK(KERN_WARNING, pci_dev, "Can't get revision id.\n");
7378 return -1;
7379 }
7380
7381 subdevice_id = pci_dev->subsystem_device;
7382
7383 /* found a controller */
7384 ha = kmalloc(sizeof (ips_ha_t), GFP_KERNEL);
7385 if (ha == NULL) {
7386 IPS_PRINTK(KERN_WARNING, pci_dev,
7387 "Unable to allocate temporary ha struct\n");
7388 return -1;
7389 }
7390
7391 memset(ha, 0, sizeof (ips_ha_t));
7392
7393 ips_sh[index] = NULL;
7394 ips_ha[index] = ha;
7395 ha->active = 1;
7396
7397 /* Store info in HA structure */
7398 ha->irq = irq;
7399 ha->io_addr = io_addr;
7400 ha->io_len = io_len;
7401 ha->mem_addr = mem_addr;
7402 ha->mem_len = mem_len;
7403 ha->mem_ptr = mem_ptr;
7404 ha->ioremap_ptr = ioremap_ptr;
7405 ha->host_num = (uint32_t) index;
7406 ha->revision_id = revision_id;
7407 ha->slot_num = PCI_SLOT(pci_dev->devfn);
7408 ha->device_id = pci_dev->device;
7409 ha->subdevice_id = subdevice_id;
7410 ha->pcidev = pci_dev;
7411
7412 /*
7413 * Set the pci_dev's dma_mask. Not all adapters support 64bit
7414 * addressing so don't enable it if the adapter can't support
7415 * it! Also, don't use 64bit addressing if dma addresses
7416 * are guaranteed to be < 4G.
7417 */
7418 if (IPS_ENABLE_DMA64 && IPS_HAS_ENH_SGLIST(ha) &&
7419 !pci_set_dma_mask(ha->pcidev, 0xffffffffffffffffULL)) {
7420 (ha)->flags |= IPS_HA_ENH_SG;
7421 } else {
7422 if (pci_set_dma_mask(ha->pcidev, 0xffffffffULL) != 0) {
7423 printk(KERN_WARNING "Unable to set DMA Mask\n");
7424 return ips_abort_init(ha, index);
7425 }
7426 }
7427 if(ips_cd_boot && !ips_FlashData){
7428 ips_FlashData = pci_alloc_consistent(pci_dev, PAGE_SIZE << 7,
7429 &ips_flashbusaddr);
7430 }
7431
7432 ha->enq = pci_alloc_consistent(pci_dev, sizeof (IPS_ENQ),
7433 &ha->enq_busaddr);
7434 if (!ha->enq) {
7435 IPS_PRINTK(KERN_WARNING, pci_dev,
7436 "Unable to allocate host inquiry structure\n");
7437 return ips_abort_init(ha, index);
7438 }
7439
7440 ha->adapt = pci_alloc_consistent(pci_dev, sizeof (IPS_ADAPTER) +
7441 sizeof (IPS_IO_CMD), &dma_address);
7442 if (!ha->adapt) {
7443 IPS_PRINTK(KERN_WARNING, pci_dev,
7444 "Unable to allocate host adapt & dummy structures\n");
7445 return ips_abort_init(ha, index);
7446 }
7447 ha->adapt->hw_status_start = dma_address;
7448 ha->dummy = (void *) (ha->adapt + 1);
7449
7450
7451
7452 ha->logical_drive_info = pci_alloc_consistent(pci_dev, sizeof (IPS_LD_INFO), &dma_address);
7453 if (!ha->logical_drive_info) {
7454 IPS_PRINTK(KERN_WARNING, pci_dev,
7455 "Unable to allocate logical drive info structure\n");
7456 return ips_abort_init(ha, index);
7457 }
7458 ha->logical_drive_info_dma_addr = dma_address;
7459
7460
7461 ha->conf = kmalloc(sizeof (IPS_CONF), GFP_KERNEL);
7462
7463 if (!ha->conf) {
7464 IPS_PRINTK(KERN_WARNING, pci_dev,
7465 "Unable to allocate host conf structure\n");
7466 return ips_abort_init(ha, index);
7467 }
7468
7469 ha->nvram = kmalloc(sizeof (IPS_NVRAM_P5), GFP_KERNEL);
7470
7471 if (!ha->nvram) {
7472 IPS_PRINTK(KERN_WARNING, pci_dev,
7473 "Unable to allocate host NVRAM structure\n");
7474 return ips_abort_init(ha, index);
7475 }
7476
7477 ha->subsys = kmalloc(sizeof (IPS_SUBSYS), GFP_KERNEL);
7478
7479 if (!ha->subsys) {
7480 IPS_PRINTK(KERN_WARNING, pci_dev,
7481 "Unable to allocate host subsystem structure\n");
7482 return ips_abort_init(ha, index);
7483 }
7484
7485 /* the ioctl buffer is now used during adapter initialization, so its
7486 * successful allocation is now required */
7487 if (ips_ioctlsize < PAGE_SIZE)
7488 ips_ioctlsize = PAGE_SIZE;
7489
7490 ha->ioctl_data = pci_alloc_consistent(pci_dev, ips_ioctlsize,
7491 &ha->ioctl_busaddr);
7492 ha->ioctl_len = ips_ioctlsize;
7493 if (!ha->ioctl_data) {
7494 IPS_PRINTK(KERN_WARNING, pci_dev,
7495 "Unable to allocate IOCTL data\n");
7496 return ips_abort_init(ha, index);
7497 }
7498
7499 /*
7500 * Setup Functions
7501 */
7502 ips_setup_funclist(ha);
7503
7504 if ((IPS_IS_MORPHEUS(ha)) || (IPS_IS_MARCO(ha))) {
7505 /* If Morpheus appears dead, reset it */
7506 IsDead = readl(ha->mem_ptr + IPS_REG_I960_MSG1);
7507 if (IsDead == 0xDEADBEEF) {
7508 ips_reset_morpheus(ha);
7509 }
7510 }
7511
7512 /*
7513 * Initialize the card if it isn't already
7514 */
7515
7516 if (!(*ha->func.isinit) (ha)) {
7517 if (!(*ha->func.init) (ha)) {
7518 /*
7519 * Initialization failed
7520 */
7521 IPS_PRINTK(KERN_WARNING, pci_dev,
7522 "Unable to initialize controller\n");
7523 return ips_abort_init(ha, index);
7524 }
7525 }
7526
7527 *indexPtr = index;
7528 return SUCCESS;
7529}
7530
7531/*---------------------------------------------------------------------------*/
7532/* Routine Name: ips_init_phase2 */
7533/* */
7534/* Routine Description: */
7535/* Adapter Initialization Phase 2 */
7536/* */
7537/* Return Value: */
7538/* 0 if Successful, else non-zero */
7539/*---------------------------------------------------------------------------*/
7540static int
7541ips_init_phase2(int index)
7542{
7543 ips_ha_t *ha;
7544
7545 ha = ips_ha[index];
7546
7547 METHOD_TRACE("ips_init_phase2", 1);
7548 if (!ha->active) {
7549 ips_ha[index] = NULL;
7550 return -1;
7551 }
7552
7553 /* Install the interrupt handler */
7554 if (request_irq(ha->irq, do_ipsintr, SA_SHIRQ, ips_name, ha)) {
7555 IPS_PRINTK(KERN_WARNING, ha->pcidev,
7556 "Unable to install interrupt handler\n");
7557 return ips_abort_init(ha, index);
7558 }
7559
7560 /*
7561 * Allocate a temporary SCB for initialization
7562 */
7563 ha->max_cmds = 1;
7564 if (!ips_allocatescbs(ha)) {
7565 IPS_PRINTK(KERN_WARNING, ha->pcidev,
7566 "Unable to allocate a CCB\n");
7567 free_irq(ha->irq, ha);
7568 return ips_abort_init(ha, index);
7569 }
7570
7571 if (!ips_hainit(ha)) {
7572 IPS_PRINTK(KERN_WARNING, ha->pcidev,
7573 "Unable to initialize controller\n");
7574 free_irq(ha->irq, ha);
7575 return ips_abort_init(ha, index);
7576 }
7577 /* Free the temporary SCB */
7578 ips_deallocatescbs(ha, 1);
7579
7580 /* allocate CCBs */
7581 if (!ips_allocatescbs(ha)) {
7582 IPS_PRINTK(KERN_WARNING, ha->pcidev,
7583 "Unable to allocate CCBs\n");
7584 free_irq(ha->irq, ha);
7585 return ips_abort_init(ha, index);
7586 }
7587
7588 return SUCCESS;
7589}
7590
7591#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,4,9)
7592MODULE_LICENSE("GPL");
7593#endif
7594
7595MODULE_DESCRIPTION("IBM ServeRAID Adapter Driver " IPS_VER_STRING);
7596
7597#ifdef MODULE_VERSION
7598MODULE_VERSION(IPS_VER_STRING);
7599#endif
7600
7601
7602/*
7603 * Overrides for Emacs so that we almost follow Linus's tabbing style.
7604 * Emacs will notice this stuff at the end of the file and automatically
7605 * adjust the settings for this buffer only. This must remain at the end
7606 * of the file.
7607 * ---------------------------------------------------------------------------
7608 * Local variables:
7609 * c-indent-level: 2
7610 * c-brace-imaginary-offset: 0
7611 * c-brace-offset: -2
7612 * c-argdecl-indent: 2
7613 * c-label-offset: -2
7614 * c-continued-statement-offset: 2
7615 * c-continued-brace-offset: 0
7616 * indent-tabs-mode: nil
7617 * tab-width: 8
7618 * End:
7619 */