Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
openeuler
Kernel
提交
c6016bde
K
Kernel
项目概览
openeuler
/
Kernel
1 年多 前同步成功
通知
8
Star
0
Fork
0
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
0
列表
看板
标记
里程碑
合并请求
0
DevOps
流水线
流水线任务
计划
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
K
Kernel
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
0
Issue
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
DevOps
DevOps
流水线
流水线任务
计划
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
流水线任务
提交
Issue看板
提交
c6016bde
编写于
11月 08, 2013
作者:
M
Mark Brown
浏览文件
操作
浏览文件
下载
差异文件
Merge remote-tracking branch 'asoc/topic/core' into asoc-next
上级
d20b09f0
988e8cc4
变更
7
显示空白变更内容
内联
并排
Showing
7 changed file
with
177 addition
and
381 deletion
+177
-381
include/sound/soc.h
include/sound/soc.h
+3
-57
include/trace/events/asoc.h
include/trace/events/asoc.h
+1
-0
sound/soc/soc-cache.c
sound/soc/soc-cache.c
+61
-202
sound/soc/soc-core.c
sound/soc/soc-core.c
+62
-69
sound/soc/soc-io.c
sound/soc/soc-io.c
+0
-26
sound/soc/soc-pcm.c
sound/soc/soc-pcm.c
+45
-26
sound/soc/soc-utils.c
sound/soc/soc-utils.c
+5
-1
未找到文件。
include/sound/soc.h
浏览文件 @
c6016bde
...
@@ -331,7 +331,6 @@ struct soc_enum;
...
@@ -331,7 +331,6 @@ struct soc_enum;
struct
snd_soc_jack
;
struct
snd_soc_jack
;
struct
snd_soc_jack_zone
;
struct
snd_soc_jack_zone
;
struct
snd_soc_jack_pin
;
struct
snd_soc_jack_pin
;
struct
snd_soc_cache_ops
;
#include <sound/soc-dapm.h>
#include <sound/soc-dapm.h>
#include <sound/soc-dpcm.h>
#include <sound/soc-dpcm.h>
...
@@ -349,10 +348,6 @@ enum snd_soc_control_type {
...
@@ -349,10 +348,6 @@ enum snd_soc_control_type {
SND_SOC_REGMAP
,
SND_SOC_REGMAP
,
};
};
enum
snd_soc_compress_type
{
SND_SOC_FLAT_COMPRESSION
=
1
,
};
enum
snd_soc_pcm_subclass
{
enum
snd_soc_pcm_subclass
{
SND_SOC_PCM_CLASS_PCM
=
0
,
SND_SOC_PCM_CLASS_PCM
=
0
,
SND_SOC_PCM_CLASS_BE
=
1
,
SND_SOC_PCM_CLASS_BE
=
1
,
...
@@ -404,12 +399,6 @@ int snd_soc_cache_write(struct snd_soc_codec *codec,
...
@@ -404,12 +399,6 @@ int snd_soc_cache_write(struct snd_soc_codec *codec,
unsigned
int
reg
,
unsigned
int
value
);
unsigned
int
reg
,
unsigned
int
value
);
int
snd_soc_cache_read
(
struct
snd_soc_codec
*
codec
,
int
snd_soc_cache_read
(
struct
snd_soc_codec
*
codec
,
unsigned
int
reg
,
unsigned
int
*
value
);
unsigned
int
reg
,
unsigned
int
*
value
);
int
snd_soc_default_volatile_register
(
struct
snd_soc_codec
*
codec
,
unsigned
int
reg
);
int
snd_soc_default_readable_register
(
struct
snd_soc_codec
*
codec
,
unsigned
int
reg
);
int
snd_soc_default_writable_register
(
struct
snd_soc_codec
*
codec
,
unsigned
int
reg
);
int
snd_soc_platform_read
(
struct
snd_soc_platform
*
platform
,
int
snd_soc_platform_read
(
struct
snd_soc_platform
*
platform
,
unsigned
int
reg
);
unsigned
int
reg
);
int
snd_soc_platform_write
(
struct
snd_soc_platform
*
platform
,
int
snd_soc_platform_write
(
struct
snd_soc_platform
*
platform
,
...
@@ -542,22 +531,6 @@ int snd_soc_get_strobe(struct snd_kcontrol *kcontrol,
...
@@ -542,22 +531,6 @@ int snd_soc_get_strobe(struct snd_kcontrol *kcontrol,
int
snd_soc_put_strobe
(
struct
snd_kcontrol
*
kcontrol
,
int
snd_soc_put_strobe
(
struct
snd_kcontrol
*
kcontrol
,
struct
snd_ctl_elem_value
*
ucontrol
);
struct
snd_ctl_elem_value
*
ucontrol
);
/**
* struct snd_soc_reg_access - Describes whether a given register is
* readable, writable or volatile.
*
* @reg: the register number
* @read: whether this register is readable
* @write: whether this register is writable
* @vol: whether this register is volatile
*/
struct
snd_soc_reg_access
{
u16
reg
;
u16
read
;
u16
write
;
u16
vol
;
};
/**
/**
* struct snd_soc_jack_pin - Describes a pin to update based on jack detection
* struct snd_soc_jack_pin - Describes a pin to update based on jack detection
*
*
...
@@ -658,19 +631,6 @@ struct snd_soc_compr_ops {
...
@@ -658,19 +631,6 @@ struct snd_soc_compr_ops {
int
(
*
trigger
)(
struct
snd_compr_stream
*
);
int
(
*
trigger
)(
struct
snd_compr_stream
*
);
};
};
/* SoC cache ops */
struct
snd_soc_cache_ops
{
const
char
*
name
;
enum
snd_soc_compress_type
id
;
int
(
*
init
)(
struct
snd_soc_codec
*
codec
);
int
(
*
exit
)(
struct
snd_soc_codec
*
codec
);
int
(
*
read
)(
struct
snd_soc_codec
*
codec
,
unsigned
int
reg
,
unsigned
int
*
value
);
int
(
*
write
)(
struct
snd_soc_codec
*
codec
,
unsigned
int
reg
,
unsigned
int
value
);
int
(
*
sync
)(
struct
snd_soc_codec
*
codec
);
};
/* component interface */
/* component interface */
struct
snd_soc_component_driver
{
struct
snd_soc_component_driver
{
const
char
*
name
;
const
char
*
name
;
...
@@ -684,10 +644,12 @@ struct snd_soc_component_driver {
...
@@ -684,10 +644,12 @@ struct snd_soc_component_driver {
struct
snd_soc_component
{
struct
snd_soc_component
{
const
char
*
name
;
const
char
*
name
;
int
id
;
int
id
;
int
num_dai
;
struct
device
*
dev
;
struct
device
*
dev
;
struct
list_head
list
;
struct
list_head
list
;
struct
snd_soc_dai_driver
*
dai_drv
;
int
num_dai
;
const
struct
snd_soc_component_driver
*
driver
;
const
struct
snd_soc_component_driver
*
driver
;
};
};
...
@@ -704,8 +666,6 @@ struct snd_soc_codec {
...
@@ -704,8 +666,6 @@ struct snd_soc_codec {
struct
list_head
list
;
struct
list_head
list
;
struct
list_head
card_list
;
struct
list_head
card_list
;
int
num_dai
;
int
num_dai
;
enum
snd_soc_compress_type
compress_type
;
size_t
reg_size
;
/* reg_cache_size * reg_word_size */
int
(
*
volatile_register
)(
struct
snd_soc_codec
*
,
unsigned
int
);
int
(
*
volatile_register
)(
struct
snd_soc_codec
*
,
unsigned
int
);
int
(
*
readable_register
)(
struct
snd_soc_codec
*
,
unsigned
int
);
int
(
*
readable_register
)(
struct
snd_soc_codec
*
,
unsigned
int
);
int
(
*
writable_register
)(
struct
snd_soc_codec
*
,
unsigned
int
);
int
(
*
writable_register
)(
struct
snd_soc_codec
*
,
unsigned
int
);
...
@@ -729,10 +689,7 @@ struct snd_soc_codec {
...
@@ -729,10 +689,7 @@ struct snd_soc_codec {
unsigned
int
(
*
hw_read
)(
struct
snd_soc_codec
*
,
unsigned
int
);
unsigned
int
(
*
hw_read
)(
struct
snd_soc_codec
*
,
unsigned
int
);
unsigned
int
(
*
read
)(
struct
snd_soc_codec
*
,
unsigned
int
);
unsigned
int
(
*
read
)(
struct
snd_soc_codec
*
,
unsigned
int
);
int
(
*
write
)(
struct
snd_soc_codec
*
,
unsigned
int
,
unsigned
int
);
int
(
*
write
)(
struct
snd_soc_codec
*
,
unsigned
int
,
unsigned
int
);
int
(
*
bulk_write_raw
)(
struct
snd_soc_codec
*
,
unsigned
int
,
const
void
*
,
size_t
);
void
*
reg_cache
;
void
*
reg_cache
;
const
void
*
reg_def_copy
;
const
struct
snd_soc_cache_ops
*
cache_ops
;
struct
mutex
cache_rw_mutex
;
struct
mutex
cache_rw_mutex
;
int
val_bytes
;
int
val_bytes
;
...
@@ -785,9 +742,6 @@ struct snd_soc_codec_driver {
...
@@ -785,9 +742,6 @@ struct snd_soc_codec_driver {
short
reg_cache_step
;
short
reg_cache_step
;
short
reg_word_size
;
short
reg_word_size
;
const
void
*
reg_cache_default
;
const
void
*
reg_cache_default
;
short
reg_access_size
;
const
struct
snd_soc_reg_access
*
reg_access_default
;
enum
snd_soc_compress_type
compress_type
;
/* codec bias level */
/* codec bias level */
int
(
*
set_bias_level
)(
struct
snd_soc_codec
*
,
int
(
*
set_bias_level
)(
struct
snd_soc_codec
*
,
...
@@ -955,12 +909,6 @@ struct snd_soc_codec_conf {
...
@@ -955,12 +909,6 @@ struct snd_soc_codec_conf {
* associated per device
* associated per device
*/
*/
const
char
*
name_prefix
;
const
char
*
name_prefix
;
/*
* set this to the desired compression type if you want to
* override the one supplied in codec->driver->compress_type
*/
enum
snd_soc_compress_type
compress_type
;
};
};
struct
snd_soc_aux_dev
{
struct
snd_soc_aux_dev
{
...
@@ -1132,8 +1080,6 @@ struct soc_enum {
...
@@ -1132,8 +1080,6 @@ struct soc_enum {
unsigned
int
snd_soc_read
(
struct
snd_soc_codec
*
codec
,
unsigned
int
reg
);
unsigned
int
snd_soc_read
(
struct
snd_soc_codec
*
codec
,
unsigned
int
reg
);
unsigned
int
snd_soc_write
(
struct
snd_soc_codec
*
codec
,
unsigned
int
snd_soc_write
(
struct
snd_soc_codec
*
codec
,
unsigned
int
reg
,
unsigned
int
val
);
unsigned
int
reg
,
unsigned
int
val
);
unsigned
int
snd_soc_bulk_write_raw
(
struct
snd_soc_codec
*
codec
,
unsigned
int
reg
,
const
void
*
data
,
size_t
len
);
/* device driver data */
/* device driver data */
...
...
include/trace/events/asoc.h
浏览文件 @
c6016bde
...
@@ -14,6 +14,7 @@ struct snd_soc_codec;
...
@@ -14,6 +14,7 @@ struct snd_soc_codec;
struct
snd_soc_platform
;
struct
snd_soc_platform
;
struct
snd_soc_card
;
struct
snd_soc_card
;
struct
snd_soc_dapm_widget
;
struct
snd_soc_dapm_widget
;
struct
snd_soc_dapm_path
;
/*
/*
* Log register events
* Log register events
...
...
sound/soc/soc-cache.c
浏览文件 @
c6016bde
...
@@ -11,12 +11,9 @@
...
@@ -11,12 +11,9 @@
* option) any later version.
* option) any later version.
*/
*/
#include <linux/i2c.h>
#include <linux/spi/spi.h>
#include <sound/soc.h>
#include <sound/soc.h>
#include <linux/bitmap.h>
#include <linux/rbtree.h>
#include <linux/export.h>
#include <linux/export.h>
#include <linux/slab.h>
#include <trace/events/asoc.h>
#include <trace/events/asoc.h>
...
@@ -66,126 +63,42 @@ static unsigned int snd_soc_get_cache_val(const void *base, unsigned int idx,
...
@@ -66,126 +63,42 @@ static unsigned int snd_soc_get_cache_val(const void *base, unsigned int idx,
return
-
1
;
return
-
1
;
}
}
static
int
snd_soc_flat_cache_sync
(
struct
snd_soc_codec
*
codec
)
int
snd_soc_cache_init
(
struct
snd_soc_codec
*
codec
)
{
{
int
i
;
const
struct
snd_soc_codec_driver
*
codec_drv
=
codec
->
driver
;
int
ret
;
size_t
reg_size
;
const
struct
snd_soc_codec_driver
*
codec_drv
;
unsigned
int
val
;
codec_drv
=
codec
->
driver
;
for
(
i
=
0
;
i
<
codec_drv
->
reg_cache_size
;
++
i
)
{
ret
=
snd_soc_cache_read
(
codec
,
i
,
&
val
);
if
(
ret
)
return
ret
;
if
(
codec
->
reg_def_copy
)
if
(
snd_soc_get_cache_val
(
codec
->
reg_def_copy
,
i
,
codec_drv
->
reg_word_size
)
==
val
)
continue
;
WARN_ON
(
!
snd_soc_codec_writable_register
(
codec
,
i
));
ret
=
snd_soc_write
(
codec
,
i
,
val
);
if
(
ret
)
return
ret
;
dev_dbg
(
codec
->
dev
,
"ASoC: Synced register %#x, value = %#x
\n
"
,
i
,
val
);
}
return
0
;
}
static
int
snd_soc_flat_cache_write
(
struct
snd_soc_codec
*
codec
,
reg_size
=
codec_drv
->
reg_cache_size
*
codec_drv
->
reg_word_size
;
unsigned
int
reg
,
unsigned
int
value
)
{
snd_soc_set_cache_val
(
codec
->
reg_cache
,
reg
,
value
,
codec
->
driver
->
reg_word_size
);
return
0
;
}
static
int
snd_soc_flat_cache_read
(
struct
snd_soc_codec
*
codec
,
mutex_init
(
&
codec
->
cache_rw_mutex
);
unsigned
int
reg
,
unsigned
int
*
value
)
{
*
value
=
snd_soc_get_cache_val
(
codec
->
reg_cache
,
reg
,
codec
->
driver
->
reg_word_size
);
return
0
;
}
static
int
snd_soc_flat_cache_exit
(
struct
snd_soc_codec
*
codec
)
dev_dbg
(
codec
->
dev
,
"ASoC: Initializing cache for %s codec
\n
"
,
{
codec
->
name
);
if
(
!
codec
->
reg_cache
)
return
0
;
kfree
(
codec
->
reg_cache
);
codec
->
reg_cache
=
NULL
;
return
0
;
}
static
int
snd_soc_flat_cache_init
(
struct
snd_soc_codec
*
codec
)
if
(
codec_drv
->
reg_cache_default
)
{
codec
->
reg_cache
=
kmemdup
(
codec_drv
->
reg_cache_default
,
if
(
codec
->
reg_def_copy
)
reg_size
,
GFP_KERNEL
);
codec
->
reg_cache
=
kmemdup
(
codec
->
reg_def_copy
,
codec
->
reg_size
,
GFP_KERNEL
);
else
else
codec
->
reg_cache
=
kzalloc
(
codec
->
reg_size
,
GFP_KERNEL
);
codec
->
reg_cache
=
kzalloc
(
reg_size
,
GFP_KERNEL
);
if
(
!
codec
->
reg_cache
)
if
(
!
codec
->
reg_cache
)
return
-
ENOMEM
;
return
-
ENOMEM
;
return
0
;
return
0
;
}
}
/* an array of all supported compression types */
static
const
struct
snd_soc_cache_ops
cache_types
[]
=
{
/* Flat *must* be the first entry for fallback */
{
.
id
=
SND_SOC_FLAT_COMPRESSION
,
.
name
=
"flat"
,
.
init
=
snd_soc_flat_cache_init
,
.
exit
=
snd_soc_flat_cache_exit
,
.
read
=
snd_soc_flat_cache_read
,
.
write
=
snd_soc_flat_cache_write
,
.
sync
=
snd_soc_flat_cache_sync
},
};
int
snd_soc_cache_init
(
struct
snd_soc_codec
*
codec
)
{
int
i
;
for
(
i
=
0
;
i
<
ARRAY_SIZE
(
cache_types
);
++
i
)
if
(
cache_types
[
i
].
id
==
codec
->
compress_type
)
break
;
/* Fall back to flat compression */
if
(
i
==
ARRAY_SIZE
(
cache_types
))
{
dev_warn
(
codec
->
dev
,
"ASoC: Could not match compress type: %d
\n
"
,
codec
->
compress_type
);
i
=
0
;
}
mutex_init
(
&
codec
->
cache_rw_mutex
);
codec
->
cache_ops
=
&
cache_types
[
i
];
if
(
codec
->
cache_ops
->
init
)
{
if
(
codec
->
cache_ops
->
name
)
dev_dbg
(
codec
->
dev
,
"ASoC: Initializing %s cache for %s codec
\n
"
,
codec
->
cache_ops
->
name
,
codec
->
name
);
return
codec
->
cache_ops
->
init
(
codec
);
}
return
-
ENOSYS
;
}
/*
/*
* NOTE: keep in mind that this function might be called
* NOTE: keep in mind that this function might be called
* multiple times.
* multiple times.
*/
*/
int
snd_soc_cache_exit
(
struct
snd_soc_codec
*
codec
)
int
snd_soc_cache_exit
(
struct
snd_soc_codec
*
codec
)
{
{
if
(
codec
->
cache_ops
&&
codec
->
cache_ops
->
exit
)
{
dev_dbg
(
codec
->
dev
,
"ASoC: Destroying cache for %s codec
\n
"
,
if
(
codec
->
cache_ops
->
name
)
codec
->
name
);
dev_dbg
(
codec
->
dev
,
"ASoC: Destroying %s cache for %s codec
\n
"
,
if
(
!
codec
->
reg_cache
)
codec
->
cache_ops
->
name
,
codec
->
name
)
;
return
0
;
return
codec
->
cache_ops
->
exit
(
codec
);
kfree
(
codec
->
reg_cache
);
}
codec
->
reg_cache
=
NULL
;
return
-
ENOSYS
;
return
0
;
}
}
/**
/**
...
@@ -198,18 +111,15 @@ int snd_soc_cache_exit(struct snd_soc_codec *codec)
...
@@ -198,18 +111,15 @@ int snd_soc_cache_exit(struct snd_soc_codec *codec)
int
snd_soc_cache_read
(
struct
snd_soc_codec
*
codec
,
int
snd_soc_cache_read
(
struct
snd_soc_codec
*
codec
,
unsigned
int
reg
,
unsigned
int
*
value
)
unsigned
int
reg
,
unsigned
int
*
value
)
{
{
int
ret
;
if
(
!
value
)
return
-
EINVAL
;
mutex_lock
(
&
codec
->
cache_rw_mutex
);
mutex_lock
(
&
codec
->
cache_rw_mutex
);
*
value
=
snd_soc_get_cache_val
(
codec
->
reg_cache
,
reg
,
if
(
value
&&
codec
->
cache_ops
&&
codec
->
cache_ops
->
read
)
{
codec
->
driver
->
reg_word_size
);
ret
=
codec
->
cache_ops
->
read
(
codec
,
reg
,
value
);
mutex_unlock
(
&
codec
->
cache_rw_mutex
);
mutex_unlock
(
&
codec
->
cache_rw_mutex
);
return
ret
;
}
mutex_unlock
(
&
codec
->
cache_rw_mutex
);
return
0
;
return
-
ENOSYS
;
}
}
EXPORT_SYMBOL_GPL
(
snd_soc_cache_read
);
EXPORT_SYMBOL_GPL
(
snd_soc_cache_read
);
...
@@ -223,20 +133,42 @@ EXPORT_SYMBOL_GPL(snd_soc_cache_read);
...
@@ -223,20 +133,42 @@ EXPORT_SYMBOL_GPL(snd_soc_cache_read);
int
snd_soc_cache_write
(
struct
snd_soc_codec
*
codec
,
int
snd_soc_cache_write
(
struct
snd_soc_codec
*
codec
,
unsigned
int
reg
,
unsigned
int
value
)
unsigned
int
reg
,
unsigned
int
value
)
{
{
mutex_lock
(
&
codec
->
cache_rw_mutex
);
snd_soc_set_cache_val
(
codec
->
reg_cache
,
reg
,
value
,
codec
->
driver
->
reg_word_size
);
mutex_unlock
(
&
codec
->
cache_rw_mutex
);
return
0
;
}
EXPORT_SYMBOL_GPL
(
snd_soc_cache_write
);
static
int
snd_soc_flat_cache_sync
(
struct
snd_soc_codec
*
codec
)
{
int
i
;
int
ret
;
int
ret
;
const
struct
snd_soc_codec_driver
*
codec_drv
;
unsigned
int
val
;
mutex_lock
(
&
codec
->
cache_rw_mutex
);
codec_drv
=
codec
->
driver
;
for
(
i
=
0
;
i
<
codec_drv
->
reg_cache_size
;
++
i
)
{
ret
=
snd_soc_cache_read
(
codec
,
i
,
&
val
);
if
(
ret
)
return
ret
;
if
(
codec_drv
->
reg_cache_default
)
if
(
snd_soc_get_cache_val
(
codec_drv
->
reg_cache_default
,
i
,
codec_drv
->
reg_word_size
)
==
val
)
continue
;
if
(
codec
->
cache_ops
&&
codec
->
cache_ops
->
write
)
{
WARN_ON
(
!
snd_soc_codec_writable_register
(
codec
,
i
));
ret
=
codec
->
cache_ops
->
write
(
codec
,
reg
,
value
);
mutex_unlock
(
&
codec
->
cache_rw_mutex
);
ret
=
snd_soc_write
(
codec
,
i
,
val
);
if
(
ret
)
return
ret
;
return
ret
;
dev_dbg
(
codec
->
dev
,
"ASoC: Synced register %#x, value = %#x
\n
"
,
i
,
val
);
}
}
return
0
;
mutex_unlock
(
&
codec
->
cache_rw_mutex
);
return
-
ENOSYS
;
}
}
EXPORT_SYMBOL_GPL
(
snd_soc_cache_write
);
/**
/**
* snd_soc_cache_sync: Sync the register cache with the hardware.
* snd_soc_cache_sync: Sync the register cache with the hardware.
...
@@ -249,92 +181,19 @@ EXPORT_SYMBOL_GPL(snd_soc_cache_write);
...
@@ -249,92 +181,19 @@ EXPORT_SYMBOL_GPL(snd_soc_cache_write);
*/
*/
int
snd_soc_cache_sync
(
struct
snd_soc_codec
*
codec
)
int
snd_soc_cache_sync
(
struct
snd_soc_codec
*
codec
)
{
{
const
char
*
name
=
"flat"
;
int
ret
;
int
ret
;
const
char
*
name
;
if
(
!
codec
->
cache_sync
)
{
if
(
!
codec
->
cache_sync
)
return
0
;
return
0
;
}
if
(
!
codec
->
cache_ops
||
!
codec
->
cache_ops
->
sync
)
dev_dbg
(
codec
->
dev
,
"ASoC: Syncing cache for %s codec
\n
"
,
return
-
ENOSYS
;
codec
->
name
);
if
(
codec
->
cache_ops
->
name
)
name
=
codec
->
cache_ops
->
name
;
else
name
=
"unknown"
;
if
(
codec
->
cache_ops
->
name
)
dev_dbg
(
codec
->
dev
,
"ASoC: Syncing %s cache for %s codec
\n
"
,
codec
->
cache_ops
->
name
,
codec
->
name
);
trace_snd_soc_cache_sync
(
codec
,
name
,
"start"
);
trace_snd_soc_cache_sync
(
codec
,
name
,
"start"
);
ret
=
codec
->
cache_ops
->
sync
(
codec
);
ret
=
snd_soc_flat_cache_
sync
(
codec
);
if
(
!
ret
)
if
(
!
ret
)
codec
->
cache_sync
=
0
;
codec
->
cache_sync
=
0
;
trace_snd_soc_cache_sync
(
codec
,
name
,
"end"
);
trace_snd_soc_cache_sync
(
codec
,
name
,
"end"
);
return
ret
;
return
ret
;
}
}
EXPORT_SYMBOL_GPL
(
snd_soc_cache_sync
);
EXPORT_SYMBOL_GPL
(
snd_soc_cache_sync
);
static
int
snd_soc_get_reg_access_index
(
struct
snd_soc_codec
*
codec
,
unsigned
int
reg
)
{
const
struct
snd_soc_codec_driver
*
codec_drv
;
unsigned
int
min
,
max
,
index
;
codec_drv
=
codec
->
driver
;
min
=
0
;
max
=
codec_drv
->
reg_access_size
-
1
;
do
{
index
=
(
min
+
max
)
/
2
;
if
(
codec_drv
->
reg_access_default
[
index
].
reg
==
reg
)
return
index
;
if
(
codec_drv
->
reg_access_default
[
index
].
reg
<
reg
)
min
=
index
+
1
;
else
max
=
index
;
}
while
(
min
<=
max
);
return
-
1
;
}
int
snd_soc_default_volatile_register
(
struct
snd_soc_codec
*
codec
,
unsigned
int
reg
)
{
int
index
;
if
(
reg
>=
codec
->
driver
->
reg_cache_size
)
return
1
;
index
=
snd_soc_get_reg_access_index
(
codec
,
reg
);
if
(
index
<
0
)
return
0
;
return
codec
->
driver
->
reg_access_default
[
index
].
vol
;
}
EXPORT_SYMBOL_GPL
(
snd_soc_default_volatile_register
);
int
snd_soc_default_readable_register
(
struct
snd_soc_codec
*
codec
,
unsigned
int
reg
)
{
int
index
;
if
(
reg
>=
codec
->
driver
->
reg_cache_size
)
return
1
;
index
=
snd_soc_get_reg_access_index
(
codec
,
reg
);
if
(
index
<
0
)
return
0
;
return
codec
->
driver
->
reg_access_default
[
index
].
read
;
}
EXPORT_SYMBOL_GPL
(
snd_soc_default_readable_register
);
int
snd_soc_default_writable_register
(
struct
snd_soc_codec
*
codec
,
unsigned
int
reg
)
{
int
index
;
if
(
reg
>=
codec
->
driver
->
reg_cache_size
)
return
1
;
index
=
snd_soc_get_reg_access_index
(
codec
,
reg
);
if
(
index
<
0
)
return
0
;
return
codec
->
driver
->
reg_access_default
[
index
].
write
;
}
EXPORT_SYMBOL_GPL
(
snd_soc_default_writable_register
);
sound/soc/soc-core.c
浏览文件 @
c6016bde
...
@@ -662,6 +662,8 @@ int snd_soc_suspend(struct device *dev)
...
@@ -662,6 +662,8 @@ int snd_soc_suspend(struct device *dev)
codec
->
cache_sync
=
1
;
codec
->
cache_sync
=
1
;
if
(
codec
->
using_regmap
)
if
(
codec
->
using_regmap
)
regcache_mark_dirty
(
codec
->
control_data
);
regcache_mark_dirty
(
codec
->
control_data
);
/* deactivate pins to sleep state */
pinctrl_pm_select_sleep_state
(
codec
->
dev
);
break
;
break
;
default:
default:
dev_dbg
(
codec
->
dev
,
dev_dbg
(
codec
->
dev
,
...
@@ -679,6 +681,9 @@ int snd_soc_suspend(struct device *dev)
...
@@ -679,6 +681,9 @@ int snd_soc_suspend(struct device *dev)
if
(
cpu_dai
->
driver
->
suspend
&&
cpu_dai
->
driver
->
ac97_control
)
if
(
cpu_dai
->
driver
->
suspend
&&
cpu_dai
->
driver
->
ac97_control
)
cpu_dai
->
driver
->
suspend
(
cpu_dai
);
cpu_dai
->
driver
->
suspend
(
cpu_dai
);
/* deactivate pins to sleep state */
pinctrl_pm_select_sleep_state
(
cpu_dai
->
dev
);
}
}
if
(
card
->
suspend_post
)
if
(
card
->
suspend_post
)
...
@@ -807,6 +812,16 @@ int snd_soc_resume(struct device *dev)
...
@@ -807,6 +812,16 @@ int snd_soc_resume(struct device *dev)
if
(
list_empty
(
&
card
->
codec_dev_list
))
if
(
list_empty
(
&
card
->
codec_dev_list
))
return
0
;
return
0
;
/* activate pins from sleep state */
for
(
i
=
0
;
i
<
card
->
num_rtd
;
i
++
)
{
struct
snd_soc_dai
*
cpu_dai
=
card
->
rtd
[
i
].
cpu_dai
;
struct
snd_soc_dai
*
codec_dai
=
card
->
rtd
[
i
].
codec_dai
;
if
(
cpu_dai
->
active
)
pinctrl_pm_select_default_state
(
cpu_dai
->
dev
);
if
(
codec_dai
->
active
)
pinctrl_pm_select_default_state
(
codec_dai
->
dev
);
}
/* AC97 devices might have other drivers hanging off them so
/* AC97 devices might have other drivers hanging off them so
* need to resume immediately. Other drivers don't have that
* need to resume immediately. Other drivers don't have that
* problem and may take a substantial amount of time to resume
* problem and may take a substantial amount of time to resume
...
@@ -1589,17 +1604,13 @@ static void soc_remove_aux_dev(struct snd_soc_card *card, int num)
...
@@ -1589,17 +1604,13 @@ static void soc_remove_aux_dev(struct snd_soc_card *card, int num)
soc_remove_codec
(
codec
);
soc_remove_codec
(
codec
);
}
}
static
int
snd_soc_init_codec_cache
(
struct
snd_soc_codec
*
codec
,
static
int
snd_soc_init_codec_cache
(
struct
snd_soc_codec
*
codec
)
enum
snd_soc_compress_type
compress_type
)
{
{
int
ret
;
int
ret
;
if
(
codec
->
cache_init
)
if
(
codec
->
cache_init
)
return
0
;
return
0
;
/* override the compress_type if necessary */
if
(
compress_type
&&
codec
->
compress_type
!=
compress_type
)
codec
->
compress_type
=
compress_type
;
ret
=
snd_soc_cache_init
(
codec
);
ret
=
snd_soc_cache_init
(
codec
);
if
(
ret
<
0
)
{
if
(
ret
<
0
)
{
dev_err
(
codec
->
dev
,
dev_err
(
codec
->
dev
,
...
@@ -1614,8 +1625,6 @@ static int snd_soc_init_codec_cache(struct snd_soc_codec *codec,
...
@@ -1614,8 +1625,6 @@ static int snd_soc_init_codec_cache(struct snd_soc_codec *codec,
static
int
snd_soc_instantiate_card
(
struct
snd_soc_card
*
card
)
static
int
snd_soc_instantiate_card
(
struct
snd_soc_card
*
card
)
{
{
struct
snd_soc_codec
*
codec
;
struct
snd_soc_codec
*
codec
;
struct
snd_soc_codec_conf
*
codec_conf
;
enum
snd_soc_compress_type
compress_type
;
struct
snd_soc_dai_link
*
dai_link
;
struct
snd_soc_dai_link
*
dai_link
;
int
ret
,
i
,
order
,
dai_fmt
;
int
ret
,
i
,
order
,
dai_fmt
;
...
@@ -1639,19 +1648,7 @@ static int snd_soc_instantiate_card(struct snd_soc_card *card)
...
@@ -1639,19 +1648,7 @@ static int snd_soc_instantiate_card(struct snd_soc_card *card)
list_for_each_entry
(
codec
,
&
codec_list
,
list
)
{
list_for_each_entry
(
codec
,
&
codec_list
,
list
)
{
if
(
codec
->
cache_init
)
if
(
codec
->
cache_init
)
continue
;
continue
;
/* by default we don't override the compress_type */
ret
=
snd_soc_init_codec_cache
(
codec
);
compress_type
=
0
;
/* check to see if we need to override the compress_type */
for
(
i
=
0
;
i
<
card
->
num_configs
;
++
i
)
{
codec_conf
=
&
card
->
codec_conf
[
i
];
if
(
!
strcmp
(
codec
->
name
,
codec_conf
->
dev_name
))
{
compress_type
=
codec_conf
->
compress_type
;
if
(
compress_type
&&
compress_type
!=
codec
->
compress_type
)
break
;
}
}
ret
=
snd_soc_init_codec_cache
(
codec
,
compress_type
);
if
(
ret
<
0
)
if
(
ret
<
0
)
goto
base_error
;
goto
base_error
;
}
}
...
@@ -1947,6 +1944,14 @@ int snd_soc_poweroff(struct device *dev)
...
@@ -1947,6 +1944,14 @@ int snd_soc_poweroff(struct device *dev)
snd_soc_dapm_shutdown
(
card
);
snd_soc_dapm_shutdown
(
card
);
/* deactivate pins to sleep state */
for
(
i
=
0
;
i
<
card
->
num_rtd
;
i
++
)
{
struct
snd_soc_dai
*
cpu_dai
=
card
->
rtd
[
i
].
cpu_dai
;
struct
snd_soc_dai
*
codec_dai
=
card
->
rtd
[
i
].
codec_dai
;
pinctrl_pm_select_sleep_state
(
codec_dai
->
dev
);
pinctrl_pm_select_sleep_state
(
cpu_dai
->
dev
);
}
return
0
;
return
0
;
}
}
EXPORT_SYMBOL_GPL
(
snd_soc_poweroff
);
EXPORT_SYMBOL_GPL
(
snd_soc_poweroff
);
...
@@ -2297,13 +2302,6 @@ unsigned int snd_soc_write(struct snd_soc_codec *codec,
...
@@ -2297,13 +2302,6 @@ unsigned int snd_soc_write(struct snd_soc_codec *codec,
}
}
EXPORT_SYMBOL_GPL
(
snd_soc_write
);
EXPORT_SYMBOL_GPL
(
snd_soc_write
);
unsigned
int
snd_soc_bulk_write_raw
(
struct
snd_soc_codec
*
codec
,
unsigned
int
reg
,
const
void
*
data
,
size_t
len
)
{
return
codec
->
bulk_write_raw
(
codec
,
reg
,
data
,
len
);
}
EXPORT_SYMBOL_GPL
(
snd_soc_bulk_write_raw
);
/**
/**
* snd_soc_update_bits - update codec register bits
* snd_soc_update_bits - update codec register bits
* @codec: audio codec
* @codec: audio codec
...
@@ -2576,8 +2574,9 @@ int snd_soc_info_enum_double(struct snd_kcontrol *kcontrol,
...
@@ -2576,8 +2574,9 @@ int snd_soc_info_enum_double(struct snd_kcontrol *kcontrol,
if
(
uinfo
->
value
.
enumerated
.
item
>
e
->
max
-
1
)
if
(
uinfo
->
value
.
enumerated
.
item
>
e
->
max
-
1
)
uinfo
->
value
.
enumerated
.
item
=
e
->
max
-
1
;
uinfo
->
value
.
enumerated
.
item
=
e
->
max
-
1
;
strcpy
(
uinfo
->
value
.
enumerated
.
name
,
strlcpy
(
uinfo
->
value
.
enumerated
.
name
,
e
->
texts
[
uinfo
->
value
.
enumerated
.
item
]);
e
->
texts
[
uinfo
->
value
.
enumerated
.
item
],
sizeof
(
uinfo
->
value
.
enumerated
.
name
));
return
0
;
return
0
;
}
}
EXPORT_SYMBOL_GPL
(
snd_soc_info_enum_double
);
EXPORT_SYMBOL_GPL
(
snd_soc_info_enum_double
);
...
@@ -3791,6 +3790,16 @@ int snd_soc_register_card(struct snd_soc_card *card)
...
@@ -3791,6 +3790,16 @@ int snd_soc_register_card(struct snd_soc_card *card)
if
(
ret
!=
0
)
if
(
ret
!=
0
)
soc_cleanup_card_debugfs
(
card
);
soc_cleanup_card_debugfs
(
card
);
/* deactivate pins to sleep state */
for
(
i
=
0
;
i
<
card
->
num_rtd
;
i
++
)
{
struct
snd_soc_dai
*
cpu_dai
=
card
->
rtd
[
i
].
cpu_dai
;
struct
snd_soc_dai
*
codec_dai
=
card
->
rtd
[
i
].
codec_dai
;
if
(
!
codec_dai
->
active
)
pinctrl_pm_select_sleep_state
(
codec_dai
->
dev
);
if
(
!
cpu_dai
->
active
)
pinctrl_pm_select_sleep_state
(
cpu_dai
->
dev
);
}
return
ret
;
return
ret
;
}
}
EXPORT_SYMBOL_GPL
(
snd_soc_register_card
);
EXPORT_SYMBOL_GPL
(
snd_soc_register_card
);
...
@@ -4063,6 +4072,7 @@ __snd_soc_register_component(struct device *dev,
...
@@ -4063,6 +4072,7 @@ __snd_soc_register_component(struct device *dev,
cmpnt
->
dev
=
dev
;
cmpnt
->
dev
=
dev
;
cmpnt
->
driver
=
cmpnt_drv
;
cmpnt
->
driver
=
cmpnt_drv
;
cmpnt
->
dai_drv
=
dai_drv
;
cmpnt
->
num_dai
=
num_dai
;
cmpnt
->
num_dai
=
num_dai
;
/*
/*
...
@@ -4287,7 +4297,6 @@ int snd_soc_register_codec(struct device *dev,
...
@@ -4287,7 +4297,6 @@ int snd_soc_register_codec(struct device *dev,
struct
snd_soc_dai_driver
*
dai_drv
,
struct
snd_soc_dai_driver
*
dai_drv
,
int
num_dai
)
int
num_dai
)
{
{
size_t
reg_size
;
struct
snd_soc_codec
*
codec
;
struct
snd_soc_codec
*
codec
;
int
ret
,
i
;
int
ret
,
i
;
...
@@ -4304,11 +4313,6 @@ int snd_soc_register_codec(struct device *dev,
...
@@ -4304,11 +4313,6 @@ int snd_soc_register_codec(struct device *dev,
goto
fail_codec
;
goto
fail_codec
;
}
}
if
(
codec_drv
->
compress_type
)
codec
->
compress_type
=
codec_drv
->
compress_type
;
else
codec
->
compress_type
=
SND_SOC_FLAT_COMPRESSION
;
codec
->
write
=
codec_drv
->
write
;
codec
->
write
=
codec_drv
->
write
;
codec
->
read
=
codec_drv
->
read
;
codec
->
read
=
codec_drv
->
read
;
codec
->
volatile_register
=
codec_drv
->
volatile_register
;
codec
->
volatile_register
=
codec_drv
->
volatile_register
;
...
@@ -4325,35 +4329,6 @@ int snd_soc_register_codec(struct device *dev,
...
@@ -4325,35 +4329,6 @@ int snd_soc_register_codec(struct device *dev,
codec
->
num_dai
=
num_dai
;
codec
->
num_dai
=
num_dai
;
mutex_init
(
&
codec
->
mutex
);
mutex_init
(
&
codec
->
mutex
);
/* allocate CODEC register cache */
if
(
codec_drv
->
reg_cache_size
&&
codec_drv
->
reg_word_size
)
{
reg_size
=
codec_drv
->
reg_cache_size
*
codec_drv
->
reg_word_size
;
codec
->
reg_size
=
reg_size
;
/* it is necessary to make a copy of the default register cache
* because in the case of using a compression type that requires
* the default register cache to be marked as the
* kernel might have freed the array by the time we initialize
* the cache.
*/
if
(
codec_drv
->
reg_cache_default
)
{
codec
->
reg_def_copy
=
kmemdup
(
codec_drv
->
reg_cache_default
,
reg_size
,
GFP_KERNEL
);
if
(
!
codec
->
reg_def_copy
)
{
ret
=
-
ENOMEM
;
goto
fail_codec_name
;
}
}
}
if
(
codec_drv
->
reg_access_size
&&
codec_drv
->
reg_access_default
)
{
if
(
!
codec
->
volatile_register
)
codec
->
volatile_register
=
snd_soc_default_volatile_register
;
if
(
!
codec
->
readable_register
)
codec
->
readable_register
=
snd_soc_default_readable_register
;
if
(
!
codec
->
writable_register
)
codec
->
writable_register
=
snd_soc_default_writable_register
;
}
for
(
i
=
0
;
i
<
num_dai
;
i
++
)
{
for
(
i
=
0
;
i
<
num_dai
;
i
++
)
{
fixup_codec_formats
(
&
dai_drv
[
i
].
playback
);
fixup_codec_formats
(
&
dai_drv
[
i
].
playback
);
fixup_codec_formats
(
&
dai_drv
[
i
].
capture
);
fixup_codec_formats
(
&
dai_drv
[
i
].
capture
);
...
@@ -4412,7 +4387,6 @@ void snd_soc_unregister_codec(struct device *dev)
...
@@ -4412,7 +4387,6 @@ void snd_soc_unregister_codec(struct device *dev)
dev_dbg
(
codec
->
dev
,
"ASoC: Unregistered codec '%s'
\n
"
,
codec
->
name
);
dev_dbg
(
codec
->
dev
,
"ASoC: Unregistered codec '%s'
\n
"
,
codec
->
name
);
snd_soc_cache_exit
(
codec
);
snd_soc_cache_exit
(
codec
);
kfree
(
codec
->
reg_def_copy
);
kfree
(
codec
->
name
);
kfree
(
codec
->
name
);
kfree
(
codec
);
kfree
(
codec
);
}
}
...
@@ -4624,12 +4598,31 @@ int snd_soc_of_get_dai_name(struct device_node *of_node,
...
@@ -4624,12 +4598,31 @@ int snd_soc_of_get_dai_name(struct device_node *of_node,
if
(
pos
->
dev
->
of_node
!=
args
.
np
)
if
(
pos
->
dev
->
of_node
!=
args
.
np
)
continue
;
continue
;
if
(
!
pos
->
driver
->
of_xlate_dai_name
)
{
if
(
pos
->
driver
->
of_xlate_dai_name
)
{
ret
=
-
ENOSYS
;
ret
=
pos
->
driver
->
of_xlate_dai_name
(
pos
,
&
args
,
dai_name
);
}
else
{
int
id
=
-
1
;
switch
(
args
.
args_count
)
{
case
0
:
id
=
0
;
/* same as dai_drv[0] */
break
;
case
1
:
id
=
args
.
args
[
0
];
break
;
default:
/* not supported */
break
;
break
;
}
}
ret
=
pos
->
driver
->
of_xlate_dai_name
(
pos
,
&
args
,
dai_name
);
if
(
id
<
0
||
id
>=
pos
->
num_dai
)
{
ret
=
-
EINVAL
;
}
else
{
*
dai_name
=
pos
->
dai_drv
[
id
].
name
;
ret
=
0
;
}
}
break
;
break
;
}
}
mutex_unlock
(
&
client_mutex
);
mutex_unlock
(
&
client_mutex
);
...
...
sound/soc/soc-io.c
浏览文件 @
c6016bde
...
@@ -65,31 +65,6 @@ static unsigned int hw_read(struct snd_soc_codec *codec, unsigned int reg)
...
@@ -65,31 +65,6 @@ static unsigned int hw_read(struct snd_soc_codec *codec, unsigned int reg)
return
val
;
return
val
;
}
}
/* Primitive bulk write support for soc-cache. The data pointed to by
* `data' needs to already be in the form the hardware expects. Any
* data written through this function will not go through the cache as
* it only handles writing to volatile or out of bounds registers.
*
* This is currently only supported for devices using the regmap API
* wrappers.
*/
static
int
snd_soc_hw_bulk_write_raw
(
struct
snd_soc_codec
*
codec
,
unsigned
int
reg
,
const
void
*
data
,
size_t
len
)
{
/* To ensure that we don't get out of sync with the cache, check
* whether the base register is volatile or if we've directly asked
* to bypass the cache. Out of bounds registers are considered
* volatile.
*/
if
(
!
codec
->
cache_bypass
&&
!
snd_soc_codec_volatile_register
(
codec
,
reg
)
&&
reg
<
codec
->
driver
->
reg_cache_size
)
return
-
EINVAL
;
return
regmap_raw_write
(
codec
->
control_data
,
reg
,
data
,
len
);
}
/**
/**
* snd_soc_codec_set_cache_io: Set up standard I/O functions.
* snd_soc_codec_set_cache_io: Set up standard I/O functions.
*
*
...
@@ -119,7 +94,6 @@ int snd_soc_codec_set_cache_io(struct snd_soc_codec *codec,
...
@@ -119,7 +94,6 @@ int snd_soc_codec_set_cache_io(struct snd_soc_codec *codec,
memset
(
&
config
,
0
,
sizeof
(
config
));
memset
(
&
config
,
0
,
sizeof
(
config
));
codec
->
write
=
hw_write
;
codec
->
write
=
hw_write
;
codec
->
read
=
hw_read
;
codec
->
read
=
hw_read
;
codec
->
bulk_write_raw
=
snd_soc_hw_bulk_write_raw
;
config
.
reg_bits
=
addr_bits
;
config
.
reg_bits
=
addr_bits
;
config
.
val_bits
=
data_bits
;
config
.
val_bits
=
data_bits
;
...
...
sound/soc/soc-pcm.c
浏览文件 @
c6016bde
...
@@ -19,6 +19,7 @@
...
@@ -19,6 +19,7 @@
#include <linux/kernel.h>
#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/init.h>
#include <linux/delay.h>
#include <linux/delay.h>
#include <linux/pinctrl/consumer.h>
#include <linux/pm_runtime.h>
#include <linux/pm_runtime.h>
#include <linux/slab.h>
#include <linux/slab.h>
#include <linux/workqueue.h>
#include <linux/workqueue.h>
...
@@ -183,6 +184,8 @@ static int soc_pcm_open(struct snd_pcm_substream *substream)
...
@@ -183,6 +184,8 @@ static int soc_pcm_open(struct snd_pcm_substream *substream)
struct
snd_soc_dai_driver
*
codec_dai_drv
=
codec_dai
->
driver
;
struct
snd_soc_dai_driver
*
codec_dai_drv
=
codec_dai
->
driver
;
int
ret
=
0
;
int
ret
=
0
;
pinctrl_pm_select_default_state
(
cpu_dai
->
dev
);
pinctrl_pm_select_default_state
(
codec_dai
->
dev
);
pm_runtime_get_sync
(
cpu_dai
->
dev
);
pm_runtime_get_sync
(
cpu_dai
->
dev
);
pm_runtime_get_sync
(
codec_dai
->
dev
);
pm_runtime_get_sync
(
codec_dai
->
dev
);
pm_runtime_get_sync
(
platform
->
dev
);
pm_runtime_get_sync
(
platform
->
dev
);
...
@@ -190,7 +193,7 @@ static int soc_pcm_open(struct snd_pcm_substream *substream)
...
@@ -190,7 +193,7 @@ static int soc_pcm_open(struct snd_pcm_substream *substream)
mutex_lock_nested
(
&
rtd
->
pcm_mutex
,
rtd
->
pcm_subclass
);
mutex_lock_nested
(
&
rtd
->
pcm_mutex
,
rtd
->
pcm_subclass
);
/* startup the audio subsystem */
/* startup the audio subsystem */
if
(
cpu_dai
->
driver
->
ops
->
startup
)
{
if
(
cpu_dai
->
driver
->
ops
&&
cpu_dai
->
driver
->
ops
->
startup
)
{
ret
=
cpu_dai
->
driver
->
ops
->
startup
(
substream
,
cpu_dai
);
ret
=
cpu_dai
->
driver
->
ops
->
startup
(
substream
,
cpu_dai
);
if
(
ret
<
0
)
{
if
(
ret
<
0
)
{
dev_err
(
cpu_dai
->
dev
,
"ASoC: can't open interface"
dev_err
(
cpu_dai
->
dev
,
"ASoC: can't open interface"
...
@@ -208,7 +211,7 @@ static int soc_pcm_open(struct snd_pcm_substream *substream)
...
@@ -208,7 +211,7 @@ static int soc_pcm_open(struct snd_pcm_substream *substream)
}
}
}
}
if
(
codec_dai
->
driver
->
ops
->
startup
)
{
if
(
codec_dai
->
driver
->
ops
&&
codec_dai
->
driver
->
ops
->
startup
)
{
ret
=
codec_dai
->
driver
->
ops
->
startup
(
substream
,
codec_dai
);
ret
=
codec_dai
->
driver
->
ops
->
startup
(
substream
,
codec_dai
);
if
(
ret
<
0
)
{
if
(
ret
<
0
)
{
dev_err
(
codec_dai
->
dev
,
"ASoC: can't open codec"
dev_err
(
codec_dai
->
dev
,
"ASoC: can't open codec"
...
@@ -317,6 +320,10 @@ static int soc_pcm_open(struct snd_pcm_substream *substream)
...
@@ -317,6 +320,10 @@ static int soc_pcm_open(struct snd_pcm_substream *substream)
pm_runtime_put
(
platform
->
dev
);
pm_runtime_put
(
platform
->
dev
);
pm_runtime_put
(
codec_dai
->
dev
);
pm_runtime_put
(
codec_dai
->
dev
);
pm_runtime_put
(
cpu_dai
->
dev
);
pm_runtime_put
(
cpu_dai
->
dev
);
if
(
!
codec_dai
->
active
)
pinctrl_pm_select_sleep_state
(
codec_dai
->
dev
);
if
(
!
cpu_dai
->
active
)
pinctrl_pm_select_sleep_state
(
cpu_dai
->
dev
);
return
ret
;
return
ret
;
}
}
...
@@ -426,6 +433,10 @@ static int soc_pcm_close(struct snd_pcm_substream *substream)
...
@@ -426,6 +433,10 @@ static int soc_pcm_close(struct snd_pcm_substream *substream)
pm_runtime_put
(
platform
->
dev
);
pm_runtime_put
(
platform
->
dev
);
pm_runtime_put
(
codec_dai
->
dev
);
pm_runtime_put
(
codec_dai
->
dev
);
pm_runtime_put
(
cpu_dai
->
dev
);
pm_runtime_put
(
cpu_dai
->
dev
);
if
(
!
codec_dai
->
active
)
pinctrl_pm_select_sleep_state
(
codec_dai
->
dev
);
if
(
!
cpu_dai
->
active
)
pinctrl_pm_select_sleep_state
(
cpu_dai
->
dev
);
return
0
;
return
0
;
}
}
...
@@ -463,7 +474,7 @@ static int soc_pcm_prepare(struct snd_pcm_substream *substream)
...
@@ -463,7 +474,7 @@ static int soc_pcm_prepare(struct snd_pcm_substream *substream)
}
}
}
}
if
(
codec_dai
->
driver
->
ops
->
prepare
)
{
if
(
codec_dai
->
driver
->
ops
&&
codec_dai
->
driver
->
ops
->
prepare
)
{
ret
=
codec_dai
->
driver
->
ops
->
prepare
(
substream
,
codec_dai
);
ret
=
codec_dai
->
driver
->
ops
->
prepare
(
substream
,
codec_dai
);
if
(
ret
<
0
)
{
if
(
ret
<
0
)
{
dev_err
(
codec_dai
->
dev
,
"ASoC: DAI prepare error: %d
\n
"
,
dev_err
(
codec_dai
->
dev
,
"ASoC: DAI prepare error: %d
\n
"
,
...
@@ -472,7 +483,7 @@ static int soc_pcm_prepare(struct snd_pcm_substream *substream)
...
@@ -472,7 +483,7 @@ static int soc_pcm_prepare(struct snd_pcm_substream *substream)
}
}
}
}
if
(
cpu_dai
->
driver
->
ops
->
prepare
)
{
if
(
cpu_dai
->
driver
->
ops
&&
cpu_dai
->
driver
->
ops
->
prepare
)
{
ret
=
cpu_dai
->
driver
->
ops
->
prepare
(
substream
,
cpu_dai
);
ret
=
cpu_dai
->
driver
->
ops
->
prepare
(
substream
,
cpu_dai
);
if
(
ret
<
0
)
{
if
(
ret
<
0
)
{
dev_err
(
cpu_dai
->
dev
,
"ASoC: DAI prepare error: %d
\n
"
,
dev_err
(
cpu_dai
->
dev
,
"ASoC: DAI prepare error: %d
\n
"
,
...
@@ -523,7 +534,7 @@ static int soc_pcm_hw_params(struct snd_pcm_substream *substream,
...
@@ -523,7 +534,7 @@ static int soc_pcm_hw_params(struct snd_pcm_substream *substream,
}
}
}
}
if
(
codec_dai
->
driver
->
ops
->
hw_params
)
{
if
(
codec_dai
->
driver
->
ops
&&
codec_dai
->
driver
->
ops
->
hw_params
)
{
ret
=
codec_dai
->
driver
->
ops
->
hw_params
(
substream
,
params
,
codec_dai
);
ret
=
codec_dai
->
driver
->
ops
->
hw_params
(
substream
,
params
,
codec_dai
);
if
(
ret
<
0
)
{
if
(
ret
<
0
)
{
dev_err
(
codec_dai
->
dev
,
"ASoC: can't set %s hw params:"
dev_err
(
codec_dai
->
dev
,
"ASoC: can't set %s hw params:"
...
@@ -532,7 +543,7 @@ static int soc_pcm_hw_params(struct snd_pcm_substream *substream,
...
@@ -532,7 +543,7 @@ static int soc_pcm_hw_params(struct snd_pcm_substream *substream,
}
}
}
}
if
(
cpu_dai
->
driver
->
ops
->
hw_params
)
{
if
(
cpu_dai
->
driver
->
ops
&&
cpu_dai
->
driver
->
ops
->
hw_params
)
{
ret
=
cpu_dai
->
driver
->
ops
->
hw_params
(
substream
,
params
,
cpu_dai
);
ret
=
cpu_dai
->
driver
->
ops
->
hw_params
(
substream
,
params
,
cpu_dai
);
if
(
ret
<
0
)
{
if
(
ret
<
0
)
{
dev_err
(
cpu_dai
->
dev
,
"ASoC: %s hw params failed: %d
\n
"
,
dev_err
(
cpu_dai
->
dev
,
"ASoC: %s hw params failed: %d
\n
"
,
...
@@ -559,11 +570,11 @@ static int soc_pcm_hw_params(struct snd_pcm_substream *substream,
...
@@ -559,11 +570,11 @@ static int soc_pcm_hw_params(struct snd_pcm_substream *substream,
return
ret
;
return
ret
;
platform_err:
platform_err:
if
(
cpu_dai
->
driver
->
ops
->
hw_free
)
if
(
cpu_dai
->
driver
->
ops
&&
cpu_dai
->
driver
->
ops
->
hw_free
)
cpu_dai
->
driver
->
ops
->
hw_free
(
substream
,
cpu_dai
);
cpu_dai
->
driver
->
ops
->
hw_free
(
substream
,
cpu_dai
);
interface_err:
interface_err:
if
(
codec_dai
->
driver
->
ops
->
hw_free
)
if
(
codec_dai
->
driver
->
ops
&&
codec_dai
->
driver
->
ops
->
hw_free
)
codec_dai
->
driver
->
ops
->
hw_free
(
substream
,
codec_dai
);
codec_dai
->
driver
->
ops
->
hw_free
(
substream
,
codec_dai
);
codec_err:
codec_err:
...
@@ -600,10 +611,10 @@ static int soc_pcm_hw_free(struct snd_pcm_substream *substream)
...
@@ -600,10 +611,10 @@ static int soc_pcm_hw_free(struct snd_pcm_substream *substream)
platform
->
driver
->
ops
->
hw_free
(
substream
);
platform
->
driver
->
ops
->
hw_free
(
substream
);
/* now free hw params for the DAIs */
/* now free hw params for the DAIs */
if
(
codec_dai
->
driver
->
ops
->
hw_free
)
if
(
codec_dai
->
driver
->
ops
&&
codec_dai
->
driver
->
ops
->
hw_free
)
codec_dai
->
driver
->
ops
->
hw_free
(
substream
,
codec_dai
);
codec_dai
->
driver
->
ops
->
hw_free
(
substream
,
codec_dai
);
if
(
cpu_dai
->
driver
->
ops
->
hw_free
)
if
(
cpu_dai
->
driver
->
ops
&&
cpu_dai
->
driver
->
ops
->
hw_free
)
cpu_dai
->
driver
->
ops
->
hw_free
(
substream
,
cpu_dai
);
cpu_dai
->
driver
->
ops
->
hw_free
(
substream
,
cpu_dai
);
mutex_unlock
(
&
rtd
->
pcm_mutex
);
mutex_unlock
(
&
rtd
->
pcm_mutex
);
...
@@ -618,7 +629,7 @@ static int soc_pcm_trigger(struct snd_pcm_substream *substream, int cmd)
...
@@ -618,7 +629,7 @@ static int soc_pcm_trigger(struct snd_pcm_substream *substream, int cmd)
struct
snd_soc_dai
*
codec_dai
=
rtd
->
codec_dai
;
struct
snd_soc_dai
*
codec_dai
=
rtd
->
codec_dai
;
int
ret
;
int
ret
;
if
(
codec_dai
->
driver
->
ops
->
trigger
)
{
if
(
codec_dai
->
driver
->
ops
&&
codec_dai
->
driver
->
ops
->
trigger
)
{
ret
=
codec_dai
->
driver
->
ops
->
trigger
(
substream
,
cmd
,
codec_dai
);
ret
=
codec_dai
->
driver
->
ops
->
trigger
(
substream
,
cmd
,
codec_dai
);
if
(
ret
<
0
)
if
(
ret
<
0
)
return
ret
;
return
ret
;
...
@@ -630,7 +641,7 @@ static int soc_pcm_trigger(struct snd_pcm_substream *substream, int cmd)
...
@@ -630,7 +641,7 @@ static int soc_pcm_trigger(struct snd_pcm_substream *substream, int cmd)
return
ret
;
return
ret
;
}
}
if
(
cpu_dai
->
driver
->
ops
->
trigger
)
{
if
(
cpu_dai
->
driver
->
ops
&&
cpu_dai
->
driver
->
ops
->
trigger
)
{
ret
=
cpu_dai
->
driver
->
ops
->
trigger
(
substream
,
cmd
,
cpu_dai
);
ret
=
cpu_dai
->
driver
->
ops
->
trigger
(
substream
,
cmd
,
cpu_dai
);
if
(
ret
<
0
)
if
(
ret
<
0
)
return
ret
;
return
ret
;
...
@@ -647,19 +658,20 @@ static int soc_pcm_bespoke_trigger(struct snd_pcm_substream *substream,
...
@@ -647,19 +658,20 @@ static int soc_pcm_bespoke_trigger(struct snd_pcm_substream *substream,
struct
snd_soc_dai
*
codec_dai
=
rtd
->
codec_dai
;
struct
snd_soc_dai
*
codec_dai
=
rtd
->
codec_dai
;
int
ret
;
int
ret
;
if
(
codec_dai
->
driver
->
ops
->
bespoke_trigger
)
{
if
(
codec_dai
->
driver
->
ops
&&
codec_dai
->
driver
->
ops
->
bespoke_trigger
)
{
ret
=
codec_dai
->
driver
->
ops
->
bespoke_trigger
(
substream
,
cmd
,
codec_dai
);
ret
=
codec_dai
->
driver
->
ops
->
bespoke_trigger
(
substream
,
cmd
,
codec_dai
);
if
(
ret
<
0
)
if
(
ret
<
0
)
return
ret
;
return
ret
;
}
}
if
(
platform
->
driver
->
bespoke_trigger
)
{
if
(
platform
->
driver
->
ops
&&
platform
->
driver
->
bespoke_trigger
)
{
ret
=
platform
->
driver
->
bespoke_trigger
(
substream
,
cmd
);
ret
=
platform
->
driver
->
bespoke_trigger
(
substream
,
cmd
);
if
(
ret
<
0
)
if
(
ret
<
0
)
return
ret
;
return
ret
;
}
}
if
(
cpu_dai
->
driver
->
ops
->
bespoke_trigger
)
{
if
(
cpu_dai
->
driver
->
ops
&&
cpu_dai
->
driver
->
ops
->
bespoke_trigger
)
{
ret
=
cpu_dai
->
driver
->
ops
->
bespoke_trigger
(
substream
,
cmd
,
cpu_dai
);
ret
=
cpu_dai
->
driver
->
ops
->
bespoke_trigger
(
substream
,
cmd
,
cpu_dai
);
if
(
ret
<
0
)
if
(
ret
<
0
)
return
ret
;
return
ret
;
...
@@ -684,10 +696,10 @@ static snd_pcm_uframes_t soc_pcm_pointer(struct snd_pcm_substream *substream)
...
@@ -684,10 +696,10 @@ static snd_pcm_uframes_t soc_pcm_pointer(struct snd_pcm_substream *substream)
if
(
platform
->
driver
->
ops
&&
platform
->
driver
->
ops
->
pointer
)
if
(
platform
->
driver
->
ops
&&
platform
->
driver
->
ops
->
pointer
)
offset
=
platform
->
driver
->
ops
->
pointer
(
substream
);
offset
=
platform
->
driver
->
ops
->
pointer
(
substream
);
if
(
cpu_dai
->
driver
->
ops
->
delay
)
if
(
cpu_dai
->
driver
->
ops
&&
cpu_dai
->
driver
->
ops
->
delay
)
delay
+=
cpu_dai
->
driver
->
ops
->
delay
(
substream
,
cpu_dai
);
delay
+=
cpu_dai
->
driver
->
ops
->
delay
(
substream
,
cpu_dai
);
if
(
codec_dai
->
driver
->
ops
->
delay
)
if
(
codec_dai
->
driver
->
ops
&&
codec_dai
->
driver
->
ops
->
delay
)
delay
+=
codec_dai
->
driver
->
ops
->
delay
(
substream
,
codec_dai
);
delay
+=
codec_dai
->
driver
->
ops
->
delay
(
substream
,
codec_dai
);
if
(
platform
->
driver
->
delay
)
if
(
platform
->
driver
->
delay
)
...
@@ -721,7 +733,7 @@ static int dpcm_be_connect(struct snd_soc_pcm_runtime *fe,
...
@@ -721,7 +733,7 @@ static int dpcm_be_connect(struct snd_soc_pcm_runtime *fe,
list_add
(
&
dpcm
->
list_be
,
&
fe
->
dpcm
[
stream
].
be_clients
);
list_add
(
&
dpcm
->
list_be
,
&
fe
->
dpcm
[
stream
].
be_clients
);
list_add
(
&
dpcm
->
list_fe
,
&
be
->
dpcm
[
stream
].
fe_clients
);
list_add
(
&
dpcm
->
list_fe
,
&
be
->
dpcm
[
stream
].
fe_clients
);
dev_dbg
(
fe
->
dev
,
"
connected new DPCM %s path %s %s %s
\n
"
,
dev_dbg
(
fe
->
dev
,
"connected new DPCM %s path %s %s %s
\n
"
,
stream
?
"capture"
:
"playback"
,
fe
->
dai_link
->
name
,
stream
?
"capture"
:
"playback"
,
fe
->
dai_link
->
name
,
stream
?
"<-"
:
"->"
,
be
->
dai_link
->
name
);
stream
?
"<-"
:
"->"
,
be
->
dai_link
->
name
);
...
@@ -749,7 +761,7 @@ static void dpcm_be_reparent(struct snd_soc_pcm_runtime *fe,
...
@@ -749,7 +761,7 @@ static void dpcm_be_reparent(struct snd_soc_pcm_runtime *fe,
if
(
dpcm
->
fe
==
fe
)
if
(
dpcm
->
fe
==
fe
)
continue
;
continue
;
dev_dbg
(
fe
->
dev
,
"
reparent %s path %s %s %s
\n
"
,
dev_dbg
(
fe
->
dev
,
"reparent %s path %s %s %s
\n
"
,
stream
?
"capture"
:
"playback"
,
stream
?
"capture"
:
"playback"
,
dpcm
->
fe
->
dai_link
->
name
,
dpcm
->
fe
->
dai_link
->
name
,
stream
?
"<-"
:
"->"
,
dpcm
->
be
->
dai_link
->
name
);
stream
?
"<-"
:
"->"
,
dpcm
->
be
->
dai_link
->
name
);
...
@@ -773,7 +785,7 @@ static void dpcm_be_disconnect(struct snd_soc_pcm_runtime *fe, int stream)
...
@@ -773,7 +785,7 @@ static void dpcm_be_disconnect(struct snd_soc_pcm_runtime *fe, int stream)
if
(
dpcm
->
state
!=
SND_SOC_DPCM_LINK_STATE_FREE
)
if
(
dpcm
->
state
!=
SND_SOC_DPCM_LINK_STATE_FREE
)
continue
;
continue
;
dev_dbg
(
fe
->
dev
,
"
freed DSP %s path %s %s %s
\n
"
,
dev_dbg
(
fe
->
dev
,
"freed DSP %s path %s %s %s
\n
"
,
stream
?
"capture"
:
"playback"
,
fe
->
dai_link
->
name
,
stream
?
"capture"
:
"playback"
,
fe
->
dai_link
->
name
,
stream
?
"<-"
:
"->"
,
dpcm
->
be
->
dai_link
->
name
);
stream
?
"<-"
:
"->"
,
dpcm
->
be
->
dai_link
->
name
);
...
@@ -1037,6 +1049,12 @@ static int dpcm_be_dai_startup(struct snd_soc_pcm_runtime *fe, int stream)
...
@@ -1037,6 +1049,12 @@ static int dpcm_be_dai_startup(struct snd_soc_pcm_runtime *fe, int stream)
struct
snd_pcm_substream
*
be_substream
=
struct
snd_pcm_substream
*
be_substream
=
snd_soc_dpcm_get_substream
(
be
,
stream
);
snd_soc_dpcm_get_substream
(
be
,
stream
);
if
(
!
be_substream
)
{
dev_err
(
be
->
dev
,
"ASoC: no backend %s stream
\n
"
,
stream
?
"capture"
:
"playback"
);
continue
;
}
/* is this op for this BE ? */
/* is this op for this BE ? */
if
(
!
snd_soc_dpcm_be_can_update
(
fe
,
be
,
stream
))
if
(
!
snd_soc_dpcm_be_can_update
(
fe
,
be
,
stream
))
continue
;
continue
;
...
@@ -1054,7 +1072,8 @@ static int dpcm_be_dai_startup(struct snd_soc_pcm_runtime *fe, int stream)
...
@@ -1054,7 +1072,8 @@ static int dpcm_be_dai_startup(struct snd_soc_pcm_runtime *fe, int stream)
(
be
->
dpcm
[
stream
].
state
!=
SND_SOC_DPCM_STATE_CLOSE
))
(
be
->
dpcm
[
stream
].
state
!=
SND_SOC_DPCM_STATE_CLOSE
))
continue
;
continue
;
dev_dbg
(
be
->
dev
,
"ASoC: open BE %s
\n
"
,
be
->
dai_link
->
name
);
dev_dbg
(
be
->
dev
,
"ASoC: open %s BE %s
\n
"
,
stream
?
"capture"
:
"playback"
,
be
->
dai_link
->
name
);
be_substream
->
runtime
=
be
->
dpcm
[
stream
].
runtime
;
be_substream
->
runtime
=
be
->
dpcm
[
stream
].
runtime
;
err
=
soc_pcm_open
(
be_substream
);
err
=
soc_pcm_open
(
be_substream
);
...
@@ -1673,7 +1692,7 @@ static int soc_pcm_ioctl(struct snd_pcm_substream *substream,
...
@@ -1673,7 +1692,7 @@ static int soc_pcm_ioctl(struct snd_pcm_substream *substream,
struct
snd_soc_pcm_runtime
*
rtd
=
substream
->
private_data
;
struct
snd_soc_pcm_runtime
*
rtd
=
substream
->
private_data
;
struct
snd_soc_platform
*
platform
=
rtd
->
platform
;
struct
snd_soc_platform
*
platform
=
rtd
->
platform
;
if
(
platform
->
driver
->
ops
->
ioctl
)
if
(
platform
->
driver
->
ops
&&
platform
->
driver
->
ops
->
ioctl
)
return
platform
->
driver
->
ops
->
ioctl
(
substream
,
cmd
,
arg
);
return
platform
->
driver
->
ops
->
ioctl
(
substream
,
cmd
,
arg
);
return
snd_pcm_lib_ioctl
(
substream
,
cmd
,
arg
);
return
snd_pcm_lib_ioctl
(
substream
,
cmd
,
arg
);
}
}
...
@@ -1934,7 +1953,7 @@ int soc_dpcm_be_digital_mute(struct snd_soc_pcm_runtime *fe, int mute)
...
@@ -1934,7 +1953,7 @@ int soc_dpcm_be_digital_mute(struct snd_soc_pcm_runtime *fe, int mute)
dev_dbg
(
be
->
dev
,
"ASoC: BE digital mute %s
\n
"
,
be
->
dai_link
->
name
);
dev_dbg
(
be
->
dev
,
"ASoC: BE digital mute %s
\n
"
,
be
->
dai_link
->
name
);
if
(
drv
->
ops
->
digital_mute
&&
dai
->
playback_active
)
if
(
drv
->
ops
&&
drv
->
ops
->
digital_mute
&&
dai
->
playback_active
)
drv
->
ops
->
digital_mute
(
dai
,
mute
);
drv
->
ops
->
digital_mute
(
dai
,
mute
);
}
}
...
@@ -2116,7 +2135,7 @@ int soc_new_pcm(struct snd_soc_pcm_runtime *rtd, int num)
...
@@ -2116,7 +2135,7 @@ int soc_new_pcm(struct snd_soc_pcm_runtime *rtd, int num)
pcm
->
private_free
=
platform
->
driver
->
pcm_free
;
pcm
->
private_free
=
platform
->
driver
->
pcm_free
;
out:
out:
dev_info
(
rtd
->
card
->
dev
,
"
%s <-> %s mapping ok
\n
"
,
codec_dai
->
name
,
dev_info
(
rtd
->
card
->
dev
,
"%s <-> %s mapping ok
\n
"
,
codec_dai
->
name
,
cpu_dai
->
name
);
cpu_dai
->
name
);
return
ret
;
return
ret
;
}
}
...
@@ -2224,7 +2243,7 @@ EXPORT_SYMBOL_GPL(snd_soc_dpcm_can_be_params);
...
@@ -2224,7 +2243,7 @@ EXPORT_SYMBOL_GPL(snd_soc_dpcm_can_be_params);
int
snd_soc_platform_trigger
(
struct
snd_pcm_substream
*
substream
,
int
snd_soc_platform_trigger
(
struct
snd_pcm_substream
*
substream
,
int
cmd
,
struct
snd_soc_platform
*
platform
)
int
cmd
,
struct
snd_soc_platform
*
platform
)
{
{
if
(
platform
->
driver
->
ops
->
trigger
)
if
(
platform
->
driver
->
ops
&&
platform
->
driver
->
ops
->
trigger
)
return
platform
->
driver
->
ops
->
trigger
(
substream
,
cmd
);
return
platform
->
driver
->
ops
->
trigger
(
substream
,
cmd
);
return
0
;
return
0
;
}
}
...
...
sound/soc/soc-utils.c
浏览文件 @
c6016bde
...
@@ -75,6 +75,10 @@ static const struct snd_pcm_hardware dummy_dma_hardware = {
...
@@ -75,6 +75,10 @@ static const struct snd_pcm_hardware dummy_dma_hardware = {
static
int
dummy_dma_open
(
struct
snd_pcm_substream
*
substream
)
static
int
dummy_dma_open
(
struct
snd_pcm_substream
*
substream
)
{
{
struct
snd_soc_pcm_runtime
*
rtd
=
substream
->
private_data
;
/* BE's dont need dummy params */
if
(
!
rtd
->
dai_link
->
no_pcm
)
snd_soc_set_runtime_hwparams
(
substream
,
&
dummy_dma_hardware
);
snd_soc_set_runtime_hwparams
(
substream
,
&
dummy_dma_hardware
);
return
0
;
return
0
;
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录