Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
openeuler
raspberrypi-kernel
提交
bad318cd
R
raspberrypi-kernel
项目概览
openeuler
/
raspberrypi-kernel
通知
13
Star
1
Fork
0
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
0
列表
看板
标记
里程碑
合并请求
0
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
R
raspberrypi-kernel
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
0
Issue
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
提交
Issue看板
提交
bad318cd
编写于
3月 25, 2014
作者:
M
Mark Brown
浏览文件
操作
浏览文件
下载
差异文件
Merge remote-tracking branch 'asoc/topic/simple' into asoc-next
上级
c61e9e8c
6a91a17b
变更
2
显示空白变更内容
内联
并排
Showing
2 changed file
with
183 addition
and
79 deletion
+183
-79
Documentation/devicetree/bindings/sound/simple-card.txt
Documentation/devicetree/bindings/sound/simple-card.txt
+41
-1
sound/soc/generic/simple-card.c
sound/soc/generic/simple-card.c
+142
-78
未找到文件。
Documentation/devicetree/bindings/sound/simple-card.txt
浏览文件 @
bad318cd
...
@@ -23,6 +23,11 @@ Optional properties:
...
@@ -23,6 +23,11 @@ Optional properties:
Required subnodes:
Required subnodes:
- simple-audio-card,dai-link : container for the CPU and CODEC sub-nodes
This container may be omitted when the
card has only one DAI link.
See the examples.
- simple-audio-card,cpu : CPU sub-node
- simple-audio-card,cpu : CPU sub-node
- simple-audio-card,codec : CODEC sub-node
- simple-audio-card,codec : CODEC sub-node
...
@@ -49,7 +54,7 @@ Note:
...
@@ -49,7 +54,7 @@ Note:
CPU and CODEC sides as we need to keep the settings identical for both ends
CPU and CODEC sides as we need to keep the settings identical for both ends
of the link.
of the link.
Example:
Example
1 - single DAI link
:
sound {
sound {
compatible = "simple-audio-card";
compatible = "simple-audio-card";
...
@@ -94,3 +99,38 @@ sh_fsi2: sh_fsi2@ec230000 {
...
@@ -94,3 +99,38 @@ sh_fsi2: sh_fsi2@ec230000 {
interrupt-parent = <&gic>;
interrupt-parent = <&gic>;
interrupts = <0 146 0x4>;
interrupts = <0 146 0x4>;
};
};
Example 2 - many DAI links:
sound {
compatible = "simple-audio-card";
simple-audio-card,name = "Cubox Audio";
simple-audio-card,format = "i2s";
simple-audio-card,dai-link@0 { /* I2S - HDMI */
simple-audio-card,cpu {
sound-dai = <&audio1 0>;
};
simple-audio-card,codec {
sound-dai = <&tda998x 0>;
};
};
simple-audio-card,dai-link@1 { /* S/PDIF - HDMI */
simple-audio-card,cpu {
sound-dai = <&audio1 1>;
};
simple-audio-card,codec {
sound-dai = <&tda998x 1>;
};
};
simple-audio-card,dai-link@2 { /* S/PDIF - S/PDIF */
simple-audio-card,cpu {
sound-dai = <&audio1 1>;
};
simple-audio-card,codec {
sound-dai = <&spdif_codec>;
};
};
};
sound/soc/generic/simple-card.c
浏览文件 @
bad318cd
...
@@ -20,9 +20,11 @@
...
@@ -20,9 +20,11 @@
struct
simple_card_data
{
struct
simple_card_data
{
struct
snd_soc_card
snd_card
;
struct
snd_soc_card
snd_card
;
struct
simple_dai_props
{
struct
asoc_simple_dai
cpu_dai
;
struct
asoc_simple_dai
cpu_dai
;
struct
asoc_simple_dai
codec_dai
;
struct
asoc_simple_dai
codec_dai
;
struct
snd_soc_dai_link
snd_link
;
}
*
dai_props
;
struct
snd_soc_dai_link
dai_link
[];
/* dynamically allocated */
};
};
static
int
__asoc_simple_card_dai_init
(
struct
snd_soc_dai
*
dai
,
static
int
__asoc_simple_card_dai_init
(
struct
snd_soc_dai
*
dai
,
...
@@ -68,13 +70,16 @@ static int asoc_simple_card_dai_init(struct snd_soc_pcm_runtime *rtd)
...
@@ -68,13 +70,16 @@ static int asoc_simple_card_dai_init(struct snd_soc_pcm_runtime *rtd)
snd_soc_card_get_drvdata
(
rtd
->
card
);
snd_soc_card_get_drvdata
(
rtd
->
card
);
struct
snd_soc_dai
*
codec
=
rtd
->
codec_dai
;
struct
snd_soc_dai
*
codec
=
rtd
->
codec_dai
;
struct
snd_soc_dai
*
cpu
=
rtd
->
cpu_dai
;
struct
snd_soc_dai
*
cpu
=
rtd
->
cpu_dai
;
int
ret
;
struct
simple_dai_props
*
dai_props
;
int
num
,
ret
;
ret
=
__asoc_simple_card_dai_init
(
codec
,
&
priv
->
codec_dai
);
num
=
rtd
-
rtd
->
card
->
rtd
;
dai_props
=
&
priv
->
dai_props
[
num
];
ret
=
__asoc_simple_card_dai_init
(
codec
,
&
dai_props
->
codec_dai
);
if
(
ret
<
0
)
if
(
ret
<
0
)
return
ret
;
return
ret
;
ret
=
__asoc_simple_card_dai_init
(
cpu
,
&
priv
->
cpu_dai
);
ret
=
__asoc_simple_card_dai_init
(
cpu
,
&
dai_props
->
cpu_dai
);
if
(
ret
<
0
)
if
(
ret
<
0
)
return
ret
;
return
ret
;
...
@@ -146,13 +151,47 @@ asoc_simple_card_sub_parse_of(struct device_node *np,
...
@@ -146,13 +151,47 @@ asoc_simple_card_sub_parse_of(struct device_node *np,
return
0
;
return
0
;
}
}
static
int
simple_card_cpu_codec_of
(
struct
device_node
*
node
,
int
daifmt
,
struct
snd_soc_dai_link
*
dai_link
,
struct
simple_dai_props
*
dai_props
)
{
struct
device_node
*
np
;
int
ret
;
/* CPU sub-node */
ret
=
-
EINVAL
;
np
=
of_get_child_by_name
(
node
,
"simple-audio-card,cpu"
);
if
(
np
)
{
ret
=
asoc_simple_card_sub_parse_of
(
np
,
daifmt
,
&
dai_props
->
cpu_dai
,
&
dai_link
->
cpu_of_node
,
&
dai_link
->
cpu_dai_name
);
of_node_put
(
np
);
}
if
(
ret
<
0
)
return
ret
;
/* CODEC sub-node */
ret
=
-
EINVAL
;
np
=
of_get_child_by_name
(
node
,
"simple-audio-card,codec"
);
if
(
np
)
{
ret
=
asoc_simple_card_sub_parse_of
(
np
,
daifmt
,
&
dai_props
->
codec_dai
,
&
dai_link
->
codec_of_node
,
&
dai_link
->
codec_dai_name
);
of_node_put
(
np
);
}
return
ret
;
}
static
int
asoc_simple_card_parse_of
(
struct
device_node
*
node
,
static
int
asoc_simple_card_parse_of
(
struct
device_node
*
node
,
struct
simple_card_data
*
priv
,
struct
simple_card_data
*
priv
,
struct
device
*
dev
)
struct
device
*
dev
,
int
multi
)
{
{
struct
snd_soc_dai_link
*
dai_link
=
priv
->
snd_card
.
dai_link
;
struct
snd_soc_dai_link
*
dai_link
=
priv
->
snd_card
.
dai_link
;
struct
asoc_simple_dai
*
codec_dai
=
&
priv
->
codec_dai
;
struct
simple_dai_props
*
dai_props
=
priv
->
dai_props
;
struct
asoc_simple_dai
*
cpu_dai
=
&
priv
->
cpu_dai
;
struct
device_node
*
np
;
struct
device_node
*
np
;
char
*
name
;
char
*
name
;
unsigned
int
daifmt
;
unsigned
int
daifmt
;
...
@@ -181,78 +220,71 @@ static int asoc_simple_card_parse_of(struct device_node *node,
...
@@ -181,78 +220,71 @@ static int asoc_simple_card_parse_of(struct device_node *node,
return
ret
;
return
ret
;
}
}
/* CPU sub-node */
/* loop on the DAI links */
ret
=
-
EINVAL
;
np
=
NULL
;
np
=
of_get_child_by_name
(
node
,
"simple-audio-card,cpu"
);
for
(;;)
{
if
(
np
)
{
if
(
multi
)
{
ret
=
asoc_simple_card_sub_parse_of
(
np
,
daifmt
,
np
=
of_get_next_child
(
node
,
np
);
cpu_dai
,
if
(
!
np
)
&
dai_link
->
cpu_of_node
,
break
;
&
dai_link
->
cpu_dai_name
);
of_node_put
(
np
);
}
}
if
(
ret
<
0
)
return
ret
;
/* CODEC sub-node */
ret
=
simple_card_cpu_codec_of
(
multi
?
np
:
node
,
ret
=
-
EINVAL
;
daifmt
,
dai_link
,
dai_props
);
np
=
of_get_child_by_name
(
node
,
"simple-audio-card,codec"
);
if
(
np
)
{
ret
=
asoc_simple_card_sub_parse_of
(
np
,
daifmt
,
codec_dai
,
&
dai_link
->
codec_of_node
,
&
dai_link
->
codec_dai_name
);
of_node_put
(
np
);
}
if
(
ret
<
0
)
if
(
ret
<
0
)
return
ret
;
goto
err
;
/*
/*
* overwrite cpu_dai->fmt as its DAIFMT_MASTER bit is based on CODEC
* overwrite cpu_dai->fmt as its DAIFMT_MASTER bit is based on CODEC
* while the other bits should be identical unless buggy SW/HW design.
* while the other bits should be identical unless buggy SW/HW design.
*/
*/
cpu_dai
->
fmt
=
codec_dai
->
fmt
;
dai_props
->
cpu_dai
.
fmt
=
dai_props
->
codec_dai
.
fmt
;
if
(
!
dai_link
->
cpu_dai_name
||
!
dai_link
->
codec_dai_name
)
if
(
!
dai_link
->
cpu_dai_name
||
!
dai_link
->
codec_dai_name
)
{
return
-
EINVAL
;
ret
=
-
EINVAL
;
goto
err
;
}
/* simple-card assumes platform == cpu */
dai_link
->
platform_of_node
=
dai_link
->
cpu_of_node
;
/* card name is created from CPU/CODEC dai name */
name
=
devm_kzalloc
(
dev
,
name
=
devm_kzalloc
(
dev
,
strlen
(
dai_link
->
cpu_dai_name
)
+
strlen
(
dai_link
->
cpu_dai_name
)
+
strlen
(
dai_link
->
codec_dai_name
)
+
2
,
strlen
(
dai_link
->
codec_dai_name
)
+
2
,
GFP_KERNEL
);
GFP_KERNEL
);
sprintf
(
name
,
"%s-%s"
,
dai_link
->
cpu_dai_name
,
sprintf
(
name
,
"%s-%s"
,
dai_link
->
cpu_dai_name
,
dai_link
->
codec_dai_name
);
dai_link
->
codec_dai_name
);
if
(
!
priv
->
snd_card
.
name
)
priv
->
snd_card
.
name
=
name
;
dai_link
->
name
=
dai_link
->
stream_name
=
name
;
dai_link
->
name
=
dai_link
->
stream_name
=
name
;
/* simple-card assumes platform == cpu */
if
(
!
multi
)
dai_link
->
platform_of_node
=
dai_link
->
cpu_of_node
;
break
;
dev_dbg
(
dev
,
"card-name : %s
\n
"
,
name
);
dai_link
++
;
dai_props
++
;
}
/* card name is created from CPU/CODEC dai name */
dai_link
=
priv
->
snd_card
.
dai_link
;
if
(
!
priv
->
snd_card
.
name
)
priv
->
snd_card
.
name
=
dai_link
->
name
;
dev_dbg
(
dev
,
"card-name : %s
\n
"
,
priv
->
snd_card
.
name
);
dev_dbg
(
dev
,
"platform : %04x
\n
"
,
daifmt
);
dev_dbg
(
dev
,
"platform : %04x
\n
"
,
daifmt
);
dai_props
=
priv
->
dai_props
;
dev_dbg
(
dev
,
"cpu : %s / %04x / %d
\n
"
,
dev_dbg
(
dev
,
"cpu : %s / %04x / %d
\n
"
,
dai_link
->
cpu_dai_name
,
dai_link
->
cpu_dai_name
,
cpu_dai
->
fmt
,
dai_props
->
cpu_dai
.
fmt
,
cpu_dai
->
sysclk
);
dai_props
->
cpu_dai
.
sysclk
);
dev_dbg
(
dev
,
"codec : %s / %04x / %d
\n
"
,
dev_dbg
(
dev
,
"codec : %s / %04x / %d
\n
"
,
dai_link
->
codec_dai_name
,
dai_link
->
codec_dai_name
,
codec_dai
->
fmt
,
dai_props
->
codec_dai
.
fmt
,
codec_dai
->
sysclk
);
dai_props
->
codec_dai
.
sysclk
);
/*
* soc_bind_dai_link() will check cpu name
* after of_node matching if dai_link has cpu_dai_name.
* but, it will never match if name was created by fmt_single_name()
* remove cpu_dai_name to escape name matching.
* see
* fmt_single_name()
* fmt_multiple_name()
*/
dai_link
->
cpu_dai_name
=
NULL
;
return
0
;
return
0
;
err:
of_node_put
(
np
);
return
ret
;
}
}
/* update the reference count of the devices nodes at end of probe */
/* update the reference count of the devices nodes at end of probe */
...
@@ -282,9 +314,21 @@ static int asoc_simple_card_probe(struct platform_device *pdev)
...
@@ -282,9 +314,21 @@ static int asoc_simple_card_probe(struct platform_device *pdev)
struct
snd_soc_dai_link
*
dai_link
;
struct
snd_soc_dai_link
*
dai_link
;
struct
device_node
*
np
=
pdev
->
dev
.
of_node
;
struct
device_node
*
np
=
pdev
->
dev
.
of_node
;
struct
device
*
dev
=
&
pdev
->
dev
;
struct
device
*
dev
=
&
pdev
->
dev
;
int
ret
;
int
num_links
,
multi
,
ret
;
priv
=
devm_kzalloc
(
dev
,
sizeof
(
*
priv
),
GFP_KERNEL
);
/* get the number of DAI links */
if
(
np
&&
of_get_child_by_name
(
np
,
"simple-audio-card,dai-link"
))
{
num_links
=
of_get_child_count
(
np
);
multi
=
1
;
}
else
{
num_links
=
1
;
multi
=
0
;
}
/* allocate the private data and the DAI link array */
priv
=
devm_kzalloc
(
dev
,
sizeof
(
*
priv
)
+
sizeof
(
*
dai_link
)
*
num_links
,
GFP_KERNEL
);
if
(
!
priv
)
if
(
!
priv
)
return
-
ENOMEM
;
return
-
ENOMEM
;
...
@@ -293,18 +337,38 @@ static int asoc_simple_card_probe(struct platform_device *pdev)
...
@@ -293,18 +337,38 @@ static int asoc_simple_card_probe(struct platform_device *pdev)
*/
*/
priv
->
snd_card
.
owner
=
THIS_MODULE
;
priv
->
snd_card
.
owner
=
THIS_MODULE
;
priv
->
snd_card
.
dev
=
dev
;
priv
->
snd_card
.
dev
=
dev
;
dai_link
=
&
priv
->
snd
_link
;
dai_link
=
priv
->
dai
_link
;
priv
->
snd_card
.
dai_link
=
dai_link
;
priv
->
snd_card
.
dai_link
=
dai_link
;
priv
->
snd_card
.
num_links
=
1
;
priv
->
snd_card
.
num_links
=
num_links
;
/* get room for the other properties */
priv
->
dai_props
=
devm_kzalloc
(
dev
,
sizeof
(
*
priv
->
dai_props
)
*
num_links
,
GFP_KERNEL
);
if
(
!
priv
->
dai_props
)
return
-
ENOMEM
;
if
(
np
&&
of_device_is_available
(
np
))
{
if
(
np
&&
of_device_is_available
(
np
))
{
ret
=
asoc_simple_card_parse_of
(
np
,
priv
,
dev
);
ret
=
asoc_simple_card_parse_of
(
np
,
priv
,
dev
,
multi
);
if
(
ret
<
0
)
{
if
(
ret
<
0
)
{
if
(
ret
!=
-
EPROBE_DEFER
)
if
(
ret
!=
-
EPROBE_DEFER
)
dev_err
(
dev
,
"parse error %d
\n
"
,
ret
);
dev_err
(
dev
,
"parse error %d
\n
"
,
ret
);
goto
err
;
goto
err
;
}
}
/*
* soc_bind_dai_link() will check cpu name
* after of_node matching if dai_link has cpu_dai_name.
* but, it will never match if name was created by fmt_single_name()
* remove cpu_dai_name to escape name matching.
* see
* fmt_single_name()
* fmt_multiple_name()
*/
if
(
num_links
==
1
)
dai_link
->
cpu_dai_name
=
NULL
;
}
else
{
}
else
{
struct
asoc_simple_card_info
*
cinfo
;
struct
asoc_simple_card_info
*
cinfo
;
...
@@ -330,13 +394,13 @@ static int asoc_simple_card_probe(struct platform_device *pdev)
...
@@ -330,13 +394,13 @@ static int asoc_simple_card_probe(struct platform_device *pdev)
dai_link
->
codec_name
=
cinfo
->
codec
;
dai_link
->
codec_name
=
cinfo
->
codec
;
dai_link
->
cpu_dai_name
=
cinfo
->
cpu_dai
.
name
;
dai_link
->
cpu_dai_name
=
cinfo
->
cpu_dai
.
name
;
dai_link
->
codec_dai_name
=
cinfo
->
codec_dai
.
name
;
dai_link
->
codec_dai_name
=
cinfo
->
codec_dai
.
name
;
memcpy
(
&
priv
->
cpu_dai
,
&
cinfo
->
cpu_dai
,
memcpy
(
&
priv
->
dai_props
->
cpu_dai
,
&
cinfo
->
cpu_dai
,
sizeof
(
priv
->
cpu_dai
));
sizeof
(
priv
->
dai_props
->
cpu_dai
));
memcpy
(
&
priv
->
codec_dai
,
&
cinfo
->
codec_dai
,
memcpy
(
&
priv
->
dai_props
->
codec_dai
,
&
cinfo
->
codec_dai
,
sizeof
(
priv
->
codec_dai
));
sizeof
(
priv
->
dai_props
->
codec_dai
));
priv
->
cpu_dai
.
fmt
|=
cinfo
->
daifmt
;
priv
->
dai_props
->
cpu_dai
.
fmt
|=
cinfo
->
daifmt
;
priv
->
codec_dai
.
fmt
|=
cinfo
->
daifmt
;
priv
->
dai_props
->
codec_dai
.
fmt
|=
cinfo
->
daifmt
;
}
}
/*
/*
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录