300 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			300 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * Copyright (c) 2013-2019 Huawei Technologies Co., Ltd. All rights reserved.
 | 
						|
 * Copyright (c) 2020-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.
 | 
						|
 */
 | 
						|
 | 
						|
#include "los_exc_info.h"
 | 
						|
#include "securec.h"
 | 
						|
#include "los_debug.h"
 | 
						|
#include "los_interrupt.h"
 | 
						|
#include "los_task.h"
 | 
						|
#include "los_exchook.h"
 | 
						|
#include "los_membox.h"
 | 
						|
#include "los_memory.h"
 | 
						|
 | 
						|
 | 
						|
#if (LOSCFG_PLATFORM_EXC == 1)
 | 
						|
#define EXC_MSG_ARRAY_INIT_VALUE   0xff
 | 
						|
#define EXC_INT_STATUS_LEN        (OS_NVIC_INT_ENABLE_SIZE + OS_NVIC_INT_PEND_SIZE + \
 | 
						|
                                   OS_NVIC_INT_ACT_SIZE + OS_NVIC_INT_PRI_SIZE + \
 | 
						|
                                   OS_NVIC_EXCPRI_SIZE + OS_NVIC_SHCSR_SIZE + \
 | 
						|
                                   OS_NVIC_INT_CTRL_SIZE)
 | 
						|
 | 
						|
UINT32 g_excQueueMaxNum;
 | 
						|
UINT32 g_excMemMaxNum;
 | 
						|
VOID *g_excContent = NULL;
 | 
						|
UINT32 g_excArraySize = MAX_EXC_MEM_SIZE;
 | 
						|
UINT8 g_excMsgArray[MAX_EXC_MEM_SIZE];
 | 
						|
STATIC ExcInfoArray g_excArray[OS_EXC_TYPE_MAX];
 | 
						|
 | 
						|
STATIC UINT32 OsExcSaveIntStatus(UINT32 type, VOID *arg)
 | 
						|
{
 | 
						|
    errno_t ret;
 | 
						|
    UINTPTR excContentEnd = (UINTPTR)MAX_INT_INFO_SIZE + (UINTPTR)g_excContent;
 | 
						|
 | 
						|
    (VOID)arg;
 | 
						|
 | 
						|
    *((UINT32 *)g_excContent) = type;
 | 
						|
    g_excContent = (UINT8 *)g_excContent + sizeof(UINT32);
 | 
						|
 | 
						|
    *((UINT32 *)g_excContent) = EXC_INT_STATUS_LEN;
 | 
						|
    g_excContent = (UINT8 *)g_excContent + sizeof(UINT32);
 | 
						|
    /* save IRQ ENABLE reg group */
 | 
						|
    ret = memcpy_s(g_excContent, excContentEnd - (UINTPTR)g_excContent,
 | 
						|
                   (const VOID *)OS_NVIC_SETENA_BASE, OS_NVIC_INT_ENABLE_SIZE);
 | 
						|
    if (ret != EOK) {
 | 
						|
        return LOS_NOK;
 | 
						|
    }
 | 
						|
    g_excContent = (UINT8 *)g_excContent + OS_NVIC_INT_ENABLE_SIZE;
 | 
						|
 | 
						|
    /* save IRQ PEND reg group */
 | 
						|
    ret = memcpy_s(g_excContent, excContentEnd - (UINTPTR)g_excContent,
 | 
						|
                   (const VOID *)OS_NVIC_SETPEND_BASE, OS_NVIC_INT_PEND_SIZE);
 | 
						|
    if (ret != EOK) {
 | 
						|
        return LOS_NOK;
 | 
						|
    }
 | 
						|
    g_excContent = (UINT8 *)g_excContent + OS_NVIC_INT_PEND_SIZE;
 | 
						|
 | 
						|
    /* save IRQ ACTIVE reg group */
 | 
						|
    ret = memcpy_s(g_excContent, excContentEnd - (UINTPTR)g_excContent,
 | 
						|
                   (const VOID *)OS_NVIC_INT_ACT_BASE, OS_NVIC_INT_ACT_SIZE);
 | 
						|
    if (ret != EOK) {
 | 
						|
        return LOS_NOK;
 | 
						|
    }
 | 
						|
    g_excContent = (UINT8 *)g_excContent + OS_NVIC_INT_ACT_SIZE;
 | 
						|
 | 
						|
    /* save IRQ Priority reg group */
 | 
						|
    ret = memcpy_s(g_excContent, excContentEnd - (UINTPTR)g_excContent,
 | 
						|
                   (const VOID *)OS_NVIC_PRI_BASE, OS_NVIC_INT_PRI_SIZE);
 | 
						|
    if (ret != EOK) {
 | 
						|
        return LOS_NOK;
 | 
						|
    }
 | 
						|
    g_excContent = (UINT8 *)g_excContent + OS_NVIC_INT_PRI_SIZE;
 | 
						|
 | 
						|
    /* save Exception Priority reg group */
 | 
						|
    ret = memcpy_s(g_excContent, excContentEnd - (UINTPTR)g_excContent,
 | 
						|
                   (const VOID *)OS_NVIC_EXCPRI_BASE, OS_NVIC_EXCPRI_SIZE);
 | 
						|
    if (ret != EOK) {
 | 
						|
        return LOS_NOK;
 | 
						|
    }
 | 
						|
    g_excContent = (UINT8 *)g_excContent + OS_NVIC_EXCPRI_SIZE;
 | 
						|
 | 
						|
    /* save IRQ Handler & SHCSR */
 | 
						|
    ret = memcpy_s(g_excContent, excContentEnd - (UINTPTR)g_excContent,
 | 
						|
                   (const VOID *)OS_NVIC_SHCSR, OS_NVIC_SHCSR_SIZE);
 | 
						|
    if (ret != EOK) {
 | 
						|
        return LOS_NOK;
 | 
						|
    }
 | 
						|
    g_excContent = (UINT8 *)g_excContent + OS_NVIC_SHCSR_SIZE;
 | 
						|
 | 
						|
    /* save IRQ Control & ICSR */
 | 
						|
    ret = memcpy_s(g_excContent, excContentEnd - (UINTPTR)g_excContent,
 | 
						|
                   (const VOID *)OS_NVIC_INT_CTRL, OS_NVIC_INT_CTRL_SIZE);
 | 
						|
    if (ret != EOK) {
 | 
						|
        return LOS_NOK;
 | 
						|
    }
 | 
						|
    g_excContent = (UINT8 *)g_excContent + OS_NVIC_INT_CTRL_SIZE;
 | 
						|
 | 
						|
    return LOS_OK;
 | 
						|
}
 | 
						|
 | 
						|
