blob: 6c55700451a678b26b58b88b3efb6c41c3b62146 [file] [log] [blame]
keunyounge18e25d2015-08-28 15:57:19 -07001/*
2 * Copyright (C) 2015 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#define LOG_TAG "VehicleNetwork"
18
19#include <memory>
20
21#include <binder/IPCThreadState.h>
22#include <private/android_filesystem_config.h>
23
24#include <utils/Log.h>
25
26#include <IVehicleNetwork.h>
27#include <VehicleNetworkProto.pb.h>
28
keunyoung15882e52015-09-16 16:57:58 -070029#include "BinderUtil.h"
keunyounge18e25d2015-08-28 15:57:19 -070030#include "VehicleNetworkProtoUtil.h"
31
32namespace android {
33
34enum {
35 LIST_PROPERTIES = IBinder::FIRST_CALL_TRANSACTION,
36 SET_PROPERTY,
37 GET_PROPERTY,
38 SUBSCRIBE,
39 UNSUBSCRIBE,
keunyoung1ab8e182015-09-24 09:25:22 -070040 INJECT_EVENT,
41 START_MOCKING,
42 STOP_MOCKING,
Keun-young Park28dd4702015-11-19 18:06:04 -080043 INJECT_HAL_ERROR,
44 START_ERROR_LISTENING,
45 STOP_ERROR_LISTENING,
46 START_HAL_RESTART_MONITORING,
47 STOP_HAL_RESTART_MONITORING
keunyounge18e25d2015-08-28 15:57:19 -070048};
49
50// ----------------------------------------------------------------------------
51
keunyoung15882e52015-09-16 16:57:58 -070052const char IVehicleNetwork::SERVICE_NAME[] = "com.android.car.vehiclenetwork.IVehicleNetwork";
keunyounge18e25d2015-08-28 15:57:19 -070053
54// ----------------------------------------------------------------------------
55
56class BpVehicleNetwork : public BpInterface<IVehicleNetwork> {
57public:
58 BpVehicleNetwork(const sp<IBinder> & impl)
59 : BpInterface<IVehicleNetwork>(impl) {
60 }
61
62 virtual sp<VehiclePropertiesHolder> listProperties(int32_t property) {
63 sp<VehiclePropertiesHolder> holder;
64 Parcel data, reply;
65 data.writeInterfaceToken(IVehicleNetwork::getInterfaceDescriptor());
66 data.writeInt32(property);
67 status_t status = remote()->transact(LIST_PROPERTIES, data, &reply);
68 if (status == NO_ERROR) {
keunyoung15882e52015-09-16 16:57:58 -070069 reply.readExceptionCode(); // for compatibility with java
keunyounge18e25d2015-08-28 15:57:19 -070070 if (reply.readInt32() == 0) { // no result
71 return holder;
72 }
73 ReadableBlobHolder blob(new Parcel::ReadableBlob());
74 if (blob.blob == NULL) {
75 ALOGE("listProperties, no memory");
76 return holder;
77 }
78 int32_t size = reply.readInt32();
79 status = reply.readBlob(size, blob.blob);
80 if (status != NO_ERROR) {
81 ALOGE("listProperties, cannot read blob %d", status);
82 return holder;
83 }
84 //TODO make this more memory efficient
85 std::unique_ptr<VehiclePropConfigs> configs(new VehiclePropConfigs());
86 if (configs.get() == NULL) {
87 return holder;
88 }
89 if(!configs->ParseFromArray(blob.blob->data(), size)) {
90 ALOGE("listProperties, cannot parse reply");
91 return holder;
92 }
keunyoungd32f4e62015-09-21 11:33:06 -070093 holder = new VehiclePropertiesHolder();
94 ASSERT_OR_HANDLE_NO_MEMORY(holder.get(), return);
95 status = VehicleNetworkProtoUtil::fromVehiclePropConfigs(*configs.get(),
96 holder->getList());
keunyounge18e25d2015-08-28 15:57:19 -070097 if (status != NO_ERROR) {
98 ALOGE("listProperties, cannot convert VehiclePropConfigs %d", status);
99 return holder;
100 }
keunyoungd32f4e62015-09-21 11:33:06 -0700101
keunyounge18e25d2015-08-28 15:57:19 -0700102 }
103 return holder;
104 }
105
106 virtual status_t setProperty(const vehicle_prop_value_t& value) {
107 Parcel data, reply;
108 data.writeInterfaceToken(IVehicleNetwork::getInterfaceDescriptor());
keunyoung7d74e6d2015-10-14 15:43:10 -0700109 status_t status = VehiclePropValueBinderUtil::writeToParcel(data, value);
110 if (status != NO_ERROR) {
111 return status;
112 }
113 status = remote()->transact(SET_PROPERTY, data, &reply);
keunyounge18e25d2015-08-28 15:57:19 -0700114 return status;
115 }
116
117 virtual status_t getProperty(vehicle_prop_value_t* value) {
118 Parcel data, reply;
keunyoung7d74e6d2015-10-14 15:43:10 -0700119 if (value == NULL) {
120 return BAD_VALUE;
121 }
keunyounge18e25d2015-08-28 15:57:19 -0700122 data.writeInterfaceToken(IVehicleNetwork::getInterfaceDescriptor());
keunyoung7d74e6d2015-10-14 15:43:10 -0700123 status_t status = VehiclePropValueBinderUtil::writeToParcel(data, *value);
124 if (status != NO_ERROR) {
125 ALOGE("getProperty, cannot write");
126 return status;
127 }
128 status = remote()->transact(GET_PROPERTY, data, &reply);
keunyounge18e25d2015-08-28 15:57:19 -0700129 if (status == NO_ERROR) {
keunyoung15882e52015-09-16 16:57:58 -0700130 reply.readExceptionCode(); // for compatibility with java
keunyoung7d74e6d2015-10-14 15:43:10 -0700131 status = VehiclePropValueBinderUtil::readFromParcel(reply, value);
keunyounge18e25d2015-08-28 15:57:19 -0700132 }
133 return status;
134 }
135
136 virtual status_t subscribe(const sp<IVehicleNetworkListener> &listener, int32_t property,
Keun-young Park0727f952015-12-21 14:30:07 -0800137 float sampleRate, int32_t zones) {
keunyounge18e25d2015-08-28 15:57:19 -0700138 Parcel data, reply;
139 data.writeInterfaceToken(IVehicleNetwork::getInterfaceDescriptor());
140 data.writeStrongBinder(IInterface::asBinder(listener));
141 data.writeInt32(property);
142 data.writeFloat(sampleRate);
Keun-young Park0727f952015-12-21 14:30:07 -0800143 data.writeInt32(zones);
keunyounge18e25d2015-08-28 15:57:19 -0700144 status_t status = remote()->transact(SUBSCRIBE, data, &reply);
145 return status;
146 }
147
148 virtual void unsubscribe(const sp<IVehicleNetworkListener> &listener, int32_t property) {
149 Parcel data, reply;
150 data.writeInterfaceToken(IVehicleNetwork::getInterfaceDescriptor());
151 data.writeStrongBinder(IInterface::asBinder(listener));
152 data.writeInt32(property);
153 status_t status = remote()->transact(UNSUBSCRIBE, data, &reply);
154 if (status != NO_ERROR) {
155 ALOGI("unsubscribing property %d failed %d", property, status);
156 }
157 }
keunyoung1ab8e182015-09-24 09:25:22 -0700158
159 virtual status_t injectEvent(const vehicle_prop_value_t& value) {
160 Parcel data, reply;
161 data.writeInterfaceToken(IVehicleNetwork::getInterfaceDescriptor());
162 data.writeInt32(1); // 0 means no value. For compatibility with aidl based code.
163 std::unique_ptr<VehiclePropValue> v(new VehiclePropValue());
164 ASSERT_OR_HANDLE_NO_MEMORY(v.get(), return NO_MEMORY);
165 VehicleNetworkProtoUtil::toVehiclePropValue(value, *v.get());
166 int size = v->ByteSize();
167 WritableBlobHolder blob(new Parcel::WritableBlob());
168 ASSERT_OR_HANDLE_NO_MEMORY(blob.blob, return NO_MEMORY);
169 data.writeInt32(size);
170 data.writeBlob(size, false, blob.blob);
171 v->SerializeToArray(blob.blob->data(), size);
172 status_t status = remote()->transact(INJECT_EVENT, data, &reply);
173 return status;
174 }
175
176 virtual status_t startMocking(const sp<IVehicleNetworkHalMock>& mock) {
177 Parcel data, reply;
178 data.writeInterfaceToken(IVehicleNetwork::getInterfaceDescriptor());
179 data.writeStrongBinder(IInterface::asBinder(mock));
180 status_t status = remote()->transact(START_MOCKING, data, &reply);
181 return status;
182 }
183
184 virtual void stopMocking(const sp<IVehicleNetworkHalMock>& mock) {
185 Parcel data, reply;
186 data.writeInterfaceToken(IVehicleNetwork::getInterfaceDescriptor());
187 data.writeStrongBinder(IInterface::asBinder(mock));
188 status_t status = remote()->transact(STOP_MOCKING, data, &reply);
189 if (status != NO_ERROR) {
190 ALOGI("stop mocking failed %d", status);
191 }
192 }
Keun-young Park28dd4702015-11-19 18:06:04 -0800193
194 status_t injectHalError(int32_t errorCode, int32_t property, int32_t operation) {
195 Parcel data, reply;
196 data.writeInterfaceToken(IVehicleNetwork::getInterfaceDescriptor());
197 data.writeInt32(errorCode);
198 data.writeInt32(property);
199 data.writeInt32(operation);
200 status_t status = remote()->transact(INJECT_HAL_ERROR, data, &reply);
201 return status;
202 }
203
204 virtual status_t startErrorListening(const sp<IVehicleNetworkListener> &listener) {
205 Parcel data, reply;
206 data.writeInterfaceToken(IVehicleNetwork::getInterfaceDescriptor());
207 data.writeStrongBinder(IInterface::asBinder(listener));
208 status_t status = remote()->transact(START_ERROR_LISTENING, data, &reply);
209 return status;
210 }
211
212 virtual void stopErrorListening(const sp<IVehicleNetworkListener> &listener) {
213 Parcel data, reply;
214 data.writeInterfaceToken(IVehicleNetwork::getInterfaceDescriptor());
215 data.writeStrongBinder(IInterface::asBinder(listener));
216 status_t status = remote()->transact(STOP_ERROR_LISTENING, data, &reply);
217 if (status != NO_ERROR) {
218 ALOGI("stopErrorListening %d", status);
219 }
220 }
221
222 virtual status_t startHalRestartMonitoring(const sp<IVehicleNetworkListener> &listener) {
223 Parcel data, reply;
224 data.writeInterfaceToken(IVehicleNetwork::getInterfaceDescriptor());
225 data.writeStrongBinder(IInterface::asBinder(listener));
226 status_t status = remote()->transact(START_HAL_RESTART_MONITORING, data, &reply);
227 return status;
228 }
229
230 virtual void stopHalRestartMonitoring(const sp<IVehicleNetworkListener> &listener) {
231 Parcel data, reply;
232 data.writeInterfaceToken(IVehicleNetwork::getInterfaceDescriptor());
233 data.writeStrongBinder(IInterface::asBinder(listener));
234 status_t status = remote()->transact(STOP_HAL_RESTART_MONITORING, data, &reply);
235 if (status != NO_ERROR) {
236 ALOGI("stopHalRestartMonitoring %d", status);
237 }
238 }
keunyounge18e25d2015-08-28 15:57:19 -0700239};
240
241IMPLEMENT_META_INTERFACE(VehicleNetwork, IVehicleNetwork::SERVICE_NAME);
242
243// ----------------------------------------------------------------------
244
245static bool isSystemUser() {
246 uid_t uid = IPCThreadState::self()->getCallingUid();
247 switch (uid) {
248 // This list will be expanded. Only those UIDs are allowed to access vehicle network
249 // for now. There can be per property based UID check built-in as well.
250 case AID_ROOT:
251 case AID_SYSTEM:
Keun-young Parkaaeffaf2015-11-25 17:24:10 -0800252 case AID_AUDIO:
253 case AID_MEDIA: {
keunyounge18e25d2015-08-28 15:57:19 -0700254 return true;
255 } break;
keunyoung15882e52015-09-16 16:57:58 -0700256 default: {
257 ALOGE("non-system user tried access, uid %d", uid);
258 } break;
keunyounge18e25d2015-08-28 15:57:19 -0700259 }
260 return false;
261}
262
263status_t BnVehicleNetwork::onTransact(uint32_t code, const Parcel& data, Parcel* reply,
264 uint32_t flags) {
265 if (!isSystemUser()) {
266 return PERMISSION_DENIED;
267 }
268 status_t r;
269 switch (code) {
270 case LIST_PROPERTIES: {
271 CHECK_INTERFACE(IVehicleNetwork, data, reply);
272 int32_t property = data.readInt32();
273 sp<VehiclePropertiesHolder> holder = listProperties(property);
274 if (holder.get() == NULL) { // given property not found
keunyoung15882e52015-09-16 16:57:58 -0700275 BinderUtil::fillObjectResultReply(reply, false /* isValid */);
276 return NO_ERROR;
keunyounge18e25d2015-08-28 15:57:19 -0700277 }
278 std::unique_ptr<VehiclePropConfigs> configs(new VehiclePropConfigs());
279 ASSERT_OR_HANDLE_NO_MEMORY(configs.get(), return NO_MEMORY);
keunyoungd32f4e62015-09-21 11:33:06 -0700280 VehicleNetworkProtoUtil::toVehiclePropConfigs(holder->getList(), *configs.get());
keunyounge18e25d2015-08-28 15:57:19 -0700281 int size = configs->ByteSize();
282 WritableBlobHolder blob(new Parcel::WritableBlob());
283 ASSERT_OR_HANDLE_NO_MEMORY(blob.blob, return NO_MEMORY);
keunyoung15882e52015-09-16 16:57:58 -0700284 BinderUtil::fillObjectResultReply(reply, true);
keunyounge18e25d2015-08-28 15:57:19 -0700285 reply->writeInt32(size);
286 reply->writeBlob(size, false, blob.blob);
287 configs->SerializeToArray(blob.blob->data(), size);
288 return NO_ERROR;
289 } break;
290 case SET_PROPERTY: {
291 CHECK_INTERFACE(IVehicleNetwork, data, reply);
keunyounge18e25d2015-08-28 15:57:19 -0700292 ScopedVehiclePropValue value;
keunyoung7d74e6d2015-10-14 15:43:10 -0700293 r = VehiclePropValueBinderUtil::readFromParcel(data, &value.value,
294 false /* deleteMembers */);
keunyounge18e25d2015-08-28 15:57:19 -0700295 if (r != NO_ERROR) {
keunyounge18e25d2015-08-28 15:57:19 -0700296 return r;
297 }
keunyounge18e25d2015-08-28 15:57:19 -0700298 r = setProperty(value.value);
keunyoung15882e52015-09-16 16:57:58 -0700299 BinderUtil::fillNoResultReply(reply);
keunyounge18e25d2015-08-28 15:57:19 -0700300 return r;
301 } break;
302 case GET_PROPERTY: {
303 CHECK_INTERFACE(IVehicleNetwork, data, reply);
304 ScopedVehiclePropValue value;
keunyoung7d74e6d2015-10-14 15:43:10 -0700305 r = VehiclePropValueBinderUtil::readFromParcel(data, &value.value,
306 false /* deleteMembers */, true /*canIgnoreNoData*/);
307 if (r != NO_ERROR) {
308 ALOGE("getProperty cannot read %d", r);
309 return r;
310 }
keunyounge18e25d2015-08-28 15:57:19 -0700311 r = getProperty(&(value.value));
312 if (r == NO_ERROR) {
keunyoung7d74e6d2015-10-14 15:43:10 -0700313 reply->writeNoException();
314 r = VehiclePropValueBinderUtil::writeToParcel(*reply, value.value);
keunyounge18e25d2015-08-28 15:57:19 -0700315 }
316 return r;
317 } break;
318 case SUBSCRIBE: {
319 CHECK_INTERFACE(IVehicleNetwork, data, reply);
320 sp<IVehicleNetworkListener> listener =
321 interface_cast<IVehicleNetworkListener>(data.readStrongBinder());
322 int32_t property = data.readInt32();
323 float sampleRate = data.readFloat();
Keun-young Park0727f952015-12-21 14:30:07 -0800324 int32_t zones = data.readInt32();
325 r = subscribe(listener, property, sampleRate, zones);
keunyoung15882e52015-09-16 16:57:58 -0700326 BinderUtil::fillNoResultReply(reply);
keunyounge18e25d2015-08-28 15:57:19 -0700327 return r;
328 } break;
329 case UNSUBSCRIBE: {
330 CHECK_INTERFACE(IVehicleNetwork, data, reply);
331 sp<IVehicleNetworkListener> listener =
332 interface_cast<IVehicleNetworkListener>(data.readStrongBinder());
333 int32_t property = data.readInt32();
334 unsubscribe(listener, property);
keunyoung15882e52015-09-16 16:57:58 -0700335 BinderUtil::fillNoResultReply(reply);
keunyounge18e25d2015-08-28 15:57:19 -0700336 return NO_ERROR;
337 } break;
keunyoung1ab8e182015-09-24 09:25:22 -0700338 case INJECT_EVENT: {
339 CHECK_INTERFACE(IVehicleNetwork, data, reply);
340 if (data.readInt32() == 0) { // java side allows passing null with this.
341 return BAD_VALUE;
342 }
343 ScopedVehiclePropValue value;
344 ReadableBlobHolder blob(new Parcel::ReadableBlob());
345 ASSERT_OR_HANDLE_NO_MEMORY(blob.blob, return NO_MEMORY);
346 int32_t size = data.readInt32();
347 r = data.readBlob(size, blob.blob);
348 if (r != NO_ERROR) {
349 ALOGE("injectEvent:service, cannot read blob");
350 return r;
351 }
352 std::unique_ptr<VehiclePropValue> v(new VehiclePropValue());
353 ASSERT_OR_HANDLE_NO_MEMORY(v.get(), return NO_MEMORY);
354 if (!v->ParseFromArray(blob.blob->data(), size)) {
355 ALOGE("injectEvent:service, cannot parse data");
356 return BAD_VALUE;
357 }
358 r = VehicleNetworkProtoUtil::fromVehiclePropValue(*v.get(), value.value);
359 if (r != NO_ERROR) {
360 ALOGE("injectEvent:service, cannot convert data");
361 return BAD_VALUE;
362 }
363 r = injectEvent(value.value);
364 BinderUtil::fillNoResultReply(reply);
365 return r;
366 } break;
367 case START_MOCKING: {
368 CHECK_INTERFACE(IVehicleNetwork, data, reply);
369 sp<IVehicleNetworkHalMock> mock =
370 interface_cast<IVehicleNetworkHalMock>(data.readStrongBinder());
371 r = startMocking(mock);
372 BinderUtil::fillNoResultReply(reply);
373 return r;
374 } break;
375 case STOP_MOCKING: {
376 CHECK_INTERFACE(IVehicleNetwork, data, reply);
377 sp<IVehicleNetworkHalMock> mock =
378 interface_cast<IVehicleNetworkHalMock>(data.readStrongBinder());
379 stopMocking(mock);
380 BinderUtil::fillNoResultReply(reply);
381 return NO_ERROR;
382 } break;
Keun-young Park28dd4702015-11-19 18:06:04 -0800383 case INJECT_HAL_ERROR: {
384 CHECK_INTERFACE(IVehicleNetwork, data, reply);
385 int32_t errorCode = data.readInt32();
386 int32_t property = data.readInt32();
387 int32_t operation = data.readInt32();
388 r = injectHalError(errorCode, property, operation);
389 BinderUtil::fillNoResultReply(reply);
390 return r;
391 } break;
392 case START_ERROR_LISTENING: {
393 CHECK_INTERFACE(IVehicleNetwork, data, reply);
394 sp<IVehicleNetworkListener> listener =
395 interface_cast<IVehicleNetworkListener>(data.readStrongBinder());
396 r = startErrorListening(listener);
397 BinderUtil::fillNoResultReply(reply);
398 return r;
399 } break;
400 case STOP_ERROR_LISTENING: {
401 CHECK_INTERFACE(IVehicleNetwork, data, reply);
402 sp<IVehicleNetworkListener> listener =
403 interface_cast<IVehicleNetworkListener>(data.readStrongBinder());
404 stopErrorListening(listener);
405 BinderUtil::fillNoResultReply(reply);
406 return NO_ERROR;
407 } break;
408 case START_HAL_RESTART_MONITORING: {
409 CHECK_INTERFACE(IVehicleNetwork, data, reply);
410 sp<IVehicleNetworkListener> listener =
411 interface_cast<IVehicleNetworkListener>(data.readStrongBinder());
412 r = startHalRestartMonitoring(listener);
413 BinderUtil::fillNoResultReply(reply);
414 return r;
415 } break;
416 case STOP_HAL_RESTART_MONITORING: {
417 CHECK_INTERFACE(IVehicleNetwork, data, reply);
418 sp<IVehicleNetworkListener> listener =
419 interface_cast<IVehicleNetworkListener>(data.readStrongBinder());
420 stopHalRestartMonitoring(listener);
421 BinderUtil::fillNoResultReply(reply);
422 return NO_ERROR;
423 } break;
keunyounge18e25d2015-08-28 15:57:19 -0700424 default:
425 return BBinder::onTransact(code, data, reply, flags);
426 }
427}
428
429}; // namespace android