slightly better, skips occasional frames but not horrid

pull/34892/head
Trey Moen 7 months ago
parent 4e0a8f3094
commit 282cebc71b
  1. 56
      tools/clip/application.cc
  2. 1
      tools/clip/application.h
  3. 24
      tools/clip/recorder/widget.cc
  4. 3
      tools/clip/recorder/widget.h

@ -2,6 +2,7 @@
#include <QApplication> #include <QApplication>
#include <QTranslator> #include <QTranslator>
#include <QWindow>
#include <selfdrive/ui/qt/util.h> #include <selfdrive/ui/qt/util.h>
#include <selfdrive/ui/qt/window.h> #include <selfdrive/ui/qt/window.h>
@ -73,30 +74,54 @@ Application::Application(int argc, char *argv[], QObject *parent) : QObject(pare
recorder->moveToThread(recorderThread); recorder->moveToThread(recorderThread);
connect(recorderThread, &QThread::finished, recorder, &QObject::deleteLater); connect(recorderThread, &QThread::finished, recorder, &QObject::deleteLater);
connect(app, &QCoreApplication::aboutToQuit, recorderThread, &QThread::quit); connect(app, &QCoreApplication::aboutToQuit, recorderThread, &QThread::quit);
recorderThread->start();
// Initialize and start replay
initReplay(route.toStdString());
replayThread = QThread::create([this, startTime] { startReplay(startTime); });
replayThread->start();
QTimer *loop = new QTimer; // Frame capture optimization
connect(loop, &QTimer::timeout, this, [&]() { QElapsedTimer frameTimer;
frameTimer.start();
int64_t lastFrameTime = 0;
const int64_t frameInterval = 1000 / UI_FREQ; // Target frame interval in ms
loop = new QTimer;
connect(loop, &QTimer::timeout, this, [&, frameTimer, lastFrameTime]() mutable {
if (!window->isVisible()) { if (!window->isVisible()) {
return; return;
} }
QElapsedTimer timer;
timer.start(); int64_t currentTime = frameTimer.elapsed();
int64_t elapsedSinceLastFrame = currentTime - lastFrameTime;
// Skip frame if we're ahead of schedule
if (elapsedSinceLastFrame < frameInterval) {
return;
}
QPixmap pixmap = window->grab(); QPixmap pixmap = window->grab();
qDebug() << "pixmap took " << timer.elapsed() << " ms";
timer.restart(); // Only process frame if capture was successful
if (!pixmap.isNull()) {
recorder->saveFrame(std::make_shared<QPixmap>(std::move(pixmap))); recorder->saveFrame(std::make_shared<QPixmap>(std::move(pixmap)));
lastFrameTime = currentTime;
}
}); });
loop->start(1000 / UI_FREQ);
// Use a higher timer resolution for more precise frame timing
loop->setTimerType(Qt::PreciseTimer);
loop->start(1); // Run at highest possible frequency, we'll control frame rate ourselves
window->setAttribute(Qt::WA_DontShowOnScreen); window->setAttribute(Qt::WA_DontShowOnScreen);
window->setAttribute(Qt::WA_OpaquePaintEvent); window->setAttribute(Qt::WA_OpaquePaintEvent);
window->setAttribute(Qt::WA_NoSystemBackground); window->setAttribute(Qt::WA_NoSystemBackground);
recorderThread->start(); window->setAttribute(Qt::WA_TranslucentBackground, false);
window->setAttribute(Qt::WA_AlwaysStackOnTop);
// Initialize and start replay window->setAttribute(Qt::WA_ShowWithoutActivating);
initReplay(route.toStdString()); window->setAttribute(Qt::WA_UpdatesDisabled);
replayThread = QThread::create([this, startTime] { startReplay(startTime); }); window->setAttribute(Qt::WA_StaticContents);
replayThread->start();
} }
void Application::initReplay(const std::string& route) { void Application::initReplay(const std::string& route) {
@ -132,15 +157,18 @@ Application::~Application() {
if (recorderThread) { if (recorderThread) {
recorderThread->quit(); recorderThread->quit();
recorderThread->wait(); recorderThread->wait();
delete recorderThread;
} }
delete recorder;
delete window; delete window;
delete loop;
delete app; delete app;
} }
int Application::exec() const { int Application::exec() const {
// TODO: modify Replay to block until all OnroadWindow required messages have been broadcast at least once // TODO: modify Replay to block until all OnroadWindow required messages have been broadcast at least once
std::this_thread::sleep_for(std::chrono::seconds(5)); std::this_thread::sleep_for(std::chrono::seconds(8));
setMainWindow(window); setMainWindow(window);
return app->exec(); return app->exec();
} }

@ -22,6 +22,7 @@ private:
QThread *recorderThread = nullptr; QThread *recorderThread = nullptr;
Recorder *recorder = nullptr; Recorder *recorder = nullptr;
OnroadWindow *window; OnroadWindow *window;
QTimer *loop;
// Replay related members // Replay related members
std::unique_ptr<Replay> replay; std::unique_ptr<Replay> replay;

@ -3,37 +3,47 @@
#include "tools/clip/recorder/ffmpeg.h" #include "tools/clip/recorder/ffmpeg.h"
Recorder::Recorder(const std::string& outputFile, QObject *parent) : QObject(parent) { Recorder::Recorder(const std::string& outputFile, QObject *parent) : QObject(parent) {
encoder = new FFmpegEncoder(outputFile, DEVICE_SCREEN_SIZE.width(), DEVICE_SCREEN_SIZE.height(), UI_FREQ); const float scale = util::getenv("SCALE", 1.0f);
encoder = new FFmpegEncoder(outputFile, DEVICE_SCREEN_SIZE.width() * scale, DEVICE_SCREEN_SIZE.height() * scale, UI_FREQ);
} }
Recorder::~Recorder() { Recorder::~Recorder() {
keepRunning = false; // Signal processing thread to stop
delete encoder; delete encoder;
} }
void Recorder::saveFrame(const std::shared_ptr<QPixmap> &frame) { void Recorder::saveFrame(const std::shared_ptr<QPixmap> &frame) {
QMutexLocker locker(&mutex); QMutexLocker locker(&mutex);
// Drop frame if queue is full
if (frameQueue.size() >= MAX_QUEUE_SIZE) {
qDebug() << "Dropping frame";
return;
}
frameQueue.enqueue(frame); frameQueue.enqueue(frame);
if (!isProcessing) { if (isProcessing.loadRelaxed() == 0) {
isProcessing = true; isProcessing.storeRelaxed(1);
QMetaObject::invokeMethod(this, &Recorder::processQueue, Qt::QueuedConnection); QMetaObject::invokeMethod(this, &Recorder::processQueue, Qt::QueuedConnection);
} }
} }
void Recorder::processQueue() { void Recorder::processQueue() {
while (true) { while (keepRunning) {
std::shared_ptr<QPixmap> frame; std::shared_ptr<QPixmap> frame;
{ {
QMutexLocker locker(&mutex); QMutexLocker locker(&mutex);
if (frameQueue.isEmpty() || !keepRunning) { if (frameQueue.isEmpty()) {
isProcessing = false; isProcessing.storeRelaxed(0);
return; return;
} }
frame = frameQueue.dequeue(); frame = frameQueue.dequeue();
} }
if (!encoder->writeFrame(frame->toImage().convertToFormat(QImage::Format_ARGB32_Premultiplied))) { if (!encoder->writeFrame(frame->toImage().convertToFormat(QImage::Format_ARGB32))) {
fprintf(stderr, "did not write\n"); fprintf(stderr, "did not write\n");
} }
} }
isProcessing.storeRelaxed(0);
} }

@ -19,10 +19,11 @@ public slots:
void saveFrame(const std::shared_ptr<QPixmap> &frame); void saveFrame(const std::shared_ptr<QPixmap> &frame);
private: private:
static constexpr int MAX_QUEUE_SIZE = 30; // Limit queue size to prevent memory growth
FFmpegEncoder *encoder; FFmpegEncoder *encoder;
QQueue<std::shared_ptr<QPixmap>> frameQueue; QQueue<std::shared_ptr<QPixmap>> frameQueue;
QMutex mutex; QMutex mutex;
bool isProcessing = false; QAtomicInt isProcessing{0}; // Use atomic for thread safety
bool keepRunning = true; bool keepRunning = true;
void processQueue(); void processQueue();
}; };

Loading…
Cancel
Save