2014-12-06 23:00:08 +01:00
|
|
|
// Copyright 2014 Citra Emulator Project
|
2014-12-17 06:38:14 +01:00
|
|
|
// Licensed under GPLv2 or any later version
|
2014-12-06 23:00:08 +01:00
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
#include "common/logging/backend.h"
|
2016-09-21 08:52:38 +02:00
|
|
|
#include "common/logging/filter.h"
|
2014-12-06 23:00:08 +01:00
|
|
|
#include "common/string_util.h"
|
|
|
|
|
|
|
|
namespace Log {
|
|
|
|
|
|
|
|
Filter::Filter(Level default_level) {
|
|
|
|
ResetAll(default_level);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Filter::ResetAll(Level level) {
|
|
|
|
class_levels.fill(level);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Filter::SetClassLevel(Class log_class, Level level) {
|
|
|
|
class_levels[static_cast<size_t>(log_class)] = level;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Filter::ParseFilterString(const std::string& filter_str) {
|
|
|
|
auto clause_begin = filter_str.cbegin();
|
|
|
|
while (clause_begin != filter_str.cend()) {
|
|
|
|
auto clause_end = std::find(clause_begin, filter_str.cend(), ' ');
|
|
|
|
|
|
|
|
// If clause isn't empty
|
|
|
|
if (clause_end != clause_begin) {
|
|
|
|
ParseFilterRule(clause_begin, clause_end);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (clause_end != filter_str.cend()) {
|
|
|
|
// Skip over the whitespace
|
|
|
|
++clause_end;
|
|
|
|
}
|
|
|
|
clause_begin = clause_end;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename It>
|
|
|
|
static Level GetLevelByName(const It begin, const It end) {
|
|
|
|
for (u8 i = 0; i < static_cast<u8>(Level::Count); ++i) {
|
2015-05-12 07:19:44 +02:00
|
|
|
const char* level_name = GetLevelName(static_cast<Level>(i));
|
2014-12-06 23:00:08 +01:00
|
|
|
if (Common::ComparePartialString(begin, end, level_name)) {
|
|
|
|
return static_cast<Level>(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Level::Count;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename It>
|
|
|
|
static Class GetClassByName(const It begin, const It end) {
|
|
|
|
for (ClassType i = 0; i < static_cast<ClassType>(Class::Count); ++i) {
|
2015-05-12 07:19:44 +02:00
|
|
|
const char* level_name = GetLogClassName(static_cast<Class>(i));
|
2014-12-06 23:00:08 +01:00
|
|
|
if (Common::ComparePartialString(begin, end, level_name)) {
|
|
|
|
return static_cast<Class>(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Class::Count;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Filter::ParseFilterRule(const std::string::const_iterator begin,
|
2016-09-18 02:38:01 +02:00
|
|
|
const std::string::const_iterator end) {
|
2014-12-06 23:00:08 +01:00
|
|
|
auto level_separator = std::find(begin, end, ':');
|
|
|
|
if (level_separator == end) {
|
2018-06-29 13:18:07 +02:00
|
|
|
LOG_ERROR(Log, "Invalid log filter. Must specify a log level after `:`: {}",
|
2018-03-25 12:35:50 +02:00
|
|
|
std::string(begin, end));
|
2014-12-06 23:00:08 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Level level = GetLevelByName(level_separator + 1, end);
|
|
|
|
if (level == Level::Count) {
|
2018-06-29 13:18:07 +02:00
|
|
|
LOG_ERROR(Log, "Unknown log level in filter: {}", std::string(begin, end));
|
2014-12-06 23:00:08 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Common::ComparePartialString(begin, level_separator, "*")) {
|
|
|
|
ResetAll(level);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-05-12 07:19:44 +02:00
|
|
|
const Class log_class = GetClassByName(begin, level_separator);
|
2014-12-06 23:00:08 +01:00
|
|
|
if (log_class == Class::Count) {
|
2018-06-29 13:18:07 +02:00
|
|
|
LOG_ERROR(Log, "Unknown log class in filter: {}", std::string(begin, end));
|
2014-12-06 23:00:08 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-05-12 07:19:44 +02:00
|
|
|
SetClassLevel(log_class, level);
|
2014-12-06 23:00:08 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Filter::CheckMessage(Class log_class, Level level) const {
|
|
|
|
return static_cast<u8>(level) >= static_cast<u8>(class_levels[static_cast<size_t>(log_class)]);
|
|
|
|
}
|
2018-03-09 18:54:43 +01:00
|
|
|
} // namespace Log
|