main.cc revision a6551079fe71b1c76505ada0e4f758f6faf651e0
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
20ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray#include <base/at_exit.h>
21ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray#include <base/command_line.h>
22f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray#include <base/logging.h>
2339a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray#include <base/macros.h>
24ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski#include <base/strings/string_number_conversions.h>
255192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray#include <base/strings/string_split.h>
265192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray#include <base/strings/string_util.h>
272e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray#include <binder/IPCThreadState.h>
2839a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray#include <binder/ProcessState.h>
29f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray
30234138e2606dd7a54fbcc540643511abc0a3598dArman Uguray#include <bluetooth/adapter_state.h>
31234138e2606dd7a54fbcc540643511abc0a3598dArman Uguray#include <bluetooth/binder/IBluetooth.h>
32234138e2606dd7a54fbcc540643511abc0a3598dArman Uguray#include <bluetooth/binder/IBluetoothCallback.h>
334fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray#include <bluetooth/binder/IBluetoothGattClient.h>
344fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray#include <bluetooth/binder/IBluetoothGattClientCallback.h>
354fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray#include <bluetooth/binder/IBluetoothLowEnergy.h>
364fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray#include <bluetooth/binder/IBluetoothLowEnergyCallback.h>
37234138e2606dd7a54fbcc540643511abc0a3598dArman Uguray#include <bluetooth/low_energy_constants.h>
38ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski#include <bluetooth/scan_filter.h>
39ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski#include <bluetooth/scan_settings.h>
4087222e0e826216c69f6a9a5bfe77689561067474Arman Uguray#include <bluetooth/uuid.h>
41f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray
42fcf2e0391950a8b140082fbe78688fa89471fbedArman Ugurayusing namespace std;
43fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
44f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Ugurayusing android::sp;
45f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray
46f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Ugurayusing ipc::binder::IBluetooth;
474fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Ugurayusing ipc::binder::IBluetoothGattClient;
482e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Ugurayusing ipc::binder::IBluetoothLowEnergy;
49f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray
502117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguraynamespace {
512117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray
522117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray#define COLOR_OFF         "\x1B[0m"
532117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray#define COLOR_RED         "\x1B[0;91m"
542117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray#define COLOR_GREEN       "\x1B[0;92m"
552117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray#define COLOR_YELLOW      "\x1B[0;93m"
562117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray#define COLOR_BLUE        "\x1B[0;94m"
572117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray#define COLOR_MAGENTA     "\x1B[0;95m"
582117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray#define COLOR_BOLDGRAY    "\x1B[1;30m"
592117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray#define COLOR_BOLDWHITE   "\x1B[1;37m"
602117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray#define COLOR_BOLDYELLOW  "\x1B[1;93m"
61fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
62fcf2e0391950a8b140082fbe78688fa89471fbedArman Ugurayconst char kCommandDisable[] = "disable";
63fcf2e0391950a8b140082fbe78688fa89471fbedArman Ugurayconst char kCommandEnable[] = "enable";
64fcf2e0391950a8b140082fbe78688fa89471fbedArman Ugurayconst char kCommandGetState[] = "get-state";
65fcf2e0391950a8b140082fbe78688fa89471fbedArman Ugurayconst char kCommandIsEnabled[] = "is-enabled";
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"))
111d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray       << COLOR_OFF << endl << endl;
112d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray}
113d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
11439a66bed960d96eca900e7e002e0d7bef0e0e151Arman Ugurayclass CLIBluetoothCallback : public ipc::binder::BnBluetoothCallback {
11539a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray public:
11639a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  CLIBluetoothCallback() = default;
11739a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  ~CLIBluetoothCallback() override = default;
11839a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
1192e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  // IBluetoothCallback overrides:
12039a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  void OnBluetoothStateChange(
12139a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray      bluetooth::AdapterState prev_state,
12239a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray      bluetooth::AdapterState new_state) override {
12339a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray    if (showing_prompt.load())
12439a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray      cout << endl;
12539a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray    cout << COLOR_BOLDWHITE "Adapter state changed: " COLOR_OFF
12639a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray         << COLOR_MAGENTA << AdapterStateToString(prev_state) << COLOR_OFF
12739a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray         << COLOR_BOLDWHITE " -> " COLOR_OFF
12839a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray         << COLOR_BOLDYELLOW << AdapterStateToString(new_state) << COLOR_OFF
12939a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray         << endl << endl;
13039a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray    if (showing_prompt.load())
13139a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray      PrintPrompt();
13239a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  }
13339a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
13439a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray private:
13539a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  DISALLOW_COPY_AND_ASSIGN(CLIBluetoothCallback);
13639a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray};
13739a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
1382e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Ugurayclass CLIBluetoothLowEnergyCallback
1392e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    : public ipc::binder::BnBluetoothLowEnergyCallback {
1402e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray public:
1412e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  CLIBluetoothLowEnergyCallback() = default;
1424fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  ~CLIBluetoothLowEnergyCallback() override = default;
1432e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
1442e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  // IBluetoothLowEnergyCallback overrides:
1454fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  void OnClientRegistered(int status, int client_id) override {
1462e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    if (showing_prompt.load())
1472e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray      cout << endl;
1482e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    if (status != bluetooth::BLE_STATUS_SUCCESS) {
1492e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray      PrintError("Failed to register BLE client");
1502e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    } else {
1514fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray      ble_client_id = client_id;
1522e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray      cout << COLOR_BOLDWHITE "Registered BLE client with ID: " COLOR_OFF
1534fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray           << COLOR_GREEN << client_id << COLOR_OFF << endl << endl;
1542e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    }
1552e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    if (showing_prompt.load())
1562e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray      PrintPrompt();
1572e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
1582e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    ble_registering = false;
1592e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
1602e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
161455dc8f535a719af6a65a7512d90f9db878f5a58Jakub Pawlowski  void OnConnectionState(int status, int client_id, const char* address,
162455dc8f535a719af6a65a7512d90f9db878f5a58Jakub Pawlowski                         bool connected) override {
163608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    if (showing_prompt.load())
164608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski      cout << endl;
165608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
166608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    cout << COLOR_BOLDWHITE "Connection state: "
167608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski         << COLOR_BOLDYELLOW "[" << address
168608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski         << " connected: " << (connected ? "true" : "false") << " ] "
169608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski         << COLOR_BOLDWHITE "- status: " << status
170608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski         << COLOR_BOLDWHITE " - client_id: " << client_id << COLOR_OFF;
171608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
172608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    cout  << endl << endl;
173608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
174608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    if (showing_prompt.load())
175608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski      PrintPrompt();
176455dc8f535a719af6a65a7512d90f9db878f5a58Jakub Pawlowski  }
177455dc8f535a719af6a65a7512d90f9db878f5a58Jakub Pawlowski
178a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski  void OnMtuChanged(int status, const char *address, int mtu) override {
179a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski    if (showing_prompt.load())
180a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski      cout << endl;
181a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski
182a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski    cout << COLOR_BOLDWHITE "MTU changed: "
183a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski         << COLOR_BOLDYELLOW "[" << address << " ] "
184a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski         << COLOR_BOLDWHITE " - status: " << status
185a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski         << COLOR_BOLDWHITE " - mtu: " << mtu << COLOR_OFF;
186a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski
187a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski    cout  << endl << endl;
188a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski
189a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski    if (showing_prompt.load())
190a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski      PrintPrompt();
191a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski  }
192a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski
193ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  void OnScanResult(const bluetooth::ScanResult& scan_result) override {
194ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    if (showing_prompt.load())
195ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski      cout << endl;
196ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
197ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    cout << COLOR_BOLDWHITE "Scan result: "
198ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski         << COLOR_BOLDYELLOW "[" << scan_result.device_address() << "] "
199ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski         << COLOR_BOLDWHITE "- RSSI: " << scan_result.rssi() << COLOR_OFF;
200ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
201ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    if (dump_scan_record) {
202ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski      cout << " - Record: "
203ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski           << base::HexEncode(scan_result.scan_record().data(),
204ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski                              scan_result.scan_record().size());
205ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    }
206ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
207ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    cout  << endl << endl;
208ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
209ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    if (showing_prompt.load())
210ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski      PrintPrompt();
211ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  }
212ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
21352bfc6060cec652a67c8989e0548225af0008be1Arman Uguray  void OnMultiAdvertiseCallback(
21452bfc6060cec652a67c8989e0548225af0008be1Arman Uguray      int status, bool is_start,
215d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      const bluetooth::AdvertiseSettings& /* settings */) {
216d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    if (showing_prompt.load())
217d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      cout << endl;
218d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
219d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    std::string op = is_start ? "start" : "stop";
220d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
221d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    PrintOpStatus("Advertising " + op, status == bluetooth::BLE_STATUS_SUCCESS);
222d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
223d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    if (showing_prompt.load())
224d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      PrintPrompt();
22552bfc6060cec652a67c8989e0548225af0008be1Arman Uguray  }
22652bfc6060cec652a67c8989e0548225af0008be1Arman Uguray
2272e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray private:
2282e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  DISALLOW_COPY_AND_ASSIGN(CLIBluetoothLowEnergyCallback);
2292e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray};
2305192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
2314fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Ugurayclass CLIGattClientCallback
2324fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    : public ipc::binder::BnBluetoothGattClientCallback {
2334fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray public:
2344fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  CLIGattClientCallback() = default;
2354fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  ~CLIGattClientCallback() override = default;
2364fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
2374fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  // IBluetoothGattClientCallback overrides:
2384fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  void OnClientRegistered(int status, int client_id) override {
2394fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    if (showing_prompt.load())
2404fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray      cout << endl;
2414fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    if (status != bluetooth::BLE_STATUS_SUCCESS) {
2424fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray      PrintError("Failed to register GATT client");
2434fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    } else {
2444fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray      gatt_client_id = client_id;
2454fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray      cout << COLOR_BOLDWHITE "Registered GATT client with ID: " COLOR_OFF
2464fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray           << COLOR_GREEN << client_id << COLOR_OFF << endl << endl;
2474fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    }
2484fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    if (showing_prompt.load())
2494fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray      PrintPrompt();
2504fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
2514fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    gatt_registering = false;
2524fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  }
2534fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
2544fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray private:
2554fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  DISALLOW_COPY_AND_ASSIGN(CLIGattClientCallback);
2564fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray};
2574fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
258fcf2e0391950a8b140082fbe78688fa89471fbedArman Ugurayvoid PrintCommandStatus(bool status) {
259d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  PrintOpStatus("Command", status);
260fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray}
261fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
2625192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid PrintFieldAndValue(const string& field, const string& value) {
2635192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  cout << COLOR_BOLDWHITE << field << ": " << COLOR_BOLDYELLOW << value
2645192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray       << COLOR_OFF << endl;
2655192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray}
2665192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
2675192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid PrintFieldAndBoolValue(const string& field, bool value) {
2685192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndValue(field, (value ? "true" : "false"));
2695192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray}
2705192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
2715192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleDisable(IBluetooth* bt_iface, const vector<string>& args) {
2725192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_NO_ARGS(args);
273fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  PrintCommandStatus(bt_iface->Disable());
274fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray}
275fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
2765192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleEnable(IBluetooth* bt_iface, const vector<string>& args) {
2775192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_NO_ARGS(args);
278fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  PrintCommandStatus(bt_iface->Enable());
279fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray}
280fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
2815192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleGetState(IBluetooth* bt_iface, const vector<string>& args) {
2825192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_NO_ARGS(args);
2832117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray  bluetooth::AdapterState state = static_cast<bluetooth::AdapterState>(
2842117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray      bt_iface->GetState());
2855192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndValue("Adapter state", bluetooth::AdapterStateToString(state));
286fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray}
287fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
2885192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleIsEnabled(IBluetooth* bt_iface, const vector<string>& args) {
2895192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_NO_ARGS(args);
290fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  bool enabled = bt_iface->IsEnabled();
2915192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndBoolValue("Adapter enabled", enabled);
292fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray}
293fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
2945192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleGetLocalAddress(IBluetooth* bt_iface, const vector<string>& args) {
2955192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_NO_ARGS(args);
2965192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  string address = bt_iface->GetAddress();
2975192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndValue("Adapter address", address);
2985192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray}
2995192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
3005192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleSetLocalName(IBluetooth* bt_iface, const vector<string>& args) {
3015192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_ARGS_COUNT(args, >=, 1, "No name was given");
3025192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
3035192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  std::string name;
3045192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  for (const auto& arg : args)
3055192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray    name += arg + " ";
3065192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
3075192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  base::TrimWhitespaceASCII(name, base::TRIM_TRAILING, &name);
3085192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
3095192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintCommandStatus(bt_iface->SetName(name));
3105192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray}
3115192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
3125192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleGetLocalName(IBluetooth* bt_iface, const vector<string>& args) {
3135192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_NO_ARGS(args);
3145192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  string name = bt_iface->GetName();
3155192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndValue("Adapter name", name);
3165192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray}
3175192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
3185192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleAdapterInfo(IBluetooth* bt_iface, const vector<string>& args) {
3195192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_NO_ARGS(args);
3205192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
3215192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  cout << COLOR_BOLDWHITE "Adapter Properties: " COLOR_OFF << endl;
3225192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
3235192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndValue("\tAddress", bt_iface->GetAddress());
3245192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndValue("\tState", bluetooth::AdapterStateToString(
3255192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray      static_cast<bluetooth::AdapterState>(bt_iface->GetState())));
3265192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndValue("\tName", bt_iface->GetName());
32710b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray  PrintFieldAndBoolValue("\tMulti-Adv. supported",
32810b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray                         bt_iface->IsMultiAdvertisementSupported());
32910b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray}
33010b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray
33110b54c4b7f1a863a27eca4158f256062ec9c3770Arman Ugurayvoid HandleSupportsMultiAdv(IBluetooth* bt_iface, const vector<string>& args) {
33210b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray  CHECK_NO_ARGS(args);
33310b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray
33410b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray  bool status = bt_iface->IsMultiAdvertisementSupported();
33510b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray  PrintFieldAndBoolValue("Multi-advertisement support", status);
3365192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray}
3375192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
3382e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Ugurayvoid HandleRegisterBLE(IBluetooth* bt_iface, const vector<string>& args) {
3392e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  CHECK_NO_ARGS(args);
3402e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3412e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  if (ble_registering.load()) {
3422e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    PrintError("In progress");
3432e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    return;
3442e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
3452e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3464fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (ble_client_id.load()) {
3472e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    PrintError("Already registered");
3482e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    return;
3492e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
3502e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3512e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
3522e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  if (!ble_iface.get()) {
3532e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
3542e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    return;
3552e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
3562e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
357ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  bool status = ble_iface->RegisterClient(new CLIBluetoothLowEnergyCallback());
358ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  ble_registering = status;
359ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  PrintCommandStatus(status);
3602e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray}
3612e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3622e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Ugurayvoid HandleUnregisterBLE(IBluetooth* bt_iface, const vector<string>& args) {
3632e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  CHECK_NO_ARGS(args);
3642e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3654fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (!ble_client_id.load()) {
3662e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    PrintError("Not registered");
3672e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    return;
3682e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
3692e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3702e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
3712e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  if (!ble_iface.get()) {
3722e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
3732e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    return;
3742e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
3752e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3764fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  ble_iface->UnregisterClient(ble_client_id.load());
3774fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  ble_client_id = 0;
3782e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  PrintCommandStatus(true);
3792e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray}
3802e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3812e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Ugurayvoid HandleUnregisterAllBLE(IBluetooth* bt_iface, const vector<string>& args) {
3822e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  CHECK_NO_ARGS(args);
3832e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3842e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
3852e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  if (!ble_iface.get()) {
3862e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
3872e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    return;
3882e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
3892e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3902e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  ble_iface->UnregisterAll();
3912e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  PrintCommandStatus(true);
3922e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray}
3932e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3944fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Ugurayvoid HandleRegisterGATT(IBluetooth* bt_iface, const vector<string>& args) {
3954fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  CHECK_NO_ARGS(args);
3964fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
3974fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (gatt_registering.load()) {
3984fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    PrintError("In progress");
3994fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    return;
4004fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  }
4014fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
4024fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (gatt_client_id.load()) {
4034fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    PrintError("Already registered");
4044fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    return;
4054fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  }
4064fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
4074fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  sp<IBluetoothGattClient> gatt_iface = bt_iface->GetGattClientInterface();
4084fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (!gatt_iface.get()) {
4094fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    PrintError("Failed to obtain handle to Bluetooth GATT Client interface");
4104fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    return;
4114fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  }
4124fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
4134fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  bool status = gatt_iface->RegisterClient(new CLIGattClientCallback());
4144fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  gatt_registering = status;
4154fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  PrintCommandStatus(status);
4164fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray}
4174fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
4184fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Ugurayvoid HandleUnregisterGATT(IBluetooth* bt_iface, const vector<string>& args) {
4194fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  CHECK_NO_ARGS(args);
4204fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
4214fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (!gatt_client_id.load()) {
4224fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    PrintError("Not registered");
4234fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    return;
4244fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  }
4254fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
4264fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  sp<IBluetoothGattClient> gatt_iface = bt_iface->GetGattClientInterface();
4274fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (!gatt_iface.get()) {
4284fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    PrintError("Failed to obtain handle to Bluetooth GATT Client interface");
4294fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    return;
4304fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  }
4314fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
4324fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  gatt_iface->UnregisterClient(gatt_client_id.load());
4334fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  gatt_client_id = 0;
4344fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  PrintCommandStatus(true);
4354fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray}
4364fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
437d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Ugurayvoid HandleStartAdv(IBluetooth* bt_iface, const vector<string>& args) {
438d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  bool include_name = false;
439d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  bool include_tx_power = false;
440d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  bool connectable = false;
441d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  bool set_manufacturer_data = false;
44287222e0e826216c69f6a9a5bfe77689561067474Arman Uguray  bool set_uuid = false;
44387222e0e826216c69f6a9a5bfe77689561067474Arman Uguray  bluetooth::UUID uuid;
444d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
44587222e0e826216c69f6a9a5bfe77689561067474Arman Uguray  for (auto iter = args.begin(); iter != args.end(); ++iter) {
44687222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    const std::string& arg = *iter;
447d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    if (arg == "-n")
448d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      include_name = true;
449d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    else if (arg == "-t")
450d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      include_tx_power = true;
451d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    else if (arg == "-c")
452d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      connectable = true;
453d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    else if (arg == "-m")
454d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      set_manufacturer_data = true;
45587222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    else if (arg == "-u") {
45687222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      // This flag has a single argument.
45787222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      ++iter;
45887222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      if (iter == args.end()) {
45987222e0e826216c69f6a9a5bfe77689561067474Arman Uguray        PrintError("Expected a UUID after -u");
46087222e0e826216c69f6a9a5bfe77689561067474Arman Uguray        return;
46187222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      }
46287222e0e826216c69f6a9a5bfe77689561067474Arman Uguray
46387222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      std::string uuid_str = *iter;
46487222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      uuid = bluetooth::UUID(uuid_str);
46587222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      if (!uuid.is_valid()) {
46687222e0e826216c69f6a9a5bfe77689561067474Arman Uguray        PrintError("Invalid UUID: " + uuid_str);
46787222e0e826216c69f6a9a5bfe77689561067474Arman Uguray        return;
46887222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      }
46987222e0e826216c69f6a9a5bfe77689561067474Arman Uguray
47087222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      set_uuid = true;
47187222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    }
472d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    else if (arg == "-h") {
473ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski      static const char kUsage[] =
474d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray          "Usage: start-adv [flags]\n"
475d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray          "\n"
476ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski          "Flags:\n"
477d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray          "\t-n\tInclude device name\n"
478d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray          "\t-t\tInclude TX power\n"
479d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray          "\t-c\tSend connectable adv. packets (default is non-connectable)\n"
480d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray          "\t-m\tInclude random manufacturer data\n"
481d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray          "\t-h\tShow this help message\n";
482d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      cout << kUsage << endl;
483d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      return;
484d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    }
485d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    else {
486d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      PrintError("Unrecognized option: " + arg);
487d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      return;
488d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    }
489d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  }
490d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
4914fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (!ble_client_id.load()) {
492d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    PrintError("BLE not registered");
493d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    return;
494d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  }
495d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
496d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
497d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  if (!ble_iface.get()) {
498d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
499d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    return;
500d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  }
501d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
502d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  std::vector<uint8_t> data;
503d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  if (set_manufacturer_data) {
504d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    data = {{
505d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      0x07, bluetooth::kEIRTypeManufacturerSpecificData,
506d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      0xe0, 0x00,
507d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      'T', 'e', 's', 't'
508d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    }};
509d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  }
510d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
51187222e0e826216c69f6a9a5bfe77689561067474Arman Uguray  if (set_uuid) {
51287222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    // Determine the type and length bytes.
51387222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    int uuid_size = uuid.GetShortestRepresentationSize();
51487222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    uint8_t type;
51587222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    if (uuid_size == bluetooth::UUID::kNumBytes128)
51687222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      type = bluetooth::kEIRTypeComplete128BitUUIDs;
51787222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    else if (uuid_size == bluetooth::UUID::kNumBytes32)
51887222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      type = bluetooth::kEIRTypeComplete32BitUUIDs;
51987222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    else if (uuid_size == bluetooth::UUID::kNumBytes16)
52087222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      type = bluetooth::kEIRTypeComplete16BitUUIDs;
52187222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    else
52287222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      NOTREACHED() << "Unexpected size: " << uuid_size;
52387222e0e826216c69f6a9a5bfe77689561067474Arman Uguray
52487222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    data.push_back(uuid_size + 1);
52587222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    data.push_back(type);
52687222e0e826216c69f6a9a5bfe77689561067474Arman Uguray
52787222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    auto uuid_bytes = uuid.GetFullLittleEndian();
52887222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    int index = (uuid_size == 16) ? 0 : 12;
52987222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    data.insert(data.end(), uuid_bytes.data() + index,
53087222e0e826216c69f6a9a5bfe77689561067474Arman Uguray                uuid_bytes.data() + index + uuid_size);
53187222e0e826216c69f6a9a5bfe77689561067474Arman Uguray  }
53287222e0e826216c69f6a9a5bfe77689561067474Arman Uguray
533d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  base::TimeDelta timeout;
534d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
535d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  bluetooth::AdvertiseSettings settings(
536d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      bluetooth::AdvertiseSettings::MODE_LOW_POWER,
537d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      timeout,
538d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      bluetooth::AdvertiseSettings::TX_POWER_LEVEL_MEDIUM,
539d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      connectable);
540d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
541d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  bluetooth::AdvertiseData adv_data(data);
542d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  adv_data.set_include_device_name(include_name);
543d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  adv_data.set_include_tx_power_level(include_tx_power);
544d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
545d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  bluetooth::AdvertiseData scan_rsp;
546d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
5474fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  bool status = ble_iface->StartMultiAdvertising(ble_client_id.load(),
548ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray                                                 adv_data, scan_rsp, settings);
549ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  PrintCommandStatus(status);
550d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray}
551d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
552d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Ugurayvoid HandleStopAdv(IBluetooth* bt_iface, const vector<string>& args) {
5534fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (!ble_client_id.load()) {
554d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    PrintError("BLE not registered");
555d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    return;
556d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  }
557d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
558d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
559d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  if (!ble_iface.get()) {
560d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
561d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    return;
562d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  }
563d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
5644fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  bool status = ble_iface->StopMultiAdvertising(ble_client_id.load());
565ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  PrintCommandStatus(status);
566d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray}
567d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
568608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowskivoid HandleConnect(IBluetooth* bt_iface, const vector<string>& args) {
569608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  string address;
570608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
571608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  if (args.size() != 1) {
572608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    PrintError("Expected MAC address as only argument");
573608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    return;
574608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  }
575608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
576608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  address = args[0];
577608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
578608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  if (!ble_client_id.load()) {
579608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    PrintError("BLE not registered");
580608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    return;
581608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  }
582608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
583608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
584608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  if (!ble_iface.get()) {
585608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
586608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    return;
587608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  }
588608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
589608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  bool status = ble_iface->Connect(ble_client_id.load(), address.c_str(),
590608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski                                   false /*  is_direct */);
591608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
592608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  PrintCommandStatus(status);
593608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski}
594608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
595608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowskivoid HandleDisconnect(IBluetooth* bt_iface, const vector<string>& args) {
596608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  string address;
597608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
598608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  if (args.size() != 1) {
599608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    PrintError("Expected MAC address as only argument");
600608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    return;
601608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  }
602608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
603608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  address = args[0];
604608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
605608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  if (!ble_client_id.load()) {
606608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    PrintError("BLE not registered");
607608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    return;
608608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  }
609608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
610608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
611608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  if (!ble_iface.get()) {
612608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
613608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    return;
614608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  }
615608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
616608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  bool status = ble_iface->Disconnect(ble_client_id.load(), address.c_str());
617608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
618608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  PrintCommandStatus(status);
619608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski}
620608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
621ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowskivoid HandleStartLeScan(IBluetooth* bt_iface, const vector<string>& args) {
622ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  if (!ble_client_id.load()) {
623ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    PrintError("BLE not registered");
624ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    return;
625ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  }
626ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
627ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  for (auto arg : args) {
628ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    if (arg == "-d") {
629ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski      dump_scan_record = true;
630ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    } else if (arg == "-h") {
631ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski      static const char kUsage[] =
632ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski          "Usage: start-le-scan [flags]\n"
633ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski          "\n"
634ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski          "Flags:\n"
635ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski          "\t-d\tDump scan record\n"
636ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski          "\t-h\tShow this help message\n";
637ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski      cout << kUsage << endl;
638ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski      return;
639ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    }
640ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  }
641ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
642ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
643ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  if (!ble_iface.get()) {
644ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
645ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    return;
646ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  }
647ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
648ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  bluetooth::ScanSettings settings;
649ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  std::vector<bluetooth::ScanFilter> filters;
650ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
651ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  bool status = ble_iface->StartScan(ble_client_id.load(), settings, filters);
652ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  PrintCommandStatus(status);
653ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski}
654ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
655ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowskivoid HandleStopLeScan(IBluetooth* bt_iface, const vector<string>& args) {
656ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  if (!ble_client_id.load()) {
657ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    PrintError("BLE not registered");
658ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    return;
659ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  }
660ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
661ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
662ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  if (!ble_iface.get()) {
663ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
664ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    return;
665ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  }
666ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
667ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  bluetooth::ScanSettings settings;
668ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  std::vector<bluetooth::ScanFilter> filters;
669ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
670ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  bool status = ble_iface->StopScan(ble_client_id.load());
671ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  PrintCommandStatus(status);
672ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski}
673ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
6745192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleHelp(IBluetooth* bt_iface, const vector<string>& args);
6752117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray
676fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguraystruct {
677fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  string command;
6785192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  void (*func)(IBluetooth*, const vector<string>& args);
6792117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray  string help;
680fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray} kCommandMap[] = {
6815192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "help", HandleHelp, "\t\t\tDisplay this message" },
6825192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "disable", HandleDisable, "\t\t\tDisable Bluetooth" },
6835192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "enable", HandleEnable, "\t\t\tEnable Bluetooth" },
6845192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "get-state", HandleGetState, "\t\tGet the current adapter state" },
6855192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "is-enabled", HandleIsEnabled, "\t\tReturn if Bluetooth is enabled" },
6865192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "get-local-address", HandleGetLocalAddress,
6875192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray    "\tGet the local adapter address" },
6885192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "set-local-name", HandleSetLocalName, "\t\tSet the local adapter name" },
6895192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "get-local-name", HandleGetLocalName, "\t\tGet the local adapter name" },
6905192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "adapter-info", HandleAdapterInfo, "\t\tPrint adapter properties" },
69110b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray  { "supports-multi-adv", HandleSupportsMultiAdv,
69210b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray    "\tWhether multi-advertisement is currently supported" },
6932e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  { "register-ble", HandleRegisterBLE,
6942e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    "\t\tRegister with the Bluetooth Low Energy interface" },
6952e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  { "unregister-ble", HandleUnregisterBLE,
6962e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    "\t\tUnregister from the Bluetooth Low Energy interface" },
6972e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  { "unregister-all-ble", HandleUnregisterAllBLE,
6982e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    "\tUnregister all clients from the Bluetooth Low Energy interface" },
6994fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  { "register-gatt", HandleRegisterGATT,
7004fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    "\t\tRegister with the Bluetooth GATT Client interface" },
7014fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  { "unregister-gatt", HandleUnregisterGATT,
7024fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    "\t\tUnregister from the Bluetooth GATT Client interface" },
703608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  { "connect-le", HandleConnect, "\t\tConnect to LE device (-h for options)"},
704608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  { "disconnect-le", HandleDisconnect,
705608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    "\t\tDisconnect LE device (-h for options)"},
706d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  { "start-adv", HandleStartAdv, "\t\tStart advertising (-h for options)" },
707d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  { "stop-adv", HandleStopAdv, "\t\tStop advertising" },
708ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  { "start-le-scan", HandleStartLeScan,
709ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    "\t\tStart LE device scan (-h for options)" },
710ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  { "stop-le-scan", HandleStopLeScan, "\t\tStop LE device scan" },
711fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  {},
712fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray};
713f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray
7145192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleHelp(IBluetooth* /* bt_iface */, const vector<string>& /* args */) {
7152117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray  cout << endl;
7162117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray  for (int i = 0; kCommandMap[i].func; i++)
7172117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray    cout << "\t" << kCommandMap[i].command << kCommandMap[i].help << endl;
7182117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray  cout << endl;
7192117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray}
7202117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray
7212117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray}  // namespace
7222117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray
7232e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Ugurayclass BluetoothDeathRecipient : public android::IBinder::DeathRecipient {
7242e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray public:
7252e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  BluetoothDeathRecipient() = default;
7262e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  ~BluetoothDeathRecipient() override = default;
7272e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
7282e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  // android::IBinder::DeathRecipient override:
7292e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  void binderDied(const android::wp<android::IBinder>& /* who */) override {
7302e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    if (showing_prompt.load())
7312e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray      cout << endl;
7322e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    cout << COLOR_BOLDWHITE "The Bluetooth daemon has died" COLOR_OFF << endl;
7332e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    cout << "\nPress 'ENTER' to exit." << endl;
7342e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    if (showing_prompt.load())
7352e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray      PrintPrompt();
7362e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
7372e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    android::IPCThreadState::self()->stopProcess();
7382e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    should_exit = true;
7392e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
7402e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
7412e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray private:
7422e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  DISALLOW_COPY_AND_ASSIGN(BluetoothDeathRecipient);
7432e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray};
7442e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
745ae43de627b9c91e54215d439149c0e01599249c0Arman Ugurayint main(int argc, char* argv[]) {
746ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  base::AtExitManager exit_manager;
747ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  base::CommandLine::Init(argc, argv);
748ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  logging::LoggingSettings log_settings;
749ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray
750ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  if (!logging::InitLogging(log_settings)) {
751ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray    LOG(ERROR) << "Failed to set up logging";
752ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray    return EXIT_FAILURE;
753ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  }
754ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray
755f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray  sp<IBluetooth> bt_iface = IBluetooth::getClientInterface();
756f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray  if (!bt_iface.get()) {
757f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray    LOG(ERROR) << "Failed to obtain handle on IBluetooth";
758f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray    return EXIT_FAILURE;
759f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray  }
760f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray
7612e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  sp<BluetoothDeathRecipient> dr(new BluetoothDeathRecipient());
7622e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  if (android::IInterface::asBinder(bt_iface.get())->linkToDeath(dr) !=
7632e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray      android::NO_ERROR) {
7642e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    LOG(ERROR) << "Failed to register DeathRecipient for IBluetooth";
7652e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    return EXIT_FAILURE;
7662e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
7672e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
76839a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  // Initialize the Binder process thread pool. We have to set this up,
76939a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  // otherwise, incoming callbacks from IBluetoothCallback will block the main
77039a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  // thread (in other words, we have to do this as we are a "Binder server").
77139a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  android::ProcessState::self()->startThreadPool();
77239a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
77339a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  // Register Adapter state-change callback
77439a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  sp<CLIBluetoothCallback> callback = new CLIBluetoothCallback();
77539a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  bt_iface->RegisterCallback(callback);
77639a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
777fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  cout << COLOR_BOLDWHITE << "Fluoride Command-Line Interface\n" << COLOR_OFF
7782117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray       << endl
7792117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray       << "Type \"help\" to see possible commands.\n"
780fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray       << endl;
781fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
782fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  while (true) {
783fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray    string command;
7845192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
78539a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray    PrintPrompt();
78639a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
78739a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray    showing_prompt = true;
78891c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray    auto& istream = getline(cin, command);
78939a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray    showing_prompt = false;
790fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
79191c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray    if (istream.eof() || should_exit.load()) {
79291c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray      cout << "\nExiting" << endl;
7932e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray      return EXIT_SUCCESS;
79491c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray    }
79591c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray
79691c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray    if (!istream.good()) {
79791c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray      LOG(ERROR) << "An error occured while reading input";
79891c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray      return EXIT_FAILURE;
79991c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray    }
8002e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
80157d7bf98ea5f7a08c93c30f5d8a0e303d78ea432Alex Vakulenko    vector<string> args =
80257d7bf98ea5f7a08c93c30f5d8a0e303d78ea432Alex Vakulenko        base::SplitString(command, " ", base::TRIM_WHITESPACE,
80357d7bf98ea5f7a08c93c30f5d8a0e303d78ea432Alex Vakulenko                          base::SPLIT_WANT_ALL);
8045192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
8055192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray    if (args.empty())
8065192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray      continue;
8075192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
8085192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray    // The first argument is the command while the remaning are what we pass to
8095192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray    // the handler functions.
8105192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray    command = args[0];
8115192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray    args.erase(args.begin());
8125192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
813fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray    bool command_handled = false;
8142117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray    for (int i = 0; kCommandMap[i].func && !command_handled; i++) {
815fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray      if (command == kCommandMap[i].command) {
8165192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray        kCommandMap[i].func(bt_iface.get(), args);
817fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray        command_handled = true;
818fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray      }
819fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray    }
820fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
821fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray    if (!command_handled)
822fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray      cout << "Unrecognized command: " << command << endl;
823fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  }
824f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray
825f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray  return EXIT_SUCCESS;
826f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray}
827