aic7xxx_osm.c 140 KB
Newer Older
Linus Torvalds's avatar
Linus Torvalds committed
1 2 3
/*
 * Adaptec AIC7xxx device driver for Linux.
 *
4
 * $Id: //depot/aic7xxx/linux/drivers/scsi/aic7xxx/aic7xxx_osm.c#235 $
Linus Torvalds's avatar
Linus Torvalds committed
5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40
 *
 * Copyright (c) 1994 John Aycock
 *   The University of Calgary Department of Computer Science.
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2, or (at your option)
 * any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; see the file COPYING.  If not, write to
 * the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
 *
 * Sources include the Adaptec 1740 driver (aha1740.c), the Ultrastor 24F
 * driver (ultrastor.c), various Linux kernel source, the Adaptec EISA
 * config file (!adp7771.cfg), the Adaptec AHA-2740A Series User's Guide,
 * the Linux Kernel Hacker's Guide, Writing a SCSI Device Driver for Linux,
 * the Adaptec 1542 driver (aha1542.c), the Adaptec EISA overlay file
 * (adp7770.ovl), the Adaptec AHA-2740 Series Technical Reference Manual,
 * the Adaptec AIC-7770 Data Book, the ANSI SCSI specification, the
 * ANSI SCSI-2 specification (draft 10c), ...
 *
 * --------------------------------------------------------------------------
 *
 *  Modifications by Daniel M. Eischen (deischen@iworks.InterWorks.org):
 *
 *  Substantially modified to include support for wide and twin bus
 *  adapters, DMAing of SCBs, tagged queueing, IRQ sharing, bug fixes,
 *  SCB paging, and other rework of the code.
 *
 * --------------------------------------------------------------------------
Linus Torvalds's avatar
Linus Torvalds committed
41 42
 * Copyright (c) 1994-2000 Justin T. Gibbs.
 * Copyright (c) 2000-2001 Adaptec Inc.
Linus Torvalds's avatar
Linus Torvalds committed
43 44 45 46 47 48 49 50
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions, and the following disclaimer,
 *    without modification.
Linus Torvalds's avatar
Linus Torvalds committed
51 52 53 54 55 56 57 58
 * 2. Redistributions in binary form must reproduce at minimum a disclaimer
 *    substantially similar to the "NO WARRANTY" disclaimer below
 *    ("Disclaimer") and any redistribution must be conditioned upon
 *    including a substantially similar Disclaimer requirement for further
 *    binary redistribution.
 * 3. Neither the names of the above-listed copyright holders nor the names
 *    of any contributors may be used to endorse or promote products derived
 *    from this software without specific prior written permission.
Linus Torvalds's avatar
Linus Torvalds committed
59 60
 *
 * Alternatively, this software may be distributed under the terms of the
Linus Torvalds's avatar
Linus Torvalds committed
61 62
 * GNU General Public License ("GPL") version 2 as published by the Free
 * Software Foundation.
Linus Torvalds's avatar
Linus Torvalds committed
63
 *
Linus Torvalds's avatar
Linus Torvalds committed
64 65 66 67 68 69
 * NO WARRANTY
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
 * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
Linus Torvalds's avatar
Linus Torvalds committed
70 71
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
Linus Torvalds's avatar
Linus Torvalds committed
72 73 74 75
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
 * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGES.
Linus Torvalds's avatar
Linus Torvalds committed
76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110
 *
 *---------------------------------------------------------------------------
 *
 *  Thanks also go to (in alphabetical order) the following:
 *
 *    Rory Bolt     - Sequencer bug fixes
 *    Jay Estabrook - Initial DEC Alpha support
 *    Doug Ledford  - Much needed abort/reset bug fixes
 *    Kai Makisara  - DMAing of SCBs
 *
 *  A Boot time option was also added for not resetting the scsi bus.
 *
 *    Form:  aic7xxx=extended
 *           aic7xxx=no_reset
 *           aic7xxx=verbose
 *
 *  Daniel M. Eischen, deischen@iworks.InterWorks.org, 1/23/97
 *
 *  Id: aic7xxx.c,v 4.1 1997/06/12 08:23:42 deang Exp
 */

/*
 * Further driver modifications made by Doug Ledford <dledford@redhat.com>
 *
 * Copyright (c) 1997-1999 Doug Ledford
 *
 * These changes are released under the same licensing terms as the FreeBSD
 * driver written by Justin Gibbs.  Please see his Copyright notice above
 * for the exact terms and conditions covering my changes as well as the
 * warranty statement.
 *
 * Modifications made to the aic7xxx.c,v 4.1 driver from Dan Eischen include
 * but are not limited to:
 *
 *  1: Import of the latest FreeBSD sequencer code for this driver
111
 *  2: Modification of kernel code to accommodate different sequencer semantics
Linus Torvalds's avatar
Linus Torvalds committed
112 113 114 115 116 117 118 119 120 121 122 123
 *  3: Extensive changes throughout kernel portion of driver to improve
 *     abort/reset processing and error hanndling
 *  4: Other work contributed by various people on the Internet
 *  5: Changes to printk information and verbosity selection code
 *  6: General reliability related changes, especially in IRQ management
 *  7: Modifications to the default probe/attach order for supported cards
 *  8: SMP friendliness has been improved
 *
 */

#include "aic7xxx_osm.h"
#include "aic7xxx_inline.h"
124
#include <scsi/scsicam.h>
Linus Torvalds's avatar
Linus Torvalds committed
125

126 127 128 129 130 131
/*
 * Include aiclib.c as part of our
 * "module dependencies are hard" work around.
 */
#include "aiclib.c"

Linus Torvalds's avatar
Linus Torvalds committed
132 133 134 135
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,3,0)
#include <linux/init.h>		/* __setup */
#endif

136 137 138 139

#if LINUX_VERSION_CODE < KERNEL_VERSION(2,5,0)
#include "sd.h"			/* For geometry detection */
#endif
Linus Torvalds's avatar
Linus Torvalds committed
140

Linus Torvalds's avatar
Linus Torvalds committed
141
#include <linux/mm.h>		/* For fetching system memory size */
142
#include <linux/blkdev.h>		/* For block_size() */
143
#include <linux/delay.h>	/* For ssleep/msleep */
144 145 146 147 148

/*
 * Lock protecting manipulation of the ahc softc list.
 */
spinlock_t ahc_list_spinlock;
Linus Torvalds's avatar
Linus Torvalds committed
149

150 151 152 153
#if LINUX_VERSION_CODE < KERNEL_VERSION(2,5,0)
/* For dynamic sglist size calculation. */
u_int ahc_linux_nseg;
#endif
Linus Torvalds's avatar
Linus Torvalds committed
154 155 156 157 158 159 160 161 162 163 164 165 166 167 168

#if LINUX_VERSION_CODE < KERNEL_VERSION(2,3,0)
struct proc_dir_entry proc_scsi_aic7xxx = {
	PROC_SCSI_AIC7XXX, 7, "aic7xxx",
	S_IFDIR | S_IRUGO | S_IXUGO, 2,
	0, 0, 0, NULL, NULL, NULL, NULL, NULL, NULL, NULL
};
#endif

/*
 * Set this to the delay in seconds after SCSI bus reset.
 * Note, we honor this only for the initial bus reset.
 * The scsi error recovery code performs its own bus settle
 * delay handling for error recovery actions.
 */
Linus Torvalds's avatar
Linus Torvalds committed
169 170
#ifdef CONFIG_AIC7XXX_RESET_DELAY_MS
#define AIC7XXX_RESET_DELAY CONFIG_AIC7XXX_RESET_DELAY_MS
Linus Torvalds's avatar
Linus Torvalds committed
171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283
#else
#define AIC7XXX_RESET_DELAY 5000
#endif

/*
 * Control collection of SCSI transfer statistics for the /proc filesystem.
 *
 * NOTE: Do NOT enable this when running on kernels version 1.2.x and below.
 * NOTE: This does affect performance since it has to maintain statistics.
 */
#ifdef CONFIG_AIC7XXX_PROC_STATS
#define AIC7XXX_PROC_STATS
#endif

/*
 * To change the default number of tagged transactions allowed per-device,
 * add a line to the lilo.conf file like:
 * append="aic7xxx=verbose,tag_info:{{32,32,32,32},{32,32,32,32}}"
 * which will result in the first four devices on the first two
 * controllers being set to a tagged queue depth of 32.
 *
 * The tag_commands is an array of 16 to allow for wide and twin adapters.
 * Twin adapters will use indexes 0-7 for channel 0, and indexes 8-15
 * for channel 1.
 */
typedef struct {
	uint8_t tag_commands[16];	/* Allow for wide/twin adapters. */
} adapter_tag_info_t;

/*
 * Modify this as you see fit for your system.
 *
 * 0			tagged queuing disabled
 * 1 <= n <= 253	n == max tags ever dispatched.
 *
 * The driver will throttle the number of commands dispatched to a
 * device if it returns queue full.  For devices with a fixed maximum
 * queue depth, the driver will eventually determine this depth and
 * lock it in (a console message is printed to indicate that a lock
 * has occurred).  On some devices, queue full is returned for a temporary
 * resource shortage.  These devices will return queue full at varying
 * depths.  The driver will throttle back when the queue fulls occur and
 * attempt to slowly increase the depth over time as the device recovers
 * from the resource shortage.
 *
 * In this example, the first line will disable tagged queueing for all
 * the devices on the first probed aic7xxx adapter.
 *
 * The second line enables tagged queueing with 4 commands/LUN for IDs
 * (0, 2-11, 13-15), disables tagged queueing for ID 12, and tells the
 * driver to attempt to use up to 64 tags for ID 1.
 *
 * The third line is the same as the first line.
 *
 * The fourth line disables tagged queueing for devices 0 and 3.  It
 * enables tagged queueing for the other IDs, with 16 commands/LUN
 * for IDs 1 and 4, 127 commands/LUN for ID 8, and 4 commands/LUN for
 * IDs 2, 5-7, and 9-15.
 */

/*
 * NOTE: The below structure is for reference only, the actual structure
 *       to modify in order to change things is just below this comment block.
adapter_tag_info_t aic7xxx_tag_info[] =
{
	{{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}},
	{{4, 64, 4, 4, 4, 4, 4, 4, 4, 4, 4, 4, 0, 4, 4, 4}},
	{{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}},
	{{0, 16, 4, 0, 16, 4, 4, 4, 127, 4, 4, 4, 4, 4, 4, 4}}
};
*/

#ifdef CONFIG_AIC7XXX_CMDS_PER_DEVICE
#define AIC7XXX_CMDS_PER_DEVICE CONFIG_AIC7XXX_CMDS_PER_DEVICE
#else
#define AIC7XXX_CMDS_PER_DEVICE AHC_MAX_QUEUE
#endif

#define AIC7XXX_CONFIGED_TAG_COMMANDS {					\
	AIC7XXX_CMDS_PER_DEVICE, AIC7XXX_CMDS_PER_DEVICE,		\
	AIC7XXX_CMDS_PER_DEVICE, AIC7XXX_CMDS_PER_DEVICE,		\
	AIC7XXX_CMDS_PER_DEVICE, AIC7XXX_CMDS_PER_DEVICE,		\
	AIC7XXX_CMDS_PER_DEVICE, AIC7XXX_CMDS_PER_DEVICE,		\
	AIC7XXX_CMDS_PER_DEVICE, AIC7XXX_CMDS_PER_DEVICE,		\
	AIC7XXX_CMDS_PER_DEVICE, AIC7XXX_CMDS_PER_DEVICE,		\
	AIC7XXX_CMDS_PER_DEVICE, AIC7XXX_CMDS_PER_DEVICE,		\
	AIC7XXX_CMDS_PER_DEVICE, AIC7XXX_CMDS_PER_DEVICE		\
}

/*
 * By default, use the number of commands specified by
 * the users kernel configuration.
 */
static adapter_tag_info_t aic7xxx_tag_info[] =
{
	{AIC7XXX_CONFIGED_TAG_COMMANDS},
	{AIC7XXX_CONFIGED_TAG_COMMANDS},
	{AIC7XXX_CONFIGED_TAG_COMMANDS},
	{AIC7XXX_CONFIGED_TAG_COMMANDS},
	{AIC7XXX_CONFIGED_TAG_COMMANDS},
	{AIC7XXX_CONFIGED_TAG_COMMANDS},
	{AIC7XXX_CONFIGED_TAG_COMMANDS},
	{AIC7XXX_CONFIGED_TAG_COMMANDS},
	{AIC7XXX_CONFIGED_TAG_COMMANDS},
	{AIC7XXX_CONFIGED_TAG_COMMANDS},
	{AIC7XXX_CONFIGED_TAG_COMMANDS},
	{AIC7XXX_CONFIGED_TAG_COMMANDS},
	{AIC7XXX_CONFIGED_TAG_COMMANDS},
	{AIC7XXX_CONFIGED_TAG_COMMANDS},
	{AIC7XXX_CONFIGED_TAG_COMMANDS},
	{AIC7XXX_CONFIGED_TAG_COMMANDS}
};

284 285 286 287 288 289 290 291 292 293 294 295 296
/*
 * DV option:
 *
 * positive value = DV Enabled
 * zero		  = DV Disabled
 * negative value = DV Default for adapter type/seeprom
 */
#ifdef CONFIG_AIC7XXX_DV_SETTING
#define AIC7XXX_CONFIGED_DV CONFIG_AIC7XXX_DV_SETTING
#else
#define AIC7XXX_CONFIGED_DV -1
#endif

297
static int8_t aic7xxx_dv_settings[] =
298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316
{
	AIC7XXX_CONFIGED_DV,
	AIC7XXX_CONFIGED_DV,
	AIC7XXX_CONFIGED_DV,
	AIC7XXX_CONFIGED_DV,
	AIC7XXX_CONFIGED_DV,
	AIC7XXX_CONFIGED_DV,
	AIC7XXX_CONFIGED_DV,
	AIC7XXX_CONFIGED_DV,
	AIC7XXX_CONFIGED_DV,
	AIC7XXX_CONFIGED_DV,
	AIC7XXX_CONFIGED_DV,
	AIC7XXX_CONFIGED_DV,
	AIC7XXX_CONFIGED_DV,
	AIC7XXX_CONFIGED_DV,
	AIC7XXX_CONFIGED_DV,
	AIC7XXX_CONFIGED_DV
};

Linus Torvalds's avatar
Linus Torvalds committed
317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362
/*
 * There should be a specific return value for this in scsi.h, but
 * it seems that most drivers ignore it.
 */
#define DID_UNDERFLOW   DID_ERROR

void
ahc_print_path(struct ahc_softc *ahc, struct scb *scb)
{
	printk("(scsi%d:%c:%d:%d): ",
	       ahc->platform_data->host->host_no,
	       scb != NULL ? SCB_GET_CHANNEL(ahc, scb) : 'X',
	       scb != NULL ? SCB_GET_TARGET(ahc, scb) : -1,
	       scb != NULL ? SCB_GET_LUN(scb) : -1);
}

/*
 * XXX - these options apply unilaterally to _all_ 274x/284x/294x
 *       cards in the system.  This should be fixed.  Exceptions to this
 *       rule are noted in the comments.
 */

/*
 * Skip the scsi bus reset.  Non 0 make us skip the reset at startup.  This
 * has no effect on any later resets that might occur due to things like
 * SCSI bus timeouts.
 */
static uint32_t aic7xxx_no_reset;

/*
 * Certain PCI motherboards will scan PCI devices from highest to lowest,
 * others scan from lowest to highest, and they tend to do all kinds of
 * strange things when they come into contact with PCI bridge chips.  The
 * net result of all this is that the PCI card that is actually used to boot
 * the machine is very hard to detect.  Most motherboards go from lowest
 * PCI slot number to highest, and the first SCSI controller found is the
 * one you boot from.  The only exceptions to this are when a controller
 * has its BIOS disabled.  So, we by default sort all of our SCSI controllers
 * from lowest PCI slot number to highest PCI slot number.  We also force
 * all controllers with their BIOS disabled to the end of the list.  This
 * works on *almost* all computers.  Where it doesn't work, we have this
 * option.  Setting this option to non-0 will reverse the order of the sort
 * to highest first, then lowest, but will still leave cards with their BIOS
 * disabled at the very end.  That should fix everyone up unless there are
 * really strange cirumstances.
 */
363
static uint32_t aic7xxx_reverse_scan;
Linus Torvalds's avatar
Linus Torvalds committed
364 365 366 367 368 369

/*
 * Should we force EXTENDED translation on a controller.
 *     0 == Use whatever is in the SEEPROM or default to off
 *     1 == Use whatever is in the SEEPROM or default to on
 */
370
static uint32_t aic7xxx_extended;
Linus Torvalds's avatar
Linus Torvalds committed
371 372 373 374 375

/*
 * PCI bus parity checking of the Adaptec controllers.  This is somewhat
 * dubious at best.  To my knowledge, this option has never actually
 * solved a PCI parity problem, but on certain machines with broken PCI
376 377
 * chipset configurations where stray PCI transactions with bad parity are
 * the norm rather than the exception, the error messages can be overwelming.
Linus Torvalds's avatar
Linus Torvalds committed
378
 * It's included in the driver for completeness.
379 380
 *   0	   = Shut off PCI parity check
 *   non-0 = reverse polarity pci parity checking
Linus Torvalds's avatar
Linus Torvalds committed
381
 */
382
static uint32_t aic7xxx_pci_parity = ~0;
Linus Torvalds's avatar
Linus Torvalds committed
383 384 385 386 387 388 389 390 391 392 393

/*
 * Certain newer motherboards have put new PCI based devices into the
 * IO spaces that used to typically be occupied by VLB or EISA cards.
 * This overlap can cause these newer motherboards to lock up when scanned
 * for older EISA and VLB devices.  Setting this option to non-0 will
 * cause the driver to skip scanning for any VLB or EISA controllers and
 * only support the PCI controllers.  NOTE: this means that if the kernel
 * os compiled with PCI support disabled, then setting this to non-0
 * would result in never finding any devices :)
 */
394
#ifndef CONFIG_AIC7XXX_PROBE_EISA_VL
395
uint32_t aic7xxx_probe_eisa_vl;
396
#else
397
uint32_t aic7xxx_probe_eisa_vl = ~0;
398 399 400 401 402 403 404 405
#endif

/*
 * There are lots of broken chipsets in the world.  Some of them will
 * violate the PCI spec when we issue byte sized memory writes to our
 * controller.  I/O mapped register access, if allowed by the given
 * platform, will work in almost all cases.
 */
406
uint32_t aic7xxx_allow_memio = ~0;
Linus Torvalds's avatar
Linus Torvalds committed
407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424

/*
 * aic7xxx_detect() has been run, so register all device arrivals
 * immediately with the system rather than deferring to the sorted
 * attachment performed by aic7xxx_detect().
 */
int aic7xxx_detect_complete;

/*
 * So that we can set how long each device is given as a selection timeout.
 * The table of values goes like this:
 *   0 - 256ms
 *   1 - 128ms
 *   2 - 64ms
 *   3 - 32ms
 * We default to 256ms because some older devices need a longer time
 * to respond to initial selection.
 */
425
static uint32_t aic7xxx_seltime;
Linus Torvalds's avatar
Linus Torvalds committed
426 427

/*
Linus Torvalds's avatar
Linus Torvalds committed
428 429 430 431 432 433 434
 * Certain devices do not perform any aging on commands.  Should the
 * device be saturated by commands in one portion of the disk, it is
 * possible for transactions on far away sectors to never be serviced.
 * To handle these devices, we can periodically send an ordered tag to
 * force all outstanding transactions to be serviced prior to a new
 * transaction.
 */
435
uint32_t aic7xxx_periodic_otag;
Linus Torvalds's avatar
Linus Torvalds committed
436 437 438

/*
 * Module information and settable options.
Linus Torvalds's avatar
Linus Torvalds committed
439 440 441 442 443 444 445 446 447
 */
static char *aic7xxx = NULL;
/*
 * Just in case someone uses commas to separate items on the insmod
 * command line, we define a dummy buffer here to avoid having insmod
 * write wild stuff into our code segment
 */
static char dummy_buffer[60] = "Please don't trounce on me insmod!!\n";

Linus Torvalds's avatar
Linus Torvalds committed
448 449 450
MODULE_AUTHOR("Maintainer: Justin T. Gibbs <gibbs@scsiguy.com>");
MODULE_DESCRIPTION("Adaptec Aic77XX/78XX SCSI Host Bus Adapter driver");
MODULE_LICENSE("Dual BSD/GPL");
451
MODULE_VERSION(AIC7XXX_DRIVER_VERSION);
Linus Torvalds's avatar
Linus Torvalds committed
452
MODULE_PARM(aic7xxx, "s");
453 454 455 456 457
MODULE_PARM_DESC(aic7xxx,
"period delimited, options string.\n"
"	verbose			Enable verbose/diagnostic logging\n"
"	allow_memio		Allow device registers to be memory mapped\n"
"	debug			Bitmask of debug values to enable\n"
458
"	no_probe		Toggle EISA/VLB controller probing\n"
459
"	probe_eisa_vl		Toggle EISA/VLB controller probing\n"
460 461 462 463 464 465 466 467 468 469
"	no_reset		Supress initial bus resets\n"
"	extended		Enable extended geometry on all controllers\n"
"	periodic_otag		Send an ordered tagged transaction\n"
"				periodically to prevent tag starvation.\n"
"				This may be required by some older disk\n"
"				drives or RAID arrays.\n"
"	reverse_scan		Sort PCI devices highest Bus/Slot to lowest\n"
"	tag_info:<tag_str>	Set per-target tag depth\n"
"	global_tag_depth:<int>	Global tag depth for every target\n"
"				on every bus\n"
470
"	dv:<dv_settings>	Set per-controller Domain Validation Setting.\n"
471 472 473
"	seltime:<int>		Selection Timeout\n"
"				(0/256ms,1/128ms,2/64ms,3/32ms)\n"
"\n"
474
"	Sample /etc/modprobe.conf line:\n"
475
"		Toggle EISA/VLB probing\n"
476
"		Set tag depth on Controller 1/Target 1 to 10 tags\n"
477 478
"		Shorten the selection timeout to 128ms\n"
"\n"
479
"	options aic7xxx 'aic7xxx=probe_eisa_vl.tag_info:{{}.{.10}}.seltime:1'\n"
480
);
Linus Torvalds's avatar
Linus Torvalds committed
481

Linus Torvalds's avatar
Linus Torvalds committed
482 483 484
static void ahc_linux_handle_scsi_status(struct ahc_softc *,
					 struct ahc_linux_device *,
					 struct scb *);
485 486 487
static void ahc_linux_queue_cmd_complete(struct ahc_softc *ahc,
					 Scsi_Cmnd *cmd);
static void ahc_linux_filter_inquiry(struct ahc_softc*, struct ahc_devinfo*);
Linus Torvalds's avatar
Linus Torvalds committed
488
static void ahc_linux_sem_timeout(u_long arg);
489 490
static void ahc_linux_freeze_simq(struct ahc_softc *ahc);
static void ahc_linux_release_simq(u_long arg);
491
static void ahc_linux_dev_timed_unfreeze(u_long arg);
Linus Torvalds's avatar
Linus Torvalds committed
492 493
static int  ahc_linux_queue_recovery_cmd(Scsi_Cmnd *cmd, scb_flag flag);
static void ahc_linux_initialize_scsi_bus(struct ahc_softc *ahc);
494
static void ahc_linux_size_nseg(void);
495 496 497 498
static void ahc_linux_thread_run_complete_queue(struct ahc_softc *ahc);
static void ahc_linux_start_dv(struct ahc_softc *ahc);
static void ahc_linux_dv_timeout(struct scsi_cmnd *cmd);
static int  ahc_linux_dv_thread(void *data);
499
static void ahc_linux_kill_dv_thread(struct ahc_softc *ahc);
500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535
static void ahc_linux_dv_target(struct ahc_softc *ahc, u_int target);
static void ahc_linux_dv_transition(struct ahc_softc *ahc,
				    struct scsi_cmnd *cmd,
				    struct ahc_devinfo *devinfo,
				    struct ahc_linux_target *targ);
static void ahc_linux_dv_fill_cmd(struct ahc_softc *ahc,
				  struct scsi_cmnd *cmd,
				  struct ahc_devinfo *devinfo);
static void ahc_linux_dv_inq(struct ahc_softc *ahc,
			     struct scsi_cmnd *cmd,
			     struct ahc_devinfo *devinfo,
			     struct ahc_linux_target *targ,
			     u_int request_length);
static void ahc_linux_dv_tur(struct ahc_softc *ahc,
			     struct scsi_cmnd *cmd,
			     struct ahc_devinfo *devinfo);
static void ahc_linux_dv_rebd(struct ahc_softc *ahc,
			      struct scsi_cmnd *cmd,
			      struct ahc_devinfo *devinfo,
			      struct ahc_linux_target *targ);
static void ahc_linux_dv_web(struct ahc_softc *ahc,
			     struct scsi_cmnd *cmd,
			     struct ahc_devinfo *devinfo,
			     struct ahc_linux_target *targ);
static void ahc_linux_dv_reb(struct ahc_softc *ahc,
			     struct scsi_cmnd *cmd,
			     struct ahc_devinfo *devinfo,
			     struct ahc_linux_target *targ);
static void ahc_linux_dv_su(struct ahc_softc *ahc,
			    struct scsi_cmnd *cmd,
			    struct ahc_devinfo *devinfo,
			    struct ahc_linux_target *targ);
static int ahc_linux_fallback(struct ahc_softc *ahc,
			      struct ahc_devinfo *devinfo);
static void ahc_linux_dv_complete(Scsi_Cmnd *cmd);
static void ahc_linux_generate_dv_pattern(struct ahc_linux_target *targ);
536 537
static u_int ahc_linux_user_tagdepth(struct ahc_softc *ahc,
				     struct ahc_devinfo *devinfo);
538
static u_int ahc_linux_user_dv_setting(struct ahc_softc *ahc);
Linus Torvalds's avatar
Linus Torvalds committed
539
static void ahc_linux_device_queue_depth(struct ahc_softc *ahc,
540
					 struct ahc_linux_device *dev);
Linus Torvalds's avatar
Linus Torvalds committed
541 542 543 544 545 546 547 548 549 550 551
static struct ahc_linux_target*	ahc_linux_alloc_target(struct ahc_softc*,
						       u_int, u_int);
static void			ahc_linux_free_target(struct ahc_softc*,
						      struct ahc_linux_target*);
static struct ahc_linux_device*	ahc_linux_alloc_device(struct ahc_softc*,
						       struct ahc_linux_target*,
						       u_int);
static void			ahc_linux_free_device(struct ahc_softc*,
						      struct ahc_linux_device*);
static void ahc_linux_run_device_queue(struct ahc_softc*,
				       struct ahc_linux_device*);
552
static void ahc_linux_setup_tag_info_global(char *p);
553 554
static aic_option_callback_t ahc_linux_setup_tag_info;
static aic_option_callback_t ahc_linux_setup_dv;
555
static int  aic7xxx_setup(char *s);
556
static int  ahc_linux_next_unit(void);
557
static void ahc_runq_tasklet(unsigned long data);
558
static struct ahc_cmd *ahc_linux_run_complete_queue(struct ahc_softc *ahc);
Linus Torvalds's avatar
Linus Torvalds committed
559

560
/********************************* Inlines ************************************/
561
static __inline void ahc_schedule_runq(struct ahc_softc *ahc);
Linus Torvalds's avatar
Linus Torvalds committed
562
static __inline struct ahc_linux_device*
Linus Torvalds's avatar
Linus Torvalds committed
563 564
		     ahc_linux_get_device(struct ahc_softc *ahc, u_int channel,
					  u_int target, u_int lun, int alloc);
565
static __inline void ahc_schedule_completeq(struct ahc_softc *ahc);
Linus Torvalds's avatar
Linus Torvalds committed
566 567
static __inline void ahc_linux_check_device_queue(struct ahc_softc *ahc,
						  struct ahc_linux_device *dev);
568 569 570
static __inline struct ahc_linux_device *
		     ahc_linux_next_device_to_run(struct ahc_softc *ahc);
static __inline void ahc_linux_run_device_queues(struct ahc_softc *ahc);
Linus Torvalds's avatar
Linus Torvalds committed
571
static __inline void ahc_linux_unmap_scb(struct ahc_softc*, struct scb*);
Linus Torvalds's avatar
Linus Torvalds committed
572

Linus Torvalds's avatar
Linus Torvalds committed
573 574 575 576
static __inline int ahc_linux_map_seg(struct ahc_softc *ahc, struct scb *scb,
		 		      struct ahc_dma_seg *sg,
				      bus_addr_t addr, bus_size_t len);

577 578
static __inline void
ahc_schedule_completeq(struct ahc_softc *ahc)
579 580 581 582 583 584 585 586
{
	if ((ahc->platform_data->flags & AHC_RUN_CMPLT_Q_TIMER) == 0) {
		ahc->platform_data->flags |= AHC_RUN_CMPLT_Q_TIMER;
		ahc->platform_data->completeq_timer.expires = jiffies;
		add_timer(&ahc->platform_data->completeq_timer);
	}
}

587 588 589
/*
 * Must be called with our lock held.
 */
590 591 592 593 594 595 596
static __inline void
ahc_schedule_runq(struct ahc_softc *ahc)
{
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,4,0)
	tasklet_schedule(&ahc->platform_data->runq_tasklet);
#else
	/*
597
	 * Tasklets are not available, so run inline.
598 599 600 601 602
	 */
	ahc_runq_tasklet((unsigned long)ahc);
#endif
}

Linus Torvalds's avatar
Linus Torvalds committed
603
static __inline struct ahc_linux_device*
Linus Torvalds's avatar
Linus Torvalds committed
604
ahc_linux_get_device(struct ahc_softc *ahc, u_int channel, u_int target,
605
		     u_int lun, int alloc)
