blob: 327cf06595b1c25872304f39204cd4fc8254963d [file] [log] [blame]
/*
* Copyright (C) 2012 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.
*/
#include <errno.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/socket.h>
#include <cutils/uevent.h>
#include <sys/poll.h>
#include <pthread.h>
#include <linux/netlink.h>
#include <stdlib.h>
#include <stdbool.h>
#define LOG_TAG "Grouper PowerHAL"
#include <utils/Log.h>
#include <hardware/hardware.h>
#include <hardware/power.h>
#define BOOST_PATH "/sys/devices/system/cpu/cpufreq/interactive/boost"
#define UEVENT_MSG_LEN 2048
#define TOTAL_CPUS 4
#define RETRY_TIME_CHANGING_FREQ 20
#define SLEEP_USEC_BETWN_RETRY 200
#define LOW_POWER_MAX_FREQ "640000"
#define LOW_POWER_MIN_FREQ "51000"
#define NORMAL_MAX_FREQ "1300000"
#define UEVENT_STRING "online@/devices/system/cpu/"
static int boost_fd = -1;
static int boost_warned;
static struct pollfd pfd;
static char *cpu_path_min[] = {
"/sys/devices/system/cpu/cpu0/cpufreq/scaling_min_freq",
"/sys/devices/system/cpu/cpu1/cpufreq/scaling_min_freq",
"/sys/devices/system/cpu/cpu2/cpufreq/scaling_min_freq",
"/sys/devices/system/cpu/cpu3/cpufreq/scaling_min_freq",
};
static char *cpu_path_max[] = {
"/sys/devices/system/cpu/cpu0/cpufreq/scaling_max_freq",
"/sys/devices/system/cpu/cpu1/cpufreq/scaling_max_freq",
"/sys/devices/system/cpu/cpu2/cpufreq/scaling_max_freq",
"/sys/devices/system/cpu/cpu3/cpufreq/scaling_max_freq",
};
static bool freq_set[TOTAL_CPUS];
static bool low_power_mode = false;
static pthread_mutex_t low_power_mode_lock = PTHREAD_MUTEX_INITIALIZER;
static int sysfs_write(char *path, char *s)
{
char buf[80];
int len;
int fd = open(path, O_WRONLY);
if (fd < 0) {
strerror_r(errno, buf, sizeof(buf));
ALOGE("Error opening %s: %s\n", path, buf);
return -1;
}
len = write(fd, s, strlen(s));
if (len < 0) {
strerror_r(errno, buf, sizeof(buf));
ALOGE("Error writing to %s: %s\n", path, buf);
return -1;
}
close(fd);
return 0;
}
static int uevent_event()
{
char msg[UEVENT_MSG_LEN];
char *cp;
int n, cpu, ret, retry = RETRY_TIME_CHANGING_FREQ;
n = recv(pfd.fd, msg, UEVENT_MSG_LEN, MSG_DONTWAIT);
if (n <= 0) {
return -1;
}
if (n >= UEVENT_MSG_LEN) { /* overflow -- discard */
return -1;
}
cp = msg;
if (strstr(cp, UEVENT_STRING)) {
n = strlen(cp);
errno = 0;
cpu = strtol(cp + n - 1, NULL, 10);
if (errno == EINVAL || errno == ERANGE || cpu < 0 || cpu >= TOTAL_CPUS) {
return -1;
}
pthread_mutex_lock(&low_power_mode_lock);
if (low_power_mode && !freq_set[cpu]) {
while (retry) {
sysfs_write(cpu_path_min[cpu], LOW_POWER_MIN_FREQ);
ret = sysfs_write(cpu_path_max[cpu], LOW_POWER_MAX_FREQ);
if (!ret) {
freq_set[cpu] = true;
break;
}
usleep(SLEEP_USEC_BETWN_RETRY);
retry--;
}
} else if (!low_power_mode && freq_set[cpu]) {
while (retry) {
ret = sysfs_write(cpu_path_max[cpu], NORMAL_MAX_FREQ);
if (!ret) {
freq_set[cpu] = false;
break;
}
usleep(SLEEP_USEC_BETWN_RETRY);
retry--;
}
}
pthread_mutex_unlock(&low_power_mode_lock);
}
return 0;
}
void *thread_uevent(__attribute__((unused)) void *x)
{
while (1) {
int nevents, ret;
nevents = poll(&pfd, 1, -1);
if (nevents == -1) {
if (errno == EINTR)
continue;
ALOGE("powerhal: thread_uevent: poll_wait failed\n");
break;
}
ret = uevent_event();
if (ret < 0)
ALOGE("Error processing the uevent event");
}
return NULL;
}
static void uevent_init()
{
struct sockaddr_nl client;
pthread_t tid;
pfd.fd = socket(PF_NETLINK, SOCK_DGRAM, NETLINK_KOBJECT_UEVENT);
if (pfd.fd < 0) {
ALOGE("%s: failed to open: %s", __func__, strerror(errno));
return;
}
memset(&client, 0, sizeof(struct sockaddr_nl));
pthread_create(&tid, NULL, thread_uevent, NULL);
client.nl_family = AF_NETLINK;
client.nl_pid = tid;
client.nl_groups = -1;
pfd.events = POLLIN;
bind(pfd.fd, (void *)&client, sizeof(struct sockaddr_nl));
return;
}
static void grouper_power_init( __attribute__((unused)) struct power_module *module)
{
/*
* cpufreq interactive governor: timer 20ms, min sample 100ms,
* hispeed 700MHz at load 40%
*/
sysfs_write("/sys/devices/system/cpu/cpufreq/interactive/timer_rate",
"50000");
sysfs_write("/sys/devices/system/cpu/cpufreq/interactive/min_sample_time",
"500000");
sysfs_write("/sys/devices/system/cpu/cpufreq/interactive/go_hispeed_load",
"75");
sysfs_write("/sys/devices/system/cpu/cpufreq/interactive/boost_factor",
"0");
sysfs_write("/sys/devices/system/cpu/cpufreq/interactive/input_boost",
"1");
uevent_init();
}
static void grouper_power_set_interactive(__attribute__((unused)) struct power_module *module,
__attribute__((unused)) int on)
{
if (on) {
sysfs_write("/sys/devices/system/cpu/cpufreq/interactive/go_hispeed_load", "75");
sysfs_write("/sys/devices/system/cpu/cpufreq/interactive/core_lock_period", "3000000");
sysfs_write("/sys/devices/system/cpu/cpufreq/interactive/core_lock_count", "2");
sysfs_write("/sys/devices/system/cpu/cpufreq/interactive/input_boost", "1");
}
else {
sysfs_write("/sys/devices/system/cpu/cpufreq/interactive/go_hispeed_load", "85");
sysfs_write("/sys/devices/system/cpu/cpufreq/interactive/core_lock_period", "200000");
sysfs_write("/sys/devices/system/cpu/cpufreq/interactive/core_lock_count", "0");
sysfs_write("/sys/devices/system/cpu/cpufreq/interactive/input_boost", "0");
}
}
static void grouper_power_hint(__attribute__((unused)) struct power_module *module, power_hint_t hint,
void *data)
{
char buf[80];
int len, cpu, ret;
switch (hint) {
case POWER_HINT_VSYNC:
break;
case POWER_HINT_LOW_POWER:
pthread_mutex_lock(&low_power_mode_lock);
if (data) {
low_power_mode = true;
for (cpu = 0; cpu < TOTAL_CPUS; cpu++) {
sysfs_write(cpu_path_min[cpu], LOW_POWER_MIN_FREQ);
ret = sysfs_write(cpu_path_max[cpu], LOW_POWER_MAX_FREQ);
if (!ret) {
freq_set[cpu] = true;
}
}
} else {
low_power_mode = false;
for (cpu = 0; cpu < TOTAL_CPUS; cpu++) {
ret = sysfs_write(cpu_path_max[cpu], NORMAL_MAX_FREQ);
if (!ret) {
freq_set[cpu] = false;
}
}
}
pthread_mutex_unlock(&low_power_mode_lock);
break;
default:
break;
}
}
static struct hw_module_methods_t power_module_methods = {
.open = NULL,
};
struct power_module HAL_MODULE_INFO_SYM = {
.common = {
.tag = HARDWARE_MODULE_TAG,
.module_api_version = POWER_MODULE_API_VERSION_0_2,
.hal_api_version = HARDWARE_HAL_API_VERSION,
.id = POWER_HARDWARE_MODULE_ID,
.name = "Grouper Power HAL",
.author = "The Android Open Source Project",
.methods = &power_module_methods,
},
.init = grouper_power_init,
.setInteractive = grouper_power_set_interactive,
.powerHint = grouper_power_hint,
};