Quellcodebibliothek Statistik Leitseite products/sources/formale Sprachen/C/Linux/sound/soc/   (Open Source Betriebssystem Version 6.17.9©)  Datei vom 24.10.2025 mit Größe 121 kB image not shown  

Quelle  soc-dapm.c   Sprache: C

 
// SPDX-License-Identifier: GPL-2.0+
//
// soc-dapm.c  --  ALSA SoC Dynamic Audio Power Management
//
// Copyright 2005 Wolfson Microelectronics PLC.
// Author: Liam Girdwood <lrg@slimlogic.co.uk>
//
//  Features:
//    o Changes power status of internal codec blocks depending on the
//      dynamic configuration of codec internal audio paths and active
//      DACs/ADCs.
//    o Platform power domain - can support external components i.e. amps and
//      mic/headphone insertion events.
//    o Automatic Mic Bias support
//    o Jack insertion power event initiation - e.g. hp insertion will enable
//      sinks, dacs, etc
//    o Delayed power down of audio subsystem to reduce pops between a quick
//      device reopen.

#include <linux/module.h>
#include <linux/init.h>
#include <linux/async.h>
#include <linux/cleanup.h>
#include <linux/delay.h>
#include <linux/pm.h>
#include <linux/bitops.h>
#include <linux/platform_device.h>
#include <linux/jiffies.h>
#include <linux/debugfs.h>
#include <linux/pm_runtime.h>
#include <linux/regulator/consumer.h>
#include <linux/pinctrl/consumer.h>
#include <linux/clk.h>
#include <linux/slab.h>
#include <sound/core.h>
#include <sound/pcm.h>
#include <sound/pcm_params.h>
#include <sound/soc.h>
#include <sound/initval.h>

#include <trace/events/asoc.h>

#define DAPM_UPDATE_STAT(widget, val) widget->dapm->card->dapm_stats.val++;

#define SND_SOC_DAPM_DIR_REVERSE(x) ((x == SND_SOC_DAPM_DIR_IN) ? \
 SND_SOC_DAPM_DIR_OUT : SND_SOC_DAPM_DIR_IN)

#define snd_soc_dapm_for_each_direction(dir) \
 for ((dir) = SND_SOC_DAPM_DIR_IN; (dir) <= SND_SOC_DAPM_DIR_OUT; \
  (dir)++)

/* dapm power sequences - make this per codec in the future */
static int dapm_up_seq[] = {
 [snd_soc_dapm_pre] = 1,
 [snd_soc_dapm_regulator_supply] = 2,
 [snd_soc_dapm_pinctrl] = 2,
 [snd_soc_dapm_clock_supply] = 2,
 [snd_soc_dapm_supply] = 3,
 [snd_soc_dapm_dai_link] = 3,
 [snd_soc_dapm_micbias] = 4,
 [snd_soc_dapm_vmid] = 4,
 [snd_soc_dapm_dai_in] = 5,
 [snd_soc_dapm_dai_out] = 5,
 [snd_soc_dapm_aif_in] = 5,
 [snd_soc_dapm_aif_out] = 5,
 [snd_soc_dapm_mic] = 6,
 [snd_soc_dapm_siggen] = 6,
 [snd_soc_dapm_input] = 6,
 [snd_soc_dapm_output] = 6,
 [snd_soc_dapm_mux] = 7,
 [snd_soc_dapm_demux] = 7,
 [snd_soc_dapm_dac] = 8,
 [snd_soc_dapm_switch] = 9,
 [snd_soc_dapm_mixer] = 9,
 [snd_soc_dapm_mixer_named_ctl] = 9,
 [snd_soc_dapm_pga] = 10,
 [snd_soc_dapm_buffer] = 10,
 [snd_soc_dapm_scheduler] = 10,
 [snd_soc_dapm_effect] = 10,
 [snd_soc_dapm_src] = 10,
 [snd_soc_dapm_asrc] = 10,
 [snd_soc_dapm_encoder] = 10,
 [snd_soc_dapm_decoder] = 10,
 [snd_soc_dapm_adc] = 11,
 [snd_soc_dapm_out_drv] = 12,
 [snd_soc_dapm_hp] = 12,
 [snd_soc_dapm_line] = 12,
 [snd_soc_dapm_sink] = 12,
 [snd_soc_dapm_spk] = 13,
 [snd_soc_dapm_kcontrol] = 14,
 [snd_soc_dapm_post] = 15,
};

static int dapm_down_seq[] = {
 [snd_soc_dapm_pre] = 1,
 [snd_soc_dapm_kcontrol] = 2,
 [snd_soc_dapm_adc] = 3,
 [snd_soc_dapm_spk] = 4,
 [snd_soc_dapm_hp] = 5,
 [snd_soc_dapm_line] = 5,
 [snd_soc_dapm_out_drv] = 5,
 [snd_soc_dapm_sink] = 6,
 [snd_soc_dapm_pga] = 6,
 [snd_soc_dapm_buffer] = 6,
 [snd_soc_dapm_scheduler] = 6,
 [snd_soc_dapm_effect] = 6,
 [snd_soc_dapm_src] = 6,
 [snd_soc_dapm_asrc] = 6,
 [snd_soc_dapm_encoder] = 6,
 [snd_soc_dapm_decoder] = 6,
 [snd_soc_dapm_switch] = 7,
 [snd_soc_dapm_mixer_named_ctl] = 7,
 [snd_soc_dapm_mixer] = 7,
 [snd_soc_dapm_dac] = 8,
 [snd_soc_dapm_mic] = 9,
 [snd_soc_dapm_siggen] = 9,
 [snd_soc_dapm_input] = 9,
 [snd_soc_dapm_output] = 9,
 [snd_soc_dapm_micbias] = 10,
 [snd_soc_dapm_vmid] = 10,
 [snd_soc_dapm_mux] = 11,
 [snd_soc_dapm_demux] = 11,
 [snd_soc_dapm_aif_in] = 12,
 [snd_soc_dapm_aif_out] = 12,
 [snd_soc_dapm_dai_in] = 12,
 [snd_soc_dapm_dai_out] = 12,
 [snd_soc_dapm_dai_link] = 13,
 [snd_soc_dapm_supply] = 14,
 [snd_soc_dapm_clock_supply] = 15,
 [snd_soc_dapm_pinctrl] = 15,
 [snd_soc_dapm_regulator_supply] = 15,
 [snd_soc_dapm_post] = 16,
};

static void dapm_assert_locked(struct snd_soc_dapm_context *dapm)
{
 if (snd_soc_card_is_instantiated(dapm->card))
  snd_soc_dapm_mutex_assert_held(dapm);
}

static void pop_wait(u32 pop_time)
{
 if (pop_time)
  schedule_timeout_uninterruptible(msecs_to_jiffies(pop_time));
}

__printf(3, 4)
static void pop_dbg(struct device *dev, u32 pop_time, const char *fmt, ...)
{
 va_list args;
 char *buf;

 if (!pop_time)
  return;

 buf = kmalloc(PAGE_SIZE, GFP_KERNEL);
 if (buf == NULL)
  return;

 va_start(args, fmt);
 vsnprintf(buf, PAGE_SIZE, fmt, args);
 dev_info(dev, "%s", buf);
 va_end(args);

 kfree(buf);
}

static bool dapm_dirty_widget(struct snd_soc_dapm_widget *w)
{
 return !list_empty(&w->dirty);
}

static void dapm_mark_dirty(struct snd_soc_dapm_widget *w, const char *reason)
{
 dapm_assert_locked(w->dapm);

 if (!dapm_dirty_widget(w)) {
  dev_vdbg(w->dapm->dev, "Marking %s dirty due to %s\n",
    w->name, reason);
  list_add_tail(&w->dirty, &w->dapm->card->dapm_dirty);
 }
}

/*
 * Common implementation for dapm_widget_invalidate_input_paths() and
 * dapm_widget_invalidate_output_paths(). The function is inlined since the
 * combined size of the two specialized functions is only marginally larger then
 * the size of the generic function and at the same time the fast path of the
 * specialized functions is significantly smaller than the generic function.
 */

static __always_inline void dapm_widget_invalidate_paths(
 struct snd_soc_dapm_widget *w, enum snd_soc_dapm_direction dir)
{
 enum snd_soc_dapm_direction rdir = SND_SOC_DAPM_DIR_REVERSE(dir);
 struct snd_soc_dapm_widget *node;
 struct snd_soc_dapm_path *p;
 LIST_HEAD(list);

 dapm_assert_locked(w->dapm);

 if (w->endpoints[dir] == -1)
  return;

 list_add_tail(&w->work_list, &list);
 w->endpoints[dir] = -1;

 list_for_each_entry(w, &list, work_list) {
  snd_soc_dapm_widget_for_each_path(w, dir, p) {
   if (p->is_supply || !p->connect)
    continue;
   node = p->node[rdir];
   if (node->endpoints[dir] != -1) {
    node->endpoints[dir] = -1;
    list_add_tail(&node->work_list, &list);
   }
  }
 }
}

/*
 * dapm_widget_invalidate_input_paths() - Invalidate the cached number of
 *  input paths
 * @w: The widget for which to invalidate the cached number of input paths
 *
 * Resets the cached number of inputs for the specified widget and all widgets
 * that can be reached via outcoming paths from the widget.
 *
 * This function must be called if the number of output paths for a widget might
 * have changed. E.g. if the source state of a widget changes or a path is added
 * or activated with the widget as the sink.
 */

static void dapm_widget_invalidate_input_paths(struct snd_soc_dapm_widget *w)
{
 dapm_widget_invalidate_paths(w, SND_SOC_DAPM_DIR_IN);
}

/*
 * dapm_widget_invalidate_output_paths() - Invalidate the cached number of
 *  output paths
 * @w: The widget for which to invalidate the cached number of output paths
 *
 * Resets the cached number of outputs for the specified widget and all widgets
 * that can be reached via incoming paths from the widget.
 *
 * This function must be called if the number of output paths for a widget might
 * have changed. E.g. if the sink state of a widget changes or a path is added
 * or activated with the widget as the source.
 */

static void dapm_widget_invalidate_output_paths(struct snd_soc_dapm_widget *w)
{
 dapm_widget_invalidate_paths(w, SND_SOC_DAPM_DIR_OUT);
}

