Skip to content

Commit 22adb08

Browse files
authored
[Linux] Handle BLE scan timeout in BLEManager instead of scanner class (project-chip#32770)
* [Linux] Handle BLE scan timeout in BLEManager instead of scanner class * Simplify scanning check * Simplify error handling in InitiateScan() * Update Python binding code * Fix compilation without logging
1 parent a0ace35 commit 22adb08

File tree

5 files changed

+70
-70
lines changed

5 files changed

+70
-70
lines changed

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

+29-4
Original file line numberDiff line numberDiff line change
@@ -19,13 +19,18 @@
1919
#include <cstdint>
2020
#include <memory>
2121

22+
#include <ble/CHIPBleServiceData.h>
2223
#include <lib/core/CHIPError.h>
23-
#include <lib/support/CHIPMem.h>
24+
#include <lib/support/CodeUtils.h>
2425
#include <platform/CHIPDeviceLayer.h>
26+
#include <platform/Linux/BlePlatformConfig.h>
2527
#include <platform/Linux/bluez/AdapterIterator.h>
2628
#include <platform/Linux/bluez/BluezObjectManager.h>
29+
#include <platform/Linux/bluez/ChipDeviceScanner.h>
2730
#include <platform/Linux/dbus/bluez/DbusBluez.h>
28-
#include <platform/internal/BLEManager.h>
31+
#include <platform/PlatformManager.h>
32+
#include <system/SystemClock.h>
33+
#include <system/SystemLayer.h>
2934

3035
using namespace chip::DeviceLayer::Internal;
3136

@@ -107,9 +112,29 @@ class ScannerDelegateImpl : public ChipDeviceScannerDelegate
107112

108113
void ScannerShutdown() { mBluezObjectManager.Shutdown(); }
109114

110-
CHIP_ERROR ScannerStartScan(chip::System::Clock::Timeout timeout) { return mScanner.StartScan(timeout); }
115+
CHIP_ERROR ScannerStartScan(chip::System::Clock::Timeout timeout)
116+
{
117+
CHIP_ERROR err = mScanner.StartScan();
118+
VerifyOrReturnError(err == CHIP_NO_ERROR, err);
119+
120+
err = chip::DeviceLayer::SystemLayer().StartTimer(timeout, HandleScannerTimer, this);
121+
VerifyOrReturnError(err == CHIP_NO_ERROR, err, mScanner.StopScan());
111122

112-
CHIP_ERROR ScannerStopScan() { return mScanner.StopScan(); }
123+
return CHIP_NO_ERROR;
124+
}
125+
126+
CHIP_ERROR ScannerStopScan()
127+
{
128+
chip::DeviceLayer::SystemLayer().CancelTimer(HandleScannerTimer, this);
129+
return mScanner.StopScan();
130+
}
131+
132+
static void HandleScannerTimer(chip::System::Layer *, void * appState)
133+
{
134+
auto * delegate = static_cast<ScannerDelegateImpl *>(appState);
135+
delegate->OnScanError(CHIP_ERROR_TIMEOUT);
136+
delegate->mScanner.StopScan();
137+
}
113138

114139
void OnDeviceScanned(BluezDevice1 & device, const chip::Ble::ChipBLEDeviceIdentificationInfo & info) override
115140
{

src/platform/Linux/BLEManagerImpl.cpp

+35-23
Original file line numberDiff line numberDiff line change
@@ -746,42 +746,50 @@ void BLEManagerImpl::HandleAdvertisingTimer(chip::System::Layer *, void * appSta
746746

747747
void BLEManagerImpl::InitiateScan(BleScanState scanType)
748748
{
749+
CHIP_ERROR err = CHIP_ERROR_INCORRECT_STATE;
750+
749751
DriveBLEState();
750752

751-
if (scanType == BleScanState::kNotScanning)
752-
{
753-
ChipLogError(Ble, "Invalid scan type requested");
754-
BleConnectionDelegate::OnConnectionError(mBLEScanConfig.mAppState, CHIP_ERROR_INCORRECT_STATE);
755-
return;
756-
}
757-
758-
if (!mFlags.Has(Flags::kBluezAdapterAvailable))
759-
{
760-
BleConnectionDelegate::OnConnectionError(mBLEScanConfig.mAppState, BLE_ERROR_ADAPTER_UNAVAILABLE);
761-
return;
762-
}
753+
VerifyOrExit(scanType != BleScanState::kNotScanning,
754+
ChipLogError(Ble, "Invalid scan type requested: %d", to_underlying(scanType)));
755+
VerifyOrExit(!mDeviceScanner.IsScanning(), ChipLogError(Ble, "BLE scan already in progress"));
756+
VerifyOrExit(mFlags.Has(Flags::kBluezAdapterAvailable), err = BLE_ERROR_ADAPTER_UNAVAILABLE);
763757

764758
mBLEScanConfig.mBleScanState = scanType;
765759

766-
CHIP_ERROR err = mDeviceScanner.Init(mAdapter.get(), this);
767-
if (err != CHIP_NO_ERROR)
768-
{
760+
err = mDeviceScanner.Init(mAdapter.get(), this);
761+
VerifyOrExit(err == CHIP_NO_ERROR, {
769762
mBLEScanConfig.mBleScanState = BleScanState::kNotScanning;
770-
ChipLogError(Ble, "Failed to create a BLE device scanner: %" CHIP_ERROR_FORMAT, err.Format());
771-
BleConnectionDelegate::OnConnectionError(mBLEScanConfig.mAppState, CHIP_ERROR_INTERNAL);
772-
return;
773-
}
763+
ChipLogError(Ble, "Failed to create BLE device scanner: %" CHIP_ERROR_FORMAT, err.Format());
764+
});
774765

775-
err = mDeviceScanner.StartScan(kNewConnectionScanTimeout);
766+
err = mDeviceScanner.StartScan();
767+
VerifyOrExit(err == CHIP_NO_ERROR, {
768+
mBLEScanConfig.mBleScanState = BleScanState::kNotScanning;
769+
ChipLogError(Ble, "Failed to start BLE scan: %" CHIP_ERROR_FORMAT, err.Format());
770+
});
771+
772+
err = DeviceLayer::SystemLayer().StartTimer(kNewConnectionScanTimeout, HandleScannerTimer, this);
773+
VerifyOrExit(err == CHIP_NO_ERROR, {
774+
mBLEScanConfig.mBleScanState = BleScanState::kNotScanning;
775+
mDeviceScanner.StopScan();
776+
ChipLogError(Ble, "Failed to start BLE scan timeout: %" CHIP_ERROR_FORMAT, err.Format());
777+
});
778+
779+
exit:
776780
if (err != CHIP_NO_ERROR)
777781
{
778-
mBLEScanConfig.mBleScanState = BleScanState::kNotScanning;
779-
ChipLogError(Ble, "Failed to start a BLE can: %" CHIP_ERROR_FORMAT, err.Format());
780782
BleConnectionDelegate::OnConnectionError(mBLEScanConfig.mAppState, err);
781-
return;
782783
}
783784
}
784785

786+
void BLEManagerImpl::HandleScannerTimer(chip::System::Layer *, void * appState)
787+
{
788+
auto * manager = static_cast<BLEManagerImpl *>(appState);
789+
manager->OnScanError(CHIP_ERROR_TIMEOUT);
790+
manager->mDeviceScanner.StopScan();
791+
}
792+
785793
void BLEManagerImpl::CleanScanConfig()
786794
{
787795
if (mBLEScanConfig.mBleScanState == BleScanState::kConnecting)
@@ -805,7 +813,10 @@ CHIP_ERROR BLEManagerImpl::CancelConnection()
805813
mEndpoint.CancelConnect();
806814
// If in discovery mode, stop scan.
807815
else if (mBLEScanConfig.mBleScanState != BleScanState::kNotScanning)
816+
{
817+
DeviceLayer::SystemLayer().CancelTimer(HandleScannerTimer, this);
808818
mDeviceScanner.StopScan();
819+
}
809820
return CHIP_NO_ERROR;
810821
}
811822

@@ -894,6 +905,7 @@ void BLEManagerImpl::OnDeviceScanned(BluezDevice1 & device, const chip::Ble::Chi
894905
// We StartScan in the ChipStack thread.
895906
// StopScan should also be performed in the ChipStack thread.
896907
// At the same time, the scan timer also needs to be canceled in the ChipStack thread.
908+
DeviceLayer::SystemLayer().CancelTimer(HandleScannerTimer, this);
897909
mDeviceScanner.StopScan();
898910
// Stop scanning and then start connecting timer
899911
DeviceLayer::SystemLayer().StartTimer(kConnectTimeout, HandleConnectTimeout, &mEndpoint);

src/platform/Linux/BLEManagerImpl.h

+1
Original file line numberDiff line numberDiff line change
@@ -187,6 +187,7 @@ class BLEManagerImpl final : public BLEManager,
187187
BluezAdvertisement::AdvertisingIntervals GetAdvertisingIntervals() const;
188188
static void HandleAdvertisingTimer(chip::System::Layer *, void * appState);
189189
void InitiateScan(BleScanState scanType);
190+
static void HandleScannerTimer(chip::System::Layer *, void * appState);
190191
void CleanScanConfig();
191192

192193
CHIPoBLEServiceMode mServiceMode;

src/platform/Linux/bluez/ChipDeviceScanner.cpp

+1-32
Original file line numberDiff line numberDiff line change
@@ -89,11 +89,10 @@ void ChipDeviceScanner::Shutdown()
8989
mScannerState = ChipDeviceScannerState::SCANNER_UNINITIALIZED;
9090
}
9191

92-
CHIP_ERROR ChipDeviceScanner::StartScan(System::Clock::Timeout timeout)
92+
CHIP_ERROR ChipDeviceScanner::StartScan()
9393
{
9494
assertChipStackLockedByCurrentThread();
9595
VerifyOrReturnError(mScannerState != ChipDeviceScannerState::SCANNER_SCANNING, CHIP_ERROR_INCORRECT_STATE);
96-
VerifyOrReturnError(mTimerState == ScannerTimerState::TIMER_CANCELED, CHIP_ERROR_INCORRECT_STATE);
9796

9897
mCancellable.reset(g_cancellable_new());
9998
CHIP_ERROR err = PlatformMgrImpl().GLibMatterContextInvokeSync(
@@ -105,34 +104,12 @@ CHIP_ERROR ChipDeviceScanner::StartScan(System::Clock::Timeout timeout)
105104
return err;
106105
}
107106

108-
// Here need to set the Bluetooth scanning status immediately.
109-
// So that if the timer fails to start in the next step,
110-
// calling StopScan will be effective.
111107
mScannerState = ChipDeviceScannerState::SCANNER_SCANNING;
112-
113-
err = chip::DeviceLayer::SystemLayer().StartTimer(timeout, TimerExpiredCallback, static_cast<void *>(this));
114-
if (err != CHIP_NO_ERROR)
115-
{
116-
ChipLogError(Ble, "Failed to schedule scan timeout: %" CHIP_ERROR_FORMAT, err.Format());
117-
StopScan();
118-
return err;
119-
}
120-
121-
mTimerState = ScannerTimerState::TIMER_STARTED;
122-
123108
ChipLogDetail(Ble, "ChipDeviceScanner has started scanning!");
124109

125110
return CHIP_NO_ERROR;
126111
}
127112

128-
void ChipDeviceScanner::TimerExpiredCallback(chip::System::Layer * layer, void * appState)
129-
{
130-
ChipDeviceScanner * chipDeviceScanner = static_cast<ChipDeviceScanner *>(appState);
131-
chipDeviceScanner->mTimerState = ScannerTimerState::TIMER_EXPIRED;
132-
chipDeviceScanner->mDelegate->OnScanError(CHIP_ERROR_TIMEOUT);
133-
chipDeviceScanner->StopScan();
134-
}
135-
136113
CHIP_ERROR ChipDeviceScanner::StopScan()
137114
{
138115
assertChipStackLockedByCurrentThread();
@@ -151,14 +128,6 @@ CHIP_ERROR ChipDeviceScanner::StopScan()
151128

152129
ChipLogDetail(Ble, "ChipDeviceScanner has stopped scanning!");
153130

154-
if (mTimerState == ScannerTimerState::TIMER_STARTED)
155-
{
156-
chip::DeviceLayer::SystemLayer().CancelTimer(TimerExpiredCallback, this);
157-
}
158-
159-
// Reset timer status
160-
mTimerState = ScannerTimerState::TIMER_CANCELED;
161-
162131
mDelegate->OnScanComplete();
163132

164133
return CHIP_NO_ERROR;

src/platform/Linux/bluez/ChipDeviceScanner.h

+4-11
Original file line numberDiff line numberDiff line change
@@ -73,11 +73,14 @@ class ChipDeviceScanner : public BluezObjectManagerAdapterNotificationsDelegate
7373
///
7474
/// This method must be called while in the Matter context (from the Matter event
7575
/// loop, or while holding the Matter stack lock).
76-
CHIP_ERROR StartScan(System::Clock::Timeout timeout);
76+
CHIP_ERROR StartScan();
7777

7878
/// Stop any currently running scan
7979
CHIP_ERROR StopScan();
8080

81+
/// Check if the scanner is active
82+
bool IsScanning() const { return mScannerState == ChipDeviceScannerState::SCANNER_SCANNING; }
83+
8184
/// Members that implement virtual methods on BluezObjectManagerAdapterNotificationsDelegate
8285
void OnDeviceAdded(BluezDevice1 & device) override;
8386
void OnDevicePropertyChanged(BluezDevice1 & device, GVariant * changedProps, const char * const * invalidatedProps) override;
@@ -91,16 +94,8 @@ class ChipDeviceScanner : public BluezObjectManagerAdapterNotificationsDelegate
9194
SCANNER_SCANNING
9295
};
9396

94-
enum ScannerTimerState
95-
{
96-
TIMER_CANCELED,
97-
TIMER_STARTED,
98-
TIMER_EXPIRED
99-
};
100-
10197
CHIP_ERROR StartScanImpl();
10298
CHIP_ERROR StopScanImpl();
103-
static void TimerExpiredCallback(chip::System::Layer * layer, void * appState);
10499

105100
/// Check if a given device is a CHIP device and if yes, report it as discovered
106101
void ReportDevice(BluezDevice1 & device);
@@ -114,8 +109,6 @@ class ChipDeviceScanner : public BluezObjectManagerAdapterNotificationsDelegate
114109

115110
ChipDeviceScannerDelegate * mDelegate = nullptr;
116111
ChipDeviceScannerState mScannerState = ChipDeviceScannerState::SCANNER_UNINITIALIZED;
117-
/// Used to track if timer has already expired and doesn't need to be canceled.
118-
ScannerTimerState mTimerState = ScannerTimerState::TIMER_CANCELED;
119112
GAutoPtr<GCancellable> mCancellable;
120113
};
121114

0 commit comments

Comments
 (0)