2021-09-01 00:33:43 +03:00
|
|
|
/* Spa A2DP codec API
|
2018-01-11 10:23:37 +01:00
|
|
|
*
|
2021-09-01 00:33:43 +03:00
|
|
|
* Copyright © 2020 Wim Taymans
|
2018-01-11 10:23:37 +01:00
|
|
|
*
|
2021-09-01 00:33:43 +03:00
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
|
|
|
|
* copy of this software and associated documentation files (the "Software"),
|
|
|
|
|
* to deal in the Software without restriction, including without limitation
|
|
|
|
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
|
|
|
* and/or sell copies of the Software, and to permit persons to whom the
|
|
|
|
|
* Software is furnished to do so, subject to the following conditions:
|
2018-01-11 10:23:37 +01:00
|
|
|
*
|
2021-09-01 00:33:43 +03:00
|
|
|
* The above copyright notice and this permission notice (including the next
|
|
|
|
|
* paragraph) shall be included in all copies or substantial portions of the
|
|
|
|
|
* Software.
|
2018-01-11 10:23:37 +01:00
|
|
|
*
|
2021-09-01 00:33:43 +03:00
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
|
|
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
|
|
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
|
|
|
* DEALINGS IN THE SOFTWARE.
|
2018-01-11 10:23:37 +01:00
|
|
|
*/
|
2021-09-01 00:33:43 +03:00
|
|
|
#ifndef SPA_BLUEZ5_A2DP_CODECS_H_
|
|
|
|
|
#define SPA_BLUEZ5_A2DP_CODECS_H_
|
2018-01-11 10:23:37 +01:00
|
|
|
|
|
|
|
|
#include <stdint.h>
|
2020-10-19 12:12:21 +02:00
|
|
|
#include <stddef.h>
|
|
|
|
|
|
|
|
|
|
#include <spa/param/audio/format.h>
|
2021-03-21 02:38:26 +02:00
|
|
|
#include <spa/param/bluetooth/audio.h>
|
2021-09-01 00:33:43 +03:00
|
|
|
#include <spa/utils/names.h>
|
|
|
|
|
#include <spa/support/plugin.h>
|
2020-12-03 18:05:57 +01:00
|
|
|
#include <spa/pod/pod.h>
|
|
|
|
|
#include <spa/pod/builder.h>
|
2022-06-12 16:11:58 +03:00
|
|
|
#include <spa/support/log.h>
|
2018-01-11 10:23:37 +01:00
|
|
|
|
2021-09-01 00:33:43 +03:00
|
|
|
#include "a2dp-codec-caps.h"
|
2020-12-04 11:34:38 +01:00
|
|
|
|
2021-09-01 00:33:43 +03:00
|
|
|
/*
|
|
|
|
|
* The codec plugin SPA interface is private. The version should be incremented
|
|
|
|
|
* when any of the structs or semantics change.
|
2021-08-15 18:17:40 +03:00
|
|
|
*/
|
2018-01-11 10:23:37 +01:00
|
|
|
|
2022-06-15 17:24:41 +02:00
|
|
|
#define SPA_TYPE_INTERFACE_Bluez5CodecMedia SPA_TYPE_INFO_INTERFACE_BASE "Bluez5:Codec:Media:Private"
|
2018-01-11 10:23:37 +01:00
|
|
|
|
2022-06-15 17:24:41 +02:00
|
|
|
#define SPA_VERSION_BLUEZ5_CODEC_MEDIA 5
|
2018-01-11 10:23:37 +01:00
|
|
|
|
2021-09-01 00:33:43 +03:00
|
|
|
struct spa_bluez5_codec_a2dp {
|
|
|
|
|
struct spa_interface iface;
|
2022-06-15 17:24:41 +02:00
|
|
|
const struct media_codec * const *codecs; /**< NULL terminated array */
|
2021-09-01 00:33:43 +03:00
|
|
|
};
|
2020-10-19 18:12:47 +02:00
|
|
|
|
2022-06-15 17:24:41 +02:00
|
|
|
#define MEDIA_CODEC_FACTORY_NAME(basename) (SPA_NAME_API_CODEC_BLUEZ5_MEDIA "." basename)
|
2020-10-19 18:12:47 +02:00
|
|
|
|
2021-09-01 00:33:43 +03:00
|
|
|
#ifdef CODEC_PLUGIN
|
2022-06-15 17:24:41 +02:00
|
|
|
#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;
|
2021-08-20 01:04:25 +03:00
|
|
|
|
2022-06-15 17:24:41 +02:00
|
|
|
extern const struct media_codec * const * const codec_plugin_media_codecs;
|
2021-09-01 00:33:43 +03:00
|
|
|
extern const char *codec_plugin_factory_name;
|
2018-01-11 10:23:37 +01:00
|
|
|
#endif
|
|
|
|
|
|
2022-06-15 17:24:41 +02:00
|
|
|
#define MEDIA_CODEC_FLAG_SINK (1 << 0)
|
2021-08-15 18:17:40 +03:00
|
|
|
|
2021-05-03 15:40:13 +08:00
|
|
|
#define A2DP_CODEC_DEFAULT_RATE 48000
|
|
|
|
|
#define A2DP_CODEC_DEFAULT_CHANNELS 2
|
|
|
|
|
|
2022-04-30 20:54:32 +03:00
|
|
|
enum {
|
|
|
|
|
NEED_FLUSH_NO = 0,
|
|
|
|
|
NEED_FLUSH_ALL = 1,
|
|
|
|
|
NEED_FLUSH_FRAGMENT = 2,
|
|
|
|
|
};
|
|
|
|
|
|
2022-06-15 17:24:41 +02:00
|
|
|
struct media_codec_audio_info {
|
2021-05-03 15:40:13 +08:00
|
|
|
uint32_t rate;
|
|
|
|
|
uint32_t channels;
|
|
|
|
|
};
|
2018-01-23 16:23:39 +01:00
|
|
|
|
2022-06-15 17:24:41 +02:00
|
|
|
struct media_codec {
|
2021-03-21 02:38:26 +02:00
|
|
|
enum spa_bluetooth_audio_codec id;
|
2020-12-04 11:34:38 +01:00
|
|
|
uint8_t codec_id;
|
|
|
|
|
a2dp_vendor_codec_t vendor;
|
2020-10-19 12:12:21 +02:00
|
|
|
|
|
|
|
|
const char *name;
|
|
|
|
|
const char *description;
|
2021-08-15 20:12:51 +03:00
|
|
|
const char *endpoint_name; /**< Endpoint name. If NULL, same as name */
|
2021-01-07 17:28:49 +01:00
|
|
|
const struct spa_dict *info;
|
2020-10-19 12:12:21 +02:00
|
|
|
|
2021-01-31 09:01:40 +08:00
|
|
|
const size_t send_buf_size;
|
2020-12-19 08:21:40 +08:00
|
|
|
|
2022-06-15 17:24:41 +02:00
|
|
|
const struct media_codec *duplex_codec; /**< Codec for non-standard A2DP duplex channel */
|
2021-08-15 19:11:12 +03:00
|
|
|
|
2022-06-15 17:24:41 +02:00
|
|
|
struct spa_log *log;
|
|
|
|
|
|
|
|
|
|
int (*fill_caps) (const struct media_codec *codec, uint32_t flags,
|
2020-12-04 11:34:38 +01:00
|
|
|
uint8_t caps[A2DP_MAX_CAPS_SIZE]);
|
2022-06-15 17:24:41 +02:00
|
|
|
int (*select_config) (const struct media_codec *codec, uint32_t flags,
|
2020-12-04 11:34:38 +01:00
|
|
|
const void *caps, size_t caps_size,
|
2022-06-15 17:24:41 +02:00
|
|
|
const struct media_codec_audio_info *info,
|
2022-06-04 19:01:51 +03:00
|
|
|
const struct spa_dict *global_settings, uint8_t config[A2DP_MAX_CAPS_SIZE]);
|
2022-06-15 17:24:41 +02:00
|
|
|
int (*enum_config) (const struct media_codec *codec, uint32_t flags,
|
2020-12-04 11:34:38 +01:00
|
|
|
const void *caps, size_t caps_size, uint32_t id, uint32_t idx,
|
2020-12-03 18:05:57 +01:00
|
|
|
struct spa_pod_builder *builder, struct spa_pod **param);
|
2022-06-15 17:24:41 +02:00
|
|
|
int (*validate_config) (const struct media_codec *codec, uint32_t flags,
|
2021-01-07 17:28:49 +01:00
|
|
|
const void *caps, size_t caps_size,
|
|
|
|
|
struct spa_audio_info *info);
|
2020-10-19 12:12:21 +02:00
|
|
|
|
2021-01-25 19:57:45 +02:00
|
|
|
/** 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.
|
|
|
|
|
*/
|
2022-06-15 17:24:41 +02:00
|
|
|
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,
|
2022-06-04 19:01:51 +03:00
|
|
|
const struct spa_dict *global_settings);
|
2021-01-25 19:57:45 +02:00
|
|
|
|
2022-06-15 17:24:41 +02:00
|
|
|
void *(*init_props) (const struct media_codec *codec, uint32_t flags, const struct spa_dict *settings);
|
2021-03-20 14:53:18 +08:00
|
|
|
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);
|
|
|
|
|
|
2022-06-15 17:24:41 +02:00
|
|
|
void *(*init) (const struct media_codec *codec, uint32_t flags, void *config, size_t config_size,
|
2021-03-20 14:53:18 +08:00
|
|
|
const struct spa_audio_info *info, void *props, size_t mtu);
|
2020-10-19 12:12:21 +02:00
|
|
|
void (*deinit) (void *data);
|
|
|
|
|
|
2021-03-20 14:53:18 +08:00
|
|
|
int (*update_props) (void *data, void *props);
|
2021-01-07 17:39:39 +01:00
|
|
|
|
2020-10-19 12:12:21 +02:00
|
|
|
int (*get_block_size) (void *data);
|
|
|
|
|
|
2020-12-19 08:21:40 +08:00
|
|
|
int (*abr_process) (void *data, size_t unsent);
|
|
|
|
|
|
2020-10-19 12:12:21 +02:00
|
|
|
int (*start_encode) (void *data,
|
|
|
|
|
void *dst, size_t dst_size, uint16_t seqnum, uint32_t timestamp);
|
|
|
|
|
int (*encode) (void *data,
|
|
|
|
|
const void *src, size_t src_size,
|
|
|
|
|
void *dst, size_t dst_size,
|
2021-03-31 17:50:15 +08:00
|
|
|
size_t *dst_out, int *need_flush);
|
2020-10-19 12:12:21 +02:00
|
|
|
|
2020-12-02 17:01:22 +01:00
|
|
|
int (*start_decode) (void *data,
|
|
|
|
|
const void *src, size_t src_size, uint16_t *seqnum, uint32_t *timestamp);
|
2020-10-19 12:12:21 +02:00
|
|
|
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);
|
2022-06-12 16:11:58 +03:00
|
|
|
|
|
|
|
|
void (*set_log) (struct spa_log *global_log);
|
2020-10-19 12:12:21 +02:00
|
|
|
};
|
|
|
|
|
|
2022-06-15 17:24:41 +02:00
|
|
|
struct media_codec_config {
|
2021-05-03 15:40:13 +08:00
|
|
|
uint32_t config;
|
|
|
|
|
int value;
|
|
|
|
|
unsigned int priority;
|
|
|
|
|
};
|
|
|
|
|
|
2022-06-15 17:24:41 +02:00
|
|
|
int media_codec_select_config(const struct media_codec_config configs[], size_t n,
|
2021-05-03 15:40:13 +08:00
|
|
|
uint32_t cap, int preferred_value);
|
|
|
|
|
|
2022-06-15 17:24:41 +02:00
|
|
|
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,
|
2022-06-04 19:01:51 +03:00
|
|
|
const struct spa_dict *global_settings);
|
2021-01-24 21:54:43 +02:00
|
|
|
|
2018-01-11 10:23:37 +01:00
|
|
|
#endif
|