Linus Torvalds's avatar
Linus Torvalds committed
606 607 608 609 610 611 612 613 614 615 616
{
	struct ahc_linux_target *targ;
	struct ahc_linux_device *dev;
	u_int target_offset;

	target_offset = target;
	if (channel != 0)
		target_offset += 8;
	targ = ahc->platform_data->targets[target_offset];
	if (targ == NULL) {
		if (alloc != 0) {
Linus Torvalds's avatar
Linus Torvalds committed
617
			targ = ahc_linux_alloc_target(ahc, channel, target);
Linus Torvalds's avatar
Linus Torvalds committed
618 619 620 621 622 623 624
			if (targ == NULL)
				return (NULL);
		} else
			return (NULL);
	}
	dev = targ->devices[lun];
	if (dev == NULL && alloc != 0)
Linus Torvalds's avatar
Linus Torvalds committed
625
		dev = ahc_linux_alloc_device(ahc, targ, lun);
Linus Torvalds's avatar
Linus Torvalds committed
626 627 628
	return (dev);
}

629 630
#define AHC_LINUX_MAX_RETURNED_ERRORS 4
static struct ahc_cmd *
631 632 633
ahc_linux_run_complete_queue(struct ahc_softc *ahc)
{
	struct	ahc_cmd *acmd;
634 635
	u_long	done_flags;
	int	with_errors;
Linus Torvalds's avatar
Linus Torvalds committed
636

637
	with_errors = 0;
638
	ahc_done_lock(ahc, &done_flags);
639
	while ((acmd = TAILQ_FIRST(&ahc->platform_data->completeq)) != NULL) {
Linus Torvalds's avatar
Linus Torvalds committed
640 641
		Scsi_Cmnd *cmd;

642 643 644 645 646 647 648 649 650
		if (with_errors > AHC_LINUX_MAX_RETURNED_ERRORS) {
			/*
			 * Linux uses stack recursion to requeue
			 * commands that need to be retried.  Avoid
			 * blowing out the stack by "spoon feeding"
			 * commands that completed with error back
			 * the operating system in case they are going
			 * to be retried. "ick"
			 */
651
			ahc_schedule_completeq(ahc);
652 653
			break;
		}
654 655 656 657 658 659 660 661 662
		TAILQ_REMOVE(&ahc->platform_data->completeq,
			     acmd, acmd_links.tqe);
		cmd = &acmd_scsi_cmd(acmd);
		cmd->host_scribble = NULL;
		if (ahc_cmd_get_transaction_status(cmd) != DID_OK
		 || (cmd->result & 0xFF) != SCSI_STATUS_OK)
			with_errors++;

		cmd->scsi_done(cmd);
Linus Torvalds's avatar
Linus Torvalds committed
663 664
	}
	ahc_done_unlock(ahc, &done_flags);
665
	return (acmd);
Linus Torvalds's avatar
Linus Torvalds committed
666 667 668
}

static __inline void
Linus Torvalds's avatar
Linus Torvalds committed
669 670
ahc_linux_check_device_queue(struct ahc_softc *ahc,
			     struct ahc_linux_device *dev)
Linus Torvalds's avatar
Linus Torvalds committed
671 672 673 674 675 676 677 678 679 680 681
{
	if ((dev->flags & AHC_DEV_FREEZE_TIL_EMPTY) != 0
	 && dev->active == 0) {
		dev->flags &= ~AHC_DEV_FREEZE_TIL_EMPTY;
		dev->qfrozen--;
	}

	if (TAILQ_FIRST(&dev->busyq) == NULL
	 || dev->openings == 0 || dev->qfrozen != 0)
		return;

Linus Torvalds's avatar
Linus Torvalds committed
682
	ahc_linux_run_device_queue(ahc, dev);
Linus Torvalds's avatar
Linus Torvalds committed
683 684
}

685 686 687 688 689
static __inline struct ahc_linux_device *
ahc_linux_next_device_to_run(struct ahc_softc *ahc)
{
	
	if ((ahc->flags & AHC_RESOURCE_SHORTAGE) != 0
690 691
	 || (ahc->platform_data->qfrozen != 0
	  && AHC_DV_SIMQ_FROZEN(ahc) == 0))
692 693 694 695
		return (NULL);
	return (TAILQ_FIRST(&ahc->platform_data->device_runq));
}

Linus Torvalds's avatar
Linus Torvalds committed
696
static __inline void
Linus Torvalds's avatar
Linus Torvalds committed
697
ahc_linux_run_device_queues(struct ahc_softc *ahc)
Linus Torvalds's avatar
Linus Torvalds committed
698 699 700
{
	struct ahc_linux_device *dev;

701
	while ((dev = ahc_linux_next_device_to_run(ahc)) != NULL) {
Linus Torvalds's avatar
Linus Torvalds committed
702
		TAILQ_REMOVE(&ahc->platform_data->device_runq, dev, links);
Linus Torvalds's avatar
Linus Torvalds committed
703
		dev->flags &= ~AHC_DEV_ON_RUN_LIST;
Linus Torvalds's avatar
Linus Torvalds committed
704
		ahc_linux_check_device_queue(ahc, dev);
Linus Torvalds's avatar
Linus Torvalds committed
705 706 707 708
	}
}

static __inline void
Linus Torvalds's avatar
Linus Torvalds committed
709
ahc_linux_unmap_scb(struct ahc_softc *ahc, struct scb *scb)
Linus Torvalds's avatar
Linus Torvalds committed
710 711 712 713
{
	Scsi_Cmnd *cmd;

	cmd = scb->io_ctx;
Linus Torvalds's avatar
Linus Torvalds committed
714
	ahc_sync_sglist(ahc, scb, BUS_DMASYNC_POSTWRITE);
Linus Torvalds's avatar
Linus Torvalds committed
715 716 717 718 719 720
	if (cmd->use_sg != 0) {
		struct scatterlist *sg;

		sg = (struct scatterlist *)cmd->request_buffer;
		pci_unmap_sg(ahc->dev_softc, sg, cmd->use_sg,
			     scsi_to_pci_dma_dir(cmd->sc_data_direction));
Linus Torvalds's avatar
Linus Torvalds committed
721
	} else if (cmd->request_bufflen != 0) {
Linus Torvalds's avatar
Linus Torvalds committed
722
		pci_unmap_single(ahc->dev_softc,
723
				 scb->platform_data->buf_busaddr,
Linus Torvalds's avatar
Linus Torvalds committed
724 725
				 cmd->request_bufflen,
				 scsi_to_pci_dma_dir(cmd->sc_data_direction));
Linus Torvalds's avatar
Linus Torvalds committed
726
	}
Linus Torvalds's avatar
Linus Torvalds committed
727 728
}

Linus Torvalds's avatar
Linus Torvalds committed
729 730 731 732 733 734 735 736 737 738 739 740 741
static __inline int
ahc_linux_map_seg(struct ahc_softc *ahc, struct scb *scb,
		  struct ahc_dma_seg *sg, bus_addr_t addr, bus_size_t len)
{
	int	 consumed;

	if ((scb->sg_count + 1) > AHC_NSEG)
		panic("Too few segs for dma mapping.  "
		      "Increase AHC_NSEG\n");

	consumed = 1;
	sg->addr = ahc_htole32(addr & 0xFFFFFFFF);
	scb->platform_data->xfer_len += len;
742

Linus Torvalds's avatar
Linus Torvalds committed
743
	if (sizeof(bus_addr_t) > 4
744 745 746
	 && (ahc->flags & AHC_39BIT_ADDRESSING) != 0)
		len |= (addr >> 8) & AHC_SG_HIGH_ADDR_MASK;

Linus Torvalds's avatar
Linus Torvalds committed
747 748 749 750
	sg->len = ahc_htole32(len);
	return (consumed);
}

751 752 753 754 755
/************************  Host template entry points *************************/
static int	   ahc_linux_detect(Scsi_Host_Template *);
static int	   ahc_linux_queue(Scsi_Cmnd *, void (*)(Scsi_Cmnd *));
static const char *ahc_linux_info(struct Scsi_Host *);
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,5,0)
756
static int	   ahc_linux_slave_alloc(Scsi_Device *);
757 758
static int	   ahc_linux_slave_configure(Scsi_Device *);
static void	   ahc_linux_slave_destroy(Scsi_Device *);
759
#if defined(__i386__)
760
static int	   ahc_linux_biosparam(struct scsi_device*,
761 762
				       struct block_device*,
				       sector_t, int[]);
763
#endif
764
#else
765
static int	   ahc_linux_release(struct Scsi_Host *);
766 767
static void	   ahc_linux_select_queue_depth(struct Scsi_Host *host,
						Scsi_Device *scsi_devs);
768
#if defined(__i386__)
769 770
static int	   ahc_linux_biosparam(Disk *, kdev_t, int[]);
#endif
771
#endif
772 773 774 775
static int	   ahc_linux_bus_reset(Scsi_Cmnd *);
static int	   ahc_linux_dev_reset(Scsi_Cmnd *);
static int	   ahc_linux_abort(Scsi_Cmnd *);

776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836
/*
 * Calculate a safe value for AHC_NSEG (as expressed through ahc_linux_nseg).
 *
 * In pre-2.5.X...
 * The midlayer allocates an S/G array dynamically when a command is issued
 * using SCSI malloc.  This array, which is in an OS dependent format that
 * must later be copied to our private S/G list, is sized to house just the
 * number of segments needed for the current transfer.  Since the code that
 * sizes the SCSI malloc pool does not take into consideration fragmentation
 * of the pool, executing transactions numbering just a fraction of our
 * concurrent transaction limit with list lengths aproaching AHC_NSEG will
 * quickly depleat the SCSI malloc pool of usable space.  Unfortunately, the
 * mid-layer does not properly handle this scsi malloc failures for the S/G
 * array and the result can be a lockup of the I/O subsystem.  We try to size
 * our S/G list so that it satisfies our drivers allocation requirements in
 * addition to avoiding fragmentation of the SCSI malloc pool.
 */
static void
ahc_linux_size_nseg(void)
{
#if LINUX_VERSION_CODE < KERNEL_VERSION(2,5,0)
	u_int cur_size;
	u_int best_size;

	/*
	 * The SCSI allocator rounds to the nearest 512 bytes
	 * an cannot allocate across a page boundary.  Our algorithm
	 * is to start at 1K of scsi malloc space per-command and
	 * loop through all factors of the PAGE_SIZE and pick the best.
	 */
	best_size = 0;
	for (cur_size = 1024; cur_size <= PAGE_SIZE; cur_size *= 2) {
		u_int nseg;

		nseg = cur_size / sizeof(struct scatterlist);
		if (nseg < AHC_LINUX_MIN_NSEG)
			continue;

		if (best_size == 0) {
			best_size = cur_size;
			ahc_linux_nseg = nseg;
		} else {
			u_int best_rem;
			u_int cur_rem;

			/*
			 * Compare the traits of the current "best_size"
			 * with the current size to determine if the
			 * current size is a better size.
			 */
			best_rem = best_size % sizeof(struct scatterlist);
			cur_rem = cur_size % sizeof(struct scatterlist);
			if (cur_rem < best_rem) {
				best_size = cur_size;
				ahc_linux_nseg = nseg;
			}
		}
	}
#endif
}

837 838 839 840 841 842 843
/*
 * Try to detect an Adaptec 7XXX controller.
 */
static int
ahc_linux_detect(Scsi_Host_Template *template)
{
	struct	ahc_softc *ahc;
844
	int     found = 0;
845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864

#if LINUX_VERSION_CODE < KERNEL_VERSION(2,5,0)
	/*
	 * It is a bug that the upper layer takes
	 * this lock just prior to calling us.
	 */
	spin_unlock_irq(&io_request_lock);
#endif

	/*
	 * Sanity checking of Linux SCSI data structures so
	 * that some of our hacks^H^H^H^H^Hassumptions aren't
	 * violated.
	 */
	if (offsetof(struct ahc_cmd_internal, end)
	  > offsetof(struct scsi_cmnd, host_scribble)) {
		printf("ahc_linux_detect: SCSI data structures changed.\n");
		printf("ahc_linux_detect: Unable to attach\n");
		return (0);
	}
865
	ahc_linux_size_nseg();
866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891
#ifdef MODULE
	/*
	 * If we've been passed any parameters, process them now.
	 */
	if (aic7xxx)
		aic7xxx_setup(aic7xxx);
	if (dummy_buffer[0] != 'P')
		printk(KERN_WARNING
"aic7xxx: Please read the file /usr/src/linux/drivers/scsi/README.aic7xxx\n"
"aic7xxx: to see the proper way to specify options to the aic7xxx module\n"
"aic7xxx: Specifically, don't use any commas when passing arguments to\n"
"aic7xxx: insmod or else it might trash certain memory areas.\n");
#endif

#if LINUX_VERSION_CODE > KERNEL_VERSION(2,3,0)
	template->proc_name = "aic7xxx";
#else
	template->proc_dir = &proc_scsi_aic7xxx;
#endif

	/*
	 * Initialize our softc list lock prior to
	 * probing for any adapters.
	 */
	ahc_list_lockinit();

892 893 894 895
	found = ahc_linux_pci_init();
	if (!ahc_linux_eisa_init())
		found++;
	
896 897 898 899 900 901 902 903 904
	/*
	 * Register with the SCSI layer all
	 * controllers we've found.
	 */
	TAILQ_FOREACH(ahc, &ahc_tailq, links) {

		if (ahc_linux_register_host(ahc, template) == 0)
			found++;
	}
905

906 907 908 909
#if LINUX_VERSION_CODE < KERNEL_VERSION(2,5,0)
	spin_lock_irq(&io_request_lock);
#endif
	aic7xxx_detect_complete++;
910

911 912 913
	return (found);
}

914
#if LINUX_VERSION_CODE < KERNEL_VERSION(2,5,0)
915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945
/*
 * Free the passed in Scsi_Host memory structures prior to unloading the
 * module.
 */
int
ahc_linux_release(struct Scsi_Host * host)
{
	struct ahc_softc *ahc;
	u_long l;

	ahc_list_lock(&l);
	if (host != NULL) {

		/*
		 * We should be able to just perform
		 * the free directly, but check our
		 * list for extra sanity.
		 */
		ahc = ahc_find_softc(*(struct ahc_softc **)host->hostdata);
		if (ahc != NULL) {
			u_long s;

			ahc_lock(ahc, &s);
			ahc_intr_enable(ahc, FALSE);
			ahc_unlock(ahc, &s);
			ahc_free(ahc);
		}
	}
	ahc_list_unlock(&l);
	return (0);
}
946
#endif
947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 983 984 985

/*
 * Return a string describing the driver.
 */
static const char *
ahc_linux_info(struct Scsi_Host *host)
{
	static char buffer[512];
	char	ahc_info[256];
	char   *bp;
	struct ahc_softc *ahc;

	bp = &buffer[0];
	ahc = *(struct ahc_softc **)host->hostdata;
	memset(bp, 0, sizeof(buffer));
	strcpy(bp, "Adaptec AIC7XXX EISA/VLB/PCI SCSI HBA DRIVER, Rev ");
	strcat(bp, AIC7XXX_DRIVER_VERSION);
	strcat(bp, "\n");
	strcat(bp, "        <");
	strcat(bp, ahc->description);
	strcat(bp, ">\n");
	strcat(bp, "        ");
	ahc_controller_info(ahc, ahc_info);
	strcat(bp, ahc_info);
	strcat(bp, "\n");

	return (bp);
}

/*
 * Queue an SCB to the controller.
 */
static int
ahc_linux_queue(Scsi_Cmnd * cmd, void (*scsi_done) (Scsi_Cmnd *))
{
	struct	 ahc_softc *ahc;
	struct	 ahc_linux_device *dev;
	u_long	 flags;

Justin T. Gibbs's avatar
Justin T. Gibbs committed
986
	ahc = *(struct ahc_softc **)cmd->device->host->hostdata;
987 988 989 990 991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005

	/*
	 * Save the callback on completion function.
	 */
	cmd->scsi_done = scsi_done;

	ahc_midlayer_entrypoint_lock(ahc, &flags);

	/*
	 * Close the race of a command that was in the process of
	 * being queued to us just as our simq was frozen.  Let
	 * DV commands through so long as we are only frozen to
	 * perform DV.
	 */
	if (ahc->platform_data->qfrozen != 0
	 && AHC_DV_CMD(cmd) == 0) {

		ahc_cmd_set_transaction_status(cmd, CAM_REQUEUE_REQ);
		ahc_linux_queue_cmd_complete(ahc, cmd);
1006
		ahc_schedule_completeq(ahc);
1007 1008 1009
		ahc_midlayer_entrypoint_unlock(ahc, &flags);
		return (0);
	}
Justin T. Gibbs's avatar
Justin T. Gibbs committed
1010 1011
	dev = ahc_linux_get_device(ahc, cmd->device->channel, cmd->device->id,
				   cmd->device->lun, /*alloc*/TRUE);
1012
	if (dev == NULL) {
1013 1014
		ahc_cmd_set_transaction_status(cmd, CAM_RESRC_UNAVAIL);
		ahc_linux_queue_cmd_complete(ahc, cmd);
1015
		ahc_schedule_completeq(ahc);
1016
		ahc_midlayer_entrypoint_unlock(ahc, &flags);
1017 1018 1019
		printf("%s: aic7xxx_linux_queue - Unable to allocate device!\n",
		       ahc_name(ahc));
		return (0);
1020 1021 1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035
	}
	cmd->result = CAM_REQ_INPROG << 16;
	TAILQ_INSERT_TAIL(&dev->busyq, (struct ahc_cmd *)cmd, acmd_links.tqe);
	if ((dev->flags & AHC_DEV_ON_RUN_LIST) == 0) {
		TAILQ_INSERT_TAIL(&ahc->platform_data->device_runq, dev, links);
		dev->flags |= AHC_DEV_ON_RUN_LIST;
		ahc_linux_run_device_queues(ahc);
	}
	ahc_midlayer_entrypoint_unlock(ahc, &flags);
	return (0);
}

#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,5,0)
static int
ahc_linux_slave_alloc(Scsi_Device *device)
{
1036 1037 1038 1039 1040
	struct	ahc_softc *ahc;

	ahc = *((struct ahc_softc **)device->host->hostdata);
	if (bootverbose)
		printf("%s: Slave Alloc %d\n", ahc_name(ahc), device->id);
1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051
	return (0);
}

static int
ahc_linux_slave_configure(Scsi_Device *device)
{
	struct	ahc_softc *ahc;
	struct	ahc_linux_device *dev;
	u_long	flags;

	ahc = *((struct ahc_softc **)device->host->hostdata);
1052 1053
	if (bootverbose)
		printf("%s: Slave Configure %d\n", ahc_name(ahc), device->id);
1054 1055 1056 1057 1058 1059 1060 1061
	ahc_midlayer_entrypoint_lock(ahc, &flags);
	/*
	 * Since Linux has attached to the device, configure
	 * it so we don't free and allocate the device
	 * structure on every command.
	 */
	dev = ahc_linux_get_device(ahc, device->channel,
				   device->id, device->lun,
1062
				   /*alloc*/TRUE);
1063 1064 1065
	if (dev != NULL) {
		dev->flags &= ~AHC_DEV_UNCONFIGURED;
		dev->scsi_device = device;
1066
		ahc_linux_device_queue_depth(ahc, dev);
1067 1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079
	}
	ahc_midlayer_entrypoint_unlock(ahc, &flags);
	return (0);
}

static void
ahc_linux_slave_destroy(Scsi_Device *device)
{
	struct	ahc_softc *ahc;
	struct	ahc_linux_device *dev;
	u_long	flags;

	ahc = *((struct ahc_softc **)device->host->hostdata);
1080 1081
	if (bootverbose)
		printf("%s: Slave Destroy %d\n", ahc_name(ahc), device->id);
1082 1083 1084 1085
	ahc_midlayer_entrypoint_lock(ahc, &flags);
	dev = ahc_linux_get_device(ahc, device->channel,
				   device->id, device->lun,
					   /*alloc*/FALSE);
1086 1087 1088 1089 1090 1091 1092 1093 1094
	/*
	 * Filter out "silly" deletions of real devices by only
	 * deleting devices that have had slave_configure()
	 * called on them.  All other devices that have not
	 * been configured will automatically be deleted by
	 * the refcounting process.
	 */
	if (dev != NULL
	 && (dev->flags & AHC_DEV_SLAVE_CONFIGURED) != 0) {
1095
		dev->flags |= AHC_DEV_UNCONFIGURED;
1096
		if (TAILQ_EMPTY(&dev->busyq)
1097 1098
		 && dev->active == 0
	 	 && (dev->flags & AHC_DEV_TIMER_ACTIVE) == 0)
1099 1100
			ahc_linux_free_device(ahc, dev);
	}
1101 1102 1103 1104 1105 1106 1107 1108
	ahc_midlayer_entrypoint_unlock(ahc, &flags);
}
#else
/*
 * Sets the queue depth for each SCSI device hanging
 * off the input host adapter.
 */
static void
1109
ahc_linux_select_queue_depth(struct Scsi_Host *host, Scsi_Device *scsi_devs)
1110 1111
{
	Scsi_Device *device;
1112
	Scsi_Device *ldev;
1113 1114 1115 1116
	struct	ahc_softc *ahc;
	u_long	flags;

	ahc = *((struct ahc_softc **)host->hostdata);
1117
	ahc_lock(ahc, &flags);
1118
	for (device = scsi_devs; device != NULL; device = device->next) {
1119 1120 1121 1122 1123 1124 1125 1126 1127 1128 1129 1130 1131 1132 1133 1134 1135

		/*
		 * Watch out for duplicate devices.  This works around
		 * some quirks in how the SCSI scanning code does its
		 * device management.
		 */
		for (ldev = scsi_devs; ldev != device; ldev = ldev->next) {
			if (ldev->host == device->host
			 && ldev->channel == device->channel
			 && ldev->id == device->id
			 && ldev->lun == device->lun)
				break;
		}
		/* Skip duplicate. */
		if (ldev != device)
			continue;

1136 1137 1138 1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156 1157 1158 1159 1160 1161 1162 1163 1164 1165 1166
		if (device->host == host) {
			struct	 ahc_linux_device *dev;

			/*
			 * Since Linux has attached to the device, configure
			 * it so we don't free and allocate the device
			 * structure on every command.
			 */
			dev = ahc_linux_get_device(ahc, device->channel,
						   device->id, device->lun,
						   /*alloc*/TRUE);
			if (dev != NULL) {
				dev->flags &= ~AHC_DEV_UNCONFIGURED;
				dev->scsi_device = device;
				ahc_linux_device_queue_depth(ahc, dev);
				device->queue_depth = dev->openings
						    + dev->active;
				if ((dev->flags & (AHC_DEV_Q_BASIC
						| AHC_DEV_Q_TAGGED)) == 0) {
					/*
					 * We allow the OS to queue 2 untagged
					 * transactions to us at any time even
					 * though we can only execute them
					 * serially on the controller/device.
					 * This should remove some latency.
					 */
					device->queue_depth = 2;
				}
			}
		}
	}
1167
	ahc_unlock(ahc, &flags);
1168 1169 1170
}
#endif

1171
#if defined(__i386__)
1172 1173 1174 1175 1176 1177 1178
/*
 * Return the disk geometry for the given SCSI device.
 */
static int
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,5,0)
ahc_linux_biosparam(struct scsi_device *sdev, struct block_device *bdev,
		    sector_t capacity, int geom[])
1179 1180
{
	uint8_t *bh;
1181 1182 1183
#else
ahc_linux_biosparam(Disk *disk, kdev_t dev, int geom[])
{
1184 1185 1186 1187
	struct	scsi_device *sdev = disk->device;
	u_long	capacity = disk->capacity;
	struct	buffer_head *bh;
#endif
1188 1189 1190 1191 1192 1193 1194 1195 1196
	int	 heads;
	int	 sectors;
	int	 cylinders;
	int	 ret;
	int	 extended;
	struct	 ahc_softc *ahc;
	u_int	 channel;

	ahc = *((struct ahc_softc **)sdev->host->hostdata);
1197
	channel = sdev->channel;
1198 1199 1200 1201 1202 1203 1204 1205 1206 1207 1208 1209 1210 1211 1212 1213 1214 1215 1216 1217 1218 1219

#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,5,0)
	bh = scsi_bios_ptable(bdev);
#elif LINUX_VERSION_CODE >= KERNEL_VERSION(2,4,17)
	bh = bread(MKDEV(MAJOR(dev), MINOR(dev) & ~0xf), 0, block_size(dev));
#else
	bh = bread(MKDEV(MAJOR(dev), MINOR(dev) & ~0xf), 0, 1024);
#endif

	if (bh) {
		ret = scsi_partsize(bh, capacity,
				    &geom[2], &geom[0], &geom[1]);
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,5,0)
		kfree(bh);
#else
		brelse(bh);
#endif
		if (ret != -1)
			return (ret);
	}
	heads = 64;
	sectors = 32;
1220
	cylinders = aic_sector_div(capacity, heads, sectors);
1221 1222 1223 1224 1225 1226 1227 1228 1229 1230

	if (aic7xxx_extended != 0)
		extended = 1;
	else if (channel == 0)
		extended = (ahc->flags & AHC_EXTENDED_TRANS_A) != 0;
	else
		extended = (ahc->flags & AHC_EXTENDED_TRANS_B) != 0;
	if (extended && cylinders >= 1024) {
		heads = 255;
		sectors = 63;
1231
		cylinders = aic_sector_div(capacity, heads, sectors);
1232 1233 1234 1235 1236 1237
	}
	geom[0] = heads;
	geom[1] = sectors;
	geom[2] = cylinders;
	return (0);
}
1238
#endif
1239 1240 1241 1242 1243 1244 1245 1246 1247 1248 1249 1250 1251 1252 1253 1254 1255 1256 1257 1258 1259 1260 1261 1262 1263 1264 1265 1266 1267 1268 1269 1270 1271 1272 1273 1274 1275 1276 1277

/*
 * Abort the current SCSI command(s).
 */
static int
ahc_linux_abort(Scsi_Cmnd *cmd)
{
	int error;

	error = ahc_linux_queue_recovery_cmd(cmd, SCB_ABORT);
	if (error != 0)
		printf("aic7xxx_abort returns 0x%x\n", error);
	return (error);
}

/*
 * Attempt to send a target reset message to the device that timed out.
 */
static int
ahc_linux_dev_reset(Scsi_Cmnd *cmd)
{
	int error;

	error = ahc_linux_queue_recovery_cmd(cmd, SCB_DEVICE_RESET);
	if (error != 0)
		printf("aic7xxx_dev_reset returns 0x%x\n", error);
	return (error);
}

/*
 * Reset the SCSI bus.
 */
static int
ahc_linux_bus_reset(Scsi_Cmnd *cmd)
{
	struct ahc_softc *ahc;
	u_long s;
	int    found;

Justin T. Gibbs's avatar
Justin T. Gibbs committed
1278
	ahc = *(struct ahc_softc **)cmd->device->host->hostdata;
1279
	ahc_midlayer_entrypoint_lock(ahc, &s);
Justin T. Gibbs's avatar
Justin T. Gibbs committed
1280
	found = ahc_reset_channel(ahc, cmd->device->channel + 'A',
1281
				  /*initiate reset*/TRUE);
1282
	ahc_linux_run_complete_queue(ahc);
1283 1284 1285 1286 1287 1288
	ahc_midlayer_entrypoint_unlock(ahc, &s);

	if (bootverbose)
		printf("%s: SCSI bus reset delivered. "
		       "%d SCBs aborted.\n", ahc_name(ahc), found);

1289 1290 1291 1292
	return SUCCESS;
}

Scsi_Host_Template aic7xxx_driver_template = {
1293 1294
	.module			= THIS_MODULE,
	.name			= "aic7xxx",
1295 1296 1297 1298 1299 1300 1301 1302 1303 1304 1305 1306 1307
	.proc_info		= ahc_linux_proc_info,
	.info			= ahc_linux_info,
	.queuecommand		= ahc_linux_queue,
	.eh_abort_handler	= ahc_linux_abort,
	.eh_device_reset_handler = ahc_linux_dev_reset,
	.eh_bus_reset_handler	= ahc_linux_bus_reset,
#if defined(__i386__)
	.bios_param		= ahc_linux_biosparam,
#endif
	.can_queue		= AHC_MAX_QUEUE,
	.this_id		= -1,
	.cmd_per_lun		= 2,
	.use_clustering		= ENABLE_CLUSTERING,
1308
	.slave_alloc		= ahc_linux_slave_alloc,
1309 1310 1311 1312
	.slave_configure	= ahc_linux_slave_configure,
	.slave_destroy		= ahc_linux_slave_destroy,
};

1313 1314
/**************************** Tasklet Handler *********************************/

1315 1316 1317 1318 1319 1320
/*
 * In 2.4.X and above, this routine is called from a tasklet,
 * so we must re-acquire our lock prior to executing this code.
 * In all prior kernels, ahc_schedule_runq() calls this routine
 * directly and ahc_schedule_runq() is called with our lock held.
 */
1321 1322 1323 1324 1325
static void
ahc_runq_tasklet(unsigned long data)
{
	struct ahc_softc* ahc;
	struct ahc_linux_device *dev;
1326
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,4,0)
1327
	u_long flags;
1328
#endif
1329 1330

	ahc = (struct ahc_softc *)data;
1331
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,4,0)
1332
	ahc_lock(ahc, &flags);
1333
#endif
1334 1335 1336 1337 1338
	while ((dev = ahc_linux_next_device_to_run(ahc)) != NULL) {
	
		TAILQ_REMOVE(&ahc->platform_data->device_runq, dev, links);
		dev->flags &= ~AHC_DEV_ON_RUN_LIST;
		ahc_linux_check_device_queue(ahc, dev);
1339
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,4,0)
1340 1341 1342
		/* Yeild to our interrupt handler */
		ahc_unlock(ahc, &flags);
		ahc_lock(ahc, &flags);
1343
#endif
1344
	}
1345
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,4,0)
1346
	ahc_unlock(ahc, &flags);
1347
#endif
1348 1349
}

Linus Torvalds's avatar
Linus Torvalds committed
1350
/******************************** Macros **************************************/
Justin T. Gibbs's avatar
Justin T. Gibbs committed
1351 1352 1353 1354
#define BUILD_SCSIID(ahc, cmd)						    \
	((((cmd)->device->id << TID_SHIFT) & TID)			    \
	| (((cmd)->device->channel == 0) ? (ahc)->our_id : (ahc)->our_id_b) \
	| (((cmd)->device->channel == 0) ? 0 : TWIN_CHNLB))
