blob: cde1536633f70c71d4444069b5da1c8a697a9769 [file] [log] [blame]
Steven Moreland9f8b5c72016-11-29 15:03:38 -08001/*
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 */
Yifan Hongf9d30342016-11-30 13:45:34 -080016
17#define LOG_TAG "light"
Mark Salyzyn3ff52602017-01-10 10:16:48 -080018
19#include <log/log.h>
Yifan Hongf9d30342016-11-30 13:45:34 -080020
Steven Moreland81f5da92016-09-30 16:32:24 -070021#include "Light.h"
22
23namespace android {
24namespace hardware {
25namespace light {
26namespace V2_0 {
27namespace implementation {
28
29static_assert(LIGHT_FLASH_NONE == static_cast<int>(Flash::NONE),
30 "Flash::NONE must match legacy value.");
31static_assert(LIGHT_FLASH_TIMED == static_cast<int>(Flash::TIMED),
32 "Flash::TIMED must match legacy value.");
33static_assert(LIGHT_FLASH_HARDWARE == static_cast<int>(Flash::HARDWARE),
34 "Flash::HARDWARE must match legacy value.");
35
36static_assert(BRIGHTNESS_MODE_USER == static_cast<int>(Brightness::USER),
37 "Brightness::USER must match legacy value.");
38static_assert(BRIGHTNESS_MODE_SENSOR == static_cast<int>(Brightness::SENSOR),
39 "Brightness::SENSOR must match legacy value.");
40static_assert(BRIGHTNESS_MODE_LOW_PERSISTENCE ==
41 static_cast<int>(Brightness::LOW_PERSISTENCE),
42 "Brightness::LOW_PERSISTENCE must match legacy value.");
43
44Light::Light(std::map<Type, light_device_t*> &&lights)
45 : mLights(std::move(lights)) {}
46
47// Methods from ::android::hardware::light::V2_0::ILight follow.
48Return<Status> Light::setLight(Type type, const LightState& state) {
49 auto it = mLights.find(type);
50
51 if (it == mLights.end()) {
52 return Status::LIGHT_NOT_SUPPORTED;
53 }
54
55 light_device_t* hwLight = it->second;
56
57 light_state_t legacyState {
58 .color = state.color,
59 .flashMode = static_cast<int>(state.flashMode),
60 .flashOnMS = state.flashOnMs,
61 .flashOffMS = state.flashOffMs,
62 .brightnessMode = static_cast<int>(state.brightnessMode),
63 };
64
65 int ret = hwLight->set_light(hwLight, &legacyState);
66
67 switch (ret) {
68 case -ENOSYS:
69 return Status::BRIGHTNESS_NOT_SUPPORTED;
70 case 0:
71 return Status::SUCCESS;
72 default:
73 return Status::UNKNOWN;
74 }
75}
76
77Return<void> Light::getSupportedTypes(getSupportedTypes_cb _hidl_cb) {
78 Type *types = new Type[mLights.size()];
79
80 int idx = 0;
81 for(auto const &pair : mLights) {
82 Type type = pair.first;
83
84 types[idx++] = type;
85 }
86
87 {
88 hidl_vec<Type> hidl_types{};
89 hidl_types.setToExternal(types, mLights.size());
90
91 _hidl_cb(hidl_types);
92 }
93
94 delete[] types;
95
96 return Void();
97}
98
99const static std::map<Type, const char*> kLogicalLights = {
100 {Type::BACKLIGHT, LIGHT_ID_BACKLIGHT},
101 {Type::KEYBOARD, LIGHT_ID_KEYBOARD},
102 {Type::BUTTONS, LIGHT_ID_BUTTONS},
103 {Type::BATTERY, LIGHT_ID_BATTERY},
104 {Type::NOTIFICATIONS, LIGHT_ID_NOTIFICATIONS},
105 {Type::ATTENTION, LIGHT_ID_ATTENTION},
106 {Type::BLUETOOTH, LIGHT_ID_BLUETOOTH},
107 {Type::WIFI, LIGHT_ID_WIFI}
108};
109
110light_device_t* getLightDevice(const char* name) {
111 light_device_t* lightDevice;
112 const hw_module_t* hwModule = NULL;
113
114 int ret = hw_get_module (LIGHTS_HARDWARE_MODULE_ID, &hwModule);
115 if (ret == 0) {
116 ret = hwModule->methods->open(hwModule, name,
117 reinterpret_cast<hw_device_t**>(&lightDevice));
118 if (ret != 0) {
119 ALOGE("light_open %s %s failed: %d", LIGHTS_HARDWARE_MODULE_ID, name, ret);
120 }
121 } else {
122 ALOGE("hw_get_module %s %s failed: %d", LIGHTS_HARDWARE_MODULE_ID, name, ret);
123 }
124
125 if (ret == 0) {
126 return lightDevice;
127 } else {
128 ALOGE("Light passthrough failed to load legacy HAL.");
129 return nullptr;
130 }
131}
132
133ILight* HIDL_FETCH_ILight(const char* /* name */) {
134 std::map<Type, light_device_t*> lights;
135
136 for(auto const &pair : kLogicalLights) {
137 Type type = pair.first;
138 const char* name = pair.second;
139
140 light_device_t* light = getLightDevice(name);
141
142 if (light != nullptr) {
143 lights[type] = light;
144 }
145 }
146
147 if (lights.size() == 0) {
148 // Log information, but still return new Light.
149 // Some devices may not have any lights.
150 ALOGI("Could not open any lights.");
151 }
152
153 return new Light(std::move(lights));
154}
155
156} // namespace implementation
157} // namespace V2_0
158} // namespace light
159} // namespace hardware
160} // namespace android