/*
 * dapm_path_invalidate() - Invalidates the cached number of inputs and outputs
 *  for the widgets connected to a path
 * @p: The path to invalidate
 *
 * Resets the cached number of inputs for the sink of the path and the cached
 * number of outputs for the source of the path.
 *
 * This function must be called when a path is added, removed or the connected
 * state changes.
 */

static void dapm_path_invalidate(struct snd_soc_dapm_path *p)
{
 /*
 * Weak paths or supply paths do not influence the number of input or
 * output paths of their neighbors.
 */

 if (p->is_supply)
  return;

 /*
 * The number of connected endpoints is the sum of the number of
 * connected endpoints of all neighbors. If a node with 0 connected
 * endpoints is either connected or disconnected that sum won't change,
 * so there is no need to re-check the path.
 */

 if (p->source->endpoints[SND_SOC_DAPM_DIR_IN] != 0)
  dapm_widget_invalidate_input_paths(p->sink);
 if (p->sink->endpoints[SND_SOC_DAPM_DIR_OUT] != 0)
  dapm_widget_invalidate_output_paths(p->source);
}

void snd_soc_dapm_mark_endpoints_dirty(struct snd_soc_card *card)
{
 struct snd_soc_dapm_widget *w;

 snd_soc_dapm_mutex_lock_root(card);

 for_each_card_widgets(card, w) {
  if (w->is_ep) {
   dapm_mark_dirty(w, "Rechecking endpoints");
   if (w->is_ep & SND_SOC_DAPM_EP_SINK)
    dapm_widget_invalidate_output_paths(w);
   if (w->is_ep & SND_SOC_DAPM_EP_SOURCE)
    dapm_widget_invalidate_input_paths(w);
  }
 }

 snd_soc_dapm_mutex_unlock(card);
}

/* create a new dapm widget */
static inline struct snd_soc_dapm_widget *dapm_cnew_widget(
 const struct snd_soc_dapm_widget *_widget,
 const char *prefix)
{
 struct snd_soc_dapm_widget *w __free(kfree) = kmemdup(_widget,
             sizeof(*_widget),
             GFP_KERNEL);
 if (!w)
  return NULL;

 if (prefix)
  w->name = kasprintf(GFP_KERNEL, "%s %s", prefix, _widget->name);
 else
  w->name = kstrdup_const(_widget->name, GFP_KERNEL);
 if (!w->name)
  return NULL;

 if (_widget->sname) {
  w->sname = kstrdup_const(_widget->sname, GFP_KERNEL);
  if (!w->sname) {
   kfree_const(w->name);
   return NULL;
  }
 }

 return_ptr(w);
}

struct dapm_kcontrol_data {
 unsigned int value;
 struct snd_soc_dapm_widget *widget;
 struct list_head paths;
 struct snd_soc_dapm_widget_list *wlist;
};

static unsigned int soc_dapm_read(struct snd_soc_dapm_context *dapm, int reg)
{
 if (!dapm->component)
  return -EIO;
 return  snd_soc_component_read(dapm->component, reg);
}

/* set up initial codec paths */
static void dapm_set_mixer_path_status(struct snd_soc_dapm_path *p, int i,
           int nth_path)
{
 struct soc_mixer_control *mc = (struct soc_mixer_control *)
  p->sink->kcontrol_news[i].private_value;
 unsigned int reg = mc->reg;
 unsigned int invert = mc->invert;

 if (reg != SND_SOC_NOPM) {
  unsigned int shift = mc->shift;
  unsigned int max = mc->max;
  unsigned int mask = (1 << fls(max)) - 1;
  unsigned int val = soc_dapm_read(p->sink->dapm, reg);

  /*
 * The nth_path argument allows this function to know
 * which path of a kcontrol it is setting the initial
 * status for. Ideally this would support any number
 * of paths and channels. But since kcontrols only come
 * in mono and stereo variants, we are limited to 2
 * channels.
 *
 * The following code assumes for stereo controls the
 * first path is the left channel, and all remaining
 * paths are the right channel.
 */

  if (snd_soc_volsw_is_stereo(mc) && nth_path > 0) {
   if (reg != mc->rreg)
    val = soc_dapm_read(p->sink->dapm, mc->rreg);
   val = (val >> mc->rshift) & mask;
  } else {
   val = (val >> shift) & mask;
  }
  if (invert)
   val = max - val;
  p->connect = !!val;
 } else {
  /* since a virtual mixer has no backing registers to
 * decide which path to connect, it will try to match
 * with initial state.  This is to ensure
 * that the default mixer choice will be
 * correctly powered up during initialization.
 */

  p->connect = invert;
 }
}

/* connect mux widget to its interconnecting audio paths */
static int dapm_connect_mux(struct snd_soc_dapm_context *dapm,
       struct snd_soc_dapm_path *path, const char *control_name,
       struct snd_soc_dapm_widget *w)
{
 const struct snd_kcontrol_new *kcontrol = &w->kcontrol_news[0];
 struct soc_enum *e = (struct soc_enum *)kcontrol->private_value;
 unsigned int item;
 int i;

 if (e->reg != SND_SOC_NOPM) {
  unsigned int val;

  val = soc_dapm_read(dapm, e->reg);
  val = (val >> e->shift_l) & e->mask;
  item = snd_soc_enum_val_to_item(e, val);
 } else {
  /* since a virtual mux has no backing registers to
 * decide which path to connect, it will try to match
 * with the first enumeration.  This is to ensure
 * that the default mux choice (the first) will be
 * correctly powered up during initialization.
 */

  item = 0;
 }

 i = match_string(e->texts, e->items, control_name);
 if (i < 0)
  return -ENODEV;

 path->name = e->texts[i];
 path->connect = (i == item);
 return 0;

}

/* connect mixer widget to its interconnecting audio paths */
static int dapm_connect_mixer(struct snd_soc_dapm_context *dapm,
         struct snd_soc_dapm_path *path, const char *control_name)
{
 int i, nth_path = 0;

 /* search for mixer kcontrol */
 for (i = 0; i < path->sink->num_kcontrols; i++) {
  if (!strcmp(control_name, path->sink->kcontrol_news[i].name)) {
   path->name = path->sink->kcontrol_news[i].name;
   dapm_set_mixer_path_status(path, i, nth_path++);
   return 0;
  }
 }
 return -ENODEV;
}

/*
 * dapm_update_widget_flags() - Re-compute widget sink and source flags
 * @w: The widget for which to update the flags
 *
 * Some widgets have a dynamic category which depends on which neighbors they
 * are connected to. This function update the category for these widgets.
 *
 * This function must be called whenever a path is added or removed to a widget.
 */

static void dapm_update_widget_flags(struct snd_soc_dapm_widget *w)
{
 enum snd_soc_dapm_direction dir;
 struct snd_soc_dapm_path *p;
 unsigned int ep;

 switch (w->id) {
 case snd_soc_dapm_input:
  /* On a fully routed card an input is never a source */
  if (w->dapm->card->fully_routed)
   return;
  ep = SND_SOC_DAPM_EP_SOURCE;
  snd_soc_dapm_widget_for_each_source_path(w, p) {
   if (p->source->id == snd_soc_dapm_micbias ||
       p->source->id == snd_soc_dapm_mic ||
       p->source->id == snd_soc_dapm_line ||
       p->source->id == snd_soc_dapm_output) {
    ep = 0;
    break;
   }
  }
  break;
 case snd_soc_dapm_output:
  /* On a fully routed card a output is never a sink */
  if (w->dapm->card->fully_routed)
   return;
  ep = SND_SOC_DAPM_EP_SINK;
  snd_soc_dapm_widget_for_each_sink_path(w, p) {
   if (p->sink->id == snd_soc_dapm_spk ||
       p->sink->id == snd_soc_dapm_hp ||
       p->sink->id == snd_soc_dapm_line ||
       p->sink->id == snd_soc_dapm_input) {
    ep = 0;
    break;
   }
  }
  break;
 case snd_soc_dapm_line:
  ep = 0;
  snd_soc_dapm_for_each_direction(dir) {
   if (!list_empty(&w->edges[dir]))
    ep |= SND_SOC_DAPM_DIR_TO_EP(dir);
  }
  break;
 default:
  return;
 }

 w->is_ep = ep;
}

static int snd_soc_dapm_check_dynamic_path(
 struct snd_soc_dapm_context *dapm,
 struct snd_soc_dapm_widget *source, struct snd_soc_dapm_widget *sink,
 const char *control)
{
 bool dynamic_source = false;
 bool dynamic_sink = false;

 if (!control)
  return 0;

 switch (source->id) {
 case snd_soc_dapm_demux:
  dynamic_source = true;
  break;
 default:
  break;
 }

 switch (sink->id) {
 case snd_soc_dapm_mux:
 case snd_soc_dapm_switch:
 case snd_soc_dapm_mixer:
 case snd_soc_dapm_mixer_named_ctl:
  dynamic_sink = true;
  break;
 default:
  break;
 }

 if (dynamic_source && dynamic_sink) {
  dev_err(dapm->dev,
   "Direct connection between demux and mixer/mux not supported for path %s -> [%s] -> %s\n",
   source->name, control, sink->name);
  return -EINVAL;
 } else if (!dynamic_source && !dynamic_sink) {
  dev_err(dapm->dev,
   "Control not supported for path %s -> [%s] -> %s\n",
   source->name, control, sink->name);
  return -EINVAL;
 }

 return 0;
}

static int snd_soc_dapm_add_path(
 struct snd_soc_dapm_context *dapm,
 struct snd_soc_dapm_widget *wsource, struct snd_soc_dapm_widget *wsink,
 const char *control,
 int (*connected)(struct snd_soc_dapm_widget *source,
    struct snd_soc_dapm_widget *sink))
{
 enum snd_soc_dapm_direction dir;
 struct snd_soc_dapm_path *path;
 int ret;

 if (wsink->is_supply && !wsource->is_supply) {
  dev_err(dapm->dev,
   "Connecting non-supply widget to supply widget is not supported (%s -> %s)\n",
   wsource->name, wsink->name);
  return -EINVAL;
 }

 if (connected && !wsource->is_supply) {
  dev_err(dapm->dev,
   "connected() callback only supported for supply widgets (%s -> %s)\n",
   wsource->name, wsink->name);
  return -EINVAL;
 }

 if (wsource->is_supply && control) {
  dev_err(dapm->dev,
   "Conditional paths are not supported for supply widgets (%s -> [%s] -> %s)\n",
   wsource->name, control, wsink->name);
  return -EINVAL;
 }

