Commit 28e10eb7 authored by James Bottomley's avatar James Bottomley

Automerge

parents fdefc9b9 1f5f4779
...@@ -47,6 +47,7 @@ ...@@ -47,6 +47,7 @@
************************************************************************ ************************************************************************
*/ */
#include <linux/module.h> #include <linux/module.h>
#include <linux/moduleparam.h>
#include <linux/delay.h> #include <linux/delay.h>
#include <linux/ctype.h> #include <linux/ctype.h>
#include <linux/blk.h> #include <linux/blk.h>
...@@ -209,16 +210,6 @@ char DC395x_traceoverflow[8] = { 0, 0, 0, 0, 0, 0, 0, 0 }; ...@@ -209,16 +210,6 @@ char DC395x_traceoverflow[8] = { 0, 0, 0, 0, 0, 0, 0, 0 };
#define PCI_DEVICE_ID_TEKRAM_TRMS1040 0x0391 /* Device ID */ #define PCI_DEVICE_ID_TEKRAM_TRMS1040 0x0391 /* Device ID */
#endif #endif
static struct pci_device_id dc395x_pci_tbl[] __devinitdata = {
{
.vendor = PCI_VENDOR_ID_TEKRAM,
.device = PCI_DEVICE_ID_TEKRAM_TRMS1040,
.subvendor = PCI_ANY_ID,
.subdevice = PCI_ANY_ID,
},
{} /* Terminating entry */
};
MODULE_DEVICE_TABLE(pci, dc395x_pci_tbl);
#define DC395x_LOCK_IO(dev) spin_lock_irqsave(((struct Scsi_Host *)dev)->host_lock, flags) #define DC395x_LOCK_IO(dev) spin_lock_irqsave(((struct Scsi_Host *)dev)->host_lock, flags)
...@@ -546,7 +537,6 @@ static inline void DC395x_SetXferRate(struct AdapterCtlBlk *pACB, ...@@ -546,7 +537,6 @@ static inline void DC395x_SetXferRate(struct AdapterCtlBlk *pACB,
struct DeviceCtlBlk *pDCB); struct DeviceCtlBlk *pDCB);
void DC395x_initDCB(struct AdapterCtlBlk *pACB, void DC395x_initDCB(struct AdapterCtlBlk *pACB,
struct DeviceCtlBlk **ppDCB, u8 target, u8 lun); struct DeviceCtlBlk **ppDCB, u8 target, u8 lun);
int DC395x_shutdown(struct Scsi_Host *host);
static void DC395x_remove_dev(struct AdapterCtlBlk *pACB, static void DC395x_remove_dev(struct AdapterCtlBlk *pACB,
struct DeviceCtlBlk *pDCB); struct DeviceCtlBlk *pDCB);
...@@ -613,116 +603,203 @@ static u8 dc395x_clock_period[] = { 12, 18, 25, 31, 37, 43, 50, 62 }; ...@@ -613,116 +603,203 @@ static u8 dc395x_clock_period[] = { 12, 18, 25, 31, 37, 43, 50, 62 };
static u16 dc395x_clock_speed[] = { 200, 133, 100, 80, 67, 58, 50, 40 }; static u16 dc395x_clock_speed[] = { 200, 133, 100, 80, 67, 58, 50, 40 };
/* real period:48ns,72ns,100ns,124ns,148ns,172ns,200ns,248ns */ /* real period:48ns,72ns,100ns,124ns,148ns,172ns,200ns,248ns */
/*---------------------------------------------------------------------------
Configuration
---------------------------------------------------------------------------*/
/* /*
* Override defaults on cmdline: * Command line parameters are stored in a structure below.
* dc395x_trm = AdaptID, MaxSpeed (Index), DevMode (Bitmapped), AdaptMode (Bitmapped), Tags (log2-1), DelayReset * These are the index's into the strcuture for the various
* command line options.
*/ */
int dc395x_trm[] = { -2, -2, -2, -2, -2, -2 }; #define CFG_ADAPTER_ID 0
#define CFG_MAX_SPEED 1
#define CFG_DEV_MODE 2
#define CFG_ADAPTER_MODE 3
#define CFG_TAGS 4
#define CFG_RESET_DELAY 5
#if defined(MODULE) #define CFG_NUM 6 /* number of configuration items */
MODULE_PARM(dc395x_trm, "1-6i");
MODULE_PARM_DESC(dc395x_trm,
"Host SCSI ID, Speed (0=20MHz), Device Flags, Adapter Flags, Max Tags (log2(tags)-1), DelayReset (s)"); /*
* Value used to indicate that a command line override
* hasn't been used to modify the value.
*/
#define CFG_PARAM_UNSET -1
/*
* Hold command line parameters.
*/
struct dc395x_config_data {
int value; /* value of this setting */
int min; /* minimum value */
int max; /* maximum value */
int def; /* default value */
int safe; /* safe value */
};
struct dc395x_config_data __initdata cfg_data[] = {
{ /* adapter id */
CFG_PARAM_UNSET,
0,
15,
7,
7
},
{ /* max speed */
CFG_PARAM_UNSET,
0,
7,
1, /* 13.3Mhz */
4, /* 6.7Hmz */
},
{ /* dev mode */
CFG_PARAM_UNSET,
0,
0x3f,
NTC_DO_PARITY_CHK | NTC_DO_DISCONNECT | NTC_DO_SYNC_NEGO |
NTC_DO_WIDE_NEGO | NTC_DO_TAG_QUEUEING |
NTC_DO_SEND_START,
NTC_DO_PARITY_CHK | NTC_DO_SEND_START
},
{ /* adapter mode */
CFG_PARAM_UNSET,
0,
0x2f,
#ifdef CONFIG_SCSI_MULTI_LUN
NAC_SCANLUN |
#endif #endif
NAC_GT2DRIVES | NAC_GREATER_1G | NAC_POWERON_SCSI_RESET
/*| NAC_ACTIVE_NEG*/,
NAC_GT2DRIVES | NAC_GREATER_1G | NAC_POWERON_SCSI_RESET | 0x08
},
{ /* tags */
CFG_PARAM_UNSET,
0,
5,
3, /* 16 tags (??) */
2,
},
{ /* reset delay */
CFG_PARAM_UNSET,
0,
180,
1, /* 1 second */
10, /* 10 seconds */
}
};
MODULE_AUTHOR("C.L. Huang / Erich Chen / Kurt Garloff");
MODULE_DESCRIPTION
("SCSI host adapter driver for Tekram TRM-S1040 based adapters: Tekram DC395 and DC315 series");
MODULE_SUPPORTED_DEVICE("sd,sr,sg,st");
MODULE_LICENSE("GPL"); /*
* Safe settings. If set to zero the the BIOS/default values with command line
* overrides will be used. If set to 1 then safe and slow settings will be used.
*/
static int dc395x_safe = 0;
module_param_named(safe, dc395x_safe, bool, 0);
MODULE_PARM_DESC(safe, "Use safe and slow settings only. Default: false");
/* Delaying after a reset */
static char __initdata DC395x_interpd[] = { 1, 3, 5, 10, 16, 30, 60, 120 };
/* Convert EEprom value to seconds */ module_param_named(adapter_id, cfg_data[CFG_ADAPTER_ID].value, int, 0);
static void __init DC395x_interpret_delay(struct NvRamType *eeprom) MODULE_PARM_DESC(adapter_id, "Adapter SCSI ID. Default 7 (0-15)");
{
/*printk (DC395X_NAME, "Debug: Delay: %i\n", eeprom->NvramDelayTime); */
eeprom->NvramDelayTime = DC395x_interpd[eeprom->NvramDelayTime];
}
/* seconds to EEProm value */ module_param_named(max_speed, cfg_data[CFG_MAX_SPEED].value, int, 0);
static int __init DC395x_uninterpret_delay(int delay) MODULE_PARM_DESC(max_speed, "Maximum bus speed. Default 1 (0-7) Speeds: 0=20, 1=13.3, 2=10, 3=8, 4=6.7, 5=5.8, 6=5, 7=4 Mhz");
{
u8 idx = 0; module_param_named(dev_mode, cfg_data[CFG_DEV_MODE].value, int, 0);
while (idx < 7 && DC395x_interpd[idx] < delay) MODULE_PARM_DESC(dev_mode, "Device mode.");
idx++;
return idx; module_param_named(adapter_mode, cfg_data[CFG_ADAPTER_MODE].value, int, 0);
} MODULE_PARM_DESC(adapter_mode, "Adapter mode.");
module_param_named(tags, cfg_data[CFG_TAGS].value, int, 0);
MODULE_PARM_DESC(tags, "Number of tags (1<<x). Default 3 (0-5)");
module_param_named(reset_delay, cfg_data[CFG_RESET_DELAY].value, int, 0);
MODULE_PARM_DESC(reset_delay, "Reset delay in seconds. Default 1 (0-180)");
/* Handle "-1" case */
static void __init DC395x_check_for_safe_settings(void) /**
* set_safe_settings - if the safe parameter is set then
* set all values to the safe and slow values.
**/
static
void __init set_safe_settings(void)
{ {
if (dc395x_trm[0] == -1 || dc395x_trm[0] > 15) { /* modules-2.0.0 passes -1 as string */ if (dc395x_safe)
dc395x_trm[0] = 7; {
dc395x_trm[1] = 4; int i;
dc395x_trm[2] = 0x09;
dc395x_trm[3] = 0x0f; dprintkl(KERN_INFO, "Using sage settings.\n");
dc395x_trm[4] = 2; for (i = 0; i < CFG_NUM; i++)
dc395x_trm[5] = 10; {
dprintkl(KERN_INFO, "Using safe settings.\n"); cfg_data[i].value = cfg_data[i].safe;
}
} }
} }
/* Defaults, to be overriden by (a) BIOS and (b) Cmnd line (kernel/module) args */
int __initdata dc395x_def[] = { 7, 1 /* 13.3MHz */ ,
NTC_DO_PARITY_CHK | NTC_DO_DISCONNECT | NTC_DO_SYNC_NEGO |
NTC_DO_WIDE_NEGO | NTC_DO_TAG_QUEUEING | NTC_DO_SEND_START,
NAC_GT2DRIVES | NAC_GREATER_1G | NAC_POWERON_SCSI_RESET
/* | NAC_ACTIVE_NEG */
#ifdef CONFIG_SCSI_MULTI_LUN
| NAC_SCANLUN
#endif
, 3 /* 16 Tags per LUN */ , 1 /* s delay after Reset */
};
/* Copy defaults over set values where missing */ /**
static void __init DC395x_fill_with_defaults(void) * fix_settings - reset any boot parmeters which are out of range
* back to the default values.
**/
static
void __init fix_settings(void)
{ {
int i; int i;
dprintkdbg(DBG_PARSE, "setup %08x %08x %08x %08x %08x %08x\n", dprintkdbg(DBG_PARSE, "setup %08x %08x %08x %08x %08x %08x\n",
dc395x_trm[0], dc395x_trm[1], dc395x_trm[2], cfg_data[CFG_ADAPTER_ID].value,
dc395x_trm[3], dc395x_trm[4], dc395x_trm[5]); cfg_data[CFG_MAX_SPEED].value,
for (i = 0; i < 6; i++) { cfg_data[CFG_DEV_MODE].value,
if (dc395x_trm[i] < 0 || dc395x_trm[i] > 255) cfg_data[CFG_ADAPTER_MODE].value,
dc395x_trm[i] = dc395x_def[i]; cfg_data[CFG_TAGS].value,
} cfg_data[CFG_RESET_DELAY].value);
/* Sanity checks */ for (i = 0; i < CFG_NUM; i++)
if (dc395x_trm[0] > 15) {
dc395x_trm[0] = 7; if (cfg_data[i].value < cfg_data[i].min ||
if (dc395x_trm[1] > 7) cfg_data[i].value > cfg_data[i].max)
dc395x_trm[1] = 4; {
if (dc395x_trm[4] > 5) cfg_data[i].value = cfg_data[i].def;
dc395x_trm[4] = 4; }
if (dc395x_trm[5] > 180) }
dc395x_trm[5] = 180;
} }
/* Read the parameters from the command line */
#if !defined(MODULE)
static int DC395x_trm_setup(char *str)
{
int i;
int im;
int ints[8];
(void) get_options(str, ARRAY_SIZE(ints), ints);
im = ints[0];
if (im > 6) {
dprintkl(KERN_NOTICE, "ignore extra params!\n");
im = 6;
}
for (i = 0; i < im; i++)
dc395x_trm[i] = ints[i + 1];
return 1; /*
* Mapping from the eeprom value (index into this array) to the
* the number of actual seconds that the delay should be for.
*/
static
char __initdata eeprom_index_to_delay_map[] = { 1, 3, 5, 10, 16, 30, 60, 120 };
/**
* eeprom_index_to_delay - Take the eeprom delay setting and convert it
* into a number of seconds.
*/
static void __init eeprom_index_to_delay(struct NvRamType *eeprom)
{
eeprom->NvramDelayTime = eeprom_index_to_delay_map[eeprom->NvramDelayTime];
} }
__setup(DC395X_NAME "=", DC395x_trm_setup);
#endif /* !MODULE */ /**
* delay_to_eeprom_index - Take a delay in seconds and return the closest
* eeprom index which will delay for at least that amount of seconds.
*/
static int __init delay_to_eeprom_index(int delay)
{
u8 idx = 0;
while (idx < 7 && eeprom_index_to_delay_map[idx] < delay) {
idx++;
}
return idx;
}
/* Overrride BIOS values with the set ones */ /* Overrride BIOS values with the set ones */
static void __init DC395x_EEprom_Override(struct NvRamType *eeprom) static void __init DC395x_EEprom_Override(struct NvRamType *eeprom)
...@@ -730,21 +807,32 @@ static void __init DC395x_EEprom_Override(struct NvRamType *eeprom) ...@@ -730,21 +807,32 @@ static void __init DC395x_EEprom_Override(struct NvRamType *eeprom)
u8 id; u8 id;
/* Adapter Settings */ /* Adapter Settings */
if (dc395x_trm[0] != -2) if (cfg_data[CFG_ADAPTER_ID].value != CFG_PARAM_UNSET) {
eeprom->NvramScsiId = (u8) dc395x_trm[0]; /* Adapter ID */ eeprom->NvramScsiId =
if (dc395x_trm[3] != -2) (u8)cfg_data[CFG_ADAPTER_ID].value;
eeprom->NvramChannelCfg = (u8) dc395x_trm[3]; }
if (dc395x_trm[5] != -2) if (cfg_data[CFG_ADAPTER_MODE].value != CFG_PARAM_UNSET) {
eeprom->NvramDelayTime = DC395x_uninterpret_delay(dc395x_trm[5]); /* Reset delay */ eeprom->NvramChannelCfg =
if (dc395x_trm[4] != -2) (u8)cfg_data[CFG_ADAPTER_MODE].value;
eeprom->NvramMaxTag = (u8) dc395x_trm[4]; /* Tagged Cmds */ }
if (cfg_data[CFG_RESET_DELAY].value != CFG_PARAM_UNSET) {
eeprom->NvramDelayTime =
delay_to_eeprom_index(cfg_data[CFG_RESET_DELAY].value);
}
if (cfg_data[CFG_TAGS].value != CFG_PARAM_UNSET) {
eeprom->NvramMaxTag = (u8)cfg_data[CFG_TAGS].value;
}
/* Device Settings */ /* Device Settings */
for (id = 0; id < DC395x_MAX_SCSI_ID; id++) { for (id = 0; id < DC395x_MAX_SCSI_ID; id++) {
if (dc395x_trm[2] != -2) if (cfg_data[CFG_DEV_MODE].value != CFG_PARAM_UNSET) {
eeprom->NvramTarget[id].NvmTarCfg0 = (u8) dc395x_trm[2]; /* Cfg0 */ eeprom->NvramTarget[id].NvmTarCfg0 =
if (dc395x_trm[1] != -2) (u8)cfg_data[CFG_DEV_MODE].value;
eeprom->NvramTarget[id].NvmTarPeriod = (u8) dc395x_trm[1]; /* Speed */ }
if (cfg_data[CFG_MAX_SPEED].value != CFG_PARAM_UNSET) {
eeprom->NvramTarget[id].NvmTarPeriod =
(u8)cfg_data[CFG_MAX_SPEED].value;
}
} }
} }
...@@ -1346,7 +1434,7 @@ DC395x_queue_command(Scsi_Cmnd * cmd, void (*done) (Scsi_Cmnd *)) ...@@ -1346,7 +1434,7 @@ DC395x_queue_command(Scsi_Cmnd * cmd, void (*done) (Scsi_Cmnd *))
#if debug_enabled(DBG_RECURSION) #if debug_enabled(DBG_RECURSION)
if (dbg_in_driver++ > NORM_REC_LVL) { if (dbg_in_driver++ > NORM_REC_LVL) {
dprintkl(KERN_DEBUG, dprintkl(KERN_DEBUG,
": %i queue_command () recursion? (pid=%li)\n", "%i queue_command () recursion? (pid=%li)\n",
dbg_in_driver, cmd->pid); dbg_in_driver, cmd->pid);
} }
#endif #endif
...@@ -1928,7 +2016,7 @@ void DC395x_selection_timeout_missed(unsigned long ptr) ...@@ -1928,7 +2016,7 @@ void DC395x_selection_timeout_missed(unsigned long ptr)
struct ScsiReqBlk *pSRB; struct ScsiReqBlk *pSRB;
dprintkl(KERN_DEBUG, "Chip forgot to produce SelTO IRQ!\n"); dprintkl(KERN_DEBUG, "Chip forgot to produce SelTO IRQ!\n");
if (!pACB->pActiveDCB || !pACB->pActiveDCB->pActiveSRB) { if (!pACB->pActiveDCB || !pACB->pActiveDCB->pActiveSRB) {
dprintkl(KERN_DEBUG, " ... but no cmd pending? Oops!\n"); dprintkl(KERN_DEBUG, "... but no cmd pending? Oops!\n");
return; return;
} }
DC395x_LOCK_IO(pACB->pScsiHost); DC395x_LOCK_IO(pACB->pScsiHost);
...@@ -4160,6 +4248,7 @@ DC395x_remove_dev(struct AdapterCtlBlk *pACB, struct DeviceCtlBlk *pDCB) ...@@ -4160,6 +4248,7 @@ DC395x_remove_dev(struct AdapterCtlBlk *pACB, struct DeviceCtlBlk *pDCB)
{ {
struct DeviceCtlBlk *pPrevDCB = pACB->pLinkDCB; struct DeviceCtlBlk *pPrevDCB = pACB->pLinkDCB;
dprintkdbg(DBG_0, "remove_dev\n");
if (pDCB->GoingSRBCnt > 1) { if (pDCB->GoingSRBCnt > 1) {
dprintkdbg(DBG_DCB, "Driver won't free DCB (ID %i, LUN %i): 0x%08x because of SRBCnt %i\n", dprintkdbg(DBG_DCB, "Driver won't free DCB (ID %i, LUN %i): 0x%08x because of SRBCnt %i\n",
pDCB->TargetID, pDCB->TargetLUN, (int) pDCB, pDCB->TargetID, pDCB->TargetLUN, (int) pDCB,
...@@ -4982,6 +5071,7 @@ int DC395x_alloc_tracebufs(struct AdapterCtlBlk *pACB) ...@@ -4982,6 +5071,7 @@ int DC395x_alloc_tracebufs(struct AdapterCtlBlk *pACB)
/* Free SG tables */ /* Free SG tables */
static
void DC395x_free_SG_tables(struct AdapterCtlBlk *pACB, int SRBIdx) void DC395x_free_SG_tables(struct AdapterCtlBlk *pACB, int SRBIdx)
{ {
int srbidx; int srbidx;
...@@ -5529,8 +5619,8 @@ DC395x_check_eeprom(struct NvRamType *eeprom, u16 io_port) ...@@ -5529,8 +5619,8 @@ DC395x_check_eeprom(struct NvRamType *eeprom, u16 io_port)
*d_eeprom = 0x00; *d_eeprom = 0x00;
/* Now load defaults (maybe set by boot/module params) */ /* Now load defaults (maybe set by boot/module params) */
DC395x_check_for_safe_settings(); set_safe_settings();
DC395x_fill_with_defaults(); fix_settings();
DC395x_EEprom_Override(eeprom); DC395x_EEprom_Override(eeprom);
eeprom->NvramCheckSum = 0x00; eeprom->NvramCheckSum = 0x00;
...@@ -5540,10 +5630,10 @@ DC395x_check_eeprom(struct NvRamType *eeprom, u16 io_port) ...@@ -5540,10 +5630,10 @@ DC395x_check_eeprom(struct NvRamType *eeprom, u16 io_port)
*w_eeprom = 0x1234 - cksum; *w_eeprom = 0x1234 - cksum;
TRM_S1040_write_all(eeprom, io_port); TRM_S1040_write_all(eeprom, io_port);
eeprom->NvramDelayTime = dc395x_trm[5]; eeprom->NvramDelayTime = cfg_data[CFG_RESET_DELAY].value;
} else { } else {
DC395x_check_for_safe_settings(); set_safe_settings();
DC395x_interpret_delay(eeprom); eeprom_index_to_delay(eeprom);
DC395x_EEprom_Override(eeprom); DC395x_EEprom_Override(eeprom);
} }
} }
...@@ -5586,6 +5676,29 @@ static void __init DC395x_print_config(struct AdapterCtlBlk *pACB) ...@@ -5586,6 +5676,29 @@ static void __init DC395x_print_config(struct AdapterCtlBlk *pACB)
} }
/**
* DC395x_print_eeprom_settings - output the eeprom settings
* to the kernel log so people can see what they were.
*
* @index: Adapter number
**/
static void __init
DC395x_print_eeprom_settings(u16 index)
{
dprintkl(KERN_INFO, "Used settings: AdapterID=%02i, Speed=%i(%02i.%01iMHz), DevMode=0x%02x\n",
dc395x_trm_eepromBuf[index].NvramScsiId,
dc395x_trm_eepromBuf[index].NvramTarget[0].NvmTarPeriod,
dc395x_clock_speed[dc395x_trm_eepromBuf[index].NvramTarget[0].NvmTarPeriod] / 10,
dc395x_clock_speed[dc395x_trm_eepromBuf[index].NvramTarget[0].NvmTarPeriod] % 10,
dc395x_trm_eepromBuf[index].NvramTarget[0].NvmTarCfg0);
dprintkl(KERN_INFO, " AdaptMode=0x%02x, Tags=%i(%02i), DelayReset=%is\n",
dc395x_trm_eepromBuf[index].NvramChannelCfg,
dc395x_trm_eepromBuf[index].NvramMaxTag,
1 << dc395x_trm_eepromBuf[index].NvramMaxTag,
dc395x_trm_eepromBuf[index].NvramDelayTime);
}
/* /*
********************************************************************* *********************************************************************
* DC395x_detect * DC395x_detect
...@@ -5610,119 +5723,44 @@ DC395x_init(Scsi_Host_Template * host_template, u32 io_port, u8 irq, ...@@ -5610,119 +5723,44 @@ DC395x_init(Scsi_Host_Template * host_template, u32 io_port, u8 irq,
*/ */
DC395x_check_eeprom(&dc395x_trm_eepromBuf[index], (u16) io_port); DC395x_check_eeprom(&dc395x_trm_eepromBuf[index], (u16) io_port);
/*$$$$$$$$$$$ MEMORY ALLOCATE FOR ADAPTER CONTROL BLOCK $$$$$$$$$$$$ */ /*
*$$$$$$$$$$$ MEMORY ALLOCATE FOR ADAPTER CONTROL BLOCK $$$$$$$$$$$$
*/
host = scsi_register(host_template, sizeof(struct AdapterCtlBlk)); host = scsi_register(host_template, sizeof(struct AdapterCtlBlk));
if (!host) { if (!host) {
dprintkl(KERN_INFO, "pSH scsi_register ERROR\n"); dprintkl(KERN_INFO, "pSH scsi_register ERROR\n");
return 0; return 0;
} }
dprintkl(KERN_INFO, DC395x_print_eeprom_settings(index);
"Used settings: AdapterID=%02i, Speed=%i(%02i.%01iMHz), DevMode=0x%02x\n",
dc395x_trm_eepromBuf[index].NvramScsiId,
dc395x_trm_eepromBuf[index].NvramTarget[0].NvmTarPeriod,
dc395x_clock_speed[dc395x_trm_eepromBuf[index].
NvramTarget[0].NvmTarPeriod] / 10,
dc395x_clock_speed[dc395x_trm_eepromBuf[index].
NvramTarget[0].NvmTarPeriod] % 10,
dc395x_trm_eepromBuf[index].NvramTarget[0].NvmTarCfg0);
dprintkl(KERN_INFO,
" AdaptMode=0x%02x, Tags=%i(%02i), DelayReset=%is\n",
dc395x_trm_eepromBuf[index].NvramChannelCfg,
dc395x_trm_eepromBuf[index].NvramMaxTag,
1 << dc395x_trm_eepromBuf[index].NvramMaxTag,
dc395x_trm_eepromBuf[index].NvramDelayTime);
pACB = (struct AdapterCtlBlk *) host->hostdata; pACB = (struct AdapterCtlBlk *) host->hostdata;
/*DC395x_ACB_INITLOCK(pACB); */
/*DC395x_ACB_LOCK(pACB,acb_flags); */
/*$$$$$$$$ INITIAL ADAPTER CONTROL BLOCK $$$$$$$$$$$$ */
if (DC395x_initACB(host, io_port, irq, index)) { if (DC395x_initACB(host, io_port, irq, index)) {
scsi_unregister(host); scsi_unregister(host);
/*DC395x_ACB_UNLOCK(pACB,acb_flags); */
return 0; return 0;
} }
DC395x_print_config(pACB); DC395x_print_config(pACB);
/*$$$$$$$$$$$$$$$$$ INITIAL ADAPTER $$$$$$$$$$$$$$$$$ */
/*
*$$$$$$$$$$$$$$$$$ INITIAL ADAPTER $$$$$$$$$$$$$$$$$
*/
if (!DC395x_initAdapter(host, io_port, irq, index)) { if (!DC395x_initAdapter(host, io_port, irq, index)) {
if (!DC395x_pACB_start) { if (!DC395x_pACB_start) {
DC395x_pACB_start = pACB; DC395x_pACB_start = pACB;
DC395x_pACB_current = pACB;
pACB->pNextACB = NULL;
} else { } else {
DC395x_pACB_current->pNextACB = pACB; DC395x_pACB_current->pNextACB = pACB;
}
DC395x_pACB_current = pACB; DC395x_pACB_current = pACB;
pACB->pNextACB = NULL; pACB->pNextACB = NULL;
}
/*DC395x_ACB_UNLOCK(pACB,acb_flags); */
return host;
} else { } else {
dprintkl(KERN_INFO, dprintkl(KERN_INFO, "DC395x_initAdapter initial ERROR\n");
"DC395x_initAdapter initial ERROR\n");
scsi_unregister(host); scsi_unregister(host);
/*DC395x_ACB_UNLOCK(pACB,acb_flags); */ host = NULL;
return 0;
} }
return host;
} }
/*
* DC395x_detect
*
* Detect TRM-S1040 cards, acquire resources and initialise the card.
* Argument is a pointer to the host driver's scsi_hosts entry.
*
* Returns the number of adapters found.
*
* This function is called during system initialization and must not
* call SCSI mid-level functions including scsi_malloc() and
* scsi_free().
*/
static int __init DC395x_detect(Scsi_Host_Template * host_template)
{
struct pci_dev *pdev = NULL;
unsigned int io_port;
u8 irq;
DC395x_pACB_start = NULL;
/* without PCI we cannot do anything */
if (pci_present() == 0) {
dprintkl(KERN_INFO, "PCI not present\n");
return 0;
}
dprintkl(KERN_INFO, "%s %s\n", DC395X_BANNER, DC395X_VERSION);
while ((pdev =
pci_find_device(PCI_VENDOR_ID_TEKRAM,
PCI_DEVICE_ID_TEKRAM_TRMS1040, pdev))) {
struct Scsi_Host *scsi_host;
if (pci_enable_device(pdev))
continue;
io_port =
pci_resource_start(pdev, 0) & PCI_BASE_ADDRESS_IO_MASK;
irq = pdev->irq;
dprintkdbg(DBG_0, "IO_PORT=%04x,IRQ=%x\n", (unsigned int) io_port, irq);
if ((scsi_host =
DC395x_init(host_template, io_port, irq,
DC395x_adapterCnt))) {
pci_set_master(pdev);
((struct AdapterCtlBlk *) (scsi_host->hostdata))->
pdev = pdev;
/*DC395x_set_pci_cfg(pdev); */
DC395x_adapterCnt++;
}
}
if (DC395x_adapterCnt) {
host_template->proc_name = DC395X_NAME;
}
dprintkl(KERN_INFO, "%s: %i adapters found\n",
DC395X_BANNER, DC395x_adapterCnt);
return DC395x_adapterCnt;
}
/* /*
* Functions: DC395x_inquiry(), DC395x_inquiry_done() * Functions: DC395x_inquiry(), DC395x_inquiry_done()
* *
...@@ -6014,114 +6052,103 @@ DC395x_proc_info(struct Scsi_Host *shpnt, char *buffer, char **start, off_t offs ...@@ -6014,114 +6052,103 @@ DC395x_proc_info(struct Scsi_Host *shpnt, char *buffer, char **start, off_t offs
} }
/* /**
* Function : int DC395x_shutdown (struct Scsi_Host *host) * DC395x_chip_shutdown - cleanly shut down the scsi controller chip,
* Purpose : does a clean (we hope) shutdown of the SCSI chip. * stopping all operations and disablig interrupt generation on the
* Use prior to dumping core, unloading the driver, etc. * card.
* Returns : 0 on success *
*/ * @acb: The scsi adapter control block of the adapter to shut down.
int DC395x_shutdown(struct Scsi_Host *host) **/
static
void DC395x_chip_shutdown(struct AdapterCtlBlk *pACB)
{ {
struct AdapterCtlBlk *pACB;
pACB = (struct AdapterCtlBlk *) (host->hostdata);
/* pACB->soft_reset(host); */
/* disable interrupt */ /* disable interrupt */
DC395x_write8(TRM_S1040_DMA_INTEN, 0); DC395x_write8(TRM_S1040_DMA_INTEN, 0);
DC395x_write8(TRM_S1040_SCSI_INTEN, 0); DC395x_write8(TRM_S1040_SCSI_INTEN, 0);
/* remove timers */
if (timer_pending(&pACB->Waiting_Timer)) if (timer_pending(&pACB->Waiting_Timer))
del_timer(&pACB->Waiting_Timer); del_timer(&pACB->Waiting_Timer);
if (timer_pending(&pACB->SelTO_Timer)) if (timer_pending(&pACB->SelTO_Timer))
del_timer(&pACB->SelTO_Timer); del_timer(&pACB->SelTO_Timer);
if (1 || pACB->Config & HCC_SCSI_RESET) /* reset the scsi bus */
if (pACB->Config & HCC_SCSI_RESET)
DC395x_ResetSCSIBus(pACB); DC395x_ResetSCSIBus(pACB);
/* clear any pending interupt state */
DC395x_read8(TRM_S1040_SCSI_INTSTATUS); DC395x_read8(TRM_S1040_SCSI_INTSTATUS);
/* release chip resources */
#if debug_enabled(DBG_TRACE|DBG_TRACEALL) #if debug_enabled(DBG_TRACE|DBG_TRACEALL)
DC395x_free_tracebufs(pACB, DC395x_MAX_SRB_CNT); DC395x_free_tracebufs(pACB, DC395x_MAX_SRB_CNT);
#endif #endif
DC395x_free_SG_tables(pACB, DC395x_MAX_SRB_CNT); DC395x_free_SG_tables(pACB, DC395x_MAX_SRB_CNT);
return 0;
} }
/* /**
* Free all DCBs * DC395x_free_DCBs - Free all of the DCBs.
*/ *
void DC395x_freeDCBs(struct Scsi_Host *host) * @pACB: Adapter to remove the DCBs for.
**/
static
void DC395x_free_DCBs(struct AdapterCtlBlk* pACB)
{ {
struct DeviceCtlBlk *pDCB; struct DeviceCtlBlk *dcb;
struct DeviceCtlBlk *nDCB; struct DeviceCtlBlk *dcb_next;
struct AdapterCtlBlk *pACB =
(struct AdapterCtlBlk *) (host->hostdata);
dprintkdbg(DBG_DCB, "Free %i DCBs\n", pACB->DCBCnt); dprintkdbg(DBG_DCB, "Free %i DCBs\n", pACB->DCBCnt);
pDCB = pACB->pLinkDCB;
if (pDCB) { for (dcb = pACB->pLinkDCB; dcb != NULL; dcb = dcb_next)
do { {
nDCB = pDCB->pNextDCB; dcb_next = dcb->pNextDCB;
dprintkdbg(DBG_DCB, "Free DCB (ID %i, LUN %i): %p\n", dprintkdbg(DBG_DCB, "Free DCB (ID %i, LUN %i): %p\n",
pDCB->TargetID, pDCB->TargetLUN, pDCB); dcb->TargetID, dcb->TargetLUN, dcb);
DC395x_remove_dev(pACB, pDCB); /* includes a dc395x_kfree(pDCB); */ /*
printk("."); * Free the DCB. This removes the entry from the
pDCB = nDCB; * pLinkDCB list and decrements the count in DCBCnt
} while (pDCB && pACB->pLinkDCB); */
DC395x_remove_dev(pACB, dcb);
} }
} }
/**
/* * DC395x_release - shutdown device and release resources that were
* Release method * allocate for it. Called once for each card as it is shutdown.
* *
* Called when we are to shutdown the controller and release all of * @host: The adapter instance to shutdown.
* it's resources. **/
*/ static
static int DC395x_release(struct Scsi_Host *host) void DC395x_release(struct Scsi_Host *host)
{ {
struct AdapterCtlBlk *pACB = struct AdapterCtlBlk *pACB = (struct AdapterCtlBlk *) (host->hostdata);
(struct AdapterCtlBlk *) (host->hostdata);
unsigned long flags; unsigned long flags;
dprintkl(KERN_DEBUG, "release"); dprintkl(KERN_DEBUG, "DC395x release\n");
DC395x_LOCK_IO(pACB->pScsiHost); DC395x_LOCK_IO(pACB->pScsiHost);
DC395x_shutdown(host); DC395x_chip_shutdown(pACB);
DC395x_freeDCBs(host); DC395x_free_DCBs(pACB);
if (host->irq != NO_IRQ) { if (host->irq != NO_IRQ) {
/*
* Find the IRQ to release. XXX Why didn't we just store the
* appropriate IRQ details when we request_irq it?
*/
int irq_count;
for (irq_count = 0, pACB = DC395x_pACB_start;
pACB;
pACB = pACB->pNextACB) {
if (pACB->IRQLevel == host->irq)
++irq_count;
}
if (irq_count == 1)
free_irq(host->irq, DC395x_pACB_start); free_irq(host->irq, DC395x_pACB_start);
} }
release_region(host->io_port, host->n_io_port); release_region(host->io_port, host->n_io_port);
DC395x_UNLOCK_IO(pACB->pScsiHost); DC395x_UNLOCK_IO(pACB->pScsiHost);
return 1;
} }
/* /*
* SCSI host template * SCSI host template
*/ */
static Scsi_Host_Template driver_template = { static Scsi_Host_Template dc395x_driver_template = {
.module = THIS_MODULE,
.proc_name = DC395X_NAME, .proc_name = DC395X_NAME,
.proc_info = DC395x_proc_info, .proc_info = DC395x_proc_info,
.name = DC395X_BANNER " " DC395X_VERSION, .name = DC395X_BANNER " " DC395X_VERSION,
.detect = DC395x_detect,
.release = DC395x_release,
.queuecommand = DC395x_queue_command, .queuecommand = DC395x_queue_command,
.bios_param = DC395x_bios_param, .bios_param = DC395x_bios_param,
.slave_alloc = DC395x_slave_alloc, .slave_alloc = DC395x_slave_alloc,
...@@ -6135,9 +6162,114 @@ static Scsi_Host_Template driver_template = { ...@@ -6135,9 +6162,114 @@ static Scsi_Host_Template driver_template = {
.unchecked_isa_dma = 0, .unchecked_isa_dma = 0,
.use_clustering = DISABLE_CLUSTERING, .use_clustering = DISABLE_CLUSTERING,
}; };
/* /*
* The following code deals with registering the above scsi host * Called to initialise a single instance of the adaptor
* template with the higher level scsi code and results in the detect
* method from the template being called during initialisation.
*/ */
#include "scsi_module.c"
static
int __devinit dc395x_init_one(struct pci_dev *pdev,
const struct pci_device_id *id)
{
unsigned int io_port;
u8 irq;
struct Scsi_Host *scsi_host;
static int banner_done = 0;
dprintkdbg(DBG_0, "Init one instance of the dc395x\n");
if (!banner_done)
{
dprintkl(KERN_INFO, "%s %s\n", DC395X_BANNER, DC395X_VERSION);
banner_done = 1;
}
if (pci_enable_device(pdev))
{
dprintkl(KERN_INFO, "PCI Enable device failed.\n");
return -ENODEV;
}
dprintkdbg(DBG_0, "Get resources...\n");
io_port = pci_resource_start(pdev, 0) & PCI_BASE_ADDRESS_IO_MASK;
irq = pdev->irq;
dprintkdbg(DBG_0, "IO_PORT=%04x,IRQ=%x\n", (unsigned int) io_port, irq);
scsi_host = DC395x_init(&dc395x_driver_template, io_port, irq, DC395x_adapterCnt);
if (!scsi_host)
{
dprintkdbg(DBG_0, "DC395x_init failed\n");
return -ENOMEM;
}
pci_set_master(pdev);
/* store pci devices in out host data object. */
((struct AdapterCtlBlk *)(scsi_host->hostdata))->pdev = pdev;
/* increment adaptor count */
DC395x_adapterCnt++;
/* store ptr to scsi host in the PCI device structure */
pci_set_drvdata(pdev, scsi_host);
/* get the scsi mid level to scan for new devices on the bus */
scsi_add_host(scsi_host, &pdev->dev);
return 0;
}
/*
* Called to remove a single instance of the adaptor
*/
static void __devexit dc395x_remove_one(struct pci_dev *pdev)
{
struct Scsi_Host *host = pci_get_drvdata(pdev);
dprintkdbg(DBG_0, "Removing instance\n");
scsi_remove_host(host);
DC395x_release(host);
pci_set_drvdata(pdev, NULL);
}
/*
* Table which identifies the PCI devices which
* are handled by this device driver.
*/
static struct pci_device_id dc395x_pci_table[] __devinitdata = {
{
.vendor = PCI_VENDOR_ID_TEKRAM,
.device = PCI_DEVICE_ID_TEKRAM_TRMS1040,
.subvendor = PCI_ANY_ID,
.subdevice = PCI_ANY_ID,
},
{} /* Terminating entry */
};
MODULE_DEVICE_TABLE(pci, dc395x_pci_table);
/*
* PCI driver operations.
* Tells the PCI sub system what can be done with the card.
*/
static struct pci_driver dc395x_driver = {
.name = DC395X_NAME,
.id_table = dc395x_pci_table,
.probe = dc395x_init_one,
.remove = __devexit_p(dc395x_remove_one),
};
static int __init dc395x_module_init(void)
{
return pci_module_init(&dc395x_driver);
}
static void __exit dc395x_module_exit(void)
{
pci_unregister_driver(&dc395x_driver);
}
module_init(dc395x_module_init);
module_exit(dc395x_module_exit);
MODULE_AUTHOR("C.L. Huang / Erich Chen / Kurt Garloff");
MODULE_DESCRIPTION("SCSI host adapter driver for Tekram TRM-S1040 based adapters: Tekram DC395 and DC315 series");
MODULE_LICENSE("GPL");
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment