/* * Copyright (C) 2020 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. */ #define LOG_TAG "PowerHalController" #include #include #include #include #include #include #include #include using namespace android::hardware::power; namespace android { namespace power { // ------------------------------------------------------------------------------------------------- std::unique_ptr HalConnector::connect() { if (std::shared_ptr halAidl = PowerHalLoader::loadAidl()) { return std::make_unique(halAidl); } // If V1_0 isn't defined, none of them are if (sp halHidlV1_0 = PowerHalLoader::loadHidlV1_0()) { if (sp halHidlV1_3 = PowerHalLoader::loadHidlV1_3()) { return std::make_unique(halHidlV1_3); } if (sp halHidlV1_2 = PowerHalLoader::loadHidlV1_2()) { return std::make_unique(halHidlV1_2); } if (sp halHidlV1_1 = PowerHalLoader::loadHidlV1_1()) { return std::make_unique(halHidlV1_1); } return std::make_unique(halHidlV1_0); } return nullptr; } void HalConnector::reset() { PowerHalLoader::unloadAll(); } int32_t HalConnector::getAidlVersion() { return PowerHalLoader::getAidlVersion(); } // ------------------------------------------------------------------------------------------------- void PowerHalController::init() { initHal(); } // Check validity of current handle to the power HAL service, and create a new // one if necessary. std::shared_ptr PowerHalController::initHal() { std::lock_guard lock(mConnectedHalMutex); if (mConnectedHal == nullptr) { mConnectedHal = mHalConnector->connect(); if (mConnectedHal == nullptr) { // Unable to connect to Power HAL service. Fallback to default. return mDefaultHal; } } return mConnectedHal; } // Using statement expression macro instead of a method lets the static be // scoped to the outer method while dodging the need for a support lookup table // This only works for AIDL methods that do not vary supported/unsupported depending // on their arguments (not setBoost, setMode) which do their own support checks #define CACHE_SUPPORT(version, method) \ ({ \ static bool support = mHalConnector->getAidlVersion() >= version; \ !support ? decltype(method)::unsupported() : ({ \ auto result = method; \ if (result.isUnsupported()) { \ support = false; \ } \ std::move(result); \ }); \ }) // Check if a call to Power HAL function failed; if so, log the failure and // invalidate the current Power HAL handle. template HalResult PowerHalController::processHalResult(HalResult&& result, const char* fnName) { if (result.isFailed()) { ALOGE("%s failed: %s", fnName, result.errorMessage()); std::lock_guard lock(mConnectedHalMutex); // Drop Power HAL handle. This will force future api calls to reconnect. mConnectedHal = nullptr; mHalConnector->reset(); } return std::move(result); } HalResult PowerHalController::setBoost(aidl::android::hardware::power::Boost boost, int32_t durationMs) { std::shared_ptr handle = initHal(); return processHalResult(handle->setBoost(boost, durationMs), "setBoost"); } HalResult PowerHalController::setMode(aidl::android::hardware::power::Mode mode, bool enabled) { std::shared_ptr handle = initHal(); return processHalResult(handle->setMode(mode, enabled), "setMode"); } // Aidl-only methods HalResult> PowerHalController::createHintSession( int32_t tgid, int32_t uid, const std::vector& threadIds, int64_t durationNanos) { std::shared_ptr handle = initHal(); return CACHE_SUPPORT(2, processHalResult(handle->createHintSession(tgid, uid, threadIds, durationNanos), "createHintSession")); } HalResult> PowerHalController::createHintSessionWithConfig( int32_t tgid, int32_t uid, const std::vector& threadIds, int64_t durationNanos, aidl::android::hardware::power::SessionTag tag, aidl::android::hardware::power::SessionConfig* config) { std::shared_ptr handle = initHal(); return CACHE_SUPPORT(5, processHalResult(handle->createHintSessionWithConfig(tgid, uid, threadIds, durationNanos, tag, config), "createHintSessionWithConfig")); } HalResult PowerHalController::getHintSessionPreferredRate() { std::shared_ptr handle = initHal(); return CACHE_SUPPORT(2, processHalResult(handle->getHintSessionPreferredRate(), "getHintSessionPreferredRate")); } HalResult PowerHalController::getSessionChannel( int tgid, int uid) { std::shared_ptr handle = initHal(); return CACHE_SUPPORT(5, processHalResult(handle->getSessionChannel(tgid, uid), "getSessionChannel")); } HalResult PowerHalController::closeSessionChannel(int tgid, int uid) { std::shared_ptr handle = initHal(); return CACHE_SUPPORT(5, processHalResult(handle->closeSessionChannel(tgid, uid), "closeSessionChannel")); } } // namespace power } // namespace android