Linus Torvalds's avatar
Linus Torvalds committed
1355 1356 1357 1358 1359 1360 1361 1362 1363 1364 1365 1366 1367 1368 1369 1370 1371 1372 1373 1374 1375 1376 1377 1378 1379 1380 1381 1382 1383 1384 1385 1386 1387 1388 1389 1390 1391 1392 1393 1394 1395 1396 1397 1398 1399 1400

/******************************** Bus DMA *************************************/
int
ahc_dma_tag_create(struct ahc_softc *ahc, bus_dma_tag_t parent,
		   bus_size_t alignment, bus_size_t boundary,
		   bus_addr_t lowaddr, bus_addr_t highaddr,
		   bus_dma_filter_t *filter, void *filterarg,
		   bus_size_t maxsize, int nsegments,
		   bus_size_t maxsegsz, int flags, bus_dma_tag_t *ret_tag)
{
	bus_dma_tag_t dmat;

	dmat = malloc(sizeof(*dmat), M_DEVBUF, M_NOWAIT);
	if (dmat == NULL)
		return (ENOMEM);

	/*
	 * Linux is very simplistic about DMA memory.  For now don't
	 * maintain all specification information.  Once Linux supplies
	 * better facilities for doing these operations, or the
	 * needs of this particular driver change, we might need to do
	 * more here.
	 */
	dmat->alignment = alignment;
	dmat->boundary = boundary;
	dmat->maxsize = maxsize;
	*ret_tag = dmat;
	return (0);
}

void
ahc_dma_tag_destroy(struct ahc_softc *ahc, bus_dma_tag_t dmat)
{
	free(dmat, M_DEVBUF);
}

int
ahc_dmamem_alloc(struct ahc_softc *ahc, bus_dma_tag_t dmat, void** vaddr,
		 int flags, bus_dmamap_t *mapp)
{
	bus_dmamap_t map;

#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,3,0)
	map = malloc(sizeof(*map), M_DEVBUF, M_NOWAIT);
	if (map == NULL)
		return (ENOMEM);
Linus Torvalds's avatar
Linus Torvalds committed
1401 1402 1403 1404 1405 1406 1407
	/*
	 * Although we can dma data above 4GB, our
	 * "consistent" memory is below 4GB for
	 * space efficiency reasons (only need a 4byte
	 * address).  For this reason, we have to reset
	 * our dma mask when doing allocations.
	 */
1408
	if (ahc->dev_softc != NULL)
1409 1410 1411 1412
		if (ahc_pci_set_dma_mask(ahc->dev_softc, 0xFFFFFFFF)) {
			printk(KERN_WARNING "aic7xxx: No suitable DMA available.\n");
			return (ENODEV);
		}
Linus Torvalds's avatar
Linus Torvalds committed
1413 1414
	*vaddr = pci_alloc_consistent(ahc->dev_softc,
				      dmat->maxsize, &map->bus_addr);
1415
	if (ahc->dev_softc != NULL)
1416 1417 1418 1419 1420
		if (ahc_pci_set_dma_mask(ahc->dev_softc,
				     ahc->platform_data->hw_dma_mask)) {
			printk(KERN_WARNING "aic7xxx: No suitable DMA available.\n");
			return (ENODEV);
		}
Linus Torvalds's avatar
Linus Torvalds committed
1421
#else /* LINUX_VERSION_CODE < KERNEL_VERSION(2,3,0) */
Linus Torvalds's avatar
Linus Torvalds committed
1422 1423
	/*
	 * At least in 2.2.14, malloc is a slab allocator so all
Linus Torvalds's avatar
Linus Torvalds committed
1424
	 * allocations are aligned.  We assume for these kernel versions
Linus Torvalds's avatar
Linus Torvalds committed
1425
	 * that all allocations will be bellow 4Gig, physically contiguous,
1426
	 * and accessible via DMA by the controller.
Linus Torvalds's avatar
Linus Torvalds committed
1427 1428 1429 1430 1431 1432 1433 1434 1435 1436 1437 1438 1439 1440 1441 1442 1443 1444 1445 1446 1447 1448 1449 1450 1451 1452 1453 1454 1455 1456 1457 1458 1459 1460 1461 1462
	 */
	map = NULL; /* No additional information to store */
	*vaddr = malloc(dmat->maxsize, M_DEVBUF, M_NOWAIT);
#endif
	if (*vaddr == NULL)
		return (ENOMEM);
	*mapp = map;
	return(0);
}

void
ahc_dmamem_free(struct ahc_softc *ahc, bus_dma_tag_t dmat,
		void* vaddr, bus_dmamap_t map)
{
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,3,0)
	pci_free_consistent(ahc->dev_softc, dmat->maxsize,
			    vaddr, map->bus_addr);
#else
	free(vaddr, M_DEVBUF);
#endif
}

int
ahc_dmamap_load(struct ahc_softc *ahc, bus_dma_tag_t dmat, bus_dmamap_t map,
		void *buf, bus_size_t buflen, bus_dmamap_callback_t *cb,
		void *cb_arg, int flags)
{
	/*
	 * Assume for now that this will only be used during
	 * initialization and not for per-transaction buffer mapping.
	 */
	bus_dma_segment_t stack_sg;

#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,3,0)
	stack_sg.ds_addr = map->bus_addr;
#else
1463
#define VIRT_TO_BUS(a) (uint32_t)virt_to_bus((void *)(a))
Linus Torvalds's avatar
Linus Torvalds committed
1464 1465 1466 1467 1468 1469 1470 1471 1472 1473 1474 1475 1476 1477 1478 1479 1480 1481 1482 1483 1484 1485 1486 1487 1488
	stack_sg.ds_addr = VIRT_TO_BUS(buf);
#endif
	stack_sg.ds_len = dmat->maxsize;
	cb(cb_arg, &stack_sg, /*nseg*/1, /*error*/0);
	return (0);
}

void
ahc_dmamap_destroy(struct ahc_softc *ahc, bus_dma_tag_t dmat, bus_dmamap_t map)
{
	/*
	 * The map may is NULL in our < 2.3.X implementation.
	 */
	if (map != NULL)
		free(map, M_DEVBUF);
}

int
ahc_dmamap_unload(struct ahc_softc *ahc, bus_dma_tag_t dmat, bus_dmamap_t map)
{
	/* Nothing to do */
	return (0);
}

/********************* Platform Dependent Functions ***************************/
1489 1490 1491 1492 1493 1494
/*
 * Compare "left hand" softc with "right hand" softc, returning:
 * < 0 - lahc has a lower priority than rahc
 *   0 - Softcs are equal
 * > 0 - lahc has a higher priority than rahc
 */
Linus Torvalds's avatar
Linus Torvalds committed
1495 1496 1497 1498 1499 1500 1501 1502 1503 1504 1505 1506 1507 1508 1509 1510 1511 1512
int
ahc_softc_comp(struct ahc_softc *lahc, struct ahc_softc *rahc)
{
	int	value;
	int	rvalue;
	int	lvalue;

	/*
	 * Under Linux, cards are ordered as follows:
	 *	1) VLB/EISA BIOS enabled devices sorted by BIOS address.
	 *	2) PCI devices with BIOS enabled sorted by bus/slot/func.
	 *	3) All remaining VLB/EISA devices sorted by ioport.
	 *	4) All remaining PCI devices sorted by bus/slot/func.
	 */
	value = (lahc->flags & AHC_BIOS_ENABLED)
	      - (rahc->flags & AHC_BIOS_ENABLED);
	if (value != 0)
		/* Controllers with BIOS enabled have a *higher* priority */
1513
		return (value);
Linus Torvalds's avatar
Linus Torvalds committed
1514 1515 1516 1517 1518 1519 1520 1521 1522 1523 1524 1525

	/*
	 * Same BIOS setting, now sort based on bus type.
	 * EISA and VL controllers sort together.  EISA/VL
	 * have higher priority than PCI.
	 */
	rvalue = (rahc->chip & AHC_BUS_MASK);
 	if (rvalue == AHC_VL)
		rvalue = AHC_EISA;
	lvalue = (lahc->chip & AHC_BUS_MASK);
 	if (lvalue == AHC_VL)
		lvalue = AHC_EISA;
1526
	value = rvalue - lvalue;
Linus Torvalds's avatar
Linus Torvalds committed
1527 1528 1529 1530 1531
	if (value != 0)
		return (value);

	/* Still equal.  Sort by BIOS address, ioport, or bus/slot/func. */
	switch (rvalue) {
1532
#ifdef CONFIG_PCI
Linus Torvalds's avatar
Linus Torvalds committed
1533
	case AHC_PCI:
Linus Torvalds's avatar
Linus Torvalds committed
1534 1535 1536
	{
		char primary_channel;

Linus Torvalds's avatar
Linus Torvalds committed
1537 1538 1539
		if (aic7xxx_reverse_scan != 0)
			value = ahc_get_pci_bus(lahc->dev_softc)
			      - ahc_get_pci_bus(rahc->dev_softc);
1540 1541 1542
		else
			value = ahc_get_pci_bus(rahc->dev_softc)
			      - ahc_get_pci_bus(lahc->dev_softc);
Linus Torvalds's avatar
Linus Torvalds committed
1543 1544
		if (value != 0)
			break;
Linus Torvalds's avatar
Linus Torvalds committed
1545 1546 1547
		if (aic7xxx_reverse_scan != 0)
			value = ahc_get_pci_slot(lahc->dev_softc)
			      - ahc_get_pci_slot(rahc->dev_softc);
1548 1549 1550
		else
			value = ahc_get_pci_slot(rahc->dev_softc)
			      - ahc_get_pci_slot(lahc->dev_softc);
Linus Torvalds's avatar
Linus Torvalds committed
1551 1552 1553 1554 1555
		if (value != 0)
			break;
		/*
		 * On multi-function devices, the user can choose
		 * to have function 1 probed before function 0.
Linus Torvalds's avatar
Linus Torvalds committed
1556
		 * Give whichever channel is the primary channel
1557
		 * the highest priority.
Linus Torvalds's avatar
Linus Torvalds committed
1558
		 */
Linus Torvalds's avatar
Linus Torvalds committed
1559
		primary_channel = (lahc->flags & AHC_PRIMARY_CHANNEL) + 'A';
1560
		value = -1;
Linus Torvalds's avatar
Linus Torvalds committed
1561
		if (lahc->channel == primary_channel)
1562
			value = 1;
Linus Torvalds's avatar
Linus Torvalds committed
1563
		break;
Linus Torvalds's avatar
Linus Torvalds committed
1564
	}
1565
#endif
Linus Torvalds's avatar
Linus Torvalds committed
1566 1567
	case AHC_EISA:
		if ((rahc->flags & AHC_BIOS_ENABLED) != 0) {
1568 1569
			value = rahc->platform_data->bios_address
			      - lahc->platform_data->bios_address; 
Linus Torvalds's avatar
Linus Torvalds committed
1570
		} else {
1571 1572
			value = rahc->bsh.ioport
			      - lahc->bsh.ioport; 
Linus Torvalds's avatar
Linus Torvalds committed
1573 1574 1575 1576 1577 1578 1579 1580
		}
		break;
	default:
		panic("ahc_softc_sort: invalid bus type");
	}
	return (value);
}

1581 1582 1583 1584 1585 1586 1587 1588 1589 1590 1591 1592 1593 1594 1595
static void
ahc_linux_setup_tag_info_global(char *p)
{
	int tags, i, j;

	tags = simple_strtoul(p + 1, NULL, 0) & 0xff;
	printf("Setting Global Tags= %d\n", tags);

	for (i = 0; i < NUM_ELEMENTS(aic7xxx_tag_info); i++) {
		for (j = 0; j < AHC_NUM_TARGETS; j++) {
			aic7xxx_tag_info[i].tag_commands[j] = tags;
		}
	}
}

1596
static void
1597
ahc_linux_setup_tag_info(u_long arg, int instance, int targ, int32_t value)
1598 1599
{

1600 1601 1602 1603 1604 1605 1606 1607 1608 1609
	if ((instance >= 0) && (targ >= 0)
	 && (instance < NUM_ELEMENTS(aic7xxx_tag_info))
	 && (targ < AHC_NUM_TARGETS)) {
		aic7xxx_tag_info[instance].tag_commands[targ] = value & 0xff;
		if (bootverbose)
			printf("tag_info[%d:%d] = %d\n", instance, targ, value);
	}
}

static void
1610
ahc_linux_setup_dv(u_long arg, int instance, int targ, int32_t value)
1611 1612 1613 1614 1615 1616 1617
{

	if ((instance >= 0)
	 && (instance < NUM_ELEMENTS(aic7xxx_dv_settings))) {
		aic7xxx_dv_settings[instance] = value;
		if (bootverbose)
			printf("dv[%d] = %d\n", instance, value);
1618 1619 1620
	}
}

Linus Torvalds's avatar
Linus Torvalds committed
1621 1622 1623
/*
 * Handle Linux boot parameters. This routine allows for assigning a value
 * to a parameter with a ':' between the parameter and the value.
Linus Torvalds's avatar
Linus Torvalds committed
1624
 * ie. aic7xxx=stpwlev:1,extended
Linus Torvalds's avatar
Linus Torvalds committed
1625
 */
1626
static int
Linus Torvalds's avatar
Linus Torvalds committed
1627 1628 1629 1630 1631 1632 1633 1634 1635 1636 1637 1638 1639
aic7xxx_setup(char *s)
{
	int	i, n;
	char   *p;
	char   *end;

	static struct {
		const char *name;
		uint32_t *flag;
	} options[] = {
		{ "extended", &aic7xxx_extended },
		{ "no_reset", &aic7xxx_no_reset },
		{ "verbose", &aic7xxx_verbose },
1640
		{ "allow_memio", &aic7xxx_allow_memio},
1641
#ifdef AHC_DEBUG
1642 1643
		{ "debug", &ahc_debug },
#endif
Linus Torvalds's avatar
Linus Torvalds committed
1644
		{ "reverse_scan", &aic7xxx_reverse_scan },
1645
		{ "no_probe", &aic7xxx_probe_eisa_vl },
1646
		{ "probe_eisa_vl", &aic7xxx_probe_eisa_vl },
Linus Torvalds's avatar
Linus Torvalds committed
1647
		{ "periodic_otag", &aic7xxx_periodic_otag },
Linus Torvalds's avatar
Linus Torvalds committed
1648 1649
		{ "pci_parity", &aic7xxx_pci_parity },
		{ "seltime", &aic7xxx_seltime },
1650
		{ "tag_info", NULL },
1651 1652
		{ "global_tag_depth", NULL },
		{ "dv", NULL }
Linus Torvalds's avatar
Linus Torvalds committed
1653 1654 1655 1656
	};

	end = strchr(s, '\0');

1657 1658 1659 1660 1661 1662
	/*
	 * XXX ia64 gcc isn't smart enough to know that NUM_ELEMENTS
	 * will never be 0 in this case.
	 */
	n = 0;

Dave Jones's avatar
Dave Jones committed
1663
	while ((p = strsep(&s, ",.")) != NULL) {
1664 1665
		if (*p == '\0')
			continue;
Linus Torvalds's avatar
Linus Torvalds committed
1666 1667
		for (i = 0; i < NUM_ELEMENTS(options); i++) {

1668 1669 1670 1671 1672 1673
			n = strlen(options[i].name);
			if (strncmp(options[i].name, p, n) == 0)
				break;
		}
		if (i == NUM_ELEMENTS(options))
			continue;
Linus Torvalds's avatar
Linus Torvalds committed
1674

1675 1676 1677 1678
		if (strncmp(p, "global_tag_depth", n) == 0) {
			ahc_linux_setup_tag_info_global(p + n);
		} else if (strncmp(p, "tag_info", n) == 0) {
			s = aic_parse_brace_option("tag_info", p + n, end,
1679
			    2, ahc_linux_setup_tag_info, 0);
1680 1681
		} else if (strncmp(p, "dv", n) == 0) {
			s = aic_parse_brace_option("dv", p + n, end, 1,
1682
			    ahc_linux_setup_dv, 0);
1683 1684 1685 1686 1687
		} else if (p[n] == ':') {
			*(options[i].flag) = simple_strtoul(p + n + 1, NULL, 0);
		} else if (strncmp(p, "verbose", n) == 0) {
			*(options[i].flag) = 1;
		} else {
1688
			*(options[i].flag) ^= 0xFFFFFFFF;
Linus Torvalds's avatar
Linus Torvalds committed
1689 1690 1691 1692 1693 1694 1695 1696 1697
		}
	}
	return 1;
}

#if LINUX_VERSION_CODE > KERNEL_VERSION(2,3,0)
__setup("aic7xxx=", aic7xxx_setup);
#endif

1698
uint32_t aic7xxx_verbose;
Linus Torvalds's avatar
Linus Torvalds committed
1699 1700

int
Linus Torvalds's avatar
Linus Torvalds committed
1701
ahc_linux_register_host(struct ahc_softc *ahc, Scsi_Host_Template *template)
Linus Torvalds's avatar
Linus Torvalds committed
1702
{
1703 1704 1705 1706
	char	 buf[80];
	struct	 Scsi_Host *host;
	char	*new_name;
	u_long	 s;
Justin T. Gibbs's avatar
Justin T. Gibbs committed
1707
	u_int	 targ_offset;
Linus Torvalds's avatar
Linus Torvalds committed
1708 1709

	template->name = ahc->description;
1710
	host = scsi_host_alloc(template, sizeof(struct ahc_softc *));
Linus Torvalds's avatar
Linus Torvalds committed
1711 1712 1713 1714
	if (host == NULL)
		return (ENOMEM);

	*((struct ahc_softc **)host->hostdata) = ahc;
Linus Torvalds's avatar
Linus Torvalds committed
1715
	ahc_lock(ahc, &s);
1716 1717
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,5,0)
	scsi_assign_lock(host, &ahc->platform_data->spin_lock);
1718 1719
#elif AHC_SCSI_HAS_HOST_LOCK != 0
	host->lock = &ahc->platform_data->spin_lock;
1720 1721
#endif
	ahc->platform_data->host = host;
Linus Torvalds's avatar
Linus Torvalds committed
1722 1723
	host->can_queue = AHC_MAX_QUEUE;
	host->cmd_per_lun = 2;
Linus Torvalds's avatar
Linus Torvalds committed
1724
	/* XXX No way to communicate the ID for multiple channels */
Linus Torvalds's avatar
Linus Torvalds committed
1725 1726 1727
	host->this_id = ahc->our_id;
	host->irq = ahc->platform_data->irq;
	host->max_id = (ahc->features & AHC_WIDE) ? 16 : 8;
Linus Torvalds's avatar
Linus Torvalds committed
1728
	host->max_lun = AHC_NUM_LUNS;
Linus Torvalds's avatar
Linus Torvalds committed
1729
	host->max_channel = (ahc->features & AHC_TWIN) ? 1 : 0;
1730
	host->sg_tablesize = AHC_NSEG;
Linus Torvalds's avatar
Linus Torvalds committed
1731
	ahc_set_unit(ahc, ahc_linux_next_unit());
Linus Torvalds's avatar
Linus Torvalds committed
1732 1733 1734 1735 1736 1737 1738
	sprintf(buf, "scsi%d", host->host_no);
	new_name = malloc(strlen(buf) + 1, M_DEVBUF, M_NOWAIT);
	if (new_name != NULL) {
		strcpy(new_name, buf);
		ahc_set_name(ahc, new_name);
	}
	host->unique_id = ahc->unit;
1739
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,4,4) && \
1740
    LINUX_VERSION_CODE  < KERNEL_VERSION(2,5,0)
Linus Torvalds's avatar
Linus Torvalds committed
1741
	scsi_set_pci_device(host, ahc->dev_softc);
Linus Torvalds's avatar
Linus Torvalds committed
1742 1743
#endif
	ahc_linux_initialize_scsi_bus(ahc);
1744 1745 1746 1747 1748 1749 1750 1751 1752 1753 1754 1755 1756 1757 1758 1759 1760 1761
	ahc_unlock(ahc, &s);
	ahc->platform_data->dv_pid = kernel_thread(ahc_linux_dv_thread, ahc, 0);
	ahc_lock(ahc, &s);
	if (ahc->platform_data->dv_pid < 0) {
		printf("%s: Failed to create DV thread, error= %d\n",
		       ahc_name(ahc), ahc->platform_data->dv_pid);
		return (-ahc->platform_data->dv_pid);
	}
	/*
	 * Initially allocate *all* of our linux target objects
	 * so that the DV thread will scan them all in parallel
	 * just after driver initialization.  Any device that
	 * does not exist will have its target object destroyed
	 * by the selection timeout handler.  In the case of a
	 * device that appears after the initial DV scan, async
	 * negotiation will occur for the first command, and DV
	 * will comence should that first command be successful.
	 */
Justin T. Gibbs's avatar
Justin T. Gibbs committed
1762 1763 1764
	for (targ_offset = 0;
	     targ_offset < host->max_id * (host->max_channel + 1);
	     targ_offset++) {
1765
		u_int channel;
Justin T. Gibbs's avatar
Justin T. Gibbs committed
1766
		u_int target;
1767 1768

		channel = 0;
Justin T. Gibbs's avatar
Justin T. Gibbs committed
1769
		target = targ_offset;
1770
		if (target > 7
Justin T. Gibbs's avatar
Justin T. Gibbs committed
1771
		 && (ahc->features & AHC_TWIN) != 0) {
1772
			channel = 1;
Justin T. Gibbs's avatar
Justin T. Gibbs committed
1773 1774
			target &= 0x7;
		}
1775 1776 1777 1778 1779 1780
		/*
		 * Skip our own ID.  Some Compaq/HP storage devices
		 * have enclosure management devices that respond to
		 * single bit selection (i.e. selecting ourselves).
		 * It is expected that either an external application
		 * or a modified kernel will be used to probe this
1781 1782 1783
		 * ID if it is appropriate.  To accommodate these
		 * installations, ahc_linux_alloc_target() will allocate
		 * for our ID if asked to do so.
1784 1785 1786 1787 1788
		 */
		if ((channel == 0 && target == ahc->our_id)
		 || (channel == 1 && target == ahc->our_id_b))
			continue;

1789 1790
		ahc_linux_alloc_target(ahc, channel, target);
	}
1791
	ahc_intr_enable(ahc, TRUE);
1792
	ahc_linux_start_dv(ahc);
Linus Torvalds's avatar
Linus Torvalds committed
1793
	ahc_unlock(ahc, &s);
1794

1795
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,5,0)
1796 1797
	scsi_add_host(host, (ahc->dev_softc ? &ahc->dev_softc->dev : NULL)); /* XXX handle failure */
	scsi_scan_host(host);
1798
#endif
Linus Torvalds's avatar
Linus Torvalds committed
1799 1800 1801
	return (0);
}

Linus Torvalds's avatar
Linus Torvalds committed
1802
uint64_t
1803
ahc_linux_get_memsize(void)
Linus Torvalds's avatar
Linus Torvalds committed
1804 1805 1806 1807
{
	struct sysinfo si;

	si_meminfo(&si);
1808
	return ((uint64_t)si.totalram << PAGE_SHIFT);
Linus Torvalds's avatar
Linus Torvalds committed
1809 1810
}

Linus Torvalds's avatar
Linus Torvalds committed
1811 1812 1813 1814 1815 1816 1817
/*
 * Find the smallest available unit number to use
 * for a new device.  We don't just use a static
 * count to handle the "repeated hot-(un)plug"
 * scenario.
 */
static int
1818
ahc_linux_next_unit(void)
Linus Torvalds's avatar
Linus Torvalds committed
1819 1820 1821 1822 1823 1824 1825 1826 1827 1828 1829 1830 1831 1832 1833 1834 1835 1836 1837 1838 1839
{
	struct ahc_softc *ahc;
	int unit;

	unit = 0;
retry:
	TAILQ_FOREACH(ahc, &ahc_tailq, links) {
		if (ahc->unit == unit) {
			unit++;
			goto retry;
		}
	}
	return (unit);
}

/*
 * Place the SCSI bus into a known state by either resetting it,
 * or forcing transfer negotiations on the next command to any
 * target.
 */
void
Linus Torvalds's avatar
Linus Torvalds committed
1840
ahc_linux_initialize_scsi_bus(struct ahc_softc *ahc)
Linus Torvalds's avatar
Linus Torvalds committed
1841 1842 1843 1844 1845 1846 1847 1848 1849 1850 1851 1852 1853 1854 1855 1856 1857 1858 1859 1860 1861 1862 1863 1864 1865 1866
{
	int i;
	int numtarg;

	i = 0;
	numtarg = 0;

	if (aic7xxx_no_reset != 0)
		ahc->flags &= ~(AHC_RESET_BUS_A|AHC_RESET_BUS_B);

	if ((ahc->flags & AHC_RESET_BUS_A) != 0)
		ahc_reset_channel(ahc, 'A', /*initiate_reset*/TRUE);
	else
		numtarg = (ahc->features & AHC_WIDE) ? 16 : 8;

	if ((ahc->features & AHC_TWIN) != 0) {

		if ((ahc->flags & AHC_RESET_BUS_B) != 0) {
			ahc_reset_channel(ahc, 'B', /*initiate_reset*/TRUE);
		} else {
			if (numtarg == 0)
				i = 8;
			numtarg += 8;
		}
	}

1867 1868 1869 1870
	/*
	 * Force negotiation to async for all targets that
	 * will not see an initial bus reset.
	 */
Linus Torvalds's avatar
Linus Torvalds committed
1871 1872 1873
	for (; i < numtarg; i++) {
		struct ahc_devinfo devinfo;
		struct ahc_initiator_tinfo *tinfo;
Linus Torvalds's avatar
Linus Torvalds committed
1874
		struct ahc_tmode_tstate *tstate;
Linus Torvalds's avatar
Linus Torvalds committed
1875 1876 1877 1878 1879 1880 1881 1882 1883 1884 1885 1886 1887 1888 1889
		u_int our_id;
		u_int target_id;
		char channel;

		channel = 'A';
		our_id = ahc->our_id;
		target_id = i;
		if (i > 7 && (ahc->features & AHC_TWIN) != 0) {
			channel = 'B';
			our_id = ahc->our_id_b;
			target_id = i % 8;
		}
		tinfo = ahc_fetch_transinfo(ahc, channel, our_id,
					    target_id, &tstate);
		ahc_compile_devinfo(&devinfo, our_id, target_id,
1890
				    CAM_LUN_WILDCARD, channel, ROLE_INITIATOR);
Linus Torvalds's avatar
Linus Torvalds committed
1891
		ahc_update_neg_request(ahc, &devinfo, tstate,
1892
				       tinfo, AHC_NEG_ALWAYS);
Linus Torvalds's avatar
Linus Torvalds committed
1893 1894 1895
	}
	/* Give the bus some time to recover */
	if ((ahc->flags & (AHC_RESET_BUS_A|AHC_RESET_BUS_B)) != 0) {
1896
		ahc_linux_freeze_simq(ahc);
Linus Torvalds's avatar
Linus Torvalds committed
1897 1898 1899 1900 1901
		init_timer(&ahc->platform_data->reset_timer);
		ahc->platform_data->reset_timer.data = (u_long)ahc;
		ahc->platform_data->reset_timer.expires =
		    jiffies + (AIC7XXX_RESET_DELAY * HZ)/1000;
		ahc->platform_data->reset_timer.function =
1902
		    ahc_linux_release_simq;
Linus Torvalds's avatar
Linus Torvalds committed
1903 1904 1905 1906 1907 1908 1909
		add_timer(&ahc->platform_data->reset_timer);
	}
}

int
ahc_platform_alloc(struct ahc_softc *ahc, void *platform_arg)
{
1910

Linus Torvalds's avatar
Linus Torvalds committed
1911 1912 1913 1914 1915 1916
	ahc->platform_data =
	    malloc(sizeof(struct ahc_platform_data), M_DEVBUF, M_NOWAIT);
	if (ahc->platform_data == NULL)
		return (ENOMEM);
	memset(ahc->platform_data, 0, sizeof(struct ahc_platform_data));
	TAILQ_INIT(&ahc->platform_data->completeq);
Linus Torvalds's avatar
Linus Torvalds committed
1917
	TAILQ_INIT(&ahc->platform_data->device_runq);
1918
	ahc->platform_data->irq = AHC_LINUX_NOIRQ;
Linus Torvalds's avatar
Linus Torvalds committed
1919
	ahc->platform_data->hw_dma_mask = 0xFFFFFFFF;
1920
	ahc_lockinit(ahc);
Linus Torvalds's avatar
Linus Torvalds committed
1921
	ahc_done_lockinit(ahc);
1922 1923 1924 1925
	init_timer(&ahc->platform_data->completeq_timer);
	ahc->platform_data->completeq_timer.data = (u_long)ahc;
	ahc->platform_data->completeq_timer.function =
	    (ahc_linux_callback_t *)ahc_linux_thread_run_complete_queue;
Linus Torvalds's avatar
Linus Torvalds committed
1926 1927
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,3,0)
	init_MUTEX_LOCKED(&ahc->platform_data->eh_sem);
1928 1929
	init_MUTEX_LOCKED(&ahc->platform_data->dv_sem);
	init_MUTEX_LOCKED(&ahc->platform_data->dv_cmd_sem);
Linus Torvalds's avatar
Linus Torvalds committed
1930 1931
#else
	ahc->platform_data->eh_sem = MUTEX_LOCKED;
1932 1933
	ahc->platform_data->dv_sem = MUTEX_LOCKED;
	ahc->platform_data->dv_cmd_sem = MUTEX_LOCKED;
1934 1935 1936 1937
#endif
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,4,0)
	tasklet_init(&ahc->platform_data->runq_tasklet, ahc_runq_tasklet,
		     (unsigned long)ahc);
Linus Torvalds's avatar
Linus Torvalds committed
1938 1939 1940
#endif
	ahc->seltime = (aic7xxx_seltime & 0x3) << 4;
	ahc->seltime_b = (aic7xxx_seltime & 0x3) << 4;
