if (reset_control_is_array(rstc))
                return reset_control_array_assert(rstc_to_array(rstc));
 
-       if (!rstc->rcdev->ops->assert)
-               return -ENOTSUPP;
-
        if (rstc->shared) {
                if (WARN_ON(atomic_read(&rstc->triggered_count) != 0))
                        return -EINVAL;
 
                if (atomic_dec_return(&rstc->deassert_count) != 0)
                        return 0;
+
+               /*
+                * Shared reset controls allow the reset line to be in any state
+                * after this call, so doing nothing is a valid option.
+                */
+               if (!rstc->rcdev->ops->assert)
+                       return 0;
+       } else {
+               /*
+                * If the reset controller does not implement .assert(), there
+                * is no way to guarantee that the reset line is asserted after
+                * this call.
+                */
+               if (!rstc->rcdev->ops->assert)
+                       return -ENOTSUPP;
        }
 
        return rstc->rcdev->ops->assert(rstc->rcdev, rstc->id);
        if (reset_control_is_array(rstc))
                return reset_control_array_deassert(rstc_to_array(rstc));
 
-       if (!rstc->rcdev->ops->deassert)
-               return -ENOTSUPP;
-
        if (rstc->shared) {
                if (WARN_ON(atomic_read(&rstc->triggered_count) != 0))
                        return -EINVAL;
                        return 0;
        }
 
+       /*
+        * If the reset controller does not implement .deassert(), we assume
+        * that it handles self-deasserting reset lines via .reset(). In that
+        * case, the reset lines are deasserted by default. If that is not the
+        * case, the reset controller driver should implement .deassert() and
+        * return -ENOTSUPP.
+        */
+       if (!rstc->rcdev->ops->deassert)
+               return 0;
+
        return rstc->rcdev->ops->deassert(rstc->rcdev, rstc->id);
 }
 EXPORT_SYMBOL_GPL(reset_control_deassert);