Home | History | Annotate | Download | only in hal
      1 /*
      2  * Copyright (C) 2016 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 package com.android.car.hal;
     17 
     18 import static com.android.car.CarServiceUtils.toByteArray;
     19 
     20 import static java.lang.Integer.toHexString;
     21 
     22 import android.car.VehicleAreaType;
     23 import android.car.hardware.CarPropertyConfig;
     24 import android.car.hardware.CarPropertyValue;
     25 import android.hardware.automotive.vehicle.V2_0.VehicleArea;
     26 import android.hardware.automotive.vehicle.V2_0.VehicleAreaConfig;
     27 import android.hardware.automotive.vehicle.V2_0.VehiclePropConfig;
     28 import android.hardware.automotive.vehicle.V2_0.VehiclePropValue;
     29 import android.hardware.automotive.vehicle.V2_0.VehiclePropertyType;
     30 
     31 import java.util.Collections;
     32 import java.util.List;
     33 
     34 /**
     35  * Utility functions to work with {@link CarPropertyConfig} and {@link CarPropertyValue}
     36  */
     37 /*package*/ final class CarPropertyUtils {
     38 
     39     /* Utility class has no public constructor */
     40     private CarPropertyUtils() {}
     41 
     42     /** Converts {@link VehiclePropValue} to {@link CarPropertyValue} */
     43     static CarPropertyValue<?> toCarPropertyValue(
     44             VehiclePropValue halValue, int propertyId) {
     45         Class<?> clazz = getJavaClass(halValue.prop & VehiclePropertyType.MASK);
     46         int areaId = halValue.areaId;
     47         int status = halValue.status;
     48         long timestamp = halValue.timestamp;
     49         VehiclePropValue.RawValue v = halValue.value;
     50 
     51         if (Boolean.class == clazz) {
     52             return new CarPropertyValue<>(propertyId, areaId, status, timestamp,
     53                                           v.int32Values.get(0) == 1);
     54         } else if (Boolean[].class == clazz) {
     55             Boolean[] values = new Boolean[v.int32Values.size()];
     56             for (int i = 0; i < values.length; i++) {
     57                 values[i] = v.int32Values.get(i) == 1;
     58             }
     59             return new CarPropertyValue<>(propertyId, areaId, status, timestamp, values);
     60         } else if (String.class == clazz) {
     61             return new CarPropertyValue<>(propertyId, areaId, status, timestamp, v.stringValue);
     62         } else if (byte[].class == clazz) {
     63             byte[] halData = toByteArray(v.bytes);
     64             return new CarPropertyValue<>(propertyId, areaId, status, timestamp, halData);
     65         } else if (Long[].class == clazz) {
     66             Long[] values = new Long[v.int64Values.size()];
     67             for (int i = 0; i < values.length; i++) {
     68                 values[i] = v.int64Values.get(i);
     69             }
     70             return new CarPropertyValue<>(propertyId, areaId, status, timestamp, values);
     71         } else /* All list properties */ {
     72             Object[] values = getRawValueList(clazz, v).toArray();
     73             return new CarPropertyValue<>(propertyId, areaId, status, timestamp,
     74                     values.length == 1 ? values[0] : values);
     75         }
     76     }
     77 
     78     /** Converts {@link CarPropertyValue} to {@link VehiclePropValue} */
     79     static VehiclePropValue toVehiclePropValue(CarPropertyValue carProp, int halPropId) {
     80         VehiclePropValue vehicleProp = new VehiclePropValue();
     81         vehicleProp.prop = halPropId;
     82         vehicleProp.areaId = carProp.getAreaId();
     83         VehiclePropValue.RawValue v = vehicleProp.value;
     84 
     85         Object o = carProp.getValue();
     86 
     87         if (o instanceof Boolean) {
     88             v.int32Values.add(((Boolean) o) ? 1 : 0);
     89         } else if (o instanceof Boolean[]) {
     90             for (Boolean b : (Boolean[]) o) {
     91                 v.int32Values.add(((Boolean) o) ? 1 : 0);
     92             }
     93         } else if (o instanceof Integer) {
     94             v.int32Values.add((Integer) o);
     95         } else if (o instanceof Integer[]) {
     96             Collections.addAll(v.int32Values, (Integer[]) o);
     97         } else if (o instanceof Float) {
     98             v.floatValues.add((Float) o);
     99         } else if (o instanceof Float[]) {
    100             Collections.addAll(v.floatValues, (Float[]) o);
    101         } else if (o instanceof Long) {
    102             v.int64Values.add((Long) o);
    103         } else if (o instanceof Long[]) {
    104             Collections.addAll(v.int64Values, (Long[]) o);
    105         } else if (o instanceof String) {
    106             v.stringValue = (String) o;
    107         } else if (o instanceof byte[]) {
    108             for (byte b : (byte[]) o) {
    109                 v.bytes.add(b);
    110             }
    111         } else {
    112             throw new IllegalArgumentException("Unexpected type in: " + carProp);
    113         }
    114 
    115         return vehicleProp;
    116     }
    117 
    118     /**
    119      * Converts {@link VehiclePropConfig} to {@link CarPropertyConfig}.
    120      */
    121     static CarPropertyConfig<?> toCarPropertyConfig(VehiclePropConfig p, int propertyId) {
    122         int areaType = getVehicleAreaType(p.prop & VehicleArea.MASK);
    123         // Create list of areaIds for this property
    124         int[] areas = new int[p.areaConfigs.size()];
    125         for (int i=0; i<p.areaConfigs.size(); i++) {
    126             areas[i] = p.areaConfigs.get(i).areaId;
    127         }
    128 
    129         Class<?> clazz = getJavaClass(p.prop & VehiclePropertyType.MASK);
    130         if (p.areaConfigs.isEmpty()) {
    131             return CarPropertyConfig
    132                     .newBuilder(clazz, propertyId, areaType, /* capacity */ 1)
    133                     .addAreas(areas)
    134                     .setAccess(p.access)
    135                     .setChangeMode(p.changeMode)
    136                     .setConfigArray(p.configArray)
    137                     .setConfigString(p.configString)
    138                     .setMaxSampleRate(p.maxSampleRate)
    139                     .setMinSampleRate(p.minSampleRate)
    140                     .build();
    141         } else {
    142             CarPropertyConfig.Builder builder = CarPropertyConfig
    143                     .newBuilder(clazz, propertyId, areaType, /* capacity */ p.areaConfigs.size())
    144                     .setAccess(p.access)
    145                     .setChangeMode(p.changeMode)
    146                     .setConfigArray(p.configArray)
    147                     .setConfigString(p.configString)
    148                     .setMaxSampleRate(p.maxSampleRate)
    149                     .setMinSampleRate(p.minSampleRate);
    150 
    151             for (VehicleAreaConfig area : p.areaConfigs) {
    152                 if (classMatched(Integer.class, clazz)) {
    153                     builder.addAreaConfig(area.areaId, area.minInt32Value, area.maxInt32Value);
    154                 } else if (classMatched(Float.class, clazz)) {
    155                     builder.addAreaConfig(area.areaId, area.minFloatValue, area.maxFloatValue);
    156                 } else if (classMatched(Long.class, clazz)) {
    157                     builder.addAreaConfig(area.areaId, area.minInt64Value, area.maxInt64Value);
    158                 } else if (classMatched(Boolean.class, clazz) ||
    159                            classMatched(Float[].class, clazz) ||
    160                            classMatched(Integer[].class, clazz) ||
    161                            classMatched(Long[].class, clazz) ||
    162                            classMatched(String.class, clazz) ||
    163                            classMatched(byte[].class, clazz) ||
    164                            classMatched(Object.class, clazz)) {
    165                     // These property types do not have min/max values
    166                     builder.addArea(area.areaId);
    167                 } else {
    168                     throw new IllegalArgumentException("Unexpected type: " + clazz);
    169                 }
    170             }
    171 
    172             return builder.build();
    173         }
    174     }
    175 
    176     private static @VehicleAreaType.VehicleAreaTypeValue int getVehicleAreaType(int halArea) {
    177         switch (halArea) {
    178             case VehicleArea.GLOBAL:
    179                 return VehicleAreaType.VEHICLE_AREA_TYPE_GLOBAL;
    180             case VehicleArea.SEAT:
    181                 return VehicleAreaType.VEHICLE_AREA_TYPE_SEAT;
    182             case VehicleArea.DOOR:
    183                 return VehicleAreaType.VEHICLE_AREA_TYPE_DOOR;
    184             case VehicleArea.WINDOW:
    185                 return VehicleAreaType.VEHICLE_AREA_TYPE_WINDOW;
    186             case VehicleArea.MIRROR:
    187                 return VehicleAreaType.VEHICLE_AREA_TYPE_MIRROR;
    188             case VehicleArea.WHEEL:
    189                 return VehicleAreaType.VEHICLE_AREA_TYPE_WHEEL;
    190             default:
    191                 throw new RuntimeException("Unsupported area type " + halArea);
    192         }
    193     }
    194 
    195     private static Class<?> getJavaClass(int halType) {
    196         switch (halType) {
    197             case VehiclePropertyType.BOOLEAN:
    198                 return Boolean.class;
    199             case VehiclePropertyType.FLOAT:
    200                 return Float.class;
    201             case VehiclePropertyType.INT32:
    202                 return Integer.class;
    203             case VehiclePropertyType.INT64:
    204                 return Long.class;
    205             case VehiclePropertyType.FLOAT_VEC:
    206                 return Float[].class;
    207             case VehiclePropertyType.INT32_VEC:
    208                 return Integer[].class;
    209             case VehiclePropertyType.INT64_VEC:
    210                 return Long[].class;
    211             case VehiclePropertyType.STRING:
    212                 return String.class;
    213             case VehiclePropertyType.BYTES:
    214                 return byte[].class;
    215             case VehiclePropertyType.MIXED:
    216                 return Object.class;
    217             default:
    218                 throw new IllegalArgumentException("Unexpected type: " + toHexString(halType));
    219         }
    220     }
    221 
    222     private static List getRawValueList(Class<?> clazz, VehiclePropValue.RawValue value) {
    223         if (classMatched(Float.class, clazz) || classMatched(Float[].class, clazz)) {
    224             return value.floatValues;
    225         } else if (classMatched(Integer.class, clazz) || classMatched(Integer[].class, clazz)) {
    226             return value.int32Values;
    227         } else if (classMatched(Long.class, clazz) || classMatched(Long[].class, clazz)) {
    228             return value.int64Values;
    229         } else {
    230             throw new IllegalArgumentException("Unexpected type: " + clazz);
    231         }
    232     }
    233 
    234     private static boolean classMatched(Class<?> class1, Class<?> class2) {
    235         return class1 == class2 || class1.getComponentType() == class2;
    236     }
    237 }
    238