Home | History | Annotate | Download | only in vehicle
      1 /*
      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 "vehicle_hw_default"
     18 #define LOG_NDEBUG 1
     19 #define RADIO_PRESET_NUM 6
     20 
     21 #define UNUSED __attribute__((__unused__))
     22 
     23 #include <errno.h>
     24 #include <inttypes.h>
     25 #include <malloc.h>
     26 #include <pthread.h>
     27 #include <stdint.h>
     28 #include <stdlib.h>
     29 #include <string.h>
     30 #include <sys/prctl.h>
     31 #include <sys/time.h>
     32 #include <time.h>
     33 
     34 #include <cutils/log.h>
     35 #include <system/radio.h>
     36 #include <hardware/hardware.h>
     37 #include <hardware/vehicle.h>
     38 
     39 extern int64_t elapsedRealtimeNano();
     40 
     41 static char VEHICLE_MAKE[] = "android_car";
     42 
     43 typedef struct vehicle_device_impl {
     44     vehicle_hw_device_t vehicle_device;
     45     uint32_t initialized_;
     46     vehicle_event_callback_fn event_fn_;
     47     vehicle_error_callback_fn error_fn_;
     48 } vehicle_device_impl_t ;
     49 
     50 static pthread_mutex_t lock_;
     51 
     52 typedef struct subscription {
     53     // Each subscription has it's own thread.
     54     pthread_t thread_id;
     55     int32_t prop;
     56     float sample_rate;
     57     pthread_mutex_t lock;
     58     // This field should be protected by the above mutex.
     59     // TODO change this to something better as flag alone takes long time to finish.
     60     uint32_t stop_thread;
     61     vehicle_device_impl_t* impl;
     62     pthread_t thread;
     63     pthread_cond_t cond;
     64     char name[100];
     65 } subscription_t;
     66 
     67 static vehicle_prop_config_t CONFIGS[] = {
     68     {
     69         .prop = VEHICLE_PROPERTY_INFO_MAKE,
     70         .access = VEHICLE_PROP_ACCESS_READ,
     71         .change_mode = VEHICLE_PROP_CHANGE_MODE_STATIC,
     72         .value_type = VEHICLE_VALUE_TYPE_STRING,
     73         .min_sample_rate = 0,
     74         .max_sample_rate = 0,
     75         .hal_data = NULL,
     76     },
     77     {
     78         .prop = VEHICLE_PROPERTY_GEAR_SELECTION,
     79         .access = VEHICLE_PROP_ACCESS_READ,
     80         .change_mode = VEHICLE_PROP_CHANGE_MODE_ON_CHANGE,
     81         .value_type = VEHICLE_VALUE_TYPE_INT32,
     82         .min_sample_rate = 0,
     83         .max_sample_rate = 0,
     84         .hal_data = NULL,
     85     },
     86     {
     87         .prop = VEHICLE_PROPERTY_DRIVING_STATUS,
     88         .access = VEHICLE_PROP_ACCESS_READ,
     89         .change_mode = VEHICLE_PROP_CHANGE_MODE_ON_CHANGE,
     90         .value_type = VEHICLE_VALUE_TYPE_INT32,
     91         .min_sample_rate = 0,
     92         .max_sample_rate = 0,
     93         .hal_data = NULL,
     94     },
     95     {
     96         .prop = VEHICLE_PROPERTY_PARKING_BRAKE_ON,
     97         .access = VEHICLE_PROP_ACCESS_READ,
     98         .change_mode = VEHICLE_PROP_CHANGE_MODE_ON_CHANGE,
     99         .value_type = VEHICLE_VALUE_TYPE_BOOLEAN,
    100         .min_sample_rate = 0,
    101         .max_sample_rate = 0,
    102         .hal_data = NULL,
    103     },
    104     {
    105         .prop = VEHICLE_PROPERTY_PERF_VEHICLE_SPEED,
    106         .access = VEHICLE_PROP_ACCESS_READ,
    107         .change_mode = VEHICLE_PROP_CHANGE_MODE_CONTINUOUS,
    108         .value_type = VEHICLE_VALUE_TYPE_FLOAT,
    109         .min_sample_rate = 0.1,
    110         .max_sample_rate = 10.0,
    111         .hal_data = NULL,
    112     },
    113     {
    114         .prop = VEHICLE_PROPERTY_RADIO_PRESET,
    115         .access = VEHICLE_PROP_ACCESS_READ_WRITE,
    116         .change_mode = VEHICLE_PROP_CHANGE_MODE_ON_CHANGE,
    117         .value_type = VEHICLE_VALUE_TYPE_INT32_VEC4,
    118         .vehicle_radio_num_presets = RADIO_PRESET_NUM,
    119         .min_sample_rate = 0,
    120         .max_sample_rate = 0,
    121         .hal_data = NULL,
    122     },
    123 };
    124 
    125 vehicle_prop_config_t* find_config(int prop) {
    126     unsigned int i;
    127     for (i = 0; i < sizeof(CONFIGS) / sizeof(vehicle_prop_config_t); i++) {
    128         if (CONFIGS[i].prop == prop) {
    129             return &CONFIGS[i];
    130         }
    131     }
    132     return NULL;
    133 }
    134 
    135 static int alloc_vehicle_str_from_cstr(const char* string, vehicle_str_t* vehicle_str) {
    136     int len = strlen(string);
    137     vehicle_str->data = (uint8_t*) malloc(len);
    138     if (vehicle_str->data == NULL) {
    139         return -ENOMEM;
    140     }
    141     memcpy(vehicle_str->data, string, len);
    142     vehicle_str->len = len;
    143     return 0;
    144 }
    145 
    146 static vehicle_prop_config_t const * vdev_list_properties(vehicle_hw_device_t* device UNUSED,
    147         int* num_properties) {
    148     ALOGD("vdev_list_properties.");
    149 
    150     *num_properties = sizeof(CONFIGS) / sizeof(vehicle_prop_config_t);
    151     return CONFIGS;
    152 }
    153 
    154 static int vdev_init(vehicle_hw_device_t* device,
    155                      vehicle_event_callback_fn event_callback_fn,
    156                      vehicle_error_callback_fn error_callback_fn) {
    157     ALOGD("vdev_init.");
    158     vehicle_device_impl_t* impl = (vehicle_device_impl_t*)device;
    159     pthread_mutex_lock(&lock_);
    160     if (impl->initialized_) {
    161         ALOGE("vdev_init: Callback and Error functions are already existing.");
    162         pthread_mutex_unlock(&lock_);
    163         return -EEXIST;
    164     }
    165 
    166     impl->initialized_ = 1;
    167     impl->event_fn_ = event_callback_fn;
    168     impl->error_fn_ = error_callback_fn;
    169     pthread_mutex_unlock(&lock_);
    170     return 0;
    171 }
    172 
    173 static int vdev_release(vehicle_hw_device_t* device) {
    174     vehicle_device_impl_t* impl = (vehicle_device_impl_t*)device;
    175     pthread_mutex_lock(&lock_);
    176     if (!impl->initialized_) {
    177         ALOGD("vdev_release: Already released before, returning early.");
    178     } else {
    179         // unsubscribe_all()
    180         impl->initialized_ = 0;
    181     }
    182     pthread_mutex_unlock(&lock_);
    183     return 0;
    184 }
    185 
    186 static int vdev_get(vehicle_hw_device_t* device UNUSED, vehicle_prop_value_t* data) {
    187     ALOGD("vdev_get.");
    188     //TODO all data supporting read should support get
    189     if (!data) {
    190         ALOGE("vdev_get: Data cannot be null.");
    191         return -EINVAL;
    192     }
    193     vehicle_prop_config_t* config = find_config(data->prop);
    194     if (config == NULL) {
    195         ALOGE("vdev_get: cannot find config 0x%x", data->prop);
    196         return -EINVAL;
    197     }
    198     data->value_type = config->value_type;
    199     // for STATIC type, time can be just 0 instead
    200     data->timestamp = elapsedRealtimeNano();
    201     int r;
    202     switch (data->prop) {
    203         case VEHICLE_PROPERTY_INFO_MAKE:
    204             r = alloc_vehicle_str_from_cstr(VEHICLE_MAKE, &(data->value.str_value));
    205             if (r != 0) {
    206                 ALOGE("vdev_get: alloc failed");
    207                 return r;
    208             }
    209             break;
    210 
    211         case VEHICLE_PROPERTY_RADIO_PRESET: {
    212               int radio_preset = data->value.int32_array[0];
    213               if (radio_preset < VEHICLE_RADIO_PRESET_MIN_VALUE ||
    214                   radio_preset >= RADIO_PRESET_NUM) {
    215                   ALOGE("%s Invalid radio preset: %d\n", __func__, radio_preset);
    216                   return -1;
    217               }
    218               ALOGD("%s Radio Preset number: %d", __func__, radio_preset);
    219               int32_t selector = radio_preset % 2 == 0;
    220               // Populate the channel and subchannel to be some variation of the
    221               // preset number for mocking.
    222 
    223               // Restore the preset number.
    224               data->value.int32_array[0] = radio_preset;
    225               // Channel type values taken from
    226               // system/core/include/system/radio.h
    227               data->value.int32_array[1] = selector ? RADIO_BAND_FM : RADIO_BAND_AM;
    228               // For FM set a value in Mhz and for AM set a value in Khz range
    229               // (channel).
    230               data->value.int32_array[2] = selector ? 99000000 : 100000;
    231               // For FM we have a sub-channel and we care about it, for AM pass
    232               // a dummy value.
    233               data->value.int32_array[3] = selector ? radio_preset : -1;
    234               break;
    235         }
    236 
    237         default:
    238             // actual implementation will be much complex than this. It should track proper last
    239             // state. Here just fill with zero.
    240             memset(&(data->value), 0, sizeof(data->value));
    241             break;
    242     }
    243     ALOGI("vdev_get, type 0x%x, time %" PRId64 ", value_type %d", data->prop, data->timestamp,
    244             data->value_type);
    245     return 0;
    246 }
    247 
    248 static void vdev_release_memory_from_get(struct vehicle_hw_device* device UNUSED,
    249         vehicle_prop_value_t *data) {
    250     switch (data->value_type) {
    251         case VEHICLE_VALUE_TYPE_STRING:
    252         case VEHICLE_VALUE_TYPE_BYTES:
    253             free(data->value.str_value.data);
    254             data->value.str_value.data = NULL;
    255             break;
    256         default:
    257             ALOGW("release_memory_from_get for property 0x%x which is not string or bytes type 0x%x"
    258                     , data->prop, data->value_type);
    259             break;
    260     }
    261 }
    262 
    263 static int vdev_set(vehicle_hw_device_t* device UNUSED, const vehicle_prop_value_t* data) {
    264     ALOGD("vdev_set.");
    265     // Just print what data will be setting here.
    266     ALOGD("Setting property %d with value type %d\n", data->prop, data->value_type);
    267     vehicle_prop_config_t* config = find_config(data->prop);
    268     if (config == NULL) {
    269         ALOGE("vdev_set: cannot find config 0x%x", data->prop);
    270         return -EINVAL;
    271     }
    272     if (config->value_type != data->value_type) {
    273         ALOGE("vdev_set: type mismatch, passed 0x%x expecting 0x%x", data->value_type,
    274                 config->value_type);
    275         return -EINVAL;
    276     }
    277     switch (data->value_type) {
    278         case VEHICLE_VALUE_TYPE_FLOAT:
    279             ALOGD("Value type: FLOAT\nValue: %f\n", data->value.float_value);
    280             break;
    281         case VEHICLE_VALUE_TYPE_INT32:
    282             ALOGD("Value type: INT32\nValue: %d\n", data->value.int32_value);
    283             break;
    284         case VEHICLE_VALUE_TYPE_INT64:
    285             ALOGD("Value type: INT64\nValue: %lld\n", data->value.int64_value);
    286             break;
    287         case VEHICLE_VALUE_TYPE_BOOLEAN:
    288             ALOGD("Value type: BOOLEAN\nValue: %d\n", data->value.boolean_value);
    289             break;
    290         case VEHICLE_VALUE_TYPE_STRING:
    291             ALOGD("Value type: STRING\n Size: %d\n", data->value.str_value.len);
    292             // NOTE: We only handle ASCII strings here.
    293             // Print the UTF-8 string.
    294             char *ascii_out = (char *) malloc ((data->value.str_value.len + 1) * sizeof (char));
    295             memcpy(ascii_out, data->value.str_value.data, data->value.str_value.len);
    296             ascii_out[data->value.str_value.len] = '\0';
    297             ALOGD("Value: %s\n", ascii_out);
    298             break;
    299         case VEHICLE_VALUE_TYPE_INT32_VEC4:
    300             ALOGD("Value type: INT32_VEC4\nValue[0]: %d Value[1] %d Value[2] %d Value[3] %d",
    301                   data->value.int32_array[0], data->value.int32_array[1],
    302                   data->value.int32_array[2], data->value.int32_array[3]);
    303             break;
    304         default:
    305             ALOGD("Value type not yet handled: %d.\n", data->value_type);
    306     }
    307     return 0;
    308 }
    309 
    310 void print_subscribe_info(vehicle_device_impl_t* impl UNUSED) {
    311     unsigned int i;
    312     for (i = 0; i < sizeof(CONFIGS) / sizeof(vehicle_prop_config_t); i++) {
    313         subscription_t* sub = (subscription_t*)CONFIGS[i].hal_data;
    314         if (sub != NULL) {
    315             ALOGD("prop: %d rate: %f", sub->prop, sub->sample_rate);
    316         }
    317     }
    318 }
    319 
    320 // This should be run in a separate thread always.
    321 void fake_event_thread(struct subscription *sub) {
    322     if (!sub) {
    323         ALOGE("oops! subscription object cannot be NULL.");
    324         exit(-1);
    325     }
    326     prctl(PR_SET_NAME, (unsigned long)sub->name, 0, 0, 0);
    327     // Emit values in a loop, every 2 seconds.
    328     while (1) {
    329         // Create a random value depending on the property type.
    330         vehicle_prop_value_t event;
    331         event.prop = sub->prop;
    332         event.timestamp = elapsedRealtimeNano();
    333         switch (sub->prop) {
    334             case VEHICLE_PROPERTY_DRIVING_STATUS:
    335                 event.value_type = VEHICLE_VALUE_TYPE_INT32;
    336                 switch ((event.timestamp & 0x30000000)>>28) {
    337                     case 0:
    338                         event.value.driving_status = VEHICLE_DRIVING_STATUS_UNRESTRICTED;
    339                         break;
    340                     case 1:
    341                         event.value.driving_status = VEHICLE_DRIVING_STATUS_NO_VIDEO;
    342                         break;
    343                     case 2:
    344                         event.value.driving_status = VEHICLE_DRIVING_STATUS_NO_KEYBOARD_INPUT;
    345                         break;
    346                     default:
    347                         event.value.driving_status = VEHICLE_DRIVING_STATUS_NO_CONFIG;
    348                 }
    349                 break;
    350             case VEHICLE_PROPERTY_GEAR_SELECTION:
    351                 event.value_type = VEHICLE_VALUE_TYPE_INT32;
    352                 switch ((event.timestamp & 0x30000000)>>28) {
    353                     case 0:
    354                         event.value.gear_selection = VEHICLE_GEAR_PARK;
    355                         break;
    356                     case 1:
    357                         event.value.gear_selection = VEHICLE_GEAR_NEUTRAL;
    358                         break;
    359                     case 2:
    360                         event.value.gear_selection = VEHICLE_GEAR_DRIVE;
    361                         break;
    362                     case 3:
    363                         event.value.gear_selection = VEHICLE_GEAR_REVERSE;
    364                         break;
    365                 }
    366                 break;
    367             case VEHICLE_PROPERTY_PARKING_BRAKE_ON:
    368                 event.value_type = VEHICLE_VALUE_TYPE_BOOLEAN;
    369                 if (event.timestamp & 0x20000000) {
    370                     event.value.parking_brake = VEHICLE_FALSE;
    371                 } else {
    372                     event.value.parking_brake = VEHICLE_TRUE;
    373                 }
    374                 break;
    375             case VEHICLE_PROPERTY_PERF_VEHICLE_SPEED:
    376                 event.value_type = VEHICLE_VALUE_TYPE_FLOAT;
    377                 event.value.vehicle_speed = (float) ((event.timestamp & 0xff000000)>>24);
    378                 break;
    379             case VEHICLE_PROPERTY_RADIO_PRESET:
    380                 event.value_type = VEHICLE_VALUE_TYPE_INT32_VEC4;
    381                 int presetInfo1[4] = {1  /* preset number */, 0  /* AM Band */, 1000, 0};
    382                 int presetInfo2[4] = {2  /* preset number */, 1  /* FM Band */, 1000, 0};
    383                 if (event.timestamp & 0x20000000) {
    384                     memcpy(event.value.int32_array, presetInfo1, sizeof(presetInfo1));
    385                 } else {
    386                     memcpy(event.value.int32_array, presetInfo2, sizeof(presetInfo2));
    387                 }
    388                 break;
    389             default: // unsupported
    390                 if (sub->impl == NULL) {
    391                     ALOGE("subscription impl NULL");
    392                     return;
    393                 }
    394                 if (sub->impl->error_fn_ != NULL) {
    395                     sub->impl->error_fn_(-EINVAL, VEHICLE_PROPERTY_INVALID,
    396                             VEHICLE_OPERATION_GENERIC);
    397                 } else {
    398                     ALOGE("Error function is null");
    399                 }
    400                 ALOGE("Unsupported prop 0x%x, quit", sub->prop);
    401                 return;
    402         }
    403         if (sub->impl->event_fn_ != NULL) {
    404             sub->impl->event_fn_(&event);
    405         } else {
    406             ALOGE("Event function is null");
    407             return;
    408         }
    409         pthread_mutex_lock(&sub->lock);
    410         if (sub->stop_thread) {
    411             ALOGD("exiting subscription request here.");
    412             // Do any cleanup here.
    413             pthread_mutex_unlock(&sub->lock);
    414             return;
    415         }
    416         struct timespec now;
    417         clock_gettime(CLOCK_REALTIME, &now);
    418         now.tv_sec += 1; // sleep for one sec
    419         pthread_cond_timedwait(&sub->cond, &sub->lock, &now);
    420         pthread_mutex_unlock(&sub->lock);
    421     }
    422 }
    423 
    424 static int vdev_subscribe(vehicle_hw_device_t* device, int32_t prop, float sample_rate,
    425         int32_t zones UNUSED) {
    426     ALOGD("vdev_subscribe 0x%x, %f", prop, sample_rate);
    427     vehicle_device_impl_t* impl = (vehicle_device_impl_t*)device;
    428     // Check that the device is initialized.
    429     pthread_mutex_lock(&lock_);
    430     if (!impl->initialized_) {
    431         pthread_mutex_unlock(&lock_);
    432         ALOGE("vdev_subscribe: have you called init()?");
    433         return -EINVAL;
    434     }
    435     vehicle_prop_config_t* config = find_config(prop);
    436     if (config == NULL) {
    437         pthread_mutex_unlock(&lock_);
    438         ALOGE("vdev_subscribe not supported property 0x%x", prop);
    439         return -EINVAL;
    440     }
    441     if ((config->access != VEHICLE_PROP_ACCESS_READ) &&
    442         (config->access != VEHICLE_PROP_ACCESS_READ_WRITE)) {
    443         pthread_mutex_unlock(&lock_);
    444         ALOGE("vdev_subscribe read not supported on the property 0x%x", prop);
    445         return -EINVAL;
    446     }
    447     if (config->change_mode == VEHICLE_PROP_CHANGE_MODE_STATIC) {
    448         pthread_mutex_unlock(&lock_);
    449         ALOGE("vdev_subscribe cannot subscribe static property 0x%x", prop);
    450         return -EINVAL;
    451     }
    452     if ((config->change_mode == VEHICLE_PROP_CHANGE_MODE_ON_CHANGE) && (sample_rate != 0)) {
    453         pthread_mutex_unlock(&lock_);
    454         ALOGE("vdev_subscribe on change type should have 0 sample rate, property 0x%x, sample rate %f",
    455                 prop, sample_rate);
    456         return -EINVAL;
    457     }
    458     if ((config->max_sample_rate < sample_rate) || (config->min_sample_rate > sample_rate)) {
    459 
    460         ALOGE("vdev_subscribe property 0x%x, invalid sample rate %f, min:%f, max:%f",
    461                 prop, sample_rate, config->min_sample_rate, config->max_sample_rate);
    462         pthread_mutex_unlock(&lock_);
    463         return -EINVAL;
    464     }
    465     subscription_t* sub = (subscription_t*)config->hal_data;
    466     if (sub == NULL) {
    467         sub = calloc(1, sizeof(subscription_t));
    468         sub->prop = prop;
    469         sub->sample_rate = sample_rate;
    470         sub->stop_thread = 0;
    471         sub->impl = impl;
    472         pthread_mutex_init(&sub->lock, NULL);
    473         pthread_cond_init(&sub->cond, NULL);
    474         config->hal_data = sub;
    475         sprintf(sub->name, "vhal0x%x", prop);
    476     } else if (sub->sample_rate != sample_rate){ // sample rate changed
    477         //TODO notify this to fake sensor thread
    478         sub->sample_rate = sample_rate;
    479         pthread_mutex_unlock(&lock_);
    480         return 0;
    481     }
    482     int ret_code = pthread_create(
    483                                   &sub->thread, NULL, (void *(*)(void*))fake_event_thread, sub);
    484     print_subscribe_info(impl);
    485     pthread_mutex_unlock(&lock_);
    486     return 0;
    487 }
    488 
    489 static int vdev_unsubscribe(vehicle_hw_device_t* device, int32_t prop) {
    490     ALOGD("vdev_unsubscribe 0x%x", prop);
    491     vehicle_device_impl_t* impl = (vehicle_device_impl_t*)device;
    492     pthread_mutex_lock(&lock_);
    493     vehicle_prop_config_t* config = find_config(prop);
    494     if (config == NULL) {
    495         pthread_mutex_unlock(&lock_);
    496         return -EINVAL;
    497     }
    498     subscription_t* sub = (subscription_t*)config->hal_data;
    499     if (sub == NULL) {
    500         pthread_mutex_unlock(&lock_);
    501         return -EINVAL;
    502     }
    503     config->hal_data = NULL;
    504     pthread_mutex_unlock(&lock_);
    505     pthread_mutex_lock(&sub->lock);
    506     sub->stop_thread = 1;
    507     pthread_cond_signal(&sub->cond);
    508     pthread_mutex_unlock(&sub->lock);
    509     pthread_join(sub->thread, NULL);
    510     pthread_cond_destroy(&sub->cond);
    511     pthread_mutex_destroy(&sub->lock);
    512     free(sub);
    513     pthread_mutex_lock(&lock_);
    514     print_subscribe_info(impl);
    515     pthread_mutex_unlock(&lock_);
    516     return 0;
    517 }
    518 
    519 static int vdev_close(hw_device_t* device) {
    520     vehicle_device_impl_t* impl = (vehicle_device_impl_t*)device;
    521     if (impl) {
    522         free(impl);
    523         return 0;
    524     } else {
    525         return -1;
    526     }
    527 }
    528 
    529 static int vdev_dump(struct vehicle_hw_device* device UNUSED, int fd UNUSED) {
    530     //TODO
    531     return 0;
    532 }
    533 
    534 /*
    535  * The open function is provided as an interface in harwdare.h which fills in
    536  * all the information about specific implementations and version specific
    537  * informations in hw_device_t structure. After calling open() the client should
    538  * use the hw_device_t to execute any Vehicle HAL device specific functions.
    539  */
    540 static int vdev_open(const hw_module_t* module, const char* name UNUSED,
    541                      hw_device_t** device) {
    542     ALOGD("vdev_open");
    543 
    544     // Oops, out of memory!
    545     vehicle_device_impl_t* vdev = calloc(1, sizeof(vehicle_device_impl_t));
    546     if (vdev == NULL) {
    547         return -ENOMEM;
    548     }
    549 
    550     // Common functions provided by harware.h to access module and device(s).
    551     vdev->vehicle_device.common.tag = HARDWARE_DEVICE_TAG;
    552     vdev->vehicle_device.common.version = VEHICLE_DEVICE_API_VERSION_1_0;
    553     vdev->vehicle_device.common.module = (hw_module_t *) module;
    554     vdev->vehicle_device.common.close = vdev_close;
    555 
    556     // Define the Vehicle HAL device specific functions.
    557     vdev->vehicle_device.list_properties = vdev_list_properties;
    558     vdev->vehicle_device.init = vdev_init;
    559     vdev->vehicle_device.release = vdev_release;
    560     vdev->vehicle_device.get = vdev_get;
    561     vdev->vehicle_device.release_memory_from_get = vdev_release_memory_from_get;
    562     vdev->vehicle_device.set = vdev_set;
    563     vdev->vehicle_device.subscribe = vdev_subscribe;
    564     vdev->vehicle_device.unsubscribe = vdev_unsubscribe;
    565     vdev->vehicle_device.dump = vdev_dump;
    566 
    567     *device = (hw_device_t *) vdev;
    568     return 0;
    569 }
    570 
    571 static struct hw_module_methods_t hal_module_methods = {
    572     .open = vdev_open,
    573 };
    574 
    575 /*
    576  * This structure is mandatory to be implemented by each HAL implementation. It
    577  * exposes the open method (see hw_module_methods_t above) which opens a device.
    578  * The vehicle HAL is supposed to be used as a single device HAL hence all the
    579  * functions should be implemented inside of the vehicle_hw_device_t struct (see
    580  * the vehicle.h in include/ folder.
    581  */
    582 vehicle_module_t HAL_MODULE_INFO_SYM = {
    583     .common = {
    584         .tag = HARDWARE_MODULE_TAG,
    585         .module_api_version = VEHICLE_MODULE_API_VERSION_1_0,
    586         .hal_api_version = HARDWARE_HAL_API_VERSION,
    587         .id = VEHICLE_HARDWARE_MODULE_ID,
    588         .name = "Default vehicle HW HAL",
    589         .author = "",
    590         .methods = &hal_module_methods,
    591     },
    592 };
    593