blob: 64ff0857b2371c5edcf053675bb8dd97ca91c6cf [file] [log] [blame]
/*
* Copyright (c) 2013-2020, Intel Corporation
*
* Permission is hereby granted, free of charge, to any person obtaining a
* copy of this software and associated documentation files (the "Software"),
* to deal in the Software without restriction, including without limitation
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
* and/or sell copies of the Software, and to permit persons to whom the
* Software is furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included
* in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
* OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
* ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
* OTHER DEALINGS IN THE SOFTWARE.
*/
//!
//! \file mos_util_debug_specific.cpp
//! \brief This module implments Linux MOS debug functionalities
//!
#include "mos_util_debug.h"
#if MOS_MESSAGES_ENABLED
#include "mos_utilities_specific.h"
#include "mos_utilities.h"
#include "string.h"
#include <time.h> //get_clocktime
#include <unistd.h> //read, lseek
#include <fcntl.h> //open
#include "mos_util_debug_specific_next.h"
#ifdef ANDROID
#include <android/log.h>
#else
#include <signal.h>
#endif // ANDROID
extern int32_t MOS_ShouldPrintMessage(
MOS_MESSAGE_LEVEL level,
MOS_COMPONENT_ID compID,
uint8_t subCompID,
const PCCHAR message);
//!
//! \brief HLT log file prefix
//!
const PCCHAR MosLogPathPrefix = "/etc/log";
const PCCHAR MosUltLogPathPrefix = "./";
extern MOS_MESSAGE_PARAMS g_MosMsgParams;
extern uint8_t MosUltFlag;
static MOS_MUTEX gMosMsgMutex = PTHREAD_MUTEX_INITIALIZER;
//!
//! \brief Form a string that will prefix MOS's log file name
//! \details Form a string that will prefix MOS's log file name
//! MosLogPathPrefix defines the system directory that
//! Mos uses to prefix the log filename by default.
//! \param char *fileNamePrefix
//! [out] Pointer to the string where the prefix is returned
//! \return MOS_STATUS
//! Returns one of the MOS_STATUS error codes if failed,
//! else MOS_STATUS_SUCCESS
//!
MOS_STATUS MOS_LogFileNamePrefix(char *fileNamePrefix, MOS_CONTEXT_HANDLE mosCtx)
{
int32_t iRet = 0;
MOS_USER_FEATURE_VALUE_DATA UserFeatureData;
MOS_USER_FEATURE_VALUE_WRITE_DATA UserFeatureWriteData;
MOS_STATUS eStatus = MOS_STATUS_UNKNOWN;
if (MosUltFlag)
{
iRet = MOS_SecureStringPrint(
fileNamePrefix,
MOS_MAX_HLT_FILENAME_LEN,
MOS_MAX_HLT_FILENAME_LEN,
MosUltLogPathPrefix);
if (iRet > 0)
{
eStatus = MOS_STATUS_SUCCESS;
}
return eStatus;
}
MOS_ZeroMemory(&UserFeatureData, sizeof(UserFeatureData));
UserFeatureData.StringData.pStringData = fileNamePrefix;
eStatus = MOS_UserFeature_ReadValue_ID(
nullptr,
__MOS_USER_FEATURE_KEY_MESSAGE_HLT_OUTPUT_DIRECTORY_ID,
&UserFeatureData,
mosCtx);
// If the user feature key was not found, create it with the default value.
if (eStatus != MOS_STATUS_SUCCESS)
{
iRet = MOS_SecureStringPrint(
fileNamePrefix,
MOS_MAX_HLT_FILENAME_LEN,
MOS_MAX_HLT_FILENAME_LEN,
MosLogPathPrefix);
if (iRet > 0)
{
MOS_ZeroMemory(&UserFeatureWriteData, sizeof(UserFeatureWriteData));
UserFeatureWriteData.Value.StringData.pStringData = fileNamePrefix;
UserFeatureWriteData.Value.StringData.uSize = strlen(fileNamePrefix) + 1;
UserFeatureWriteData.ValueID = __MOS_USER_FEATURE_KEY_MESSAGE_HLT_OUTPUT_DIRECTORY_ID;
eStatus = MOS_UserFeature_WriteValues_ID(nullptr, &UserFeatureWriteData, 1, mosCtx);
}
else
{
return MOS_STATUS_UNKNOWN;
}
}
return eStatus;
}
//!
//! \brief Prints debug messages when enabled
//! \details Prints debug messages if prints are enabled and the level of the comp and sub-comp is
//! set to less than the message level.
//! \param MOS_MESSAGE_LEVEL level
//! [in] Level of the message
//! \param const PCCHAR logtag
//! [in] For Linux only, used for tagging the message.
//! \param MOS_COMPONENT_ID compID
//! [in] Indicates which component
//! \param uint8_t subCompID
//! [in] Indicates which sub-component
//! \param const char *functionName
//! [in] pointer to the function name
//! \param int32_t lineNum
//! [in] Indicates which line the message locate, -1 for no line output
//! \param const char *message
//! [in] pointer to the message format string
//! \param var_args
//! [in] variable list of arguments for the message
//! \return void
//!
void MOS_Message(
MOS_MESSAGE_LEVEL level,
const PCCHAR logtag,
MOS_COMPONENT_ID compID,
uint8_t subCompID,
const PCCHAR functionName,
int32_t lineNum,
const PCCHAR message,
...)
{
va_list var_args;
uint32_t nLen = 0;
PCCHAR func = functionName;
if (MosUtilDebugSpecific::MosShouldPrintMessage(level, compID, subCompID, message) == false)
{
return;
}
va_start(var_args, message);
MosUtilities::MosLockMutex(&MosUtilDebugSpecific::m_mosMsgMutex);
// Proceed to print the message
if (functionName == nullptr)
{
MosUtilities::MosSecureStringPrint(MosUtilDebug::m_mosMsgParams.g_MosMsgBuffer,
MOS_MAX_MSG_BUF_SIZE,
(MOS_MAX_MSG_BUF_SIZE-1),
"%s%s - ",
MosUtilDebug::m_mosComponentName[compID],
MosUtilDebug::m_mosLogLevelName[level]);
nLen = strlen(MosUtilDebug::m_mosMsgParams.g_MosMsgBuffer);
}
else
{
#if USE_PRETTY_FUNCTION
// call MosGetClassMethod to convert pretty function to class::function
// return string locate in static memory, mutex should be hold.
func = MosUtilDebugSpecific::MosGetClassMethod(functionName);
#endif //USE_PRETTY_FUNCTION
if (lineNum < 0)
{
// no line number output
MosUtilities::MosSecureStringPrint(MosUtilDebug::m_mosMsgParams.g_MosMsgBuffer,
MOS_MAX_MSG_BUF_SIZE,
(MOS_MAX_MSG_BUF_SIZE-1),
"%s%s - %s",
MosUtilDebug::m_mosComponentName[compID],
MosUtilDebug::m_mosLogLevelName[level],
func);
nLen = strlen(MosUtilDebug::m_mosMsgParams.g_MosMsgBuffer);
}
else
{
MosUtilities::MosSecureStringPrint(MosUtilDebug::m_mosMsgParams.g_MosMsgBuffer,
MOS_MAX_MSG_BUF_SIZE,
(MOS_MAX_MSG_BUF_SIZE-1),
"%s%s - %s:%d: ",
MosUtilDebug::m_mosComponentName[compID],
MosUtilDebug::m_mosLogLevelName[level],
func,
lineNum);
nLen = strlen(MosUtilDebug::m_mosMsgParams.g_MosMsgBuffer);
}
}
MosUtilities::MosSecureVStringPrint(MosUtilDebug::m_mosMsgParams.g_MosMsgBuffer + nLen,
MOS_MAX_MSG_BUF_SIZE - nLen,
(MOS_MAX_MSG_BUF_SIZE - 1 - nLen),
message,
var_args);
// Dump message to debugger if print to output window enabled
if (MosUtilDebug::m_mosMsgParams.bUseOutputDebugString)
{
printf("%s\n", MosUtilDebug::m_mosMsgParams.g_MosMsgBuffer);
}
// Write to log file if HLT enabled. File already open to add preface information
if (MosUtilDebug::m_mosMsgParams.bUseHybridLogTrace)
{
if (MosUtilDebug::m_mosMsgParams.pLogFile != nullptr)
{
nLen = strlen(MosUtilDebug::m_mosMsgParams.g_MosMsgBuffer);
fwrite(MosUtilDebug::m_mosMsgParams.g_MosMsgBuffer, nLen, 1, MosUtilDebug::m_mosMsgParams.pLogFile);
fprintf(MosUtilDebug::m_mosMsgParams.pLogFile, "\n");
}
else
{
printf("ERROR: m_mosMsgParams.pLogFile is NULL!\n");
}
}
MosUtilities::MosUnlockMutex(&MosUtilDebugSpecific::m_mosMsgMutex);
va_end(var_args);
return;
}
#if MOS_ASSERT_ENABLED
extern int32_t MOS_ShouldAssert(MOS_COMPONENT_ID compID, uint8_t subCompID);
//!
//! \brief MOS assert function for MOS internal use
//! \details Halts the cpu in debug mode when expression resolves to zero
//! and only if assert enabled for both comp and sub-comp.
//! Nop in release version
//! Called by MOS_ASSERT macro only
//! \param MOS_COMPONENT_ID compID
//! [in] Indicates which component
//! \param uint8_t subCompID
//! [in] Indicates which sub-component
//! \return void
//!
void _MOS_Assert(MOS_COMPONENT_ID compID, uint8_t subCompID)
{
return MosUtilDebug::MosAssert(compID,subCompID);
}
#endif // MOS_ASSERT_ENABLED
#endif // MOS_MESSAGES_ENABLED