Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
linux
Commits
0334a537
Commit
0334a537
authored
Mar 28, 2018
by
Mark Brown
Browse files
Options
Browse Files
Download
Plain Diff
Merge remote-tracking branch 'asoc/topic/pcm1789' into asoc-next
parents
338f803a
e4fe156a
Changes
6
Hide whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
391 additions
and
0 deletions
+391
-0
Documentation/devicetree/bindings/sound/pcm1789.txt
Documentation/devicetree/bindings/sound/pcm1789.txt
+22
-0
sound/soc/codecs/Kconfig
sound/soc/codecs/Kconfig
+12
-0
sound/soc/codecs/Makefile
sound/soc/codecs/Makefile
+4
-0
sound/soc/codecs/pcm1789-i2c.c
sound/soc/codecs/pcm1789-i2c.c
+62
-0
sound/soc/codecs/pcm1789.c
sound/soc/codecs/pcm1789.c
+274
-0
sound/soc/codecs/pcm1789.h
sound/soc/codecs/pcm1789.h
+17
-0
No files found.
Documentation/devicetree/bindings/sound/pcm1789.txt
0 → 100644
View file @
0334a537
Texas Instruments pcm1789 DT bindings
PCM1789 is a simple audio codec that can be connected via
I2C or SPI. Currently, only I2C bus is supported.
Required properties:
- compatible: "ti,pcm1789"
Required properties on I2C:
- reg: the I2C address
- reset-gpios: GPIO to control the RESET pin
Examples:
audio-codec@4c {
compatible = "ti,pcm1789";
reg = <0x4c>;
reset-gpios = <&gpio2 14 GPIO_ACTIVE_LOW>;
#sound-dai-cells = <0>;
};
sound/soc/codecs/Kconfig
View file @
0334a537
...
@@ -108,6 +108,7 @@ config SND_SOC_ALL_CODECS
...
@@ -108,6 +108,7 @@ config SND_SOC_ALL_CODECS
select SND_SOC_NAU8825 if I2C
select SND_SOC_NAU8825 if I2C
select SND_SOC_HDMI_CODEC
select SND_SOC_HDMI_CODEC
select SND_SOC_PCM1681 if I2C
select SND_SOC_PCM1681 if I2C
select SND_SOC_PCM1789_I2C if I2C
select SND_SOC_PCM179X_I2C if I2C
select SND_SOC_PCM179X_I2C if I2C
select SND_SOC_PCM179X_SPI if SPI_MASTER
select SND_SOC_PCM179X_SPI if SPI_MASTER
select SND_SOC_PCM186X_I2C if I2C
select SND_SOC_PCM186X_I2C if I2C
...
@@ -650,6 +651,17 @@ config SND_SOC_PCM1681
...
@@ -650,6 +651,17 @@ config SND_SOC_PCM1681
tristate "Texas Instruments PCM1681 CODEC"
tristate "Texas Instruments PCM1681 CODEC"
depends on I2C
depends on I2C
config SND_SOC_PCM1789
tristate
config SND_SOC_PCM1789_I2C
tristate "Texas Instruments PCM1789 CODEC (I2C)"
depends on I2C
select SND_SOC_PCM1789
help
Enable support for Texas Instruments PCM1789 CODEC.
Select this if your PCM1789 is connected via an I2C bus.
config SND_SOC_PCM179X
config SND_SOC_PCM179X
tristate
tristate
...
...
sound/soc/codecs/Makefile
View file @
0334a537
...
@@ -103,6 +103,8 @@ snd-soc-nau8824-objs := nau8824.o
...
@@ -103,6 +103,8 @@ snd-soc-nau8824-objs := nau8824.o
snd-soc-nau8825-objs
:=
nau8825.o
snd-soc-nau8825-objs
:=
nau8825.o
snd-soc-hdmi-codec-objs
:=
hdmi-codec.o
snd-soc-hdmi-codec-objs
:=
hdmi-codec.o
snd-soc-pcm1681-objs
:=
pcm1681.o
snd-soc-pcm1681-objs
:=
pcm1681.o
snd-soc-pcm1789-codec-objs
:=
pcm1789.o
snd-soc-pcm1789-i2c-objs
:=
pcm1789-i2c.o
snd-soc-pcm179x-codec-objs
:=
pcm179x.o
snd-soc-pcm179x-codec-objs
:=
pcm179x.o
snd-soc-pcm179x-i2c-objs
:=
pcm179x-i2c.o
snd-soc-pcm179x-i2c-objs
:=
pcm179x-i2c.o
snd-soc-pcm179x-spi-objs
:=
pcm179x-spi.o
snd-soc-pcm179x-spi-objs
:=
pcm179x-spi.o
...
@@ -349,6 +351,8 @@ obj-$(CONFIG_SND_SOC_NAU8825) += snd-soc-nau8825.o
...
@@ -349,6 +351,8 @@ obj-$(CONFIG_SND_SOC_NAU8825) += snd-soc-nau8825.o
obj-$(CONFIG_SND_SOC_HDMI_CODEC)
+=
snd-soc-hdmi-codec.o
obj-$(CONFIG_SND_SOC_HDMI_CODEC)
+=
snd-soc-hdmi-codec.o
obj-$(CONFIG_SND_SOC_PCM1681)
+=
snd-soc-pcm1681.o
obj-$(CONFIG_SND_SOC_PCM1681)
+=
snd-soc-pcm1681.o
obj-$(CONFIG_SND_SOC_PCM179X)
+=
snd-soc-pcm179x-codec.o
obj-$(CONFIG_SND_SOC_PCM179X)
+=
snd-soc-pcm179x-codec.o
obj-$(CONFIG_SND_SOC_PCM1789_I2C)
+=
snd-soc-pcm1789-i2c.o
obj-$(CONFIG_SND_SOC_PCM1789)
+=
snd-soc-pcm1789-codec.o
obj-$(CONFIG_SND_SOC_PCM179X_I2C)
+=
snd-soc-pcm179x-i2c.o
obj-$(CONFIG_SND_SOC_PCM179X_I2C)
+=
snd-soc-pcm179x-i2c.o
obj-$(CONFIG_SND_SOC_PCM179X_SPI)
+=
snd-soc-pcm179x-spi.o
obj-$(CONFIG_SND_SOC_PCM179X_SPI)
+=
snd-soc-pcm179x-spi.o
obj-$(CONFIG_SND_SOC_PCM186X)
+=
snd-soc-pcm186x.o
obj-$(CONFIG_SND_SOC_PCM186X)
+=
snd-soc-pcm186x.o
...
...
sound/soc/codecs/pcm1789-i2c.c
0 → 100644
View file @
0334a537
// SPDX-License-Identifier: GPL-2.0
// Audio driver for PCM1789 I2C
// Copyright (C) 2018 Bootlin
// Mylène Josserand <mylene.josserand@bootlin.com>
#include <linux/clk.h>
#include <linux/delay.h>
#include <linux/i2c.h>
#include <linux/module.h>
#include <linux/of.h>
#include <linux/regmap.h>
#include "pcm1789.h"
static
int
pcm1789_i2c_probe
(
struct
i2c_client
*
client
,
const
struct
i2c_device_id
*
id
)
{
struct
regmap
*
regmap
;
int
ret
;
regmap
=
devm_regmap_init_i2c
(
client
,
&
pcm1789_regmap_config
);
if
(
IS_ERR
(
regmap
))
{
ret
=
PTR_ERR
(
regmap
);
dev_err
(
&
client
->
dev
,
"Failed to allocate regmap: %d
\n
"
,
ret
);
return
ret
;
}
return
pcm1789_common_init
(
&
client
->
dev
,
regmap
);
}
static
int
pcm1789_i2c_remove
(
struct
i2c_client
*
client
)
{
return
pcm1789_common_exit
(
&
client
->
dev
);
}
static
const
struct
of_device_id
pcm1789_of_match
[]
=
{
{
.
compatible
=
"ti,pcm1789"
,
},
{
}
};
MODULE_DEVICE_TABLE
(
of
,
pcm1789_of_match
);
static
const
struct
i2c_device_id
pcm1789_i2c_ids
[]
=
{
{
"pcm1789"
,
0
},
{
}
};
MODULE_DEVICE_TABLE
(
i2c
,
pcm1789_i2c_ids
);
static
struct
i2c_driver
pcm1789_i2c_driver
=
{
.
driver
=
{
.
name
=
"pcm1789"
,
.
of_match_table
=
of_match_ptr
(
pcm1789_of_match
),
},
.
id_table
=
pcm1789_i2c_ids
,
.
probe
=
pcm1789_i2c_probe
,
.
remove
=
pcm1789_i2c_remove
,
};
module_i2c_driver
(
pcm1789_i2c_driver
);
MODULE_DESCRIPTION
(
"ASoC PCM1789 I2C driver"
);
MODULE_AUTHOR
(
"Mylène Josserand <mylene.josserand@bootlin.com>"
);
MODULE_LICENSE
(
"GPL"
);
sound/soc/codecs/pcm1789.c
0 → 100644
View file @
0334a537
// SPDX-License-Identifier: GPL-2.0
// Audio driver for PCM1789
// Copyright (C) 2018 Bootlin
// Mylène Josserand <mylene.josserand@bootlin.com>
#include <linux/gpio.h>
#include <linux/module.h>
#include <linux/workqueue.h>
#include <sound/pcm_params.h>
#include <sound/soc.h>
#include <sound/tlv.h>
#include "pcm1789.h"
#define PCM1789_MUTE_CONTROL 0x10
#define PCM1789_FMT_CONTROL 0x11
#define PCM1789_SOFT_MUTE 0x14
#define PCM1789_DAC_VOL_LEFT 0x18
#define PCM1789_DAC_VOL_RIGHT 0x19
#define PCM1789_FMT_MASK 0x07
#define PCM1789_MUTE_MASK 0x03
#define PCM1789_MUTE_SRET 0x06
struct
pcm1789_private
{
struct
regmap
*
regmap
;
unsigned
int
format
;
unsigned
int
rate
;
struct
gpio_desc
*
reset
;
struct
work_struct
work
;
struct
device
*
dev
;
};
static
const
struct
reg_default
pcm1789_reg_defaults
[]
=
{
{
PCM1789_FMT_CONTROL
,
0x00
},
{
PCM1789_SOFT_MUTE
,
0x00
},
{
PCM1789_DAC_VOL_LEFT
,
0xff
},
{
PCM1789_DAC_VOL_RIGHT
,
0xff
},
};
static
bool
pcm1789_accessible_reg
(
struct
device
*
dev
,
unsigned
int
reg
)
{
return
reg
>=
PCM1789_MUTE_CONTROL
&&
reg
<=
PCM1789_DAC_VOL_RIGHT
;
}
static
bool
pcm1789_writeable_reg
(
struct
device
*
dev
,
unsigned
int
reg
)
{
return
pcm1789_accessible_reg
(
dev
,
reg
);
}
static
int
pcm1789_set_dai_fmt
(
struct
snd_soc_dai
*
codec_dai
,
unsigned
int
format
)
{
struct
snd_soc_component
*
component
=
codec_dai
->
component
;
struct
pcm1789_private
*
priv
=
snd_soc_component_get_drvdata
(
component
);
priv
->
format
=
format
;
return
0
;
}
static
int
pcm1789_digital_mute
(
struct
snd_soc_dai
*
codec_dai
,
int
mute
)
{
struct
snd_soc_component
*
component
=
codec_dai
->
component
;
struct
pcm1789_private
*
priv
=
snd_soc_component_get_drvdata
(
component
);
return
regmap_update_bits
(
priv
->
regmap
,
PCM1789_SOFT_MUTE
,
PCM1789_MUTE_MASK
,
mute
?
0
:
PCM1789_MUTE_MASK
);
}
static
int
pcm1789_hw_params
(
struct
snd_pcm_substream
*
substream
,
struct
snd_pcm_hw_params
*
params
,
struct
snd_soc_dai
*
codec_dai
)
{
struct
snd_soc_component
*
component
=
codec_dai
->
component
;
struct
pcm1789_private
*
priv
=
snd_soc_component_get_drvdata
(
component
);
int
val
=
0
,
ret
;
priv
->
rate
=
params_rate
(
params
);
switch
(
priv
->
format
&
SND_SOC_DAIFMT_FORMAT_MASK
)
{
case
SND_SOC_DAIFMT_RIGHT_J
:
switch
(
params_width
(
params
))
{
case
24
:
val
=
2
;
break
;
case
16
:
val
=
3
;
break
;
default:
return
-
EINVAL
;
}
break
;
case
SND_SOC_DAIFMT_I2S
:
switch
(
params_width
(
params
))
{
case
16
:
case
24
:
case
32
:
val
=
0
;
break
;
default:
return
-
EINVAL
;
}
break
;
case
SND_SOC_DAIFMT_LEFT_J
:
switch
(
params_width
(
params
))
{
case
16
:
case
24
:
case
32
:
val
=
1
;
break
;
default:
return
-
EINVAL
;
}
break
;
default:
dev_err
(
component
->
dev
,
"Invalid DAI format
\n
"
);
return
-
EINVAL
;
}
ret
=
regmap_update_bits
(
priv
->
regmap
,
PCM1789_FMT_CONTROL
,
PCM1789_FMT_MASK
,
val
);
if
(
ret
<
0
)
return
ret
;
return
0
;
}
static
void
pcm1789_work_queue
(
struct
work_struct
*
work
)
{
struct
pcm1789_private
*
priv
=
container_of
(
work
,
struct
pcm1789_private
,
work
);
/* Perform a software reset to remove codec from desynchronized state */
if
(
regmap_update_bits
(
priv
->
regmap
,
PCM1789_MUTE_CONTROL
,
0x3
<<
PCM1789_MUTE_SRET
,
0
)
<
0
)
dev_err
(
priv
->
dev
,
"Error while setting SRET"
);
}
static
int
pcm1789_trigger
(
struct
snd_pcm_substream
*
substream
,
int
cmd
,
struct
snd_soc_dai
*
dai
)
{
struct
snd_soc_component
*
component
=
dai
->
component
;
struct
pcm1789_private
*
priv
=
snd_soc_component_get_drvdata
(
component
);
int
ret
=
0
;
switch
(
cmd
)
{
case
SNDRV_PCM_TRIGGER_START
:
case
SNDRV_PCM_TRIGGER_RESUME
:
case
SNDRV_PCM_TRIGGER_PAUSE_RELEASE
:
schedule_work
(
&
priv
->
work
);
break
;
case
SNDRV_PCM_TRIGGER_STOP
:
case
SNDRV_PCM_TRIGGER_SUSPEND
:
case
SNDRV_PCM_TRIGGER_PAUSE_PUSH
:
break
;
default:
ret
=
-
EINVAL
;
}
return
ret
;
}
static
const
struct
snd_soc_dai_ops
pcm1789_dai_ops
=
{
.
set_fmt
=
pcm1789_set_dai_fmt
,
.
hw_params
=
pcm1789_hw_params
,
.
digital_mute
=
pcm1789_digital_mute
,
.
trigger
=
pcm1789_trigger
,
};
static
const
DECLARE_TLV_DB_SCALE
(
pcm1789_dac_tlv
,
-
12000
,
50
,
1
);
static
const
struct
snd_kcontrol_new
pcm1789_controls
[]
=
{
SOC_DOUBLE_R_RANGE_TLV
(
"DAC Playback Volume"
,
PCM1789_DAC_VOL_LEFT
,
PCM1789_DAC_VOL_RIGHT
,
0
,
0xf
,
0xff
,
0
,
pcm1789_dac_tlv
),
};
static
const
struct
snd_soc_dapm_widget
pcm1789_dapm_widgets
[]
=
{
SND_SOC_DAPM_OUTPUT
(
"IOUTL+"
),
SND_SOC_DAPM_OUTPUT
(
"IOUTL-"
),
SND_SOC_DAPM_OUTPUT
(
"IOUTR+"
),
SND_SOC_DAPM_OUTPUT
(
"IOUTR-"
),
};
static
const
struct
snd_soc_dapm_route
pcm1789_dapm_routes
[]
=
{
{
"IOUTL+"
,
NULL
,
"Playback"
},
{
"IOUTL-"
,
NULL
,
"Playback"
},
{
"IOUTR+"
,
NULL
,
"Playback"
},
{
"IOUTR-"
,
NULL
,
"Playback"
},
};
static
struct
snd_soc_dai_driver
pcm1789_dai
=
{
.
name
=
"pcm1789-hifi"
,
.
playback
=
{
.
stream_name
=
"Playback"
,
.
channels_min
=
2
,
.
channels_max
=
2
,
.
rates
=
SNDRV_PCM_RATE_CONTINUOUS
,
.
rate_min
=
10000
,
.
rate_max
=
200000
,
.
formats
=
PCM1789_FORMATS
,
},
.
ops
=
&
pcm1789_dai_ops
,
};
const
struct
regmap_config
pcm1789_regmap_config
=
{
.
reg_bits
=
8
,
.
val_bits
=
8
,
.
max_register
=
PCM1789_DAC_VOL_RIGHT
,
.
reg_defaults
=
pcm1789_reg_defaults
,
.
num_reg_defaults
=
ARRAY_SIZE
(
pcm1789_reg_defaults
),
.
writeable_reg
=
pcm1789_writeable_reg
,
.
readable_reg
=
pcm1789_accessible_reg
,
};
EXPORT_SYMBOL_GPL
(
pcm1789_regmap_config
);
static
const
struct
snd_soc_component_driver
soc_component_dev_pcm1789
=
{
.
controls
=
pcm1789_controls
,
.
num_controls
=
ARRAY_SIZE
(
pcm1789_controls
),
.
dapm_widgets
=
pcm1789_dapm_widgets
,
.
num_dapm_widgets
=
ARRAY_SIZE
(
pcm1789_dapm_widgets
),
.
dapm_routes
=
pcm1789_dapm_routes
,
.
num_dapm_routes
=
ARRAY_SIZE
(
pcm1789_dapm_routes
),
.
idle_bias_on
=
1
,
.
use_pmdown_time
=
1
,
.
endianness
=
1
,
.
non_legacy_dai_naming
=
1
,
};
int
pcm1789_common_init
(
struct
device
*
dev
,
struct
regmap
*
regmap
)
{
struct
pcm1789_private
*
pcm1789
;
pcm1789
=
devm_kzalloc
(
dev
,
sizeof
(
struct
pcm1789_private
),
GFP_KERNEL
);
if
(
!
pcm1789
)
return
-
ENOMEM
;
pcm1789
->
regmap
=
regmap
;
pcm1789
->
dev
=
dev
;
dev_set_drvdata
(
dev
,
pcm1789
);
pcm1789
->
reset
=
devm_gpiod_get_optional
(
dev
,
"reset"
,
GPIOD_OUT_HIGH
);
if
(
IS_ERR
(
pcm1789
->
reset
))
return
PTR_ERR
(
pcm1789
->
reset
);
gpiod_set_value_cansleep
(
pcm1789
->
reset
,
0
);
msleep
(
300
);
INIT_WORK
(
&
pcm1789
->
work
,
pcm1789_work_queue
);
return
devm_snd_soc_register_component
(
dev
,
&
soc_component_dev_pcm1789
,
&
pcm1789_dai
,
1
);
}
EXPORT_SYMBOL_GPL
(
pcm1789_common_init
);
int
pcm1789_common_exit
(
struct
device
*
dev
)
{
struct
pcm1789_private
*
priv
=
dev_get_drvdata
(
dev
);
if
(
&
priv
->
work
)
flush_work
(
&
priv
->
work
);
return
0
;
}
EXPORT_SYMBOL_GPL
(
pcm1789_common_exit
);
MODULE_DESCRIPTION
(
"ASoC PCM1789 driver"
);
MODULE_AUTHOR
(
"Mylène Josserand <mylene.josserand@free-electrons.com>"
);
MODULE_LICENSE
(
"GPL"
);
sound/soc/codecs/pcm1789.h
0 → 100644
View file @
0334a537
// SPDX-License-Identifier: GPL-2.0
// Definitions for PCM1789 audio driver
// Copyright (C) 2018 Bootlin
// Mylène Josserand <mylene.josserand@bootlin.com>
#ifndef __PCM1789_H__
#define __PCM1789_H__
#define PCM1789_FORMATS (SNDRV_PCM_FMTBIT_S32_LE | SNDRV_PCM_FMTBIT_S24_LE | \
SNDRV_PCM_FMTBIT_S16_LE)
extern
const
struct
regmap_config
pcm1789_regmap_config
;
int
pcm1789_common_init
(
struct
device
*
dev
,
struct
regmap
*
regmap
);
int
pcm1789_common_exit
(
struct
device
*
dev
);
#endif
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment