Blame src/sdpd-service.c

Packit 34410b
/*
Packit 34410b
 *
Packit 34410b
 *  BlueZ - Bluetooth protocol stack for Linux
Packit 34410b
 *
Packit 34410b
 *  Copyright (C) 2001-2002  Nokia Corporation
Packit 34410b
 *  Copyright (C) 2002-2003  Maxim Krasnyansky <maxk@qualcomm.com>
Packit 34410b
 *  Copyright (C) 2002-2010  Marcel Holtmann <marcel@holtmann.org>
Packit 34410b
 *  Copyright (C) 2002-2003  Stephen Crane <steve.crane@rococosoft.com>
Packit 34410b
 *
Packit 34410b
 *
Packit 34410b
 *  This program is free software; you can redistribute it and/or modify
Packit 34410b
 *  it under the terms of the GNU General Public License as published by
Packit 34410b
 *  the Free Software Foundation; either version 2 of the License, or
Packit 34410b
 *  (at your option) any later version.
Packit 34410b
 *
Packit 34410b
 *  This program is distributed in the hope that it will be useful,
Packit 34410b
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 34410b
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 34410b
 *  GNU General Public License for more details.
Packit 34410b
 *
Packit 34410b
 *  You should have received a copy of the GNU General Public License
Packit 34410b
 *  along with this program; if not, write to the Free Software
Packit 34410b
 *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
Packit 34410b
 *
Packit 34410b
 */
Packit 34410b
Packit 34410b
#ifdef HAVE_CONFIG_H
Packit 34410b
#include <config.h>
Packit 34410b
#endif
Packit 34410b
Packit 34410b
#include <errno.h>
Packit 34410b
#include <stdlib.h>
Packit 34410b
#include <assert.h>
Packit 34410b
#include <sys/time.h>
Packit 34410b
#include <stdbool.h>
Packit 34410b
Packit 34410b
#include <glib.h>
Packit 34410b
Packit 34410b
#include "lib/bluetooth.h"
Packit 34410b
#include "lib/sdp.h"
Packit 34410b
#include "lib/sdp_lib.h"
Packit 34410b
Packit 34410b
#include "src/shared/util.h"
Packit 34410b
#include "sdpd.h"
Packit 34410b
#include "log.h"
Packit 34410b
Packit 34410b
#define MPSD_HFP_AG_A2DP_SRC_ANSWER_CALL_DURING_AUDIO		(1ULL << 0)
Packit 34410b
#define MPSD_HFP_HF_A2DP_SNK_ANSWER_CALL_DURING_AUDIO		(1ULL << 1)
Packit 34410b
#define MPSD_HFP_AG_A2DP_SRC_OUTGOING_CALL_DURING_AUDIO		(1ULL << 2)
Packit 34410b
#define MPSD_HFP_HF_A2DP_SNK_OUTGOING_CALL_DURING_AUDIO		(1ULL << 3)
Packit 34410b
#define MPSD_HFP_AG_A2DP_SRC_REJECT_CALL_DURING_AUDIO		(1ULL << 4)
Packit 34410b
#define MPSD_HFP_HF_A2DP_SNK_SRC_REJECT_CALL_DURING_AUDIO	(1ULL << 5)
Packit 34410b
#define MPSD_HFP_AG_A2DP_SRC_TERMINATE_CALL_DURING_AVP		(1ULL << 6)
Packit 34410b
#define MPSD_HFP_HF_A2DP_SNK_TERMINATE_CALL_DURING_AVP		(1ULL << 7)
Packit 34410b
#define MPSD_HFP_AG_A2DP_SRC_PRESS_PLAY_DURING_ACTIVE_CALL	(1ULL << 8)
Packit 34410b
#define MPSD_HFP_HF_A2DP_SNK_PRESS_PLAY_DURING_ACTIVE_CALL	(1ULL << 9)
Packit 34410b
#define MPSD_HFP_AG_A2DP_SRC_START_AUDIO_STREAM_AFTER_PLAY	(1ULL << 10)
Packit 34410b
#define MPSD_HFP_HF_A2DP_SNK_START_AUDIO_STREAM_AFTER_PLAY	(1ULL << 11)
Packit 34410b
#define MPSD_HFP_AG_A2DP_SRC_SUSPEND_AUDIO_STREAM_ON_PAUSE	(1ULL << 12)
Packit 34410b
#define MPSD_HFP_HF_A2DP_SNK_SUSPEND_AUDIO_STREAM_ON_PAUSE	(1ULL << 13)
Packit 34410b
#define MPSD_HFP_AG_DUN_GW_DATA_COMM_DURING_VOICE_CALL		(1ULL << 14)
Packit 34410b
#define MPSD_HFP_HF_DUN_DT_DATA_COMM_DURING_VOICE_CALL		(1ULL << 15)
Packit 34410b
#define MPSD_HFP_AG_DUN_GW_OUTGOING_CALL_DURING_DATA_COMM	(1ULL << 16)
Packit 34410b
#define MPSD_HFP_HF_DUN_DT_OUTGOING_CALL_DURING_DATA_COMM	(1ULL << 17)
Packit 34410b
#define MPSD_HFP_AG_DUN_GW_INCOMING_CALL_DURING_DATA_COMM	(1ULL << 18)
Packit 34410b
#define MPSD_HFP_HF_DUN_DT_INCOMING_CALL_DURING_DATA_COMM	(1ULL << 19)
Packit 34410b
#define MPSD_A2DP_SRC_DUN_GW_START_AUDIO_DURING_DATA_COMM	(1ULL << 20)
Packit 34410b
#define MPSD_A2DP_SNK_DUN_DT_START_AUDIO_DURING_DATA_COMM	(1ULL << 21)
Packit 34410b
#define MPSD_A2DP_SRC_DUN_GW_DATA_COMM_DURING_AUDIO_STREAM	(1ULL << 22)
Packit 34410b
#define MPSD_A2DP_SNK_DUN_DT_DATA_COMM_DURING_AUDIO_STREAM	(1ULL << 23)
Packit 34410b
#define MPSD_HFP_AG_DUN_GW_TERMINATE_CALL_DURING_DATA_COMM	(1ULL << 24)
Packit 34410b
#define MPSD_HFP_HF_DUN_DT_TERMINATE_CALL_DURING_DATA_COMM	(1ULL << 25)
Packit 34410b
#define MPSD_HFP_AG_PAN_NAP_DATA_COMM_DURING_VOICE_CALL		(1ULL << 26)
Packit 34410b
#define MPSD_HFP_HF_PAN_PANU_DATA_COMM_DURING_VOICE_CALL	(1ULL << 27)
Packit 34410b
#define MPSD_HFP_AG_PAN_NAP_OUTGOING_CALL_DURING_DATA_COMM	(1ULL << 28)
Packit 34410b
#define MPSD_HFP_HF_PAN_PANU_OUTGOING_CALL_DURING_DATA_COMM	(1ULL << 29)
Packit 34410b
#define MPSD_HFP_AG_PAN_NAP_INCOMING_CALL_DURING_DATA_COMM	(1ULL << 30)
Packit 34410b
#define MPSD_HFP_HF_PAN_PANU_INCOMING_CALL_DURING_DATA_COMM	(1ULL << 31)
Packit 34410b
#define MPSD_A2DP_SRC_PAN_NAP_START_AUDIO_DURING_DATA_COMM	(1ULL << 32)
Packit 34410b
#define MPSD_A2DP_SNK_PAN_PANU_START_AUDIO_DURING_DATA_COMM	(1ULL << 33)
Packit 34410b
#define MPSD_A2DP_SRC_PAN_NAP_DATA_COMM_DURING_AUDIO_STREAM	(1ULL << 34)
Packit 34410b
#define MPSD_A2DP_SNK_PAN_PANU_DATA_COMM_DURING_AUDIO_STREAM	(1ULL << 35)
Packit 34410b
#define MPSD_A2DP_SRC_PBAP_SRV_PB_DL_DURING_AUDIO_STREAM	(1ULL << 36)
Packit 34410b
#define MPSD_A2DP_SNK_PBAP_CLI_PB_DL_DURING_AUDIO_STREAM	(1ULL << 37)
Packit 34410b
Packit 34410b
#define MPMD_HFP_HF_A2DP_SNK_AVRCP_CT_ANSWER_CALL_DURING_AUDIO	(1ULL << 0)
Packit 34410b
#define MPMD_A2DP_SRC_AVRCP_TG_ANSWER_CALL_DURING_AUDIO		(1ULL << 1)
Packit 34410b
#define MPMD_HFP_HF_A2DP_SNK_AVRCP_CT_OUTGOING_CALL_DURING_AUDIO (1ULL << 2)
Packit 34410b
#define MPMD_A2DP_SRC_AVRCP_TG_OUTGOING_CALL_DURING_AUDIO	(1ULL << 3)
Packit 34410b
#define MPMD_HFP_HF_A2DP_SNK_AVRCP_CT_REJECT_CALL_DURING_AUDIO	(1ULL << 4)
Packit 34410b
#define MPMD_A2DP_SRC_AVRCP_TG_REJECT_CALL_DURING_AUDIO		(1ULL << 5)
Packit 34410b
#define MPMD_HFP_AG_CALL_TERMINATION_DURING_AVP			(1ULL << 6)
Packit 34410b
#define MPMD_HFP_HF_A2DP_SNK_AVRCP_CT_TERMINATION_DURING_AVP	(1ULL << 7)
Packit 34410b
#define MPMD_A2DP_SRC_AVRCP_TG_TERMINATION_DURING_AVP		(1ULL << 8)
Packit 34410b
#define MPMD_HFP_HF_A2DP_SNK_AVRCP_CT_PLAY_DURING_CALL		(1ULL << 9)
Packit 34410b
#define MPMD_A2DP_SRC_AVRCP_TG_PRESS_PLAY_DURING_CALL		(1ULL << 10)
Packit 34410b
#define MPMD_AVRCP_CT_NO_A2DP_SNK_START_AUDIO_AFTER_PLAY	(1ULL << 11)
Packit 34410b
#define MPMD_A2DP_SRC_AVRCP_TG_START_AUDIO_AFTER_PLAY		(1ULL << 12)
Packit 34410b
#define MPMD_AVRCP_CT_NO_A2DP_SNK_SUSPEND_AUDIO_AFTER_PAUSE	(1ULL << 13)
Packit 34410b
#define MPMD_A2DP_SRC_AVRCP_TG_SUSPEND_AUDIO_AFTER_PAUSE	(1ULL << 14)
Packit 34410b
#define MPMD_A2DP_SRC_AVRCP_TG_START_AUDIO_DURING_DATA_COMM	(1ULL << 15)
Packit 34410b
#define MPMD_A2DP_SNK_AVRCP_CT_DUN_DT_AUDIO_DURING_DATA_COMM	(1ULL << 16)
Packit 34410b
#define MPMD_A2DP_SRC_AVRCP_TG_START_DATA_DURING_AUDIO		(1ULL << 17)
Packit 34410b
#define MPMD_A2DP_SNK_AVRCP_CT_DUN_DT_START_DATA_DURING_AUDIO	(1ULL << 18)
Packit 34410b
Packit 34410b
/* Note: in spec dependency bit position starts from 1 (bit 0 unused?) */
Packit 34410b
#define MPS_DEPS_SNIFF_MODE_DURRING_STREAMING	(1ULL << 1)
Packit 34410b
#define MPS_DEPS_GAVDP_REQUIREMENTS		(1ULL << 2)
Packit 34410b
#define MPS_DEPS_DIS_CONNECTION_ORDER_BEHAVIOR	(1ULL << 3)
Packit 34410b
Packit 34410b
/*
Packit 34410b
 * default MPS features are all disabled, will be updated if relevant service
Packit 34410b
 * is (un)registered
Packit 34410b
 */
Packit 34410b
#define MPS_MPSD_DEFAULT_FEATURES 0
Packit 34410b
#define MPS_MPMD_DEFAULT_FEATURES 0
Packit 34410b
Packit 34410b
/*
Packit 34410b
 * Those defines bits for all features that depend on specific profile and role.
Packit 34410b
 * If profile is not supported then all those bits should not be set in record
Packit 34410b
 */
Packit 34410b
#define MPS_MPSD_HFP_AG (MPSD_HFP_AG_A2DP_SRC_ANSWER_CALL_DURING_AUDIO | \
Packit 34410b
			MPSD_HFP_AG_A2DP_SRC_OUTGOING_CALL_DURING_AUDIO | \
Packit 34410b
			MPSD_HFP_AG_A2DP_SRC_REJECT_CALL_DURING_AUDIO | \
Packit 34410b
			MPSD_HFP_AG_A2DP_SRC_TERMINATE_CALL_DURING_AVP | \
Packit 34410b
			MPSD_HFP_AG_A2DP_SRC_PRESS_PLAY_DURING_ACTIVE_CALL | \
Packit 34410b
			MPSD_HFP_AG_A2DP_SRC_START_AUDIO_STREAM_AFTER_PLAY | \
Packit 34410b
			MPSD_HFP_AG_DUN_GW_DATA_COMM_DURING_VOICE_CALL | \
Packit 34410b
			MPSD_HFP_AG_A2DP_SRC_SUSPEND_AUDIO_STREAM_ON_PAUSE | \
Packit 34410b
			MPSD_HFP_AG_DUN_GW_OUTGOING_CALL_DURING_DATA_COMM | \
Packit 34410b
			MPSD_HFP_AG_DUN_GW_INCOMING_CALL_DURING_DATA_COMM | \
Packit 34410b
			MPSD_HFP_AG_DUN_GW_TERMINATE_CALL_DURING_DATA_COMM | \
Packit 34410b
			MPSD_HFP_AG_PAN_NAP_DATA_COMM_DURING_VOICE_CALL | \
Packit 34410b
			MPSD_HFP_AG_PAN_NAP_OUTGOING_CALL_DURING_DATA_COMM | \
Packit 34410b
			MPSD_HFP_AG_PAN_NAP_INCOMING_CALL_DURING_DATA_COMM)
Packit 34410b
Packit 34410b
#define MPS_MPSD_HFP_HF (MPSD_HFP_HF_A2DP_SNK_ANSWER_CALL_DURING_AUDIO | \
Packit 34410b
			MPSD_HFP_HF_A2DP_SNK_OUTGOING_CALL_DURING_AUDIO | \
Packit 34410b
			MPSD_HFP_HF_A2DP_SNK_SRC_REJECT_CALL_DURING_AUDIO | \
Packit 34410b
			MPSD_HFP_HF_A2DP_SNK_TERMINATE_CALL_DURING_AVP | \
Packit 34410b
			MPSD_HFP_HF_A2DP_SNK_PRESS_PLAY_DURING_ACTIVE_CALL | \
Packit 34410b
			MPSD_HFP_HF_A2DP_SNK_START_AUDIO_STREAM_AFTER_PLAY | \
Packit 34410b
			MPSD_HFP_HF_A2DP_SNK_SUSPEND_AUDIO_STREAM_ON_PAUSE | \
Packit 34410b
			MPSD_HFP_HF_DUN_DT_DATA_COMM_DURING_VOICE_CALL | \
Packit 34410b
			MPSD_HFP_HF_DUN_DT_OUTGOING_CALL_DURING_DATA_COMM | \
Packit 34410b
			MPSD_HFP_HF_DUN_DT_INCOMING_CALL_DURING_DATA_COMM | \
Packit 34410b
			MPSD_HFP_HF_DUN_DT_TERMINATE_CALL_DURING_DATA_COMM | \
Packit 34410b
			MPSD_HFP_HF_PAN_PANU_DATA_COMM_DURING_VOICE_CALL | \
Packit 34410b
			MPSD_HFP_HF_PAN_PANU_OUTGOING_CALL_DURING_DATA_COMM | \
Packit 34410b
			MPSD_HFP_HF_PAN_PANU_INCOMING_CALL_DURING_DATA_COMM)
Packit 34410b
Packit 34410b
#define MPS_MPSD_A2DP_SRC (MPSD_HFP_AG_A2DP_SRC_ANSWER_CALL_DURING_AUDIO | \
Packit 34410b
			MPSD_HFP_AG_A2DP_SRC_OUTGOING_CALL_DURING_AUDIO | \
Packit 34410b
			MPSD_HFP_AG_A2DP_SRC_REJECT_CALL_DURING_AUDIO | \
