main.cc revision 64401bf539bdef652ddcfc25138ad5e353aea1c3
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"
6164401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski#define CLEAR_LINE        "\x1B[2K"
62fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
63fcf2e0391950a8b140082fbe78688fa89471fbedArman Ugurayconst char kCommandDisable[] = "disable";
64fcf2e0391950a8b140082fbe78688fa89471fbedArman Ugurayconst char kCommandEnable[] = "enable";
65fcf2e0391950a8b140082fbe78688fa89471fbedArman Ugurayconst char kCommandGetState[] = "get-state";
66fcf2e0391950a8b140082fbe78688fa89471fbedArman Ugurayconst char kCommandIsEnabled[] = "is-enabled";
67fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
685192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray#define CHECK_ARGS_COUNT(args, op, num, msg) \
695192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  if (!(args.size() op num)) { \
705192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray    PrintError(msg); \
715192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray    return; \
725192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  }
735192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray#define CHECK_NO_ARGS(args) \
745192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_ARGS_COUNT(args, ==, 0, "Expected no arguments")
755192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
7639a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray// TODO(armansito): Clean up this code. Right now everything is in this
7739a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray// monolithic file. We should organize this into different classes for command
7839a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray// handling, console output/printing, callback handling, etc.
7939a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray// (See http://b/23387611)
8039a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
8139a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray// Used to synchronize the printing of the command-line prompt and incoming
8239a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray// Binder callbacks.
8339a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguraystd::atomic_bool showing_prompt(false);
8439a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
852e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray// The registered IBluetoothLowEnergy client handle. If |ble_registering| is
862e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray// true then an operation to register the client is in progress.
872e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguraystd::atomic_bool ble_registering(false);
884fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguraystd::atomic_int ble_client_id(0);
894fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
904fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray// The registered IBluetoothGattClient client handle. If |gatt_registering| is
914fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray// true then an operation to register the client is in progress.
924fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguraystd::atomic_bool gatt_registering(false);
934fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguraystd::atomic_int gatt_client_id(0);
942e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
95ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski// True if we should dump the scan record bytes for incoming scan results.
96ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowskistd::atomic_bool dump_scan_record(false);
97ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
982e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray// True if the remote process has died and we should exit.
992e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguraystd::atomic_bool should_exit(false);
1002e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
10139a66bed960d96eca900e7e002e0d7bef0e0e151Arman Ugurayvoid PrintPrompt() {
10239a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  cout << COLOR_BLUE "[FCLI] " COLOR_OFF << flush;
10339a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray}
10439a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
1052e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Ugurayvoid PrintError(const string& message) {
1062e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  cout << COLOR_RED << message << COLOR_OFF << endl;
1072e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray}
1082e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
109d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Ugurayvoid PrintOpStatus(const std::string& op, bool status) {
110d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  cout << COLOR_BOLDWHITE << op << " status: " COLOR_OFF
111d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray       << (status ? (COLOR_GREEN "success") : (COLOR_RED "failure"))
11264401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski       << COLOR_OFF << endl;
11364401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski}
11464401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski
11564401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowskiinline void BeginAsyncOut() {
11664401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski  if (showing_prompt.load())
11764401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    cout << CLEAR_LINE << "\r";
11864401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski}
11964401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski
12064401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowskiinline void EndAsyncOut() {
12164401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski  std::flush(cout);
12264401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski  if (showing_prompt.load())
12364401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski      PrintPrompt();
12464401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski  else
12564401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    cout << endl;
126d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray}
127d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
12839a66bed960d96eca900e7e002e0d7bef0e0e151Arman Ugurayclass CLIBluetoothCallback : public ipc::binder::BnBluetoothCallback {
12939a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray public:
13039a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  CLIBluetoothCallback() = default;
13139a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  ~CLIBluetoothCallback() override = default;
13239a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
1332e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  // IBluetoothCallback overrides:
13439a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  void OnBluetoothStateChange(
13539a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray      bluetooth::AdapterState prev_state,
13639a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray      bluetooth::AdapterState new_state) override {
13764401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski
13864401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    BeginAsyncOut();
13939a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray    cout << COLOR_BOLDWHITE "Adapter state changed: " COLOR_OFF
14039a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray         << COLOR_MAGENTA << AdapterStateToString(prev_state) << COLOR_OFF
14139a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray         << COLOR_BOLDWHITE " -> " COLOR_OFF
14264401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski         << COLOR_BOLDYELLOW << AdapterStateToString(new_state) << COLOR_OFF;
14364401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    EndAsyncOut();
14464401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski   }
14539a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
14639a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray private:
14739a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  DISALLOW_COPY_AND_ASSIGN(CLIBluetoothCallback);
14839a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray};
14939a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
1502e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Ugurayclass CLIBluetoothLowEnergyCallback
1512e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    : public ipc::binder::BnBluetoothLowEnergyCallback {
1522e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray public:
1532e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  CLIBluetoothLowEnergyCallback() = default;
1544fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  ~CLIBluetoothLowEnergyCallback() override = default;
1552e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
1562e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  // IBluetoothLowEnergyCallback overrides:
1574fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  void OnClientRegistered(int status, int client_id) override {
15864401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    BeginAsyncOut();
1592e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    if (status != bluetooth::BLE_STATUS_SUCCESS) {
1602e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray      PrintError("Failed to register BLE client");
1612e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    } else {
1624fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray      ble_client_id = client_id;
1632e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray      cout << COLOR_BOLDWHITE "Registered BLE client with ID: " COLOR_OFF
16464401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski           << COLOR_GREEN << client_id << COLOR_OFF;
1652e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    }
16664401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    EndAsyncOut();
1672e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
1682e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    ble_registering = false;
1692e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
1702e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
171455dc8f535a719af6a65a7512d90f9db878f5a58Jakub Pawlowski  void OnConnectionState(int status, int client_id, const char* address,
172455dc8f535a719af6a65a7512d90f9db878f5a58Jakub Pawlowski                         bool connected) override {
17364401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    BeginAsyncOut();
174608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    cout << COLOR_BOLDWHITE "Connection state: "
175608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski         << COLOR_BOLDYELLOW "[" << address
176608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski         << " connected: " << (connected ? "true" : "false") << " ] "
177608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski         << COLOR_BOLDWHITE "- status: " << status
178608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski         << COLOR_BOLDWHITE " - client_id: " << client_id << COLOR_OFF;
17964401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    EndAsyncOut();
180455dc8f535a719af6a65a7512d90f9db878f5a58Jakub Pawlowski  }
181455dc8f535a719af6a65a7512d90f9db878f5a58Jakub Pawlowski
182a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski  void OnMtuChanged(int status, const char *address, int mtu) override {
18364401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    BeginAsyncOut();
184a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski    cout << COLOR_BOLDWHITE "MTU changed: "
185a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski         << COLOR_BOLDYELLOW "[" << address << " ] "
186a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski         << COLOR_BOLDWHITE " - status: " << status
187a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski         << COLOR_BOLDWHITE " - mtu: " << mtu << COLOR_OFF;
18864401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    EndAsyncOut();
189a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski  }
190a6551079fe71b1c76505ada0e4f758f6faf651e0Jakub Pawlowski
191ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  void OnScanResult(const bluetooth::ScanResult& scan_result) override {
19264401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    BeginAsyncOut();
193ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    cout << COLOR_BOLDWHITE "Scan result: "
194ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski         << COLOR_BOLDYELLOW "[" << scan_result.device_address() << "] "
195ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski         << COLOR_BOLDWHITE "- RSSI: " << scan_result.rssi() << COLOR_OFF;
196ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
197ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    if (dump_scan_record) {
198ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski      cout << " - Record: "
199ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski           << base::HexEncode(scan_result.scan_record().data(),
200ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski                              scan_result.scan_record().size());
201ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    }
20264401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    EndAsyncOut();
203ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  }
204ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
20552bfc6060cec652a67c8989e0548225af0008be1Arman Uguray  void OnMultiAdvertiseCallback(
20652bfc6060cec652a67c8989e0548225af0008be1Arman Uguray      int status, bool is_start,
207d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      const bluetooth::AdvertiseSettings& /* settings */) {
20864401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    BeginAsyncOut();
209d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    std::string op = is_start ? "start" : "stop";
210d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
211d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    PrintOpStatus("Advertising " + op, status == bluetooth::BLE_STATUS_SUCCESS);
21264401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    EndAsyncOut();
21352bfc6060cec652a67c8989e0548225af0008be1Arman Uguray  }
21452bfc6060cec652a67c8989e0548225af0008be1Arman Uguray
2152e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray private:
2162e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  DISALLOW_COPY_AND_ASSIGN(CLIBluetoothLowEnergyCallback);
2172e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray};
2185192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
2194fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Ugurayclass CLIGattClientCallback
2204fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    : public ipc::binder::BnBluetoothGattClientCallback {
2214fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray public:
2224fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  CLIGattClientCallback() = default;
2234fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  ~CLIGattClientCallback() override = default;
2244fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
2254fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  // IBluetoothGattClientCallback overrides:
2264fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  void OnClientRegistered(int status, int client_id) override {
22764401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    BeginAsyncOut();
2284fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    if (status != bluetooth::BLE_STATUS_SUCCESS) {
2294fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray      PrintError("Failed to register GATT client");
2304fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    } else {
2314fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray      gatt_client_id = client_id;
2324fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray      cout << COLOR_BOLDWHITE "Registered GATT client with ID: " COLOR_OFF
23364401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski           << COLOR_GREEN << client_id << COLOR_OFF;
2344fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    }
23564401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    EndAsyncOut();
2364fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
2374fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    gatt_registering = false;
2384fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  }
2394fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
2404fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray private:
2414fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  DISALLOW_COPY_AND_ASSIGN(CLIGattClientCallback);
2424fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray};
2434fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
244fcf2e0391950a8b140082fbe78688fa89471fbedArman Ugurayvoid PrintCommandStatus(bool status) {
245d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  PrintOpStatus("Command", status);
246fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray}
247fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
2485192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid PrintFieldAndValue(const string& field, const string& value) {
2495192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  cout << COLOR_BOLDWHITE << field << ": " << COLOR_BOLDYELLOW << value
2505192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray       << COLOR_OFF << endl;
2515192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray}
2525192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
2535192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid PrintFieldAndBoolValue(const string& field, bool value) {
2545192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndValue(field, (value ? "true" : "false"));
2555192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray}
2565192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
2575192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleDisable(IBluetooth* bt_iface, const vector<string>& args) {
2585192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_NO_ARGS(args);
259fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  PrintCommandStatus(bt_iface->Disable());
260fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray}
261fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
2625192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleEnable(IBluetooth* bt_iface, const vector<string>& args) {
2635192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_NO_ARGS(args);
264fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  PrintCommandStatus(bt_iface->Enable());
265fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray}
266fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
2675192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleGetState(IBluetooth* bt_iface, const vector<string>& args) {
2685192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_NO_ARGS(args);
2692117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray  bluetooth::AdapterState state = static_cast<bluetooth::AdapterState>(
2702117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray      bt_iface->GetState());
2715192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndValue("Adapter state", bluetooth::AdapterStateToString(state));
272fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray}
273fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
2745192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleIsEnabled(IBluetooth* bt_iface, const vector<string>& args) {
2755192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_NO_ARGS(args);
276fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  bool enabled = bt_iface->IsEnabled();
2775192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndBoolValue("Adapter enabled", enabled);
278fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray}
279fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
2805192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleGetLocalAddress(IBluetooth* bt_iface, const vector<string>& args) {
2815192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_NO_ARGS(args);
2825192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  string address = bt_iface->GetAddress();
2835192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndValue("Adapter address", address);
2845192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray}
2855192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
2865192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleSetLocalName(IBluetooth* bt_iface, const vector<string>& args) {
2875192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_ARGS_COUNT(args, >=, 1, "No name was given");
2885192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
2895192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  std::string name;
2905192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  for (const auto& arg : args)
2915192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray    name += arg + " ";
2925192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
2935192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  base::TrimWhitespaceASCII(name, base::TRIM_TRAILING, &name);
2945192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
2955192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintCommandStatus(bt_iface->SetName(name));
2965192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray}
2975192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
2985192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleGetLocalName(IBluetooth* bt_iface, const vector<string>& args) {
2995192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_NO_ARGS(args);
3005192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  string name = bt_iface->GetName();
3015192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndValue("Adapter name", name);
3025192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray}
3035192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
3045192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleAdapterInfo(IBluetooth* bt_iface, const vector<string>& args) {
3055192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  CHECK_NO_ARGS(args);
3065192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
3075192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  cout << COLOR_BOLDWHITE "Adapter Properties: " COLOR_OFF << endl;
3085192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
3095192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndValue("\tAddress", bt_iface->GetAddress());
3105192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndValue("\tState", bluetooth::AdapterStateToString(
3115192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray      static_cast<bluetooth::AdapterState>(bt_iface->GetState())));
3125192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  PrintFieldAndValue("\tName", bt_iface->GetName());
31310b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray  PrintFieldAndBoolValue("\tMulti-Adv. supported",
31410b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray                         bt_iface->IsMultiAdvertisementSupported());
31510b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray}
31610b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray
31710b54c4b7f1a863a27eca4158f256062ec9c3770Arman Ugurayvoid HandleSupportsMultiAdv(IBluetooth* bt_iface, const vector<string>& args) {
31810b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray  CHECK_NO_ARGS(args);
31910b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray
32010b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray  bool status = bt_iface->IsMultiAdvertisementSupported();
32110b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray  PrintFieldAndBoolValue("Multi-advertisement support", status);
3225192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray}
3235192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
3242e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Ugurayvoid HandleRegisterBLE(IBluetooth* bt_iface, const vector<string>& args) {
3252e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  CHECK_NO_ARGS(args);
3262e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3272e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  if (ble_registering.load()) {
3282e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    PrintError("In progress");
3292e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    return;
3302e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
3312e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3324fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (ble_client_id.load()) {
3332e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    PrintError("Already registered");
3342e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    return;
3352e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
3362e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3372e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
3382e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  if (!ble_iface.get()) {
3392e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
3402e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    return;
3412e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
3422e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
343ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  bool status = ble_iface->RegisterClient(new CLIBluetoothLowEnergyCallback());
344ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  ble_registering = status;
345ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  PrintCommandStatus(status);
3462e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray}
3472e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3482e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Ugurayvoid HandleUnregisterBLE(IBluetooth* bt_iface, const vector<string>& args) {
3492e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  CHECK_NO_ARGS(args);
3502e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3514fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (!ble_client_id.load()) {
3522e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    PrintError("Not registered");
3532e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    return;
3542e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
3552e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3562e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
3572e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  if (!ble_iface.get()) {
3582e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
3592e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    return;
3602e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
3612e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3624fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  ble_iface->UnregisterClient(ble_client_id.load());
3634fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  ble_client_id = 0;
3642e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  PrintCommandStatus(true);
3652e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray}
3662e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3672e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Ugurayvoid HandleUnregisterAllBLE(IBluetooth* bt_iface, const vector<string>& args) {
3682e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  CHECK_NO_ARGS(args);
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
3762e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  ble_iface->UnregisterAll();
3772e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  PrintCommandStatus(true);
3782e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray}
3792e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
3804fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Ugurayvoid HandleRegisterGATT(IBluetooth* bt_iface, const vector<string>& args) {
3814fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  CHECK_NO_ARGS(args);
3824fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
3834fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (gatt_registering.load()) {
3844fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    PrintError("In progress");
3854fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    return;
3864fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  }
3874fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
3884fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (gatt_client_id.load()) {
3894fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    PrintError("Already registered");
3904fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    return;
3914fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  }
3924fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
3934fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  sp<IBluetoothGattClient> gatt_iface = bt_iface->GetGattClientInterface();
3944fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (!gatt_iface.get()) {
3954fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    PrintError("Failed to obtain handle to Bluetooth GATT Client interface");
3964fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    return;
3974fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  }
3984fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
3994fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  bool status = gatt_iface->RegisterClient(new CLIGattClientCallback());
4004fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  gatt_registering = status;
4014fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  PrintCommandStatus(status);
4024fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray}
4034fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
4044fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Ugurayvoid HandleUnregisterGATT(IBluetooth* bt_iface, const vector<string>& args) {
4054fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  CHECK_NO_ARGS(args);
4064fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
4074fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (!gatt_client_id.load()) {
4084fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    PrintError("Not registered");
4094fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    return;
4104fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  }
4114fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
4124fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  sp<IBluetoothGattClient> gatt_iface = bt_iface->GetGattClientInterface();
4134fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (!gatt_iface.get()) {
4144fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    PrintError("Failed to obtain handle to Bluetooth GATT Client interface");
4154fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    return;
4164fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  }
4174fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
4184fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  gatt_iface->UnregisterClient(gatt_client_id.load());
4194fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  gatt_client_id = 0;
4204fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  PrintCommandStatus(true);
4214fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray}
4224fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray
423d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Ugurayvoid HandleStartAdv(IBluetooth* bt_iface, const vector<string>& args) {
424d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  bool include_name = false;
425d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  bool include_tx_power = false;
426d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  bool connectable = false;
427d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  bool set_manufacturer_data = false;
42887222e0e826216c69f6a9a5bfe77689561067474Arman Uguray  bool set_uuid = false;
42987222e0e826216c69f6a9a5bfe77689561067474Arman Uguray  bluetooth::UUID uuid;
430d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
43187222e0e826216c69f6a9a5bfe77689561067474Arman Uguray  for (auto iter = args.begin(); iter != args.end(); ++iter) {
43287222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    const std::string& arg = *iter;
433d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    if (arg == "-n")
434d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      include_name = true;
435d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    else if (arg == "-t")
436d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      include_tx_power = true;
437d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    else if (arg == "-c")
438d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      connectable = true;
439d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    else if (arg == "-m")
440d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      set_manufacturer_data = true;
44187222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    else if (arg == "-u") {
44287222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      // This flag has a single argument.
44387222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      ++iter;
44487222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      if (iter == args.end()) {
44587222e0e826216c69f6a9a5bfe77689561067474Arman Uguray        PrintError("Expected a UUID after -u");
44687222e0e826216c69f6a9a5bfe77689561067474Arman Uguray        return;
44787222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      }
44887222e0e826216c69f6a9a5bfe77689561067474Arman Uguray
44987222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      std::string uuid_str = *iter;
45087222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      uuid = bluetooth::UUID(uuid_str);
45187222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      if (!uuid.is_valid()) {
45287222e0e826216c69f6a9a5bfe77689561067474Arman Uguray        PrintError("Invalid UUID: " + uuid_str);
45387222e0e826216c69f6a9a5bfe77689561067474Arman Uguray        return;
45487222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      }
45587222e0e826216c69f6a9a5bfe77689561067474Arman Uguray
45687222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      set_uuid = true;
45787222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    }
458d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    else if (arg == "-h") {
459ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski      static const char kUsage[] =
460d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray          "Usage: start-adv [flags]\n"
461d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray          "\n"
462ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski          "Flags:\n"
463d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray          "\t-n\tInclude device name\n"
464d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray          "\t-t\tInclude TX power\n"
465d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray          "\t-c\tSend connectable adv. packets (default is non-connectable)\n"
466d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray          "\t-m\tInclude random manufacturer data\n"
467d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray          "\t-h\tShow this help message\n";
468d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      cout << kUsage << endl;
469d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      return;
470d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    }
471d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    else {
472d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      PrintError("Unrecognized option: " + arg);
473d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      return;
474d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    }
475d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  }
476d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
4774fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (!ble_client_id.load()) {
478d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    PrintError("BLE not registered");
479d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    return;
480d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  }
481d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
482d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
483d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  if (!ble_iface.get()) {
484d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
485d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    return;
486d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  }
487d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
488d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  std::vector<uint8_t> data;
489d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  if (set_manufacturer_data) {
490d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    data = {{
491d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      0x07, bluetooth::kEIRTypeManufacturerSpecificData,
492d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      0xe0, 0x00,
493d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      'T', 'e', 's', 't'
494d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    }};
495d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  }
496d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
49787222e0e826216c69f6a9a5bfe77689561067474Arman Uguray  if (set_uuid) {
49887222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    // Determine the type and length bytes.
49987222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    int uuid_size = uuid.GetShortestRepresentationSize();
50087222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    uint8_t type;
50187222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    if (uuid_size == bluetooth::UUID::kNumBytes128)
50287222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      type = bluetooth::kEIRTypeComplete128BitUUIDs;
50387222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    else if (uuid_size == bluetooth::UUID::kNumBytes32)
50487222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      type = bluetooth::kEIRTypeComplete32BitUUIDs;
50587222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    else if (uuid_size == bluetooth::UUID::kNumBytes16)
50687222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      type = bluetooth::kEIRTypeComplete16BitUUIDs;
50787222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    else
50887222e0e826216c69f6a9a5bfe77689561067474Arman Uguray      NOTREACHED() << "Unexpected size: " << uuid_size;
50987222e0e826216c69f6a9a5bfe77689561067474Arman Uguray
51087222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    data.push_back(uuid_size + 1);
51187222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    data.push_back(type);
51287222e0e826216c69f6a9a5bfe77689561067474Arman Uguray
51387222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    auto uuid_bytes = uuid.GetFullLittleEndian();
51487222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    int index = (uuid_size == 16) ? 0 : 12;
51587222e0e826216c69f6a9a5bfe77689561067474Arman Uguray    data.insert(data.end(), uuid_bytes.data() + index,
51687222e0e826216c69f6a9a5bfe77689561067474Arman Uguray                uuid_bytes.data() + index + uuid_size);
51787222e0e826216c69f6a9a5bfe77689561067474Arman Uguray  }
51887222e0e826216c69f6a9a5bfe77689561067474Arman Uguray
519d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  base::TimeDelta timeout;
520d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
521d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  bluetooth::AdvertiseSettings settings(
522d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      bluetooth::AdvertiseSettings::MODE_LOW_POWER,
523d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      timeout,
524d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      bluetooth::AdvertiseSettings::TX_POWER_LEVEL_MEDIUM,
525d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray      connectable);
526d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
527d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  bluetooth::AdvertiseData adv_data(data);
528d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  adv_data.set_include_device_name(include_name);
529d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  adv_data.set_include_tx_power_level(include_tx_power);
530d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
531d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  bluetooth::AdvertiseData scan_rsp;
532d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
5334fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  bool status = ble_iface->StartMultiAdvertising(ble_client_id.load(),
534ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray                                                 adv_data, scan_rsp, settings);
535ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  PrintCommandStatus(status);
536d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray}
537d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
538d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Ugurayvoid HandleStopAdv(IBluetooth* bt_iface, const vector<string>& args) {
5394fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  if (!ble_client_id.load()) {
540d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    PrintError("BLE not registered");
541d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    return;
542d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  }
543d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
544d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
545d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  if (!ble_iface.get()) {
546d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
547d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray    return;
548d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  }
549d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
5504fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  bool status = ble_iface->StopMultiAdvertising(ble_client_id.load());
551ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  PrintCommandStatus(status);
552d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray}
553d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray
554608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowskivoid HandleConnect(IBluetooth* bt_iface, const vector<string>& args) {
555608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  string address;
556608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
557608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  if (args.size() != 1) {
558608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    PrintError("Expected MAC address as only argument");
559608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    return;
560608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  }
561608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
562608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  address = args[0];
563608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
564608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  if (!ble_client_id.load()) {
565608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    PrintError("BLE not registered");
566608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    return;
567608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  }
568608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
569608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
570608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  if (!ble_iface.get()) {
571608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
572608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    return;
573608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  }
574608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
575608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  bool status = ble_iface->Connect(ble_client_id.load(), address.c_str(),
576608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski                                   false /*  is_direct */);
577608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
578608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  PrintCommandStatus(status);
579608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski}
580608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
581608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowskivoid HandleDisconnect(IBluetooth* bt_iface, const vector<string>& args) {
582608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  string address;
583608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
584608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  if (args.size() != 1) {
585608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    PrintError("Expected MAC address as only argument");
586608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    return;
587608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  }
588608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
589608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  address = args[0];
590608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
591608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  if (!ble_client_id.load()) {
592608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    PrintError("BLE not registered");
593608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    return;
594608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  }
595608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
596608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
597608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  if (!ble_iface.get()) {
598608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
599608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    return;
600608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  }
601608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
602608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  bool status = ble_iface->Disconnect(ble_client_id.load(), address.c_str());
603608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
604608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  PrintCommandStatus(status);
605608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski}
606608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski
607756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowskivoid HandleSetMtu(IBluetooth* bt_iface, const vector<string>& args) {
608756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  string address;
609756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  int mtu;
610756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski
611756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  if (args.size() != 2) {
612756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski    PrintError("Usage: set-mtu [address] [mtu]");
613756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski    return;
614756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  }
615756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski
616756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  address = args[0];
617756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  mtu = std::stoi(args[1]);
618756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski
619756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  if (mtu < 23) {
620756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski    PrintError("MTU must be 23 or larger");
621756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski    return;
622756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  }
623756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski
624756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  if (!ble_client_id.load()) {
625756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski    PrintError("BLE not registered");
626756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski    return;
627756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  }
628756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski
629756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
630756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  if (!ble_iface.get()) {
631756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
632756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski    return;
633756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  }
634756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski
635756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  bool status = ble_iface->SetMtu(ble_client_id.load(), address.c_str(), mtu);
636756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  PrintCommandStatus(status);
637756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski}
638756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski
639ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowskivoid HandleStartLeScan(IBluetooth* bt_iface, const vector<string>& args) {
640ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  if (!ble_client_id.load()) {
641ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    PrintError("BLE not registered");
642ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    return;
643ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  }
644ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
645ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  for (auto arg : args) {
646ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    if (arg == "-d") {
647ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski      dump_scan_record = true;
648ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    } else if (arg == "-h") {
649ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski      static const char kUsage[] =
650ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski          "Usage: start-le-scan [flags]\n"
651ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski          "\n"
652ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski          "Flags:\n"
653ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski          "\t-d\tDump scan record\n"
654ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski          "\t-h\tShow this help message\n";
655ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski      cout << kUsage << endl;
656ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski      return;
657ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    }
658ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  }
659ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
660ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
661ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  if (!ble_iface.get()) {
662ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
663ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    return;
664ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  }
665ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
666ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  bluetooth::ScanSettings settings;
667ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  std::vector<bluetooth::ScanFilter> filters;
668ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
669ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  bool status = ble_iface->StartScan(ble_client_id.load(), settings, filters);
670ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  PrintCommandStatus(status);
671ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski}
672ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
673ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowskivoid HandleStopLeScan(IBluetooth* bt_iface, const vector<string>& args) {
674ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  if (!ble_client_id.load()) {
675ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    PrintError("BLE not registered");
676ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    return;
677ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  }
678ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
679ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  sp<IBluetoothLowEnergy> ble_iface = bt_iface->GetLowEnergyInterface();
680ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  if (!ble_iface.get()) {
681ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    PrintError("Failed to obtain handle to Bluetooth Low Energy interface");
682ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    return;
683ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  }
684ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
685ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  bluetooth::ScanSettings settings;
686ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  std::vector<bluetooth::ScanFilter> filters;
687ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
688ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  bool status = ble_iface->StopScan(ble_client_id.load());
689ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  PrintCommandStatus(status);
690ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski}
691ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski
6925192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleHelp(IBluetooth* bt_iface, const vector<string>& args);
6932117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray
694fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguraystruct {
695fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  string command;
6965192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  void (*func)(IBluetooth*, const vector<string>& args);
6972117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray  string help;
698fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray} kCommandMap[] = {
6995192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "help", HandleHelp, "\t\t\tDisplay this message" },
7005192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "disable", HandleDisable, "\t\t\tDisable Bluetooth" },
7015192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "enable", HandleEnable, "\t\t\tEnable Bluetooth" },
7025192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "get-state", HandleGetState, "\t\tGet the current adapter state" },
7035192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "is-enabled", HandleIsEnabled, "\t\tReturn if Bluetooth is enabled" },
7045192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "get-local-address", HandleGetLocalAddress,
7055192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray    "\tGet the local adapter address" },
7065192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "set-local-name", HandleSetLocalName, "\t\tSet the local adapter name" },
7075192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "get-local-name", HandleGetLocalName, "\t\tGet the local adapter name" },
7085192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray  { "adapter-info", HandleAdapterInfo, "\t\tPrint adapter properties" },
70910b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray  { "supports-multi-adv", HandleSupportsMultiAdv,
71010b54c4b7f1a863a27eca4158f256062ec9c3770Arman Uguray    "\tWhether multi-advertisement is currently supported" },
7112e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  { "register-ble", HandleRegisterBLE,
7122e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    "\t\tRegister with the Bluetooth Low Energy interface" },
7132e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  { "unregister-ble", HandleUnregisterBLE,
7142e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    "\t\tUnregister from the Bluetooth Low Energy interface" },
7152e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  { "unregister-all-ble", HandleUnregisterAllBLE,
7162e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    "\tUnregister all clients from the Bluetooth Low Energy interface" },
7174fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  { "register-gatt", HandleRegisterGATT,
7184fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    "\t\tRegister with the Bluetooth GATT Client interface" },
7194fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray  { "unregister-gatt", HandleUnregisterGATT,
7204fbbf6047f182b9dfbf11d5d8da7281845fce99eArman Uguray    "\t\tUnregister from the Bluetooth GATT Client interface" },
721608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  { "connect-le", HandleConnect, "\t\tConnect to LE device (-h for options)"},
722608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski  { "disconnect-le", HandleDisconnect,
723608762d5769b948387c8d76e7f1d5a60db1850aeJakub Pawlowski    "\t\tDisconnect LE device (-h for options)"},
724756cc4bdedca24e4ba157902a3d0b411cfa109beJakub Pawlowski  { "set-mtu", HandleSetMtu, "\t\tSet MTU (-h for options)"},
725d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  { "start-adv", HandleStartAdv, "\t\tStart advertising (-h for options)" },
726d19bc0457a9b6519acd6a79c3ac7de653894f5ecArman Uguray  { "stop-adv", HandleStopAdv, "\t\tStop advertising" },
727ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  { "start-le-scan", HandleStartLeScan,
728ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski    "\t\tStart LE device scan (-h for options)" },
729ba197a21d810b0ef814ed9c23dd3b290613751c7Jakub Pawlowski  { "stop-le-scan", HandleStopLeScan, "\t\tStop LE device scan" },
730fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  {},
731fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray};
732f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray
7335192309af14408c3f170f15c1282ae5c1eb5abffArman Ugurayvoid HandleHelp(IBluetooth* /* bt_iface */, const vector<string>& /* args */) {
7342117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray  cout << endl;
7352117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray  for (int i = 0; kCommandMap[i].func; i++)
7362117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray    cout << "\t" << kCommandMap[i].command << kCommandMap[i].help << endl;
7372117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray  cout << endl;
7382117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray}
7392117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray
7402117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray}  // namespace
7412117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray
7422e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Ugurayclass BluetoothDeathRecipient : public android::IBinder::DeathRecipient {
7432e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray public:
7442e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  BluetoothDeathRecipient() = default;
7452e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  ~BluetoothDeathRecipient() override = default;
7462e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
7472e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  // android::IBinder::DeathRecipient override:
7482e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  void binderDied(const android::wp<android::IBinder>& /* who */) override {
74964401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    BeginAsyncOut();
7502e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    cout << COLOR_BOLDWHITE "The Bluetooth daemon has died" COLOR_OFF << endl;
75164401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    cout << "\nPress 'ENTER' to exit.";
75264401bf539bdef652ddcfc25138ad5e353aea1c3Jakub Pawlowski    EndAsyncOut();
7532e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
7542e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    android::IPCThreadState::self()->stopProcess();
7552e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    should_exit = true;
7562e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
7572e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
7582e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray private:
7592e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  DISALLOW_COPY_AND_ASSIGN(BluetoothDeathRecipient);
7602e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray};
7612e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
762ae43de627b9c91e54215d439149c0e01599249c0Arman Ugurayint main(int argc, char* argv[]) {
763ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  base::AtExitManager exit_manager;
764ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  base::CommandLine::Init(argc, argv);
765ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  logging::LoggingSettings log_settings;
766ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray
767ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  if (!logging::InitLogging(log_settings)) {
768ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray    LOG(ERROR) << "Failed to set up logging";
769ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray    return EXIT_FAILURE;
770ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray  }
771ae43de627b9c91e54215d439149c0e01599249c0Arman Uguray
772f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray  sp<IBluetooth> bt_iface = IBluetooth::getClientInterface();
773f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray  if (!bt_iface.get()) {
774f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray    LOG(ERROR) << "Failed to obtain handle on IBluetooth";
775f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray    return EXIT_FAILURE;
776f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray  }
777f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray
7782e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  sp<BluetoothDeathRecipient> dr(new BluetoothDeathRecipient());
7792e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  if (android::IInterface::asBinder(bt_iface.get())->linkToDeath(dr) !=
7802e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray      android::NO_ERROR) {
7812e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    LOG(ERROR) << "Failed to register DeathRecipient for IBluetooth";
7822e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray    return EXIT_FAILURE;
7832e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray  }
7842e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
78539a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  // Initialize the Binder process thread pool. We have to set this up,
78639a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  // otherwise, incoming callbacks from IBluetoothCallback will block the main
78739a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  // thread (in other words, we have to do this as we are a "Binder server").
78839a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  android::ProcessState::self()->startThreadPool();
78939a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
79039a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  // Register Adapter state-change callback
79139a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  sp<CLIBluetoothCallback> callback = new CLIBluetoothCallback();
79239a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray  bt_iface->RegisterCallback(callback);
79339a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
794fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  cout << COLOR_BOLDWHITE << "Fluoride Command-Line Interface\n" << COLOR_OFF
7952117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray       << endl
7962117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray       << "Type \"help\" to see possible commands.\n"
797fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray       << endl;
798fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
799fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  while (true) {
800fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray    string command;
8015192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
80239a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray    PrintPrompt();
80339a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray
80439a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray    showing_prompt = true;
80591c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray    auto& istream = getline(cin, command);
80639a66bed960d96eca900e7e002e0d7bef0e0e151Arman Uguray    showing_prompt = false;
807fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
80891c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray    if (istream.eof() || should_exit.load()) {
80991c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray      cout << "\nExiting" << endl;
8102e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray      return EXIT_SUCCESS;
81191c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray    }
81291c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray
81391c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray    if (!istream.good()) {
81491c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray      LOG(ERROR) << "An error occured while reading input";
81591c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray      return EXIT_FAILURE;
81691c59c9febbf4a49664b8a06620d4a4882605131Arman Uguray    }
8172e4341488064be7e8f4d575c8de0a1670ddc81a0Arman Uguray
81857d7bf98ea5f7a08c93c30f5d8a0e303d78ea432Alex Vakulenko    vector<string> args =
81957d7bf98ea5f7a08c93c30f5d8a0e303d78ea432Alex Vakulenko        base::SplitString(command, " ", base::TRIM_WHITESPACE,
82057d7bf98ea5f7a08c93c30f5d8a0e303d78ea432Alex Vakulenko                          base::SPLIT_WANT_ALL);
8215192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
8225192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray    if (args.empty())
8235192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray      continue;
8245192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
8255192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray    // The first argument is the command while the remaning are what we pass to
8265192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray    // the handler functions.
8275192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray    command = args[0];
8285192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray    args.erase(args.begin());
8295192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray
830fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray    bool command_handled = false;
8312117e520c9f5b105ade7e92c4ab4928ea905f176Arman Uguray    for (int i = 0; kCommandMap[i].func && !command_handled; i++) {
832fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray      if (command == kCommandMap[i].command) {
8335192309af14408c3f170f15c1282ae5c1eb5abffArman Uguray        kCommandMap[i].func(bt_iface.get(), args);
834fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray        command_handled = true;
835fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray      }
836fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray    }
837fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray
838fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray    if (!command_handled)
839fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray      cout << "Unrecognized command: " << command << endl;
840fcf2e0391950a8b140082fbe78688fa89471fbedArman Uguray  }
841f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray
842f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray  return EXIT_SUCCESS;
843f52095257e36b887d7ddfd1f00871b9311dbfa1bArman Uguray}
844