ASoC: dapm: Allow routes to be deleted at runtime
authorMark Brown <broonie@opensource.wolfsonmicro.com>
Thu, 5 Jul 2012 16:24:19 +0000 (17:24 +0100)
committerMark Brown <broonie@opensource.wolfsonmicro.com>
Fri, 6 Jul 2012 18:06:59 +0000 (19:06 +0100)
Since we're now relying on DAPM for things like enabling clocks when we
reparent the clocks for widgets we need to either use conditional routes
(which are expensive) or remove routes at runtime. Add a route removal
API to support this use case.

Signed-off-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
Acked-by: Liam Girdwood <lrg@ti.com>
include/sound/soc-dapm.h
sound/soc/soc-dapm.c

index 05559e571d44d78e09df8e2ea1f42961f71274fb..abe373d57adca91889784d9b1a69b27cb59b662f 100644 (file)
@@ -374,6 +374,8 @@ int snd_soc_dapm_new_widgets(struct snd_soc_dapm_context *dapm);
 void snd_soc_dapm_free(struct snd_soc_dapm_context *dapm);
 int snd_soc_dapm_add_routes(struct snd_soc_dapm_context *dapm,
                            const struct snd_soc_dapm_route *route, int num);
+int snd_soc_dapm_del_routes(struct snd_soc_dapm_context *dapm,
+                           const struct snd_soc_dapm_route *route, int num);
 int snd_soc_dapm_weak_routes(struct snd_soc_dapm_context *dapm,
                             const struct snd_soc_dapm_route *route, int num);
 
index 19fda1339510b612e55827ba8032371edd34b9fd..4ba47aab980156b29562b77a969491fbf026ae1d 100644 (file)
@@ -2264,6 +2264,59 @@ err:
        return ret;
 }
 
+static int snd_soc_dapm_del_route(struct snd_soc_dapm_context *dapm,
+                                 const struct snd_soc_dapm_route *route)
+{
+       struct snd_soc_dapm_path *path, *p;
+       const char *sink;
+       const char *source;
+       char prefixed_sink[80];
+       char prefixed_source[80];
+
+       if (route->control) {
+               dev_err(dapm->dev,
+                       "Removal of routes with controls not supported\n");
+               return -EINVAL;
+       }
+
+       if (dapm->codec && dapm->codec->name_prefix) {
+               snprintf(prefixed_sink, sizeof(prefixed_sink), "%s %s",
+                        dapm->codec->name_prefix, route->sink);
+               sink = prefixed_sink;
+               snprintf(prefixed_source, sizeof(prefixed_source), "%s %s",
+                        dapm->codec->name_prefix, route->source);
+               source = prefixed_source;
+       } else {
+               sink = route->sink;
+               source = route->source;
+       }
+
+       path = NULL;
+       list_for_each_entry(p, &dapm->card->paths, list) {
+               if (strcmp(p->source->name, source) != 0)
+                       continue;
+               if (strcmp(p->sink->name, sink) != 0)
+                       continue;
+               path = p;
+               break;
+       }
+
+       if (path) {
+               dapm_mark_dirty(path->source, "Route removed");
+               dapm_mark_dirty(path->sink, "Route removed");
+
+               list_del(&path->list);
+               list_del(&path->list_sink);
+               list_del(&path->list_source);
+               kfree(path);
+       } else {
+               dev_warn(dapm->dev, "Route %s->%s does not exist\n",
+                        source, sink);
+       }
+
+       return 0;
+}
+
 /**
  * snd_soc_dapm_add_routes - Add routes between DAPM widgets
  * @dapm: DAPM context
@@ -2298,6 +2351,30 @@ int snd_soc_dapm_add_routes(struct snd_soc_dapm_context *dapm,
 }
 EXPORT_SYMBOL_GPL(snd_soc_dapm_add_routes);
 
+/**
+ * snd_soc_dapm_del_routes - Remove routes between DAPM widgets
+ * @dapm: DAPM context
+ * @route: audio routes
+ * @num: number of routes
+ *
+ * Removes routes from the DAPM context.
+ */
+int snd_soc_dapm_del_routes(struct snd_soc_dapm_context *dapm,
+                           const struct snd_soc_dapm_route *route, int num)
+{
+       int i, ret = 0;
+
+       mutex_lock_nested(&dapm->card->dapm_mutex, SND_SOC_DAPM_CLASS_INIT);
+       for (i = 0; i < num; i++) {
+               snd_soc_dapm_del_route(dapm, route);
+               route++;
+       }
+       mutex_unlock(&dapm->card->dapm_mutex);
+
+       return ret;
+}
+EXPORT_SYMBOL_GPL(snd_soc_dapm_del_routes);
+
 static int snd_soc_dapm_weak_route(struct snd_soc_dapm_context *dapm,
                                   const struct snd_soc_dapm_route *route)
 {