//
// Copyright (C) 2015 Google, Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at:
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
#pragma once
#include <mutex>
#include <unordered_map>
#include <base/macros.h>
#include <base/memory/ref_counted.h>
#include <base/memory/weak_ptr.h>
#include <base/single_thread_task_runner.h>
#include <android/bluetooth/BnBluetoothGattServerCallback.h>
#include <android/bluetooth/IBluetooth.h>
using android::binder::Status;
using android::String16;
namespace heart_rate {
// Implements an example GATT Heart Rate service. This class emulates the
// behavior of a heart rate service by sending fake heart-rate pulses.
class HeartRateServer
: public android::bluetooth::BnBluetoothGattServerCallback {
public:
HeartRateServer(android::sp<android::bluetooth::IBluetooth> bluetooth,
scoped_refptr<base::SingleThreadTaskRunner> main_task_runner,
bool advertise);
~HeartRateServer() override;
// Set up the server and register the GATT services with the stack. This
// initiates a set of asynchronous procedures. Invokes |callback|
// asynchronously with the result of the operation.
using RunCallback = std::function<void(bool success)>;
bool Run(const RunCallback& callback);
private:
// Helpers for posting heart rate measurement notifications.
void ScheduleNextMeasurement();
void SendHeartRateMeasurement();
void BuildHeartRateMeasurementValue(std::vector<uint8_t>* out_value);
// ipc::binder::IBluetoothGattServerCallback override:
Status OnServerRegistered(int status, int server_id) override;
Status OnServiceAdded(
int status,
const android::bluetooth::BluetoothGattService& service) override;
Status OnCharacteristicReadRequest(const String16& device_address,
int request_id, int offset, bool is_long,
int handle) override;
Status OnDescriptorReadRequest(const String16& device_address, int request_id,
int offset, bool is_long, int handle) override;
Status OnCharacteristicWriteRequest(const String16& device_address,
int request_id, int offset,
bool is_prepare_write, bool need_response,
const std::vector<uint8_t>& value,
int handle) override;
Status OnDescriptorWriteRequest(const String16& device_address,
int request_id, int offset,
bool is_prepare_write, bool need_response,
const std::vector<uint8_t>& value,
int handle) override;
Status OnExecuteWriteRequest(const String16& device_address, int request_id,
bool is_execute) override;
Status OnNotificationSent(const String16& device_address,
int status) override;
Status OnConnectionStateChanged(const String16& device_address,
bool connected) override;
// Single mutex to protect all variables below.
std::mutex mutex_;
// This stores whether or not at least one remote device has written to the
// CCC descriptor.
bool simulation_started_;
// The IBluetooth and IBluetoothGattServer binders that we use to communicate
// with the Bluetooth daemon's GATT server features.
android::sp<android::bluetooth::IBluetooth> bluetooth_;
android::sp<android::bluetooth::IBluetoothGattServer> gatt_;
// ID assigned to us by the daemon to operate on our dedicated GATT server
// instance.
int server_if_;
// Callback passed to Run(). We use this to tell main that all attributes have
// been registered with the daemon.
RunCallback pending_run_cb_;
// Stores whether or not an outgoing notification is still pending. We use
// this to throttle notifications so that we don't accidentally congest the
// connection.
std::unordered_map<std::string, bool> pending_notification_map_;
// The current HR notification count.
int hr_notification_count_;
// The Energy Expended value we use in our notifications.
uint16_t energy_expended_;
// Handles that refer to Heart Rate Service GATT objects.
// These returned to us from the Bluetooth daemon as we populate the database.
uint16_t hr_service_handle_;
uint16_t hr_measurement_handle_;
uint16_t hr_measurement_cccd_handle_;
uint16_t body_sensor_loc_handle_;
uint16_t hr_control_point_handle_;
// The daemon itself doesn't maintain a Client Characteristic Configuration
// mapping, so we do it ourselves here.
std::unordered_map<std::string, uint8_t> device_ccc_map_;
// Wether we should also start advertising
bool advertise_;
// libchrome task runner that we use to post heart rate measurement
// notifications on the main thread.
scoped_refptr<base::SingleThreadTaskRunner> main_task_runner_;
// We use this to pass weak_ptr's to base::Bind, which won't execute if the
// HeartRateServer object gets deleted. This is a convenience utility from
// libchrome and we use it here since base::TaskRunner uses base::Callback.
// Note: This should remain the last member so that it'll be destroyed and
// invalidate its weak pointers before any other members are destroyed.
base::WeakPtrFactory<HeartRateServer> weak_ptr_factory_;
DISALLOW_COPY_AND_ASSIGN(HeartRateServer);
};
} // namespace heart_rate