Implemented Log app (#144)
This commit is contained in:
parent
737c0f7447
commit
a7d15056d8
@ -51,6 +51,7 @@ if (DEFINED ENV{ESP_IDF_VERSION})
|
|||||||
|
|
||||||
add_compile_definitions(LV_CONF_PATH="${LVGL_CONFIG_FULL_PATH}/lv_conf_kconfig.h")
|
add_compile_definitions(LV_CONF_PATH="${LVGL_CONFIG_FULL_PATH}/lv_conf_kconfig.h")
|
||||||
idf_build_set_property(LINK_OPTIONS "-Wl,--wrap=esp_panic_handler" APPEND)
|
idf_build_set_property(LINK_OPTIONS "-Wl,--wrap=esp_panic_handler" APPEND)
|
||||||
|
idf_build_set_property(LINK_OPTIONS "-Wl,--wrap=esp_log_write" APPEND)
|
||||||
|
|
||||||
file(READ version.txt TACTILITY_VERSION)
|
file(READ version.txt TACTILITY_VERSION)
|
||||||
add_compile_definitions(TT_VERSION="${TACTILITY_VERSION}")
|
add_compile_definitions(TT_VERSION="${TACTILITY_VERSION}")
|
||||||
|
|||||||
Binary file not shown.
|
Before Width: | Height: | Size: 74 KiB After Width: | Height: | Size: 19 KiB |
@ -43,11 +43,12 @@ namespace app {
|
|||||||
namespace desktop { extern const AppManifest manifest; }
|
namespace desktop { extern const AppManifest manifest; }
|
||||||
namespace files { extern const AppManifest manifest; }
|
namespace files { extern const AppManifest manifest; }
|
||||||
namespace gpio { extern const AppManifest manifest; }
|
namespace gpio { extern const AppManifest manifest; }
|
||||||
namespace imageviewer { extern const AppManifest manifest; }
|
|
||||||
namespace display { extern const AppManifest manifest; }
|
namespace display { extern const AppManifest manifest; }
|
||||||
namespace i2cscanner { extern const AppManifest manifest; }
|
namespace i2cscanner { extern const AppManifest manifest; }
|
||||||
namespace i2csettings { extern const AppManifest manifest; }
|
namespace i2csettings { extern const AppManifest manifest; }
|
||||||
|
namespace imageviewer { extern const AppManifest manifest; }
|
||||||
namespace inputdialog { extern const AppManifest manifest; }
|
namespace inputdialog { extern const AppManifest manifest; }
|
||||||
|
namespace log { extern const AppManifest manifest; }
|
||||||
namespace power { extern const AppManifest manifest; }
|
namespace power { extern const AppManifest manifest; }
|
||||||
namespace selectiondialog { extern const AppManifest manifest; }
|
namespace selectiondialog { extern const AppManifest manifest; }
|
||||||
namespace settings { extern const AppManifest manifest; }
|
namespace settings { extern const AppManifest manifest; }
|
||||||
@ -78,8 +79,9 @@ static const std::vector<const app::AppManifest*> system_apps = {
|
|||||||
&app::gpio::manifest,
|
&app::gpio::manifest,
|
||||||
&app::i2cscanner::manifest,
|
&app::i2cscanner::manifest,
|
||||||
&app::i2csettings::manifest,
|
&app::i2csettings::manifest,
|
||||||
&app::inputdialog::manifest,
|
|
||||||
&app::imageviewer::manifest,
|
&app::imageviewer::manifest,
|
||||||
|
&app::inputdialog::manifest,
|
||||||
|
&app::log::manifest,
|
||||||
&app::settings::manifest,
|
&app::settings::manifest,
|
||||||
&app::selectiondialog::manifest,
|
&app::selectiondialog::manifest,
|
||||||
&app::systeminfo::manifest,
|
&app::systeminfo::manifest,
|
||||||
|
|||||||
150
Tactility/Source/app/log/Log.cpp
Normal file
150
Tactility/Source/app/log/Log.cpp
Normal file
@ -0,0 +1,150 @@
|
|||||||
|
#include <TactilityCore.h>
|
||||||
|
#include <sstream>
|
||||||
|
#include <vector>
|
||||||
|
#include "lvgl.h"
|
||||||
|
#include "lvgl/Style.h"
|
||||||
|
#include "lvgl/Toolbar.h"
|
||||||
|
#include "app/selectiondialog/SelectionDialog.h"
|
||||||
|
#include "service/loader/Loader.h"
|
||||||
|
#include "lvgl/LvglSync.h"
|
||||||
|
|
||||||
|
#define TAG "text_viewer"
|
||||||
|
|
||||||
|
namespace tt::app::log {
|
||||||
|
|
||||||
|
struct LogAppData {
|
||||||
|
LogLevel filterLevel = LogLevelInfo;
|
||||||
|
lv_obj_t* labelWidget = nullptr;
|
||||||
|
};
|
||||||
|
|
||||||
|
static bool shouldShowLog(LogLevel filterLevel, LogLevel logLevel) {
|
||||||
|
if (filterLevel == LogLevelNone || logLevel == LogLevelNone) {
|
||||||
|
return false;
|
||||||
|
} else {
|
||||||
|
return filterLevel >= logLevel;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void setLogEntries(lv_obj_t* label) {
|
||||||
|
auto app = service::loader::getCurrentApp();
|
||||||
|
if (app == nullptr) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
auto data = std::static_pointer_cast<LogAppData>(app->getData());
|
||||||
|
auto filterLevel = data->filterLevel;
|
||||||
|
|
||||||
|
unsigned int index;
|
||||||
|
auto* entries = copyLogEntries(index);
|
||||||
|
std::stringstream buffer;
|
||||||
|
if (entries != nullptr) {
|
||||||
|
for (unsigned int i = index; i < TT_LOG_ENTRY_COUNT; ++i) {
|
||||||
|
if (shouldShowLog(filterLevel, entries[i].level)) {
|
||||||
|
buffer << entries[i].message;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (index != 0) {
|
||||||
|
for (unsigned int i = 0; i < index; ++i) {
|
||||||
|
if (shouldShowLog(filterLevel, entries[i].level)) {
|
||||||
|
buffer << entries[i].message;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
delete entries;
|
||||||
|
if (!buffer.str().empty()) {
|
||||||
|
lv_label_set_text(label, buffer.str().c_str());
|
||||||
|
} else {
|
||||||
|
lv_label_set_text(label, "No logs for the selected log level");
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
lv_label_set_text(label, "Failed to load log");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void onLevelFilterPressed(TT_UNUSED lv_event_t* event) {
|
||||||
|
std::vector<std::string> items = {
|
||||||
|
"Verbose",
|
||||||
|
"Debug",
|
||||||
|
"Info",
|
||||||
|
"Warning",
|
||||||
|
"Error",
|
||||||
|
};
|
||||||
|
app::selectiondialog::start("Log Level", items);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void updateViews() {
|
||||||
|
auto app = service::loader::getCurrentApp();
|
||||||
|
if (app == nullptr) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
auto data = std::static_pointer_cast<LogAppData>(app->getData());
|
||||||
|
assert(data != nullptr);
|
||||||
|
|
||||||
|
if (lvgl::lock(100 / portTICK_PERIOD_MS)) {
|
||||||
|
setLogEntries(data->labelWidget);
|
||||||
|
lvgl::unlock();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void onShow(AppContext& app, lv_obj_t* parent) {
|
||||||
|
auto data = std::static_pointer_cast<LogAppData>(app.getData());
|
||||||
|
|
||||||
|
lv_obj_set_flex_flow(parent, LV_FLEX_FLOW_COLUMN);
|
||||||
|
auto* toolbar = lvgl::toolbar_create(parent, app);
|
||||||
|
lvgl::toolbar_add_button_action(toolbar, LV_SYMBOL_EDIT, onLevelFilterPressed, nullptr);
|
||||||
|
|
||||||
|
auto* wrapper = lv_obj_create(parent);
|
||||||
|
lv_obj_set_width(wrapper, LV_PCT(100));
|
||||||
|
lv_obj_set_flex_grow(wrapper, 1);
|
||||||
|
lv_obj_set_flex_flow(wrapper, LV_FLEX_FLOW_COLUMN);
|
||||||
|
lvgl::obj_set_style_no_padding(wrapper);
|
||||||
|
lvgl::obj_set_style_bg_invisible(wrapper);
|
||||||
|
|
||||||
|
data->labelWidget = lv_label_create(wrapper);
|
||||||
|
lv_obj_align(data->labelWidget, LV_ALIGN_CENTER, 0, 0);
|
||||||
|
setLogEntries(data->labelWidget);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void onStart(AppContext& app) {
|
||||||
|
auto data = std::make_shared<LogAppData>();
|
||||||
|
app.setData(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void onResult(AppContext& app, Result result, const Bundle& bundle) {
|
||||||
|
auto resultIndex = selectiondialog::getResultIndex(bundle);
|
||||||
|
auto data = std::static_pointer_cast<LogAppData>(app.getData());
|
||||||
|
if (result == ResultOk) {
|
||||||
|
switch (resultIndex) {
|
||||||
|
case 0:
|
||||||
|
data->filterLevel = LogLevelVerbose;
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
data->filterLevel = LogLevelDebug;
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
data->filterLevel = LogLevelInfo;
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
data->filterLevel = LogLevelWarning;
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
data->filterLevel = LogLevelError;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
updateViews();
|
||||||
|
}
|
||||||
|
|
||||||
|
extern const AppManifest manifest = {
|
||||||
|
.id = "Log",
|
||||||
|
.name = "Log",
|
||||||
|
.icon = LV_SYMBOL_LIST,
|
||||||
|
.type = TypeSystem,
|
||||||
|
.onStart = onStart,
|
||||||
|
.onShow = onShow,
|
||||||
|
.onResult = onResult
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace
|
||||||
@ -1,9 +1,60 @@
|
|||||||
|
#include "Mutex.h"
|
||||||
|
#include <cstring>
|
||||||
|
|
||||||
|
namespace tt {
|
||||||
|
|
||||||
|
static LogEntry* logEntries = nullptr;
|
||||||
|
static unsigned int nextLogEntryIndex;
|
||||||
|
static Mutex logMutex;
|
||||||
|
|
||||||
|
static void ensureLogEntriesExist() {
|
||||||
|
if (logEntries == nullptr) {
|
||||||
|
logEntries = new LogEntry[TT_LOG_ENTRY_COUNT];
|
||||||
|
assert(logEntries != nullptr);
|
||||||
|
nextLogEntryIndex = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void storeLog(LogLevel level, const char* format, va_list args) {
|
||||||
|
if (logMutex.lock(5 / portTICK_PERIOD_MS)) {
|
||||||
|
ensureLogEntriesExist();
|
||||||
|
|
||||||
|
logEntries[nextLogEntryIndex].level = level;
|
||||||
|
vsnprintf(logEntries[nextLogEntryIndex].message, TT_LOG_MESSAGE_SIZE, format, args);
|
||||||
|
|
||||||
|
nextLogEntryIndex++;
|
||||||
|
if (nextLogEntryIndex == TT_LOG_ENTRY_COUNT) {
|
||||||
|
nextLogEntryIndex = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
logMutex.unlock();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
LogEntry* copyLogEntries(unsigned int& outIndex) {
|
||||||
|
if (logMutex.lock(5 / portTICK_PERIOD_MS)) {
|
||||||
|
auto* newEntries = new LogEntry[TT_LOG_ENTRY_COUNT];
|
||||||
|
assert(newEntries != nullptr);
|
||||||
|
for (int i = 0; i < TT_LOG_ENTRY_COUNT; ++i) {
|
||||||
|
memcpy(&newEntries[i], &logEntries[i], sizeof(LogEntry));
|
||||||
|
}
|
||||||
|
outIndex = nextLogEntryIndex;
|
||||||
|
logMutex.unlock();
|
||||||
|
return newEntries;
|
||||||
|
} else {
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace tt
|
||||||
|
|
||||||
#ifndef ESP_PLATFORM
|
#ifndef ESP_PLATFORM
|
||||||
|
|
||||||
#include "Log.h"
|
#include "Log.h"
|
||||||
|
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
#include <sys/time.h>
|
#include <sys/time.h>
|
||||||
|
#include <sstream>
|
||||||
|
|
||||||
namespace tt {
|
namespace tt {
|
||||||
|
|
||||||
@ -17,7 +68,7 @@ static char toPrefix(LogLevel level) {
|
|||||||
return 'I';
|
return 'I';
|
||||||
case LogLevelDebug:
|
case LogLevelDebug:
|
||||||
return 'D';
|
return 'D';
|
||||||
case LogLevelTrace:
|
case LogLevelVerbose:
|
||||||
return 'T';
|
return 'T';
|
||||||
default:
|
default:
|
||||||
return '?';
|
return '?';
|
||||||
@ -34,7 +85,7 @@ static const char* toColour(LogLevel level) {
|
|||||||
return "\033[32m";
|
return "\033[32m";
|
||||||
case LogLevelDebug:
|
case LogLevelDebug:
|
||||||
return "\033[1;37m";
|
return "\033[1;37m";
|
||||||
case LogLevelTrace:
|
case LogLevelVerbose:
|
||||||
return "\033[37m";
|
return "\033[37m";
|
||||||
default:
|
default:
|
||||||
return "";
|
return "";
|
||||||
@ -66,22 +117,38 @@ static uint64_t getTimestamp() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void log(LogLevel level, const char* tag, const char* format, ...) {
|
void log(LogLevel level, const char* tag, const char* format, ...) {
|
||||||
printf(
|
std::stringstream buffer;
|
||||||
"%s%c (%lu) %s: ",
|
buffer << toColour(level) << toPrefix(level) << " (" << getTimestamp() << ") " << tag << " " << format << "\033[0m\n";
|
||||||
toColour(level),
|
|
||||||
toPrefix(level),
|
|
||||||
getTimestamp(),
|
|
||||||
tag
|
|
||||||
);
|
|
||||||
|
|
||||||
va_list args;
|
va_list args;
|
||||||
va_start(args, format);
|
va_start(args, format);
|
||||||
vprintf(format, args);
|
vprintf(buffer.str().c_str(), args);
|
||||||
va_end(args);
|
va_end(args);
|
||||||
|
|
||||||
printf("\033[0m\n");
|
va_start(args, format);
|
||||||
|
tt::storeLog(level, buffer.str().c_str(), args);
|
||||||
|
va_end(args);
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
#endif
|
#else // ESP_PLATFORM
|
||||||
|
|
||||||
|
#include <esp_log.h>
|
||||||
|
|
||||||
|
extern "C" {
|
||||||
|
|
||||||
|
extern void __real_esp_log_write(esp_log_level_t level, const char* tag, const char* format, ...);
|
||||||
|
|
||||||
|
void __wrap_esp_log_write(esp_log_level_t level, const char* tag, const char* format, ...) {
|
||||||
|
va_list args;
|
||||||
|
va_start(args, format);
|
||||||
|
tt::storeLog((tt::LogLevel)level, format, args);
|
||||||
|
esp_log_writev(level, tag, format, args);
|
||||||
|
va_end(args);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|||||||
@ -2,6 +2,34 @@
|
|||||||
|
|
||||||
#include "LogMessages.h"
|
#include "LogMessages.h"
|
||||||
|
|
||||||
|
#if CONFIG_SPIRAM_USE_MALLOC == 1 or not defined(ESP_PLATFORM)
|
||||||
|
#define TT_LOG_ENTRY_COUNT 200
|
||||||
|
#define TT_LOG_MESSAGE_SIZE 128
|
||||||
|
#else
|
||||||
|
#define TT_LOG_ENTRY_COUNT 50
|
||||||
|
#define TT_LOG_MESSAGE_SIZE 50
|
||||||
|
#endif
|
||||||
|
|
||||||
|
namespace tt {
|
||||||
|
|
||||||
|
enum LogLevel {
|
||||||
|
LogLevelNone, /*!< No log output */
|
||||||
|
LogLevelError, /*!< Critical errors, software module can not recover on its own */
|
||||||
|
LogLevelWarning, /*!< Error conditions from which recovery measures have been taken */
|
||||||
|
LogLevelInfo, /*!< Information messages which describe normal flow of events */
|
||||||
|
LogLevelDebug, /*!< Extra information which is not necessary for normal use (values, pointers, sizes, etc). */
|
||||||
|
LogLevelVerbose /*!< Bigger chunks of debugging information, or frequent messages which can potentially flood the output. */
|
||||||
|
};
|
||||||
|
|
||||||
|
struct LogEntry {
|
||||||
|
LogLevel level = LogLevelNone;
|
||||||
|
char message[TT_LOG_MESSAGE_SIZE] = { 0 };
|
||||||
|
};
|
||||||
|
|
||||||
|
LogEntry* copyLogEntries(unsigned int& outIndex);
|
||||||
|
|
||||||
|
} // namespace tt
|
||||||
|
|
||||||
#ifdef ESP_TARGET
|
#ifdef ESP_TARGET
|
||||||
#include "esp_log.h"
|
#include "esp_log.h"
|
||||||
#else
|
#else
|
||||||
@ -26,14 +54,6 @@
|
|||||||
|
|
||||||
namespace tt {
|
namespace tt {
|
||||||
|
|
||||||
typedef enum {
|
|
||||||
LogLevelError,
|
|
||||||
LogLevelWarning,
|
|
||||||
LogLevelInfo,
|
|
||||||
LogLevelDebug,
|
|
||||||
LogLevelTrace
|
|
||||||
} LogLevel;
|
|
||||||
|
|
||||||
void log(LogLevel level, const char* tag, const char* format, ...);
|
void log(LogLevel level, const char* tag, const char* format, ...);
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user