Justin T. Gibbs's avatar
Justin T. Gibbs committed
1941 1942 1943
	if (aic7xxx_pci_parity == 0)
		ahc->flags |= AHC_DISABLE_PCI_PERR;

Linus Torvalds's avatar
Linus Torvalds committed
1944 1945 1946 1947 1948 1949
	return (0);
}

void
ahc_platform_free(struct ahc_softc *ahc)
{
Justin T. Gibbs's avatar
Justin T. Gibbs committed
1950 1951 1952
	struct ahc_linux_target *targ;
	struct ahc_linux_device *dev;
	int i, j;
1953

Linus Torvalds's avatar
Linus Torvalds committed
1954
	if (ahc->platform_data != NULL) {
1955
		del_timer_sync(&ahc->platform_data->completeq_timer);
1956
		ahc_linux_kill_dv_thread(ahc);
1957 1958 1959
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,4,0)
		tasklet_kill(&ahc->platform_data->runq_tasklet);
#endif
1960
		if (ahc->platform_data->host != NULL) {
1961
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,5,0)
1962 1963
			scsi_remove_host(ahc->platform_data->host);
#endif
1964
			scsi_host_put(ahc->platform_data->host);
1965
		}
Justin T. Gibbs's avatar
Justin T. Gibbs committed
1966 1967 1968 1969 1970

		/* destroy all of the device and target objects */
		for (i = 0; i < AHC_NUM_TARGETS; i++) {
			targ = ahc->platform_data->targets[i];
			if (targ != NULL) {
1971 1972
				/* Keep target around through the loop. */
				targ->refcount++;
Justin T. Gibbs's avatar
Justin T. Gibbs committed
1973
				for (j = 0; j < AHC_NUM_LUNS; j++) {
1974 1975 1976 1977 1978

					if (targ->devices[j] == NULL)
						continue;
					dev = targ->devices[j];
					ahc_linux_free_device(ahc, dev);
Justin T. Gibbs's avatar
Justin T. Gibbs committed
1979
				}
1980 1981 1982 1983 1984 1985 1986
				/*
				 * Forcibly free the target now that
				 * all devices are gone.
				 */
				ahc_linux_free_target(ahc, targ);
 			}
 		}
Justin T. Gibbs's avatar
Justin T. Gibbs committed
1987

1988
		if (ahc->platform_data->irq != AHC_LINUX_NOIRQ)
Linus Torvalds's avatar
Linus Torvalds committed
1989 1990 1991 1992 1993 1994 1995 1996 1997 1998 1999 2000 2001 2002 2003 2004
			free_irq(ahc->platform_data->irq, ahc);
		if (ahc->tag == BUS_SPACE_PIO
		 && ahc->bsh.ioport != 0)
			release_region(ahc->bsh.ioport, 256);
		if (ahc->tag == BUS_SPACE_MEMIO
		 && ahc->bsh.maddr != NULL) {
			u_long base_addr;

			base_addr = (u_long)ahc->bsh.maddr;
			base_addr &= PAGE_MASK;
			iounmap((void *)base_addr);
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,4,0)
			release_mem_region(ahc->platform_data->mem_busaddr,
					   0x1000);
#endif
		}
2005 2006
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,4,0) && \
    LINUX_VERSION_CODE  < KERNEL_VERSION(2,5,0)
2007 2008
		/*
		 * In 2.4 we detach from the scsi midlayer before the PCI
2009 2010 2011
		 * layer invokes our remove callback.  No per-instance
		 * detach is provided, so we must reach inside the PCI
		 * subsystem's internals and detach our driver manually.
2012
		 */
Linus Torvalds's avatar
Linus Torvalds committed
2013 2014
		if (ahc->dev_softc != NULL)
			ahc->dev_softc->driver = NULL;
2015
#endif
2016
		free(ahc->platform_data, M_DEVBUF);
2017
	}
Linus Torvalds's avatar
Linus Torvalds committed
2018 2019 2020 2021 2022 2023 2024 2025 2026 2027 2028 2029
}

void
ahc_platform_freeze_devq(struct ahc_softc *ahc, struct scb *scb)
{
	ahc_platform_abort_scbs(ahc, SCB_GET_TARGET(ahc, scb),
				SCB_GET_CHANNEL(ahc, scb),
				SCB_GET_LUN(scb), SCB_LIST_NULL,
				ROLE_UNKNOWN, CAM_REQUEUE_REQ);
}

void
Linus Torvalds's avatar
Linus Torvalds committed
2030 2031
ahc_platform_set_tags(struct ahc_softc *ahc, struct ahc_devinfo *devinfo,
		      ahc_queue_alg alg)
Linus Torvalds's avatar
Linus Torvalds committed
2032 2033
{
	struct ahc_linux_device *dev;
Linus Torvalds's avatar
Linus Torvalds committed
2034 2035 2036 2037 2038 2039
	int was_queuing;
	int now_queuing;

	dev = ahc_linux_get_device(ahc, devinfo->channel - 'A',
				   devinfo->target,
				   devinfo->lun, /*alloc*/FALSE);
Linus Torvalds's avatar
Linus Torvalds committed
2040 2041
	if (dev == NULL)
		return;
Linus Torvalds's avatar
Linus Torvalds committed
2042
	was_queuing = dev->flags & (AHC_DEV_Q_BASIC|AHC_DEV_Q_TAGGED);
2043 2044 2045 2046 2047 2048 2049 2050 2051 2052 2053 2054
	switch (alg) {
	default:
	case AHC_QUEUE_NONE:
		now_queuing = 0;
		break; 
	case AHC_QUEUE_BASIC:
		now_queuing = AHC_DEV_Q_BASIC;
		break;
	case AHC_QUEUE_TAGGED:
		now_queuing = AHC_DEV_Q_TAGGED;
		break;
	}
Linus Torvalds's avatar
Linus Torvalds committed
2055
	if ((dev->flags & AHC_DEV_FREEZE_TIL_EMPTY) == 0
Linus Torvalds's avatar
Linus Torvalds committed
2056
	 && (was_queuing != now_queuing)
Linus Torvalds's avatar
Linus Torvalds committed
2057 2058 2059 2060 2061
	 && (dev->active != 0)) {
		dev->flags |= AHC_DEV_FREEZE_TIL_EMPTY;
		dev->qfrozen++;
	}

Linus Torvalds's avatar
Linus Torvalds committed
2062
	dev->flags &= ~(AHC_DEV_Q_BASIC|AHC_DEV_Q_TAGGED|AHC_DEV_PERIODIC_OTAG);
Linus Torvalds's avatar
Linus Torvalds committed
2063
	if (now_queuing) {
2064
		u_int usertags;
Linus Torvalds's avatar
Linus Torvalds committed
2065

2066
		usertags = ahc_linux_user_tagdepth(ahc, devinfo);
Linus Torvalds's avatar
Linus Torvalds committed
2067 2068 2069 2070 2071 2072
		if (!was_queuing) {
			/*
			 * Start out agressively and allow our
			 * dynamic queue depth algorithm to take
			 * care of the rest.
			 */
2073
			dev->maxtags = usertags;
Linus Torvalds's avatar
Linus Torvalds committed
2074 2075
			dev->openings = dev->maxtags - dev->active;
		}
2076 2077 2078 2079 2080 2081
		if (dev->maxtags == 0) {
			/*
			 * Queueing is disabled by the user.
			 */
			dev->openings = 1;
		} else if (alg == AHC_QUEUE_TAGGED) {
Linus Torvalds's avatar
Linus Torvalds committed
2082
			dev->flags |= AHC_DEV_Q_TAGGED;
Linus Torvalds's avatar
Linus Torvalds committed
2083 2084 2085
			if (aic7xxx_periodic_otag != 0)
				dev->flags |= AHC_DEV_PERIODIC_OTAG;
		} else
Linus Torvalds's avatar
Linus Torvalds committed
2086
			dev->flags |= AHC_DEV_Q_BASIC;
Linus Torvalds's avatar
Linus Torvalds committed
2087
	} else {
2088
		/* We can only have one opening. */
Linus Torvalds's avatar
Linus Torvalds committed
2089 2090 2091
		dev->maxtags = 0;
		dev->openings =  1 - dev->active;
	}
2092 2093 2094 2095 2096 2097 2098 2099 2100 2101 2102 2103 2104 2105 2106 2107 2108 2109 2110 2111 2112 2113 2114 2115 2116 2117 2118
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,5,0)
	if (dev->scsi_device != NULL) {
		switch ((dev->flags & (AHC_DEV_Q_BASIC|AHC_DEV_Q_TAGGED))) {
		case AHC_DEV_Q_BASIC:
			scsi_adjust_queue_depth(dev->scsi_device,
						MSG_SIMPLE_TASK,
						dev->openings + dev->active);
			break;
		case AHC_DEV_Q_TAGGED:
			scsi_adjust_queue_depth(dev->scsi_device,
						MSG_ORDERED_TASK,
						dev->openings + dev->active);
			break;
		default:
			/*
			 * We allow the OS to queue 2 untagged transactions to
			 * us at any time even though we can only execute them
			 * serially on the controller/device.  This should
			 * remove some latency.
			 */
			scsi_adjust_queue_depth(dev->scsi_device,
						/*NON-TAGGED*/0,
						/*queue depth*/2);
			break;
		}
	}
#endif
Linus Torvalds's avatar
Linus Torvalds committed
2119 2120 2121 2122 2123 2124 2125 2126 2127 2128 2129 2130 2131 2132 2133 2134 2135 2136 2137 2138 2139 2140 2141 2142 2143 2144 2145 2146 2147 2148 2149 2150 2151 2152 2153 2154
}

int
ahc_platform_abort_scbs(struct ahc_softc *ahc, int target, char channel,
			int lun, u_int tag, role_t role, uint32_t status)
{
	int chan;
	int maxchan;
	int targ;
	int maxtarg;
	int clun;
	int maxlun;
	int count;

	if (tag != SCB_LIST_NULL)
		return (0);

	chan = 0;
	if (channel != ALL_CHANNELS) {
		chan = channel - 'A';
		maxchan = chan + 1;
	} else {
		maxchan = (ahc->features & AHC_TWIN) ? 2 : 1;
	}
	targ = 0;
	if (target != CAM_TARGET_WILDCARD) {
		targ = target;
		maxtarg = targ + 1;
	} else {
		maxtarg = (ahc->features & AHC_WIDE) ? 16 : 8;
	}
	clun = 0;
	if (lun != CAM_LUN_WILDCARD) {
		clun = lun;
		maxlun = clun + 1;
	} else {
2155
		maxlun = AHC_NUM_LUNS;
Linus Torvalds's avatar
Linus Torvalds committed
2156 2157 2158 2159
	}

	count = 0;
	for (; chan < maxchan; chan++) {
2160

Linus Torvalds's avatar
Linus Torvalds committed
2161
		for (; targ < maxtarg; targ++) {
2162

Linus Torvalds's avatar
Linus Torvalds committed
2163 2164 2165 2166 2167
			for (; clun < maxlun; clun++) {
				struct ahc_linux_device *dev;
				struct ahc_busyq *busyq;
				struct ahc_cmd *acmd;

Linus Torvalds's avatar
Linus Torvalds committed
2168 2169 2170
				dev = ahc_linux_get_device(ahc, chan,
							   targ, clun,
							   /*alloc*/FALSE);
Linus Torvalds's avatar
Linus Torvalds committed
2171 2172 2173 2174 2175 2176 2177 2178 2179 2180 2181 2182
				if (dev == NULL)
					continue;

				busyq = &dev->busyq;
				while ((acmd = TAILQ_FIRST(busyq)) != NULL) {
					Scsi_Cmnd *cmd;

					cmd = &acmd_scsi_cmd(acmd);
					TAILQ_REMOVE(busyq, acmd,
						     acmd_links.tqe);
					count++;
					cmd->result = status << 16;
Linus Torvalds's avatar
Linus Torvalds committed
2183
					ahc_linux_queue_cmd_complete(ahc, cmd);
Linus Torvalds's avatar
Linus Torvalds committed
2184 2185 2186 2187 2188 2189 2190 2191 2192
				}
			}
		}
	}

	return (count);
}

static void
2193
ahc_linux_thread_run_complete_queue(struct ahc_softc *ahc)
Linus Torvalds's avatar
Linus Torvalds committed
2194
{
2195
	u_long flags;
Linus Torvalds's avatar
Linus Torvalds committed
2196

2197 2198 2199
	ahc_lock(ahc, &flags);
	del_timer(&ahc->platform_data->completeq_timer);
	ahc->platform_data->flags &= ~AHC_RUN_CMPLT_Q_TIMER;
2200
	ahc_linux_run_complete_queue(ahc);
2201
	ahc_unlock(ahc, &flags);
Linus Torvalds's avatar
Linus Torvalds committed
2202 2203
}

2204 2205
static void
ahc_linux_start_dv(struct ahc_softc *ahc)
Linus Torvalds's avatar
Linus Torvalds committed
2206 2207
{

2208 2209
	/*
	 * Freeze the simq and signal ahc_linux_queue to not let any
2210
	 * more commands through.
2211 2212 2213 2214 2215 2216
	 */
	if ((ahc->platform_data->flags & AHC_DV_ACTIVE) == 0) {
#ifdef AHC_DEBUG
		if (ahc_debug & AHC_SHOW_DV)
			printf("%s: Waking DV thread\n", ahc_name(ahc));
#endif
Linus Torvalds's avatar
Linus Torvalds committed
2217

2218 2219
		ahc->platform_data->flags |= AHC_DV_ACTIVE;
		ahc_linux_freeze_simq(ahc);
Linus Torvalds's avatar
Linus Torvalds committed
2220

2221 2222
		/* Wake up the DV kthread */
		up(&ahc->platform_data->dv_sem);
Linus Torvalds's avatar
Linus Torvalds committed
2223
	}
2224 2225
}

2226 2227 2228 2229 2230 2231 2232 2233 2234 2235 2236 2237 2238 2239 2240 2241 2242 2243 2244 2245 2246 2247 2248 2249 2250 2251 2252 2253 2254 2255 2256 2257 2258 2259 2260 2261 2262 2263 2264 2265 2266 2267
static void
ahc_linux_kill_dv_thread(struct ahc_softc *ahc)
{
	u_long s;

	ahc_lock(ahc, &s);
	if (ahc->platform_data->dv_pid != 0) {
		ahc->platform_data->flags |= AHC_DV_SHUTDOWN;
		ahc_unlock(ahc, &s);
		up(&ahc->platform_data->dv_sem);

		/*
		 * Use the eh_sem as an indicator that the
		 * dv thread is exiting.  Note that the dv
		 * thread must still return after performing
		 * the up on our semaphore before it has
		 * completely exited this module.  Unfortunately,
		 * there seems to be no easy way to wait for the
		 * exit of a thread for which you are not the
		 * parent (dv threads are parented by init).
		 * Cross your fingers...
		 */
		down(&ahc->platform_data->eh_sem);

		/*
		 * Mark the dv thread as already dead.  This
		 * avoids attempting to kill it a second time.
		 * This is necessary because we must kill the
		 * DV thread before calling ahc_free() in the
		 * module shutdown case to avoid bogus locking
		 * in the SCSI mid-layer, but we ahc_free() is
		 * called without killing the DV thread in the
		 * instance detach case, so ahc_platform_free()
		 * calls us again to verify that the DV thread
		 * is dead.
		 */
		ahc->platform_data->dv_pid = 0;
	} else {
		ahc_unlock(ahc, &s);
	}
}

2268 2269
static int
ahc_linux_dv_thread(void *data)
2270
{
2271 2272 2273
	struct	ahc_softc *ahc;
	int	target;
	u_long	s;
2274

2275
	ahc = (struct ahc_softc *)data;
2276

2277 2278 2279 2280 2281
#ifdef AHC_DEBUG
	if (ahc_debug & AHC_SHOW_DV)
		printf("Launching DV Thread\n");
#endif

2282 2283 2284 2285
	/*
	 * Complete thread creation.
	 */
	lock_kernel();
2286 2287 2288 2289 2290 2291 2292 2293 2294
#if LINUX_VERSION_CODE < KERNEL_VERSION(2,5,0)
	/*
	 * Don't care about any signals.
	 */
	siginitsetinv(&current->blocked, 0);

	daemonize();
	sprintf(current->comm, "ahc_dv_%d", ahc->unit);
#else
2295
	daemonize("ahc_dv_%d", ahc->unit);
2296
	current->flags |= PF_FREEZE;
2297
#endif
2298 2299
	unlock_kernel();

2300
	while (1) {
2301 2302 2303 2304 2305
		/*
		 * Use down_interruptible() rather than down() to
		 * avoid inclusion in the load average.
		 */
		down_interruptible(&ahc->platform_data->dv_sem);
2306 2307 2308 2309 2310

		/* Check to see if we've been signaled to exit */
		ahc_lock(ahc, &s);
		if ((ahc->platform_data->flags & AHC_DV_SHUTDOWN) != 0) {
			ahc_unlock(ahc, &s);
2311
			break;
2312 2313 2314 2315 2316 2317 2318 2319 2320 2321 2322 2323 2324 2325 2326 2327
		}
		ahc_unlock(ahc, &s);

#ifdef AHC_DEBUG
		if (ahc_debug & AHC_SHOW_DV)
			printf("%s: Beginning Domain Validation\n",
			       ahc_name(ahc));
#endif

		/*
		 * Wait for any pending commands to drain before proceeding.
		 */
		ahc_lock(ahc, &s);
		while (LIST_FIRST(&ahc->pending_scbs) != NULL) {
			ahc->platform_data->flags |= AHC_DV_WAIT_SIMQ_EMPTY;
			ahc_unlock(ahc, &s);
2328
			down_interruptible(&ahc->platform_data->dv_sem);
2329 2330 2331 2332 2333 2334 2335 2336 2337 2338
			ahc_lock(ahc, &s);
		}

		/*
		 * Wait for the SIMQ to be released so that DV is the
		 * only reason the queue is frozen.
		 */
		while (AHC_DV_SIMQ_FROZEN(ahc) == 0) {
			ahc->platform_data->flags |= AHC_DV_WAIT_SIMQ_RELEASE;
			ahc_unlock(ahc, &s);
2339
			down_interruptible(&ahc->platform_data->dv_sem);
2340 2341 2342 2343 2344 2345 2346 2347 2348 2349 2350 2351 2352 2353 2354 2355 2356
			ahc_lock(ahc, &s);
		}
		ahc_unlock(ahc, &s);

		for (target = 0; target < AHC_NUM_TARGETS; target++)
			ahc_linux_dv_target(ahc, target);

		ahc_lock(ahc, &s);
		ahc->platform_data->flags &= ~AHC_DV_ACTIVE;
		ahc_unlock(ahc, &s);

		/*
		 * Release the SIMQ so that normal commands are
		 * allowed to continue on the bus.
		 */
		ahc_linux_release_simq((u_long)ahc);
	}
2357
	up(&ahc->platform_data->eh_sem);
2358 2359 2360 2361 2362 2363 2364 2365 2366 2367 2368 2369 2370 2371 2372 2373 2374 2375 2376 2377 2378 2379 2380 2381 2382 2383 2384 2385 2386 2387 2388 2389 2390 2391 2392 2393
	return (0);
}

#define AHC_LINUX_DV_INQ_SHORT_LEN	36
#define AHC_LINUX_DV_INQ_LEN		256
#define AHC_LINUX_DV_TIMEOUT		(HZ / 4)

#define AHC_SET_DV_STATE(ahc, targ, newstate) \
	ahc_set_dv_state(ahc, targ, newstate, __LINE__)

static __inline void
ahc_set_dv_state(struct ahc_softc *ahc, struct ahc_linux_target *targ,
		 ahc_dv_state newstate, u_int line)
{
	ahc_dv_state oldstate;

	oldstate = targ->dv_state;
#ifdef AHC_DEBUG
	if (ahc_debug & AHC_SHOW_DV)
		printf("%s:%d: Going from state %d to state %d\n",
		       ahc_name(ahc), line, oldstate, newstate);
#endif

	if (oldstate == newstate)
		targ->dv_state_retry++;
	else
		targ->dv_state_retry = 0;
	targ->dv_state = newstate;
}

static void
ahc_linux_dv_target(struct ahc_softc *ahc, u_int target_offset)
{
	struct	 ahc_devinfo devinfo;
	struct	 ahc_linux_target *targ;
	struct	 scsi_cmnd *cmd;
Justin T. Gibbs's avatar
Justin T. Gibbs committed
2394
	struct	 scsi_device *scsi_dev;
2395 2396 2397 2398 2399 2400 2401 2402 2403 2404 2405 2406 2407 2408 2409
	struct	 scsi_sense_data *sense;
	uint8_t *buffer;
	u_long	 s;
	u_int	 timeout;
	int	 echo_size;

	sense = NULL;
	buffer = NULL;
	echo_size = 0;
	ahc_lock(ahc, &s);
	targ = ahc->platform_data->targets[target_offset];
	if (targ == NULL || (targ->flags & AHC_DV_REQUIRED) == 0) {
		ahc_unlock(ahc, &s);
		return;
	}
Justin T. Gibbs's avatar
Justin T. Gibbs committed
2410 2411 2412 2413
	ahc_compile_devinfo(&devinfo,
			    targ->channel == 0 ? ahc->our_id : ahc->our_id_b,
			    targ->target, /*lun*/0, targ->channel + 'A',
			    ROLE_INITIATOR);
2414 2415 2416 2417 2418 2419 2420 2421 2422 2423
#ifdef AHC_DEBUG
	if (ahc_debug & AHC_SHOW_DV) {
		ahc_print_devinfo(ahc, &devinfo);
		printf("Performing DV\n");
	}
#endif

	ahc_unlock(ahc, &s);

	cmd = malloc(sizeof(struct scsi_cmnd), M_DEVBUF, M_WAITOK);
Justin T. Gibbs's avatar
Justin T. Gibbs committed
2424 2425 2426 2427 2428 2429
	scsi_dev = malloc(sizeof(struct scsi_device), M_DEVBUF, M_WAITOK);
	scsi_dev->host = ahc->platform_data->host;
	scsi_dev->id = devinfo.target;
	scsi_dev->lun = devinfo.lun;
	scsi_dev->channel = devinfo.channel - 'A';
	ahc->platform_data->dv_scsi_dev = scsi_dev;
2430 2431 2432 2433 2434 2435 2436 2437 2438 2439 2440 2441 2442 2443 2444 2445 2446 2447 2448 2449

	AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_INQ_SHORT_ASYNC);

	while (targ->dv_state != AHC_DV_STATE_EXIT) {
		timeout = AHC_LINUX_DV_TIMEOUT;
		switch (targ->dv_state) {
		case AHC_DV_STATE_INQ_SHORT_ASYNC:
		case AHC_DV_STATE_INQ_ASYNC:
		case AHC_DV_STATE_INQ_ASYNC_VERIFY:
			/*
			 * Set things to async narrow to reduce the
			 * chance that the INQ will fail.
			 */
			ahc_lock(ahc, &s);
			ahc_set_syncrate(ahc, &devinfo, NULL, 0, 0, 0,
					 AHC_TRANS_GOAL, /*paused*/FALSE);
			ahc_set_width(ahc, &devinfo, MSG_EXT_WDTR_BUS_8_BIT,
				      AHC_TRANS_GOAL, /*paused*/FALSE);
			ahc_unlock(ahc, &s);
			timeout = 10 * HZ;
2450
			targ->flags &= ~AHC_INQ_VALID;
2451 2452 2453 2454 2455 2456 2457 2458 2459 2460 2461 2462 2463 2464
			/* FALLTHROUGH */
		case AHC_DV_STATE_INQ_VERIFY:
		{
			u_int inq_len;

			if (targ->dv_state == AHC_DV_STATE_INQ_SHORT_ASYNC)
				inq_len = AHC_LINUX_DV_INQ_SHORT_LEN;
			else
				inq_len = targ->inq_data->additional_length + 5;
			ahc_linux_dv_inq(ahc, cmd, &devinfo, targ, inq_len);
			break;
		}
		case AHC_DV_STATE_TUR:
		case AHC_DV_STATE_BUSY:
2465
			timeout = 5 * HZ;
2466 2467 2468 2469 2470 2471 2472 2473 2474 2475 2476 2477 2478 2479 2480 2481 2482 2483 2484 2485 2486 2487 2488 2489 2490 2491 2492 2493 2494 2495 2496 2497 2498 2499 2500 2501 2502 2503 2504 2505
			ahc_linux_dv_tur(ahc, cmd, &devinfo);
			break;
		case AHC_DV_STATE_REBD:
			ahc_linux_dv_rebd(ahc, cmd, &devinfo, targ);
			break;
		case AHC_DV_STATE_WEB:
			ahc_linux_dv_web(ahc, cmd, &devinfo, targ);
			break;

		case AHC_DV_STATE_REB:
			ahc_linux_dv_reb(ahc, cmd, &devinfo, targ);
			break;

		case AHC_DV_STATE_SU:
			ahc_linux_dv_su(ahc, cmd, &devinfo, targ);
			timeout = 50 * HZ;
			break;

		default:
			ahc_print_devinfo(ahc, &devinfo);
			printf("Unknown DV state %d\n", targ->dv_state);
			goto out;
		}

		/* Queue the command and wait for it to complete */
		/* Abuse eh_timeout in the scsi_cmnd struct for our purposes */
		init_timer(&cmd->eh_timeout);
#ifdef AHC_DEBUG
		if ((ahc_debug & AHC_SHOW_MESSAGES) != 0)
			/*
			 * All of the printfs during negotiation
			 * really slow down the negotiation.
			 * Add a bit of time just to be safe.
			 */
			timeout += HZ;
#endif
		scsi_add_timer(cmd, timeout, ahc_linux_dv_timeout);
		/*
		 * In 2.5.X, it is assumed that all calls from the
		 * "midlayer" (which we are emulating) will have the
2506 2507
		 * ahc host lock held.  For other kernels, the
		 * io_request_lock must be held.
2508
		 */
2509
#if AHC_SCSI_HAS_HOST_LOCK != 0
2510
		ahc_lock(ahc, &s);
2511 2512
#else
		spin_lock_irqsave(&io_request_lock, s);
2513 2514
#endif
		ahc_linux_queue(cmd, ahc_linux_dv_complete);
2515
#if AHC_SCSI_HAS_HOST_LOCK != 0
2516
		ahc_unlock(ahc, &s);
2517 2518
#else
		spin_unlock_irqrestore(&io_request_lock, s);
2519
#endif
2520
		down_interruptible(&ahc->platform_data->dv_cmd_sem);
2521 2522 2523 2524 2525 2526 2527 2528
		/*
		 * Wait for the SIMQ to be released so that DV is the
		 * only reason the queue is frozen.
		 */
		ahc_lock(ahc, &s);
		while (AHC_DV_SIMQ_FROZEN(ahc) == 0) {
			ahc->platform_data->flags |= AHC_DV_WAIT_SIMQ_RELEASE;
			ahc_unlock(ahc, &s);
2529
			down_interruptible(&ahc->platform_data->dv_sem);
2530 2531 2532 2533 2534 2535 2536 2537
			ahc_lock(ahc, &s);
		}
		ahc_unlock(ahc, &s);

		ahc_linux_dv_transition(ahc, cmd, &devinfo, targ);
	}

out:
2538 2539 2540 2541 2542 2543 2544 2545 2546 2547 2548 2549 2550 2551 2552 2553 2554 2555 2556
	if ((targ->flags & AHC_INQ_VALID) != 0
	 && ahc_linux_get_device(ahc, devinfo.channel - 'A',
				 devinfo.target, devinfo.lun,
				 /*alloc*/FALSE) == NULL) {
		/*
		 * The DV state machine failed to configure this device.  
		 * This is normal if DV is disabled.  Since we have inquiry
		 * data, filter it and use the "optimistic" negotiation
		 * parameters found in the inquiry string.
		 */
		ahc_linux_filter_inquiry(ahc, &devinfo);
		if ((targ->flags & (AHC_BASIC_DV|AHC_ENHANCED_DV)) != 0) {
			ahc_print_devinfo(ahc, &devinfo);
			printf("DV failed to configure device.  "
			       "Please file a bug report against "
			       "this driver.\n");
		}
	}

2557 2558 2559
	if (cmd != NULL)
		free(cmd, M_DEVBUF);

Justin T. Gibbs's avatar
Justin T. Gibbs committed
2560 2561 2562 2563 2564
	if (ahc->platform_data->dv_scsi_dev != NULL) {
		free(ahc->platform_data->dv_scsi_dev, M_DEVBUF);
		ahc->platform_data->dv_scsi_dev = NULL;
	}

2565
	ahc_lock(ahc, &s);
2566
	if (targ->dv_buffer != NULL) {
2567
		free(targ->dv_buffer, M_DEVBUF);
2568 2569 2570
		targ->dv_buffer = NULL;
	}
	if (targ->dv_buffer1 != NULL) {
2571
		free(targ->dv_buffer1, M_DEVBUF);
2572 2573
		targ->dv_buffer1 = NULL;
	}
2574 2575 2576 2577 2578 2579 2580 2581 2582 2583 2584 2585 2586
	targ->flags &= ~AHC_DV_REQUIRED;
	if (targ->refcount == 0)
		ahc_linux_free_target(ahc, targ);
	ahc_unlock(ahc, &s);
}

static void
ahc_linux_dv_transition(struct ahc_softc *ahc, struct scsi_cmnd *cmd,
			struct ahc_devinfo *devinfo,
			struct ahc_linux_target *targ)
{
	u_int32_t status;

2587 2588 2589
	status = aic_error_action(cmd, targ->inq_data,
				  ahc_cmd_get_transaction_status(cmd),
				  ahc_cmd_get_scsi_status(cmd));
2590 2591 2592 2593 2594 2595 2596 2597 2598 2599 2600 2601 2602 2603 2604 2605 2606 2607 2608 2609 2610 2611 2612 2613 2614 2615 2616 2617 2618 2619 2620 2621 2622 2623 2624 2625 2626 2627 2628 2629 2630 2631 2632 2633 2634 2635 2636 2637 2638
	
#ifdef AHC_DEBUG
	if (ahc_debug & AHC_SHOW_DV) {
		ahc_print_devinfo(ahc, devinfo);
		printf("Entering ahc_linux_dv_transition, state= %d, "
		       "status= 0x%x, cmd->result= 0x%x\n", targ->dv_state,
		       status, cmd->result);
	}
#endif