 ret = snd_soc_dapm_check_dynamic_path(dapm, wsource, wsink, control);
 if (ret)
  return ret;

 path = kzalloc(sizeof(struct snd_soc_dapm_path), GFP_KERNEL);
 if (!path)
  return -ENOMEM;

 path->node[SND_SOC_DAPM_DIR_IN] = wsource;
 path->node[SND_SOC_DAPM_DIR_OUT] = wsink;

 path->connected = connected;
 INIT_LIST_HEAD(&path->list);
 INIT_LIST_HEAD(&path->list_kcontrol);

 if (wsource->is_supply || wsink->is_supply)
  path->is_supply = 1;

 /* connect static paths */
 if (control == NULL) {
  path->connect = 1;
 } else {
  switch (wsource->id) {
  case snd_soc_dapm_demux:
   ret = dapm_connect_mux(dapm, path, control, wsource);
   if (ret)
    goto err;
   break;
  default:
   break;
  }

  switch (wsink->id) {
  case snd_soc_dapm_mux:
   ret = dapm_connect_mux(dapm, path, control, wsink);
   if (ret != 0)
    goto err;
   break;
  case snd_soc_dapm_switch:
  case snd_soc_dapm_mixer:
  case snd_soc_dapm_mixer_named_ctl:
   ret = dapm_connect_mixer(dapm, path, control);
   if (ret != 0)
    goto err;
   break;
  default:
   break;
  }
 }

 list_add(&path->list, &dapm->card->paths);

 snd_soc_dapm_for_each_direction(dir)
  list_add(&path->list_node[dir], &path->node[dir]->edges[dir]);

 snd_soc_dapm_for_each_direction(dir) {
  dapm_update_widget_flags(path->node[dir]);
  dapm_mark_dirty(path->node[dir], "Route added");
 }

 if (snd_soc_card_is_instantiated(dapm->card) && path->connect)
  dapm_path_invalidate(path);

 return 0;
err:
 kfree(path);
 return ret;
}

static int dapm_kcontrol_data_alloc(struct snd_soc_dapm_widget *widget,
 struct snd_kcontrol *kcontrol, const char *ctrl_name)
{
 struct dapm_kcontrol_data *data;
 struct soc_mixer_control *mc;
 struct soc_enum *e;
 const char *name;
 int ret;

 data = kzalloc(sizeof(*data), GFP_KERNEL);
 if (!data)
  return -ENOMEM;

 INIT_LIST_HEAD(&data->paths);

 switch (widget->id) {
 case snd_soc_dapm_switch:
 case snd_soc_dapm_mixer:
 case snd_soc_dapm_mixer_named_ctl:
  mc = (struct soc_mixer_control *)kcontrol->private_value;

  if (mc->autodisable) {
   struct snd_soc_dapm_widget template;

   if (snd_soc_volsw_is_stereo(mc))
    dev_warn(widget->dapm->dev,
      "ASoC: Unsupported stereo autodisable control '%s'\n",
      ctrl_name);

   name = kasprintf(GFP_KERNEL, "%s %s", ctrl_name,
      "Autodisable");
   if (!name) {
    ret = -ENOMEM;
    goto err_data;
   }

   memset(&template, 0, sizeof(template));
   template.reg = mc->reg;
   template.mask = (1 << fls(mc->max)) - 1;
   template.shift = mc->shift;
   if (mc->invert)
    template.off_val = mc->max;
   else
    template.off_val = 0;
   template.on_val = template.off_val;
   template.id = snd_soc_dapm_kcontrol;
   template.name = name;

   data->value = template.on_val;

   data->widget =
    snd_soc_dapm_new_control_unlocked(widget->dapm,
    &template);
   kfree(name);
   if (IS_ERR(data->widget)) {
    ret = PTR_ERR(data->widget);
    goto err_data;
   }
  }
  break;
 case snd_soc_dapm_demux:
 case snd_soc_dapm_mux:
  e = (struct soc_enum *)kcontrol->private_value;

  if (e->autodisable) {
   struct snd_soc_dapm_widget template;

   name = kasprintf(GFP_KERNEL, "%s %s", ctrl_name,
      "Autodisable");
   if (!name) {
    ret = -ENOMEM;
    goto err_data;
   }

   memset(&template, 0, sizeof(template));
   template.reg = e->reg;
   template.mask = e->mask;
   template.shift = e->shift_l;
   template.off_val = snd_soc_enum_item_to_val(e, 0);
   template.on_val = template.off_val;
   template.id = snd_soc_dapm_kcontrol;
   template.name = name;

   data->value = template.on_val;

   data->widget = snd_soc_dapm_new_control_unlocked(
      widget->dapm, &template);
   kfree(name);
   if (IS_ERR(data->widget)) {
    ret = PTR_ERR(data->widget);
    goto err_data;
   }

   snd_soc_dapm_add_path(widget->dapm, data->widget,
           widget, NULL, NULL);
  } else if (e->reg != SND_SOC_NOPM) {
   data->value = soc_dapm_read(widget->dapm, e->reg) &
          (e->mask << e->shift_l);
  }
  break;
 default:
  break;
 }

 kcontrol->private_data = data;

 return 0;

err_data:
 kfree(data);
 return ret;
}

static void dapm_kcontrol_free(struct snd_kcontrol *kctl)
{
 struct dapm_kcontrol_data *data = snd_kcontrol_chip(kctl);

 list_del(&data->paths);
 kfree(data->wlist);
 kfree(data);
}

static struct snd_soc_dapm_widget_list *dapm_kcontrol_get_wlist(
 const struct snd_kcontrol *kcontrol)
{
 struct dapm_kcontrol_data *data = snd_kcontrol_chip(kcontrol);

 return data->wlist;
}

static int dapm_kcontrol_add_widget(struct snd_kcontrol *kcontrol,
 struct snd_soc_dapm_widget *widget)
{
 struct dapm_kcontrol_data *data = snd_kcontrol_chip(kcontrol);
 struct snd_soc_dapm_widget_list *new_wlist;
 unsigned int n;

 if (data->wlist)
  n = data->wlist->num_widgets + 1;
 else
  n = 1;

 new_wlist = krealloc(data->wlist,
        struct_size(new_wlist, widgets, n),
        GFP_KERNEL);
 if (!new_wlist)
  return -ENOMEM;

 new_wlist->num_widgets = n;
 new_wlist->widgets[n - 1] = widget;

 data->wlist = new_wlist;

 return 0;
}

static void dapm_kcontrol_add_path(const struct snd_kcontrol *kcontrol,
 struct snd_soc_dapm_path *path)
{
 struct dapm_kcontrol_data *data = snd_kcontrol_chip(kcontrol);

 list_add_tail(&path->list_kcontrol, &data->paths);
}

static bool dapm_kcontrol_is_powered(const struct snd_kcontrol *kcontrol)
{
 struct dapm_kcontrol_data *data = snd_kcontrol_chip(kcontrol);

 if (!data->widget)
  return true;

 return data->widget->power;
}

static struct list_head *dapm_kcontrol_get_path_list(
 const struct snd_kcontrol *kcontrol)
{
 struct dapm_kcontrol_data *data = snd_kcontrol_chip(kcontrol);

 return &data->paths;
}

#define dapm_kcontrol_for_each_path(path, kcontrol) \
 list_for_each_entry(path, dapm_kcontrol_get_path_list(kcontrol), \
  list_kcontrol)

unsigned int dapm_kcontrol_get_value(const struct snd_kcontrol *kcontrol)
{
 struct dapm_kcontrol_data *data = snd_kcontrol_chip(kcontrol);

 return data->value;
}
EXPORT_SYMBOL_GPL(dapm_kcontrol_get_value);

static bool dapm_kcontrol_set_value(const struct snd_kcontrol *kcontrol,
 unsigned int value)
{
 struct dapm_kcontrol_data *data = snd_kcontrol_chip(kcontrol);

 if (data->value == value)
  return false;

 if (data->widget) {
  switch (dapm_kcontrol_get_wlist(kcontrol)->widgets[0]->id) {
  case snd_soc_dapm_switch:
  case snd_soc_dapm_mixer:
  case snd_soc_dapm_mixer_named_ctl:
   data->widget->on_val = value & data->widget->mask;
   break;
  case snd_soc_dapm_demux:
  case snd_soc_dapm_mux:
   data->widget->on_val = value >> data->widget->shift;
   break;
  default:
   data->widget->on_val = value;
   break;
  }
 }

 data->value = value;

 return true;
}

/**
 * snd_soc_dapm_kcontrol_widget() - Returns the widget associated to a
 *   kcontrol
 * @kcontrol: The kcontrol
 */

struct snd_soc_dapm_widget *snd_soc_dapm_kcontrol_widget(
    struct snd_kcontrol *kcontrol)
{
 return dapm_kcontrol_get_wlist(kcontrol)->widgets[0];
}
EXPORT_SYMBOL_GPL(snd_soc_dapm_kcontrol_widget);

/**
 * snd_soc_dapm_kcontrol_dapm() - Returns the dapm context associated to a
 *  kcontrol
 * @kcontrol: The kcontrol
 *
 * Note: This function must only be used on kcontrols that are known to have
 * been registered for a CODEC. Otherwise the behaviour is undefined.
 */

struct snd_soc_dapm_context *snd_soc_dapm_kcontrol_dapm(
 struct snd_kcontrol *kcontrol)
{
 return dapm_kcontrol_get_wlist(kcontrol)->widgets[0]->dapm;
}
EXPORT_SYMBOL_GPL(snd_soc_dapm_kcontrol_dapm);

static void dapm_reset(struct snd_soc_card *card)
{
 struct snd_soc_dapm_widget *w;

 snd_soc_dapm_mutex_assert_held(card);

 memset(&card->dapm_stats, 0, sizeof(card->dapm_stats));

 for_each_card_widgets(card, w) {
  w->new_power = w->power;
  w->power_checked = false;
 }
}

static const char *soc_dapm_prefix(struct snd_soc_dapm_context *dapm)
{
 if (!dapm->component)
  return NULL;
 return dapm->component->name_prefix;
}

static int soc_dapm_update_bits(struct snd_soc_dapm_context *dapm,
 int reg, unsigned int mask, unsigned int value)
{
 if (!dapm->component)
  return -EIO;
 return snd_soc_component_update_bits(dapm->component, reg,
          mask, value);
}

static int soc_dapm_test_bits(struct snd_soc_dapm_context *dapm,
 int reg, unsigned int mask, unsigned int value)
{
 if (!dapm->component)
  return -EIO;
 return snd_soc_component_test_bits(dapm->component, reg, mask, value);
}

static void soc_dapm_async_complete(struct snd_soc_dapm_context *dapm)
{
 if (dapm->component)
  snd_soc_component_async_complete(dapm->component);
}

static struct snd_soc_dapm_widget *
dapm_wcache_lookup(struct snd_soc_dapm_widget *w, const char *name)
{
 if (w) {
  struct list_head *wlist = &w->dapm->card->widgets;
  const int depth = 2;
  int i = 0;

  list_for_each_entry_from(w, wlist, list) {
   if (!strcmp(name, w->name))
    return w;

   if (++i == depth)
    break;
  }
 }

 return NULL;
}

/**
 * snd_soc_dapm_force_bias_level() - Sets the DAPM bias level
 * @dapm: The DAPM context for which to set the level
 * @level: The level to set
 *
 * Forces the DAPM bias level to a specific state. It will call the bias level
 * callback of DAPM context with the specified level. This will even happen if
 * the context is already at the same level. Furthermore it will not go through
 * the normal bias level sequencing, meaning any intermediate states between the
 * current and the target state will not be entered.
 *
 * Note that the change in bias level is only temporary and the next time
 * snd_soc_dapm_sync() is called the state will be set to the level as
 * determined by the DAPM core. The function is mainly intended to be used to
 * used during probe or resume from suspend to power up the device so
 * initialization can be done, before the DAPM core takes over.
 */

int snd_soc_dapm_force_bias_level(struct snd_soc_dapm_context *dapm,
 enum snd_soc_bias_level level)
{
 int ret = 0;

 if (dapm->component)
  ret = snd_soc_component_set_bias_level(dapm->component, level);

 if (ret == 0)
  dapm->bias_level = level;

 return ret;
}
EXPORT_SYMBOL_GPL(snd_soc_dapm_force_bias_level);

/**
 * snd_soc_dapm_set_bias_level - set the bias level for the system
 * @dapm: DAPM context
 * @level: level to configure
 *
 * Configure the bias (power) levels for the SoC audio device.
 *
 * Returns 0 for success else error.
 */

static int snd_soc_dapm_set_bias_level(struct snd_soc_dapm_context *dapm,
           enum snd_soc_bias_level level)
{
 struct snd_soc_card *card = dapm->card;
 int ret = 0;

 trace_snd_soc_bias_level_start(dapm, level);

 ret = snd_soc_card_set_bias_level(card, dapm, level);
 if (ret != 0)
  goto out;

 if (dapm != &card->dapm)
  ret = snd_soc_dapm_force_bias_level(dapm, level);

 if (ret != 0)
  goto out;

 ret = snd_soc_card_set_bias_level_post(card, dapm, level);
out:
 trace_snd_soc_bias_level_done(dapm, level);

 /* success */
 if (ret == 0)
  snd_soc_dapm_init_bias_level(dapm, level);

 return ret;
}

static int dapm_is_shared_kcontrol(struct snd_soc_dapm_context *dapm,
 struct snd_soc_dapm_widget *kcontrolw,
 const struct snd_kcontrol_new *kcontrol_new,
 struct snd_kcontrol **kcontrol)
{
 struct snd_soc_dapm_widget *w;
 int i;

 *kcontrol = NULL;

 for_each_card_widgets(dapm->card, w) {
  if (w == kcontrolw || w->dapm != kcontrolw->dapm)
   continue;
  for (i = 0; i < w->num_kcontrols; i++) {
   if (&w->kcontrol_news[i] == kcontrol_new) {
    if (w->kcontrols)
     *kcontrol = w->kcontrols[i];
    return 1;
   }
  }
 }

 return 0;
}

/*
 * Determine if a kcontrol is shared. If it is, look it up. If it isn't,
 * create it. Either way, add the widget into the control's widget list
 */

static int dapm_create_or_share_kcontrol(struct snd_soc_dapm_widget *w,
 int kci)
{
 struct snd_soc_dapm_context *dapm = w->dapm;
 struct snd_card *card = dapm->card->snd_card;
 const char *prefix;
 size_t prefix_len;
 int shared;
 struct snd_kcontrol *kcontrol;
 bool wname_in_long_name, kcname_in_long_name;
 char *long_name = NULL;
 const char *name;
 int ret = 0;

 prefix = soc_dapm_prefix(dapm);
 if (prefix)
  prefix_len = strlen(prefix) + 1;
 else
  prefix_len = 0;

 shared = dapm_is_shared_kcontrol(dapm, w, &w->kcontrol_news[kci],
      &kcontrol);

 if (!kcontrol) {
  if (shared) {
   wname_in_long_name = false;
   kcname_in_long_name = true;
  } else {
   switch (w->id) {
   case snd_soc_dapm_switch:
   case snd_soc_dapm_mixer:
   case snd_soc_dapm_pga:
   case snd_soc_dapm_effect:
   case snd_soc_dapm_out_drv:
    wname_in_long_name = true;
    kcname_in_long_name = true;
    break;
   case snd_soc_dapm_mixer_named_ctl:
    wname_in_long_name = false;
    kcname_in_long_name = true;
    break;
   case snd_soc_dapm_demux:
   case snd_soc_dapm_mux:
    wname_in_long_name = true;
    kcname_in_long_name = false;
    break;
   default:
    return -EINVAL;
   }
  }
  if (w->no_wname_in_kcontrol_name)
   wname_in_long_name = false;

  if (wname_in_long_name && kcname_in_long_name) {
   /*
 * The control will get a prefix from the control
 * creation process but we're also using the same
 * prefix for widgets so cut the prefix off the
 * front of the widget name.
 */

   long_name = kasprintf(GFP_KERNEL, "%s %s",
     w->name + prefix_len,
     w->kcontrol_news[kci].name);
   if (long_name == NULL)
    return -ENOMEM;

   name = long_name;
  } else if (wname_in_long_name) {
   long_name = NULL;
   name = w->name + prefix_len;
  } else {
   long_name = NULL;
   name = w->kcontrol_news[kci].name;
  }

  kcontrol = snd_soc_cnew(&w->kcontrol_news[kci], NULL, name,
     prefix);
  if (!kcontrol) {
   ret = -ENOMEM;
   goto exit_free;
  }

  kcontrol->private_free = dapm_kcontrol_free;

  ret = dapm_kcontrol_data_alloc(w, kcontrol, name);
  if (ret) {
   snd_ctl_free_one(kcontrol);
   goto exit_free;
  }

  ret = snd_ctl_add(card, kcontrol);
  if (ret < 0) {
   dev_err(dapm->dev,
    "ASoC: failed to add widget %s dapm kcontrol %s: %d\n",
    w->name, name, ret);
   goto exit_free;
  }
 }

 ret = dapm_kcontrol_add_widget(kcontrol, w);
 if (ret == 0)
  w->kcontrols[kci] = kcontrol;

exit_free:
 kfree(long_name);

 return ret;
}

/* create new dapm mixer control */
static int dapm_new_mixer(struct snd_soc_dapm_widget *w)
{
 int i, ret;
 struct snd_soc_dapm_path *path;
 struct dapm_kcontrol_data *data;

 /* add kcontrol */
 for (i = 0; i < w->num_kcontrols; i++) {
  /* match name */
  snd_soc_dapm_widget_for_each_source_path(w, path) {
   /* mixer/mux paths name must match control name */
   if (path->name != (char *)w->kcontrol_news[i].name)
    continue;

   if (!w->kcontrols[i]) {
    ret = dapm_create_or_share_kcontrol(w, i);
    if (ret < 0)
     return ret;
   }

   dapm_kcontrol_add_path(w->kcontrols[i], path);

   data = snd_kcontrol_chip(w->kcontrols[i]);
   if (data->widget)
    snd_soc_dapm_add_path(data->widget->dapm,
            data->widget,
            path->source,
            NULL, NULL);
  }
 }

 return 0;
}

/* create new dapm mux control */
static int dapm_new_mux(struct snd_soc_dapm_widget *w)
{
 struct snd_soc_dapm_context *dapm = w->dapm;
 enum snd_soc_dapm_direction dir;
 struct snd_soc_dapm_path *path;
 const char *type;
 int ret;

 switch (w->id) {
 case snd_soc_dapm_mux:
  dir = SND_SOC_DAPM_DIR_OUT;
  type = "mux";
  break;
 case snd_soc_dapm_demux:
  dir = SND_SOC_DAPM_DIR_IN;
  type = "demux";
  break;
 default:
  return -EINVAL;
 }

 if (w->num_kcontrols != 1) {
  dev_err(dapm->dev,
   "ASoC: %s %s has incorrect number of controls\n", type,
   w->name);
  return -EINVAL;
 }

 if (list_empty(&w->edges[dir])) {
  dev_err(dapm->dev, "ASoC: %s %s has no paths\n", type, w->name);
  return -EINVAL;
 }

 ret = dapm_create_or_share_kcontrol(w, 0);
 if (ret < 0)
  return ret;

 snd_soc_dapm_widget_for_each_path(w, dir, path) {
  if (path->name)
   dapm_kcontrol_add_path(w->kcontrols[0], path);
 }

 return 0;
}

/* create new dapm volume control */
static int dapm_new_pga(struct snd_soc_dapm_widget *w)
{
 int i;

 for (i = 0; i < w->num_kcontrols; i++) {
  int ret = dapm_create_or_share_kcontrol(w, i);
  if (ret < 0)
   return ret;
 }

 return 0;
}

/* create new dapm dai link control */
static int dapm_new_dai_link(struct snd_soc_dapm_widget *w)
{
 int i;
 struct snd_soc_pcm_runtime *rtd = w->priv;

 /* create control for links with > 1 config */
 if (rtd->dai_link->num_c2c_params <= 1)
  return 0;

 /* add kcontrol */
 for (i = 0; i < w->num_kcontrols; i++) {
  struct snd_soc_dapm_context *dapm = w->dapm;
  struct snd_card *card = dapm->card->snd_card;
  struct snd_kcontrol *kcontrol = snd_soc_cnew(&w->kcontrol_news[i],
            w, w->name, NULL);
  int ret = snd_ctl_add(card, kcontrol);

  if (ret < 0) {
   dev_err(dapm->dev,
    "ASoC: failed to add widget %s dapm kcontrol %s: %d\n",
    w->name, w->kcontrol_news[i].name, ret);
   return ret;
  }
  kcontrol->private_data = w;
  w->kcontrols[i] = kcontrol;
 }

 return 0;
}

/* We implement power down on suspend by checking the power state of
 * the ALSA card - when we are suspending the ALSA state for the card
 * is set to D3.
 */

static int snd_soc_dapm_suspend_check(struct snd_soc_dapm_widget *widget)
{
 int level = snd_power_get_state(widget->dapm->card->snd_card);

 switch (level) {
 case SNDRV_CTL_POWER_D3hot:
 case SNDRV_CTL_POWER_D3cold:
  if (widget->ignore_suspend)
   dev_dbg(widget->dapm->dev, "ASoC: %s ignoring suspend\n",
    widget->name);
  return widget->ignore_suspend;
 default:
  return 1;
 }
}

static void dapm_widget_list_free(struct snd_soc_dapm_widget_list **list)
{
 kfree(*list);
}

static int dapm_widget_list_create(struct snd_soc_dapm_widget_list **list,
 struct list_head *widgets)
{
 struct snd_soc_dapm_widget *w;
 struct list_head *it;
 unsigned int size = 0;
 unsigned int i = 0;

 list_for_each(it, widgets)
  size++;

 *list = kzalloc(struct_size(*list, widgets, size), GFP_KERNEL);
 if (*list == NULL)
  return -ENOMEM;

 (*list)->num_widgets = size;

 list_for_each_entry(w, widgets, work_list)
  (*list)->widgets[i++] = w;

 (*list)->num_widgets = i;

 return 0;
}

/*
 * Recursively reset the cached number of inputs or outputs for the specified
 * widget and all widgets that can be reached via incoming or outcoming paths
 * from the widget.
 */

static void invalidate_paths_ep(struct snd_soc_dapm_widget *widget,
 enum snd_soc_dapm_direction dir)
{
 enum snd_soc_dapm_direction rdir = SND_SOC_DAPM_DIR_REVERSE(dir);
 struct snd_soc_dapm_path *path;

 widget->endpoints[dir] = -1;

 snd_soc_dapm_widget_for_each_path(widget, rdir, path) {
  if (path->is_supply)
   continue;

  if (path->walking)
   return;

  if (path->connect) {
   path->walking = 1;
   invalidate_paths_ep(path->node[dir], dir);
   path->walking = 0;
  }
 }
}

/*
 * Common implementation for is_connected_output_ep() and
 * is_connected_input_ep(). The function is inlined since the combined size of
 * the two specialized functions is only marginally larger then the size of the
 * generic function and at the same time the fast path of the specialized
 * functions is significantly smaller than the generic function.
 */

static __always_inline int is_connected_ep(struct snd_soc_dapm_widget *widget,
 struct list_head *list, enum snd_soc_dapm_direction dir,
 int (*fn)(struct snd_soc_dapm_widget *, struct list_head *,
    bool (*custom_stop_condition)(struct snd_soc_dapm_widget *,
      enum snd_soc_dapm_direction)),
 bool (*custom_stop_condition)(struct snd_soc_dapm_widget *,
          enum snd_soc_dapm_direction))
{
 enum snd_soc_dapm_direction rdir = SND_SOC_DAPM_DIR_REVERSE(dir);
 struct snd_soc_dapm_path *path;
 int con = 0;

 if (widget->endpoints[dir] >= 0)
  return widget->endpoints[dir];

 DAPM_UPDATE_STAT(widget, path_checks);

 /* do we need to add this widget to the list ? */
 if (list)
  list_add_tail(&widget->work_list, list);

 if (custom_stop_condition && custom_stop_condition(widget, dir)) {
  list = NULL;
  custom_stop_condition = NULL;
 }

 if ((widget->is_ep & SND_SOC_DAPM_DIR_TO_EP(dir)) && widget->connected) {
  widget->endpoints[dir] = snd_soc_dapm_suspend_check(widget);
  return widget->endpoints[dir];
 }

 snd_soc_dapm_widget_for_each_path(widget, rdir, path) {
  DAPM_UPDATE_STAT(widget, neighbour_checks);

  if (path->is_supply)
   continue;

  if (path->walking)
   return 1;

  trace_snd_soc_dapm_path(widget, dir, path);

  if (path->connect) {
   path->walking = 1;
   con += fn(path->node[dir], list, custom_stop_condition);
   path->walking = 0;
  }
 }

 widget->endpoints[dir] = con;

 return con;
}

/*
 * Recursively check for a completed path to an active or physically connected
 * output widget. Returns number of complete paths.
 *
 * Optionally, can be supplied with a function acting as a stopping condition.
 * This function takes the dapm widget currently being examined and the walk
 * direction as an arguments, it should return true if widgets from that point
 * in the graph onwards should not be added to the widget list.
 */

static int is_connected_output_ep(struct snd_soc_dapm_widget *widget,
 struct list_head *list,
 bool (*custom_stop_condition)(struct snd_soc_dapm_widget *i,
          enum snd_soc_dapm_direction))
{
 return is_connected_ep(widget, list, SND_SOC_DAPM_DIR_OUT,
   is_connected_output_ep, custom_stop_condition);
}

/*
 * Recursively check for a completed path to an active or physically connected
 * input widget. Returns number of complete paths.
 *
 * Optionally, can be supplied with a function acting as a stopping condition.
 * This function takes the dapm widget currently being examined and the walk
 * direction as an arguments, it should return true if the walk should be
 * stopped and false otherwise.
 */

static int is_connected_input_ep(struct snd_soc_dapm_widget *widget,
 struct list_head *list,
 bool (*custom_stop_condition)(struct snd_soc_dapm_widget *i,
          enum snd_soc_dapm_direction))
{
 return is_connected_ep(widget, list, SND_SOC_DAPM_DIR_IN,
   is_connected_input_ep, custom_stop_condition);
}

/**
 * snd_soc_dapm_dai_get_connected_widgets - query audio path and it's widgets.
 * @dai: the soc DAI.
 * @stream: stream direction.
 * @list: list of active widgets for this stream.
 * @custom_stop_condition: (optional) a function meant to stop the widget graph
 *                         walk based on custom logic.
 *
 * Queries DAPM graph as to whether a valid audio stream path exists for
 * the initial stream specified by name. This takes into account
 * current mixer and mux kcontrol settings. Creates list of valid widgets.
 *
 * Optionally, can be supplied with a function acting as a stopping condition.
 * This function takes the dapm widget currently being examined and the walk
 * direction as an arguments, it should return true if the walk should be
 * stopped and false otherwise.
 *
 * Returns the number of valid paths or negative error.
 */

int snd_soc_dapm_dai_get_connected_widgets(struct snd_soc_dai *dai, int stream,
 struct snd_soc_dapm_widget_list **list,
 bool (*custom_stop_condition)(struct snd_soc_dapm_widget *,
          enum snd_soc_dapm_direction))
{
 struct snd_soc_card *card = dai->component->card;
 struct snd_soc_dapm_widget *w = snd_soc_dai_get_widget(dai, stream);
 LIST_HEAD(widgets);
 int paths;
 int ret;

 snd_soc_dapm_mutex_lock(card);

 if (stream == SNDRV_PCM_STREAM_PLAYBACK) {
  invalidate_paths_ep(w, SND_SOC_DAPM_DIR_OUT);
  paths = is_connected_output_ep(w, &widgets,
    custom_stop_condition);
 } else {
  invalidate_paths_ep(w, SND_SOC_DAPM_DIR_IN);
  paths = is_connected_input_ep(w, &widgets,
    custom_stop_condition);
 }

 /* Drop starting point */
 list_del(widgets.next);

 ret = dapm_widget_list_create(list, &widgets);
 if (ret)
  paths = ret;

 trace_snd_soc_dapm_connected(paths, stream);
 snd_soc_dapm_mutex_unlock(card);

 return paths;
}
EXPORT_SYMBOL_GPL(snd_soc_dapm_dai_get_connected_widgets);

void snd_soc_dapm_dai_free_widgets(struct snd_soc_dapm_widget_list **list)
{
 dapm_widget_list_free(list);
}
EXPORT_SYMBOL_GPL(snd_soc_dapm_dai_free_widgets);

/*
 * Handler for regulator supply widget.
 */

int snd_soc_dapm_regulator_event(struct snd_soc_dapm_widget *w,
     struct snd_kcontrol *kcontrol, int event)
{
 int ret;

 soc_dapm_async_complete(w->dapm);

 if (SND_SOC_DAPM_EVENT_ON(event)) {
  if (w->on_val & SND_SOC_DAPM_REGULATOR_BYPASS) {
   ret = regulator_allow_bypass(w->regulator, false);
   if (ret != 0)
    dev_warn(w->dapm->dev,
      "ASoC: Failed to unbypass %s: %d\n",
      w->name, ret);
  }

  return regulator_enable(w->regulator);
 } else {
  if (w->on_val & SND_SOC_DAPM_REGULATOR_BYPASS) {
   ret = regulator_allow_bypass(w->regulator, true);
   if (ret != 0)
    dev_warn(w->dapm->dev,
      "ASoC: Failed to bypass %s: %d\n",
      w->name, ret);
  }

  return regulator_disable_deferred(w->regulator, w->shift);
 }
}
EXPORT_SYMBOL_GPL(snd_soc_dapm_regulator_event);

/*
 * Handler for pinctrl widget.
 */

int snd_soc_dapm_pinctrl_event(struct snd_soc_dapm_widget *w,
          struct snd_kcontrol *kcontrol, int event)
{
 struct snd_soc_dapm_pinctrl_priv *priv = w->priv;
 struct pinctrl *p = w->pinctrl;
 struct pinctrl_state *s;

 if (!p || !priv)
  return -EIO;

 if (SND_SOC_DAPM_EVENT_ON(event))
  s = pinctrl_lookup_state(p, priv->active_state);
 else
  s = pinctrl_lookup_state(p, priv->sleep_state);

 if (IS_ERR(s))
  return PTR_ERR(s);

 return pinctrl_select_state(p, s);
}
EXPORT_SYMBOL_GPL(snd_soc_dapm_pinctrl_event);

/*
 * Handler for clock supply widget.
 */

int snd_soc_dapm_clock_event(struct snd_soc_dapm_widget *w,
        struct snd_kcontrol *kcontrol, int event)
{
 if (!w->clk)
  return -EIO;

 soc_dapm_async_complete(w->dapm);

 if (SND_SOC_DAPM_EVENT_ON(event)) {
  return clk_prepare_enable(w->clk);
 } else {
  clk_disable_unprepare(w->clk);
  return 0;
 }

 return 0;
}
EXPORT_SYMBOL_GPL(snd_soc_dapm_clock_event);

static int dapm_widget_power_check(struct snd_soc_dapm_widget *w)
{
 if (w->power_checked)
  return w->new_power;

 if (w->force)
  w->new_power = 1;
 else
  w->new_power = w->power_check(w);

 w->power_checked = true;

 return w->new_power;
}

/* Generic check to see if a widget should be powered. */
static int dapm_generic_check_power(struct snd_soc_dapm_widget *w)
{
 int in, out;

 DAPM_UPDATE_STAT(w, power_checks);

 in = is_connected_input_ep(w, NULL, NULL);
 out = is_connected_output_ep(w, NULL, NULL);
 return out != 0 && in != 0;
}

/* Check to see if a power supply is needed */
static int dapm_supply_check_power(struct snd_soc_dapm_widget *w)
{
 struct snd_soc_dapm_path *path;

 DAPM_UPDATE_STAT(w, power_checks);

 /* Check if one of our outputs is connected */
 snd_soc_dapm_widget_for_each_sink_path(w, path) {
  DAPM_UPDATE_STAT(w, neighbour_checks);

  if (path->connected &&
      !path->connected(path->source, path->sink))
   continue;

  if (dapm_widget_power_check(path->sink))
   return 1;
 }

 return 0;
}

static int dapm_always_on_check_power(struct snd_soc_dapm_widget *w)
{
 return w->connected;
}

static int dapm_seq_compare(struct snd_soc_dapm_widget *a,
       struct snd_soc_dapm_widget *b,
       bool power_up)
{
 int *sort;

 BUILD_BUG_ON(ARRAY_SIZE(dapm_up_seq) != SND_SOC_DAPM_TYPE_COUNT);
 BUILD_BUG_ON(ARRAY_SIZE(dapm_down_seq) != SND_SOC_DAPM_TYPE_COUNT);

 if (power_up)
  sort = dapm_up_seq;
 else
  sort = dapm_down_seq;

 WARN_ONCE(sort[a->id] == 0, "offset a->id %d not initialized\n", a->id);
 WARN_ONCE(sort[b->id] == 0, "offset b->id %d not initialized\n", b->id);

 if (sort[a->id] != sort[b->id])
  return sort[a->id] - sort[b->id];
 if (a->subseq != b->subseq) {
  if (power_up)
   return a->subseq - b->subseq;
  else
   return b->subseq - a->subseq;
 }
 if (a->reg != b->reg)
  return a->reg - b->reg;
 if (a->dapm != b->dapm)
  return (unsigned long)a->dapm - (unsigned long)b->dapm;

 return 0;
}

/* Insert a widget in order into a DAPM power sequence. */
static void dapm_seq_insert(struct snd_soc_dapm_widget *new_widget,
       struct list_head *list,
       bool power_up)
{
 struct snd_soc_dapm_widget *w;

 list_for_each_entry(w, list, power_list)
  if (dapm_seq_compare(new_widget, w, power_up) < 0) {
   list_add_tail(&new_widget->power_list, &w->power_list);
   return;
  }

 list_add_tail(&new_widget->power_list, list);
}

static void dapm_seq_check_event(struct snd_soc_card *card,
     struct snd_soc_dapm_widget *w, int event)
{
 const char *ev_name;
 int power;

 switch (event) {
 case SND_SOC_DAPM_PRE_PMU:
  ev_name = "PRE_PMU";
  power = 1;
  break;
 case SND_SOC_DAPM_POST_PMU:
  ev_name = "POST_PMU";
  power = 1;
  break;
 case SND_SOC_DAPM_PRE_PMD:
  ev_name = "PRE_PMD";
  power = 0;
  break;
 case SND_SOC_DAPM_POST_PMD:
  ev_name = "POST_PMD";
  power = 0;
  break;
 case SND_SOC_DAPM_WILL_PMU:
  ev_name = "WILL_PMU";
  power = 1;
  break;
 case SND_SOC_DAPM_WILL_PMD:
  ev_name = "WILL_PMD";
  power = 0;
  break;
 default:
  WARN(1, "Unknown event %d\n", event);
  return;
 }

 if (w->new_power != power)
  return;

 if (w->event && (w->event_flags & event)) {
  int ret;

  pop_dbg(w->dapm->dev, card->pop_time, "pop test : %s %s\n",
   w->name, ev_name);
  soc_dapm_async_complete(w->dapm);
  trace_snd_soc_dapm_widget_event_start(w, event);
  ret = w->event(w, NULL, event);
  trace_snd_soc_dapm_widget_event_done(w, event);
  if (ret < 0)
   dev_err(w->dapm->dev, "ASoC: %s: %s event failed: %d\n",
          ev_name, w->name, ret);
 }
}

/* Apply the coalesced changes from a DAPM sequence */
static void dapm_seq_run_coalesced(struct snd_soc_card *card,
       struct list_head *pending)
{
 struct snd_soc_dapm_context *dapm;
 struct snd_soc_dapm_widget *w;
 int reg;
 unsigned int value = 0;
 unsigned int mask = 0;

 w = list_first_entry(pending, struct snd_soc_dapm_widget, power_list);
 reg = w->reg;
 dapm = w->dapm;

 list_for_each_entry(w, pending, power_list) {
  WARN_ON(reg != w->reg || dapm != w->dapm);
  w->power = w->new_power;

  mask |= w->mask << w->shift;
  if (w->power)
   value |= w->on_val << w->shift;
  else
   value |= w->off_val << w->shift;

  pop_dbg(dapm->dev, card->pop_time,
   "pop test : Queue %s: reg=0x%x, 0x%x/0x%x\n",
   w->name, reg, value, mask);

  /* Check for events */
  dapm_seq_check_event(card, w, SND_SOC_DAPM_PRE_PMU);
  dapm_seq_check_event(card, w, SND_SOC_DAPM_PRE_PMD);
 }

 if (reg >= 0) {
  /* Any widget will do, they should all be updating the
 * same register.
 */


  pop_dbg(dapm->dev, card->pop_time,
   "pop test : Applying 0x%x/0x%x to %x in %dms\n",
   value, mask, reg, card->pop_time);
  pop_wait(card->pop_time);
  soc_dapm_update_bits(dapm, reg, mask, value);
 }

 list_for_each_entry(w, pending, power_list) {
  dapm_seq_check_event(card, w, SND_SOC_DAPM_POST_PMU);
  dapm_seq_check_event(card, w, SND_SOC_DAPM_POST_PMD);
 }
}

/* Apply a DAPM power sequence.
 *
 * We walk over a pre-sorted list of widgets to apply power to.  In
 * order to minimise the number of writes to the device required
 * multiple widgets will be updated in a single write where possible.
 * Currently anything that requires more than a single write is not
 * handled.
 */

static void dapm_seq_run(struct snd_soc_card *card,
 struct list_head *list, int event, bool power_up)
{
 struct snd_soc_dapm_widget *w, *n;
 struct snd_soc_dapm_context *d;
 LIST_HEAD(pending);
 int cur_sort = -1;
 int cur_subseq = -1;
 int cur_reg = SND_SOC_NOPM;
 struct snd_soc_dapm_context *cur_dapm = NULL;
 int i;
 int *sort;

 if (power_up)
  sort = dapm_up_seq;
 else
  sort = dapm_down_seq;

 list_for_each_entry_safe(w, n, list, power_list) {
  int ret = 0;

  /* Do we need to apply any queued changes? */
  if (sort[w->id] != cur_sort || w->reg != cur_reg ||
      w->dapm != cur_dapm || w->subseq != cur_subseq) {
   if (!list_empty(&pending))
    dapm_seq_run_coalesced(card, &pending);

   if (cur_dapm && cur_dapm->component) {
    for (i = 0; i < ARRAY_SIZE(dapm_up_seq); i++)
     if (sort[i] == cur_sort)
      snd_soc_component_seq_notifier(
       cur_dapm->component,
       i, cur_subseq);
   }

   if (cur_dapm && w->dapm != cur_dapm)
    soc_dapm_async_complete(cur_dapm);

   INIT_LIST_HEAD(&pending);
   cur_sort = -1;
   cur_subseq = INT_MIN;
   cur_reg = SND_SOC_NOPM;
   cur_dapm = NULL;
  }

  switch (w->id) {
  case snd_soc_dapm_pre:
   if (!w->event)
    continue;

   if (event == SND_SOC_DAPM_STREAM_START)
    ret = w->event(w,
            NULL, SND_SOC_DAPM_PRE_PMU);
   else if (event == SND_SOC_DAPM_STREAM_STOP)
    ret = w->event(w,
            NULL, SND_SOC_DAPM_PRE_PMD);
   break;

  case snd_soc_dapm_post:
   if (!w->event)
    continue;

   if (event == SND_SOC_DAPM_STREAM_START)
    ret = w->event(w,
            NULL, SND_SOC_DAPM_POST_PMU);
   else if (event == SND_SOC_DAPM_STREAM_STOP)
    ret = w->event(w,
            NULL, SND_SOC_DAPM_POST_PMD);
   break;

  default:
   /* Queue it up for application */
   cur_sort = sort[w->id];
   cur_subseq = w->subseq;
   cur_reg = w->reg;
   cur_dapm = w->dapm;
   list_move(&w->power_list, &pending);
   break;
  }

  if (ret < 0)
   dev_err(w->dapm->dev,
    "ASoC: Failed to apply widget power: %d\n", ret);
 }

 if (!list_empty(&pending))
  dapm_seq_run_coalesced(card, &pending);

 if (cur_dapm && cur_dapm->component) {
  for (i = 0; i < ARRAY_SIZE(dapm_up_seq); i++)
   if (sort[i] == cur_sort)
    snd_soc_component_seq_notifier(
     cur_dapm->component,
     i, cur_subseq);
 }

