diff options
3 files changed, 810 insertions, 135 deletions
| diff --git a/core/tests/coretests/src/android/bluetooth/BluetoothStressTest.java b/core/tests/coretests/src/android/bluetooth/BluetoothStressTest.java index d8d9eba14dc3..f019599a4716 100644 --- a/core/tests/coretests/src/android/bluetooth/BluetoothStressTest.java +++ b/core/tests/coretests/src/android/bluetooth/BluetoothStressTest.java @@ -42,6 +42,10 @@ public class BluetoothStressTest extends InstrumentationTestCase {      public void testEnable() {          int iterations = BluetoothTestRunner.sEnableIterations; +        if (iterations == 0) { +            return; +        } +          BluetoothAdapter adapter = BluetoothAdapter.getDefaultAdapter();          for (int i = 0; i < iterations; i++) { @@ -53,6 +57,10 @@ public class BluetoothStressTest extends InstrumentationTestCase {      public void testDiscoverable() {          int iterations = BluetoothTestRunner.sDiscoverableIterations; +        if (iterations == 0) { +            return; +        } +          BluetoothAdapter adapter = BluetoothAdapter.getDefaultAdapter();          mTestUtils.enable(adapter); @@ -67,6 +75,10 @@ public class BluetoothStressTest extends InstrumentationTestCase {      public void testScan() {          int iterations = BluetoothTestRunner.sScanIterations; +        if (iterations == 0) { +            return; +        } +          BluetoothAdapter adapter = BluetoothAdapter.getDefaultAdapter();          mTestUtils.enable(adapter); @@ -78,4 +90,67 @@ public class BluetoothStressTest extends InstrumentationTestCase {          mTestUtils.disable(adapter);      } + +    public void testPair() { +        int iterations = BluetoothTestRunner.sPairIterations; +        if (iterations == 0) { +            return; +        } + +        BluetoothAdapter adapter = BluetoothAdapter.getDefaultAdapter(); +        BluetoothDevice device = adapter.getRemoteDevice(BluetoothTestRunner.sPairAddress); +        mTestUtils.enable(adapter); + +        for (int i = 0; i < iterations; i++) { +            mTestUtils.writeOutput("pair iteration " + (i + 1) + " of " + iterations); +            mTestUtils.pair(adapter, device, BluetoothTestRunner.sPairPasskey, +                    BluetoothTestRunner.sPairPin); +            mTestUtils.unpair(adapter, device); +        } +        mTestUtils.disable(adapter); +    } + +    public void testConnectA2dp() { +        int iterations = BluetoothTestRunner.sConnectA2dpIterations; +        if (iterations == 0) { +            return; +        } + +        BluetoothAdapter adapter = BluetoothAdapter.getDefaultAdapter(); +        BluetoothDevice device = adapter.getRemoteDevice(BluetoothTestRunner.sA2dpAddress); +        mTestUtils.enable(adapter); +        mTestUtils.pair(adapter, device, BluetoothTestRunner.sPairPasskey, +                BluetoothTestRunner.sPairPin); + +        for (int i = 0; i < iterations; i++) { +            mTestUtils.writeOutput("connectA2dp iteration " + (i + 1) + " of " + iterations); +            mTestUtils.connectProfile(adapter, device, BluetoothProfile.A2DP); +            mTestUtils.disconnectProfile(adapter, device, BluetoothProfile.A2DP); +        } + +        // TODO: Unpair from device if device can accept pairing after unpairing +        mTestUtils.disable(adapter); +    } + +    public void testConnectHeadset() { +        int iterations = BluetoothTestRunner.sConnectHeadsetIterations; +        if (iterations == 0) { +            return; +        } + +        BluetoothAdapter adapter = BluetoothAdapter.getDefaultAdapter(); +        BluetoothDevice device = adapter.getRemoteDevice(BluetoothTestRunner.sHeadsetAddress); +        mTestUtils.enable(adapter); +        mTestUtils.pair(adapter, device, BluetoothTestRunner.sPairPasskey, +                BluetoothTestRunner.sPairPin); + +        for (int i = 0; i < iterations; i++) { +            mTestUtils.writeOutput("connectHeadset iteration " + (i + 1) + " of " + iterations); +            mTestUtils.connectProfile(adapter, device, BluetoothProfile.HEADSET); +            mTestUtils.disconnectProfile(adapter, device, BluetoothProfile.HEADSET); +        } + +        // TODO: Unpair from device if device can accept pairing after unpairing +        mTestUtils.disable(adapter); +    }  } diff --git a/core/tests/coretests/src/android/bluetooth/BluetoothTestRunner.java b/core/tests/coretests/src/android/bluetooth/BluetoothTestRunner.java index cf0ff99d1283..3e589fcb826c 100644 --- a/core/tests/coretests/src/android/bluetooth/BluetoothTestRunner.java +++ b/core/tests/coretests/src/android/bluetooth/BluetoothTestRunner.java @@ -21,11 +21,24 @@ import junit.framework.TestSuite;  import android.os.Bundle;  import android.test.InstrumentationTestRunner;  import android.test.InstrumentationTestSuite; +import android.util.Log;  public class BluetoothTestRunner extends InstrumentationTestRunner { +    private static final String TAG = "BluetoothTestRunner"; +      public static int sEnableIterations = 100;      public static int sDiscoverableIterations = 1000;      public static int sScanIterations = 1000; +    public static int sPairIterations = 100; +    public static int sConnectHeadsetIterations = 100; +    public static int sConnectA2dpIterations = 100; + +    public static String sPairAddress = ""; +    public static String sHeadsetAddress = ""; +    public static String sA2dpAddress = ""; + +    public static byte[] sPairPin = {'1', '2', '3', '4'}; +    public static int sPairPasskey = 123456;      @Override      public TestSuite getAllTests() { @@ -41,8 +54,6 @@ public class BluetoothTestRunner extends InstrumentationTestRunner {      @Override      public void onCreate(Bundle arguments) { -        super.onCreate(arguments); -          String val = arguments.getString("enable_iterations");          if (val != null) {              try { @@ -69,5 +80,76 @@ public class BluetoothTestRunner extends InstrumentationTestRunner {                  // Invalid argument, fall back to default value              }          } + +        val = arguments.getString("pair_iterations"); +        if (val != null) { +            try { +                sPairIterations = Integer.parseInt(val); +            } catch (NumberFormatException e) { +                // Invalid argument, fall back to default value +            } +        } + +        val = arguments.getString("connect_a2dp_iterations"); +        if (val != null) { +            try { +                sConnectA2dpIterations = Integer.parseInt(val); +            } catch (NumberFormatException e) { +                // Invalid argument, fall back to default value +            } +        } + +        val = arguments.getString("connect_headset_iterations"); +        if (val != null) { +            try { +                sConnectHeadsetIterations = Integer.parseInt(val); +            } catch (NumberFormatException e) { +                // Invalid argument, fall back to default value +            } +        } + +        val = arguments.getString("pair_address"); +        if (val != null) { +            sPairAddress = val; +        } + +        val = arguments.getString("headset_address"); +        if (val != null) { +            sHeadsetAddress = val; +        } + +        val = arguments.getString("a2dp_address"); +        if (val != null) { +            sA2dpAddress = val; +        } + +        val = arguments.getString("pair_pin"); +        if (val != null) { +            sPairPin = BluetoothDevice.convertPinToBytes(val); +        } + +        val = arguments.getString("pair_passkey"); +        if (val != null) { +            try { +                sPairPasskey = Integer.parseInt(val); +            } catch (NumberFormatException e) { +                // Invalid argument, fall back to default value +            } +        } + +        Log.i(TAG, String.format("enable_iterations=%d", sEnableIterations)); +        Log.i(TAG, String.format("discoverable_iterations=%d", sDiscoverableIterations)); +        Log.i(TAG, String.format("scan_iterations=%d", sScanIterations)); +        Log.i(TAG, String.format("pair_iterations=%d", sPairIterations)); +        Log.i(TAG, String.format("connect_a2dp_iterations=%d", sConnectA2dpIterations)); +        Log.i(TAG, String.format("connect_headset_iterations=%d", sConnectHeadsetIterations)); +        Log.i(TAG, String.format("pair_address=%s", sPairAddress)); +        Log.i(TAG, String.format("a2dp_address=%s", sA2dpAddress)); +        Log.i(TAG, String.format("headset_address=%s", sHeadsetAddress)); +        Log.i(TAG, String.format("pair_pin=%s", new String(sPairPin))); +        Log.i(TAG, String.format("pair_passkey=%d", sPairPasskey)); + +        // Call onCreate last since we want to set the static variables first. +        super.onCreate(arguments);      }  } diff --git a/core/tests/coretests/src/android/bluetooth/BluetoothTestUtils.java b/core/tests/coretests/src/android/bluetooth/BluetoothTestUtils.java index cddf63da5261..328891c004bc 100644 --- a/core/tests/coretests/src/android/bluetooth/BluetoothTestUtils.java +++ b/core/tests/coretests/src/android/bluetooth/BluetoothTestUtils.java @@ -29,6 +29,8 @@ import java.io.BufferedWriter;  import java.io.File;  import java.io.FileWriter;  import java.io.IOException; +import java.util.ArrayList; +import java.util.List;  public class BluetoothTestUtils extends Assert { @@ -57,89 +59,277 @@ public class BluetoothTestUtils extends Assert {       */      private static final int CANCEL_DISCOVERY_TIMEOUT = 5000; -    private static final int DISCOVERY_STARTED_FLAG = 1; -    private static final int DISCOVERY_FINISHED_FLAG = 1 << 1; -    private static final int SCAN_MODE_NONE_FLAG = 1 << 2; -    private static final int SCAN_MODE_CONNECTABLE_FLAG = 1 << 3; -    private static final int SCAN_MODE_CONNECTABLE_DISCOVERABLE_FLAG = 1 << 4; -    private static final int STATE_OFF_FLAG = 1 << 5; -    private static final int STATE_TURNING_ON_FLAG = 1 << 6; -    private static final int STATE_ON_FLAG = 1 << 7; -    private static final int STATE_TURNING_OFF_FLAG = 1 << 8; +    /** +     * Timeout for {@link BluetoothDevice#createBond()} in ms. +     */ +    private static final int PAIR_TIMEOUT = 20000; + +    /** +     * Timeout for {@link BluetoothDevice#removeBond()} in ms. +     */ +    private static final int UNPAIR_TIMEOUT = 20000; + +    /** +     * Timeout for {@link BluetoothProfile#connect(BluetoothDevice)} in ms. +     */ +    private static final int CONNECT_PROFILE_TIMEOUT = 20000; + +    /** +     * Timeout for {@link BluetoothProfile#disconnect(BluetoothDevice)} in ms. +     */ +    private static final int DISCONNECT_PROFILE_TIMEOUT = 20000; + +    /** +     * Timeout to connect a profile proxy in ms. +     */ +    private static final int CONNECT_PROXY_TIMEOUT = 5000;      /**       * Time between polls in ms.       */      private static final int POLL_TIME = 100; -    private Context mContext; +    private abstract class FlagReceiver extends BroadcastReceiver { +        private int mExpectedFlags = 0; +        private int mFiredFlags = 0; +        private long mCompletedTime = -1; -    private BufferedWriter mOutputWriter; +        public FlagReceiver(int expectedFlags) { +            mExpectedFlags = expectedFlags; +        } -    private String mOutputFile; -    private String mTag; +        public int getFiredFlags() { +            synchronized (this) { +                return mFiredFlags; +            } +        } -    private class BluetoothReceiver extends BroadcastReceiver { -        private int mFiredFlags = 0; +        public long getCompletedTime() { +            synchronized (this) { +                return mCompletedTime; +            } +        } + +        protected void setFiredFlag(int flag) { +            synchronized (this) { +                mFiredFlags |= flag; +                if (mFiredFlags == mExpectedFlags) { +                    mCompletedTime = System.currentTimeMillis(); +                } +            } +        } +    } + +    private class BluetoothReceiver extends FlagReceiver { +        private static final int DISCOVERY_STARTED_FLAG = 1; +        private static final int DISCOVERY_FINISHED_FLAG = 1 << 1; +        private static final int SCAN_MODE_NONE_FLAG = 1 << 2; +        private static final int SCAN_MODE_CONNECTABLE_FLAG = 1 << 3; +        private static final int SCAN_MODE_CONNECTABLE_DISCOVERABLE_FLAG = 1 << 4; +        private static final int STATE_OFF_FLAG = 1 << 5; +        private static final int STATE_TURNING_ON_FLAG = 1 << 6; +        private static final int STATE_ON_FLAG = 1 << 7; +        private static final int STATE_TURNING_OFF_FLAG = 1 << 8; + +        public BluetoothReceiver(int expectedFlags) { +            super(expectedFlags); +        }          @Override          public void onReceive(Context context, Intent intent) { -            synchronized (this) { -                if (BluetoothAdapter.ACTION_DISCOVERY_STARTED.equals(intent.getAction())) { -                    mFiredFlags |= DISCOVERY_STARTED_FLAG; -                } else if (BluetoothAdapter.ACTION_DISCOVERY_FINISHED.equals(intent.getAction())) { -                    mFiredFlags |= DISCOVERY_FINISHED_FLAG; -                } else if (BluetoothAdapter.ACTION_SCAN_MODE_CHANGED.equals(intent.getAction())) { -                    int mode = intent.getIntExtra(BluetoothAdapter.EXTRA_SCAN_MODE, -                            BluetoothAdapter.ERROR); -                    assertNotSame(mode, BluetoothAdapter.ERROR); -                    switch (mode) { -                        case BluetoothAdapter.SCAN_MODE_NONE: -                            mFiredFlags |= SCAN_MODE_NONE_FLAG; -                            break; -                        case BluetoothAdapter.SCAN_MODE_CONNECTABLE: -                            mFiredFlags |= SCAN_MODE_CONNECTABLE_FLAG; -                            break; -                        case BluetoothAdapter.SCAN_MODE_CONNECTABLE_DISCOVERABLE: -                            mFiredFlags |= SCAN_MODE_CONNECTABLE_DISCOVERABLE_FLAG; -                            break; -                    } -                } else if (BluetoothAdapter.ACTION_STATE_CHANGED.equals(intent.getAction())) { -                    int state = intent.getIntExtra(BluetoothAdapter.EXTRA_STATE, -                            BluetoothAdapter.ERROR); -                    assertNotSame(state, BluetoothAdapter.ERROR); -                    switch (state) { -                        case BluetoothAdapter.STATE_OFF: -                            mFiredFlags |= STATE_OFF_FLAG; -                            break; -                        case BluetoothAdapter.STATE_TURNING_ON: -                            mFiredFlags |= STATE_TURNING_ON_FLAG; -                            break; -                        case BluetoothAdapter.STATE_ON: -                            mFiredFlags |= STATE_ON_FLAG; -                            break; -                        case BluetoothAdapter.STATE_TURNING_OFF: -                            mFiredFlags |= STATE_TURNING_OFF_FLAG; -                            break; -                    } +            Log.i("BT", intent.toString()); +            if (BluetoothAdapter.ACTION_DISCOVERY_STARTED.equals(intent.getAction())) { +                setFiredFlag(DISCOVERY_STARTED_FLAG); +            } else if (BluetoothAdapter.ACTION_DISCOVERY_FINISHED.equals(intent.getAction())) { +                setFiredFlag(DISCOVERY_FINISHED_FLAG); +            } else if (BluetoothAdapter.ACTION_SCAN_MODE_CHANGED.equals(intent.getAction())) { +                int mode = intent.getIntExtra(BluetoothAdapter.EXTRA_SCAN_MODE, -1); +                assertNotSame(-1, mode); +                switch (mode) { +                    case BluetoothAdapter.SCAN_MODE_NONE: +                        setFiredFlag(SCAN_MODE_NONE_FLAG); +                        break; +                    case BluetoothAdapter.SCAN_MODE_CONNECTABLE: +                        setFiredFlag(SCAN_MODE_CONNECTABLE_FLAG); +                        break; +                    case BluetoothAdapter.SCAN_MODE_CONNECTABLE_DISCOVERABLE: +                        setFiredFlag(SCAN_MODE_CONNECTABLE_DISCOVERABLE_FLAG); +                        break; +                } +            } else if (BluetoothAdapter.ACTION_STATE_CHANGED.equals(intent.getAction())) { +                int state = intent.getIntExtra(BluetoothAdapter.EXTRA_STATE, -1); +                assertNotSame(-1, state); +                switch (state) { +                    case BluetoothAdapter.STATE_OFF: +                        setFiredFlag(STATE_OFF_FLAG); +                        break; +                    case BluetoothAdapter.STATE_TURNING_ON: +                        setFiredFlag(STATE_TURNING_ON_FLAG); +                        break; +                    case BluetoothAdapter.STATE_ON: +                        setFiredFlag(STATE_ON_FLAG); +                        break; +                    case BluetoothAdapter.STATE_TURNING_OFF: +                        setFiredFlag(STATE_TURNING_OFF_FLAG); +                        break;                  }              }          } +    } -        public int getFiredFlags() { +    private class PairReceiver extends FlagReceiver { +        private static final int STATE_BONDED_FLAG = 1; +        private static final int STATE_BONDING_FLAG = 1 << 1; +        private static final int STATE_NONE_FLAG = 1 << 2; + +        private BluetoothDevice mDevice; +        private int mPasskey; +        private byte[] mPin; + +        public PairReceiver(BluetoothDevice device, int passkey, byte[] pin, int expectedFlags) { +            super(expectedFlags); + +            mDevice = device; +            mPasskey = passkey; +            mPin = pin; +        } + +        @Override +        public void onReceive(Context context, Intent intent) { +            if (!mDevice.equals(intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE))) { +                return; +            } + +            if (BluetoothDevice.ACTION_PAIRING_REQUEST.equals(intent.getAction())) { +                int varient = intent.getIntExtra(BluetoothDevice.EXTRA_PAIRING_VARIANT, -1); +                assertNotSame(-1, varient); +                switch(varient) { +                    case BluetoothDevice.PAIRING_VARIANT_PIN: +                        mDevice.setPin(mPin); +                        break; +                    case BluetoothDevice.PAIRING_VARIANT_PASSKEY: +                        mDevice.setPasskey(mPasskey); +                        break; +                    case BluetoothDevice.PAIRING_VARIANT_PASSKEY_CONFIRMATION: +                    case BluetoothDevice.PAIRING_VARIANT_CONSENT: +                        mDevice.setPairingConfirmation(true); +                        break; +                    case BluetoothDevice.PAIRING_VARIANT_OOB_CONSENT: +                        mDevice.setRemoteOutOfBandData(); +                        break; +                } +            } else if (BluetoothDevice.ACTION_BOND_STATE_CHANGED.equals(intent.getAction())) { +                int state = intent.getIntExtra(BluetoothDevice.EXTRA_BOND_STATE, -1); +                assertNotSame(-1, state); +                switch (state) { +                    case BluetoothDevice.BOND_NONE: +                        setFiredFlag(STATE_NONE_FLAG); +                        break; +                    case BluetoothDevice.BOND_BONDING: +                        setFiredFlag(STATE_BONDING_FLAG); +                        break; +                    case BluetoothDevice.BOND_BONDED: +                        setFiredFlag(STATE_BONDED_FLAG); +                        break; +                } +            } +        } +    } + +    private class ConnectProfileReceiver extends FlagReceiver { +        private static final int STATE_DISCONNECTED_FLAG = 1; +        private static final int STATE_CONNECTING_FLAG = 1 << 1; +        private static final int STATE_CONNECTED_FLAG = 1 << 2; +        private static final int STATE_DISCONNECTING_FLAG = 1 << 3; + +        private BluetoothDevice mDevice; +        private int mProfile; +        private String mConnectionAction; + +        public ConnectProfileReceiver(BluetoothDevice device, int profile, int expectedFlags) { +            super(expectedFlags); + +            mDevice = device; +            mProfile = profile; + +            switch(mProfile) { +                case BluetoothProfile.A2DP: +                    mConnectionAction = BluetoothA2dp.ACTION_CONNECTION_STATE_CHANGED; +                    break; +                case BluetoothProfile.HEADSET: +                    mConnectionAction = BluetoothHeadset.ACTION_CONNECTION_STATE_CHANGED; +                    break; +                default: +                    mConnectionAction = null; +            } +        } + +        @Override +        public void onReceive(Context context, Intent intent) { + +            if (mConnectionAction != null && mConnectionAction.equals(intent.getAction())) { +                if (!mDevice.equals(intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE))) { +                    return; +                } + +                int state = intent.getIntExtra(BluetoothProfile.EXTRA_STATE, -1); +                assertNotSame(-1, state); +                switch (state) { +                    case BluetoothProfile.STATE_DISCONNECTED: +                        setFiredFlag(STATE_DISCONNECTED_FLAG); +                        break; +                    case BluetoothProfile.STATE_CONNECTING: +                        setFiredFlag(STATE_CONNECTING_FLAG); +                        break; +                    case BluetoothProfile.STATE_CONNECTED: +                        setFiredFlag(STATE_CONNECTED_FLAG); +                        break; +                    case BluetoothProfile.STATE_DISCONNECTING: +                        setFiredFlag(STATE_DISCONNECTING_FLAG); +                        break; +                } +            } +        } +    } + +    private BluetoothProfile.ServiceListener mServiceListener = +            new BluetoothProfile.ServiceListener() { +        public void onServiceConnected(int profile, BluetoothProfile proxy) {              synchronized (this) { -                return mFiredFlags; +                switch (profile) { +                    case BluetoothProfile.A2DP: +                        mA2dp = (BluetoothA2dp) proxy; +                        break; +                    case BluetoothProfile.HEADSET: +                        mHeadset = (BluetoothHeadset) proxy; +                        break; +                }              }          } -        public void resetFiredFlags() { +        public void onServiceDisconnected(int profile) {              synchronized (this) { -                mFiredFlags = 0; +                switch (profile) { +                    case BluetoothProfile.A2DP: +                        mA2dp = null; +                        break; +                    case BluetoothProfile.HEADSET: +                        mHeadset = null; +                        break; +                }              }          } -    } +    }; + +    private List<BroadcastReceiver> mReceivers = new ArrayList<BroadcastReceiver>(); + +    private BufferedWriter mOutputWriter; +    private String mTag; +    private String mOutputFile; -    private BluetoothReceiver mReceiver = new BluetoothReceiver(); +    private Context mContext; +    private BluetoothA2dp mA2dp; +    private BluetoothHeadset mHeadset;      public BluetoothTestUtils(Context context, String tag) {          this(context, tag, null); @@ -161,17 +351,12 @@ public class BluetoothTestUtils extends Assert {                  mOutputWriter = null;              }          } - -        IntentFilter filter = new IntentFilter(); -        filter.addAction(BluetoothAdapter.ACTION_DISCOVERY_FINISHED); -        filter.addAction(BluetoothAdapter.ACTION_DISCOVERY_STARTED); -        filter.addAction(BluetoothAdapter.ACTION_SCAN_MODE_CHANGED); -        filter.addAction(BluetoothAdapter.ACTION_STATE_CHANGED); -        mContext.registerReceiver(mReceiver, filter);      }      public void close() { -        mContext.unregisterReceiver(mReceiver); +        while (!mReceivers.isEmpty()) { +            mContext.unregisterReceiver(mReceivers.remove(0)); +        }          if (mOutputWriter != null) {              try { @@ -183,25 +368,30 @@ public class BluetoothTestUtils extends Assert {      }      public void enable(BluetoothAdapter adapter) { -        int mask = STATE_TURNING_ON_FLAG | STATE_ON_FLAG | SCAN_MODE_CONNECTABLE_FLAG; -        mReceiver.resetFiredFlags(); +        int mask = (BluetoothReceiver.STATE_TURNING_ON_FLAG | BluetoothReceiver.STATE_ON_FLAG +                | BluetoothReceiver.SCAN_MODE_CONNECTABLE_FLAG); +        long start = -1; +        BluetoothReceiver receiver = getBluetoothReceiver(mask);          int state = adapter.getState();          switch (state) {              case BluetoothAdapter.STATE_ON:                  assertTrue(adapter.isEnabled()); +                removeReceiver(receiver);                  return; +            case BluetoothAdapter.STATE_TURNING_ON: +                assertFalse(adapter.isEnabled()); +                mask = 0; // Don't check for received intents since we might have missed them. +                break;              case BluetoothAdapter.STATE_OFF:              case BluetoothAdapter.STATE_TURNING_OFF:                  assertFalse(adapter.isEnabled()); +                start = System.currentTimeMillis();                  assertTrue(adapter.enable());                  break; -            case BluetoothAdapter.STATE_TURNING_ON: -                assertFalse(adapter.isEnabled()); -                mask = 0; // Don't check for received intents since we might have missed them. -                break;              default: -                fail("enable() invalid state: state=" + state); +                removeReceiver(receiver); +                fail(String.format("enable() invalid state: state=%d", state));          }          long s = System.currentTimeMillis(); @@ -209,10 +399,14 @@ public class BluetoothTestUtils extends Assert {              state = adapter.getState();              if (state == BluetoothAdapter.STATE_ON) {                  assertTrue(adapter.isEnabled()); -                if ((mReceiver.getFiredFlags() & mask) == mask) { -                    mReceiver.resetFiredFlags(); -                    writeOutput(String.format("enable() completed in %d ms", -                            (System.currentTimeMillis() - s))); +                if ((receiver.getFiredFlags() & mask) == mask) { +                    long finish = receiver.getCompletedTime(); +                    if (start != -1 && finish != -1) { +                        writeOutput(String.format("enable() completed in %d ms", (finish - start))); +                    } else { +                        writeOutput("enable() completed"); +                    } +                    removeReceiver(receiver);                      return;                  }              } else { @@ -222,27 +416,32 @@ public class BluetoothTestUtils extends Assert {              sleep(POLL_TIME);          } -        int firedFlags = mReceiver.getFiredFlags(); -        mReceiver.resetFiredFlags(); +        int firedFlags = receiver.getFiredFlags(); +        removeReceiver(receiver);          fail(String.format("enable() timeout: state=%d (expected %d), flags=0x%x (expected 0x%x)",                  state, BluetoothAdapter.STATE_ON, firedFlags, mask));      }      public void disable(BluetoothAdapter adapter) { -        int mask = STATE_TURNING_OFF_FLAG | STATE_OFF_FLAG | SCAN_MODE_NONE_FLAG; -        mReceiver.resetFiredFlags(); +        int mask = (BluetoothReceiver.STATE_TURNING_OFF_FLAG | BluetoothReceiver.STATE_OFF_FLAG +                | BluetoothReceiver.SCAN_MODE_NONE_FLAG); +        long start = -1; +        BluetoothReceiver receiver = getBluetoothReceiver(mask);          int state = adapter.getState();          switch (state) {              case BluetoothAdapter.STATE_OFF:                  assertFalse(adapter.isEnabled()); +                removeReceiver(receiver);                  return; -            case BluetoothAdapter.STATE_ON: -                assertTrue(adapter.isEnabled()); -                assertTrue(adapter.disable()); -                break;              case BluetoothAdapter.STATE_TURNING_ON:                  assertFalse(adapter.isEnabled()); +                start = System.currentTimeMillis(); +                assertTrue(adapter.disable()); +                break; +            case BluetoothAdapter.STATE_ON: +                assertTrue(adapter.isEnabled()); +                start = System.currentTimeMillis();                  assertTrue(adapter.disable());                  break;              case BluetoothAdapter.STATE_TURNING_OFF: @@ -250,7 +449,8 @@ public class BluetoothTestUtils extends Assert {                  mask = 0; // Don't check for received intents since we might have missed them.                  break;              default: -                fail("disable() invalid state: state=" + state); +                removeReceiver(receiver); +                fail(String.format("disable() invalid state: state=%d", state));          }          long s = System.currentTimeMillis(); @@ -258,10 +458,15 @@ public class BluetoothTestUtils extends Assert {              state = adapter.getState();              if (state == BluetoothAdapter.STATE_OFF) {                  assertFalse(adapter.isEnabled()); -                if ((mReceiver.getFiredFlags() & mask) == mask) { -                    mReceiver.resetFiredFlags(); -                    writeOutput(String.format("disable() completed in %d ms", -                            (System.currentTimeMillis() - s))); +                if ((receiver.getFiredFlags() & mask) == mask) { +                    long finish = receiver.getCompletedTime(); +                    if (start != -1 && finish != -1) { +                        writeOutput(String.format("disable() completed in %d ms", +                                (finish - start))); +                    } else { +                        writeOutput("disable() completed"); +                    } +                    removeReceiver(receiver);                      return;                  }              } else { @@ -271,15 +476,14 @@ public class BluetoothTestUtils extends Assert {              sleep(POLL_TIME);          } -        int firedFlags = mReceiver.getFiredFlags(); -        mReceiver.resetFiredFlags(); +        int firedFlags = receiver.getFiredFlags(); +        removeReceiver(receiver);          fail(String.format("disable() timeout: state=%d (expected %d), flags=0x%x (expected 0x%x)",                  state, BluetoothAdapter.STATE_OFF, firedFlags, mask));      }      public void discoverable(BluetoothAdapter adapter) { -        int mask = SCAN_MODE_CONNECTABLE_DISCOVERABLE_FLAG; -        mReceiver.resetFiredFlags(); +        int mask = BluetoothReceiver.SCAN_MODE_CONNECTABLE_DISCOVERABLE_FLAG;          if (!adapter.isEnabled()) {              fail("discoverable() bluetooth not enabled"); @@ -290,35 +494,36 @@ public class BluetoothTestUtils extends Assert {              return;          } -        assertEquals(scanMode, BluetoothAdapter.SCAN_MODE_CONNECTABLE); +        BluetoothReceiver receiver = getBluetoothReceiver(mask); + +        assertEquals(BluetoothAdapter.SCAN_MODE_CONNECTABLE, scanMode); +        long start = System.currentTimeMillis();          assertTrue(adapter.setScanMode(BluetoothAdapter.SCAN_MODE_CONNECTABLE_DISCOVERABLE)); -        long s = System.currentTimeMillis(); -        while (System.currentTimeMillis() - s < SET_SCAN_MODE_TIMEOUT) { +        while (System.currentTimeMillis() - start < SET_SCAN_MODE_TIMEOUT) {              scanMode = adapter.getScanMode();              if (scanMode == BluetoothAdapter.SCAN_MODE_CONNECTABLE_DISCOVERABLE) { -                if ((mReceiver.getFiredFlags() & mask) == mask) { -                    mReceiver.resetFiredFlags(); +                if ((receiver.getFiredFlags() & mask) == mask) {                      writeOutput(String.format("discoverable() completed in %d ms", -                            (System.currentTimeMillis() - s))); +                            (receiver.getCompletedTime() - start))); +                    removeReceiver(receiver);                      return;                  }              } else { -                assertEquals(scanMode, BluetoothAdapter.SCAN_MODE_CONNECTABLE); +                assertEquals(BluetoothAdapter.SCAN_MODE_CONNECTABLE, scanMode);              }              sleep(POLL_TIME);          } -        int firedFlags = mReceiver.getFiredFlags(); -        mReceiver.resetFiredFlags(); +        int firedFlags = receiver.getFiredFlags(); +        removeReceiver(receiver);          fail(String.format("discoverable() timeout: scanMode=%d (expected %d), flags=0x%x "                  + "(expected 0x%x)", scanMode, BluetoothAdapter.SCAN_MODE_CONNECTABLE_DISCOVERABLE,                  firedFlags, mask));      }      public void undiscoverable(BluetoothAdapter adapter) { -        int mask = SCAN_MODE_CONNECTABLE_FLAG; -        mReceiver.resetFiredFlags(); +        int mask = BluetoothReceiver.SCAN_MODE_CONNECTABLE_FLAG;          if (!adapter.isEnabled()) {              fail("undiscoverable() bluetooth not enabled"); @@ -329,35 +534,36 @@ public class BluetoothTestUtils extends Assert {              return;          } -        assertEquals(scanMode, BluetoothAdapter.SCAN_MODE_CONNECTABLE_DISCOVERABLE); +        BluetoothReceiver receiver = getBluetoothReceiver(mask); + +        assertEquals(BluetoothAdapter.SCAN_MODE_CONNECTABLE_DISCOVERABLE, scanMode); +        long start = System.currentTimeMillis();          assertTrue(adapter.setScanMode(BluetoothAdapter.SCAN_MODE_CONNECTABLE)); -        long s = System.currentTimeMillis(); -        while (System.currentTimeMillis() - s < SET_SCAN_MODE_TIMEOUT) { +        while (System.currentTimeMillis() - start < SET_SCAN_MODE_TIMEOUT) {              scanMode = adapter.getScanMode();              if (scanMode == BluetoothAdapter.SCAN_MODE_CONNECTABLE) { -                if ((mReceiver.getFiredFlags() & mask) == mask) { -                    mReceiver.resetFiredFlags(); +                if ((receiver.getFiredFlags() & mask) == mask) {                      writeOutput(String.format("undiscoverable() completed in %d ms", -                            (System.currentTimeMillis() - s))); +                            (receiver.getCompletedTime() - start))); +                    removeReceiver(receiver);                      return;                  }              } else { -                assertEquals(scanMode, BluetoothAdapter.SCAN_MODE_CONNECTABLE_DISCOVERABLE); +                assertEquals(BluetoothAdapter.SCAN_MODE_CONNECTABLE_DISCOVERABLE, scanMode);              }              sleep(POLL_TIME);          } -        int firedFlags = mReceiver.getFiredFlags(); -        mReceiver.resetFiredFlags(); +        int firedFlags = receiver.getFiredFlags(); +        removeReceiver(receiver);          fail(String.format("undiscoverable() timeout: scanMode=%d (expected %d), flags=0x%x "                  + "(expected 0x%x)", scanMode, BluetoothAdapter.SCAN_MODE_CONNECTABLE, firedFlags,                  mask));      }      public void startScan(BluetoothAdapter adapter) { -        int mask = DISCOVERY_STARTED_FLAG; -        mReceiver.resetFiredFlags(); +        int mask = BluetoothReceiver.DISCOVERY_STARTED_FLAG;          if (!adapter.isEnabled()) {              fail("startScan() bluetooth not enabled"); @@ -367,28 +573,29 @@ public class BluetoothTestUtils extends Assert {              return;          } +        BluetoothReceiver receiver = getBluetoothReceiver(mask); + +        long start = System.currentTimeMillis();          assertTrue(adapter.startDiscovery()); -        long s = System.currentTimeMillis(); -        while (System.currentTimeMillis() - s < START_DISCOVERY_TIMEOUT) { -            if (adapter.isDiscovering() && ((mReceiver.getFiredFlags() & mask) == mask)) { -                mReceiver.resetFiredFlags(); +        while (System.currentTimeMillis() - start < START_DISCOVERY_TIMEOUT) { +            if (adapter.isDiscovering() && ((receiver.getFiredFlags() & mask) == mask)) {                  writeOutput(String.format("startScan() completed in %d ms", -                        (System.currentTimeMillis() - s))); +                        (receiver.getCompletedTime() - start))); +                removeReceiver(receiver);                  return;              }              sleep(POLL_TIME);          } -        int firedFlags = mReceiver.getFiredFlags(); -        mReceiver.resetFiredFlags(); +        int firedFlags = receiver.getFiredFlags(); +        removeReceiver(receiver);          fail(String.format("startScan() timeout: isDiscovering=%b, flags=0x%x (expected 0x%x)",                  adapter.isDiscovering(), firedFlags, mask));      }      public void stopScan(BluetoothAdapter adapter) { -        int mask = DISCOVERY_FINISHED_FLAG; -        mReceiver.resetFiredFlags(); +        int mask = BluetoothReceiver.DISCOVERY_FINISHED_FLAG;          if (!adapter.isEnabled()) {              fail("stopScan() bluetooth not enabled"); @@ -398,27 +605,277 @@ public class BluetoothTestUtils extends Assert {              return;          } +        BluetoothReceiver receiver = getBluetoothReceiver(mask); + +        long start = System.currentTimeMillis();          // TODO: put assertTrue() around cancelDiscovery() once it starts returning true.          adapter.cancelDiscovery(); -        long s = System.currentTimeMillis(); -        while (System.currentTimeMillis() - s < CANCEL_DISCOVERY_TIMEOUT) { -            if (!adapter.isDiscovering() && ((mReceiver.getFiredFlags() & mask) == mask)) { -                mReceiver.resetFiredFlags(); +        while (System.currentTimeMillis() - start < CANCEL_DISCOVERY_TIMEOUT) { +            if (!adapter.isDiscovering() && ((receiver.getFiredFlags() & mask) == mask)) {                  writeOutput(String.format("stopScan() completed in %d ms", -                        (System.currentTimeMillis() - s))); +                        (receiver.getCompletedTime() - start))); +                removeReceiver(receiver);                  return;              }              sleep(POLL_TIME);          } -        int firedFlags = mReceiver.getFiredFlags(); -        mReceiver.resetFiredFlags(); +        int firedFlags = receiver.getFiredFlags(); +        removeReceiver(receiver);          fail(String.format("stopScan() timeout: isDiscovering=%b, flags=0x%x (expected 0x%x)",                  adapter.isDiscovering(), firedFlags, mask));      } +    public void pair(BluetoothAdapter adapter, BluetoothDevice device, int passkey, byte[] pin) { +        int mask = PairReceiver.STATE_BONDING_FLAG | PairReceiver.STATE_BONDED_FLAG; +        long start = -1; + +        if (!adapter.isEnabled()) { +            fail("pair() bluetooth not enabled"); +        } + +        PairReceiver receiver = getPairReceiver(device, passkey, pin, mask); + +        int state = device.getBondState(); +        switch (state) { +            case BluetoothDevice.BOND_NONE: +                assertFalse(adapter.getBondedDevices().contains(device)); +                start = System.currentTimeMillis(); +                assertTrue(device.createBond()); +                break; +            case BluetoothDevice.BOND_BONDING: +                mask = 0; // Don't check for received intents since we might have missed them. +                break; +            case BluetoothDevice.BOND_BONDED: +                assertTrue(adapter.getBondedDevices().contains(device)); +                return; +            default: +                removeReceiver(receiver); +                fail(String.format("pair() invalid state: device=%s, state=%d", device, state)); +        } + +        long s = System.currentTimeMillis(); +        while (System.currentTimeMillis() - s < PAIR_TIMEOUT) { +            state = device.getBondState(); +            if (state == BluetoothDevice.BOND_BONDED) { +                assertTrue(adapter.getBondedDevices().contains(device)); +                if ((receiver.getFiredFlags() & mask) == mask) { +                    long finish = receiver.getCompletedTime(); +                    if (start != -1 && finish != -1) { +                        writeOutput(String.format("pair() completed in %d ms: device=%s", +                                (finish - start), device)); +                    } else { +                        writeOutput(String.format("pair() completed: device=%s", device)); +                    } +                    removeReceiver(receiver); +                    return; +                } +            } +            sleep(POLL_TIME); +        } + +        int firedFlags = receiver.getFiredFlags(); +        removeReceiver(receiver); +        fail(String.format("pair() timeout: device=%s, state=%d (expected %d), " +                + "flags=0x%x (expected 0x%x)", device, state, BluetoothDevice.BOND_BONDED, +                firedFlags, mask)); +    } + +    public void unpair(BluetoothAdapter adapter, BluetoothDevice device) { +        int mask = PairReceiver.STATE_NONE_FLAG; +        long start = -1; + +        if (!adapter.isEnabled()) { +            fail("unpair() bluetooth not enabled"); +        } + +        PairReceiver receiver = getPairReceiver(device, 0, null, mask); + +        int state = device.getBondState(); +        switch (state) { +            case BluetoothDevice.BOND_NONE: +                assertFalse(adapter.getBondedDevices().contains(device)); +                removeReceiver(receiver); +                return; +            case BluetoothDevice.BOND_BONDING: +                start = System.currentTimeMillis(); +                assertTrue(device.removeBond()); +                break; +            case BluetoothDevice.BOND_BONDED: +                assertTrue(adapter.getBondedDevices().contains(device)); +                start = System.currentTimeMillis(); +                assertTrue(device.removeBond()); +                break; +            default: +                removeReceiver(receiver); +                fail(String.format("unpair() invalid state: device=%s, state=%d", device, state)); +        } + +        long s = System.currentTimeMillis(); +        while (System.currentTimeMillis() - s < UNPAIR_TIMEOUT) { +            if (device.getBondState() == BluetoothDevice.BOND_NONE) { +                assertFalse(adapter.getBondedDevices().contains(device)); +                if ((receiver.getFiredFlags() & mask) == mask) { +                    long finish = receiver.getCompletedTime(); +                    if (start != -1 && finish != -1) { +                        writeOutput(String.format("unpair() completed in %d ms: device=%s", +                                (finish - start), device)); +                    } else { +                        writeOutput(String.format("unpair() completed: device=%s", device)); +                    } +                    removeReceiver(receiver); +                    return; +                } +            } +        } + +        int firedFlags = receiver.getFiredFlags(); +        removeReceiver(receiver); +        fail(String.format("unpair() timeout: device=%s, state=%d (expected %d), " +                + "flags=0x%x (expected 0x%x)", device, state, BluetoothDevice.BOND_BONDED, +                firedFlags, mask)); +    } + +    public void connectProfile(BluetoothAdapter adapter, BluetoothDevice device, int profile) { +        int mask = (ConnectProfileReceiver.STATE_CONNECTING_FLAG +                | ConnectProfileReceiver.STATE_CONNECTED_FLAG); +        long start = -1; + +        if (!adapter.isEnabled()) { +            fail(String.format("connectProfile() bluetooth not enabled: device=%s, profile=%d", +                    device, profile)); +        } + +        if (!adapter.getBondedDevices().contains(device)) { +            fail(String.format("connectProfile() device not paired: device=%s, profile=%d", +                    device, profile)); +        } + +        BluetoothProfile proxy = connectProxy(adapter, profile); +        if (proxy == null) { +            fail(String.format("connectProfile() unknown profile: device=%s, profile=%d", +                    device, profile)); +        } + +        ConnectProfileReceiver receiver = getConnectProfileReceiver(device, profile, mask); + +        int state = proxy.getConnectionState(device); +        switch (state) { +            case BluetoothProfile.STATE_CONNECTED: +                removeReceiver(receiver); +                return; +            case BluetoothProfile.STATE_CONNECTING: +                mask = 0; // Don't check for received intents since we might have missed them. +                break; +            case BluetoothProfile.STATE_DISCONNECTED: +            case BluetoothProfile.STATE_DISCONNECTING: +                start = System.currentTimeMillis(); +                assertTrue(proxy.connect(device)); +                break; +            default: +                removeReceiver(receiver); +                fail(String.format("connectProfile() invalid state: device=%s, profile=%d, " +                        + "state=%d", device, profile, state)); +        } + +        long s = System.currentTimeMillis(); +        while (System.currentTimeMillis() - s < CONNECT_PROFILE_TIMEOUT) { +            state = proxy.getConnectionState(device); +            if (state == BluetoothProfile.STATE_CONNECTED) { +                if ((receiver.getFiredFlags() & mask) == mask) { +                    long finish = receiver.getCompletedTime(); +                    if (start != -1 && finish != -1) { +                        writeOutput(String.format("connectProfile() completed in %d ms: " +                                +"device=%s, profile=%d", (finish - start), device, profile)); +                    } else { +                        writeOutput(String.format("connectProfile() completed: device=%s, " +                                + "profile=%d", device, profile)); +                    } +                    removeReceiver(receiver); +                    return; +                } +            } +            sleep(POLL_TIME); +        } + +        int firedFlags = receiver.getFiredFlags(); +        removeReceiver(receiver); +        fail(String.format("connectProfile() timeout: device=%s, profile=%s, " +                + "state=%d (expected %d), flags=0x%x (expected 0x%x)", device, profile, state, +                BluetoothProfile.STATE_CONNECTED, firedFlags, mask)); +    } + +    public void disconnectProfile(BluetoothAdapter adapter, BluetoothDevice device, int profile) { +        int mask = (ConnectProfileReceiver.STATE_DISCONNECTING_FLAG +                | ConnectProfileReceiver.STATE_DISCONNECTED_FLAG); +        long start = -1; + +        if (!adapter.isEnabled()) { +            fail(String.format("disconnectProfile() bluetooth not enabled: device=%s, profile=%d", +                    device, profile)); +        } + +        if (!adapter.getBondedDevices().contains(device)) { +            fail(String.format("disconnectProfile() device not paired: device=%s, profile=%d", +                    device, profile)); +        } + +        BluetoothProfile proxy = connectProxy(adapter, profile); +        if (proxy == null) { +            fail(String.format("disconnectProfile() unknown profile: device=%s, profile=%d", +                    device, profile)); +        } + +        ConnectProfileReceiver receiver = getConnectProfileReceiver(device, profile, mask); + +        int state = proxy.getConnectionState(device); +        switch (state) { +            case BluetoothProfile.STATE_CONNECTED: +            case BluetoothProfile.STATE_CONNECTING: +                start = System.currentTimeMillis(); +                assertTrue(proxy.disconnect(device)); +                break; +            case BluetoothProfile.STATE_DISCONNECTED: +                removeReceiver(receiver); +                return; +            case BluetoothProfile.STATE_DISCONNECTING: +                mask = 0; // Don't check for received intents since we might have missed them. +                break; +            default: +                removeReceiver(receiver); +                fail(String.format("disconnectProfile() invalid state: device=%s, profile=%d, " +                        + "state=%d", device, profile, state)); +        } + +        long s = System.currentTimeMillis(); +        while (System.currentTimeMillis() - s < DISCONNECT_PROFILE_TIMEOUT) { +            state = proxy.getConnectionState(device); +            if (state == BluetoothProfile.STATE_DISCONNECTED) { +                if ((receiver.getFiredFlags() & mask) == mask) { +                    long finish = receiver.getCompletedTime(); +                    if (start != -1 && finish != -1) { +                        writeOutput(String.format("disconnectProfile() completed in %d ms: " +                                +"device=%s, profile=%d", (finish - start), device, profile)); +                    } else { +                        writeOutput(String.format("disconnectProfile() completed: device=%s, " +                                + "profile=%d", device, profile)); +                    } +                    removeReceiver(receiver); +                    return; +                } +            } +            sleep(POLL_TIME); +        } + +        int firedFlags = receiver.getFiredFlags(); +        removeReceiver(receiver); +        fail(String.format("disconnectProfile() timeout: device=%s, profile=%s, " +                + "state=%d (expected %d), flags=0x%x (expected 0x%x)", device, profile, state, +                BluetoothProfile.STATE_DISCONNECTED, firedFlags, mask)); +    } +      public void writeOutput(String s) {          Log.i(mTag, s);          if (mOutputWriter == null) { @@ -432,6 +889,67 @@ public class BluetoothTestUtils extends Assert {          }      } +    private BluetoothReceiver getBluetoothReceiver(int expectedFlags) { +        BluetoothReceiver receiver = new BluetoothReceiver(expectedFlags); +        IntentFilter filter = new IntentFilter(); +        filter.addAction(BluetoothAdapter.ACTION_DISCOVERY_FINISHED); +        filter.addAction(BluetoothAdapter.ACTION_DISCOVERY_STARTED); +        filter.addAction(BluetoothAdapter.ACTION_SCAN_MODE_CHANGED); +        filter.addAction(BluetoothAdapter.ACTION_STATE_CHANGED); +        mContext.registerReceiver(receiver, filter); +        mReceivers.add(receiver); +        return receiver; +    } + +    private PairReceiver getPairReceiver(BluetoothDevice device, int passkey, byte[] pin, +            int expectedFlags) { +        PairReceiver receiver = new PairReceiver(device, passkey, pin, expectedFlags); +        IntentFilter filter = new IntentFilter(); +        filter.addAction(BluetoothDevice.ACTION_PAIRING_REQUEST); +        filter.addAction(BluetoothDevice.ACTION_BOND_STATE_CHANGED); +        mContext.registerReceiver(receiver, filter); +        mReceivers.add(receiver); +        return receiver; +    } + +    private ConnectProfileReceiver getConnectProfileReceiver(BluetoothDevice device, int profile, +            int expectedFlags) { +        ConnectProfileReceiver receiver = new ConnectProfileReceiver(device, profile, +                expectedFlags); +        IntentFilter filter = new IntentFilter(); +        filter.addAction(BluetoothA2dp.ACTION_CONNECTION_STATE_CHANGED); +        filter.addAction(BluetoothHeadset.ACTION_CONNECTION_STATE_CHANGED); +        mContext.registerReceiver(receiver, filter); +        mReceivers.add(receiver); +        return receiver; +    } + +    private void removeReceiver(BroadcastReceiver receiver) { +        mContext.unregisterReceiver(receiver); +        mReceivers.remove(receiver); +    } + +    private BluetoothProfile connectProxy(BluetoothAdapter adapter, int profile) { +        adapter.getProfileProxy(mContext, mServiceListener, profile); +        long s = System.currentTimeMillis(); +        switch (profile) { +            case BluetoothProfile.A2DP: +                while (mA2dp != null +                        && System.currentTimeMillis() - s < CONNECT_PROXY_TIMEOUT) { +                    sleep(POLL_TIME); +                } +                return mA2dp; +            case BluetoothProfile.HEADSET: +                while (mHeadset != null +                        && System.currentTimeMillis() - s < CONNECT_PROXY_TIMEOUT) { +                    sleep(POLL_TIME); +                } +                return mHeadset; +            default: +                return null; +        } +    } +      private void sleep(long time) {          try {              Thread.sleep(time); |