/* * 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. */ #define LOG_TAG "android.hardware.light@2.0-service.xiaomi_sdm660" #include #include "Light.h" #include #define LEDS "/sys/class/leds/" #define LCD_LED LEDS "lcd-backlight/" #define WHITE_LED LEDS "red/" #define BLINK "blink" #define BRIGHTNESS "brightness" #define MAX_BRIGHTNESS "max_brightness" #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" /* * 15 duty percent steps. */ #define RAMP_STEPS 15 /* * Each step will stay on for 150ms by default. */ #define RAMP_STEP_DURATION 150 /* * 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, 85, 72, 50, 37, 25, 12, 0}; namespace { /* * Write value to path and close file. */ static void set(std::string path, std::string value) { std::ofstream file(path); if (!file.is_open()) { ALOGW("failed to write %s to %s", value.c_str(), path.c_str()); return; } file << value; } static void set(std::string path, int value) { set(path, std::to_string(value)); } static int get(std::string path) { std::ifstream file(path); int value; if (!file.is_open()) { ALOGW("failed to read from %s", path.c_str()); return 0; } file >> value; return value; } static int getMaxBrightness(std::string path) { int value = get(path); ALOGW("Got max brightness %d", value); return 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; } 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 handleBacklight(const LightState& state) { uint32_t brightness = getScaledBrightness(state, getMaxBrightness(LCD_LED MAX_BRIGHTNESS)); set(LCD_LED BRIGHTNESS, brightness); } /* * 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 void handleNotification(const LightState& state) { uint32_t whiteBrightness = getScaledBrightness(state, getMaxBrightness(WHITE_LED MAX_BRIGHTNESS)); /* Disable blinking */ set(WHITE_LED 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) { //stepDuration = state.flashOnMs / (RAMP_STEPS * 2); pauseHi = 0; } /* White */ set(WHITE_LED START_IDX, 0 * RAMP_STEPS); set(WHITE_LED DUTY_PCTS, getScaledRamp(whiteBrightness)); set(WHITE_LED PAUSE_LO, pauseLo); set(WHITE_LED PAUSE_HI, pauseHi); set(WHITE_LED RAMP_STEP_MS, stepDuration); /* Enable blinking */ set(WHITE_LED BLINK, 1); } else { set(WHITE_LED BRIGHTNESS, whiteBrightness); } } static inline bool isLit(const LightState& state) { return state.color & 0x00ffffff; } /* Keep sorted in the order of importance. */ static std::vector backends = { { Type::ATTENTION, handleNotification }, { Type::NOTIFICATIONS, handleNotification }, { Type::BATTERY, handleNotification }, { Type::BACKLIGHT, handleBacklight }, }; } // anonymous namespace namespace android { namespace hardware { namespace light { namespace V2_0 { namespace implementation { Return Light::setLight(Type type, const LightState& state) { LightStateHandler handler = nullptr; /* Lock global mutex until light state is updated. */ std::lock_guard lock(globalLock); /* Update the cached state value for the current type. */ for (LightBackend& backend : backends) { if (backend.type == type) { backend.state = state; handler = backend.handler; } } /* If no handler has been found, then the type is not supported. */ if (!handler) { return Status::LIGHT_NOT_SUPPORTED; } /* Light up the type with the highest priority that matches the current handler. */ for (LightBackend& backend : backends) { if (handler == backend.handler && isLit(backend.state)) { handler(backend.state); return Status::SUCCESS; } } /* If no type has been lit up, then turn off the hardware. */ handler(state); return Status::SUCCESS; } Return Light::getSupportedTypes(getSupportedTypes_cb _hidl_cb) { std::vector types; for (const LightBackend& backend : backends) { types.push_back(backend.type); } _hidl_cb(types); return Void(); } } // namespace implementation } // namespace V2_0 } // namespace light } // namespace hardware } // namespace android