static void ahci_thaw(struct ata_port *ap);
 static void ahci_pmp_attach(struct ata_port *ap);
 static void ahci_pmp_detach(struct ata_port *ap);
-static int ahci_pmp_read(struct ata_device *dev, int pmp, int reg, u32 *r_val);
-static int ahci_pmp_write(struct ata_device *dev, int pmp, int reg, u32 val);
 static void ahci_error_handler(struct ata_port *ap);
 static void ahci_vt8251_error_handler(struct ata_port *ap);
 static void ahci_post_internal_cmd(struct ata_queued_cmd *qc);
 
        .pmp_attach             = ahci_pmp_attach,
        .pmp_detach             = ahci_pmp_detach,
-       .pmp_read               = ahci_pmp_read,
-       .pmp_write              = ahci_pmp_write,
 
 #ifdef CONFIG_PM
        .port_suspend           = ahci_port_suspend,
 
        .pmp_attach             = ahci_pmp_attach,
        .pmp_detach             = ahci_pmp_detach,
-       .pmp_read               = ahci_pmp_read,
-       .pmp_write              = ahci_pmp_write,
 
 #ifdef CONFIG_PM
        .port_suspend           = ahci_port_suspend,
        struct ata_eh_info *ehi = &ap->link.eh_info;
        struct ahci_port_priv *pp = ap->private_data;
        struct ahci_host_priv *hpriv = ap->host->private_data;
+       int resetting = !!(ap->pflags & ATA_PFLAG_RESETTING);
        u32 status, qc_active;
        int rc, known_irq = 0;
 
        status = readl(port_mmio + PORT_IRQ_STAT);
        writel(status, port_mmio + PORT_IRQ_STAT);
 
