blob: 6a105dadbb3eda5e5411b725a078188f1ab0d046 [file] [log] [blame]
Jan Altensen53afa332019-07-09 22:43:47 +02001/*
2 * Copyright (C) 2019 The LineageOS 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#define LOG_TAG "android.hardware.light@2.0-service.samsung"
17
Jan Altensen83cc4602020-02-03 18:57:34 +010018#include <android-base/stringprintf.h>
Jan Altensen53afa332019-07-09 22:43:47 +020019#include <iomanip>
20
21#include "Light.h"
22
23#define COLOR_MASK 0x00ffffff
24#define MAX_INPUT_BRIGHTNESS 255
25
26using android::hardware::light::V2_0::LightState;
27using android::hardware::light::V2_0::Status;
28using android::hardware::light::V2_0::Type;
29
30namespace android {
31namespace hardware {
32namespace light {
33namespace V2_0 {
34namespace implementation {
35
36/*
37 * Write value to path and close file.
38 */
39template <typename T>
40static void set(const std::string& path, const T& value) {
41 std::ofstream file(path);
42 file << value << std::endl;
43}
44
45template <typename T>
46static T get(const std::string& path, const T& def) {
47 std::ifstream file(path);
48 T result;
49
50 file >> result;
51 return file.fail() ? def : result;
52}
53
54Light::Light() {
55 mLights.emplace(Type::BACKLIGHT,
56 std::bind(&Light::handleBacklight, this, std::placeholders::_1));
57#ifdef BUTTON_BRIGHTNESS_NODE
58 mLights.emplace(Type::BUTTONS, std::bind(&Light::handleButtons, this, std::placeholders::_1));
Andreas Schneidereeb666c2020-03-13 20:26:02 +010059#endif /* BUTTON_BRIGHTNESS_NODE */
60#ifdef LED_BLINK_NODE
Jan Altensen53afa332019-07-09 22:43:47 +020061 mLights.emplace(Type::BATTERY, std::bind(&Light::handleBattery, this, std::placeholders::_1));
62 mLights.emplace(Type::NOTIFICATIONS,
63 std::bind(&Light::handleNotifications, this, std::placeholders::_1));
64 mLights.emplace(Type::ATTENTION,
65 std::bind(&Light::handleAttention, this, std::placeholders::_1));
Andreas Schneidereeb666c2020-03-13 20:26:02 +010066#endif /* LED_BLINK_NODE */
Jan Altensen53afa332019-07-09 22:43:47 +020067}
68
69Return<Status> Light::setLight(Type type, const LightState& state) {
70 auto it = mLights.find(type);
71
72 if (it == mLights.end()) {
73 return Status::LIGHT_NOT_SUPPORTED;
74 }
75
76 /*
77 * Lock global mutex until light state is updated.
78 */
79 std::lock_guard<std::mutex> lock(mLock);
80
81 it->second(state);
82
83 return Status::SUCCESS;
84}
85
86void Light::handleBacklight(const LightState& state) {
87 uint32_t max_brightness = get(PANEL_MAX_BRIGHTNESS_NODE, MAX_INPUT_BRIGHTNESS);
88 uint32_t brightness = rgbToBrightness(state);
89
90 if (max_brightness != MAX_INPUT_BRIGHTNESS) {
91 brightness = brightness * max_brightness / MAX_INPUT_BRIGHTNESS;
92 }
93
94 set(PANEL_BRIGHTNESS_NODE, brightness);
95}
96
97#ifdef BUTTON_BRIGHTNESS_NODE
98void Light::handleButtons(const LightState& state) {
99#ifdef VAR_BUTTON_BRIGHTNESS
100 uint32_t brightness = rgbToBrightness(state);
101#else
102 uint32_t brightness = (state.color & COLOR_MASK) ? 1 : 0;
103#endif
104
105 set(BUTTON_BRIGHTNESS_NODE, brightness);
106}
107#endif
108
Andreas Schneidereeb666c2020-03-13 20:26:02 +0100109#ifdef LED_BLINK_NODE
Jan Altensen53afa332019-07-09 22:43:47 +0200110void Light::handleBattery(const LightState& state) {
111 mBatteryState = state;
112 setNotificationLED();
113}
114
115void Light::handleNotifications(const LightState& state) {
116 mNotificationState = state;
117 setNotificationLED();
118}
119
120void Light::handleAttention(const LightState& state) {
121 mAttentionState = state;
122 setNotificationLED();
123}
124
125void Light::setNotificationLED() {
126 int32_t adjusted_brightness = MAX_INPUT_BRIGHTNESS;
127 LightState state;
128#ifdef LED_BLN_NODE
129 bool bln = false;
Andreas Schneidereeb666c2020-03-13 20:26:02 +0100130#endif /* LED_BLN_NODE */
Jan Altensen53afa332019-07-09 22:43:47 +0200131
Jan Altensen7ffb4952019-08-07 11:33:32 +0200132 if (mNotificationState.color & COLOR_MASK) {
Jan Altensen53afa332019-07-09 22:43:47 +0200133 adjusted_brightness = LED_BRIGHTNESS_NOTIFICATION;
134 state = mNotificationState;
135#ifdef LED_BLN_NODE
136 bln = true;
Andreas Schneidereeb666c2020-03-13 20:26:02 +0100137#endif /* LED_BLN_NODE */
Jan Altensen53afa332019-07-09 22:43:47 +0200138 } else if (mAttentionState.color & COLOR_MASK) {
139 adjusted_brightness = LED_BRIGHTNESS_ATTENTION;
140 state = mAttentionState;
141 if (state.flashMode == Flash::HARDWARE) {
142 if (state.flashOnMs > 0 && state.flashOffMs == 0) state.flashMode = Flash::NONE;
143 state.color = 0x000000ff;
144 }
145 if (state.flashMode == Flash::NONE) {
146 state.color = 0;
147 }
Jan Altensen7ffb4952019-08-07 11:33:32 +0200148 } else if (mBatteryState.color & COLOR_MASK) {
149 adjusted_brightness = LED_BRIGHTNESS_BATTERY;
150 state = mBatteryState;
Jan Altensen53afa332019-07-09 22:43:47 +0200151 } else {
152 set(LED_BLINK_NODE, "0x00000000 0 0");
153 return;
154 }
155
156 if (state.flashMode == Flash::NONE) {
157 state.flashOnMs = 0;
158 state.flashOffMs = 0;
159 }
160
161 state.color = calibrateColor(state.color & COLOR_MASK, adjusted_brightness);
Jan Altensen83cc4602020-02-03 18:57:34 +0100162 set(LED_BLINK_NODE, android::base::StringPrintf("0x%08x %d %d", state.color, state.flashOnMs,
163 state.flashOffMs));
Jan Altensen53afa332019-07-09 22:43:47 +0200164
165#ifdef LED_BLN_NODE
166 if (bln) {
167 set(LED_BLN_NODE, (state.color & COLOR_MASK) ? 1 : 0);
168 }
Andreas Schneidereeb666c2020-03-13 20:26:02 +0100169#endif /* LED_BLN_NODE */
Jan Altensen53afa332019-07-09 22:43:47 +0200170}
171
Andreas Schneidereeb666c2020-03-13 20:26:02 +0100172uint32_t Light::calibrateColor(uint32_t color, int32_t brightness) {
173 uint32_t red = ((color >> 16) & 0xFF) * LED_ADJUSTMENT_R;
174 uint32_t green = ((color >> 8) & 0xFF) * LED_ADJUSTMENT_G;
175 uint32_t blue = (color & 0xFF) * LED_ADJUSTMENT_B;
176
177 return (((red * brightness) / 255) << 16) + (((green * brightness) / 255) << 8) +
178 ((blue * brightness) / 255);
179}
180#endif /* LED_BLINK_NODE */
181
Jan Altensen53afa332019-07-09 22:43:47 +0200182Return<void> Light::getSupportedTypes(getSupportedTypes_cb _hidl_cb) {
183 std::vector<Type> types;
184
185 for (auto const& light : mLights) {
186 types.push_back(light.first);
187 }
188
189 _hidl_cb(types);
190
191 return Void();
192}
193
194uint32_t Light::rgbToBrightness(const LightState& state) {
195 uint32_t color = state.color & COLOR_MASK;
196
197 return ((77 * ((color >> 16) & 0xff)) + (150 * ((color >> 8) & 0xff)) + (29 * (color & 0xff))) >>
198 8;
199}
200
Jan Altensen53afa332019-07-09 22:43:47 +0200201} // namespace implementation
202} // namespace V2_0
203} // namespace light
204} // namespace hardware
205} // namespace android