From eb304063e05967f7024fb3b2b6f0d38a7df520af Mon Sep 17 00:00:00 2001 From: Volodymyr Huti Date: Tue, 29 Mar 2022 03:43:47 +0300 Subject: vyos-debug: T4394: extend profiling support 1. Use chrono library instead of basic C library to improve accuracy 2. Introduce the tracing helpers 3. Add tracing calls to the main processing points NOTE: loger is writing to the stdout, and for bootloader it is redirected to - /var/log/vyatta/vyos-boot-config-loader.log --- src/commit/commit-algorithm.cpp | 78 +++++++++++++++++++++++++++++++++++------ src/cstore/cstore.cpp | 11 ------ 2 files changed, 67 insertions(+), 22 deletions(-) diff --git a/src/commit/commit-algorithm.cpp b/src/commit/commit-algorithm.cpp index 1909ca9..13d3500 100644 --- a/src/commit/commit-algorithm.cpp +++ b/src/commit/commit-algorithm.cpp @@ -17,6 +17,7 @@ #include #include #include +#include #include #include @@ -24,6 +25,37 @@ using namespace commit; using namespace std; +using namespace chrono; + +////// debugging helpers +bool debug_on = false; +int stack_depth = 0; + +// You can display a file/function/line if required as follows +// OUTPUT_USER("[%s:%d:%s]" __FILE__, __LINE__, __FUNCTION__) + +#define TRACE_RESET(fmt, ...) \ + if (debug_on) { \ + stack_depth++; \ + OUTPUT_USER("%*s" fmt "\n", stack_depth, " ", ##__VA_ARGS__); \ + start_time = high_resolution_clock::now(); \ + } + +#define TRACE_INIT(fmt, ...) \ + time_point start_time; \ + TRACE_RESET(fmt, ##__VA_ARGS__); + +#define TRACE_DISPLAY(fmt, ...) \ + if (debug_on) { \ + time_point stop_time = high_resolution_clock::now(); \ + int total_ms = duration_cast(stop_time - start_time).count(); \ + int sec_elapsed = total_ms / 1000; \ + int ms_elapsed = total_ms % 1000; \ + OUTPUT_USER("%*s" "Elapsed %d.%03d sec: " fmt "\n", \ + stack_depth, " ", sec_elapsed, ms_elapsed, \ + ##__VA_ARGS__); \ + stack_depth--; \ + } ////// static @@ -278,11 +310,15 @@ _exec_tmpl_actions(Cstore& cs, CommitState s, char *at_str, default: break; } + + TRACE_INIT("Executing the \"%s\" ...", disp_path.to_string().c_str()); setenv("COMMIT_ACTION", aenv, 1); set_in_delete_action((act == delete_act)); bool ret = cs.executeTmplActions(at_str, path, disp_path, actions, def); set_in_delete_action(false); unsetenv("COMMIT_ACTION"); + TRACE_DISPLAY(""); + return ret; } @@ -662,19 +698,33 @@ static bool _commit_exec_prio_subtree(Cstore& cs, PrioNode *proot) { CfgNode *cfg = proot->getCfgNode(); + CommittedPathListT clist; + bool ret = false; if (cfg) { if (proot->getCommitState() == COMMIT_STATE_ADDED && proot->parentCreateFailed()) { // can't create if parent create failed - proot->setSucceeded(false); - return false; + goto commit_failed; } - CommittedPathListT clist; - if (!_commit_check_cfg_node(cs, cfg, clist) - || !_commit_exec_cfg_node(cs, cfg)) { - // subtree commit failed - proot->setSucceeded(false); - return false; + + if (!debug_on) { + if (!_commit_check_cfg_node(cs, cfg, clist) + || !_commit_exec_cfg_node(cs, cfg)) { + // subtree commit failed + goto commit_failed; + } + } else { + TRACE_INIT("Entering the _commit_check_cfg_node"); + ret = _commit_check_cfg_node(cs, cfg, clist); + TRACE_DISPLAY("_commit_check_cfg_node"); + if (!ret) + goto commit_failed; + + TRACE_RESET("Entering the _commit_exec_cfg_node"); + ret = _commit_exec_cfg_node(cs, cfg); + TRACE_DISPLAY("_commit_exec_cfg_node"); + if (!ret) + goto commit_failed; } // subtree succeeded, mark nodes committed for (size_t i = 0; i < clist.size(); i++) { @@ -682,13 +732,16 @@ _commit_exec_prio_subtree(Cstore& cs, PrioNode *proot) (clist[i].first == COMMIT_STATE_DELETED))) { fprintf(stderr, "Failed to mark path committed\n"); - proot->setSucceeded(false); - return false; + goto commit_failed; } } } proot->setSucceeded(true); return true; + +commit_failed: + proot->setSucceeded(false); + return false; } static CfgNode * @@ -1248,7 +1301,9 @@ commit::doCommit(Cstore& cs, CfgNode& cfg1, CfgNode& cfg2) DelPrioQueueT dpq; _get_commit_prio_queue(&proot, pq, dpq); size_t s = 0, f = 0; - + + debug_on = !!getenv("VYOS_DEBUG"); + TRACE_INIT("Processing the Priority Queue"); while (!dpq.empty()) { PrioNode *p = dpq.top(); if (!_commit_exec_prio_subtree(cs, p)) { @@ -1275,6 +1330,7 @@ commit::doCommit(Cstore& cs, CfgNode& cfg1, CfgNode& cfg2) } pq.pop(); } + TRACE_DISPLAY("Commit execute priority tree"); bool ret = true; const char *cst = "SUCCESS"; if (f > 0) { diff --git a/src/cstore/cstore.cpp b/src/cstore/cstore.cpp index 4750ba0..71d30d1 100644 --- a/src/cstore/cstore.cpp +++ b/src/cstore/cstore.cpp @@ -2050,20 +2050,9 @@ Cstore::executeTmplActions(char *at_str, const Cpath& path, var_ref_handle = (void *) this; // const_cast for legacy code - std::time_t start_time = std::time(0); - bool ret = execute_list(const_cast(actions), def, sdisp.c_str()); var_ref_handle = NULL; - - char* debug_on = getenv("VYOS_DEBUG"); - if (debug_on != NULL) - { - std::time_t stop_time = std::time(0); - unsigned long int exec_time = stop_time - start_time; - output_internal("Action for \"%s\" took %u seconds to execute\n", sdisp.c_str(), exec_time); - } - return ret; } -- cgit v1.2.3