	switch (targ->dv_state) {
	case AHC_DV_STATE_INQ_SHORT_ASYNC:
	case AHC_DV_STATE_INQ_ASYNC:
		switch (status & SS_MASK) {
		case SS_NOP:
		{
			AHC_SET_DV_STATE(ahc, targ, targ->dv_state+1);
			break;
		}
		case SS_INQ_REFRESH:
			AHC_SET_DV_STATE(ahc, targ,
					 AHC_DV_STATE_INQ_SHORT_ASYNC);
			break;
		case SS_TUR:
		case SS_RETRY:
			AHC_SET_DV_STATE(ahc, targ, targ->dv_state);
			if (ahc_cmd_get_transaction_status(cmd)
			 == CAM_REQUEUE_REQ)
				targ->dv_state_retry--;
			if ((status & SS_ERRMASK) == EBUSY)
				AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_BUSY);
			if (targ->dv_state_retry < 10)
				break;
			/* FALLTHROUGH */
		default:
			AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_EXIT);
#ifdef AHC_DEBUG
			if (ahc_debug & AHC_SHOW_DV) {
				ahc_print_devinfo(ahc, devinfo);
				printf("Failed DV inquiry, skipping\n");
			}
#endif
			break;
		}
		break;
	case AHC_DV_STATE_INQ_ASYNC_VERIFY:
		switch (status & SS_MASK) {
		case SS_NOP:
		{
2639
			u_int xportflags;
2640 2641 2642 2643 2644 2645 2646 2647 2648 2649 2650 2651 2652
			u_int spi3data;

			if (memcmp(targ->inq_data, targ->dv_buffer,
				   AHC_LINUX_DV_INQ_LEN) != 0) {
				/*
				 * Inquiry data must have changed.
				 * Try from the top again.
				 */
				AHC_SET_DV_STATE(ahc, targ,
						 AHC_DV_STATE_INQ_SHORT_ASYNC);
				break;
			}

2653 2654 2655
			AHC_SET_DV_STATE(ahc, targ, targ->dv_state+1);
			targ->flags |= AHC_INQ_VALID;
			if (ahc_linux_user_dv_setting(ahc) == 0)
2656 2657
				break;

2658 2659 2660 2661
			xportflags = targ->inq_data->flags;
			if ((xportflags & (SID_Sync|SID_WBus16)) == 0)
				break;

2662 2663 2664 2665 2666
			spi3data = targ->inq_data->spi3data;
			switch (spi3data & SID_SPI_CLOCK_DT_ST) {
			default:
			case SID_SPI_CLOCK_ST:
				/* Assume only basic DV is supported. */
2667
				targ->flags |= AHC_BASIC_DV;
2668 2669 2670
				break;
			case SID_SPI_CLOCK_DT:
			case SID_SPI_CLOCK_DT_ST:
2671
				targ->flags |= AHC_ENHANCED_DV;
2672 2673 2674 2675 2676 2677 2678 2679 2680 2681 2682 2683 2684 2685 2686 2687 2688 2689 2690 2691 2692 2693 2694 2695 2696 2697 2698 2699 2700 2701 2702 2703 2704 2705 2706 2707 2708 2709 2710 2711 2712
				break;
			}
			break;
		}
		case SS_INQ_REFRESH:
			AHC_SET_DV_STATE(ahc, targ,
					 AHC_DV_STATE_INQ_SHORT_ASYNC);
			break;
		case SS_TUR:
		case SS_RETRY:
			AHC_SET_DV_STATE(ahc, targ, targ->dv_state);
			if (ahc_cmd_get_transaction_status(cmd)
			 == CAM_REQUEUE_REQ)
				targ->dv_state_retry--;

			if ((status & SS_ERRMASK) == EBUSY)
				AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_BUSY);
			if (targ->dv_state_retry < 10)
				break;
			/* FALLTHROUGH */
		default:
			AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_EXIT);
#ifdef AHC_DEBUG
			if (ahc_debug & AHC_SHOW_DV) {
				ahc_print_devinfo(ahc, devinfo);
				printf("Failed DV inquiry, skipping\n");
			}
#endif
			break;
		}
		break;
	case AHC_DV_STATE_INQ_VERIFY:
		switch (status & SS_MASK) {
		case SS_NOP:
		{

			if (memcmp(targ->inq_data, targ->dv_buffer,
				   AHC_LINUX_DV_INQ_LEN) == 0) {
				AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_EXIT);
				break;
			}
2713 2714 2715 2716 2717 2718 2719 2720 2721 2722 2723 2724 2725 2726 2727 2728
#ifdef AHC_DEBUG
			if (ahc_debug & AHC_SHOW_DV) {
				int i;

				ahc_print_devinfo(ahc, devinfo);
				printf("Inquiry buffer mismatch:");
				for (i = 0; i < AHC_LINUX_DV_INQ_LEN; i++) {
					if ((i & 0xF) == 0)
						printf("\n        ");
					printf("0x%x:0x0%x ",
					       ((uint8_t *)targ->inq_data)[i], 
					       targ->dv_buffer[i]);
				}
				printf("\n");
			}
#endif
2729 2730 2731 2732 2733 2734 2735 2736 2737 2738 2739 2740 2741 2742 2743 2744 2745 2746 2747 2748 2749 2750 2751 2752 2753 2754 2755 2756 2757 2758 2759 2760 2761 2762 2763 2764 2765 2766 2767 2768 2769 2770 2771 2772 2773 2774 2775 2776 2777 2778 2779 2780 2781 2782

			if (ahc_linux_fallback(ahc, devinfo) != 0) {
				AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_EXIT);
				break;
			}
			/*
			 * Do not count "falling back"
			 * against our retries.
			 */
			targ->dv_state_retry = 0;
			AHC_SET_DV_STATE(ahc, targ, targ->dv_state);
			break;
		}
		case SS_INQ_REFRESH:
			AHC_SET_DV_STATE(ahc, targ,
					 AHC_DV_STATE_INQ_SHORT_ASYNC);
			break;
		case SS_TUR:
		case SS_RETRY:
			AHC_SET_DV_STATE(ahc, targ, targ->dv_state);
			if (ahc_cmd_get_transaction_status(cmd)
			 == CAM_REQUEUE_REQ) {
				targ->dv_state_retry--;
			} else if ((status & SSQ_FALLBACK) != 0) {
				if (ahc_linux_fallback(ahc, devinfo) != 0) {
					AHC_SET_DV_STATE(ahc, targ,
							 AHC_DV_STATE_EXIT);
					break;
				}
				/*
				 * Do not count "falling back"
				 * against our retries.
				 */
				targ->dv_state_retry = 0;
			} else if ((status & SS_ERRMASK) == EBUSY)
				AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_BUSY);
			if (targ->dv_state_retry < 10)
				break;
			/* FALLTHROUGH */
		default:
			AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_EXIT);
#ifdef AHC_DEBUG
			if (ahc_debug & AHC_SHOW_DV) {
				ahc_print_devinfo(ahc, devinfo);
				printf("Failed DV inquiry, skipping\n");
			}
#endif
			break;
		}
		break;

	case AHC_DV_STATE_TUR:
		switch (status & SS_MASK) {
		case SS_NOP:
2783 2784 2785 2786 2787 2788 2789 2790 2791
			if ((targ->flags & AHC_BASIC_DV) != 0) {
				ahc_linux_filter_inquiry(ahc, devinfo);
				AHC_SET_DV_STATE(ahc, targ,
						 AHC_DV_STATE_INQ_VERIFY);
			} else if ((targ->flags & AHC_ENHANCED_DV) != 0) {
				AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_REBD);
			} else {
				AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_EXIT);
			}
2792 2793 2794 2795 2796 2797 2798 2799 2800 2801 2802 2803 2804 2805 2806 2807 2808 2809 2810 2811 2812 2813 2814 2815 2816 2817 2818 2819 2820 2821 2822 2823 2824 2825
			break;
		case SS_RETRY:
		case SS_TUR:
			if ((status & SS_ERRMASK) == EBUSY) {
				AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_BUSY);
				break;
			}
			AHC_SET_DV_STATE(ahc, targ, targ->dv_state);
			if (ahc_cmd_get_transaction_status(cmd)
			 == CAM_REQUEUE_REQ) {
				targ->dv_state_retry--;
			} else if ((status & SSQ_FALLBACK) != 0) {
				if (ahc_linux_fallback(ahc, devinfo) != 0) {
					AHC_SET_DV_STATE(ahc, targ,
							 AHC_DV_STATE_EXIT);
					break;
				}
				/*
				 * Do not count "falling back"
				 * against our retries.
				 */
				targ->dv_state_retry = 0;
			}
			if (targ->dv_state_retry >= 10) {
#ifdef AHC_DEBUG
				if (ahc_debug & AHC_SHOW_DV) {
					ahc_print_devinfo(ahc, devinfo);
					printf("DV TUR reties exhausted\n");
				}
#endif
				AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_EXIT);
				break;
			}
			if (status & SSQ_DELAY)
2826
				ssleep(1);
2827 2828 2829 2830 2831 2832 2833 2834 2835 2836 2837 2838 2839 2840 2841 2842 2843 2844 2845 2846 2847 2848 2849 2850 2851 2852 2853 2854 2855 2856 2857 2858 2859 2860 2861 2862 2863 2864 2865 2866 2867 2868 2869 2870 2871 2872 2873 2874 2875 2876 2877 2878 2879 2880 2881 2882 2883 2884 2885 2886 2887 2888 2889 2890 2891 2892 2893 2894 2895 2896 2897 2898 2899 2900 2901 2902 2903 2904 2905 2906 2907 2908 2909 2910 2911 2912 2913 2914 2915 2916 2917 2918 2919 2920 2921 2922 2923 2924 2925 2926 2927 2928 2929 2930 2931 2932 2933 2934 2935 2936 2937 2938 2939 2940 2941 2942 2943 2944 2945 2946 2947 2948 2949 2950 2951 2952 2953 2954 2955 2956 2957 2958 2959 2960 2961 2962 2963 2964 2965 2966 2967 2968 2969 2970 2971 2972 2973 2974 2975 2976 2977 2978 2979 2980 2981 2982 2983 2984 2985

			break;
		case SS_START:
			AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_SU);
			break;
		case SS_INQ_REFRESH:
			AHC_SET_DV_STATE(ahc, targ,
					 AHC_DV_STATE_INQ_SHORT_ASYNC);
			break;
		default:
			AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_EXIT);
			break;
		}
		break;

	case AHC_DV_STATE_REBD:
		switch (status & SS_MASK) {
		case SS_NOP:
		{
			uint32_t echo_size;

			AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_WEB);
			echo_size = scsi_3btoul(&targ->dv_buffer[1]);
			echo_size &= 0x1FFF;
#ifdef AHC_DEBUG
			if (ahc_debug & AHC_SHOW_DV) {
				ahc_print_devinfo(ahc, devinfo);
				printf("Echo buffer size= %d\n", echo_size);
			}
#endif
			if (echo_size == 0) {
				AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_EXIT);
				break;
			}

			/* Generate the buffer pattern */
			targ->dv_echo_size = echo_size;
			ahc_linux_generate_dv_pattern(targ);
			/*
			 * Setup initial negotiation values.
			 */
			ahc_linux_filter_inquiry(ahc, devinfo);
			break;
		}
		case SS_INQ_REFRESH:
			AHC_SET_DV_STATE(ahc, targ,
					 AHC_DV_STATE_INQ_SHORT_ASYNC);
			break;
		case SS_RETRY:
			AHC_SET_DV_STATE(ahc, targ, targ->dv_state);
			if (ahc_cmd_get_transaction_status(cmd)
			 == CAM_REQUEUE_REQ)
				targ->dv_state_retry--;
			if (targ->dv_state_retry <= 10)
				break;
#ifdef AHC_DEBUG
			if (ahc_debug & AHC_SHOW_DV) {
				ahc_print_devinfo(ahc, devinfo);
				printf("DV REBD reties exhausted\n");
			}
#endif
			/* FALLTHROUGH */
		case SS_FATAL:
		default:
			/*
			 * Setup initial negotiation values
			 * and try level 1 DV.
			 */
			ahc_linux_filter_inquiry(ahc, devinfo);
			AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_INQ_VERIFY);
			targ->dv_echo_size = 0;
			break;
		}
		break;

	case AHC_DV_STATE_WEB:
		switch (status & SS_MASK) {
		case SS_NOP:
			AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_REB);
			break;
		case SS_INQ_REFRESH:
			AHC_SET_DV_STATE(ahc, targ,
					 AHC_DV_STATE_INQ_SHORT_ASYNC);
			break;
		case SS_RETRY:
			AHC_SET_DV_STATE(ahc, targ, targ->dv_state);
			if (ahc_cmd_get_transaction_status(cmd)
			 == CAM_REQUEUE_REQ) {
				targ->dv_state_retry--;
			} else if ((status & SSQ_FALLBACK) != 0) {
				if (ahc_linux_fallback(ahc, devinfo) != 0) {
					AHC_SET_DV_STATE(ahc, targ,
							 AHC_DV_STATE_EXIT);
					break;
				}
				/*
				 * Do not count "falling back"
				 * against our retries.
				 */
				targ->dv_state_retry = 0;
			}
			if (targ->dv_state_retry <= 10)
				break;
			/* FALLTHROUGH */
#ifdef AHC_DEBUG
			if (ahc_debug & AHC_SHOW_DV) {
				ahc_print_devinfo(ahc, devinfo);
				printf("DV WEB reties exhausted\n");
			}
#endif
		default:
			AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_EXIT);
			break;
		}
		break;

	case AHC_DV_STATE_REB:
		switch (status & SS_MASK) {
		case SS_NOP:
			if (memcmp(targ->dv_buffer, targ->dv_buffer1,
				   targ->dv_echo_size) != 0) {
				if (ahc_linux_fallback(ahc, devinfo) != 0)
					AHC_SET_DV_STATE(ahc, targ,
							 AHC_DV_STATE_EXIT);
				else
					AHC_SET_DV_STATE(ahc, targ,
							 AHC_DV_STATE_WEB);
				break;
			}
			
			if (targ->dv_buffer != NULL) {
				free(targ->dv_buffer, M_DEVBUF);
				targ->dv_buffer = NULL;
			}
			if (targ->dv_buffer1 != NULL) {
				free(targ->dv_buffer1, M_DEVBUF);
				targ->dv_buffer1 = NULL;
			}
			AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_EXIT);
			break;
		case SS_INQ_REFRESH:
			AHC_SET_DV_STATE(ahc, targ,
					 AHC_DV_STATE_INQ_SHORT_ASYNC);
			break;
		case SS_RETRY:
			AHC_SET_DV_STATE(ahc, targ, targ->dv_state);
			if (ahc_cmd_get_transaction_status(cmd)
			 == CAM_REQUEUE_REQ) {
				targ->dv_state_retry--;
			} else if ((status & SSQ_FALLBACK) != 0) {
				if (ahc_linux_fallback(ahc, devinfo) != 0) {
					AHC_SET_DV_STATE(ahc, targ,
							 AHC_DV_STATE_EXIT);
					break;
				}
				AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_WEB);
			}
			if (targ->dv_state_retry <= 10) {
				if ((status & (SSQ_DELAY_RANDOM|SSQ_DELAY))!= 0)
2986
					msleep(ahc->our_id*1000/10);
2987 2988 2989 2990 2991 2992 2993 2994 2995 2996 2997 2998 2999 3000 3001 3002 3003 3004 3005 3006 3007 3008 3009 3010 3011 3012 3013 3014 3015 3016 3017 3018 3019 3020 3021 3022 3023 3024 3025 3026 3027 3028 3029
				break;
			}
#ifdef AHC_DEBUG
			if (ahc_debug & AHC_SHOW_DV) {
				ahc_print_devinfo(ahc, devinfo);
				printf("DV REB reties exhausted\n");
			}
#endif
			/* FALLTHROUGH */
		default:
			AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_EXIT);
			break;
		}
		break;

	case AHC_DV_STATE_SU:
		switch (status & SS_MASK) {
		case SS_NOP:
		case SS_INQ_REFRESH:
			AHC_SET_DV_STATE(ahc, targ,
					 AHC_DV_STATE_INQ_SHORT_ASYNC);
			break;
		default:
			AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_EXIT);
			break;
		}
		break;

	case AHC_DV_STATE_BUSY:
		switch (status & SS_MASK) {
		case SS_NOP:
		case SS_INQ_REFRESH:
			AHC_SET_DV_STATE(ahc, targ,
					 AHC_DV_STATE_INQ_SHORT_ASYNC);
			break;
		case SS_TUR:
		case SS_RETRY:
			AHC_SET_DV_STATE(ahc, targ, targ->dv_state);
			if (ahc_cmd_get_transaction_status(cmd)
			 == CAM_REQUEUE_REQ) {
				targ->dv_state_retry--;
			} else if (targ->dv_state_retry < 60) {
				if ((status & SSQ_DELAY) != 0)
3030
					ssleep(1);
3031 3032 3033 3034 3035 3036 3037 3038 3039 3040 3041 3042 3043 3044 3045 3046 3047 3048 3049 3050 3051 3052 3053 3054 3055 3056 3057 3058 3059
			} else {
#ifdef AHC_DEBUG
				if (ahc_debug & AHC_SHOW_DV) {
					ahc_print_devinfo(ahc, devinfo);
					printf("DV BUSY reties exhausted\n");
				}
#endif
				AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_EXIT);
			}
			break;
		default:
			AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_EXIT);
			break;
		}
		break;

	default:
		printf("%s: Invalid DV completion state %d\n", ahc_name(ahc),
		       targ->dv_state);
		AHC_SET_DV_STATE(ahc, targ, AHC_DV_STATE_EXIT);
		break;
	}
}

static void
ahc_linux_dv_fill_cmd(struct ahc_softc *ahc, struct scsi_cmnd *cmd,
		      struct ahc_devinfo *devinfo)
{
	memset(cmd, 0, sizeof(struct scsi_cmnd));
Justin T. Gibbs's avatar
Justin T. Gibbs committed
3060
	cmd->device = ahc->platform_data->dv_scsi_dev;
3061 3062 3063 3064 3065 3066 3067 3068 3069 3070 3071 3072 3073 3074 3075 3076 3077 3078 3079 3080 3081 3082 3083 3084 3085 3086 3087 3088 3089 3090 3091 3092 3093 3094 3095 3096 3097 3098 3099 3100 3101 3102 3103 3104 3105 3106 3107 3108 3109 3110 3111 3112 3113 3114 3115 3116 3117 3118 3119 3120 3121 3122 3123 3124 3125 3126 3127 3128 3129 3130 3131 3132 3133 3134 3135 3136 3137 3138 3139 3140 3141 3142 3143 3144 3145 3146 3147 3148 3149 3150 3151 3152 3153 3154 3155 3156 3157 3158 3159 3160 3161 3162 3163 3164 3165 3166 3167 3168 3169 3170 3171 3172 3173 3174 3175 3176 3177 3178 3179 3180 3181 3182 3183 3184 3185 3186 3187 3188 3189 3190 3191 3192 3193 3194 3195 3196 3197 3198 3199 3200 3201 3202 3203 3204 3205 3206 3207 3208 3209 3210 3211 3212 3213 3214 3215 3216 3217 3218 3219 3220 3221 3222 3223 3224 3225 3226 3227 3228 3229 3230 3231 3232 3233 3234 3235 3236 3237 3238 3239 3240 3241 3242 3243 3244 3245 3246 3247 3248 3249 3250 3251 3252 3253
	cmd->scsi_done = ahc_linux_dv_complete;
}

/*
 * Synthesize an inquiry command.  On the return trip, it'll be
 * sniffed and the device transfer settings set for us.
 */
static void
ahc_linux_dv_inq(struct ahc_softc *ahc, struct scsi_cmnd *cmd,
		 struct ahc_devinfo *devinfo, struct ahc_linux_target *targ,
		 u_int request_length)
{

#ifdef AHC_DEBUG
	if (ahc_debug & AHC_SHOW_DV) {
		ahc_print_devinfo(ahc, devinfo);
		printf("Sending INQ\n");
	}
#endif
	if (targ->inq_data == NULL)
		targ->inq_data = malloc(AHC_LINUX_DV_INQ_LEN,
					M_DEVBUF, M_WAITOK);
	if (targ->dv_state > AHC_DV_STATE_INQ_ASYNC) {
		if (targ->dv_buffer != NULL)
			free(targ->dv_buffer, M_DEVBUF);
		targ->dv_buffer = malloc(AHC_LINUX_DV_INQ_LEN,
					 M_DEVBUF, M_WAITOK);
	}

	ahc_linux_dv_fill_cmd(ahc, cmd, devinfo);
	cmd->sc_data_direction = SCSI_DATA_READ;
	cmd->cmd_len = 6;
	cmd->cmnd[0] = INQUIRY;
	cmd->cmnd[4] = request_length;
	cmd->request_bufflen = request_length;
	if (targ->dv_state > AHC_DV_STATE_INQ_ASYNC)
		cmd->request_buffer = targ->dv_buffer;
	else
		cmd->request_buffer = targ->inq_data;
	memset(cmd->request_buffer, 0, AHC_LINUX_DV_INQ_LEN);
}

static void
ahc_linux_dv_tur(struct ahc_softc *ahc, struct scsi_cmnd *cmd,
		 struct ahc_devinfo *devinfo)
{

#ifdef AHC_DEBUG
	if (ahc_debug & AHC_SHOW_DV) {
		ahc_print_devinfo(ahc, devinfo);
		printf("Sending TUR\n");
	}
#endif
	/* Do a TUR to clear out any non-fatal transitional state */
	ahc_linux_dv_fill_cmd(ahc, cmd, devinfo);
	cmd->sc_data_direction = SCSI_DATA_NONE;
	cmd->cmd_len = 6;
	cmd->cmnd[0] = TEST_UNIT_READY;
}

#define AHC_REBD_LEN 4

static void
ahc_linux_dv_rebd(struct ahc_softc *ahc, struct scsi_cmnd *cmd,
		 struct ahc_devinfo *devinfo, struct ahc_linux_target *targ)
{

#ifdef AHC_DEBUG
	if (ahc_debug & AHC_SHOW_DV) {
		ahc_print_devinfo(ahc, devinfo);
		printf("Sending REBD\n");
	}
#endif
	if (targ->dv_buffer != NULL)
		free(targ->dv_buffer, M_DEVBUF);
	targ->dv_buffer = malloc(AHC_REBD_LEN, M_DEVBUF, M_WAITOK);
	ahc_linux_dv_fill_cmd(ahc, cmd, devinfo);
	cmd->sc_data_direction = SCSI_DATA_READ;
	cmd->cmd_len = 10;
	cmd->cmnd[0] = READ_BUFFER;
	cmd->cmnd[1] = 0x0b;
	scsi_ulto3b(AHC_REBD_LEN, &cmd->cmnd[6]);
	cmd->request_bufflen = AHC_REBD_LEN;
	cmd->underflow = cmd->request_bufflen;
	cmd->request_buffer = targ->dv_buffer;
}

static void
ahc_linux_dv_web(struct ahc_softc *ahc, struct scsi_cmnd *cmd,
		 struct ahc_devinfo *devinfo, struct ahc_linux_target *targ)
{

#ifdef AHC_DEBUG
	if (ahc_debug & AHC_SHOW_DV) {
		ahc_print_devinfo(ahc, devinfo);
		printf("Sending WEB\n");
	}
#endif
	ahc_linux_dv_fill_cmd(ahc, cmd, devinfo);
	cmd->sc_data_direction = SCSI_DATA_WRITE;
	cmd->cmd_len = 10;
	cmd->cmnd[0] = WRITE_BUFFER;
	cmd->cmnd[1] = 0x0a;
	scsi_ulto3b(targ->dv_echo_size, &cmd->cmnd[6]);
	cmd->request_bufflen = targ->dv_echo_size;
	cmd->underflow = cmd->request_bufflen;
	cmd->request_buffer = targ->dv_buffer;
}

static void
ahc_linux_dv_reb(struct ahc_softc *ahc, struct scsi_cmnd *cmd,
		 struct ahc_devinfo *devinfo, struct ahc_linux_target *targ)
{

#ifdef AHC_DEBUG
	if (ahc_debug & AHC_SHOW_DV) {
		ahc_print_devinfo(ahc, devinfo);
		printf("Sending REB\n");
	}
#endif
	ahc_linux_dv_fill_cmd(ahc, cmd, devinfo);
	cmd->sc_data_direction = SCSI_DATA_READ;
	cmd->cmd_len = 10;
	cmd->cmnd[0] = READ_BUFFER;
	cmd->cmnd[1] = 0x0a;
	scsi_ulto3b(targ->dv_echo_size, &cmd->cmnd[6]);
	cmd->request_bufflen = targ->dv_echo_size;
	cmd->underflow = cmd->request_bufflen;
	cmd->request_buffer = targ->dv_buffer1;
}

static void
ahc_linux_dv_su(struct ahc_softc *ahc, struct scsi_cmnd *cmd,
		struct ahc_devinfo *devinfo,
		struct ahc_linux_target *targ)
{
	u_int le;

	le = SID_IS_REMOVABLE(targ->inq_data) ? SSS_LOEJ : 0;

#ifdef AHC_DEBUG
	if (ahc_debug & AHC_SHOW_DV) {
		ahc_print_devinfo(ahc, devinfo);
		printf("Sending SU\n");
	}
#endif
	ahc_linux_dv_fill_cmd(ahc, cmd, devinfo);
	cmd->sc_data_direction = SCSI_DATA_NONE;
	cmd->cmd_len = 6;
	cmd->cmnd[0] = START_STOP_UNIT;
	cmd->cmnd[4] = le | SSS_START;
}

static int
ahc_linux_fallback(struct ahc_softc *ahc, struct ahc_devinfo *devinfo)
{
	struct	ahc_linux_target *targ;
	struct	ahc_initiator_tinfo *tinfo;
	struct	ahc_transinfo *goal;
	struct	ahc_tmode_tstate *tstate;
	struct	ahc_syncrate *syncrate;
	u_long	s;
	u_int	width;
	u_int	period;
	u_int	offset;
	u_int	ppr_options;
	u_int	cur_speed;
	u_int	wide_speed;
	u_int	narrow_speed;
	u_int	fallback_speed;

#ifdef AHC_DEBUG
	if (ahc_debug & AHC_SHOW_DV) {
		ahc_print_devinfo(ahc, devinfo);
		printf("Trying to fallback\n");
	}
#endif
	ahc_lock(ahc, &s);
	targ = ahc->platform_data->targets[devinfo->target_offset];
	tinfo = ahc_fetch_transinfo(ahc, devinfo->channel,
				    devinfo->our_scsiid,
				    devinfo->target, &tstate);
	goal = &tinfo->goal;
	width = goal->width;
	period = goal->period;
	offset = goal->offset;
	ppr_options = goal->ppr_options;
	if (offset == 0)
		period = AHC_ASYNC_XFER_PERIOD;
	if (targ->dv_next_narrow_period == 0)
		targ->dv_next_narrow_period = MAX(period, AHC_SYNCRATE_ULTRA2);
	if (targ->dv_next_wide_period == 0)
		targ->dv_next_wide_period = period;
3254 3255
	if (targ->dv_max_width == 0)
		targ->dv_max_width = width;
3256 3257 3258 3259 3260
	if (targ->dv_max_ppr_options == 0)
		targ->dv_max_ppr_options = ppr_options;
	if (targ->dv_last_ppr_options == 0)
		targ->dv_last_ppr_options = ppr_options;

Justin T. Gibbs's avatar
Justin T. Gibbs committed
3261 3262
	cur_speed = aic_calc_speed(width, period, offset, AHC_SYNCRATE_MIN);
	wide_speed = aic_calc_speed(MSG_EXT_WDTR_BUS_16_BIT,
3263
					  targ->dv_next_wide_period,
Justin T. Gibbs's avatar
Justin T. Gibbs committed
3264 3265 3266
					  MAX_OFFSET,
					  AHC_SYNCRATE_MIN);
	narrow_speed = aic_calc_speed(MSG_EXT_WDTR_BUS_8_BIT,
3267
					    targ->dv_next_narrow_period,
Justin T. Gibbs's avatar
Justin T. Gibbs committed
3268 3269 3270 3271
					    MAX_OFFSET,
					    AHC_SYNCRATE_MIN);
	fallback_speed = aic_calc_speed(width, period+1, offset,
					AHC_SYNCRATE_MIN);
3272 3273 3274 3275 3276 3277
#ifdef AHC_DEBUG
	if (ahc_debug & AHC_SHOW_DV) {
		printf("cur_speed= %d, wide_speed= %d, narrow_speed= %d, "
		       "fallback_speed= %d\n", cur_speed, wide_speed,
		       narrow_speed, fallback_speed);
	}
Justin T. Gibbs's avatar
Justin T. Gibbs committed
3278
#endif
3279 3280 3281 3282 3283 3284 3285 3286 3287 3288 3289 3290 3291 3292 3293 3294 3295 3296 3297 3298 3299 3300 3301 3302 3303 3304 3305 3306 3307 3308 3309 3310 3311 3312 3313 3314 3315 3316 3317 3318 3319 3320 3321 3322 3323 3324 3325 3326 3327 3328 3329 3330 3331 3332 3333 3334 3335 3336 3337 3338 3339 3340 3341 3342 3343 3344 3345 3346 3347 3348 3349 3350 3351 3352

	if (cur_speed > 160000) {
		/*
		 * Paced/DT/IU_REQ only transfer speeds.  All we
		 * can do is fallback in terms of syncrate.
		 */
		period++;
	} else if (cur_speed > 80000) {
		if ((ppr_options & MSG_EXT_PPR_IU_REQ) != 0) {
			/*
			 * Try without IU_REQ as it may be confusing
			 * an expander.
			 */
			ppr_options &= ~MSG_EXT_PPR_IU_REQ;
		} else {
			/*
			 * Paced/DT only transfer speeds.  All we
			 * can do is fallback in terms of syncrate.
			 */
			period++;
			ppr_options = targ->dv_max_ppr_options;
		}
	} else if (cur_speed > 3300) {

		/*
		 * In this range we the following
		 * options ordered from highest to
		 * lowest desireability:
		 *
		 * o Wide/DT
		 * o Wide/non-DT
		 * o Narrow at a potentally higher sync rate.
		 *
		 * All modes are tested with and without IU_REQ
		 * set since using IUs may confuse an expander.
		 */
		if ((ppr_options & MSG_EXT_PPR_IU_REQ) != 0) {

			ppr_options &= ~MSG_EXT_PPR_IU_REQ;
		} else if ((ppr_options & MSG_EXT_PPR_DT_REQ) != 0) {
			/*
			 * Try going non-DT.
			 */
			ppr_options = targ->dv_max_ppr_options;
			ppr_options &= ~MSG_EXT_PPR_DT_REQ;
		} else if (targ->dv_last_ppr_options != 0) {
			/*
			 * Try without QAS or any other PPR options.
			 * We may need a non-PPR message to work with
			 * an expander.  We look at the "last PPR options"
			 * so we will perform this fallback even if the
			 * target responded to our PPR negotiation with
			 * no option bits set.
			 */
			ppr_options = 0;
		} else if (width == MSG_EXT_WDTR_BUS_16_BIT) {
			/*
			 * If the next narrow speed is greater than
			 * the next wide speed, fallback to narrow.
			 * Otherwise fallback to the next DT/Wide setting.
			 * The narrow async speed will always be smaller
			 * than the wide async speed, so handle this case
			 * specifically.
			 */
			ppr_options = targ->dv_max_ppr_options;
			if (narrow_speed > fallback_speed
			 || period >= AHC_ASYNC_XFER_PERIOD) {
				targ->dv_next_wide_period = period+1;
				width = MSG_EXT_WDTR_BUS_8_BIT;
				period = targ->dv_next_narrow_period;
			} else {
				period++;
			}
		} else if ((ahc->features & AHC_WIDE) != 0
3353
			&& targ->dv_max_width != 0
3354 3355 3356 3357 3358 3359 3360 3361 3362 3363 3364 3365 3366 3367 3368 3369 3370 3371 3372 3373 3374 3375 3376 3377 3378 3379 3380 3381 3382 3383 3384 3385 3386 3387 3388 3389 3390 3391 3392 3393 3394 3395 3396 3397 3398 3399 3400 3401 3402
			&& wide_speed >= fallback_speed
			&& (targ->dv_next_wide_period <= AHC_ASYNC_XFER_PERIOD
			 || period >= AHC_ASYNC_XFER_PERIOD)) {

			/*
			 * We are narrow.  Try falling back
			 * to the next wide speed with 
			 * all supported ppr options set.
			 */
			targ->dv_next_narrow_period = period+1;
			width = MSG_EXT_WDTR_BUS_16_BIT;
			period = targ->dv_next_wide_period;
			ppr_options = targ->dv_max_ppr_options;
		} else {
			/* Only narrow fallback is allowed. */
			period++;
			ppr_options = targ->dv_max_ppr_options;
		}
	} else {
		ahc_unlock(ahc, &s);
		return (-1);
	}
	offset = MAX_OFFSET;
	syncrate = ahc_find_syncrate(ahc, &period, &ppr_options,
				     AHC_SYNCRATE_DT);
	ahc_set_width(ahc, devinfo, width, AHC_TRANS_GOAL, FALSE);
	if (period == 0) {
		period = 0;
		offset = 0;
		ppr_options = 0;
		if (width == MSG_EXT_WDTR_BUS_8_BIT)
			targ->dv_next_narrow_period = AHC_ASYNC_XFER_PERIOD;
		else
			targ->dv_next_wide_period = AHC_ASYNC_XFER_PERIOD;
	}
	ahc_set_syncrate(ahc, devinfo, syncrate, period, offset,
			 ppr_options, AHC_TRANS_GOAL, FALSE);
	targ->dv_last_ppr_options = ppr_options;
	ahc_unlock(ahc, &s);
	return (0);
}

