mirror of
https://github.com/espressif/esp-idf.git
synced 2024-10-05 20:47:46 -04:00
408 lines
12 KiB
C++
408 lines
12 KiB
C++
/* LOG unit tests
|
|
|
|
This example code is in the Public Domain (or CC0 licensed, at your option.)
|
|
|
|
Unless required by applicable law or agreed to in writing, this
|
|
software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
|
|
CONDITIONS OF ANY KIND, either express or implied.
|
|
*/
|
|
#include <cstdio>
|
|
#include <regex>
|
|
#include <iostream>
|
|
#include "esp_rom_sys.h"
|
|
#include "esp_log.h"
|
|
#include "esp_private/log_util.h"
|
|
#include "esp_private/log_timestamp.h"
|
|
#include "sdkconfig.h"
|
|
|
|
#include <catch2/catch_test_macros.hpp>
|
|
|
|
using namespace std;
|
|
|
|
#define EARLY_TIMESTAMP "[0-9]*"
|
|
|
|
#if CONFIG_LOG_TIMESTAMP_SOURCE_RTOS
|
|
#define TIMESTAMP "[0-9]*"
|
|
#elif CONFIG_LOG_TIMESTAMP_SOURCE_SYSTEM
|
|
#define TIMESTAMP "[0-9]{2}:[0-9]{2}:[0-9]{2}\\.[0-9]{3}"
|
|
#elif CONFIG_LOG_TIMESTAMP_SOURCE_SYSTEM_FULL
|
|
#define TIMESTAMP "[0-9]{2}-[0-9]{2}-[0-9]{2} [0-9]{2}:[0-9]{2}:[0-9]{2}\\.[0-9]{3}"
|
|
#else
|
|
#define TIMESTAMP ""
|
|
#endif
|
|
|
|
static const char *TEST_TAG = "test";
|
|
|
|
class BasicLogFixture {
|
|
public:
|
|
static const size_t BUFFER_SIZE = 4096;
|
|
|
|
BasicLogFixture(esp_log_level_t log_level = ESP_LOG_VERBOSE)
|
|
{
|
|
std::memset(print_buffer, 0, BUFFER_SIZE);
|
|
buffer_idx = 0;
|
|
esp_log_level_set("*", log_level);
|
|
}
|
|
|
|
virtual ~BasicLogFixture()
|
|
{
|
|
esp_log_level_set("*", ESP_LOG_INFO);
|
|
}
|
|
|
|
string get_print_buffer_string() const
|
|
{
|
|
return string(print_buffer);
|
|
}
|
|
|
|
void reset_buffer()
|
|
{
|
|
std::memset(print_buffer, 0, BUFFER_SIZE);
|
|
buffer_idx = 0;
|
|
additional_reset();
|
|
}
|
|
|
|
protected:
|
|
char print_buffer [BUFFER_SIZE];
|
|
int buffer_idx;
|
|
|
|
virtual void additional_reset() { }
|
|
};
|
|
|
|
struct PrintFixture : BasicLogFixture {
|
|
PrintFixture(esp_log_level_t log_level = ESP_LOG_VERBOSE) : BasicLogFixture(log_level)
|
|
{
|
|
if (instance != nullptr) {
|
|
throw exception();
|
|
}
|
|
|
|
instance = this;
|
|
|
|
old_vprintf = esp_log_set_vprintf(print_callback);
|
|
}
|
|
|
|
virtual ~PrintFixture()
|
|
{
|
|
esp_log_set_vprintf(old_vprintf);
|
|
instance = nullptr;
|
|
}
|
|
|
|
private:
|
|
static int print_callback(const char *format, va_list args)
|
|
{
|
|
return instance->print_to_buffer(format, args);
|
|
}
|
|
|
|
int print_to_buffer(const char *format, va_list args)
|
|
{
|
|
// Added support for multi-line log, for example ESP_LOG_BUFFER...
|
|
int ret = vsnprintf(&print_buffer[buffer_idx], BUFFER_SIZE, format, args);
|
|
buffer_idx += ret;
|
|
return ret;
|
|
}
|
|
|
|
static PrintFixture *instance;
|
|
|
|
vprintf_like_t old_vprintf;
|
|
};
|
|
|
|
struct PutcFixture : BasicLogFixture {
|
|
PutcFixture(esp_log_level_t log_level = ESP_LOG_VERBOSE) : BasicLogFixture(log_level), counter(0)
|
|
{
|
|
if (instance != nullptr) {
|
|
throw exception();
|
|
}
|
|
|
|
esp_rom_install_channel_putc(1, putc_callback);
|
|
|
|
instance = this;
|
|
}
|
|
|
|
~PutcFixture()
|
|
{
|
|
esp_rom_install_uart_printf();
|
|
instance = nullptr;
|
|
}
|
|
|
|
void additional_reset() override
|
|
{
|
|
counter = 0;
|
|
}
|
|
|
|
size_t counter;
|
|
|
|
private:
|
|
static void putc_callback(char c)
|
|
{
|
|
return instance->putc_to_buffer(c);
|
|
}
|
|
|
|
void putc_to_buffer(char c)
|
|
{
|
|
print_buffer[counter++] = c;
|
|
}
|
|
|
|
static PutcFixture *instance;
|
|
};
|
|
|
|
PrintFixture *PrintFixture::instance = nullptr;
|
|
PutcFixture *PutcFixture::instance = nullptr;
|
|
|
|
TEST_CASE("verbose log level")
|
|
{
|
|
PrintFixture fix(ESP_LOG_VERBOSE);
|
|
const std::regex test_print("V \\(" TIMESTAMP "\\) test: verbose", std::regex::ECMAScript);
|
|
|
|
ESP_LOGV(TEST_TAG, "verbose");
|
|
CHECK(regex_search(fix.get_print_buffer_string(), test_print) == true);
|
|
}
|
|
|
|
TEST_CASE("debug log level")
|
|
{
|
|
PrintFixture fix(ESP_LOG_DEBUG);
|
|
const std::regex test_print("D \\(" TIMESTAMP "\\) test: debug", std::regex::ECMAScript);
|
|
|
|
ESP_LOGD(TEST_TAG, "debug");
|
|
CHECK(regex_search(fix.get_print_buffer_string(), test_print) == true);
|
|
}
|
|
|
|
TEST_CASE("info log level")
|
|
{
|
|
PrintFixture fix(ESP_LOG_INFO);
|
|
const std::regex test_print("I \\(" TIMESTAMP "\\) test: info", std::regex::ECMAScript);
|
|
|
|
ESP_LOGI(TEST_TAG, "info");
|
|
CHECK(regex_search(fix.get_print_buffer_string(), test_print) == true);
|
|
}
|
|
|
|
TEST_CASE("warn log level")
|
|
{
|
|
PrintFixture fix(ESP_LOG_WARN);
|
|
const std::regex test_print("W \\(" TIMESTAMP "\\) test: warn", std::regex::ECMAScript);
|
|
|
|
ESP_LOGW(TEST_TAG, "warn");
|
|
CHECK(regex_search(fix.get_print_buffer_string(), test_print) == true);
|
|
}
|
|
|
|
TEST_CASE("error log level")
|
|
{
|
|
PrintFixture fix(ESP_LOG_ERROR);
|
|
const std::regex test_print("E \\(" TIMESTAMP "\\) test: error", std::regex::ECMAScript);
|
|
|
|
ESP_LOGE(TEST_TAG, "error");
|
|
CHECK(regex_search(fix.get_print_buffer_string(), test_print) == true);
|
|
}
|
|
|
|
#if CONFIG_LOG_DYNAMIC_LEVEL_CONTROL
|
|
TEST_CASE("changing log level")
|
|
{
|
|
PrintFixture fix(ESP_LOG_INFO);
|
|
const std::regex test_print("I \\(" TIMESTAMP "\\) test: must indeed be printed", std::regex::ECMAScript);
|
|
|
|
ESP_LOGI(TEST_TAG, "must indeed be printed");
|
|
CHECK(regex_search(fix.get_print_buffer_string(), test_print) == true);
|
|
|
|
fix.reset_buffer();
|
|
esp_log_level_set(TEST_TAG, ESP_LOG_WARN);
|
|
|
|
ESP_LOGI(TEST_TAG, "must not be printed");
|
|
CHECK(fix.get_print_buffer_string().size() == 0);
|
|
|
|
fix.reset_buffer();
|
|
esp_log_level_set(TEST_TAG, ESP_LOG_INFO);
|
|
|
|
ESP_LOGI(TEST_TAG, "must indeed be printed");
|
|
CHECK(regex_search(fix.get_print_buffer_string(), test_print) == true);
|
|
}
|
|
#endif // CONFIG_LOG_DYNAMIC_LEVEL_CONTROL
|
|
|
|
TEST_CASE("log buffer")
|
|
{
|
|
PrintFixture fix(ESP_LOG_INFO);
|
|
const uint8_t buffer[] = {
|
|
0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08,
|
|
0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18,
|
|
};
|
|
ESP_LOG_BUFFER_HEX(TEST_TAG, buffer, sizeof(buffer));
|
|
const std::regex buffer_regex("I \\(" TIMESTAMP "\\) test: 01 02 03 04 05 06 07 08 11 12 13 14 15 16 17 18", std::regex::ECMAScript);
|
|
CHECK(regex_search(fix.get_print_buffer_string(), buffer_regex));
|
|
}
|
|
|
|
TEST_CASE("log bytes > 127")
|
|
{
|
|
PrintFixture fix(ESP_LOG_INFO);
|
|
const uint8_t buffer[] = {
|
|
0xff, 0x80,
|
|
};
|
|
ESP_LOG_BUFFER_HEX(TEST_TAG, buffer, sizeof(buffer));
|
|
const std::regex buffer_regex("I \\(" TIMESTAMP "\\) test: ff 80", std::regex::ECMAScript);
|
|
CHECK(regex_search(fix.get_print_buffer_string(), buffer_regex));
|
|
}
|
|
|
|
TEST_CASE("log buffer char")
|
|
{
|
|
PrintFixture fix(ESP_LOG_INFO);
|
|
const char g[] = "The way to get started is to quit talking and begin doing. - Walt Disney";
|
|
const std::regex buffer_regex("I \\(" TIMESTAMP "\\) test: The way to get s.*\n\
|
|
.*I \\(" TIMESTAMP "\\) test: tarted is to qui.*\n\
|
|
.*I \\(" TIMESTAMP "\\) test: t talking and be.*\n\
|
|
.*I \\(" TIMESTAMP "\\) test: gin doing. - Wal.*\n\
|
|
.*I \\(" TIMESTAMP "\\) test: t Disney", std::regex::ECMAScript);
|
|
ESP_LOG_BUFFER_CHAR(TEST_TAG, g, sizeof(g));
|
|
CHECK(regex_search(fix.get_print_buffer_string(), buffer_regex) == true);
|
|
}
|
|
|
|
TEST_CASE("log buffer dump")
|
|
{
|
|
PrintFixture fix(ESP_LOG_INFO);
|
|
const uint8_t buffer[] = {
|
|
0x00, 0x00, 0x00, 0x00, 0x05, 0x06, 0x07, 0x08,
|
|
0xff, 0xfe, 0xfd, 0xfc, 0xfb, 0xfa, 0xf9, 0xf8
|
|
};
|
|
ESP_LOG_BUFFER_HEXDUMP(TEST_TAG, buffer, sizeof(buffer), ESP_LOG_INFO);
|
|
const std::regex buffer_regex("I \\(" TIMESTAMP "\\) test: 0x[0-9a-f]+\\s+"
|
|
"00 00 00 00 05 06 07 08 ff fe fd fc fb fa f9 f8 "
|
|
"\\s+|[\\.]{16}|", std::regex::ECMAScript);
|
|
CHECK(regex_search(fix.get_print_buffer_string(), buffer_regex));
|
|
}
|
|
|
|
TEST_CASE("rom printf")
|
|
{
|
|
PutcFixture fix;
|
|
int printed_chars = esp_rom_printf("info");
|
|
|
|
CHECK(printed_chars == 4);
|
|
CHECK(fix.get_print_buffer_string() == "info");
|
|
}
|
|
|
|
TEST_CASE("early verbose log level")
|
|
{
|
|
PutcFixture fix;
|
|
const std::regex test_print("V \\(" EARLY_TIMESTAMP "\\) test: verbose", std::regex::ECMAScript);
|
|
|
|
ESP_EARLY_LOGV(TEST_TAG, "verbose");
|
|
CHECK(regex_search(fix.get_print_buffer_string(), test_print) == true);
|
|
}
|
|
|
|
TEST_CASE("early debug log level")
|
|
{
|
|
PutcFixture fix;
|
|
const std::regex test_print("D \\(" EARLY_TIMESTAMP "\\) test: debug", std::regex::ECMAScript);
|
|
|
|
ESP_EARLY_LOGD(TEST_TAG, "debug");
|
|
CHECK(regex_search(fix.get_print_buffer_string(), test_print) == true);
|
|
}
|
|
|
|
TEST_CASE("early info log level")
|
|
{
|
|
PutcFixture fix;
|
|
const std::regex test_print("I \\(" EARLY_TIMESTAMP "\\) test: info", std::regex::ECMAScript);
|
|
|
|
ESP_EARLY_LOGI(TEST_TAG, "info");
|
|
CHECK(regex_search(fix.get_print_buffer_string(), test_print) == true);
|
|
}
|
|
|
|
TEST_CASE("early warn log level")
|
|
{
|
|
PutcFixture fix;
|
|
const std::regex test_print("W \\(" EARLY_TIMESTAMP "\\) test: warn", std::regex::ECMAScript);
|
|
|
|
ESP_EARLY_LOGW(TEST_TAG, "warn");
|
|
CHECK(regex_search(fix.get_print_buffer_string(), test_print) == true);
|
|
}
|
|
|
|
TEST_CASE("early error log level")
|
|
{
|
|
PutcFixture fix;
|
|
const std::regex test_print("E \\(" EARLY_TIMESTAMP "\\) test: error", std::regex::ECMAScript);
|
|
|
|
ESP_EARLY_LOGE(TEST_TAG, "error");
|
|
CHECK(regex_search(fix.get_print_buffer_string(), test_print) == true);
|
|
}
|
|
|
|
#if CONFIG_LOG_DYNAMIC_LEVEL_CONTROL
|
|
TEST_CASE("changing early log level")
|
|
{
|
|
PutcFixture fix(ESP_LOG_INFO);
|
|
const std::regex test_print("I \\(" EARLY_TIMESTAMP "\\) test: must indeed be printed", std::regex::ECMAScript);
|
|
|
|
ESP_EARLY_LOGI(TEST_TAG, "must indeed be printed");
|
|
CHECK(regex_search(fix.get_print_buffer_string(), test_print) == true);
|
|
|
|
fix.reset_buffer();
|
|
esp_log_level_set("*", ESP_LOG_WARN);
|
|
|
|
ESP_EARLY_LOGI(TEST_TAG, "must not be printed");
|
|
CHECK(fix.get_print_buffer_string().size() == 0);
|
|
|
|
fix.reset_buffer();
|
|
esp_log_level_set("*", ESP_LOG_INFO);
|
|
|
|
ESP_EARLY_LOGI(TEST_TAG, "must indeed be printed");
|
|
CHECK(regex_search(fix.get_print_buffer_string(), test_print) == true);
|
|
}
|
|
#endif // CONFIG_LOG_DYNAMIC_LEVEL_CONTROL
|
|
|
|
TEST_CASE("esp_log_util_cvt")
|
|
{
|
|
char buf[128];
|
|
CHECK(esp_log_util_cvt_dec(123456, 0, buf) == 6);
|
|
CHECK(strcmp(buf, "123456") == 0);
|
|
memset(buf, 0, sizeof(buf));
|
|
|
|
CHECK(esp_log_util_cvt_dec(123456, 2, buf) == 6);
|
|
CHECK(strcmp(buf, "123456") == 0);
|
|
memset(buf, 0, sizeof(buf));
|
|
|
|
CHECK(esp_log_util_cvt_dec(123456, 8, buf) == 8);
|
|
CHECK(strcmp(buf, "00123456") == 0);
|
|
memset(buf, 0, sizeof(buf));
|
|
|
|
CHECK(esp_log_util_cvt_dec(1, 0, buf) == 1);
|
|
CHECK(strcmp(buf, "1") == 0);
|
|
memset(buf, 0, sizeof(buf));
|
|
|
|
CHECK(esp_log_util_cvt_dec(0, 0, buf) == 1);
|
|
CHECK(strcmp(buf, "0") == 0);
|
|
memset(buf, 0, sizeof(buf));
|
|
|
|
CHECK(esp_log_util_cvt_dec(0, 4, buf) == 4);
|
|
CHECK(strcmp(buf, "0000") == 0);
|
|
memset(buf, 0, sizeof(buf));
|
|
|
|
CHECK(esp_log_util_cvt_dec(1, 2, buf) == 2);
|
|
CHECK(strcmp(buf, "01") == 0);
|
|
memset(buf, 0, sizeof(buf));
|
|
|
|
CHECK(esp_log_util_cvt_hex(0x73f, -1, buf) == 3);
|
|
CHECK(strcmp(buf, "73f") == 0);
|
|
memset(buf, 0, sizeof(buf));
|
|
|
|
CHECK(esp_log_util_cvt_hex(0x73f, 2, buf) == 3);
|
|
CHECK(strcmp(buf, "73f") == 0);
|
|
memset(buf, 0, sizeof(buf));
|
|
|
|
CHECK(esp_log_util_cvt_hex(0x73f, 3, buf) == 3);
|
|
CHECK(strcmp(buf, "73f") == 0);
|
|
memset(buf, 0, sizeof(buf));
|
|
|
|
CHECK(esp_log_util_cvt_hex(0x73f, 4, buf) == 4);
|
|
CHECK(strcmp(buf, "073f") == 0);
|
|
memset(buf, 0, sizeof(buf));
|
|
}
|
|
|
|
TEST_CASE("esp_log_timestamp_str")
|
|
{
|
|
char buffer[64];
|
|
bool critical = true;
|
|
uint64_t timestamp_ms = esp_log_timestamp64(critical);
|
|
esp_log_timestamp_str(critical, timestamp_ms, buffer);
|
|
const std::regex test_print(EARLY_TIMESTAMP, std::regex::ECMAScript);
|
|
CHECK(regex_search(string(buffer), test_print) == true);
|
|
|
|
critical = false;
|
|
timestamp_ms = esp_log_timestamp64(critical);
|
|
esp_log_timestamp_str(critical, timestamp_ms, buffer);
|
|
const std::regex test_print2(TIMESTAMP, std::regex::ECMAScript);
|
|
CHECK(regex_search(string(buffer), test_print2) == true);
|
|
}
|