Packit 34410b
			MPSD_HFP_AG_A2DP_SRC_TERMINATE_CALL_DURING_AVP | \
Packit 34410b
			MPSD_HFP_AG_A2DP_SRC_PRESS_PLAY_DURING_ACTIVE_CALL | \
Packit 34410b
			MPSD_HFP_AG_A2DP_SRC_START_AUDIO_STREAM_AFTER_PLAY | \
Packit 34410b
			MPSD_HFP_AG_A2DP_SRC_SUSPEND_AUDIO_STREAM_ON_PAUSE | \
Packit 34410b
			MPSD_A2DP_SRC_DUN_GW_START_AUDIO_DURING_DATA_COMM | \
Packit 34410b
			MPSD_A2DP_SRC_DUN_GW_DATA_COMM_DURING_AUDIO_STREAM | \
Packit 34410b
			MPSD_A2DP_SRC_PAN_NAP_START_AUDIO_DURING_DATA_COMM | \
Packit 34410b
			MPSD_A2DP_SRC_PAN_NAP_DATA_COMM_DURING_AUDIO_STREAM | \
Packit 34410b
			MPSD_A2DP_SRC_PBAP_SRV_PB_DL_DURING_AUDIO_STREAM)
Packit 34410b
Packit 34410b
#define MPS_MPSD_A2DP_SNK (MPSD_HFP_HF_A2DP_SNK_ANSWER_CALL_DURING_AUDIO | \
Packit 34410b
			MPSD_HFP_HF_A2DP_SNK_OUTGOING_CALL_DURING_AUDIO | \
Packit 34410b
			MPSD_HFP_HF_A2DP_SNK_SRC_REJECT_CALL_DURING_AUDIO | \
Packit 34410b
			MPSD_HFP_HF_A2DP_SNK_TERMINATE_CALL_DURING_AVP | \
Packit 34410b
			MPSD_HFP_HF_A2DP_SNK_PRESS_PLAY_DURING_ACTIVE_CALL | \
Packit 34410b
			MPSD_HFP_HF_A2DP_SNK_START_AUDIO_STREAM_AFTER_PLAY | \
Packit 34410b
			MPSD_HFP_HF_A2DP_SNK_SUSPEND_AUDIO_STREAM_ON_PAUSE | \
Packit 34410b
			MPSD_A2DP_SNK_DUN_DT_START_AUDIO_DURING_DATA_COMM | \
Packit 34410b
			MPSD_A2DP_SNK_DUN_DT_DATA_COMM_DURING_AUDIO_STREAM | \
Packit 34410b
			MPSD_A2DP_SNK_PAN_PANU_START_AUDIO_DURING_DATA_COMM | \
Packit 34410b
			MPSD_A2DP_SNK_PAN_PANU_DATA_COMM_DURING_AUDIO_STREAM | \
Packit 34410b
			MPSD_A2DP_SNK_PBAP_CLI_PB_DL_DURING_AUDIO_STREAM)
Packit 34410b
Packit 34410b
#define MPS_MPSD_AVRCP_CT MPS_MPSD_A2DP_SNK
Packit 34410b
Packit 34410b
#define MPS_MPSD_AVRCP_TG MPS_MPSD_A2DP_SRC
Packit 34410b
Packit 34410b
#define MPS_MPSD_DUN_GW (MPSD_HFP_AG_DUN_GW_DATA_COMM_DURING_VOICE_CALL | \
Packit 34410b
			MPSD_HFP_AG_DUN_GW_OUTGOING_CALL_DURING_DATA_COMM | \
Packit 34410b
			MPSD_HFP_AG_DUN_GW_INCOMING_CALL_DURING_DATA_COMM | \
Packit 34410b
			MPSD_A2DP_SRC_DUN_GW_START_AUDIO_DURING_DATA_COMM | \
Packit 34410b
			MPSD_A2DP_SRC_DUN_GW_DATA_COMM_DURING_AUDIO_STREAM | \
Packit 34410b
			MPSD_HFP_AG_DUN_GW_TERMINATE_CALL_DURING_DATA_COMM)
Packit 34410b
Packit 34410b
#define MPS_MPSD_DUN_DT (MPSD_HFP_HF_DUN_DT_DATA_COMM_DURING_VOICE_CALL | \
Packit 34410b
			MPSD_HFP_HF_DUN_DT_OUTGOING_CALL_DURING_DATA_COMM | \
Packit 34410b
			MPSD_HFP_HF_DUN_DT_INCOMING_CALL_DURING_DATA_COMM | \
Packit 34410b
			MPSD_A2DP_SNK_DUN_DT_START_AUDIO_DURING_DATA_COMM | \
Packit 34410b
			MPSD_A2DP_SNK_DUN_DT_DATA_COMM_DURING_AUDIO_STREAM | \
Packit 34410b
			MPSD_HFP_HF_DUN_DT_TERMINATE_CALL_DURING_DATA_COMM)
Packit 34410b
Packit 34410b
#define MPS_MPSD_PAN_NAP (MPSD_HFP_AG_PAN_NAP_DATA_COMM_DURING_VOICE_CALL | \
Packit 34410b
			MPSD_HFP_AG_PAN_NAP_OUTGOING_CALL_DURING_DATA_COMM | \
Packit 34410b
			MPSD_HFP_AG_PAN_NAP_INCOMING_CALL_DURING_DATA_COMM | \
Packit 34410b
			MPSD_A2DP_SRC_PAN_NAP_START_AUDIO_DURING_DATA_COMM | \
Packit 34410b
			MPSD_A2DP_SRC_PAN_NAP_DATA_COMM_DURING_AUDIO_STREAM)
Packit 34410b
Packit 34410b
#define MPS_MPSD_PAN_PANU (MPSD_HFP_HF_PAN_PANU_DATA_COMM_DURING_VOICE_CALL | \
Packit 34410b
			MPSD_HFP_HF_PAN_PANU_OUTGOING_CALL_DURING_DATA_COMM | \
Packit 34410b
			MPSD_HFP_HF_PAN_PANU_INCOMING_CALL_DURING_DATA_COMM | \
Packit 34410b
			MPSD_A2DP_SNK_PAN_PANU_START_AUDIO_DURING_DATA_COMM | \
Packit 34410b
			MPSD_A2DP_SNK_PAN_PANU_DATA_COMM_DURING_AUDIO_STREAM)
Packit 34410b
Packit 34410b
#define MPS_MPSD_PBAP_SRC MPSD_A2DP_SRC_PBAP_SRV_PB_DL_DURING_AUDIO_STREAM
Packit 34410b
Packit 34410b
#define MPS_MPSD_PBAP_CLI MPSD_A2DP_SNK_PBAP_CLI_PB_DL_DURING_AUDIO_STREAM
Packit 34410b
Packit 34410b
#define MPS_MPSD_ALL (MPS_MPSD_HFP_AG | MPS_MPSD_HFP_HF | \
Packit 34410b
			MPS_MPSD_A2DP_SRC | MPS_MPSD_A2DP_SNK | \
Packit 34410b
			MPS_MPSD_AVRCP_CT | MPS_MPSD_AVRCP_TG | \
Packit 34410b
			MPS_MPSD_DUN_GW | MPS_MPSD_DUN_DT | \
Packit 34410b
			MPS_MPSD_PAN_NAP | MPS_MPSD_PAN_PANU | \
Packit 34410b
			MPS_MPSD_PBAP_SRC | MPS_MPSD_PBAP_CLI)
Packit 34410b
Packit 34410b
#define MPS_MPMD_HFP_AG MPMD_HFP_AG_CALL_TERMINATION_DURING_AVP
Packit 34410b
Packit 34410b
#define MPS_MPMD_HFP_HF ( \
Packit 34410b
		MPMD_HFP_HF_A2DP_SNK_AVRCP_CT_ANSWER_CALL_DURING_AUDIO | \
Packit 34410b
		MPMD_HFP_HF_A2DP_SNK_AVRCP_CT_OUTGOING_CALL_DURING_AUDIO | \
Packit 34410b
		MPMD_HFP_HF_A2DP_SNK_AVRCP_CT_REJECT_CALL_DURING_AUDIO | \
Packit 34410b
		MPMD_HFP_HF_A2DP_SNK_AVRCP_CT_TERMINATION_DURING_AVP | \