 for_each_card_dapms(card, d)
  soc_dapm_async_complete(d);
}

static void dapm_widget_update(struct snd_soc_card *card, struct snd_soc_dapm_update *update)
{
 struct snd_soc_dapm_widget_list *wlist;
 struct snd_soc_dapm_widget *w = NULL;
 unsigned int wi;
 int ret;

 if (!update || !dapm_kcontrol_is_powered(update->kcontrol))
  return;

 wlist = dapm_kcontrol_get_wlist(update->kcontrol);

 for_each_dapm_widgets(wlist, wi, w) {
  if (w->event && (w->event_flags & SND_SOC_DAPM_PRE_REG)) {
   ret = w->event(w, update->kcontrol, SND_SOC_DAPM_PRE_REG);
   if (ret != 0)
    dev_err(w->dapm->dev, "ASoC: %s DAPM pre-event failed: %d\n",
        w->name, ret);
  }
 }

 if (!w)
  return;

 ret = soc_dapm_update_bits(w->dapm, update->reg, update->mask,
  update->val);
 if (ret < 0)
  dev_err(w->dapm->dev, "ASoC: %s DAPM update failed: %d\n",
   w->name, ret);

 if (update->has_second_set) {
  ret = soc_dapm_update_bits(w->dapm, update->reg2,
        update->mask2, update->val2);
  if (ret < 0)
   dev_err(w->dapm->dev,
    "ASoC: %s DAPM update failed: %d\n",
    w->name, ret);
 }

 for_each_dapm_widgets(wlist, wi, w) {
  if (w->event && (w->event_flags & SND_SOC_DAPM_POST_REG)) {
   ret = w->event(w, update->kcontrol, SND_SOC_DAPM_POST_REG);
   if (ret != 0)
    dev_err(w->dapm->dev, "ASoC: %s DAPM post-event failed: %d\n",
        w->name, ret);
  }
 }
}

/* Async callback run prior to DAPM sequences - brings to _PREPARE if
 * they're changing state.
 */

static void dapm_pre_sequence_async(void *data, async_cookie_t cookie)
{
 struct snd_soc_dapm_context *dapm = data;
 int ret;

 /* If we're off and we're not supposed to go into STANDBY */
 if (dapm->bias_level == SND_SOC_BIAS_OFF &&
     dapm->target_bias_level != SND_SOC_BIAS_OFF) {
  if (dapm->dev && cookie)
   pm_runtime_get_sync(dapm->dev);

  ret = snd_soc_dapm_set_bias_level(dapm, SND_SOC_BIAS_STANDBY);
  if (ret != 0)
   dev_err(dapm->dev,
    "ASoC: Failed to turn on bias: %d\n", ret);
 }

 /* Prepare for a transition to ON or away from ON */
 if ((dapm->target_bias_level == SND_SOC_BIAS_ON &&
      dapm->bias_level != SND_SOC_BIAS_ON) ||
     (dapm->target_bias_level != SND_SOC_BIAS_ON &&
      dapm->bias_level == SND_SOC_BIAS_ON)) {
  ret = snd_soc_dapm_set_bias_level(dapm, SND_SOC_BIAS_PREPARE);
  if (ret != 0)
   dev_err(dapm->dev,
    "ASoC: Failed to prepare bias: %d\n", ret);
 }
}

/* Async callback run prior to DAPM sequences - brings to their final
 * state.
 */

static void dapm_post_sequence_async(void *data, async_cookie_t cookie)
{
 struct snd_soc_dapm_context *dapm = data;
 int ret;

 /* If we just powered the last thing off drop to standby bias */
 if (dapm->bias_level == SND_SOC_BIAS_PREPARE &&
     (dapm->target_bias_level == SND_SOC_BIAS_STANDBY ||
      dapm->target_bias_level == SND_SOC_BIAS_OFF)) {
  ret = snd_soc_dapm_set_bias_level(dapm, SND_SOC_BIAS_STANDBY);
  if (ret != 0)
   dev_err(dapm->dev, "ASoC: Failed to apply standby bias: %d\n",
    ret);
 }

 /* If we're in standby and can support bias off then do that */
 if (dapm->bias_level == SND_SOC_BIAS_STANDBY &&
     dapm->target_bias_level == SND_SOC_BIAS_OFF) {
  ret = snd_soc_dapm_set_bias_level(dapm, SND_SOC_BIAS_OFF);
  if (ret != 0)
   dev_err(dapm->dev, "ASoC: Failed to turn off bias: %d\n",
    ret);

  if (dapm->dev && cookie)
   pm_runtime_put(dapm->dev);
 }

 /* If we just powered up then move to active bias */
 if (dapm->bias_level == SND_SOC_BIAS_PREPARE &&
     dapm->target_bias_level == SND_SOC_BIAS_ON) {
  ret = snd_soc_dapm_set_bias_level(dapm, SND_SOC_BIAS_ON);
  if (ret != 0)
   dev_err(dapm->dev, "ASoC: Failed to apply active bias: %d\n",
    ret);
 }
}

static void dapm_widget_set_peer_power(struct snd_soc_dapm_widget *peer,
           bool power, bool connect)
{
 /* If a connection is being made or broken then that update
 * will have marked the peer dirty, otherwise the widgets are
 * not connected and this update has no impact. */

 if (!connect)
  return;

 /* If the peer is already in the state we're moving to then we
 * won't have an impact on it. */

 if (power != peer->power)
  dapm_mark_dirty(peer, "peer state change");
}

static void dapm_power_one_widget(struct snd_soc_dapm_widget *w,
      struct list_head *up_list,
      struct list_head *down_list)
{
 struct snd_soc_dapm_path *path;
 int power;

 switch (w->id) {
 case snd_soc_dapm_pre:
  power = 0;
  goto end;
 case snd_soc_dapm_post:
  power = 1;
  goto end;
 default:
  break;
 }

 power = dapm_widget_power_check(w);

 if (w->power == power)
  return;

 trace_snd_soc_dapm_widget_power(w, power);

 /*
 * If we changed our power state perhaps our neigbours
 * changed also.
 */

 snd_soc_dapm_widget_for_each_source_path(w, path)
  dapm_widget_set_peer_power(path->source, power, path->connect);

 /*
 * Supplies can't affect their outputs, only their inputs
 */

 if (!w->is_supply)
  snd_soc_dapm_widget_for_each_sink_path(w, path)
   dapm_widget_set_peer_power(path->sink, power, path->connect);

end:
 if (power)
  dapm_seq_insert(w, up_list, true);
 else
  dapm_seq_insert(w, down_list, false);
}

static bool dapm_idle_bias_off(struct snd_soc_dapm_context *dapm)
{
 if (dapm->idle_bias_off)
  return true;

 switch (snd_power_get_state(dapm->card->snd_card)) {
 case SNDRV_CTL_POWER_D3hot:
 case SNDRV_CTL_POWER_D3cold:
  return dapm->suspend_bias_off;
 default:
  break;
 }

 return false;
}

/*
 * Scan each dapm widget for complete audio path.
 * A complete path is a route that has valid endpoints i.e.:-
 *
 *  o DAC to output pin.
 *  o Input pin to ADC.
 *  o Input pin to Output pin (bypass, sidetone)
 *  o DAC to ADC (loopback).
 */

static int dapm_power_widgets(struct snd_soc_card *card, int event,
         struct snd_soc_dapm_update *update)
{
 struct snd_soc_dapm_widget *w;
 struct snd_soc_dapm_context *d;
 LIST_HEAD(up_list);
 LIST_HEAD(down_list);
 ASYNC_DOMAIN_EXCLUSIVE(async_domain);
 enum snd_soc_bias_level bias;
 int ret;

 snd_soc_dapm_mutex_assert_held(card);

 trace_snd_soc_dapm_start(card, event);

 for_each_card_dapms(card, d) {
  if (dapm_idle_bias_off(d))
   d->target_bias_level = SND_SOC_BIAS_OFF;
  else
   d->target_bias_level = SND_SOC_BIAS_STANDBY;
 }

 dapm_reset(card);

 /* Check which widgets we need to power and store them in
 * lists indicating if they should be powered up or down.  We
 * only check widgets that have been flagged as dirty but note
 * that new widgets may be added to the dirty list while we
 * iterate.
 */

 list_for_each_entry(w, &card->dapm_dirty, dirty) {
  dapm_power_one_widget(w, &up_list, &down_list);
 }

 for_each_card_widgets(card, w) {
  switch (w->id) {
  case snd_soc_dapm_pre:
  case snd_soc_dapm_post:
   /* These widgets always need to be powered */
   break;
  default:
   list_del_init(&w->dirty);
   break;
  }

  if (w->new_power) {
   d = w->dapm;

   /* Supplies and micbiases only bring the
 * context up to STANDBY as unless something
 * else is active and passing audio they
 * generally don't require full power.  Signal
 * generators are virtual pins and have no
 * power impact themselves.
 */

   switch (w->id) {
   case snd_soc_dapm_siggen:
   case snd_soc_dapm_vmid:
    break;
   case snd_soc_dapm_supply:
   case snd_soc_dapm_regulator_supply:
   case snd_soc_dapm_pinctrl:
   case snd_soc_dapm_clock_supply:
   case snd_soc_dapm_micbias:
    if (d->target_bias_level < SND_SOC_BIAS_STANDBY)
     d->target_bias_level = SND_SOC_BIAS_STANDBY;
    break;
   default:
    d->target_bias_level = SND_SOC_BIAS_ON;
    break;
   }
  }

 }

 /* Force all contexts in the card to the same bias state if
 * they're not ground referenced.
 */

 bias = SND_SOC_BIAS_OFF;
 for_each_card_dapms(card, d)
  if (d->target_bias_level > bias)
   bias = d->target_bias_level;
 for_each_card_dapms(card, d)
  if (!dapm_idle_bias_off(d))
   d->target_bias_level = bias;

 trace_snd_soc_dapm_walk_done(card);

 /* Run card bias changes at first */
 dapm_pre_sequence_async(&card->dapm, 0);
 /* Run other bias changes in parallel */
 for_each_card_dapms(card, d) {
  if (d != &card->dapm && d->bias_level != d->target_bias_level)
   async_schedule_domain(dapm_pre_sequence_async, d,
      &async_domain);
 }
 async_synchronize_full_domain(&async_domain);

