#ifndef ANDROID_DVR_SERVICES_DISPLAYD_VSYNC_SERVICE_H_ #define ANDROID_DVR_SERVICES_DISPLAYD_VSYNC_SERVICE_H_ #include <pdx/service.h> #include <list> #include <memory> #include <mutex> #include <thread> #include "display_service.h" namespace android { namespace dvr { // VSyncWaiter encapsulates a client blocked waiting for the next vsync. // It is used to enqueue the Message to reply to when the next vsync event // occurs. class VSyncWaiter { public: explicit VSyncWaiter(pdx::Message& message) : message_(std::move(message)) {} void Notify(int64_t timestamp); private: pdx::Status<int64_t> OnWait(pdx::Message& message); pdx::Message message_; int64_t timestamp_ = 0; VSyncWaiter(const VSyncWaiter&) = delete; void operator=(const VSyncWaiter&) = delete; }; // VSyncChannel manages the service-side per-client context for each client // using the service. class VSyncChannel : public pdx::Channel { public: VSyncChannel(pdx::Service& service, int pid, int cid) : service_(service), pid_(pid), cid_(cid) {} void Ack(); void Signal(); private: pdx::Service& service_; pid_t pid_; int cid_; VSyncChannel(const VSyncChannel&) = delete; void operator=(const VSyncChannel&) = delete; }; // VSyncService implements the displayd vsync service over ServiceFS. class VSyncService : public pdx::ServiceBase<VSyncService> { public: ~VSyncService() override; pdx::Status<void> HandleMessage(pdx::Message& message) override; std::shared_ptr<pdx::Channel> OnChannelOpen(pdx::Message& message) override; void OnChannelClose(pdx::Message& message, const std::shared_ptr<pdx::Channel>& channel) override; // Called by the hardware composer HAL, or similar, whenever a vsync event // occurs on the primary display. |compositor_time_ns| is the number of ns // before the next vsync when the compositor will preempt the GPU to do EDS // and lens warp. void VSyncEvent(int64_t timestamp_ns, int64_t compositor_time_ns, uint32_t vsync_count); private: friend BASE; VSyncService(); pdx::Status<int64_t> OnGetLastTimestamp(pdx::Message& message); pdx::Status<display::VSyncSchedInfo> OnGetSchedInfo(pdx::Message& message); pdx::Status<void> OnAcknowledge(pdx::Message& message); void NotifierThreadFunction(); void AddWaiter(pdx::Message& message); void NotifyWaiters(); void UpdateClients(); void AddClient(const std::shared_ptr<VSyncChannel>& client); void RemoveClient(const std::shared_ptr<VSyncChannel>& client); int64_t last_vsync_; int64_t current_vsync_; int64_t compositor_time_ns_; uint32_t current_vsync_count_; std::mutex mutex_; std::list<std::unique_ptr<VSyncWaiter>> waiters_; std::list<std::shared_ptr<VSyncChannel>> clients_; VSyncService(const VSyncService&) = delete; void operator=(VSyncService&) = delete; }; } // namespace dvr } // namespace android #endif // ANDROID_DVR_SERVICES_DISPLAYD_VSYNC_SERVICE_H_