/* * Copyright (C) 2018 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ //Author := dev_harsh1998, Isaac Chen #define LOG_TAG "android.hardware.light@2.0-service.xiaomi_sdm660" #include #include #include "Light.h" namespace android { namespace hardware { namespace light { namespace V2_0 { namespace implementation { #define LEDS "/sys/class/leds/" #define LCD_LED LEDS "lcd-backlight/" #define BRIGHTNESS "brightness" #define WHITE LEDS "white/" #define BLINK "blink" #define DUTY_PCTS "duty_pcts" #define PAUSE_HI "pause_hi" #define PAUSE_LO "pause_lo" #define RAMP_STEP_MS "ramp_step_ms" #define START_IDX "start_idx" #define MAX_LED_BRIGHTNESS 255 #define MAX_LCD_BRIGHTNESS 4095 /* * 8 duty percent steps. */ #define RAMP_STEPS 8 /* * Each step will stay on for 50ms by default. */ #define RAMP_STEP_DURATION 50 /* * Each value represents a duty percent (0 - 100) for the led pwm. */ static int32_t BRIGHTNESS_RAMP[RAMP_STEPS] = {0, 12, 25, 37, 50, 72, 85, 100}; /* * Write value to path and close file. */ static void set(std::string path, std::string value) { std::ofstream file(path); /* Only write brightness value if stream is open, alive & well */ if (file.is_open()) { file << value; } else { /* Fire a warning a bail out */ ALOGE("failed to write %s to %s", value.c_str(), path.c_str()); return; } } static void set(std::string path, int value) { set(path, std::to_string(value)); } static uint32_t getBrightness(const LightState& state) { uint32_t alpha, red, green, blue; /* * Extract brightness from AARRGGBB. */ alpha = (state.color >> 24) & 0xFF; red = (state.color >> 16) & 0xFF; green = (state.color >> 8) & 0xFF; blue = state.color & 0xFF; /* * Scale RGB brightness if Alpha brightness is not 0xFF. */ if (alpha != 0xFF) { red = red * alpha / 0xFF; green = green * alpha / 0xFF; blue = blue * alpha / 0xFF; } return (77 * red + 150 * green + 29 * blue) >> 8; } /* * Scale each value of the brightness ramp according to the * brightness of the color. */ static std::string getScaledRamp(uint32_t brightness) { std::string ramp, pad; for (auto const& step : BRIGHTNESS_RAMP) { ramp += pad + std::to_string(step * brightness / 0xFF); pad = ","; } return ramp; } static inline uint32_t scaleBrightness(uint32_t brightness, uint32_t maxBrightness) { return brightness * maxBrightness / 0xFF; } static inline uint32_t getScaledBrightness(const LightState& state, uint32_t maxBrightness) { return scaleBrightness(getBrightness(state), maxBrightness); } static void handleXiaomiBacklight(Type /*type*/, const LightState& state) { uint32_t brightness = getScaledBrightness(state, MAX_LCD_BRIGHTNESS); set(LCD_LED BRIGHTNESS, brightness); } static void setNotification(const LightState& state) { uint32_t whiteBrightness = getScaledBrightness(state, MAX_LED_BRIGHTNESS); /* Turn off the leds (initially) */ set(WHITE BLINK, 0); if (state.flashMode == Flash::TIMED) { /* * If the flashOnMs duration is not long enough to fit ramping up * and down at the default step duration, step duration is modified * to fit. */ int32_t stepDuration = RAMP_STEP_DURATION; int32_t pauseHi = state.flashOnMs - (stepDuration * RAMP_STEPS * 2); int32_t pauseLo = state.flashOffMs; if (pauseHi < 0) { pauseHi = 0; } /* White */ set(WHITE START_IDX, 0 * RAMP_STEPS); set(WHITE DUTY_PCTS, getScaledRamp(whiteBrightness)); set(WHITE PAUSE_LO, pauseLo); set(WHITE PAUSE_HI, pauseHi); set(WHITE RAMP_STEP_MS, stepDuration); set(WHITE BLINK, 1); } else { set(WHITE BRIGHTNESS, whiteBrightness); } } static inline bool isLit(const LightState& state) { return state.color & 0x00ffffff; } /* * Keep sorted in the order of importance. */ static const LightState offState = {}; static std::vector> notificationStates = { { Type::ATTENTION, offState }, { Type::NOTIFICATIONS, offState }, { Type::BATTERY, offState }, }; static void handleXiaomiNotification(Type type, const LightState& state) { for(auto it : notificationStates) { if (it.first == type) { it.second = state; } if (isLit(it.second)) { setNotification(it.second); return; } } setNotification(offState); } static std::map> lights = { {Type::BACKLIGHT, handleXiaomiBacklight}, {Type::NOTIFICATIONS, handleXiaomiNotification}, {Type::BATTERY, handleXiaomiNotification}, {Type::ATTENTION, handleXiaomiNotification}, }; Light::Light() {} Return Light::setLight(Type type, const LightState& state) { auto it = lights.find(type); if (it == lights.end()) { return Status::LIGHT_NOT_SUPPORTED; } /* * Lock global mutex until light state is updated. */ std::lock_guard lock(globalLock); it->second(type, state); return Status::SUCCESS; } Return Light::getSupportedTypes(getSupportedTypes_cb _hidl_cb) { std::vector types; for (auto const& light : lights) types.push_back(light.first); _hidl_cb(types); return Void(); } } // namespace implementation } // namespace V2_0 } // namespace light } // namespace hardware } // namespace android