Change-Id: I993353943dd1be7104b6790bfd2d73cd6f63260atirimbino
parent
e893e32fe5
commit
e0dd329e29
@ -0,0 +1,47 @@ |
||||
#
|
||||
# Copyright (C) 2020 The LineageOS 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.
|
||||
#
|
||||
|
||||
LOCAL_PATH := $(call my-dir)
|
||||
|
||||
include $(CLEAR_VARS) |
||||
|
||||
LOCAL_SRC_FILES := \
|
||||
Power.cpp \
|
||||
service.cpp
|
||||
|
||||
LOCAL_C_INCLUDES := \
|
||||
$(LOCAL_PATH)/include \
|
||||
hardware/samsung/hidl/light/include
|
||||
|
||||
LOCAL_SHARED_LIBRARIES := \
|
||||
libbase \
|
||||
libbinder \
|
||||
libhidlbase \
|
||||
libhidltransport \
|
||||
libutils \
|
||||
android.hardware.power@1.0 \
|
||||
vendor.lineage.power@1.0
|
||||
|
||||
LOCAL_STATIC_LIBRARIES := libc++fs
|
||||
|
||||
LOCAL_MODULE := android.hardware.power@1.0-service.exynos
|
||||
LOCAL_INIT_RC := android.hardware.power@1.0-service.exynos.rc
|
||||
LOCAL_MODULE_RELATIVE_PATH := hw
|
||||
LOCAL_MODULE_TAGS := optional
|
||||
LOCAL_MODULE_OWNER := samsung
|
||||
LOCAL_VENDOR_MODULE := true
|
||||
|
||||
include $(BUILD_EXECUTABLE) |
@ -0,0 +1,247 @@ |
||||
/*
|
||||
* Copyright (C) 2020 The LineageOS 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. |
||||
*/ |
||||
|
||||
#define LOG_TAG "android.hardware.power@1.0-service.exynos" |
||||
|
||||
#include "Power.h" |
||||
#include <android-base/logging.h> |
||||
#include <filesystem> |
||||
#include <fstream> |
||||
#include <iostream> |
||||
#include "samsung_lights.h" |
||||
#include "samsung_power.h" |
||||
|
||||
namespace android { |
||||
namespace hardware { |
||||
namespace power { |
||||
namespace V1_0 { |
||||
namespace implementation { |
||||
|
||||
/*
|
||||
* Write value to path and close file. |
||||
*/ |
||||
template <typename T> |
||||
static void set(const std::string& path, const T& value) { |
||||
std::ofstream file(path); |
||||
file << value << std::endl; |
||||
} |
||||
|
||||
template <typename T> |
||||
static T get(const std::string& path, const T& def) { |
||||
std::ifstream file(path); |
||||
T result; |
||||
|
||||
file >> result; |
||||
return file.fail() ? def : result; |
||||
} |
||||
|
||||
Return<void> Power::setInteractive(bool interactive) { |
||||
if (!initialized) { |
||||
initialize(); |
||||
} |
||||
|
||||
if (!interactive) { |
||||
int32_t panel_brightness = get(PANEL_BRIGHTNESS_NODE, -1); |
||||
|
||||
if (panel_brightness > 0) { |
||||
LOG(VERBOSE) << "Moving to non-interactive state, but screen is still on," |
||||
<< "not disabling input devices"; |
||||
return Void(); |
||||
} |
||||
} |
||||
|
||||
if (!sec_touchscreen.empty()) { |
||||
set(sec_touchscreen, interactive ? "1" : "0"); |
||||
} |
||||
|
||||
if (!sec_touchkey.empty()) { |
||||
if (!interactive) { |
||||
int button_state = get(sec_touchkey, -1); |
||||
|
||||
if (button_state < 0) { |
||||
LOG(ERROR) << "Failed to read touchkey state"; |
||||
goto out; |
||||
} |
||||
|
||||
/*
|
||||
* If button_state is 0, the keys have been disabled by another component |
||||
* (for example lineagehw), which means we don't want them to be enabled when resuming |
||||
* from suspend. |
||||
*/ |
||||
if (button_state == 0) { |
||||
touchkeys_blocked = true; |
||||
} |
||||
} |
||||
|
||||
if (!touchkeys_blocked) { |
||||
set(sec_touchkey, interactive ? "1" : "0"); |
||||
} |
||||
} |
||||
|
||||
out: |
||||
for (const std::string& interactivePath : cpuInteractivePaths) { |
||||
set(interactivePath + "/io_is_busy", interactive ? "1" : "0"); |
||||
} |
||||
|
||||
return Void(); |
||||
} |
||||
|
||||
Return<void> Power::powerHint(PowerHint hint, int32_t data) { |
||||
if (!initialized) { |
||||
initialize(); |
||||
} |
||||
|
||||
/* Bail out if low-power mode is active */ |
||||
if (current_profile == PowerProfile::POWER_SAVE && hint != PowerHint::LOW_POWER && |
||||
hint != static_cast<PowerHint>(LineagePowerHint::SET_PROFILE)) { |
||||
LOG(VERBOSE) << "PROFILE_POWER_SAVE active, ignoring hint " << static_cast<int32_t>(hint); |
||||
return Void(); |
||||
} |
||||
|
||||
switch (hint) { |
||||
case PowerHint::INTERACTION: |
||||
case PowerHint::LAUNCH: |
||||
sendBoostpulse(); |
||||
break; |
||||
case PowerHint::LOW_POWER: |
||||
setProfile(data ? PowerProfile::POWER_SAVE : PowerProfile::BALANCED); |
||||
break; |
||||
default: |
||||
if (hint == static_cast<PowerHint>(LineagePowerHint::SET_PROFILE)) { |
||||
setProfile(static_cast<PowerProfile>(data)); |
||||
} else if (hint == static_cast<PowerHint>(LineagePowerHint::CPU_BOOST)) { |
||||
sendBoost(data); |
||||
} else { |
||||
LOG(INFO) << "Unknown power hint: " << static_cast<int32_t>(hint); |
||||
} |
||||
break; |
||||
} |
||||
return Void(); |
||||
} |
||||
|
||||
Return<void> Power::setFeature(Feature feature __unused, bool activate __unused) { |
||||
if (!initialized) { |
||||
initialize(); |
||||
} |
||||
|
||||
#ifdef TAP_TO_WAKE_NODE |
||||
if (feature == Feature::POWER_FEATURE_DOUBLE_TAP_TO_WAKE) { |
||||
set(TAP_TO_WAKE_NODE, activate ? "1" : "0"); |
||||
} |
||||
#endif |
||||
|
||||
return Void(); |
||||
} |
||||
|
||||
Return<void> Power::getPlatformLowPowerStats(getPlatformLowPowerStats_cb _hidl_cb) { |
||||
_hidl_cb({}, Status::SUCCESS); |
||||
return Void(); |
||||
} |
||||
|
||||
Return<int32_t> Power::getFeature(LineageFeature feature) { |
||||
switch (feature) { |
||||
case LineageFeature::SUPPORTED_PROFILES: |
||||
return static_cast<int32_t>(PowerProfile::MAX); |
||||
default: |
||||
return -1; |
||||
} |
||||
} |
||||
|
||||
void Power::initialize() { |
||||
findInputNodes(); |
||||
|
||||
current_profile = PowerProfile::BALANCED; |
||||
|
||||
for (const std::string& interactivePath : cpuInteractivePaths) { |
||||
hispeed_freqs.emplace_back(get<std::string>(interactivePath + "/hispeed_freq", "")); |
||||
} |
||||
|
||||
for (const std::string& sysfsPath : cpuSysfsPaths) { |
||||
max_freqs.emplace_back(get<std::string>(sysfsPath + "/cpufreq/scaling_max_freq", "")); |
||||
} |
||||
|
||||
initialized = true; |
||||
} |
||||
|
||||
void Power::findInputNodes() { |
||||
std::error_code ec; |
||||
for (auto& de : std::filesystem::directory_iterator("/sys/class/input/", ec)) { |
||||
/* we are only interested in the input devices that we can access */ |
||||
if (ec || de.path().string().find("/sys/class/input/input") == std::string::npos) { |
||||
continue; |
||||
} |
||||
|
||||
for (auto& de2 : std::filesystem::directory_iterator(de.path(), ec)) { |
||||
if (!ec && de2.path().string().find("/name") != std::string::npos) { |
||||
std::string content = get<std::string>(de2.path(), ""); |
||||
if (content == "sec_touchkey") { |
||||
sec_touchkey = de.path().string().append("/enabled"); |
||||
LOG(INFO) << "found sec_touchkey: " << sec_touchkey; |
||||
} else if (content == "sec_touchscreen") { |
||||
sec_touchscreen = de.path().string().append("/enabled"); |
||||
LOG(INFO) << "found sec_touchscreen: " << sec_touchscreen; |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
||||
|
||||
void Power::setProfile(PowerProfile profile) { |
||||
if (current_profile == profile) { |
||||
return; |
||||
} |
||||
|
||||
switch (profile) { |
||||
case PowerProfile::POWER_SAVE: |
||||
// Limit to hispeed freq
|
||||
for (int i = 0; i < cpuSysfsPaths.size(); i++) { |
||||
if (hispeed_freqs.size() > i && !hispeed_freqs.at(i).empty()) { |
||||
set(cpuSysfsPaths.at(i) + "/cpufreq/scaling_max_freq", hispeed_freqs.at(i)); |
||||
} |
||||
} |
||||
break; |
||||
case PowerProfile::BALANCED: |
||||
case PowerProfile::HIGH_PERFORMANCE: |
||||
// Restore normal max freq
|
||||
for (int i = 0; i < cpuSysfsPaths.size(); i++) { |
||||
if (max_freqs.size() > i && !max_freqs.at(i).empty()) { |
||||
set(cpuSysfsPaths.at(i) + "/cpufreq/scaling_max_freq", max_freqs.at(i)); |
||||
} |
||||
} |
||||
break; |
||||
default: |
||||
break; |
||||
} |
||||
} |
||||
|
||||
void Power::sendBoostpulse() { |
||||
// the boostpulse node is only valid for the LITTLE cluster
|
||||
set(cpuInteractivePaths.front() + "/boostpulse", "1"); |
||||
} |
||||
|
||||
void Power::sendBoost(int duration_us) { |
||||
set(cpuInteractivePaths.front() + "/boost", "1"); |
||||
|
||||
usleep(duration_us); |
||||
|
||||
set(cpuInteractivePaths.front() + "/boost", "0"); |
||||
} |
||||
|
||||
} // namespace implementation
|
||||
} // namespace V1_0
|
||||
} // namespace power
|
||||
} // namespace hardware
|
||||
} // namespace android
|
@ -0,0 +1,82 @@ |
||||
/*
|
||||
* Copyright (C) 2020 The LineageOS 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. |
||||
*/ |
||||
|
||||
#ifndef ANDROID_HARDWARE_POWER_V1_0_POWER_H |
||||
#define ANDROID_HARDWARE_POWER_V1_0_POWER_H |
||||
|
||||
#include <android/hardware/power/1.0/IPower.h> |
||||
#include <hidl/MQDescriptor.h> |
||||
#include <hidl/Status.h> |
||||
#include <vendor/lineage/power/1.0/ILineagePower.h> |
||||
|
||||
namespace android { |
||||
namespace hardware { |
||||
namespace power { |
||||
namespace V1_0 { |
||||
namespace implementation { |
||||
|
||||
using ::android::sp; |
||||
using ::android::hardware::hidl_array; |
||||
using ::android::hardware::hidl_memory; |
||||
using ::android::hardware::hidl_string; |
||||
using ::android::hardware::hidl_vec; |
||||
using ::android::hardware::Return; |
||||
using ::android::hardware::Void; |
||||
|
||||
using ::vendor::lineage::power::V1_0::ILineagePower; |
||||
using ::vendor::lineage::power::V1_0::LineageFeature; |
||||
using ::vendor::lineage::power::V1_0::LineagePowerHint; |
||||
|
||||
// clang-format off
|
||||
enum PowerProfile { |
||||
POWER_SAVE = 0, |
||||
BALANCED, |
||||
HIGH_PERFORMANCE, |
||||
MAX |
||||
}; |
||||
// clang-format on
|
||||
|
||||
struct Power : public IPower, public ILineagePower { |
||||
Return<void> setInteractive(bool interactive) override; |
||||
Return<void> powerHint(PowerHint hint, int32_t data) override; |
||||
Return<void> setFeature(Feature feature, bool activate) override; |
||||
Return<void> getPlatformLowPowerStats(getPlatformLowPowerStats_cb _hidl_cb) override; |
||||
|
||||
Return<int32_t> getFeature(LineageFeature feature) override; |
||||
|
||||
private: |
||||
void initialize(); |
||||
void findInputNodes(); |
||||
void setProfile(PowerProfile profile); |
||||
void sendBoostpulse(); |
||||
void sendBoost(int duration_us); |
||||
|
||||
bool initialized; |
||||
bool touchkeys_blocked; |
||||
std::string sec_touchkey; |
||||
std::string sec_touchscreen; |
||||
PowerProfile current_profile; |
||||
std::vector<std::string> hispeed_freqs; |
||||
std::vector<std::string> max_freqs; |
||||
}; |
||||
|
||||
} // namespace implementation
|
||||
} // namespace V1_0
|
||||
} // namespace power
|
||||
} // namespace hardware
|
||||
} // namespace android
|
||||
|
||||
#endif // ANDROID_HARDWARE_POWER_V1_0_POWER_H
|
@ -0,0 +1,4 @@ |
||||
service vendor.power-hal-1-0 /vendor/bin/hw/android.hardware.power@1.0-service.exynos |
||||
class hal |
||||
user system |
||||
group system |
@ -0,0 +1,42 @@ |
||||
/*
|
||||
* Copyright (C) 2016 The CyanogenMod Project |
||||
* Copyright (C) 2020 The LineageOS 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. |
||||
*/ |
||||
|
||||
#ifndef SAMSUNG_POWER_H |
||||
#define SAMSUNG_POWER_H |
||||
|
||||
/*
|
||||
* Board specific nodes |
||||
* |
||||
* If your kernel exposes these controls in another place, you can either |
||||
* symlink to the locations given here, or override this header in your |
||||
* device tree. |
||||
*/ |
||||
|
||||
static const std::vector<std::string> cpuSysfsPaths = { |
||||
"/sys/devices/system/cpu/cpu0", |
||||
"/sys/devices/system/cpu/cpu4" |
||||
}; |
||||
|
||||
static const std::vector<std::string> cpuInteractivePaths = { |
||||
"/sys/devices/system/cpu/cpu0/cpufreq/interactive", |
||||
"/sys/devices/system/cpu/cpu4/cpufreq/interactive" |
||||
}; |
||||
|
||||
/* double tap to wake node */ |
||||
//#define TAP_TO_WAKE_NODE "/sys/class/sec/tsp/dt2w_enable"
|
||||
|
||||
#endif // SAMSUNG_POWER_H
|
@ -0,0 +1,60 @@ |
||||
/*
|
||||
* Copyright (C) 2020 The LineageOS 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. |
||||
*/ |
||||
|
||||
#define LOG_TAG "android.hardware.power@1.0-service.exynos" |
||||
|
||||
#include <android-base/logging.h> |
||||
#include <hidl/HidlTransportSupport.h> |
||||
#include <utils/Errors.h> |
||||
|
||||
#include "Power.h" |
||||
|
||||
using android::hardware::configureRpcThreadpool; |
||||
using android::hardware::joinRpcThreadpool; |
||||
|
||||
using android::hardware::power::V1_0::IPower; |
||||
using android::hardware::power::V1_0::implementation::Power; |
||||
|
||||
using android::OK; |
||||
using android::sp; |
||||
using android::status_t; |
||||
|
||||
int main() { |
||||
sp<Power> power = new Power(); |
||||
status_t status = 0; |
||||
|
||||
configureRpcThreadpool(1, true); |
||||
|
||||
status = power->IPower::registerAsService(); |
||||
if (status != OK) { |
||||
LOG(ERROR) << "Could not register service (IPower) for Power HAL"; |
||||
goto shutdown; |
||||
} |
||||
|
||||
status = power->ILineagePower::registerAsService(); |
||||
if (status != OK) { |
||||
LOG(ERROR) << "Could not register service (ILineagePower) for Power HAL"; |
||||
goto shutdown; |
||||
} |
||||
|
||||
LOG(INFO) << "Power HAL service is Ready."; |
||||
joinRpcThreadpool(); |
||||
|
||||
shutdown: |
||||
// In normal operation, we don't expect the thread pool to shutdown
|
||||
LOG(ERROR) << "Power HAL failed to join thread pool."; |
||||
return 1; |
||||
} |
Loading…
Reference in new issue