onnxruntime/include/onnxruntime/core/common/logging/logging.h
RandySheriffH 75584c5fa8
Enabling thread pool to be numa-aware (#13778)
The PR enables ort thread pool to be numa-aware, so that threads could
be evenly created and distributed among numa nodes.
In addition, to facilitate performance tuning, the PR opens a new API
allowing customers to attach threads to certain logical processors.
Please check the API
[definition](https://github.com/microsoft/onnxruntime/pull/13778/files#diff-5845a5c76fb64abdc8f0cffe21b37f8da1712674eb3abc4cd87190891be1bd48)
for details.

Co-authored-by: Randy Shuai <rashuai@microsoft.com>
2022-12-12 10:33:55 -08:00

337 lines
12 KiB
C++

// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License.
#pragma once
#include <atomic>
#include <chrono>
#include <climits>
#include <map>
#include <memory>
#include <mutex>
#include <string>
#include "core/common/common.h"
#include "core/common/profiler_common.h"
#include "core/common/logging/capture.h"
#include "core/common/logging/severity.h"
#include "core/common/logging/macros.h"
/*
Logging overview and expected usage:
At program startup:
* Create one or more ISink instances. If multiple, combine using composite_sink.
* Create a LoggingManager instance with the sink/s with is_default_instance set to true
* Only one instance should be created in this way, and it should remain valid for
until the program no longer needs to produce log output.
You can either use the static default Logger which LoggingManager will create when constructed
via LoggingManager::DefaultLogger(), or separate Logger instances each with different log ids
via LoggingManager::CreateLogger.
The log id is passed to the ISink instance with the sink determining how the log id is used
in the output.
LoggingManager
* creates the Logger instances used by the application
* provides a static default logger instance
* owns the log sink instance
* applies checks on severity and output of user data
The log macros create a Capture instance to capture the information to log.
If the severity and/or user filtering settings would prevent logging, no evaluation
of the log arguments will occur, so no performance cost beyond the severity and user
filtering check.
A sink can do further filter as needed.
*/
namespace onnxruntime {
namespace logging {
using Timestamp = std::chrono::time_point<std::chrono::system_clock>;
#ifndef NDEBUG
ORT_ATTRIBUTE_UNUSED static bool vlog_enabled = true; // Set directly based on your needs.
#else
constexpr bool vlog_enabled = false; // no VLOG output
#endif
enum class DataType {
SYSTEM = 0, ///< System data.
USER = 1 ///< Contains potentially sensitive user data.
};
// Internal log categories.
// Logging interface takes const char* so arbitrary values can also be used.
struct Category {
static const char* onnxruntime; ///< General output
static const char* System; ///< Log output regarding interactions with the host system
// TODO: What other high level categories are meaningful? Model? Optimizer? Execution?
};
class ISink;
class Logger;
class Capture;
/// <summary>
/// The logging manager.
/// Owns the log sink and potentially provides a default Logger instance.
/// Provides filtering based on a minimum LogSeverity level, and of messages with DataType::User if enabled.
/// </summary>
class LoggingManager final {
public:
enum InstanceType {
Default, ///< Default instance of LoggingManager that should exist for the lifetime of the program
Temporal ///< Temporal instance. CreateLogger(...) should be used, however DefaultLogger() will NOT be provided via this instance.
};
/**
Initializes a new instance of the LoggingManager class.
@param sink The sink to write to. Use CompositeSink if you need to write to multiple places.
@param default_min_severity The default minimum severity. Messages with lower severity will be ignored unless
overridden in CreateLogger.
@param default_filter_user_data If set to true ignore messages with DataType::USER unless overridden in CreateLogger.
@param instance_type If InstanceType::Default, this is the default instance of the LoggingManager
and is expected to exist for the lifetime of the program.
It creates and owns the default logger that calls to the static DefaultLogger method return.
@param default_logger_id Logger Id to use for the default logger. nullptr/ignored if instance_type == Temporal.
@param default_max_vlog_level Default maximum level for VLOG messages to be created unless overridden in CreateLogger.
Requires a severity of kVERBOSE for VLOG messages to be logged.
*/
LoggingManager(std::unique_ptr<ISink> sink, Severity default_min_severity, bool default_filter_user_data,
InstanceType instance_type,
const std::string* default_logger_id = nullptr,
int default_max_vlog_level = -1);
/**
Creates a new logger instance which will use the provided logger_id and default severity and vlog levels.
@param logger_id The log identifier.
@returns A new Logger instance that the caller owns.
*/
std::unique_ptr<Logger> CreateLogger(const std::string& logger_id);
/**
Creates a new logger instance which will use the provided logger_id, severity and vlog levels.
@param logger_id The log identifier.
@param min_severity The minimum severity. Requests to create messages with lower severity will be ignored.
@param filter_user_data If set to true ignore messages with DataType::USER.
@param max_vlog_level Maximum level for VLOG messages to be created.
@returns A new Logger instance that the caller owns.
*/
std::unique_ptr<Logger> CreateLogger(const std::string& logger_id,
Severity min_severity, bool filter_user_data, int max_vlog_level = -1);
/**
Gets the default logger instance if set. Throws if no default logger is currently registered.
@remarks
Creating a LoggingManager instance with is_default_instance == true registers a default logger.
Note that the default logger is only valid until the LoggerManager that registered it is destroyed.
@returns The default logger if available.
*/
static const Logger& DefaultLogger();
/**
Return a boolean indicating if the default logger has been initialized
*/
static bool HasDefaultLogger() { return nullptr != s_default_logger_; }
/**
Change the minimum severity level for log messages to be output by the default logger.
@param severity The severity.
*/
static void SetDefaultLoggerSeverity(Severity severity);
/**
Change the maximum verbosity level for log messages to be output by the default logger.
@remarks
To activate the verbose log, the logger severity must also be set to kVERBOSE.
@param vlog_level The verbosity level.
*/
static void SetDefaultLoggerVerbosity(int vlog_level);
/**
Logs a FATAL level message and creates an exception that can be thrown with error information.
@param category The log category.
@param location The location the log message was generated.
@param format_str The printf format string.
@param ... The printf arguments.
@returns A new Logger instance that the caller owns.
*/
static std::exception LogFatalAndCreateException(const char* category,
const CodeLocation& location,
const char* format_str, ...);
/**
Logs the message using the provided logger id.
@param logger_id The log identifier.
@param message The log message.
*/
void Log(const std::string& logger_id, const Capture& message) const;
/**
Sends a Profiling Event Record to the sink.
@param Profiling Event Record
*/
void SendProfileEvent(profiling::EventRecord& eventRecord) const;
~LoggingManager();
private:
ORT_DISALLOW_COPY_ASSIGNMENT_AND_MOVE(LoggingManager);
Timestamp GetTimestamp() const noexcept;
void CreateDefaultLogger(const std::string& logger_id);
std::unique_ptr<ISink> sink_;
const Severity default_min_severity_;
const bool default_filter_user_data_;
const int default_max_vlog_level_;
bool owns_default_logger_;
static Logger* s_default_logger_;
struct Epochs {
const std::chrono::time_point<std::chrono::high_resolution_clock> high_res;
const std::chrono::time_point<std::chrono::system_clock> system;
const std::chrono::minutes localtime_offset_from_utc;
};
static const Epochs& GetEpochs() noexcept;
};
/**
Logger provides a per-instance log id. Everything else is passed back up to the LoggingManager
*/
class Logger {
public:
/**
Initializes a new instance of the Logger class.
@param loggingManager The logging manager.
@param id The identifier for messages coming from this Logger.
@param severity Minimum severity for messages to be created and logged.
@param filter_user_data Should USER data be filtered from output.
@param vlog_level Minimum level for VLOG messages to be created. Note that a severity of kVERBOSE must be provided
for VLOG messages to be logged.
*/
Logger(const LoggingManager& loggingManager, std::string id,
Severity severity, bool filter_user_data, int vlog_level)
: logging_manager_{&loggingManager},
id_{id},
min_severity_{severity},
filter_user_data_{filter_user_data},
max_vlog_level_{vlog_level} {
}
/**
Get the minimum severity level for log messages to be output.
@returns The severity.
*/
Severity GetSeverity() const noexcept { return min_severity_; }
/**
Change the minimum severity level for log messages to be output.
@param severity The severity.
*/
void SetSeverity(Severity severity) noexcept { min_severity_ = severity; }
/**
Change the maximum verbosity level for log messages to be output.
@remarks
To activate the verbose log, the logger severity must also be set to kVERBOSE.
@param vlog_level The verbosity.
*/
void SetVerbosity(int vlog_level) noexcept { max_vlog_level_ = vlog_level; }
/**
Check if output is enabled for the provided LogSeverity and DataType values.
@param severity The severity.
@param data_type Type of the data.
@returns True if a message with these values will be logged.
*/
bool OutputIsEnabled(Severity severity, DataType data_type) const noexcept {
return (severity >= min_severity_ && (data_type != DataType::USER || !filter_user_data_));
}
/**
Return the maximum VLOG level allowed. Disabled unless logging VLOG messages
*/
int VLOGMaxLevel() const noexcept {
return min_severity_ > Severity::kVERBOSE ? -1 : max_vlog_level_;
}
/**
Logs the captured message.
@param message The log message.
*/
void Log(const Capture& message) const {
logging_manager_->Log(id_, message);
}
/**
Sends a Profiling Event Record to the sink.
@param Profiling Event Record
*/
void SendProfileEvent(profiling::EventRecord& eventRecord) const {
logging_manager_->SendProfileEvent(eventRecord);
}
private:
const LoggingManager* logging_manager_;
const std::string id_;
Severity min_severity_;
const bool filter_user_data_;
int max_vlog_level_;
};
inline const Logger& LoggingManager::DefaultLogger() {
if (s_default_logger_ == nullptr) {
// fail early for attempted misuse. don't use logging macros as we have no logger.
ORT_THROW("Attempt to use DefaultLogger but none has been registered.");
}
return *s_default_logger_;
}
inline void LoggingManager::SetDefaultLoggerSeverity(Severity severity) {
if (s_default_logger_ == nullptr) {
// fail early for attempted misuse. don't use logging macros as we have no logger.
ORT_THROW("Attempt to use DefaultLogger but none has been registered.");
}
s_default_logger_->SetSeverity(severity);
}
inline void LoggingManager::SetDefaultLoggerVerbosity(int vlog_level) {
if (s_default_logger_ == nullptr) {
// fail early for attempted misuse. don't use logging macros as we have no logger.
ORT_THROW("Attempt to use DefaultLogger but none has been registered.");
}
s_default_logger_->SetVerbosity(vlog_level);
}
inline Timestamp LoggingManager::GetTimestamp() const noexcept {
static const Epochs& epochs = GetEpochs();
const auto high_res_now = std::chrono::high_resolution_clock::now();
return std::chrono::time_point_cast<std::chrono::system_clock::duration>(
epochs.system + (high_res_now - epochs.high_res) + epochs.localtime_offset_from_utc);
}
/**
Return the current thread id.
*/
unsigned int GetThreadId();
/**
Return the current process id.
*/
unsigned int GetProcessId();
} // namespace logging
} // namespace onnxruntime