blob: 52a84c470ce55cc17cccec3232eb70e632c6ecd6 [file] [log] [blame]
/*
* Copyright 2024 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 "InputThread.h"
#include "InputTracingPerfettoBackend.h"
#include <android-base/thread_annotations.h>
#include <mutex>
#include <variant>
#include <vector>
namespace android::inputdispatcher::trace::impl {
/**
* A wrapper around an InputTracingBackend implementation that writes to the inner tracing backend
* from a single new thread that it creates. The new tracing thread is started when the
* ThreadedBackend is created, and is stopped when it is destroyed. The ThreadedBackend is
* thread-safe.
*/
template <typename Backend>
class ThreadedBackend : public InputTracingBackendInterface {
public:
ThreadedBackend(Backend&& innerBackend);
~ThreadedBackend() override;
void traceKeyEvent(const TracedKeyEvent&, const TracedEventMetadata&) override;
void traceMotionEvent(const TracedMotionEvent&, const TracedEventMetadata&) override;
void traceWindowDispatch(const WindowDispatchArgs&, const TracedEventMetadata&) override;
/** Returns a function that, when called, will block until the tracing thread is idle. */
std::function<void()> getIdleWaiterForTesting();
private:
std::mutex mLock;
bool mThreadExit GUARDED_BY(mLock){false};
std::condition_variable mThreadWakeCondition;
Backend mBackend;
using TraceEntry =
std::pair<std::variant<TracedKeyEvent, TracedMotionEvent, WindowDispatchArgs>,
TracedEventMetadata>;
std::vector<TraceEntry> mQueue GUARDED_BY(mLock);
struct IdleWaiter {
std::mutex idleLock;
std::condition_variable threadIdleCondition;
bool isIdle GUARDED_BY(idleLock){false};
};
// The lazy-initialized object used to wait for the tracing thread to idle.
std::shared_ptr<IdleWaiter> mIdleWaiter GUARDED_BY(mLock);
// InputThread stops when its destructor is called. Initialize it last so that it is the
// first thing to be destructed. This will guarantee the thread will not access other
// members that have already been destructed.
InputThread mTracerThread;
void threadLoop();
void setIdleStatus(bool isIdle) REQUIRES(mLock);
};
} // namespace android::inputdispatcher::trace::impl