STATIC UINT32 OsExcMemMsgGet(UINT32 type, VOID *arg)
 | 
						|
{
 | 
						|
    UINT32 count;
 | 
						|
    UINT32 memNum = *(UINT32 *)arg;
 | 
						|
    UINTPTR memMsgEnd = (UINTPTR)g_excContent + MAX_MEM_INFO_SIZE;
 | 
						|
 | 
						|
    *((UINT32 *)g_excContent) = type;
 | 
						|
    g_excContent = (UINT8 *)g_excContent + sizeof(UINT32);
 | 
						|
 | 
						|
    *((UINT32 *)g_excContent) = sizeof(MemInfoCB) * memNum;
 | 
						|
    g_excContent = (UINT8 *)g_excContent + sizeof(UINT32);
 | 
						|
 | 
						|
    count = OsMemboxExcInfoGet(memNum, g_excContent);
 | 
						|
    if (count < memNum) {
 | 
						|
        g_excContent = (UINT8 *)g_excContent + sizeof(MemInfoCB) * count;
 | 
						|
        (VOID)OsMemExcInfoGet(memNum - count, g_excContent);
 | 
						|
    }
 | 
						|
    g_excContent = (VOID *)memMsgEnd;
 | 
						|
 | 
						|
    return LOS_OK;
 | 
						|
}
 | 
						|
 | 
						|
