Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
linux
Commits
1f5f4779
Commit
1f5f4779
authored
May 28, 2003
by
Oliver Neukum
Committed by
Greg Kroah-Hartman
May 28, 2003
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
- major cleanup of the module code
parent
99c5cb0f
Changes
1
Show whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
420 additions
and
288 deletions
+420
-288
drivers/scsi/dc395x.c
drivers/scsi/dc395x.c
+420
-288
No files found.
drivers/scsi/dc395x.c
View file @
1f5f4779
...
...
@@ -47,6 +47,7 @@
************************************************************************
*/
#include <linux/module.h>
#include <linux/moduleparam.h>
#include <linux/delay.h>
#include <linux/ctype.h>
#include <linux/blk.h>
...
...
@@ -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 */
#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)
...
...
@@ -546,7 +537,6 @@ static inline void DC395x_SetXferRate(struct AdapterCtlBlk *pACB,
struct
DeviceCtlBlk
*
pDCB
);
void
DC395x_initDCB
(
struct
AdapterCtlBlk
*
pACB
,
struct
DeviceCtlBlk
**
ppDCB
,
u8
target
,
u8
lun
);
int
DC395x_shutdown
(
struct
Scsi_Host
*
host
);
static
void
DC395x_remove_dev
(
struct
AdapterCtlBlk
*
pACB
,
struct
DeviceCtlBlk
*
pDCB
);
...
...
@@ -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
};
/* real period:48ns,72ns,100ns,124ns,148ns,172ns,200ns,248ns */
/*---------------------------------------------------------------------------
Configuration
---------------------------------------------------------------------------*/
/*
* Override defaults on cmdline:
* dc395x_trm = AdaptID, MaxSpeed (Index), DevMode (Bitmapped), AdaptMode (Bitmapped), Tags (log2-1), DelayReset
* Command line parameters are stored in a structure below.
* 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)
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)"
);
#define CFG_NUM 6
/* number of configuration items */
/*
* 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
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 */
static
void
__init
DC395x_interpret_delay
(
struct
NvRamType
*
eeprom
)
{
/*printk (DC395X_NAME, "Debug: Delay: %i\n", eeprom->NvramDelayTime); */
eeprom
->
NvramDelayTime
=
DC395x_interpd
[
eeprom
->
NvramDelayTime
];
}
module_param_named
(
adapter_id
,
cfg_data
[
CFG_ADAPTER_ID
].
value
,
int
,
0
);
MODULE_PARM_DESC
(
adapter_id
,
"Adapter SCSI ID. Default 7 (0-15)"
);
/* seconds to EEProm value */
static
int
__init
DC395x_uninterpret_delay
(
int
delay
)
{
u8
idx
=
0
;
while
(
idx
<
7
&&
DC395x_interpd
[
idx
]
<
delay
)
idx
++
;
return
idx
;
}
module_param_named
(
max_speed
,
cfg_data
[
CFG_MAX_SPEED
].
value
,
int
,
0
);
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"
);
module_param_named
(
dev_mode
,
cfg_data
[
CFG_DEV_MODE
].
value
,
int
,
0
);
MODULE_PARM_DESC
(
dev_mode
,
"Device mode."
);
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 */
dc395x_trm
[
0
]
=
7
;
dc395x_trm
[
1
]
=
4
;
dc395x_trm
[
2
]
=
0x09
;
dc395x_trm
[
3
]
=
0x0f
;
dc395x_trm
[
4
]
=
2
;
dc395x_trm
[
5
]
=
10
;
dprintkl
(
KERN_INFO
,
"Using safe settings.
\n
"
);
if
(
dc395x_safe
)
{
int
i
;
dprintkl
(
KERN_INFO
,
"Using sage settings.
\n
"
);
for
(
i
=
0
;
i
<
CFG_NUM
;
i
++
)
{
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
;
dprintkdbg
(
DBG_PARSE
,
"setup %08x %08x %08x %08x %08x %08x
\n
"
,
dc395x_trm
[
0
],
dc395x_trm
[
1
],
dc395x_trm
[
2
],
dc395x_trm
[
3
],
dc395x_trm
[
4
],
dc395x_trm
[
5
]);
for
(
i
=
0
;
i
<
6
;
i
++
)
{
if
(
dc395x_trm
[
i
]
<
0
||
dc395x_trm
[
i
]
>
255
)
dc395x_trm
[
i
]
=
dc395x_def
[
i
];
}
/* Sanity checks */
if
(
dc395x_trm
[
0
]
>
15
)
dc395x_trm
[
0
]
=
7
;
if
(
dc395x_trm
[
1
]
>
7
)
dc395x_trm
[
1
]
=
4
;
if
(
dc395x_trm
[
4
]
>
5
)
dc395x_trm
[
4
]
=
4
;
if
(
dc395x_trm
[
5
]
>
180
)
dc395x_trm
[
5
]
=
180
;
cfg_data
[
CFG_ADAPTER_ID
].
value
,
cfg_data
[
CFG_MAX_SPEED
].
value
,
cfg_data
[
CFG_DEV_MODE
].
value
,
cfg_data
[
CFG_ADAPTER_MODE
].
value
,
cfg_data
[
CFG_TAGS
].
value
,
cfg_data
[
CFG_RESET_DELAY
].
value
);
for
(
i
=
0
;
i
<
CFG_NUM
;
i
++
)
{
if
(
cfg_data
[
i
].
value
<
cfg_data
[
i
].
min
||
cfg_data
[
i
].
value
>
cfg_data
[
i
].
max
)
{
cfg_data
[
i
].
value
=
cfg_data
[
i
].
def
;
}
}
}
/* 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 */
static
void
__init
DC395x_EEprom_Override
(
struct
NvRamType
*
eeprom
)
...
...
@@ -730,21 +807,32 @@ static void __init DC395x_EEprom_Override(struct NvRamType *eeprom)
u8
id
;
/* Adapter Settings */
if
(
dc395x_trm
[
0
]
!=
-
2
)
eeprom
->
NvramScsiId
=
(
u8
)
dc395x_trm
[
0
];
/* Adapter ID */
if
(
dc395x_trm
[
3
]
!=
-
2
)
eeprom
->
NvramChannelCfg
=
(
u8
)
dc395x_trm
[
3
];
if
(
dc395x_trm
[
5
]
!=
-
2
)
eeprom
->
NvramDelayTime
=
DC395x_uninterpret_delay
(
dc395x_trm
[
5
]);
/* Reset delay */
if
(
dc395x_trm
[
4
]
!=
-
2
)
eeprom
->
NvramMaxTag
=
(
u8
)
dc395x_trm
[
4
];
/* Tagged Cmds */
if
(
cfg_data
[
CFG_ADAPTER_ID
].
value
!=
CFG_PARAM_UNSET
)
{
eeprom
->
NvramScsiId
=
(
u8
)
cfg_data
[
CFG_ADAPTER_ID
].
value
;
}
if
(
cfg_data
[
CFG_ADAPTER_MODE
].
value
!=
CFG_PARAM_UNSET
)
{
eeprom
->
NvramChannelCfg
=
(
u8
)
cfg_data
[
CFG_ADAPTER_MODE
].
value
;
}
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 */
for
(
id
=
0
;
id
<
DC395x_MAX_SCSI_ID
;
id
++
)
{
if
(
dc395x_trm
[
2
]
!=
-
2
)
eeprom
->
NvramTarget
[
id
].
NvmTarCfg0
=
(
u8
)
dc395x_trm
[
2
];
/* Cfg0 */
if
(
dc395x_trm
[
1
]
!=
-
2
)
eeprom
->
NvramTarget
[
id
].
NvmTarPeriod
=
(
u8
)
dc395x_trm
[
1
];
/* Speed */
if
(
cfg_data
[
CFG_DEV_MODE
].
value
!=
CFG_PARAM_UNSET
)
{
eeprom
->
NvramTarget
[
id
].
NvmTarCfg0
=
(
u8
)
cfg_data
[
CFG_DEV_MODE
].
value
;
}
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 *))
#if debug_enabled(DBG_RECURSION)
if
(
dbg_in_driver
++
>
NORM_REC_LVL
)
{
dprintkl
(
KERN_DEBUG
,
"
:
%i queue_command () recursion? (pid=%li)
\n
"
,
"%i queue_command () recursion? (pid=%li)
\n
"
,
dbg_in_driver
,
cmd
->
pid
);
}
#endif
...
...
@@ -1928,7 +2016,7 @@ void DC395x_selection_timeout_missed(unsigned long ptr)
struct ScsiReqBlk *pSRB;
dprintkl(KERN_DEBUG, "Chip forgot to produce SelTO IRQ!\n");
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;
}
DC395x_LOCK_IO(pACB->pScsiHost);
...
...
@@ -4160,6 +4248,7 @@ DC395x_remove_dev(struct AdapterCtlBlk *pACB, struct DeviceCtlBlk *pDCB)
{
struct
DeviceCtlBlk
*
pPrevDCB
=
pACB
->
pLinkDCB
;
dprintkdbg
(
DBG_0
,
"remove_dev
\n
"
);
if
(
pDCB
->
GoingSRBCnt
>
1
)
{
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
,
...
...
@@ -4982,6 +5071,7 @@ int DC395x_alloc_tracebufs(struct AdapterCtlBlk *pACB)
/* Free SG tables */
static
void
DC395x_free_SG_tables
(
struct
AdapterCtlBlk
*
pACB
,
int
SRBIdx
)
{
int
srbidx
;
...
...
@@ -5529,8 +5619,8 @@ DC395x_check_eeprom(struct NvRamType *eeprom, u16 io_port)
*
d_eeprom
=
0x00
;
/* Now load defaults (maybe set by boot/module params) */
DC395x_check_for
_safe_settings
();
DC395x_fill_with_default
s
();
set
_safe_settings
();
fix_setting
s
();
DC395x_EEprom_Override
(
eeprom
);
eeprom
->
NvramCheckSum
=
0x00
;
...
...
@@ -5540,10 +5630,10 @@ DC395x_check_eeprom(struct NvRamType *eeprom, u16 io_port)
*
w_eeprom
=
0x1234
-
cksum
;
TRM_S1040_write_all
(
eeprom
,
io_port
);
eeprom
->
NvramDelayTime
=
dc395x_trm
[
5
]
;
eeprom
->
NvramDelayTime
=
cfg_data
[
CFG_RESET_DELAY
].
value
;
}
else
{
DC395x_check_for
_safe_settings
();
DC395x_interpret
_delay
(
eeprom
);
set
_safe_settings
();
eeprom_index_to
_delay
(
eeprom
);
DC395x_EEprom_Override
(
eeprom
);
}
}
...
...
@@ -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
...
...
@@ -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
);
/*$$$$$$$$$$$ MEMORY ALLOCATE FOR ADAPTER CONTROL BLOCK $$$$$$$$$$$$ */
/*
*$$$$$$$$$$$ MEMORY ALLOCATE FOR ADAPTER CONTROL BLOCK $$$$$$$$$$$$
*/
host
=
scsi_register
(
host_template
,
sizeof
(
struct
AdapterCtlBlk
));
if
(
!
host
)
{
dprintkl
(
KERN_INFO
,
"pSH scsi_register ERROR
\n
"
);
return
0
;
}
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_print_eeprom_settings
(
index
);
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
))
{
scsi_unregister
(
host
);
/*DC395x_ACB_UNLOCK(pACB,acb_flags); */
return
0
;
}
DC395x_print_config
(
pACB
);
/*$$$$$$$$$$$$$$$$$ INITIAL ADAPTER $$$$$$$$$$$$$$$$$ */
/*
*$$$$$$$$$$$$$$$$$ INITIAL ADAPTER $$$$$$$$$$$$$$$$$
*/
if
(
!
DC395x_initAdapter
(
host
,
io_port
,
irq
,
index
))
{
if
(
!
DC395x_pACB_start
)
{
DC395x_pACB_start
=
pACB
;
DC395x_pACB_current
=
pACB
;
pACB
->
pNextACB
=
NULL
;
}
else
{
DC395x_pACB_current
->
pNextACB
=
pACB
;
}
DC395x_pACB_current
=
pACB
;
pACB
->
pNextACB
=
NULL
;
}
/*DC395x_ACB_UNLOCK(pACB,acb_flags); */
return
host
;
}
else
{
dprintkl
(
KERN_INFO
,
"DC395x_initAdapter initial ERROR
\n
"
);
dprintkl
(
KERN_INFO
,
"DC395x_initAdapter initial ERROR
\n
"
);
scsi_unregister
(
host
);
/*DC395x_ACB_UNLOCK(pACB,acb_flags); */
return
0
;
host
=
NULL
;
}
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()
*
...
...
@@ -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)
* Purpose : does a clean (we hope) shutdown of the SCSI chip.
* Use prior to dumping core, unloading the driver, etc.
* Returns : 0 on success
*/
int
DC395x_shutdown
(
struct
Scsi_Host
*
host
)
/**
* DC395x_chip_shutdown - cleanly shut down the scsi controller chip,
* stopping all operations and disablig interrupt generation on the
* card.
*
* @acb: The scsi adapter control block of the adapter to shut down.
**/
static
void
DC395x_chip_shutdown
(
struct
AdapterCtlBlk
*
pACB
)
{
struct
AdapterCtlBlk
*
pACB
;
pACB
=
(
struct
AdapterCtlBlk
*
)
(
host
->
hostdata
);
/* pACB->soft_reset(host); */
/* disable interrupt */
DC395x_write8
(
TRM_S1040_DMA_INTEN
,
0
);
DC395x_write8
(
TRM_S1040_SCSI_INTEN
,
0
);
/* remove timers */
if
(
timer_pending
(
&
pACB
->
Waiting_Timer
))
del_timer
(
&
pACB
->
Waiting_Timer
);
if
(
timer_pending
(
&
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
);
/* clear any pending interupt state */
DC395x_read8
(
TRM_S1040_SCSI_INTSTATUS
);
/* release chip resources */
#if debug_enabled(DBG_TRACE|DBG_TRACEALL)
DC395x_free_tracebufs
(
pACB
,
DC395x_MAX_SRB_CNT
);
#endif
DC395x_free_SG_tables
(
pACB
,
DC395x_MAX_SRB_CNT
);
return
0
;
}
/*
* Free all DCBs
*/
void
DC395x_freeDCBs
(
struct
Scsi_Host
*
host
)
/**
* DC395x_free_DCBs - Free all of the DCBs.
*
* @pACB: Adapter to remove the DCBs for.
**/
static
void
DC395x_free_DCBs
(
struct
AdapterCtlBlk
*
pACB
)
{
struct
DeviceCtlBlk
*
pDCB
;
struct
DeviceCtlBlk
*
nDCB
;
struct
AdapterCtlBlk
*
pACB
=
(
struct
AdapterCtlBlk
*
)
(
host
->
hostdata
);
struct
DeviceCtlBlk
*
dcb
;
struct
DeviceCtlBlk
*
dcb_next
;
dprintkdbg
(
DBG_DCB
,
"Free %i DCBs
\n
"
,
pACB
->
DCBCnt
);
pDCB
=
pACB
->
pLinkDCB
;
if
(
pDCB
)
{
do
{
nDCB
=
pDCB
->
pNextDCB
;
for
(
dcb
=
pACB
->
pLinkDCB
;
dcb
!=
NULL
;
dcb
=
dcb_next
)
{
dcb_next
=
dcb
->
pNextDCB
;
dprintkdbg
(
DBG_DCB
,
"Free DCB (ID %i, LUN %i): %p
\n
"
,
pDCB
->
TargetID
,
pDCB
->
TargetLUN
,
pDCB
);
DC395x_remove_dev
(
pACB
,
pDCB
);
/* includes a dc395x_kfree(pDCB); */
printk
(
"."
);
pDCB
=
nDCB
;
}
while
(
pDCB
&&
pACB
->
pLinkDCB
);
dcb
->
TargetID
,
dcb
->
TargetLUN
,
dcb
);
/*
* Free the DCB. This removes the entry from the
* pLinkDCB list and decrements the count in DCBCnt
*/
DC395x_remove_dev
(
pACB
,
dcb
);
}
}
/*
*
Release method
/**
* DC395x_release - shutdown device and release resources that were
*
allocate for it. Called once for each card as it is shutdown.
*
*
Called when we are to shutdown the controller and release all of
*
it's resources.
*/
static
int
DC395x_release
(
struct
Scsi_Host
*
host
)
*
@host: The adapter instance to shutdown.
*
*/
static
void
DC395x_release
(
struct
Scsi_Host
*
host
)
{
struct
AdapterCtlBlk
*
pACB
=
(
struct
AdapterCtlBlk
*
)
(
host
->
hostdata
);
struct
AdapterCtlBlk
*
pACB
=
(
struct
AdapterCtlBlk
*
)
(
host
->
hostdata
);
unsigned
long
flags
;
dprintkl
(
KERN_DEBUG
,
"
release
"
);
dprintkl
(
KERN_DEBUG
,
"
DC395x release
\n
"
);
DC395x_LOCK_IO
(
pACB
->
pScsiHost
);
DC395x_
shutdown
(
host
);
DC395x_free
DCBs
(
host
);
DC395x_
chip_shutdown
(
pACB
);
DC395x_free
_DCBs
(
pACB
);
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
);
}
release_region
(
host
->
io_port
,
host
->
n_io_port
);
DC395x_UNLOCK_IO
(
pACB
->
pScsiHost
);
return
1
;
}
/*
* SCSI host template
*/
static
Scsi_Host_Template
driver_template
=
{
static
Scsi_Host_Template
dc395x_driver_template
=
{
.
module
=
THIS_MODULE
,
.
proc_name
=
DC395X_NAME
,
.
proc_info
=
DC395x_proc_info
,
.
name
=
DC395X_BANNER
" "
DC395X_VERSION
,
.
detect
=
DC395x_detect
,
.
release
=
DC395x_release
,
.
queuecommand
=
DC395x_queue_command
,
.
bios_param
=
DC395x_bios_param
,
.
slave_alloc
=
DC395x_slave_alloc
,
...
...
@@ -6135,9 +6162,114 @@ static Scsi_Host_Template driver_template = {
.
unchecked_isa_dma
=
0
,
.
use_clustering
=
DISABLE_CLUSTERING
,
};
/*
* The following code deals with registering the above scsi host
* template with the higher level scsi code and results in the detect
* method from the template being called during initialisation.
* Called to initialise a single instance of the adaptor
*/
#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"
);
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment