//
// Copyright (C) 2016 The Android Open Source Project
//
// 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.
//
namespace tpm_manager {
// Handles protobuf conversions for a binder proxy. Implements ITpmManagerClient
// and acts as a response observer for async requests it sends out. The
// |RequestProtobufType| and |ResponseProtobufType| types can be any protobufs.
template<typename RequestProtobufType, typename ResponseProtobufType>
class BinderProxyHelper {
public:
using CallbackType = base::Callback<void(const ResponseProtobufType&)>;
using GetErrorResponseCallbackType =
base::Callback<void(ResponseProtobufType*)>;
using BinderMethodType = base::Callback<android::binder::Status(
const std::vector<uint8_t>& command_proto,
const android::sp<android::tpm_manager::ITpmManagerClient>& client)>;
BinderProxyHelper(const BinderMethodType& method,
const CallbackType& callback,
const GetErrorResponseCallbackType& get_error_response)
: method_(method),
callback_(callback),
get_error_response_(get_error_response) {}
void SendErrorResponse() {
ResponseProtobufType response_proto;
get_error_response_.Run(&response_proto);
callback_.Run(response_proto);
}
void SendRequest(const RequestProtobufType& request) {
VLOG(2) << __func__;
android::sp<ResponseObserver> observer(
new ResponseObserver(callback_, get_error_response_));
std::vector<uint8_t> request_bytes;
request_bytes.resize(request.ByteSize());
if (!request.SerializeToArray(request_bytes.data(), request_bytes.size())) {
LOG(ERROR) << "BinderProxyHelper: Failed to serialize protobuf.";
SendErrorResponse();
return;
}
android::binder::Status status = method_.Run(request_bytes, observer);
if (!status.isOk()) {
LOG(ERROR) << "BinderProxyHelper: Binder error: " << status.toString8();
SendErrorResponse();
return;
}
}
private:
class ResponseObserver : public android::tpm_manager::BnTpmManagerClient {
public:
ResponseObserver(const CallbackType& callback,
const GetErrorResponseCallbackType& get_error_response)
: callback_(callback), get_error_response_(get_error_response) {}
// ITpmManagerClient interface.
android::binder::Status OnCommandResponse(
const std::vector<uint8_t>& response_proto_data) override {
VLOG(2) << __func__;
ResponseProtobufType response_proto;
if (!response_proto.ParseFromArray(response_proto_data.data(),
response_proto_data.size())) {
LOG(ERROR) << "BinderProxyHelper: Bad response data.";
get_error_response_.Run(&response_proto);
}
callback_.Run(response_proto);
return android::binder::Status::ok();
}
private:
CallbackType callback_;
GetErrorResponseCallbackType get_error_response_;
};
BinderMethodType method_;
CallbackType callback_;
GetErrorResponseCallbackType get_error_response_;
};
} // tpm_manager