blob: f53f058c5372111d580ccccd1beb033534cf5c1d [file] [log] [blame]
Christopher N. Hessede5e3c62015-12-21 21:28:23 +01001/*
2 * Copyright (C) 2012 The Android Open Source Project
3 * Copyright (C) 2014 The CyanogenMod Project
4 * Copyright (C) 2014-2015 Andreas Schneider <asn@cryptomilk.org>
Christopher N. Hessef05f9022017-01-16 22:49:06 +01005 * Copyright (C) 2014-2017 Christopher N. Hesse <raymanfx@gmail.com>
Christopher N. Hessede5e3c62015-12-21 21:28:23 +01006 *
7 * Licensed under the Apache License, Version 2.0 (the "License");
8 * you may not use this file except in compliance with the License.
9 * You may obtain a copy of the License at
10 *
11 * http://www.apache.org/licenses/LICENSE-2.0
12 *
13 * Unless required by applicable law or agreed to in writing, software
14 * distributed under the License is distributed on an "AS IS" BASIS,
15 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16 * See the License for the specific language governing permissions and
17 * limitations under the License.
18 */
19
Christopher N. Hesse3360b092016-07-11 15:48:35 +020020#include <ctype.h>
Christopher N. Hessede5e3c62015-12-21 21:28:23 +010021#include <dirent.h>
22#include <errno.h>
23#include <fcntl.h>
24#include <malloc.h>
25#include <stdbool.h>
Christopher N. Hessee1434192016-11-18 18:56:17 +010026#include <stdlib.h>
Christopher N. Hessede5e3c62015-12-21 21:28:23 +010027#include <string.h>
28#include <unistd.h>
29
30#include <sys/types.h>
Christopher N. Hessede5e3c62015-12-21 21:28:23 +010031
32#define LOG_TAG "SamsungPowerHAL"
33/* #define LOG_NDEBUG 0 */
34#include <utils/Log.h>
35
36#include <hardware/hardware.h>
37#include <hardware/power.h>
Jan Altensen39e99622019-11-26 20:34:28 +010038#include "samsung_lights.h"
Christopher N. Hessede5e3c62015-12-21 21:28:23 +010039
Christopher N. Hesse4139d852016-12-07 12:21:44 +010040#include "samsung_power.h"
Christopher N. Hessede5e3c62015-12-21 21:28:23 +010041
Christopher N. Hesse7ad73bc2017-06-28 22:53:03 +020042#define BOOST_PATH "/boost"
43#define BOOSTPULSE_PATH "/boostpulse"
Christopher N. Hesse81f714d2017-03-12 17:11:04 +010044
Christopher N. Hesse7ad73bc2017-06-28 22:53:03 +020045#define IO_IS_BUSY_PATH "/io_is_busy"
46#define HISPEED_FREQ_PATH "/hispeed_freq"
Christopher N. Hesse81f714d2017-03-12 17:11:04 +010047
Christopher N. Hesse7ad73bc2017-06-28 22:53:03 +020048#define MAX_FREQ_PATH "/cpufreq/scaling_max_freq"
49
50#define CLUSTER_COUNT ARRAY_SIZE(CPU_SYSFS_PATHS)
51#define PARAM_MAXLEN 10
52
53#define ARRAY_SIZE(a) sizeof(a) / sizeof(a[0])
Christopher N. Hessed9106e92017-03-04 01:05:14 +010054
Christopher N. Hessede5e3c62015-12-21 21:28:23 +010055struct samsung_power_module {
56 struct power_module base;
57 pthread_mutex_t lock;
Christopher N. Hessefb5669e2017-07-08 12:44:39 +020058 int boost_fd;
Christopher N. Hessede5e3c62015-12-21 21:28:23 +010059 int boostpulse_fd;
Christopher N. Hesse7ad73bc2017-06-28 22:53:03 +020060 char hispeed_freqs[CLUSTER_COUNT][PARAM_MAXLEN];
61 char max_freqs[CLUSTER_COUNT][PARAM_MAXLEN];
Christopher N. Hessede5e3c62015-12-21 21:28:23 +010062 char* touchscreen_power_path;
63 char* touchkey_power_path;
Christopher N. Hessede5e3c62015-12-21 21:28:23 +010064};
65
Andreas Schneiderf15d7f42016-02-03 10:43:47 +010066enum power_profile_e {
67 PROFILE_POWER_SAVE = 0,
68 PROFILE_BALANCED,
Christopher N. Hesse58f2ca02017-01-17 00:01:15 +010069 PROFILE_HIGH_PERFORMANCE,
70 PROFILE_MAX
Andreas Schneiderf15d7f42016-02-03 10:43:47 +010071};
Christopher N. Hesseccd970d2017-01-25 22:41:05 +010072
Andreas Schneiderf15d7f42016-02-03 10:43:47 +010073static enum power_profile_e current_power_profile = PROFILE_BALANCED;
Christopher N. Hessede5e3c62015-12-21 21:28:23 +010074
Stricted404b41c2018-09-07 01:16:30 +020075// Custom Lineage hints
76const static power_hint_t POWER_HINT_CPU_BOOST = (power_hint_t)0x00000110;
77const static power_hint_t POWER_HINT_SET_PROFILE = (power_hint_t)0x00000111;
78
Christopher N. Hessede5e3c62015-12-21 21:28:23 +010079/**********************************************************
80 *** HELPER FUNCTIONS
81 **********************************************************/
82
83static int sysfs_read(char *path, char *s, int num_bytes)
84{
85 char errno_str[64];
86 int len;
87 int ret = 0;
88 int fd;
89
90 fd = open(path, O_RDONLY);
91 if (fd < 0) {
92 strerror_r(errno, errno_str, sizeof(errno_str));
Christopher N. Hesse354f7132017-03-04 01:06:22 +010093 ALOGE("Error opening %s: %s", path, errno_str);
Christopher N. Hessede5e3c62015-12-21 21:28:23 +010094
95 return -1;
96 }
97
98 len = read(fd, s, num_bytes - 1);
99 if (len < 0) {
100 strerror_r(errno, errno_str, sizeof(errno_str));
Christopher N. Hesse354f7132017-03-04 01:06:22 +0100101 ALOGE("Error reading from %s: %s", path, errno_str);
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100102
103 ret = -1;
104 } else {
Christopher N. Hesse0da5fbf2017-07-08 13:56:25 +0200105 // do not store newlines, but terminate the string instead
106 if (s[len-1] == '\n') {
107 s[len-1] = '\0';
108 } else {
109 s[len] = '\0';
110 }
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100111 }
112
113 close(fd);
114
115 return ret;
116}
117
118static void sysfs_write(const char *path, char *s)
119{
120 char errno_str[64];
121 int len;
122 int fd;
123
124 fd = open(path, O_WRONLY);
125 if (fd < 0) {
126 strerror_r(errno, errno_str, sizeof(errno_str));
Christopher N. Hesse354f7132017-03-04 01:06:22 +0100127 ALOGE("Error opening %s: %s", path, errno_str);
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100128 return;
129 }
130
131 len = write(fd, s, strlen(s));
132 if (len < 0) {
133 strerror_r(errno, errno_str, sizeof(errno_str));
Christopher N. Hesse354f7132017-03-04 01:06:22 +0100134 ALOGE("Error writing to %s: %s", path, errno_str);
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100135 }
136
137 close(fd);
138}
139
Christopher N. Hesse7ad73bc2017-06-28 22:53:03 +0200140static void cpu_sysfs_read(const char *param, char s[CLUSTER_COUNT][PARAM_MAXLEN])
141{
142 char path[PATH_MAX];
143
144 for (unsigned int i = 0; i < ARRAY_SIZE(CPU_SYSFS_PATHS); i++) {
145 sprintf(path, "%s%s", CPU_SYSFS_PATHS[i], param);
146 sysfs_read(path, s[i], PARAM_MAXLEN);
147 }
148}
149
150static void cpu_sysfs_write(const char *param, char s[CLUSTER_COUNT][PARAM_MAXLEN])
151{
152 char path[PATH_MAX];
153
154 for (unsigned int i = 0; i < ARRAY_SIZE(CPU_SYSFS_PATHS); i++) {
155 sprintf(path, "%s%s", CPU_SYSFS_PATHS[i], param);
156 sysfs_write(path, s[i]);
157 }
158}
159
160static void cpu_interactive_read(const char *param, char s[CLUSTER_COUNT][PARAM_MAXLEN])
161{
162 char path[PATH_MAX];
163
164 for (unsigned int i = 0; i < ARRAY_SIZE(CPU_INTERACTIVE_PATHS); i++) {
165 sprintf(path, "%s%s", CPU_INTERACTIVE_PATHS[i], param);
166 sysfs_read(path, s[i], PARAM_MAXLEN);
167 }
168}
169
170static void cpu_interactive_write(const char *param, char s[CLUSTER_COUNT][PARAM_MAXLEN])
171{
172 char path[PATH_MAX];
173
174 for (unsigned int i = 0; i < ARRAY_SIZE(CPU_INTERACTIVE_PATHS); i++) {
175 sprintf(path, "%s%s", CPU_INTERACTIVE_PATHS[i], param);
176 sysfs_write(path, s[i]);
177 }
178}
179
Christopher N. Hessefb5669e2017-07-08 12:44:39 +0200180static void send_boost(int boost_fd, int32_t duration_us)
Christopher N. Hesse5fada9b2017-01-16 23:39:48 +0100181{
Christopher N. Hessefb5669e2017-07-08 12:44:39 +0200182 int len;
Christopher N. Hesse5fada9b2017-01-16 23:39:48 +0100183
Christopher N. Hessefb5669e2017-07-08 12:44:39 +0200184 if (boost_fd < 0) {
Christopher N. Hesse5fada9b2017-01-16 23:39:48 +0100185 return;
186 }
187
Christopher N. Hessefb5669e2017-07-08 12:44:39 +0200188 len = write(boost_fd, "1", 1);
189 if (len < 0) {
190 ALOGE("Error writing to %s%s: %s", CPU_INTERACTIVE_PATHS[0], BOOST_PATH, strerror(errno));
191 return;
192 }
193
Christopher N. Hesse5fada9b2017-01-16 23:39:48 +0100194 usleep(duration_us);
Christopher N. Hessefb5669e2017-07-08 12:44:39 +0200195 len = write(boost_fd, "0", 1);
Christopher N. Hessef21fbdb2017-03-04 01:41:58 +0100196}
197
198static void send_boostpulse(int boostpulse_fd)
199{
200 int len;
201
202 if (boostpulse_fd < 0) {
203 return;
204 }
205
206 len = write(boostpulse_fd, "1", 1);
207 if (len < 0) {
Christopher N. Hesse7ad73bc2017-06-28 22:53:03 +0200208 ALOGE("Error writing to %s%s: %s", CPU_INTERACTIVE_PATHS[0], BOOSTPULSE_PATH,
209 strerror(errno));
Christopher N. Hessef21fbdb2017-03-04 01:41:58 +0100210 }
211}
212
Jan Altensen39e99622019-11-26 20:34:28 +0100213static int get_cur_panel_brightness() {
214 int ret = 0;
215 int read_status;
216 // brightness can range from 0 to 255, so max. 3 chars + '\0'
217 char panel_brightness[4];
218 // for strtol
219 char *dummy;
220 const int base = 10;
221
222 read_status = sysfs_read(PANEL_BRIGHTNESS_NODE, panel_brightness, sizeof(PANEL_BRIGHTNESS_NODE));
223 if (read_status < 0) {
224 ALOGE("%s: Failed to read panel brightness from %s!\n", __func__, PANEL_BRIGHTNESS_NODE);
225 return -1;
226 }
227
228 ret = strtol(panel_brightness, &dummy, base);
229 ALOGV("%s: Panel brightness is: %d", __func__, ret);
230
231 return ret;
232}
233
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100234/**********************************************************
235 *** POWER FUNCTIONS
236 **********************************************************/
237
Andreas Schneiderf15d7f42016-02-03 10:43:47 +0100238static void set_power_profile(struct samsung_power_module *samsung_pwr,
Christopher N. Hesse58f2ca02017-01-17 00:01:15 +0100239 int profile)
Andreas Schneiderf15d7f42016-02-03 10:43:47 +0100240{
Christopher N. Hesse58f2ca02017-01-17 00:01:15 +0100241 if (profile < 0 || profile >= PROFILE_MAX) {
242 return;
243 }
244
Andreas Schneiderf15d7f42016-02-03 10:43:47 +0100245 if (current_power_profile == profile) {
246 return;
247 }
248
249 ALOGV("%s: profile=%d", __func__, profile);
250
251 switch (profile) {
252 case PROFILE_POWER_SAVE:
Martin Bouchet4bf8a792017-03-28 04:00:04 -0300253 // Grab value set by init.*.rc
Christopher N. Hesse7ad73bc2017-06-28 22:53:03 +0200254 cpu_interactive_read(HISPEED_FREQ_PATH, samsung_pwr->hispeed_freqs);
Andreas Schneiderf15d7f42016-02-03 10:43:47 +0100255 // Limit to hispeed freq
Christopher N. Hesse7ad73bc2017-06-28 22:53:03 +0200256 cpu_sysfs_write(MAX_FREQ_PATH, samsung_pwr->hispeed_freqs);
Christopher N. Hessef05f9022017-01-16 22:49:06 +0100257 ALOGV("%s: set powersave mode", __func__);
Andreas Schneiderf15d7f42016-02-03 10:43:47 +0100258 break;
259 case PROFILE_BALANCED:
260 // Restore normal max freq
Christopher N. Hesse7ad73bc2017-06-28 22:53:03 +0200261 cpu_sysfs_write(MAX_FREQ_PATH, samsung_pwr->max_freqs);
Christopher N. Hessef05f9022017-01-16 22:49:06 +0100262 ALOGV("%s: set balanced mode", __func__);
Andreas Schneiderf15d7f42016-02-03 10:43:47 +0100263 break;
264 case PROFILE_HIGH_PERFORMANCE:
265 // Restore normal max freq
Christopher N. Hesse7ad73bc2017-06-28 22:53:03 +0200266 cpu_sysfs_write(MAX_FREQ_PATH, samsung_pwr->max_freqs);
Christopher N. Hessef05f9022017-01-16 22:49:06 +0100267 ALOGV("%s: set performance mode", __func__);
Andreas Schneiderf15d7f42016-02-03 10:43:47 +0100268 break;
Christopher N. Hesse2981f792017-03-12 17:08:47 +0100269 default:
270 ALOGW("%s: Unknown power profile: %d", __func__, profile);
271 return;
Andreas Schneiderf15d7f42016-02-03 10:43:47 +0100272 }
273
274 current_power_profile = profile;
275}
276
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100277static void find_input_nodes(struct samsung_power_module *samsung_pwr, char *dir)
278{
279 const char filename[] = "name";
280 char errno_str[64];
281 struct dirent *de;
282 char file_content[20];
283 char *path = NULL;
284 char *node_path = NULL;
285 size_t pathsize;
286 size_t node_pathsize;
287 DIR *d;
288
289 d = opendir(dir);
290 if (d == NULL) {
291 return;
292 }
293
294 while ((de = readdir(d)) != NULL) {
295 if (strncmp(filename, de->d_name, sizeof(filename)) == 0) {
296 pathsize = strlen(dir) + strlen(de->d_name) + 2;
297 node_pathsize = strlen(dir) + strlen("enabled") + 2;
298
299 path = malloc(pathsize);
300 node_path = malloc(node_pathsize);
301 if (path == NULL || node_path == NULL) {
302 strerror_r(errno, errno_str, sizeof(errno_str));
Christopher N. Hesse354f7132017-03-04 01:06:22 +0100303 ALOGE("Out of memory: %s", errno_str);
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100304 return;
305 }
306
307 snprintf(path, pathsize, "%s/%s", dir, filename);
308 sysfs_read(path, file_content, sizeof(file_content));
309
310 snprintf(node_path, node_pathsize, "%s/%s", dir, "enabled");
311
312 if (strncmp(file_content, "sec_touchkey", 12) == 0) {
Christopher N. Hesse354f7132017-03-04 01:06:22 +0100313 ALOGV("%s: found touchkey path: %s", __func__, node_path);
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100314 samsung_pwr->touchkey_power_path = malloc(node_pathsize);
315 if (samsung_pwr->touchkey_power_path == NULL) {
316 strerror_r(errno, errno_str, sizeof(errno_str));
Christopher N. Hesse354f7132017-03-04 01:06:22 +0100317 ALOGE("Out of memory: %s", errno_str);
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100318 return;
319 }
Christopher N. Hesse22da3132016-02-01 12:36:54 +0100320 snprintf(samsung_pwr->touchkey_power_path, node_pathsize,
321 "%s", node_path);
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100322 }
323
324 if (strncmp(file_content, "sec_touchscreen", 15) == 0) {
Christopher N. Hesse354f7132017-03-04 01:06:22 +0100325 ALOGV("%s: found touchscreen path: %s", __func__, node_path);
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100326 samsung_pwr->touchscreen_power_path = malloc(node_pathsize);
327 if (samsung_pwr->touchscreen_power_path == NULL) {
328 strerror_r(errno, errno_str, sizeof(errno_str));
Christopher N. Hesse354f7132017-03-04 01:06:22 +0100329 ALOGE("Out of memory: %s", errno_str);
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100330 return;
331 }
Christopher N. Hesse22da3132016-02-01 12:36:54 +0100332 snprintf(samsung_pwr->touchscreen_power_path, node_pathsize,
333 "%s", node_path);
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100334 }
335 }
336 }
337
338 if (path)
339 free(path);
340 if (node_path)
341 free(node_path);
342 closedir(d);
343}
344
345/**********************************************************
346 *** INIT FUNCTIONS
347 **********************************************************/
348
349static void init_cpufreqs(struct samsung_power_module *samsung_pwr)
350{
Christopher N. Hesse7ad73bc2017-06-28 22:53:03 +0200351 cpu_interactive_read(HISPEED_FREQ_PATH, samsung_pwr->hispeed_freqs);
352 cpu_sysfs_read(MAX_FREQ_PATH, samsung_pwr->max_freqs);
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100353}
354
355static void init_touch_input_power_path(struct samsung_power_module *samsung_pwr)
356{
357 char dir[1024];
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100358 uint32_t i;
359
360 for (i = 0; i < 20; i++) {
361 snprintf(dir, sizeof(dir), "/sys/class/input/input%d", i);
362 find_input_nodes(samsung_pwr, dir);
363 }
364}
365
Christopher N. Hessefb5669e2017-07-08 12:44:39 +0200366static void boost_open(struct samsung_power_module *samsung_pwr)
367{
368 char path[PATH_MAX];
369
370 // the boost node is only valid for the LITTLE cluster
371 sprintf(path, "%s%s", CPU_INTERACTIVE_PATHS[0], BOOST_PATH);
372
373 samsung_pwr->boost_fd = open(path, O_WRONLY);
374 if (samsung_pwr->boost_fd < 0) {
375 ALOGE("Error opening %s: %s\n", path, strerror(errno));
376 }
377}
378
379static void boostpulse_open(struct samsung_power_module *samsung_pwr)
380{
381 char path[PATH_MAX];
382
383 // the boostpulse node is only valid for the LITTLE cluster
384 sprintf(path, "%s%s", CPU_INTERACTIVE_PATHS[0], BOOSTPULSE_PATH);
385
386 samsung_pwr->boostpulse_fd = open(path, O_WRONLY);
387 if (samsung_pwr->boostpulse_fd < 0) {
388 ALOGE("Error opening %s: %s\n", path, strerror(errno));
389 }
390}
391
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100392static void samsung_power_init(struct power_module *module)
393{
394 struct samsung_power_module *samsung_pwr = (struct samsung_power_module *) module;
395
396 init_cpufreqs(samsung_pwr);
Christopher N. Hessed9106e92017-03-04 01:05:14 +0100397
Christopher N. Hessefb5669e2017-07-08 12:44:39 +0200398 // keep interactive boost fds opened
399 boost_open(samsung_pwr);
Christopher N. Hessed9106e92017-03-04 01:05:14 +0100400 boostpulse_open(samsung_pwr);
401
Christopher N. Hesse65c65bd2017-03-07 23:01:12 +0100402 samsung_pwr->touchscreen_power_path = NULL;
403 samsung_pwr->touchkey_power_path = NULL;
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100404 init_touch_input_power_path(samsung_pwr);
Christopher N. Hesse775bce82017-07-08 13:46:35 +0200405
406 ALOGI("Initialized settings:");
Christopher N. Hesse1146f1a2017-07-13 21:14:57 +0200407 char max_freqs[PATH_MAX];
408 sprintf(max_freqs, "max_freqs: cluster[0]: %s", samsung_pwr->max_freqs[0]);
409 for (unsigned int i = 1; i < CLUSTER_COUNT; i++) {
410 sprintf(max_freqs, "%s, %s[%d]: %s", max_freqs, "cluster", i, samsung_pwr->max_freqs[i]);
411 }
412 ALOGI("%s", max_freqs);
413 char hispeed_freqs[PATH_MAX];
414 sprintf(hispeed_freqs, "hispeed_freqs: cluster[0]: %s", samsung_pwr->hispeed_freqs[0]);
415 for (unsigned int i = 1; i < CLUSTER_COUNT; i++) {
416 sprintf(hispeed_freqs, "%s, %s[%d]: %s", hispeed_freqs, "cluster", i,
417 samsung_pwr->hispeed_freqs[i]);
418 }
419 ALOGI("%s", hispeed_freqs);
Christopher N. Hesseff868462017-07-09 23:07:09 +0200420 ALOGI("boostpulse_fd: %d", samsung_pwr->boostpulse_fd);
Christopher N. Hesse775bce82017-07-08 13:46:35 +0200421 ALOGI("touchscreen_power_path: %s",
422 samsung_pwr->touchscreen_power_path ? samsung_pwr->touchscreen_power_path : "NULL");
423 ALOGI("touchkey_power_path: %s",
424 samsung_pwr->touchkey_power_path ? samsung_pwr->touchkey_power_path : "NULL");
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100425}
426
Christopher N. Hessef6296722017-01-16 22:47:11 +0100427/**********************************************************
428 *** API FUNCTIONS
429 ***
430 *** Refer to power.h for documentation.
431 **********************************************************/
432
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100433static void samsung_power_set_interactive(struct power_module *module, int on)
434{
435 struct samsung_power_module *samsung_pwr = (struct samsung_power_module *) module;
Christopher N. Hessed9106e92017-03-04 01:05:14 +0100436 int panel_brightness;
437 char button_state[2];
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100438 int rc;
Christopher N. Hessed9106e92017-03-04 01:05:14 +0100439 static bool touchkeys_blocked = false;
Christopher N. Hesse7ad73bc2017-06-28 22:53:03 +0200440 char ON[CLUSTER_COUNT][PARAM_MAXLEN] = {"1", "1"};
441 char OFF[CLUSTER_COUNT][PARAM_MAXLEN] = {"0", "0"};
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100442
Christopher N. Hesse354f7132017-03-04 01:06:22 +0100443 ALOGV("power_set_interactive: %d", on);
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100444
Christopher N. Hessed9106e92017-03-04 01:05:14 +0100445 /*
446 * Do not disable any input devices if the screen is on but we are in a non-interactive
447 * state.
448 */
Christopher N. Hesse3360b092016-07-11 15:48:35 +0200449 if (!on) {
Christopher N. Hessed9106e92017-03-04 01:05:14 +0100450 panel_brightness = get_cur_panel_brightness();
451 if (panel_brightness < 0) {
452 ALOGE("%s: Failed to read panel brightness", __func__);
453 } else if (panel_brightness > 0) {
Christopher N. Hesse3360b092016-07-11 15:48:35 +0200454 ALOGV("%s: Moving to non-interactive state, but screen is still on,"
Christopher N. Hesse354f7132017-03-04 01:06:22 +0100455 " not disabling input devices", __func__);
Christopher N. Hesse3360b092016-07-11 15:48:35 +0200456 goto out;
457 }
458 }
459
Paul Keith5e4fbe02017-04-21 21:34:17 -0500460 /* Sanity check the touchscreen path */
461 if (samsung_pwr->touchscreen_power_path) {
462 sysfs_write(samsung_pwr->touchscreen_power_path, on ? "1" : "0");
463 }
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100464
Christopher N. Hessed9106e92017-03-04 01:05:14 +0100465 /* Bail out if the device does not have touchkeys */
466 if (samsung_pwr->touchkey_power_path == NULL) {
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100467 goto out;
468 }
469
470 if (!on) {
Christopher N. Hessed9106e92017-03-04 01:05:14 +0100471 rc = sysfs_read(samsung_pwr->touchkey_power_path, button_state, ARRAY_SIZE(button_state));
472 if (rc < 0) {
473 ALOGE("%s: Failed to read touchkey state", __func__);
474 goto out;
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100475 }
Christopher N. Hessed9106e92017-03-04 01:05:14 +0100476 /*
477 * If button_state is 0, the keys have been disabled by another component
478 * (for example cmhw), which means we don't want them to be enabled when resuming
479 * from suspend.
480 */
481 if (button_state[0] == '0') {
482 touchkeys_blocked = true;
483 } else {
484 touchkeys_blocked = false;
485 }
486 }
487
488 if (!touchkeys_blocked) {
489 sysfs_write(samsung_pwr->touchkey_power_path, on ? "1" : "0");
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100490 }
491
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100492out:
Christopher N. Hesse7ad73bc2017-06-28 22:53:03 +0200493 cpu_interactive_write(IO_IS_BUSY_PATH, on ? ON : OFF);
Christopher N. Hesse63b5bd72017-05-29 20:45:14 +0200494
Christopher N. Hesse354f7132017-03-04 01:06:22 +0100495 ALOGV("power_set_interactive: %d done", on);
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100496}
497
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100498static void samsung_power_hint(struct power_module *module,
499 power_hint_t hint,
500 void *data)
501{
502 struct samsung_power_module *samsung_pwr = (struct samsung_power_module *) module;
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100503
Christopher N. Hesse15b63092017-03-12 17:04:53 +0100504 /* Bail out if low-power mode is active */
Christopher N. Hesse5eca5aa2017-04-26 21:42:10 +0200505 if (current_power_profile == PROFILE_POWER_SAVE && hint != POWER_HINT_LOW_POWER
Christopher N. Hesse62b70f82018-02-19 23:20:14 +0100506 && hint != POWER_HINT_SET_PROFILE && hint != POWER_HINT_DISABLE_TOUCH) {
Christopher N. Hesse4f7dc6f2018-02-19 23:17:09 +0100507 ALOGV("%s: PROFILE_POWER_SAVE active, ignoring hint %d", __func__, hint);
Christopher N. Hesse15b63092017-03-12 17:04:53 +0100508 return;
509 }
510
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100511 switch (hint) {
Christopher N. Hesse2981f792017-03-12 17:08:47 +0100512 case POWER_HINT_VSYNC:
513 ALOGV("%s: POWER_HINT_VSYNC", __func__);
514 break;
515 case POWER_HINT_INTERACTION:
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100516 ALOGV("%s: POWER_HINT_INTERACTION", __func__);
Christopher N. Hessef21fbdb2017-03-04 01:41:58 +0100517 send_boostpulse(samsung_pwr->boostpulse_fd);
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100518 break;
Christopher N. Hesse2981f792017-03-12 17:08:47 +0100519 case POWER_HINT_LOW_POWER:
520 ALOGV("%s: POWER_HINT_LOW_POWER", __func__);
Christopher N. Hesse3fa2b692017-04-05 18:52:48 +0200521 set_power_profile(samsung_pwr, data ? PROFILE_POWER_SAVE : PROFILE_BALANCED);
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100522 break;
Christopher N. Hesse2981f792017-03-12 17:08:47 +0100523 case POWER_HINT_LAUNCH:
Christopher N. Hesse3fa2b692017-04-05 18:52:48 +0200524 ALOGV("%s: POWER_HINT_LAUNCH", __func__);
525 send_boostpulse(samsung_pwr->boostpulse_fd);
526 break;
Christopher N. Hesse5fada9b2017-01-16 23:39:48 +0100527 case POWER_HINT_CPU_BOOST:
Christopher N. Hesse3fa2b692017-04-05 18:52:48 +0200528 ALOGV("%s: POWER_HINT_CPU_BOOST", __func__);
Christopher N. Hessefb5669e2017-07-08 12:44:39 +0200529 int32_t duration_us = *((int32_t *)data);
530 send_boost(samsung_pwr->boost_fd, duration_us);
Christopher N. Hesse5fada9b2017-01-16 23:39:48 +0100531 break;
Christopher N. Hesse2981f792017-03-12 17:08:47 +0100532 case POWER_HINT_SET_PROFILE:
Andreas Schneiderf15d7f42016-02-03 10:43:47 +0100533 ALOGV("%s: POWER_HINT_SET_PROFILE", __func__);
Christopher N. Hesse2981f792017-03-12 17:08:47 +0100534 int profile = *((intptr_t *)data);
Andreas Schneiderf15d7f42016-02-03 10:43:47 +0100535 set_power_profile(samsung_pwr, profile);
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100536 break;
Christopher N. Hessef9b8e832017-01-17 00:04:45 +0100537 case POWER_HINT_DISABLE_TOUCH:
538 ALOGV("%s: POWER_HINT_DISABLE_TOUCH", __func__);
Christopher N. Hesse3fa2b692017-04-05 18:52:48 +0200539 sysfs_write(samsung_pwr->touchscreen_power_path, data ? "0" : "1");
Christopher N. Hessef9b8e832017-01-17 00:04:45 +0100540 break;
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100541 default:
Christopher N. Hesse2981f792017-03-12 17:08:47 +0100542 ALOGW("%s: Unknown power hint: %d", __func__, hint);
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100543 break;
544 }
545}
546
Stricted1a871752018-08-28 07:30:40 +0200547static void samsung_set_feature(struct power_module *module __unused, feature_t feature, int state __unused)
Christopher N. Hessee480d892016-06-22 23:04:39 +0200548{
Christopher N. Hessee480d892016-06-22 23:04:39 +0200549 switch (feature) {
ishantvivek987dcca2016-11-21 06:05:47 +0000550#ifdef TARGET_TAP_TO_WAKE_NODE
Christopher N. Hessee480d892016-06-22 23:04:39 +0200551 case POWER_FEATURE_DOUBLE_TAP_TO_WAKE:
552 ALOGV("%s: %s double tap to wake", __func__, state ? "enabling" : "disabling");
ishantvivek987dcca2016-11-21 06:05:47 +0000553 sysfs_write(TARGET_TAP_TO_WAKE_NODE, state > 0 ? "1" : "0");
Christopher N. Hessee480d892016-06-22 23:04:39 +0200554 break;
555#endif
556 default:
557 break;
558 }
559}
560
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100561static struct hw_module_methods_t power_module_methods = {
562 .open = NULL,
563};
564
565struct samsung_power_module HAL_MODULE_INFO_SYM = {
566 .base = {
567 .common = {
568 .tag = HARDWARE_MODULE_TAG,
569 .module_api_version = POWER_MODULE_API_VERSION_0_2,
570 .hal_api_version = HARDWARE_HAL_API_VERSION,
571 .id = POWER_HARDWARE_MODULE_ID,
572 .name = "Samsung Power HAL",
Christopher N. Hesseaa75be42017-01-16 22:47:59 +0100573 .author = "The LineageOS Project",
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100574 .methods = &power_module_methods,
575 },
576
577 .init = samsung_power_init,
578 .setInteractive = samsung_power_set_interactive,
579 .powerHint = samsung_power_hint,
Christopher N. Hessee480d892016-06-22 23:04:39 +0200580 .setFeature = samsung_set_feature
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100581 },
582
583 .lock = PTHREAD_MUTEX_INITIALIZER,
584 .boostpulse_fd = -1,
Christopher N. Hessede5e3c62015-12-21 21:28:23 +0100585};