Commit 17600073 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'ata-6.12-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/libata/linux

Pull ata updates from Damien Le Moal:

 - Convert the qcom AHCI controller DT bindings to DT schema (from
   Rayyan)

 - Cleanup of libata core and drivers code handling controller and
   device quirks to rename "blacklist" to the more neutral "quirk" and
   to replace the rarely used "horkage" term with the more common
   "quirk" naming (me)

 - Add libata-core message to print the quirks applied to a controller
   or device (me)

 - Remove the not-so-useful function ata_noop_qc_prep() from libata core
   (me)

 - ahci_imx driver cleanup, improvements and DT bindings compatible
   strings update (Richard and Dan)

 - libahci_platform improvements (Zhang)

 - Remove obsolete functions declarations from libata header files (from
   Gaosheng)

 - Improve teh ahci_brcm driver using managed device resources funetions
   (Zhang)

 - Introduce new helper function to improve libata EH code readability
   (Niklas)

 - Enable module autoloading for the pata_ftide010, pata_ixp4xx and
   sata_gemini drivers (Liao)

 - Move SATA related functions and data declaraions from libata-core to
   libata-sata (me)

 - Rename the function handling the sense data for successful NCQ
   commands log to better reflect that function actions (me)

 - Reduce libata memory usage by moving port resources to struct
   ata_device and by optimizing the management of resources for CDL
   capable devices (me)

 - Improve libata-eh handling of failed ATA passthrough commands
   (Niklas)

* tag 'ata-6.12-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/libata/linux: (39 commits)
  ata: libata: Clear DID_TIME_OUT for ATA PT commands with sense data
  ata: libata: Fix W=1 compilation warning
  ata: libata: Improve CDL resource management
  ata: libata: Introduce ata_dev_free_resources
  ata: libata: Move sector_buf from struct ata_port to struct ata_device
  ata: libata: Rename ata_eh_read_sense_success_ncq_log()
  ata: libata: Move sata_std_hardreset() definition to libata-sata.c
  ata: libata: Move sata_down_spd_limit() to libata-sata.c
  ata: libata: Improve __ata_qc_complete()
  ata: libata-scsi: Improve ata_scsi_handle_link_detach()
  ata: libata: Cleanup libata-transport
  ata: sata_gemini: Enable module autoloading
  ata: pata_ixp4xx: Enable module autoloading
  ata: pata_ftide010: Enable module autoloading
  ata: libata: Add helper ata_eh_decide_disposition()
  ata: ahci_brcm: Use devm_platform_ioremap_resource_byname() helper function
  ata: libata: Remove obsolete function declarations
  ata: ahci_imx: Fix error code in probe()
  ata: libahci_platform: Simplify code with for_each_child_of_node_scoped()
  ata: ahci_imx: Correct the email address
  ...
parents a65b3c3e e5dd410a
...@@ -30,6 +30,8 @@ select: ...@@ -30,6 +30,8 @@ select:
- marvell,armada-3700-ahci - marvell,armada-3700-ahci
- marvell,armada-8k-ahci - marvell,armada-8k-ahci
- marvell,berlin2q-ahci - marvell,berlin2q-ahci
- qcom,apq8064-ahci
- qcom,ipq806x-ahci
- socionext,uniphier-pro4-ahci - socionext,uniphier-pro4-ahci
- socionext,uniphier-pxs2-ahci - socionext,uniphier-pxs2-ahci
- socionext,uniphier-pxs3-ahci - socionext,uniphier-pxs3-ahci
...@@ -45,6 +47,8 @@ properties: ...@@ -45,6 +47,8 @@ properties:
- marvell,armada-8k-ahci - marvell,armada-8k-ahci
- marvell,berlin2-ahci - marvell,berlin2-ahci
- marvell,berlin2q-ahci - marvell,berlin2q-ahci
- qcom,apq8064-ahci
- qcom,ipq806x-ahci
- socionext,uniphier-pro4-ahci - socionext,uniphier-pro4-ahci
- socionext,uniphier-pxs2-ahci - socionext,uniphier-pxs2-ahci
- socionext,uniphier-pxs3-ahci - socionext,uniphier-pxs3-ahci
...@@ -64,11 +68,11 @@ properties: ...@@ -64,11 +68,11 @@ properties:
clocks: clocks:
minItems: 1 minItems: 1
maxItems: 3 maxItems: 5
clock-names: clock-names:
minItems: 1 minItems: 1
maxItems: 3 maxItems: 5
interrupts: interrupts:
maxItems: 1 maxItems: 1
...@@ -97,6 +101,31 @@ required: ...@@ -97,6 +101,31 @@ required:
allOf: allOf:
- $ref: ahci-common.yaml# - $ref: ahci-common.yaml#
- if:
properties:
compatible:
contains:
enum:
- qcom,apq8064-ahci
- qcom,ipq806x-ahci
then:
properties:
clocks:
minItems: 5
clock-names:
items:
- const: slave_iface
- const: iface
- const: core
- const: rxoob
- const: pmalive
required:
- phys
- phy-names
- clocks
- clock-names
- if: - if:
properties: properties:
compatible: compatible:
......
...@@ -19,6 +19,7 @@ properties: ...@@ -19,6 +19,7 @@ properties:
- fsl,imx53-ahci - fsl,imx53-ahci
- fsl,imx6q-ahci - fsl,imx6q-ahci
- fsl,imx6qp-ahci - fsl,imx6qp-ahci
- fsl,imx8qm-ahci
reg: reg:
maxItems: 1 maxItems: 1
...@@ -27,12 +28,14 @@ properties: ...@@ -27,12 +28,14 @@ properties:
maxItems: 1 maxItems: 1
clocks: clocks:
minItems: 2
items: items:
- description: sata clock - description: sata clock
- description: sata reference clock - description: sata reference clock
- description: ahb clock - description: ahb clock
clock-names: clock-names:
minItems: 2
items: items:
- const: sata - const: sata
- const: sata_ref - const: sata_ref
...@@ -58,6 +61,25 @@ properties: ...@@ -58,6 +61,25 @@ properties:
$ref: /schemas/types.yaml#/definitions/flag $ref: /schemas/types.yaml#/definitions/flag
description: if present, disable spread-spectrum clocking on the SATA link. description: if present, disable spread-spectrum clocking on the SATA link.
phys:
items:
- description: phandle to SATA PHY.
Since "REXT" pin is only present for first lane of i.MX8QM PHY, it's
calibration result will be stored, passed through second lane, and
shared with all three lanes PHY. The first two lanes PHY are used as
calibration PHYs, although only the third lane PHY is used by SATA.
- description: phandle to the first lane PHY of i.MX8QM.
- description: phandle to the second lane PHY of i.MX8QM.
phy-names:
items:
- const: sata-phy
- const: cali-phy0
- const: cali-phy1
power-domains:
maxItems: 1
required: required:
- compatible - compatible
- reg - reg
...@@ -65,6 +87,31 @@ required: ...@@ -65,6 +87,31 @@ required:
- clocks - clocks
- clock-names - clock-names
allOf:
- if:
properties:
compatible:
contains:
enum:
- fsl,imx53-ahci
- fsl,imx6q-ahci
- fsl,imx6qp-ahci
then:
properties:
clock-names:
minItems: 3
- if:
properties:
compatible:
contains:
enum:
- fsl,imx8qm-ahci
then:
properties:
clock-names:
minItems: 2
additionalProperties: false additionalProperties: false
examples: examples:
......
* Qualcomm AHCI SATA Controller
SATA nodes are defined to describe on-chip Serial ATA controllers.
Each SATA controller should have its own node.
Required properties:
- compatible : compatible list, must contain "generic-ahci"
- interrupts : <interrupt mapping for SATA IRQ>
- reg : <registers mapping>
- phys : Must contain exactly one entry as specified
in phy-bindings.txt
- phy-names : Must be "sata-phy"
Required properties for "qcom,ipq806x-ahci" compatible:
- clocks : Must contain an entry for each entry in clock-names.
- clock-names : Shall be:
"slave_iface" - Fabric port AHB clock for SATA
"iface" - AHB clock
"core" - core clock
"rxoob" - RX out-of-band clock
"pmalive" - Power Module Alive clock
- assigned-clocks : Shall be:
SATA_RXOOB_CLK
SATA_PMALIVE_CLK
- assigned-clock-rates : Shall be:
100Mhz (100000000) for SATA_RXOOB_CLK
100Mhz (100000000) for SATA_PMALIVE_CLK
Example:
sata@29000000 {
compatible = "qcom,ipq806x-ahci", "generic-ahci";
reg = <0x29000000 0x180>;
interrupts = <0 209 0x0>;
clocks = <&gcc SFAB_SATA_S_H_CLK>,
<&gcc SATA_H_CLK>,
<&gcc SATA_A_CLK>,
<&gcc SATA_RXOOB_CLK>,
<&gcc SATA_PMALIVE_CLK>;
clock-names = "slave_iface", "iface", "core",
"rxoob", "pmalive";
assigned-clocks = <&gcc SATA_RXOOB_CLK>, <&gcc SATA_PMALIVE_CLK>;
assigned-clock-rates = <100000000>, <100000000>;
phys = <&sata_phy>;
phy-names = "sata-phy";
};
...@@ -1370,7 +1370,7 @@ static bool ahci_broken_suspend(struct pci_dev *pdev) ...@@ -1370,7 +1370,7 @@ static bool ahci_broken_suspend(struct pci_dev *pdev)
* V1.03 is known to be broken. V3.04 is known to * V1.03 is known to be broken. V3.04 is known to
* work. Between, there are V1.06, V2.06 and V3.03 * work. Between, there are V1.06, V2.06 and V3.03
* that we don't have much idea about. For now, * that we don't have much idea about. For now,
* blacklist anything older than V3.04. * assume that anything older than V3.04 is broken.
* *
* http://bugzilla.kernel.org/show_bug.cgi?id=15104 * http://bugzilla.kernel.org/show_bug.cgi?id=15104
*/ */
......
...@@ -437,7 +437,6 @@ static int brcm_ahci_probe(struct platform_device *pdev) ...@@ -437,7 +437,6 @@ static int brcm_ahci_probe(struct platform_device *pdev)
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
struct brcm_ahci_priv *priv; struct brcm_ahci_priv *priv;
struct ahci_host_priv *hpriv; struct ahci_host_priv *hpriv;
struct resource *res;
int ret; int ret;
priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL); priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
...@@ -451,8 +450,7 @@ static int brcm_ahci_probe(struct platform_device *pdev) ...@@ -451,8 +450,7 @@ static int brcm_ahci_probe(struct platform_device *pdev)
priv->version = (unsigned long)of_id->data; priv->version = (unsigned long)of_id->data;
priv->dev = dev; priv->dev = dev;
res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "top-ctrl"); priv->top_ctrl = devm_platform_ioremap_resource_byname(pdev, "top-ctrl");
priv->top_ctrl = devm_ioremap_resource(dev, res);
if (IS_ERR(priv->top_ctrl)) if (IS_ERR(priv->top_ctrl))
return PTR_ERR(priv->top_ctrl); return PTR_ERR(priv->top_ctrl);
......
This diff is collapsed.
...@@ -1446,7 +1446,6 @@ static int piix_init_sidpr(struct ata_host *host) ...@@ -1446,7 +1446,6 @@ static int piix_init_sidpr(struct ata_host *host)
if (hpriv->map[i] == IDE) if (hpriv->map[i] == IDE)
return 0; return 0;
/* is it blacklisted? */
if (piix_no_sidpr(host)) if (piix_no_sidpr(host))
return 0; return 0;
......
...@@ -410,7 +410,6 @@ static int ahci_platform_get_regulator(struct ahci_host_priv *hpriv, u32 port, ...@@ -410,7 +410,6 @@ static int ahci_platform_get_regulator(struct ahci_host_priv *hpriv, u32 port,
static int ahci_platform_get_firmware(struct ahci_host_priv *hpriv, static int ahci_platform_get_firmware(struct ahci_host_priv *hpriv,
struct device *dev) struct device *dev)
{ {
struct device_node *child;
u32 port; u32 port;
if (!of_property_read_u32(dev->of_node, "hba-cap", &hpriv->saved_cap)) if (!of_property_read_u32(dev->of_node, "hba-cap", &hpriv->saved_cap))
...@@ -419,14 +418,12 @@ static int ahci_platform_get_firmware(struct ahci_host_priv *hpriv, ...@@ -419,14 +418,12 @@ static int ahci_platform_get_firmware(struct ahci_host_priv *hpriv,
of_property_read_u32(dev->of_node, of_property_read_u32(dev->of_node,
"ports-implemented", &hpriv->saved_port_map); "ports-implemented", &hpriv->saved_port_map);
for_each_child_of_node(dev->of_node, child) { for_each_child_of_node_scoped(dev->of_node, child) {
if (!of_device_is_available(child)) if (!of_device_is_available(child))
continue; continue;
if (of_property_read_u32(child, "reg", &port)) { if (of_property_read_u32(child, "reg", &port))
of_node_put(child);
return -EINVAL; return -EINVAL;
}
if (!of_property_read_u32(child, "hba-port-cap", &hpriv->saved_port_cap[port])) if (!of_property_read_u32(child, "hba-port-cap", &hpriv->saved_port_cap[port]))
hpriv->saved_port_cap[port] &= PORT_CMD_CAP; hpriv->saved_port_cap[port] &= PORT_CMD_CAP;
...@@ -460,7 +457,6 @@ struct ahci_host_priv *ahci_platform_get_resources(struct platform_device *pdev, ...@@ -460,7 +457,6 @@ struct ahci_host_priv *ahci_platform_get_resources(struct platform_device *pdev,
int child_nodes, rc = -ENOMEM, enabled_ports = 0; int child_nodes, rc = -ENOMEM, enabled_ports = 0;
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
struct ahci_host_priv *hpriv; struct ahci_host_priv *hpriv;
struct device_node *child;
u32 mask_port_map = 0; u32 mask_port_map = 0;
if (!devres_open_group(dev, NULL, GFP_KERNEL)) if (!devres_open_group(dev, NULL, GFP_KERNEL))
...@@ -579,7 +575,7 @@ struct ahci_host_priv *ahci_platform_get_resources(struct platform_device *pdev, ...@@ -579,7 +575,7 @@ struct ahci_host_priv *ahci_platform_get_resources(struct platform_device *pdev,
} }
if (child_nodes) { if (child_nodes) {
for_each_child_of_node(dev->of_node, child) { for_each_child_of_node_scoped(dev->of_node, child) {
u32 port; u32 port;
struct platform_device *port_dev __maybe_unused; struct platform_device *port_dev __maybe_unused;
...@@ -588,7 +584,6 @@ struct ahci_host_priv *ahci_platform_get_resources(struct platform_device *pdev, ...@@ -588,7 +584,6 @@ struct ahci_host_priv *ahci_platform_get_resources(struct platform_device *pdev,
if (of_property_read_u32(child, "reg", &port)) { if (of_property_read_u32(child, "reg", &port)) {
rc = -EINVAL; rc = -EINVAL;
of_node_put(child);
goto err_out; goto err_out;
} }
...@@ -606,18 +601,14 @@ struct ahci_host_priv *ahci_platform_get_resources(struct platform_device *pdev, ...@@ -606,18 +601,14 @@ struct ahci_host_priv *ahci_platform_get_resources(struct platform_device *pdev,
if (port_dev) { if (port_dev) {
rc = ahci_platform_get_regulator(hpriv, port, rc = ahci_platform_get_regulator(hpriv, port,
&port_dev->dev); &port_dev->dev);
if (rc == -EPROBE_DEFER) { if (rc == -EPROBE_DEFER)
of_node_put(child);
goto err_out; goto err_out;
}
} }
#endif #endif
rc = ahci_platform_get_phy(hpriv, port, dev, child); rc = ahci_platform_get_phy(hpriv, port, dev, child);
if (rc) { if (rc)
of_node_put(child);
goto err_out; goto err_out;
}
enabled_ports++; enabled_ports++;
} }
......
This diff is collapsed.
...@@ -500,10 +500,13 @@ static void ata_eh_dev_disable(struct ata_device *dev) ...@@ -500,10 +500,13 @@ static void ata_eh_dev_disable(struct ata_device *dev)
ata_down_xfermask_limit(dev, ATA_DNXFER_FORCE_PIO0 | ATA_DNXFER_QUIET); ata_down_xfermask_limit(dev, ATA_DNXFER_FORCE_PIO0 | ATA_DNXFER_QUIET);
dev->class++; dev->class++;
/* From now till the next successful probe, ering is used to /*
* From now till the next successful probe, ering is used to
* track probe failures. Clear accumulated device error info. * track probe failures. Clear accumulated device error info.
*/ */
ata_ering_clear(&dev->ering); ata_ering_clear(&dev->ering);
ata_dev_free_resources(dev);
} }
static void ata_eh_unload(struct ata_port *ap) static void ata_eh_unload(struct ata_port *ap)
...@@ -630,6 +633,14 @@ void ata_scsi_cmd_error_handler(struct Scsi_Host *host, struct ata_port *ap, ...@@ -630,6 +633,14 @@ void ata_scsi_cmd_error_handler(struct Scsi_Host *host, struct ata_port *ap,
list_for_each_entry_safe(scmd, tmp, eh_work_q, eh_entry) { list_for_each_entry_safe(scmd, tmp, eh_work_q, eh_entry) {
struct ata_queued_cmd *qc; struct ata_queued_cmd *qc;
/*
* If the scmd was added to EH, via ata_qc_schedule_eh() ->
* scsi_timeout() -> scsi_eh_scmd_add(), scsi_timeout() will
* have set DID_TIME_OUT (since libata does not have an abort
* handler). Thus, to clear DID_TIME_OUT, clear the host byte.
*/
set_host_byte(scmd, DID_OK);
ata_qc_for_each_raw(ap, qc, i) { ata_qc_for_each_raw(ap, qc, i) {
if (qc->flags & ATA_QCFLAG_ACTIVE && if (qc->flags & ATA_QCFLAG_ACTIVE &&
qc->scsicmd == scmd) qc->scsicmd == scmd)
...@@ -1401,6 +1412,43 @@ unsigned int atapi_eh_tur(struct ata_device *dev, u8 *r_sense_key) ...@@ -1401,6 +1412,43 @@ unsigned int atapi_eh_tur(struct ata_device *dev, u8 *r_sense_key)
return err_mask; return err_mask;
} }
/**
* ata_eh_decide_disposition - Disposition a qc based on sense data
* @qc: qc to examine
*
* For a regular SCSI command, the SCSI completion callback (scsi_done())
* will call scsi_complete(), which will call scsi_decide_disposition(),
* which will call scsi_check_sense(). scsi_complete() finally calls
* scsi_finish_command(). This is fine for SCSI, since any eventual sense
* data is usually returned in the completion itself (without invoking SCSI
* EH). However, for a QC, we always need to fetch the sense data
* explicitly using SCSI EH.
*
* A command that is completed via SCSI EH will instead be completed using
* scsi_eh_flush_done_q(), which will call scsi_finish_command() directly
* (without ever calling scsi_check_sense()).
*
* For a command that went through SCSI EH, it is the responsibility of the
* SCSI EH strategy handler to call scsi_decide_disposition(), see e.g. how
* scsi_eh_get_sense() calls scsi_decide_disposition() for SCSI LLDDs that
* do not get the sense data as part of the completion.
*
* Thus, for QC commands that went via SCSI EH, we need to call
* scsi_check_sense() ourselves, similar to how scsi_eh_get_sense() calls
* scsi_decide_disposition(), which calls scsi_check_sense(), in order to
* set the correct SCSI ML byte (if any).
*
* LOCKING:
* EH context.
*
* RETURNS:
* SUCCESS or FAILED or NEEDS_RETRY or ADD_TO_MLQUEUE
*/
enum scsi_disposition ata_eh_decide_disposition(struct ata_queued_cmd *qc)
{
return scsi_check_sense(qc->scsicmd);
}
/** /**
* ata_eh_request_sense - perform REQUEST_SENSE_DATA_EXT * ata_eh_request_sense - perform REQUEST_SENSE_DATA_EXT
* @qc: qc to perform REQUEST_SENSE_SENSE_DATA_EXT to * @qc: qc to perform REQUEST_SENSE_SENSE_DATA_EXT to
...@@ -1627,7 +1675,8 @@ static unsigned int ata_eh_analyze_tf(struct ata_queued_cmd *qc) ...@@ -1627,7 +1675,8 @@ static unsigned int ata_eh_analyze_tf(struct ata_queued_cmd *qc)
} }
if (qc->flags & ATA_QCFLAG_SENSE_VALID) { if (qc->flags & ATA_QCFLAG_SENSE_VALID) {
enum scsi_disposition ret = scsi_check_sense(qc->scsicmd); enum scsi_disposition ret = ata_eh_decide_disposition(qc);
/* /*
* SUCCESS here means that the sense code could be * SUCCESS here means that the sense code could be
* evaluated and should be passed to the upper layers * evaluated and should be passed to the upper layers
...@@ -1924,7 +1973,7 @@ static inline bool ata_eh_quiet(struct ata_queued_cmd *qc) ...@@ -1924,7 +1973,7 @@ static inline bool ata_eh_quiet(struct ata_queued_cmd *qc)
return qc->flags & ATA_QCFLAG_QUIET; return qc->flags & ATA_QCFLAG_QUIET;
} }
static int ata_eh_read_sense_success_non_ncq(struct ata_link *link) static int ata_eh_get_non_ncq_success_sense(struct ata_link *link)
{ {
struct ata_port *ap = link->ap; struct ata_port *ap = link->ap;
struct ata_queued_cmd *qc; struct ata_queued_cmd *qc;
...@@ -1942,11 +1991,10 @@ static int ata_eh_read_sense_success_non_ncq(struct ata_link *link) ...@@ -1942,11 +1991,10 @@ static int ata_eh_read_sense_success_non_ncq(struct ata_link *link)
return -EIO; return -EIO;
/* /*
* If we have sense data, call scsi_check_sense() in order to set the * No point in checking the return value, since the command has already
* correct SCSI ML byte (if any). No point in checking the return value, * completed successfully.
* since the command has already completed successfully.
*/ */
scsi_check_sense(qc->scsicmd); ata_eh_decide_disposition(qc);
return 0; return 0;
} }
...@@ -1976,9 +2024,9 @@ static void ata_eh_get_success_sense(struct ata_link *link) ...@@ -1976,9 +2024,9 @@ static void ata_eh_get_success_sense(struct ata_link *link)
* request sense ext command to retrieve the sense data. * request sense ext command to retrieve the sense data.
*/ */
if (link->sactive) if (link->sactive)
ret = ata_eh_read_sense_success_ncq_log(link); ret = ata_eh_get_ncq_success_sense(link);
else else
ret = ata_eh_read_sense_success_non_ncq(link); ret = ata_eh_get_non_ncq_success_sense(link);
if (ret) if (ret)
goto out; goto out;
...@@ -3247,7 +3295,7 @@ static int atapi_eh_clear_ua(struct ata_device *dev) ...@@ -3247,7 +3295,7 @@ static int atapi_eh_clear_ua(struct ata_device *dev)
int i; int i;
for (i = 0; i < ATA_EH_UA_TRIES; i++) { for (i = 0; i < ATA_EH_UA_TRIES; i++) {
u8 *sense_buffer = dev->link->ap->sector_buf; u8 *sense_buffer = dev->sector_buf;
u8 sense_key = 0; u8 sense_key = 0;
unsigned int err_mask; unsigned int err_mask;
......
...@@ -648,8 +648,7 @@ static int sata_pmp_same_pmp(struct ata_device *dev, const u32 *new_gscr) ...@@ -648,8 +648,7 @@ static int sata_pmp_same_pmp(struct ata_device *dev, const u32 *new_gscr)
static int sata_pmp_revalidate(struct ata_device *dev, unsigned int new_class) static int sata_pmp_revalidate(struct ata_device *dev, unsigned int new_class)
{ {
struct ata_link *link = dev->link; struct ata_link *link = dev->link;
struct ata_port *ap = link->ap; u32 *gscr = (void *)dev->sector_buf;
u32 *gscr = (void *)ap->sector_buf;
int rc; int rc;
ata_eh_about_to_do(link, NULL, ATA_EH_REVALIDATE); ata_eh_about_to_do(link, NULL, ATA_EH_REVALIDATE);
......
...@@ -517,6 +517,86 @@ int sata_set_spd(struct ata_link *link) ...@@ -517,6 +517,86 @@ int sata_set_spd(struct ata_link *link)
} }
EXPORT_SYMBOL_GPL(sata_set_spd); EXPORT_SYMBOL_GPL(sata_set_spd);
/**
* sata_down_spd_limit - adjust SATA spd limit downward
* @link: Link to adjust SATA spd limit for
* @spd_limit: Additional limit
*
* Adjust SATA spd limit of @link downward. Note that this
* function only adjusts the limit. The change must be applied
* using sata_set_spd().
*
* If @spd_limit is non-zero, the speed is limited to equal to or
* lower than @spd_limit if such speed is supported. If
* @spd_limit is slower than any supported speed, only the lowest
* supported speed is allowed.
*
* LOCKING:
* Inherited from caller.
*
* RETURNS:
* 0 on success, negative errno on failure
*/
int sata_down_spd_limit(struct ata_link *link, u32 spd_limit)
{
u32 sstatus, spd, mask;
int rc, bit;
if (!sata_scr_valid(link))
return -EOPNOTSUPP;
/* If SCR can be read, use it to determine the current SPD.
* If not, use cached value in link->sata_spd.
*/
rc = sata_scr_read(link, SCR_STATUS, &sstatus);
if (rc == 0 && ata_sstatus_online(sstatus))
spd = (sstatus >> 4) & 0xf;
else
spd = link->sata_spd;
mask = link->sata_spd_limit;
if (mask <= 1)
return -EINVAL;
/* unconditionally mask off the highest bit */
bit = fls(mask) - 1;
mask &= ~(1 << bit);
/*
* Mask off all speeds higher than or equal to the current one. At
* this point, if current SPD is not available and we previously
* recorded the link speed from SStatus, the driver has already
* masked off the highest bit so mask should already be 1 or 0.
* Otherwise, we should not force 1.5Gbps on a link where we have
* not previously recorded speed from SStatus. Just return in this
* case.
*/
if (spd > 1)
mask &= (1 << (spd - 1)) - 1;
else if (link->sata_spd)
return -EINVAL;
/* were we already at the bottom? */
if (!mask)
return -EINVAL;
if (spd_limit) {
if (mask & ((1 << spd_limit) - 1))
mask &= (1 << spd_limit) - 1;
else {
bit = ffs(mask) - 1;
mask = 1 << bit;
}
}
link->sata_spd_limit = mask;
ata_link_warn(link, "limiting SATA link speed to %s\n",
sata_spd_string(fls(mask)));
return 0;
}
/** /**
* sata_link_hardreset - reset link via SATA phy reset * sata_link_hardreset - reset link via SATA phy reset
* @link: link to reset * @link: link to reset
...@@ -626,6 +706,34 @@ int sata_link_hardreset(struct ata_link *link, const unsigned int *timing, ...@@ -626,6 +706,34 @@ int sata_link_hardreset(struct ata_link *link, const unsigned int *timing,
} }
EXPORT_SYMBOL_GPL(sata_link_hardreset); EXPORT_SYMBOL_GPL(sata_link_hardreset);
/**
* sata_std_hardreset - COMRESET w/o waiting or classification
* @link: link to reset
* @class: resulting class of attached device
* @deadline: deadline jiffies for the operation
*
* Standard SATA COMRESET w/o waiting or classification.
*
* LOCKING:
* Kernel thread context (may sleep)
*
* RETURNS:
* 0 if link offline, -EAGAIN if link online, -errno on errors.
*/
int sata_std_hardreset(struct ata_link *link, unsigned int *class,
unsigned long deadline)
{
const unsigned int *timing = sata_ehc_deb_timing(&link->eh_context);
bool online;
int rc;
rc = sata_link_hardreset(link, timing, deadline, &online, NULL);
if (online)
return -EAGAIN;
return rc;
}
EXPORT_SYMBOL_GPL(sata_std_hardreset);
/** /**
* ata_qc_complete_multiple - Complete multiple qcs successfully * ata_qc_complete_multiple - Complete multiple qcs successfully
* @ap: port in question * @ap: port in question
...@@ -818,7 +926,7 @@ static ssize_t ata_scsi_lpm_store(struct device *device, ...@@ -818,7 +926,7 @@ static ssize_t ata_scsi_lpm_store(struct device *device,
ata_for_each_link(link, ap, EDGE) { ata_for_each_link(link, ap, EDGE) {
ata_for_each_dev(dev, &ap->link, ENABLED) { ata_for_each_dev(dev, &ap->link, ENABLED) {
if (dev->horkage & ATA_HORKAGE_NOLPM) { if (dev->quirks & ATA_QUIRK_NOLPM) {
count = -EOPNOTSUPP; count = -EOPNOTSUPP;
goto out_unlock; goto out_unlock;
} }
...@@ -1340,7 +1448,7 @@ EXPORT_SYMBOL_GPL(sata_async_notification); ...@@ -1340,7 +1448,7 @@ EXPORT_SYMBOL_GPL(sata_async_notification);
static int ata_eh_read_log_10h(struct ata_device *dev, static int ata_eh_read_log_10h(struct ata_device *dev,
int *tag, struct ata_taskfile *tf) int *tag, struct ata_taskfile *tf)
{ {
u8 *buf = dev->link->ap->sector_buf; u8 *buf = dev->sector_buf;
unsigned int err_mask; unsigned int err_mask;
u8 csum; u8 csum;
int i; int i;
...@@ -1379,8 +1487,8 @@ static int ata_eh_read_log_10h(struct ata_device *dev, ...@@ -1379,8 +1487,8 @@ static int ata_eh_read_log_10h(struct ata_device *dev,
} }
/** /**
* ata_eh_read_sense_success_ncq_log - Read the sense data for successful * ata_eh_get_ncq_success_sense - Read and process the sense data for
* NCQ commands log * successful NCQ commands log page
* @link: ATA link to get sense data for * @link: ATA link to get sense data for
* *
* Read the sense data for successful NCQ commands log page to obtain * Read the sense data for successful NCQ commands log page to obtain
...@@ -1393,11 +1501,11 @@ static int ata_eh_read_log_10h(struct ata_device *dev, ...@@ -1393,11 +1501,11 @@ static int ata_eh_read_log_10h(struct ata_device *dev,
* RETURNS: * RETURNS:
* 0 on success, -errno otherwise. * 0 on success, -errno otherwise.
*/ */
int ata_eh_read_sense_success_ncq_log(struct ata_link *link) int ata_eh_get_ncq_success_sense(struct ata_link *link)
{ {
struct ata_device *dev = link->device; struct ata_device *dev = link->device;
struct ata_port *ap = dev->link->ap; struct ata_port *ap = dev->link->ap;
u8 *buf = ap->ncq_sense_buf; u8 *buf = dev->cdl->ncq_sense_log_buf;
struct ata_queued_cmd *qc; struct ata_queued_cmd *qc;
unsigned int err_mask, tag; unsigned int err_mask, tag;
u8 *sense, sk = 0, asc = 0, ascq = 0; u8 *sense, sk = 0, asc = 0, ascq = 0;
...@@ -1455,17 +1563,14 @@ int ata_eh_read_sense_success_ncq_log(struct ata_link *link) ...@@ -1455,17 +1563,14 @@ int ata_eh_read_sense_success_ncq_log(struct ata_link *link)
qc->flags |= ATA_QCFLAG_SENSE_VALID; qc->flags |= ATA_QCFLAG_SENSE_VALID;
/* /*
* If we have sense data, call scsi_check_sense() in order to * No point in checking the return value, since the command has
* set the correct SCSI ML byte (if any). No point in checking * already completed successfully.
* the return value, since the command has already completed
* successfully.
*/ */
scsi_check_sense(qc->scsicmd); ata_eh_decide_disposition(qc);
} }
return ret; return ret;
} }
EXPORT_SYMBOL_GPL(ata_eh_read_sense_success_ncq_log);
/** /**
* ata_eh_analyze_ncq_error - analyze NCQ error * ata_eh_analyze_ncq_error - analyze NCQ error
...@@ -1576,3 +1681,11 @@ void ata_eh_analyze_ncq_error(struct ata_link *link) ...@@ -1576,3 +1681,11 @@ void ata_eh_analyze_ncq_error(struct ata_link *link)
ehc->i.err_mask &= ~AC_ERR_DEV; ehc->i.err_mask &= ~AC_ERR_DEV;
} }
EXPORT_SYMBOL_GPL(ata_eh_analyze_ncq_error); EXPORT_SYMBOL_GPL(ata_eh_analyze_ncq_error);
const struct ata_port_operations sata_port_ops = {
.inherits = &ata_base_port_ops,
.qc_defer = ata_std_qc_defer,
.hardreset = sata_std_hardreset,
};
EXPORT_SYMBOL_GPL(sata_port_ops);
...@@ -1691,9 +1691,6 @@ static void ata_scsi_qc_complete(struct ata_queued_cmd *qc) ...@@ -1691,9 +1691,6 @@ static void ata_scsi_qc_complete(struct ata_queued_cmd *qc)
set_status_byte(qc->scsicmd, SAM_STAT_CHECK_CONDITION); set_status_byte(qc->scsicmd, SAM_STAT_CHECK_CONDITION);
} else if (is_error && !have_sense) { } else if (is_error && !have_sense) {
ata_gen_ata_sense(qc); ata_gen_ata_sense(qc);
} else {
/* Keep the SCSI ML and status byte, clear host byte. */
cmd->result &= 0x0000ffff;
} }
ata_qc_done(qc); ata_qc_done(qc);
...@@ -2094,7 +2091,7 @@ static unsigned int ata_scsiop_inq_b0(struct ata_scsi_args *args, u8 *rbuf) ...@@ -2094,7 +2091,7 @@ static unsigned int ata_scsiop_inq_b0(struct ata_scsi_args *args, u8 *rbuf)
if (ata_id_has_trim(args->id)) { if (ata_id_has_trim(args->id)) {
u64 max_blocks = 65535 * ATA_MAX_TRIM_RNUM; u64 max_blocks = 65535 * ATA_MAX_TRIM_RNUM;
if (dev->horkage & ATA_HORKAGE_MAX_TRIM_128M) if (dev->quirks & ATA_QUIRK_MAX_TRIM_128M)
max_blocks = 128 << (20 - SECTOR_SHIFT); max_blocks = 128 << (20 - SECTOR_SHIFT);
put_unaligned_be64(max_blocks, &rbuf[36]); put_unaligned_be64(max_blocks, &rbuf[36]);
...@@ -2259,7 +2256,7 @@ static inline u16 ata_xlat_cdl_limit(u8 *buf) ...@@ -2259,7 +2256,7 @@ static inline u16 ata_xlat_cdl_limit(u8 *buf)
static unsigned int ata_msense_control_spgt2(struct ata_device *dev, u8 *buf, static unsigned int ata_msense_control_spgt2(struct ata_device *dev, u8 *buf,
u8 spg) u8 spg)
{ {
u8 *b, *cdl = dev->cdl, *desc; u8 *b, *cdl = dev->cdl->desc_log_buf, *desc;
u32 policy; u32 policy;
int i; int i;
...@@ -2572,11 +2569,11 @@ static unsigned int ata_scsiop_read_cap(struct ata_scsi_args *args, u8 *rbuf) ...@@ -2572,11 +2569,11 @@ static unsigned int ata_scsiop_read_cap(struct ata_scsi_args *args, u8 *rbuf)
rbuf[15] = lowest_aligned; rbuf[15] = lowest_aligned;
if (ata_id_has_trim(args->id) && if (ata_id_has_trim(args->id) &&
!(dev->horkage & ATA_HORKAGE_NOTRIM)) { !(dev->quirks & ATA_QUIRK_NOTRIM)) {
rbuf[14] |= 0x80; /* LBPME */ rbuf[14] |= 0x80; /* LBPME */
if (ata_id_has_zero_after_trim(args->id) && if (ata_id_has_zero_after_trim(args->id) &&
dev->horkage & ATA_HORKAGE_ZERO_AFTER_TRIM) { dev->quirks & ATA_QUIRK_ZERO_AFTER_TRIM) {
ata_dev_info(dev, "Enabling discard_zeroes_data\n"); ata_dev_info(dev, "Enabling discard_zeroes_data\n");
rbuf[14] |= 0x40; /* LBPRZ */ rbuf[14] |= 0x40; /* LBPRZ */
} }
...@@ -3240,8 +3237,7 @@ static unsigned int ata_scsi_write_same_xlat(struct ata_queued_cmd *qc) ...@@ -3240,8 +3237,7 @@ static unsigned int ata_scsi_write_same_xlat(struct ata_queued_cmd *qc)
} }
scsi_16_lba_len(cdb, &block, &n_block); scsi_16_lba_len(cdb, &block, &n_block);
if (!unmap || if (!unmap || (dev->quirks & ATA_QUIRK_NOTRIM) ||
(dev->horkage & ATA_HORKAGE_NOTRIM) ||
!ata_id_has_trim(dev->id)) { !ata_id_has_trim(dev->id)) {
fp = 1; fp = 1;
bp = 3; bp = 3;
...@@ -4617,16 +4613,15 @@ static void ata_scsi_handle_link_detach(struct ata_link *link) ...@@ -4617,16 +4613,15 @@ static void ata_scsi_handle_link_detach(struct ata_link *link)
ata_for_each_dev(dev, link, ALL) { ata_for_each_dev(dev, link, ALL) {
unsigned long flags; unsigned long flags;
if (!(dev->flags & ATA_DFLAG_DETACHED)) spin_lock_irqsave(ap->lock, flags);
if (!(dev->flags & ATA_DFLAG_DETACHED)) {
spin_unlock_irqrestore(ap->lock, flags);
continue; continue;
}
spin_lock_irqsave(ap->lock, flags);
dev->flags &= ~ATA_DFLAG_DETACHED; dev->flags &= ~ATA_DFLAG_DETACHED;
spin_unlock_irqrestore(ap->lock, flags); spin_unlock_irqrestore(ap->lock, flags);
if (zpodd_dev_enabled(dev))
zpodd_exit(dev);
ata_scsi_remove_dev(dev); ata_scsi_remove_dev(dev);
} }
} }
......
...@@ -26,7 +26,6 @@ static struct workqueue_struct *ata_sff_wq; ...@@ -26,7 +26,6 @@ static struct workqueue_struct *ata_sff_wq;
const struct ata_port_operations ata_sff_port_ops = { const struct ata_port_operations ata_sff_port_ops = {
.inherits = &ata_base_port_ops, .inherits = &ata_base_port_ops,
.qc_prep = ata_noop_qc_prep,
.qc_issue = ata_sff_qc_issue, .qc_issue = ata_sff_qc_issue,
.qc_fill_rtf = ata_sff_qc_fill_rtf, .qc_fill_rtf = ata_sff_qc_fill_rtf,
...@@ -970,7 +969,7 @@ int ata_sff_hsm_move(struct ata_port *ap, struct ata_queued_cmd *qc, ...@@ -970,7 +969,7 @@ int ata_sff_hsm_move(struct ata_port *ap, struct ata_queued_cmd *qc,
* We ignore ERR here to workaround and proceed sending * We ignore ERR here to workaround and proceed sending
* the CDB. * the CDB.
*/ */
if (!(qc->dev->horkage & ATA_HORKAGE_STUCK_ERR)) { if (!(qc->dev->quirks & ATA_QUIRK_STUCK_ERR)) {
ata_ehi_push_desc(ehi, "ST_FIRST: " ata_ehi_push_desc(ehi, "ST_FIRST: "
"DRQ=1 with device error, " "DRQ=1 with device error, "
"dev_stat 0x%X", status); "dev_stat 0x%X", status);
...@@ -1045,8 +1044,8 @@ int ata_sff_hsm_move(struct ata_port *ap, struct ata_queued_cmd *qc, ...@@ -1045,8 +1044,8 @@ int ata_sff_hsm_move(struct ata_port *ap, struct ata_queued_cmd *qc,
* IDENTIFY, it's likely a phantom * IDENTIFY, it's likely a phantom
* device. Mark hint. * device. Mark hint.
*/ */
if (qc->dev->horkage & if (qc->dev->quirks &
ATA_HORKAGE_DIAGNOSTIC) ATA_QUIRK_DIAGNOSTIC)
qc->err_mask |= qc->err_mask |=
AC_ERR_NODEV_HINT; AC_ERR_NODEV_HINT;
} else { } else {
...@@ -1762,7 +1761,7 @@ unsigned int ata_sff_dev_classify(struct ata_device *dev, int present, ...@@ -1762,7 +1761,7 @@ unsigned int ata_sff_dev_classify(struct ata_device *dev, int present,
/* see if device passed diags: continue and warn later */ /* see if device passed diags: continue and warn later */
if (err == 0) if (err == 0)
/* diagnostic fail : do nothing _YET_ */ /* diagnostic fail : do nothing _YET_ */
dev->horkage |= ATA_HORKAGE_DIAGNOSTIC; dev->quirks |= ATA_QUIRK_DIAGNOSTIC;
else if (err == 1) else if (err == 1)
/* do nothing */ ; /* do nothing */ ;
else if ((dev->devno == 0) && (err == 0x81)) else if ((dev->devno == 0) && (err == 0x81))
...@@ -1781,7 +1780,7 @@ unsigned int ata_sff_dev_classify(struct ata_device *dev, int present, ...@@ -1781,7 +1780,7 @@ unsigned int ata_sff_dev_classify(struct ata_device *dev, int present,
* device signature is invalid with diagnostic * device signature is invalid with diagnostic
* failure. * failure.
*/ */
if (present && (dev->horkage & ATA_HORKAGE_DIAGNOSTIC)) if (present && (dev->quirks & ATA_QUIRK_DIAGNOSTIC))
class = ATA_DEV_ATA; class = ATA_DEV_ATA;
else else
class = ATA_DEV_NONE; class = ATA_DEV_NONE;
......
...@@ -80,12 +80,6 @@ struct ata_internal { ...@@ -80,12 +80,6 @@ struct ata_internal {
#define transport_class_to_port(dev) \ #define transport_class_to_port(dev) \
tdev_to_port((dev)->parent) tdev_to_port((dev)->parent)
/* Device objects are always created whit link objects */
static int ata_tdev_add(struct ata_device *dev);
static void ata_tdev_delete(struct ata_device *dev);
/* /*
* Hack to allow attributes of the same name in different objects. * Hack to allow attributes of the same name in different objects.
*/ */
...@@ -364,135 +358,6 @@ unsigned int ata_port_classify(struct ata_port *ap, ...@@ -364,135 +358,6 @@ unsigned int ata_port_classify(struct ata_port *ap,
} }
EXPORT_SYMBOL_GPL(ata_port_classify); EXPORT_SYMBOL_GPL(ata_port_classify);
/*
* ATA link attributes
*/
static int noop(int x) { return x; }
#define ata_link_show_linkspeed(field, format) \
static ssize_t \
show_ata_link_##field(struct device *dev, \
struct device_attribute *attr, char *buf) \
{ \
struct ata_link *link = transport_class_to_link(dev); \
\
return sprintf(buf, "%s\n", sata_spd_string(format(link->field))); \
}
#define ata_link_linkspeed_attr(field, format) \
ata_link_show_linkspeed(field, format) \
static DEVICE_ATTR(field, S_IRUGO, show_ata_link_##field, NULL)
ata_link_linkspeed_attr(hw_sata_spd_limit, fls);
ata_link_linkspeed_attr(sata_spd_limit, fls);
ata_link_linkspeed_attr(sata_spd, noop);
static DECLARE_TRANSPORT_CLASS(ata_link_class,
"ata_link", NULL, NULL, NULL);
static void ata_tlink_release(struct device *dev)
{
}
/**
* ata_is_link -- check if a struct device represents a ATA link
* @dev: device to check
*
* Returns:
* %1 if the device represents a ATA link, %0 else
*/
static int ata_is_link(const struct device *dev)
{
return dev->release == ata_tlink_release;
}
static int ata_tlink_match(struct attribute_container *cont,
struct device *dev)
{
struct ata_internal* i = to_ata_internal(ata_scsi_transport_template);
if (!ata_is_link(dev))
return 0;
return &i->link_attr_cont.ac == cont;
}
/**
* ata_tlink_delete -- remove ATA LINK
* @link: ATA LINK to remove
*
* Removes the specified ATA LINK. remove associated ATA device(s) as well.
*/
void ata_tlink_delete(struct ata_link *link)
{
struct device *dev = &link->tdev;
struct ata_device *ata_dev;
ata_for_each_dev(ata_dev, link, ALL) {
ata_tdev_delete(ata_dev);
}
transport_remove_device(dev);
device_del(dev);
transport_destroy_device(dev);
put_device(dev);
}
/**
* ata_tlink_add -- initialize a transport ATA link structure
* @link: allocated ata_link structure.
*
* Initialize an ATA LINK structure for sysfs. It will be added in the
* device tree below the ATA PORT it belongs to.
*
* Returns %0 on success
*/
int ata_tlink_add(struct ata_link *link)
{
struct device *dev = &link->tdev;
struct ata_port *ap = link->ap;
struct ata_device *ata_dev;
int error;
device_initialize(dev);
dev->parent = &ap->tdev;
dev->release = ata_tlink_release;
if (ata_is_host_link(link))
dev_set_name(dev, "link%d", ap->print_id);
else
dev_set_name(dev, "link%d.%d", ap->print_id, link->pmp);
transport_setup_device(dev);
error = device_add(dev);
if (error) {
goto tlink_err;
}
error = transport_add_device(dev);
if (error)
goto tlink_transport_err;
transport_configure_device(dev);
ata_for_each_dev(ata_dev, link, ALL) {
error = ata_tdev_add(ata_dev);
if (error) {
goto tlink_dev_err;
}
}
return 0;
tlink_dev_err:
while (--ata_dev >= link->device) {
ata_tdev_delete(ata_dev);
}
transport_remove_device(dev);
tlink_transport_err:
device_del(dev);
tlink_err:
transport_destroy_device(dev);
put_device(dev);
return error;
}
/* /*
* ATA device attributes * ATA device attributes
*/ */
...@@ -617,10 +482,10 @@ show_ata_dev_trim(struct device *dev, ...@@ -617,10 +482,10 @@ show_ata_dev_trim(struct device *dev,
if (!ata_id_has_trim(ata_dev->id)) if (!ata_id_has_trim(ata_dev->id))
mode = "unsupported"; mode = "unsupported";
else if (ata_dev->horkage & ATA_HORKAGE_NOTRIM) else if (ata_dev->quirks & ATA_QUIRK_NOTRIM)
mode = "forced_unsupported"; mode = "forced_unsupported";
else if (ata_dev->horkage & ATA_HORKAGE_NO_NCQ_TRIM) else if (ata_dev->quirks & ATA_QUIRK_NO_NCQ_TRIM)
mode = "forced_unqueued"; mode = "forced_unqueued";
else if (ata_fpdma_dsm_supported(ata_dev)) else if (ata_fpdma_dsm_supported(ata_dev))
mode = "queued"; mode = "queued";
else else
...@@ -643,9 +508,9 @@ static void ata_tdev_release(struct device *dev) ...@@ -643,9 +508,9 @@ static void ata_tdev_release(struct device *dev)
* @dev: device to check * @dev: device to check
* *
* Returns: * Returns:
* %1 if the device represents a ATA device, %0 else * true if the device represents a ATA device, false otherwise
*/ */
static int ata_is_ata_dev(const struct device *dev) static bool ata_is_ata_dev(const struct device *dev)
{ {
return dev->release == ata_tdev_release; return dev->release == ata_tdev_release;
} }
...@@ -653,21 +518,22 @@ static int ata_is_ata_dev(const struct device *dev) ...@@ -653,21 +518,22 @@ static int ata_is_ata_dev(const struct device *dev)
static int ata_tdev_match(struct attribute_container *cont, static int ata_tdev_match(struct attribute_container *cont,
struct device *dev) struct device *dev)
{ {
struct ata_internal* i = to_ata_internal(ata_scsi_transport_template); struct ata_internal *i = to_ata_internal(ata_scsi_transport_template);
if (!ata_is_ata_dev(dev)) if (!ata_is_ata_dev(dev))
return 0; return 0;
return &i->dev_attr_cont.ac == cont; return &i->dev_attr_cont.ac == cont;
} }
/** /**
* ata_tdev_free -- free a ATA LINK * ata_tdev_free -- free an ATA transport device
* @dev: ATA PHY to free * @dev: struct ata_device owning the transport device to free
* *
* Frees the specified ATA PHY. * Free the ATA transport device for the specified ATA device.
* *
* Note: * Note:
* This function must only be called on a PHY that has not * This function must only be called for a ATA transport device that has not
* successfully been added using ata_tdev_add(). * yet successfully been added using ata_tdev_add().
*/ */
static void ata_tdev_free(struct ata_device *dev) static void ata_tdev_free(struct ata_device *dev)
{ {
...@@ -676,10 +542,10 @@ static void ata_tdev_free(struct ata_device *dev) ...@@ -676,10 +542,10 @@ static void ata_tdev_free(struct ata_device *dev)
} }
/** /**
* ata_tdev_delete -- remove ATA device * ata_tdev_delete -- remove an ATA transport device
* @ata_dev: ATA device to remove * @ata_dev: struct ata_device owning the transport device to delete
* *
* Removes the specified ATA device. * Removes the ATA transport device for the specified ATA device.
*/ */
static void ata_tdev_delete(struct ata_device *ata_dev) static void ata_tdev_delete(struct ata_device *ata_dev)
{ {
...@@ -690,15 +556,14 @@ static void ata_tdev_delete(struct ata_device *ata_dev) ...@@ -690,15 +556,14 @@ static void ata_tdev_delete(struct ata_device *ata_dev)
ata_tdev_free(ata_dev); ata_tdev_free(ata_dev);
} }
/** /**
* ata_tdev_add -- initialize a transport ATA device structure. * ata_tdev_add -- initialize an ATA transport device
* @ata_dev: ata_dev structure. * @ata_dev: struct ata_device owning the transport device to add
* *
* Initialize an ATA device structure for sysfs. It will be added in the * Initialize an ATA transport device for sysfs. It will be added in the
* device tree below the ATA LINK device it belongs to. * device tree below the ATA link device it belongs to.
* *
* Returns %0 on success * Returns %0 on success and a negative error code on error.
*/ */
static int ata_tdev_add(struct ata_device *ata_dev) static int ata_tdev_add(struct ata_device *ata_dev)
{ {
...@@ -734,6 +599,136 @@ static int ata_tdev_add(struct ata_device *ata_dev) ...@@ -734,6 +599,136 @@ static int ata_tdev_add(struct ata_device *ata_dev)
return 0; return 0;
} }
/*
* ATA link attributes
*/
static int noop(int x)
{
return x;
}
#define ata_link_show_linkspeed(field, format) \
static ssize_t \
show_ata_link_##field(struct device *dev, \
struct device_attribute *attr, char *buf) \
{ \
struct ata_link *link = transport_class_to_link(dev); \
\
return sprintf(buf, "%s\n", \
sata_spd_string(format(link->field))); \
}
#define ata_link_linkspeed_attr(field, format) \
ata_link_show_linkspeed(field, format) \
static DEVICE_ATTR(field, 0444, show_ata_link_##field, NULL)
ata_link_linkspeed_attr(hw_sata_spd_limit, fls);
ata_link_linkspeed_attr(sata_spd_limit, fls);
ata_link_linkspeed_attr(sata_spd, noop);
static DECLARE_TRANSPORT_CLASS(ata_link_class,
"ata_link", NULL, NULL, NULL);
static void ata_tlink_release(struct device *dev)
{
}
/**
* ata_is_link -- check if a struct device represents a ATA link
* @dev: device to check
*
* Returns:
* true if the device represents a ATA link, false otherwise
*/
static bool ata_is_link(const struct device *dev)
{
return dev->release == ata_tlink_release;
}
static int ata_tlink_match(struct attribute_container *cont,
struct device *dev)
{
struct ata_internal *i = to_ata_internal(ata_scsi_transport_template);
if (!ata_is_link(dev))
return 0;
return &i->link_attr_cont.ac == cont;
}
/**
* ata_tlink_delete -- remove an ATA link transport device
* @link: struct ata_link owning the link transport device to remove
*
* Removes the link transport device of the specified ATA link. This also
* removes the ATA device(s) associated with the link as well.
*/
void ata_tlink_delete(struct ata_link *link)
{
struct device *dev = &link->tdev;
struct ata_device *ata_dev;
ata_for_each_dev(ata_dev, link, ALL) {
ata_tdev_delete(ata_dev);
}
transport_remove_device(dev);
device_del(dev);
transport_destroy_device(dev);
put_device(dev);
}
/**
* ata_tlink_add -- initialize an ATA link transport device
* @link: struct ata_link owning the link transport device to initialize
*
* Initialize an ATA link transport device for sysfs. It will be added in the
* device tree below the ATA port it belongs to.
*
* Returns %0 on success and a negative error code on error.
*/
int ata_tlink_add(struct ata_link *link)
{
struct device *dev = &link->tdev;
struct ata_port *ap = link->ap;
struct ata_device *ata_dev;
int error;
device_initialize(dev);
dev->parent = &ap->tdev;
dev->release = ata_tlink_release;
if (ata_is_host_link(link))
dev_set_name(dev, "link%d", ap->print_id);
else
dev_set_name(dev, "link%d.%d", ap->print_id, link->pmp);
transport_setup_device(dev);
error = device_add(dev);
if (error)
goto tlink_err;
error = transport_add_device(dev);
if (error)
goto tlink_transport_err;
transport_configure_device(dev);
ata_for_each_dev(ata_dev, link, ALL) {
error = ata_tdev_add(ata_dev);
if (error)
goto tlink_dev_err;
}
return 0;
tlink_dev_err:
while (--ata_dev >= link->device)
ata_tdev_delete(ata_dev);
transport_remove_device(dev);
tlink_transport_err:
device_del(dev);
tlink_err:
transport_destroy_device(dev);
put_device(dev);
return error;
}
/* /*
* Setup / Teardown code * Setup / Teardown code
......
...@@ -112,7 +112,7 @@ static bool zpready(struct ata_device *dev) ...@@ -112,7 +112,7 @@ static bool zpready(struct ata_device *dev)
if (!ret || sense_key != NOT_READY) if (!ret || sense_key != NOT_READY)
return false; return false;
sense_buf = dev->link->ap->sector_buf; sense_buf = dev->sector_buf;
ret = atapi_eh_request_sense(dev, sense_buf, sense_key); ret = atapi_eh_request_sense(dev, sense_buf, sense_key);
if (ret) if (ret)
return false; return false;
......
...@@ -38,6 +38,12 @@ extern int libata_noacpi; ...@@ -38,6 +38,12 @@ extern int libata_noacpi;
extern int libata_allow_tpm; extern int libata_allow_tpm;
extern const struct device_type ata_port_type; extern const struct device_type ata_port_type;
extern struct ata_link *ata_dev_phys_link(struct ata_device *dev); extern struct ata_link *ata_dev_phys_link(struct ata_device *dev);
static inline bool ata_sstatus_online(u32 sstatus)
{
return (sstatus & 0xf) == 0x3;
}
#ifdef CONFIG_ATA_FORCE #ifdef CONFIG_ATA_FORCE
extern void ata_force_cbl(struct ata_port *ap); extern void ata_force_cbl(struct ata_port *ap);
#else #else
...@@ -65,7 +71,7 @@ extern bool ata_dev_power_init_tf(struct ata_device *dev, ...@@ -65,7 +71,7 @@ extern bool ata_dev_power_init_tf(struct ata_device *dev,
struct ata_taskfile *tf, bool set_active); struct ata_taskfile *tf, bool set_active);
extern void ata_dev_power_set_standby(struct ata_device *dev); extern void ata_dev_power_set_standby(struct ata_device *dev);
extern void ata_dev_power_set_active(struct ata_device *dev); extern void ata_dev_power_set_active(struct ata_device *dev);
extern int sata_down_spd_limit(struct ata_link *link, u32 spd_limit); void ata_dev_free_resources(struct ata_device *dev);
extern int ata_down_xfermask_limit(struct ata_device *dev, unsigned int sel); extern int ata_down_xfermask_limit(struct ata_device *dev, unsigned int sel);
extern unsigned int ata_dev_set_feature(struct ata_device *dev, extern unsigned int ata_dev_set_feature(struct ata_device *dev,
u8 subcmd, u8 action); u8 subcmd, u8 action);
...@@ -84,9 +90,25 @@ extern int ata_cmd_ioctl(struct scsi_device *scsidev, void __user *arg); ...@@ -84,9 +90,25 @@ extern int ata_cmd_ioctl(struct scsi_device *scsidev, void __user *arg);
extern const char *sata_spd_string(unsigned int spd); extern const char *sata_spd_string(unsigned int spd);
extern unsigned int ata_read_log_page(struct ata_device *dev, u8 log, extern unsigned int ata_read_log_page(struct ata_device *dev, u8 log,
u8 page, void *buf, unsigned int sectors); u8 page, void *buf, unsigned int sectors);
void ata_dev_cleanup_cdl_resources(struct ata_device *dev);
#define to_ata_port(d) container_of(d, struct ata_port, tdev) #define to_ata_port(d) container_of(d, struct ata_port, tdev)
/* libata-sata.c */
#ifdef CONFIG_SATA_HOST
int sata_down_spd_limit(struct ata_link *link, u32 spd_limit);
int ata_eh_get_ncq_success_sense(struct ata_link *link);
#else
static inline int sata_down_spd_limit(struct ata_link *link, u32 spd_limit)
{
return -EOPNOTSUPP;
}
static inline int ata_eh_get_ncq_success_sense(struct ata_link *link)
{
return -EOPNOTSUPP;
}
#endif
/* libata-acpi.c */ /* libata-acpi.c */
#ifdef CONFIG_ATA_ACPI #ifdef CONFIG_ATA_ACPI
extern unsigned int ata_acpi_gtf_filter; extern unsigned int ata_acpi_gtf_filter;
...@@ -124,7 +146,6 @@ extern void ata_scsi_set_sense_information(struct ata_device *dev, ...@@ -124,7 +146,6 @@ extern void ata_scsi_set_sense_information(struct ata_device *dev,
const struct ata_taskfile *tf); const struct ata_taskfile *tf);
extern void ata_scsi_media_change_notify(struct ata_device *dev); extern void ata_scsi_media_change_notify(struct ata_device *dev);
extern void ata_scsi_hotplug(struct work_struct *work); extern void ata_scsi_hotplug(struct work_struct *work);
extern void ata_schedule_scsi_eh(struct Scsi_Host *shost);
extern void ata_scsi_dev_rescan(struct work_struct *work); extern void ata_scsi_dev_rescan(struct work_struct *work);
extern int ata_scsi_user_scan(struct Scsi_Host *shost, unsigned int channel, extern int ata_scsi_user_scan(struct Scsi_Host *shost, unsigned int channel,
unsigned int id, u64 lun); unsigned int id, u64 lun);
...@@ -162,6 +183,7 @@ extern void ata_eh_finish(struct ata_port *ap); ...@@ -162,6 +183,7 @@ extern void ata_eh_finish(struct ata_port *ap);
extern int ata_ering_map(struct ata_ering *ering, extern int ata_ering_map(struct ata_ering *ering,
int (*map_fn)(struct ata_ering_entry *, void *), int (*map_fn)(struct ata_ering_entry *, void *),
void *arg); void *arg);
enum scsi_disposition ata_eh_decide_disposition(struct ata_queued_cmd *qc);
extern unsigned int atapi_eh_tur(struct ata_device *dev, u8 *r_sense_key); extern unsigned int atapi_eh_tur(struct ata_device *dev, u8 *r_sense_key);
extern unsigned int atapi_eh_request_sense(struct ata_device *dev, extern unsigned int atapi_eh_request_sense(struct ata_device *dev,
u8 *sense_buf, u8 dfl_sense_key); u8 *sense_buf, u8 dfl_sense_key);
......
...@@ -8,9 +8,9 @@ ...@@ -8,9 +8,9 @@
* PIO mode and smarter silicon. * PIO mode and smarter silicon.
* *
* The practical upshot of this is that we must always tune the * The practical upshot of this is that we must always tune the
* drive for the right PIO mode. We must also ignore all the blacklists * drive for the right PIO mode and ignore the drive bus mastering DMA
* and the drive bus mastering DMA information. Also to confuse matters * information. Also to confuse matters further we can do DMA on PIO only
* further we can do DMA on PIO only drives. * drives.
* *
* DMA on the 5510 also requires we disable_hlt() during DMA on early * DMA on the 5510 also requires we disable_hlt() during DMA on early
* revisions. * revisions.
......
...@@ -884,8 +884,6 @@ static const struct scsi_host_template ep93xx_pata_sht = { ...@@ -884,8 +884,6 @@ static const struct scsi_host_template ep93xx_pata_sht = {
static struct ata_port_operations ep93xx_pata_port_ops = { static struct ata_port_operations ep93xx_pata_port_ops = {
.inherits = &ata_bmdma_port_ops, .inherits = &ata_bmdma_port_ops,
.qc_prep = ata_noop_qc_prep,
.softreset = ep93xx_pata_softreset, .softreset = ep93xx_pata_softreset,
.hardreset = ATA_OP_NULL, .hardreset = ATA_OP_NULL,
......
...@@ -549,6 +549,7 @@ static const struct of_device_id pata_ftide010_of_match[] = { ...@@ -549,6 +549,7 @@ static const struct of_device_id pata_ftide010_of_match[] = {
{ .compatible = "faraday,ftide010", }, { .compatible = "faraday,ftide010", },
{ /* sentinel */ } { /* sentinel */ }
}; };
MODULE_DEVICE_TABLE(of, pata_ftide010_of_match);
static struct platform_driver pata_ftide010_driver = { static struct platform_driver pata_ftide010_driver = {
.driver = { .driver = {
......
...@@ -170,8 +170,8 @@ static const char * const bad_ata66_3[] = { ...@@ -170,8 +170,8 @@ static const char * const bad_ata66_3[] = {
NULL NULL
}; };
static int hpt_dma_blacklisted(const struct ata_device *dev, char *modestr, static int hpt_dma_broken(const struct ata_device *dev, char *modestr,
const char * const list[]) const char * const list[])
{ {
unsigned char model_num[ATA_ID_PROD_LEN + 1]; unsigned char model_num[ATA_ID_PROD_LEN + 1];
int i; int i;
...@@ -197,11 +197,11 @@ static int hpt_dma_blacklisted(const struct ata_device *dev, char *modestr, ...@@ -197,11 +197,11 @@ static int hpt_dma_blacklisted(const struct ata_device *dev, char *modestr,
static unsigned int hpt366_filter(struct ata_device *adev, unsigned int mask) static unsigned int hpt366_filter(struct ata_device *adev, unsigned int mask)
{ {
if (adev->class == ATA_DEV_ATA) { if (adev->class == ATA_DEV_ATA) {
if (hpt_dma_blacklisted(adev, "UDMA", bad_ata33)) if (hpt_dma_broken(adev, "UDMA", bad_ata33))
mask &= ~ATA_MASK_UDMA; mask &= ~ATA_MASK_UDMA;
if (hpt_dma_blacklisted(adev, "UDMA3", bad_ata66_3)) if (hpt_dma_broken(adev, "UDMA3", bad_ata66_3))
mask &= ~(0xF8 << ATA_SHIFT_UDMA); mask &= ~(0xF8 << ATA_SHIFT_UDMA);
if (hpt_dma_blacklisted(adev, "UDMA4", bad_ata66_4)) if (hpt_dma_broken(adev, "UDMA4", bad_ata66_4))
mask &= ~(0xF0 << ATA_SHIFT_UDMA); mask &= ~(0xF0 << ATA_SHIFT_UDMA);
} else if (adev->class == ATA_DEV_ATAPI) } else if (adev->class == ATA_DEV_ATAPI)
mask &= ~(ATA_MASK_MWDMA | ATA_MASK_UDMA); mask &= ~(ATA_MASK_MWDMA | ATA_MASK_UDMA);
......
...@@ -218,8 +218,8 @@ static u32 hpt37x_find_mode(struct ata_port *ap, int speed) ...@@ -218,8 +218,8 @@ static u32 hpt37x_find_mode(struct ata_port *ap, int speed)
return 0xffffffffU; /* silence compiler warning */ return 0xffffffffU; /* silence compiler warning */
} }
static int hpt_dma_blacklisted(const struct ata_device *dev, char *modestr, static int hpt_dma_broken(const struct ata_device *dev, char *modestr,
const char * const list[]) const char * const list[])
{ {
unsigned char model_num[ATA_ID_PROD_LEN + 1]; unsigned char model_num[ATA_ID_PROD_LEN + 1];
int i; int i;
...@@ -281,9 +281,9 @@ static const char * const bad_ata100_5[] = { ...@@ -281,9 +281,9 @@ static const char * const bad_ata100_5[] = {
static unsigned int hpt370_filter(struct ata_device *adev, unsigned int mask) static unsigned int hpt370_filter(struct ata_device *adev, unsigned int mask)
{ {
if (adev->class == ATA_DEV_ATA) { if (adev->class == ATA_DEV_ATA) {
if (hpt_dma_blacklisted(adev, "UDMA", bad_ata33)) if (hpt_dma_broken(adev, "UDMA", bad_ata33))
mask &= ~ATA_MASK_UDMA; mask &= ~ATA_MASK_UDMA;
if (hpt_dma_blacklisted(adev, "UDMA100", bad_ata100_5)) if (hpt_dma_broken(adev, "UDMA100", bad_ata100_5))
mask &= ~(0xE0 << ATA_SHIFT_UDMA); mask &= ~(0xE0 << ATA_SHIFT_UDMA);
} }
return mask; return mask;
...@@ -300,7 +300,7 @@ static unsigned int hpt370_filter(struct ata_device *adev, unsigned int mask) ...@@ -300,7 +300,7 @@ static unsigned int hpt370_filter(struct ata_device *adev, unsigned int mask)
static unsigned int hpt370a_filter(struct ata_device *adev, unsigned int mask) static unsigned int hpt370a_filter(struct ata_device *adev, unsigned int mask)
{ {
if (adev->class == ATA_DEV_ATA) { if (adev->class == ATA_DEV_ATA) {
if (hpt_dma_blacklisted(adev, "UDMA100", bad_ata100_5)) if (hpt_dma_broken(adev, "UDMA100", bad_ata100_5))
mask &= ~(0xE0 << ATA_SHIFT_UDMA); mask &= ~(0xE0 << ATA_SHIFT_UDMA);
} }
return mask; return mask;
......
...@@ -328,8 +328,6 @@ static void pata_icside_postreset(struct ata_link *link, unsigned int *classes) ...@@ -328,8 +328,6 @@ static void pata_icside_postreset(struct ata_link *link, unsigned int *classes)
static struct ata_port_operations pata_icside_port_ops = { static struct ata_port_operations pata_icside_port_ops = {
.inherits = &ata_bmdma_port_ops, .inherits = &ata_bmdma_port_ops,
/* no need to build any PRD tables for DMA */
.qc_prep = ata_noop_qc_prep,
.sff_data_xfer = ata_sff_data_xfer32, .sff_data_xfer = ata_sff_data_xfer32,
.bmdma_setup = pata_icside_bmdma_setup, .bmdma_setup = pata_icside_bmdma_setup,
.bmdma_start = pata_icside_bmdma_start, .bmdma_start = pata_icside_bmdma_start,
......
...@@ -519,9 +519,9 @@ static void it821x_dev_config(struct ata_device *adev) ...@@ -519,9 +519,9 @@ static void it821x_dev_config(struct ata_device *adev)
} }
/* This is a controller firmware triggered funny, don't /* This is a controller firmware triggered funny, don't
report the drive faulty! */ report the drive faulty! */
adev->horkage &= ~ATA_HORKAGE_DIAGNOSTIC; adev->quirks &= ~ATA_QUIRK_DIAGNOSTIC;
/* No HPA in 'smart' mode */ /* No HPA in 'smart' mode */
adev->horkage |= ATA_HORKAGE_BROKEN_HPA; adev->quirks |= ATA_QUIRK_BROKEN_HPA;
} }
/** /**
......
...@@ -290,6 +290,7 @@ static const struct of_device_id ixp4xx_pata_of_match[] = { ...@@ -290,6 +290,7 @@ static const struct of_device_id ixp4xx_pata_of_match[] = {
{ .compatible = "intel,ixp4xx-compact-flash", }, { .compatible = "intel,ixp4xx-compact-flash", },
{ /* sentinel */ } { /* sentinel */ }
}; };
MODULE_DEVICE_TABLE(of, ixp4xx_pata_of_match);
static struct platform_driver ixp4xx_pata_platform_driver = { static struct platform_driver ixp4xx_pata_platform_driver = {
.driver = { .driver = {
......
...@@ -620,7 +620,6 @@ static struct ata_port_operations mpc52xx_ata_port_ops = { ...@@ -620,7 +620,6 @@ static struct ata_port_operations mpc52xx_ata_port_ops = {
.bmdma_start = mpc52xx_bmdma_start, .bmdma_start = mpc52xx_bmdma_start,
.bmdma_stop = mpc52xx_bmdma_stop, .bmdma_stop = mpc52xx_bmdma_stop,
.bmdma_status = mpc52xx_bmdma_status, .bmdma_status = mpc52xx_bmdma_status,
.qc_prep = ata_noop_qc_prep,
}; };
static int mpc52xx_ata_init_one(struct device *dev, static int mpc52xx_ata_init_one(struct device *dev,
......
...@@ -789,7 +789,6 @@ static unsigned int octeon_cf_qc_issue(struct ata_queued_cmd *qc) ...@@ -789,7 +789,6 @@ static unsigned int octeon_cf_qc_issue(struct ata_queued_cmd *qc)
static struct ata_port_operations octeon_cf_ops = { static struct ata_port_operations octeon_cf_ops = {
.inherits = &ata_sff_port_ops, .inherits = &ata_sff_port_ops,
.check_atapi_dma = octeon_cf_check_atapi_dma, .check_atapi_dma = octeon_cf_check_atapi_dma,
.qc_prep = ata_noop_qc_prep,
.qc_issue = octeon_cf_qc_issue, .qc_issue = octeon_cf_qc_issue,
.sff_dev_select = octeon_cf_dev_select, .sff_dev_select = octeon_cf_dev_select,
.sff_irq_on = octeon_cf_ata_port_noaction, .sff_irq_on = octeon_cf_ata_port_noaction,
......
...@@ -46,10 +46,11 @@ ...@@ -46,10 +46,11 @@
#define SVWKS_CSB5_REVISION_NEW 0x92 /* min PCI_REVISION_ID for UDMA5 (A2.0) */ #define SVWKS_CSB5_REVISION_NEW 0x92 /* min PCI_REVISION_ID for UDMA5 (A2.0) */
#define SVWKS_CSB6_REVISION 0xa0 /* min PCI_REVISION_ID for UDMA4 (A1.0) */ #define SVWKS_CSB6_REVISION 0xa0 /* min PCI_REVISION_ID for UDMA4 (A1.0) */
/* Seagate Barracuda ATA IV Family drives in UDMA mode 5 /*
* can overrun their FIFOs when used with the CSB5 */ * Seagate Barracuda ATA IV Family drives in UDMA mode 5
* can overrun their FIFOs when used with the CSB5.
static const char *csb_bad_ata100[] = { */
static const char * const csb_bad_ata100[] = {
"ST320011A", "ST320011A",
"ST340016A", "ST340016A",
"ST360021A", "ST360021A",
...@@ -163,10 +164,11 @@ static unsigned int serverworks_osb4_filter(struct ata_device *adev, unsigned in ...@@ -163,10 +164,11 @@ static unsigned int serverworks_osb4_filter(struct ata_device *adev, unsigned in
* @adev: ATA device * @adev: ATA device
* @mask: Mask of proposed modes * @mask: Mask of proposed modes
* *
* Check the blacklist and disable UDMA5 if matched * Check the list of devices with broken UDMA5 and
* disable UDMA5 if matched.
*/ */
static unsigned int serverworks_csb_filter(struct ata_device *adev,
static unsigned int serverworks_csb_filter(struct ata_device *adev, unsigned int mask) unsigned int mask)
{ {
const char *p; const char *p;
char model_num[ATA_ID_PROD_LEN + 1]; char model_num[ATA_ID_PROD_LEN + 1];
......
...@@ -417,6 +417,7 @@ static const struct of_device_id gemini_sata_of_match[] = { ...@@ -417,6 +417,7 @@ static const struct of_device_id gemini_sata_of_match[] = {
{ .compatible = "cortina,gemini-sata-bridge", }, { .compatible = "cortina,gemini-sata-bridge", },
{ /* sentinel */ } { /* sentinel */ }
}; };
MODULE_DEVICE_TABLE(of, gemini_sata_of_match);
static struct platform_driver gemini_sata_driver = { static struct platform_driver gemini_sata_driver = {
.driver = { .driver = {
......
...@@ -128,7 +128,7 @@ static const struct pci_device_id sil_pci_tbl[] = { ...@@ -128,7 +128,7 @@ static const struct pci_device_id sil_pci_tbl[] = {
static const struct sil_drivelist { static const struct sil_drivelist {
const char *product; const char *product;
unsigned int quirk; unsigned int quirk;
} sil_blacklist [] = { } sil_quirks[] = {
{ "ST320012AS", SIL_QUIRK_MOD15WRITE }, { "ST320012AS", SIL_QUIRK_MOD15WRITE },
{ "ST330013AS", SIL_QUIRK_MOD15WRITE }, { "ST330013AS", SIL_QUIRK_MOD15WRITE },
{ "ST340017AS", SIL_QUIRK_MOD15WRITE }, { "ST340017AS", SIL_QUIRK_MOD15WRITE },
...@@ -600,8 +600,8 @@ static void sil_thaw(struct ata_port *ap) ...@@ -600,8 +600,8 @@ static void sil_thaw(struct ata_port *ap)
* list, and apply the fixups to only the specific * list, and apply the fixups to only the specific
* devices/hosts/firmwares that need it. * devices/hosts/firmwares that need it.
* *
* 20040111 - Seagate drives affected by the Mod15Write bug are blacklisted * 20040111 - Seagate drives affected by the Mod15Write bug are quirked
* The Maxtor quirk is in the blacklist, but I'm keeping the original * The Maxtor quirk is in sil_quirks, but I'm keeping the original
* pessimistic fix for the following reasons... * pessimistic fix for the following reasons...
* - There seems to be less info on it, only one device gleaned off the * - There seems to be less info on it, only one device gleaned off the
* Windows driver, maybe only one is affected. More info would be greatly * Windows driver, maybe only one is affected. More info would be greatly
...@@ -616,13 +616,13 @@ static void sil_dev_config(struct ata_device *dev) ...@@ -616,13 +616,13 @@ static void sil_dev_config(struct ata_device *dev)
unsigned char model_num[ATA_ID_PROD_LEN + 1]; unsigned char model_num[ATA_ID_PROD_LEN + 1];
/* This controller doesn't support trim */ /* This controller doesn't support trim */
dev->horkage |= ATA_HORKAGE_NOTRIM; dev->quirks |= ATA_QUIRK_NOTRIM;
ata_id_c_string(dev->id, model_num, ATA_ID_PROD, sizeof(model_num)); ata_id_c_string(dev->id, model_num, ATA_ID_PROD, sizeof(model_num));
for (n = 0; sil_blacklist[n].product; n++) for (n = 0; sil_quirks[n].product; n++)
if (!strcmp(sil_blacklist[n].product, model_num)) { if (!strcmp(sil_quirks[n].product, model_num)) {
quirks = sil_blacklist[n].quirk; quirks = sil_quirks[n].quirk;
break; break;
} }
......
...@@ -564,7 +564,6 @@ static struct ata_port_operations sas_sata_ops = { ...@@ -564,7 +564,6 @@ static struct ata_port_operations sas_sata_ops = {
.error_handler = ata_std_error_handler, .error_handler = ata_std_error_handler,
.post_internal_cmd = sas_ata_post_internal, .post_internal_cmd = sas_ata_post_internal,
.qc_defer = ata_std_qc_defer, .qc_defer = ata_std_qc_defer,
.qc_prep = ata_noop_qc_prep,
.qc_issue = sas_ata_qc_issue, .qc_issue = sas_ata_qc_issue,
.qc_fill_rtf = sas_ata_qc_fill_rtf, .qc_fill_rtf = sas_ata_qc_fill_rtf,
.set_dmamode = sas_ata_set_dmamode, .set_dmamode = sas_ata_set_dmamode,
......
This diff is collapsed.
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