static void
ahc_linux_dv_timeout(struct scsi_cmnd *cmd)
{
	struct	ahc_softc *ahc;
	struct	scb *scb;
	u_long	flags;

Justin T. Gibbs's avatar
Justin T. Gibbs committed
3403
	ahc = *((struct ahc_softc **)cmd->device->host->hostdata);
3404 3405 3406
	ahc_lock(ahc, &flags);

#ifdef AHC_DEBUG
3407
	if (ahc_debug & AHC_SHOW_DV) {
3408 3409
		printf("%s: Timeout while doing DV command %x.\n",
		       ahc_name(ahc), cmd->cmnd[0]);
3410 3411
		ahc_dump_card_state(ahc);
	}
3412 3413 3414 3415 3416 3417 3418 3419 3420 3421 3422 3423 3424 3425 3426 3427 3428 3429 3430 3431 3432
#endif
	
	/*
	 * Guard against "done race".  No action is
	 * required if we just completed.
	 */
	if ((scb = (struct scb *)cmd->host_scribble) == NULL) {
		ahc_unlock(ahc, &flags);
		return;
	}

	/*
	 * Command has not completed.  Mark this
	 * SCB as having failing status prior to
	 * resetting the bus, so we get the correct
	 * error code.
	 */
	if ((scb->flags & SCB_SENSE) != 0)
		ahc_set_transaction_status(scb, CAM_AUTOSENSE_FAIL);
	else
		ahc_set_transaction_status(scb, CAM_CMD_TIMEOUT);
Justin T. Gibbs's avatar
Justin T. Gibbs committed
3433
	ahc_reset_channel(ahc, cmd->device->channel + 'A', /*initiate*/TRUE);
3434 3435 3436 3437 3438 3439 3440 3441 3442 3443 3444 3445

	/*
	 * Add a minimal bus settle delay for devices that are slow to
	 * respond after bus resets.
	 */
	ahc_linux_freeze_simq(ahc);
	init_timer(&ahc->platform_data->reset_timer);
	ahc->platform_data->reset_timer.data = (u_long)ahc;
	ahc->platform_data->reset_timer.expires = jiffies + HZ / 2;
	ahc->platform_data->reset_timer.function =
	    (ahc_linux_callback_t *)ahc_linux_release_simq;
	add_timer(&ahc->platform_data->reset_timer);
3446
	if (ahc_linux_next_device_to_run(ahc) != NULL)
3447
		ahc_schedule_runq(ahc);
3448
	ahc_linux_run_complete_queue(ahc);
3449 3450 3451 3452 3453 3454 3455 3456
	ahc_unlock(ahc, &flags);
}

static void
ahc_linux_dv_complete(struct scsi_cmnd *cmd)
{
	struct ahc_softc *ahc;

Justin T. Gibbs's avatar
Justin T. Gibbs committed
3457
	ahc = *((struct ahc_softc **)cmd->device->host->hostdata);
3458 3459 3460 3461 3462 3463 3464

	/* Delete the DV timer before it goes off! */
	scsi_delete_timer(cmd);

#ifdef AHC_DEBUG
	if (ahc_debug & AHC_SHOW_DV)
		printf("%s:%d:%d: Command completed, status= 0x%x\n",
Justin T. Gibbs's avatar
Justin T. Gibbs committed
3465 3466
		       ahc_name(ahc), cmd->device->channel,
		       cmd->device->id, cmd->result);
3467 3468 3469 3470 3471 3472 3473 3474 3475 3476 3477 3478 3479 3480 3481 3482 3483 3484 3485 3486 3487 3488 3489 3490 3491 3492 3493 3494 3495 3496 3497 3498 3499 3500 3501 3502 3503 3504 3505 3506 3507 3508 3509 3510 3511 3512 3513 3514 3515 3516 3517 3518 3519 3520 3521 3522 3523 3524 3525 3526 3527 3528 3529 3530 3531 3532 3533 3534 3535 3536
#endif

	/* Wake up the state machine */
	up(&ahc->platform_data->dv_cmd_sem);
}

static void
ahc_linux_generate_dv_pattern(struct ahc_linux_target *targ)
{
	uint16_t b;
	u_int	 i;
	u_int	 j;

	if (targ->dv_buffer != NULL)
		free(targ->dv_buffer, M_DEVBUF);
	targ->dv_buffer = malloc(targ->dv_echo_size, M_DEVBUF, M_WAITOK);
	if (targ->dv_buffer1 != NULL)
		free(targ->dv_buffer1, M_DEVBUF);
	targ->dv_buffer1 = malloc(targ->dv_echo_size, M_DEVBUF, M_WAITOK);

	i = 0;
	b = 0x0001;
	for (j = 0 ; i < targ->dv_echo_size; j++) {
		if (j < 32) {
			/*
			 * 32bytes of sequential numbers.
			 */
			targ->dv_buffer[i++] = j & 0xff;
		} else if (j < 48) {
			/*
			 * 32bytes of repeating 0x0000, 0xffff.
			 */
			targ->dv_buffer[i++] = (j & 0x02) ? 0xff : 0x00;
		} else if (j < 64) {
			/*
			 * 32bytes of repeating 0x5555, 0xaaaa.
			 */
			targ->dv_buffer[i++] = (j & 0x02) ? 0xaa : 0x55;
		} else {
			/*
			 * Remaining buffer is filled with a repeating
			 * patter of:
			 *
			 *	 0xffff
			 *	~0x0001 << shifted once in each loop.
			 */
			if (j & 0x02) {
				if (j & 0x01) {
					targ->dv_buffer[i++] = ~(b >> 8) & 0xff;
					b <<= 1;
					if (b == 0x0000)
						b = 0x0001;
				} else {
					targ->dv_buffer[i++] = (~b & 0xff);
				}
			} else {
				targ->dv_buffer[i++] = 0xff;
			}
		}
	}
}

static u_int
ahc_linux_user_tagdepth(struct ahc_softc *ahc, struct ahc_devinfo *devinfo)
{
	static int warned_user;
	u_int tags;

	tags = 0;
	if ((ahc->user_discenable & devinfo->target_mask) != 0) {
3537 3538 3539 3540 3541 3542 3543 3544 3545 3546
		if (ahc->unit >= NUM_ELEMENTS(aic7xxx_tag_info)) {
			if (warned_user == 0) {

				printf(KERN_WARNING
"aic7xxx: WARNING: Insufficient tag_info instances\n"
"aic7xxx: for installed controllers. Using defaults\n"
"aic7xxx: Please update the aic7xxx_tag_info array in\n"
"aic7xxx: the aic7xxx_osm..c source file.\n");
				warned_user++;
			}
3547 3548 3549 3550 3551 3552 3553 3554 3555 3556 3557 3558 3559 3560 3561 3562 3563 3564 3565
			tags = AHC_MAX_QUEUE;
		} else {
			adapter_tag_info_t *tag_info;

			tag_info = &aic7xxx_tag_info[ahc->unit];
			tags = tag_info->tag_commands[devinfo->target_offset];
			if (tags > AHC_MAX_QUEUE)
				tags = AHC_MAX_QUEUE;
		}
	}
	return (tags);
}

static u_int
ahc_linux_user_dv_setting(struct ahc_softc *ahc)
{
	static int warned_user;
	int dv;

3566 3567
	if (ahc->unit >= NUM_ELEMENTS(aic7xxx_dv_settings)) {
		if (warned_user == 0) {
3568

3569 3570 3571 3572 3573 3574 3575
			printf(KERN_WARNING
"aic7xxx: WARNING: Insufficient dv settings instances\n"
"aic7xxx: for installed controllers. Using defaults\n"
"aic7xxx: Please update the aic7xxx_dv_settings array\n"
"aic7xxx: in the aic7xxx_osm.c source file.\n");
			warned_user++;
		}
3576
		dv = -1;
Linus Torvalds's avatar
Linus Torvalds committed
3577
	} else {
3578 3579 3580 3581 3582 3583 3584 3585 3586 3587

		dv = aic7xxx_dv_settings[ahc->unit];
	}

	if (dv < 0) {
		u_long s;

		/*
		 * Apply the default.
		 */
Linus Torvalds's avatar
Linus Torvalds committed
3588
		/*
3589 3590
		 * XXX - Enable DV on non-U160 controllers once it
		 *       has been tested there.
Linus Torvalds's avatar
Linus Torvalds committed
3591
		 */
3592 3593 3594 3595 3596 3597 3598 3599 3600 3601 3602 3603 3604
		ahc_lock(ahc, &s);
		dv = (ahc->features & AHC_DT);
		if (ahc->seep_config != 0
		 && ahc->seep_config->signature >= CFSIGNATURE2)
			dv = (ahc->seep_config->adapter_control & CFENABLEDV);
		ahc_unlock(ahc, &s);
	}
	return (dv);
}

/*
 * Determines the queue depth for a given device.
 */
3605 3606 3607 3608 3609 3610
static void
ahc_linux_device_queue_depth(struct ahc_softc *ahc,
			     struct ahc_linux_device *dev)
{
	struct	ahc_devinfo devinfo;
	u_int	tags;
3611

3612 3613 3614 3615 3616 3617 3618 3619 3620 3621
	ahc_compile_devinfo(&devinfo,
			    dev->target->channel == 0
			  ? ahc->our_id : ahc->our_id_b,
			    dev->target->target, dev->lun,
			    dev->target->channel == 0 ? 'A' : 'B',
			    ROLE_INITIATOR);
	tags = ahc_linux_user_tagdepth(ahc, &devinfo);
	if (tags != 0
	 && dev->scsi_device != NULL
	 && dev->scsi_device->tagged_supported != 0) {
3622

3623
		ahc_set_tags(ahc, &devinfo, AHC_QUEUE_TAGGED);
3624 3625
		ahc_print_devinfo(ahc, &devinfo);
		printf("Tagged Queuing enabled.  Depth %d\n", tags);
3626 3627
	} else {
		ahc_set_tags(ahc, &devinfo, AHC_QUEUE_NONE);
Linus Torvalds's avatar
Linus Torvalds committed
3628
	}
Linus Torvalds's avatar
Linus Torvalds committed
3629 3630 3631
}

static void
Linus Torvalds's avatar
Linus Torvalds committed
3632
ahc_linux_run_device_queue(struct ahc_softc *ahc, struct ahc_linux_device *dev)
Linus Torvalds's avatar
Linus Torvalds committed
3633 3634 3635 3636 3637 3638
{
	struct	 ahc_cmd *acmd;
	struct	 scsi_cmnd *cmd;
	struct	 scb *scb;
	struct	 hardware_scb *hscb;
	struct	 ahc_initiator_tinfo *tinfo;
Linus Torvalds's avatar
Linus Torvalds committed
3639
	struct	 ahc_tmode_tstate *tstate;
Linus Torvalds's avatar
Linus Torvalds committed
3640 3641
	uint16_t mask;

Linus Torvalds's avatar
Linus Torvalds committed
3642 3643 3644
	if ((dev->flags & AHC_DEV_ON_RUN_LIST) != 0)
		panic("running device on run list");

Linus Torvalds's avatar
Linus Torvalds committed
3645 3646 3647 3648 3649 3650 3651
	while ((acmd = TAILQ_FIRST(&dev->busyq)) != NULL
	    && dev->openings > 0 && dev->qfrozen == 0) {

		/*
		 * Schedule us to run later.  The only reason we are not
		 * running is because the whole controller Q is frozen.
		 */
3652 3653
		if (ahc->platform_data->qfrozen != 0
	 	 && AHC_DV_SIMQ_FROZEN(ahc) == 0) {
Linus Torvalds's avatar
Linus Torvalds committed
3654 3655
			TAILQ_INSERT_TAIL(&ahc->platform_data->device_runq,
					  dev, links);
Linus Torvalds's avatar
Linus Torvalds committed
3656 3657 3658 3659 3660 3661 3662
			dev->flags |= AHC_DEV_ON_RUN_LIST;
			return;
		}
		/*
		 * Get an scb to use.
		 */
		if ((scb = ahc_get_scb(ahc)) == NULL) {
Linus Torvalds's avatar
Linus Torvalds committed
3663
			TAILQ_INSERT_TAIL(&ahc->platform_data->device_runq,
Linus Torvalds's avatar
Linus Torvalds committed
3664 3665 3666 3667 3668 3669 3670 3671 3672 3673 3674 3675 3676 3677 3678 3679 3680
					 dev, links);
			dev->flags |= AHC_DEV_ON_RUN_LIST;
			ahc->flags |= AHC_RESOURCE_SHORTAGE;
			return;
		}
		TAILQ_REMOVE(&dev->busyq, acmd, acmd_links.tqe);
		cmd = &acmd_scsi_cmd(acmd);
		scb->io_ctx = cmd;
		scb->platform_data->dev = dev;
		hscb = scb->hscb;
		cmd->host_scribble = (char *)scb;

		/*
		 * Fill out basics of the HSCB.
		 */
		hscb->control = 0;
		hscb->scsiid = BUILD_SCSIID(ahc, cmd);
Luben Tuikov's avatar
Luben Tuikov committed
3681
		hscb->lun = cmd->device->lun;
Linus Torvalds's avatar
Linus Torvalds committed
3682 3683 3684 3685 3686
		mask = SCB_GET_TARGET_MASK(ahc, scb);
		tinfo = ahc_fetch_transinfo(ahc, SCB_GET_CHANNEL(ahc, scb),
					    SCB_GET_OUR_ID(scb),
					    SCB_GET_TARGET(ahc, scb), &tstate);
		hscb->scsirate = tinfo->scsirate;
Linus Torvalds's avatar
Linus Torvalds committed
3687
		hscb->scsioffset = tinfo->curr.offset;
Linus Torvalds's avatar
Linus Torvalds committed
3688 3689 3690
		if ((tstate->ultraenb & mask) != 0)
			hscb->control |= ULTRAENB;

Linus Torvalds's avatar
Linus Torvalds committed
3691
		if ((ahc->user_discenable & mask) != 0)
Linus Torvalds's avatar
Linus Torvalds committed
3692 3693
			hscb->control |= DISCENB;

3694 3695 3696
	 	if (AHC_DV_CMD(cmd) != 0)
			scb->flags |= SCB_SILENT;

Linus Torvalds's avatar
Linus Torvalds committed
3697 3698 3699 3700 3701 3702
		if ((tstate->auto_negotiate & mask) != 0) {
			scb->flags |= SCB_AUTO_NEGOTIATE;
			scb->hscb->control |= MK_MESSAGE;
		}

		if ((dev->flags & (AHC_DEV_Q_TAGGED|AHC_DEV_Q_BASIC)) != 0) {
3703 3704 3705 3706 3707 3708 3709 3710 3711 3712 3713
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,5,0)
			int	msg_bytes;
			uint8_t tag_msgs[2];

			msg_bytes = scsi_populate_tag_msg(cmd, tag_msgs);
			if (msg_bytes && tag_msgs[0] != MSG_SIMPLE_TASK) {
				hscb->control |= tag_msgs[0];
				if (tag_msgs[0] == MSG_ORDERED_TASK)
					dev->commands_since_idle_or_otag = 0;
			} else
#endif
Linus Torvalds's avatar
Linus Torvalds committed
3714 3715 3716 3717 3718 3719 3720
			if (dev->commands_since_idle_or_otag == AHC_OTAG_THRESH
			 && (dev->flags & AHC_DEV_Q_TAGGED) != 0) {
				hscb->control |= MSG_ORDERED_TASK;
				dev->commands_since_idle_or_otag = 0;
			} else {
				hscb->control |= MSG_SIMPLE_TASK;
			}
Linus Torvalds's avatar
Linus Torvalds committed
3721 3722 3723 3724 3725 3726 3727 3728 3729 3730 3731
		}

		hscb->cdb_len = cmd->cmd_len;
		if (hscb->cdb_len <= 12) {
			memcpy(hscb->shared_data.cdb, cmd->cmnd, hscb->cdb_len);
		} else {
			memcpy(hscb->cdb32, cmd->cmnd, hscb->cdb_len);
			scb->flags |= SCB_CDB32_PTR;
		}

		scb->platform_data->xfer_len = 0;
Linus Torvalds's avatar
Linus Torvalds committed
3732 3733
		ahc_set_residual(scb, 0);
		ahc_set_sense_residual(scb, 0);
3734
		scb->sg_count = 0;
Linus Torvalds's avatar
Linus Torvalds committed
3735 3736 3737 3738 3739 3740 3741 3742
		if (cmd->use_sg != 0) {
			struct	ahc_dma_seg *sg;
			struct	scatterlist *cur_seg;
			struct	scatterlist *end_seg;
			int	nseg;

			cur_seg = (struct scatterlist *)cmd->request_buffer;
			nseg = pci_map_sg(ahc->dev_softc, cur_seg, cmd->use_sg,
Justin T. Gibbs's avatar
Justin T. Gibbs committed
3743
			    scsi_to_pci_dma_dir(cmd->sc_data_direction));
Linus Torvalds's avatar
Linus Torvalds committed
3744 3745 3746
			end_seg = cur_seg + nseg;
			/* Copy the segments into the SG list. */
			sg = scb->sg_list;
Linus Torvalds's avatar
Linus Torvalds committed
3747 3748 3749 3750
			/*
			 * The sg_count may be larger than nseg if
			 * a transfer crosses a 32bit page.
			 */ 
3751
			while (cur_seg < end_seg) {
Linus Torvalds's avatar
Linus Torvalds committed
3752 3753 3754 3755 3756 3757 3758 3759 3760 3761
				bus_addr_t addr;
				bus_size_t len;
				int consumed;

				addr = sg_dma_address(cur_seg);
				len = sg_dma_len(cur_seg);
				consumed = ahc_linux_map_seg(ahc, scb,
							     sg, addr, len);
				sg += consumed;
				scb->sg_count += consumed;
Linus Torvalds's avatar
Linus Torvalds committed
3762 3763 3764 3765 3766 3767 3768 3769 3770 3771 3772 3773 3774 3775 3776 3777 3778 3779 3780
				cur_seg++;
			}
			sg--;
			sg->len |= ahc_htole32(AHC_DMA_LAST_SEG);

			/*
			 * Reset the sg list pointer.
			 */
			scb->hscb->sgptr =
			    ahc_htole32(scb->sg_list_phys | SG_FULL_RESID);

			/*
			 * Copy the first SG into the "current"
			 * data pointer area.
			 */
			scb->hscb->dataptr = scb->sg_list->addr;
			scb->hscb->datacnt = scb->sg_list->len;
		} else if (cmd->request_bufflen != 0) {
			struct	 ahc_dma_seg *sg;
Linus Torvalds's avatar
Linus Torvalds committed
3781
			bus_addr_t addr;
Linus Torvalds's avatar
Linus Torvalds committed
3782 3783

			sg = scb->sg_list;
Linus Torvalds's avatar
Linus Torvalds committed
3784
			addr = pci_map_single(ahc->dev_softc,
Linus Torvalds's avatar
Linus Torvalds committed
3785 3786 3787
			       cmd->request_buffer,
			       cmd->request_bufflen,
			       scsi_to_pci_dma_dir(cmd->sc_data_direction));
3788
			scb->platform_data->buf_busaddr = addr;
Linus Torvalds's avatar
Linus Torvalds committed
3789 3790 3791 3792
			scb->sg_count = ahc_linux_map_seg(ahc, scb,
							  sg, addr,
							  cmd->request_bufflen);
			sg->len |= ahc_htole32(AHC_DMA_LAST_SEG);
Linus Torvalds's avatar
Linus Torvalds committed
3793 3794 3795 3796 3797 3798 3799 3800 3801 3802 3803 3804 3805 3806 3807 3808 3809 3810 3811 3812

			/*
			 * Reset the sg list pointer.
			 */
			scb->hscb->sgptr =
			    ahc_htole32(scb->sg_list_phys | SG_FULL_RESID);

			/*
			 * Copy the first SG into the "current"
			 * data pointer area.
			 */
			scb->hscb->dataptr = sg->addr;
			scb->hscb->datacnt = sg->len;
		} else {
			scb->hscb->sgptr = ahc_htole32(SG_LIST_NULL);
			scb->hscb->dataptr = 0;
			scb->hscb->datacnt = 0;
			scb->sg_count = 0;
		}

Linus Torvalds's avatar
Linus Torvalds committed
3813
		ahc_sync_sglist(ahc, scb, BUS_DMASYNC_PREWRITE);
Linus Torvalds's avatar
Linus Torvalds committed
3814 3815 3816
		LIST_INSERT_HEAD(&ahc->pending_scbs, scb, pending_links);
		dev->openings--;
		dev->active++;
Linus Torvalds's avatar
Linus Torvalds committed
3817
		dev->commands_issued++;
Linus Torvalds's avatar
Linus Torvalds committed
3818 3819
		if ((dev->flags & AHC_DEV_PERIODIC_OTAG) != 0)
			dev->commands_since_idle_or_otag++;
Linus Torvalds's avatar
Linus Torvalds committed
3820 3821 3822 3823 3824 3825 3826 3827 3828 3829 3830 3831 3832 3833 3834 3835 3836 3837 3838 3839 3840 3841 3842 3843 3844 3845 3846

		/*
		 * We only allow one untagged transaction
		 * per target in the initiator role unless
		 * we are storing a full busy target *lun*
		 * table in SCB space.
		 */
		if ((scb->hscb->control & (TARGET_SCB|TAG_ENB)) == 0
		 && (ahc->features & AHC_SCB_BTT) == 0) {
			struct scb_tailq *untagged_q;
			int target_offset;

			target_offset = SCB_GET_TARGET_OFFSET(ahc, scb);
			untagged_q = &(ahc->untagged_queues[target_offset]);
			TAILQ_INSERT_TAIL(untagged_q, scb, links.tqe);
			scb->flags |= SCB_UNTAGGEDQ;
			if (TAILQ_FIRST(untagged_q) != scb)
				continue;
		}
		scb->flags |= SCB_ACTIVE;
		ahc_queue_scb(ahc, scb);
	}
}

/*
 * SCSI controller interrupt handler.
 */
3847
irqreturn_t
Linus Torvalds's avatar
Linus Torvalds committed
3848
ahc_linux_isr(int irq, void *dev_id, struct pt_regs * regs)
Linus Torvalds's avatar
Linus Torvalds committed
3849
{
3850 3851
	struct	ahc_softc *ahc;
	u_long	flags;
3852
	int	ours;
Linus Torvalds's avatar
Linus Torvalds committed
3853 3854 3855

	ahc = (struct ahc_softc *) dev_id;
	ahc_lock(ahc, &flags); 
3856
	ours = ahc_intr(ahc);
3857
	if (ahc_linux_next_device_to_run(ahc) != NULL)
3858
		ahc_schedule_runq(ahc);
3859
	ahc_linux_run_complete_queue(ahc);
3860
	ahc_unlock(ahc, &flags);
3861
	return IRQ_RETVAL(ours);
Linus Torvalds's avatar
Linus Torvalds committed
3862 3863 3864 3865 3866 3867
}

void
ahc_platform_flushwork(struct ahc_softc *ahc)
{

3868 3869
	while (ahc_linux_run_complete_queue(ahc) != NULL)
		;
Linus Torvalds's avatar
Linus Torvalds committed
3870 3871 3872
}

static struct ahc_linux_target*
Linus Torvalds's avatar
Linus Torvalds committed
3873
ahc_linux_alloc_target(struct ahc_softc *ahc, u_int channel, u_int target)
Linus Torvalds's avatar
Linus Torvalds committed
3874 3875 3876 3877
{
	struct ahc_linux_target *targ;
	u_int target_offset;

3878 3879 3880 3881
	target_offset = target;
	if (channel != 0)
		target_offset += 8;

Linus Torvalds's avatar
Linus Torvalds committed
3882 3883 3884 3885 3886 3887
	targ = malloc(sizeof(*targ), M_DEVBUG, M_NOWAIT);
	if (targ == NULL)
		return (NULL);
	memset(targ, 0, sizeof(*targ));
	targ->channel = channel;
	targ->target = target;
3888
	targ->ahc = ahc;
3889
	targ->flags = AHC_DV_REQUIRED;
Linus Torvalds's avatar
Linus Torvalds committed
3890 3891 3892 3893 3894
	ahc->platform_data->targets[target_offset] = targ;
	return (targ);
}

static void
Linus Torvalds's avatar
Linus Torvalds committed
3895
ahc_linux_free_target(struct ahc_softc *ahc, struct ahc_linux_target *targ)
Linus Torvalds's avatar
Linus Torvalds committed
3896
{
3897 3898 3899 3900
	struct ahc_devinfo devinfo;
	struct ahc_initiator_tinfo *tinfo;
	struct ahc_tmode_tstate *tstate;
	u_int our_id;
Linus Torvalds's avatar
Linus Torvalds committed
3901
	u_int target_offset;
3902
	char channel;
Linus Torvalds's avatar
Linus Torvalds committed
3903

3904 3905 3906 3907 3908 3909 3910
	/*
	 * Force a negotiation to async/narrow on any
	 * future command to this device unless a bus
	 * reset occurs between now and that command.
	 */
	channel = 'A' + targ->channel;
	our_id = ahc->our_id;
Linus Torvalds's avatar
Linus Torvalds committed
3911
	target_offset = targ->target;
3912
	if (targ->channel != 0) {
Linus Torvalds's avatar
Linus Torvalds committed
3913
		target_offset += 8;
3914 3915 3916 3917 3918 3919 3920 3921 3922 3923 3924
		our_id = ahc->our_id_b;
	}
	tinfo = ahc_fetch_transinfo(ahc, channel, our_id,
				    targ->target, &tstate);
	ahc_compile_devinfo(&devinfo, our_id, targ->target, CAM_LUN_WILDCARD,
			    channel, ROLE_INITIATOR);
	ahc_set_syncrate(ahc, &devinfo, NULL, 0, 0, 0,
			 AHC_TRANS_GOAL, /*paused*/FALSE);
	ahc_set_width(ahc, &devinfo, MSG_EXT_WDTR_BUS_8_BIT,
		      AHC_TRANS_GOAL, /*paused*/FALSE);
	ahc_update_neg_request(ahc, &devinfo, tstate, tinfo, AHC_NEG_ALWAYS);
Linus Torvalds's avatar
Linus Torvalds committed
3925
	ahc->platform_data->targets[target_offset] = NULL;
3926 3927 3928 3929 3930 3931
	if (targ->inq_data != NULL)
		free(targ->inq_data, M_DEVBUF);
	if (targ->dv_buffer != NULL)
		free(targ->dv_buffer, M_DEVBUF);
	if (targ->dv_buffer1 != NULL)
		free(targ->dv_buffer1, M_DEVBUF);
Linus Torvalds's avatar
Linus Torvalds committed
3932 3933 3934 3935
	free(targ, M_DEVBUF);
}

