]> git.karo-electronics.de Git - karo-tx-linux.git/blobdiff - drivers/ata/sata_fsl.c
Merge tag 'upstream-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/jgarzik...
[karo-tx-linux.git] / drivers / ata / sata_fsl.c
index dfc6fd0ac119668ff3fb351fcc4d4ac19a841519..d40e403e82dd259a34ab6a155a88619ba3b3542a 100644 (file)
@@ -285,6 +285,7 @@ struct sata_fsl_host_priv {
        int irq;
        int data_snoop;
        struct device_attribute intr_coalescing;
+       struct device_attribute rx_watermark;
 };
 
 static void fsl_sata_set_irq_coalescing(struct ata_host *host,
@@ -343,6 +344,48 @@ static ssize_t fsl_sata_intr_coalescing_store(struct device *dev,
        return strlen(buf);
 }
 
+static ssize_t fsl_sata_rx_watermark_show(struct device *dev,
+               struct device_attribute *attr, char *buf)
+{
+       unsigned int rx_watermark;
+       unsigned long flags;
+       struct ata_host *host = dev_get_drvdata(dev);
+       struct sata_fsl_host_priv *host_priv = host->private_data;
+       void __iomem *csr_base = host_priv->csr_base;
+
+       spin_lock_irqsave(&host->lock, flags);
+       rx_watermark = ioread32(csr_base + TRANSCFG);
+       rx_watermark &= 0x1f;
+
+       spin_unlock_irqrestore(&host->lock, flags);
+       return sprintf(buf, "%d\n", rx_watermark);
+}
+
+static ssize_t fsl_sata_rx_watermark_store(struct device *dev,
+               struct device_attribute *attr,
+               const char *buf, size_t count)
+{
+       unsigned int rx_watermark;
+       unsigned long flags;
+       struct ata_host *host = dev_get_drvdata(dev);
+       struct sata_fsl_host_priv *host_priv = host->private_data;
+       void __iomem *csr_base = host_priv->csr_base;
+       u32 temp;
+
+       if (sscanf(buf, "%d", &rx_watermark) != 1) {
+               printk(KERN_ERR "fsl-sata: wrong parameter format.\n");
+               return -EINVAL;
+       }
+
+       spin_lock_irqsave(&host->lock, flags);
+       temp = ioread32(csr_base + TRANSCFG);
+       temp &= 0xffffffe0;
+       iowrite32(temp | rx_watermark, csr_base + TRANSCFG);
+
+       spin_unlock_irqrestore(&host->lock, flags);
+       return strlen(buf);
+}
+
 static inline unsigned int sata_fsl_tag(unsigned int tag,
                                        void __iomem *hcr_base)
 {
@@ -1500,6 +1543,17 @@ static int sata_fsl_probe(struct platform_device *ofdev)
        if (retval)
                goto error_exit_with_cleanup;
 
+       host_priv->rx_watermark.show = fsl_sata_rx_watermark_show;
+       host_priv->rx_watermark.store = fsl_sata_rx_watermark_store;
+       sysfs_attr_init(&host_priv->rx_watermark.attr);
+       host_priv->rx_watermark.attr.name = "rx_watermark";
+       host_priv->rx_watermark.attr.mode = S_IRUGO | S_IWUSR;
+       retval = device_create_file(host->dev, &host_priv->rx_watermark);
+       if (retval) {
+               device_remove_file(&ofdev->dev, &host_priv->intr_coalescing);
+               goto error_exit_with_cleanup;
+       }
+
        return 0;
 
 error_exit_with_cleanup:
@@ -1522,6 +1576,7 @@ static int sata_fsl_remove(struct platform_device *ofdev)
        struct sata_fsl_host_priv *host_priv = host->private_data;
 
        device_remove_file(&ofdev->dev, &host_priv->intr_coalescing);
+       device_remove_file(&ofdev->dev, &host_priv->rx_watermark);
 
        ata_host_detach(host);