#ifndef _SYS_USB_AC_H
#define _SYS_USB_AC_H
#ifdef __cplusplus
extern "C" {
#endif
#include <sys/sunldi.h>
#include <sys/sysmacros.h>
#include <sys/usb/usba/usbai_private.h>
int usb_ac_open(dev_info_t *);
void usb_ac_close(dev_info_t *);
typedef struct usb_ac_unit_list {
uint_t acu_type;
void *acu_descriptor;
size_t acu_descr_length;
} usb_ac_unit_list_t;
#define USB_AC_ID_NONE 0
#define USB_AC_FIND_ONE 0
#define USB_AC_FIND_ALL 1
#define USB_AC_MAX_DEPTH 8
typedef struct usb_ac_plumbed {
struct usb_ac_state *acp_uacp;
dev_info_t *acp_dip;
uint_t acp_ifno;
int acp_driver;
ldi_handle_t acp_lh;
dev_t acp_devt;
ddi_taskq_t *acp_tqp;
int acp_flags;
#define ACP_ENABLED 1
void *acp_data;
} usb_ac_plumbed_t;
typedef struct usb_ac_to_as_req {
usb_audio_formats_t acr_curr_format;
} usb_ac_to_as_req_t;
typedef struct usb_ac_streams_info {
usb_ac_plumbed_t *acs_plumbed;
uint_t acs_rcvd_reg_data;
usb_as_registration_t acs_streams_reg;
int acs_setup_teardown_count;
uint8_t acs_default_gain;
} usb_ac_streams_info_t;
typedef struct usb_ac_power {
void *acpm_state;
int acpm_pm_busy;
uint8_t acpm_wakeup_enabled;
uint8_t acpm_pwr_states;
uint8_t acpm_capabilities;
uint8_t acpm_current_power;
} usb_ac_power_t;
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_ac_power_t::acpm_state))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_ac_power_t::acpm_wakeup_enabled))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_ac_power_t::acpm_pwr_states))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_ac_power_t::acpm_capabilities))
typedef struct usb_audio_format {
int sr;
uint_t ch;
uint_t prec;
uint_t enc;
} usb_audio_format_t;
typedef struct usb_audio_eng {
void *statep;
usb_ac_streams_info_t *streams;
audio_engine_t *af_engp;
int af_eflags;
usb_audio_format_t fmt;
uint64_t af_defgain;
unsigned intrate;
unsigned sampsz;
unsigned framesz;
unsigned fragsz;
unsigned nfrags;
unsigned fragfr;
unsigned frsmshift;
unsigned smszshift;
caddr_t bufp;
unsigned bufsz;
caddr_t bufpos;
caddr_t bufendp;
uint64_t frames;
uint64_t io_count;
uint64_t bufio_count;
boolean_t started;
boolean_t busy;
kcondvar_t usb_audio_cv;
kmutex_t lock;
} usb_audio_eng_t;
#define USB_AC_MAX_PLUMBED 3
#define USB_AC_MAX_AS_PLUMBED 2
typedef struct usb_ac_state usb_ac_state_t;
typedef struct usb_audio_ctrl {
audio_ctrl_t *af_ctrlp;
usb_ac_state_t *statep;
kmutex_t ctrl_mutex;
uint64_t cval;
} usb_audio_ctrl_t;
enum {
CTL_VOLUME_MONO = 0,
CTL_VOLUME_STERO,
CTL_REC_MONO,
CTL_REC_STERO,
CTL_REC_SRC,
CTL_MONITOR_GAIN,
CTL_MIC_BOOST,
CTL_NUM
};
#define USB_AC_ENG_MAX 2
struct usb_ac_state {
dev_info_t *usb_ac_dip;
uint_t usb_ac_instance;
usb_log_handle_t usb_ac_log_handle;
uint_t usb_ac_dev_state;
uint_t usb_ac_ifno;
kmutex_t usb_ac_mutex;
usb_client_dev_data_t *usb_ac_dev_data;
audio_dev_t *usb_ac_audio_dev;
usb_audio_eng_t engines[USB_AC_ENG_MAX];
int flags;
usb_audio_ctrl_t *controls[CTL_NUM];
usb_if_descr_t usb_ac_if_descr;
uint_t usb_ac_max_unit;
usb_ac_unit_list_t *usb_ac_units;
uchar_t **usb_ac_connections;
size_t usb_ac_connections_len;
uchar_t *usb_ac_connections_a;
size_t usb_ac_connections_a_len;
uchar_t *usb_ac_unit_type;
uchar_t *usb_ac_traverse_path;
uchar_t usb_ac_traverse_path_index;
uint64_t usb_ac_input_ports;
uint64_t usb_ac_output_ports;
usb_pipe_handle_t usb_ac_default_ph;
usb_serialization_t usb_ac_ser_acc;
usb_ac_power_t *usb_ac_pm;
uint_t usb_ac_registered_with_mixer;
uint_t usb_ac_plumbing_state;
ushort_t usb_ac_busy_count;
usb_ac_plumbed_t usb_ac_plumbed[USB_AC_MAX_PLUMBED];
int usb_ac_current_plumbed_index;
usb_ac_streams_info_t usb_ac_streams[USB_AC_MAX_AS_PLUMBED];
ddi_taskq_t *tqp;
char dstr[64];
};
_NOTE(MUTEX_PROTECTS_DATA(usb_ac_state_t::usb_ac_mutex, usb_ac_state_t))
_NOTE(MUTEX_PROTECTS_DATA(usb_ac_state_t::usb_ac_mutex, usb_ac_power_t))
_NOTE(MUTEX_PROTECTS_DATA(usb_ac_state_t::usb_ac_mutex, usb_ac_plumbed_t))
_NOTE(MUTEX_PROTECTS_DATA(usb_audio_eng_t::lock, usb_audio_eng_t))
_NOTE(MUTEX_PROTECTS_DATA(usb_audio_eng_t::lock, usb_audio_format_t))
_NOTE(MUTEX_PROTECTS_DATA(usb_audio_ctrl_t::ctrl_mutex, usb_audio_ctrl_t))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_ac_state_t::usb_ac_dip))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_ac_state_t::usb_ac_ser_acc))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_ac_state_t::usb_ac_pm))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_ac_state_t::usb_ac_instance))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_ac_state_t::usb_ac_default_ph))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_ac_state_t::usb_ac_log_handle))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_ac_state_t::usb_ac_if_descr))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_ac_state_t::usb_ac_dev_data))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_ac_state_t::usb_ac_ifno))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_ac_state_t::flags))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_ac_state_t::usb_ac_input_ports))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_ac_state_t::engines))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_ac_state_t::usb_ac_audio_dev))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_ac_state_t::controls))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_audio_eng_t::af_eflags))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_audio_eng_t::streams))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_audio_eng_t::statep))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_audio_eng_t::fmt))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_audio_eng_t::fragfr))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_audio_eng_t::frsmshift))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_audio_eng_t::started))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_audio_eng_t::af_engp))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_audio_eng_t::io_count))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_audio_eng_t::intrate))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_audio_ctrl_t::statep))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_audio_ctrl_t::af_ctrlp))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_audio_ctrl_t::cval))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_ac_plumbed_t::acp_tqp))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_ac_plumbed_t::acp_uacp))
_NOTE(DATA_READABLE_WITHOUT_LOCK(usb_audio_format_t::ch))
#define USB_AC_STATE_UNPLUMBED 0
#define USB_AC_STATE_PLUMBED 1
#define USB_AC_STATE_PLUMBED_RESTORING 2
#define USB_AC_DEF_CLOSED 0
#define USB_AC_DEF_OPENED 1
#define USB_AC_BUFFER_SIZE 256
#define USB_AC_RESTORE_DELAY drv_usectohz(1000000)
#define USB_AS_PLUMBED 1
#define USB_AH_PLUMBED 2
#define UNKNOWN_PLUMBED 3
#define AF_REGISTERED 0x1
#define AD_SETUP 0x10
int usb_audio_attach(usb_ac_state_t *);
#define AUDIO_CTRL_STEREO_VAL(l, r) (((l) & 0xff) | (((r) & 0xff) << 8))
#define AUDIO_CTRL_STEREO_LEFT(v) ((uint8_t)((v) & 0xff))
#define AUDIO_CTRL_STEREO_RIGHT(v) ((uint8_t)(((v) >> 8) & 0xff))
#define AF_MAX_GAIN 100
#define AF_MIN_GAIN 0
int usb_ac_get_audio(void *, void *, int);
void usb_ac_send_audio(void *, void *, int);
void usb_ac_stop_play(usb_ac_state_t *, usb_audio_eng_t *);
#ifdef __cplusplus
}
#endif
#endif