mirror of
https://git.kernel.org/pub/scm/bluetooth/bluez.git
synced 2024-12-18 16:33:30 +08:00
0a259dd05b
This patch adds SPDX License Identifier and removes the license text. ------------------------------------- License COUNT ------------------------------------- GPL-2.0-or-later : 97 LGPL-2.1-or-later : 38 GPL-2.0-only : 2 License: GPL-2.0-or-later tools/l2cap-tester.c tools/hcisecfilter.c tools/ciptool.c tools/btsnoop.c tools/check-selftest.c tools/btpclientctl.c tools/hci-tester.c tools/hcitool.c tools/btiotest.c tools/oobtest.c tools/btinfo.c tools/hwdb.c tools/hciattach_bcm43xx.c tools/mgmt-tester.c tools/hex2hcd.c tools/hciattach_st.c tools/smp-tester.c tools/bluetooth-player.c tools/hciattach_tialt.c tools/gap-tester.c tools/bluemoon.c tools/bneptest.c tools/gatt-service.c tools/rctest.c tools/rfcomm-tester.c tools/hcieventmask.c tools/hciattach_ti.c tools/seq2bseq.c tools/scotest.c tools/bcmfw.c tools/hciconfig.c tools/btattach.c tools/l2ping.c tools/obexctl.c tools/l2test.c tools/hciattach_intel.c tools/hciattach.h tools/create-image.c tools/bnep-tester.c tools/userchan-tester.c tools/rfcomm.c tools/btmon-logger.c tools/hcidump.c tools/rtlfw.c tools/hciattach_qualcomm.c tools/btproxy.c tools/nokfw.c tools/hciattach_ath3k.c tools/3dsp.c tools/bdaddr.c tools/sco-tester.c tools/hciattach.c tools/amptest.c tools/btgatt-server.c tools/btgatt-client.c tools/cltest.c tools/ibeacon.c tools/mcaptest.c tools/hid2hci.c tools/btmgmt.c tools/advtest.c tools/eddystone.c tools/avtest.c tools/mpris-proxy.c tools/avinfo.c tools/sdptool.c tools/btconfig.c tools/update_compids.sh tools/parser/parser.h tools/parser/obex.c tools/parser/amp.c tools/parser/sdp.c tools/parser/tcpip.c tools/parser/sap.c tools/parser/cmtp.c tools/parser/avctp.c tools/parser/lmp.c tools/parser/ppp.c tools/parser/rfcomm.h tools/parser/hci.c tools/parser/sdp.h tools/parser/parser.c tools/parser/rfcomm.c tools/parser/avdtp.c tools/parser/avrcp.c tools/parser/ericsson.c tools/parser/hcrp.c tools/parser/bpa.c tools/parser/hidp.c tools/parser/bnep.c tools/parser/capi.c tools/parser/att.c tools/parser/l2cap.c tools/parser/smp.c tools/parser/csr.c tools/parser/l2cap.h tools/parse_companies.pl License: LGPL-2.1-or-later tools/test-runner.c tools/btpclient.c tools/meshctl.c tools/mesh-cfgclient.c tools/mesh/model.h tools/mesh/util.h tools/mesh/config-model.h tools/mesh/cfgcli.h tools/mesh/mesh-db.c tools/mesh/mesh-db.h tools/mesh/keys.c tools/mesh/util.c tools/mesh/agent.h tools/mesh/remote.c tools/mesh/keys.h tools/mesh/agent.c tools/mesh/cfgcli.c tools/mesh/remote.h tools/mesh-gatt/prov.c tools/mesh-gatt/util.h tools/mesh-gatt/prov.h tools/mesh-gatt/net.c tools/mesh-gatt/util.c tools/mesh-gatt/prov-db.h tools/mesh-gatt/crypto.c tools/mesh-gatt/crypto.h tools/mesh-gatt/gatt.c tools/mesh-gatt/config-server.c tools/mesh-gatt/keys.h tools/mesh-gatt/onoff-model.c tools/mesh-gatt/net.h tools/mesh-gatt/gatt.h tools/mesh-gatt/node.c tools/mesh-gatt/config-client.c tools/mesh-gatt/mesh-net.h tools/mesh-gatt/node.h tools/mesh-gatt/onoff-model.h tools/mesh-gatt/prov-db.c License: GPL-2.0-only tools/obex-server-tool.c tools/obex-client-tool.c
656 lines
14 KiB
C
656 lines
14 KiB
C
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
/*
|
|
*
|
|
* BlueZ - Bluetooth protocol stack for Linux
|
|
*
|
|
* Copyright (C) 2011-2012 Intel Corporation
|
|
* Copyright (C) 2004-2010 Marcel Holtmann <marcel@holtmann.org>
|
|
*
|
|
*
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
|
|
#include <stdio.h>
|
|
#include <errno.h>
|
|
#include <unistd.h>
|
|
#include <alloca.h>
|
|
#include <stdlib.h>
|
|
#include <stdbool.h>
|
|
#include <poll.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/socket.h>
|
|
|
|
#include "lib/bluetooth.h"
|
|
#include "lib/hci.h"
|
|
#include "lib/hci_lib.h"
|
|
|
|
static int activate_amp_controller(int dev_id)
|
|
{
|
|
struct hci_dev_info di;
|
|
struct hci_filter flt;
|
|
int fd;
|
|
|
|
printf("hci%d: Activating controller\n", dev_id);
|
|
|
|
fd = socket(AF_BLUETOOTH, SOCK_RAW, BTPROTO_HCI);
|
|
if (fd < 0) {
|
|
perror("Failed to open raw HCI socket");
|
|
return -1;
|
|
}
|
|
|
|
di.dev_id = dev_id;
|
|
|
|
if (ioctl(fd, HCIGETDEVINFO, (void *) &di) < 0) {
|
|
perror("Failed to get HCI device info");
|
|
close(fd);
|
|
return -1;
|
|
}
|
|
|
|
if (!hci_test_bit(HCI_UP, &di.flags)) {
|
|
if (ioctl(fd, HCIDEVUP, dev_id) < 0) {
|
|
if (errno != EALREADY) {
|
|
perror("Failed to bring up HCI device");
|
|
close(fd);
|
|
return -1;
|
|
}
|
|
}
|
|
}
|
|
|
|
close(fd);
|
|
|
|
fd = hci_open_dev(dev_id);
|
|
if (fd < 0) {
|
|
perror("Failed to open HCI device");
|
|
return -1;
|
|
}
|
|
|
|
hci_filter_clear(&flt);
|
|
hci_filter_set_ptype(HCI_EVENT_PKT, &flt);
|
|
hci_filter_set_event(EVT_CHANNEL_SELECTED, &flt);
|
|
hci_filter_set_event(EVT_PHYSICAL_LINK_COMPLETE, &flt);
|
|
hci_filter_set_event(EVT_DISCONNECT_PHYSICAL_LINK_COMPLETE, &flt);
|
|
|
|
if (setsockopt(fd, SOL_HCI, HCI_FILTER, &flt, sizeof(flt)) < 0) {
|
|
perror("Failed to setup HCI device filter");
|
|
close(fd);
|
|
return -1;
|
|
}
|
|
|
|
return fd;
|
|
}
|
|
|
|
static bool read_local_amp_info(int dev_id, uint16_t *max_assoc_len)
|
|
{
|
|
read_local_amp_info_rp rp;
|
|
struct hci_request rq;
|
|
int fd;
|
|
|
|
printf("hci%d: Reading local AMP information\n", dev_id);
|
|
|
|
fd = hci_open_dev(dev_id);
|
|
if (fd < 0) {
|
|
perror("Failed to open HCI device");
|
|
return false;
|
|
}
|
|
|
|
memset(&rp, 0, sizeof(rp));
|
|
|
|
memset(&rq, 0, sizeof(rq));
|
|
rq.ogf = OGF_STATUS_PARAM;
|
|
rq.ocf = OCF_READ_LOCAL_AMP_INFO;
|
|
rq.rparam = &rp;
|
|
rq.rlen = READ_LOCAL_AMP_INFO_RP_SIZE;
|
|
|
|
if (hci_send_req(fd, &rq, 1000) < 0) {
|
|
perror("Failed sending HCI request");
|
|
hci_close_dev(fd);
|
|
return false;
|
|
}
|
|
|
|
if (rp.status) {
|
|
fprintf(stderr, "Failed HCI command: 0x%02x\n", rp.status);
|
|
hci_close_dev(fd);
|
|
return false;
|
|
}
|
|
|
|
printf("\tAMP status: 0x%02x\n", rp.amp_status);
|
|
printf("\tController type: 0x%02x\n", rp.controller_type);
|
|
printf("\tMax ASSOC length: %d\n", btohs(rp.max_amp_assoc_length));
|
|
|
|
*max_assoc_len = btohs(rp.max_amp_assoc_length);
|
|
|
|
hci_close_dev(fd);
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool read_local_amp_assoc(int dev_id, uint8_t phy_handle,
|
|
uint16_t max_assoc_len,
|
|
uint8_t *assoc_data,
|
|
uint16_t *assoc_len)
|
|
{
|
|
read_local_amp_assoc_cp cp;
|
|
read_local_amp_assoc_rp rp;
|
|
struct hci_request rq;
|
|
int fd;
|
|
|
|
printf("hci%d: Reading local AMP association\n", dev_id);
|
|
|
|
fd = hci_open_dev(dev_id);
|
|
if (fd < 0) {
|
|
perror("Failed to open HCI device");
|
|
return false;
|
|
}
|
|
|
|
memset(&cp, 0, sizeof(cp));
|
|
cp.handle = phy_handle;
|
|
cp.length_so_far = htobs(0);
|
|
cp.assoc_length = htobs(max_assoc_len);
|
|
memset(&rp, 0, sizeof(rp));
|
|
|
|
memset(&rq, 0, sizeof(rq));
|
|
rq.ogf = OGF_STATUS_PARAM;
|
|
rq.ocf = OCF_READ_LOCAL_AMP_ASSOC;
|
|
rq.cparam = &cp;
|
|
rq.clen = READ_LOCAL_AMP_ASSOC_CP_SIZE;
|
|
rq.rparam = &rp;
|
|
rq.rlen = READ_LOCAL_AMP_ASSOC_RP_SIZE;
|
|
|
|
if (hci_send_req(fd, &rq, 1000) < 0) {
|
|
perror("Failed sending HCI request");
|
|
hci_close_dev(fd);
|
|
return false;
|
|
}
|
|
|
|
if (rp.status) {
|
|
fprintf(stderr, "Failed HCI command: 0x%02x\n", rp.status);
|
|
hci_close_dev(fd);
|
|
return false;
|
|
}
|
|
|
|
printf("\tRemain ASSOC length: %d\n", btohs(rp.length));
|
|
|
|
*assoc_len = btohs(rp.length);
|
|
memcpy(assoc_data, rp.fragment, *assoc_len);
|
|
|
|
hci_close_dev(fd);
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool write_remote_amp_assoc(int dev_id, uint8_t phy_handle,
|
|
uint8_t *assoc_data,
|
|
uint16_t assoc_len)
|
|
{
|
|
write_remote_amp_assoc_cp cp;
|
|
write_remote_amp_assoc_rp rp;
|
|
struct hci_request rq;
|
|
int fd;
|
|
|
|
printf("hci%d: Writing remote AMP association\n", dev_id);
|
|
|
|
fd = hci_open_dev(dev_id);
|
|
if (fd < 0) {
|
|
perror("Failed to open HCI device");
|
|
return false;
|
|
}
|
|
|
|
memset(&cp, 0, sizeof(cp));
|
|
cp.handle = phy_handle;
|
|
cp.length_so_far = htobs(0);
|
|
cp.remaining_length = htobs(assoc_len);
|
|
memcpy(cp.fragment, assoc_data, assoc_len);
|
|
memset(&rp, 0, sizeof(rp));
|
|
|
|
memset(&rq, 0, sizeof(rq));
|
|
rq.ogf = OGF_STATUS_PARAM;
|
|
rq.ocf = OCF_WRITE_REMOTE_AMP_ASSOC;
|
|
rq.cparam = &cp;
|
|
rq.clen = 5 + assoc_len;
|
|
rq.rparam = &rp;
|
|
rq.rlen = WRITE_REMOTE_AMP_ASSOC_RP_SIZE;
|
|
|
|
if (hci_send_req(fd, &rq, 1000) < 0) {
|
|
perror("Failed sending HCI request");
|
|
hci_close_dev(fd);
|
|
return false;
|
|
}
|
|
|
|
if (rp.status) {
|
|
fprintf(stderr, "Failed HCI command: 0x%02x\n", rp.status);
|
|
hci_close_dev(fd);
|
|
return false;
|
|
}
|
|
|
|
hci_close_dev(fd);
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool channel_selected_event(int dev_id, int fd, uint8_t phy_handle)
|
|
{
|
|
printf("hci%d: Waiting for channel selected event\n", dev_id);
|
|
|
|
while (1) {
|
|
uint8_t buf[HCI_MAX_EVENT_SIZE];
|
|
hci_event_hdr *hdr;
|
|
struct pollfd p;
|
|
int n, len;
|
|
|
|
p.fd = fd;
|
|
p.events = POLLIN;
|
|
|
|
n = poll(&p, 1, 10000);
|
|
if (n < 0) {
|
|
if (errno == EAGAIN || errno == EINTR)
|
|
continue;
|
|
|
|
perror("Failed to poll HCI device");
|
|
return false;
|
|
}
|
|
|
|
if (n == 0) {
|
|
fprintf(stderr, "Failure to receive event\n");
|
|
return false;
|
|
}
|
|
|
|
len = read(fd, buf, sizeof(buf));
|
|
if (len < 0) {
|
|
if (errno == EAGAIN || errno == EINTR)
|
|
continue;
|
|
|
|
perror("Failed to read from HCI device");
|
|
return false;
|
|
}
|
|
|
|
hdr = (void *) (buf + 1);
|
|
|
|
if (hdr->evt == EVT_CHANNEL_SELECTED)
|
|
break;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool create_physical_link(int dev_id, uint8_t phy_handle)
|
|
{
|
|
create_physical_link_cp cp;
|
|
evt_cmd_status evt;
|
|
struct hci_request rq;
|
|
int i, fd;
|
|
|
|
printf("hci%d: Creating physical link\n", dev_id);
|
|
|
|
fd = hci_open_dev(dev_id);
|
|
if (fd < 0) {
|
|
perror("Failed to open HCI device");
|
|
return false;
|
|
}
|
|
|
|
memset(&cp, 0, sizeof(cp));
|
|
cp.handle = phy_handle;
|
|
cp.key_length = 32;
|
|
cp.key_type = 0x03;
|
|
for (i = 0; i < cp.key_length; i++)
|
|
cp.key[i] = 0x23;
|
|
memset(&evt, 0, sizeof(evt));
|
|
|
|
memset(&rq, 0, sizeof(rq));
|
|
rq.ogf = OGF_LINK_CTL;
|
|
rq.ocf = OCF_CREATE_PHYSICAL_LINK;
|
|
rq.event = EVT_CMD_STATUS;
|
|
rq.cparam = &cp;
|
|
rq.clen = CREATE_PHYSICAL_LINK_CP_SIZE;
|
|
rq.rparam = &evt;
|
|
rq.rlen = EVT_CMD_STATUS_SIZE;
|
|
|
|
if (hci_send_req(fd, &rq, 1000) < 0) {
|
|
perror("Failed sending HCI request");
|
|
hci_close_dev(fd);
|
|
return false;
|
|
}
|
|
|
|
if (evt.status) {
|
|
fprintf(stderr, "Failed HCI command: 0x%02x\n", evt.status);
|
|
hci_close_dev(fd);
|
|
return false;
|
|
}
|
|
|
|
hci_close_dev(fd);
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool accept_physical_link(int dev_id, uint8_t phy_handle)
|
|
{
|
|
accept_physical_link_cp cp;
|
|
evt_cmd_status evt;
|
|
struct hci_request rq;
|
|
int i, fd;
|
|
|
|
printf("hci%d: Accepting physical link\n", dev_id);
|
|
|
|
fd = hci_open_dev(dev_id);
|
|
if (fd < 0) {
|
|
perror("Failed to open HCI device");
|
|
return false;
|
|
}
|
|
|
|
memset(&cp, 0, sizeof(cp));
|
|
cp.handle = phy_handle;
|
|
cp.key_length = 32;
|
|
cp.key_type = 0x03;
|
|
for (i = 0; i < cp.key_length; i++)
|
|
cp.key[i] = 0x23;
|
|
memset(&evt, 0, sizeof(evt));
|
|
|
|
memset(&rq, 0, sizeof(rq));
|
|
rq.ogf = OGF_LINK_CTL;
|
|
rq.ocf = OCF_ACCEPT_PHYSICAL_LINK;
|
|
rq.event = EVT_CMD_STATUS;
|
|
rq.cparam = &cp;
|
|
rq.clen = ACCEPT_PHYSICAL_LINK_CP_SIZE;
|
|
rq.rparam = &evt;
|
|
rq.rlen = EVT_CMD_STATUS_SIZE;
|
|
|
|
if (hci_send_req(fd, &rq, 1000) < 0) {
|
|
perror("Failed sending HCI request");
|
|
hci_close_dev(fd);
|
|
return false;
|
|
}
|
|
|
|
if (evt.status) {
|
|
fprintf(stderr, "Failed HCI command: 0x%02x\n", evt.status);
|
|
hci_close_dev(fd);
|
|
return false;
|
|
}
|
|
|
|
hci_close_dev(fd);
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool disconnect_physical_link(int dev_id, uint8_t phy_handle,
|
|
uint8_t reason)
|
|
{
|
|
disconnect_physical_link_cp cp;
|
|
evt_cmd_status evt;
|
|
struct hci_request rq;
|
|
int fd;
|
|
|
|
printf("hci%d: Disconnecting physical link\n", dev_id);
|
|
|
|
fd = hci_open_dev(dev_id);
|
|
if (fd < 0) {
|
|
perror("Failed to open HCI device");
|
|
return false;
|
|
}
|
|
|
|
memset(&cp, 0, sizeof(cp));
|
|
cp.handle = phy_handle;
|
|
cp.reason = reason;
|
|
|
|
memset(&rq, 0, sizeof(rq));
|
|
rq.ogf = OGF_LINK_CTL;
|
|
rq.ocf = OCF_DISCONNECT_PHYSICAL_LINK;
|
|
rq.event = EVT_CMD_STATUS;
|
|
rq.cparam = &cp;
|
|
rq.clen = DISCONNECT_PHYSICAL_LINK_CP_SIZE;
|
|
rq.rparam = &evt;
|
|
rq.rlen = EVT_CMD_STATUS_SIZE;
|
|
|
|
if (hci_send_req(fd, &rq, 1000) < 0) {
|
|
perror("Failed sending HCI request");
|
|
hci_close_dev(fd);
|
|
return false;
|
|
}
|
|
|
|
if (evt.status) {
|
|
fprintf(stderr, "Failed HCI command: 0x%02x\n", evt.status);
|
|
hci_close_dev(fd);
|
|
return false;
|
|
}
|
|
|
|
hci_close_dev(fd);
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool physical_link_complete_event(int dev_id, int fd,
|
|
uint8_t phy_handle)
|
|
{
|
|
printf("hci%d: Waiting for physical link complete event\n", dev_id);
|
|
|
|
while (1) {
|
|
uint8_t buf[HCI_MAX_EVENT_SIZE];
|
|
hci_event_hdr *hdr;
|
|
int len;
|
|
|
|
len = read(fd, buf, sizeof(buf));
|
|
if (len < 0) {
|
|
if (errno == EAGAIN || errno == EINTR)
|
|
continue;
|
|
|
|
perror("Failed to read from HCI device");
|
|
return false;
|
|
}
|
|
|
|
hdr = (void *) (buf + 1);
|
|
|
|
if (hdr->evt == EVT_PHYSICAL_LINK_COMPLETE)
|
|
break;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool disconnect_physical_link_complete_event(int dev_id, int fd,
|
|
uint8_t phy_handle)
|
|
{
|
|
printf("hci%d: Waiting for physical link disconnect event\n", dev_id);
|
|
|
|
while (1) {
|
|
uint8_t buf[HCI_MAX_EVENT_SIZE];
|
|
hci_event_hdr *hdr;
|
|
int len;
|
|
|
|
len = read(fd, buf, sizeof(buf));
|
|
if (len < 0) {
|
|
if (errno == EAGAIN || errno == EINTR)
|
|
continue;
|
|
|
|
perror("Failed to read from HCI device");
|
|
return false;
|
|
}
|
|
|
|
hdr = (void *) (buf + 1);
|
|
|
|
if (hdr->evt == EVT_DISCONNECT_PHYSICAL_LINK_COMPLETE)
|
|
break;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static int amp1_dev_id = -1;
|
|
static int amp2_dev_id = -1;
|
|
|
|
static bool find_amp_controller(void)
|
|
{
|
|
struct hci_dev_list_req *dl;
|
|
struct hci_dev_req *dr;
|
|
int fd, i;
|
|
bool result;
|
|
|
|
fd = socket(AF_BLUETOOTH, SOCK_RAW, BTPROTO_HCI);
|
|
if (fd < 0) {
|
|
perror("Failed to open raw HCI socket");
|
|
return false;
|
|
}
|
|
|
|
dl = malloc(HCI_MAX_DEV * sizeof(struct hci_dev_req) + sizeof(uint16_t));
|
|
if (!dl) {
|
|
perror("Failed allocate HCI device request memory");
|
|
close(fd);
|
|
return false;
|
|
}
|
|
|
|
dl->dev_num = HCI_MAX_DEV;
|
|
dr = dl->dev_req;
|
|
|
|
if (ioctl(fd, HCIGETDEVLIST, (void *) dl) < 0) {
|
|
perror("Failed to get HCI device list");
|
|
result = false;
|
|
goto done;
|
|
}
|
|
|
|
for (i = 0; i< dl->dev_num; i++) {
|
|
struct hci_dev_info di;
|
|
|
|
di.dev_id = (dr + i)->dev_id;
|
|
|
|
if (ioctl(fd, HCIGETDEVINFO, (void *) &di) < 0)
|
|
continue;
|
|
|
|
if (((di.type & 0x30) >> 4) != HCI_AMP)
|
|
continue;
|
|
|
|
if (amp1_dev_id < 0)
|
|
amp1_dev_id = di.dev_id;
|
|
else if (amp2_dev_id < 0) {
|
|
if (di.dev_id < amp1_dev_id) {
|
|
amp2_dev_id = amp1_dev_id;
|
|
amp1_dev_id = di.dev_id;
|
|
} else
|
|
amp2_dev_id = di.dev_id;
|
|
}
|
|
}
|
|
|
|
result = true;
|
|
|
|
done:
|
|
free(dl);
|
|
close(fd);
|
|
return result;
|
|
}
|
|
|
|
int main(int argc ,char *argv[])
|
|
{
|
|
int amp1_event_fd, amp2_event_fd;
|
|
uint16_t amp1_max_assoc_len, amp2_max_assoc_len;
|
|
uint8_t *amp1_assoc_data, *amp2_assoc_data;
|
|
uint16_t amp1_assoc_len, amp2_assoc_len;
|
|
uint8_t amp1_phy_handle, amp2_phy_handle;
|
|
|
|
if (!find_amp_controller())
|
|
return EXIT_FAILURE;
|
|
|
|
if (amp1_dev_id < 0 || amp2_dev_id < 0) {
|
|
fprintf(stderr, "Two AMP controllers are required\n");
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
printf("hci%d: AMP initiator\n", amp1_dev_id);
|
|
printf("hci%d: AMP acceptor\n", amp2_dev_id);
|
|
|
|
amp1_event_fd = activate_amp_controller(amp1_dev_id);
|
|
if (amp1_event_fd < 0)
|
|
return EXIT_FAILURE;
|
|
|
|
amp2_event_fd = activate_amp_controller(amp2_dev_id);
|
|
if (amp2_event_fd < 0) {
|
|
hci_close_dev(amp1_event_fd);
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
if (!read_local_amp_info(amp1_dev_id, &1_max_assoc_len))
|
|
return EXIT_FAILURE;
|
|
|
|
amp1_assoc_data = alloca(amp1_max_assoc_len);
|
|
|
|
printf("--> AMP_Get_Info_Request (Amp_ID B)\n");
|
|
|
|
if (!read_local_amp_info(amp2_dev_id, &2_max_assoc_len))
|
|
return EXIT_FAILURE;
|
|
|
|
amp2_assoc_data = alloca(amp2_max_assoc_len);
|
|
|
|
printf("<-- AMP_Get_Info_Response (Amp_ID B, Status)\n");
|
|
|
|
printf("--> AMP_Get_AMP_Assoc_Request (Amp_ID B)\n");
|
|
|
|
if (!read_local_amp_assoc(amp2_dev_id, 0x00, amp2_max_assoc_len,
|
|
amp2_assoc_data, &2_assoc_len))
|
|
return EXIT_FAILURE;
|
|
|
|
printf("<-- AMP_Get_AMP_Assoc_Response (Amp_ID B, AMP_Assoc B)\n");
|
|
|
|
amp1_phy_handle = 0x04;
|
|
|
|
if (!create_physical_link(amp1_dev_id, amp1_phy_handle))
|
|
return EXIT_FAILURE;
|
|
|
|
if (!write_remote_amp_assoc(amp1_dev_id, amp1_phy_handle,
|
|
amp2_assoc_data, amp2_assoc_len))
|
|
return EXIT_FAILURE;
|
|
|
|
printf("hci%d: Signal MAC to scan\n", amp1_dev_id);
|
|
|
|
printf("hci%d: Signal MAC to start\n", amp1_dev_id);
|
|
|
|
if (!channel_selected_event(amp1_dev_id, amp1_event_fd,
|
|
amp1_phy_handle))
|
|
return EXIT_FAILURE;
|
|
|
|
if (!read_local_amp_assoc(amp1_dev_id, amp1_phy_handle,
|
|
amp1_max_assoc_len,
|
|
amp1_assoc_data, &1_assoc_len))
|
|
return EXIT_FAILURE;
|
|
|
|
printf("--> AMP_Create_Physical_Link_Request (Remote-Amp-ID B, AMP_Assoc A)\n");
|
|
|
|
amp2_phy_handle = 0x05;
|
|
|
|
if (!accept_physical_link(amp2_dev_id, amp2_phy_handle))
|
|
return EXIT_FAILURE;
|
|
|
|
if (!write_remote_amp_assoc(amp2_dev_id, amp2_phy_handle,
|
|
amp1_assoc_data, amp1_assoc_len))
|
|
return EXIT_FAILURE;
|
|
|
|
printf("hci%d: Signal MAC to start\n", amp2_dev_id);
|
|
|
|
printf("<-- AMP_Create_Physical_Link_Response (Local-Amp-ID B, Status)\n");
|
|
|
|
if (!physical_link_complete_event(amp2_dev_id, amp2_event_fd,
|
|
amp2_phy_handle))
|
|
return EXIT_FAILURE;
|
|
|
|
if (!physical_link_complete_event(amp1_dev_id, amp1_event_fd,
|
|
amp1_phy_handle))
|
|
return EXIT_FAILURE;
|
|
|
|
/* physical link established */
|
|
|
|
if (!disconnect_physical_link(amp1_dev_id, amp1_phy_handle, 0x13))
|
|
return EXIT_FAILURE;
|
|
|
|
if (!disconnect_physical_link_complete_event(amp1_dev_id,
|
|
amp1_event_fd,
|
|
amp1_phy_handle))
|
|
return EXIT_FAILURE;
|
|
|
|
if (!disconnect_physical_link_complete_event(amp2_dev_id,
|
|
amp2_event_fd,
|
|
amp2_phy_handle))
|
|
return EXIT_FAILURE;
|
|
|
|
hci_close_dev(amp2_event_fd);
|
|
hci_close_dev(amp1_event_fd);
|
|
|
|
return EXIT_SUCCESS;
|
|
}
|