2015-11-23 10:27:45 +00:00
|
|
|
/*
|
|
|
|
* Greybus audio driver
|
|
|
|
* Copyright 2015 Google Inc.
|
|
|
|
* Copyright 2015 Linaro Ltd.
|
|
|
|
*
|
|
|
|
* Released under the GPLv2 only.
|
|
|
|
*/
|
|
|
|
|
2016-01-13 21:07:49 +00:00
|
|
|
#ifndef __LINUX_GBAUDIO_CODEC_H
|
|
|
|
#define __LINUX_GBAUDIO_CODEC_H
|
2015-11-23 10:27:45 +00:00
|
|
|
|
|
|
|
#include <sound/soc.h>
|
2016-03-29 11:02:36 +00:00
|
|
|
#include <sound/jack.h>
|
2015-11-23 10:27:45 +00:00
|
|
|
|
2016-01-13 21:07:46 +00:00
|
|
|
#include "greybus.h"
|
|
|
|
#include "greybus_protocols.h"
|
|
|
|
|
2015-11-23 10:27:45 +00:00
|
|
|
#define NAME_SIZE 32
|
2016-01-13 21:07:50 +00:00
|
|
|
#define MAX_DAIS 2 /* APB1, APB2 */
|
2015-11-23 10:27:45 +00:00
|
|
|
|
|
|
|
enum {
|
|
|
|
APB1_PCM = 0,
|
|
|
|
APB2_PCM,
|
|
|
|
NUM_CODEC_DAIS,
|
|
|
|
};
|
|
|
|
|
|
|
|
enum gbcodec_reg_index {
|
|
|
|
GBCODEC_CTL_REG,
|
|
|
|
GBCODEC_MUTE_REG,
|
|
|
|
GBCODEC_PB_LVOL_REG,
|
|
|
|
GBCODEC_PB_RVOL_REG,
|
|
|
|
GBCODEC_CAP_LVOL_REG,
|
|
|
|
GBCODEC_CAP_RVOL_REG,
|
|
|
|
GBCODEC_APB1_MUX_REG,
|
|
|
|
GBCODEC_APB2_MUX_REG,
|
|
|
|
GBCODEC_REG_COUNT
|
|
|
|
};
|
|
|
|
|
2016-03-30 07:53:56 +00:00
|
|
|
/* device_type should be same as defined in audio.h (Android media layer) */
|
|
|
|
enum {
|
|
|
|
GBAUDIO_DEVICE_NONE = 0x0,
|
|
|
|
/* reserved bits */
|
|
|
|
GBAUDIO_DEVICE_BIT_IN = 0x80000000,
|
|
|
|
GBAUDIO_DEVICE_BIT_DEFAULT = 0x40000000,
|
|
|
|
/* output devices */
|
|
|
|
GBAUDIO_DEVICE_OUT_SPEAKER = 0x2,
|
|
|
|
GBAUDIO_DEVICE_OUT_WIRED_HEADSET = 0x4,
|
|
|
|
GBAUDIO_DEVICE_OUT_WIRED_HEADPHONE = 0x8,
|
|
|
|
/* input devices */
|
|
|
|
GBAUDIO_DEVICE_IN_BUILTIN_MIC = GBAUDIO_DEVICE_BIT_IN | 0x4,
|
|
|
|
GBAUDIO_DEVICE_IN_WIRED_HEADSET = GBAUDIO_DEVICE_BIT_IN | 0x10,
|
|
|
|
};
|
|
|
|
|
2015-11-23 10:27:45 +00:00
|
|
|
/* bit 0-SPK, 1-HP, 2-DAC,
|
|
|
|
* 4-MIC, 5-HSMIC, 6-MIC2
|
|
|
|
*/
|
|
|
|
#define GBCODEC_CTL_REG_DEFAULT 0x00
|
|
|
|
|
|
|
|
/* bit 0,1 - APB1-PB-L/R
|
|
|
|
* bit 2,3 - APB2-PB-L/R
|
|
|
|
* bit 4,5 - APB1-Cap-L/R
|
|
|
|
* bit 6,7 - APB2-Cap-L/R
|
|
|
|
*/
|
|
|
|
#define GBCODEC_MUTE_REG_DEFAULT 0x00
|
|
|
|
|
|
|
|
/* 0-127 steps */
|
|
|
|
#define GBCODEC_PB_VOL_REG_DEFAULT 0x00
|
|
|
|
#define GBCODEC_CAP_VOL_REG_DEFAULT 0x00
|
|
|
|
|
|
|
|
/* bit 0,1,2 - PB stereo, left, right
|
|
|
|
* bit 8,9,10 - Cap stereo, left, right
|
|
|
|
*/
|
|
|
|
#define GBCODEC_APB1_MUX_REG_DEFAULT 0x00
|
|
|
|
#define GBCODEC_APB2_MUX_REG_DEFAULT 0x00
|
|
|
|
|
2016-03-29 11:02:36 +00:00
|
|
|
#define GBCODEC_JACK_MASK (SND_JACK_HEADSET | SND_JACK_LINEOUT | \
|
|
|
|
SND_JACK_LINEIN | SND_JACK_UNSUPPORTED)
|
|
|
|
#define GBCODEC_JACK_BUTTON_MASK (SND_JACK_BTN_0 | SND_JACK_BTN_1 | \
|
|
|
|
SND_JACK_BTN_2 | SND_JACK_BTN_3)
|
|
|
|
|
2015-11-23 10:27:45 +00:00
|
|
|
static const u8 gbcodec_reg_defaults[GBCODEC_REG_COUNT] = {
|
|
|
|
GBCODEC_CTL_REG_DEFAULT,
|
|
|
|
GBCODEC_MUTE_REG_DEFAULT,
|
|
|
|
GBCODEC_PB_VOL_REG_DEFAULT,
|
|
|
|
GBCODEC_PB_VOL_REG_DEFAULT,
|
|
|
|
GBCODEC_CAP_VOL_REG_DEFAULT,
|
|
|
|
GBCODEC_CAP_VOL_REG_DEFAULT,
|
|
|
|
GBCODEC_APB1_MUX_REG_DEFAULT,
|
|
|
|
GBCODEC_APB2_MUX_REG_DEFAULT,
|
|
|
|
};
|
|
|
|
|
2016-03-29 18:01:41 +00:00
|
|
|
enum gbaudio_codec_state {
|
|
|
|
GBAUDIO_CODEC_SHUTDOWN = 0,
|
|
|
|
GBAUDIO_CODEC_STARTUP,
|
|
|
|
GBAUDIO_CODEC_HWPARAMS,
|
|
|
|
GBAUDIO_CODEC_PREPARE,
|
|
|
|
GBAUDIO_CODEC_START,
|
|
|
|
GBAUDIO_CODEC_STOP,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct gbaudio_stream {
|
|
|
|
const char *dai_name;
|
|
|
|
int state;
|
|
|
|
uint8_t sig_bits, channels;
|
|
|
|
uint32_t format, rate;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct gbaudio_codec_info {
|
|
|
|
struct device *dev;
|
|
|
|
struct snd_soc_codec *codec;
|
|
|
|
struct list_head module_list;
|
|
|
|
struct gbaudio_stream stream[2]; /* PB/CAP */
|
|
|
|
struct mutex lock;
|
|
|
|
u8 reg[GBCODEC_REG_COUNT];
|
|
|
|
};
|
|
|
|
|
2016-01-13 21:07:51 +00:00
|
|
|
struct gbaudio_widget {
|
|
|
|
__u8 id;
|
|
|
|
const char *name;
|
|
|
|
struct list_head list;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct gbaudio_control {
|
|
|
|
__u8 id;
|
|
|
|
char *name;
|
2016-07-05 22:09:20 +00:00
|
|
|
char *wname;
|
2016-01-13 21:07:51 +00:00
|
|
|
const char * const *texts;
|
|
|
|
struct list_head list;
|
|
|
|
};
|
|
|
|
|
2016-03-29 18:01:41 +00:00
|
|
|
struct gbaudio_data_connection {
|
2016-01-13 21:07:50 +00:00
|
|
|
__le16 data_cport;
|
2016-03-29 18:01:41 +00:00
|
|
|
int cport_configured;
|
2016-01-13 21:07:50 +00:00
|
|
|
char name[NAME_SIZE];
|
|
|
|
struct gb_connection *connection;
|
|
|
|
struct list_head list;
|
|
|
|
};
|
|
|
|
|
2016-03-29 18:01:41 +00:00
|
|
|
/* stream direction */
|
|
|
|
#define GB_PLAYBACK BIT(0)
|
|
|
|
#define GB_CAPTURE BIT(1)
|
|
|
|
|
|
|
|
enum gbaudio_module_state {
|
|
|
|
GBAUDIO_MODULE_OFF = 0,
|
|
|
|
GBAUDIO_MODULE_ON,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct gbaudio_module_info {
|
2016-01-13 21:07:50 +00:00
|
|
|
/* module info */
|
2016-03-29 18:01:41 +00:00
|
|
|
struct device *dev;
|
2016-01-13 21:07:50 +00:00
|
|
|
int dev_id; /* check if it should be bundle_id/hd_cport_id */
|
|
|
|
int vid;
|
|
|
|
int pid;
|
|
|
|
int slot;
|
|
|
|
int type;
|
2016-01-13 21:07:53 +00:00
|
|
|
int set_uevent;
|
2016-01-13 21:07:50 +00:00
|
|
|
char vstr[NAME_SIZE];
|
|
|
|
char pstr[NAME_SIZE];
|
|
|
|
struct list_head list;
|
2016-01-13 21:07:53 +00:00
|
|
|
/* need to share this info to above user space */
|
|
|
|
int manager_id;
|
2016-01-13 21:07:50 +00:00
|
|
|
char name[NAME_SIZE];
|
2016-03-30 07:53:56 +00:00
|
|
|
unsigned int ip_devices;
|
|
|
|
unsigned int op_devices;
|
2015-11-23 10:27:45 +00:00
|
|
|
|
2016-03-29 11:02:36 +00:00
|
|
|
/* jack related */
|
|
|
|
char jack_name[NAME_SIZE];
|
|
|
|
char button_name[NAME_SIZE];
|
|
|
|
int num_jacks;
|
|
|
|
int jack_type;
|
|
|
|
int button_status;
|
|
|
|
struct snd_soc_jack headset_jack;
|
|
|
|
struct snd_soc_jack button_jack;
|
|
|
|
|
2016-03-29 18:01:41 +00:00
|
|
|
/* used by codec_ops */
|
|
|
|
int ctrlstate[2]; /* PB/CAP */
|
2016-01-28 15:45:40 +00:00
|
|
|
|
2016-03-29 18:01:41 +00:00
|
|
|
/* connection info */
|
2016-01-13 21:07:50 +00:00
|
|
|
struct gb_connection *mgmt_connection;
|
2016-01-28 15:45:39 +00:00
|
|
|
size_t num_data_connections;
|
2016-03-29 18:01:41 +00:00
|
|
|
struct list_head data_list;
|
|
|
|
|
2016-01-28 15:45:39 +00:00
|
|
|
/* topology related */
|
2016-01-13 21:07:50 +00:00
|
|
|
int num_dais;
|
2016-03-29 18:01:41 +00:00
|
|
|
int num_controls;
|
2015-11-23 10:27:45 +00:00
|
|
|
int num_dapm_widgets;
|
|
|
|
int num_dapm_routes;
|
2016-01-13 21:07:51 +00:00
|
|
|
unsigned long dai_offset;
|
|
|
|
unsigned long widget_offset;
|
|
|
|
unsigned long control_offset;
|
|
|
|
unsigned long route_offset;
|
2016-03-29 18:01:41 +00:00
|
|
|
struct snd_kcontrol_new *controls;
|
|
|
|
struct snd_soc_dapm_widget *dapm_widgets;
|
|
|
|
struct snd_soc_dapm_route *dapm_routes;
|
2016-01-13 21:07:50 +00:00
|
|
|
struct snd_soc_dai_driver *dais;
|
|
|
|
|
2016-01-13 21:07:51 +00:00
|
|
|
struct list_head widget_list;
|
2016-03-29 18:01:41 +00:00
|
|
|
struct list_head ctl_list;
|
2016-01-13 21:07:51 +00:00
|
|
|
struct list_head widget_ctl_list;
|
2016-03-29 18:01:41 +00:00
|
|
|
|
|
|
|
struct gb_audio_topology *topology;
|
2015-11-23 10:27:45 +00:00
|
|
|
};
|
|
|
|
|
2016-03-29 18:01:41 +00:00
|
|
|
int gbaudio_tplg_parse_data(struct gbaudio_module_info *module,
|
2016-01-13 21:07:51 +00:00
|
|
|
struct gb_audio_topology *tplg_data);
|
2016-03-29 18:01:41 +00:00
|
|
|
void gbaudio_tplg_release(struct gbaudio_module_info *module);
|
|
|
|
|
|
|
|
int gbaudio_module_update(struct gbaudio_codec_info *codec,
|
2016-08-04 09:44:32 +00:00
|
|
|
struct snd_soc_dapm_widget *w,
|
|
|
|
struct gbaudio_module_info *module,
|
|
|
|
int enable);
|
2016-03-29 18:01:41 +00:00
|
|
|
int gbaudio_register_module(struct gbaudio_module_info *module);
|
|
|
|
void gbaudio_unregister_module(struct gbaudio_module_info *module);
|
2016-01-13 21:07:51 +00:00
|
|
|
|
2016-01-13 21:07:50 +00:00
|
|
|
/* protocol related */
|
2016-01-13 21:07:51 +00:00
|
|
|
extern int gb_audio_gb_get_topology(struct gb_connection *connection,
|
|
|
|
struct gb_audio_topology **topology);
|
2016-01-13 21:07:46 +00:00
|
|
|
extern int gb_audio_gb_get_control(struct gb_connection *connection,
|
|
|
|
uint8_t control_id, uint8_t index,
|
|
|
|
struct gb_audio_ctl_elem_value *value);
|
|
|
|
extern int gb_audio_gb_set_control(struct gb_connection *connection,
|
|
|
|
uint8_t control_id, uint8_t index,
|
|
|
|
struct gb_audio_ctl_elem_value *value);
|
|
|
|
extern int gb_audio_gb_enable_widget(struct gb_connection *connection,
|
|
|
|
uint8_t widget_id);
|
|
|
|
extern int gb_audio_gb_disable_widget(struct gb_connection *connection,
|
|
|
|
uint8_t widget_id);
|
|
|
|
extern int gb_audio_gb_get_pcm(struct gb_connection *connection,
|
|
|
|
uint16_t data_cport, uint32_t *format,
|
|
|
|
uint32_t *rate, uint8_t *channels,
|
|
|
|
uint8_t *sig_bits);
|
|
|
|
extern int gb_audio_gb_set_pcm(struct gb_connection *connection,
|
|
|
|
uint16_t data_cport, uint32_t format,
|
|
|
|
uint32_t rate, uint8_t channels,
|
|
|
|
uint8_t sig_bits);
|
|
|
|
extern int gb_audio_gb_set_tx_data_size(struct gb_connection *connection,
|
|
|
|
uint16_t data_cport, uint16_t size);
|
|
|
|
extern int gb_audio_gb_activate_tx(struct gb_connection *connection,
|
|
|
|
uint16_t data_cport);
|
|
|
|
extern int gb_audio_gb_deactivate_tx(struct gb_connection *connection,
|
|
|
|
uint16_t data_cport);
|
|
|
|
extern int gb_audio_gb_set_rx_data_size(struct gb_connection *connection,
|
|
|
|
uint16_t data_cport, uint16_t size);
|
|
|
|
extern int gb_audio_gb_activate_rx(struct gb_connection *connection,
|
|
|
|
uint16_t data_cport);
|
|
|
|
extern int gb_audio_gb_deactivate_rx(struct gb_connection *connection,
|
|
|
|
uint16_t data_cport);
|
2016-01-13 21:07:47 +00:00
|
|
|
extern int gb_audio_apbridgea_set_config(struct gb_connection *connection,
|
|
|
|
__u16 i2s_port, __u32 format,
|
|
|
|
__u32 rate, __u32 mclk_freq);
|
|
|
|
extern int gb_audio_apbridgea_register_cport(struct gb_connection *connection,
|
2016-02-27 00:04:36 +00:00
|
|
|
__u16 i2s_port, __u16 cportid,
|
|
|
|
__u8 direction);
|
2016-01-13 21:07:47 +00:00
|
|
|
extern int gb_audio_apbridgea_unregister_cport(struct gb_connection *connection,
|
2016-02-27 00:04:36 +00:00
|
|
|
__u16 i2s_port, __u16 cportid,
|
|
|
|
__u8 direction);
|
2016-01-13 21:07:47 +00:00
|
|
|
extern int gb_audio_apbridgea_set_tx_data_size(struct gb_connection *connection,
|
|
|
|
__u16 i2s_port, __u16 size);
|
2016-02-29 22:31:02 +00:00
|
|
|
extern int gb_audio_apbridgea_prepare_tx(struct gb_connection *connection,
|
|
|
|
__u16 i2s_port);
|
2016-01-13 21:07:47 +00:00
|
|
|
extern int gb_audio_apbridgea_start_tx(struct gb_connection *connection,
|
|
|
|
__u16 i2s_port, __u64 timestamp);
|
|
|
|
extern int gb_audio_apbridgea_stop_tx(struct gb_connection *connection,
|
|
|
|
__u16 i2s_port);
|
2016-02-29 22:31:02 +00:00
|
|
|
extern int gb_audio_apbridgea_shutdown_tx(struct gb_connection *connection,
|
|
|
|
__u16 i2s_port);
|
2016-01-13 21:07:47 +00:00
|
|
|
extern int gb_audio_apbridgea_set_rx_data_size(struct gb_connection *connection,
|
|
|
|
__u16 i2s_port, __u16 size);
|
2016-02-29 22:31:02 +00:00
|
|
|
extern int gb_audio_apbridgea_prepare_rx(struct gb_connection *connection,
|
|
|
|
__u16 i2s_port);
|
2016-01-13 21:07:47 +00:00
|
|
|
extern int gb_audio_apbridgea_start_rx(struct gb_connection *connection,
|
|
|
|
__u16 i2s_port);
|
|
|
|
extern int gb_audio_apbridgea_stop_rx(struct gb_connection *connection,
|
|
|
|
__u16 i2s_port);
|
2016-02-29 22:31:02 +00:00
|
|
|
extern int gb_audio_apbridgea_shutdown_rx(struct gb_connection *connection,
|
|
|
|
__u16 i2s_port);
|
2016-01-13 21:07:46 +00:00
|
|
|
|
2016-01-13 21:07:49 +00:00
|
|
|
#endif /* __LINUX_GBAUDIO_CODEC_H */
|