From add3b8c512a48498f0110a7d178c48e510164479 Mon Sep 17 00:00:00 2001 From: Dave Date: Thu, 16 Jan 2025 20:33:15 -0500 Subject: [PATCH 1/2] Refactor/refine 10 Hz update logic --- gui/src/chcanv.cpp | 26 +++++--- gui/src/ocpn_frame.cpp | 132 ++++++++++++++++++++++++++++++----------- 2 files changed, 114 insertions(+), 44 deletions(-) diff --git a/gui/src/chcanv.cpp b/gui/src/chcanv.cpp index 8a8c78fc8e..da8e5c5937 100644 --- a/gui/src/chcanv.cpp +++ b/gui/src/chcanv.cpp @@ -325,6 +325,7 @@ extern int g_nAIS_activity_timer; extern bool g_bskew_comp; extern float g_compass_scalefactor; extern int g_COGAvgSec; // COG average period (sec.) for Course Up Mode +extern bool g_btenhertz; wxGLContext *g_pGLcontext; // shared common context @@ -1694,7 +1695,7 @@ bool ChartCanvas::DoCanvasUpdate(void) { GetCanvasScaleFactor() / proposed_scale_onscreen, 0, GetVPRotation()); } - if (m_bFollow) { + if (m_bFollow && g_btenhertz) { StartTimedMovementVP(vpLat, vpLon); } else { bNewView |= SetViewPoint(vpLat, vpLon, GetVPScale(), 0, GetVPRotation()); @@ -3370,9 +3371,9 @@ bool ChartCanvas::DoCanvasCOGSet(void) { m_VPRotate = -g_COGAvg * PI / 180.; SetVPRotation(m_VPRotate); - bool bnew_chart = DoCanvasUpdate(); + // bool bnew_chart = DoCanvasUpdate(); - if ((bnew_chart) || (old_VPRotate != m_VPRotate)) ReloadVP(); + // if ((bnew_chart) || (old_VPRotate != m_VPRotate)) ReloadVP(); return true; } @@ -3419,7 +3420,7 @@ bool ChartCanvas::StartTimedMovement(bool stoptimer) { return true; } - +int stvpc; void ChartCanvas::StartTimedMovementVP(double target_lat, double target_lon) { // Save the target m_target_lat = target_lat; @@ -3431,21 +3432,30 @@ void ChartCanvas::StartTimedMovementVP(double target_lat, double target_lon) { m_VPMovementTimer.Start(1, true); // oneshot m_timed_move_vp_active = true; + stvpc = 0; } void ChartCanvas::DoTimedMovementVP() { if (!m_timed_move_vp_active) return; // not active // Stop condition double one_pix = (1. / (1852 * 60)) / GetVP().view_scale_ppm; + double d2 = + pow(m_run_lat - m_target_lat, 2) + pow(m_run_lon - m_target_lon, 2); - if ((fabs(m_run_lat - m_target_lat) < one_pix) || - (fabs(m_run_lat - m_target_lat) < one_pix)) { + if (d2 < one_pix) { + SetViewPoint(m_target_lat, m_target_lon); // Embeds a refresh StopMovementVP(); return; } - double new_lat = GetVP().clat + (m_target_lat - m_start_lat) / 10; - double new_lon = GetVP().clon + (m_target_lon - m_start_lon) / 10; + // if ((fabs(m_run_lat - m_target_lat) < one_pix) && + // (fabs(m_run_lon - m_target_lon) < one_pix)) { + // StopMovementVP(); + // return; + // } + + double new_lat = GetVP().clat + (m_target_lat - m_start_lat) / 5; + double new_lon = GetVP().clon + (m_target_lon - m_start_lon) / 5; m_run_lat = new_lat; m_run_lon = new_lon; diff --git a/gui/src/ocpn_frame.cpp b/gui/src/ocpn_frame.cpp index b507b4fe62..6bca25c4f4 100644 --- a/gui/src/ocpn_frame.cpp +++ b/gui/src/ocpn_frame.cpp @@ -5143,8 +5143,20 @@ void MyFrame::HandleBasicNavMsg(std::shared_ptr msg) { ((msg->vflag & POS_VALID) == POS_VALID)) { // Save the reported fix as the best available "ground truth" uint64_t fix_time_gt_last = fix_time_gt; - fix_time_gt = msg->set_time.tv_sec * 1e9 + msg->set_time.tv_nsec; - fix_time_interval = (fix_time_gt - fix_time_gt_last) / (double)1e9; + uint64_t fix_time_gt_now = + msg->set_time.tv_sec * 1e9 + msg->set_time.tv_nsec; + fix_time_interval = (fix_time_gt_now - fix_time_gt_last) / (double)1e9; + + // Calculate an implied SOG from the position change and time interval + double dist, brg; + DistanceBearingMercator(gLat, gLon, gLat_gt, gLon_gt, &brg, &dist); + double implied_sog = dist / (fix_time_interval / 3600); + + // printf("-------------- SOG: %g %g %g\n", fix_time_interval, gSog, + // implied_sog); + if (dist < .001) { + return; // probably a duplicate message, ignore it + } // shuffle history data gLat_gt_m2 = gLat_gt_m1; @@ -5157,8 +5169,19 @@ void MyFrame::HandleBasicNavMsg(std::shared_ptr msg) { gLon_gt = gLon; gCog_gt = gCog; gSog_gt = gSog; + fix_time_gt = fix_time_gt_now; if (std::isnan(gCog_gt_m1)) return; // Startup + + if ((fabs(gSog - implied_sog) / gSog) > 0.5) { + // Probably a synthetic data stream, with multiple position sources. + // Do not try to interpolate position at 10 Hz. + gSog_gt = 0; + cog_rate_gt = 0; + // printf("---Skip SOG\n"); + return; + } + // Calculate an estimated Rate-of-turn double diff = gCog_gt - gCog_gt_m1; double tentative_cog_rate_gt = diff / (fix_time_gt - fix_time_gt_last); @@ -5497,51 +5520,79 @@ void MyFrame::CheckToolbarPosition() { } void MyFrame::OnFrameTenHzTimer(wxTimerEvent &event) { - if (!g_btenhertz) return; - - if (std::isnan(gCog)) return; - if (std::isnan(gSog)) return; + // Check to see if in non-North-Up mode + bool b_rotate = false; + for (ChartCanvas *cc : g_canvasArray) { + if (cc) b_rotate |= (cc->GetUpMode() != NORTH_UP_MODE); + } - // Estimate current state by extrapolating from last "ground truth" state + if (!b_rotate && !g_btenhertz) return; // Nothing to do - struct timespec now; - clock_gettime(CLOCK_MONOTONIC, &now); - uint64_t diff = 1e9 * (now.tv_sec) + now.tv_nsec - fix_time_gt; - double diffc = diff / 1e9; // sec + bool b_update = false; + if (g_btenhertz) { + if (std::isnan(gCog)) return; + if (std::isnan(gSog)) return; - // Set gCog as estimated from last two ground truth fixes - gCog = gCog_gt + (cog_rate_gt * diffc); + // Estimate current state by extrapolating from last "ground truth" state - // And the same for gHdt - if (!std::isnan(gHdt_gt)) { - uint64_t diff = 1e9 * (now.tv_sec) + now.tv_nsec - hdt_time_gt; + struct timespec now; + clock_gettime(CLOCK_MONOTONIC, &now); + uint64_t diff = 1e9 * (now.tv_sec) + now.tv_nsec - fix_time_gt; double diffc = diff / 1e9; // sec - gHdt = gHdt_gt + (hdt_rate_gt * diffc); - } - // Estimate lat/lon position - double delta_t = diffc / 3600; // hours - double distance = gSog_gt * delta_t; // NMi + // Set gCog as estimated from last two ground truth fixes + gCog = gCog_gt + (cog_rate_gt * diffc); - // spherical (close enough) - double angr = gCog_gt / 180 * M_PI; - double latr = gLat_gt * M_PI / 180; - double D = distance / 3443; // earth radius in nm - double sD = sin(D), cD = cos(D); - double sy = sin(latr), cy = cos(latr); - double sa = sin(angr), ca = cos(angr); + // And the same for gHdt + if (!std::isnan(gHdt_gt)) { + uint64_t diff = 1e9 * (now.tv_sec) + now.tv_nsec - hdt_time_gt; + double diffc = diff / 1e9; // sec + gHdt = gHdt_gt + (hdt_rate_gt * diffc); + } - gLon = gLon_gt + asin(sa * sD / cy) * 180 / M_PI; - gLat = asin(sy * cD + cy * sD * ca) * 180 / M_PI; + // Estimate lat/lon position + if (gSog_gt) { + double delta_t = diffc / 3600; // hours + double distance = gSog_gt * delta_t; // NMi - for (unsigned int i = 0; i < g_canvasArray.GetCount(); i++) { - ChartCanvas *cc = g_canvasArray.Item(i); - if (cc) { - if (g_bopengl) { - cc->Refresh(false); + // spherical (close enough) + double angr = gCog_gt / 180 * M_PI; + double latr = gLat_gt * M_PI / 180; + double D = distance / 3443; // earth radius in nm + double sD = sin(D), cD = cos(D); + double sy = sin(latr), cy = cos(latr); + double sa = sin(angr), ca = cos(angr); + + gLon = gLon_gt + asin(sa * sD / cy) * 180 / M_PI; + gLat = asin(sy * cD + cy * sD * ca) * 180 / M_PI; + } + + b_update = true; + } + + // In a valid rotation mode ? + if (b_rotate) { + for (ChartCanvas *cc : g_canvasArray) { + if (cc) cc->DoCanvasCOGSet(); + } + b_update = true; + } + + if (b_update) { + // printf("10 Hz update\n"); + + for (ChartCanvas *cc : g_canvasArray) { + if (cc) { + if (g_bopengl) { + if (b_rotate || cc->m_bFollow) + cc->DoCanvasUpdate(); + else + cc->Refresh(); + } } } } + FrameTenHzTimer.Start(100, wxTIMER_CONTINUOUS); } @@ -5902,7 +5953,14 @@ void MyFrame::OnFrameTimer1(wxTimerEvent &event) { if (g_bopengl) { #ifdef ocpnUSE_GL if (cc->GetglCanvas()) { - cc->Refresh(false); + bool b_rotate = cc->GetUpMode() != NORTH_UP_MODE; + if (!b_rotate && !g_btenhertz) { + // printf("...........1 Hz update\n"); + if (cc->m_bFollow) + cc->DoCanvasUpdate(); + else + cc->Refresh(false); + } } #endif } else { @@ -6052,6 +6110,8 @@ void MyFrame::OnFrameTCTimer(wxTimerEvent &event) { // Keep and update the Viewport rotation angle according to average COG for // COGUP mode void MyFrame::OnFrameCOGTimer(wxTimerEvent &event) { + return; + // ..For each canvas... bool b_rotate = false; for (unsigned int i = 0; i < g_canvasArray.GetCount(); i++) { From fb991b50a6c0157bc3b3f087893125f9da9674d8 Mon Sep 17 00:00:00 2001 From: Dave Date: Thu, 16 Jan 2025 23:58:40 -0500 Subject: [PATCH 2/2] Refactor 1 Hz heartbeat timer tick, mainly housekeeping. --- gui/include/gui/ocpn_frame.h | 8 ++ gui/src/ocpn_frame.cpp | 251 ++++++++++++++++++----------------- 2 files changed, 137 insertions(+), 122 deletions(-) diff --git a/gui/include/gui/ocpn_frame.h b/gui/include/gui/ocpn_frame.h index e4e436ce7f..a842cb8c7a 100644 --- a/gui/include/gui/ocpn_frame.h +++ b/gui/include/gui/ocpn_frame.h @@ -339,7 +339,15 @@ class MyFrame : public wxFrame { void ConfigureStatusBar(); private: + void ProcessUnitTest(); + void ProcessQuitFlag(); + void ProcessDeferredTrackOn(); + void SendFixToPlugins(); + void ProcessAnchorWatch(); + void ProcessLogAndBells(); + void CheckToolbarPosition(); + void ODoSetSize(void); void DoCOGSet(void); diff --git a/gui/src/ocpn_frame.cpp b/gui/src/ocpn_frame.cpp index 6bca25c4f4..d41013e86e 100644 --- a/gui/src/ocpn_frame.cpp +++ b/gui/src/ocpn_frame.cpp @@ -5519,6 +5519,80 @@ void MyFrame::CheckToolbarPosition() { #endif } +void MyFrame::ProcessUnitTest() { + if (!g_bPauseTest && (g_unit_test_1 || g_unit_test_2)) { + // if((0 == ut_index) && GetQuiltMode()) + // ToggleQuiltMode(); + + // We use only one canvas for the unit tests, so far... + ChartCanvas *cc = GetPrimaryCanvas(); + + cc->m_bFollow = false; + if (g_MainToolbar && g_MainToolbar->GetToolbar()) + g_MainToolbar->GetToolbar()->ToggleTool(ID_FOLLOW, cc->m_bFollow); + int ut_index_max = ((g_unit_test_1 > 0) ? (g_unit_test_1 - 1) : INT_MAX); + + if (ChartData) { + if (cc->m_groupIndex > 0) { + while (ut_index < ChartData->GetChartTableEntries() && + !ChartData->IsChartInGroup(ut_index, cc->m_groupIndex)) { + ut_index++; + } + } + if (ut_index < ChartData->GetChartTableEntries()) { + // printf("%d / %d\n", ut_index, ChartData->GetChartTableEntries()); + const ChartTableEntry *cte = &ChartData->GetChartTableEntry(ut_index); + + double clat = (cte->GetLatMax() + cte->GetLatMin()) / 2; + double clon = (cte->GetLonMax() + cte->GetLonMin()) / 2; + + vLat = clat; + vLon = clon; + + cc->SetViewPoint(clat, clon); + + if (cc->GetQuiltMode()) { + if (cc->IsChartQuiltableRef(ut_index)) + cc->SelectQuiltRefdbChart(ut_index); + } else + cc->SelectdbChart(ut_index); + + double ppm; // final ppm scale to use + if (g_unit_test_1) { + ppm = cc->GetCanvasScaleFactor() / cte->GetScale(); + ppm /= 2; + } else { + double rw, rh; // width, height + int ww, wh; // chart window width, height + + // width in nm + DistanceBearingMercator(cte->GetLatMin(), cte->GetLonMin(), + cte->GetLatMin(), cte->GetLonMax(), NULL, + &rw); + + // height in nm + DistanceBearingMercator(cte->GetLatMin(), cte->GetLonMin(), + cte->GetLatMax(), cte->GetLonMin(), NULL, + &rh); + + cc->GetSize(&ww, &wh); + ppm = wxMin(ww / (rw * 1852), wh / (rh * 1852)) * (100 - fabs(clat)) / + 90; + ppm = wxMin(ppm, 1.0); + } + cc->SetVPScale(ppm); + + cc->ReloadVP(); + + ut_index++; + if (ut_index > ut_index_max) exit(0); + } else { + _exit(0); + } + } + } +} + void MyFrame::OnFrameTenHzTimer(wxTimerEvent &event) { // Check to see if in non-North-Up mode bool b_rotate = false; @@ -5596,82 +5670,7 @@ void MyFrame::OnFrameTenHzTimer(wxTimerEvent &event) { FrameTenHzTimer.Start(100, wxTIMER_CONTINUOUS); } -void MyFrame::OnFrameTimer1(wxTimerEvent &event) { - CheckToolbarPosition(); - - if (!g_bPauseTest && (g_unit_test_1 || g_unit_test_2)) { - // if((0 == ut_index) && GetQuiltMode()) - // ToggleQuiltMode(); - - // We use only one canvas for the unit tests, so far... - ChartCanvas *cc = GetPrimaryCanvas(); - - cc->m_bFollow = false; - if (g_MainToolbar && g_MainToolbar->GetToolbar()) - g_MainToolbar->GetToolbar()->ToggleTool(ID_FOLLOW, cc->m_bFollow); - int ut_index_max = ((g_unit_test_1 > 0) ? (g_unit_test_1 - 1) : INT_MAX); - - if (ChartData) { - if (cc->m_groupIndex > 0) { - while (ut_index < ChartData->GetChartTableEntries() && - !ChartData->IsChartInGroup(ut_index, cc->m_groupIndex)) { - ut_index++; - } - } - if (ut_index < ChartData->GetChartTableEntries()) { - // printf("%d / %d\n", ut_index, ChartData->GetChartTableEntries()); - const ChartTableEntry *cte = &ChartData->GetChartTableEntry(ut_index); - - double clat = (cte->GetLatMax() + cte->GetLatMin()) / 2; - double clon = (cte->GetLonMax() + cte->GetLonMin()) / 2; - - vLat = clat; - vLon = clon; - - cc->SetViewPoint(clat, clon); - - if (cc->GetQuiltMode()) { - if (cc->IsChartQuiltableRef(ut_index)) - cc->SelectQuiltRefdbChart(ut_index); - } else - cc->SelectdbChart(ut_index); - - double ppm; // final ppm scale to use - if (g_unit_test_1) { - ppm = cc->GetCanvasScaleFactor() / cte->GetScale(); - ppm /= 2; - } else { - double rw, rh; // width, height - int ww, wh; // chart window width, height - - // width in nm - DistanceBearingMercator(cte->GetLatMin(), cte->GetLonMin(), - cte->GetLatMin(), cte->GetLonMax(), NULL, - &rw); - - // height in nm - DistanceBearingMercator(cte->GetLatMin(), cte->GetLonMin(), - cte->GetLatMax(), cte->GetLonMin(), NULL, - &rh); - - cc->GetSize(&ww, &wh); - ppm = wxMin(ww / (rw * 1852), wh / (rh * 1852)) * (100 - fabs(clat)) / - 90; - ppm = wxMin(ppm, 1.0); - } - cc->SetVPScale(ppm); - - cc->ReloadVP(); - - ut_index++; - if (ut_index > ut_index_max) exit(0); - } else { - _exit(0); - } - } - } - g_tick++; - +void MyFrame::ProcessQuitFlag() { // Listen for quitflag to be set, requesting application close if (quitflag) { wxLogMessage(_T("Got quitflag from SIGNAL")); @@ -5681,12 +5680,9 @@ void MyFrame::OnFrameTimer1(wxTimerEvent &event) { Close(); return; } +} - if (bDBUpdateInProgress) return; - - FrameTimer1.Stop(); - FrameTenHzTimer.Stop(); - +void MyFrame::ProcessDeferredTrackOn() { // If tracking carryover was found in config file, enable tracking as soon as // GPS become valid if (g_bDeferredStartTrack) { @@ -5699,35 +5695,9 @@ void MyFrame::OnFrameTimer1(wxTimerEvent &event) { g_bDeferredStartTrack = false; } } +} - // Build and send a Position Fix event to PlugIns - if (g_pi_manager) { - GenericPosDatEx GPSData; - GPSData.kLat = gLat; - GPSData.kLon = gLon; - GPSData.kCog = gCog; - GPSData.kSog = gSog; - GPSData.kVar = gVar; - GPSData.kHdm = gHdm; - GPSData.kHdt = gHdt; - GPSData.nSats = g_SatsInView; - - wxDateTime tCheck((time_t)m_fixtime); - if (tCheck.IsValid()) { - // As a special case, when no GNSS data is available, m_fixtime is set to - // zero. Note wxDateTime(0) is valid, so the zero value is passed to the - // plugins. The plugins should check for zero and not use the time in that - // case. - GPSData.FixTime = m_fixtime; - } else { - // Note: I don't think this is ever reached, as m_fixtime can never be set - // to wxLongLong(wxINT64_MIN), which is the only way to get here. - GPSData.FixTime = wxDateTime::Now().GetTicks(); - } - - SendPositionFixToAllPlugIns(&GPSData); - } - +void MyFrame::ProcessAnchorWatch() { // Check for anchorwatch alarms // pjotrc // 2010.02.15 if (pAnchorWatchPoint1) { @@ -5773,10 +5743,41 @@ void MyFrame::OnFrameTimer1(wxTimerEvent &event) { if ((pAnchorWatchPoint1 || pAnchorWatchPoint2) && !bGPSValid) AnchorAlertOn1 = true; +} + +void MyFrame::SendFixToPlugins() { + // Build and send a Position Fix event to PlugIns + if (g_pi_manager) { + GenericPosDatEx GPSData; + GPSData.kLat = gLat; + GPSData.kLon = gLon; + GPSData.kCog = gCog; + GPSData.kSog = gSog; + GPSData.kVar = gVar; + GPSData.kHdm = gHdm; + GPSData.kHdt = gHdt; + GPSData.nSats = g_SatsInView; + + wxDateTime tCheck((time_t)m_fixtime); + if (tCheck.IsValid()) { + // As a special case, when no GNSS data is available, m_fixtime is set to + // zero. Note wxDateTime(0) is valid, so the zero value is passed to the + // plugins. The plugins should check for zero and not use the time in that + // case. + GPSData.FixTime = m_fixtime; + } else { + // Note: I don't think this is ever reached, as m_fixtime can never be set + // to wxLongLong(wxINT64_MIN), which is the only way to get here. + GPSData.FixTime = wxDateTime::Now().GetTicks(); + } + SendPositionFixToAllPlugIns(&GPSData); + } +} + +void MyFrame::ProcessLogAndBells() { // Send current nav status data to log file on every half hour // pjotrc // 2010.02.09 - wxDateTime lognow = wxDateTime::Now(); // pjotrc 2010.02.09 int hourLOC = lognow.GetHour(); int minuteLOC = lognow.GetMinute(); @@ -5834,6 +5835,24 @@ void MyFrame::OnFrameTimer1(wxTimerEvent &event) { } } } +} + +void MyFrame::OnFrameTimer1(wxTimerEvent &event) { + CheckToolbarPosition(); + + ProcessUnitTest(); + g_tick++; + ProcessQuitFlag(); + + if (bDBUpdateInProgress) return; + + FrameTimer1.Stop(); + FrameTenHzTimer.Stop(); + + ProcessDeferredTrackOn(); + SendFixToPlugins(); + ProcessAnchorWatch(); + ProcessLogAndBells(); if (ShouldRestartTrack()) TrackDailyRestart(); @@ -5855,18 +5874,6 @@ void MyFrame::OnFrameTimer1(wxTimerEvent &event) { gCog = 0.0; // say speed is zero to kill ownship predictor } -// TODO -// Not needed? -#if 0 -#if !defined(__WXGTK__) && !defined(__WXQT__) - { - double cursor_lat, cursor_lon; - GetPrimaryCanvas()->GetCursorLatLon( &cursor_lat, &cursor_lon ); - GetPrimaryCanvas()->SetCursorStatus(cursor_lat, cursor_lon); - } -#endif -#endif - // Update the chart database and displayed chart bool bnew_view = false;