Home | History | Annotate | Download | only in default
      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 
     17 #define LOG_TAG "light"
     18 
     19 #include <log/log.h>
     20 
     21 #include <stdio.h>
     22 
     23 #include "Light.h"
     24 
     25 namespace android {
     26 namespace hardware {
     27 namespace light {
     28 namespace V2_0 {
     29 namespace implementation {
     30 
     31 static_assert(LIGHT_FLASH_NONE == static_cast<int>(Flash::NONE),
     32     "Flash::NONE must match legacy value.");
     33 static_assert(LIGHT_FLASH_TIMED == static_cast<int>(Flash::TIMED),
     34     "Flash::TIMED must match legacy value.");
     35 static_assert(LIGHT_FLASH_HARDWARE == static_cast<int>(Flash::HARDWARE),
     36     "Flash::HARDWARE must match legacy value.");
     37 
     38 static_assert(BRIGHTNESS_MODE_USER == static_cast<int>(Brightness::USER),
     39     "Brightness::USER must match legacy value.");
     40 static_assert(BRIGHTNESS_MODE_SENSOR == static_cast<int>(Brightness::SENSOR),
     41     "Brightness::SENSOR must match legacy value.");
     42 static_assert(BRIGHTNESS_MODE_LOW_PERSISTENCE ==
     43     static_cast<int>(Brightness::LOW_PERSISTENCE),
     44     "Brightness::LOW_PERSISTENCE must match legacy value.");
     45 
     46 Light::Light(std::map<Type, light_device_t*> &&lights)
     47   : mLights(std::move(lights)) {}
     48 
     49 // Methods from ::android::hardware::light::V2_0::ILight follow.
     50 Return<Status> Light::setLight(Type type, const LightState& state)  {
     51     auto it = mLights.find(type);
     52 
     53     if (it == mLights.end()) {
     54         return Status::LIGHT_NOT_SUPPORTED;
     55     }
     56 
     57     light_device_t* hwLight = it->second;
     58 
     59     light_state_t legacyState {
     60         .color = state.color,
     61         .flashMode = static_cast<int>(state.flashMode),
     62         .flashOnMS = state.flashOnMs,
     63         .flashOffMS = state.flashOffMs,
     64         .brightnessMode = static_cast<int>(state.brightnessMode),
     65     };
     66 
     67     int ret = hwLight->set_light(hwLight, &legacyState);
     68 
     69     switch (ret) {
     70         case -ENOSYS:
     71             return Status::BRIGHTNESS_NOT_SUPPORTED;
     72         case 0:
     73             return Status::SUCCESS;
     74         default:
     75             return Status::UNKNOWN;
     76     }
     77 }
     78 
     79 Return<void> Light::getSupportedTypes(getSupportedTypes_cb _hidl_cb)  {
     80     Type *types = new Type[mLights.size()];
     81 
     82     int idx = 0;
     83     for(auto const &pair : mLights) {
     84         Type type = pair.first;
     85 
     86         types[idx++] = type;
     87     }
     88 
     89     {
     90         hidl_vec<Type> hidl_types{};
     91         hidl_types.setToExternal(types, mLights.size());
     92 
     93         _hidl_cb(hidl_types);
     94     }
     95 
     96     delete[] types;
     97 
     98     return Void();
     99 }
    100 
    101 const static std::map<Type, const char*> kLogicalLights = {
    102     {Type::BACKLIGHT,     LIGHT_ID_BACKLIGHT},
    103     {Type::KEYBOARD,      LIGHT_ID_KEYBOARD},
    104     {Type::BUTTONS,       LIGHT_ID_BUTTONS},
    105     {Type::BATTERY,       LIGHT_ID_BATTERY},
    106     {Type::NOTIFICATIONS, LIGHT_ID_NOTIFICATIONS},
    107     {Type::ATTENTION,     LIGHT_ID_ATTENTION},
    108     {Type::BLUETOOTH,     LIGHT_ID_BLUETOOTH},
    109     {Type::WIFI,          LIGHT_ID_WIFI}
    110 };
    111 
    112 Return<void> Light::debug(const hidl_handle& handle, const hidl_vec<hidl_string>& /* options */) {
    113     if (handle == nullptr || handle->numFds < 1) {
    114         ALOGE("debug called with no handle\n");
    115         return Void();
    116     }
    117 
    118     int fd = handle->data[0];
    119     if (fd < 0) {
    120         ALOGE("invalid FD: %d\n", handle->data[0]);
    121         return Void();
    122     }
    123 
    124     dprintf(fd, "The following lights are registered: ");
    125     for (auto const& pair : mLights) {
    126         const Type type = pair.first;
    127         dprintf(fd, "%s,", kLogicalLights.at(type));
    128     }
    129     dprintf(fd, ".\n");
    130     fsync(fd);
    131     return Void();
    132 }
    133 
    134 light_device_t* getLightDevice(const char* name) {
    135     light_device_t* lightDevice;
    136     const hw_module_t* hwModule = NULL;
    137 
    138     int ret = hw_get_module (LIGHTS_HARDWARE_MODULE_ID, &hwModule);
    139     if (ret == 0) {
    140         ret = hwModule->methods->open(hwModule, name,
    141             reinterpret_cast<hw_device_t**>(&lightDevice));
    142         if (ret != 0) {
    143             ALOGE("light_open %s %s failed: %d", LIGHTS_HARDWARE_MODULE_ID, name, ret);
    144         }
    145     } else {
    146         ALOGE("hw_get_module %s %s failed: %d", LIGHTS_HARDWARE_MODULE_ID, name, ret);
    147     }
    148 
    149     if (ret == 0) {
    150         return lightDevice;
    151     } else {
    152         ALOGE("Light passthrough failed to load legacy HAL.");
    153         return nullptr;
    154     }
    155 }
    156 
    157 ILight* HIDL_FETCH_ILight(const char* /* name */) {
    158     std::map<Type, light_device_t*> lights;
    159 
    160     for(auto const &pair : kLogicalLights) {
    161         Type type = pair.first;
    162         const char* name = pair.second;
    163 
    164         light_device_t* light = getLightDevice(name);
    165 
    166         if (light != nullptr) {
    167             lights[type] = light;
    168         }
    169     }
    170 
    171     if (lights.size() == 0) {
    172         // Log information, but still return new Light.
    173         // Some devices may not have any lights.
    174         ALOGI("Could not open any lights.");
    175     }
    176 
    177     return new Light(std::move(lights));
    178 }
    179 
    180 } // namespace implementation
    181 }  // namespace V2_0
    182 }  // namespace light
    183 }  // namespace hardware
    184 }  // namespace android
    185