| /* | 
 |  * Copyright (C) 2009 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 <cstdint> | 
 | #include <future> | 
 | #include <type_traits> | 
 | #include <utility> | 
 |  | 
 | #include <android-base/thread_annotations.h> | 
 | #include <gui/IDisplayEventConnection.h> | 
 | #include <private/gui/BitTube.h> | 
 | #include <utils/Looper.h> | 
 | #include <utils/Timers.h> | 
 |  | 
 | #include "EventThread.h" | 
 | #include "TracedOrdinal.h" | 
 | #include "VSyncDispatch.h" | 
 |  | 
 | namespace android { | 
 |  | 
 | class SurfaceFlinger; | 
 |  | 
 | template <typename F> | 
 | class Task : public MessageHandler { | 
 |     template <typename G> | 
 |     friend auto makeTask(G&&); | 
 |  | 
 |     explicit Task(F&& f) : mTask(std::move(f)) {} | 
 |  | 
 |     void handleMessage(const Message&) override { mTask(); } | 
 |  | 
 |     using T = std::invoke_result_t<F>; | 
 |     std::packaged_task<T()> mTask; | 
 | }; | 
 |  | 
 | template <typename F> | 
 | inline auto makeTask(F&& f) { | 
 |     sp<Task<F>> task = new Task<F>(std::move(f)); | 
 |     return std::make_pair(task, task->mTask.get_future()); | 
 | } | 
 |  | 
 | class MessageQueue { | 
 | public: | 
 |     enum { | 
 |         INVALIDATE = 0, | 
 |         REFRESH = 1, | 
 |     }; | 
 |  | 
 |     virtual ~MessageQueue() = default; | 
 |  | 
 |     virtual void init(const sp<SurfaceFlinger>& flinger) = 0; | 
 |     virtual void initVsync(scheduler::VSyncDispatch&, frametimeline::TokenManager&, | 
 |                            std::chrono::nanoseconds workDuration) = 0; | 
 |     virtual void setDuration(std::chrono::nanoseconds workDuration) = 0; | 
 |     virtual void setInjector(sp<EventThreadConnection>) = 0; | 
 |     virtual void waitMessage() = 0; | 
 |     virtual void postMessage(sp<MessageHandler>&&) = 0; | 
 |     virtual void invalidate() = 0; | 
 |     virtual void refresh() = 0; | 
 |     virtual std::optional<std::chrono::steady_clock::time_point> nextExpectedInvalidate() = 0; | 
 | }; | 
 |  | 
 | // --------------------------------------------------------------------------- | 
 |  | 
 | namespace impl { | 
 |  | 
 | class MessageQueue : public android::MessageQueue { | 
 | protected: | 
 |     class Handler : public MessageHandler { | 
 |         enum : uint32_t { | 
 |             eventMaskInvalidate = 0x1, | 
 |             eventMaskRefresh = 0x2, | 
 |             eventMaskTransaction = 0x4 | 
 |         }; | 
 |         MessageQueue& mQueue; | 
 |         std::atomic<uint32_t> mEventMask; | 
 |         std::atomic<int64_t> mVsyncId; | 
 |         std::atomic<nsecs_t> mExpectedVSyncTime; | 
 |  | 
 |     public: | 
 |         explicit Handler(MessageQueue& queue) : mQueue(queue), mEventMask(0) {} | 
 |         void handleMessage(const Message& message) override; | 
 |         virtual void dispatchRefresh(); | 
 |         virtual void dispatchInvalidate(int64_t vsyncId, nsecs_t expectedVSyncTimestamp); | 
 |         virtual bool invalidatePending(); | 
 |     }; | 
 |  | 
 |     friend class Handler; | 
 |  | 
 |     sp<SurfaceFlinger> mFlinger; | 
 |     sp<Looper> mLooper; | 
 |  | 
 |     struct Vsync { | 
 |         frametimeline::TokenManager* tokenManager = nullptr; | 
 |         std::unique_ptr<scheduler::VSyncCallbackRegistration> registration; | 
 |  | 
 |         std::mutex mutex; | 
 |         TracedOrdinal<std::chrono::nanoseconds> workDuration | 
 |                 GUARDED_BY(mutex) = {"VsyncWorkDuration-sf", std::chrono::nanoseconds(0)}; | 
 |         std::chrono::nanoseconds lastCallbackTime GUARDED_BY(mutex) = std::chrono::nanoseconds{0}; | 
 |         bool scheduled GUARDED_BY(mutex) = false; | 
 |         std::optional<nsecs_t> expectedWakeupTime GUARDED_BY(mutex); | 
 |         TracedOrdinal<int> value = {"VSYNC-sf", 0}; | 
 |     }; | 
 |  | 
 |     struct Injector { | 
 |         gui::BitTube tube; | 
 |         std::mutex mutex; | 
 |         sp<EventThreadConnection> connection GUARDED_BY(mutex); | 
 |     }; | 
 |  | 
 |     Vsync mVsync; | 
 |     Injector mInjector; | 
 |  | 
 |     sp<Handler> mHandler; | 
 |  | 
 |     void vsyncCallback(nsecs_t vsyncTime, nsecs_t targetWakeupTime, nsecs_t readyTime); | 
 |     void injectorCallback(); | 
 |  | 
 | public: | 
 |     ~MessageQueue() override = default; | 
 |     void init(const sp<SurfaceFlinger>& flinger) override; | 
 |     void initVsync(scheduler::VSyncDispatch&, frametimeline::TokenManager&, | 
 |                    std::chrono::nanoseconds workDuration) override; | 
 |     void setDuration(std::chrono::nanoseconds workDuration) override; | 
 |     void setInjector(sp<EventThreadConnection>) override; | 
 |  | 
 |     void waitMessage() override; | 
 |     void postMessage(sp<MessageHandler>&&) override; | 
 |  | 
 |     // sends INVALIDATE message at next VSYNC | 
 |     void invalidate() override; | 
 |  | 
 |     // sends REFRESH message at next VSYNC | 
 |     void refresh() override; | 
 |  | 
 |     std::optional<std::chrono::steady_clock::time_point> nextExpectedInvalidate() override; | 
 | }; | 
 |  | 
 | } // namespace impl | 
 | } // namespace android |