diff --git a/lib/mgmt.h b/lib/mgmt.h
index f784dca..6a39764 100644
--- a/lib/mgmt.h
+++ b/lib/mgmt.h
#define DEVICE_FLAG_REMOTE_WAKEUP BIT(0)
#define DEVICE_FLAG_DEVICE_PRIVACY BIT(1)
+#define DEVICE_FLAG_ADDRESS_RESOLUTION BIT(2)
#define MGMT_OP_SET_DEVICE_FLAGS 0x0050
#define MGMT_SET_DEVICE_FLAGS_SIZE 11
diff --git a/src/adapter.c b/src/adapter.c
index f422bba..5d4117a 100644
--- a/src/adapter.c
+++ b/src/adapter.c
if (btd_opts.device_privacy && !(flags & DEVICE_FLAG_DEVICE_PRIVACY))
flags |= DEVICE_FLAG_DEVICE_PRIVACY & supported & ~pending;
+ /* Set Address Resolution if it has not been set the flag yet. */
+ if (btd_opts.defaults.le.addr_resolution &&
+ device_address_is_private(device) &&
+ !(flags & DEVICE_FLAG_ADDRESS_RESOLUTION))
+ flags |= DEVICE_FLAG_ADDRESS_RESOLUTION & supported & ~pending;
+
bdaddr = device_get_address(device);
bdaddr_type = btd_device_get_bdaddr_type(device);
diff --git a/src/btd.h b/src/btd.h
index a443f66..18a5eb8 100644
--- a/src/btd.h
+++ b/src/btd.h
};
struct btd_le_defaults {
+ uint8_t addr_resolution;
+
uint16_t min_adv_interval;
uint16_t max_adv_interval;
uint16_t adv_rotation_interval;
diff --git a/src/device.c b/src/device.c
index 2b3d19f..2d3ac71 100644
--- a/src/device.c
+++ b/src/device.c
return FALSE;
}
-static bool device_address_is_private(struct btd_device *dev)
+bool device_address_is_private(struct btd_device *dev)
{
if (dev->bdaddr_type != BDADDR_LE_RANDOM)
return false;
store_device_info(device);
}
+bool btd_device_flags_enabled(struct btd_device *dev, uint32_t flags)
+{
+ const char *ll_privacy = "15c0a148-c273-11ea-b3de-0242ac130004";
+
+ if (!dev)
+ return false;
+
+ if (dev->current_flags & flags)
+ return true;
+
+ /* For backward compatibility check for LL Privacy experimental UUID
+ * since that shall be equivalent to DEVICE_FLAG_ADDRESS_RESOLUTION on
+ * older kernels.
+ */
+ if ((flags & DEVICE_FLAG_ADDRESS_RESOLUTION) &&
+ btd_kernel_experimental_enabled(ll_privacy))
+ return true;
+
+ return false;
+}
+
uint32_t btd_device_get_current_flags(struct btd_device *dev)
{
return dev->current_flags;
diff --git a/src/device.h b/src/device.h
index 9753677..2e4a977 100644
--- a/src/device.h
+++ b/src/device.h
bool device_is_name_resolve_allowed(struct btd_device *device);
void device_name_resolve_fail(struct btd_device *device);
void device_set_class(struct btd_device *device, uint32_t class);
+bool device_address_is_private(struct btd_device *dev);
void device_set_rpa(struct btd_device *device, bool value);
void device_update_addr(struct btd_device *device, const bdaddr_t *bdaddr,
uint8_t bdaddr_type);
int device_discover_services(struct btd_device *device);
int btd_device_connect_services(struct btd_device *dev, GSList *services);
+bool btd_device_flags_enabled(struct btd_device *dev, uint32_t flags);
uint32_t btd_device_get_current_flags(struct btd_device *dev);
uint32_t btd_device_get_supported_flags(struct btd_device *dev);
uint32_t btd_device_get_pending_flags(struct btd_device *dev);
diff --git a/src/gatt-database.c b/src/gatt-database.c
index a86e528..a5a01ad 100644
--- a/src/gatt-database.c
+++ b/src/gatt-database.c
#include "lib/sdp.h"
#include "lib/sdp_lib.h"
#include "lib/uuid.h"
+#include "lib/mgmt.h"
#include "btio/btio.h"
#include "gdbus/gdbus.h"
#include "src/shared/util.h"
uint8_t opcode, struct bt_att *att,
void *user_data)
{
- uint8_t value = 0x01;
+ uint8_t value = 0x00;
DBG("GAP Central Address Resolution read request\n");
+ if (btd_opts.defaults.le.addr_resolution) {
+ struct btd_device *device;
+
+ device = btd_adapter_find_device_by_fd(bt_att_get_fd(att));
+ if (device)
+ value = btd_device_flags_enabled(device,
+ DEVICE_FLAG_ADDRESS_RESOLUTION);
+ }
+
gatt_db_attribute_read_result(attrib, id, 0, &value, sizeof(value));
}
diff --git a/src/main.c b/src/main.c
index b92b22e..d13ada2 100644
--- a/src/main.c
+++ b/src/main.c
};
static const char *le_options[] = {
+ "CentralAddressResolution",
"MinAdvertisementInterval",
"MaxAdvertisementInterval",
"MultiAdvertisementRotationInterval",
static void parse_le_config(GKeyFile *config)
{
static const struct config_param params[] = {
+ { "CentralAddressResolution",
+ &btd_opts.defaults.le.addr_resolution,
+ sizeof(btd_opts.defaults.le.addr_resolution),
+ 0,
+ 1},
{ "MinAdvertisementInterval",
&btd_opts.defaults.le.min_adv_interval,
sizeof(btd_opts.defaults.le.min_adv_interval),
bool btd_kernel_experimental_enabled(const char *uuid)
{
if (!btd_opts.kernel)
- false;
+ goto done;
+
+ if (queue_find(btd_opts.kernel, match_experimental, uuid))
+ return true;
+
+done:
+ /* For backward compatibility set LL Privacy as enabled if
+ * CentralAddressResolution has been set so old kernel LL Privacy is
+ * enabled.
+ */
+ if (!strcmp(uuid, "15c0a148-c273-11ea-b3de-0242ac130004") &&
+ btd_opts.defaults.le.addr_resolution)
+ return true;
- return queue_find(btd_opts.kernel, match_experimental, uuid);
+ return false;
}
static const char *valid_uuids[] = {
btd_opts.defaults.num_entries = 0;
btd_opts.defaults.br.page_scan_type = 0xFFFF;
btd_opts.defaults.br.scan_type = 0xFFFF;
+ btd_opts.defaults.le.addr_resolution = 0x01;
btd_opts.defaults.le.enable_advmon_interleave_scan = 0xFF;
if (sscanf(VERSION, "%hhu.%hhu", &major, &minor) != 2)
diff --git a/src/main.conf b/src/main.conf
index 80e72e1..60ab60e 100644
--- a/src/main.conf
+++ b/src/main.conf
#MaxSniffInterval=
[LE]
+# Enable/Disable Central Address Resolution.
+# 0: disable
+# 1: enable
+# Defaults to 1
+#CentralAddressResolution = 1
+
# The following values are used to load default adapter parameters for LE.
# BlueZ loads the values into the kernel before the adapter is powered if the
# kernel supports the MGMT_LOAD_DEFAULT_PARAMETERS command. If a value isn't