Skip to content

Commit

Permalink
Unified BLETransmitter stop advert method
Browse files Browse the repository at this point in the history
- Using the same function for stopping advert, when triggered by internal timer and external events
- Stop advert function makes best effort for graceful stop where possible
- Previous experiments have consistently shown BluetoothState==PowerOff implies bluetoothGattServer is no longer valid
- This has been confirmed again (Android reports bluetoothGattServer as dead object during graceful stop)
- Using stop advert function (instead of bluetoothGattServer=null) following BluetoothState==PowerOff is not expected to offer any benefit, but it is certainly harmless, and enables logging of this previously unseen condition for investigation
- Tested on Pixel 2 (Android 29), J6 (Android 28), with iPhone 6s (iOS 12.1.4)

Signed-off-by: c19x <[email protected]>
  • Loading branch information
c19x committed Dec 24, 2020
1 parent 701552d commit 5ec7bd2
Show file tree
Hide file tree
Showing 2 changed files with 39 additions and 25 deletions.
15 changes: 12 additions & 3 deletions .idea/codeStyles/Project.xml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Original file line number Diff line number Diff line change
Expand Up @@ -148,10 +148,16 @@ private long timeSincelastStateChange(final long now) {
public void bleTimer(final long now) {
if (!isSupported() || bluetoothStateManager.state() == BluetoothState.poweredOff) {
if (advertLoopState != AdvertLoopState.stopped) {
advertiseCallback = null;
bluetoothGattServer = null;
state(now, AdvertLoopState.stopped);
logger.debug("advertLoopTask, stop advert (advert={}ms)", timeSincelastStateChange(now));
logger.debug("advertLoopTask, stopping advert following bluetooth state change (isSupported={},bluetoothPowerOff={})", isSupported(), bluetoothStateManager.state() == BluetoothState.poweredOff);
stopAdvert(bluetoothLeAdvertiser(), advertiseCallback, bluetoothGattServer, new Callback<Boolean>() {
@Override
public void accept(Boolean value) {
advertiseCallback = null;
bluetoothGattServer = null;
state(now, AdvertLoopState.stopped);
logger.debug("advertLoopTask, stop advert (advert={}ms)", timeSincelastStateChange(now));
}
});
}
return;
}
Expand Down Expand Up @@ -206,29 +212,28 @@ public void accept(Boolean value) {

// MARK:- Start and stop advert

private void stopExistingGattServer() {
if (null != bluetoothGattServer) {
// Stop old version, if there's already a proxy reference
try {
bluetoothGattServer.clearServices();
bluetoothGattServer.close();
bluetoothGattServer = null;
} catch (Throwable e2) {
logger.fault("stopGattServer failed to stop EXISTING GATT server", e2);
bluetoothGattServer = null;
}
}
}

private void startAdvert(final BluetoothLeAdvertiser bluetoothLeAdvertiser, final Callback<Triple<Boolean, AdvertiseCallback, BluetoothGattServer>> callback) {
logger.debug("startAdvert");
operationQueue.execute(new Runnable() {
@Override
public void run() {
boolean result = true;

stopExistingGattServer();

// Stop existing advert if there is already a proxy reference.
// This should never happen because only the AdvertLoopTask calls
// startAdvert and it should only call startAdvert after stopAdvert
// has been called previously. Logging this condition to verify if
// this condition can ever occur to support investigation.
if (bluetoothGattServer != null) {
logger.fault("startAdvert found existing GATT server");
try {
bluetoothGattServer.clearServices();
bluetoothGattServer.close();
} catch (Throwable e) {
logger.fault("startAdvert found existing GATT server but failed to stop the server", e);
}
bluetoothGattServer = null;
}
// Start new GATT server
try {
bluetoothGattServer = startGattServer(logger, context, payloadDataSupplier, database);
} catch (Throwable e) {
Expand Down Expand Up @@ -289,7 +294,7 @@ private void stopAdvert(final BluetoothLeAdvertiser bluetoothLeAdvertiser, final
public void run() {
boolean result = true;
try {
if (advertiseCallback != null) {
if (bluetoothLeAdvertiser != null && advertiseCallback != null) {
bluetoothLeAdvertiser.stopAdvertising(advertiseCallback);
}
} catch (Throwable e) {
Expand Down

0 comments on commit 5ec7bd2

Please sign in to comment.