summaryrefslogtreecommitdiffstats
path: root/src/bluetooth/osx/osxbtcentralmanager.mm
diff options
context:
space:
mode:
authorTimur Pocheptsov <Timur.Pocheptsov@digia.com>2014-11-07 14:00:40 +0100
committerTimur Pocheptsov <Timur.Pocheptsov@digia.com>2014-11-12 14:32:35 +0100
commit51dbe4a457c0bf81b96c898aab5d83b421e7a4f0 (patch)
treecc1c878a608c0d03ea7c67367548cc7b73322aa8 /src/bluetooth/osx/osxbtcentralmanager.mm
parentbe5883540fe61123da3fb39dd39cfd34eec442b8 (diff)
QLowEnergyController - connectTo/disconnectFrom/Device (OS X, iOS)
Implement connect to device/disconnect from device methods, add CentralManager class dealing with CBCentralManager (with additional external logic). - Improve error handling while in 'connecting' state. - Use the proper error (ConnectionError) instead of UnknownError + make the error handling more uniform while isConnecting == true. - Use isNull on QBluetoothUuid, no need in ugly 'hasUuid'. Change-Id: I84a704a746b4677ccb870b9c132db5f7359030c6 Reviewed-by: Alex Blasche <alexander.blasche@theqtcompany.com>
Diffstat (limited to 'src/bluetooth/osx/osxbtcentralmanager.mm')
-rw-r--r--src/bluetooth/osx/osxbtcentralmanager.mm448
1 files changed, 448 insertions, 0 deletions
diff --git a/src/bluetooth/osx/osxbtcentralmanager.mm b/src/bluetooth/osx/osxbtcentralmanager.mm
new file mode 100644
index 00000000..d2e45832
--- /dev/null
+++ b/src/bluetooth/osx/osxbtcentralmanager.mm
@@ -0,0 +1,448 @@
+/****************************************************************************
+**
+** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies).
+** Contact: http://www.qt-project.org/legal
+**
+** This file is part of the QtBluetooth module of the Qt Toolkit.
+**
+** $QT_BEGIN_LICENSE:LGPL$
+** Commercial License Usage
+** Licensees holding valid commercial Qt licenses may use this file in
+** accordance with the commercial license agreement provided with the
+** Software or, alternatively, in accordance with the terms contained in
+** a written agreement between you and Digia. For licensing terms and
+** conditions see http://qt.digia.com/licensing. For further information
+** use the contact form at http://qt.digia.com/contact-us.
+**
+** GNU Lesser General Public License Usage
+** Alternatively, this file may be used under the terms of the GNU Lesser
+** General Public License version 2.1 as published by the Free Software
+** Foundation and appearing in the file LICENSE.LGPL included in the
+** packaging of this file. Please review the following information to
+** ensure the GNU Lesser General Public License version 2.1 requirements
+** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
+**
+** In addition, as a special exception, Digia gives you certain additional
+** rights. These rights are described in the Digia Qt LGPL Exception
+** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
+**
+** GNU General Public License Usage
+** Alternatively, this file may be used under the terms of the GNU
+** General Public License version 3.0 as published by the Free Software
+** Foundation and appearing in the file LICENSE.GPL included in the
+** packaging of this file. Please review the following information to
+** ensure the GNU General Public License version 3.0 requirements will be
+** met: http://www.gnu.org/copyleft/gpl.html.
+**
+**
+** $QT_END_LICENSE$
+**
+****************************************************************************/
+
+#include "osxbtcentralmanagerdelegate_p.h"
+#include "osxbtcentralmanager_p.h"
+
+#include <QtCore/qloggingcategory.h>
+#include <QtCore/qdebug.h>
+
+#include <algorithm>
+
+QT_BEGIN_NAMESPACE
+
+namespace OSXBluetooth {
+
+CentralManagerDelegate::~CentralManagerDelegate()
+{
+}
+
+}
+
+QT_END_NAMESPACE
+
+
+#ifdef QT_NAMESPACE
+using namespace QT_NAMESPACE;
+#endif
+
+@interface QT_MANGLE_NAMESPACE(OSXBTCentralManager) (PrivateAPI)
+
+- (QLowEnergyController::Error)connectToDevice; // "Device" is in Qt's world ...
+- (void)connectToPeripheral; // "Peripheral" is in Core Bluetooth.
+- (bool)connected;
+
+@end
+
+@implementation QT_MANGLE_NAMESPACE(OSXBTCentralManager)
+
+- (id)initWithDelegate:(QT_PREPEND_NAMESPACE(OSXBluetooth)::CentralManagerDelegate *)aDelegate
+{
+ Q_ASSERT_X(aDelegate, "-initWithDelegate:", "invalid delegate (null)");
+
+ if (self = [super init]) {
+ manager = nil;
+ managerState = OSXBluetooth::CentralManagerIdle;
+ disconnectPending = false;
+ peripheral = nil;
+ delegate = aDelegate;
+ }
+
+ return self;
+}
+
+- (void)dealloc
+{
+ typedef QT_MANGLE_NAMESPACE(OSXBTCentralManagerTransientDelegate) TransientDelegate;
+
+ if (managerState == OSXBluetooth::CentralManagerUpdating) {
+ // Here we have to trick with a transient delegate not to
+ // delete the manager too early - or Core Bluetooth will crash.
+ // State was not updated yet, too early to release.
+ TransientDelegate *const transient = [[TransientDelegate alloc] initWithManager:manager];
+ // On ARC the lifetime of a transient delegate will become a problem, since delegate itself
+ // is a weak reference in a manager.
+ [manager setDelegate:transient];
+ } else {
+ [manager setDelegate:nil];
+ [manager release];
+ }
+
+ [peripheral setDelegate:nil];
+ [peripheral release];
+
+ [super dealloc];
+}
+
+- (QLowEnergyController::Error)connectToDevice:(const QBluetoothUuid &)aDeviceUuid
+{
+ Q_ASSERT_X(managerState == OSXBluetooth::CentralManagerIdle, "-connectToDevice",
+ "invalid state"); // QLowEnergyController connects only if in UnconnectedState.
+
+ deviceUuid = aDeviceUuid;
+
+ if (!manager) {
+ managerState = OSXBluetooth::CentralManagerUpdating; // We'll have to wait for updated state.
+ manager = [[CBCentralManager alloc] initWithDelegate:self queue:nil];
+ if (!manager) {
+ managerState = OSXBluetooth::CentralManagerIdle;
+ qCWarning(QT_BT_OSX) << "-connectToDevice:, failed to allocate a "
+ "central manager";
+ return QLowEnergyController::ConnectionError;
+ }
+ return QLowEnergyController::NoError;
+ } else {
+ return [self connectToDevice];
+ }
+}
+
+- (QLowEnergyController::Error)connectToDevice
+{
+ Q_ASSERT_X(manager, "-connectToDevice", "invalid central manager (nil)");
+ Q_ASSERT_X(managerState == OSXBluetooth::CentralManagerIdle,
+ "-connectToDevice", "invalid state");
+
+ if ([self isConnected]) {
+ qCDebug(QT_BT_OSX) << "-connectToDevice, already connected";
+ delegate->connectSuccess();
+ return QLowEnergyController::NoError;
+ } else if (peripheral) {
+ // Was retrieved already, but not connected
+ // or disconnected.
+ [self connectToPeripheral];
+ return QLowEnergyController::NoError;
+ }
+
+ using namespace OSXBluetooth;
+
+ // Retrieve a peripheral first ...
+ ObjCScopedPointer<NSMutableArray> uuids([[NSMutableArray alloc] init]);
+ if (!uuids) {
+ qCWarning(QT_BT_OSX) << "-connectToDevice, failed to allocate identifiers";
+ return QLowEnergyController::ConnectionError;
+ }
+
+#if QT_MAC_PLATFORM_SDK_EQUAL_OR_ABOVE(__MAC_10_9, __IPHONE_6_0)
+ const quint128 qtUuidData(deviceUuid.toUInt128());
+ // STATIC_ASSERT on sizes would be handy!
+ uuid_t uuidData = {};
+ std::copy(qtUuidData.data, qtUuidData.data + 16, uuidData);
+ const ObjCScopedPointer<NSUUID> nsUuid([[NSUUID alloc] initWithUUIDBytes:uuidData]);
+ if (!nsUuid) {
+ qCWarning(QT_BT_OSX) << "-connectToDevice, failed to allocate NSUUID identifier";
+ return QLowEnergyController::ConnectionError;
+ }
+
+ [uuids addObject:nsUuid];
+
+ // With the latest CoreBluetooth, we can synchronously retrive peripherals:
+ QT_BT_MAC_AUTORELEASEPOOL;
+ NSArray *const peripherals = [manager retrievePeripheralsWithIdentifiers:uuids];
+ if (!peripherals || peripherals.count != 1) {
+ qCWarning(QT_BT_OSX) << "-connectToDevice, failed to retrive a peripheral";
+ return QLowEnergyController::UnknownRemoteDeviceError;
+ }
+
+ peripheral = [static_cast<CBPeripheral *>([peripherals objectAtIndex:0]) retain];
+ [self connectToPeripheral];
+
+ return QLowEnergyController::NoError;
+#else
+ OSXBluetooth::CFStrongReference<CFUUIDRef> cfUuid(OSXBluetooth::cf_uuid(deviceUuid));
+ if (!cfUuid) {
+ qCWarning(QT_BT_OSX) << "-connectToDevice:, failed to create CFUUID object";
+ return QLowEnergyController::ConnectionError;
+ }
+ // TODO: With ARC this cast will be illegal:
+ [uuids addObject:(id)cfUuid.data()];
+ // Unfortunately, with old Core Bluetooth this call is asynchronous ...
+ managerState = OSXBluetooth::CentralManagerConnecting;
+ [manager retrievePeripherals:uuids];
+
+ return QLowEnergyController::NoError;
+#endif
+}
+
+- (void)connectToPeripheral
+{
+ Q_ASSERT_X(manager, "-connectToPeripheral", "invalid central manager (nil)");
+ Q_ASSERT_X(peripheral, "-connectToPeripheral", "invalid peripheral (nil)");
+ Q_ASSERT_X(managerState == OSXBluetooth::CentralManagerIdle,
+ "-connectToPeripheral", "invalid state");
+
+ // The state is still the same - connecting.
+ if ([self isConnected]) {
+ qCDebug(QT_BT_OSX) << "-connectToPeripheral, already connected";
+ delegate->connectSuccess();
+ } else {
+ qCDebug(QT_BT_OSX) << "-connectToPeripheral, trying to connect";
+ managerState = OSXBluetooth::CentralManagerConnecting;
+ [manager connectPeripheral:peripheral options:nil];
+ }
+}
+
+- (bool)isConnected
+{
+ if (!peripheral)
+ return false;
+
+#if QT_MAC_PLATFORM_SDK_EQUAL_OR_ABOVE(__MAC_10_9, __IPHONE_7_0)
+ return peripheral.state == CBPeripheralStateConnected;
+#else
+ return peripheral.isConnected;
+#endif
+}
+
+- (void)disconnectFromDevice
+{
+ if (managerState == OSXBluetooth::CentralManagerUpdating) {
+ disconnectPending = true;
+ } else {
+ disconnectPending = false;
+
+ if ([self isConnected]) {
+ Q_ASSERT_X(peripheral, "-disconnectFromDevice", "invalid peripheral (nil)");
+ Q_ASSERT_X(manager, "-disconnectFromDevice", "invalid central manager (nil)");
+ managerState = OSXBluetooth::CentralManagerDisconnecting;
+ [manager cancelPeripheralConnection:peripheral];
+ } else {
+ managerState = OSXBluetooth::CentralManagerIdle;
+ delegate->disconnected();
+ }
+ }
+}
+
+- (void)discoverServices
+{
+}
+
+- (bool)discoverServiceDetails:(const QBluetoothUuid &)serviceUuid
+{
+ Q_UNUSED(serviceUuid)
+ return false;
+}
+
+- (bool)discoverCharacteristics:(const QBluetoothUuid &)serviceUuid
+{
+ Q_UNUSED(serviceUuid)
+ return false;
+}
+
+// CBCentralManagerDelegate (the real one).
+
+- (void)centralManagerDidUpdateState:(CBCentralManager *)central
+{
+ Q_ASSERT_X(delegate, "-centralManagerDidUpdateState:", "invalid delegate (null)");
+
+ using namespace OSXBluetooth;
+
+ const CBCentralManagerState state = central.state;
+
+ if (state == CBCentralManagerStateUnknown
+ || state == CBCentralManagerStateResetting) {
+ // We still have to wait, docs say:
+ // "The current state of the central manager is unknown;
+ // an update is imminent." or
+ // "The connection with the system service was momentarily
+ // lost; an update is imminent."
+ return;
+ }
+
+ if (disconnectPending) {
+ disconnectPending = false;
+ managerState = OSXBluetooth::CentralManagerIdle;
+ return [self disconnectFromDevice];
+ }
+
+ // Let's check some states we do not like first:
+ if (state == CBCentralManagerStateUnsupported || state == CBCentralManagerStateUnauthorized) {
+ if (managerState == CentralManagerUpdating) {
+ // We tried to connect just to realize, LE is not supported. Report this.
+ managerState = CentralManagerIdle;
+ delegate->LEnotSupported();
+ } else {
+ // TODO: if we are here, LE _was_ supported and we first managed to update
+ // and reset managerState from CentralManagerUpdating.
+ managerState = CentralManagerIdle;
+ delegate->error(QLowEnergyController::InvalidBluetoothAdapterError);
+ }
+ return;
+ }
+
+ if (state == CBCentralManagerStatePoweredOff) {
+ if (managerState == CentralManagerUpdating) {
+ // I've seen this instead of Unsopported on OS X.
+ managerState = CentralManagerIdle;
+ delegate->LEnotSupported();
+ } else {
+ managerState = CentralManagerIdle;
+ // TODO: we need a better error +
+ // what will happen if later the state changes to PoweredOn???
+ delegate->error(QLowEnergyController::InvalidBluetoothAdapterError);
+ }
+ return;
+ }
+
+ if (state == CBCentralManagerStatePoweredOn) {
+ if (managerState == CentralManagerUpdating) {
+ managerState = CentralManagerIdle;
+ const QLowEnergyController::Error status = [self connectToDevice];
+ if (status != QLowEnergyController::NoError)// An allocation problem?
+ delegate->error(status);
+ }
+ } else {
+ // We actually handled all known states, but .. Core Bluetooth can change?
+ Q_ASSERT_X(0, "-centralManagerDidUpdateState:", "invalid centra's state");
+ }
+}
+
+- (void)centralManager:(CBCentralManager *)central didRetrievePeripherals:(NSArray *)peripherals
+{
+ Q_UNUSED(central)
+
+ // This method is required for iOS before 7.0 and OS X below 10.9.
+ Q_ASSERT_X(manager, "-centralManager:didRetrivePeripherals:", "invalid central manager (nil)");
+
+ if (managerState != OSXBluetooth::CentralManagerConnecting) {
+ // Canceled by calling -disconnectFromDevice method.
+ return;
+ }
+
+ managerState = OSXBluetooth::CentralManagerIdle;
+
+ if (!peripherals || peripherals.count != 1) {
+ Q_ASSERT_X(delegate, "-centralManager:didRetrievePeripherals:",
+ "invalid delegate (null)");
+
+ qCDebug(QT_BT_OSX) << "-centralManager:didRetrievePeripherals:, "
+ "unexpected number of peripherals (!= 1)";
+
+ delegate->error(QLowEnergyController::UnknownRemoteDeviceError);
+ } else {
+ peripheral = [static_cast<CBPeripheral *>([peripherals objectAtIndex:0]) retain];
+ [self connectToPeripheral];
+ }
+}
+
+- (void)centralManager:(CBCentralManager *)central didConnectPeripheral:(CBPeripheral *)aPeripheral
+{
+ Q_UNUSED(central)
+ Q_UNUSED(aPeripheral)
+
+ Q_ASSERT_X(delegate, "-centralManager:didConnectPeripheral:",
+ "invalid delegate (null)");
+
+ if (managerState != OSXBluetooth::CentralManagerConnecting) {
+ // We called cancel but before disconnected, managed to connect?
+ return;
+ }
+
+ managerState = OSXBluetooth::CentralManagerIdle;
+ delegate->connectSuccess();
+}
+
+- (void)centralManager:(CBCentralManager *)central didFailToConnectPeripheral:(CBPeripheral *)aPeripheral
+ error:(NSError *)error
+{
+ Q_UNUSED(central)
+ Q_UNUSED(aPeripheral)
+ Q_UNUSED(error)
+
+ Q_ASSERT_X(delegate, "-centralManager:didFailToConnectPeripheral:",
+ "invalid delegate (null)");
+
+ if (managerState != OSXBluetooth::CentralManagerConnecting) {
+ // Canceled already.
+ return;
+ }
+
+ managerState = OSXBluetooth::CentralManagerIdle;
+ // TODO: better error mapping is required.
+ delegate->error(QLowEnergyController::UnknownRemoteDeviceError);
+}
+
+- (void)centralManager:(CBCentralManager *)central didDisconnectPeripheral:(CBPeripheral *)aPeripheral
+ error:(NSError *)error
+{
+ Q_UNUSED(central)
+ Q_UNUSED(aPeripheral)
+
+ Q_ASSERT_X(delegate, "-centralManager:didDisconnectPeripheral:error:",
+ "invalid delegate (null)");
+
+ if (error && managerState == OSXBluetooth::CentralManagerDisconnecting) {
+ managerState = OSXBluetooth::CentralManagerIdle;
+ qCWarning(QT_BT_OSX) << "-centralManager:didDisconnectPeripheral:, "
+ "failed to disconnect";
+ // TODO: instead of 'unknown' - .. ?
+ delegate->error(QLowEnergyController::UnknownRemoteDeviceError);
+ } else {
+ managerState = OSXBluetooth::CentralManagerIdle;
+ delegate->disconnected();
+ }
+}
+
+// CBPeripheralDelegate.
+
+- (void)peripheral:(CBPeripheral *)aPeripheral didDiscoverServices:(NSError *)error
+{
+ Q_UNUSED(aPeripheral)
+ Q_UNUSED(error)
+}
+
+
+- (void)peripheral:(CBPeripheral *)aPeripheral didDiscoverIncludedServicesForService:(CBService *)service
+ error:(NSError *)error
+{
+ Q_UNUSED(aPeripheral)
+ Q_UNUSED(service)
+ Q_UNUSED(error)
+}
+
+- (void)peripheral:(CBPeripheral *)aPeripheral didDiscoverCharacteristicsForService:(CBService *)service
+ error:(NSError *)error
+{
+ Q_UNUSED(aPeripheral)
+ Q_UNUSED(service)
+ Q_UNUSED(error)
+}
+
+@end