Packit 34410b
		MPMD_HFP_HF_A2DP_SNK_AVRCP_CT_PLAY_DURING_CALL)
Packit 34410b
Packit 34410b
#define MPS_MPMD_A2DP_SRC (MPMD_A2DP_SRC_AVRCP_TG_ANSWER_CALL_DURING_AUDIO | \
Packit 34410b
			MPMD_A2DP_SRC_AVRCP_TG_OUTGOING_CALL_DURING_AUDIO | \
Packit 34410b
			MPMD_A2DP_SRC_AVRCP_TG_REJECT_CALL_DURING_AUDIO | \
Packit 34410b
			MPMD_A2DP_SRC_AVRCP_TG_TERMINATION_DURING_AVP | \
Packit 34410b
			MPMD_A2DP_SRC_AVRCP_TG_PRESS_PLAY_DURING_CALL | \
Packit 34410b
			MPMD_A2DP_SRC_AVRCP_TG_START_AUDIO_AFTER_PLAY | \
Packit 34410b
			MPMD_A2DP_SRC_AVRCP_TG_SUSPEND_AUDIO_AFTER_PAUSE | \
Packit 34410b
			MPMD_A2DP_SRC_AVRCP_TG_START_AUDIO_DURING_DATA_COMM | \
Packit 34410b
			MPMD_A2DP_SRC_AVRCP_TG_START_DATA_DURING_AUDIO)
Packit 34410b
Packit 34410b
#define MPS_MPMD_A2DP_SNK ( \
Packit 34410b
		MPMD_HFP_HF_A2DP_SNK_AVRCP_CT_ANSWER_CALL_DURING_AUDIO | \
Packit 34410b
		MPMD_HFP_HF_A2DP_SNK_AVRCP_CT_OUTGOING_CALL_DURING_AUDIO | \
Packit 34410b
		MPMD_HFP_HF_A2DP_SNK_AVRCP_CT_REJECT_CALL_DURING_AUDIO | \
Packit 34410b
		MPMD_HFP_HF_A2DP_SNK_AVRCP_CT_TERMINATION_DURING_AVP | \
Packit 34410b
		MPMD_HFP_HF_A2DP_SNK_AVRCP_CT_PLAY_DURING_CALL | \
Packit 34410b
		MPMD_A2DP_SNK_AVRCP_CT_DUN_DT_AUDIO_DURING_DATA_COMM | \
Packit 34410b
		MPMD_A2DP_SNK_AVRCP_CT_DUN_DT_START_DATA_DURING_AUDIO)
Packit 34410b
Packit 34410b
#define MPS_MPMD_AVRCP_CT MPS_MPMD_A2DP_SNK
Packit 34410b
Packit 34410b
/* should be set only if CT is supported but SNK is not supported */
Packit 34410b
#define MPS_MPMD_AVRCP_CT_ONLY ( \
Packit 34410b
		MPMD_AVRCP_CT_NO_A2DP_SNK_START_AUDIO_AFTER_PLAY | \
Packit 34410b
		MPMD_AVRCP_CT_NO_A2DP_SNK_SUSPEND_AUDIO_AFTER_PAUSE)
Packit 34410b
Packit 34410b
#define MPS_MPMD_AVRCP_TG MPS_MPMD_A2DP_SRC
Packit 34410b
Packit 34410b
#define MPS_MPMD_DUN_DT ( \
Packit 34410b
		MPMD_A2DP_SNK_AVRCP_CT_DUN_DT_AUDIO_DURING_DATA_COMM | \
Packit 34410b
		MPMD_A2DP_SNK_AVRCP_CT_DUN_DT_START_DATA_DURING_AUDIO)
Packit 34410b
Packit 34410b
#define MPS_MPMD_ALL (MPS_MPMD_HFP_AG | MPS_MPMD_HFP_HF | MPS_MPMD_A2DP_SRC | \
Packit 34410b
			MPS_MPMD_A2DP_SNK | MPS_MPMD_AVRCP_CT | \
Packit 34410b
			MPS_MPMD_AVRCP_CT_ONLY | MPS_MPMD_AVRCP_TG | \
Packit 34410b
			MPS_MPMD_DUN_DT)
Packit 34410b
Packit 34410b
/* Assume all dependencies are supported */
Packit 34410b
#define MPS_DEFAULT_DEPS (MPS_DEPS_SNIFF_MODE_DURRING_STREAMING | \
Packit 34410b
			MPS_DEPS_GAVDP_REQUIREMENTS | \
Packit 34410b
			MPS_DEPS_DIS_CONNECTION_ORDER_BEHAVIOR)
Packit 34410b
Packit 34410b
static sdp_record_t *server = NULL;
Packit 34410b
static uint32_t fixed_dbts = 0;
Packit 34410b
Packit 34410b
/*
Packit 34410b
 * List of version numbers supported by the SDP server.
Packit 34410b
 * Add to this list when newer versions are supported.
Packit 34410b
 */
Packit 34410b
static sdp_version_t sdpVnumArray[1] = {
Packit 34410b
	{ 1, 0 }
Packit 34410b
};
Packit 34410b
static const int sdpServerVnumEntries = 1;
Packit 34410b
Packit 34410b
static uint32_t mps_handle = 0;
Packit 34410b
static bool mps_mpmd = false;
Packit 34410b
Packit 34410b
/*
Packit 34410b
 * A simple function which returns the time of day in
Packit 34410b
 * seconds. Used for updating the service db state
Packit 34410b
 * attribute of the service record of the SDP server
Packit 34410b
 */
Packit 34410b
uint32_t sdp_get_time(void)
Packit 34410b
{
Packit 34410b
	/*
Packit 34410b
	 * To handle failure in gettimeofday, so an old
Packit 34410b
	 * value is returned and service does not fail
Packit 34410b
	 */
Packit 34410b
	static struct timeval tm;
Packit 34410b
Packit 34410b
	gettimeofday(&tm, NULL);
Packit 34410b
	return (uint32_t) tm.tv_sec;
Packit 34410b
}
Packit 34410b
Packit 34410b
/*
Packit 34410b
 * The service database state is an attribute of the service record
Packit 34410b
 * of the SDP server itself. This attribute is guaranteed to
Packit 34410b
 * change if any of the contents of the service repository
Packit 34410b
 * changes. This function updates the timestamp of value of
Packit 34410b
 * the svcDBState attribute
Packit 34410b
 * Set the SDP server DB. Simply a timestamp which is the marker
Packit 34410b
 * when the DB was modified.
Packit 34410b
 */
Packit 34410b
static void update_db_timestamp(void)
Packit 34410b
{
Packit 34410b
	if (fixed_dbts) {
Packit 34410b
		sdp_data_t *d = sdp_data_alloc(SDP_UINT32, &fixed_dbts);
Packit 34410b
		sdp_attr_replace(server, SDP_ATTR_SVCDB_STATE, d);
Packit 34410b
	} else {
Packit 34410b
		uint32_t dbts = sdp_get_time();
Packit 34410b
		sdp_data_t *d = sdp_data_alloc(SDP_UINT32, &dbts);
Packit 34410b
		sdp_attr_replace(server, SDP_ATTR_SVCDB_STATE, d);
Packit 34410b
	}
Packit 34410b
}
Packit 34410b
Packit 34410b
void set_fixed_db_timestamp(uint32_t dbts)
Packit 34410b
{
Packit 34410b
	fixed_dbts = dbts;
Packit 34410b
}
Packit 34410b
Packit 34410b
void register_public_browse_group(void)
Packit 34410b
{
Packit 34410b
	sdp_list_t *browselist;
Packit 34410b
	uuid_t bgscid, pbgid;
Packit 34410b
	sdp_data_t *sdpdata;
Packit 34410b
	sdp_record_t *browse = sdp_record_alloc();
Packit 34410b
Packit 34410b
	browse->handle = SDP_SERVER_RECORD_HANDLE + 1;
Packit 34410b
Packit 34410b
	sdp_record_add(BDADDR_ANY, browse);
Packit 34410b
	sdpdata = sdp_data_alloc(SDP_UINT32, &browse->handle);
Packit 34410b
	sdp_attr_add(browse, SDP_ATTR_RECORD_HANDLE, sdpdata);
Packit 34410b
Packit 34410b
	sdp_uuid16_create(&bgscid, BROWSE_GRP_DESC_SVCLASS_ID);
Packit 34410b
	browselist = sdp_list_append(0, &bgscid);
Packit 34410b
	sdp_set_service_classes(browse, browselist);
Packit 34410b
	sdp_list_free(browselist, 0);
Packit 34410b
Packit 34410b
	sdp_uuid16_create(&pbgid, PUBLIC_BROWSE_GROUP);
Packit 34410b
	sdp_attr_add_new(browse, SDP_ATTR_GROUP_ID,
Packit 34410b
				SDP_UUID16, &pbgid.value.uuid16);
Packit 34410b
}
Packit 34410b
Packit 34410b
/*
Packit 34410b
 * The SDP server must present its own service record to
Packit 34410b
 * the service repository. This can be accessed by service
Packit 34410b
 * discovery clients. This method constructs a service record
Packit 34410b
 * and stores it in the repository
Packit 34410b
 */
Packit 34410b
void register_server_service(void)
Packit 34410b
{
Packit 34410b
	sdp_list_t *classIDList;
Packit 34410b
	uuid_t classID;
Packit 34410b
	void **versions, **versionDTDs;
Packit 34410b
	uint8_t dtd;
Packit 34410b
	sdp_data_t *pData;
Packit 34410b
	int i;
Packit 34410b
Packit 34410b
	server = sdp_record_alloc();
Packit 34410b
	server->pattern = NULL;
Packit 34410b
Packit 34410b
	/* Force the record to be SDP_SERVER_RECORD_HANDLE */
Packit 34410b
	server->handle = SDP_SERVER_RECORD_HANDLE;
Packit 34410b
Packit 34410b
	sdp_record_add(BDADDR_ANY, server);
Packit 34410b
	sdp_attr_add(server, SDP_ATTR_RECORD_HANDLE,
Packit 34410b
				sdp_data_alloc(SDP_UINT32, &server->handle));
Packit 34410b
Packit 34410b
	sdp_uuid16_create(&classID, SDP_SERVER_SVCLASS_ID);
Packit 34410b
	classIDList = sdp_list_append(0, &classID);
Packit 34410b
	sdp_set_service_classes(server, classIDList);
Packit 34410b
	sdp_list_free(classIDList, 0);
Packit 34410b
Packit 34410b
	/*
Packit 34410b
	 * Set the version numbers supported, these are passed as arguments
Packit 34410b
	 * to the server on command line. Now defaults to 1.0
Packit 34410b
	 * Build the version number sequence first
Packit 34410b
	 */
Packit 34410b
	versions = malloc(sdpServerVnumEntries * sizeof(void *));
Packit 34410b
	versionDTDs = malloc(sdpServerVnumEntries * sizeof(void *));
Packit 34410b
	dtd = SDP_UINT16;
Packit 34410b
	for (i = 0; i < sdpServerVnumEntries; i++) {
Packit 34410b
		uint16_t *version = malloc(sizeof(uint16_t));
Packit 34410b
		*version = sdpVnumArray[i].major;
Packit 34410b
		*version = (*version << 8);
Packit 34410b
		*version |= sdpVnumArray[i].minor;
Packit 34410b
		versions[i] = version;
Packit 34410b
		versionDTDs[i] = &dtd;
Packit 34410b
	}
Packit 34410b
	pData = sdp_seq_alloc(versionDTDs, versions, sdpServerVnumEntries);
Packit 34410b
	for (i = 0; i < sdpServerVnumEntries; i++)
Packit 34410b
		free(versions[i]);
Packit 34410b
	free(versions);
Packit 34410b
	free(versionDTDs);
Packit 34410b
	sdp_attr_add(server, SDP_ATTR_VERSION_NUM_LIST, pData);
Packit 34410b
Packit 34410b
	update_db_timestamp();
Packit 34410b
}
Packit 34410b
Packit 34410b
void register_device_id(uint16_t source, uint16_t vendor,
Packit 34410b
					uint16_t product, uint16_t version)
