mirror of
https://git.kernel.org/pub/scm/bluetooth/bluez.git
synced 2024-12-19 17:03:26 +08:00
7a2d604673
Use common style for multi-line comments consistent with rest of codebase.
438 lines
13 KiB
C
438 lines
13 KiB
C
/*
|
|
*
|
|
* BlueZ - Bluetooth protocol stack for Linux
|
|
*
|
|
* Copyright (C) 2010 GSyC/LibreSoft, Universidad Rey Juan Carlos.
|
|
* Copyright (C) 2010 Signove
|
|
* Copyright (C) 2014 Intel Corporation. All rights reserved.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*
|
|
*/
|
|
|
|
#define MCAP_VERSION 0x0100 /* current version 01.00 */
|
|
|
|
/* bytes to get MCAP Supported Procedures */
|
|
#define MCAP_SUP_PROC 0x06
|
|
|
|
/* maximum transmission unit for channels */
|
|
#define MCAP_CC_MTU 48
|
|
#define MCAP_DC_MTU 65535
|
|
|
|
/* MCAP Standard Op Codes */
|
|
#define MCAP_ERROR_RSP 0x00
|
|
#define MCAP_MD_CREATE_MDL_REQ 0x01
|
|
#define MCAP_MD_CREATE_MDL_RSP 0x02
|
|
#define MCAP_MD_RECONNECT_MDL_REQ 0x03
|
|
#define MCAP_MD_RECONNECT_MDL_RSP 0x04
|
|
#define MCAP_MD_ABORT_MDL_REQ 0x05
|
|
#define MCAP_MD_ABORT_MDL_RSP 0x06
|
|
#define MCAP_MD_DELETE_MDL_REQ 0x07
|
|
#define MCAP_MD_DELETE_MDL_RSP 0x08
|
|
|
|
/* MCAP Clock Sync Op Codes */
|
|
#define MCAP_MD_SYNC_CAP_REQ 0x11
|
|
#define MCAP_MD_SYNC_CAP_RSP 0x12
|
|
#define MCAP_MD_SYNC_SET_REQ 0x13
|
|
#define MCAP_MD_SYNC_SET_RSP 0x14
|
|
#define MCAP_MD_SYNC_INFO_IND 0x15
|
|
|
|
/* MCAP Response codes */
|
|
#define MCAP_SUCCESS 0x00
|
|
#define MCAP_INVALID_OP_CODE 0x01
|
|
#define MCAP_INVALID_PARAM_VALUE 0x02
|
|
#define MCAP_INVALID_MDEP 0x03
|
|
#define MCAP_MDEP_BUSY 0x04
|
|
#define MCAP_INVALID_MDL 0x05
|
|
#define MCAP_MDL_BUSY 0x06
|
|
#define MCAP_INVALID_OPERATION 0x07
|
|
#define MCAP_RESOURCE_UNAVAILABLE 0x08
|
|
#define MCAP_UNSPECIFIED_ERROR 0x09
|
|
#define MCAP_REQUEST_NOT_SUPPORTED 0x0A
|
|
#define MCAP_CONFIGURATION_REJECTED 0x0B
|
|
|
|
/* MDL IDs */
|
|
#define MCAP_MDLID_RESERVED 0x0000
|
|
#define MCAP_MDLID_INITIAL 0x0001
|
|
#define MCAP_MDLID_FINAL 0xFEFF
|
|
#define MCAP_ALL_MDLIDS 0xFFFF
|
|
|
|
/* MDEP IDs */
|
|
#define MCAP_MDEPID_INITIAL 0x00
|
|
#define MCAP_MDEPID_FINAL 0x7F
|
|
|
|
/* CSP special values */
|
|
#define MCAP_BTCLOCK_IMMEDIATE 0xffffffffUL
|
|
#define MCAP_TMSTAMP_DONTSET 0xffffffffffffffffULL
|
|
#define MCAP_BTCLOCK_MAX 0x0fffffff
|
|
#define MCAP_BTCLOCK_FIELD (MCAP_BTCLOCK_MAX + 1)
|
|
|
|
#define MCAP_CTRL_CACHED 0x01 /* MCL is cached */
|
|
#define MCAP_CTRL_STD_OP 0x02 /* Support for standard op codes */
|
|
#define MCAP_CTRL_SYNC_OP 0x04 /* Support for synchronization commands */
|
|
#define MCAP_CTRL_CONN 0x08 /* MCL is in connecting process */
|
|
#define MCAP_CTRL_FREE 0x10 /* MCL is marked as releasable */
|
|
#define MCAP_CTRL_NOCACHE 0x20 /* MCL is marked as not cacheable */
|
|
|
|
/*
|
|
* MCAP Request Packet Format
|
|
*/
|
|
|
|
typedef struct {
|
|
uint8_t op;
|
|
uint16_t mdl;
|
|
uint8_t mdep;
|
|
uint8_t conf;
|
|
} __attribute__ ((packed)) mcap_md_create_mdl_req;
|
|
|
|
typedef struct {
|
|
uint8_t op;
|
|
uint16_t mdl;
|
|
} __attribute__ ((packed)) mcap_md_req;
|
|
|
|
/* MCAP Response Packet Format */
|
|
|
|
typedef struct {
|
|
uint8_t op;
|
|
uint8_t rc;
|
|
uint16_t mdl;
|
|
uint8_t data[0];
|
|
} __attribute__ ((packed)) mcap_rsp;
|
|
|
|
/* MCAP Clock Synchronization Protocol */
|
|
|
|
typedef struct {
|
|
uint8_t op;
|
|
uint16_t timest;
|
|
} __attribute__ ((packed)) mcap_md_sync_cap_req;
|
|
|
|
typedef struct {
|
|
uint8_t op;
|
|
uint8_t rc;
|
|
} __attribute__ ((packed)) mcap_md_sync_rsp;
|
|
|
|
typedef struct {
|
|
uint8_t op;
|
|
uint8_t rc;
|
|
uint8_t btclock;
|
|
uint16_t sltime;
|
|
uint16_t timestnr;
|
|
uint16_t timestna;
|
|
} __attribute__ ((packed)) mcap_md_sync_cap_rsp;
|
|
|
|
typedef struct {
|
|
uint8_t op;
|
|
uint8_t timestui;
|
|
uint32_t btclock;
|
|
uint64_t timestst;
|
|
} __attribute__ ((packed)) mcap_md_sync_set_req;
|
|
|
|
typedef struct {
|
|
int8_t op;
|
|
uint8_t rc;
|
|
uint32_t btclock;
|
|
uint64_t timestst;
|
|
uint16_t timestsa;
|
|
} __attribute__ ((packed)) mcap_md_sync_set_rsp;
|
|
|
|
typedef struct {
|
|
uint8_t op;
|
|
uint32_t btclock;
|
|
uint64_t timestst;
|
|
uint16_t timestsa;
|
|
} __attribute__ ((packed)) mcap_md_sync_info_ind;
|
|
|
|
typedef enum {
|
|
/* MCAP Error Response Codes */
|
|
MCAP_ERROR_INVALID_OP_CODE = 1,
|
|
MCAP_ERROR_INVALID_PARAM_VALUE,
|
|
MCAP_ERROR_INVALID_MDEP,
|
|
MCAP_ERROR_MDEP_BUSY,
|
|
MCAP_ERROR_INVALID_MDL,
|
|
MCAP_ERROR_MDL_BUSY,
|
|
MCAP_ERROR_INVALID_OPERATION,
|
|
MCAP_ERROR_RESOURCE_UNAVAILABLE,
|
|
MCAP_ERROR_UNSPECIFIED_ERROR,
|
|
MCAP_ERROR_REQUEST_NOT_SUPPORTED,
|
|
MCAP_ERROR_CONFIGURATION_REJECTED,
|
|
/* MCAP Internal Errors */
|
|
MCAP_ERROR_INVALID_ARGS,
|
|
MCAP_ERROR_ALREADY_EXISTS,
|
|
MCAP_ERROR_REQ_IGNORED,
|
|
MCAP_ERROR_MCL_CLOSED,
|
|
MCAP_ERROR_FAILED
|
|
} McapError;
|
|
|
|
typedef enum {
|
|
MCAP_MDL_CB_INVALID,
|
|
MCAP_MDL_CB_CONNECTED, /* mcap_mdl_event_cb */
|
|
MCAP_MDL_CB_CLOSED, /* mcap_mdl_event_cb */
|
|
MCAP_MDL_CB_DELETED, /* mcap_mdl_event_cb */
|
|
MCAP_MDL_CB_ABORTED, /* mcap_mdl_event_cb */
|
|
MCAP_MDL_CB_REMOTE_CONN_REQ, /* mcap_remote_mdl_conn_req_cb */
|
|
MCAP_MDL_CB_REMOTE_RECONN_REQ /* mcap_remote_mdl_reconn_req_cb */
|
|
} McapMclCb;
|
|
|
|
typedef enum {
|
|
MCL_CONNECTED,
|
|
MCL_PENDING,
|
|
MCL_ACTIVE,
|
|
MCL_IDLE
|
|
} MCLState;
|
|
|
|
typedef enum {
|
|
MCL_ACCEPTOR,
|
|
MCL_INITIATOR
|
|
} MCLRole;
|
|
|
|
typedef enum {
|
|
MCL_AVAILABLE,
|
|
MCL_WAITING_RSP
|
|
} MCAPCtrl;
|
|
|
|
typedef enum {
|
|
MDL_WAITING,
|
|
MDL_CONNECTED,
|
|
MDL_DELETING,
|
|
MDL_CLOSED
|
|
} MDLState;
|
|
|
|
struct mcap_csp;
|
|
struct mcap_mdl_op_cb;
|
|
struct mcap_instance;
|
|
struct mcap_mcl;
|
|
struct mcap_mdl;
|
|
struct sync_info_ind_data;
|
|
|
|
/************ Callbacks ************/
|
|
|
|
/* MDL callbacks */
|
|
|
|
typedef void (* mcap_mdl_event_cb) (struct mcap_mdl *mdl, gpointer data);
|
|
typedef void (* mcap_mdl_operation_conf_cb) (struct mcap_mdl *mdl, uint8_t conf,
|
|
GError *err, gpointer data);
|
|
typedef void (* mcap_mdl_operation_cb) (struct mcap_mdl *mdl, GError *err,
|
|
gpointer data);
|
|
typedef void (* mcap_mdl_notify_cb) (GError *err, gpointer data);
|
|
|
|
/* Next function should return an MCAP appropriate response code */
|
|
typedef uint8_t (* mcap_remote_mdl_conn_req_cb) (struct mcap_mcl *mcl,
|
|
uint8_t mdepid, uint16_t mdlid,
|
|
uint8_t *conf, gpointer data);
|
|
typedef uint8_t (* mcap_remote_mdl_reconn_req_cb) (struct mcap_mdl *mdl,
|
|
gpointer data);
|
|
|
|
/* MCL callbacks */
|
|
|
|
typedef void (* mcap_mcl_event_cb) (struct mcap_mcl *mcl, gpointer data);
|
|
typedef void (* mcap_mcl_connect_cb) (struct mcap_mcl *mcl, GError *err,
|
|
gpointer data);
|
|
|
|
/* CSP callbacks */
|
|
|
|
typedef void (* mcap_info_ind_event_cb) (struct mcap_mcl *mcl,
|
|
struct sync_info_ind_data *data);
|
|
|
|
typedef void (* mcap_sync_cap_cb) (struct mcap_mcl *mcl,
|
|
uint8_t mcap_err,
|
|
uint8_t btclockres,
|
|
uint16_t synclead,
|
|
uint16_t tmstampres,
|
|
uint16_t tmstampacc,
|
|
GError *err,
|
|
gpointer data);
|
|
|
|
typedef void (* mcap_sync_set_cb) (struct mcap_mcl *mcl,
|
|
uint8_t mcap_err,
|
|
uint32_t btclock,
|
|
uint64_t timestamp,
|
|
uint16_t accuracy,
|
|
GError *err,
|
|
gpointer data);
|
|
|
|
struct mcap_mdl_cb {
|
|
mcap_mdl_event_cb mdl_connected; /* Remote device has created a MDL */
|
|
mcap_mdl_event_cb mdl_closed; /* Remote device has closed a MDL */
|
|
mcap_mdl_event_cb mdl_deleted; /* Remote device requested deleting a MDL */
|
|
mcap_mdl_event_cb mdl_aborted; /* Remote device aborted the mdl creation */
|
|
mcap_remote_mdl_conn_req_cb mdl_conn_req; /* Remote device requested creating a MDL */
|
|
mcap_remote_mdl_reconn_req_cb mdl_reconn_req; /* Remote device requested reconnecting a MDL */
|
|
gpointer user_data; /* User data */
|
|
};
|
|
|
|
struct mcap_instance {
|
|
bdaddr_t src; /* Source address */
|
|
GIOChannel *ccio; /* Control Channel IO */
|
|
GIOChannel *dcio; /* Data Channel IO */
|
|
GSList *mcls; /* MCAP instance list */
|
|
GSList *cached; /* List with all cached MCLs (MAX_CACHED macro) */
|
|
BtIOSecLevel sec; /* Security level */
|
|
mcap_mcl_event_cb mcl_connected_cb; /* New MCL connected */
|
|
mcap_mcl_event_cb mcl_reconnected_cb; /* Old MCL has been reconnected */
|
|
mcap_mcl_event_cb mcl_disconnected_cb; /* MCL disconnected */
|
|
mcap_mcl_event_cb mcl_uncached_cb; /* MCL has been removed from MCAP cache */
|
|
mcap_info_ind_event_cb mcl_sync_infoind_cb; /* (CSP Master) Received info indication */
|
|
gpointer user_data; /* Data to be provided in callbacks */
|
|
int ref; /* Reference counter */
|
|
|
|
gboolean csp_enabled; /* CSP: functionality enabled */
|
|
};
|
|
|
|
struct mcap_mcl {
|
|
struct mcap_instance *mi; /* MCAP instance where this MCL belongs */
|
|
bdaddr_t addr; /* Device address */
|
|
GIOChannel *cc; /* MCAP Control Channel IO */
|
|
guint wid; /* MCL Watcher id */
|
|
GSList *mdls; /* List of Data Channels shorted by mdlid */
|
|
MCLState state; /* Current MCL State */
|
|
MCLRole role; /* Initiator or acceptor of this MCL */
|
|
MCAPCtrl req; /* Request control flag */
|
|
struct mcap_mdl_op_cb *priv_data; /* Temporal data to manage responses */
|
|
struct mcap_mdl_cb *cb; /* MDL callbacks */
|
|
guint tid; /* Timer id for waiting for a response */
|
|
uint8_t *lcmd; /* Last command sent */
|
|
int ref; /* References counter */
|
|
uint8_t ctrl; /* MCL control flag */
|
|
uint16_t next_mdl; /* id used to create next MDL */
|
|
struct mcap_csp *csp; /* CSP control structure */
|
|
};
|
|
|
|
struct mcap_mdl {
|
|
struct mcap_mcl *mcl; /* MCL where this MDL belongs */
|
|
GIOChannel *dc; /* MCAP Data Channel IO */
|
|
guint wid; /* MDL Watcher id */
|
|
uint16_t mdlid; /* MDL id */
|
|
uint8_t mdep_id; /* MCAP Data End Point */
|
|
MDLState state; /* MDL state */
|
|
int ref; /* References counter */
|
|
};
|
|
|
|
struct sync_info_ind_data {
|
|
uint32_t btclock;
|
|
uint64_t timestamp;
|
|
uint16_t accuracy;
|
|
};
|
|
|
|
/************ Operations ************/
|
|
|
|
/* MDL operations */
|
|
|
|
gboolean mcap_create_mdl(struct mcap_mcl *mcl,
|
|
uint8_t mdepid,
|
|
uint8_t conf,
|
|
mcap_mdl_operation_conf_cb connect_cb,
|
|
gpointer user_data,
|
|
GDestroyNotify destroy,
|
|
GError **err);
|
|
gboolean mcap_reconnect_mdl(struct mcap_mdl *mdl,
|
|
mcap_mdl_operation_cb reconnect_cb,
|
|
gpointer user_data,
|
|
GDestroyNotify destroy,
|
|
GError **err);
|
|
gboolean mcap_delete_all_mdls(struct mcap_mcl *mcl,
|
|
mcap_mdl_notify_cb delete_cb,
|
|
gpointer user_data,
|
|
GDestroyNotify destroy,
|
|
GError **err);
|
|
gboolean mcap_delete_mdl(struct mcap_mdl *mdl,
|
|
mcap_mdl_notify_cb delete_cb,
|
|
gpointer user_data,
|
|
GDestroyNotify destroy,
|
|
GError **err);
|
|
gboolean mcap_connect_mdl(struct mcap_mdl *mdl,
|
|
uint8_t mode,
|
|
uint16_t dcpsm,
|
|
mcap_mdl_operation_cb connect_cb,
|
|
gpointer user_data,
|
|
GDestroyNotify destroy,
|
|
GError **err);
|
|
gboolean mcap_mdl_abort(struct mcap_mdl *mdl,
|
|
mcap_mdl_notify_cb abort_cb,
|
|
gpointer user_data,
|
|
GDestroyNotify destroy,
|
|
GError **err);
|
|
|
|
int mcap_mdl_get_fd(struct mcap_mdl *mdl);
|
|
uint16_t mcap_mdl_get_mdlid(struct mcap_mdl *mdl);
|
|
struct mcap_mdl *mcap_mdl_ref(struct mcap_mdl *mdl);
|
|
void mcap_mdl_unref(struct mcap_mdl *mdl);
|
|
|
|
/* MCL operations */
|
|
|
|
gboolean mcap_create_mcl(struct mcap_instance *mi,
|
|
const bdaddr_t *addr,
|
|
uint16_t ccpsm,
|
|
mcap_mcl_connect_cb connect_cb,
|
|
gpointer user_data,
|
|
GDestroyNotify destroy,
|
|
GError **err);
|
|
void mcap_close_mcl(struct mcap_mcl *mcl, gboolean cache);
|
|
gboolean mcap_mcl_set_cb(struct mcap_mcl *mcl, gpointer user_data,
|
|
GError **gerr, McapMclCb cb1, ...);
|
|
void mcap_mcl_get_addr(struct mcap_mcl *mcl, bdaddr_t *addr);
|
|
struct mcap_mcl *mcap_mcl_ref(struct mcap_mcl *mcl);
|
|
void mcap_mcl_unref(struct mcap_mcl *mcl);
|
|
|
|
/* CSP operations */
|
|
|
|
void mcap_enable_csp(struct mcap_instance *mi);
|
|
void mcap_disable_csp(struct mcap_instance *mi);
|
|
uint64_t mcap_get_timestamp(struct mcap_mcl *mcl,
|
|
struct timespec *given_time);
|
|
uint32_t mcap_get_btclock(struct mcap_mcl *mcl);
|
|
|
|
void mcap_sync_cap_req(struct mcap_mcl *mcl,
|
|
uint16_t reqacc,
|
|
mcap_sync_cap_cb cb,
|
|
gpointer user_data,
|
|
GError **err);
|
|
|
|
void mcap_sync_set_req(struct mcap_mcl *mcl,
|
|
uint8_t update,
|
|
uint32_t btclock,
|
|
uint64_t timestamp,
|
|
mcap_sync_set_cb cb,
|
|
gpointer user_data,
|
|
GError **err);
|
|
|
|
/* MCAP main operations */
|
|
|
|
struct mcap_instance *mcap_create_instance(const bdaddr_t *src,
|
|
BtIOSecLevel sec, uint16_t ccpsm,
|
|
uint16_t dcpsm,
|
|
mcap_mcl_event_cb mcl_connected,
|
|
mcap_mcl_event_cb mcl_reconnected,
|
|
mcap_mcl_event_cb mcl_disconnected,
|
|
mcap_mcl_event_cb mcl_uncached,
|
|
mcap_info_ind_event_cb mcl_sync_info_ind,
|
|
gpointer user_data,
|
|
GError **gerr);
|
|
void mcap_release_instance(struct mcap_instance *mi);
|
|
|
|
struct mcap_instance *mcap_instance_ref(struct mcap_instance *mi);
|
|
void mcap_instance_unref(struct mcap_instance *mi);
|
|
|
|
uint16_t mcap_get_ctrl_psm(struct mcap_instance *mi, GError **err);
|
|
uint16_t mcap_get_data_psm(struct mcap_instance *mi, GError **err);
|
|
|
|
gboolean mcap_set_data_chan_mode(struct mcap_instance *mi, uint8_t mode,
|
|
GError **err);
|
|
|
|
int mcap_send_data(int sock, const void *buf, uint32_t size);
|
|
|
|
void proc_sync_cmd(struct mcap_mcl *mcl, uint8_t *cmd, uint32_t len);
|
|
void mcap_sync_init(struct mcap_mcl *mcl);
|
|
void mcap_sync_stop(struct mcap_mcl *mcl);
|