main.cc revision 5d80cb938d7fb3312870a047749cc839ba5bafca
1f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray//
2f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray//  Copyright (C) 2015 Google, Inc.
3f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray//
4f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray//  Licensed under the Apache License, Version 2.0 (the "License");
5f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray//  you may not use this file except in compliance with the License.
6f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray//  You may obtain a copy of the License at:
7f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray//
8f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray//  http://www.apache.org/licenses/LICENSE-2.0
9f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray//
10f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray//  Unless required by applicable law or agreed to in writing, software
11f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray//  distributed under the License is distributed on an "AS IS" BASIS,
12f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray//  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray//  See the License for the specific language governing permissions and
14f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray//  limitations under the License.
15f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray//
16f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray
17fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray#include <iostream>
18fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray#include <string>
19fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
201377f938abf30ab3104571fb8372c657e65237d0Pavlin Radoslavov#ifdef BT_LIBCHROME_NDEBUG
211377f938abf30ab3104571fb8372c657e65237d0Pavlin Radoslavov#define NDEBUG 1
221377f938abf30ab3104571fb8372c657e65237d0Pavlin Radoslavov#endif
231377f938abf30ab3104571fb8372c657e65237d0Pavlin Radoslavov
24ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray#include <base/at_exit.h>
25ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray#include <base/command_line.h>
26f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray#include <base/logging.h>
2739a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray#include <base/macros.h>
28ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski#include <base/strings/string_number_conversions.h>
295192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray#include <base/strings/string_split.h>
305192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray#include <base/strings/string_util.h>
312e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray#include <binder/IPCThreadState.h>
3239a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray#include <binder/ProcessState.h>
33f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray
34234138e2606dd7a54fbcc540643511abc0a3598dArman Uguray#include <bluetooth/adapter_state.h>
35234138e2606dd7a54fbcc540643511abc0a3598dArman Uguray#include <bluetooth/binder/IBluetooth.h>
36234138e2606dd7a54fbcc540643511abc0a3598dArman Uguray#include <bluetooth/binder/IBluetoothCallback.h>
374fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray#include <bluetooth/binder/IBluetoothGattClient.h>
384fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray#include <bluetooth/binder/IBluetoothGattClientCallback.h>
394fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray#include <bluetooth/binder/IBluetoothLowEnergy.h>
404fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray#include <bluetooth/binder/IBluetoothLowEnergyCallback.h>
41234138e2606dd7a54fbcc540643511abc0a3598dArman Uguray#include <bluetooth/low_energy_constants.h>
42ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski#include <bluetooth/scan_filter.h>
43ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski#include <bluetooth/scan_settings.h>
4487222e0e826216c69f6a9a5bfe77689561067474Arman Uguray#include <bluetooth/uuid.h>
45f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray
46fcf2e0391950a8b140082fbe78688fa89471fbedArman Ugurayusing namespace std;
47fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
48f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Ugurayusing android::sp;
49f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray
50f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Ugurayusing ipc::binder::IBluetooth;
514fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Ugurayusing ipc::binder::IBluetoothGattClient;
522e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Ugurayusing ipc::binder::IBluetoothLowEnergy;
53f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray
542117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguraynamespace {
552117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray
562117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray#define COLOR_OFF         "\x1B[0m"
572117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray#define COLOR_RED         "\x1B[0;91m"
582117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray#define COLOR_GREEN       "\x1B[0;92m"
592117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray#define COLOR_YELLOW      "\x1B[0;93m"
602117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray#define COLOR_BLUE        "\x1B[0;94m"
612117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray#define COLOR_MAGENTA     "\x1B[0;95m"
622117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray#define COLOR_BOLDGRAY    "\x1B[1;30m"
632117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray#define COLOR_BOLDWHITE   "\x1B[1;37m"
642117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray#define COLOR_BOLDYELLOW  "\x1B[1;93m"
6564401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski#define CLEAR_LINE        "\x1B[2K"
66fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
675192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray#define CHECK_ARGS_COUNT(args, op, num, msg) \
685192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  if (!(args.size() op num)) { \
695192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray    PrintError(msg); \
705192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray    return; \
715192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  }
725192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray#define CHECK_NO_ARGS(args) \
735192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_ARGS_COUNT(args, ==, 0, "Expected no arguments")
745192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
7539a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray// TODO(armansito): Clean up this code. Right now everything is in this
7639a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray// monolithic file. We should organize this into different classes for command
7739a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray// handling, console output/printing, callback handling, etc.
7839a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray// (See http://b/23387611)
7939a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
8039a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray// Used to synchronize the printing of the command-line prompt and incoming
8139a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray// Binder callbacks.
8239a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguraystd::atomic_bool showing_prompt(false);
8339a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
842e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray// The registered IBluetoothLowEnergy client handle. If |ble_registering| is
852e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray// true then an operation to register the client is in progress.
862e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguraystd::atomic_bool ble_registering(false);
874fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguraystd::atomic_int ble_client_id(0);
884fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
894fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray// The registered IBluetoothGattClient client handle. If |gatt_registering| is
904fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray// true then an operation to register the client is in progress.
914fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguraystd::atomic_bool gatt_registering(false);
924fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguraystd::atomic_int gatt_client_id(0);
932e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
94ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski// True if we should dump the scan record bytes for incoming scan results.
95ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowskistd::atomic_bool dump_scan_record(false);
96ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
972e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray// True if the remote process has died and we should exit.
982e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguraystd::atomic_bool should_exit(false);
992e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
10039a66bed960d96eca900e7e002e0d7bef0e0e151Arman Ugurayvoid PrintPrompt() {
10139a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  cout << COLOR_BLUE "[FCLI] " COLOR_OFF << flush;
10239a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray}
10339a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
1042e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Ugurayvoid PrintError(const string& message) {
1052e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  cout << COLOR_RED << message << COLOR_OFF << endl;
1062e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray}
1072e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
108d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Ugurayvoid PrintOpStatus(const std::string& op, bool status) {
109d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  cout << COLOR_BOLDWHITE << op << " status: " COLOR_OFF
110d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray       << (status ? (COLOR_GREEN "success") : (COLOR_RED "failure"))
11164401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski       << COLOR_OFF << endl;
11264401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski}
11364401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski
11464401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowskiinline void BeginAsyncOut() {
11564401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski  if (showing_prompt.load())
11664401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    cout << CLEAR_LINE << "\r";
11764401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski}
11864401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski
11964401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowskiinline void EndAsyncOut() {
12064401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski  std::flush(cout);
12164401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski  if (showing_prompt.load())
12264401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski      PrintPrompt();
12364401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski  else
12464401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    cout << endl;
125d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray}
126d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
12739a66bed960d96eca900e7e002e0d7bef0e0e151Arman Ugurayclass CLIBluetoothCallback : public ipc::binder::BnBluetoothCallback {
12839a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray public:
12939a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  CLIBluetoothCallback() = default;
13039a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  ~CLIBluetoothCallback() override = default;
13139a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
1322e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  // IBluetoothCallback overrides:
13339a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  void OnBluetoothStateChange(
13439a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray      bluetooth::AdapterState prev_state,
13539a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray      bluetooth::AdapterState new_state) override {
13664401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski
13764401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    BeginAsyncOut();
13839a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray    cout << COLOR_BOLDWHITE "Adapter state changed: " COLOR_OFF
13939a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray         << COLOR_MAGENTA << AdapterStateToString(prev_state) << COLOR_OFF
14039a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray         << COLOR_BOLDWHITE " -> " COLOR_OFF
14164401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski         << COLOR_BOLDYELLOW << AdapterStateToString(new_state) << COLOR_OFF;
14264401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    EndAsyncOut();
14364401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski   }
14439a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
14539a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray private:
14639a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  DISALLOW_COPY_AND_ASSIGN(CLIBluetoothCallback);
14739a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray};
14839a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
1492e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Ugurayclass CLIBluetoothLowEnergyCallback
1502e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    : public ipc::binder::BnBluetoothLowEnergyCallback {
1512e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray public:
1522e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  CLIBluetoothLowEnergyCallback() = default;
1534fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  ~CLIBluetoothLowEnergyCallback() override = default;
1542e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
1552e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  // IBluetoothLowEnergyCallback overrides:
1564fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  void OnClientRegistered(int status, int client_id) override {
15764401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    BeginAsyncOut();
1582e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    if (status != bluetooth::BLE_STATUS_SUCCESS) {
1592e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray      PrintError("Failed to register BLE client");
1602e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    } else {
1614fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray      ble_client_id = client_id;
1622e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray      cout << COLOR_BOLDWHITE "Registered BLE client with ID: " COLOR_OFF
16364401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski           << COLOR_GREEN << client_id << COLOR_OFF;
1642e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    }
16564401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    EndAsyncOut();
1662e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
1672e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    ble_registering = false;
1682e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
1692e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
170455dc8f535a719af6a65a7512d90f9db878f5a58Jakub Pawlowski  void OnConnectionState(int status, int client_id, const char* address,
171455dc8f535a719af6a65a7512d90f9db878f5a58Jakub Pawlowski                         bool connected) override {
17264401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    BeginAsyncOut();
173608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    cout << COLOR_BOLDWHITE "Connection state: "
174608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski         << COLOR_BOLDYELLOW "[" << address
175608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski         << " connected: " << (connected ? "true" : "false") << " ] "
176608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski         << COLOR_BOLDWHITE "- status: " << status
177608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski         << COLOR_BOLDWHITE " - client_id: " << client_id << COLOR_OFF;
17864401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    EndAsyncOut();
179455dc8f535a719af6a65a7512d90f9db878f5a58Jakub Pawlowski  }
180455dc8f535a719af6a65a7512d90f9db878f5a58Jakub Pawlowski
181a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski  void OnMtuChanged(int status, const char *address, int mtu) override {
18264401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    BeginAsyncOut();
183a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski    cout << COLOR_BOLDWHITE "MTU changed: "
184a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski         << COLOR_BOLDYELLOW "[" << address << " ] "
185a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski         << COLOR_BOLDWHITE " - status: " << status
186a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski         << COLOR_BOLDWHITE " - mtu: " << mtu << COLOR_OFF;
18764401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    EndAsyncOut();
188a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski  }
189a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski
190ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  void OnScanResult(const bluetooth::ScanResult& scan_result) override {
19164401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    BeginAsyncOut();
192ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    cout << COLOR_BOLDWHITE "Scan result: "
193ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski         << COLOR_BOLDYELLOW "[" << scan_result.device_address() << "] "
194ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski         << COLOR_BOLDWHITE "- RSSI: " << scan_result.rssi() << COLOR_OFF;
195ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
196ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    if (dump_scan_record) {
197ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski      cout << " - Record: "
198ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski           << base::HexEncode(scan_result.scan_record().data(),
199ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski                              scan_result.scan_record().size());
200ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    }
20164401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    EndAsyncOut();
202ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  }
203ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
20452bfc6060cec652a67c8989e0548225af0008be1Arman Uguray  void OnMultiAdvertiseCallback(
20552bfc6060cec652a67c8989e0548225af0008be1Arman Uguray      int status, bool is_start,
206d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      const bluetooth::AdvertiseSettings& /* settings */) {
20764401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    BeginAsyncOut();
208d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    std::string op = is_start ? "start" : "stop";
209d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
210d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    PrintOpStatus("Advertising " + op, status == bluetooth::BLE_STATUS_SUCCESS);
21164401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    EndAsyncOut();
21252bfc6060cec652a67c8989e0548225af0008be1Arman Uguray  }
21352bfc6060cec652a67c8989e0548225af0008be1Arman Uguray
2142e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray private:
2152e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  DISALLOW_COPY_AND_ASSIGN(CLIBluetoothLowEnergyCallback);
2162e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray};
2175192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
2184fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Ugurayclass CLIGattClientCallback
2194fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    : public ipc::binder::BnBluetoothGattClientCallback {
2204fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray public:
2214fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  CLIGattClientCallback() = default;
2224fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  ~CLIGattClientCallback() override = default;
2234fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
2244fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  // IBluetoothGattClientCallback overrides:
2254fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  void OnClientRegistered(int status, int client_id) override {
22664401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    BeginAsyncOut();
2274fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    if (status != bluetooth::BLE_STATUS_SUCCESS) {
2284fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray      PrintError("Failed to register GATT client");
2294fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    } else {
2304fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray      gatt_client_id = client_id;
2314fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray      cout << COLOR_BOLDWHITE "Registered GATT client with ID: " COLOR_OFF
23264401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski           << COLOR_GREEN << client_id << COLOR_OFF;
2334fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    }
23464401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    EndAsyncOut();
2354fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
2364fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    gatt_registering = false;
2374fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  }
2384fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
2394fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray private:
2404fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  DISALLOW_COPY_AND_ASSIGN(CLIGattClientCallback);
2414fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray};
2424fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
243fcf2e0391950a8b140082fbe78688fa89471fbedArman Ugurayvoid PrintCommandStatus(bool status) {
244d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  PrintOpStatus("Command", status);
245fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray}
246fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
2475192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid PrintFieldAndValue(const string& field, const string& value) {
2485192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  cout << COLOR_BOLDWHITE << field << ": " << COLOR_BOLDYELLOW << value
2495192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray       << COLOR_OFF << endl;
2505192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray}
2515192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
2525192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid PrintFieldAndBoolValue(const string& field, bool value) {
2535192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndValue(field, (value ? "true" : "false"));
2545192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray}
2555192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
2565192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleDisable(IBluetooth* bt_iface, const vector<string>& args) {
2575192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_NO_ARGS(args);
258fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  PrintCommandStatus(bt_iface->Disable());
259fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray}
260fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
2615192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleEnable(IBluetooth* bt_iface, const vector<string>& args) {
2625192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_NO_ARGS(args);
263fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  PrintCommandStatus(bt_iface->Enable());
264fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray}
265fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
2665192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleGetState(IBluetooth* bt_iface, const vector<string>& args) {
2675192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_NO_ARGS(args);
2682117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray  bluetooth::AdapterState state = static_cast<bluetooth::AdapterState>(
2692117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray      bt_iface->GetState());
2705192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndValue("Adapter state", bluetooth::AdapterStateToString(state));
271fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray}
272fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
2735192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleIsEnabled(IBluetooth* bt_iface, const vector<string>& args) {
2745192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_NO_ARGS(args);
275fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  bool enabled = bt_iface->IsEnabled();
2765192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndBoolValue("Adapter enabled", enabled);
277fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray}
278fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
2795192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleGetLocalAddress(IBluetooth* bt_iface, const vector<string>& args) {
2805192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_NO_ARGS(args);
2815192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  string address = bt_iface->GetAddress();
2825192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndValue("Adapter address", address);
2835192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray}
2845192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
2855192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleSetLocalName(IBluetooth* bt_iface, const vector<string>& args) {
2865192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_ARGS_COUNT(args, >=, 1, "No name was given");
2875192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
2885192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  std::string name;
2895192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  for (const auto& arg : args)
2905192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray    name += arg + " ";
2915192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
2925192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  base::TrimWhitespaceASCII(name, base::TRIM_TRAILING, &name);
2935192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
2945192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintCommandStatus(bt_iface->SetName(name));
2955192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray}
2965192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
2975192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleGetLocalName(IBluetooth* bt_iface, const vector<string>& args) {
2985192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_NO_ARGS(args);
2995192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  string name = bt_iface->GetName();
3005192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndValue("Adapter name", name);
3015192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray}
3025192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
3035192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleAdapterInfo(IBluetooth* bt_iface, const vector<string>& args) {
3045192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_NO_ARGS(args);
3055192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
3065192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  cout << COLOR_BOLDWHITE "Adapter Properties: " COLOR_OFF << endl;
3075192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
3085192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndValue("\tAddress", bt_iface->GetAddress());
3095192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndValue("\tState", bluetooth::AdapterStateToString(
3105192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray      static_cast<bluetooth::AdapterState>(bt_iface->GetState())));
3115192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndValue("\tName", bt_iface->GetName());
31210b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray  PrintFieldAndBoolValue("\tMulti-Adv. supported",
31310b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray                         bt_iface->IsMultiAdvertisementSupported());
31410b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray}
31510b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray
31610b54c4b7f1a863a27eca4158f256062ec9c3770Arman Ugurayvoid HandleSupportsMultiAdv(IBluetooth* bt_iface, const vector<string>& args) {
31710b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray  CHECK_NO_ARGS(args);
31810b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray
31910b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray  bool status = bt_iface->IsMultiAdvertisementSupported();
32010b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray  PrintFieldAndBoolValue("Multi-advertisement support", status);
3215192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray}
3225192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
3232e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Ugurayvoid HandleRegisterBLE(IBluetooth* bt_iface, const vector<string>& args) {
3242e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  CHECK_NO_ARGS(args);
3252e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3262e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  if (ble_registering.load()) {
3272e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    PrintError("In progress");
3282e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    return;
3292e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
3302e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3314fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (ble_client_id.load()) {
3322e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    PrintError("Already registered");
3332e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    return;
3342e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
3352e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3362e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
3372e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  if (!ble_iface.get()) {
3382e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
3392e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    return;
3402e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
3412e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
342ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  bool status = ble_iface->RegisterClient(new CLIBluetoothLowEnergyCallback());
343ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  ble_registering = status;
344ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  PrintCommandStatus(status);
3452e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray}
3462e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3472e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Ugurayvoid HandleUnregisterBLE(IBluetooth* bt_iface, const vector<string>& args) {
3482e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  CHECK_NO_ARGS(args);
3492e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3504fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (!ble_client_id.load()) {
3512e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    PrintError("Not registered");
3522e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    return;
3532e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
3542e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3552e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
3562e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  if (!ble_iface.get()) {
3572e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
3582e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    return;
3592e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
3602e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3614fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  ble_iface->UnregisterClient(ble_client_id.load());
3624fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  ble_client_id = 0;
3632e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  PrintCommandStatus(true);
3642e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray}
3652e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3662e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Ugurayvoid HandleUnregisterAllBLE(IBluetooth* bt_iface, const vector<string>& args) {
3672e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  CHECK_NO_ARGS(args);
3682e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3692e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
3702e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  if (!ble_iface.get()) {
3712e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
3722e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    return;
3732e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
3742e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3752e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  ble_iface->UnregisterAll();
3762e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  PrintCommandStatus(true);
3772e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray}
3782e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3794fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Ugurayvoid HandleRegisterGATT(IBluetooth* bt_iface, const vector<string>& args) {
3804fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  CHECK_NO_ARGS(args);
3814fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
3824fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (gatt_registering.load()) {
3834fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    PrintError("In progress");
3844fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    return;
3854fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  }
3864fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
3874fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (gatt_client_id.load()) {
3884fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    PrintError("Already registered");
3894fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    return;
3904fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  }
3914fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
3924fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  sp<IBluetoothGattClient> gatt_iface = bt_iface->GetGattClientInterface();
3934fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (!gatt_iface.get()) {
3944fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    PrintError("Failed to obtain handle to Bluetooth GATT Client interface");
3954fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    return;
3964fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  }
3974fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
3984fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  bool status = gatt_iface->RegisterClient(new CLIGattClientCallback());
3994fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  gatt_registering = status;
4004fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  PrintCommandStatus(status);
4014fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray}
4024fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
4034fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Ugurayvoid HandleUnregisterGATT(IBluetooth* bt_iface, const vector<string>& args) {
4044fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  CHECK_NO_ARGS(args);
4054fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
4064fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (!gatt_client_id.load()) {
4074fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    PrintError("Not registered");
4084fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    return;
4094fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  }
4104fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
4114fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  sp<IBluetoothGattClient> gatt_iface = bt_iface->GetGattClientInterface();
4124fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (!gatt_iface.get()) {
4134fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    PrintError("Failed to obtain handle to Bluetooth GATT Client interface");
4144fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    return;
4154fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  }
4164fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
4174fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  gatt_iface->UnregisterClient(gatt_client_id.load());
4184fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  gatt_client_id = 0;
4194fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  PrintCommandStatus(true);
4204fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray}
4214fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
422d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Ugurayvoid HandleStartAdv(IBluetooth* bt_iface, const vector<string>& args) {
423d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  bool include_name = false;
424d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  bool include_tx_power = false;
425d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  bool connectable = false;
426d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  bool set_manufacturer_data = false;
42787222e0e826216c69f6a9a5bfe77689561067474Arman Uguray  bool set_uuid = false;
42887222e0e826216c69f6a9a5bfe77689561067474Arman Uguray  bluetooth::UUID uuid;
429d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
43087222e0e826216c69f6a9a5bfe77689561067474Arman Uguray  for (auto iter = args.begin(); iter != args.end(); ++iter) {
43187222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    const std::string& arg = *iter;
432d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    if (arg == "-n")
433d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      include_name = true;
434d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    else if (arg == "-t")
435d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      include_tx_power = true;
436d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    else if (arg == "-c")
437d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      connectable = true;
438d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    else if (arg == "-m")
439d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      set_manufacturer_data = true;
44087222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    else if (arg == "-u") {
44187222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      // This flag has a single argument.
44287222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      ++iter;
44387222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      if (iter == args.end()) {
44487222e0e826216c69f6a9a5bfe77689561067474Arman Uguray        PrintError("Expected a UUID after -u");
44587222e0e826216c69f6a9a5bfe77689561067474Arman Uguray        return;
44687222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      }
44787222e0e826216c69f6a9a5bfe77689561067474Arman Uguray
44887222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      std::string uuid_str = *iter;
44987222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      uuid = bluetooth::UUID(uuid_str);
45087222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      if (!uuid.is_valid()) {
45187222e0e826216c69f6a9a5bfe77689561067474Arman Uguray        PrintError("Invalid UUID: " + uuid_str);
45287222e0e826216c69f6a9a5bfe77689561067474Arman Uguray        return;
45387222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      }
45487222e0e826216c69f6a9a5bfe77689561067474Arman Uguray
45587222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      set_uuid = true;
45687222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    }
457d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    else if (arg == "-h") {
458ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski      static const char kUsage[] =
459d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray          "Usage: start-adv [flags]\n"
460d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray          "\n"
461ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski          "Flags:\n"
462d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray          "\t-n\tInclude device name\n"
463d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray          "\t-t\tInclude TX power\n"
464d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray          "\t-c\tSend connectable adv. packets (default is non-connectable)\n"
465d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray          "\t-m\tInclude random manufacturer data\n"
466d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray          "\t-h\tShow this help message\n";
467d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      cout << kUsage << endl;
468d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      return;
469d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    }
470d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    else {
471d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      PrintError("Unrecognized option: " + arg);
472d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      return;
473d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    }
474d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  }
475d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
4764fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (!ble_client_id.load()) {
477d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    PrintError("BLE not registered");
478d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    return;
479d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  }
480d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
481d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
482d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  if (!ble_iface.get()) {
483d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
484d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    return;
485d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  }
486d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
487d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  std::vector<uint8_t> data;
488d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  if (set_manufacturer_data) {
489d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    data = {{
490d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      0x07, bluetooth::kEIRTypeManufacturerSpecificData,
491d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      0xe0, 0x00,
492d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      'T', 'e', 's', 't'
493d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    }};
494d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  }
495d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
49687222e0e826216c69f6a9a5bfe77689561067474Arman Uguray  if (set_uuid) {
49787222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    // Determine the type and length bytes.
49887222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    int uuid_size = uuid.GetShortestRepresentationSize();
49987222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    uint8_t type;
50087222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    if (uuid_size == bluetooth::UUID::kNumBytes128)
50187222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      type = bluetooth::kEIRTypeComplete128BitUUIDs;
50287222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    else if (uuid_size == bluetooth::UUID::kNumBytes32)
50387222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      type = bluetooth::kEIRTypeComplete32BitUUIDs;
50487222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    else if (uuid_size == bluetooth::UUID::kNumBytes16)
50587222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      type = bluetooth::kEIRTypeComplete16BitUUIDs;
50687222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    else
50787222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      NOTREACHED() << "Unexpected size: " << uuid_size;
50887222e0e826216c69f6a9a5bfe77689561067474Arman Uguray
50987222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    data.push_back(uuid_size + 1);
51087222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    data.push_back(type);
51187222e0e826216c69f6a9a5bfe77689561067474Arman Uguray
51287222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    auto uuid_bytes = uuid.GetFullLittleEndian();
51387222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    int index = (uuid_size == 16) ? 0 : 12;
51487222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    data.insert(data.end(), uuid_bytes.data() + index,
51587222e0e826216c69f6a9a5bfe77689561067474Arman Uguray                uuid_bytes.data() + index + uuid_size);
51687222e0e826216c69f6a9a5bfe77689561067474Arman Uguray  }
51787222e0e826216c69f6a9a5bfe77689561067474Arman Uguray
518d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  base::TimeDelta timeout;
519d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
520d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  bluetooth::AdvertiseSettings settings(
521d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      bluetooth::AdvertiseSettings::MODE_LOW_POWER,
522d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      timeout,
523d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      bluetooth::AdvertiseSettings::TX_POWER_LEVEL_MEDIUM,
524d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      connectable);
525d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
526d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  bluetooth::AdvertiseData adv_data(data);
527d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  adv_data.set_include_device_name(include_name);
528d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  adv_data.set_include_tx_power_level(include_tx_power);
529d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
530d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  bluetooth::AdvertiseData scan_rsp;
531d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
5324fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  bool status = ble_iface->StartMultiAdvertising(ble_client_id.load(),
533ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray                                                 adv_data, scan_rsp, settings);
534ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  PrintCommandStatus(status);
535d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray}
536d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
537d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Ugurayvoid HandleStopAdv(IBluetooth* bt_iface, const vector<string>& args) {
5384fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (!ble_client_id.load()) {
539d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    PrintError("BLE not registered");
540d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    return;
541d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  }
542d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
543d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
544d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  if (!ble_iface.get()) {
545d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
546d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    return;
547d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  }
548d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
5494fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  bool status = ble_iface->StopMultiAdvertising(ble_client_id.load());
550ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  PrintCommandStatus(status);
551d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray}
552d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
553608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowskivoid HandleConnect(IBluetooth* bt_iface, const vector<string>& args) {
554608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  string address;
555608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
556608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  if (args.size() != 1) {
557608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    PrintError("Expected MAC address as only argument");
558608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    return;
559608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  }
560608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
561608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  address = args[0];
562608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
563608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  if (!ble_client_id.load()) {
564608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    PrintError("BLE not registered");
565608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    return;
566608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  }
567608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
568608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
569608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  if (!ble_iface.get()) {
570608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
571608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    return;
572608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  }
573608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
574608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  bool status = ble_iface->Connect(ble_client_id.load(), address.c_str(),
575608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski                                   false /*  is_direct */);
576608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
577608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  PrintCommandStatus(status);
578608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski}
579608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
580608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowskivoid HandleDisconnect(IBluetooth* bt_iface, const vector<string>& args) {
581608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  string address;
582608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
583608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  if (args.size() != 1) {
584608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    PrintError("Expected MAC address as only argument");
585608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    return;
586608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  }
587608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
588608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  address = args[0];
589608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
590608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  if (!ble_client_id.load()) {
591608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    PrintError("BLE not registered");
592608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    return;
593608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  }
594608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
595608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
596608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  if (!ble_iface.get()) {
597608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
598608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    return;
599608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  }
600608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
601608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  bool status = ble_iface->Disconnect(ble_client_id.load(), address.c_str());
602608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
603608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  PrintCommandStatus(status);
604608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski}
605608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
606756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowskivoid HandleSetMtu(IBluetooth* bt_iface, const vector<string>& args) {
607756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  string address;
608756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  int mtu;
609756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski
610756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  if (args.size() != 2) {
611756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski    PrintError("Usage: set-mtu [address] [mtu]");
612756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski    return;
613756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  }
614756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski
615756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  address = args[0];
616756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  mtu = std::stoi(args[1]);
617756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski
618756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  if (mtu < 23) {
619756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski    PrintError("MTU must be 23 or larger");
620756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski    return;
621756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  }
622756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski
623756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  if (!ble_client_id.load()) {
624756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski    PrintError("BLE not registered");
625756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski    return;
626756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  }
627756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski
628756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
629756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  if (!ble_iface.get()) {
630756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
631756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski    return;
632756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  }
633756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski
634756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  bool status = ble_iface->SetMtu(ble_client_id.load(), address.c_str(), mtu);
635756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  PrintCommandStatus(status);
636756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski}
637756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski
638ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowskivoid HandleStartLeScan(IBluetooth* bt_iface, const vector<string>& args) {
639ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  if (!ble_client_id.load()) {
640ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    PrintError("BLE not registered");
641ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    return;
642ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  }
643ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
644ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  for (auto arg : args) {
645ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    if (arg == "-d") {
646ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski      dump_scan_record = true;
647ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    } else if (arg == "-h") {
648ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski      static const char kUsage[] =
649ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski          "Usage: start-le-scan [flags]\n"
650ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski          "\n"
651ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski          "Flags:\n"
652ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski          "\t-d\tDump scan record\n"
653ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski          "\t-h\tShow this help message\n";
654ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski      cout << kUsage << endl;
655ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski      return;
656ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    }
657ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  }
658ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
659ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
660ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  if (!ble_iface.get()) {
661ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
662ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    return;
663ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  }
664ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
665ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  bluetooth::ScanSettings settings;
666ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  std::vector<bluetooth::ScanFilter> filters;
667ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
668ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  bool status = ble_iface->StartScan(ble_client_id.load(), settings, filters);
669ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  PrintCommandStatus(status);
670ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski}
671ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
672ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowskivoid HandleStopLeScan(IBluetooth* bt_iface, const vector<string>& args) {
673ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  if (!ble_client_id.load()) {
674ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    PrintError("BLE not registered");
675ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    return;
676ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  }
677ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
678ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
679ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  if (!ble_iface.get()) {
680ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
681ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    return;
682ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  }
683ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
684ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  bluetooth::ScanSettings settings;
685ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  std::vector<bluetooth::ScanFilter> filters;
686ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
687ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  bool status = ble_iface->StopScan(ble_client_id.load());
688ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  PrintCommandStatus(status);
689ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski}
690ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
6915192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleHelp(IBluetooth* bt_iface, const vector<string>& args);
6922117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray
693fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguraystruct {
694fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  string command;
6955192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  void (*func)(IBluetooth*, const vector<string>& args);
6962117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray  string help;
697fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray} kCommandMap[] = {
6985192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "help", HandleHelp, "\t\t\tDisplay this message" },
6995192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "disable", HandleDisable, "\t\t\tDisable Bluetooth" },
7005192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "enable", HandleEnable, "\t\t\tEnable Bluetooth" },
7015192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "get-state", HandleGetState, "\t\tGet the current adapter state" },
7025192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "is-enabled", HandleIsEnabled, "\t\tReturn if Bluetooth is enabled" },
7035192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "get-local-address", HandleGetLocalAddress,
7045192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray    "\tGet the local adapter address" },
7055192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "set-local-name", HandleSetLocalName, "\t\tSet the local adapter name" },
7065192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "get-local-name", HandleGetLocalName, "\t\tGet the local adapter name" },
7075192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "adapter-info", HandleAdapterInfo, "\t\tPrint adapter properties" },
70810b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray  { "supports-multi-adv", HandleSupportsMultiAdv,
70910b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray    "\tWhether multi-advertisement is currently supported" },
7102e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  { "register-ble", HandleRegisterBLE,
7112e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    "\t\tRegister with the Bluetooth Low Energy interface" },
7122e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  { "unregister-ble", HandleUnregisterBLE,
7132e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    "\t\tUnregister from the Bluetooth Low Energy interface" },
7142e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  { "unregister-all-ble", HandleUnregisterAllBLE,
7152e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    "\tUnregister all clients from the Bluetooth Low Energy interface" },
7164fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  { "register-gatt", HandleRegisterGATT,
7174fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    "\t\tRegister with the Bluetooth GATT Client interface" },
7184fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  { "unregister-gatt", HandleUnregisterGATT,
7194fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    "\t\tUnregister from the Bluetooth GATT Client interface" },
720608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  { "connect-le", HandleConnect, "\t\tConnect to LE device (-h for options)"},
721608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  { "disconnect-le", HandleDisconnect,
722608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    "\t\tDisconnect LE device (-h for options)"},
723756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  { "set-mtu", HandleSetMtu, "\t\tSet MTU (-h for options)"},
724d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  { "start-adv", HandleStartAdv, "\t\tStart advertising (-h for options)" },
725d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  { "stop-adv", HandleStopAdv, "\t\tStop advertising" },
726ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  { "start-le-scan", HandleStartLeScan,
727ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    "\t\tStart LE device scan (-h for options)" },
728ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  { "stop-le-scan", HandleStopLeScan, "\t\tStop LE device scan" },
729fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  {},
730fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray};
731f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray
7325192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleHelp(IBluetooth* /* bt_iface */, const vector<string>& /* args */) {
7332117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray  cout << endl;
7342117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray  for (int i = 0; kCommandMap[i].func; i++)
7352117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray    cout << "\t" << kCommandMap[i].command << kCommandMap[i].help << endl;
7362117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray  cout << endl;
7372117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray}
7382117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray
7395d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssenconst char kExecuteLong[] = "exec";
7405d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssenconst char kExecuteShort[] = "e";
7415d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen
7425d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssenbool ExecuteCommand(sp<IBluetooth> bt_iface, std::string &command) {
7435d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen  vector<string> args =
7445d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen      base::SplitString(command, " ", base::TRIM_WHITESPACE,
7455d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen                        base::SPLIT_WANT_ALL);
7465d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen
7475d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen  if (args.empty())
7485d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen    return true;
7495d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen
7505d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen  // The first argument is the command while the remaining are what we pass to
7515d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen  // the handler functions.
7525d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen  command = args[0];
7535d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen  args.erase(args.begin());
7545d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen
7555d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen  for (int i = 0; kCommandMap[i].func; i++) {
7565d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen    if (command == kCommandMap[i].command) {
7575d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen      kCommandMap[i].func(bt_iface.get(), args);
7585d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen      return true;
7595d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen    }
7605d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen  }
7615d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen
7625d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen  cout << "Unrecognized command: " << command << endl;
7635d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen  return false;
7645d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen}
7655d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen
7662117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray}  // namespace
7672117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray
7682e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Ugurayclass BluetoothDeathRecipient : public android::IBinder::DeathRecipient {
7692e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray public:
7702e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  BluetoothDeathRecipient() = default;
7712e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  ~BluetoothDeathRecipient() override = default;
7722e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
7732e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  // android::IBinder::DeathRecipient override:
7742e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  void binderDied(const android::wp<android::IBinder>& /* who */) override {
77564401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    BeginAsyncOut();
7762e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    cout << COLOR_BOLDWHITE "The Bluetooth daemon has died" COLOR_OFF << endl;
77764401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    cout << "\nPress 'ENTER' to exit.";
77864401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    EndAsyncOut();
7792e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
7802e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    android::IPCThreadState::self()->stopProcess();
7812e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    should_exit = true;
7822e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
7832e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
7842e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray private:
7852e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  DISALLOW_COPY_AND_ASSIGN(BluetoothDeathRecipient);
7862e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray};
7872e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
7885d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen
789ae43de627b9c91e54215d439149c0e01599249c0Arman Ugurayint main(int argc, char* argv[]) {
790ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  base::AtExitManager exit_manager;
791ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  base::CommandLine::Init(argc, argv);
792ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  logging::LoggingSettings log_settings;
793ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray
794ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  if (!logging::InitLogging(log_settings)) {
795ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray    LOG(ERROR) << "Failed to set up logging";
796ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray    return EXIT_FAILURE;
797ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  }
798ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray
799f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray  sp<IBluetooth> bt_iface = IBluetooth::getClientInterface();
800f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray  if (!bt_iface.get()) {
801f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray    LOG(ERROR) << "Failed to obtain handle on IBluetooth";
802f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray    return EXIT_FAILURE;
803f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray  }
804f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray
8052e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  sp<BluetoothDeathRecipient> dr(new BluetoothDeathRecipient());
8062e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  if (android::IInterface::asBinder(bt_iface.get())->linkToDeath(dr) !=
8072e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray      android::NO_ERROR) {
8082e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    LOG(ERROR) << "Failed to register DeathRecipient for IBluetooth";
8092e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    return EXIT_FAILURE;
8102e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
8112e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
81239a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  // Initialize the Binder process thread pool. We have to set this up,
81339a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  // otherwise, incoming callbacks from IBluetoothCallback will block the main
81439a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  // thread (in other words, we have to do this as we are a "Binder server").
81539a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  android::ProcessState::self()->startThreadPool();
81639a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
81739a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  // Register Adapter state-change callback
81839a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  sp<CLIBluetoothCallback> callback = new CLIBluetoothCallback();
81939a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  bt_iface->RegisterCallback(callback);
82039a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
821fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  cout << COLOR_BOLDWHITE << "Fluoride Command-Line Interface\n" << COLOR_OFF
8222117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray       << endl
8232117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray       << "Type \"help\" to see possible commands.\n"
824fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray       << endl;
825fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
8265d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen  string command;
8275d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen
8285d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen  // Add commands from the command line, if they exist.
8295d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen  auto command_line = base::CommandLine::ForCurrentProcess();
8305d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen  if (command_line->HasSwitch(kExecuteLong)) {
8315d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen    command += command_line->GetSwitchValueASCII(kExecuteLong);
8325d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen  }
8335d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen
8345d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen  if (command_line->HasSwitch(kExecuteShort)) {
8355d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen    if (!command.empty())
8365d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen      command += " ; ";
8375d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen    command += command_line->GetSwitchValueASCII(kExecuteShort);
8385d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen  }
8395d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen
840fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  while (true) {
8415d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen    vector<string> commands = base::SplitString(command, ";",
8425d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen                                                base::TRIM_WHITESPACE,
8435d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen                                                base::SPLIT_WANT_ALL);
8445d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen    for (string command : commands) {
8455d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen      if (!ExecuteCommand(bt_iface, command))
8465d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen        break;
8475d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen    }
8485d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen
8495d80cb938d7fb3312870a047749cc839ba5bafcaMarie Janssen    commands.clear();
8505192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
85139a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray    PrintPrompt();
85239a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
85339a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray    showing_prompt = true;
85491c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray    auto& istream = getline(cin, command);
85539a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray    showing_prompt = false;
856fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
85791c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray    if (istream.eof() || should_exit.load()) {
85891c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray      cout << "\nExiting" << endl;
8592e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray      return EXIT_SUCCESS;
86091c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray    }
86191c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray
86291c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray    if (!istream.good()) {
86391c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray      LOG(ERROR) << "An error occured while reading input";
86491c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray      return EXIT_FAILURE;
86591c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray    }
8662e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
867fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  }
868f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray
869f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray  return EXIT_SUCCESS;
870f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray}
871