/* Media proxy */ /* * Copyright (c) 2021 Nordic Semiconductor ASA * * SPDX-License-Identifier: Apache-2.0 */ #include #include #include #include #include #include #include #include #include #include #include #include "media_proxy_internal.h" #include "mcs_internal.h" LOG_MODULE_REGISTER(bt_media_proxy, CONFIG_MCTL_LOG_LEVEL); /* Media player */ struct media_player { struct media_proxy_pl_calls *calls; struct bt_conn *conn; /* TODO: Treat local and remote player differently */ bool registered; }; /* Synchronous controller - controller using the synchronous API */ struct scontroller { struct media_proxy_sctrl_cbs *cbs; }; /* Asynchronous controller - controller using the asynchronous API */ struct controller { struct media_proxy_ctrl_cbs *cbs; }; /* Media proxy */ struct mprx { #if defined(CONFIG_MCTL_LOCAL_PLAYER_REMOTE_CONTROL) /* Controller using the synchronous interface - e.g. remote controller via * the media control service */ struct scontroller sctrlr; #endif /* CONFIG_MCTL_LOCAL_PLAYER_REMOTE_CONTROL */ #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) /* Controller using the callback interface - i.e. upper layer */ struct controller ctrlr; #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_LOCAL_PLAYER_CONTROL) /* The local media player */ struct media_player local_player; #endif /* CONFIG_MCTL_LOCAL_PLAYER_CONTROL */ /* Control of remote players is currently only supported over Bluetooth, via MCC */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL) /* Remote media player - to have a player instance for the user */ struct media_player remote_player; /* MCC callbacks */ struct bt_mcc_cb mcc_cbs; #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL */ }; static struct mprx mprx = { 0 }; #if defined(CONFIG_MCTL_LOCAL_PLAYER_REMOTE_CONTROL) /* Synchronous controller calls ***********************************************/ /* The synchronous controller is the media control service, representing a remote controller. * It only gets access to the local player, not remote players. * I.e., the calls from the synchronous controller are always routed to the local_player. */ int media_proxy_sctrl_register(struct media_proxy_sctrl_cbs *sctrl_cbs) { mprx.sctrlr.cbs = sctrl_cbs; return 0; }; const char *media_proxy_sctrl_get_player_name(void) { /* TODO: Add check for whether function pointer is non-NULL everywhere */ return mprx.local_player.calls->get_player_name(); } #ifdef CONFIG_BT_MPL_OBJECTS uint64_t media_proxy_sctrl_get_icon_id(void) { return mprx.local_player.calls->get_icon_id(); } #endif /* CONFIG_BT_MPL_OBJECTS */ const char *media_proxy_sctrl_get_icon_url(void) { return mprx.local_player.calls->get_icon_url(); } const char *media_proxy_sctrl_get_track_title(void) { return mprx.local_player.calls->get_track_title(); } int32_t media_proxy_sctrl_get_track_duration(void) { return mprx.local_player.calls->get_track_duration(); } int32_t media_proxy_sctrl_get_track_position(void) { return mprx.local_player.calls->get_track_position(); } void media_proxy_sctrl_set_track_position(int32_t position) { mprx.local_player.calls->set_track_position(position); } int8_t media_proxy_sctrl_get_playback_speed(void) { return mprx.local_player.calls->get_playback_speed(); } void media_proxy_sctrl_set_playback_speed(int8_t speed) { mprx.local_player.calls->set_playback_speed(speed); } int8_t media_proxy_sctrl_get_seeking_speed(void) { return mprx.local_player.calls->get_seeking_speed(); } #ifdef CONFIG_BT_MPL_OBJECTS uint64_t media_proxy_sctrl_get_track_segments_id(void) { return mprx.local_player.calls->get_track_segments_id(); } uint64_t media_proxy_sctrl_get_current_track_id(void) { return mprx.local_player.calls->get_current_track_id(); } void media_proxy_sctrl_set_current_track_id(uint64_t id) { mprx.local_player.calls->set_current_track_id(id); } uint64_t media_proxy_sctrl_get_next_track_id(void) { return mprx.local_player.calls->get_next_track_id(); } void media_proxy_sctrl_set_next_track_id(uint64_t id) { mprx.local_player.calls->set_next_track_id(id); } uint64_t media_proxy_sctrl_get_parent_group_id(void) { return mprx.local_player.calls->get_parent_group_id(); } uint64_t media_proxy_sctrl_get_current_group_id(void) { return mprx.local_player.calls->get_current_group_id(); } void media_proxy_sctrl_set_current_group_id(uint64_t id) { mprx.local_player.calls->set_current_group_id(id); } #endif /* CONFIG_BT_MPL_OBJECTS */ uint8_t media_proxy_sctrl_get_playing_order(void) { return mprx.local_player.calls->get_playing_order(); } void media_proxy_sctrl_set_playing_order(uint8_t order) { mprx.local_player.calls->set_playing_order(order); } uint16_t media_proxy_sctrl_get_playing_orders_supported(void) { return mprx.local_player.calls->get_playing_orders_supported(); } uint8_t media_proxy_sctrl_get_media_state(void) { return mprx.local_player.calls->get_media_state(); } void media_proxy_sctrl_send_command(const struct mpl_cmd *cmd) { mprx.local_player.calls->send_command(cmd); } uint32_t media_proxy_sctrl_get_commands_supported(void) { return mprx.local_player.calls->get_commands_supported(); } #ifdef CONFIG_BT_MPL_OBJECTS void media_proxy_sctrl_send_search(const struct mpl_search *search) { mprx.local_player.calls->send_search(search); } uint64_t media_proxy_sctrl_get_search_results_id(void) { return mprx.local_player.calls->get_search_results_id(); } void media_proxy_sctrl_search_results_id_cb(uint64_t id); #endif /* CONFIG_BT_MPL_OBJECTS */ uint8_t media_proxy_sctrl_get_content_ctrl_id(void) { return mprx.local_player.calls->get_content_ctrl_id(); } #endif /* CONFIG_MCTL_LOCAL_PLAYER_REMOTE_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL) /* Media control client callbacks *********************************************/ static void mcc_discover_mcs_cb(struct bt_conn *conn, int err) { if (err) { LOG_ERR("Discovery failed (%d)", err); } LOG_DBG("Discovered player"); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->discover_player) { mprx.ctrlr.cbs->discover_player(&mprx.remote_player, err); } else { LOG_DBG("No callback"); } } static void mcc_read_player_name_cb(struct bt_conn *conn, int err, const char *name) { /* Debug statements for at least a couple of the callbacks, to show flow */ LOG_DBG("MCC player name callback"); if (err) { LOG_ERR("Player name failed"); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->player_name_recv) { mprx.ctrlr.cbs->player_name_recv(&mprx.remote_player, err, name); } else { LOG_DBG("No callback"); } } #ifdef CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS static void mcc_read_icon_obj_id_cb(struct bt_conn *conn, int err, uint64_t id) { LOG_DBG("Icon Object ID callback"); if (err) { LOG_ERR("Icon Object ID read failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->icon_id_recv) { mprx.ctrlr.cbs->icon_id_recv(&mprx.remote_player, err, id); } else { LOG_DBG("No callback"); } } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS */ #if defined(CONFIG_BT_MCC_READ_MEDIA_PLAYER_ICON_URL) static void mcc_read_icon_url_cb(struct bt_conn *conn, int err, const char *url) { if (err) { LOG_ERR("Icon URL read failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->icon_url_recv) { mprx.ctrlr.cbs->icon_url_recv(&mprx.remote_player, err, url); } else { LOG_DBG("No callback"); } } #endif /* defined(CONFIG_BT_MCC_READ_MEDIA_PLAYER_ICON_URL) */ static void mcc_track_changed_ntf_cb(struct bt_conn *conn, int err) { if (err) { LOG_ERR("Track change notification failed (%d)", err); return; } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->track_changed_recv) { mprx.ctrlr.cbs->track_changed_recv(&mprx.remote_player, err); } else { LOG_DBG("No callback"); } } #if defined(CONFIG_BT_MCC_READ_TRACK_TITLE) static void mcc_read_track_title_cb(struct bt_conn *conn, int err, const char *title) { if (err) { LOG_ERR("Track title read failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->track_title_recv) { mprx.ctrlr.cbs->track_title_recv(&mprx.remote_player, err, title); } else { LOG_DBG("No callback"); } } #endif /* defined(CONFIG_BT_MCC_READ_TRACK_TITLE) */ #if defined(CONFIG_BT_MCC_READ_TRACK_DURATION) static void mcc_read_track_duration_cb(struct bt_conn *conn, int err, int32_t dur) { if (err) { LOG_ERR("Track duration read failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->track_duration_recv) { mprx.ctrlr.cbs->track_duration_recv(&mprx.remote_player, err, dur); } else { LOG_DBG("No callback"); } } #endif /* defined(CONFIG_BT_MCC_READ_TRACK_DURATION) */ #if defined(CONFIG_BT_MCC_READ_TRACK_POSITION) static void mcc_read_track_position_cb(struct bt_conn *conn, int err, int32_t pos) { if (err) { LOG_ERR("Track position read failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->track_position_recv) { mprx.ctrlr.cbs->track_position_recv(&mprx.remote_player, err, pos); } else { LOG_DBG("No callback"); } } #endif /* defined(CONFIG_BT_MCC_READ_TRACK_POSITION) */ #if defined(CONFIG_BT_MCC_SET_TRACK_POSITION) static void mcc_set_track_position_cb(struct bt_conn *conn, int err, int32_t pos) { if (err) { LOG_ERR("Track Position set failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->track_position_write) { mprx.ctrlr.cbs->track_position_write(&mprx.remote_player, err, pos); } else { LOG_DBG("No callback"); } } #endif /* defined(CONFIG_BT_MCC_SET_TRACK_POSITION) */ #if defined(CONFIG_BT_MCC_READ_PLAYBACK_SPEED) static void mcc_read_playback_speed_cb(struct bt_conn *conn, int err, int8_t speed) { if (err) { LOG_ERR("Playback speed read failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->playback_speed_recv) { mprx.ctrlr.cbs->playback_speed_recv(&mprx.remote_player, err, speed); } else { LOG_DBG("No callback"); } } #endif /* defined (CONFIG_BT_MCC_READ_PLAYBACK_SPEED) */ #if defined(CONFIG_BT_MCC_SET_PLAYBACK_SPEED) static void mcc_set_playback_speed_cb(struct bt_conn *conn, int err, int8_t speed) { if (err) { LOG_ERR("Playback speed set failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->playback_speed_write) { mprx.ctrlr.cbs->playback_speed_write(&mprx.remote_player, err, speed); } else { LOG_DBG("No callback"); } } #endif /* defined (CONFIG_BT_MCC_SET_PLAYBACK_SPEED) */ #if defined(CONFIG_BT_MCC_READ_SEEKING_SPEED) static void mcc_read_seeking_speed_cb(struct bt_conn *conn, int err, int8_t speed) { if (err) { LOG_ERR("Seeking speed read failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->seeking_speed_recv) { mprx.ctrlr.cbs->seeking_speed_recv(&mprx.remote_player, err, speed); } else { LOG_DBG("No callback"); } } #endif /* defined (CONFIG_BT_MCC_READ_SEEKING_SPEED) */ #ifdef CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS static void mcc_read_segments_obj_id_cb(struct bt_conn *conn, int err, uint64_t id) { if (err) { LOG_ERR("Track Segments Object ID read failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->track_segments_id_recv) { mprx.ctrlr.cbs->track_segments_id_recv(&mprx.remote_player, err, id); } else { LOG_DBG("No callback"); } } static void mcc_read_current_track_obj_id_cb(struct bt_conn *conn, int err, uint64_t id) { if (err) { LOG_ERR("Current Track Object ID read failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->current_track_id_recv) { mprx.ctrlr.cbs->current_track_id_recv(&mprx.remote_player, err, id); } else { LOG_DBG("No callback"); } } /* TODO: current track set callback - must be added to MCC first */ static void mcc_read_next_track_obj_id_cb(struct bt_conn *conn, int err, uint64_t id) { if (err) { LOG_ERR("Next Track Object ID read failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->next_track_id_recv) { mprx.ctrlr.cbs->next_track_id_recv(&mprx.remote_player, err, id); } else { LOG_DBG("No callback"); } } /* TODO: next track set callback - must be added to MCC first */ static void mcc_read_parent_group_obj_id_cb(struct bt_conn *conn, int err, uint64_t id) { if (err) { LOG_ERR("Parent Group Object ID read failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->parent_group_id_recv) { mprx.ctrlr.cbs->parent_group_id_recv(&mprx.remote_player, err, id); } else { LOG_DBG("No callback"); } } static void mcc_read_current_group_obj_id_cb(struct bt_conn *conn, int err, uint64_t id) { if (err) { LOG_ERR("Current Group Object ID read failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->current_group_id_recv) { mprx.ctrlr.cbs->current_group_id_recv(&mprx.remote_player, err, id); } else { LOG_DBG("No callback"); } } /* TODO: current group set callback - must be added to MCC first */ #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS */ #if defined(CONFIG_BT_MCC_READ_PLAYING_ORDER) static void mcc_read_playing_order_cb(struct bt_conn *conn, int err, uint8_t order) { if (err) { LOG_ERR("Playing order read failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->playing_order_recv) { mprx.ctrlr.cbs->playing_order_recv(&mprx.remote_player, err, order); } else { LOG_DBG("No callback"); } } #endif /* defined(CONFIG_BT_MCC_READ_PLAYING_ORDER) */ #if defined(CONFIG_BT_MCC_SET_PLAYING_ORDER) static void mcc_set_playing_order_cb(struct bt_conn *conn, int err, uint8_t order) { if (err) { LOG_ERR("Playing order set failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->playing_order_write) { mprx.ctrlr.cbs->playing_order_write(&mprx.remote_player, err, order); } else { LOG_DBG("No callback"); } } #endif /* defined(CONFIG_BT_MCC_SET_PLAYING_ORDER) */ #if defined(CONFIG_BT_MCC_READ_PLAYING_ORDER_SUPPORTED) static void mcc_read_playing_orders_supported_cb(struct bt_conn *conn, int err, uint16_t orders) { if (err) { LOG_ERR("Playing orders supported read failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->playing_orders_supported_recv) { mprx.ctrlr.cbs->playing_orders_supported_recv(&mprx.remote_player, err, orders); } else { LOG_DBG("No callback"); } } #endif /* defined(CONFIG_BT_MCC_READ_PLAYING_ORDER_SUPPORTED) */ #if defined(CONFIG_BT_MCC_READ_MEDIA_STATE) static void mcc_read_media_state_cb(struct bt_conn *conn, int err, uint8_t state) { if (err) { LOG_ERR("Media State read failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->media_state_recv) { mprx.ctrlr.cbs->media_state_recv(&mprx.remote_player, err, state); } else { LOG_DBG("No callback"); } } #endif /* defined(CONFIG_BT_MCC_READ_MEDIA_STATE) */ #if defined(CONFIG_BT_MCC_SET_MEDIA_CONTROL_POINT) static void mcc_send_cmd_cb(struct bt_conn *conn, int err, const struct mpl_cmd *cmd) { if (err) { LOG_ERR("Command send failed (%d) - opcode: %d, param: %d", err, cmd->opcode, cmd->param); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->command_send) { mprx.ctrlr.cbs->command_send(&mprx.remote_player, err, cmd); } else { LOG_DBG("No callback"); } } #endif /* defined(CONFIG_BT_MCC_SET_MEDIA_CONTROL_POINT) */ static void mcc_cmd_ntf_cb(struct bt_conn *conn, int err, const struct mpl_cmd_ntf *ntf) { if (err) { LOG_ERR("Command notification error (%d) - command opcode: %d, result: %d", err, ntf->requested_opcode, ntf->result_code); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->command_recv) { mprx.ctrlr.cbs->command_recv(&mprx.remote_player, err, ntf); } else { LOG_DBG("No callback"); } } #if defined(CONFIG_BT_MCC_READ_MEDIA_CONTROL_POINT_OPCODES_SUPPORTED) static void mcc_read_opcodes_supported_cb(struct bt_conn *conn, int err, uint32_t opcodes) { if (err) { LOG_ERR("Opcodes supported read failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->commands_supported_recv) { mprx.ctrlr.cbs->commands_supported_recv(&mprx.remote_player, err, opcodes); } else { LOG_DBG("No callback"); } } #endif /* defined(CONFIG_BT_MCC_READ_MEDIA_CONTROL_POINT_OPCODES_SUPPORTED) */ #ifdef CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS static void mcc_send_search_cb(struct bt_conn *conn, int err, const struct mpl_search *search) { if (err) { LOG_ERR("Search send failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->search_send) { mprx.ctrlr.cbs->search_send(&mprx.remote_player, err, search); } else { LOG_DBG("No callback"); } } static void mcc_search_ntf_cb(struct bt_conn *conn, int err, uint8_t result_code) { if (err) { LOG_ERR("Search notification error (%d), result code: %d", err, result_code); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->search_recv) { mprx.ctrlr.cbs->search_recv(&mprx.remote_player, err, result_code); } else { LOG_DBG("No callback"); } } static void mcc_read_search_results_obj_id_cb(struct bt_conn *conn, int err, uint64_t id) { if (err) { LOG_ERR("Search Results Object ID read failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->search_results_id_recv) { mprx.ctrlr.cbs->search_results_id_recv(&mprx.remote_player, err, id); } else { LOG_DBG("No callback"); } } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS */ #if defined(CONFIG_BT_MCC_READ_CONTENT_CONTROL_ID) static void mcc_read_content_control_id_cb(struct bt_conn *conn, int err, uint8_t ccid) { if (err) { LOG_ERR("Content Control ID read failed (%d)", err); } if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->content_ctrl_id_recv) { mprx.ctrlr.cbs->content_ctrl_id_recv(&mprx.remote_player, err, ccid); } else { LOG_DBG("No callback"); } } #endif /* defined(CONFIG_BT_MCC_READ_CONTENT_CONTROL_ID) */ #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL */ /* Asynchronous controller calls **********************************************/ #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) int media_proxy_ctrl_register(struct media_proxy_ctrl_cbs *ctrl_cbs) { CHECKIF(ctrl_cbs == NULL) { LOG_DBG("NULL callback pointer"); return -EINVAL; } mprx.ctrlr.cbs = ctrl_cbs; if (mprx.local_player.registered) { if (mprx.ctrlr.cbs->local_player_instance) { mprx.ctrlr.cbs->local_player_instance(&mprx.local_player, 0); } } /* TODO: Return error code if too many controllers registered */ return 0; }; #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #ifdef CONFIG_MCTL_REMOTE_PLAYER_CONTROL static void disconnected(struct bt_conn *conn, uint8_t reason) { if (mprx.remote_player.conn == conn) { bt_conn_unref(mprx.remote_player.conn); mprx.remote_player.conn = NULL; } } BT_CONN_CB_DEFINE(conn_callbacks) = { .disconnected = disconnected, }; int media_proxy_ctrl_discover_player(struct bt_conn *conn) { int err; CHECKIF(!conn) { LOG_DBG("NUll conn pointer"); return -EINVAL; } /* Initialize MCC */ mprx.mcc_cbs.discover_mcs = mcc_discover_mcs_cb; mprx.mcc_cbs.read_player_name = mcc_read_player_name_cb; #ifdef CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS mprx.mcc_cbs.read_icon_obj_id = mcc_read_icon_obj_id_cb; #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS */ #if defined(CONFIG_BT_MCC_READ_MEDIA_PLAYER_ICON_URL) mprx.mcc_cbs.read_icon_url = mcc_read_icon_url_cb; #endif /* defined(CONFIG_BT_MCC_READ_MEDIA_PLAYER_ICON_URL) */ mprx.mcc_cbs.track_changed_ntf = mcc_track_changed_ntf_cb; #if defined(CONFIG_BT_MCC_READ_TRACK_TITLE) mprx.mcc_cbs.read_track_title = mcc_read_track_title_cb; #endif /* defined(CONFIG_BT_MCC_READ_TRACK_TITLE) */ #if defined(CONFIG_BT_MCC_READ_TRACK_DURATION) mprx.mcc_cbs.read_track_duration = mcc_read_track_duration_cb; #endif /* defined(CONFIG_BT_MCC_READ_TRACK_DURATION) */ #if defined(CONFIG_BT_MCC_READ_TRACK_POSITION) mprx.mcc_cbs.read_track_position = mcc_read_track_position_cb; #endif /* defined(CONFIG_BT_MCC_READ_TRACK_POSITION) */ #if defined(CONFIG_BT_MCC_SET_TRACK_POSITION) mprx.mcc_cbs.set_track_position = mcc_set_track_position_cb; #endif /* defined(CONFIG_BT_MCC_SET_TRACK_POSITION) */ #if defined(CONFIG_BT_MCC_READ_PLAYBACK_SPEED) mprx.mcc_cbs.read_playback_speed = mcc_read_playback_speed_cb; #endif /* defined (CONFIG_BT_MCC_READ_PLAYBACK_SPEED) */ #if defined(CONFIG_BT_MCC_SET_PLAYBACK_SPEED) mprx.mcc_cbs.set_playback_speed = mcc_set_playback_speed_cb; #endif /* defined (CONFIG_BT_MCC_SET_PLAYBACK_SPEED) */ #if defined(CONFIG_BT_MCC_READ_SEEKING_SPEED) mprx.mcc_cbs.read_seeking_speed = mcc_read_seeking_speed_cb; #endif /* defined (CONFIG_BT_MCC_READ_SEEKING_SPEED) */ #ifdef CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS mprx.mcc_cbs.read_segments_obj_id = mcc_read_segments_obj_id_cb; mprx.mcc_cbs.read_current_track_obj_id = mcc_read_current_track_obj_id_cb; mprx.mcc_cbs.read_next_track_obj_id = mcc_read_next_track_obj_id_cb; mprx.mcc_cbs.read_parent_group_obj_id = mcc_read_parent_group_obj_id_cb; mprx.mcc_cbs.read_current_group_obj_id = mcc_read_current_group_obj_id_cb; #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS */ #if defined(CONFIG_BT_MCC_READ_PLAYING_ORDER) mprx.mcc_cbs.read_playing_order = mcc_read_playing_order_cb; #endif /* defined(CONFIG_BT_MCC_READ_PLAYING_ORDER) */ #if defined(CONFIG_BT_MCC_SET_PLAYING_ORDER) mprx.mcc_cbs.set_playing_order = mcc_set_playing_order_cb; #endif /* defined(CONFIG_BT_MCC_SET_PLAYING_ORDER) */ #if defined(CONFIG_BT_MCC_READ_PLAYING_ORDER_SUPPORTED) mprx.mcc_cbs.read_playing_orders_supported = mcc_read_playing_orders_supported_cb; #endif /* defined(CONFIG_BT_MCC_READ_PLAYING_ORDER_SUPPORTED) */ #if defined(CONFIG_BT_MCC_READ_MEDIA_STATE) mprx.mcc_cbs.read_media_state = mcc_read_media_state_cb; #endif /* defined(CONFIG_BT_MCC_READ_MEDIA_STATE) */ #if defined(CONFIG_BT_MCC_SET_MEDIA_CONTROL_POINT) mprx.mcc_cbs.send_cmd = mcc_send_cmd_cb; #endif /* defined(CONFIG_BT_MCC_SET_MEDIA_CONTROL_POINT) */ mprx.mcc_cbs.cmd_ntf = mcc_cmd_ntf_cb; #if defined(CONFIG_BT_MCC_READ_MEDIA_CONTROL_POINT_OPCODES_SUPPORTED) mprx.mcc_cbs.read_opcodes_supported = mcc_read_opcodes_supported_cb; #endif /* defined(CONFIG_BT_MCC_READ_MEDIA_CONTROL_POINT_OPCODES_SUPPORTED) */ #ifdef CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS mprx.mcc_cbs.send_search = mcc_send_search_cb; mprx.mcc_cbs.search_ntf = mcc_search_ntf_cb; mprx.mcc_cbs.read_search_results_obj_id = mcc_read_search_results_obj_id_cb; #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS */ #if defined(CONFIG_BT_MCC_READ_CONTENT_CONTROL_ID) mprx.mcc_cbs.read_content_control_id = mcc_read_content_control_id_cb; #endif /* defined(CONFIG_BT_MCC_READ_CONTENT_CONTROL_ID) */ err = bt_mcc_init(&mprx.mcc_cbs); if (err) { LOG_ERR("Failed to initialize MCC"); return err; } /* Start discovery of remote MCS, subscribe to notifications */ err = bt_mcc_discover_mcs(conn, 1); if (err) { LOG_ERR("Discovery failed"); return err; } if (mprx.remote_player.conn != NULL) { bt_conn_unref(mprx.remote_player.conn); } mprx.remote_player.conn = bt_conn_ref(conn); mprx.remote_player.registered = true; /* TODO: Do MCC init and "registration" at startup */ return 0; } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL */ #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) || defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL) int media_proxy_ctrl_get_player_name(struct media_player *player) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { LOG_DBG("Local player"); if (mprx.local_player.calls->get_player_name) { if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->player_name_recv) { const char *name = mprx.local_player.calls->get_player_name(); mprx.ctrlr.cbs->player_name_recv(&mprx.local_player, 0, name); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL) if (mprx.remote_player.registered && player == &mprx.remote_player) { LOG_DBG("Remote player"); return bt_mcc_read_player_name(mprx.remote_player.conn); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL */ return -EINVAL; } int media_proxy_ctrl_get_icon_id(struct media_player *player) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->get_icon_id) { if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->icon_id_recv) { const uint64_t id = mprx.local_player.calls->get_icon_id(); mprx.ctrlr.cbs->icon_id_recv(&mprx.local_player, 0, id); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS) if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_read_icon_obj_id(mprx.remote_player.conn); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS */ return -EINVAL; } int media_proxy_ctrl_get_icon_url(struct media_player *player) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->get_icon_url) { if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->icon_url_recv) { const char *url = mprx.local_player.calls->get_icon_url(); mprx.ctrlr.cbs->icon_url_recv(player, 0, url); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL) && defined(CONFIG_BT_MCC_READ_MEDIA_PLAYER_ICON_URL) if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_read_icon_url(mprx.remote_player.conn); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL && CONFIG_BT_MCC_READ_MEDIA_PLAYER_ICON_URL */ return -EINVAL; } int media_proxy_ctrl_get_track_title(struct media_player *player) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->get_track_title) { if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->track_title_recv) { const char *title = mprx.local_player.calls->get_track_title(); mprx.ctrlr.cbs->track_title_recv(player, 0, title); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL) && defined(CONFIG_BT_MCC_READ_TRACK_TITLE) if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_read_track_title(mprx.remote_player.conn); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL && CONFIG_BT_MCC_READ_TRACK_TITLE */ return -EINVAL; } int media_proxy_ctrl_get_track_duration(struct media_player *player) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->get_track_duration) { if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->track_duration_recv) { const int32_t duration = mprx.local_player.calls->get_track_duration(); mprx.ctrlr.cbs->track_duration_recv(player, 0, duration); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL) && defined(CONFIG_BT_MCC_READ_TRACK_DURATION) if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_read_track_duration(mprx.remote_player.conn); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL && CONFIG_BT_MCC_READ_TRACK_DURATION */ return -EINVAL; } int media_proxy_ctrl_get_track_position(struct media_player *player) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->get_track_position) { if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->track_position_recv) { const int32_t position = mprx.local_player.calls->get_track_position(); mprx.ctrlr.cbs->track_position_recv(player, 0, position); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL) && defined(CONFIG_BT_MCC_READ_TRACK_POSITION) if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_read_track_position(mprx.remote_player.conn); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL && CONFIG_BT_MCC_READ_TRACK_POSITION */ return -EINVAL; } int media_proxy_ctrl_set_track_position(struct media_player *player, int32_t position) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->set_track_position) { mprx.local_player.calls->set_track_position(position); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->track_position_write) { mprx.ctrlr.cbs->track_position_write(player, 0, position); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL) && CONFIG_BT_MCC_SET_TRACK_POSITION if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_set_track_position(mprx.remote_player.conn, position); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL && CONFIG_BT_MCC_SET_TRACK_POSITION */ return -EINVAL; } int media_proxy_ctrl_get_playback_speed(struct media_player *player) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->get_playback_speed) { if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->playback_speed_recv) { const int8_t speed = mprx.local_player.calls->get_playback_speed(); mprx.ctrlr.cbs->playback_speed_recv(player, 0, speed); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL) && defined(CONFIG_BT_MCC_READ_PLAYBACK_SPEED) if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_read_playback_speed(mprx.remote_player.conn); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL && CONFIG_BT_MCC_READ_PLAYBACK_SPEED */ return -EINVAL; } int media_proxy_ctrl_set_playback_speed(struct media_player *player, int8_t speed) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->set_playback_speed) { mprx.local_player.calls->set_playback_speed(speed); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->playback_speed_write) { mprx.ctrlr.cbs->playback_speed_write(player, 0, speed); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL) && defined(CONFIG_BT_MCC_SET_PLAYBACK_SPEED) if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_set_playback_speed(mprx.remote_player.conn, speed); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL && CONFIG_BT_MCC_SET_PLAYBACK_SPEED */ return -EINVAL; } int media_proxy_ctrl_get_seeking_speed(struct media_player *player) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->get_seeking_speed) { if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->seeking_speed_recv) { const int8_t speed = mprx.local_player.calls->get_seeking_speed(); mprx.ctrlr.cbs->seeking_speed_recv(player, 0, speed); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL) && defined(CONFIG_BT_MCC_READ_SEEKING_SPEED) if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_read_seeking_speed(mprx.remote_player.conn); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL && CONFIG_BT_MCC_READ_SEEKING_SPEED */ return -EINVAL; } int media_proxy_ctrl_get_track_segments_id(struct media_player *player) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->get_track_segments_id) { if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->track_segments_id_recv) { const uint64_t id = mprx.local_player.calls->get_track_segments_id(); mprx.ctrlr.cbs->track_segments_id_recv(player, 0, id); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS) if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_read_segments_obj_id(mprx.remote_player.conn); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS */ return -EINVAL; } int media_proxy_ctrl_get_current_track_id(struct media_player *player) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->get_current_track_id) { if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->current_track_id_recv) { const uint64_t id = mprx.local_player.calls->get_current_track_id(); mprx.ctrlr.cbs->current_track_id_recv(player, 0, id); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS) if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_read_current_track_obj_id(mprx.remote_player.conn); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS */ return -EINVAL; } int media_proxy_ctrl_set_current_track_id(struct media_player *player, uint64_t id) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) CHECKIF(!BT_MCS_VALID_OBJ_ID(id)) { LOG_DBG("Object ID 0x%016llx invalid", id); return -EINVAL; } if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->set_current_track_id) { mprx.local_player.calls->set_current_track_id(id); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->current_track_id_write) { mprx.ctrlr.cbs->current_track_id_write(player, 0, id); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS) if (mprx.remote_player.registered && player == &mprx.remote_player) { /* TODO: Uncomment when function is implemented */ /* return bt_mcc_set_current_track_obj_id(mprx.remote_player.conn, position); */ } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS */ return -EINVAL; } int media_proxy_ctrl_get_next_track_id(struct media_player *player) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->get_next_track_id) { if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->next_track_id_recv) { const uint64_t id = mprx.local_player.calls->get_next_track_id(); mprx.ctrlr.cbs->next_track_id_recv(player, 0, id); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS) if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_read_next_track_obj_id(mprx.remote_player.conn); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS */ return -EINVAL; } int media_proxy_ctrl_set_next_track_id(struct media_player *player, uint64_t id) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) CHECKIF(!BT_MCS_VALID_OBJ_ID(id)) { LOG_DBG("Object ID 0x%016llx invalid", id); return -EINVAL; } if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->set_next_track_id) { mprx.local_player.calls->set_next_track_id(id); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->next_track_id_write) { mprx.ctrlr.cbs->next_track_id_write(player, 0, id); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS) if (mprx.remote_player.registered && player == &mprx.remote_player) { /* TODO: Uncomment when function is implemented */ /* return bt_mcc_set_next_track_obj_id(mprx.remote_player.conn, position); */ } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS */ return -EINVAL; } int media_proxy_ctrl_get_parent_group_id(struct media_player *player) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->get_parent_group_id) { if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->parent_group_id_recv) { const uint64_t id = mprx.local_player.calls->get_parent_group_id(); mprx.ctrlr.cbs->parent_group_id_recv(player, 0, id); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS) if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_read_parent_group_obj_id(mprx.remote_player.conn); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS */ return -EINVAL; } int media_proxy_ctrl_get_current_group_id(struct media_player *player) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->get_current_group_id) { if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->current_group_id_recv) { const uint64_t id = mprx.local_player.calls->get_current_group_id(); mprx.ctrlr.cbs->current_group_id_recv(player, 0, id); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS) if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_read_current_group_obj_id(mprx.remote_player.conn); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS */ return -EINVAL; } int media_proxy_ctrl_set_current_group_id(struct media_player *player, uint64_t id) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) CHECKIF(!BT_MCS_VALID_OBJ_ID(id)) { LOG_DBG("Object ID 0x%016llx invalid", id); return -EINVAL; } if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->set_current_group_id) { mprx.local_player.calls->set_current_group_id(id); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->current_group_id_write) { mprx.ctrlr.cbs->current_group_id_write(player, 0, id); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS) if (mprx.remote_player.registered && player == &mprx.remote_player) { /* TODO: Uncomment when function is implemented */ /* return bt_mcc_set_current_group_obj_id(mprx.remote_player.conn, position); */ } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS */ return -EINVAL; } int media_proxy_ctrl_get_playing_order(struct media_player *player) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->get_playing_order) { if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->playing_order_recv) { const uint8_t order = mprx.local_player.calls->get_playing_order(); mprx.ctrlr.cbs->playing_order_recv(player, 0, order); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL) && defined(CONFIG_BT_MCC_READ_PLAYING_ORDER) if (mprx.remote_player.registered && player == &mprx.remote_player) { LOG_DBG("Remote player"); return bt_mcc_read_playing_order(mprx.remote_player.conn); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL && CONFIG_BT_MCC_READ_PLAYING_ORDER */ return -EINVAL; } int media_proxy_ctrl_set_playing_order(struct media_player *player, uint8_t order) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->set_playing_order) { mprx.local_player.calls->set_playing_order(order); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->playing_order_write) { mprx.ctrlr.cbs->playing_order_write(player, 0, order); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL) && defined(CONFIG_BT_MCC_SET_PLAYING_ORDER) if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_set_playing_order(mprx.remote_player.conn, order); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL && CONFIG_BT_MCC_SET_PLAYING_ORDER */ return -EINVAL; } int media_proxy_ctrl_get_playing_orders_supported(struct media_player *player) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->get_playing_orders_supported) { if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->playing_orders_supported_recv) { const uint16_t orders = mprx.local_player.calls->get_playing_orders_supported(); mprx.ctrlr.cbs->playing_orders_supported_recv(player, 0, orders); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL) && \ defined(CONFIG_BT_MCC_READ_PLAYING_ORDER_SUPPORTED) if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_read_playing_orders_supported(mprx.remote_player.conn); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL && CONFIG_BT_MCC_READ_PLAYING_ORDER_SUPPORTED */ return -EINVAL; } int media_proxy_ctrl_get_media_state(struct media_player *player) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->get_media_state) { if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->media_state_recv) { const uint8_t state = mprx.local_player.calls->get_media_state(); mprx.ctrlr.cbs->media_state_recv(player, 0, state); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL) && defined(CONFIG_BT_MCC_READ_MEDIA_STATE) if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_read_media_state(mprx.remote_player.conn); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL && CONFIG_BT_MCC_READ_MEDIA_STATE */ return -EINVAL; } int media_proxy_ctrl_send_command(struct media_player *player, const struct mpl_cmd *cmd) { CHECKIF(player == NULL || cmd == NULL) { LOG_DBG("NULL pointer"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->send_command) { mprx.local_player.calls->send_command(cmd); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->command_send) { mprx.ctrlr.cbs->command_send(player, 0, cmd); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL) && defined(CONFIG_BT_MCC_SET_MEDIA_CONTROL_POINT) if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_send_cmd(mprx.remote_player.conn, cmd); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL && CONFIG_BT_MCC_SET_MEDIA_CONTROL_POINT */ return -EINVAL; } int media_proxy_ctrl_get_commands_supported(struct media_player *player) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->get_commands_supported) { if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->commands_supported_recv) { const uint32_t opcodes = mprx.local_player.calls->get_commands_supported(); mprx.ctrlr.cbs->commands_supported_recv(player, 0, opcodes); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL) && \ defined(CONFIG_BT_MCC_READ_MEDIA_CONTROL_POINT_OPCODES_SUPPORTED) if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_read_opcodes_supported(mprx.remote_player.conn); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL && * CONFIG_BT_MCC_READ_MEDIA_CONTROL_POINT_OPCODES_SUPPORTED */ return -EINVAL; } int media_proxy_ctrl_send_search(struct media_player *player, const struct mpl_search *search) { CHECKIF(player == NULL || search == NULL) { LOG_DBG("NULL pointer"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->send_search) { mprx.local_player.calls->send_search(search); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->search_send) { mprx.ctrlr.cbs->search_send(player, 0, search); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS) if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_send_search(mprx.remote_player.conn, search); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS */ return -EINVAL; } int media_proxy_ctrl_get_search_results_id(struct media_player *player) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->get_search_results_id) { if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->search_results_id_recv) { const uint64_t id = mprx.local_player.calls->get_search_results_id(); mprx.ctrlr.cbs->search_results_id_recv(player, 0, id); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS) if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_read_search_results_obj_id(mprx.remote_player.conn); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL_OBJECTS */ return -EINVAL; } uint8_t media_proxy_ctrl_get_content_ctrl_id(struct media_player *player) { CHECKIF(player == NULL) { LOG_DBG("player is NULL"); return -EINVAL; } #if defined(CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL) if (mprx.local_player.registered && player == &mprx.local_player) { if (mprx.local_player.calls->get_content_ctrl_id) { if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->content_ctrl_id_recv) { const uint8_t ccid = mprx.local_player.calls->get_content_ctrl_id(); mprx.ctrlr.cbs->content_ctrl_id_recv(player, 0, ccid); } else { LOG_DBG("No callback"); } return 0; } LOG_DBG("No call"); return -EOPNOTSUPP; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL */ #if defined(CONFIG_MCTL_REMOTE_PLAYER_CONTROL) && defined(CONFIG_BT_MCC_READ_CONTENT_CONTROL_ID) if (mprx.remote_player.registered && player == &mprx.remote_player) { return bt_mcc_read_content_control_id(mprx.remote_player.conn); } #endif /* CONFIG_MCTL_REMOTE_PLAYER_CONTROL && CONFIG_BT_MCC_READ_CONTENT_CONTROL_ID */ return -EINVAL; } #endif /* CONFIG_MCTL_LOCAL_PLAYER_LOCAL_CONTROL || CONFIG_MCTL_REMOTE_PLAYER_CONTROL */ #if defined(CONFIG_MCTL_LOCAL_PLAYER_CONTROL) /* Player calls *******************************************/ static bool pl_calls_is_valid(const struct media_proxy_pl_calls *pl_calls) { if (pl_calls == NULL) { LOG_DBG("pl_calls is NULL"); return false; } if (pl_calls->get_player_name == NULL) { LOG_DBG("get_player_name is NULL"); return false; } #ifdef CONFIG_BT_MPL_OBJECTS if (pl_calls->get_icon_id == NULL) { LOG_DBG("get_icon_id is NULL"); return false; } #endif /* CONFIG_BT_MPL_OBJECTS */ if (pl_calls->get_icon_url == NULL) { LOG_DBG("get_icon_url is NULL"); return false; } if (pl_calls->get_track_title == NULL) { LOG_DBG("get_track_title is NULL"); return false; } if (pl_calls->get_track_duration == NULL) { LOG_DBG("get_track_duration is NULL"); return false; } if (pl_calls->get_track_position == NULL) { LOG_DBG("get_track_position is NULL"); return false; } if (pl_calls->set_track_position == NULL) { LOG_DBG("set_track_position is NULL"); return false; } if (pl_calls->get_playback_speed == NULL) { LOG_DBG("get_playback_speed is NULL"); return false; } if (pl_calls->set_playback_speed == NULL) { LOG_DBG("set_playback_speed is NULL"); return false; } if (pl_calls->get_seeking_speed == NULL) { LOG_DBG("get_seeking_speed is NULL"); return false; } #ifdef CONFIG_BT_MPL_OBJECTS if (pl_calls->get_track_segments_id == NULL) { LOG_DBG("get_track_segments_id is NULL"); return false; } if (pl_calls->get_current_track_id == NULL) { LOG_DBG("get_current_track_id is NULL"); return false; } if (pl_calls->set_current_track_id == NULL) { LOG_DBG("set_current_track_id is NULL"); return false; } if (pl_calls->get_next_track_id == NULL) { LOG_DBG("get_next_track_id is NULL"); return false; } if (pl_calls->set_next_track_id == NULL) { LOG_DBG("set_next_track_id is NULL"); return false; } if (pl_calls->get_parent_group_id == NULL) { LOG_DBG("get_parent_group_id is NULL"); return false; } if (pl_calls->get_current_group_id == NULL) { LOG_DBG("get_current_group_id is NULL"); return false; } if (pl_calls->set_current_group_id == NULL) { LOG_DBG("set_current_group_id is NULL"); return false; } #endif /* CONFIG_BT_MPL_OBJECTS */ if (pl_calls->get_playing_order == NULL) { LOG_DBG("get_playing_order is NULL"); return false; } if (pl_calls->set_playing_order == NULL) { LOG_DBG("set_playing_order is NULL"); return false; } if (pl_calls->get_playing_orders_supported == NULL) { LOG_DBG("get_playing_orders_supported is NULL"); return false; } if (pl_calls->get_media_state == NULL) { LOG_DBG("get_media_state is NULL"); return false; } if (pl_calls->send_command == NULL) { LOG_DBG("send_command is NULL"); return false; } if (pl_calls->get_commands_supported == NULL) { LOG_DBG("get_commands_supported is NULL"); return false; } #ifdef CONFIG_BT_MPL_OBJECTS if (pl_calls->send_search == NULL) { LOG_DBG("send_search is NULL"); return false; } if (pl_calls->get_search_results_id == NULL) { LOG_DBG("get_search_results_id is NULL"); return false; } #endif /* CONFIG_BT_MPL_OBJECTS */ if (pl_calls->get_content_ctrl_id == NULL) { LOG_DBG("get_content_ctrl_id is NULL"); return false; } return true; } int media_proxy_pl_register(struct media_proxy_pl_calls *pl_calls) { CHECKIF(!pl_calls_is_valid(pl_calls)) { return -EINVAL; } if (mprx.local_player.registered) { LOG_DBG("Player already registered"); return -EALREADY; } mprx.local_player.calls = pl_calls; mprx.local_player.registered = true; if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->local_player_instance) { mprx.ctrlr.cbs->local_player_instance(&mprx.local_player, 0); } return 0; }; /* Player callbacks ********************************/ /* All callbacks here must come from the local player - mprx.local_player */ void media_proxy_pl_name_cb(const char *name) { mprx.sctrlr.cbs->player_name(name); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->player_name_recv) { mprx.ctrlr.cbs->player_name_recv(&mprx.local_player, 0, name); } else { LOG_DBG("No ctrlr player name callback"); } } void media_proxy_pl_icon_url_cb(const char *url) { mprx.sctrlr.cbs->icon_url(url); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->player_name_recv) { mprx.ctrlr.cbs->icon_url_recv(&mprx.local_player, 0, url); } else { LOG_DBG("No ctrlr player icon URL callback"); } } void media_proxy_pl_track_changed_cb(void) { mprx.sctrlr.cbs->track_changed(); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->track_changed_recv) { mprx.ctrlr.cbs->track_changed_recv(&mprx.local_player, 0); } else { LOG_DBG("No ctrlr track changed callback"); } } void media_proxy_pl_track_title_cb(char *title) { mprx.sctrlr.cbs->track_title(title); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->track_title_recv) { mprx.ctrlr.cbs->track_title_recv(&mprx.local_player, 0, title); } else { LOG_DBG("No ctrlr track title callback"); } } void media_proxy_pl_track_duration_cb(int32_t duration) { mprx.sctrlr.cbs->track_duration(duration); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->track_duration_recv) { mprx.ctrlr.cbs->track_duration_recv(&mprx.local_player, 0, duration); } else { LOG_DBG("No ctrlr track duration callback"); } } void media_proxy_pl_track_position_cb(int32_t position) { mprx.sctrlr.cbs->track_position(position); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->track_position_recv) { mprx.ctrlr.cbs->track_position_recv(&mprx.local_player, 0, position); } else { LOG_DBG("No ctrlr track position callback"); } } void media_proxy_pl_playback_speed_cb(int8_t speed) { mprx.sctrlr.cbs->playback_speed(speed); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->playback_speed_recv) { mprx.ctrlr.cbs->playback_speed_recv(&mprx.local_player, 0, speed); } else { LOG_DBG("No ctrlr playback speed callback"); } } void media_proxy_pl_seeking_speed_cb(int8_t speed) { mprx.sctrlr.cbs->seeking_speed(speed); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->seeking_speed_recv) { mprx.ctrlr.cbs->seeking_speed_recv(&mprx.local_player, 0, speed); } else { LOG_DBG("No ctrlr seeking speed callback"); } } #ifdef CONFIG_BT_MPL_OBJECTS void media_proxy_pl_current_track_id_cb(uint64_t id) { mprx.sctrlr.cbs->current_track_id(id); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->current_track_id_recv) { mprx.ctrlr.cbs->current_track_id_recv(&mprx.local_player, 0, id); } else { LOG_DBG("No ctrlr current track id callback"); } } void media_proxy_pl_next_track_id_cb(uint64_t id) { mprx.sctrlr.cbs->next_track_id(id); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->next_track_id_recv) { mprx.ctrlr.cbs->next_track_id_recv(&mprx.local_player, 0, id); } else { LOG_DBG("No ctrlr next track id callback"); } } void media_proxy_pl_parent_group_id_cb(uint64_t id) { mprx.sctrlr.cbs->parent_group_id(id); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->parent_group_id_recv) { mprx.ctrlr.cbs->parent_group_id_recv(&mprx.local_player, 0, id); } else { LOG_DBG("No ctrlr parent group id callback"); } } void media_proxy_pl_current_group_id_cb(uint64_t id) { mprx.sctrlr.cbs->current_group_id(id); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->current_group_id_recv) { mprx.ctrlr.cbs->current_group_id_recv(&mprx.local_player, 0, id); } else { LOG_DBG("No ctrlr current group id callback"); } } #endif /* CONFIG_BT_MPL_OBJECTS */ void media_proxy_pl_playing_order_cb(uint8_t order) { mprx.sctrlr.cbs->playing_order(order); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->playing_order_recv) { mprx.ctrlr.cbs->playing_order_recv(&mprx.local_player, 0, order); } else { LOG_DBG("No ctrlr playing order callback"); } } void media_proxy_pl_media_state_cb(uint8_t state) { mprx.sctrlr.cbs->media_state(state); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->media_state_recv) { mprx.ctrlr.cbs->media_state_recv(&mprx.local_player, 0, state); } else { LOG_DBG("No ctrlr media state callback"); } } void media_proxy_pl_command_cb(const struct mpl_cmd_ntf *cmd_ntf) { CHECKIF(cmd_ntf == NULL) { LOG_WRN("cmd_ntf is NULL"); return; } mprx.sctrlr.cbs->command(cmd_ntf); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->command_recv) { mprx.ctrlr.cbs->command_recv(&mprx.local_player, 0, cmd_ntf); } else { LOG_DBG("No ctrlr command callback"); } } void media_proxy_pl_commands_supported_cb(uint32_t opcodes) { mprx.sctrlr.cbs->commands_supported(opcodes); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->commands_supported_recv) { mprx.ctrlr.cbs->commands_supported_recv(&mprx.local_player, 0, opcodes); } else { LOG_DBG("No ctrlr commands supported callback"); } } #ifdef CONFIG_BT_MPL_OBJECTS void media_proxy_pl_search_cb(uint8_t result_code) { mprx.sctrlr.cbs->search(result_code); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->search_recv) { mprx.ctrlr.cbs->search_recv(&mprx.local_player, 0, result_code); } else { LOG_DBG("No ctrlr search callback"); } } void media_proxy_pl_search_results_id_cb(uint64_t id) { mprx.sctrlr.cbs->search_results_id(id); if (mprx.ctrlr.cbs && mprx.ctrlr.cbs->search_results_id_recv) { mprx.ctrlr.cbs->search_results_id_recv(&mprx.local_player, 0, id); } else { LOG_DBG("No ctrlr search results id callback"); } } #endif /* CONFIG_BT_MPL_OBJECTS */ #endif /* CONFIG_MCTL_LOCAL_PLAYER_CONTROL */