160 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			160 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			C++
		
	
	
	
/*
 | 
						|
 * Copyright 2021 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.
 | 
						|
 */
 | 
						|
 | 
						|
#pragma once
 | 
						|
 | 
						|
#include <aidl/android/hardware/power/BnPowerHintSession.h>
 | 
						|
#include <aidl/android/hardware/power/WorkDuration.h>
 | 
						|
#include <utils/Looper.h>
 | 
						|
#include <utils/Thread.h>
 | 
						|
 | 
						|
#include <mutex>
 | 
						|
#include <unordered_map>
 | 
						|
 | 
						|
#include "adaptivecpu/AdaptiveCpu.h"
 | 
						|
 | 
						|
namespace aidl {
 | 
						|
namespace google {
 | 
						|
namespace hardware {
 | 
						|
namespace power {
 | 
						|
namespace impl {
 | 
						|
namespace pixel {
 | 
						|
 | 
						|
using aidl::android::hardware::power::BnPowerHintSession;
 | 
						|
using aidl::android::hardware::power::WorkDuration;
 | 
						|
using ::android::Message;
 | 
						|
using ::android::MessageHandler;
 | 
						|
using ::android::sp;
 | 
						|
using std::chrono::milliseconds;
 | 
						|
using std::chrono::nanoseconds;
 | 
						|
using std::chrono::steady_clock;
 | 
						|
using std::chrono::time_point;
 | 
						|
 | 
						|
struct AppHintDesc {
 | 
						|
    AppHintDesc(int32_t tgid, int32_t uid, std::vector<int> threadIds)
 | 
						|
        : tgid(tgid),
 | 
						|
          uid(uid),
 | 
						|
          threadIds(std::move(threadIds)),
 | 
						|
          duration(0LL),
 | 
						|
          current_min(0),
 | 
						|
          is_active(true),
 | 
						|
          update_count(0),
 | 
						|
          integral_error(0),
 | 
						|
          previous_error(0) {}
 | 
						|
    std::string toString() const;
 | 
						|
    const int32_t tgid;
 | 
						|
    const int32_t uid;
 | 
						|
    const std::vector<int> threadIds;
 | 
						|
    nanoseconds duration;
 | 
						|
    int current_min;
 | 
						|
    // status
 | 
						|
    std::atomic<bool> is_active;
 | 
						|
    // pid
 | 
						|
    uint64_t update_count;
 | 
						|
    int64_t integral_error;
 | 
						|
    int64_t previous_error;
 | 
						|
};
 | 
						|
 | 
						|
class PowerHintSession : public BnPowerHintSession {
 | 
						|
  public:
 | 
						|
    explicit PowerHintSession(std::shared_ptr<AdaptiveCpu> adaptiveCpu, int32_t tgid, int32_t uid,
 | 
						|
                              const std::vector<int32_t> &threadIds, int64_t durationNanos);
 | 
						|
    ~PowerHintSession();
 | 
						|
    ndk::ScopedAStatus close() override;
 | 
						|
    ndk::ScopedAStatus pause() override;
 | 
						|
    ndk::ScopedAStatus resume() override;
 | 
						|
    ndk::ScopedAStatus updateTargetWorkDuration(int64_t targetDurationNanos) override;
 | 
						|
    ndk::ScopedAStatus reportActualWorkDuration(
 | 
						|
            const std::vector<WorkDuration> &actualDurations) override;
 | 
						|
    bool isActive();
 | 
						|
    bool isTimeout();
 | 
						|
    void wakeup();
 | 
						|
    void setStale();
 | 
						|
    // Is this hint session for a user application
 | 
						|
    bool isAppSession();
 | 
						|
    const std::vector<int> &getTidList() const;
 | 
						|
    int getUclampMin();
 | 
						|
    void dumpToStream(std::ostream &stream);
 | 
						|
 | 
						|
    void updateWorkPeriod(const std::vector<WorkDuration> &actualDurations);
 | 
						|
    time_point<steady_clock> getEarlyBoostTime();
 | 
						|
    time_point<steady_clock> getStaleTime();
 | 
						|
 | 
						|
  private:
 | 
						|
    class StaleTimerHandler : public MessageHandler {
 | 
						|
      public:
 | 
						|
        StaleTimerHandler(PowerHintSession *session)
 | 
						|
            : mSession(session), mIsMonitoring(false), mIsSessionDead(false) {}
 | 
						|
        void updateTimer();
 | 
						|
        void updateTimer(time_point<steady_clock> staleTime);
 | 
						|
        void handleMessage(const Message &message) override;
 | 
						|
        void setSessionDead();
 | 
						|
 | 
						|
      private:
 | 
						|
        PowerHintSession *mSession;
 | 
						|
        std::mutex mClosedLock;
 | 
						|
        std::mutex mMessageLock;
 | 
						|
        std::atomic<time_point<steady_clock>> mStaleTime;
 | 
						|
        std::atomic<bool> mIsMonitoring;
 | 
						|
        bool mIsSessionDead;
 | 
						|
    };
 | 
						|
 | 
						|
    class EarlyBoostHandler : public MessageHandler {
 | 
						|
      public:
 | 
						|
        EarlyBoostHandler(PowerHintSession *session)
 | 
						|
            : mSession(session), mIsMonitoring(false), mIsSessionDead(false) {}
 | 
						|
        void updateTimer(time_point<steady_clock> boostTime);
 | 
						|
        void handleMessage(const Message &message) override;
 | 
						|
        void setSessionDead();
 | 
						|
 | 
						|
      private:
 | 
						|
        PowerHintSession *mSession;
 | 
						|
        std::mutex mBoostLock;
 | 
						|
        std::mutex mMessageLock;
 | 
						|
        std::atomic<time_point<steady_clock>> mBoostTime;
 | 
						|
        std::atomic<bool> mIsMonitoring;
 | 
						|
        bool mIsSessionDead;
 | 
						|
    };
 | 
						|
 | 
						|
  private:
 | 
						|
    void updateUniveralBoostMode();
 | 
						|
    int setSessionUclampMin(int32_t min);
 | 
						|
    void tryToSendPowerHint(std::string hint);
 | 
						|
    std::string getIdString() const;
 | 
						|
    const std::shared_ptr<AdaptiveCpu> mAdaptiveCpu;
 | 
						|
    AppHintDesc *mDescriptor = nullptr;
 | 
						|
    sp<StaleTimerHandler> mStaleTimerHandler;
 | 
						|
    sp<EarlyBoostHandler> mEarlyBoostHandler;
 | 
						|
    std::atomic<time_point<steady_clock>> mLastUpdatedTime;
 | 
						|
    sp<MessageHandler> mPowerManagerHandler;
 | 
						|
    std::mutex mSessionLock;
 | 
						|
    std::atomic<bool> mSessionClosed = false;
 | 
						|
    // These 3 variables are for earlyboost work period estimation.
 | 
						|
    int64_t mLastStartedTimeNs;
 | 
						|
    int64_t mLastDurationNs;
 | 
						|
    int64_t mWorkPeriodNs;
 | 
						|
 | 
						|
    // To cache the status of whether ADPF hints are supported.
 | 
						|
    std::unordered_map<std::string, std::optional<bool>> mSupportedHints;
 | 
						|
};
 | 
						|
 | 
						|
}  // namespace pixel
 | 
						|
}  // namespace impl
 | 
						|
}  // namespace power
 | 
						|
}  // namespace hardware
 | 
						|
}  // namespace google
 | 
						|
}  // namespace aidl
 |