static int coresight_enable_sink(struct coresight_device *csdev,
                                 enum cs_mode mode, void *data)
 {
-       int ret = sink_ops(csdev)->enable(csdev, mode, data);
-
-       if (ret)
-               return ret;
-
-       csdev->enable = true;
-
-       return 0;
+       return sink_ops(csdev)->enable(csdev, mode, data);
 }
 
 static void coresight_disable_sink(struct coresight_device *csdev)
 {
-       int ret = sink_ops(csdev)->disable(csdev);
-       if (ret)
-               return;
-       csdev->enable = false;
+       sink_ops(csdev)->disable(csdev);
 }
 
 static int coresight_enable_link(struct coresight_device *csdev,
                                 struct coresight_device *parent,
                                 struct coresight_device *child)
 {
-       int ret = 0;
        int link_subtype;
        struct coresight_connection *inconn, *outconn;
 
        if (link_subtype == CORESIGHT_DEV_SUBTYPE_LINK_SPLIT && IS_ERR(outconn))
                return PTR_ERR(outconn);
 
-       ret = link_ops(csdev)->enable(csdev, inconn, outconn);
-       if (!ret)
-               csdev->enable = true;
-
-       return ret;
+       return link_ops(csdev)->enable(csdev, inconn, outconn);
 }
 
 static void coresight_disable_link(struct coresight_device *csdev,
                                   struct coresight_device *parent,
                                   struct coresight_device *child)
 {
-       int i;
-       int link_subtype;
        struct coresight_connection *inconn, *outconn;
 
        if (!parent || !child)
 
        inconn = coresight_find_out_connection(parent, csdev);
        outconn = coresight_find_out_connection(csdev, child);
-       link_subtype = csdev->subtype.link_subtype;
 
        link_ops(csdev)->disable(csdev, inconn, outconn);
-
-       if (link_subtype == CORESIGHT_DEV_SUBTYPE_LINK_MERG) {
-               for (i = 0; i < csdev->pdata->nr_inconns; i++)
-                       if (atomic_read(&csdev->pdata->in_conns[i]->dest_refcnt) !=
-                           0)
-                               return;
-       } else if (link_subtype == CORESIGHT_DEV_SUBTYPE_LINK_SPLIT) {
-               for (i = 0; i < csdev->pdata->nr_outconns; i++)
-                       if (atomic_read(&csdev->pdata->out_conns[i]->src_refcnt) !=
-                           0)
-                               return;
-       } else {
-               if (atomic_read(&csdev->refcnt) != 0)
-                       return;
-       }
-
-       csdev->enable = false;
 }
 
 int coresight_enable_source(struct coresight_device *csdev, enum cs_mode mode,
 {
        int ret;
 
-       if (!csdev->enable) {
+       /*
+        * Comparison with CS_MODE_SYSFS works without taking any device
+        * specific spinlock because the truthyness of that comparison can only
+        * change with coresight_mutex held, which we already have here.
+        */
+       lockdep_assert_held(&coresight_mutex);
+       if (local_read(&csdev->mode) != CS_MODE_SYSFS) {
                ret = source_ops(csdev)->enable(csdev, data, mode);
                if (ret)
                        return ret;
-               csdev->enable = true;
        }
 
        atomic_inc(&csdev->refcnt);
 static int coresight_enable_helper(struct coresight_device *csdev,
                                   enum cs_mode mode, void *data)
 {
-       int ret = helper_ops(csdev)->enable(csdev, mode, data);
-
-       if (ret)
-               return ret;
-
-       csdev->enable = true;
-       return 0;
+       return helper_ops(csdev)->enable(csdev, mode, data);
 }
 
 static void coresight_disable_helper(struct coresight_device *csdev)
 {
-       int ret = helper_ops(csdev)->disable(csdev, NULL);
-
-       if (ret)
-               return;
-       csdev->enable = false;
+       helper_ops(csdev)->disable(csdev, NULL);
 }
 
 static void coresight_disable_helpers(struct coresight_device *csdev)
 static bool coresight_disable_source_sysfs(struct coresight_device *csdev,
                                           void *data)
 {
+       lockdep_assert_held(&coresight_mutex);
+       if (local_read(&csdev->mode) != CS_MODE_SYSFS)
+               return false;
+
        if (atomic_dec_return(&csdev->refcnt) == 0) {
                coresight_disable_source(csdev, data);
-               csdev->enable = false;
+               return true;
        }
-       return !csdev->enable;
+       return false;
 }
 
 /*
        if (ret)
                goto out;
 
-       if (csdev->enable) {
+       /*
+        * mode == SYSFS implies that it's already enabled. Don't look at the
+        * refcount to determine this because we don't claim the source until
+        * coresight_enable_source() so can still race with Perf mode which
+        * doesn't hold coresight_mutex.
+        */
+       if (local_read(&csdev->mode) == CS_MODE_SYSFS) {
                /*
                 * There could be multiple applications driving the software
                 * source. So keep the refcount for each such user when the
        if (ret)
                goto out;
 
-       if (!csdev->enable || !coresight_disable_source_sysfs(csdev, NULL))
+       if (!coresight_disable_source_sysfs(csdev, NULL))
                goto out;
 
        switch (csdev->subtype.source_subtype) {
 {
        struct coresight_device *csdev = to_coresight_device(dev);
 
-       return scnprintf(buf, PAGE_SIZE, "%u\n", csdev->enable);
+       guard(mutex)(&coresight_mutex);
+       return scnprintf(buf, PAGE_SIZE, "%u\n",
+                        local_read(&csdev->mode) == CS_MODE_SYSFS);
 }
 
 static ssize_t enable_source_store(struct device *dev,
 
 
        CS_UNLOCK(drvdata->base);
 
-       if (!drvdata->csdev->enable)
+       /*
+        * Only do pre-port enable for first port that calls enable when the
+        * device's main refcount is still 0
+        */
+       if (!atomic_read(&drvdata->csdev->refcnt))
                tpda_enable_pre_port(drvdata);
 
        ret = tpda_enable_port(drvdata, port);
                ret = __tpda_enable(drvdata, in->dest_port);
                if (!ret) {
                        atomic_inc(&in->dest_refcnt);
+                       atomic_inc(&csdev->refcnt);
                        dev_dbg(drvdata->dev, "TPDA inport %d enabled.\n", in->dest_port);
                }
        }
        struct tpda_drvdata *drvdata = dev_get_drvdata(csdev->dev.parent);
 
        spin_lock(&drvdata->spinlock);
-       if (atomic_dec_return(&in->dest_refcnt) == 0)
+       if (atomic_dec_return(&in->dest_refcnt) == 0) {
                __tpda_disable(drvdata, in->dest_port);
-
+               atomic_dec(&csdev->refcnt);
+       }
        spin_unlock(&drvdata->spinlock);
 
        dev_dbg(drvdata->dev, "TPDA inport %d disabled\n", in->dest_port);