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
971db06e
Commit
971db06e
authored
Oct 14, 2004
by
Jeff Garzik
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
[libata] add AHCI driver
parent
e58d2eb0
Changes
3
Show whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
982 additions
and
0 deletions
+982
-0
drivers/scsi/Kconfig
drivers/scsi/Kconfig
+8
-0
drivers/scsi/Makefile
drivers/scsi/Makefile
+1
-0
drivers/scsi/ahci.c
drivers/scsi/ahci.c
+973
-0
No files found.
drivers/scsi/Kconfig
View file @
971db06e
...
@@ -406,6 +406,14 @@ config SCSI_SATA
...
@@ -406,6 +406,14 @@ config SCSI_SATA
If unsure, say N.
If unsure, say N.
config SCSI_SATA_AHCI
tristate "AHCI SATA support"
depends on SCSI_SATA && PCI && EXPERIMENTAL
help
This option enables support for AHCI Serial ATA.
If unsure, say N.
config SCSI_SATA_SVW
config SCSI_SATA_SVW
tristate "ServerWorks Frodo / Apple K2 SATA support (EXPERIMENTAL)"
tristate "ServerWorks Frodo / Apple K2 SATA support (EXPERIMENTAL)"
depends on SCSI_SATA && PCI && EXPERIMENTAL
depends on SCSI_SATA && PCI && EXPERIMENTAL
...
...
drivers/scsi/Makefile
View file @
971db06e
...
@@ -121,6 +121,7 @@ obj-$(CONFIG_SCSI_LASI700) += 53c700.o lasi700.o
...
@@ -121,6 +121,7 @@ obj-$(CONFIG_SCSI_LASI700) += 53c700.o lasi700.o
obj-$(CONFIG_SCSI_NSP32)
+=
nsp32.o
obj-$(CONFIG_SCSI_NSP32)
+=
nsp32.o
obj-$(CONFIG_SCSI_IPR)
+=
ipr.o
obj-$(CONFIG_SCSI_IPR)
+=
ipr.o
obj-$(CONFIG_SCSI_IBMVSCSI)
+=
ibmvscsi/
obj-$(CONFIG_SCSI_IBMVSCSI)
+=
ibmvscsi/
obj-$(CONFIG_SCSI_SATA_AHCI)
+=
libata.o ahci.o
obj-$(CONFIG_SCSI_SATA_SVW)
+=
libata.o sata_svw.o
obj-$(CONFIG_SCSI_SATA_SVW)
+=
libata.o sata_svw.o
obj-$(CONFIG_SCSI_ATA_PIIX)
+=
libata.o ata_piix.o
obj-$(CONFIG_SCSI_ATA_PIIX)
+=
libata.o ata_piix.o
obj-$(CONFIG_SCSI_SATA_PROMISE)
+=
libata.o sata_promise.o
obj-$(CONFIG_SCSI_SATA_PROMISE)
+=
libata.o sata_promise.o
...
...
drivers/scsi/ahci.c
0 → 100644
View file @
971db06e
/*
* ahci.c - AHCI SATA support
*
* Copyright 2004 Red Hat, Inc.
*
* The contents of this file are subject to the Open
* Software License version 1.1 that can be found at
* http://www.opensource.org/licenses/osl-1.1.txt and is included herein
* by reference.
*
* Alternatively, the contents of this file may be used under the terms
* of the GNU General Public License version 2 (the "GPL") as distributed
* in the kernel source COPYING file, in which case the provisions of
* the GPL are applicable instead of the above. If you wish to allow
* the use of your version of this file only under the terms of the
* GPL and not to allow others to use your version of this file under
* the OSL, indicate your decision by deleting the provisions above and
* replace them with the notice and other provisions required by the GPL.
* 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 1.0 of the AHCI specification:
* http://www.intel.com/technology/serialata/pdf/rev1_0.pdf
*
*/
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/pci.h>
#include <linux/init.h>
#include <linux/blkdev.h>
#include <linux/delay.h>
#include <linux/interrupt.h>
#include <linux/sched.h>
#include "scsi.h"
#include <scsi/scsi_host.h>
#include <linux/libata.h>
#include <asm/io.h>
#define DRV_NAME "ahci"
#define DRV_VERSION "0.10"
enum
{
AHCI_PCI_BAR
=
5
,
AHCI_MAX_SG
=
512
,
/* hardware max is 64K */
AHCI_DMA_BOUNDARY
=
0xffffffff
,
AHCI_USE_CLUSTERING
=
0
,
AHCI_CMD_SLOT_SZ
=
32
,
AHCI_RX_FIS_SZ
=
256
,
AHCI_CMD_TBL_SZ
=
128
+
(
AHCI_MAX_SG
*
16
),
AHCI_PORT_PRIV_DMA_SZ
=
AHCI_CMD_SLOT_SZ
+
AHCI_CMD_TBL_SZ
+
AHCI_RX_FIS_SZ
,
AHCI_IRQ_ON_SG
=
(
1
<<
31
),
AHCI_CMD_WRITE
=
(
1
<<
6
),
RX_FIS_D2H_REG
=
0x40
,
/* offset of D2H Register FIS data */
board_ahci
=
0
,
/* global controller registers */
HOST_CAP
=
0x00
,
/* host capabilities */
HOST_CTL
=
0x04
,
/* global host control */
HOST_IRQ_STAT
=
0x08
,
/* interrupt status */
HOST_PORTS_IMPL
=
0x0c
,
/* bitmap of implemented ports */
HOST_VERSION
=
0x10
,
/* AHCI spec. version compliancy */
/* HOST_CTL bits */
HOST_RESET
=
(
1
<<
0
),
/* reset controller; self-clear */
HOST_IRQ_EN
=
(
1
<<
1
),
/* global IRQ enable */
HOST_AHCI_EN
=
(
1
<<
31
),
/* AHCI enabled */
/* HOST_CAP bits */
HOST_CAP_64
=
(
1
<<
31
),
/* PCI DAC (64-bit DMA) support */
/* registers for each SATA port */
PORT_LST_ADDR
=
0x00
,
/* command list DMA addr */
PORT_LST_ADDR_HI
=
0x04
,
/* command list DMA addr hi */
PORT_FIS_ADDR
=
0x08
,
/* FIS rx buf addr */
PORT_FIS_ADDR_HI
=
0x0c
,
/* FIS rx buf addr hi */
PORT_IRQ_STAT
=
0x10
,
/* interrupt status */
PORT_IRQ_MASK
=
0x14
,
/* interrupt enable/disable mask */
PORT_CMD
=
0x18
,
/* port command */
PORT_TFDATA
=
0x20
,
/* taskfile data */
PORT_CMD_ISSUE
=
0x38
,
/* command issue */
PORT_SCR
=
0x28
,
/* SATA phy register block */
PORT_SCR_CTL
=
0x2c
,
/* SATA phy register: SControl */
PORT_SCR_ERR
=
0x30
,
/* SATA phy register: SError */
/* PORT_IRQ_{STAT,MASK} bits */
PORT_IRQ_COLD_PRES
=
(
1
<<
31
),
/* cold presence detect */
PORT_IRQ_TF_ERR
=
(
1
<<
30
),
/* task file error */
PORT_IRQ_HBUS_ERR
=
(
1
<<
29
),
/* host bus fatal error */
PORT_IRQ_HBUS_DATA_ERR
=
(
1
<<
28
),
/* host bus data error */
PORT_IRQ_IF_ERR
=
(
1
<<
27
),
/* interface fatal error */
PORT_IRQ_IF_NONFATAL
=
(
1
<<
26
),
/* interface non-fatal error */
PORT_IRQ_OVERFLOW
=
(
1
<<
24
),
/* xfer exhausted available S/G */
PORT_IRQ_BAD_PMP
=
(
1
<<
23
),
/* incorrect port multiplier */
PORT_IRQ_PHYRDY
=
(
1
<<
22
),
/* PhyRdy changed */
PORT_IRQ_DEV_ILCK
=
(
1
<<
7
),
/* device interlock */
PORT_IRQ_CONNECT
=
(
1
<<
6
),
/* port connect change status */
PORT_IRQ_SG_DONE
=
(
1
<<
5
),
/* descriptor processed */
PORT_IRQ_UNK_FIS
=
(
1
<<
4
),
/* unknown FIS rx'd */
PORT_IRQ_SDB_FIS
=
(
1
<<
3
),
/* Set Device Bits FIS rx'd */
PORT_IRQ_DMAS_FIS
=
(
1
<<
2
),
/* DMA Setup FIS rx'd */
PORT_IRQ_PIOS_FIS
=
(
1
<<
1
),
/* PIO Setup FIS rx'd */
PORT_IRQ_D2H_REG_FIS
=
(
1
<<
0
),
/* D2H Register FIS rx'd */
PORT_IRQ_FATAL
=
PORT_IRQ_TF_ERR
|
PORT_IRQ_HBUS_ERR
|
PORT_IRQ_HBUS_DATA_ERR
|
PORT_IRQ_IF_ERR
,
DEF_PORT_IRQ
=
PORT_IRQ_FATAL
|
PORT_IRQ_PHYRDY
|
PORT_IRQ_D2H_REG_FIS
,
/* PORT_CMD bits */
PORT_CMD_LIST_ON
=
(
1
<<
15
),
/* cmd list DMA engine running */
PORT_CMD_FIS_ON
=
(
1
<<
14
),
/* FIS DMA engine running */
PORT_CMD_FIS_RX
=
(
1
<<
4
),
/* Enable FIS receive DMA engine */
PORT_CMD_POWER_ON
=
(
1
<<
2
),
/* Power up device */
PORT_CMD_SPIN_UP
=
(
1
<<
1
),
/* Spin up device */
PORT_CMD_START
=
(
1
<<
0
),
/* Enable port DMA engine */
PORT_CMD_ICC_ACTIVE
=
(
0x1
<<
28
),
/* Put i/f in active state */
PORT_CMD_ICC_PARTIAL
=
(
0x2
<<
28
),
/* Put i/f in partial state */
PORT_CMD_ICC_SLUMBER
=
(
0x6
<<
28
),
/* Put i/f in slumber state */
};
struct
ahci_cmd_hdr
{
u32
opts
;
u32
status
;
u32
tbl_addr
;
u32
tbl_addr_hi
;
u32
reserved
[
4
];
};
struct
ahci_sg
{
u32
addr
;
u32
addr_hi
;
u32
reserved
;
u32
flags_size
;
};
struct
ahci_host_priv
{
unsigned
long
flags
;
u32
cap
;
/* cache of HOST_CAP register */
u32
port_map
;
/* cache of HOST_PORTS_IMPL reg */
};
struct
ahci_port_priv
{
struct
ahci_cmd_hdr
*
cmd_slot
;
dma_addr_t
cmd_slot_dma
;
void
*
cmd_tbl
;
dma_addr_t
cmd_tbl_dma
;
struct
ahci_sg
*
cmd_tbl_sg
;
void
*
rx_fis
;
dma_addr_t
rx_fis_dma
;
};
static
u32
ahci_scr_read
(
struct
ata_port
*
ap
,
unsigned
int
sc_reg
);
static
void
ahci_scr_write
(
struct
ata_port
*
ap
,
unsigned
int
sc_reg
,
u32
val
);
static
int
ahci_init_one
(
struct
pci_dev
*
pdev
,
const
struct
pci_device_id
*
ent
);
static
void
ahci_dma_setup
(
struct
ata_queued_cmd
*
qc
);
static
void
ahci_dma_start
(
struct
ata_queued_cmd
*
qc
);
static
irqreturn_t
ahci_interrupt
(
int
irq
,
void
*
dev_instance
,
struct
pt_regs
*
regs
);
static
void
ahci_irq_clear
(
struct
ata_port
*
ap
);
static
void
ahci_eng_timeout
(
struct
ata_port
*
ap
);
static
int
ahci_port_start
(
struct
ata_port
*
ap
);
static
void
ahci_port_stop
(
struct
ata_port
*
ap
);
static
void
ahci_host_stop
(
struct
ata_host_set
*
host_set
);
static
void
ahci_qc_prep
(
struct
ata_queued_cmd
*
qc
);
static
u8
ahci_check_status
(
struct
ata_port
*
ap
);
static
void
ahci_tf_load
(
struct
ata_port
*
ap
,
struct
ata_taskfile
*
tf
);
static
void
ahci_tf_read
(
struct
ata_port
*
ap
,
struct
ata_taskfile
*
tf
);
static
void
ahci_exec_command
(
struct
ata_port
*
ap
,
struct
ata_taskfile
*
tf
);
static
inline
int
ahci_host_intr
(
struct
ata_port
*
ap
,
struct
ata_queued_cmd
*
qc
);
static
Scsi_Host_Template
ahci_sht
=
{
.
module
=
THIS_MODULE
,
.
name
=
DRV_NAME
,
.
queuecommand
=
ata_scsi_queuecmd
,
.
eh_strategy_handler
=
ata_scsi_error
,
.
can_queue
=
ATA_DEF_QUEUE
,
.
this_id
=
ATA_SHT_THIS_ID
,
.
sg_tablesize
=
AHCI_MAX_SG
,
.
max_sectors
=
ATA_MAX_SECTORS
,
.
cmd_per_lun
=
ATA_SHT_CMD_PER_LUN
,
.
emulated
=
ATA_SHT_EMULATED
,
.
use_clustering
=
AHCI_USE_CLUSTERING
,
.
proc_name
=
DRV_NAME
,
.
dma_boundary
=
AHCI_DMA_BOUNDARY
,
.
slave_configure
=
ata_scsi_slave_config
,
.
bios_param
=
ata_std_bios_param
,
};
static
struct
ata_port_operations
ahci_ops
=
{
.
port_disable
=
ata_port_disable
,
.
tf_load
=
ahci_tf_load
,
.
tf_read
=
ahci_tf_read
,
.
check_status
=
ahci_check_status
,
.
exec_command
=
ahci_exec_command
,
.
phy_reset
=
sata_phy_reset
,
.
bmdma_setup
=
ahci_dma_setup
,
.
bmdma_start
=
ahci_dma_start
,
.
qc_prep
=
ahci_qc_prep
,
.
qc_issue
=
ata_qc_issue_prot
,
.
eng_timeout
=
ahci_eng_timeout
,
.
irq_handler
=
ahci_interrupt
,
.
irq_clear
=
ahci_irq_clear
,
.
scr_read
=
ahci_scr_read
,
.
scr_write
=
ahci_scr_write
,
.
port_start
=
ahci_port_start
,
.
port_stop
=
ahci_port_stop
,
.
host_stop
=
ahci_host_stop
,
};
static
struct
ata_port_info
ahci_port_info
[]
=
{
/* board_ahci */
{
.
sht
=
&
ahci_sht
,
.
host_flags
=
ATA_FLAG_SATA
|
ATA_FLAG_NO_LEGACY
|
ATA_FLAG_SATA_RESET
|
ATA_FLAG_MMIO
,
.
pio_mask
=
0x03
,
/* pio3-4 */
.
udma_mask
=
0x7f
,
/* udma0-6 ; FIXME */
.
port_ops
=
&
ahci_ops
,
},
};
static
struct
pci_device_id
ahci_pci_tbl
[]
=
{
{
PCI_VENDOR_ID_INTEL
,
0x0000
,
PCI_ANY_ID
,
PCI_ANY_ID
,
0
,
0
,
board_ahci
},
{
}
/* terminate list */
};
static
struct
pci_driver
ahci_pci_driver
=
{
.
name
=
DRV_NAME
,
.
id_table
=
ahci_pci_tbl
,
.
probe
=
ahci_init_one
,
.
remove
=
ata_pci_remove_one
,
};
static
inline
unsigned
long
ahci_port_base_ul
(
unsigned
long
base
,
unsigned
int
port
)
{
return
base
+
0x100
+
(
port
*
0x80
);
}
static
inline
void
*
ahci_port_base
(
void
*
base
,
unsigned
int
port
)
{
return
(
void
*
)
ahci_port_base_ul
((
unsigned
long
)
base
,
port
);
}
static
void
ahci_host_stop
(
struct
ata_host_set
*
host_set
)
{
struct
ahci_host_priv
*
hpriv
=
host_set
->
private_data
;
kfree
(
hpriv
);
}
static
int
ahci_port_start
(
struct
ata_port
*
ap
)
{
struct
pci_dev
*
pdev
=
ap
->
host_set
->
pdev
;
struct
ahci_host_priv
*
hpriv
=
ap
->
host_set
->
private_data
;
struct
ahci_port_priv
*
pp
;
int
rc
;
void
*
mem
,
*
mem2
,
*
mmio
=
ap
->
host_set
->
mmio_base
;
void
*
port_mmio
=
ahci_port_base
(
mmio
,
ap
->
port_no
);
dma_addr_t
mem_dma
;
rc
=
ata_port_start
(
ap
);
if
(
rc
)
return
rc
;
pp
=
kmalloc
(
sizeof
(
*
pp
),
GFP_KERNEL
);
if
(
!
pp
)
{
rc
=
-
ENOMEM
;
goto
err_out
;
}
memset
(
pp
,
0
,
sizeof
(
*
pp
));
mem
=
pci_alloc_consistent
(
pdev
,
AHCI_PORT_PRIV_DMA_SZ
,
&
mem_dma
);
if
(
!
mem
)
{
rc
=
-
ENOMEM
;
goto
err_out_kfree
;
}
pp
->
rx_fis
=
mem
;
pp
->
rx_fis_dma
=
mem_dma
;
mem
+=
AHCI_RX_FIS_SZ
;
mem_dma
+=
AHCI_RX_FIS_SZ
;
pp
->
cmd_tbl
=
mem
;
pp
->
cmd_tbl_dma
=
mem_dma
;
mem2
=
mem
+
128
;
pp
->
cmd_tbl_sg
=
mem2
;
mem
+=
AHCI_CMD_TBL_SZ
;
mem_dma
+=
AHCI_CMD_TBL_SZ
;
pp
->
cmd_slot
=
mem
;
pp
->
cmd_slot_dma
=
mem_dma
;
ap
->
private_data
=
pp
;
if
(
hpriv
->
cap
&
HOST_CAP_64
)
writel
((
pp
->
cmd_slot_dma
>>
16
)
>>
16
,
port_mmio
+
PORT_LST_ADDR_HI
);
writel
(
pp
->
cmd_slot_dma
&
0xffffffff
,
port_mmio
+
PORT_LST_ADDR
);
readl
(
port_mmio
+
PORT_LST_ADDR
);
/* flush */
if
(
hpriv
->
cap
&
HOST_CAP_64
)
writel
((
pp
->
rx_fis_dma
>>
16
)
>>
16
,
port_mmio
+
PORT_FIS_ADDR_HI
);
writel
(
pp
->
rx_fis_dma
&
0xffffffff
,
port_mmio
+
PORT_LST_ADDR
);
readl
(
port_mmio
+
PORT_LST_ADDR
);
/* flush */
writel
(
PORT_CMD_ICC_ACTIVE
|
PORT_CMD_FIS_RX
|
PORT_CMD_POWER_ON
|
PORT_CMD_SPIN_UP
|
PORT_CMD_START
,
port_mmio
+
PORT_CMD
);
readl
(
port_mmio
+
PORT_CMD
);
/* flush */
return
0
;
err_out_kfree:
kfree
(
pp
);
err_out:
ata_port_stop
(
ap
);
return
rc
;
}
static
void
ahci_port_stop
(
struct
ata_port
*
ap
)
{
struct
pci_dev
*
pdev
=
ap
->
host_set
->
pdev
;
struct
ahci_port_priv
*
pp
=
ap
->
private_data
;
void
*
mmio
=
ap
->
host_set
->
mmio_base
;
void
*
port_mmio
=
ahci_port_base
(
mmio
,
ap
->
port_no
);
u32
tmp
;
tmp
=
readl
(
port_mmio
+
PORT_CMD
);
tmp
&=
~
(
PORT_CMD_START
|
PORT_CMD_FIS_RX
);
writel
(
tmp
,
port_mmio
+
PORT_CMD
);
readl
(
port_mmio
+
PORT_CMD
);
/* flush */
/* spec says 500 msecs for each PORT_CMD_{START,FIS_RX} bit, so
* this is slightly incorrect.
*/
set_current_state
(
TASK_UNINTERRUPTIBLE
);
schedule_timeout
((
HZ
/
2
)
+
1
);
ap
->
private_data
=
NULL
;
pci_free_consistent
(
pdev
,
AHCI_PORT_PRIV_DMA_SZ
,
pp
->
rx_fis
,
pp
->
rx_fis_dma
);
kfree
(
pp
);
ata_port_stop
(
ap
);
}
static
u32
ahci_scr_read
(
struct
ata_port
*
ap
,
unsigned
int
sc_reg_in
)
{
unsigned
int
sc_reg
;
switch
(
sc_reg_in
)
{
case
SCR_STATUS
:
sc_reg
=
0
;
break
;
case
SCR_CONTROL
:
sc_reg
=
1
;
break
;
case
SCR_ERROR
:
sc_reg
=
2
;
break
;
case
SCR_ACTIVE
:
sc_reg
=
3
;
break
;
default:
return
0xffffffffU
;
}
return
readl
((
void
*
)
ap
->
ioaddr
.
scr_addr
+
(
sc_reg
*
4
));
}
static
void
ahci_scr_write
(
struct
ata_port
*
ap
,
unsigned
int
sc_reg_in
,
u32
val
)
{
unsigned
int
sc_reg
;
switch
(
sc_reg_in
)
{
case
SCR_STATUS
:
sc_reg
=
0
;
break
;
case
SCR_CONTROL
:
sc_reg
=
1
;
break
;
case
SCR_ERROR
:
sc_reg
=
2
;
break
;
case
SCR_ACTIVE
:
sc_reg
=
3
;
break
;
default:
return
;
}
writel
(
val
,
(
void
*
)
ap
->
ioaddr
.
scr_addr
+
(
sc_reg
*
4
));
}
static
u8
ahci_check_status
(
struct
ata_port
*
ap
)
{
void
*
mmio
=
(
void
*
)
ap
->
ioaddr
.
cmd_addr
;
return
readl
(
mmio
+
PORT_TFDATA
)
&
0xFF
;
}
static
void
ahci_fill_sg
(
struct
ata_queued_cmd
*
qc
)
{
struct
ahci_port_priv
*
pp
=
qc
->
ap
->
private_data
;
unsigned
int
i
;
VPRINTK
(
"ENTER
\n
"
);
/*
* Next, the S/G list.
*/
for
(
i
=
0
;
i
<
qc
->
n_elem
;
i
++
)
{
u32
sg_len
;
dma_addr_t
addr
;
addr
=
sg_dma_address
(
&
qc
->
sg
[
i
]);
sg_len
=
sg_dma_len
(
&
qc
->
sg
[
i
]);
pp
->
cmd_tbl_sg
[
i
].
addr
=
cpu_to_le32
(
addr
&
0xffffffff
);
pp
->
cmd_tbl_sg
[
i
].
addr_hi
=
cpu_to_le32
((
addr
>>
16
)
>>
16
);
pp
->
cmd_tbl_sg
[
i
].
flags_size
=
cpu_to_le32
(
sg_len
);
}
}
static
void
ahci_qc_prep
(
struct
ata_queued_cmd
*
qc
)
{
struct
ahci_port_priv
*
pp
=
qc
->
ap
->
private_data
;
u32
opts
;
const
u32
cmd_fis_len
=
5
;
/* five dwords */
/*
* Fill in command slot information (currently only one slot,
* slot 0, is currently since we don't do queueing)
*/
opts
=
(
qc
->
n_elem
<<
16
)
|
cmd_fis_len
;
if
(
qc
->
tf
.
flags
&
ATA_TFLAG_WRITE
)
opts
|=
AHCI_CMD_WRITE
;
pp
->
cmd_slot
->
opts
=
cpu_to_le32
(
opts
);
pp
->
cmd_slot
->
status
=
0
;
pp
->
cmd_slot
->
tbl_addr
=
cpu_to_le32
(
pp
->
cmd_tbl_dma
&
0xffffffff
);
pp
->
cmd_slot
->
tbl_addr_hi
=
cpu_to_le32
((
pp
->
cmd_tbl_dma
>>
16
)
>>
16
);
/*
* Fill in command table information. First, the header,
* a SATA Register - Host to Device command FIS.
*/
ata_tf_to_fis
(
&
qc
->
tf
,
pp
->
cmd_tbl
,
0
);
if
(
!
(
qc
->
flags
&
ATA_QCFLAG_DMAMAP
))
return
;
ahci_fill_sg
(
qc
);
}
static
inline
void
ahci_dma_complete
(
struct
ata_port
*
ap
,
struct
ata_queued_cmd
*
qc
,
int
have_err
)
{
/* get drive status; clear intr; complete txn */
ata_qc_complete
(
ata_qc_from_tag
(
ap
,
ap
->
active_tag
),
have_err
?
ATA_ERR
:
0
);
}
static
void
ahci_intr_error
(
struct
ata_port
*
ap
,
u32
irq_stat
)
{
void
*
mmio
=
ap
->
host_set
->
mmio_base
;
void
*
port_mmio
=
ahci_port_base
(
mmio
,
ap
->
port_no
);
u32
tmp
;
int
work
;
/* stop DMA */
tmp
=
readl
(
port_mmio
+
PORT_CMD
);
tmp
&=
PORT_CMD_START
|
PORT_CMD_FIS_RX
;
writel
(
tmp
,
port_mmio
+
PORT_CMD
);
/* wait for engine to stop. TODO: this could be
* as long as 500 msec
*/
work
=
1000
;
while
(
work
--
>
0
)
{
tmp
=
readl
(
port_mmio
+
PORT_CMD
);
if
((
tmp
&
PORT_CMD_LIST_ON
)
==
0
)
break
;
udelay
(
10
);
}
/* clear SATA phy error, if any */
tmp
=
readl
(
port_mmio
+
PORT_SCR_ERR
);
writel
(
tmp
,
port_mmio
+
PORT_SCR_ERR
);
/* if DRQ/BSY is set, device needs to be reset.
* if so, issue COMRESET
*/
tmp
=
readl
(
port_mmio
+
PORT_TFDATA
);
if
(
tmp
&
(
ATA_BUSY
|
ATA_DRQ
))
{
writel
(
0x301
,
port_mmio
+
PORT_SCR_CTL
);
readl
(
port_mmio
+
PORT_SCR_CTL
);
/* flush */
udelay
(
10
);
writel
(
0x300
,
port_mmio
+
PORT_SCR_CTL
);
readl
(
port_mmio
+
PORT_SCR_CTL
);
/* flush */
}
/* re-start DMA */
tmp
=
readl
(
port_mmio
+
PORT_CMD
);
tmp
|=
PORT_CMD_START
|
PORT_CMD_FIS_RX
;
writel
(
tmp
,
port_mmio
+
PORT_CMD
);
readl
(
port_mmio
+
PORT_CMD
);
/* flush */
printk
(
KERN_WARNING
"ata%u: error occurred, port reset
\n
"
,
ap
->
port_no
);
}
static
void
ahci_eng_timeout
(
struct
ata_port
*
ap
)
{
void
*
mmio
=
ap
->
host_set
->
mmio_base
;
void
*
port_mmio
=
ahci_port_base
(
mmio
,
ap
->
port_no
);
struct
ata_queued_cmd
*
qc
;
DPRINTK
(
"ENTER
\n
"
);
ahci_intr_error
(
ap
,
readl
(
port_mmio
+
PORT_IRQ_STAT
));
qc
=
ata_qc_from_tag
(
ap
,
ap
->
active_tag
);
if
(
!
qc
)
{
printk
(
KERN_ERR
"ata%u: BUG: timeout without command
\n
"
,
ap
->
id
);
}
else
{
/* hack alert! We cannot use the supplied completion
* function from inside the ->eh_strategy_handler() thread.
* libata is the only user of ->eh_strategy_handler() in
* any kernel, so the default scsi_done() assumes it is
* not being called from the SCSI EH.
*/
qc
->
scsidone
=
scsi_finish_command
;
ata_qc_complete
(
qc
,
ATA_ERR
);
}
}
static
inline
int
ahci_host_intr
(
struct
ata_port
*
ap
,
struct
ata_queued_cmd
*
qc
)
{
void
*
mmio
=
ap
->
host_set
->
mmio_base
;
void
*
port_mmio
=
ahci_port_base
(
mmio
,
ap
->
port_no
);
u32
status
,
ci
;
status
=
readl
(
port_mmio
+
PORT_IRQ_STAT
);
if
(
!
status
)
return
0
;
writel
(
status
,
port_mmio
+
PORT_IRQ_STAT
);
ci
=
readl
(
port_mmio
+
PORT_CMD_ISSUE
);
if
(
likely
((
ci
&
0x1
)
==
0
))
{
if
(
qc
)
{
ata_qc_complete
(
qc
,
0
);
qc
=
NULL
;
}
}
if
(
status
&
PORT_IRQ_FATAL
)
{
ahci_intr_error
(
ap
,
status
);
if
(
qc
)
ata_qc_complete
(
qc
,
ATA_ERR
);
}
return
1
;
}
static
void
ahci_irq_clear
(
struct
ata_port
*
ap
)
{
/* TODO */
}
static
irqreturn_t
ahci_interrupt
(
int
irq
,
void
*
dev_instance
,
struct
pt_regs
*
regs
)
{
struct
ata_host_set
*
host_set
=
dev_instance
;
struct
ahci_host_priv
*
hpriv
;
unsigned
int
i
,
handled
=
0
;
void
*
mmio
;
u32
irq_stat
,
irq_ack
=
0
;
VPRINTK
(
"ENTER
\n
"
);
hpriv
=
host_set
->
private_data
;
mmio
=
host_set
->
mmio_base
;
/* sigh. 0xffffffff is a valid return from h/w */
irq_stat
=
readl
(
mmio
+
HOST_IRQ_STAT
);
irq_stat
&=
hpriv
->
port_map
;
if
(
!
irq_stat
)
return
IRQ_NONE
;
spin_lock
(
&
host_set
->
lock
);
for
(
i
=
0
;
i
<
host_set
->
n_ports
;
i
++
)
{
struct
ata_port
*
ap
;
u32
tmp
;
VPRINTK
(
"port %u
\n
"
,
i
);
ap
=
host_set
->
ports
[
i
];
tmp
=
irq_stat
&
(
1
<<
i
);
if
(
tmp
&&
ap
)
{
struct
ata_queued_cmd
*
qc
;
qc
=
ata_qc_from_tag
(
ap
,
ap
->
active_tag
);
if
(
ahci_host_intr
(
ap
,
qc
))
irq_ack
|=
(
1
<<
i
);
}
}
if
(
irq_ack
)
{
writel
(
irq_ack
,
mmio
+
HOST_IRQ_STAT
);
handled
=
1
;
}
spin_unlock
(
&
host_set
->
lock
);
VPRINTK
(
"EXIT
\n
"
);
return
IRQ_RETVAL
(
handled
);
}
static
void
ahci_dma_setup
(
struct
ata_queued_cmd
*
qc
)
{
/* do nothing... for now */
}
static
void
ahci_dma_start
(
struct
ata_queued_cmd
*
qc
)
{
struct
ata_port
*
ap
=
qc
->
ap
;
void
*
mmio
=
(
void
*
)
ap
->
ioaddr
.
cmd_addr
;
writel
(
1
,
mmio
+
PORT_CMD_ISSUE
);
readl
(
mmio
+
PORT_CMD_ISSUE
);
/* flush */
}
static
void
ahci_tf_read
(
struct
ata_port
*
ap
,
struct
ata_taskfile
*
tf
)
{
struct
ahci_port_priv
*
pp
=
ap
->
private_data
;
ata_tf_from_fis
(
pp
->
rx_fis
+
RX_FIS_D2H_REG
,
tf
);
}
static
void
ahci_tf_load
(
struct
ata_port
*
ap
,
struct
ata_taskfile
*
tf
)
{
/* do everything in exec command */
}
static
void
ahci_exec_command
(
struct
ata_port
*
ap
,
struct
ata_taskfile
*
tf
)
{
struct
ahci_port_priv
*
pp
=
ap
->
private_data
;
void
*
mmio
=
(
void
*
)
ap
->
ioaddr
.
cmd_addr
;
ata_tf_to_fis
(
tf
,
pp
->
cmd_tbl
,
0
);
writel
(
1
,
mmio
+
PORT_CMD_ISSUE
);
readl
(
mmio
+
PORT_CMD_ISSUE
);
/* flush */
}
static
void
ahci_setup_port
(
struct
ata_ioports
*
port
,
unsigned
long
base
,
unsigned
int
port_idx
)
{
base
=
ahci_port_base_ul
(
base
,
port_idx
);
port
->
cmd_addr
=
base
;
port
->
scr_addr
=
base
+
PORT_SCR
;
}
static
int
ahci_host_init
(
struct
ata_probe_ent
*
probe_ent
)
{
struct
ahci_host_priv
*
hpriv
=
probe_ent
->
private_data
;
struct
pci_dev
*
pdev
=
probe_ent
->
pdev
;
void
*
mmio
=
probe_ent
->
mmio_base
;
u32
tmp
;
unsigned
int
i
,
using_dac
;
int
rc
;
/* global controller reset */
tmp
=
readl
(
mmio
+
HOST_CTL
);
if
((
tmp
&
HOST_RESET
)
==
0
)
{
writel
(
tmp
|
HOST_RESET
,
mmio
+
HOST_CTL
);
readl
(
mmio
+
HOST_CTL
);
/* flush */
}
/* reset must complete within 1 second, or
* the hardware should be considered fried.
*/
set_current_state
(
TASK_UNINTERRUPTIBLE
);
schedule_timeout
(
HZ
+
1
);
tmp
=
readl
(
mmio
+
HOST_CTL
);
if
(
tmp
&
HOST_RESET
)
{
printk
(
KERN_ERR
DRV_NAME
"(%s): controller reset failed (0x%x)
\n
"
,
pci_name
(
pdev
),
tmp
);
return
-
EIO
;
}
/* make sure we're in AHCI mode, with irq disabled */
if
((
tmp
&
(
HOST_AHCI_EN
|
HOST_IRQ_EN
))
!=
HOST_AHCI_EN
)
{
tmp
|=
HOST_AHCI_EN
;
tmp
&=
~
HOST_IRQ_EN
;
writel
(
tmp
,
mmio
+
HOST_CTL
);
readl
(
mmio
+
HOST_CTL
);
/* flush */
}
tmp
=
readl
(
mmio
+
HOST_CTL
);
if
((
tmp
&
(
HOST_AHCI_EN
|
HOST_IRQ_EN
))
!=
HOST_AHCI_EN
)
{
printk
(
KERN_ERR
DRV_NAME
"(%s): AHCI enable failed (0x%x)
\n
"
,
pci_name
(
pdev
),
tmp
);
return
-
EIO
;
}
hpriv
->
cap
=
readl
(
mmio
+
HOST_CAP
);
hpriv
->
port_map
=
readl
(
mmio
+
HOST_PORTS_IMPL
);
probe_ent
->
n_ports
=
hpriv
->
cap
&
0x1f
;
using_dac
=
hpriv
->
cap
&
HOST_CAP_64
;
if
(
using_dac
&&
!
pci_set_dma_mask
(
pdev
,
0xffffffffffffffffULL
))
{
rc
=
pci_set_consistent_dma_mask
(
pdev
,
0xffffffffffffffffULL
);
if
(
rc
)
{
rc
=
pci_set_consistent_dma_mask
(
pdev
,
0xffffffffULL
);
if
(
rc
)
{
printk
(
KERN_ERR
DRV_NAME
"(%s): 64-bit DMA enable failed
\n
"
,
pci_name
(
pdev
));
return
rc
;
}
}
hpriv
->
flags
|=
HOST_CAP_64
;
}
else
{
rc
=
pci_set_dma_mask
(
pdev
,
0xffffffffULL
);
if
(
rc
)
{
printk
(
KERN_ERR
DRV_NAME
"(%s): 32-bit DMA enable failed
\n
"
,
pci_name
(
pdev
));
return
rc
;
}
rc
=
pci_set_consistent_dma_mask
(
pdev
,
0xffffffffULL
);
if
(
rc
)
{
printk
(
KERN_ERR
DRV_NAME
"(%s): 32-bit consistent DMA enable failed
\n
"
,
pci_name
(
pdev
));
return
rc
;
}
}
for
(
i
=
0
;
i
<
probe_ent
->
n_ports
;
i
++
)
{
void
*
port_mmio
;
if
(
!
(
hpriv
->
port_map
&
(
1
<<
i
)))
continue
;
port_mmio
=
ahci_port_base
(
mmio
,
i
);
ahci_setup_port
(
&
probe_ent
->
port
[
i
],
(
unsigned
long
)
mmio
,
i
);
/* make sure port is not active */
tmp
=
readl
(
port_mmio
+
PORT_CMD
);
if
(
tmp
&
(
PORT_CMD_LIST_ON
|
PORT_CMD_FIS_ON
|
PORT_CMD_FIS_RX
|
PORT_CMD_START
))
{
tmp
&=
~
(
PORT_CMD_LIST_ON
|
PORT_CMD_FIS_ON
|
PORT_CMD_FIS_RX
|
PORT_CMD_START
);
writel
(
tmp
,
port_mmio
+
PORT_CMD
);
readl
(
port_mmio
+
PORT_CMD
);
/* flush */
/* spec says 500 msecs for each bit, so
* this is slightly incorrect.
*/
set_current_state
(
TASK_UNINTERRUPTIBLE
);
schedule_timeout
((
HZ
/
2
)
+
1
);
}
/* ack any pending irq events for this port */
tmp
=
readl
(
port_mmio
+
PORT_IRQ_STAT
);
if
(
tmp
)
writel
(
tmp
,
port_mmio
+
PORT_IRQ_STAT
);
tmp
=
readl
(
port_mmio
+
PORT_SCR_ERR
);
writel
(
tmp
,
port_mmio
+
PORT_SCR_ERR
);
writel
(
1
<<
i
,
mmio
+
HOST_IRQ_STAT
);
/* set irq mask (enables interrupts) */
writel
(
DEF_PORT_IRQ
,
port_mmio
+
PORT_IRQ_MASK
);
}
tmp
=
readl
(
mmio
+
HOST_CTL
);
writel
(
tmp
|
HOST_IRQ_EN
,
mmio
+
HOST_CTL
);
pci_set_master
(
pdev
);
return
0
;
}
/* move to PCI layer, integrate w/ MSI stuff */
static
void
pci_enable_intx
(
struct
pci_dev
*
pdev
)
{
u16
pci_command
;
pci_read_config_word
(
pdev
,
PCI_COMMAND
,
&
pci_command
);
if
(
pci_command
&
PCI_COMMAND_INTX_DISABLE
)
{
pci_command
&=
~
PCI_COMMAND_INTX_DISABLE
;
pci_write_config_word
(
pdev
,
PCI_COMMAND
,
pci_command
);
}
}
static
void
ahci_print_info
(
struct
ata_probe_ent
*
probe_ent
)
{
struct
ahci_host_priv
*
hpriv
=
probe_ent
->
private_data
;
struct
pci_dev
*
pdev
=
probe_ent
->
pdev
;
void
*
mmio
=
probe_ent
->
mmio_base
;
u32
vers
,
cap
,
impl
,
speed
;
const
char
*
speed_s
;
vers
=
readl
(
mmio
+
HOST_VERSION
);
cap
=
hpriv
->
cap
;
impl
=
hpriv
->
port_map
;
speed
=
(
cap
>>
20
)
&
0xf
;
if
(
speed
==
1
)
speed_s
=
"1.5"
;
else
if
(
speed
==
2
)
speed_s
=
"3"
;
else
speed_s
=
"?"
;
printk
(
KERN_INFO
DRV_NAME
"(%s) AHCI %02x%02x.%02x%02x "
"%u slots %u ports %s Gbps 0x%x impl
\n
"
,
pci_name
(
pdev
),
(
vers
>>
24
)
&
0xff
,
(
vers
>>
16
)
&
0xff
,
(
vers
>>
8
)
&
0xff
,
vers
&
0xff
,
((
cap
>>
8
)
&
0x1f
)
+
1
,
(
cap
&
0x1f
)
+
1
,
speed_s
,
impl
);
printk
(
KERN_INFO
DRV_NAME
"(%s) flags: "
"%s%s%s%s%s%s"
"%s%s%s%s%s%s%s
\n
"
,
pci_name
(
pdev
),
cap
&
(
1
<<
31
)
?
"64bit "
:
""
,
cap
&
(
1
<<
30
)
?
"ncq "
:
""
,
cap
&
(
1
<<
28
)
?
"ilck "
:
""
,
cap
&
(
1
<<
27
)
?
"stag "
:
""
,
cap
&
(
1
<<
26
)
?
"pm "
:
""
,
cap
&
(
1
<<
25
)
?
"led "
:
""
,
cap
&
(
1
<<
24
)
?
"clo "
:
""
,
cap
&
(
1
<<
19
)
?
"nz "
:
""
,
cap
&
(
1
<<
18
)
?
"only "
:
""
,
cap
&
(
1
<<
17
)
?
"pmp "
:
""
,
cap
&
(
1
<<
15
)
?
"pio "
:
""
,
cap
&
(
1
<<
14
)
?
"slum "
:
""
,
cap
&
(
1
<<
13
)
?
"part "
:
""
);
}
static
int
ahci_init_one
(
struct
pci_dev
*
pdev
,
const
struct
pci_device_id
*
ent
)
{
static
int
printed_version
;
struct
ata_probe_ent
*
probe_ent
=
NULL
;
struct
ahci_host_priv
*
hpriv
;
unsigned
long
base
;
void
*
mmio_base
;
unsigned
int
board_idx
=
(
unsigned
int
)
ent
->
driver_data
;
int
rc
;
if
(
!
printed_version
++
)
printk
(
KERN_DEBUG
DRV_NAME
" version "
DRV_VERSION
"
\n
"
);
/*
* If this driver happens to only be useful on Apple's K2, then
* we should check that here as it has a normal Serverworks ID
*/
rc
=
pci_enable_device
(
pdev
);
if
(
rc
)
return
rc
;
rc
=
pci_request_regions
(
pdev
,
DRV_NAME
);
if
(
rc
)
goto
err_out
;
pci_enable_intx
(
pdev
);
probe_ent
=
kmalloc
(
sizeof
(
*
probe_ent
),
GFP_KERNEL
);
if
(
probe_ent
==
NULL
)
{
rc
=
-
ENOMEM
;
goto
err_out_regions
;
}
memset
(
probe_ent
,
0
,
sizeof
(
*
probe_ent
));
probe_ent
->
pdev
=
pdev
;
INIT_LIST_HEAD
(
&
probe_ent
->
node
);
mmio_base
=
ioremap
(
pci_resource_start
(
pdev
,
AHCI_PCI_BAR
),
pci_resource_len
(
pdev
,
AHCI_PCI_BAR
));
if
(
mmio_base
==
NULL
)
{
rc
=
-
ENOMEM
;
goto
err_out_free_ent
;
}
base
=
(
unsigned
long
)
mmio_base
;
hpriv
=
kmalloc
(
sizeof
(
*
hpriv
),
GFP_KERNEL
);
if
(
!
hpriv
)
{
rc
=
-
ENOMEM
;
goto
err_out_iounmap
;
}
memset
(
hpriv
,
0
,
sizeof
(
*
hpriv
));
probe_ent
->
sht
=
ahci_port_info
[
board_idx
].
sht
;
probe_ent
->
host_flags
=
ahci_port_info
[
board_idx
].
host_flags
;
probe_ent
->
pio_mask
=
ahci_port_info
[
board_idx
].
pio_mask
;
probe_ent
->
udma_mask
=
ahci_port_info
[
board_idx
].
udma_mask
;
probe_ent
->
port_ops
=
ahci_port_info
[
board_idx
].
port_ops
;
probe_ent
->
irq
=
pdev
->
irq
;
probe_ent
->
irq_flags
=
SA_SHIRQ
;
probe_ent
->
mmio_base
=
mmio_base
;
probe_ent
->
private_data
=
hpriv
;
/* initialize adapter */
rc
=
ahci_host_init
(
probe_ent
);
if
(
rc
)
goto
err_out_hpriv
;
ahci_print_info
(
probe_ent
);
/* FIXME: check ata_device_add return value */
ata_device_add
(
probe_ent
);
kfree
(
probe_ent
);
return
0
;
err_out_hpriv:
kfree
(
hpriv
);
err_out_iounmap:
iounmap
(
mmio_base
);
err_out_free_ent:
kfree
(
probe_ent
);
err_out_regions:
pci_release_regions
(
pdev
);
err_out:
pci_disable_device
(
pdev
);
return
rc
;
}
static
int
__init
ahci_init
(
void
)
{
return
pci_module_init
(
&
ahci_pci_driver
);
}
static
void
__exit
ahci_exit
(
void
)
{
pci_unregister_driver
(
&
ahci_pci_driver
);
}
MODULE_AUTHOR
(
"Jeff Garzik"
);
MODULE_DESCRIPTION
(
"AHCI SATA low-level driver"
);
MODULE_LICENSE
(
"GPL"
);
MODULE_DEVICE_TABLE
(
pci
,
ahci_pci_tbl
);
module_init
(
ahci_init
);
module_exit
(
ahci_exit
);
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