+       /* ignore BAD_PMP while resetting */
+       if (unlikely(resetting))
+               status &= ~PORT_IRQ_BAD_PMP;
+
        if (unlikely(status & PORT_IRQ_ERROR)) {
                ahci_error_intr(ap, status);
                return;
                qc_active = readl(port_mmio + PORT_CMD_ISSUE);
 
        rc = ata_qc_complete_multiple(ap, qc_active, NULL);
+
+       /* If resetting, spurious or invalid completions are expected,
+        * return unconditionally.
+        */
+       if (resetting)
+               return;
+
        if (rc > 0)
                return;
        if (rc < 0) {
        writel(pp->intr_mask, port_mmio + PORT_IRQ_MASK);
 }
 
-static int ahci_pmp_read(struct ata_device *dev, int pmp, int reg, u32 *r_val)
-{
-       struct ata_port *ap = dev->link->ap;
-       struct ata_taskfile tf;
-       int rc;
-
-       ahci_kick_engine(ap, 0);
-
-       sata_pmp_read_init_tf(&tf, dev, pmp, reg);
-       rc = ahci_exec_polled_cmd(ap, SATA_PMP_CTRL_PORT, &tf, 1, 0,
-                                 SATA_PMP_SCR_TIMEOUT);
-       if (rc == 0) {
-               ahci_tf_read(ap, &tf);
-               *r_val = sata_pmp_read_val(&tf);
-       }
-       return rc;
-}
-
-static int ahci_pmp_write(struct ata_device *dev, int pmp, int reg, u32 val)
-{
-       struct ata_port *ap = dev->link->ap;
-       struct ata_taskfile tf;
-
-       ahci_kick_engine(ap, 0);
-
-       sata_pmp_write_init_tf(&tf, dev, pmp, reg, val);
-       return ahci_exec_polled_cmd(ap, SATA_PMP_CTRL_PORT, &tf, 1, 0,
-                                   SATA_PMP_SCR_TIMEOUT);
-}
-
 static int ahci_port_resume(struct ata_port *ap)
 {
        ahci_power_up(ap);
 
  *     @reg: register to read
  *     @r_val: resulting value
  *
- *     Wrapper around ap->ops->pmp_read to make it easier to call and
- *     nomarlize error return value.
+ *     Read PMP register.
  *
  *     LOCKING:
  *     Kernel thread context (may sleep).
  *
  *     RETURNS:
- *     0 on success, -errno on failure.
+ *     0 on success, AC_ERR_* mask on failure.
  */
-static int sata_pmp_read(struct ata_link *link, int reg, u32 *r_val)
+static unsigned int sata_pmp_read(struct ata_link *link, int reg, u32 *r_val)
 {
        struct ata_port *ap = link->ap;
        struct ata_device *pmp_dev = ap->link.device;
-       int rc;
-
-       might_sleep();
-
-       rc = ap->ops->pmp_read(pmp_dev, link->pmp, reg, r_val);
-       if (rc)
-               rc = -EIO;
-       return rc;
+       struct ata_taskfile tf;
+       unsigned int err_mask;
+
+       ata_tf_init(pmp_dev, &tf);
+       tf.command = ATA_CMD_PMP_READ;
+       tf.protocol = ATA_PROT_NODATA;
+       tf.flags |= ATA_TFLAG_ISADDR | ATA_TFLAG_DEVICE;
+       tf.feature = reg;
+       tf.device = link->pmp;
+
+       err_mask = ata_exec_internal(pmp_dev, &tf, NULL, DMA_NONE, NULL, 0,
+                                    SATA_PMP_SCR_TIMEOUT);
+       if (err_mask)
+               return err_mask;
+
+       *r_val = tf.nsect | tf.lbal << 8 | tf.lbam << 16 | tf.lbah << 24;
+       return 0;
 }
 
 /**
  *     @reg: register to write
  *     @r_val: value to write
  *
- *     Wrapper around ap->ops->pmp_write to make it easier to call
- *     and nomarlize error return value.
+ *     Write PMP register.
  *
  *     LOCKING:
  *     Kernel thread context (may sleep).
  *
  *     RETURNS:
- *     0 on success, -errno on failure.
+ *     0 on success, AC_ERR_* mask on failure.
  */
-static int sata_pmp_write(struct ata_link *link, int reg, u32 val)
+static unsigned int sata_pmp_write(struct ata_link *link, int reg, u32 val)
 {
        struct ata_port *ap = link->ap;
        struct ata_device *pmp_dev = ap->link.device;
-       int rc;
-
-       might_sleep();
-
-       rc = ap->ops->pmp_write(pmp_dev, link->pmp, reg, val);
-       if (rc)
-               rc = -EIO;
-       return rc;
+       struct ata_taskfile tf;
+
+       ata_tf_init(pmp_dev, &tf);
+       tf.command = ATA_CMD_PMP_WRITE;
+       tf.protocol = ATA_PROT_NODATA;
+       tf.flags |= ATA_TFLAG_ISADDR | ATA_TFLAG_DEVICE;
+       tf.feature = reg;
+       tf.device = link->pmp;
+       tf.nsect = val & 0xff;
+       tf.lbal = (val >> 8) & 0xff;
+       tf.lbam = (val >> 16) & 0xff;
+       tf.lbah = (val >> 24) & 0xff;
+
+       return ata_exec_internal(pmp_dev, &tf, NULL, DMA_NONE, NULL, 0,
+                                SATA_PMP_SCR_TIMEOUT);
 }
 
 /**
        return ATA_DEFER_PORT;
 }
 
-/**
- *     sata_pmp_read_init_tf - initialize TF for PMP read
- *     @tf: taskfile to initialize
- *     @dev: PMP dev
- *     @pmp: port multiplier port number
- *     @reg: register to read
- *
- *     Initialize @tf for PMP read command.
- *
- *     LOCKING:
- *     None.
- */
-void sata_pmp_read_init_tf(struct ata_taskfile *tf,
-                          struct ata_device *dev, int pmp, int reg)
-{
-       ata_tf_init(dev, tf);
-       tf->command = ATA_CMD_PMP_READ;
-       tf->protocol = ATA_PROT_NODATA;
-       tf->flags |= ATA_TFLAG_ISADDR | ATA_TFLAG_DEVICE;
-       tf->feature = reg;
-       tf->device = pmp;
-}
-
-/**
- *     sata_pmp_read_val - extract PMP read result from TF
- *     @tf: target TF
- *
- *     Determine PMP read result from @tf.
- *
- *     LOCKING:
- *     None.
- */
-u32 sata_pmp_read_val(const struct ata_taskfile *tf)
-{
-       return tf->nsect | tf->lbal << 8 | tf->lbam << 16 | tf->lbah << 24;
-}
-
-/**
- *     sata_pmp_read_init_tf - initialize TF for PMP write
- *     @tf: taskfile to initialize
- *     @dev: PMP dev
- *     @pmp: port multiplier port number
- *     @reg: register to read
- *     @val: value to write
- *
- *     Initialize @tf for PMP write command.
- *
- *     LOCKING:
- *     None.
- */
-void sata_pmp_write_init_tf(struct ata_taskfile *tf,
-                           struct ata_device *dev, int pmp, int reg, u32 val)
-{
-       ata_tf_init(dev, tf);
-       tf->command = ATA_CMD_PMP_WRITE;
-       tf->protocol = ATA_PROT_NODATA;
-       tf->flags |= ATA_TFLAG_ISADDR | ATA_TFLAG_DEVICE;
-       tf->feature = reg;
-       tf->device = pmp;
-       tf->nsect = val & 0xff;
-       tf->lbal = (val >> 8) & 0xff;
-       tf->lbam = (val >> 16) & 0xff;
-       tf->lbah = (val >> 24) & 0xff;
-}
-
 /**
  *     sata_pmp_scr_read - read PSCR
  *     @link: ATA link to read PSCR for
  */
 int sata_pmp_scr_read(struct ata_link *link, int reg, u32 *r_val)
 {
+       unsigned int err_mask;
+
        if (reg > SATA_PMP_PSCR_CONTROL)
                return -EINVAL;
 
-       return sata_pmp_read(link, reg, r_val);
+       err_mask = sata_pmp_read(link, reg, r_val);
+       if (err_mask) {
+               ata_link_printk(link, KERN_WARNING, "failed to read SCR %d "
+                               "(Emask=0x%x)\n", reg, err_mask);
+               return -EIO;
+       }
+       return 0;
 }
 
 /**
  */
 int sata_pmp_scr_write(struct ata_link *link, int reg, u32 val)
 {
+       unsigned int err_mask;
+
        if (reg > SATA_PMP_PSCR_CONTROL)
                return -EINVAL;
 
-       return sata_pmp_write(link, reg, val);
+       err_mask = sata_pmp_write(link, reg, val);
+       if (err_mask) {
+               ata_link_printk(link, KERN_WARNING, "failed to write SCR %d "
+                               "(Emask=0x%x)\n", reg, err_mask);
+               return -EIO;
+       }
+       return 0;
 }
 
 /**
 static int sata_pmp_read_gscr(struct ata_device *dev, u32 *gscr)
 {
        static const int gscr_to_read[] = { 0, 1, 2, 32, 33, 64, 96 };
-       int i, rc;
+       int i;
 
        for (i = 0; i < ARRAY_SIZE(gscr_to_read); i++) {
                int reg = gscr_to_read[i];
+               unsigned int err_mask;
 
-               rc = sata_pmp_read(dev->link, reg, &gscr[reg]);
-               if (rc) {
-                       ata_dev_printk(dev, KERN_ERR, "failed to read "
-                                      "PMP GSCR[%d] (errno=%d)\n", reg, rc);
-                       return rc;
+               err_mask = sata_pmp_read(dev->link, reg, &gscr[reg]);
+               if (err_mask) {
+                       ata_dev_printk(dev, KERN_ERR, "failed to read PMP "
+                               "GSCR[%d] (Emask=0x%x)\n", reg, err_mask);
+                       return -EIO;
                }
        }
 
 {
        struct ata_port *ap = dev->link->ap;
        u32 *gscr = dev->gscr;
+       unsigned int err_mask = 0;
        const char *reason;
        int nr_ports, rc;
 
                dev->flags |= ATA_DFLAG_AN;
 
        /* monitor SERR_PHYRDY_CHG on fan-out ports */
-       rc = sata_pmp_write(dev->link, SATA_PMP_GSCR_ERROR_EN, SERR_PHYRDY_CHG);
-       if (rc) {
+       err_mask = sata_pmp_write(dev->link, SATA_PMP_GSCR_ERROR_EN,
+                                 SERR_PHYRDY_CHG);
+       if (err_mask) {
+               rc = -EIO;
                reason = "failed to write GSCR_ERROR_EN";
                goto fail;
        }
        if (gscr[SATA_PMP_GSCR_FEAT_EN] & SATA_PMP_FEAT_NOTIFY) {
                gscr[SATA_PMP_GSCR_FEAT_EN] &= ~SATA_PMP_FEAT_NOTIFY;
 
-               rc = sata_pmp_write(dev->link, SATA_PMP_GSCR_FEAT_EN,
-                                   gscr[SATA_PMP_GSCR_FEAT_EN]);
-               if (rc) {
+               err_mask = sata_pmp_write(dev->link, SATA_PMP_GSCR_FEAT_EN,
+                                         gscr[SATA_PMP_GSCR_FEAT_EN]);
+               if (err_mask) {
+                       rc = -EIO;
                        reason = "failed to write GSCR_FEAT_EN";
                        goto fail;
                }
 
  fail:
        ata_dev_printk(dev, KERN_ERR,
-                      "failed to configure Port Multiplier (%s)\n", reason);
+                      "failed to configure Port Multiplier (%s, Emask=0x%x)\n",
+                      reason, err_mask);
        return rc;
 }
 
  */
 static int sata_pmp_revalidate_quick(struct ata_device *dev)
 {
+       unsigned int err_mask;
        u32 prod_id;
-       int rc;
 
-       rc = sata_pmp_read(dev->link, SATA_PMP_GSCR_PROD_ID, &prod_id);
-       if (rc) {
-               ata_dev_printk(dev, KERN_ERR, "failed to read PMP product ID\n");
-               return rc;
+       err_mask = sata_pmp_read(dev->link, SATA_PMP_GSCR_PROD_ID, &prod_id);
+       if (err_mask) {
+               ata_dev_printk(dev, KERN_ERR, "failed to read PMP product ID "
+                              "(Emask=0x%x)\n", err_mask);
+               return -EIO;
        }
 
        if (prod_id != dev->gscr[SATA_PMP_GSCR_PROD_ID]) {
        struct ata_eh_context *pmp_ehc = &pmp_link->eh_context;
        struct ata_link *link;
        struct ata_device *dev;
+       unsigned int err_mask;
        u32 gscr_error, sntf;
        int cnt, rc;
 
        if (pmp_dev->flags & ATA_DFLAG_AN) {
                pmp_dev->gscr[SATA_PMP_GSCR_FEAT_EN] |= SATA_PMP_FEAT_NOTIFY;
 
-               rc = sata_pmp_write(pmp_dev->link, SATA_PMP_GSCR_FEAT_EN,
-                                   pmp_dev->gscr[SATA_PMP_GSCR_FEAT_EN]);
-               if (rc) {
-                       ata_dev_printk(pmp_dev, KERN_ERR,
-                                      "failed to write PMP_FEAT_EN\n");
+               err_mask = sata_pmp_write(pmp_dev->link, SATA_PMP_GSCR_FEAT_EN,
+                                         pmp_dev->gscr[SATA_PMP_GSCR_FEAT_EN]);
+               if (err_mask) {
+                       ata_dev_printk(pmp_dev, KERN_ERR, "failed to write "
+                                      "PMP_FEAT_EN (Emask=0x%x)\n", err_mask);
+                       rc = -EIO;
                        goto pmp_fail;
                }
        }
 
        /* check GSCR_ERROR */
-       rc = sata_pmp_read(pmp_link, SATA_PMP_GSCR_ERROR, &gscr_error);
-       if (rc) {
-               ata_dev_printk(pmp_dev, KERN_ERR,
-                              "failed to read PMP_GSCR_ERROR\n");
+       err_mask = sata_pmp_read(pmp_link, SATA_PMP_GSCR_ERROR, &gscr_error);
+       if (err_mask) {
+               ata_dev_printk(pmp_dev, KERN_ERR, "failed to read "
+                              "PMP_GSCR_ERROR (Emask=0x%x)\n", err_mask);
+               rc = -EIO;
                goto pmp_fail;
        }
 
 
 static void sil24_irq_clear(struct ata_port *ap);
 static void sil24_pmp_attach(struct ata_port *ap);
 static void sil24_pmp_detach(struct ata_port *ap);
-static int sil24_pmp_read(struct ata_device *dev, int pmp, int reg, u32 *r_val);
-static int sil24_pmp_write(struct ata_device *dev, int pmp, int reg, u32 val);
 static void sil24_freeze(struct ata_port *ap);
 static void sil24_thaw(struct ata_port *ap);
 static void sil24_error_handler(struct ata_port *ap);
 
        .pmp_attach             = sil24_pmp_attach,
        .pmp_detach             = sil24_pmp_detach,
-       .pmp_read               = sil24_pmp_read,
-       .pmp_write              = sil24_pmp_write,
 
        .freeze                 = sil24_freeze,
        .thaw                   = sil24_thaw,
        sil24_config_pmp(ap, 0);
 }
 
-static int sil24_pmp_read(struct ata_device *dev, int pmp, int reg, u32 *r_val)
-{
-       struct ata_port *ap = dev->link->ap;
-       struct ata_taskfile tf;
-       int rc;
-
-       sata_pmp_read_init_tf(&tf, dev, pmp, reg);
-       rc = sil24_exec_polled_cmd(ap, SATA_PMP_CTRL_PORT, &tf, 1, 0,
-                                  SATA_PMP_SCR_TIMEOUT);
-       if (rc == 0) {
-               sil24_read_tf(ap, 0, &tf);
-               *r_val = sata_pmp_read_val(&tf);
-       }
-       return rc;
-}
-
-static int sil24_pmp_write(struct ata_device *dev, int pmp, int reg, u32 val)
-{
-       struct ata_port *ap = dev->link->ap;
-       struct ata_taskfile tf;
-
-       sata_pmp_write_init_tf(&tf, dev, pmp, reg, val);
-       return sil24_exec_polled_cmd(ap, SATA_PMP_CTRL_PORT, &tf, 1, 0,
-                                    SATA_PMP_SCR_TIMEOUT);
-}
-
 static int sil24_pmp_softreset(struct ata_link *link, unsigned int *class,
                               unsigned long deadline)
 {