STATIC UINT32 OsExcContentGet(UINT32 type, VOID *arg)
 | 
						|
{
 | 
						|
    UINTPTR excContentEnd = MAX_EXC_MEM_SIZE + (UINTPTR)g_excMsgArray;
 | 
						|
    errno_t ret;
 | 
						|
 | 
						|
    (VOID)arg;
 | 
						|
 | 
						|
    /* save exception info */
 | 
						|
    *((UINT32 *)g_excContent) = type;
 | 
						|
    g_excContent = (UINT8 *)g_excContent + sizeof(UINT32);
 | 
						|
    *((UINT32 *)g_excContent) = sizeof(ExcInfo) + sizeof(EXC_CONTEXT_S);
 | 
						|
 | 
						|
    g_excContent = (UINT8 *)g_excContent + sizeof(UINT32);
 | 
						|
    ret = memcpy_s((VOID *)g_excContent, excContentEnd - (UINTPTR)g_excContent,
 | 
						|
                   (VOID *)&g_excInfo, sizeof(ExcInfo));
 | 
						|
    if (ret != EOK) {
 | 
						|
        return LOS_NOK;
 | 
						|
    }
 | 
						|
    g_excContent = (UINT8 *)g_excContent + sizeof(ExcInfo);
 | 
						|
 | 
						|
    ret = memcpy_s((VOID *)g_excContent, excContentEnd - (UINTPTR)g_excContent,
 | 
						|
                   g_excInfo.context, sizeof(EXC_CONTEXT_S));
 | 
						|
    if (ret != EOK) {
 | 
						|
        return LOS_NOK;
 | 
						|
    }
 | 
						|
    g_excContent = (UINT8 *)g_excContent + sizeof(EXC_CONTEXT_S);
 | 
						|
 | 
						|
    return LOS_OK;
 | 
						|
}
 | 
						|
 | 
						|
STATIC UINT32 OsExcHookFunc(UINT32 type, UINT32 index, UINT32 loop, UINT32 len, EXC_INFO_SAVE_CALLBACK func)
 | 
						|
{
 | 
						|
    UINT32 i;
 | 
						|
 | 
						|
    *((UINT32 *)g_excContent) = (UINT32)type;
 | 
						|
    g_excContent = (UINT8 *)g_excContent + sizeof(UINT32);
 | 
						|
 | 
						|
    *((UINT32 *)g_excContent) = len * loop;
 | 
						|
    g_excContent = (UINT8 *)g_excContent + sizeof(UINT32);
 | 
						|
 | 
						|
    for (i = index; i < loop; i++) {
 | 
						|
        (VOID)func(i, g_excContent);
 | 
						|
        g_excContent = (UINT8 *)g_excContent + len;
 | 
						|
    }
 | 
						|
 | 
						|
    return LOS_OK;
 | 
						|
}
 | 
						|
 | 
						|
#if (LOSCFG_BASE_IPC_QUEUE == 1)
 | 
						|
