libcamera: Switch to the std::chrono API
Replace the clock_gettime()-based API with durations expressed as integers with the std::chrono API. Signed-off-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com> Reviewed-by: Niklas Söderlund <niklas.soderlund@ragnatech.se>
This commit is contained in:
parent
98dff063f2
commit
cecfeed61e
9 changed files with 67 additions and 75 deletions
|
@ -7,6 +7,7 @@
|
||||||
#ifndef __LIBCAMERA_TIMER_H__
|
#ifndef __LIBCAMERA_TIMER_H__
|
||||||
#define __LIBCAMERA_TIMER_H__
|
#define __LIBCAMERA_TIMER_H__
|
||||||
|
|
||||||
|
#include <chrono>
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
|
|
||||||
#include <libcamera/object.h>
|
#include <libcamera/object.h>
|
||||||
|
@ -22,12 +23,13 @@ public:
|
||||||
Timer(Object *parent = nullptr);
|
Timer(Object *parent = nullptr);
|
||||||
~Timer();
|
~Timer();
|
||||||
|
|
||||||
void start(unsigned int msec);
|
void start(unsigned int msec) { start(std::chrono::milliseconds(msec)); }
|
||||||
|
void start(std::chrono::milliseconds interval);
|
||||||
void stop();
|
void stop();
|
||||||
bool isRunning() const;
|
bool isRunning() const;
|
||||||
|
|
||||||
unsigned int interval() const { return interval_; }
|
std::chrono::milliseconds interval() const { return interval_; }
|
||||||
uint64_t deadline() const { return deadline_; }
|
std::chrono::steady_clock::time_point deadline() const { return deadline_; }
|
||||||
|
|
||||||
Signal<Timer *> timeout;
|
Signal<Timer *> timeout;
|
||||||
|
|
||||||
|
@ -38,8 +40,8 @@ private:
|
||||||
void registerTimer();
|
void registerTimer();
|
||||||
void unregisterTimer();
|
void unregisterTimer();
|
||||||
|
|
||||||
unsigned int interval_;
|
std::chrono::milliseconds interval_;
|
||||||
uint64_t deadline_;
|
std::chrono::steady_clock::time_point deadline_;
|
||||||
};
|
};
|
||||||
|
|
||||||
} /* namespace libcamera */
|
} /* namespace libcamera */
|
||||||
|
|
|
@ -5,6 +5,7 @@
|
||||||
* capture.cpp - Cam capture
|
* capture.cpp - Cam capture
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include <chrono>
|
||||||
#include <climits>
|
#include <climits>
|
||||||
#include <iomanip>
|
#include <iomanip>
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
|
@ -16,7 +17,7 @@
|
||||||
using namespace libcamera;
|
using namespace libcamera;
|
||||||
|
|
||||||
Capture::Capture(Camera *camera, CameraConfiguration *config)
|
Capture::Capture(Camera *camera, CameraConfiguration *config)
|
||||||
: camera_(camera), config_(config), writer_(nullptr), last_(0)
|
: camera_(camera), config_(config), writer_(nullptr)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -135,17 +136,13 @@ int Capture::capture(EventLoop *loop)
|
||||||
|
|
||||||
void Capture::requestComplete(Request *request, const std::map<Stream *, Buffer *> &buffers)
|
void Capture::requestComplete(Request *request, const std::map<Stream *, Buffer *> &buffers)
|
||||||
{
|
{
|
||||||
double fps = 0.0;
|
|
||||||
uint64_t now;
|
|
||||||
|
|
||||||
if (request->status() == Request::RequestCancelled)
|
if (request->status() == Request::RequestCancelled)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
struct timespec time;
|
std::chrono::steady_clock::time_point now = std::chrono::steady_clock::now();
|
||||||
clock_gettime(CLOCK_MONOTONIC, &time);
|
double fps = std::chrono::duration_cast<std::chrono::milliseconds>(now - last_).count();
|
||||||
now = time.tv_sec * 1000 + time.tv_nsec / 1000000;
|
fps = last_ != std::chrono::steady_clock::time_point() && fps
|
||||||
fps = now - last_;
|
? 1000.0 / fps : 0.0;
|
||||||
fps = last_ && fps ? 1000.0 / fps : 0.0;
|
|
||||||
last_ = now;
|
last_ = now;
|
||||||
|
|
||||||
std::stringstream info;
|
std::stringstream info;
|
||||||
|
|
|
@ -7,6 +7,7 @@
|
||||||
#ifndef __CAM_CAPTURE_H__
|
#ifndef __CAM_CAPTURE_H__
|
||||||
#define __CAM_CAPTURE_H__
|
#define __CAM_CAPTURE_H__
|
||||||
|
|
||||||
|
#include <chrono>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
|
||||||
#include <libcamera/camera.h>
|
#include <libcamera/camera.h>
|
||||||
|
@ -35,7 +36,7 @@ private:
|
||||||
|
|
||||||
std::map<libcamera::Stream *, std::string> streamName_;
|
std::map<libcamera::Stream *, std::string> streamName_;
|
||||||
BufferWriter *writer_;
|
BufferWriter *writer_;
|
||||||
uint64_t last_;
|
std::chrono::steady_clock::time_point last_;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif /* __CAM_CAPTURE_H__ */
|
#endif /* __CAM_CAPTURE_H__ */
|
||||||
|
|
|
@ -8,6 +8,7 @@
|
||||||
#include "event_dispatcher_poll.h"
|
#include "event_dispatcher_poll.h"
|
||||||
|
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
|
#include <chrono>
|
||||||
#include <iomanip>
|
#include <iomanip>
|
||||||
#include <poll.h>
|
#include <poll.h>
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
@ -20,6 +21,7 @@
|
||||||
|
|
||||||
#include "log.h"
|
#include "log.h"
|
||||||
#include "thread.h"
|
#include "thread.h"
|
||||||
|
#include "utils.h"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \file event_dispatcher_poll.h
|
* \file event_dispatcher_poll.h
|
||||||
|
@ -206,17 +208,12 @@ int EventDispatcherPoll::poll(std::vector<struct pollfd> *pollfds)
|
||||||
struct timespec timeout;
|
struct timespec timeout;
|
||||||
|
|
||||||
if (nextTimer) {
|
if (nextTimer) {
|
||||||
clock_gettime(CLOCK_MONOTONIC, &timeout);
|
utils::time_point now = utils::clock::now();
|
||||||
uint64_t now = timeout.tv_sec * 1000000000ULL + timeout.tv_nsec;
|
|
||||||
|
|
||||||
if (nextTimer->deadline() > now) {
|
if (nextTimer->deadline() > now)
|
||||||
uint64_t delta = nextTimer->deadline() - now;
|
timeout = utils::duration_to_timespec(nextTimer->deadline() - now);
|
||||||
timeout.tv_sec = delta / 1000000000ULL;
|
else
|
||||||
timeout.tv_nsec = delta % 1000000000ULL;
|
timeout = { 0, 0 };
|
||||||
} else {
|
|
||||||
timeout.tv_sec = 0;
|
|
||||||
timeout.tv_nsec = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
LOG(Event, Debug)
|
LOG(Event, Debug)
|
||||||
<< "timeout " << timeout.tv_sec << "."
|
<< "timeout " << timeout.tv_sec << "."
|
||||||
|
@ -295,10 +292,7 @@ void EventDispatcherPoll::processNotifiers(const std::vector<struct pollfd> &pol
|
||||||
|
|
||||||
void EventDispatcherPoll::processTimers()
|
void EventDispatcherPoll::processTimers()
|
||||||
{
|
{
|
||||||
struct timespec ts;
|
utils::time_point now = utils::clock::now();
|
||||||
uint64_t now;
|
|
||||||
clock_gettime(CLOCK_MONOTONIC, &ts);
|
|
||||||
now = ts.tv_sec * 1000000000ULL + ts.tv_nsec;
|
|
||||||
|
|
||||||
while (!timers_.empty()) {
|
while (!timers_.empty()) {
|
||||||
Timer *timer = timers_.front();
|
Timer *timer = timers_.front();
|
||||||
|
|
|
@ -7,8 +7,11 @@
|
||||||
#ifndef __LIBCAMERA_LOG_H__
|
#ifndef __LIBCAMERA_LOG_H__
|
||||||
#define __LIBCAMERA_LOG_H__
|
#define __LIBCAMERA_LOG_H__
|
||||||
|
|
||||||
|
#include <chrono>
|
||||||
#include <sstream>
|
#include <sstream>
|
||||||
|
|
||||||
|
#include "utils.h"
|
||||||
|
|
||||||
namespace libcamera {
|
namespace libcamera {
|
||||||
|
|
||||||
enum LogSeverity {
|
enum LogSeverity {
|
||||||
|
@ -60,7 +63,7 @@ public:
|
||||||
|
|
||||||
std::ostream &stream() { return msgStream_; }
|
std::ostream &stream() { return msgStream_; }
|
||||||
|
|
||||||
const struct timespec ×tamp() const { return timestamp_; }
|
const utils::time_point ×tamp() const { return timestamp_; }
|
||||||
LogSeverity severity() const { return severity_; }
|
LogSeverity severity() const { return severity_; }
|
||||||
const LogCategory &category() const { return category_; }
|
const LogCategory &category() const { return category_; }
|
||||||
const std::string &fileInfo() const { return fileInfo_; }
|
const std::string &fileInfo() const { return fileInfo_; }
|
||||||
|
@ -72,7 +75,7 @@ private:
|
||||||
std::ostringstream msgStream_;
|
std::ostringstream msgStream_;
|
||||||
const LogCategory &category_;
|
const LogCategory &category_;
|
||||||
LogSeverity severity_;
|
LogSeverity severity_;
|
||||||
struct timespec timestamp_;
|
utils::time_point timestamp_;
|
||||||
std::string fileInfo_;
|
std::string fileInfo_;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -11,7 +11,6 @@
|
||||||
#include <cstdlib>
|
#include <cstdlib>
|
||||||
#include <ctime>
|
#include <ctime>
|
||||||
#include <fstream>
|
#include <fstream>
|
||||||
#include <iomanip>
|
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <list>
|
#include <list>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
@ -78,17 +77,6 @@ static int log_severity_to_syslog(LogSeverity severity)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static std::string log_timespec_to_string(const struct timespec ×tamp)
|
|
||||||
{
|
|
||||||
std::ostringstream ossTimestamp;
|
|
||||||
ossTimestamp.fill('0');
|
|
||||||
ossTimestamp << "[" << timestamp.tv_sec / (60 * 60) << ":"
|
|
||||||
<< std::setw(2) << (timestamp.tv_sec / 60) % 60 << ":"
|
|
||||||
<< std::setw(2) << timestamp.tv_sec % 60 << "."
|
|
||||||
<< std::setw(9) << timestamp.tv_nsec << "]";
|
|
||||||
return ossTimestamp.str();
|
|
||||||
}
|
|
||||||
|
|
||||||
static const char *log_severity_name(LogSeverity severity)
|
static const char *log_severity_name(LogSeverity severity)
|
||||||
{
|
{
|
||||||
static const char *const names[] = {
|
static const char *const names[] = {
|
||||||
|
@ -216,10 +204,10 @@ void LogOutput::writeSyslog(const LogMessage &msg)
|
||||||
|
|
||||||
void LogOutput::writeStream(const LogMessage &msg)
|
void LogOutput::writeStream(const LogMessage &msg)
|
||||||
{
|
{
|
||||||
std::string str = std::string(log_timespec_to_string(msg.timestamp()) +
|
std::string str = "[" + utils::time_point_to_string(msg.timestamp()) +
|
||||||
log_severity_name(msg.severity()) + " " +
|
"]" + log_severity_name(msg.severity()) + " " +
|
||||||
msg.category().name() + " " + msg.fileInfo() + " " +
|
msg.category().name() + " " + msg.fileInfo() + " " +
|
||||||
msg.msg());
|
msg.msg();
|
||||||
stream_->write(str.c_str(), str.size());
|
stream_->write(str.c_str(), str.size());
|
||||||
stream_->flush();
|
stream_->flush();
|
||||||
}
|
}
|
||||||
|
@ -777,7 +765,7 @@ LogMessage::LogMessage(LogMessage &&other)
|
||||||
void LogMessage::init(const char *fileName, unsigned int line)
|
void LogMessage::init(const char *fileName, unsigned int line)
|
||||||
{
|
{
|
||||||
/* Log the timestamp, severity and file information. */
|
/* Log the timestamp, severity and file information. */
|
||||||
clock_gettime(CLOCK_MONOTONIC, ×tamp_);
|
timestamp_ = utils::clock::now();
|
||||||
|
|
||||||
std::ostringstream ossFileInfo;
|
std::ostringstream ossFileInfo;
|
||||||
ossFileInfo << utils::basename(fileName) << ":" << line;
|
ossFileInfo << utils::basename(fileName) << ":" << line;
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
|
|
||||||
#include <libcamera/timer.h>
|
#include <libcamera/timer.h>
|
||||||
|
|
||||||
#include <time.h>
|
#include <chrono>
|
||||||
|
|
||||||
#include <libcamera/camera_manager.h>
|
#include <libcamera/camera_manager.h>
|
||||||
#include <libcamera/event_dispatcher.h>
|
#include <libcamera/event_dispatcher.h>
|
||||||
|
@ -15,6 +15,7 @@
|
||||||
#include "log.h"
|
#include "log.h"
|
||||||
#include "message.h"
|
#include "message.h"
|
||||||
#include "thread.h"
|
#include "thread.h"
|
||||||
|
#include "utils.h"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \file timer.h
|
* \file timer.h
|
||||||
|
@ -42,7 +43,7 @@ LOG_DEFINE_CATEGORY(Timer)
|
||||||
* \param[in] parent The parent Object
|
* \param[in] parent The parent Object
|
||||||
*/
|
*/
|
||||||
Timer::Timer(Object *parent)
|
Timer::Timer(Object *parent)
|
||||||
: Object(parent), interval_(0), deadline_(0)
|
: Object(parent)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -52,22 +53,28 @@ Timer::~Timer()
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* \fn Timer::start(unsigned int msec)
|
||||||
* \brief Start or restart the timer with a timeout of \a msec
|
* \brief Start or restart the timer with a timeout of \a msec
|
||||||
* \param[in] msec The timer duration in milliseconds
|
* \param[in] msec The timer duration in milliseconds
|
||||||
*
|
*
|
||||||
* If the timer is already running it will be stopped and restarted.
|
* If the timer is already running it will be stopped and restarted.
|
||||||
*/
|
*/
|
||||||
void Timer::start(unsigned int msec)
|
|
||||||
{
|
|
||||||
struct timespec tp;
|
|
||||||
clock_gettime(CLOCK_MONOTONIC, &tp);
|
|
||||||
|
|
||||||
interval_ = msec;
|
/**
|
||||||
deadline_ = tp.tv_sec * 1000000000ULL + tp.tv_nsec + msec * 1000000ULL;
|
* \brief Start or restart the timer with a timeout of \a interval
|
||||||
|
* \param[in] interval The timer duration in milliseconds
|
||||||
|
*
|
||||||
|
* If the timer is already running it will be stopped and restarted.
|
||||||
|
*/
|
||||||
|
void Timer::start(std::chrono::milliseconds interval)
|
||||||
|
{
|
||||||
|
interval_ = interval;
|
||||||
|
deadline_ = utils::clock::now() + interval;
|
||||||
|
|
||||||
LOG(Timer, Debug)
|
LOG(Timer, Debug)
|
||||||
<< "Starting timer " << this << " with interval "
|
<< "Starting timer " << this << " with interval "
|
||||||
<< msec << ": deadline " << deadline_;
|
<< interval.count() << ": deadline "
|
||||||
|
<< utils::time_point_to_string(deadline_);
|
||||||
|
|
||||||
registerTimer();
|
registerTimer();
|
||||||
}
|
}
|
||||||
|
@ -84,7 +91,7 @@ void Timer::stop()
|
||||||
{
|
{
|
||||||
unregisterTimer();
|
unregisterTimer();
|
||||||
|
|
||||||
deadline_ = 0;
|
deadline_ = utils::time_point();
|
||||||
}
|
}
|
||||||
|
|
||||||
void Timer::registerTimer()
|
void Timer::registerTimer()
|
||||||
|
@ -103,7 +110,7 @@ void Timer::unregisterTimer()
|
||||||
*/
|
*/
|
||||||
bool Timer::isRunning() const
|
bool Timer::isRunning() const
|
||||||
{
|
{
|
||||||
return deadline_ != 0;
|
return deadline_ != utils::time_point();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -115,7 +122,7 @@ bool Timer::isRunning() const
|
||||||
/**
|
/**
|
||||||
* \fn Timer::deadline()
|
* \fn Timer::deadline()
|
||||||
* \brief Retrieve the timer deadline
|
* \brief Retrieve the timer deadline
|
||||||
* \return The timer deadline in nanoseconds
|
* \return The timer deadline
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -128,7 +135,7 @@ bool Timer::isRunning() const
|
||||||
void Timer::message(Message *msg)
|
void Timer::message(Message *msg)
|
||||||
{
|
{
|
||||||
if (msg->type() == Message::ThreadMoveMessage) {
|
if (msg->type() == Message::ThreadMoveMessage) {
|
||||||
if (deadline_) {
|
if (isRunning()) {
|
||||||
unregisterTimer();
|
unregisterTimer();
|
||||||
invokeMethod(&Timer::registerTimer);
|
invokeMethod(&Timer::registerTimer);
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,6 +5,7 @@
|
||||||
* event-dispatcher.cpp - Event dispatcher test
|
* event-dispatcher.cpp - Event dispatcher test
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include <chrono>
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <signal.h>
|
#include <signal.h>
|
||||||
#include <sys/time.h>
|
#include <sys/time.h>
|
||||||
|
@ -47,8 +48,7 @@ protected:
|
||||||
Timer timer;
|
Timer timer;
|
||||||
|
|
||||||
/* Event processing interruption by signal. */
|
/* Event processing interruption by signal. */
|
||||||
struct timespec start;
|
std::chrono::steady_clock::time_point start = std::chrono::steady_clock::now();
|
||||||
clock_gettime(CLOCK_MONOTONIC, &start);
|
|
||||||
|
|
||||||
timer.start(1000);
|
timer.start(1000);
|
||||||
|
|
||||||
|
@ -59,12 +59,11 @@ protected:
|
||||||
|
|
||||||
dispatcher->processEvents();
|
dispatcher->processEvents();
|
||||||
|
|
||||||
struct timespec stop;
|
std::chrono::steady_clock::time_point stop = std::chrono::steady_clock::now();
|
||||||
clock_gettime(CLOCK_MONOTONIC, &stop);
|
std::chrono::steady_clock::duration duration = stop - start;
|
||||||
int duration = (stop.tv_sec - start.tv_sec) * 1000;
|
int msecs = std::chrono::duration_cast<std::chrono::milliseconds>(duration).count();
|
||||||
duration += (stop.tv_nsec - start.tv_nsec) / 1000000;
|
|
||||||
|
|
||||||
if (abs(duration - 1000) > 50) {
|
if (abs(msecs - 1000) > 50) {
|
||||||
cout << "Event processing restart test failed" << endl;
|
cout << "Event processing restart test failed" << endl;
|
||||||
return TestFail;
|
return TestFail;
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,6 +5,7 @@
|
||||||
* timer.cpp - Timer test
|
* timer.cpp - Timer test
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include <chrono>
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
|
|
||||||
#include <libcamera/event_dispatcher.h>
|
#include <libcamera/event_dispatcher.h>
|
||||||
|
@ -28,28 +29,28 @@ public:
|
||||||
void start(int msec)
|
void start(int msec)
|
||||||
{
|
{
|
||||||
interval_ = msec;
|
interval_ = msec;
|
||||||
clock_gettime(CLOCK_MONOTONIC, &start_);
|
start_ = std::chrono::steady_clock::now();
|
||||||
expiration_ = { 0, 0 };
|
expiration_ = std::chrono::steady_clock::time_point();
|
||||||
|
|
||||||
Timer::start(msec);
|
Timer::start(msec);
|
||||||
}
|
}
|
||||||
|
|
||||||
int jitter()
|
int jitter()
|
||||||
{
|
{
|
||||||
int duration = (expiration_.tv_sec - start_.tv_sec) * 1000;
|
std::chrono::steady_clock::duration duration = expiration_ - start_;
|
||||||
duration += (expiration_.tv_nsec - start_.tv_nsec) / 1000000;
|
int msecs = std::chrono::duration_cast<std::chrono::milliseconds>(duration).count();
|
||||||
return abs(duration - interval_);
|
return abs(msecs - interval_);
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void timeoutHandler(Timer *timer)
|
void timeoutHandler(Timer *timer)
|
||||||
{
|
{
|
||||||
clock_gettime(CLOCK_MONOTONIC, &expiration_);
|
expiration_ = std::chrono::steady_clock::now();
|
||||||
}
|
}
|
||||||
|
|
||||||
int interval_;
|
int interval_;
|
||||||
struct timespec start_;
|
std::chrono::steady_clock::time_point start_;
|
||||||
struct timespec expiration_;
|
std::chrono::steady_clock::time_point expiration_;
|
||||||
};
|
};
|
||||||
|
|
||||||
class TimerTest : public Test
|
class TimerTest : public Test
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue