236 lines
7.0 KiB
C++
236 lines
7.0 KiB
C++
/*
|
|
* Copyright (C) 2019 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 <processgroup/sched_policy.h>
|
|
|
|
#define LOG_TAG "SchedPolicy"
|
|
|
|
#include <errno.h>
|
|
#include <unistd.h>
|
|
|
|
#include <android-base/logging.h>
|
|
#include <android-base/threads.h>
|
|
#include <cgroup_map.h>
|
|
#include <processgroup/processgroup.h>
|
|
|
|
using android::base::GetThreadId;
|
|
|
|
/* Re-map SP_DEFAULT to the system default policy, and leave other values unchanged.
|
|
* Call this any place a SchedPolicy is used as an input parameter.
|
|
* Returns the possibly re-mapped policy.
|
|
*/
|
|
static inline SchedPolicy _policy(SchedPolicy p) {
|
|
return p == SP_DEFAULT ? SP_SYSTEM_DEFAULT : p;
|
|
}
|
|
|
|
#if defined(__ANDROID__)
|
|
|
|
int set_cpuset_policy(int tid, SchedPolicy policy) {
|
|
if (tid == 0) {
|
|
tid = GetThreadId();
|
|
}
|
|
policy = _policy(policy);
|
|
|
|
switch (policy) {
|
|
case SP_BACKGROUND:
|
|
return SetTaskProfiles(tid,
|
|
{"HighEnergySaving", "ProcessCapacityLow", "LowIoPriority",
|
|
"TimerSlackHigh"},
|
|
true)
|
|
? 0
|
|
: -1;
|
|
case SP_FOREGROUND:
|
|
case SP_AUDIO_APP:
|
|
case SP_AUDIO_SYS:
|
|
return SetTaskProfiles(tid,
|
|
{"HighPerformance", "ProcessCapacityHigh", "HighIoPriority",
|
|
"TimerSlackNormal"},
|
|
true)
|
|
? 0
|
|
: -1;
|
|
case SP_TOP_APP:
|
|
return SetTaskProfiles(tid,
|
|
{"MaxPerformance", "ProcessCapacityMax", "MaxIoPriority",
|
|
"TimerSlackNormal"},
|
|
true)
|
|
? 0
|
|
: -1;
|
|
case SP_SYSTEM:
|
|
return SetTaskProfiles(tid, {"ServiceCapacityLow", "TimerSlackNormal"}, true) ? 0 : -1;
|
|
case SP_RESTRICTED:
|
|
return SetTaskProfiles(tid, {"ServiceCapacityRestricted", "TimerSlackNormal"}, true)
|
|
? 0
|
|
: -1;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int set_sched_policy(int tid, SchedPolicy policy) {
|
|
if (tid == 0) {
|
|
tid = GetThreadId();
|
|
}
|
|
policy = _policy(policy);
|
|
|
|
#if POLICY_DEBUG
|
|
char statfile[64];
|
|
char statline[1024];
|
|
char thread_name[255];
|
|
|
|
snprintf(statfile, sizeof(statfile), "/proc/%d/stat", tid);
|
|
memset(thread_name, 0, sizeof(thread_name));
|
|
|
|
unique_fd fd(TEMP_FAILURE_RETRY(open(statfile, O_RDONLY | O_CLOEXEC)));
|
|
if (fd >= 0) {
|
|
int rc = read(fd, statline, 1023);
|
|
statline[rc] = 0;
|
|
char* p = statline;
|
|
char* q;
|
|
|
|
for (p = statline; *p != '('; p++)
|
|
;
|
|
p++;
|
|
for (q = p; *q != ')'; q++)
|
|
;
|
|
|
|
strncpy(thread_name, p, (q - p));
|
|
}
|
|
switch (policy) {
|
|
case SP_BACKGROUND:
|
|
SLOGD("vvv tid %d (%s)", tid, thread_name);
|
|
break;
|
|
case SP_FOREGROUND:
|
|
case SP_AUDIO_APP:
|
|
case SP_AUDIO_SYS:
|
|
case SP_TOP_APP:
|
|
SLOGD("^^^ tid %d (%s)", tid, thread_name);
|
|
break;
|
|
case SP_SYSTEM:
|
|
SLOGD("/// tid %d (%s)", tid, thread_name);
|
|
break;
|
|
case SP_RT_APP:
|
|
SLOGD("RT tid %d (%s)", tid, thread_name);
|
|
break;
|
|
default:
|
|
SLOGD("??? tid %d (%s)", tid, thread_name);
|
|
break;
|
|
}
|
|
#endif
|
|
|
|
switch (policy) {
|
|
case SP_BACKGROUND:
|
|
return SetTaskProfiles(tid, {"HighEnergySaving", "TimerSlackHigh"}, true) ? 0 : -1;
|
|
case SP_FOREGROUND:
|
|
case SP_AUDIO_APP:
|
|
case SP_AUDIO_SYS:
|
|
return SetTaskProfiles(tid, {"HighPerformance", "TimerSlackNormal"}, true) ? 0 : -1;
|
|
case SP_TOP_APP:
|
|
return SetTaskProfiles(tid, {"MaxPerformance", "TimerSlackNormal"}, true) ? 0 : -1;
|
|
case SP_RT_APP:
|
|
return SetTaskProfiles(tid, {"RealtimePerformance", "TimerSlackNormal"}, true) ? 0 : -1;
|
|
default:
|
|
return SetTaskProfiles(tid, {"TimerSlackNormal"}, true) ? 0 : -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
bool cpusets_enabled() {
|
|
static bool enabled = (CgroupMap::GetInstance().FindController("cpuset").HasValue());
|
|
return enabled;
|
|
}
|
|
|
|
bool schedboost_enabled() {
|
|
static bool enabled = (CgroupMap::GetInstance().FindController("schedtune").HasValue());
|
|
return enabled;
|
|
}
|
|
|
|
static int getCGroupSubsys(int tid, const char* subsys, std::string& subgroup) {
|
|
auto controller = CgroupMap::GetInstance().FindController(subsys);
|
|
|
|
if (!controller.HasValue()) return -1;
|
|
|
|
if (!controller.GetTaskGroup(tid, &subgroup)) {
|
|
LOG(ERROR) << "Failed to find cgroup for tid " << tid;
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int get_sched_policy(int tid, SchedPolicy* policy) {
|
|
if (tid == 0) {
|
|
tid = GetThreadId();
|
|
}
|
|
|
|
std::string group;
|
|
if (schedboost_enabled()) {
|
|
if (getCGroupSubsys(tid, "schedtune", group) < 0) return -1;
|
|
}
|
|
if (group.empty() && cpusets_enabled()) {
|
|
if (getCGroupSubsys(tid, "cpuset", group) < 0) return -1;
|
|
}
|
|
|
|
// TODO: replace hardcoded directories
|
|
if (group.empty()) {
|
|
*policy = SP_FOREGROUND;
|
|
} else if (group == "foreground") {
|
|
*policy = SP_FOREGROUND;
|
|
} else if (group == "system-background") {
|
|
*policy = SP_SYSTEM;
|
|
} else if (group == "background") {
|
|
*policy = SP_BACKGROUND;
|
|
} else if (group == "top-app") {
|
|
*policy = SP_TOP_APP;
|
|
} else if (group == "restricted") {
|
|
*policy = SP_RESTRICTED;
|
|
} else {
|
|
errno = ERANGE;
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
#else
|
|
|
|
/* Stubs for non-Android targets. */
|
|
|
|
int set_sched_policy(int, SchedPolicy) {
|
|
return 0;
|
|
}
|
|
|
|
int get_sched_policy(int, SchedPolicy* policy) {
|
|
*policy = SP_SYSTEM_DEFAULT;
|
|
return 0;
|
|
}
|
|
|
|
#endif
|
|
|
|
const char* get_sched_policy_name(SchedPolicy policy) {
|
|
policy = _policy(policy);
|
|
static const char* const kSchedPolicyNames[] = {
|
|
[SP_BACKGROUND] = "bg", [SP_FOREGROUND] = "fg", [SP_SYSTEM] = " ",
|
|
[SP_AUDIO_APP] = "aa", [SP_AUDIO_SYS] = "as", [SP_TOP_APP] = "ta",
|
|
[SP_RT_APP] = "rt", [SP_RESTRICTED] = "rs",
|
|
};
|
|
static_assert(arraysize(kSchedPolicyNames) == SP_CNT, "missing name");
|
|
if (policy < SP_BACKGROUND || policy >= SP_CNT) {
|
|
return "error";
|
|
}
|
|
return kSchedPolicyNames[policy];
|
|
}
|