/* * Copyright (c) 2021-2021 Huawei Device Co., Ltd. All rights reserved. * * Redistribution and use in source and binary forms, with or without modification, * are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this list of * conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, this list * of conditions and the following disclaimer in the documentation and/or other materials * provided with the distribution. * * 3. Neither the name of the copyright holder nor the names of its contributors may be used * to endorse or promote products derived from this software without specific prior written * permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ /* ------------ includes ------------ */ #include "los_blackbox_common.h" #ifdef LOSCFG_LIB_LIBC #include "stdlib.h" #include "unistd.h" #endif #ifdef LOSCFG_FS_VFS #include "fs/fs.h" #include "fs/mount.h" #endif #include "securec.h" #include "los_memory.h" /* ------------ local macroes ------------ */ #ifdef LOSCFG_FS_VFS #define BBOX_DIR_MODE 0777 #endif /* ------------ local prototypes ------------ */ /* ------------ local function declarations ------------ */ /* ------------ global function declarations ------------ */ /* ------------ local variables ------------ */ static bool g_isLogPartReady = FALSE; /* ------------ function definitions ------------ */ int FullWriteFile(const char *filePath, const char *buf, size_t bufSize, int isAppend) { #ifdef LOSCFG_FS_VFS int fd; int totalToWrite = (int)bufSize; int totalWrite = 0; if (filePath == NULL || buf == NULL || bufSize == 0) { BBOX_PRINT_ERR("filePath: %p, buf: %p, bufSize: %lu!\n", filePath, buf, bufSize); return -1; } if (!IsLogPartReady()) { BBOX_PRINT_ERR("log path [%s] isn't ready to be written!\n", LOSCFG_BLACKBOX_LOG_ROOT_PATH); return -1; } fd = open(filePath, O_CREAT | O_RDWR | (isAppend ? O_APPEND : O_TRUNC), 0644); if (fd < 0) { BBOX_PRINT_ERR("Create file [%s] failed, fd: %d!\n", filePath, fd); return -1; } while (totalToWrite > 0) { int writeThisTime = write(fd, buf, totalToWrite); if (writeThisTime < 0) { BBOX_PRINT_ERR("Failed to write file [%s]!\n", filePath); (void)close(fd); return -1; } buf += writeThisTime; totalToWrite -= writeThisTime; totalWrite += writeThisTime; } (void)fsync(fd); (void)close(fd); return (totalWrite == (int)bufSize) ? 0 : -1; #else (VOID)filePath; (VOID)buf; (VOID)bufSize; (VOID)isAppend; return -1; #endif } int SaveBasicErrorInfo(const char *filePath, struct ErrorInfo *info) { char *buf = NULL; if (filePath == NULL || info == NULL) { BBOX_PRINT_ERR("filePath: %p, event: %p!\n", filePath, info); return -1; } buf = LOS_MemAlloc(m_aucSysMem1, ERROR_INFO_MAX_LEN); if (buf == NULL) { BBOX_PRINT_ERR("LOS_MemAlloc failed!\n"); return -1; } (void)memset_s(buf, ERROR_INFO_MAX_LEN, 0, ERROR_INFO_MAX_LEN); if (snprintf_s(buf, ERROR_INFO_MAX_LEN, ERROR_INFO_MAX_LEN - 1, ERROR_INFO_HEADER_FORMAT, info->event, info->module, info->errorDesc) != -1) { *(buf + ERROR_INFO_MAX_LEN - 1) = '\0'; (void)FullWriteFile(filePath, buf, strlen(buf), 0); } else { BBOX_PRINT_ERR("buf is not enough or snprintf_s failed!\n"); } (void)LOS_MemFree(m_aucSysMem1, buf); return 0; } #ifdef LOSCFG_FS_VFS static int IsLogPartMounted(const char *devPoint, const char *mountPoint, struct statfs *statBuf, void *arg) { (void)devPoint; (void)statBuf; (void)arg; if (mountPoint != NULL && arg != NULL) { if (strcmp(mountPoint, (char *)arg) == 0) { g_isLogPartReady = TRUE; } } return 0; } bool IsLogPartReady(void) { if (!g_isLogPartReady) { (void)foreach_mountpoint((foreach_mountpoint_t)IsLogPartMounted, LOSCFG_BLACKBOX_LOG_PART_MOUNT_POINT); } return g_isLogPartReady; } #else bool IsLogPartReady(void) { return TRUE; } #endif #ifdef LOSCFG_FS_VFS int CreateNewDir(const char *dirPath) { int ret; if (dirPath == NULL) { BBOX_PRINT_ERR("dirPath is NULL!\n"); return -1; } ret = access(dirPath, 0); if (ret == 0) { return 0; } ret = mkdir(dirPath, BBOX_DIR_MODE); if (ret != 0) { BBOX_PRINT_ERR("mkdir [%s] failed!\n", dirPath); return -1; } return 0; } int CreateLogDir(const char *dirPath) { const char *temp = dirPath; char curPath[PATH_MAX_LEN]; int idx = 0; if (dirPath == NULL) { BBOX_PRINT_ERR("dirPath is NULL!\n"); return -1; } if (*dirPath != '/') { BBOX_PRINT_ERR("Invalid dirPath: %s\n", dirPath); return -1; } (void)memset_s(curPath, sizeof(curPath), 0, sizeof(curPath)); curPath[idx++] = *dirPath++; while (*dirPath != '\0' && idx < sizeof(curPath)) { if (*dirPath == '/') { if (CreateNewDir(curPath) != 0) { return -1; } } curPath[idx] = *dirPath; dirPath++; idx++; } if (*dirPath != '\0') { BBOX_PRINT_ERR("dirPath [%s] is too long!\n", temp); return -1; } return CreateNewDir(curPath); } #else int CreateLogDir(const char *dirPath) { (void)dirPath; return -1; } #endif