mirror of
https://gitlab.freedesktop.org/pipewire/pipewire.git
synced 2025-10-29 05:40:27 -04:00
Parse BAP settings in a single place, and simplify QoS customization a bit. Ensure the selected preset gets selected. For all the BAP codec settings, use device settings instead of global monitor ones.
232 lines
7.9 KiB
C
232 lines
7.9 KiB
C
/* Spa A2DP codec API */
|
|
/* SPDX-FileCopyrightText: Copyright © 2020 Wim Taymans */
|
|
/* SPDX-License-Identifier: MIT */
|
|
|
|
#ifndef SPA_BLUEZ5_A2DP_CODECS_H_
|
|
#define SPA_BLUEZ5_A2DP_CODECS_H_
|
|
|
|
#include <stdint.h>
|
|
#include <stddef.h>
|
|
|
|
#include <spa/param/audio/format.h>
|
|
#include <spa/param/bluetooth/audio.h>
|
|
#include <spa/utils/names.h>
|
|
#include <spa/support/plugin.h>
|
|
#include <spa/pod/pod.h>
|
|
#include <spa/pod/builder.h>
|
|
#include <spa/support/log.h>
|
|
|
|
#include "a2dp-codec-caps.h"
|
|
#include "bap-codec-caps.h"
|
|
|
|
/*
|
|
* The codec plugin SPA interface is private. The version should be incremented
|
|
* when any of the structs or semantics change.
|
|
*/
|
|
|
|
#define SPA_TYPE_INTERFACE_Bluez5CodecMedia SPA_TYPE_INFO_INTERFACE_BASE "Bluez5:Codec:Media:Private"
|
|
|
|
#define SPA_VERSION_BLUEZ5_CODEC_MEDIA 13
|
|
|
|
struct spa_bluez5_codec_a2dp {
|
|
struct spa_interface iface;
|
|
const struct media_codec * const *codecs; /**< NULL terminated array */
|
|
};
|
|
|
|
#define MEDIA_CODEC_FACTORY_NAME(basename) (SPA_NAME_API_CODEC_BLUEZ5_MEDIA "." basename)
|
|
|
|
#ifdef CODEC_PLUGIN
|
|
#define MEDIA_CODEC_EXPORT_DEF(basename,...) \
|
|
const char *codec_plugin_factory_name = MEDIA_CODEC_FACTORY_NAME(basename); \
|
|
static const struct media_codec * const codec_plugin_media_codec_list[] = { __VA_ARGS__, NULL }; \
|
|
const struct media_codec * const * const codec_plugin_media_codecs = codec_plugin_media_codec_list; \
|
|
SPA_LOG_TOPIC_DEFINE(codec_plugin_log_topic, "spa.bluez5.codecs." basename);
|
|
|
|
extern const struct media_codec * const * const codec_plugin_media_codecs;
|
|
extern const char *codec_plugin_factory_name;
|
|
extern struct spa_log_topic codec_plugin_log_topic;
|
|
#undef SPA_LOG_TOPIC_DEFAULT
|
|
#define SPA_LOG_TOPIC_DEFAULT &codec_plugin_log_topic
|
|
#endif
|
|
|
|
#define MEDIA_CODEC_FLAG_SINK (1 << 0)
|
|
|
|
#define A2DP_CODEC_DEFAULT_RATE 48000
|
|
#define A2DP_CODEC_DEFAULT_CHANNELS 2
|
|
|
|
enum {
|
|
NEED_FLUSH_NO = 0,
|
|
NEED_FLUSH_ALL = 1,
|
|
NEED_FLUSH_FRAGMENT = 2,
|
|
};
|
|
|
|
struct media_codec_audio_info {
|
|
uint32_t rate;
|
|
uint32_t channels;
|
|
};
|
|
|
|
struct media_codec {
|
|
enum spa_bluetooth_audio_codec id;
|
|
uint8_t codec_id;
|
|
a2dp_vendor_codec_t vendor;
|
|
|
|
bool bap;
|
|
bool asha;
|
|
|
|
const char *name;
|
|
const char *description;
|
|
const char *endpoint_name; /**< Endpoint name. If NULL, same as name */
|
|
const struct spa_dict *info;
|
|
|
|
const size_t send_buf_size;
|
|
|
|
const struct media_codec *duplex_codec; /**< Codec for non-standard A2DP duplex channel */
|
|
|
|
int (*get_bis_config)(const struct media_codec *codec, uint8_t *caps,
|
|
uint8_t *caps_size, struct spa_dict *settings,
|
|
struct bap_codec_qos *qos);
|
|
|
|
/** If fill_caps is NULL, no endpoint is registered (for sharing with another codec). */
|
|
int (*fill_caps) (const struct media_codec *codec, uint32_t flags,
|
|
const struct spa_dict *settings, uint8_t caps[A2DP_MAX_CAPS_SIZE]);
|
|
|
|
int (*select_config) (const struct media_codec *codec, uint32_t flags,
|
|
const void *caps, size_t caps_size,
|
|
const struct media_codec_audio_info *info,
|
|
const struct spa_dict *global_settings, uint8_t config[A2DP_MAX_CAPS_SIZE]);
|
|
int (*enum_config) (const struct media_codec *codec, uint32_t flags,
|
|
const void *caps, size_t caps_size, uint32_t id, uint32_t idx,
|
|
struct spa_pod_builder *builder, struct spa_pod **param);
|
|
int (*validate_config) (const struct media_codec *codec, uint32_t flags,
|
|
const void *caps, size_t caps_size,
|
|
struct spa_audio_info *info);
|
|
int (*get_qos)(const struct media_codec *codec,
|
|
const void *config, size_t config_size,
|
|
const struct bap_endpoint_qos *endpoint_qos,
|
|
struct bap_codec_qos *qos, const struct spa_dict *settings);
|
|
|
|
/** qsort comparison sorting caps in order of preference for the codec.
|
|
* Used in codec switching to select best remote endpoints.
|
|
* The caps handed in correspond to this codec_id, but are
|
|
* otherwise not checked beforehand.
|
|
*/
|
|
int (*caps_preference_cmp) (const struct media_codec *codec, uint32_t flags, const void *caps1, size_t caps1_size,
|
|
const void *caps2, size_t caps2_size, const struct media_codec_audio_info *info,
|
|
const struct spa_dict *global_settings);
|
|
|
|
void *(*init_props) (const struct media_codec *codec, uint32_t flags, const struct spa_dict *settings);
|
|
void (*clear_props) (void *);
|
|
int (*enum_props) (void *props, const struct spa_dict *settings, uint32_t id, uint32_t idx,
|
|
struct spa_pod_builder *builder, struct spa_pod **param);
|
|
int (*set_props) (void *props, const struct spa_pod *param);
|
|
|
|
void *(*init) (const struct media_codec *codec, uint32_t flags, void *config, size_t config_size,
|
|
const struct spa_audio_info *info, void *props, size_t mtu);
|
|
void (*deinit) (void *data);
|
|
|
|
int (*update_props) (void *data, void *props);
|
|
|
|
/** Number of bytes needed for encoding */
|
|
int (*get_block_size) (void *data);
|
|
|
|
/**
|
|
* Duration of the next packet in nanoseconds.
|
|
*
|
|
* For BAP this shall be constant and equal to the SDU interval.
|
|
*
|
|
* \param data Codec data from init()
|
|
* \return Duration in nanoseconds.
|
|
*/
|
|
uint64_t (*get_interval) (void *data);
|
|
|
|
int (*abr_process) (void *data, size_t unsent);
|
|
|
|
/**
|
|
* Start encoding new packet.
|
|
*
|
|
* \param data Codec data from init()
|
|
* \param timestamp Packet time stamp (in samples played)
|
|
* \return Size of packet header written to dst in bytes, or < 0 for error
|
|
*/
|
|
int (*start_encode) (void *data,
|
|
void *dst, size_t dst_size, uint16_t seqnum, uint32_t timestamp);
|
|
|
|
/**
|
|
* Consume data from input buffer, encode to output buffer.
|
|
*
|
|
* \param data Codec data from init()
|
|
* \param src Source data. NULL if encoding packet fragment.
|
|
* \param dst Output buffer position. The memory region passed to the
|
|
* previous start_encode() is still valid, and this position is inside
|
|
* that region. The caller does not modify the contents of the buffer.
|
|
* \param dst_size Remaining buffer space after dst
|
|
* \param dst_out Bytes written to dst
|
|
* \param need_flush
|
|
* - NEED_FLUSH_NO: don't flush this packet,
|
|
* - NEED_FLUSH_ALL: flush this packet,
|
|
* - NEED_FLUSH_FRAGMENT: flush packet fragment. The next start_encode()
|
|
* and encode() are expected to produce more fragments or the final
|
|
* fragment with NEED_FLUSH_ALL, without consuming source data.
|
|
* The fragment start_encode() is called with the same output buffer
|
|
* as previous. The fragment encode() will be called with NULL src.
|
|
* No new source data will be fed in before NEED_FLUSH_ALL.
|
|
* \return Number of bytes consumed from src, or < 0 for error
|
|
*/
|
|
int (*encode) (void *data,
|
|
const void *src, size_t src_size,
|
|
void *dst, size_t dst_size,
|
|
size_t *dst_out, int *need_flush);
|
|
|
|
/**
|
|
* Start decoding received packet.
|
|
*
|
|
* \return Number of bytes consumed from source data, or < 0 for error
|
|
*/
|
|
int (*start_decode) (void *data,
|
|
const void *src, size_t src_size, uint16_t *seqnum, uint32_t *timestamp);
|
|
|
|
/**
|
|
* Decode received packet data.
|
|
*
|
|
* \param dst_out Number of bytes output to dst
|
|
* \return Number of bytes consumed from src, or < 0 for error
|
|
*/
|
|
int (*decode) (void *data,
|
|
const void *src, size_t src_size,
|
|
void *dst, size_t dst_size,
|
|
size_t *dst_out);
|
|
|
|
int (*reduce_bitpool) (void *data);
|
|
int (*increase_bitpool) (void *data);
|
|
|
|
void (*set_log) (struct spa_log *global_log);
|
|
|
|
/**
|
|
* Get codec internal delays, in samples at input/output rates.
|
|
*
|
|
* The delay does not include the duration of the PCM input/output
|
|
* audio data, but is that internal to the codec.
|
|
*
|
|
* \param[out] encoder Encoder delay in samples, or NULL
|
|
* \param[out] decoder Decoder delay in samples, or NULL
|
|
*/
|
|
void (*get_delay) (void *data, uint32_t *encoder, uint32_t *decoder);
|
|
};
|
|
|
|
struct media_codec_config {
|
|
uint32_t config;
|
|
int value;
|
|
unsigned int priority;
|
|
};
|
|
|
|
int media_codec_select_config(const struct media_codec_config configs[], size_t n,
|
|
uint32_t cap, int preferred_value);
|
|
|
|
int media_codec_get_config(const struct media_codec_config configs[], size_t n,
|
|
uint32_t conf);
|
|
|
|
bool media_codec_check_caps(const struct media_codec *codec, unsigned int codec_id,
|
|
const void *caps, size_t caps_size, const struct media_codec_audio_info *info,
|
|
const struct spa_dict *global_settings);
|
|
|
|
#endif
|