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
ba7f30a8
Commit
ba7f30a8
authored
Aug 09, 2004
by
Linus Torvalds
Browse files
Options
Browse Files
Download
Plain Diff
Merge
bk://kernel.bkbits.net/jgarzik/libata-upstream-2.6
into ppc970.osdl.org:/home/torvalds/v2.6/linux
parents
384278b0
37da1f70
Changes
2
Show whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
250 additions
and
68 deletions
+250
-68
drivers/scsi/libata-core.c
drivers/scsi/libata-core.c
+2
-2
drivers/scsi/sata_nv.c
drivers/scsi/sata_nv.c
+248
-66
No files found.
drivers/scsi/libata-core.c
View file @
ba7f30a8
...
@@ -3252,10 +3252,10 @@ void ata_pci_remove_one (struct pci_dev *pdev)
...
@@ -3252,10 +3252,10 @@ void ata_pci_remove_one (struct pci_dev *pdev)
}
}
free_irq
(
host_set
->
irq
,
host_set
);
free_irq
(
host_set
->
irq
,
host_set
);
if
(
host_set
->
mmio_base
)
iounmap
(
host_set
->
mmio_base
);
if
(
host_set
->
ops
->
host_stop
)
if
(
host_set
->
ops
->
host_stop
)
host_set
->
ops
->
host_stop
(
host_set
);
host_set
->
ops
->
host_stop
(
host_set
);
if
(
host_set
->
mmio_base
)
iounmap
(
host_set
->
mmio_base
);
for
(
i
=
0
;
i
<
host_set
->
n_ports
;
i
++
)
{
for
(
i
=
0
;
i
<
host_set
->
n_ports
;
i
++
)
{
ap
=
host_set
->
ports
[
i
];
ap
=
host_set
->
ports
[
i
];
...
...
drivers/scsi/sata_nv.c
View file @
ba7f30a8
...
@@ -20,6 +20,11 @@
...
@@ -20,6 +20,11 @@
* If you do not delete the provisions above, a recipient may use your
* If you do not delete the provisions above, a recipient may use your
* version of this file under either the OSL or the GPL.
* version of this file under either the OSL or the GPL.
*
*
* 0.02
* - Added support for CK804 SATA controller.
*
* 0.01
* - Initial revision.
*/
*/
#include <linux/config.h>
#include <linux/config.h>
...
@@ -35,7 +40,7 @@
...
@@ -35,7 +40,7 @@
#include <linux/libata.h>
#include <linux/libata.h>
#define DRV_NAME "sata_nv"
#define DRV_NAME "sata_nv"
#define DRV_VERSION "0.0
1
"
#define DRV_VERSION "0.0
2
"
#define NV_PORTS 2
#define NV_PORTS 2
#define NV_PIO_MASK 0x1f
#define NV_PIO_MASK 0x1f
...
@@ -46,6 +51,7 @@
...
@@ -46,6 +51,7 @@
#define NV_PORT1_SCR_REG_OFFSET 0x40
#define NV_PORT1_SCR_REG_OFFSET 0x40
#define NV_INT_STATUS 0x10
#define NV_INT_STATUS 0x10
#define NV_INT_STATUS_CK804 0x440
#define NV_INT_STATUS_PDEV_INT 0x01
#define NV_INT_STATUS_PDEV_INT 0x01
#define NV_INT_STATUS_PDEV_PM 0x02
#define NV_INT_STATUS_PDEV_PM 0x02
#define NV_INT_STATUS_PDEV_ADDED 0x04
#define NV_INT_STATUS_PDEV_ADDED 0x04
...
@@ -62,6 +68,7 @@
...
@@ -62,6 +68,7 @@
NV_INT_STATUS_SDEV_HOTPLUG)
NV_INT_STATUS_SDEV_HOTPLUG)
#define NV_INT_ENABLE 0x11
#define NV_INT_ENABLE 0x11
#define NV_INT_ENABLE_CK804 0x441
#define NV_INT_ENABLE_PDEV_MASK 0x01
#define NV_INT_ENABLE_PDEV_MASK 0x01
#define NV_INT_ENABLE_PDEV_PM 0x02
#define NV_INT_ENABLE_PDEV_PM 0x02
#define NV_INT_ENABLE_PDEV_ADDED 0x04
#define NV_INT_ENABLE_PDEV_ADDED 0x04
...
@@ -80,30 +87,86 @@
...
@@ -80,30 +87,86 @@
#define NV_INT_CONFIG 0x12
#define NV_INT_CONFIG 0x12
#define NV_INT_CONFIG_METHD 0x01 // 0 = INT, 1 = SMI
#define NV_INT_CONFIG_METHD 0x01 // 0 = INT, 1 = SMI
// For PCI config register 20
#define NV_MCP_SATA_CFG_20 0x50
#define NV_MCP_SATA_CFG_20_SATA_SPACE_EN 0x04
static
int
nv_init_one
(
struct
pci_dev
*
pdev
,
const
struct
pci_device_id
*
ent
);
static
int
nv_init_one
(
struct
pci_dev
*
pdev
,
const
struct
pci_device_id
*
ent
);
irqreturn_t
nv_interrupt
(
int
irq
,
void
*
dev_instance
,
struct
pt_regs
*
regs
);
irqreturn_t
nv_interrupt
(
int
irq
,
void
*
dev_instance
,
struct
pt_regs
*
regs
);
static
u32
nv_scr_read
(
struct
ata_port
*
ap
,
unsigned
int
sc_reg
);
static
u32
nv_scr_read
(
struct
ata_port
*
ap
,
unsigned
int
sc_reg
);
static
void
nv_scr_write
(
struct
ata_port
*
ap
,
unsigned
int
sc_reg
,
u32
val
);
static
void
nv_scr_write
(
struct
ata_port
*
ap
,
unsigned
int
sc_reg
,
u32
val
);
static
void
nv_host_stop
(
struct
ata_host_set
*
host_set
);
static
void
nv_host_stop
(
struct
ata_host_set
*
host_set
);
static
void
nv_enable_hotplug
(
struct
ata_probe_ent
*
probe_ent
);
static
void
nv_disable_hotplug
(
struct
ata_host_set
*
host_set
);
static
void
nv_check_hotplug
(
struct
ata_host_set
*
host_set
);
static
void
nv_enable_hotplug_ck804
(
struct
ata_probe_ent
*
probe_ent
);
static
void
nv_disable_hotplug_ck804
(
struct
ata_host_set
*
host_set
);
static
void
nv_check_hotplug_ck804
(
struct
ata_host_set
*
host_set
);
enum
nv_host_type
{
NFORCE2
,
NFORCE3
,
CK804
};
static
struct
pci_device_id
nv_pci_tbl
[]
=
{
static
struct
pci_device_id
nv_pci_tbl
[]
=
{
{
PCI_VENDOR_ID_NVIDIA
,
PCI_DEVICE_ID_NVIDIA_NFORCE2S_SATA
,
{
PCI_VENDOR_ID_NVIDIA
,
PCI_DEVICE_ID_NVIDIA_NFORCE2S_SATA
,
PCI_ANY_ID
,
PCI_ANY_ID
,
},
PCI_ANY_ID
,
PCI_ANY_ID
,
0
,
0
,
NFORCE2
},
{
PCI_VENDOR_ID_NVIDIA
,
PCI_DEVICE_ID_NVIDIA_NFORCE3S_SATA
,
{
PCI_VENDOR_ID_NVIDIA
,
PCI_DEVICE_ID_NVIDIA_NFORCE3S_SATA
,
PCI_ANY_ID
,
PCI_ANY_ID
,
},
PCI_ANY_ID
,
PCI_ANY_ID
,
0
,
0
,
NFORCE3
},
{
PCI_VENDOR_ID_NVIDIA
,
PCI_DEVICE_ID_NVIDIA_NFORCE3S_SATA2
,
{
PCI_VENDOR_ID_NVIDIA
,
PCI_DEVICE_ID_NVIDIA_NFORCE3S_SATA2
,
PCI_ANY_ID
,
PCI_ANY_ID
,
},
PCI_ANY_ID
,
PCI_ANY_ID
,
0
,
0
,
NFORCE3
},
{
PCI_VENDOR_ID_NVIDIA
,
PCI_DEVICE_ID_NVIDIA_NFORCE_CK804_SATA
,
{
PCI_VENDOR_ID_NVIDIA
,
PCI_DEVICE_ID_NVIDIA_NFORCE_CK804_SATA
,
PCI_ANY_ID
,
PCI_ANY_ID
,
},
PCI_ANY_ID
,
PCI_ANY_ID
,
0
,
0
,
CK804
},
{
PCI_VENDOR_ID_NVIDIA
,
PCI_DEVICE_ID_NVIDIA_NFORCE_CK804_SATA2
,
{
PCI_VENDOR_ID_NVIDIA
,
PCI_DEVICE_ID_NVIDIA_NFORCE_CK804_SATA2
,
PCI_ANY_ID
,
PCI_ANY_ID
,
},
PCI_ANY_ID
,
PCI_ANY_ID
,
0
,
0
,
CK804
},
{
PCI_VENDOR_ID_NVIDIA
,
PCI_DEVICE_ID_NVIDIA_NFORCE_MCP04_SATA
,
{
PCI_VENDOR_ID_NVIDIA
,
PCI_DEVICE_ID_NVIDIA_NFORCE_MCP04_SATA
,
PCI_ANY_ID
,
PCI_ANY_ID
,
},
PCI_ANY_ID
,
PCI_ANY_ID
,
0
,
0
,
CK804
},
{
PCI_VENDOR_ID_NVIDIA
,
PCI_DEVICE_ID_NVIDIA_NFORCE_MCP04_SATA2
,
{
PCI_VENDOR_ID_NVIDIA
,
PCI_DEVICE_ID_NVIDIA_NFORCE_MCP04_SATA2
,
PCI_ANY_ID
,
PCI_ANY_ID
,
},
PCI_ANY_ID
,
PCI_ANY_ID
,
0
,
0
,
CK804
},
{
0
,
}
/* terminate list */
{
0
,
}
/* terminate list */
};
};
#define NV_HOST_FLAGS_SCR_MMIO 0x00000001
struct
nv_host_desc
{
enum
nv_host_type
host_type
;
unsigned
long
host_flags
;
void
(
*
enable_hotplug
)(
struct
ata_probe_ent
*
probe_ent
);
void
(
*
disable_hotplug
)(
struct
ata_host_set
*
host_set
);
void
(
*
check_hotplug
)(
struct
ata_host_set
*
host_set
);
};
static
struct
nv_host_desc
nv_device_tbl
[]
=
{
{
.
host_type
=
NFORCE2
,
.
host_flags
=
0x00000000
,
.
enable_hotplug
=
nv_enable_hotplug
,
.
disable_hotplug
=
nv_disable_hotplug
,
.
check_hotplug
=
nv_check_hotplug
,
},
{
.
host_type
=
NFORCE3
,
.
host_flags
=
0x00000000
,
.
enable_hotplug
=
nv_enable_hotplug
,
.
disable_hotplug
=
nv_disable_hotplug
,
.
check_hotplug
=
nv_check_hotplug
,
},
{
.
host_type
=
CK804
,
.
host_flags
=
NV_HOST_FLAGS_SCR_MMIO
,
.
enable_hotplug
=
nv_enable_hotplug_ck804
,
.
disable_hotplug
=
nv_disable_hotplug_ck804
,
.
check_hotplug
=
nv_check_hotplug_ck804
,
},
};
struct
nv_host
{
struct
nv_host_desc
*
host_desc
;
};
static
struct
pci_driver
nv_pci_driver
=
{
static
struct
pci_driver
nv_pci_driver
=
{
.
name
=
DRV_NAME
,
.
name
=
DRV_NAME
,
.
id_table
=
nv_pci_tbl
,
.
id_table
=
nv_pci_tbl
,
...
@@ -158,11 +221,10 @@ MODULE_DEVICE_TABLE(pci, nv_pci_tbl);
...
@@ -158,11 +221,10 @@ MODULE_DEVICE_TABLE(pci, nv_pci_tbl);
irqreturn_t
nv_interrupt
(
int
irq
,
void
*
dev_instance
,
struct
pt_regs
*
regs
)
irqreturn_t
nv_interrupt
(
int
irq
,
void
*
dev_instance
,
struct
pt_regs
*
regs
)
{
{
struct
ata_host_set
*
host_set
=
dev_instance
;
struct
ata_host_set
*
host_set
=
dev_instance
;
struct
nv_host
*
host
=
host_set
->
private_data
;
unsigned
int
i
;
unsigned
int
i
;
unsigned
int
handled
=
0
;
unsigned
int
handled
=
0
;
unsigned
long
flags
;
unsigned
long
flags
;
u8
intr_status
;
u8
intr_enable
;
spin_lock_irqsave
(
&
host_set
->
lock
,
flags
);
spin_lock_irqsave
(
&
host_set
->
lock
,
flags
);
...
@@ -178,34 +240,10 @@ irqreturn_t nv_interrupt (int irq, void *dev_instance, struct pt_regs *regs)
...
@@ -178,34 +240,10 @@ irqreturn_t nv_interrupt (int irq, void *dev_instance, struct pt_regs *regs)
handled
+=
ata_host_intr
(
ap
,
qc
);
handled
+=
ata_host_intr
(
ap
,
qc
);
}
}
intr_status
=
inb
(
ap
->
ioaddr
.
scr_addr
+
NV_INT_STATUS
);
intr_enable
=
inb
(
ap
->
ioaddr
.
scr_addr
+
NV_INT_ENABLE
);
// Clear interrupt status.
outb
(
0xff
,
ap
->
ioaddr
.
scr_addr
+
NV_INT_STATUS
);
if
(
intr_status
&
NV_INT_STATUS_HOTPLUG
)
{
if
(
intr_status
&
NV_INT_STATUS_PDEV_ADDED
)
{
printk
(
KERN_WARNING
"ata%u: "
"Primary device added
\n
"
,
ap
->
id
);
}
if
(
intr_status
&
NV_INT_STATUS_PDEV_REMOVED
)
{
printk
(
KERN_WARNING
"ata%u: "
"Primary device removed
\n
"
,
ap
->
id
);
}
}
if
(
intr_status
&
NV_INT_STATUS_SDEV_ADDED
)
{
if
(
host
->
host_desc
->
check_hotplug
)
printk
(
KERN_WARNING
"ata%u: "
host
->
host_desc
->
check_hotplug
(
host_set
);
"Secondary device added
\n
"
,
ap
->
id
);
}
if
(
intr_status
&
NV_INT_STATUS_SDEV_REMOVED
)
{
printk
(
KERN_WARNING
"ata%u: "
"Secondary device removed
\n
"
,
ap
->
id
);
}
}
}
spin_unlock_irqrestore
(
&
host_set
->
lock
,
flags
);
spin_unlock_irqrestore
(
&
host_set
->
lock
,
flags
);
...
@@ -214,41 +252,48 @@ irqreturn_t nv_interrupt (int irq, void *dev_instance, struct pt_regs *regs)
...
@@ -214,41 +252,48 @@ irqreturn_t nv_interrupt (int irq, void *dev_instance, struct pt_regs *regs)
static
u32
nv_scr_read
(
struct
ata_port
*
ap
,
unsigned
int
sc_reg
)
static
u32
nv_scr_read
(
struct
ata_port
*
ap
,
unsigned
int
sc_reg
)
{
{
struct
ata_host_set
*
host_set
=
ap
->
host_set
;
struct
nv_host
*
host
=
host_set
->
private_data
;
if
(
sc_reg
>
SCR_CONTROL
)
if
(
sc_reg
>
SCR_CONTROL
)
return
0xffffffffU
;
return
0xffffffffU
;
if
(
host
->
host_desc
->
host_flags
&
NV_HOST_FLAGS_SCR_MMIO
)
return
readl
(
ap
->
ioaddr
.
scr_addr
+
(
sc_reg
*
4
));
else
return
inl
(
ap
->
ioaddr
.
scr_addr
+
(
sc_reg
*
4
));
return
inl
(
ap
->
ioaddr
.
scr_addr
+
(
sc_reg
*
4
));
}
}
static
void
nv_scr_write
(
struct
ata_port
*
ap
,
unsigned
int
sc_reg
,
u32
val
)
static
void
nv_scr_write
(
struct
ata_port
*
ap
,
unsigned
int
sc_reg
,
u32
val
)
{
{
struct
ata_host_set
*
host_set
=
ap
->
host_set
;
struct
nv_host
*
host
=
host_set
->
private_data
;
if
(
sc_reg
>
SCR_CONTROL
)
if
(
sc_reg
>
SCR_CONTROL
)
return
;
return
;
if
(
host
->
host_desc
->
host_flags
&
NV_HOST_FLAGS_SCR_MMIO
)
writel
(
val
,
ap
->
ioaddr
.
scr_addr
+
(
sc_reg
*
4
));
else
outl
(
val
,
ap
->
ioaddr
.
scr_addr
+
(
sc_reg
*
4
));
outl
(
val
,
ap
->
ioaddr
.
scr_addr
+
(
sc_reg
*
4
));
}
}
static
void
nv_host_stop
(
struct
ata_host_set
*
host_set
)
static
void
nv_host_stop
(
struct
ata_host_set
*
host_set
)
{
{
int
i
;
struct
nv_host
*
host
=
host_set
->
private_data
;
for
(
i
=
0
;
i
<
host_set
->
n_ports
;
i
++
)
{
u8
intr_mask
;
// Disable hotplug event interrupts.
// Disable hotplug event interrupts.
intr_mask
=
inb
(
host_set
->
ports
[
i
]
->
ioaddr
.
scr_addr
+
if
(
host
->
host_desc
->
disable_hotplug
)
NV_INT_ENABLE
);
host
->
host_desc
->
disable_hotplug
(
host_set
);
intr_mask
&=
~
(
NV_INT_ENABLE_HOTPLUG
);
outb
(
intr_mask
,
host_set
->
ports
[
i
]
->
ioaddr
.
scr_addr
+
kfree
(
host
);
NV_INT_ENABLE
);
}
}
}
static
int
nv_init_one
(
struct
pci_dev
*
pdev
,
const
struct
pci_device_id
*
ent
)
static
int
nv_init_one
(
struct
pci_dev
*
pdev
,
const
struct
pci_device_id
*
ent
)
{
{
static
int
printed_version
=
0
;
static
int
printed_version
=
0
;
struct
nv_host
*
host
;
struct
ata_probe_ent
*
probe_ent
=
NULL
;
struct
ata_probe_ent
*
probe_ent
=
NULL
;
int
i
;
int
rc
;
int
rc
;
if
(
!
printed_version
++
)
if
(
!
printed_version
++
)
...
@@ -275,6 +320,14 @@ static int nv_init_one (struct pci_dev *pdev, const struct pci_device_id *ent)
...
@@ -275,6 +320,14 @@ static int nv_init_one (struct pci_dev *pdev, const struct pci_device_id *ent)
goto
err_out_regions
;
goto
err_out_regions
;
}
}
host
=
kmalloc
(
sizeof
(
struct
nv_host
),
GFP_KERNEL
);
if
(
!
host
)
{
rc
=
-
ENOMEM
;
goto
err_out_free_ent
;
}
host
->
host_desc
=
&
nv_device_tbl
[
ent
->
driver_data
];
memset
(
probe_ent
,
0
,
sizeof
(
*
probe_ent
));
memset
(
probe_ent
,
0
,
sizeof
(
*
probe_ent
));
INIT_LIST_HEAD
(
&
probe_ent
->
node
);
INIT_LIST_HEAD
(
&
probe_ent
->
node
);
...
@@ -284,6 +337,7 @@ static int nv_init_one (struct pci_dev *pdev, const struct pci_device_id *ent)
...
@@ -284,6 +337,7 @@ static int nv_init_one (struct pci_dev *pdev, const struct pci_device_id *ent)
ATA_FLAG_SATA_RESET
|
ATA_FLAG_SATA_RESET
|
ATA_FLAG_SRST
|
ATA_FLAG_SRST
|
ATA_FLAG_NO_LEGACY
;
ATA_FLAG_NO_LEGACY
;
probe_ent
->
port_ops
=
&
nv_ops
;
probe_ent
->
port_ops
=
&
nv_ops
;
probe_ent
->
n_ports
=
NV_PORTS
;
probe_ent
->
n_ports
=
NV_PORTS
;
probe_ent
->
irq
=
pdev
->
irq
;
probe_ent
->
irq
=
pdev
->
irq
;
...
@@ -298,8 +352,6 @@ static int nv_init_one (struct pci_dev *pdev, const struct pci_device_id *ent)
...
@@ -298,8 +352,6 @@ static int nv_init_one (struct pci_dev *pdev, const struct pci_device_id *ent)
pci_resource_start
(
pdev
,
1
)
|
ATA_PCI_CTL_OFS
;
pci_resource_start
(
pdev
,
1
)
|
ATA_PCI_CTL_OFS
;
probe_ent
->
port
[
0
].
bmdma_addr
=
probe_ent
->
port
[
0
].
bmdma_addr
=
pci_resource_start
(
pdev
,
4
)
|
NV_PORT0_BMDMA_REG_OFFSET
;
pci_resource_start
(
pdev
,
4
)
|
NV_PORT0_BMDMA_REG_OFFSET
;
probe_ent
->
port
[
0
].
scr_addr
=
pci_resource_start
(
pdev
,
5
)
|
NV_PORT0_SCR_REG_OFFSET
;
probe_ent
->
port
[
1
].
cmd_addr
=
pci_resource_start
(
pdev
,
2
);
probe_ent
->
port
[
1
].
cmd_addr
=
pci_resource_start
(
pdev
,
2
);
ata_std_ports
(
&
probe_ent
->
port
[
1
]);
ata_std_ports
(
&
probe_ent
->
port
[
1
]);
...
@@ -308,31 +360,48 @@ static int nv_init_one (struct pci_dev *pdev, const struct pci_device_id *ent)
...
@@ -308,31 +360,48 @@ static int nv_init_one (struct pci_dev *pdev, const struct pci_device_id *ent)
pci_resource_start
(
pdev
,
3
)
|
ATA_PCI_CTL_OFS
;
pci_resource_start
(
pdev
,
3
)
|
ATA_PCI_CTL_OFS
;
probe_ent
->
port
[
1
].
bmdma_addr
=
probe_ent
->
port
[
1
].
bmdma_addr
=
pci_resource_start
(
pdev
,
4
)
|
NV_PORT1_BMDMA_REG_OFFSET
;
pci_resource_start
(
pdev
,
4
)
|
NV_PORT1_BMDMA_REG_OFFSET
;
probe_ent
->
private_data
=
host
;
if
(
host
->
host_desc
->
host_flags
&
NV_HOST_FLAGS_SCR_MMIO
)
{
unsigned
long
base
;
probe_ent
->
mmio_base
=
ioremap
(
pci_resource_start
(
pdev
,
5
),
pci_resource_len
(
pdev
,
5
));
if
(
probe_ent
->
mmio_base
==
NULL
)
goto
err_out_free_ent
;
base
=
(
unsigned
long
)
probe_ent
->
mmio_base
;
probe_ent
->
port
[
0
].
scr_addr
=
base
+
NV_PORT0_SCR_REG_OFFSET
;
probe_ent
->
port
[
1
].
scr_addr
=
base
+
NV_PORT1_SCR_REG_OFFSET
;
}
else
{
probe_ent
->
port
[
0
].
scr_addr
=
pci_resource_start
(
pdev
,
5
)
|
NV_PORT0_SCR_REG_OFFSET
;
probe_ent
->
port
[
1
].
scr_addr
=
probe_ent
->
port
[
1
].
scr_addr
=
pci_resource_start
(
pdev
,
5
)
|
NV_PORT1_SCR_REG_OFFSET
;
pci_resource_start
(
pdev
,
5
)
|
NV_PORT1_SCR_REG_OFFSET
;
}
pci_set_master
(
pdev
);
pci_set_master
(
pdev
);
rc
=
ata_device_add
(
probe_ent
);
if
(
rc
!=
NV_PORTS
)
goto
err_out_regions
;
// Enable hotplug event interrupts.
// Enable hotplug event interrupts.
for
(
i
=
0
;
i
<
probe_ent
->
n_ports
;
i
++
)
{
if
(
host
->
host_desc
->
enable_hotplug
)
u8
intr_mask
;
host
->
host_desc
->
enable_hotplug
(
probe_ent
);
outb
(
NV_INT_STATUS_HOTPLUG
,
probe_ent
->
port
[
i
].
scr_addr
+
NV_INT_STATUS
);
intr_mask
=
inb
(
probe_ent
->
port
[
i
].
scr_addr
+
NV_INT_ENABLE
);
rc
=
ata_device_add
(
probe_ent
);
intr_mask
|=
NV_INT_ENABLE_HOTPLUG
;
if
(
rc
!=
NV_PORTS
)
outb
(
intr_mask
,
probe_ent
->
port
[
i
].
scr_addr
+
NV_INT_ENABLE
);
goto
err_out_free_ent
;
}
kfree
(
probe_ent
);
kfree
(
probe_ent
);
return
0
;
return
0
;
err_out_free_ent:
kfree
(
probe_ent
);
err_out_regions:
err_out_regions:
pci_release_regions
(
pdev
);
pci_release_regions
(
pdev
);
...
@@ -341,6 +410,119 @@ static int nv_init_one (struct pci_dev *pdev, const struct pci_device_id *ent)
...
@@ -341,6 +410,119 @@ static int nv_init_one (struct pci_dev *pdev, const struct pci_device_id *ent)
return
rc
;
return
rc
;
}
}
static
void
nv_enable_hotplug
(
struct
ata_probe_ent
*
probe_ent
)
{
u8
intr_mask
;
outb
(
NV_INT_STATUS_HOTPLUG
,
(
unsigned
long
)
probe_ent
->
mmio_base
+
NV_INT_STATUS
);
intr_mask
=
inb
((
unsigned
long
)
probe_ent
->
mmio_base
+
NV_INT_ENABLE
);
intr_mask
|=
NV_INT_ENABLE_HOTPLUG
;
outb
(
intr_mask
,
(
unsigned
long
)
probe_ent
->
mmio_base
+
NV_INT_ENABLE
);
}
static
void
nv_disable_hotplug
(
struct
ata_host_set
*
host_set
)
{
u8
intr_mask
;
intr_mask
=
inb
((
unsigned
long
)
host_set
->
mmio_base
+
NV_INT_ENABLE
);
intr_mask
&=
~
(
NV_INT_ENABLE_HOTPLUG
);
outb
(
intr_mask
,
(
unsigned
long
)
host_set
->
mmio_base
+
NV_INT_ENABLE
);
}
static
void
nv_check_hotplug
(
struct
ata_host_set
*
host_set
)
{
u8
intr_status
;
intr_status
=
inb
((
unsigned
long
)
host_set
->
mmio_base
+
NV_INT_STATUS
);
// Clear interrupt status.
outb
(
0xff
,
(
unsigned
long
)
host_set
->
mmio_base
+
NV_INT_STATUS
);
if
(
intr_status
&
NV_INT_STATUS_HOTPLUG
)
{
if
(
intr_status
&
NV_INT_STATUS_PDEV_ADDED
)
printk
(
KERN_WARNING
"nv_sata: "
"Primary device added
\n
"
);
if
(
intr_status
&
NV_INT_STATUS_PDEV_REMOVED
)
printk
(
KERN_WARNING
"nv_sata: "
"Primary device removed
\n
"
);
if
(
intr_status
&
NV_INT_STATUS_SDEV_ADDED
)
printk
(
KERN_WARNING
"nv_sata: "
"Secondary device added
\n
"
);
if
(
intr_status
&
NV_INT_STATUS_SDEV_REMOVED
)
printk
(
KERN_WARNING
"nv_sata: "
"Secondary device removed
\n
"
);
}
}
static
void
nv_enable_hotplug_ck804
(
struct
ata_probe_ent
*
probe_ent
)
{
u8
intr_mask
;
u8
regval
;
pci_read_config_byte
(
probe_ent
->
pdev
,
NV_MCP_SATA_CFG_20
,
&
regval
);
regval
|=
NV_MCP_SATA_CFG_20_SATA_SPACE_EN
;
pci_write_config_byte
(
probe_ent
->
pdev
,
NV_MCP_SATA_CFG_20
,
regval
);
writeb
(
NV_INT_STATUS_HOTPLUG
,
probe_ent
->
mmio_base
+
NV_INT_STATUS_CK804
);
intr_mask
=
readb
(
probe_ent
->
mmio_base
+
NV_INT_ENABLE_CK804
);
intr_mask
|=
NV_INT_ENABLE_HOTPLUG
;
writeb
(
intr_mask
,
probe_ent
->
mmio_base
+
NV_INT_ENABLE_CK804
);
}
static
void
nv_disable_hotplug_ck804
(
struct
ata_host_set
*
host_set
)
{
u8
intr_mask
;
u8
regval
;
intr_mask
=
readb
(
host_set
->
mmio_base
+
NV_INT_ENABLE_CK804
);
intr_mask
&=
~
(
NV_INT_ENABLE_HOTPLUG
);
writeb
(
intr_mask
,
host_set
->
mmio_base
+
NV_INT_ENABLE_CK804
);
pci_read_config_byte
(
host_set
->
pdev
,
NV_MCP_SATA_CFG_20
,
&
regval
);
regval
&=
~
NV_MCP_SATA_CFG_20_SATA_SPACE_EN
;
pci_write_config_byte
(
host_set
->
pdev
,
NV_MCP_SATA_CFG_20
,
regval
);
}
static
void
nv_check_hotplug_ck804
(
struct
ata_host_set
*
host_set
)
{
u8
intr_status
;
intr_status
=
readb
(
host_set
->
mmio_base
+
NV_INT_STATUS_CK804
);
// Clear interrupt status.
writeb
(
0xff
,
host_set
->
mmio_base
+
NV_INT_STATUS_CK804
);
if
(
intr_status
&
NV_INT_STATUS_HOTPLUG
)
{
if
(
intr_status
&
NV_INT_STATUS_PDEV_ADDED
)
printk
(
KERN_WARNING
"nv_sata: "
"Primary device added
\n
"
);
if
(
intr_status
&
NV_INT_STATUS_PDEV_REMOVED
)
printk
(
KERN_WARNING
"nv_sata: "
"Primary device removed
\n
"
);
if
(
intr_status
&
NV_INT_STATUS_SDEV_ADDED
)
printk
(
KERN_WARNING
"nv_sata: "
"Secondary device added
\n
"
);
if
(
intr_status
&
NV_INT_STATUS_SDEV_REMOVED
)
printk
(
KERN_WARNING
"nv_sata: "
"Secondary device removed
\n
"
);
}
}
static
int
__init
nv_init
(
void
)
static
int
__init
nv_init
(
void
)
{
{
return
pci_module_init
(
&
nv_pci_driver
);
return
pci_module_init
(
&
nv_pci_driver
);
...
...
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