static struct ahc_linux_device*
Linus Torvalds's avatar
Linus Torvalds committed
3936
ahc_linux_alloc_device(struct ahc_softc *ahc,
Linus Torvalds's avatar
Linus Torvalds committed
3937 3938 3939 3940 3941 3942 3943 3944
		 struct ahc_linux_target *targ, u_int lun)
{
	struct ahc_linux_device *dev;

	dev = malloc(sizeof(*dev), M_DEVBUG, M_NOWAIT);
	if (dev == NULL)
		return (NULL);
	memset(dev, 0, sizeof(*dev));
3945
	init_timer(&dev->timer);
Linus Torvalds's avatar
Linus Torvalds committed
3946 3947 3948 3949 3950 3951 3952 3953 3954 3955 3956 3957 3958 3959 3960 3961 3962 3963 3964 3965 3966 3967 3968 3969
	TAILQ_INIT(&dev->busyq);
	dev->flags = AHC_DEV_UNCONFIGURED;
	dev->lun = lun;
	dev->target = targ;

	/*
	 * We start out life using untagged
	 * transactions of which we allow one.
	 */
	dev->openings = 1;

	/*
	 * Set maxtags to 0.  This will be changed if we
	 * later determine that we are dealing with
	 * a tagged queuing capable device.
	 */
	dev->maxtags = 0;
	
	targ->refcount++;
	targ->devices[lun] = dev;
	return (dev);
}

static void
Andrew Morton's avatar
Andrew Morton committed
3970
__ahc_linux_free_device(struct ahc_softc *ahc, struct ahc_linux_device *dev)
Linus Torvalds's avatar
Linus Torvalds committed
3971 3972 3973 3974 3975 3976 3977
{
	struct ahc_linux_target *targ;

	targ = dev->target;
	targ->devices[dev->lun] = NULL;
	free(dev, M_DEVBUF);
	targ->refcount--;
3978 3979
	if (targ->refcount == 0
	 && (targ->flags & AHC_DV_REQUIRED) == 0)
Linus Torvalds's avatar
Linus Torvalds committed
3980
		ahc_linux_free_target(ahc, targ);
Linus Torvalds's avatar
Linus Torvalds committed
3981 3982
}

Andrew Morton's avatar
Andrew Morton committed
3983 3984 3985 3986 3987 3988 3989
static void
ahc_linux_free_device(struct ahc_softc *ahc, struct ahc_linux_device *dev)
{
	del_timer_sync(&dev->timer);
	__ahc_linux_free_device(ahc, dev);
}

Linus Torvalds's avatar
Linus Torvalds committed
3990 3991
void
ahc_send_async(struct ahc_softc *ahc, char channel,
Linus Torvalds's avatar
Linus Torvalds committed
3992
	       u_int target, u_int lun, ac_code code, void *arg)
Linus Torvalds's avatar
Linus Torvalds committed
3993 3994 3995 3996 3997 3998 3999 4000
{
	switch (code) {
	case AC_TRANSFER_NEG:
	{
		char	buf[80];
		struct	ahc_linux_target *targ;
		struct	info_str info;
		struct	ahc_initiator_tinfo *tinfo;
Linus Torvalds's avatar
Linus Torvalds committed
4001
		struct	ahc_tmode_tstate *tstate;
Linus Torvalds's avatar
Linus Torvalds committed
4002 4003 4004 4005 4006 4007 4008 4009 4010 4011 4012 4013 4014 4015 4016
		int	target_offset;

		info.buffer = buf;
		info.length = sizeof(buf);
		info.offset = 0;
		info.pos = 0;
		tinfo = ahc_fetch_transinfo(ahc, channel,
						channel == 'A' ? ahc->our_id
							       : ahc->our_id_b,
						target, &tstate);

		/*
		 * Don't bother reporting results while
		 * negotiations are still pending.
		 */
Linus Torvalds's avatar
Linus Torvalds committed
4017 4018 4019 4020
		if (tinfo->curr.period != tinfo->goal.period
		 || tinfo->curr.width != tinfo->goal.width
		 || tinfo->curr.offset != tinfo->goal.offset
		 || tinfo->curr.ppr_options != tinfo->goal.ppr_options)
Linus Torvalds's avatar
Linus Torvalds committed
4021 4022 4023 4024 4025 4026 4027 4028 4029 4030 4031
			if (bootverbose == 0)
				break;

		/*
		 * Don't bother reporting results that
		 * are identical to those last reported.
		 */
		target_offset = target;
		if (channel == 'B')
			target_offset += 8;
		targ = ahc->platform_data->targets[target_offset];
4032 4033 4034
		if (targ == NULL)
			break;
		if (tinfo->curr.period == targ->last_tinfo.period
Linus Torvalds's avatar
Linus Torvalds committed
4035 4036 4037
		 && tinfo->curr.width == targ->last_tinfo.width
		 && tinfo->curr.offset == targ->last_tinfo.offset
		 && tinfo->curr.ppr_options == targ->last_tinfo.ppr_options)
Linus Torvalds's avatar
Linus Torvalds committed
4038 4039 4040
			if (bootverbose == 0)
				break;

Linus Torvalds's avatar
Linus Torvalds committed
4041 4042 4043 4044
		targ->last_tinfo.period = tinfo->curr.period;
		targ->last_tinfo.width = tinfo->curr.width;
		targ->last_tinfo.offset = tinfo->curr.offset;
		targ->last_tinfo.ppr_options = tinfo->curr.ppr_options;
Linus Torvalds's avatar
Linus Torvalds committed
4045 4046 4047 4048 4049 4050

		printf("(%s:%c:", ahc_name(ahc), channel);
		if (target == CAM_TARGET_WILDCARD)
			printf("*): ");
		else
			printf("%d): ", target);
Linus Torvalds's avatar
Linus Torvalds committed
4051
		ahc_format_transinfo(&info, &tinfo->curr);
Linus Torvalds's avatar
Linus Torvalds committed
4052 4053 4054 4055 4056 4057 4058 4059
		if (info.pos < info.length)
			*info.buffer = '\0';
		else
			buf[info.length - 1] = '\0';
		printf("%s", buf);
		break;
	}
        case AC_SENT_BDR:
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4060
	{
4061 4062 4063 4064 4065
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,5,0)
		WARN_ON(lun != CAM_LUN_WILDCARD);
		scsi_report_device_reset(ahc->platform_data->host,
					 channel - 'A', target);
#elif LINUX_VERSION_CODE >= KERNEL_VERSION(2,3,0)
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4066 4067 4068 4069 4070 4071 4072 4073 4074 4075 4076 4077 4078 4079 4080 4081 4082
		Scsi_Device *scsi_dev;

		/*
		 * Find the SCSI device associated with this
		 * request and indicate that a UA is expected.
		 */
		for (scsi_dev = ahc->platform_data->host->host_queue;
		     scsi_dev != NULL; scsi_dev = scsi_dev->next) {
			if (channel - 'A' == scsi_dev->channel
			 && target == scsi_dev->id
			 && (lun == CAM_LUN_WILDCARD
			  || lun == scsi_dev->lun)) {
				scsi_dev->was_reset = 1;
				scsi_dev->expecting_cc_ua = 1;
			}
		}
#endif
Linus Torvalds's avatar
Linus Torvalds committed
4083
		break;
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4084
	}
Linus Torvalds's avatar
Linus Torvalds committed
4085 4086
        case AC_BUS_RESET:
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,3,0)
Linus Torvalds's avatar
Linus Torvalds committed
4087 4088 4089 4090
		if (ahc->platform_data->host != NULL) {
			scsi_report_bus_reset(ahc->platform_data->host,
					      channel - 'A');
		}
Linus Torvalds's avatar
Linus Torvalds committed
4091 4092 4093 4094 4095 4096 4097 4098 4099 4100 4101
#endif
                break;
        default:
                panic("ahc_send_async: Unexpected async event");
        }
}

/*
 * Calls the higher level scsi done function and frees the scb.
 */
void
4102
ahc_done(struct ahc_softc *ahc, struct scb *scb)
Linus Torvalds's avatar
Linus Torvalds committed
4103 4104
{
	Scsi_Cmnd *cmd;
4105
	struct	   ahc_linux_device *dev;
Linus Torvalds's avatar
Linus Torvalds committed
4106 4107 4108 4109 4110 4111 4112 4113 4114 4115 4116 4117 4118 4119 4120 4121 4122 4123 4124 4125 4126

	LIST_REMOVE(scb, pending_links);
	if ((scb->flags & SCB_UNTAGGEDQ) != 0) {
		struct scb_tailq *untagged_q;
		int target_offset;

		target_offset = SCB_GET_TARGET_OFFSET(ahc, scb);
		untagged_q = &(ahc->untagged_queues[target_offset]);
		TAILQ_REMOVE(untagged_q, scb, links.tqe);
		ahc_run_untagged_queue(ahc, untagged_q);
	}

	if ((scb->flags & SCB_ACTIVE) == 0) {
		printf("SCB %d done'd twice\n", scb->hscb->tag);
		ahc_dump_card_state(ahc);
		panic("Stopping for safety");
	}
	cmd = scb->io_ctx;
	dev = scb->platform_data->dev;
	dev->active--;
	dev->openings++;
4127 4128 4129
	if ((cmd->result & (CAM_DEV_QFRZN << 16)) != 0) {
		cmd->result &= ~(CAM_DEV_QFRZN << 16);
		dev->qfrozen--;
Linus Torvalds's avatar
Linus Torvalds committed
4130
	}
4131 4132 4133 4134 4135 4136 4137 4138 4139
	ahc_linux_unmap_scb(ahc, scb);

	/*
	 * Guard against stale sense data.
	 * The Linux mid-layer assumes that sense
	 * was retrieved anytime the first byte of
	 * the sense buffer looks "sane".
	 */
	cmd->sense_buffer[0] = 0;
Linus Torvalds's avatar
Linus Torvalds committed
4140 4141 4142 4143 4144
	if (ahc_get_transaction_status(scb) == CAM_REQ_INPROG) {
		uint32_t amount_xferred;

		amount_xferred =
		    ahc_get_transfer_length(scb) - ahc_get_residual(scb);
4145 4146 4147 4148 4149 4150 4151 4152
		if ((scb->flags & SCB_TRANSMISSION_ERROR) != 0) {
#ifdef AHC_DEBUG
			if ((ahc_debug & AHC_SHOW_MISC) != 0) {
				ahc_print_path(ahc, scb);
				printf("Set CAM_UNCOR_PARITY\n");
			}
#endif
			ahc_set_transaction_status(scb, CAM_UNCOR_PARITY);
4153 4154 4155 4156 4157 4158 4159 4160 4161 4162
#ifdef AHC_REPORT_UNDERFLOWS
		/*
		 * This code is disabled by default as some
		 * clients of the SCSI system do not properly
		 * initialize the underflow parameter.  This
		 * results in spurious termination of commands
		 * that complete as expected (e.g. underflow is
		 * allowed as command can return variable amounts
		 * of data.
		 */
4163
		} else if (amount_xferred < scb->io_ctx->underflow) {
4164 4165 4166 4167 4168 4169 4170 4171
			u_int i;

			ahc_print_path(ahc, scb);
			printf("CDB:");
			for (i = 0; i < scb->io_ctx->cmd_len; i++)
				printf(" 0x%x", scb->io_ctx->cmnd[i]);
			printf("\n");
			ahc_print_path(ahc, scb);
Linus Torvalds's avatar
Linus Torvalds committed
4172 4173 4174 4175 4176
			printf("Saw underflow (%ld of %ld bytes). "
			       "Treated as error\n",
				ahc_get_residual(scb),
				ahc_get_transfer_length(scb));
			ahc_set_transaction_status(scb, CAM_DATA_RUN_ERR);
4177
#endif
Linus Torvalds's avatar
Linus Torvalds committed
4178 4179 4180
		} else {
			ahc_set_transaction_status(scb, CAM_REQ_CMP);
		}
4181
	} else if (ahc_get_transaction_status(scb) == CAM_SCSI_STATUS_ERROR) {
Linus Torvalds's avatar
Linus Torvalds committed
4182
		ahc_linux_handle_scsi_status(ahc, dev, scb);
4183
	} else if (ahc_get_transaction_status(scb) == CAM_SEL_TIMEOUT) {
Linus Torvalds's avatar
Linus Torvalds committed
4184
		dev->flags |= AHC_DEV_UNCONFIGURED;
4185 4186
		if (AHC_DV_CMD(cmd) == FALSE)
			dev->target->flags &= ~AHC_DV_REQUIRED;
Linus Torvalds's avatar
Linus Torvalds committed
4187
	}
4188 4189 4190 4191 4192 4193 4194
	/*
	 * Start DV for devices that require it assuming the first command
	 * sent does not result in a selection timeout.
	 */
	if (ahc_get_transaction_status(scb) != CAM_SEL_TIMEOUT
	 && (dev->target->flags & AHC_DV_REQUIRED) != 0)
		ahc_linux_start_dv(ahc);
Linus Torvalds's avatar
Linus Torvalds committed
4195 4196 4197 4198 4199 4200 4201 4202 4203 4204 4205 4206 4207 4208 4209 4210 4211

	if (dev->openings == 1
	 && ahc_get_transaction_status(scb) == CAM_REQ_CMP
	 && ahc_get_scsi_status(scb) != SCSI_STATUS_QUEUE_FULL)
		dev->tag_success_count++;
	/*
	 * Some devices deal with temporary internal resource
	 * shortages by returning queue full.  When the queue
	 * full occurrs, we throttle back.  Slowly try to get
	 * back to our previous queue depth.
	 */
	if ((dev->openings + dev->active) < dev->maxtags
	 && dev->tag_success_count > AHC_TAG_SUCCESS_INTERVAL) {
		dev->tag_success_count = 0;
		dev->openings++;
	}

Linus Torvalds's avatar
Linus Torvalds committed
4212 4213 4214 4215 4216
	if (dev->active == 0)
		dev->commands_since_idle_or_otag = 0;

	if (TAILQ_EMPTY(&dev->busyq)) {
		if ((dev->flags & AHC_DEV_UNCONFIGURED) != 0
4217 4218
		 && dev->active == 0
	 	 && (dev->flags & AHC_DEV_TIMER_ACTIVE) == 0)
Linus Torvalds's avatar
Linus Torvalds committed
4219 4220
			ahc_linux_free_device(ahc, dev);
	} else if ((dev->flags & AHC_DEV_ON_RUN_LIST) == 0) {
Linus Torvalds's avatar
Linus Torvalds committed
4221
		TAILQ_INSERT_TAIL(&ahc->platform_data->device_runq, dev, links);
Linus Torvalds's avatar
Linus Torvalds committed
4222 4223 4224 4225 4226
		dev->flags |= AHC_DEV_ON_RUN_LIST;
	}

	if ((scb->flags & SCB_RECOVERY_SCB) != 0) {
		printf("Recovery SCB completes\n");
4227 4228 4229
		if (ahc_get_transaction_status(scb) == CAM_BDR_SENT
		 || ahc_get_transaction_status(scb) == CAM_REQ_ABORTED)
			ahc_set_transaction_status(scb, CAM_CMD_TIMEOUT);
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4230 4231
		if ((ahc->platform_data->flags & AHC_UP_EH_SEMAPHORE) != 0) {
			ahc->platform_data->flags &= ~AHC_UP_EH_SEMAPHORE;
4232 4233
			up(&ahc->platform_data->eh_sem);
		}
Linus Torvalds's avatar
Linus Torvalds committed
4234 4235 4236
	}

	ahc_free_scb(ahc, scb);
Linus Torvalds's avatar
Linus Torvalds committed
4237
	ahc_linux_queue_cmd_complete(ahc, cmd);
4238 4239 4240 4241 4242 4243 4244

	if ((ahc->platform_data->flags & AHC_DV_WAIT_SIMQ_EMPTY) != 0
	 && LIST_FIRST(&ahc->pending_scbs) == NULL) {
		ahc->platform_data->flags &= ~AHC_DV_WAIT_SIMQ_EMPTY;
		up(&ahc->platform_data->dv_sem);
	}
		
Linus Torvalds's avatar
Linus Torvalds committed
4245 4246 4247
}

static void
Linus Torvalds's avatar
Linus Torvalds committed
4248 4249
ahc_linux_handle_scsi_status(struct ahc_softc *ahc,
			     struct ahc_linux_device *dev, struct scb *scb)
Linus Torvalds's avatar
Linus Torvalds committed
4250
{
4251 4252 4253 4254 4255 4256 4257 4258
	struct	ahc_devinfo devinfo;

	ahc_compile_devinfo(&devinfo,
			    ahc->our_id,
			    dev->target->target, dev->lun,
			    dev->target->channel == 0 ? 'A' : 'B',
			    ROLE_INITIATOR);
	
Linus Torvalds's avatar
Linus Torvalds committed
4259 4260 4261 4262 4263 4264 4265 4266 4267 4268 4269 4270 4271
	/*
	 * We don't currently trust the mid-layer to
	 * properly deal with queue full or busy.  So,
	 * when one occurs, we tell the mid-layer to
	 * unconditionally requeue the command to us
	 * so that we can retry it ourselves.  We also
	 * implement our own throttling mechanism so
	 * we don't clobber the device with too many
	 * commands.
	 */
	switch (ahc_get_scsi_status(scb)) {
	default:
		break;
4272 4273 4274 4275 4276 4277 4278 4279 4280 4281 4282 4283 4284 4285 4286 4287 4288 4289 4290 4291 4292 4293
	case SCSI_STATUS_CHECK_COND:
	case SCSI_STATUS_CMD_TERMINATED:
	{
		Scsi_Cmnd *cmd;

		/*
		 * Copy sense information to the OS's cmd
		 * structure if it is available.
		 */
		cmd = scb->io_ctx;
		if (scb->flags & SCB_SENSE) {
			u_int sense_size;

			sense_size = MIN(sizeof(struct scsi_sense_data)
				       - ahc_get_sense_residual(scb),
					 sizeof(cmd->sense_buffer));
			memcpy(cmd->sense_buffer,
			       ahc_get_sense_buf(ahc, scb), sense_size);
			if (sense_size < sizeof(cmd->sense_buffer))
				memset(&cmd->sense_buffer[sense_size], 0,
				       sizeof(cmd->sense_buffer) - sense_size);
			cmd->result |= (DRIVER_SENSE << 24);
4294 4295 4296 4297 4298 4299
#ifdef AHC_DEBUG
			if (ahc_debug & AHC_SHOW_SENSE) {
				int i;

				printf("Copied %d bytes of sense data:",
				       sense_size);
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4300 4301 4302 4303 4304
				for (i = 0; i < sense_size; i++) {
					if ((i & 0xF) == 0)
						printf("\n");
					printf("0x%x ", cmd->sense_buffer[i]);
				}
4305 4306 4307
				printf("\n");
			}
#endif
4308 4309 4310
		}
		break;
	}
Linus Torvalds's avatar
Linus Torvalds committed
4311 4312 4313 4314 4315 4316 4317 4318 4319 4320 4321 4322 4323 4324 4325 4326 4327 4328 4329 4330 4331 4332 4333 4334 4335 4336 4337 4338 4339 4340 4341 4342 4343 4344 4345 4346 4347 4348 4349 4350 4351 4352 4353 4354 4355
	case SCSI_STATUS_QUEUE_FULL:
	{
		/*
		 * By the time the core driver has returned this
		 * command, all other commands that were queued
		 * to us but not the device have been returned.
		 * This ensures that dev->active is equal to
		 * the number of commands actually queued to
		 * the device.
		 */
		dev->tag_success_count = 0;
		if (dev->active != 0) {
			/*
			 * Drop our opening count to the number
			 * of commands currently outstanding.
			 */
			dev->openings = 0;
/*
			ahc_print_path(ahc, scb);
			printf("Dropping tag count to %d\n", dev->active);
 */
			if (dev->active == dev->tags_on_last_queuefull) {

				dev->last_queuefull_same_count++;
				/*
				 * If we repeatedly see a queue full
				 * at the same queue depth, this
				 * device has a fixed number of tag
				 * slots.  Lock in this tag depth
				 * so we stop seeing queue fulls from
				 * this device.
				 */
				if (dev->last_queuefull_same_count
				 == AHC_LOCK_TAGS_COUNT) {
					dev->maxtags = dev->active;
					ahc_print_path(ahc, scb);
					printf("Locking max tag count at %d\n",
					       dev->active);
				}
			} else {
				dev->tags_on_last_queuefull = dev->active;
				dev->last_queuefull_same_count = 0;
			}
			ahc_set_transaction_status(scb, CAM_REQUEUE_REQ);
			ahc_set_scsi_status(scb, SCSI_STATUS_OK);
4356
			ahc_platform_set_tags(ahc, &devinfo,
4357 4358
				     (dev->flags & AHC_DEV_Q_BASIC)
				   ? AHC_QUEUE_BASIC : AHC_QUEUE_TAGGED);
Linus Torvalds's avatar
Linus Torvalds committed
4359 4360 4361 4362
			break;
		}
		/*
		 * Drop down to a single opening, and treat this
4363
		 * as if the target returned BUSY SCSI status.
Linus Torvalds's avatar
Linus Torvalds committed
4364 4365
		 */
		dev->openings = 1;
4366
		ahc_set_scsi_status(scb, SCSI_STATUS_BUSY);
4367
		ahc_platform_set_tags(ahc, &devinfo,
4368 4369
			     (dev->flags & AHC_DEV_Q_BASIC)
			   ? AHC_QUEUE_BASIC : AHC_QUEUE_TAGGED);
Linus Torvalds's avatar
Linus Torvalds committed
4370 4371 4372
		/* FALLTHROUGH */
	}
	case SCSI_STATUS_BUSY:
4373
	{
Linus Torvalds's avatar
Linus Torvalds committed
4374
		/*
4375 4376
		 * Set a short timer to defer sending commands for
		 * a bit since Linux will not delay in this case.
Linus Torvalds's avatar
Linus Torvalds committed
4377
		 */
4378 4379 4380 4381 4382 4383 4384 4385 4386 4387 4388 4389 4390
		if ((dev->flags & AHC_DEV_TIMER_ACTIVE) != 0) {
			printf("%s:%c:%d: Device Timer still active during "
			       "busy processing\n", ahc_name(ahc),
				dev->target->channel, dev->target->target);
			break;
		}
		dev->flags |= AHC_DEV_TIMER_ACTIVE;
		dev->qfrozen++;
		init_timer(&dev->timer);
		dev->timer.data = (u_long)dev;
		dev->timer.expires = jiffies + (HZ/2);
		dev->timer.function = ahc_linux_dev_timed_unfreeze;
		add_timer(&dev->timer);
Linus Torvalds's avatar
Linus Torvalds committed
4391 4392
		break;
	}
4393
	}
Linus Torvalds's avatar
Linus Torvalds committed
4394 4395 4396
}

static void
4397
ahc_linux_queue_cmd_complete(struct ahc_softc *ahc, Scsi_Cmnd *cmd)
Linus Torvalds's avatar
Linus Torvalds committed
4398
{
4399 4400 4401 4402 4403 4404 4405 4406 4407 4408 4409 4410 4411 4412 4413
	/*
	 * Typically, the complete queue has very few entries
	 * queued to it before the queue is emptied by
	 * ahc_linux_run_complete_queue, so sorting the entries
	 * by generation number should be inexpensive.
	 * We perform the sort so that commands that complete
	 * with an error are retuned in the order origionally
	 * queued to the controller so that any subsequent retries
	 * are performed in order.  The underlying ahc routines do
	 * not guarantee the order that aborted commands will be
	 * returned to us.
	 */
	struct ahc_completeq *completeq;
	struct ahc_cmd *list_cmd;
	struct ahc_cmd *acmd;
Linus Torvalds's avatar
Linus Torvalds committed
4414

4415 4416 4417 4418 4419 4420 4421 4422 4423 4424 4425 4426 4427 4428 4429 4430 4431 4432 4433 4434 4435 4436 4437 4438 4439 4440 4441 4442 4443 4444 4445 4446 4447 4448 4449 4450 4451 4452 4453 4454 4455 4456 4457 4458 4459 4460 4461 4462 4463 4464 4465 4466 4467 4468 4469 4470 4471 4472 4473 4474 4475 4476 4477
	/*
	 * Map CAM error codes into Linux Error codes.  We
	 * avoid the conversion so that the DV code has the
	 * full error information available when making
	 * state change decisions.
	 */
	if (AHC_DV_CMD(cmd) == FALSE) {
		u_int new_status;

		switch (ahc_cmd_get_transaction_status(cmd)) {
		case CAM_REQ_INPROG:
		case CAM_REQ_CMP:
		case CAM_SCSI_STATUS_ERROR:
			new_status = DID_OK;
			break;
		case CAM_REQ_ABORTED:
			new_status = DID_ABORT;
			break;
		case CAM_BUSY:
			new_status = DID_BUS_BUSY;
			break;
		case CAM_REQ_INVALID:
		case CAM_PATH_INVALID:
			new_status = DID_BAD_TARGET;
			break;
		case CAM_SEL_TIMEOUT:
			new_status = DID_NO_CONNECT;
			break;
		case CAM_SCSI_BUS_RESET:
		case CAM_BDR_SENT:
			new_status = DID_RESET;
			break;
		case CAM_UNCOR_PARITY:
			new_status = DID_PARITY;
			break;
		case CAM_CMD_TIMEOUT:
			new_status = DID_TIME_OUT;
			break;
		case CAM_UA_ABORT:
		case CAM_REQ_CMP_ERR:
		case CAM_AUTOSENSE_FAIL:
		case CAM_NO_HBA:
		case CAM_DATA_RUN_ERR:
		case CAM_UNEXP_BUSFREE:
		case CAM_SEQUENCE_FAIL:
		case CAM_CCB_LEN_ERR:
		case CAM_PROVIDE_FAIL:
		case CAM_REQ_TERMIO:
		case CAM_UNREC_HBA_ERROR:
		case CAM_REQ_TOO_BIG:
			new_status = DID_ERROR;
			break;
		case CAM_REQUEUE_REQ:
			/*
			 * If we want the request requeued, make sure there
			 * are sufficent retries.  In the old scsi error code,
			 * we used to be able to specify a result code that
			 * bypassed the retry count.  Now we must use this
			 * hack.  We also "fake" a check condition with
			 * a sense code of ABORTED COMMAND.  This seems to
			 * evoke a retry even if this command is being sent
			 * via the eh thread.  Ick!  Ick!  Ick!
			 */
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4478 4479
			if (cmd->retries > 0)
				cmd->retries--;
4480 4481 4482 4483 4484 4485 4486 4487 4488 4489 4490 4491
			new_status = DID_OK;
			ahc_cmd_set_scsi_status(cmd, SCSI_STATUS_CHECK_COND);
			cmd->result |= (DRIVER_SENSE << 24);
			memset(cmd->sense_buffer, 0,
			       sizeof(cmd->sense_buffer));
			cmd->sense_buffer[0] = SSD_ERRCODE_VALID
					     | SSD_CURRENT_ERROR;
			cmd->sense_buffer[2] = SSD_KEY_ABORTED_COMMAND;
			break;
		default:
			/* We should never get here */
			new_status = DID_ERROR;
Linus Torvalds's avatar
Linus Torvalds committed
4492 4493 4494
			break;
		}

4495 4496
		ahc_cmd_set_transaction_status(cmd, new_status);
	}
Linus Torvalds's avatar
Linus Torvalds committed
4497

4498 4499 4500 4501 4502 4503 4504 4505 4506 4507 4508 4509
	completeq = &ahc->platform_data->completeq;
	list_cmd = TAILQ_FIRST(completeq);
	acmd = (struct ahc_cmd *)cmd;
	while (list_cmd != NULL
	    && acmd_scsi_cmd(list_cmd).serial_number
	     < acmd_scsi_cmd(acmd).serial_number)
		list_cmd = TAILQ_NEXT(list_cmd, acmd_links.tqe);
	if (list_cmd != NULL)
		TAILQ_INSERT_BEFORE(list_cmd, acmd, acmd_links.tqe);
	else
		TAILQ_INSERT_TAIL(completeq, acmd, acmd_links.tqe);
}
Linus Torvalds's avatar
Linus Torvalds committed
4510

