blob: 015157d425b0f2cbb52698b4c5aedd2074dfc8e4 [file] [log] [blame]
Hao Chen6cb86892023-04-10 15:21:59 -07001/*
2 * Copyright (C) 2023 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include <GRPCVehicleHardware.h>
18
19#include "ProtoMessageConverter.h"
20
21#include <android-base/logging.h>
22#include <grpc++/grpc++.h>
23
24#include <cstdlib>
25#include <mutex>
26#include <shared_mutex>
27#include <utility>
28
29namespace android::hardware::automotive::vehicle::virtualization {
30
31static std::shared_ptr<::grpc::ChannelCredentials> getChannelCredentials() {
32 // TODO(chenhaosjtuacm): get secured credentials here
33 return ::grpc::InsecureChannelCredentials();
34}
35
36GRPCVehicleHardware::GRPCVehicleHardware(std::string service_addr)
37 : mServiceAddr(std::move(service_addr)),
38 mGrpcChannel(::grpc::CreateChannel(mServiceAddr, getChannelCredentials())),
39 mGrpcStub(proto::VehicleServer::NewStub(mGrpcChannel)),
40 mValuePollingThread([this] { ValuePollingLoop(); }) {}
41
42GRPCVehicleHardware::~GRPCVehicleHardware() {
43 {
44 std::lock_guard lck(mShutdownMutex);
45 mShuttingDownFlag.store(true);
46 }
47 mShutdownCV.notify_all();
48 mValuePollingThread.join();
49}
50
51std::vector<aidlvhal::VehiclePropConfig> GRPCVehicleHardware::getAllPropertyConfigs() const {
52 std::vector<aidlvhal::VehiclePropConfig> configs;
53 ::grpc::ClientContext context;
54 auto config_stream = mGrpcStub->GetAllPropertyConfig(&context, ::google::protobuf::Empty());
55 proto::VehiclePropConfig protoConfig;
56 while (config_stream->Read(&protoConfig)) {
57 aidlvhal::VehiclePropConfig config;
58 proto_msg_converter::protoToAidl(protoConfig, &config);
59 configs.push_back(std::move(config));
60 }
61 auto grpc_status = config_stream->Finish();
62 if (!grpc_status.ok()) {
63 LOG(ERROR) << __func__
64 << ": GRPC GetAllPropertyConfig Failed: " << grpc_status.error_message();
65 }
66 return configs;
67}
68
69aidlvhal::StatusCode GRPCVehicleHardware::setValues(
70 std::shared_ptr<const SetValuesCallback> callback,
71 const std::vector<aidlvhal::SetValueRequest>& requests) {
72 ::grpc::ClientContext context;
73 proto::VehiclePropValueRequests protoRequests;
74 proto::SetValueResults protoResults;
75 for (const auto& request : requests) {
76 auto& protoRequest = *protoRequests.add_requests();
77 protoRequest.set_request_id(request.requestId);
78 proto_msg_converter::aidlToProto(request.value, protoRequest.mutable_value());
79 }
80 // TODO(chenhaosjtuacm): Make it Async.
81 auto grpc_status = mGrpcStub->SetValues(&context, protoRequests, &protoResults);
82 if (!grpc_status.ok()) {
83 LOG(ERROR) << __func__ << ": GRPC SetValues Failed: " << grpc_status.error_message();
84 {
85 std::shared_lock lck(mCallbackMutex);
86 // TODO(chenhaosjtuacm): call on-set-error callback.
87 }
88 return aidlvhal::StatusCode::INTERNAL_ERROR;
89 }
90 std::vector<aidlvhal::SetValueResult> results;
91 for (const auto& protoResult : protoResults.results()) {
92 auto& result = results.emplace_back();
93 result.requestId = protoResult.request_id();
94 result.status = static_cast<aidlvhal::StatusCode>(protoResult.status());
95 // TODO(chenhaosjtuacm): call on-set-error callback.
96 }
97 (*callback)(std::move(results));
98
99 return aidlvhal::StatusCode::OK;
100}
101
102aidlvhal::StatusCode GRPCVehicleHardware::getValues(
103 std::shared_ptr<const GetValuesCallback> callback,
104 const std::vector<aidlvhal::GetValueRequest>& requests) const {
105 ::grpc::ClientContext context;
106 proto::VehiclePropValueRequests protoRequests;
107 proto::GetValueResults protoResults;
108 for (const auto& request : requests) {
109 auto& protoRequest = *protoRequests.add_requests();
110 protoRequest.set_request_id(request.requestId);
111 proto_msg_converter::aidlToProto(request.prop, protoRequest.mutable_value());
112 }
113 // TODO(chenhaosjtuacm): Make it Async.
114 auto grpc_status = mGrpcStub->GetValues(&context, protoRequests, &protoResults);
115 if (!grpc_status.ok()) {
116 LOG(ERROR) << __func__ << ": GRPC GetValues Failed: " << grpc_status.error_message();
117 return aidlvhal::StatusCode::INTERNAL_ERROR;
118 }
119 std::vector<aidlvhal::GetValueResult> results;
120 for (const auto& protoResult : protoResults.results()) {
121 auto& result = results.emplace_back();
122 result.requestId = protoResult.request_id();
123 result.status = static_cast<aidlvhal::StatusCode>(protoResult.status());
124 if (protoResult.has_value()) {
125 aidlvhal::VehiclePropValue value;
126 proto_msg_converter::protoToAidl(protoResult.value(), &value);
127 result.prop = std::move(value);
128 }
129 }
130 (*callback)(std::move(results));
131
132 return aidlvhal::StatusCode::OK;
133}
134
135void GRPCVehicleHardware::registerOnPropertyChangeEvent(
136 std::unique_ptr<const PropertyChangeCallback> callback) {
137 std::lock_guard lck(mCallbackMutex);
138 if (mOnPropChange) {
139 LOG(ERROR) << __func__ << " must only be called once.";
140 return;
141 }
142 mOnPropChange = std::move(callback);
143}
144
145void GRPCVehicleHardware::registerOnPropertySetErrorEvent(
146 std::unique_ptr<const PropertySetErrorCallback> callback) {
147 std::lock_guard lck(mCallbackMutex);
148 if (mOnSetErr) {
149 LOG(ERROR) << __func__ << " must only be called once.";
150 return;
151 }
152 mOnSetErr = std::move(callback);
153}
154
155DumpResult GRPCVehicleHardware::dump(const std::vector<std::string>& /* options */) {
156 // TODO(chenhaosjtuacm): To be implemented.
157 return {};
158}
159
160aidlvhal::StatusCode GRPCVehicleHardware::checkHealth() {
161 // TODO(chenhaosjtuacm): To be implemented.
162 return aidlvhal::StatusCode::OK;
163}
164
165aidlvhal::StatusCode GRPCVehicleHardware::updateSampleRate(int32_t /* propId */,
166 int32_t /* areaId */,
167 float /* sampleRate */) {
168 // TODO(chenhaosjtuacm): To be implemented.
169 return aidlvhal::StatusCode::OK;
170}
171
172bool GRPCVehicleHardware::waitForConnected(std::chrono::milliseconds waitTime) {
173 return mGrpcChannel->WaitForConnected(gpr_time_add(
174 gpr_now(GPR_CLOCK_MONOTONIC), gpr_time_from_millis(waitTime.count(), GPR_TIMESPAN)));
175}
176
177void GRPCVehicleHardware::ValuePollingLoop() {
178 while (!mShuttingDownFlag.load()) {
179 ::grpc::ClientContext context;
180
181 bool rpc_stopped{false};
182 std::thread shuttingdown_watcher([this, &rpc_stopped, &context]() {
183 std::unique_lock<std::mutex> lck(mShutdownMutex);
184 mShutdownCV.wait(lck, [this, &rpc_stopped]() {
185 return rpc_stopped || mShuttingDownFlag.load();
186 });
187 context.TryCancel();
188 });
189
190 auto value_stream =
191 mGrpcStub->StartPropertyValuesStream(&context, ::google::protobuf::Empty());
192 LOG(INFO) << __func__ << ": GRPC Value Streaming Started";
193 proto::VehiclePropValues protoValues;
194 while (!mShuttingDownFlag.load() && value_stream->Read(&protoValues)) {
195 std::vector<aidlvhal::VehiclePropValue> values;
196 for (const auto protoValue : protoValues.values()) {
197 values.push_back(aidlvhal::VehiclePropValue());
198 proto_msg_converter::protoToAidl(protoValue, &values.back());
199 }
200 std::shared_lock lck(mCallbackMutex);
201 if (mOnPropChange) {
202 (*mOnPropChange)(values);
203 }
204 }
205
206 {
207 std::lock_guard lck(mShutdownMutex);
208 rpc_stopped = true;
209 }
210 mShutdownCV.notify_all();
211 shuttingdown_watcher.join();
212
213 auto grpc_status = value_stream->Finish();
214 // never reach here until connection lost
215 LOG(ERROR) << __func__ << ": GRPC Value Streaming Failed: " << grpc_status.error_message();
216
217 // try to reconnect
218 }
219}
220
221} // namespace android::hardware::automotive::vehicle::virtualization