#pragma once #include #include #ifdef __ANDROID__ #include #endif namespace Log { // Our logger class template class Logger { public: void log(const char* fmt, ...) { if constexpr (!enabled) return; std::va_list args; va_start(args, fmt); #ifdef __ANDROID__ __android_log_vprint(ANDROID_LOG_DEFAULT, "Panda3DS", fmt, args); #else std::vprintf(fmt, args); #endif va_end(args); } }; // Our loggers here. Enable/disable by toggling the template param static Logger kernelLogger; // Enables output for the outputDebugString SVC static Logger debugStringLogger; static Logger errorLogger; static Logger fileIOLogger; static Logger svcLogger; static Logger threadLogger; static Logger gpuLogger; static Logger rendererLogger; static Logger shaderJITLogger; static Logger dspLogger; // Service loggers static Logger acLogger; static Logger actLogger; static Logger amLogger; static Logger aptLogger; static Logger bossLogger; static Logger camLogger; static Logger cecdLogger; static Logger cfgLogger; static Logger csndLogger; static Logger dspServiceLogger; static Logger dlpSrvrLogger; static Logger frdLogger; static Logger fsLogger; static Logger hidLogger; static Logger httpLogger; static Logger irUserLogger; static Logger gspGPULogger; static Logger gspLCDLogger; static Logger ldrLogger; static Logger mcuLogger; static Logger micLogger; static Logger newsLogger; static Logger nfcLogger; static Logger nwmUdsLogger; static Logger nimLogger; static Logger ndmLogger; static Logger nsLogger; static Logger ptmLogger; static Logger socLogger; static Logger sslLogger; static Logger y2rLogger; static Logger srvLogger; // We have 2 ways to create a log function // MAKE_LOG_FUNCTION: Creates a log function which is toggleable but always killed for user-facing builds // MAKE_LOG_FUNCTION_USER: Creates a log function which is toggleable, may be on for user builds as well // We need this because sadly due to the loggers taking variadic arguments, compilers will not properly // Kill them fully even when they're disabled. The only way they will is if the function with varargs is totally empty #define MAKE_LOG_FUNCTION_USER(functionName, logger) \ template \ void functionName(const char* fmt, Args&&... args) { \ Log::logger.log(fmt, args...); \ } #ifdef PANDA3DS_USER_BUILD #define MAKE_LOG_FUNCTION(functionName, logger) \ template \ void functionName(const char* fmt, Args&&... args) {} #else #define MAKE_LOG_FUNCTION(functionName, logger) MAKE_LOG_FUNCTION_USER(functionName, logger) #endif }