4511 4512 4513 4514 4515 4516 4517 4518 4519 4520 4521 4522 4523 4524 4525 4526 4527 4528
static void
ahc_linux_filter_inquiry(struct ahc_softc *ahc, struct ahc_devinfo *devinfo)
{
	struct	scsi_inquiry_data *sid;
	struct	ahc_initiator_tinfo *tinfo;
	struct	ahc_transinfo *user;
	struct	ahc_transinfo *goal;
	struct	ahc_transinfo *curr;
	struct	ahc_tmode_tstate *tstate;
	struct	ahc_syncrate *syncrate;
	struct	ahc_linux_device *dev;
	u_int	maxsync;
	u_int	width;
	u_int	period;
	u_int	offset;
	u_int	ppr_options;
	u_int	trans_version;
	u_int	prot_version;
Linus Torvalds's avatar
Linus Torvalds committed
4529

4530 4531 4532 4533 4534 4535 4536 4537 4538 4539
	/*
	 * Determine if this lun actually exists.  If so,
	 * hold on to its corresponding device structure.
	 * If not, make sure we release the device and
	 * don't bother processing the rest of this inquiry
	 * command.
	 */
	dev = ahc_linux_get_device(ahc, devinfo->channel - 'A',
				   devinfo->target, devinfo->lun,
				   /*alloc*/TRUE);
Linus Torvalds's avatar
Linus Torvalds committed
4540

4541 4542
	sid = (struct scsi_inquiry_data *)dev->target->inq_data;
	if (SID_QUAL(sid) == SID_QUAL_LU_CONNECTED) {
Linus Torvalds's avatar
Linus Torvalds committed
4543

4544 4545 4546 4547
		dev->flags &= ~AHC_DEV_UNCONFIGURED;
	} else {
		dev->flags |= AHC_DEV_UNCONFIGURED;
		return;
Linus Torvalds's avatar
Linus Torvalds committed
4548
	}
4549 4550 4551 4552 4553 4554 4555 4556 4557 4558 4559 4560 4561 4562 4563 4564 4565 4566 4567 4568 4569 4570 4571 4572 4573 4574 4575 4576 4577 4578 4579 4580 4581 4582 4583 4584 4585 4586 4587 4588 4589 4590 4591 4592 4593 4594 4595 4596 4597 4598 4599 4600 4601 4602 4603 4604 4605 4606 4607 4608 4609

	/*
	 * Update our notion of this device's transfer
	 * negotiation capabilities.
	 */
	tinfo = ahc_fetch_transinfo(ahc, devinfo->channel,
				    devinfo->our_scsiid,
				    devinfo->target, &tstate);
	user = &tinfo->user;
	goal = &tinfo->goal;
	curr = &tinfo->curr;
	width = user->width;
	period = user->period;
	offset = user->offset;
	ppr_options = user->ppr_options;
	trans_version = user->transport_version;
	prot_version = MIN(user->protocol_version, SID_ANSI_REV(sid));

	/*
	 * Only attempt SPI3/4 once we've verified that
	 * the device claims to support SPI3/4 features.
	 */
	if (prot_version < SCSI_REV_2)
		trans_version = SID_ANSI_REV(sid);
	else
		trans_version = SCSI_REV_2;

	if ((sid->flags & SID_WBus16) == 0)
		width = MSG_EXT_WDTR_BUS_8_BIT;
	if ((sid->flags & SID_Sync) == 0) {
		period = 0;
		offset = 0;
		ppr_options = 0;
	}
	if ((sid->spi3data & SID_SPI_QAS) == 0)
		ppr_options &= ~MSG_EXT_PPR_QAS_REQ;
	if ((sid->spi3data & SID_SPI_CLOCK_DT) == 0)
		ppr_options &= MSG_EXT_PPR_QAS_REQ;
	if ((sid->spi3data & SID_SPI_IUS) == 0)
		ppr_options &= (MSG_EXT_PPR_DT_REQ
			      | MSG_EXT_PPR_QAS_REQ);

	if (prot_version > SCSI_REV_2
	 && ppr_options != 0)
		trans_version = user->transport_version;

	ahc_validate_width(ahc, /*tinfo limit*/NULL, &width, ROLE_UNKNOWN);
	if ((ahc->features & AHC_ULTRA2) != 0)
		maxsync = AHC_SYNCRATE_DT;
	else if ((ahc->features & AHC_ULTRA) != 0)
		maxsync = AHC_SYNCRATE_ULTRA;
	else
		maxsync = AHC_SYNCRATE_FAST;

	syncrate = ahc_find_syncrate(ahc, &period, &ppr_options, maxsync);
	ahc_validate_offset(ahc, /*tinfo limit*/NULL, syncrate,
			    &offset, width, ROLE_UNKNOWN);
	if (offset == 0 || period == 0) {
		period = 0;
		offset = 0;
		ppr_options = 0;
Linus Torvalds's avatar
Linus Torvalds committed
4610
	}
4611 4612 4613 4614 4615 4616 4617
	/* Apply our filtered user settings. */
	curr->transport_version = trans_version;
	curr->protocol_version = prot_version;
	ahc_set_width(ahc, devinfo, width, AHC_TRANS_GOAL, /*paused*/FALSE);
	ahc_set_syncrate(ahc, devinfo, syncrate, period,
			 offset, ppr_options, AHC_TRANS_GOAL,
			 /*paused*/FALSE);
Linus Torvalds's avatar
Linus Torvalds committed
4618 4619 4620
}

static void
Linus Torvalds's avatar
Linus Torvalds committed
4621
ahc_linux_sem_timeout(u_long arg)
Linus Torvalds's avatar
Linus Torvalds committed
4622
{
4623 4624
	struct	ahc_softc *ahc;
	u_long	s;
Linus Torvalds's avatar
Linus Torvalds committed
4625

4626 4627 4628 4629 4630 4631 4632 4633
	ahc = (struct ahc_softc *)arg;

	ahc_lock(ahc, &s);
	if ((ahc->platform_data->flags & AHC_UP_EH_SEMAPHORE) != 0) {
		ahc->platform_data->flags &= ~AHC_UP_EH_SEMAPHORE;
		up(&ahc->platform_data->eh_sem);
	}
	ahc_unlock(ahc, &s);
Linus Torvalds's avatar
Linus Torvalds committed
4634 4635 4636
}

static void
4637
ahc_linux_freeze_simq(struct ahc_softc *ahc)
Linus Torvalds's avatar
Linus Torvalds committed
4638 4639
{
	ahc->platform_data->qfrozen++;
4640
	if (ahc->platform_data->qfrozen == 1) {
Linus Torvalds's avatar
Linus Torvalds committed
4641
		scsi_block_requests(ahc->platform_data->host);
4642 4643 4644 4645 4646 4647

		/* XXX What about Twin channels? */
		ahc_platform_abort_scbs(ahc, CAM_TARGET_WILDCARD, ALL_CHANNELS,
					CAM_LUN_WILDCARD, SCB_LIST_NULL,
					ROLE_INITIATOR, CAM_REQUEUE_REQ);
	}
Linus Torvalds's avatar
Linus Torvalds committed
4648 4649 4650
}

static void
4651
ahc_linux_release_simq(u_long arg)
Linus Torvalds's avatar
Linus Torvalds committed
4652 4653 4654
{
	struct ahc_softc *ahc;
	u_long s;
Linus Torvalds's avatar
Linus Torvalds committed
4655
	int    unblock_reqs;
Linus Torvalds's avatar
Linus Torvalds committed
4656 4657

	ahc = (struct ahc_softc *)arg;
4658

Linus Torvalds's avatar
Linus Torvalds committed
4659
	unblock_reqs = 0;
Linus Torvalds's avatar
Linus Torvalds committed
4660 4661 4662
	ahc_lock(ahc, &s);
	if (ahc->platform_data->qfrozen > 0)
		ahc->platform_data->qfrozen--;
4663
	if (ahc->platform_data->qfrozen == 0)
Linus Torvalds's avatar
Linus Torvalds committed
4664
		unblock_reqs = 1;
4665 4666 4667 4668
	if (AHC_DV_SIMQ_FROZEN(ahc)
	 && ((ahc->platform_data->flags & AHC_DV_WAIT_SIMQ_RELEASE) != 0)) {
		ahc->platform_data->flags &= ~AHC_DV_WAIT_SIMQ_RELEASE;
		up(&ahc->platform_data->dv_sem);
Linus Torvalds's avatar
Linus Torvalds committed
4669
	}
4670
	ahc_schedule_runq(ahc);
Linus Torvalds's avatar
Linus Torvalds committed
4671
	ahc_unlock(ahc, &s);
Linus Torvalds's avatar
Linus Torvalds committed
4672 4673 4674 4675 4676 4677
	/*
	 * There is still a race here.  The mid-layer
	 * should keep its own freeze count and use
	 * a bottom half handler to run the queues
	 * so we can unblock with our own lock held.
	 */
4678
	if (unblock_reqs)
Linus Torvalds's avatar
Linus Torvalds committed
4679
		scsi_unblock_requests(ahc->platform_data->host);
4680 4681 4682 4683 4684 4685 4686 4687 4688 4689 4690 4691 4692 4693 4694 4695 4696 4697
}

static void
ahc_linux_dev_timed_unfreeze(u_long arg)
{
	struct ahc_linux_device *dev;
	struct ahc_softc *ahc;
	u_long s;

	dev = (struct ahc_linux_device *)arg;
	ahc = dev->target->ahc;
	ahc_lock(ahc, &s);
	dev->flags &= ~AHC_DEV_TIMER_ACTIVE;
	if (dev->qfrozen > 0)
		dev->qfrozen--;
	if (dev->qfrozen == 0
	 && (dev->flags & AHC_DEV_ON_RUN_LIST) == 0)
		ahc_linux_run_device_queue(ahc, dev);
4698 4699
	if (TAILQ_EMPTY(&dev->busyq)
	 && dev->active == 0)
Andrew Morton's avatar
Andrew Morton committed
4700
		__ahc_linux_free_device(ahc, dev);
4701
	ahc_unlock(ahc, &s);
Linus Torvalds's avatar
Linus Torvalds committed
4702 4703 4704
}

static int
Linus Torvalds's avatar
Linus Torvalds committed
4705
ahc_linux_queue_recovery_cmd(Scsi_Cmnd *cmd, scb_flag flag)
Linus Torvalds's avatar
Linus Torvalds committed
4706 4707 4708 4709 4710 4711 4712 4713 4714 4715
{
	struct ahc_softc *ahc;
	struct ahc_cmd *acmd;
	struct ahc_cmd *list_acmd;
	struct ahc_linux_device *dev;
	struct scb *pending_scb;
	u_long s;
	u_int  saved_scbptr;
	u_int  active_scb_index;
	u_int  last_phase;
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4716
	u_int  saved_scsiid;
4717
	u_int  cdb_byte;
Linus Torvalds's avatar
Linus Torvalds committed
4718
	int    retval;
4719
	int    was_paused;
Linus Torvalds's avatar
Linus Torvalds committed
4720 4721 4722 4723
	int    paused;
	int    wait;
	int    disconnected;

4724
	pending_scb = NULL;
Linus Torvalds's avatar
Linus Torvalds committed
4725 4726
	paused = FALSE;
	wait = FALSE;
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4727
	ahc = *(struct ahc_softc **)cmd->device->host->hostdata;
Linus Torvalds's avatar
Linus Torvalds committed
4728 4729 4730
	acmd = (struct ahc_cmd *)cmd;

	printf("%s:%d:%d:%d: Attempting to queue a%s message\n",
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4731 4732
	       ahc_name(ahc), cmd->device->channel,
	       cmd->device->id, cmd->device->lun,
Linus Torvalds's avatar
Linus Torvalds committed
4733 4734
	       flag == SCB_ABORT ? "n ABORT" : " TARGET RESET");

4735 4736 4737 4738 4739
	printf("CDB:");
	for (cdb_byte = 0; cdb_byte < cmd->cmd_len; cdb_byte++)
		printf(" 0x%x", cmd->cmnd[cdb_byte]);
	printf("\n");

4740 4741 4742 4743 4744 4745 4746 4747 4748 4749 4750 4751 4752 4753
	/*
	 * In all versions of Linux, we have to work around
	 * a major flaw in how the mid-layer is locked down
	 * if we are to sleep successfully in our error handler
	 * while allowing our interrupt handler to run.  Since
	 * the midlayer acquires either the io_request_lock or
	 * our lock prior to calling us, we must use the
	 * spin_unlock_irq() method for unlocking our lock.
	 * This will force interrupts to be enabled on the
	 * current CPU.  Since the EH thread should not have
	 * been running with CPU interrupts disabled other than
	 * by acquiring either the io_request_lock or our own
	 * lock, this *should* be safe.
	 */
4754
	ahc_midlayer_entrypoint_lock(ahc, &s);
Linus Torvalds's avatar
Linus Torvalds committed
4755 4756 4757 4758 4759 4760

	/*
	 * First determine if we currently own this command.
	 * Start by searching the device queue.  If not found
	 * there, check the pending_scb list.  If not found
	 * at all, and the system wanted us to just abort the
4761
	 * command, return success.
Linus Torvalds's avatar
Linus Torvalds committed
4762
	 */
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4763 4764
	dev = ahc_linux_get_device(ahc, cmd->device->channel, cmd->device->id,
				   cmd->device->lun, /*alloc*/FALSE);
Linus Torvalds's avatar
Linus Torvalds committed
4765 4766 4767 4768 4769 4770 4771

	if (dev == NULL) {
		/*
		 * No target device for this command exists,
		 * so we must not still own the command.
		 */
		printf("%s:%d:%d:%d: Is not an active device\n",
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4772 4773
		       ahc_name(ahc), cmd->device->channel, cmd->device->id,
		       cmd->device->lun);
Linus Torvalds's avatar
Linus Torvalds committed
4774 4775 4776 4777 4778 4779 4780 4781 4782 4783 4784
		retval = SUCCESS;
		goto no_cmd;
	}

	TAILQ_FOREACH(list_acmd, &dev->busyq, acmd_links.tqe) {
		if (list_acmd == acmd)
			break;
	}

	if (list_acmd != NULL) {
		printf("%s:%d:%d:%d: Command found on device queue\n",
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4785 4786
		       ahc_name(ahc), cmd->device->channel, cmd->device->id,
		       cmd->device->lun);
Linus Torvalds's avatar
Linus Torvalds committed
4787 4788 4789
		if (flag == SCB_ABORT) {
			TAILQ_REMOVE(&dev->busyq, list_acmd, acmd_links.tqe);
			cmd->result = DID_ABORT << 16;
Linus Torvalds's avatar
Linus Torvalds committed
4790
			ahc_linux_queue_cmd_complete(ahc, cmd);
Linus Torvalds's avatar
Linus Torvalds committed
4791 4792 4793 4794 4795
			retval = SUCCESS;
			goto done;
		}
	}

4796
	if ((dev->flags & (AHC_DEV_Q_BASIC|AHC_DEV_Q_TAGGED)) == 0
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4797 4798 4799
	 && ahc_search_untagged_queues(ahc, cmd, cmd->device->id,
				       cmd->device->channel + 'A',
				       cmd->device->lun,
4800 4801
				       CAM_REQ_ABORTED, SEARCH_COMPLETE) != 0) {
		printf("%s:%d:%d:%d: Command found on untagged queue\n",
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4802 4803
		       ahc_name(ahc), cmd->device->channel, cmd->device->id,
		       cmd->device->lun);
4804 4805 4806 4807
		retval = SUCCESS;
		goto done;
	}

Linus Torvalds's avatar
Linus Torvalds committed
4808 4809 4810 4811 4812 4813 4814 4815 4816 4817 4818 4819
	/*
	 * See if we can find a matching cmd in the pending list.
	 */
	LIST_FOREACH(pending_scb, &ahc->pending_scbs, pending_links) {
		if (pending_scb->io_ctx == cmd)
			break;
	}

	if (pending_scb == NULL && flag == SCB_DEVICE_RESET) {

		/* Any SCB for this device will do for a target reset */
		LIST_FOREACH(pending_scb, &ahc->pending_scbs, pending_links) {
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4820 4821 4822
		  	if (ahc_match_scb(ahc, pending_scb, cmd->device->id,
					  cmd->device->channel + 'A',
					  CAM_LUN_WILDCARD,
Linus Torvalds's avatar
Linus Torvalds committed
4823 4824 4825 4826 4827 4828 4829
					  SCB_LIST_NULL, ROLE_INITIATOR) == 0)
				break;
		}
	}

	if (pending_scb == NULL) {
		printf("%s:%d:%d:%d: Command not found\n",
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4830 4831
		       ahc_name(ahc), cmd->device->channel, cmd->device->id,
		       cmd->device->lun);
Linus Torvalds's avatar
Linus Torvalds committed
4832 4833 4834 4835 4836 4837 4838 4839 4840 4841 4842 4843 4844
		goto no_cmd;
	}

	if ((pending_scb->flags & SCB_RECOVERY_SCB) != 0) {
		/*
		 * We can't queue two recovery actions using the same SCB
		 */
		retval = FAILED;
		goto  done;
	}

	/*
	 * Ensure that the card doesn't do anything
4845 4846
	 * behind our back and that we didn't "just" miss
	 * an interrupt that would affect this cmd.
Linus Torvalds's avatar
Linus Torvalds committed
4847
	 */
4848
	was_paused = ahc_is_paused(ahc);
4849
	ahc_pause_and_flushwork(ahc);
4850
	paused = TRUE;
Linus Torvalds's avatar
Linus Torvalds committed
4851 4852 4853

	if ((pending_scb->flags & SCB_ACTIVE) == 0) {
		printf("%s:%d:%d:%d: Command already completed\n",
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4854 4855
		       ahc_name(ahc), cmd->device->channel, cmd->device->id,
		       cmd->device->lun);
Linus Torvalds's avatar
Linus Torvalds committed
4856 4857 4858
		goto no_cmd;
	}

4859
	printf("%s: At time of recovery, card was %spaused\n",
4860
	       ahc_name(ahc), was_paused ? "" : "not ");
4861 4862
	ahc_dump_card_state(ahc);

Linus Torvalds's avatar
Linus Torvalds committed
4863 4864
	disconnected = TRUE;
	if (flag == SCB_ABORT) {
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4865 4866 4867 4868
		if (ahc_search_qinfifo(ahc, cmd->device->id,
				       cmd->device->channel + 'A',
				       cmd->device->lun,
				       pending_scb->hscb->tag,
Linus Torvalds's avatar
Linus Torvalds committed
4869 4870 4871
				       ROLE_INITIATOR, CAM_REQ_ABORTED,
				       SEARCH_COMPLETE) > 0) {
			printf("%s:%d:%d:%d: Cmd aborted from QINFIFO\n",
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4872 4873
			       ahc_name(ahc), cmd->device->channel,
					cmd->device->id, cmd->device->lun);
Linus Torvalds's avatar
Linus Torvalds committed
4874 4875 4876
			retval = SUCCESS;
			goto done;
		}
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4877 4878 4879
	} else if (ahc_search_qinfifo(ahc, cmd->device->id,
				      cmd->device->channel + 'A',
				      cmd->device->lun, pending_scb->hscb->tag,
Linus Torvalds's avatar
Linus Torvalds committed
4880 4881 4882 4883 4884
				      ROLE_INITIATOR, /*status*/0,
				      SEARCH_COUNT) > 0) {
		disconnected = FALSE;
	}

4885
	if (disconnected && (ahc_inb(ahc, SEQ_FLAGS) & NOT_IDENTIFIED) == 0) {
4886 4887 4888 4889 4890 4891 4892
		struct scb *bus_scb;

		bus_scb = ahc_lookup_scb(ahc, ahc_inb(ahc, SCB_TAG));
		if (bus_scb == pending_scb)
			disconnected = FALSE;
		else if (flag != SCB_ABORT
		      && ahc_inb(ahc, SAVED_SCSIID) == pending_scb->hscb->scsiid
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4893
		      && ahc_inb(ahc, SAVED_LUN) == SCB_GET_LUN(pending_scb))
4894 4895 4896
			disconnected = FALSE;
	}

Linus Torvalds's avatar
Linus Torvalds committed
4897 4898 4899 4900 4901 4902 4903 4904 4905 4906 4907
	/*
	 * At this point, pending_scb is the scb associated with the
	 * passed in command.  That command is currently active on the
	 * bus, is in the disconnected state, or we're hoping to find
	 * a command for the same target active on the bus to abuse to
	 * send a BDR.  Queue the appropriate message based on which of
	 * these states we are in.
	 */
	last_phase = ahc_inb(ahc, LASTPHASE);
	saved_scbptr = ahc_inb(ahc, SCBPTR);
	active_scb_index = ahc_inb(ahc, SCB_TAG);
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4908
	saved_scsiid = ahc_inb(ahc, SAVED_SCSIID);
Linus Torvalds's avatar
Linus Torvalds committed
4909 4910 4911
	if (last_phase != P_BUSFREE
	 && (pending_scb->hscb->tag == active_scb_index
	  || (flag == SCB_DEVICE_RESET
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4912
	   && SCSIID_TARGET(ahc, saved_scsiid) == cmd->device->id))) {
Linus Torvalds's avatar
Linus Torvalds committed
4913 4914 4915 4916 4917 4918 4919 4920 4921 4922

		/*
		 * We're active on the bus, so assert ATN
		 * and hope that the target responds.
		 */
		pending_scb = ahc_lookup_scb(ahc, active_scb_index);
		pending_scb->flags |= SCB_RECOVERY_SCB|flag;
		ahc_outb(ahc, MSG_OUT, HOST_MSG);
		ahc_outb(ahc, SCSISIGO, last_phase|ATNO);
		printf("%s:%d:%d:%d: Device is active, asserting ATN\n",
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4923 4924
		       ahc_name(ahc), cmd->device->channel, cmd->device->id,
		       cmd->device->lun);
Linus Torvalds's avatar
Linus Torvalds committed
4925 4926 4927 4928 4929 4930 4931
		wait = TRUE;
	} else if (disconnected) {

		/*
		 * Actually re-queue this SCB in an attempt
		 * to select the device before it reconnects.
		 * In either case (selection or reselection),
Linus Torvalds's avatar
Linus Torvalds committed
4932
		 * we will now issue the approprate message
Linus Torvalds's avatar
Linus Torvalds committed
4933 4934 4935 4936 4937 4938 4939 4940 4941 4942 4943 4944 4945 4946 4947 4948 4949 4950 4951 4952 4953
		 * to the timed-out device.
		 *
		 * Set the MK_MESSAGE control bit indicating
		 * that we desire to send a message.  We
		 * also set the disconnected flag since
		 * in the paging case there is no guarantee
		 * that our SCB control byte matches the
		 * version on the card.  We don't want the
		 * sequencer to abort the command thinking
		 * an unsolicited reselection occurred.
		 */
		pending_scb->hscb->control |= MK_MESSAGE|DISCONNECTED;
		pending_scb->flags |= SCB_RECOVERY_SCB|flag;

		/*
		 * Remove any cached copy of this SCB in the
		 * disconnected list in preparation for the
		 * queuing of our abort SCB.  We use the
		 * same element in the SCB, SCB_NEXT, for
		 * both the qinfifo and the disconnected list.
		 */
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4954 4955 4956
		ahc_search_disc_list(ahc, cmd->device->id,
				     cmd->device->channel + 'A',
				     cmd->device->lun, pending_scb->hscb->tag,
Linus Torvalds's avatar
Linus Torvalds committed
4957 4958 4959 4960 4961 4962 4963 4964 4965 4966 4967 4968 4969 4970 4971 4972 4973 4974 4975 4976 4977 4978
				     /*stop_on_first*/TRUE,
				     /*remove*/TRUE,
				     /*save_state*/FALSE);

		/*
		 * In the non-paging case, the sequencer will
		 * never re-reference the in-core SCB.
		 * To make sure we are notified during
		 * reslection, set the MK_MESSAGE flag in
		 * the card's copy of the SCB.
		 */
		if ((ahc->flags & AHC_PAGESCBS) == 0) {
			ahc_outb(ahc, SCBPTR, pending_scb->hscb->tag);
			ahc_outb(ahc, SCB_CONTROL,
				 ahc_inb(ahc, SCB_CONTROL)|MK_MESSAGE);
		}

		/*
		 * Clear out any entries in the QINFIFO first
		 * so we are the next SCB for this target
		 * to run.
		 */
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4979 4980 4981 4982 4983
		ahc_search_qinfifo(ahc, cmd->device->id,
				   cmd->device->channel + 'A',
				   cmd->device->lun, SCB_LIST_NULL,
				   ROLE_INITIATOR, CAM_REQUEUE_REQ,
				   SEARCH_COMPLETE);
Linus Torvalds's avatar
Linus Torvalds committed
4984 4985
		ahc_qinfifo_requeue_tail(ahc, pending_scb);
		ahc_outb(ahc, SCBPTR, saved_scbptr);
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4986 4987
		ahc_print_path(ahc, pending_scb);
		printf("Device is disconnected, re-queuing SCB\n");
Linus Torvalds's avatar
Linus Torvalds committed
4988 4989 4990
		wait = TRUE;
	} else {
		printf("%s:%d:%d:%d: Unable to deliver message\n",
James Bottomley's avatar
James Bottomley committed
4991
		       ahc_name(ahc), cmd->device->channel, cmd->device->id,
Justin T. Gibbs's avatar
Justin T. Gibbs committed
4992
		       cmd->device->lun);
Linus Torvalds's avatar
Linus Torvalds committed
4993 4994 4995 4996 4997 4998 4999 5000 5001 5002 5003 5004 5005 5006
		retval = FAILED;
		goto done;
	}

no_cmd:
	/*
	 * Our assumption is that if we don't have the command, no
	 * recovery action was required, so we return success.  Again,
	 * the semantics of the mid-layer recovery engine are not
	 * well defined, so this may change in time.
	 */
	retval = SUCCESS;
done:
	if (paused)
Linus Torvalds's avatar
Linus Torvalds committed
5007
		ahc_unpause(ahc);
Linus Torvalds's avatar
Linus Torvalds committed
5008
	if (wait) {
Linus Torvalds's avatar
Linus Torvalds committed
5009
		struct timer_list timer;
Linus Torvalds's avatar
Linus Torvalds committed
5010 5011
		int ret;

Justin T. Gibbs's avatar
Justin T. Gibbs committed
5012
		ahc->platform_data->flags |= AHC_UP_EH_SEMAPHORE;
5013
		spin_unlock_irq(&ahc->platform_data->spin_lock);
Linus Torvalds's avatar
Linus Torvalds committed
5014
		init_timer(&timer);
5015
		timer.data = (u_long)ahc;
Linus Torvalds's avatar
Linus Torvalds committed
5016
		timer.expires = jiffies + (5 * HZ);
Linus Torvalds's avatar
Linus Torvalds committed
5017
		timer.function = ahc_linux_sem_timeout;
Linus Torvalds's avatar
Linus Torvalds committed
5018 5019 5020 5021
		add_timer(&timer);
		printf("Recovery code sleeping\n");
		down(&ahc->platform_data->eh_sem);
		printf("Recovery code awake\n");
5022
        	ret = del_timer_sync(&timer);
Linus Torvalds's avatar
Linus Torvalds committed
5023 5024 5025 5026
		if (ret == 0) {
			printf("Timer Expired\n");
			retval = FAILED;
		}
5027
		spin_lock_irq(&ahc->platform_data->spin_lock);
Linus Torvalds's avatar
Linus Torvalds committed
5028
	}
5029
	ahc_schedule_runq(ahc);
5030
	ahc_linux_run_complete_queue(ahc);
5031
	ahc_midlayer_entrypoint_unlock(ahc, &s);
Linus Torvalds's avatar
Linus Torvalds committed
5032 5033 5034 5035 5036 5037 5038 5039 5040 5041 5042 5043 5044 5045 5046 5047 5048
	return (retval);
}

void
ahc_platform_dump_card_state(struct ahc_softc *ahc)
{
	struct ahc_linux_device *dev;
	int channel;
	int maxchannel;
	int target;
	int maxtarget;
	int lun;
	int i;

	maxchannel = (ahc->features & AHC_TWIN) ? 1 : 0;
	maxtarget = (ahc->features & AHC_WIDE) ? 15 : 7;
	for (channel = 0; channel <= maxchannel; channel++) {
5049

Linus Torvalds's avatar
Linus Torvalds committed
5050
		for (target = 0; target <=maxtarget; target++) {
5051

Linus Torvalds's avatar
Linus Torvalds committed
5052 5053 5054
			for (lun = 0; lun < AHC_NUM_LUNS; lun++) {
				struct ahc_cmd *acmd;

Linus Torvalds's avatar
Linus Torvalds committed
5055 5056
				dev = ahc_linux_get_device(ahc, channel, target,
							   lun, /*alloc*/FALSE);
Linus Torvalds's avatar
Linus Torvalds committed
5057 5058 5059 5060 5061 5062 5063 5064
				if (dev == NULL)
					continue;

				printf("DevQ(%d:%d:%d): ",
				       channel, target, lun);
				i = 0;
				TAILQ_FOREACH(acmd, &dev->busyq,
					      acmd_links.tqe) {
5065
					if (i++ > AHC_SCB_MAX)
Linus Torvalds's avatar
Linus Torvalds committed
5066 5067 5068 5069 5070 5071 5072
						break;
				}
				printf("%d waiting\n", i);
			}
		}
	}
}
5073

5074
static void ahc_linux_exit(void);
5075

5076 5077
static int __init
ahc_linux_init(void)
5078
{
5079
#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,5,0)
5080 5081 5082 5083 5084
	int rc = ahc_linux_detect(&aic7xxx_driver_template);
	if (rc)
		return rc;
	ahc_linux_exit();
	return -ENODEV;
5085 5086
#else
	scsi_register_module(MODULE_SCSI_HA, &aic7xxx_driver_template);
5087
	if (aic7xxx_driver_template.present == 0) {
5088 5089 5090 5091 5092 5093 5094 5095 5096
		scsi_unregister_module(MODULE_SCSI_HA,
				       &aic7xxx_driver_template);
		return (-ENODEV);
	}

	return (0);
#endif
}

5097
static void
5098
ahc_linux_exit(void)
5099
{
5100 5101 5102 5103 5104 5105 5106 5107 5108 5109 5110 5111 5112 5113 5114
	struct ahc_softc *ahc;
	u_long l;

	/*
	 * Shutdown DV threads before going into the SCSI mid-layer.
	 * This avoids situations where the mid-layer locks the entire
	 * kernel so that waiting for our DV threads to exit leads
	 * to deadlock.
	 */
	ahc_list_lock(&l);
	TAILQ_FOREACH(ahc, &ahc_tailq, links) {

		ahc_linux_kill_dv_thread(ahc);
	}
	ahc_list_unlock(&l);
5115

5116
#if LINUX_VERSION_CODE < KERNEL_VERSION(2,5,0)
5117 5118
	/*
	 * In 2.4 we have to unregister from the PCI core _after_
5119 5120
	 * unregistering from the scsi midlayer to avoid dangling
	 * references.
5121
	 */
5122 5123
	scsi_unregister_module(MODULE_SCSI_HA, &aic7xxx_driver_template);
#endif
5124
	ahc_linux_pci_exit();
5125
	ahc_linux_eisa_exit();
5126 5127 5128 5129
}

module_init(ahc_linux_init);
module_exit(ahc_linux_exit);