return 0;
 }
 
+/*
+ * Read/Write IO functions.
+ * no_pm versions can only be called by the bus, e.g. while enumerating or
+ * handling suspend-resume sequences.
+ * all clients need to use the pm versions
+ */
+
+static int
+sdw_nread_no_pm(struct sdw_slave *slave, u32 addr, size_t count, u8 *val)
+{
+       struct sdw_msg msg;
+       int ret;
+
+       ret = sdw_fill_msg(&msg, slave, addr, count,
+                          slave->dev_num, SDW_MSG_FLAG_READ, val);
+       if (ret < 0)
+               return ret;
+
+       return sdw_transfer(slave->bus, &msg);
+}
+
+static int
+sdw_nwrite_no_pm(struct sdw_slave *slave, u32 addr, size_t count, u8 *val)
+{
+       struct sdw_msg msg;
+       int ret;
+
+       ret = sdw_fill_msg(&msg, slave, addr, count,
+                          slave->dev_num, SDW_MSG_FLAG_WRITE, val);
+       if (ret < 0)
+               return ret;
+
+       return sdw_transfer(slave->bus, &msg);
+}
+
+static int sdw_write_no_pm(struct sdw_slave *slave, u32 addr, u8 value)
+{
+       return sdw_nwrite_no_pm(slave, addr, 1, &value);
+}
+
 /**
  * sdw_nread() - Read "n" contiguous SDW Slave registers
  * @slave: SDW Slave
  */
 int sdw_nread(struct sdw_slave *slave, u32 addr, size_t count, u8 *val)
 {
-       struct sdw_msg msg;
        int ret;
 
-       ret = sdw_fill_msg(&msg, slave, addr, count,
-                          slave->dev_num, SDW_MSG_FLAG_READ, val);
-       if (ret < 0)
-               return ret;
-
        ret = pm_runtime_get_sync(slave->bus->dev);
-       if (ret < 0)
+       if (ret < 0 && ret != -EACCES) {
+               pm_runtime_put_noidle(slave->bus->dev);
                return ret;
+       }
+
+       ret = sdw_nread_no_pm(slave, addr, count, val);
 
-       ret = sdw_transfer(slave->bus, &msg);
+       pm_runtime_mark_last_busy(slave->bus->dev);
        pm_runtime_put(slave->bus->dev);
 
        return ret;
  */
 int sdw_nwrite(struct sdw_slave *slave, u32 addr, size_t count, u8 *val)
 {
-       struct sdw_msg msg;
        int ret;
 
-       ret = sdw_fill_msg(&msg, slave, addr, count,
-                          slave->dev_num, SDW_MSG_FLAG_WRITE, val);
-       if (ret < 0)
-               return ret;
-
        ret = pm_runtime_get_sync(slave->bus->dev);
-       if (ret < 0)
+       if (ret < 0 && ret != -EACCES) {
+               pm_runtime_put_noidle(slave->bus->dev);
                return ret;
+       }
+
+       ret = sdw_nwrite_no_pm(slave, addr, count, val);
 
-       ret = sdw_transfer(slave->bus, &msg);
+       pm_runtime_mark_last_busy(slave->bus->dev);
        pm_runtime_put(slave->bus->dev);
 
        return ret;