STATIC UINT32 OsExcQueueMsgGet(UINT32 type, VOID *arg)
 | 
						|
{
 | 
						|
    UINT32 len = sizeof(QUEUE_INFO_S);
 | 
						|
    UINT32 loop = *(UINT32 *)arg;
 | 
						|
 | 
						|
    return OsExcHookFunc(type, 1, loop, len, (EXC_INFO_SAVE_CALLBACK)LOS_QueueInfoGet);
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
STATIC UINT32 OsExcTaskMsgGet(UINT32 type, VOID *arg)
 | 
						|
{
 | 
						|
    UINT32 len = sizeof(TSK_INFO_S);
 | 
						|
    UINT32 loop = *(UINT32 *)arg;
 | 
						|
 | 
						|
    return OsExcHookFunc(type, 0, loop, len, (EXC_INFO_SAVE_CALLBACK)LOS_TaskInfoGet);
 | 
						|
}
 | 
						|
 | 
						|
#if (LOSCFG_BASE_CORE_EXC_TSK_SWITCH == 1)
 | 
						|
STATIC UINT32 OsExcTskSwitchMsgGet(UINT32 type, VOID *arg)
 | 
						|
{
 | 
						|
    UINT32 len;
 | 
						|
    UINT32 loop;
 | 
						|
    UINT32 taskSwitchCount;
 | 
						|
    TaskSwitchInfo *taskSwitchInfo = NULL;
 | 
						|
    UINT32 i;
 | 
						|
 | 
						|
    /* not necessary, just for macro int library */
 | 
						|
    taskSwitchInfo = (TaskSwitchInfo *)arg;
 | 
						|
    taskSwitchCount = taskSwitchInfo->cntInfo.maxCnt;
 | 
						|
    len = sizeof(UINT32) + (sizeof(CHAR) * LOS_TASK_NAMELEN);
 | 
						|
    if (taskSwitchInfo->cntInfo.isFull) {
 | 
						|
        i = taskSwitchInfo->idx;
 | 
						|
        loop = i + taskSwitchCount;
 | 
						|
    } else {
 | 
						|
        i = 0;
 | 
						|
        loop = taskSwitchInfo->idx;
 | 
						|
    }
 | 
						|
 | 
						|
    return OsExcHookFunc(type, i, loop, len, (EXC_INFO_SAVE_CALLBACK)LOS_TaskSwitchInfoGet);
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
VOID OsExcRegister(ExcInfoType type, EXC_INFO_SAVE_CALLBACK func, VOID *arg)
 | 
						|
{
 | 
						|
    ExcInfoArray *excInfo = NULL;
 | 
						|
    if ((type >= OS_EXC_TYPE_MAX) || (func == NULL)) {
 | 
						|
        PRINT_ERR("OsExcRegister ERROR!\n");
 | 
						|
        return;
 | 
						|
    }
 | 
						|
    excInfo = &(g_excArray[type]);
 | 
						|
    if (excInfo->valid == TRUE) {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    excInfo->type = type;
 | 
						|
    excInfo->fnExcInfoCb = func;
 | 
						|
    excInfo->arg = arg;
 | 
						|
    excInfo->valid = TRUE;
 | 
						|
}
 | 
						|
 | 
						|
STATIC VOID OsExcMsgDump(VOID)
 | 
						|
{
 | 
						|
    UINT32 index;
 | 
						|
 | 
						|
    /* Ignore the return code when matching CSEC rule 6.6(4). */
 | 
						|
    (VOID)memset_s(g_excMsgArray, g_excArraySize, EXC_MSG_ARRAY_INIT_VALUE, g_excArraySize);
 | 
						|
 | 
						|
    *((UINT32 *)g_excContent) = MAX_EXC_MEM_SIZE;  /* The total length of exception information. */
 | 
						|
    g_excContent = (UINT8 *)g_excContent + sizeof(UINT32);
 | 
						|
 | 
						|
    for (index = 0; index < OS_EXC_TYPE_MAX; index++) {
 | 
						|
        if (!g_excArray[index].valid) {
 | 
						|
            continue;
 | 
						|
        }
 | 
						|
        g_excArray[index].fnExcInfoCb(g_excArray[index].type, g_excArray[index].arg);
 | 
						|
    }
 | 
						|
 | 
						|
    *((UINT32 *)g_excContent) = OS_EXC_TYPE_MAX;
 | 
						|
    g_excContent = (UINT8 *)g_excContent + sizeof(UINT32);
 | 
						|
    return;
 | 
						|
}
 | 
						|
 | 
						|
VOID OsExcMsgDumpInit(VOID)
 | 
						|
{
 | 
						|
    g_excQueueMaxNum = LOSCFG_BASE_IPC_QUEUE_LIMIT;
 | 
						|
    g_excMemMaxNum = OS_SYS_MEM_NUM;
 | 
						|
    g_excContent = (VOID *)g_excMsgArray;
 | 
						|
 | 
						|
    OsExcRegister(OS_EXC_TYPE_CONTEXT, OsExcContentGet, NULL);
 | 
						|
    OsExcRegister(OS_EXC_TYPE_TSK, OsExcTaskMsgGet, &g_taskMaxNum);
 | 
						|
#if (LOSCFG_BASE_IPC_QUEUE == 1)
 | 
						|
    OsExcRegister(OS_EXC_TYPE_QUE, OsExcQueueMsgGet, &g_excQueueMaxNum);
 | 
						|
#endif
 | 
						|
    OsExcRegister(OS_EXC_TYPE_NVIC, OsExcSaveIntStatus, NULL);
 | 
						|
#if (LOSCFG_BASE_CORE_EXC_TSK_SWITCH == 1)
 | 
						|
    OsExcRegister(OS_EXC_TYPE_TSK_SWITCH, OsExcTskSwitchMsgGet, &g_taskSwitchInfo);
 | 
						|
#endif
 | 
						|
    OsExcRegister(OS_EXC_TYPE_MEM, OsExcMemMsgGet, &g_excMemMaxNum);
 | 
						|
 | 
						|
    (VOID)LOS_RegExcHook(EXC_INTERRUPT, (ExcHookFn)OsExcMsgDump);
 | 
						|
}
 | 
						|
#endif
 |