Skip to content

Commit 3ae2ee2

Browse files
authoredMar 25, 2024
[Linux] Extract BlueZ client manager to separate file (project-chip#32647)
* Cleanup BLE event processing function * Disallow copying BluezObjectList class * Log before BleConnectionDelegate::OnConnectionError call * Extract BlueZ client manager to separate file * Update Python binding implementation * Handle scan start error codes properly * Helper function to convert D-Bus error to CHIP error
1 parent 60edbf5 commit 3ae2ee2

16 files changed

+384
-264
lines changed
 

‎src/controller/python/chip/ble/LinuxImpl.cpp

+23-3
Original file line numberDiff line numberDiff line change
@@ -16,9 +16,15 @@
1616
* limitations under the License.
1717
*/
1818

19+
#include <cstdint>
20+
#include <memory>
21+
22+
#include <lib/core/CHIPError.h>
1923
#include <lib/support/CHIPMem.h>
2024
#include <platform/CHIPDeviceLayer.h>
2125
#include <platform/Linux/bluez/AdapterIterator.h>
26+
#include <platform/Linux/bluez/BluezObjectManager.h>
27+
#include <platform/Linux/dbus/bluez/DbusBluez.h>
2228
#include <platform/internal/BLEManager.h>
2329

2430
using namespace chip::DeviceLayer::Internal;
@@ -93,9 +99,18 @@ class ScannerDelegateImpl : public ChipDeviceScannerDelegate
9399
mScanCallback(scanCallback), mCompleteCallback(completeCallback), mErrorCallback(errorCallback)
94100
{}
95101

96-
CHIP_ERROR ScannerInit(BluezAdapter1 * adapter) { return mScanner.Init(adapter, this); }
102+
CHIP_ERROR ScannerInit(BluezAdapter1 * adapter)
103+
{
104+
ReturnErrorOnFailure(mBluezObjectManager.Init());
105+
return mScanner.Init(adapter, this);
106+
}
107+
108+
void ScannerShutdown() { mBluezObjectManager.Shutdown(); }
109+
97110
CHIP_ERROR ScannerStartScan(chip::System::Clock::Timeout timeout) { return mScanner.StartScan(timeout); }
98111

112+
CHIP_ERROR ScannerStopScan() { return mScanner.StopScan(); }
113+
99114
void OnDeviceScanned(BluezDevice1 & device, const chip::Ble::ChipBLEDeviceIdentificationInfo & info) override
100115
{
101116
if (mScanCallback)
@@ -122,7 +137,8 @@ class ScannerDelegateImpl : public ChipDeviceScannerDelegate
122137
}
123138

124139
private:
125-
ChipDeviceScanner mScanner;
140+
BluezObjectManager mBluezObjectManager;
141+
ChipDeviceScanner mScanner{ mBluezObjectManager };
126142
PyObject * const mContext;
127143
const DeviceScannedCallback mScanCallback;
128144
const ScanCompleteCallback mCompleteCallback;
@@ -152,6 +168,10 @@ extern "C" void * pychip_ble_scanner_start(PyObject * context, void * adapter, u
152168

153169
extern "C" void pychip_ble_scanner_delete(void * scanner)
154170
{
171+
auto * delegate = static_cast<ScannerDelegateImpl *>(scanner);
155172
chip::DeviceLayer::StackLock lock;
156-
delete static_cast<ScannerDelegateImpl *>(scanner);
173+
// Make sure that the scanner is stopped before deleting the delegate.
174+
delegate->ScannerStopScan();
175+
delegate->ScannerShutdown();
176+
delete delegate;
157177
}

‎src/platform/Linux/BLEManagerImpl.cpp

+35-34
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,7 @@
3535
#include <type_traits>
3636
#include <utility>
3737

38+
#include <ble/BleError.h>
3839
#include <ble/CHIPBleServiceData.h>
3940
#include <lib/support/CodeUtils.h>
4041
#include <lib/support/SafeInt.h>
@@ -123,6 +124,7 @@ void BLEManagerImpl::_Shutdown()
123124
DeviceLayer::SystemLayer().CancelTimer(HandleConnectTimeout, &mEndpoint);
124125
// Release BLE connection resources (unregister from BlueZ).
125126
mEndpoint.Shutdown();
127+
mBluezObjectManager.Shutdown();
126128
mFlags.Clear(Flags::kBluezBLELayerInitialized);
127129
}
128130

@@ -251,8 +253,7 @@ void BLEManagerImpl::_OnPlatformEvent(const ChipDeviceEvent * event)
251253

252254
void BLEManagerImpl::HandlePlatformSpecificBLEEvent(const ChipDeviceEvent * apEvent)
253255
{
254-
CHIP_ERROR err = CHIP_NO_ERROR;
255-
bool controlOpComplete = false;
256+
CHIP_ERROR err = CHIP_NO_ERROR;
256257
ChipLogDetail(DeviceLayer, "HandlePlatformSpecificBLEEvent %d", apEvent->Type);
257258
switch (apEvent->Type)
258259
{
@@ -288,37 +289,37 @@ void BLEManagerImpl::HandlePlatformSpecificBLEEvent(const ChipDeviceEvent * apEv
288289
break;
289290
case DeviceEventType::kPlatformLinuxBLEPeripheralAdvStartComplete:
290291
SuccessOrExit(err = apEvent->Platform.BLEPeripheralAdvStartComplete.mError);
291-
sInstance.mFlags.Clear(Flags::kControlOpInProgress).Clear(Flags::kAdvertisingRefreshNeeded);
292+
mFlags.Clear(Flags::kControlOpInProgress).Clear(Flags::kAdvertisingRefreshNeeded);
292293
// Do not restart the timer if it is still active. This is to avoid the timer from being restarted
293294
// if the advertising is stopped due to a premature release.
294295
if (!DeviceLayer::SystemLayer().IsTimerActive(HandleAdvertisingTimer, this))
295296
{
296297
// Start a timer to make sure that the fast advertising is stopped after specified timeout.
297298
SuccessOrExit(err = DeviceLayer::SystemLayer().StartTimer(kFastAdvertiseTimeout, HandleAdvertisingTimer, this));
298299
}
299-
sInstance.mFlags.Set(Flags::kAdvertising);
300+
mFlags.Set(Flags::kAdvertising);
300301
break;
301302
case DeviceEventType::kPlatformLinuxBLEPeripheralAdvStopComplete:
302303
SuccessOrExit(err = apEvent->Platform.BLEPeripheralAdvStopComplete.mError);
303-
sInstance.mFlags.Clear(Flags::kControlOpInProgress).Clear(Flags::kAdvertisingRefreshNeeded);
304+
mFlags.Clear(Flags::kControlOpInProgress).Clear(Flags::kAdvertisingRefreshNeeded);
304305
DeviceLayer::SystemLayer().CancelTimer(HandleAdvertisingTimer, this);
305-
306306
// Transition to the not Advertising state...
307-
if (sInstance.mFlags.Has(Flags::kAdvertising))
307+
if (mFlags.Has(Flags::kAdvertising))
308308
{
309-
sInstance.mFlags.Clear(Flags::kAdvertising);
309+
mFlags.Clear(Flags::kAdvertising);
310310
ChipLogProgress(DeviceLayer, "CHIPoBLE advertising stopped");
311311
}
312312
break;
313313
case DeviceEventType::kPlatformLinuxBLEPeripheralAdvReleased:
314314
// If the advertising was stopped due to a premature release, check if it needs to be restarted.
315-
sInstance.mFlags.Clear(Flags::kAdvertising);
315+
mFlags.Clear(Flags::kAdvertising);
316316
DriveBLEState();
317317
break;
318318
case DeviceEventType::kPlatformLinuxBLEPeripheralRegisterAppComplete:
319319
SuccessOrExit(err = apEvent->Platform.BLEPeripheralRegisterAppComplete.mError);
320+
mFlags.Clear(Flags::kControlOpInProgress);
320321
mFlags.Set(Flags::kAppRegistered);
321-
controlOpComplete = true;
322+
DriveBLEState();
322323
break;
323324
default:
324325
break;
@@ -330,13 +331,7 @@ void BLEManagerImpl::HandlePlatformSpecificBLEEvent(const ChipDeviceEvent * apEv
330331
ChipLogError(DeviceLayer, "Disabling CHIPoBLE service due to error: %s", ErrorStr(err));
331332
mServiceMode = ConnectivityManager::kCHIPoBLEServiceMode_Disabled;
332333
DeviceLayer::SystemLayer().CancelTimer(HandleAdvertisingTimer, this);
333-
sInstance.mFlags.Clear(Flags::kControlOpInProgress);
334-
}
335-
336-
if (controlOpComplete)
337-
{
338334
mFlags.Clear(Flags::kControlOpInProgress);
339-
DriveBLEState();
340335
}
341336
}
342337

@@ -587,11 +582,25 @@ void BLEManagerImpl::DriveBLEState()
587582
// If there's already a control operation in progress, wait until it completes.
588583
VerifyOrExit(!mFlags.Has(Flags::kControlOpInProgress), /* */);
589584

590-
// Initializes the Bluez BLE layer if needed.
591-
if (mServiceMode == ConnectivityManager::kCHIPoBLEServiceMode_Enabled && !mFlags.Has(Flags::kBluezBLELayerInitialized))
585+
// Initializes the BlueZ BLE layer if needed.
586+
if (mServiceMode == ConnectivityManager::kCHIPoBLEServiceMode_Enabled)
592587
{
593-
SuccessOrExit(err = mEndpoint.Init(mIsCentral, mAdapterId));
594-
mFlags.Set(Flags::kBluezBLELayerInitialized);
588+
if (!mFlags.Has(Flags::kBluezManagerInitialized))
589+
{
590+
SuccessOrExit(err = mBluezObjectManager.Init());
591+
mFlags.Set(Flags::kBluezManagerInitialized);
592+
}
593+
if (!mFlags.Has(Flags::kBluezAdapterAvailable))
594+
{
595+
mAdapter.reset(mBluezObjectManager.GetAdapter(mAdapterId));
596+
VerifyOrExit(mAdapter, err = BLE_ERROR_ADAPTER_UNAVAILABLE);
597+
mFlags.Set(Flags::kBluezAdapterAvailable);
598+
}
599+
if (!mFlags.Has(Flags::kBluezBLELayerInitialized))
600+
{
601+
SuccessOrExit(err = mEndpoint.Init(mAdapter.get(), mIsCentral));
602+
mFlags.Set(Flags::kBluezBLELayerInitialized);
603+
}
595604
}
596605

597606
// Register the CHIPoBLE application with the Bluez BLE layer if needed.
@@ -614,7 +623,7 @@ void BLEManagerImpl::DriveBLEState()
614623
// Configure advertising data if it hasn't been done yet.
615624
if (!mFlags.Has(Flags::kAdvertisingConfigured))
616625
{
617-
SuccessOrExit(err = mBLEAdvertisement.Init(mEndpoint, mpBLEAdvUUID, mDeviceName));
626+
SuccessOrExit(err = mBLEAdvertisement.Init(mAdapter.get(), mpBLEAdvUUID, mDeviceName));
618627
mFlags.Set(Flags::kAdvertisingConfigured);
619628
}
620629

@@ -724,41 +733,33 @@ void BLEManagerImpl::InitiateScan(BleScanState scanType)
724733

725734
if (scanType == BleScanState::kNotScanning)
726735
{
727-
BleConnectionDelegate::OnConnectionError(mBLEScanConfig.mAppState, CHIP_ERROR_INCORRECT_STATE);
728736
ChipLogError(Ble, "Invalid scan type requested");
729-
return;
730-
}
731-
732-
if (!mFlags.Has(Flags::kBluezBLELayerInitialized))
733-
{
734737
BleConnectionDelegate::OnConnectionError(mBLEScanConfig.mAppState, CHIP_ERROR_INCORRECT_STATE);
735-
ChipLogError(Ble, "BLE Layer is not yet initialized");
736738
return;
737739
}
738740

739-
if (mEndpoint.GetAdapter() == nullptr)
741+
if (!mFlags.Has(Flags::kBluezAdapterAvailable))
740742
{
741-
BleConnectionDelegate::OnConnectionError(mBLEScanConfig.mAppState, CHIP_ERROR_INCORRECT_STATE);
742-
ChipLogError(Ble, "No adapter available for new connection establishment");
743+
BleConnectionDelegate::OnConnectionError(mBLEScanConfig.mAppState, BLE_ERROR_ADAPTER_UNAVAILABLE);
743744
return;
744745
}
745746

746747
mBLEScanConfig.mBleScanState = scanType;
747748

748-
CHIP_ERROR err = mDeviceScanner.Init(mEndpoint.GetAdapter(), this);
749+
CHIP_ERROR err = mDeviceScanner.Init(mAdapter.get(), this);
749750
if (err != CHIP_NO_ERROR)
750751
{
751752
mBLEScanConfig.mBleScanState = BleScanState::kNotScanning;
753+
ChipLogError(Ble, "Failed to create a BLE device scanner: %" CHIP_ERROR_FORMAT, err.Format());
752754
BleConnectionDelegate::OnConnectionError(mBLEScanConfig.mAppState, CHIP_ERROR_INTERNAL);
753-
ChipLogError(Ble, "Failed to create a BLE device scanner");
754755
return;
755756
}
756757

757758
err = mDeviceScanner.StartScan(kNewConnectionScanTimeout);
758759
if (err != CHIP_NO_ERROR)
759760
{
760761
mBLEScanConfig.mBleScanState = BleScanState::kNotScanning;
761-
ChipLogError(Ble, "Failed to start a BLE can: %s", chip::ErrorStr(err));
762+
ChipLogError(Ble, "Failed to start a BLE can: %" CHIP_ERROR_FORMAT, err.Format());
762763
BleConnectionDelegate::OnConnectionError(mBLEScanConfig.mAppState, err);
763764
return;
764765
}

‎src/platform/Linux/BLEManagerImpl.h

+19-13
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@
3131

3232
#include "bluez/BluezAdvertisement.h"
3333
#include "bluez/BluezEndpoint.h"
34+
#include "bluez/BluezObjectManager.h"
3435
#include "bluez/ChipDeviceScanner.h"
3536
#include "bluez/Types.h"
3637

@@ -159,16 +160,18 @@ class BLEManagerImpl final : public BLEManager,
159160
enum class Flags : uint16_t
160161
{
161162
kAsyncInitCompleted = 0x0001, /**< One-time asynchronous initialization actions have been performed. */
162-
kBluezBLELayerInitialized = 0x0002, /**< The Bluez layer has been initialized. */
163-
kAppRegistered = 0x0004, /**< The CHIPoBLE application has been registered with the Bluez layer. */
164-
kAdvertisingConfigured = 0x0008, /**< CHIPoBLE advertising has been configured in the Bluez layer. */
165-
kAdvertising = 0x0010, /**< The system is currently CHIPoBLE advertising. */
166-
kControlOpInProgress = 0x0020, /**< An async control operation has been issued to the ESP BLE layer. */
167-
kAdvertisingEnabled = 0x0040, /**< The application has enabled CHIPoBLE advertising. */
168-
kFastAdvertisingEnabled = 0x0080, /**< The application has enabled fast advertising. */
169-
kUseCustomDeviceName = 0x0100, /**< The application has configured a custom BLE device name. */
170-
kAdvertisingRefreshNeeded = 0x0200, /**< The advertising configuration/state in BLE layer needs to be updated. */
171-
kExtAdvertisingEnabled = 0x0400, /**< The application has enabled CHIPoBLE extended advertising. */
163+
kBluezManagerInitialized = 0x0002, /**< The BlueZ object manager has been initialized. */
164+
kBluezAdapterAvailable = 0x0004, /**< Selected BlueZ adapter is available for use. */
165+
kBluezBLELayerInitialized = 0x0008, /**< The BlueZ BLE layer has been initialized. */
166+
kAppRegistered = 0x0010, /**< The CHIPoBLE application has been registered with the Bluez layer. */
167+
kAdvertisingConfigured = 0x0020, /**< CHIPoBLE advertising has been configured in the Bluez layer. */
168+
kAdvertising = 0x0040, /**< The system is currently CHIPoBLE advertising. */
169+
kControlOpInProgress = 0x0080, /**< An async control operation has been issued to the ESP BLE layer. */
170+
kAdvertisingEnabled = 0x0100, /**< The application has enabled CHIPoBLE advertising. */
171+
kFastAdvertisingEnabled = 0x0200, /**< The application has enabled fast advertising. */
172+
kUseCustomDeviceName = 0x0400, /**< The application has configured a custom BLE device name. */
173+
kAdvertisingRefreshNeeded = 0x0800, /**< The advertising configuration/state in BLE layer needs to be updated. */
174+
kExtAdvertisingEnabled = 0x1000, /**< The application has enabled CHIPoBLE extended advertising. */
172175
};
173176

174177
enum
@@ -187,15 +190,18 @@ class BLEManagerImpl final : public BLEManager,
187190
CHIPoBLEServiceMode mServiceMode;
188191
BitFlags<Flags> mFlags;
189192

193+
BluezObjectManager mBluezObjectManager;
194+
GAutoPtr<BluezAdapter1> mAdapter;
190195
uint32_t mAdapterId = 0;
196+
191197
char mDeviceName[kMaxDeviceNameLength + 1];
192198
bool mIsCentral = false;
193-
BluezEndpoint mEndpoint;
199+
BluezEndpoint mEndpoint{ mBluezObjectManager };
194200

195-
BluezAdvertisement mBLEAdvertisement;
201+
BluezAdvertisement mBLEAdvertisement{ mEndpoint };
196202
const char * mpBLEAdvUUID = nullptr;
197203

198-
ChipDeviceScanner mDeviceScanner;
204+
ChipDeviceScanner mDeviceScanner{ mBluezObjectManager };
199205
BLEScanConfig mBLEScanConfig;
200206
};
201207

‎src/platform/Linux/BUILD.gn

+2
Original file line numberDiff line numberDiff line change
@@ -103,6 +103,8 @@ static_library("Linux") {
103103
"bluez/BluezEndpoint.h",
104104
"bluez/BluezObjectIterator.h",
105105
"bluez/BluezObjectList.h",
106+
"bluez/BluezObjectManager.cpp",
107+
"bluez/BluezObjectManager.h",
106108
"bluez/ChipDeviceScanner.cpp",
107109
"bluez/ChipDeviceScanner.h",
108110
"bluez/Types.h",

‎src/platform/Linux/bluez/AdapterIterator.cpp

+19-38
Original file line numberDiff line numberDiff line change
@@ -17,48 +17,25 @@
1717

1818
#include "AdapterIterator.h"
1919

20+
#include <cstdio>
21+
22+
#include <gio/gio.h>
23+
24+
#include <lib/core/CHIPError.h>
2025
#include <lib/support/CodeUtils.h>
2126
#include <lib/support/logging/CHIPLogging.h>
2227
#include <platform/GLibTypeDeleter.h>
23-
#include <platform/PlatformManager.h>
28+
#include <platform/Linux/dbus/bluez/DbusBluez.h>
29+
30+
#include "BluezObjectIterator.h"
31+
#include "BluezObjectList.h"
32+
#include "BluezObjectManager.h"
33+
#include "Types.h"
2434

2535
namespace chip {
2636
namespace DeviceLayer {
2737
namespace Internal {
2838

29-
CHIP_ERROR AdapterIterator::Initialize()
30-
{
31-
// When creating D-Bus proxy object, the thread default context must be initialized. Otherwise,
32-
// all D-Bus signals will be delivered to the GLib global default main context.
33-
VerifyOrDie(g_main_context_get_thread_default() != nullptr);
34-
35-
GAutoPtr<GError> error;
36-
mManager.reset(g_dbus_object_manager_client_new_for_bus_sync(
37-
G_BUS_TYPE_SYSTEM, G_DBUS_OBJECT_MANAGER_CLIENT_FLAGS_NONE, BLUEZ_INTERFACE, "/",
38-
bluez_object_manager_client_get_proxy_type, nullptr /* unused user data in the Proxy Type Func */,
39-
nullptr /* destroy notify */, nullptr /* cancellable */, &error.GetReceiver()));
40-
41-
VerifyOrReturnError(mManager, CHIP_ERROR_INTERNAL,
42-
ChipLogError(DeviceLayer, "Failed to get D-Bus object manager for listing adapters: %s", error->message));
43-
44-
mObjectList.Init(mManager.get());
45-
mIterator = mObjectList.begin();
46-
47-
return CHIP_NO_ERROR;
48-
}
49-
50-
CHIP_ERROR AdapterIterator::Shutdown()
51-
{
52-
// Release resources on the glib thread to synchronize with potential signal handlers
53-
// attached to the manager client object that may run on the glib thread.
54-
return PlatformMgrImpl().GLibMatterContextInvokeSync(
55-
+[](AdapterIterator * self) {
56-
self->mManager.reset();
57-
return CHIP_NO_ERROR;
58-
},
59-
this);
60-
}
61-
6239
bool AdapterIterator::Advance()
6340
{
6441
for (; mIterator != BluezObjectList::end(); ++mIterator)
@@ -93,11 +70,15 @@ uint32_t AdapterIterator::GetIndex() const
9370

9471
bool AdapterIterator::Next()
9572
{
96-
if (!mManager)
73+
if (!mIsInitialized)
9774
{
98-
CHIP_ERROR err = PlatformMgrImpl().GLibMatterContextInvokeSync(
99-
+[](AdapterIterator * self) { return self->Initialize(); }, this);
100-
VerifyOrReturnError(err == CHIP_NO_ERROR, false, ChipLogError(DeviceLayer, "Failed to initialize adapter iterator"));
75+
CHIP_ERROR err = mObjectManager.Init();
76+
VerifyOrReturnError(
77+
err == CHIP_NO_ERROR, false,
78+
ChipLogError(DeviceLayer, "Failed to initialize BlueZ object manager: %" CHIP_ERROR_FORMAT, err.Format()));
79+
mObjectList.Init(mObjectManager.GetObjectManager());
80+
mIterator = mObjectList.begin();
81+
mIsInitialized = true;
10182
}
10283

10384
return Advance();

0 commit comments

Comments
 (0)