Skip to content

Commit

Permalink
Merge pull request #491 from AltBeacon/no-scanning-restarts-on-duplic…
Browse files Browse the repository at this point in the history
…ate-detection-devices

Don't restart scanning if device detects duplicate advertisements per scan
  • Loading branch information
davidgyoung authored Apr 21, 2017
2 parents ef90455 + 1e49cd6 commit 342bfff
Show file tree
Hide file tree
Showing 5 changed files with 157 additions and 19 deletions.
6 changes: 6 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,11 @@
### Development

Enhancements:

- Don't restart BLE scanning periodically if the library confrims device can detect duplicate
advertisements in a single scan, leading to more reliable detections with short scan cycles
(#491, David G. Young)

Bug Fixes:

- Deprecate misspelled methods `removeMonitoreNotifier` and
Expand Down
12 changes: 11 additions & 1 deletion src/main/java/org/altbeacon/beacon/service/BeaconService.java
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,7 @@
import org.altbeacon.beacon.logging.LogManager;
import org.altbeacon.beacon.service.scanner.CycledLeScanCallback;
import org.altbeacon.beacon.service.scanner.CycledLeScanner;
import org.altbeacon.beacon.service.scanner.DistinctPacketDetector;
import org.altbeacon.beacon.service.scanner.NonBeaconLeScanCallback;
import org.altbeacon.beacon.startup.StartupBroadcastReceiver;
import org.altbeacon.bluetooth.BluetoothCrashResolver;
Expand Down Expand Up @@ -90,7 +91,8 @@ public class BeaconService extends Service {
private boolean mBackgroundFlag = false;
private ExtraDataBeaconTracker mExtraDataBeaconTracker;
private ExecutorService mExecutor;

private final DistinctPacketDetector mDistinctPacketDetector = new DistinctPacketDetector();

/*
* The scan period is how long we wait between restarting the BLE advertisement scans
* Each time we restart we only see the unique advertisements once (e.g. unique beacons)
Expand Down Expand Up @@ -359,6 +361,7 @@ public void onLeScan(BluetoothDevice device, int rssi, byte[] scanRecord) {

@Override
public void onCycleEnd() {
mDistinctPacketDetector.clearDetections();
monitoringStatus.updateNewlyOutside();
processRangeData();
// If we want to use simulated scanning data, do it here. This is used for testing in an emulator
Expand Down Expand Up @@ -479,6 +482,13 @@ protected Void doInBackground(ScanData... params) {
}
if (beacon != null) {
mDetectionTracker.recordDetection();
if (!mCycledScanner.getDistinctPacketsDetectedPerScan()) {
if (!mDistinctPacketDetector.isPacketDistinct(scanData.device.getAddress(),
scanData.scanRecord)) {
LogManager.i(TAG, "Non-distinct packets detected in a single scan. Restarting scans unecessary.");
mCycledScanner.setDistinctPacketsDetectedPerScan(true);
}
}
trackedBeaconsPacketCount++;
processBeaconFromScan(beacon);
} else {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,6 @@
import org.altbeacon.beacon.logging.LogManager;
import org.altbeacon.beacon.startup.StartupBroadcastReceiver;
import org.altbeacon.bluetooth.BluetoothCrashResolver;

import java.util.Date;

@TargetApi(18)
Expand Down Expand Up @@ -52,6 +51,7 @@ public abstract class CycledLeScanner {
protected boolean mBackgroundFlag = false;
protected boolean mRestartNeeded = false;

private boolean mDistinctPacketsDetectedPerScan = false;
private static final long ANDROID_N_MIN_SCAN_CYCLE_MILLIS = 6000l;

protected CycledLeScanner(Context context, long scanPeriod, long betweenScanPeriod, boolean backgroundFlag, CycledLeScanCallback cycledLeScanCallback, BluetoothCrashResolver crashResolver) {
Expand Down Expand Up @@ -162,6 +162,14 @@ public void stop() {
}
}

public boolean getDistinctPacketsDetectedPerScan() {
return mDistinctPacketsDetectedPerScan;
}

public void setDistinctPacketsDetectedPerScan(boolean detected) {
mDistinctPacketsDetectedPerScan = detected;
}

public void destroy() {
mScanThread.quit();
}
Expand Down Expand Up @@ -268,25 +276,39 @@ private void finishScanCycle() {
if (mScanning) {
if (getBluetoothAdapter() != null) {
if (getBluetoothAdapter().isEnabled()) {
long now = SystemClock.elapsedRealtime();
if (android.os.Build.VERSION.SDK_INT >= Build.VERSION_CODES.N &&
mBetweenScanPeriod+mScanPeriod < ANDROID_N_MIN_SCAN_CYCLE_MILLIS &&
now-mLastScanCycleStartTime < ANDROID_N_MIN_SCAN_CYCLE_MILLIS) {
// As of Android N, only 5 scans may be started in a 30 second period (6
// seconds per cycle) otherwise they are blocked. So we check here to see
// if the scan period is 6 seconds or less, and if we last stopped scanning
// fewer than 6 seconds ag and if so, we simply do not stop scanning
LogManager.d(TAG, "Not stopping scan because this is Android N and we" +
" keep scanning for a minimum of 6 seconds at a time. "+
"We will stop in "+(ANDROID_N_MIN_SCAN_CYCLE_MILLIS-(now-mLastScanCycleStartTime))+" millisconds.");
// Determine if we need to restart scanning. Restarting scanning is only
// needed on devices incapable of detecting multiple distinct BLE advertising
// packets in a single cycle, typically older Android devices (e.g. Nexus 4)
// On such devices, it is necessary to stop scanning and restart to detect
// multiple beacon packets in the same scan, allowing collection of multiple
// rssi measurements. Restarting however, causes brief detection dropouts
// so it is best avoided. If we know the device has detected to distinct
// packets in the same cycle, we will not restart scanning and just keep it
// going.
if (!getDistinctPacketsDetectedPerScan()) {
long now = SystemClock.elapsedRealtime();
if (android.os.Build.VERSION.SDK_INT >= Build.VERSION_CODES.N &&
mBetweenScanPeriod+mScanPeriod < ANDROID_N_MIN_SCAN_CYCLE_MILLIS &&
now-mLastScanCycleStartTime < ANDROID_N_MIN_SCAN_CYCLE_MILLIS) {
// As of Android N, only 5 scans may be started in a 30 second period (6
// seconds per cycle) otherwise they are blocked. So we check here to see
// if the scan period is 6 seconds or less, and if we last stopped scanning
// fewer than 6 seconds ag and if so, we simply do not stop scanning
LogManager.d(TAG, "Not stopping scan because this is Android N and we" +
" keep scanning for a minimum of 6 seconds at a time. "+
"We will stop in "+(ANDROID_N_MIN_SCAN_CYCLE_MILLIS-(now-mLastScanCycleStartTime))+" millisconds.");
}
else {
try {
LogManager.d(TAG, "stopping bluetooth le scan");
finishScan();
} catch (Exception e) {
LogManager.w(e, TAG, "Internal Android exception scanning for beacons");
}
}
}
else {
try {
LogManager.d(TAG, "stopping bluetooth le scan");
finishScan();
} catch (Exception e) {
LogManager.w(e, TAG, "Internal Android exception scanning for beacons");
}
LogManager.d(TAG, "Not stopping scanning. Device capable of multiple indistinct detections per scan.");
}

mLastScanCycleEndTime = SystemClock.elapsedRealtime();
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
package org.altbeacon.beacon.service.scanner;

import android.util.Log;

import java.nio.ByteBuffer;
import java.util.HashSet;
import java.util.Set;

/**
* Created by dyoung on 4/8/17.
*
* This class tracks whether multiple distinct BLE packets have been seen, with the purpose of
* determining if the Android device supports detecting multiple distinct packets in a single scan.
* Some older devices are not capable of this (e.g. Nexus 4, Moto G1), so detecting multiple packets
* requires stopping and restarting scanning on these devices. This allows detecting if that is
* neessary
*/
public class DistinctPacketDetector {
// Sanity limit for the number of packets to track, so we don't use too much memory
private static final int MAX_PACKETS_TO_TRACK = 1000;
protected Set<ByteBuffer> mDistinctPacketsDetected = new HashSet<ByteBuffer>();

public void clearDetections() {
mDistinctPacketsDetected.clear();
}

public boolean isPacketDistinct(String originMacAddress, byte[] scanRecord) {
byte[] macBytes = originMacAddress.getBytes();
ByteBuffer buffer = ByteBuffer.allocate(macBytes.length+scanRecord.length);
buffer.put(macBytes);
buffer.put(scanRecord);
buffer.rewind(); // rewind puts position back to beginning so .equals and .hashCode work

boolean distinct = !mDistinctPacketsDetected.contains(buffer);
if (mDistinctPacketsDetected.size() == MAX_PACKETS_TO_TRACK) {
return mDistinctPacketsDetected.contains(buffer);
}
else {
return mDistinctPacketsDetected.add(buffer);
}
}

}
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
package org.altbeacon.beacon.service.scanner;
import org.junit.AfterClass;
import org.junit.BeforeClass;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.robolectric.RobolectricTestRunner;
import org.robolectric.annotation.Config;
import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertTrue;

@Config(sdk = 18)

@RunWith(RobolectricTestRunner.class)
public class DistinctPacketDetectorTest {
@BeforeClass
public static void testSetup() {
}

@AfterClass
public static void testCleanup() {

}

@Test
public void testSecondDuplicatePacketIsNotDistinct() throws Exception {
DistinctPacketDetector dpd = new DistinctPacketDetector();
dpd.isPacketDistinct("01:02:03:04:05:06", new byte[] {0x01, 0x02});
boolean secondResult = dpd.isPacketDistinct("01:02:03:04:05:06", new byte[] {0x01, 0x02});
assertFalse("second call with same packet should not be distinct", secondResult);
}

@Test
public void testSecondNonDuplicatePacketIsDistinct() throws Exception {
DistinctPacketDetector dpd = new DistinctPacketDetector();
dpd.isPacketDistinct("01:02:03:04:05:06", new byte[] {0x01, 0x02});
boolean secondResult = dpd.isPacketDistinct("01:02:03:04:05:06", new byte[] {0x03, 0x04});
assertTrue("second call with different packet should be distinct", secondResult);
}

@Test
public void testSamePacketForDifferentMacIsDistinct() throws Exception {
DistinctPacketDetector dpd = new DistinctPacketDetector();
dpd.isPacketDistinct("01:02:03:04:05:06", new byte[] {0x01, 0x02});
boolean secondResult = dpd.isPacketDistinct("01:01:01:01:01:01", new byte[] {0x01, 0x02});
assertTrue("second packet with different mac should be distinct", secondResult);
}

@Test
public void clearingDetectionsPreventsDistinctDetection() throws Exception {
DistinctPacketDetector dpd = new DistinctPacketDetector();
dpd.isPacketDistinct("01:02:03:04:05:06", new byte[] {0x01, 0x02});
dpd.clearDetections();
boolean secondResult = dpd.isPacketDistinct("01:02:03:04:05:06", new byte[] {0x01, 0x02});
assertTrue("second call with same packet after clear should be distinct", secondResult);
}

}

0 comments on commit 342bfff

Please sign in to comment.