diff --git a/CMakeLists.txt b/CMakeLists.txt index c3c25a47751f03c062f58cf93ca38c0210d4e92c..53dc9b883599a657c3c971487ac298bbdf0b46cf 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -29,7 +29,7 @@ set(CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} "${CMAKE_SOURCE_DIR}/cmake/") include(Findglog) -find_package( OpenCV REQUIRED ) +find_package( OpenCV REQUIRED COMPONENTS core imgproc highgui cudaimgproc calib3d imgcodecs videoio aruco cudaarithm cudastereo cudaoptflow face tracking quality) find_package( Threads REQUIRED ) find_package( URIParser REQUIRED ) find_package( MsgPack REQUIRED ) diff --git a/applications/vision/src/main.cpp b/applications/vision/src/main.cpp index fde5bf127e39999da6d3429b4ec1c6cc041b8433..3f8dca5abd7fa299d903f0c2cf4969b1c5dda4f6 100644 --- a/applications/vision/src/main.cpp +++ b/applications/vision/src/main.cpp @@ -55,6 +55,8 @@ static void run(ftl::Configurable *root) { Universe *net = ftl::create<Universe>(root, "net"); ftl::ctrl::Master ctrl(root, net); + ftl::timer::setHighPrecision(true); + auto paths = root->get<vector<string>>("paths"); string file = ""; if (paths && (*paths).size() > 0) file = (*paths)[(*paths).size()-1]; diff --git a/components/common/cpp/include/ftl/timer.hpp b/components/common/cpp/include/ftl/timer.hpp index 378cd3d273621a6c40ac07f0f03dec14675192e5..bf378425d191991da8a9317d2878a12dc6a92105 100644 --- a/components/common/cpp/include/ftl/timer.hpp +++ b/components/common/cpp/include/ftl/timer.hpp @@ -69,9 +69,20 @@ int64_t get_time(); */ void setInterval(int ms); +/** + * The highprec parameter sets whether or not this + * timer should be high precision on the calling interval. A high precision + * timer involves spinning the cpu to achieve millisecond accuracy. + */ +void setHighPrecision(bool hp); + int getInterval(); +/** + * Get current (monotonic) time in milliseconds. + */ int64_t get_time(); + int64_t get_time_micro(); double get_time_seconds(); diff --git a/components/common/cpp/src/timer.cpp b/components/common/cpp/src/timer.cpp index d58386c0174444471d9279425dad8093bebcf677..79fb8720bd43fb7bbd35f1fff3fa0f7a537d195e 100644 --- a/components/common/cpp/src/timer.cpp +++ b/components/common/cpp/src/timer.cpp @@ -20,6 +20,7 @@ using namespace ftl::timer; static int64_t last_frame = 0; static int64_t mspf = 50; +static bool hprec_ = false; static int64_t clock_adjust = 0; static bool active = false; static std::atomic<int> active_jobs = 0; @@ -59,7 +60,7 @@ static void waitTimePoint() { int64_t msdelay = mspf - (now % mspf); int64_t sincelast = now - last_frame*mspf; - if (sincelast > mspf) LOG(WARNING) << "Frame " << "(" << (target-last_frame) << ") dropped by " << sincelast << "ms"; + if (hprec_ && sincelast > mspf) LOG(WARNING) << "Frame " << "(" << (target-last_frame) << ") dropped by " << (sincelast-mspf) << "ms"; // Use sleep_for for larger delays @@ -89,15 +90,21 @@ static void waitTimePoint() { } } - // Spin loop until exact grab time - //LOG(INFO) << "Spin Delay: " << (now / 40) << " = " << (40 - (now%40)); - - if (sincelast != mspf) { - target = now / mspf; - while ((now/mspf) == target) { - _mm_pause(); // SSE2 nano pause intrinsic - now = get_time(); - }; + if (hprec_) { + // Spin loop until exact grab time + // Accurate to around 4 micro seconds. + if (sincelast != mspf) { + // TODO: Try using sleep_for for msdelay-1 + target = now / mspf; + while ((now/mspf) == target) { + _mm_pause(); // SSE2 nano pause intrinsic + now = get_time(); + }; + } + } else { + // Accurate to just under 1 millisecond usually + if (sincelast != mspf) sleep_for(milliseconds(msdelay)); + now = get_time(); } last_frame = now/mspf; } @@ -106,6 +113,10 @@ void ftl::timer::setInterval(int ms) { mspf = ms; } +void ftl::timer::setHighPrecision(bool hp) { + hprec_ = hp; +} + int ftl::timer::getInterval() { return mspf; }