diff options
-rw-r--r-- | drivers/ata/libata-core.c | 1 | ||||
-rw-r--r-- | drivers/ata/libata-pmp.c | 30 | ||||
-rw-r--r-- | include/linux/libata.h | 1 |
3 files changed, 32 insertions, 0 deletions
diff --git a/drivers/ata/libata-core.c b/drivers/ata/libata-core.c index d3e78d97529d..5532a6564d05 100644 --- a/drivers/ata/libata-core.c +++ b/drivers/ata/libata-core.c | |||
@@ -7303,6 +7303,7 @@ EXPORT_SYMBOL_GPL(ata_pci_default_filter); | |||
7303 | EXPORT_SYMBOL_GPL(ata_pci_clear_simplex); | 7303 | EXPORT_SYMBOL_GPL(ata_pci_clear_simplex); |
7304 | #endif /* CONFIG_PCI */ | 7304 | #endif /* CONFIG_PCI */ |
7305 | 7305 | ||
7306 | EXPORT_SYMBOL_GPL(sata_pmp_qc_defer_cmd_switch); | ||
7306 | EXPORT_SYMBOL_GPL(sata_pmp_read_init_tf); | 7307 | EXPORT_SYMBOL_GPL(sata_pmp_read_init_tf); |
7307 | EXPORT_SYMBOL_GPL(sata_pmp_read_val); | 7308 | EXPORT_SYMBOL_GPL(sata_pmp_read_val); |
7308 | EXPORT_SYMBOL_GPL(sata_pmp_write_init_tf); | 7309 | EXPORT_SYMBOL_GPL(sata_pmp_write_init_tf); |
diff --git a/drivers/ata/libata-pmp.c b/drivers/ata/libata-pmp.c index eeffce636d0c..f6c4b11336e9 100644 --- a/drivers/ata/libata-pmp.c +++ b/drivers/ata/libata-pmp.c | |||
@@ -70,6 +70,36 @@ static int sata_pmp_write(struct ata_link *link, int reg, u32 val) | |||
70 | } | 70 | } |
71 | 71 | ||
72 | /** | 72 | /** |
73 | * sata_pmp_qc_defer_cmd_switch - qc_defer for command switching PMP | ||
74 | * @qc: ATA command in question | ||
75 | * | ||
76 | * A host which has command switching PMP support cannot issue | ||
77 | * commands to multiple links simultaneously. | ||
78 | * | ||
79 | * LOCKING: | ||
80 | * spin_lock_irqsave(host lock) | ||
81 | * | ||
82 | * RETURNS: | ||
83 | * ATA_DEFER_* if deferring is needed, 0 otherwise. | ||
84 | */ | ||
85 | int sata_pmp_qc_defer_cmd_switch(struct ata_queued_cmd *qc) | ||
86 | { | ||
87 | struct ata_link *link = qc->dev->link; | ||
88 | struct ata_port *ap = link->ap; | ||
89 | |||
90 | if (ap->excl_link == NULL || ap->excl_link == link) { | ||
91 | if (ap->nr_active_links == 0 || ata_link_active(link)) { | ||
92 | qc->flags |= ATA_QCFLAG_CLEAR_EXCL; | ||
93 | return ata_std_qc_defer(qc); | ||
94 | } | ||
95 | |||
96 | ap->excl_link = link; | ||
97 | } | ||
98 | |||
99 | return ATA_DEFER_PORT; | ||
100 | } | ||
101 | |||
102 | /** | ||
73 | * sata_pmp_read_init_tf - initialize TF for PMP read | 103 | * sata_pmp_read_init_tf - initialize TF for PMP read |
74 | * @tf: taskfile to initialize | 104 | * @tf: taskfile to initialize |
75 | * @dev: PMP dev | 105 | * @dev: PMP dev |
diff --git a/include/linux/libata.h b/include/linux/libata.h index 3cfdb5f67621..ca296a575c4a 100644 --- a/include/linux/libata.h +++ b/include/linux/libata.h | |||
@@ -947,6 +947,7 @@ extern unsigned long ata_pci_default_filter(struct ata_device *, unsigned long); | |||
947 | /* | 947 | /* |
948 | * PMP | 948 | * PMP |
949 | */ | 949 | */ |
950 | extern int sata_pmp_qc_defer_cmd_switch(struct ata_queued_cmd *qc); | ||
950 | extern void sata_pmp_read_init_tf(struct ata_taskfile *tf, | 951 | extern void sata_pmp_read_init_tf(struct ata_taskfile *tf, |
951 | struct ata_device *dev, int pmp, int reg); | 952 | struct ata_device *dev, int pmp, int reg); |
952 | extern u32 sata_pmp_read_val(const struct ata_taskfile *tf); | 953 | extern u32 sata_pmp_read_val(const struct ata_taskfile *tf); |