1119 lines
31 KiB
C
1119 lines
31 KiB
C
/** @file log.c
|
|
@brief 系统日志接口文件
|
|
@version 1.0.0
|
|
*/
|
|
#include <sys/time.h>
|
|
#include <fcntl.h>
|
|
#include <stdarg.h>
|
|
#include <sys/stat.h>
|
|
#include <errno.h>
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <time.h>
|
|
#include <unistd.h>
|
|
#include <pthread.h>
|
|
#include <uthash/utlist.h>
|
|
#include <uv.h>
|
|
#include <uthash/utstring.h>
|
|
#include <uthash/utringbuffer.h>
|
|
#include <linux/mii.h>
|
|
#include <linux/sockios.h>
|
|
#include <sys/ioctl.h>
|
|
#include <net/if.h>
|
|
#ifdef PLATFORM_R16
|
|
#include <cutils/cpu_info.h>
|
|
#endif
|
|
|
|
#include "log.h"
|
|
#include "smart_sound.h"
|
|
#include "inet_api.h"
|
|
#include "crypto.h"
|
|
#include "libuv_dbus.h"
|
|
#include "server_addr.h"
|
|
|
|
#define SHOW_CONSOLE_RED ("\033[31;48m\033[1m%s\033[0m%c")
|
|
#define SHOW_CONSOLE_YELLOW ("\033[33;48m\033[1m%s\033[0m%c")
|
|
#define SHOW_CONSOLE_GREEN ("\033[32;48m\033[1m%s\033[0m%c")
|
|
#define SHOW_CONSOLE_BLUE ("\033[34;48m\033[1m%s\033[0m%c")
|
|
|
|
#ifdef PLATFORM_CPU
|
|
#define LOG_FILE_BASEDIR (".")
|
|
#else
|
|
#define LOG_FILE_BASEDIR ("/tmp")
|
|
#endif
|
|
|
|
#ifdef PLATFORM_R16
|
|
#define NIC_NAME ("wlan0")
|
|
#else
|
|
#define NIC_NAME ("enp3s0")
|
|
#endif
|
|
|
|
#define MAX_LOG_ITEM (1000)
|
|
#define MAX_LOG_FILE_SIZE (1024 * 1024)
|
|
#define LOG_PRE_SIZE (512)
|
|
|
|
#if 0
|
|
#define LOG_MAIL_USERNAME ("pv1_es2@hotmail.com")
|
|
#define LOG_MAIL_PASSWORD ("netEase163")
|
|
#define LOG_MAIL_SERVER ("smtp://smtp-mail.outlook.com")
|
|
#define LOG_MAIL_PORT (587)
|
|
#else
|
|
#define LOG_MAIL_USERNAME ("pv1_es2@163.com")
|
|
#define LOG_MAIL_LOGPASSWD ("netEase163")
|
|
#define LOG_MAIL_PASSWORD ("pv1Dev163")
|
|
#define LOG_MAIL_SERVER ("smtp.163.com")
|
|
#define LOG_MAIL_PORT (25)
|
|
#endif
|
|
|
|
#define LOG_MAIL_RECEIVER ("pv1_es2@163.com")
|
|
|
|
#define SYS_POINT_UPLOAD_TIME (600)
|
|
|
|
typedef struct
|
|
{
|
|
char* pDeviceId;
|
|
char savePath[MAX_PATH];
|
|
int iMaxCacheSize;
|
|
int iEveryUploadTime;
|
|
time_t lastPostTime;
|
|
time_t lastMarkTime;
|
|
FILE* pMarkFile;
|
|
uv_rwlock_t uvRwLock;
|
|
uv_thread_t uvIOThread;
|
|
} SYSPOINT_INFO, *PSYSPOINT_INFO;
|
|
|
|
typedef struct SYSPOINT_ITEM
|
|
{
|
|
char* pContent;
|
|
struct SYSPOINT_ITEM *next, *prev;
|
|
} *PSYSPOINT_ITEM;
|
|
|
|
typedef struct LOG_ITEM
|
|
{
|
|
LOG_LEVEL level;
|
|
int isPrinted;
|
|
int isAddTags;
|
|
struct timeval timestamp;
|
|
char *pLogContent;
|
|
struct LOG_ITEM *next, *prev;
|
|
}*PLOG_ITEM;
|
|
|
|
typedef struct LOG_BACKUP
|
|
{
|
|
time_t timestamp;
|
|
char *plogFilePath;
|
|
char *pGzFilePath;
|
|
unsigned int sendTimes;
|
|
unsigned int tolLogs;
|
|
struct LOG_BACKUP *next, *prev;
|
|
} *PLOG_BACKUP;
|
|
|
|
typedef struct
|
|
{
|
|
pid_t pid;
|
|
char* pChipId;
|
|
char* pChipSerial;
|
|
char exeName[MAX_PATH];
|
|
char logFilePath[MAX_PATH];
|
|
FILE *pLogFile;
|
|
} LOG_PROCESS_INFO, *PLOG_PROCESS_INFO;
|
|
|
|
static int g_bEnableLog = FALSE; // 是否启用 Log 功能
|
|
static int g_bEnMailBackup = FALSE;
|
|
static char* g_pEmailBox = NULL;
|
|
static int g_bEnLogToFile = TRUE;
|
|
static int g_bEnLogToServer = TRUE;
|
|
static char g_strLogTag[32]; // Log 标志
|
|
static unsigned int g_LogRdPos = 0;
|
|
static pthread_t g_logThreadId;
|
|
static pthread_t g_backupThreadId;
|
|
static LOG_PROCESS_INFO g_LogProcessInfo;
|
|
static uv_rwlock_t g_uvLogLock;
|
|
static uv_rwlock_t g_uvLogBackLock;
|
|
static PLOG_ITEM g_pLogItemList = NULL;
|
|
static PLOG_BACKUP g_pLogBackupList = NULL;
|
|
static int g_logSock = -1;
|
|
struct sockaddr_in g_logAddr;
|
|
|
|
static SYSPOINT_INFO g_SysPonitInfo;
|
|
static PSYSPOINT_ITEM g_pSysPointArray = NULL;
|
|
static PHTTP_POST_ATTACH g_pLogParamsArray = NULL;
|
|
|
|
|
|
static int g_iMinLevel = LOG_Fatal | LOG_Error | LOG_Warn | LOG_Debug | LOG_Info | LOG_Step;
|
|
|
|
/**
|
|
* @brief Log 调试等级转字符串
|
|
* @param level 调试等级
|
|
* @return 调试等级对应的字符串
|
|
*/
|
|
const char* LogLevelToStr(LOG_LEVEL level)
|
|
{
|
|
switch(level)
|
|
{
|
|
case LOG_Test:
|
|
return "T";
|
|
|
|
case LOG_Info:
|
|
return "I";
|
|
|
|
case LOG_Call:
|
|
return "C";
|
|
|
|
case LOG_Debug:
|
|
return "D";
|
|
|
|
case LOG_Warn:
|
|
return "W";
|
|
|
|
case LOG_Error:
|
|
return "E";
|
|
|
|
case LOG_Fatal:
|
|
return "F";
|
|
|
|
case LOG_Step:
|
|
return "S";
|
|
|
|
case LOG_Devp:
|
|
return "V";
|
|
|
|
case LOG_Unknown:
|
|
return "U";
|
|
|
|
case LOG_All:
|
|
return "A";
|
|
|
|
default:
|
|
return "?";
|
|
}
|
|
|
|
return "U";
|
|
}
|
|
|
|
#pragma pack (push)
|
|
#pragma pack (1)
|
|
typedef struct
|
|
{
|
|
unsigned short logSeq;
|
|
unsigned int pid;
|
|
unsigned int timeStamp;
|
|
unsigned int nanotime;
|
|
unsigned int logLevel;
|
|
char logContent[0];
|
|
} LOG_PROTO_CONTENT, *PLOG_PROTO_CONTENT;
|
|
#pragma pack (pop)
|
|
|
|
static struct HTTP_POST_ATTACH g_PostLogParams[] =
|
|
{
|
|
{"CPUID", "", NULL, NULL},
|
|
{"MAC", "", NULL, NULL},
|
|
{"Version", "", NULL, NULL},
|
|
{"BuildTime", "", NULL, NULL},
|
|
{"SerialNumber", "", NULL, NULL},
|
|
{"Datetime", "", NULL, NULL},
|
|
{"content", "", NULL, NULL},
|
|
};
|
|
|
|
static void __initPostLogParams(void)
|
|
{
|
|
char *pMacAddr = NULL;
|
|
struct tm localTime;
|
|
time_t timeStamp = time((time_t*)NULL);
|
|
|
|
GetShellExecResult("ifconfig wlan0 | grep HWaddr | awk \'{print $5}\'", &pMacAddr);
|
|
|
|
localtime_r(&timeStamp, &localTime);
|
|
|
|
strncpy(g_PostLogParams[0].keyValue, g_LogProcessInfo.pChipId, MAX_HTTP_POST_SIZE);
|
|
strncpy(g_PostLogParams[1].keyValue, (pMacAddr && strlen(pMacAddr) > 0) ? pMacAddr : "00:00:00:00:00:00", MAX_HTTP_POST_SIZE);
|
|
strncpy(g_PostLogParams[2].keyValue, GetCurrentVersion(), MAX_HTTP_POST_SIZE);
|
|
sprintf(g_PostLogParams[3].keyValue, "Build: %s %s", __DATE__, __TIME__);
|
|
strncpy(g_PostLogParams[4].keyValue, g_LogProcessInfo.pChipSerial, MAX_HTTP_POST_SIZE);
|
|
|
|
for(int i = 0; i < sizeof(g_PostLogParams) / sizeof(g_PostLogParams[0]); i++)
|
|
{
|
|
LL_APPEND(g_pLogParamsArray, &g_PostLogParams[i]);
|
|
}
|
|
|
|
if(pMacAddr)
|
|
{
|
|
free(pMacAddr);
|
|
}
|
|
}
|
|
|
|
int HttpPostLogFile(char* pSession)
|
|
{
|
|
int ret;
|
|
struct tm localTime;
|
|
const char* pFileName = "/tmp/backuplog.tar.gz";
|
|
const char* pLastFile = "/mnt/UDISK/backuplog_nand.tar.gz";
|
|
char* pLogServerURL = GetCurServerAddr(LOG_MODULE);
|
|
time_t timeStamp = time((time_t*)NULL);
|
|
|
|
ret = system(" /usr/sbin/backuplocalfiles.sh");
|
|
|
|
if(access(pFileName, F_OK) != 0
|
|
&& access(pLastFile, F_OK) != 0)
|
|
{
|
|
return -ERR_FILE_NOT_EXISTS;
|
|
}
|
|
|
|
localtime_r(&timeStamp, &localTime);
|
|
|
|
memset(g_PostLogParams[5].keyValue, 0, MAX_HTTP_POST_SIZE);
|
|
sprintf(g_PostLogParams[5].keyValue, "%04u-%02u-%02u %02u:%02u:%02u",
|
|
localTime.tm_year + 1900,
|
|
localTime.tm_mon + 1,
|
|
localTime.tm_mday,
|
|
localTime.tm_hour,
|
|
localTime.tm_min,
|
|
localTime.tm_sec);
|
|
|
|
memset(g_PostLogParams[6].keyValue, 0, MAX_HTTP_POST_SIZE);
|
|
|
|
if(pSession && strlen(pSession) >= 2)
|
|
{
|
|
strncpy(g_PostLogParams[6].keyValue, pSession, MAX_HTTP_POST_SIZE);
|
|
}
|
|
else
|
|
{
|
|
strncpy(g_PostLogParams[6].keyValue, "{}", MAX_HTTP_POST_SIZE);
|
|
}
|
|
|
|
if(access(pFileName, F_OK) == 0)
|
|
{
|
|
ret = InetHttpUploadFileSync(pLogServerURL, pFileName, g_pLogParamsArray);
|
|
|
|
if(ret == 0)
|
|
{
|
|
LOG_EX(LOG_Debug, "Upload Log Data [%s] To Server [%s]\n", pFileName, pLogServerURL);
|
|
unlink(pFileName);
|
|
}
|
|
else
|
|
{
|
|
LOG_EX(LOG_Error, "Upload Log Data [%s] To Server [%s]\n", pFileName, pLogServerURL);
|
|
}
|
|
}
|
|
|
|
if(access(pLastFile, F_OK) == 0)
|
|
{
|
|
ret = InetHttpUploadFileSync(pLogServerURL, pLastFile, g_pLogParamsArray);
|
|
|
|
if(ret == 0)
|
|
{
|
|
LOG_EX(LOG_Debug, "Upload Log Data [%s] To Server [%s]\n", pLastFile, pLogServerURL);
|
|
unlink(pLastFile);
|
|
}
|
|
else
|
|
{
|
|
LOG_EX(LOG_Error, "Upload Log Data [%s] To Server [%s]\n", pLastFile, pLogServerURL);
|
|
}
|
|
}
|
|
|
|
//fprintf(stdout, "Upload Log Data [%s] To Server [%s], size = %d\n", pFileName, UPL_HTTP_URL, fileSize);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int __logNetworkSend(PLOG_ITEM pItem)
|
|
{
|
|
PLOG_PROTO_CONTENT pLogInfo;
|
|
unsigned char *pBuf;
|
|
int ret, totalSize = 0;
|
|
static unsigned short sendSeq = 0;
|
|
|
|
if(pItem == NULL || pItem->pLogContent == NULL || strlen(pItem->pLogContent) == 0)
|
|
{
|
|
return (-ERR_INPUT_PARAMS);
|
|
}
|
|
|
|
totalSize = sizeof(LOG_PROTO_CONTENT) + strlen(pItem->pLogContent);
|
|
|
|
pBuf = (unsigned char *)malloc(totalSize);
|
|
|
|
if(pBuf == NULL)
|
|
{
|
|
return (-ERR_MALLOC_MEMORY);
|
|
}
|
|
|
|
memset(pBuf, 0, totalSize);
|
|
|
|
pLogInfo = (PLOG_PROTO_CONTENT)pBuf;
|
|
|
|
pLogInfo->logSeq = htons(sendSeq++);
|
|
pLogInfo->timeStamp = htonl(pItem->timestamp.tv_sec);
|
|
pLogInfo->nanotime = htonl(pItem->timestamp.tv_usec);
|
|
pLogInfo->pid = htonl(g_LogProcessInfo.pid);
|
|
pLogInfo->logLevel = htonl(pItem->level);
|
|
|
|
memcpy(pLogInfo->logContent, pItem->pLogContent, strlen(pItem->pLogContent));
|
|
|
|
ret = sendto(g_logSock, pBuf, totalSize, 0,
|
|
(struct sockaddr *)&g_logAddr, sizeof(g_logAddr));
|
|
|
|
if(ret == totalSize)
|
|
{
|
|
return (0);
|
|
}
|
|
else
|
|
{
|
|
//LOG_EX(LOG_Error, "Send %d bytes, need %d bytes\n", ret, totalSize);
|
|
return (-ERR_NETWORK_SEND);
|
|
}
|
|
}
|
|
|
|
static void __cleanupBackupItem(PLOG_BACKUP pItem)
|
|
{
|
|
if(pItem)
|
|
{
|
|
LL_DELETE(g_pLogBackupList, pItem);
|
|
|
|
if(pItem->plogFilePath)
|
|
{
|
|
if(access(pItem->plogFilePath, F_OK) == 0)
|
|
{
|
|
unlink(pItem->plogFilePath);
|
|
}
|
|
|
|
free(pItem->plogFilePath);
|
|
}
|
|
|
|
if(pItem->pGzFilePath)
|
|
{
|
|
if(access(pItem->pGzFilePath, F_OK) == 0)
|
|
{
|
|
unlink(pItem->pGzFilePath);
|
|
}
|
|
|
|
free(pItem->pGzFilePath);
|
|
}
|
|
|
|
free(pItem);
|
|
}
|
|
}
|
|
|
|
static void* __uvLogBackupProc(void *pParams)
|
|
{
|
|
SMTP_MAIL_CONFIG smtpCfg;
|
|
char gzLogPath[MAX_PATH];
|
|
struct tm localTime;
|
|
const char *pFrom = LOG_MAIL_USERNAME;
|
|
const char *pTo[] = {g_pEmailBox, NULL};
|
|
char *pMacAddr = NULL;
|
|
|
|
//const char *pCc[] = {"xajhuang@qq.com", "xajhuang@163.com", NULL};
|
|
|
|
memset(&smtpCfg, 0, sizeof(SMTP_MAIL_CONFIG));
|
|
|
|
smtpCfg.pUserName = LOG_MAIL_USERNAME;
|
|
smtpCfg.pPassword = LOG_MAIL_PASSWORD;
|
|
smtpCfg.pSmtpServer = LOG_MAIL_SERVER;
|
|
smtpCfg.smtpPort = LOG_MAIL_PORT;
|
|
|
|
#ifdef PLATFORM_CPU
|
|
GetShellExecResult("ifconfig enp3s0 | grep HWaddr | awk \'{print $5}\'", &pMacAddr);
|
|
#else
|
|
GetShellExecResult("ifconfig wlan0 | grep HWaddr | awk \'{print $5}\'", &pMacAddr);
|
|
#endif
|
|
|
|
if(pMacAddr == NULL || strlen(pMacAddr) == 0)
|
|
{
|
|
pMacAddr = strdup("00:00:00:00:00:00");
|
|
}
|
|
|
|
while(TRUE)
|
|
{
|
|
PLOG_BACKUP pItem = NULL, pTmp = NULL;
|
|
time_t timeStamp = time((time_t*)NULL);
|
|
localtime_r(&timeStamp, &localTime);
|
|
|
|
uv_rwlock_wrlock(&g_uvLogBackLock);
|
|
|
|
LL_FOREACH_SAFE(g_pLogBackupList, pItem, pTmp)
|
|
{
|
|
const char *pAttact[2];
|
|
UT_string *pTitle, *pMessage, *pDatetime;
|
|
char* pMD5Val;
|
|
|
|
pAttact[1] = NULL;
|
|
|
|
if(pItem->plogFilePath == NULL || strlen(pItem->plogFilePath) == 0)
|
|
{
|
|
__cleanupBackupItem(pItem);
|
|
continue;
|
|
}
|
|
|
|
if(timeStamp - pItem->timestamp >= 3600)
|
|
{
|
|
__cleanupBackupItem(pItem);
|
|
continue;
|
|
}
|
|
|
|
if(pItem->sendTimes >= 3)
|
|
{
|
|
__cleanupBackupItem(pItem);
|
|
continue;
|
|
}
|
|
#if 0
|
|
if(g_bEnLogToServer)
|
|
{
|
|
__logHttpPostFile(pItem->plogFilePath);
|
|
//InetHttpUploadFileSync(UPL_HTTP_URL, pItem->pGzFilePath, NULL);
|
|
}
|
|
#endif
|
|
if(pItem->pGzFilePath == NULL)
|
|
{
|
|
memset(gzLogPath, 0, MAX_PATH);
|
|
sprintf(gzLogPath, "%s/%d_%s_%u.log.gz",
|
|
LOG_FILE_BASEDIR, g_LogProcessInfo.pid, g_LogProcessInfo.exeName, pItem->tolLogs);
|
|
|
|
if(GZipFileCompress(pItem->plogFilePath, gzLogPath) != 0)
|
|
{
|
|
LOG_EX(LOG_Error, "Create Gzip File Error: %s\n", gzLogPath);
|
|
continue;
|
|
}
|
|
else
|
|
{
|
|
pItem->pGzFilePath = strdup(gzLogPath);
|
|
}
|
|
}
|
|
|
|
pMD5Val = (char*)EvpMD5HashFile(pItem->pGzFilePath);
|
|
|
|
utstring_new(pTitle);
|
|
utstring_new(pMessage);
|
|
utstring_new(pDatetime);
|
|
|
|
utstring_printf(pDatetime, "%04u-%02u-%02u %02u:%02u:%02u",
|
|
localTime.tm_year + 1900,
|
|
localTime.tm_mon + 1,
|
|
localTime.tm_mday,
|
|
localTime.tm_hour,
|
|
localTime.tm_min,
|
|
localTime.tm_sec);
|
|
|
|
utstring_printf(pTitle, "Log [%s]-[%s]-[%s]",
|
|
g_LogProcessInfo.exeName,
|
|
g_LogProcessInfo.pChipSerial,
|
|
utstring_body(pDatetime));
|
|
|
|
utstring_printf(pMessage, "<html><body>\r\n"
|
|
"<p><span style=\"font-size:24px;\"><strong>Summary information</strong></span><span style=\"font-size:24px;\">:</span></p>"
|
|
"<p><br /></p>"
|
|
"<p><table style=\"width:35%%;\" cellpadding=\"2\" cellspacing=\"0\" border=\"1\" bordercolor=\"#000000\">"
|
|
"<tbody>"
|
|
"<tr><td><b>Items</b><br /></td><td>%u<br /></td></tr>"
|
|
"<tr><td><b>Process PID</b><br /></td><td>%d<br /></td></tr>"
|
|
"<tr><td><b>Process Name</b><br /></td><td>%s<br /></td></tr>"
|
|
"<tr><td><b>CPU ID</b><br /></td><td>%s<br /></td></tr>"
|
|
"<tr><td><b>Serial</b><br /></td><td>%s<br /></td></tr>"
|
|
"<tr><td><b>Mac Address</b><br /></td><td>%s<br /></td></tr>"
|
|
"<tr><td><b>File Checksum</b><br /></td><td>%s<br /></td></tr>"
|
|
"<tr><td><b>Log Level</b><br /></td><td>0x%08X<br /></td></tr>"
|
|
"<tr><td><b>Datetime</b><br /></td><td>%s<br /></td></tr>"
|
|
"</tbody>"
|
|
"</table>"
|
|
"</p>"
|
|
"</body></html>\r\n",
|
|
pItem->tolLogs,
|
|
g_LogProcessInfo.pid,
|
|
g_LogProcessInfo.exeName,
|
|
g_LogProcessInfo.pChipId,
|
|
g_LogProcessInfo.pChipSerial,
|
|
pMacAddr,
|
|
SAFE_STRING_VALUE(pMD5Val),
|
|
g_iMinLevel,
|
|
utstring_body(pDatetime));
|
|
|
|
pAttact[0] = pItem->pGzFilePath;
|
|
#if 0
|
|
if(g_bEnMailBackup &&
|
|
InetSmtpSendEmail(pFrom, pTo, NULL, utstring_body(pTitle),
|
|
utstring_body(pMessage), pAttact, &smtpCfg) == 0)
|
|
{
|
|
__cleanupBackupItem(pItem);
|
|
}
|
|
else
|
|
{
|
|
LOG_EX(LOG_Error, "Send email error: %s\n", pItem->pGzFilePath);
|
|
}
|
|
#endif
|
|
pItem->sendTimes++;
|
|
|
|
if(pMD5Val)
|
|
{
|
|
free(pMD5Val);
|
|
}
|
|
|
|
utstring_free(pTitle);
|
|
utstring_free(pMessage);
|
|
utstring_free(pDatetime);
|
|
}
|
|
|
|
uv_rwlock_wrunlock(&g_uvLogBackLock);
|
|
|
|
sleep(1);
|
|
}
|
|
|
|
free(pMacAddr);
|
|
pthread_detach(pthread_self());
|
|
return (NULL);
|
|
}
|
|
|
|
static void __logColorOutput(const char* pColFmt, UT_string* pLog)
|
|
{
|
|
if(pLog == NULL)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if(pColFmt == NULL)
|
|
{
|
|
print("%s", utstring_body(pLog));
|
|
}
|
|
else
|
|
{
|
|
if(utstring_find(pLog, -1, "\n", 1) == utstring_len(pLog) - 1)
|
|
{
|
|
char* pLogArray = utstring_body(pLog);
|
|
pLogArray[utstring_len(pLog) - 1] = 0;
|
|
|
|
print(pColFmt, pLogArray, '\n');
|
|
|
|
strcat(pLogArray, "\n");
|
|
}
|
|
else
|
|
{
|
|
print(pColFmt, utstring_body(pLog), '\0');
|
|
}
|
|
}
|
|
}
|
|
|
|
static void* __logOutputThread(void *p)
|
|
{
|
|
while(TRUE)
|
|
{
|
|
int isWriteLog = FALSE;
|
|
PLOG_ITEM pItem = NULL, pTmp = NULL;
|
|
|
|
uv_rwlock_wrlock(&g_uvLogLock);
|
|
|
|
LL_FOREACH_SAFE(g_pLogItemList, pItem, pTmp)
|
|
{
|
|
UT_string *pLogStr;
|
|
struct tm lTime;
|
|
int logFileSize = 0;
|
|
|
|
if(++g_LogRdPos % 100 == 0)
|
|
{
|
|
GET_FILE_SIZE(g_LogProcessInfo.logFilePath, logFileSize);
|
|
}
|
|
|
|
localtime_r(&(pItem->timestamp.tv_sec), &lTime);
|
|
|
|
utstring_new(pLogStr);
|
|
|
|
if(pItem->isAddTags)
|
|
{
|
|
utstring_printf(pLogStr, "[%04d-%02d-%02d %02d:%02d:%02d.%03ld] [%s] %s",
|
|
lTime.tm_year + 1900, lTime.tm_mon + 1, lTime.tm_mday,
|
|
lTime.tm_hour, lTime.tm_min, lTime.tm_sec, pItem->timestamp.tv_usec / 1000,
|
|
LogLevelToStr(pItem->level), pItem->pLogContent);
|
|
}
|
|
else
|
|
{
|
|
utstring_printf(pLogStr, "%s", pItem->pLogContent);
|
|
}
|
|
|
|
if(pItem->isPrinted == FALSE)
|
|
{
|
|
if(pItem->level & LOG_Error
|
|
|| pItem->level & LOG_Fatal)
|
|
{
|
|
__logColorOutput(SHOW_CONSOLE_RED, pLogStr);
|
|
}
|
|
else if(pItem->level & LOG_Warn
|
|
|| pItem->level & LOG_Unknown)
|
|
{
|
|
__logColorOutput(SHOW_CONSOLE_YELLOW, pLogStr);
|
|
}
|
|
else if(pItem->level & LOG_Test
|
|
|| pItem->level & LOG_Call)
|
|
{
|
|
__logColorOutput(SHOW_CONSOLE_BLUE, pLogStr);
|
|
}
|
|
else if(pItem->level & LOG_Devp)
|
|
{
|
|
__logColorOutput(SHOW_CONSOLE_GREEN, pLogStr);
|
|
}
|
|
else
|
|
{
|
|
print("%s", utstring_body(pLogStr));
|
|
}
|
|
|
|
pItem->isPrinted = TRUE;
|
|
}
|
|
|
|
if(g_logSock != -1 && GetCurrWIFIConnStatus() == WIFI_CONNECTED)
|
|
{
|
|
__logNetworkSend(pItem);
|
|
}
|
|
|
|
if(g_LogProcessInfo.pLogFile != NULL && g_bEnLogToFile)
|
|
{
|
|
if(logFileSize >= MAX_LOG_FILE_SIZE)
|
|
{
|
|
fflush(g_LogProcessInfo.pLogFile);
|
|
fclose(g_LogProcessInfo.pLogFile);
|
|
#if 0
|
|
if(g_bEnMailBackup)
|
|
{
|
|
PLOG_BACKUP pBackup = (PLOG_BACKUP)malloc(sizeof(struct LOG_BACKUP));
|
|
char path[MAX_PATH];
|
|
|
|
memset(path, 0, MAX_PATH);
|
|
sprintf(path, "%s/%s.bak_%u.log",
|
|
LOG_FILE_BASEDIR,
|
|
basename_v2(g_LogProcessInfo.logFilePath),
|
|
g_LogRdPos);
|
|
rename(g_LogProcessInfo.logFilePath, path);
|
|
|
|
if(pBackup)
|
|
{
|
|
memset(pBackup, 0, sizeof(struct LOG_BACKUP));
|
|
pBackup->timestamp = time(NULL);
|
|
pBackup->plogFilePath = strdup(path);
|
|
pBackup->sendTimes = 0;
|
|
pBackup->tolLogs = g_LogRdPos - 1;
|
|
|
|
uv_rwlock_wrlock(&g_uvLogBackLock);
|
|
LL_APPEND(g_pLogBackupList, pBackup);
|
|
uv_rwlock_wrunlock(&g_uvLogBackLock);
|
|
}
|
|
}
|
|
#endif
|
|
g_LogProcessInfo.pLogFile = fopen(g_LogProcessInfo.logFilePath, "w+");
|
|
}
|
|
|
|
if(g_LogProcessInfo.pLogFile)
|
|
{
|
|
fwrite(utstring_body(pLogStr), 1, utstring_len(pLogStr), g_LogProcessInfo.pLogFile);
|
|
}
|
|
|
|
isWriteLog = TRUE;
|
|
}
|
|
|
|
LL_DELETE(g_pLogItemList, pItem);
|
|
|
|
utstring_free(pLogStr);
|
|
free(pItem->pLogContent);
|
|
free(pItem);
|
|
|
|
if(g_LogRdPos % 100 == 0)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
uv_rwlock_wrunlock(&g_uvLogLock);
|
|
|
|
usleep(1000);
|
|
if(g_LogProcessInfo.pLogFile != NULL && isWriteLog)
|
|
{
|
|
fflush(g_LogProcessInfo.pLogFile);
|
|
}
|
|
}
|
|
|
|
pthread_detach(pthread_self());
|
|
return (NULL);
|
|
}
|
|
|
|
/**
|
|
* @brief 设置调试等级
|
|
* @param level 调试等级
|
|
* @param iEnable 1 打开调试等级, 0 关闭调试等级
|
|
*/
|
|
void IHW_EnableLogLevel(LOG_LEVEL level, int iEnable)
|
|
{
|
|
if(iEnable > 0)
|
|
{
|
|
g_iMinLevel |= level;
|
|
}
|
|
else
|
|
{
|
|
g_iMinLevel &= ~(level);
|
|
}
|
|
}
|
|
|
|
static int __getCfgFromCfgFile(void)
|
|
{
|
|
int ret = 0;
|
|
g_bEnableLog = CfgGetBoolValue("Global.Log.Enable", g_bEnableLog);
|
|
g_iMinLevel = CfgGetIntValue("Global.Log.Level", g_iMinLevel);
|
|
g_bEnMailBackup = CfgGetBoolValue("Global.Log.LogToEMail.Enable", g_bEnMailBackup);
|
|
g_pEmailBox = CfgGetStringValue("Global.Log.LogToEMail.EMail", LOG_MAIL_RECEIVER);
|
|
g_bEnLogToFile = CfgGetBoolValue("Global.Log.LogToFile", g_bEnLogToFile);
|
|
g_bEnLogToServer = CfgGetBoolValue("Global.Log.LogToServer", g_bEnLogToServer);
|
|
|
|
#if 0
|
|
LOG_EX(LOG_Debug, "g_bEnableLog = %d\n", g_bEnableLog);
|
|
LOG_EX(LOG_Debug, "g_iMinLevel = 0x%X\n", g_iMinLevel);
|
|
LOG_EX(LOG_Debug, "g_bEnMailBackup = %d\n", g_bEnMailBackup);
|
|
LOG_EX(LOG_Debug, "g_pEmailBox = %s\n", g_pEmailBox);
|
|
LOG_EX(LOG_Debug, "g_bEnLogToFile = %d\n", g_bEnLogToFile);
|
|
LOG_EX(LOG_Debug, "g_bEnLogToServer = %d\n", g_bEnLogToServer);
|
|
#endif
|
|
|
|
ret = CfgGetBoolValue("Global.Log.LogToUDPServer.Enable", FALSE);
|
|
|
|
if(ret)
|
|
{
|
|
char *pIpAddr = NULL;
|
|
int udpPortBase = CfgGetBoolValue("Global.Log.LogToUDPServer.UdpBasePort", 10000);
|
|
int port = udpPortBase + DBusLibGetModName();
|
|
|
|
if(g_logSock != -1)
|
|
{
|
|
close(g_logSock);
|
|
g_logSock = -1;
|
|
}
|
|
|
|
pIpAddr = CfgGetStringValue("Global.Log.LogToUDPServer.UdpServerIp", NULL);
|
|
|
|
LOG_EX(LOG_Debug, "UDP Ipaddr = %s, port = %d\n", pIpAddr, port);
|
|
|
|
if(pIpAddr)
|
|
{
|
|
memset(&g_logAddr, 0, sizeof(g_logAddr));
|
|
g_logAddr.sin_family = AF_INET;
|
|
g_logAddr.sin_port = htons(port);
|
|
g_logAddr.sin_addr.s_addr = inet_addr(pIpAddr);
|
|
|
|
g_logSock = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
|
|
}
|
|
}
|
|
}
|
|
/**
|
|
* @brief 初始化系统日志功能
|
|
* @param pLogTag 系统日志标志
|
|
* @param pPath 系统日志保存路径
|
|
* @param bEnable 打开/关闭调试信息
|
|
*/
|
|
void IHW_InitLOG(const char* pLogTag, const char* pPath, int bEnable)
|
|
{
|
|
int ret;
|
|
char strPath[MAX_PATH];
|
|
|
|
memset(&g_logAddr, 0, sizeof(g_logAddr));
|
|
uv_rwlock_init(&g_uvLogLock);
|
|
g_LogRdPos = 0;
|
|
memset(g_strLogTag, 0, 32);
|
|
|
|
memset(&g_LogProcessInfo, 0, sizeof(LOG_PROCESS_INFO));
|
|
|
|
if(pLogTag == NULL)
|
|
{
|
|
strcpy(g_strLogTag, "");
|
|
}
|
|
else
|
|
{
|
|
strncpy(g_strLogTag, pLogTag, 31);
|
|
}
|
|
|
|
memset(strPath, 0, MAX_PATH);
|
|
|
|
g_LogProcessInfo.pid = getpid();
|
|
if(readlink("/proc/self/exe", strPath, MAX_PATH) == -1)
|
|
{
|
|
strcpy(g_LogProcessInfo.exeName, pLogTag);
|
|
}
|
|
else
|
|
{
|
|
char *pExeName = strrchr(strPath, '/');
|
|
|
|
if(pExeName == NULL)
|
|
{
|
|
strncpy(g_LogProcessInfo.exeName, strPath, MAX_PATH - 1);
|
|
}
|
|
else
|
|
{
|
|
strncpy(g_LogProcessInfo.exeName, pExeName + 1, MAX_PATH - 1);
|
|
}
|
|
}
|
|
|
|
//fprintf(stdout, "pid = %d, name = %s\n", g_LogProcessInfo.pid, g_LogProcessInfo.exeName);
|
|
g_LogProcessInfo.pChipId = GetCpuChipId();
|
|
g_LogProcessInfo.pChipSerial = GetCpuSerial();
|
|
memset(g_LogProcessInfo.logFilePath, 0, MAX_PATH);
|
|
sprintf(g_LogProcessInfo.logFilePath, "%s/%s_%d.log", LOG_FILE_BASEDIR, g_LogProcessInfo.exeName, g_LogProcessInfo.pid);
|
|
|
|
memset(strPath, 0, MAX_PATH);
|
|
sprintf(strPath, "rm -f %s/%s_*.log > /dev/zero", LOG_FILE_BASEDIR, g_LogProcessInfo.exeName);
|
|
ret = system(strPath);
|
|
|
|
g_LogProcessInfo.pLogFile = fopen(g_LogProcessInfo.logFilePath, "w+");
|
|
|
|
g_bEnableLog = bEnable;
|
|
|
|
__getCfgFromCfgFile();
|
|
|
|
#if 0
|
|
LOG_CFG_PROTOCOL logCfg;
|
|
|
|
memset(&logCfg, 0, sizeof(LOG_CFG_PROTOCOL));
|
|
logCfg.cfgCmd = CMD_LOG_NETWORK;
|
|
logCfg.iParams1 = inet_addr("10.240.84.163");
|
|
logCfg.iParams2 = htons(10000);
|
|
UpgradLogConfigure(&logCfg);
|
|
#endif
|
|
|
|
__initPostLogParams();
|
|
}
|
|
|
|
void IHW_RunLogService(void)
|
|
{
|
|
pthread_create(&g_logThreadId, NULL, __logOutputThread, NULL);
|
|
pthread_create(&g_backupThreadId, NULL, __uvLogBackupProc, NULL);
|
|
}
|
|
|
|
static void __logTo(LOG_LEVEL level, int isAddTag, char* pMsg, int isPrint)
|
|
{
|
|
PLOG_ITEM pLogItem;
|
|
|
|
pLogItem = (PLOG_ITEM)malloc(sizeof(struct LOG_ITEM));
|
|
|
|
if(pLogItem == NULL)
|
|
{
|
|
return;
|
|
}
|
|
|
|
pLogItem->pLogContent = strdup(pMsg);
|
|
pLogItem->isPrinted = isPrint ? FALSE : TRUE;
|
|
pLogItem->level = level;
|
|
pLogItem->isAddTags = isAddTag ? TRUE : FALSE;
|
|
gettimeofday(&(pLogItem->timestamp), NULL);
|
|
|
|
uv_rwlock_wrlock(&g_uvLogLock);
|
|
LL_APPEND(g_pLogItemList, pLogItem);
|
|
uv_rwlock_wrunlock(&g_uvLogLock);
|
|
}
|
|
|
|
void IHW_LogStrWithoutPrint(int level, char* pMsg)
|
|
{
|
|
__logTo(level, TRUE, pMsg, FALSE);
|
|
}
|
|
|
|
void IHW_LogRawString(int level, char* pMsg)
|
|
{
|
|
__logTo(level, TRUE, pMsg, TRUE);
|
|
}
|
|
|
|
/**
|
|
* @brief 输出调试信息
|
|
* @param cFlag 调试信息开关
|
|
* @param pMsg 调试信息内容
|
|
*/
|
|
void IHW_LOG_UNTAG(LOG_LEVEL level, const char* pMsg, ...)
|
|
{
|
|
va_list arg_ptr;
|
|
UT_string *pLogContent;
|
|
|
|
if(!g_bEnableLog)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// 检查调试等级
|
|
if(!(g_iMinLevel & level))
|
|
{
|
|
return;
|
|
}
|
|
|
|
utstring_new(pLogContent);
|
|
va_start(arg_ptr, pMsg);
|
|
utstring_printf_va(pLogContent, pMsg, arg_ptr);
|
|
va_end(arg_ptr);
|
|
|
|
__logTo(level, FALSE, utstring_body(pLogContent), TRUE);
|
|
|
|
utstring_free(pLogContent);
|
|
}
|
|
|
|
/**
|
|
* @brief 输出调试信息
|
|
* @param cFlag 调试信息开关
|
|
* @param pMsg 调试信息内容
|
|
*/
|
|
void IHW_LOG(LOG_LEVEL level, const char* pMsg, ...)
|
|
{
|
|
UT_string* pLogContent = NULL;
|
|
va_list arg_ptr;
|
|
|
|
if(!g_bEnableLog)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// 检查调试等级
|
|
if(!(g_iMinLevel & level))
|
|
{
|
|
return;
|
|
}
|
|
|
|
utstring_new(pLogContent);
|
|
va_start(arg_ptr, pMsg);
|
|
utstring_printf_va(pLogContent, pMsg, arg_ptr);
|
|
va_end(arg_ptr);
|
|
|
|
__logTo(level, TRUE, utstring_body(pLogContent), TRUE);
|
|
|
|
utstring_free(pLogContent);
|
|
}
|
|
|
|
void IHW_DisableLogOut(void)
|
|
{
|
|
g_bEnableLog = FALSE;
|
|
}
|
|
|
|
void IHW_EnableLogOut(void)
|
|
{
|
|
g_bEnableLog = TRUE;
|
|
}
|
|
|
|
#if 0
|
|
void LogUploadCurLogFile(void)
|
|
{
|
|
UT_string *pFn = NULL, *pCmd = NULL;
|
|
utstring_new(pFn);
|
|
utstring_new(pCmd);
|
|
|
|
utstring_printf(pFn, "/tmp/%s_%s.bak.log", g_LogProcessInfo.exeName, g_LogProcessInfo.pChipId);
|
|
utstring_printf(pCmd, "cp %s %s", g_LogProcessInfo.logFilePath, utstring_body(pFn));
|
|
|
|
__logHttpPostFile(utstring_body(pFn));
|
|
//InetHttpUploadFileSync(UPL_HTTP_URL, utstring_body(pFn), NULL);
|
|
|
|
utstring_free(pFn);
|
|
utstring_free(pCmd);
|
|
}
|
|
#endif
|
|
|
|
#if 0
|
|
void UploadLogFile(char* pFilePath)
|
|
{
|
|
int fileSize = 0;
|
|
|
|
if(pFilePath == NULL || strlen(pFilePath) == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
GET_FILE_SIZE(pFilePath, fileSize);
|
|
|
|
if(fileSize <= 0)
|
|
{
|
|
LOG_EX(LOG_Warn, "File Size Error: %d\n", fileSize);
|
|
return;
|
|
}
|
|
|
|
__logHttpPostFile(pFilePath);
|
|
}
|
|
#endif
|
|
|
|
void UpgradLogConfigure(PLOG_CFG_PROTOCOL pCfg)
|
|
{
|
|
switch(pCfg->cfgCmd)
|
|
{
|
|
case CMD_LOG_ENABLE:
|
|
g_bEnableLog = pCfg->iParams1;
|
|
break;
|
|
|
|
case CMD_LOG_FILE:
|
|
if(pCfg->iParams1 == FALSE)
|
|
{
|
|
g_bEnMailBackup = g_bEnLogToFile = FALSE;
|
|
}
|
|
else
|
|
{
|
|
g_bEnLogToFile = TRUE;
|
|
}
|
|
break;
|
|
|
|
case CMD_LOG_MAIL:
|
|
g_bEnMailBackup = pCfg->iParams1;
|
|
break;
|
|
|
|
case CMD_LOG_LEVEL:
|
|
if(pCfg->iParams2 == FALSE)
|
|
{
|
|
if(pCfg->iParams1 == 0)
|
|
{
|
|
g_iMinLevel = LOG_Fatal | LOG_Error | LOG_Devp;
|
|
}
|
|
else
|
|
{
|
|
g_iMinLevel &= ~(pCfg->iParams1);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
g_iMinLevel |= pCfg->iParams1;
|
|
}
|
|
break;
|
|
|
|
case CMD_LOG_NETWORK:
|
|
if(pCfg->iParams1 == 0)
|
|
{
|
|
memset(&g_logAddr, 0, sizeof(g_logAddr));
|
|
close(g_logSock);
|
|
g_logSock = -1;
|
|
}
|
|
else
|
|
{
|
|
if(g_logSock != -1)
|
|
{
|
|
close(g_logSock);
|
|
}
|
|
|
|
memset(&g_logAddr, 0, sizeof(g_logAddr));
|
|
g_logAddr.sin_family = AF_INET;
|
|
g_logAddr.sin_port = htons((pCfg->iParams2 & 0xFFFF));
|
|
g_logAddr.sin_addr.s_addr = (pCfg->iParams1);
|
|
|
|
g_logSock = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
|
|
}
|
|
break;
|
|
|
|
case CMD_LOG_SERVER:
|
|
g_bEnLogToServer = pCfg->iParams1;
|
|
break;
|
|
|
|
default: break;
|
|
}
|
|
}
|
|
|
|
const char* LogLeveToString(LOG_LEVEL lv)
|
|
{
|
|
switch(lv)
|
|
{
|
|
case LOG_Fatal: return "LOG_Fatal";
|
|
case LOG_Error: return "LOG_Error";
|
|
case LOG_Warn: return "LOG_Warn";
|
|
case LOG_Debug: return "LOG_Debug";
|
|
case LOG_Info: return "LOG_Info";
|
|
case LOG_Test: return "LOG_Test";
|
|
case LOG_Call: return "LOG_Call";
|
|
case LOG_Devp: return "LOG_Devp";
|
|
case LOG_Step: return "LOG_Step";
|
|
case LOG_Unknown: return "LOG_Unknown";
|
|
case LOG_All: return "LOG_All";
|
|
case LOG_Close: return "LOG_Close";
|
|
}
|
|
|
|
return "Unknown";
|
|
}
|
|
|