 list_for_each_entry(w, &down_list, power_list) {
  dapm_seq_check_event(card, w, SND_SOC_DAPM_WILL_PMD);
 }

 list_for_each_entry(w, &up_list, power_list) {
  dapm_seq_check_event(card, w, SND_SOC_DAPM_WILL_PMU);
 }

 /* Power down widgets first; try to avoid amplifying pops. */
 dapm_seq_run(card, &down_list, event, false);

 dapm_widget_update(card, update);

 /* Now power up. */
 dapm_seq_run(card, &up_list, event, true);

 /* Run all the bias changes in parallel */
 for_each_card_dapms(card, d) {
  if (d != &card->dapm && d->bias_level != d->target_bias_level)
   async_schedule_domain(dapm_post_sequence_async, d,
      &async_domain);
 }
 async_synchronize_full_domain(&async_domain);
 /* Run card bias changes at last */
 dapm_post_sequence_async(&card->dapm, 0);

 /* do we need to notify any clients that DAPM event is complete */
 for_each_card_dapms(card, d) {
  if (!d->component)
   continue;

  ret = snd_soc_component_stream_event(d->component, event);
  if (ret < 0)
   return ret;
 }

 pop_dbg(card->dev, card->pop_time,
  "DAPM sequencing finished, waiting %dms\n", card->pop_time);
 pop_wait(card->pop_time);

 trace_snd_soc_dapm_done(card, event);

 return 0;
}

#ifdef CONFIG_DEBUG_FS

static const char * const snd_soc_dapm_type_name[] = {
 [snd_soc_dapm_input]            = "input",
 [snd_soc_dapm_output]           = "output",
 [snd_soc_dapm_mux]              = "mux",
 [snd_soc_dapm_demux]            = "demux",
 [snd_soc_dapm_mixer]            = "mixer",
 [snd_soc_dapm_mixer_named_ctl]  = "mixer_named_ctl",
 [snd_soc_dapm_pga]              = "pga",
 [snd_soc_dapm_out_drv]          = "out_drv",
 [snd_soc_dapm_adc]              = "adc",
 [snd_soc_dapm_dac]              = "dac",
 [snd_soc_dapm_micbias]          = "micbias",
 [snd_soc_dapm_mic]              = "mic",
 [snd_soc_dapm_hp]               = "hp",
 [snd_soc_dapm_spk]              = "spk",
 [snd_soc_dapm_line]             = "line",
 [snd_soc_dapm_switch]           = "switch",
 [snd_soc_dapm_vmid]             = "vmid",
 [snd_soc_dapm_pre]              = "pre",
 [snd_soc_dapm_post]             = "post",
 [snd_soc_dapm_supply]           = "supply",
 [snd_soc_dapm_pinctrl]          = "pinctrl",
 [snd_soc_dapm_regulator_supply] = "regulator_supply",
 [snd_soc_dapm_clock_supply]     = "clock_supply",
 [snd_soc_dapm_aif_in]           = "aif_in",
 [snd_soc_dapm_aif_out]          = "aif_out",
 [snd_soc_dapm_siggen]           = "siggen",
 [snd_soc_dapm_sink]             = "sink",
 [snd_soc_dapm_dai_in]           = "dai_in",
 [snd_soc_dapm_dai_out]          = "dai_out",
 [snd_soc_dapm_dai_link]         = "dai_link",
 [snd_soc_dapm_kcontrol]         = "kcontrol",
 [snd_soc_dapm_buffer]           = "buffer",
 [snd_soc_dapm_scheduler]        = "scheduler",
 [snd_soc_dapm_effect]           = "effect",
 [snd_soc_dapm_src]              = "src",
 [snd_soc_dapm_asrc]             = "asrc",
 [snd_soc_dapm_encoder]          = "encoder",
 [snd_soc_dapm_decoder]          = "decoder",
};

static ssize_t dapm_widget_power_read_file(struct file *file,
        char __user *user_buf,
        size_t count, loff_t *ppos)
{
 struct snd_soc_dapm_widget *w = file->private_data;
 enum snd_soc_dapm_direction dir, rdir;
 char *buf;
 int in, out;
 ssize_t ret;
 struct snd_soc_dapm_path *p = NULL;
 const char *c_name;

 BUILD_BUG_ON(ARRAY_SIZE(snd_soc_dapm_type_name) != SND_SOC_DAPM_TYPE_COUNT);

 buf = kmalloc(PAGE_SIZE, GFP_KERNEL);
 if (!buf)
  return -ENOMEM;

 snd_soc_dapm_mutex_lock_root(w->dapm);

 /* Supply widgets are not handled by is_connected_{input,output}_ep() */
 if (w->is_supply) {
  in = 0;
  out = 0;
 } else {
  in = is_connected_input_ep(w, NULL, NULL);
  out = is_connected_output_ep(w, NULL, NULL);
 }

 ret = scnprintf(buf, PAGE_SIZE, "%s: %s%s in %d out %d",
         w->name, w->power ? "On" : "Off",
         w->force ? " (forced)" : "", in, out);

 if (w->reg >= 0)
  ret += scnprintf(buf + ret, PAGE_SIZE - ret,
    " - R%d(0x%x) mask 0x%x",
    w->reg, w->reg, w->mask << w->shift);

 ret += scnprintf(buf + ret, PAGE_SIZE - ret, "\n");

 if (w->sname)
  ret += scnprintf(buf + ret, PAGE_SIZE - ret, " stream %s %s\n",
    w->sname,
    w->active ? "active" : "inactive");

 ret += scnprintf(buf + ret, PAGE_SIZE - ret, " widget-type %s\n",
    snd_soc_dapm_type_name[w->id]);

 snd_soc_dapm_for_each_direction(dir) {
  rdir = SND_SOC_DAPM_DIR_REVERSE(dir);
  snd_soc_dapm_widget_for_each_path(w, dir, p) {
   if (p->connected && !p->connected(p->source, p->sink))
    continue;

   if (!p->connect)
    continue;

   c_name = p->node[rdir]->dapm->component ?
    p->node[rdir]->dapm->component->name : NULL;
   ret += scnprintf(buf + ret, PAGE_SIZE - ret,
     " %s \"%s\" \"%s\" \"%s\"\n",
     (rdir == SND_SOC_DAPM_DIR_IN) ? "in" : "out",
     p->name ? p->name : "static",
     p->node[rdir]->name, c_name);
  }
 }

 snd_soc_dapm_mutex_unlock(w->dapm);

 ret = simple_read_from_buffer(user_buf, count, ppos, buf, ret);

 kfree(buf);
 return ret;
}

static const struct file_operations dapm_widget_power_fops = {
 .open = simple_open,
 .read = dapm_widget_power_read_file,
 .llseek = default_llseek,
};

static ssize_t dapm_bias_read_file(struct file *file, char __user *user_buf,
       size_t count, loff_t *ppos)
{
 struct snd_soc_dapm_context *dapm = file->private_data;
 char *level;

 switch (dapm->bias_level) {
 case SND_SOC_BIAS_ON:
  level = "On\n";
  break;
 case SND_SOC_BIAS_PREPARE:
  level = "Prepare\n";
  break;
 case SND_SOC_BIAS_STANDBY:
  level = "Standby\n";
  break;
 case SND_SOC_BIAS_OFF:
  level = "Off\n";
  break;
 default:
  WARN(1, "Unknown bias_level %d\n", dapm->bias_level);
  level = "Unknown\n";
  break;
 }

 return simple_read_from_buffer(user_buf, count, ppos, level,
           strlen(level));
}

static const struct file_operations dapm_bias_fops = {
 .open = simple_open,
 .read = dapm_bias_read_file,
 .llseek = default_llseek,
};

void snd_soc_dapm_debugfs_init(struct snd_soc_dapm_context *dapm,
 struct dentry *parent)
{
 if (IS_ERR_OR_NULL(parent))
  return;

 dapm->debugfs_dapm = debugfs_create_dir("dapm", parent);

 debugfs_create_file("bias_level", 0444, dapm->debugfs_dapm, dapm,
       &dapm_bias_fops);
}

static void dapm_debugfs_add_widget(struct snd_soc_dapm_widget *w)
{
 struct snd_soc_dapm_context *dapm = w->dapm;

 if (!dapm->debugfs_dapm || !w->name)
  return;

 debugfs_create_file(w->name, 0444, dapm->debugfs_dapm, w,
       &dapm_widget_power_fops);
}

static void dapm_debugfs_free_widget(struct snd_soc_dapm_widget *w)
{
 struct snd_soc_dapm_context *dapm = w->dapm;

 if (!dapm->debugfs_dapm || !w->name)
  return;

 debugfs_lookup_and_remove(w->name, dapm->debugfs_dapm);
}

static void dapm_debugfs_cleanup(struct snd_soc_dapm_context *dapm)
{
 debugfs_remove_recursive(dapm->debugfs_dapm);
 dapm->debugfs_dapm = NULL;
}

#else
void snd_soc_dapm_debugfs_init(struct snd_soc_dapm_context *dapm,
 struct dentry *parent)
{
}

static inline void dapm_debugfs_add_widget(struct snd_soc_dapm_widget *w)
{
}

static inline void dapm_debugfs_free_widget(struct snd_soc_dapm_widget *w)
{
}

static inline void dapm_debugfs_cleanup(struct snd_soc_dapm_context *dapm)
{
}

#endif

/*
 * soc_dapm_connect_path() - Connects or disconnects a path
 * @path: The path to update
 * @connect: The new connect state of the path. True if the path is connected,
 *  false if it is disconnected.
 * @reason: The reason why the path changed (for debugging only)
 */

static void soc_dapm_connect_path(struct snd_soc_dapm_path *path,
 bool connect, const char *reason)
{
 if (path->connect == connect)
  return;

 path->connect = connect;
--> --------------------

--> maximum size reached

--> --------------------

Messung V0.5
C=96 H=97 G=96

¤ Dauer der Verarbeitung: 0.25 Sekunden  (vorverarbeitet)  ¤

*© Formatika GbR, Deutschland






Wurzel

Suchen

Beweissystem der NASA

Beweissystem Isabelle

NIST Cobol Testsuite

Cephes Mathematical Library

Wiener Entwicklungsmethode

Haftungshinweis

Die Informationen auf dieser Webseite wurden nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit, noch Qualität der bereit gestellten Informationen zugesichert.

Bemerkung:

Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.