diff options
-rw-r--r-- | power-8084.c | 109 | ||||
-rw-r--r-- | power-8226.c | 77 | ||||
-rw-r--r-- | power-8610.c | 73 | ||||
-rw-r--r-- | power-8916.c | 88 | ||||
-rw-r--r-- | power-8952.c | 92 | ||||
-rw-r--r-- | power-8960.c | 99 | ||||
-rw-r--r-- | power-8974.c | 103 | ||||
-rw-r--r-- | power-8992.c | 106 | ||||
-rw-r--r-- | power-8994.c | 114 |
9 files changed, 861 insertions, 0 deletions
diff --git a/power-8084.c b/power-8084.c index 4ed01a7..c9d7d13 100644 --- a/power-8084.c +++ b/power-8084.c @@ -1,5 +1,6 @@ /* * Copyright (c) 2014, The Linux Foundation. All rights reserved. + * Copyright (C) 2018-2019 The LineageOS Project * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are @@ -50,6 +51,101 @@ static int first_display_off_hint; +static int current_power_profile = PROFILE_BALANCED; + +// clang-format off +/* power save mode: max 2 CPUs, max 1.2 GHz */ +static int profile_power_save[] = { + 0x0A03, + CPUS_ONLINE_MAX_LIMIT_2, + CPU0_MAX_FREQ_NONTURBO_MAX + 1, + CPU1_MAX_FREQ_NONTURBO_MAX + 1, + CPU2_MAX_FREQ_NONTURBO_MAX + 1, + CPU3_MAX_FREQ_NONTURBO_MAX + 1 +}; + +/* efficiency mode: max 2 CPUs, max 2.4 GHz */ +static int profile_bias_power[] = { + 0x0A03, + CPUS_ONLINE_MAX_LIMIT_2, + CPU0_MAX_FREQ_NONTURBO_MAX + 14, + CPU1_MAX_FREQ_NONTURBO_MAX + 14, + CPU2_MAX_FREQ_NONTURBO_MAX + 14, + CPU3_MAX_FREQ_NONTURBO_MAX + 14, +}; + +/* quick mode: min 2 CPUs, min 1.1 GHz */ +static int profile_bias_performance[] = { + CPUS_ONLINE_MIN_2, + CPU0_MIN_FREQ_NONTURBO_MAX + 1, + CPU1_MIN_FREQ_NONTURBO_MAX + 1, + CPU2_MIN_FREQ_NONTURBO_MAX + 1, + CPU3_MIN_FREQ_NONTURBO_MAX + 1 +}; + +/* performance mode: min 4 CPUs, min 1.5 GHz */ +static int profile_high_performance[] = { + 0x0901, + CPUS_ONLINE_MIN_4, + CPU0_MIN_FREQ_NONTURBO_MAX + 5, + CPU1_MIN_FREQ_NONTURBO_MAX + 5, + CPU2_MIN_FREQ_NONTURBO_MAX + 5, + CPU3_MIN_FREQ_NONTURBO_MAX + 5 +}; +// clang-format on + +#ifdef INTERACTION_BOOST +int get_number_of_profiles() { + return 5; +} +#endif + +static int set_power_profile(void* data) { + int profile = data ? *((int*)data) : 0; + int ret = -EINVAL; + const char* profile_name = NULL; + + if (profile == current_power_profile) return 0; + + ALOGV("%s: Profile=%d", __func__, profile); + + if (current_power_profile != PROFILE_BALANCED) { + undo_hint_action(DEFAULT_PROFILE_HINT_ID); + ALOGV("%s: Hint undone", __func__); + current_power_profile = PROFILE_BALANCED; + } + + if (profile == PROFILE_POWER_SAVE) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_power_save, + ARRAY_SIZE(profile_power_save)); + profile_name = "powersave"; + + } else if (profile == PROFILE_HIGH_PERFORMANCE) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_high_performance, + ARRAY_SIZE(profile_high_performance)); + profile_name = "performance"; + + } else if (profile == PROFILE_BIAS_POWER) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_bias_power, + ARRAY_SIZE(profile_bias_power)); + profile_name = "bias power"; + + } else if (profile == PROFILE_BIAS_PERFORMANCE) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_bias_performance, + ARRAY_SIZE(profile_bias_performance)); + profile_name = "bias perf"; + } else if (profile == PROFILE_BALANCED) { + ret = 0; + profile_name = "balanced"; + } + + if (ret == 0) { + current_power_profile = profile; + ALOGD("%s: Set %s mode", __func__, profile_name); + } + return ret; +} + static int process_video_encode_hint(void* metadata) { char governor[80]; struct video_encode_metadata_t video_encode_metadata; @@ -222,6 +318,19 @@ static int process_activity_launch_hint(void* data) { int power_hint_override(power_hint_t hint, void* data) { int ret_val = HINT_NONE; + + if (hint == POWER_HINT_SET_PROFILE) { + if (set_power_profile(data) < 0) + ALOGE("Setting power profile failed. mpdecision not started?"); + return HINT_HANDLED; + } + + // Skip other hints in high/low power modes + if (current_power_profile == PROFILE_POWER_SAVE || + current_power_profile == PROFILE_HIGH_PERFORMANCE) { + return HINT_HANDLED; + } + switch (hint) { case POWER_HINT_VIDEO_ENCODE: ret_val = process_video_encode_hint(data); diff --git a/power-8226.c b/power-8226.c index 7f5ff72..99a2e91 100644 --- a/power-8226.c +++ b/power-8226.c @@ -1,5 +1,6 @@ /* * Copyright (c) 2013, The Linux Foundation. All rights reserved. + * Copyright (C) 2018-2019 The LineageOS Project * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are @@ -48,6 +49,69 @@ #include "power-common.h" #include "utils.h" +static int current_power_profile = PROFILE_BALANCED; + +// clang-format off +static int profile_high_performance[] = { + CPUS_ONLINE_MIN_4, + CPU0_MIN_FREQ_TURBO_MAX, + CPU1_MIN_FREQ_TURBO_MAX, + CPU2_MIN_FREQ_TURBO_MAX, + CPU3_MIN_FREQ_TURBO_MAX +}; + +static int profile_power_save[] = { + CPUS_ONLINE_MAX_LIMIT_2, + CPU0_MAX_FREQ_NONTURBO_MAX, + CPU1_MAX_FREQ_NONTURBO_MAX, + CPU2_MAX_FREQ_NONTURBO_MAX, + CPU3_MAX_FREQ_NONTURBO_MAX +}; +// clang-format on + +#ifdef INTERACTION_BOOST +int get_number_of_profiles() { + return 3; +} +#endif + +static int set_power_profile(void* data) { + int profile = data ? *((int*)data) : 0; + int ret = -EINVAL; + const char* profile_name = NULL; + + if (profile == current_power_profile) return 0; + + ALOGV("%s: Profile=%d", __func__, profile); + + if (current_power_profile != PROFILE_BALANCED) { + undo_hint_action(DEFAULT_PROFILE_HINT_ID); + ALOGV("%s: Hint undone", __func__); + current_power_profile = PROFILE_BALANCED; + } + + if (profile == PROFILE_POWER_SAVE) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_power_save, + ARRAY_SIZE(profile_power_save)); + profile_name = "powersave"; + + } else if (profile == PROFILE_HIGH_PERFORMANCE) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_high_performance, + ARRAY_SIZE(profile_high_performance)); + profile_name = "performance"; + + } else if (profile == PROFILE_BALANCED) { + ret = 0; + profile_name = "balanced"; + } + + if (ret == 0) { + current_power_profile = profile; + ALOGD("%s: Set %s mode", __func__, profile_name); + } + return ret; +} + static int process_video_encode_hint(void* metadata) { char governor[80]; struct video_encode_metadata_t video_encode_metadata; @@ -166,6 +230,19 @@ static void process_interaction_hint(void* data) { int power_hint_override(power_hint_t hint, void* data) { int ret_val = HINT_NONE; + + if (hint == POWER_HINT_SET_PROFILE) { + if (set_power_profile(data) < 0) + ALOGE("Setting power profile failed. mpdecision not started?"); + return HINT_HANDLED; + } + + // Skip other hints in high/low power modes + if (current_power_profile == PROFILE_POWER_SAVE || + current_power_profile == PROFILE_HIGH_PERFORMANCE) { + return HINT_HANDLED; + } + switch (hint) { case POWER_HINT_VIDEO_ENCODE: ret_val = process_video_encode_hint(data); diff --git a/power-8610.c b/power-8610.c index 7f5ff72..f5f28ff 100644 --- a/power-8610.c +++ b/power-8610.c @@ -1,5 +1,6 @@ /* * Copyright (c) 2013, The Linux Foundation. All rights reserved. + * Copyright (C) 2018-2019 The LineageOS Project * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are @@ -48,6 +49,65 @@ #include "power-common.h" #include "utils.h" +static int current_power_profile = PROFILE_BALANCED; + +// clang-format off +static int profile_high_performance[] = { + CPUS_ONLINE_MIN_2, + CPU0_MIN_FREQ_TURBO_MAX, + CPU1_MIN_FREQ_TURBO_MAX +}; + +static int profile_power_save[] = { + CPUS_ONLINE_MAX_LIMIT_2, + CPU0_MAX_FREQ_NONTURBO_MAX, + CPU1_MAX_FREQ_NONTURBO_MAX +}; +// clang-format on + +#ifdef INTERACTION_BOOST +int get_number_of_profiles() { + return 3; +} +#endif + +static int set_power_profile(void* data) { + int profile = data ? *((int*)data) : 0; + int ret = -EINVAL; + const char* profile_name = NULL; + + if (profile == current_power_profile) return 0; + + ALOGV("%s: Profile=%d", __func__, profile); + + if (current_power_profile != PROFILE_BALANCED) { + undo_hint_action(DEFAULT_PROFILE_HINT_ID); + ALOGV("%s: Hint undone", __func__); + current_power_profile = PROFILE_BALANCED; + } + + if (profile == PROFILE_POWER_SAVE) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_power_save, + ARRAY_SIZE(profile_power_save)); + profile_name = "powersave"; + + } else if (profile == PROFILE_HIGH_PERFORMANCE) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_high_performance, + ARRAY_SIZE(profile_high_performance)); + profile_name = "performance"; + + } else if (profile == PROFILE_BALANCED) { + ret = 0; + profile_name = "balanced"; + } + + if (ret == 0) { + current_power_profile = profile; + ALOGD("%s: Set %s mode", __func__, profile_name); + } + return ret; +} + static int process_video_encode_hint(void* metadata) { char governor[80]; struct video_encode_metadata_t video_encode_metadata; @@ -166,6 +226,19 @@ static void process_interaction_hint(void* data) { int power_hint_override(power_hint_t hint, void* data) { int ret_val = HINT_NONE; + + if (hint == POWER_HINT_SET_PROFILE) { + if (set_power_profile(data) < 0) + ALOGE("Setting power profile failed. mpdecision not started?"); + return HINT_HANDLED; + } + + // Skip other hints in high/low power modes + if (current_power_profile == PROFILE_POWER_SAVE || + current_power_profile == PROFILE_HIGH_PERFORMANCE) { + return HINT_HANDLED; + } + switch (hint) { case POWER_HINT_VIDEO_ENCODE: ret_val = process_video_encode_hint(data); diff --git a/power-8916.c b/power-8916.c index 8ffb647..08340ba 100644 --- a/power-8916.c +++ b/power-8916.c @@ -1,5 +1,6 @@ /* * Copyright (c) 2015, The Linux Foundation. All rights reserved. + * Copyright (C) 2018-2019 The LineageOS Project * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are @@ -64,6 +65,81 @@ static bool is_target_8916(void) { return is_8916; } +static int current_power_profile = PROFILE_BALANCED; + +// clang-format off +static int profile_high_performance_8916[3] = { + 0x1C00, 0x0901, CPU0_MIN_FREQ_TURBO_MAX, +}; + +static int profile_high_performance_8939[11] = { + SCHED_BOOST_ON, 0x1C00, 0x0901, + CPU0_MIN_FREQ_TURBO_MAX, CPU1_MIN_FREQ_TURBO_MAX, + CPU2_MIN_FREQ_TURBO_MAX, CPU3_MIN_FREQ_TURBO_MAX, + CPU4_MIN_FREQ_TURBO_MAX, CPU5_MIN_FREQ_TURBO_MAX, + CPU6_MIN_FREQ_TURBO_MAX, CPU7_MIN_FREQ_TURBO_MAX, +}; + +static int profile_power_save_8916[1] = { + CPU0_MAX_FREQ_NONTURBO_MAX, +}; + +static int profile_power_save_8939[5] = { + CPUS_ONLINE_MAX_LIMIT_2, + CPU0_MAX_FREQ_NONTURBO_MAX, CPU1_MAX_FREQ_NONTURBO_MAX, + CPU2_MAX_FREQ_NONTURBO_MAX, CPU3_MAX_FREQ_NONTURBO_MAX, +}; +// clang-format on + +#ifdef INTERACTION_BOOST +int get_number_of_profiles() { + return 3; +} +#endif + +static int set_power_profile(void* data) { + int profile = data ? *((int*)data) : 0; + int ret = -EINVAL; + const char* profile_name = NULL; + + if (profile == current_power_profile) return 0; + + ALOGV("%s: Profile=%d", __func__, profile); + + if (current_power_profile != PROFILE_BALANCED) { + undo_hint_action(DEFAULT_PROFILE_HINT_ID); + ALOGV("%s: Hint undone", __func__); + current_power_profile = PROFILE_BALANCED; + } + + if (profile == PROFILE_POWER_SAVE) { + ret = perform_hint_action( + DEFAULT_PROFILE_HINT_ID, + is_target_8916() ? profile_power_save_8916 : profile_power_save_8939, + is_target_8916() ? ARRAY_SIZE(profile_power_save_8916) + : ARRAY_SIZE(profile_power_save_8939)); + profile_name = "powersave"; + + } else if (profile == PROFILE_HIGH_PERFORMANCE) { + ret = perform_hint_action( + DEFAULT_PROFILE_HINT_ID, + is_target_8916() ? profile_high_performance_8916 : profile_high_performance_8939, + is_target_8916() ? ARRAY_SIZE(profile_high_performance_8916) + : ARRAY_SIZE(profile_high_performance_8939)); + profile_name = "performance"; + + } else if (profile == PROFILE_BALANCED) { + ret = 0; + profile_name = "balanced"; + } + + if (ret == 0) { + current_power_profile = profile; + ALOGD("%s: Set %s mode", __func__, profile_name); + } + return ret; +} + // clang-format off static int resources_interaction_fling_boost[] = { ALL_CPUS_PWR_CLPS_DIS, @@ -157,6 +233,18 @@ static int process_activity_launch_hint(void* data) { int power_hint_override(power_hint_t hint, void* data) { int ret_val = HINT_NONE; + + if (hint == POWER_HINT_SET_PROFILE) { + if (set_power_profile(data) < 0) ALOGE("Setting power profile failed. perfd not started?"); + return HINT_HANDLED; + } + + // Skip other hints in high/low power modes + if (current_power_profile == PROFILE_POWER_SAVE || + current_power_profile == PROFILE_HIGH_PERFORMANCE) { + return HINT_HANDLED; + } + switch (hint) { case POWER_HINT_VIDEO_ENCODE: /* Do nothing for encode case */ ret_val = HINT_HANDLED; diff --git a/power-8952.c b/power-8952.c index 3e7941b..4bcea61 100644 --- a/power-8952.c +++ b/power-8952.c @@ -1,5 +1,6 @@ /* * Copyright (c) 2015,2018 The Linux Foundation. All rights reserved. + * Copyright (C) 2018-2019 The LineageOS Project * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are @@ -49,6 +50,85 @@ #include "power-common.h" #include "utils.h" +static int current_power_profile = PROFILE_BALANCED; + +// clang-format off +static int profile_high_performance[] = { + CPUS_ONLINE_MIN_BIG, 0x4, + CPUS_ONLINE_MIN_LITTLE, 0x4, + CPU0_MIN_FREQ_TURBO_MAX, + CPU4_MIN_FREQ_TURBO_MAX, +}; + +static int profile_power_save[] = { + CPUS_ONLINE_MAX_BIG, 0x0, + CPU0_MAX_FREQ_NONTURBO_MAX, + CPU4_MAX_FREQ_NONTURBO_MAX, +}; + +static int profile_bias_power[] = { + CPU0_MAX_FREQ_NONTURBO_MAX, + CPU4_MAX_FREQ_NONTURBO_MAX, +}; + +static int profile_bias_performance[] = { + CPU0_MIN_FREQ_NONTURBO_MAX + 1, + CPU4_MIN_FREQ_NONTURBO_MAX + 1, +}; +// clang-format on + +#ifdef INTERACTION_BOOST +int get_number_of_profiles() { + return 5; +} +#endif + +static int set_power_profile(void* data) { + int profile = data ? *((int*)data) : 0; + int ret = -EINVAL; + const char* profile_name = NULL; + + if (profile == current_power_profile) return 0; + + ALOGV("%s: Profile=%d", __func__, profile); + + if (current_power_profile != PROFILE_BALANCED) { + undo_hint_action(DEFAULT_PROFILE_HINT_ID); + ALOGV("%s: Hint undone", __func__); + current_power_profile = PROFILE_BALANCED; + } + + if (profile == PROFILE_POWER_SAVE) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_power_save, + ARRAY_SIZE(profile_power_save)); + profile_name = "powersave"; + + } else if (profile == PROFILE_HIGH_PERFORMANCE) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_high_performance, + ARRAY_SIZE(profile_high_performance)); + profile_name = "performance"; + + } else if (profile == PROFILE_BIAS_POWER) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_bias_power, + ARRAY_SIZE(profile_bias_power)); + profile_name = "bias power"; + + } else if (profile == PROFILE_BIAS_PERFORMANCE) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_bias_performance, + ARRAY_SIZE(profile_bias_performance)); + profile_name = "bias perf"; + } else if (profile == PROFILE_BALANCED) { + ret = 0; + profile_name = "balanced"; + } + + if (ret == 0) { + current_power_profile = profile; + ALOGD("%s: Set %s mode", __func__, profile_name); + } + return ret; +} + static int process_video_encode_hint(void* metadata) { char governor[80]; struct video_encode_metadata_t video_encode_metadata; @@ -131,6 +211,18 @@ static int process_video_decode_hint(void* metadata) { int power_hint_override(power_hint_t hint, void* data) { int ret_val = HINT_NONE; + + if (hint == POWER_HINT_SET_PROFILE) { + if (set_power_profile(data) < 0) ALOGE("Setting power profile failed. perfd not started?"); + return HINT_HANDLED; + } + + // Skip other hints in high/low power modes + if (current_power_profile == PROFILE_POWER_SAVE || + current_power_profile == PROFILE_HIGH_PERFORMANCE) { + return HINT_HANDLED; + } + switch (hint) { case POWER_HINT_VIDEO_ENCODE: ret_val = process_video_encode_hint(data); diff --git a/power-8960.c b/power-8960.c index 6c29b06..6ea0568 100644 --- a/power-8960.c +++ b/power-8960.c @@ -1,5 +1,6 @@ /* * Copyright (c) 2013, The Linux Foundation. All rights reserved. + * Copyright (C) 2018-2019 The LineageOS Project * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are @@ -48,6 +49,92 @@ #include "power-common.h" #include "utils.h" +/** + * Returns true if the target is APQ8064. + */ +static bool is_target_8064(void) { + static int is_8064 = -1; + int soc_id; + + if (is_8064 >= 0) return is_8064; + + soc_id = get_soc_id(); + is_8064 = soc_id == 153; + + return is_8064; +} + +static int current_power_profile = PROFILE_BALANCED; + +// clang-format off +static int profile_high_performance_8960[] = { + CPUS_ONLINE_MIN_2, +}; + +static int profile_high_performance_8064[] = { + CPUS_ONLINE_MIN_4, +}; + +static int profile_power_save_8960[] = { + /* Don't do anything for now */ +}; + +static int profile_power_save_8064[] = { + CPUS_ONLINE_MAX_LIMIT_2, +}; +// clang-format on + +#ifdef INTERACTION_BOOST +int get_number_of_profiles() { + return 3; +} +#endif + +static int set_power_profile(void* data) { + int profile = data ? *((int*)data) : 0; + int ret = -EINVAL; + const char* profile_name = NULL; + + if (profile == current_power_profile) return 0; + + ALOGV("%s: Profile=%d", __func__, profile); + + if (current_power_profile != PROFILE_BALANCED) { + undo_hint_action(DEFAULT_PROFILE_HINT_ID); + ALOGV("%s: Hint undone", __func__); + current_power_profile = PROFILE_BALANCED; + } + + if (profile == PROFILE_POWER_SAVE) { + if (is_target_8064()) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_power_save_8064, + ARRAY_SIZE(profile_power_save_8064)); + } else { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_power_save_8960, + ARRAY_SIZE(profile_power_save_8960)); + } + profile_name = "powersave"; + } else if (profile == PROFILE_HIGH_PERFORMANCE) { + if (is_target_8064()) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_high_performance_8064, + ARRAY_SIZE(profile_high_performance_8064)); + } else { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_high_performance_8960, + ARRAY_SIZE(profile_high_performance_8960)); + } + profile_name = "performance"; + } else if (profile == PROFILE_BALANCED) { + ret = 0; + profile_name = "balanced"; + } + + if (ret == 0) { + current_power_profile = profile; + ALOGD("%s: Set %s mode", __func__, profile_name); + } + return ret; +} + static int process_video_encode_hint(void* metadata) { char governor[80]; struct video_encode_metadata_t video_encode_metadata; @@ -126,6 +213,18 @@ static int process_video_decode_hint(void* metadata) { int power_hint_override(power_hint_t hint, void* data) { int ret_val = HINT_NONE; + + if (hint == POWER_HINT_SET_PROFILE) { + if (set_power_profile(data) < 0) ALOGE("Setting power profile failed. perfd not started?"); + return HINT_HANDLED; + } + + // Skip other hints in high/low power modes + if (current_power_profile == PROFILE_POWER_SAVE || + current_power_profile == PROFILE_HIGH_PERFORMANCE) { + return HINT_HANDLED; + } + switch (hint) { case POWER_HINT_VIDEO_ENCODE: ret_val = process_video_encode_hint(data); diff --git a/power-8974.c b/power-8974.c index 9b9a52d..6269438 100644 --- a/power-8974.c +++ b/power-8974.c @@ -1,5 +1,6 @@ /* * Copyright (c) 2013, The Linux Foundation. All rights reserved. + * Copyright (C) 2018-2019 The LineageOS Project * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are @@ -65,6 +66,95 @@ static bool is_target_8974pro(void) { return is_8974pro; } +static int current_power_profile = PROFILE_BALANCED; + +// clang-format off +static int profile_high_performance[] = { + CPUS_ONLINE_MIN_4, + 0x0901, + CPU0_MIN_FREQ_TURBO_MAX, + CPU1_MIN_FREQ_TURBO_MAX, + CPU2_MIN_FREQ_TURBO_MAX, + CPU3_MIN_FREQ_TURBO_MAX +}; + +static int profile_power_save[] = { + 0x0A03, + CPUS_ONLINE_MAX_LIMIT_2, + CPU0_MAX_FREQ_NONTURBO_MAX, + CPU1_MAX_FREQ_NONTURBO_MAX, + CPU2_MAX_FREQ_NONTURBO_MAX, + CPU3_MAX_FREQ_NONTURBO_MAX +}; + +static int profile_bias_power[] = { + 0x0A03, + CPU0_MAX_FREQ_NONTURBO_MAX, + CPU1_MAX_FREQ_NONTURBO_MAX, + CPU2_MAX_FREQ_NONTURBO_MAX, + CPU3_MAX_FREQ_NONTURBO_MAX +}; + +static int profile_bias_performance[] = { + CPU0_MIN_FREQ_NONTURBO_MAX + 1, + CPU1_MIN_FREQ_NONTURBO_MAX + 1, + CPU2_MIN_FREQ_NONTURBO_MAX + 1, + CPU3_MIN_FREQ_NONTURBO_MAX + 1 +}; +// clang-format on + +#ifdef INTERACTION_BOOST +int get_number_of_profiles() { + return 5; +} +#endif + +static int set_power_profile(void* data) { + int profile = data ? *((int*)data) : 0; + int ret = -EINVAL; + const char* profile_name = NULL; + + if (profile == current_power_profile) return 0; + + ALOGV("%s: Profile=%d", __func__, profile); + + if (current_power_profile != PROFILE_BALANCED) { + undo_hint_action(DEFAULT_PROFILE_HINT_ID); + ALOGV("%s: Hint undone", __func__); + current_power_profile = PROFILE_BALANCED; + } + + if (profile == PROFILE_POWER_SAVE) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_power_save, + ARRAY_SIZE(profile_power_save)); + profile_name = "powersave"; + + } else if (profile == PROFILE_HIGH_PERFORMANCE) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_high_performance, + ARRAY_SIZE(profile_high_performance)); + profile_name = "performance"; + + } else if (profile == PROFILE_BIAS_POWER) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_bias_power, + ARRAY_SIZE(profile_bias_power)); + profile_name = "bias power"; + + } else if (profile == PROFILE_BIAS_PERFORMANCE) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_bias_performance, + ARRAY_SIZE(profile_bias_performance)); + profile_name = "bias perf"; + } else if (profile == PROFILE_BALANCED) { + ret = 0; + profile_name = "balanced"; + } + + if (ret == 0) { + current_power_profile = profile; + ALOGD("%s: Set %s mode", __func__, profile_name); + } + return ret; +} + static int process_video_encode_hint(void* metadata) { char governor[80]; struct video_encode_metadata_t video_encode_metadata; @@ -234,6 +324,19 @@ static int process_activity_launch_hint(void* data) { int power_hint_override(power_hint_t hint, void* data) { int ret_val = HINT_NONE; + + if (hint == POWER_HINT_SET_PROFILE) { + if (set_power_profile(data) < 0) + ALOGE("Setting power profile failed. mpdecision not started?"); + return HINT_HANDLED; + } + + // Skip other hints in high/low power modes + if (current_power_profile == PROFILE_POWER_SAVE || + current_power_profile == PROFILE_HIGH_PERFORMANCE) { + return HINT_HANDLED; + } + switch (hint) { case POWER_HINT_VIDEO_ENCODE: ret_val = process_video_encode_hint(data); diff --git a/power-8992.c b/power-8992.c index 6fc62d3..049dcf3 100644 --- a/power-8992.c +++ b/power-8992.c @@ -1,5 +1,6 @@ /* * Copyright (c) 2015, The Linux Foundation. All rights reserved. + * Copyright (C) 2018-2019 The LineageOS Project * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are @@ -48,6 +49,99 @@ #include "power-common.h" #include "utils.h" +static int current_power_profile = PROFILE_BALANCED; + +// clang-format off +static int profile_high_performance[] = { + SCHED_BOOST_ON, CPUS_ONLINE_MAX, + ALL_CPUS_PWR_CLPS_DIS, 0x0901, + CPU0_MIN_FREQ_TURBO_MAX, + CPU1_MIN_FREQ_TURBO_MAX, + CPU2_MIN_FREQ_TURBO_MAX, + CPU3_MIN_FREQ_TURBO_MAX, + CPU4_MIN_FREQ_TURBO_MAX, + CPU5_MIN_FREQ_TURBO_MAX +}; + +static int profile_power_save[] = { + CPUS_ONLINE_MPD_OVERRIDE, 0x0A03, + CPU0_MAX_FREQ_NONTURBO_MAX - 2, + CPU1_MAX_FREQ_NONTURBO_MAX - 2, + CPU2_MAX_FREQ_NONTURBO_MAX - 2, + CPU3_MAX_FREQ_NONTURBO_MAX - 2, + CPU4_MAX_FREQ_NONTURBO_MAX - 2, + CPU5_MAX_FREQ_NONTURBO_MAX - 2 +}; + +static int profile_bias_power[] = { + 0x0A03, 0x0902, + CPU0_MAX_FREQ_NONTURBO_MAX - 2, + CPU1_MAX_FREQ_NONTURBO_MAX - 2, + CPU2_MAX_FREQ_NONTURBO_MAX - 2, + CPU3_MAX_FREQ_NONTURBO_MAX - 2, + CPU4_MAX_FREQ_NONTURBO_MAX, + CPU5_MAX_FREQ_NONTURBO_MAX +}; + +static int profile_bias_performance[] = { + CPUS_ONLINE_MAX_LIMIT_MAX, + CPU4_MIN_FREQ_NONTURBO_MAX + 1, + CPU5_MIN_FREQ_NONTURBO_MAX + 1 +}; +// clang-format on + +#ifdef INTERACTION_BOOST +int get_number_of_profiles() { + return 5; +} +#endif + +static int set_power_profile(void* data) { + int profile = data ? *((int*)data) : 0; + int ret = -EINVAL; + const char* profile_name = NULL; + + if (profile == current_power_profile) return 0; + + ALOGV("%s: Profile=%d", __func__, profile); + + if (current_power_profile != PROFILE_BALANCED) { + undo_hint_action(DEFAULT_PROFILE_HINT_ID); + ALOGV("%s: Hint undone", __func__); + current_power_profile = PROFILE_BALANCED; + } + + if (profile == PROFILE_POWER_SAVE) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_power_save, + ARRAY_SIZE(profile_power_save)); + profile_name = "powersave"; + + } else if (profile == PROFILE_HIGH_PERFORMANCE) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_high_performance, + ARRAY_SIZE(profile_high_performance)); + profile_name = "performance"; + + } else if (profile == PROFILE_BIAS_POWER) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_bias_power, + ARRAY_SIZE(profile_bias_power)); + profile_name = "bias power"; + + } else if (profile == PROFILE_BIAS_PERFORMANCE) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_bias_performance, + ARRAY_SIZE(profile_bias_performance)); + profile_name = "bias perf"; + } else if (profile == PROFILE_BALANCED) { + ret = 0; + profile_name = "balanced"; + } + + if (ret == 0) { + current_power_profile = profile; + ALOGD("%s: Set %s mode", __func__, profile_name); + } + return ret; +} + static int process_video_encode_hint(void* metadata) { char governor[80]; struct video_encode_metadata_t video_encode_metadata; @@ -214,6 +308,18 @@ static int process_activity_launch_hint(void* data) { int power_hint_override(power_hint_t hint, void* data) { int ret_val = HINT_NONE; + + if (hint == POWER_HINT_SET_PROFILE) { + if (set_power_profile(data) < 0) ALOGE("Setting power profile failed. perfd not started?"); + return HINT_HANDLED; + } + + // Skip other hints in high/low power modes + if (current_power_profile == PROFILE_POWER_SAVE || + current_power_profile == PROFILE_HIGH_PERFORMANCE) { + return HINT_HANDLED; + } + switch (hint) { case POWER_HINT_VIDEO_ENCODE: ret_val = process_video_encode_hint(data); diff --git a/power-8994.c b/power-8994.c index ccc025a..22f747d 100644 --- a/power-8994.c +++ b/power-8994.c @@ -1,5 +1,6 @@ /* * Copyright (c) 2014, The Linux Foundation. All rights reserved. + * Copyright (C) 2018-2019 The LineageOS Project * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are @@ -48,6 +49,107 @@ #include "power-common.h" #include "utils.h" +static int current_power_profile = PROFILE_BALANCED; + +// clang-format off +static int profile_high_performance[] = { + SCHED_BOOST_ON, CPUS_ONLINE_MAX, + ALL_CPUS_PWR_CLPS_DIS, 0x0901, + CPU0_MIN_FREQ_TURBO_MAX, + CPU1_MIN_FREQ_TURBO_MAX, + CPU2_MIN_FREQ_TURBO_MAX, + CPU3_MIN_FREQ_TURBO_MAX, + CPU4_MIN_FREQ_TURBO_MAX, + CPU5_MIN_FREQ_TURBO_MAX, + CPU6_MIN_FREQ_TURBO_MAX, + CPU7_MIN_FREQ_TURBO_MAX +}; + +static int profile_power_save[] = { + CPUS_ONLINE_MPD_OVERRIDE, 0x0A03, + CPU0_MAX_FREQ_NONTURBO_MAX - 2, + CPU1_MAX_FREQ_NONTURBO_MAX - 2, + CPU2_MAX_FREQ_NONTURBO_MAX - 2, + CPU3_MAX_FREQ_NONTURBO_MAX - 2, + CPU4_MAX_FREQ_NONTURBO_MAX - 2, + CPU5_MAX_FREQ_NONTURBO_MAX - 2, + CPU6_MAX_FREQ_NONTURBO_MAX - 2, + CPU7_MAX_FREQ_NONTURBO_MAX - 2 +}; + +static int profile_bias_power[] = { + 0x0A03, 0x0902, + CPU0_MAX_FREQ_NONTURBO_MAX - 2, + CPU1_MAX_FREQ_NONTURBO_MAX - 2, + CPU2_MAX_FREQ_NONTURBO_MAX - 2, + CPU3_MAX_FREQ_NONTURBO_MAX - 2, + CPU4_MAX_FREQ_NONTURBO_MAX, + CPU5_MAX_FREQ_NONTURBO_MAX, + CPU6_MAX_FREQ_NONTURBO_MAX, + CPU7_MAX_FREQ_NONTURBO_MAX +}; + +static int profile_bias_performance[] = { + CPUS_ONLINE_MAX_LIMIT_MAX, + CPU4_MIN_FREQ_NONTURBO_MAX + 1, + CPU5_MIN_FREQ_NONTURBO_MAX + 1, + CPU6_MIN_FREQ_NONTURBO_MAX + 1, + CPU7_MIN_FREQ_NONTURBO_MAX + 1 +}; +// clang-format on + +#ifdef INTERACTION_BOOST +int get_number_of_profiles() { + return 5; +} +#endif + +static int set_power_profile(void* data) { + int profile = data ? *((int*)data) : 0; + int ret = -EINVAL; + const char* profile_name = NULL; + + if (profile == current_power_profile) return 0; + + ALOGV("%s: Profile=%d", __func__, profile); + + if (current_power_profile != PROFILE_BALANCED) { + undo_hint_action(DEFAULT_PROFILE_HINT_ID); + ALOGV("%s: Hint undone", __func__); + current_power_profile = PROFILE_BALANCED; + } + + if (profile == PROFILE_POWER_SAVE) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_power_save, + ARRAY_SIZE(profile_power_save)); + profile_name = "powersave"; + + } else if (profile == PROFILE_HIGH_PERFORMANCE) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_high_performance, + ARRAY_SIZE(profile_high_performance)); + profile_name = "performance"; + + } else if (profile == PROFILE_BIAS_POWER) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_bias_power, + ARRAY_SIZE(profile_bias_power)); + profile_name = "bias power"; + + } else if (profile == PROFILE_BIAS_PERFORMANCE) { + ret = perform_hint_action(DEFAULT_PROFILE_HINT_ID, profile_bias_performance, + ARRAY_SIZE(profile_bias_performance)); + profile_name = "bias perf"; + } else if (profile == PROFILE_BALANCED) { + ret = 0; + profile_name = "balanced"; + } + + if (ret == 0) { + current_power_profile = profile; + ALOGD("%s: Set %s mode", __func__, profile_name); + } + return ret; +} + static int process_video_encode_hint(void* metadata) { char governor[80]; struct video_encode_metadata_t video_encode_metadata; @@ -214,6 +316,18 @@ static int process_activity_launch_hint(void* data) { int power_hint_override(power_hint_t hint, void* data) { int ret_val = HINT_NONE; + + if (hint == POWER_HINT_SET_PROFILE) { + if (set_power_profile(data) < 0) ALOGE("Setting power profile failed. perfd not started?"); + return HINT_HANDLED; + } + + // Skip other hints in high/low power modes + if (current_power_profile == PROFILE_POWER_SAVE || + current_power_profile == PROFILE_HIGH_PERFORMANCE) { + return HINT_HANDLED; + } + switch (hint) { case POWER_HINT_VIDEO_ENCODE: ret_val = process_video_encode_hint(data); |