Packit 34410b
{
Packit 34410b
	const uint16_t spec = 0x0103;
Packit 34410b
	const uint8_t primary = 1;
Packit 34410b
	sdp_list_t *class_list, *group_list, *profile_list;
Packit 34410b
	uuid_t class_uuid, group_uuid;
Packit 34410b
	sdp_data_t *sdp_data, *primary_data, *source_data;
Packit 34410b
	sdp_data_t *spec_data, *vendor_data, *product_data, *version_data;
Packit 34410b
	sdp_profile_desc_t profile;
Packit 34410b
	sdp_record_t *record = sdp_record_alloc();
Packit 34410b
Packit 34410b
	DBG("Adding device id record for %04x:%04x:%04x:%04x",
Packit 34410b
					source, vendor, product, version);
Packit 34410b
Packit 34410b
	record->handle = sdp_next_handle();
Packit 34410b
Packit 34410b
	sdp_record_add(BDADDR_ANY, record);
Packit 34410b
	sdp_data = sdp_data_alloc(SDP_UINT32, &record->handle);
Packit 34410b
	sdp_attr_add(record, SDP_ATTR_RECORD_HANDLE, sdp_data);
Packit 34410b
Packit 34410b
	sdp_uuid16_create(&class_uuid, PNP_INFO_SVCLASS_ID);
Packit 34410b
	class_list = sdp_list_append(0, &class_uuid);
Packit 34410b
	sdp_set_service_classes(record, class_list);
Packit 34410b
	sdp_list_free(class_list, NULL);
Packit 34410b
Packit 34410b
	sdp_uuid16_create(&group_uuid, PUBLIC_BROWSE_GROUP);
Packit 34410b
	group_list = sdp_list_append(NULL, &group_uuid);
Packit 34410b
	sdp_set_browse_groups(record, group_list);
Packit 34410b
	sdp_list_free(group_list, NULL);
Packit 34410b
Packit 34410b
	sdp_uuid16_create(&profile.uuid, PNP_INFO_PROFILE_ID);
Packit 34410b
	profile.version = spec;
Packit 34410b
	profile_list = sdp_list_append(NULL, &profile);
Packit 34410b
	sdp_set_profile_descs(record, profile_list);
Packit 34410b
	sdp_list_free(profile_list, NULL);
Packit 34410b
Packit 34410b
	spec_data = sdp_data_alloc(SDP_UINT16, &spec);
Packit 34410b
	sdp_attr_add(record, 0x0200, spec_data);
Packit 34410b
Packit 34410b
	vendor_data = sdp_data_alloc(SDP_UINT16, &vendor);
Packit 34410b
	sdp_attr_add(record, 0x0201, vendor_data);
Packit 34410b
Packit 34410b
	product_data = sdp_data_alloc(SDP_UINT16, &product);
Packit 34410b
	sdp_attr_add(record, 0x0202, product_data);
Packit 34410b
Packit 34410b
	version_data = sdp_data_alloc(SDP_UINT16, &version);
Packit 34410b
	sdp_attr_add(record, 0x0203, version_data);
Packit 34410b
Packit 34410b
	primary_data = sdp_data_alloc(SDP_BOOL, &primary);
Packit 34410b
	sdp_attr_add(record, 0x0204, primary_data);
Packit 34410b
Packit 34410b
	source_data = sdp_data_alloc(SDP_UINT16, &source);
Packit 34410b
	sdp_attr_add(record, 0x0205, source_data);
Packit 34410b
Packit 34410b
	update_db_timestamp();
Packit 34410b
}
Packit 34410b
Packit 34410b
static bool class_supported(uint16_t class)
Packit 34410b
{
Packit 34410b
	sdp_list_t *list;
Packit 34410b
	uuid_t uuid;
Packit 34410b
Packit 34410b
	sdp_uuid16_create(&uuid, class);
Packit 34410b
Packit 34410b
	for (list = sdp_get_record_list(); list; list = list->next) {
Packit 34410b
		sdp_record_t *rec = list->data;
Packit 34410b
Packit 34410b
		if (sdp_uuid_cmp(&rec->svclass, &uuid) == 0)
Packit 34410b
			return true;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	return false;
Packit 34410b
}
Packit 34410b
Packit 34410b
static uint64_t mps_mpsd_features(void)
Packit 34410b
{
Packit 34410b
	uint64_t feat = MPS_MPSD_ALL;
Packit 34410b
Packit 34410b
	if (!class_supported(HANDSFREE_AGW_SVCLASS_ID))
Packit 34410b
		feat &= ~MPS_MPSD_HFP_AG;
Packit 34410b
Packit 34410b
	if (!class_supported(HANDSFREE_SVCLASS_ID))
Packit 34410b
		feat &= ~MPS_MPSD_HFP_HF;
Packit 34410b
Packit 34410b
	if (!class_supported(AUDIO_SOURCE_SVCLASS_ID))
Packit 34410b
		feat &= ~MPS_MPSD_A2DP_SRC;
Packit 34410b
Packit 34410b
	if (!class_supported(AUDIO_SINK_SVCLASS_ID))
Packit 34410b
		feat &= ~MPS_MPSD_A2DP_SNK;
Packit 34410b
Packit 34410b
	if (!class_supported(AV_REMOTE_CONTROLLER_SVCLASS_ID))
Packit 34410b
		feat &= ~MPS_MPSD_AVRCP_CT;
Packit 34410b
Packit 34410b
	if (!class_supported(AV_REMOTE_TARGET_SVCLASS_ID))
Packit 34410b
		feat &= ~MPS_MPSD_AVRCP_TG;
Packit 34410b
Packit 34410b
	if (!class_supported(DIALUP_NET_SVCLASS_ID))
Packit 34410b
		feat &= ~MPS_MPSD_DUN_GW;
Packit 34410b
Packit 34410b
	/* TODO */
Packit 34410b
	feat &= ~MPS_MPSD_DUN_DT;
Packit 34410b
Packit 34410b
	if (!class_supported(NAP_SVCLASS_ID))
Packit 34410b
		feat &= ~MPS_MPSD_PAN_NAP;
Packit 34410b
Packit 34410b
	if (!class_supported(PANU_SVCLASS_ID))
Packit 34410b
		feat &= ~MPS_MPSD_PAN_PANU;
Packit 34410b
Packit 34410b
	if (!class_supported(PBAP_PSE_SVCLASS_ID))
Packit 34410b
		feat &= ~MPS_MPSD_PBAP_SRC;
Packit 34410b
Packit 34410b
	if (!class_supported(PBAP_PCE_SVCLASS_ID))
Packit 34410b
		feat &= ~MPS_MPSD_PBAP_CLI;
Packit 34410b
Packit 34410b
	return feat;
Packit 34410b
}
Packit 34410b
Packit 34410b
static uint64_t mps_mpmd_features(void)
Packit 34410b
{
Packit 34410b
	uint64_t feat = MPS_MPMD_ALL;
Packit 34410b
Packit 34410b
	if (!class_supported(HANDSFREE_AGW_SVCLASS_ID))
Packit 34410b
		feat &= ~MPS_MPMD_HFP_AG;
Packit 34410b
Packit 34410b
	if (!class_supported(HANDSFREE_SVCLASS_ID))
Packit 34410b
		feat &= ~MPS_MPMD_HFP_HF;
Packit 34410b
Packit 34410b
	if (!class_supported(AUDIO_SOURCE_SVCLASS_ID))
Packit 34410b
		feat &= ~MPS_MPMD_A2DP_SRC;
Packit 34410b
Packit 34410b
	if (!class_supported(AUDIO_SINK_SVCLASS_ID))
Packit 34410b
		feat &= ~MPS_MPMD_A2DP_SNK;
Packit 34410b
	else
Packit 34410b
		feat &= ~MPS_MPMD_AVRCP_CT_ONLY;
Packit 34410b
Packit 34410b
	if (!class_supported(AV_REMOTE_CONTROLLER_SVCLASS_ID)) {
Packit 34410b
		feat &= ~MPS_MPMD_AVRCP_CT;
Packit 34410b
		feat &= ~MPS_MPMD_AVRCP_CT_ONLY;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	if (!class_supported(AV_REMOTE_TARGET_SVCLASS_ID))
Packit 34410b
		feat &= ~MPS_MPMD_AVRCP_TG;
Packit 34410b
Packit 34410b
	/* TODO */
Packit 34410b
	feat &= ~MPS_MPMD_DUN_DT;
Packit 34410b
Packit 34410b
	return feat;
Packit 34410b
}
Packit 34410b
Packit 34410b
static sdp_record_t *mps_record(int mpmd)
Packit 34410b
{
Packit 34410b
	sdp_data_t *mpsd_features, *mpmd_features, *dependencies;
Packit 34410b
	sdp_list_t *svclass_id, *pfseq, *root;
Packit 34410b
	uuid_t root_uuid, svclass_uuid;
Packit 34410b
	sdp_profile_desc_t profile;
Packit 34410b
	sdp_record_t *record;
Packit 34410b
	uint64_t mpsd_feat = MPS_MPSD_DEFAULT_FEATURES;
Packit 34410b
	uint64_t mpmd_feat = MPS_MPMD_DEFAULT_FEATURES;
Packit 34410b
	uint16_t deps = MPS_DEFAULT_DEPS;
Packit 34410b
Packit 34410b
	record = sdp_record_alloc();
Packit 34410b
	if (!record)
Packit 34410b
		return NULL;
Packit 34410b
Packit 34410b
	sdp_uuid16_create(&root_uuid, PUBLIC_BROWSE_GROUP);
Packit 34410b
	root = sdp_list_append(NULL, &root_uuid);
Packit 34410b
	sdp_set_browse_groups(record, root);
Packit 34410b
	sdp_list_free(root, NULL);
Packit 34410b
Packit 34410b
	sdp_uuid16_create(&svclass_uuid, MPS_SVCLASS_ID);
Packit 34410b
	svclass_id = sdp_list_append(NULL, &svclass_uuid);
Packit 34410b
	sdp_set_service_classes(record, svclass_id);
Packit 34410b
	sdp_list_free(svclass_id, NULL);
Packit 34410b
Packit 34410b
	sdp_uuid16_create(&profile.uuid, MPS_PROFILE_ID);
Packit 34410b
	profile.version = 0x0100;
Packit 34410b
	pfseq = sdp_list_append(NULL, &profile);
Packit 34410b
	sdp_set_profile_descs(record, pfseq);
Packit 34410b
	sdp_list_free(pfseq, NULL);
Packit 34410b
Packit 34410b
	mpsd_features = sdp_data_alloc(SDP_UINT64, &mpsd_feat);
Packit 34410b
	sdp_attr_add(record, SDP_ATTR_MPSD_SCENARIOS, mpsd_features);
Packit 34410b
Packit 34410b
	if (mpmd) {
Packit 34410b
		mpmd_features = sdp_data_alloc(SDP_UINT64, &mpmd_feat);
Packit 34410b
		sdp_attr_add(record, SDP_ATTR_MPMD_SCENARIOS, mpmd_features);
Packit 34410b
	}
Packit 34410b
Packit 34410b
	dependencies = sdp_data_alloc(SDP_UINT16, &deps);
Packit 34410b
	sdp_attr_add(record, SDP_ATTR_MPS_DEPENDENCIES, dependencies);
Packit 34410b
Packit 34410b
	sdp_set_info_attr(record, "Multi Profile", 0, 0);
Packit 34410b
Packit 34410b
	return record;
Packit 34410b
}
Packit 34410b
Packit 34410b
void register_mps(bool mpmd)
Packit 34410b
{
Packit 34410b
	sdp_record_t *record;
Packit 34410b
Packit 34410b
	record = mps_record(mpmd);
Packit 34410b
	if (!record)
Packit 34410b
		return;
Packit 34410b
Packit 34410b
	if (add_record_to_server(BDADDR_ANY, record) < 0) {
Packit 34410b
		sdp_record_free(record);
Packit 34410b
		return;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	mps_handle = record->handle;
Packit 34410b
	mps_mpmd = mpmd;
Packit 34410b
}
Packit 34410b
Packit 34410b
static void update_mps(void)
Packit 34410b
{
Packit 34410b
	sdp_record_t *rec;
Packit 34410b
	sdp_data_t *data;
Packit 34410b
	uint64_t mpsd_feat, mpmd_feat;
Packit 34410b
Packit 34410b
	if (!mps_handle)
Packit 34410b
		return;
Packit 34410b
Packit 34410b
	rec = sdp_record_find(mps_handle);
Packit 34410b
	if (!rec)
Packit 34410b
		return;
Packit 34410b
Packit 34410b
	mpsd_feat = mps_mpsd_features();
Packit 34410b
	data = sdp_data_alloc(SDP_UINT64, &mpsd_feat);
Packit 34410b
	sdp_attr_replace(rec, SDP_ATTR_MPSD_SCENARIOS, data);
Packit 34410b
Packit 34410b
	if (mps_mpmd) {
Packit 34410b
		mpmd_feat = mps_mpmd_features();
Packit 34410b
		data = sdp_data_alloc(SDP_UINT64, &mpmd_feat);
Packit 34410b
		sdp_attr_replace(rec, SDP_ATTR_MPMD_SCENARIOS, data);
Packit 34410b
	}
Packit 34410b
}
Packit 34410b
Packit 34410b
int add_record_to_server(const bdaddr_t *src, sdp_record_t *rec)
Packit 34410b
{
Packit 34410b
	sdp_data_t *data;
Packit 34410b
	sdp_list_t *pattern;
Packit 34410b
Packit 34410b
	if (rec->handle == 0xffffffff) {
Packit 34410b
		rec->handle = sdp_next_handle();
Packit 34410b
		if (rec->handle < 0x10000)
Packit 34410b
			return -ENOSPC;
Packit 34410b
	} else {
Packit 34410b
		if (sdp_record_find(rec->handle))
Packit 34410b
			return -EEXIST;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	DBG("Adding record with handle 0x%05x", rec->handle);
Packit 34410b
Packit 34410b
	sdp_record_add(src, rec);
Packit 34410b
Packit 34410b
	data = sdp_data_alloc(SDP_UINT32, &rec->handle);
Packit 34410b
	sdp_attr_replace(rec, SDP_ATTR_RECORD_HANDLE, data);
Packit 34410b
Packit 34410b
	if (sdp_data_get(rec, SDP_ATTR_BROWSE_GRP_LIST) == NULL) {
Packit 34410b
		uuid_t uuid;
Packit 34410b
		sdp_uuid16_create(&uuid, PUBLIC_BROWSE_GROUP);
Packit 34410b
		sdp_pattern_add_uuid(rec, &uuid);
Packit 34410b
	}
Packit 34410b
Packit 34410b
	for (pattern = rec->pattern; pattern; pattern = pattern->next) {
Packit 34410b
		char uuid[32];
Packit 34410b
Packit 34410b
		if (pattern->data == NULL)
Packit 34410b
			continue;
Packit 34410b
Packit 34410b
		sdp_uuid2strn((uuid_t *) pattern->data, uuid, sizeof(uuid));
Packit 34410b
		DBG("Record pattern UUID %s", uuid);
Packit 34410b
	}
Packit 34410b
Packit 34410b
	update_mps();
Packit 34410b
	update_db_timestamp();
Packit 34410b
Packit 34410b
	return 0;
Packit 34410b
}
Packit 34410b
Packit 34410b
int remove_record_from_server(uint32_t handle)
Packit 34410b
{
Packit 34410b
	sdp_record_t *rec;
Packit 34410b
Packit 34410b
	/* Refuse to remove the server's own record */
Packit 34410b
	if (handle == SDP_SERVER_RECORD_HANDLE)
Packit 34410b
		return -EINVAL;
Packit 34410b
Packit 34410b
	DBG("Removing record with handle 0x%05x", handle);
Packit 34410b
Packit 34410b
	rec = sdp_record_find(handle);
Packit 34410b
	if (!rec)
Packit 34410b
		return -ENOENT;
Packit 34410b
Packit 34410b
	if (sdp_record_remove(handle) == 0) {
Packit 34410b
		update_mps();
Packit 34410b
		update_db_timestamp();
Packit 34410b
	}
Packit 34410b
Packit 34410b
	sdp_record_free(rec);
Packit 34410b
Packit 34410b
	return 0;
Packit 34410b
}
Packit 34410b
Packit 34410b
/* FIXME: refactor for server-side */
Packit 34410b
static sdp_record_t *extract_pdu_server(bdaddr_t *device, uint8_t *p,
Packit 34410b
					unsigned int bufsize,
Packit 34410b
					uint32_t handleExpected, int *scanned)
Packit 34410b
{
Packit 34410b
	int extractStatus = -1, localExtractedLength = 0;
Packit 34410b
	uint8_t dtd;
Packit 34410b
	int seqlen = 0;
Packit 34410b
	sdp_record_t *rec = NULL;
Packit 34410b
	uint16_t attrId, lookAheadAttrId;
Packit 34410b
	sdp_data_t *pAttr = NULL;
Packit 34410b
	uint32_t handle = 0xffffffff;
Packit 34410b
Packit 34410b
	*scanned = sdp_extract_seqtype(p, bufsize, &dtd, &seqlen);
Packit 34410b
	p += *scanned;
Packit 34410b
	bufsize -= *scanned;
Packit 34410b
Packit 34410b
	if (bufsize < sizeof(uint8_t) + sizeof(uint8_t)) {
Packit 34410b
		SDPDBG("Unexpected end of packet");
Packit 34410b
		return NULL;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	lookAheadAttrId = get_be16(p + sizeof(uint8_t));
Packit 34410b
Packit 34410b
	SDPDBG("Look ahead attr id : %d", lookAheadAttrId);
Packit 34410b
Packit 34410b
	if (lookAheadAttrId == SDP_ATTR_RECORD_HANDLE) {
Packit 34410b
		if (bufsize < (sizeof(uint8_t) * 2) +
Packit 34410b
					sizeof(uint16_t) + sizeof(uint32_t)) {
Packit 34410b
			SDPDBG("Unexpected end of packet");
Packit 34410b
			return NULL;
Packit 34410b
		}
Packit 34410b
		handle = get_be32(p + sizeof(uint8_t) + sizeof(uint16_t) +
Packit 34410b
							sizeof(uint8_t));
Packit 34410b
		SDPDBG("SvcRecHandle : 0x%x", handle);
Packit 34410b
		rec = sdp_record_find(handle);
Packit 34410b
	} else if (handleExpected != 0xffffffff)
Packit 34410b
		rec = sdp_record_find(handleExpected);
Packit 34410b
Packit 34410b
	if (!rec) {
Packit 34410b
		rec = sdp_record_alloc();
Packit 34410b
		rec->attrlist = NULL;
Packit 34410b
		if (lookAheadAttrId == SDP_ATTR_RECORD_HANDLE) {
Packit 34410b
			rec->handle = handle;
Packit 34410b
			sdp_record_add(device, rec);
Packit 34410b
		} else if (handleExpected != 0xffffffff) {
Packit 34410b
			rec->handle = handleExpected;
Packit 34410b
			sdp_record_add(device, rec);
Packit 34410b
		}
Packit 34410b
	} else {
Packit 34410b
		sdp_list_free(rec->attrlist, (sdp_free_func_t) sdp_data_free);
Packit 34410b
		rec->attrlist = NULL;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	while (localExtractedLength < seqlen) {
Packit 34410b
		int attrSize = sizeof(uint8_t);
Packit 34410b
		int attrValueLength = 0;
Packit 34410b
Packit 34410b
		if (bufsize < attrSize + sizeof(uint16_t)) {
Packit 34410b
			SDPDBG("Unexpected end of packet: Terminating extraction of attributes");
Packit 34410b
			break;
Packit 34410b
		}
Packit 34410b
Packit 34410b
		SDPDBG("Extract PDU, sequenceLength: %d localExtractedLength: %d",
Packit 34410b
							seqlen, localExtractedLength);
Packit 34410b
		dtd = *(uint8_t *) p;
Packit 34410b
Packit 34410b
		attrId = get_be16(p + attrSize);
Packit 34410b
		attrSize += sizeof(uint16_t);
Packit 34410b
Packit 34410b
		SDPDBG("DTD of attrId : %d Attr id : 0x%x", dtd, attrId);
Packit 34410b
Packit 34410b
		pAttr = sdp_extract_attr(p + attrSize, bufsize - attrSize,
Packit 34410b
							&attrValueLength, rec);
Packit 34410b
Packit 34410b
		SDPDBG("Attr id : 0x%x attrValueLength : %d", attrId, attrValueLength);
Packit 34410b
Packit 34410b
		attrSize += attrValueLength;
Packit 34410b
		if (pAttr == NULL) {
Packit 34410b
			SDPDBG("Terminating extraction of attributes");
Packit 34410b
			break;
Packit 34410b
		}
Packit 34410b
		localExtractedLength += attrSize;
Packit 34410b
		p += attrSize;
Packit 34410b
		bufsize -= attrSize;
Packit 34410b
		sdp_attr_replace(rec, attrId, pAttr);
Packit 34410b
		extractStatus = 0;
Packit 34410b
		SDPDBG("Extract PDU, seqLength: %d localExtractedLength: %d",
Packit 34410b
					seqlen, localExtractedLength);
Packit 34410b
	}
Packit 34410b
Packit 34410b
	if (extractStatus == 0) {
Packit 34410b
		SDPDBG("Successful extracting of Svc Rec attributes");
Packit 34410b
#ifdef SDP_DEBUG
Packit 34410b
		sdp_print_service_attr(rec->attrlist);
Packit 34410b
#endif
Packit 34410b
		*scanned += seqlen;
Packit 34410b
	}
Packit 34410b
	return rec;
Packit 34410b
}
Packit 34410b
Packit 34410b
/*
Packit 34410b
 * Add the newly created service record to the service repository
Packit 34410b
 */
Packit 34410b
int service_register_req(sdp_req_t *req, sdp_buf_t *rsp)
Packit 34410b
{
Packit 34410b
	int scanned = 0;
Packit 34410b
	sdp_data_t *handle;
Packit 34410b
	uint8_t *p = req->buf + sizeof(sdp_pdu_hdr_t);
Packit 34410b
	int bufsize = req->len - sizeof(sdp_pdu_hdr_t);
Packit 34410b
	sdp_record_t *rec;
Packit 34410b
Packit 34410b
	req->flags = *p++;
Packit 34410b
	if (req->flags & SDP_DEVICE_RECORD) {
Packit 34410b
		bacpy(&req->device, (bdaddr_t *) p);
Packit 34410b
		p += sizeof(bdaddr_t);
Packit 34410b
		bufsize -= sizeof(bdaddr_t);
Packit 34410b
	}
Packit 34410b
Packit 34410b
	/* save image of PDU: we need it when clients request this attribute */
Packit 34410b
	rec = extract_pdu_server(&req->device, p, bufsize, 0xffffffff, &scanned);
Packit 34410b
	if (!rec)
Packit 34410b
		goto invalid;
Packit 34410b
Packit 34410b
	if (rec->handle == 0xffffffff) {
Packit 34410b
		rec->handle = sdp_next_handle();
Packit 34410b
		if (rec->handle < 0x10000) {
Packit 34410b
			sdp_record_free(rec);
Packit 34410b
			goto invalid;
Packit 34410b
		}
Packit 34410b
	} else {
Packit 34410b
		if (sdp_record_find(rec->handle)) {
Packit 34410b
			/* extract_pdu_server will add the record handle
Packit 34410b
			 * if it is missing. So instead of failing, skip
Packit 34410b
			 * the record adding to avoid duplication. */
Packit 34410b
			goto success;
Packit 34410b
		}
Packit 34410b
	}
Packit 34410b
Packit 34410b
	sdp_record_add(&req->device, rec);
Packit 34410b
	if (!(req->flags & SDP_RECORD_PERSIST))
Packit 34410b
		sdp_svcdb_set_collectable(rec, req->sock);
Packit 34410b
Packit 34410b
	handle = sdp_data_alloc(SDP_UINT32, &rec->handle);
Packit 34410b
	sdp_attr_replace(rec, SDP_ATTR_RECORD_HANDLE, handle);
Packit 34410b
Packit 34410b
success:
Packit 34410b
	/* if the browse group descriptor is NULL,
Packit 34410b
	 * ensure that the record belongs to the ROOT group */
Packit 34410b
	if (sdp_data_get(rec, SDP_ATTR_BROWSE_GRP_LIST) == NULL) {
Packit 34410b
		uuid_t uuid;
Packit 34410b
		sdp_uuid16_create(&uuid, PUBLIC_BROWSE_GROUP);
Packit 34410b
		sdp_pattern_add_uuid(rec, &uuid);
Packit 34410b
	}
Packit 34410b
Packit 34410b
	update_db_timestamp();
Packit 34410b
Packit 34410b
	/* Build a rsp buffer */
Packit 34410b
	put_be32(rec->handle, rsp->data);
Packit 34410b
	rsp->data_size = sizeof(uint32_t);
Packit 34410b
Packit 34410b
	return 0;
Packit 34410b
Packit 34410b
invalid:
Packit 34410b
	put_be16(SDP_INVALID_SYNTAX, rsp->data);
Packit 34410b
	rsp->data_size = sizeof(uint16_t);
Packit 34410b
Packit 34410b
	return -1;
Packit 34410b
}
Packit 34410b
Packit 34410b
/*
Packit 34410b
 * Update a service record
Packit 34410b
 */
Packit 34410b
int service_update_req(sdp_req_t *req, sdp_buf_t *rsp)
Packit 34410b
{
Packit 34410b
	sdp_record_t *orec, *nrec;
Packit 34410b
	int status = 0, scanned = 0;
Packit 34410b
	uint8_t *p = req->buf + sizeof(sdp_pdu_hdr_t);
Packit 34410b
	int bufsize = req->len - sizeof(sdp_pdu_hdr_t);
Packit 34410b
	uint32_t handle = get_be32(p);
Packit 34410b
Packit 34410b
	SDPDBG("Svc Rec Handle: 0x%x", handle);
Packit 34410b
Packit 34410b
	p += sizeof(uint32_t);
Packit 34410b
	bufsize -= sizeof(uint32_t);
Packit 34410b
Packit 34410b
	orec = sdp_record_find(handle);
Packit 34410b
Packit 34410b
	SDPDBG("SvcRecOld: %p", orec);
Packit 34410b
Packit 34410b
	if (!orec) {
Packit 34410b
		status = SDP_INVALID_RECORD_HANDLE;
Packit 34410b
		goto done;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	nrec = extract_pdu_server(BDADDR_ANY, p, bufsize, handle, &scanned);
Packit 34410b
	if (!nrec) {
Packit 34410b
		status = SDP_INVALID_SYNTAX;
Packit 34410b
		goto done;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	assert(nrec == orec);
Packit 34410b
Packit 34410b
	update_db_timestamp();
Packit 34410b
Packit 34410b
done:
Packit 34410b
	p = rsp->data;
Packit 34410b
	put_be16(status, p);
Packit 34410b
	rsp->data_size = sizeof(uint16_t);
Packit 34410b
	return status;
Packit 34410b
}
Packit 34410b
Packit 34410b
/*
Packit 34410b
 * Remove a registered service record
Packit 34410b
 */
Packit 34410b
int service_remove_req(sdp_req_t *req, sdp_buf_t *rsp)
Packit 34410b
{
Packit 34410b
	uint8_t *p = req->buf + sizeof(sdp_pdu_hdr_t);
Packit 34410b
	uint32_t handle = get_be32(p);
Packit 34410b
	sdp_record_t *rec;
Packit 34410b
	int status = 0;
Packit 34410b
Packit 34410b
	/* extract service record handle */
Packit 34410b
Packit 34410b
	rec = sdp_record_find(handle);
Packit 34410b
	if (rec) {
Packit 34410b
		sdp_svcdb_collect(rec);
Packit 34410b
		status = sdp_record_remove(handle);
Packit 34410b
		sdp_record_free(rec);
Packit 34410b
		if (status == 0)
Packit 34410b
			update_db_timestamp();
Packit 34410b
	} else {
Packit 34410b
		status = SDP_INVALID_RECORD_HANDLE;
Packit 34410b
		SDPDBG("Could not find record : 0x%x", handle);
Packit 34410b
	}
Packit 34410b
Packit 34410b
	p = rsp->data;
Packit 34410b
	put_be16(status, p);
Packit 34410b
	rsp->data_size = sizeof(uint16_t);
Packit 34410b
Packit 34410b
	return status;
Packit 34410b
}