310 lines
9.9 KiB
C
310 lines
9.9 KiB
C
/* ALSA Card Profile */
|
|
/* SPDX-FileCopyrightText: Copyright © 2020 Wim Taymans */
|
|
/* SPDX-License-Identifier: MIT */
|
|
|
|
#ifndef ACP_H
|
|
#define ACP_H
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#else
|
|
#include <stdbool.h>
|
|
#endif
|
|
|
|
#include <stdio.h>
|
|
#include <stdarg.h>
|
|
#include <stdint.h>
|
|
#include <poll.h>
|
|
#include <string.h>
|
|
|
|
#ifdef __GNUC__
|
|
#define ACP_PRINTF_FUNC(fmt, arg1) __attribute__((format(printf, fmt, arg1)))
|
|
#else
|
|
#define ACP_PRINTF_FUNC(fmt, arg1)
|
|
#endif
|
|
|
|
#define ACP_INVALID_INDEX ((uint32_t)-1)
|
|
#define ACP_MAX_CHANNELS 64
|
|
|
|
struct acp_dict_item {
|
|
const char *key;
|
|
const char *value;
|
|
};
|
|
#define ACP_DICT_ITEM_INIT(key,value) ((struct acp_dict_item) { (key), (value) })
|
|
|
|
struct acp_dict {
|
|
uint32_t flags;
|
|
uint32_t n_items;
|
|
const struct acp_dict_item *items;
|
|
};
|
|
|
|
enum acp_channel {
|
|
ACP_CHANNEL_UNKNOWN, /**< unspecified */
|
|
ACP_CHANNEL_NA, /**< N/A, silent */
|
|
|
|
ACP_CHANNEL_MONO, /**< mono stream */
|
|
|
|
ACP_CHANNEL_FL, /**< front left */
|
|
ACP_CHANNEL_FR, /**< front right */
|
|
ACP_CHANNEL_FC, /**< front center */
|
|
ACP_CHANNEL_LFE, /**< LFE */
|
|
ACP_CHANNEL_SL, /**< side left */
|
|
ACP_CHANNEL_SR, /**< side right */
|
|
ACP_CHANNEL_FLC, /**< front left center */
|
|
ACP_CHANNEL_FRC, /**< front right center */
|
|
ACP_CHANNEL_RC, /**< rear center */
|
|
ACP_CHANNEL_RL, /**< rear left */
|
|
ACP_CHANNEL_RR, /**< rear right */
|
|
ACP_CHANNEL_TC, /**< top center */
|
|
ACP_CHANNEL_TFL, /**< top front left */
|
|
ACP_CHANNEL_TFC, /**< top front center */
|
|
ACP_CHANNEL_TFR, /**< top front right */
|
|
ACP_CHANNEL_TRL, /**< top rear left */
|
|
ACP_CHANNEL_TRC, /**< top rear center */
|
|
ACP_CHANNEL_TRR, /**< top rear right */
|
|
ACP_CHANNEL_RLC, /**< rear left center */
|
|
ACP_CHANNEL_RRC, /**< rear right center */
|
|
ACP_CHANNEL_FLW, /**< front left wide */
|
|
ACP_CHANNEL_FRW, /**< front right wide */
|
|
ACP_CHANNEL_LFE2, /**< LFE 2 */
|
|
ACP_CHANNEL_FLH, /**< front left high */
|
|
ACP_CHANNEL_FCH, /**< front center high */
|
|
ACP_CHANNEL_FRH, /**< front right high */
|
|
ACP_CHANNEL_TFLC, /**< top front left center */
|
|
ACP_CHANNEL_TFRC, /**< top front right center */
|
|
ACP_CHANNEL_TSL, /**< top side left */
|
|
ACP_CHANNEL_TSR, /**< top side right */
|
|
ACP_CHANNEL_LLFE, /**< left LFE */
|
|
ACP_CHANNEL_RLFE, /**< right LFE */
|
|
ACP_CHANNEL_BC, /**< bottom center */
|
|
ACP_CHANNEL_BLC, /**< bottom left center */
|
|
ACP_CHANNEL_BRC, /**< bottom right center */
|
|
|
|
ACP_CHANNEL_START_Aux = 0x1000,
|
|
ACP_CHANNEL_LAST_Aux = 0x1fff,
|
|
|
|
ACP_CHANNEL_START_Custom = 0x10000,
|
|
};
|
|
|
|
char *acp_channel_str(char *buf, size_t len, enum acp_channel ch);
|
|
|
|
struct acp_format {
|
|
uint32_t flags;
|
|
uint32_t format_mask;
|
|
uint32_t rate_mask;
|
|
uint32_t channels;
|
|
uint32_t map[ACP_MAX_CHANNELS];
|
|
};
|
|
|
|
#define ACP_DICT_INIT(items,n_items) ((struct acp_dict) { 0, (n_items), (items) })
|
|
#define ACP_DICT_INIT_ARRAY(items) ((struct acp_dict) { 0, sizeof(items)/sizeof((items)[0]), (items) })
|
|
|
|
#define acp_dict_for_each(item, dict) \
|
|
for ((item) = (dict)->items; \
|
|
(item) < &(dict)->items[(dict)->n_items]; \
|
|
(item)++)
|
|
|
|
static inline const char *acp_dict_lookup(const struct acp_dict *dict, const char *key)
|
|
{
|
|
const struct acp_dict_item *it;
|
|
acp_dict_for_each(it, dict)
|
|
if (strcmp(key, it->key) == 0)
|
|
return it->value;
|
|
return NULL;
|
|
}
|
|
|
|
enum acp_direction {
|
|
ACP_DIRECTION_PLAYBACK = 1,
|
|
ACP_DIRECTION_CAPTURE = 2
|
|
};
|
|
|
|
const char *acp_direction_str(enum acp_direction direction);
|
|
|
|
enum acp_available {
|
|
ACP_AVAILABLE_UNKNOWN = 0,
|
|
ACP_AVAILABLE_NO = 1,
|
|
ACP_AVAILABLE_YES = 2
|
|
};
|
|
|
|
const char *acp_available_str(enum acp_available status);
|
|
|
|
#define ACP_KEY_PORT_TYPE "port.type" /**< a Port type, like "aux", "speaker", ... */
|
|
#define ACP_KEY_PORT_AVAILABILITY_GROUP "port.availability-group"
|
|
/**< An identifier for the group of ports that share their availability status with
|
|
* each other. This is meant especially for handling cases where one 3.5 mm connector
|
|
* is used for headphones, headsets and microphones, and the hardware can only tell
|
|
* that something was plugged in but not what exactly. In this situation the ports for
|
|
* all those devices share their availability status, and ACP can't tell which
|
|
* one is actually plugged in, and some application may ask the user what was plugged
|
|
* in. Such applications should get a list of all card ports and compare their
|
|
* `available_group` fields. Ports that have the same group are those that need
|
|
* input from the user to determine which device was plugged in. The application should
|
|
* then activate the user-chosen port.
|
|
*
|
|
* May be NULL, in which case the port is not part of any availability group (which is
|
|
* the same as having a group with only one member).
|
|
*
|
|
* The group identifier must be treated as an opaque identifier. The string may look
|
|
* like an ALSA control name, but applications must not assume any such relationship.
|
|
* The group naming scheme can change without a warning.
|
|
*/
|
|
#define ACP_KEY_IEC958_CODECS_DETECTED "iec958.codecs.detected"
|
|
/**< A list of IEC958 passthrough formats which have been auto-detected as being
|
|
* supported by a given node. This only serves as a hint, as the auto-detected
|
|
* values may be incorrect and/or might change, e.g. when external devices such
|
|
* as receivers are powered on or off.
|
|
*/
|
|
|
|
struct acp_device;
|
|
|
|
struct acp_card_events {
|
|
#define ACP_VERSION_CARD_EVENTS 0
|
|
uint32_t version;
|
|
|
|
void (*destroy) (void *data);
|
|
|
|
void (*props_changed) (void *data);
|
|
|
|
void (*profile_changed) (void *data, uint32_t old_index, uint32_t new_index);
|
|
|
|
void (*profile_available) (void *data, uint32_t index,
|
|
enum acp_available old, enum acp_available available);
|
|
|
|
void (*port_changed) (void *data, uint32_t old_index, uint32_t new_index);
|
|
|
|
void (*port_available) (void *data, uint32_t index,
|
|
enum acp_available old, enum acp_available available);
|
|
|
|
void (*volume_changed) (void *data, struct acp_device *dev);
|
|
void (*mute_changed) (void *data, struct acp_device *dev);
|
|
};
|
|
|
|
struct acp_port {
|
|
uint32_t index; /**< unique index for this port */
|
|
#define ACP_PORT_ACTIVE (1<<0)
|
|
#define ACP_PORT_SAVE (1<<1) /* if the port needs saving */
|
|
#define ACP_PORT_HIDDEN (1<<2)
|
|
uint32_t flags; /**< extra port flags */
|
|
|
|
const char *name; /**< Name of this port */
|
|
const char *description; /**< Description of this port */
|
|
uint32_t priority; /**< The higher this value is, the more useful this port is as a default. */
|
|
enum acp_direction direction;
|
|
enum acp_available available; /**< A flags (see #acp_port_available), indicating availability status of this port. */
|
|
struct acp_dict props; /**< extra port properties */
|
|
|
|
uint32_t n_profiles; /**< number of elements in profiles array */
|
|
struct acp_card_profile **profiles; /**< array of profiles for this port */
|
|
|
|
uint32_t n_devices; /**< number of elements in devices array */
|
|
struct acp_device **devices; /**< array of devices */
|
|
};
|
|
|
|
struct acp_device {
|
|
uint32_t index;
|
|
#define ACP_DEVICE_ACTIVE (1<<0)
|
|
#define ACP_DEVICE_HW_VOLUME (1<<1)
|
|
#define ACP_DEVICE_HW_MUTE (1<<2)
|
|
#define ACP_DEVICE_UCM_DEVICE (1<<3)
|
|
#define ACP_DEVICE_IEC958 (1<<4)
|
|
#define ACP_DEVICE_HIDDEN (1<<5)
|
|
uint32_t flags;
|
|
|
|
const char *name;
|
|
const char *description;
|
|
uint32_t priority;
|
|
enum acp_direction direction;
|
|
struct acp_dict props;
|
|
|
|
const char **device_strings;
|
|
struct acp_format format;
|
|
|
|
float base_volume;
|
|
float volume_step;
|
|
|
|
uint32_t n_ports;
|
|
struct acp_port **ports;
|
|
|
|
int64_t latency_ns;
|
|
uint32_t codecs[32];
|
|
uint32_t n_codecs;
|
|
};
|
|
|
|
struct acp_card_profile {
|
|
uint32_t index;
|
|
#define ACP_PROFILE_ACTIVE (1<<0)
|
|
#define ACP_PROFILE_OFF (1<<1) /* the Off profile */
|
|
#define ACP_PROFILE_SAVE (1<<2) /* if the profile needs saving */
|
|
#define ACP_PROFILE_PRO (1<<3) /* the Pro profile */
|
|
#define ACP_PROFILE_HIDDEN (1<<4) /* don't show the profile */
|
|
uint32_t flags;
|
|
|
|
const char *name;
|
|
const char *description;
|
|
uint32_t priority;
|
|
enum acp_available available;
|
|
struct acp_dict props;
|
|
|
|
uint32_t n_devices;
|
|
struct acp_device **devices;
|
|
};
|
|
|
|
struct acp_card {
|
|
uint32_t index;
|
|
uint32_t flags;
|
|
|
|
struct acp_dict props;
|
|
|
|
uint32_t n_profiles;
|
|
uint32_t active_profile_index;
|
|
struct acp_card_profile **profiles;
|
|
|
|
uint32_t n_devices;
|
|
struct acp_device **devices;
|
|
|
|
uint32_t n_ports;
|
|
struct acp_port **ports;
|
|
uint32_t preferred_input_port_index;
|
|
uint32_t preferred_output_port_index;
|
|
};
|
|
|
|
struct acp_card *acp_card_new(uint32_t index, const struct acp_dict *props);
|
|
|
|
void acp_card_add_listener(struct acp_card *card,
|
|
const struct acp_card_events *events, void *user_data);
|
|
|
|
void acp_card_destroy(struct acp_card *card);
|
|
|
|
int acp_card_poll_descriptors_count(struct acp_card *card);
|
|
int acp_card_poll_descriptors(struct acp_card *card, struct pollfd *pfds, unsigned int space);
|
|
int acp_card_poll_descriptors_revents(struct acp_card *card, struct pollfd *pfds,
|
|
unsigned int nfds, unsigned short *revents);
|
|
int acp_card_handle_events(struct acp_card *card);
|
|
|
|
uint32_t acp_card_find_best_profile_index(struct acp_card *card, const char *name);
|
|
int acp_card_set_profile(struct acp_card *card, uint32_t profile_index, uint32_t flags);
|
|
|
|
uint32_t acp_device_find_best_port_index(struct acp_device *dev, const char *name);
|
|
int acp_device_set_port(struct acp_device *dev, uint32_t port_index, uint32_t flags);
|
|
|
|
int acp_device_set_volume(struct acp_device *dev, const float *volume, uint32_t n_volume);
|
|
int acp_device_get_soft_volume(struct acp_device *dev, float *volume, uint32_t n_volume);
|
|
int acp_device_get_volume(struct acp_device *dev, float *volume, uint32_t n_volume);
|
|
int acp_device_set_mute(struct acp_device *dev, bool mute);
|
|
int acp_device_get_mute(struct acp_device *dev, bool *mute);
|
|
|
|
typedef void (*acp_log_func) (void *data,
|
|
int level, const char *file, int line, const char *func,
|
|
const char *fmt, va_list arg) ACP_PRINTF_FUNC(6,0);
|
|
|
|
void acp_set_log_func(acp_log_func, void *data);
|
|
void acp_set_log_level(int level);
|
|
|
|
void acp_iec958_codecs_to_json(const uint32_t *codecs, size_t n_codecs, char *buf, size_t maxsize);
|
|
size_t acp_iec958_codecs_from_json(const char *str, uint32_t *codecs, size_t max_codecs);
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#endif /* ACP_H */
|