|
|
/* ----------------------------------------------------------------------------
|
|
|
* Copyright (c) Huawei Technologies Co., Ltd. 2013-2019. All rights reserved.
|
|
|
* Description: Mutex Deadlock Check
|
|
|
* Author: Huawei LiteOS Team
|
|
|
* Create: 2013-01-01
|
|
|
* 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_mux_debug_pri.h"
|
|
|
#include "los_typedef.h"
|
|
|
#include "los_task.h"
|
|
|
#include "arch/exception.h"
|
|
|
#ifdef LOSCFG_SHELL
|
|
|
#include "shcmd.h"
|
|
|
#endif /* LOSCFG_SHELL */
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
#if __cplusplus
|
|
|
extern "C" {
|
|
|
#endif
|
|
|
#endif /* __cplusplus */nbb
|
|
|
|
|
|
typedef struct {
|
|
|
LOS_DL_LIST muxListHead; /* Task-held mutexs list */ //<2F><><EFBFBD>ڹ<EFBFBD><DAB9><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>еĻ<D0B5><C4BB><EFBFBD><EFBFBD><EFBFBD><EFBFBD>б<EFBFBD>//
|
|
|
UINT64 lastAccessTime; /* The last operation time */ //<2F><>¼<EFBFBD><C2BC><EFBFBD><EFBFBD>һ<EFBFBD>β<EFBFBD><CEB2><EFBFBD><EFBFBD><EFBFBD>ʱ<EFBFBD><CAB1>//
|
|
|
} MuxDLinkCB;
|
|
|
|
|
|
/*<2A><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>еĻ<D0B5><C4BB><EFBFBD><EFBFBD><EFBFBD><EFBFBD>б<EFBFBD><D0B1><EFBFBD><CDB8><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>*/
|
|
|
typedef struct {
|
|
|
LOS_DL_LIST muxList; /* Insert mutex into the owner task CB */ //<2F><><EFBFBD>ڽ<EFBFBD><DABD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>뵽<EFBFBD><EBB5BD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ŀ<EFBFBD><C4BF>ƿ<EFBFBD><C6BF><EFBFBD>//
|
|
|
VOID *muxCB; /* The Mutex CB pointer */ //ָ<><EFBFBD><F2BBA5B3><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ƿ<EFBFBD><C6BF><EFBFBD>ָ<EFBFBD><D6B8>//
|
|
|
} MuxDLinkNode;
|
|
|
|
|
|
STATIC MuxDLinkCB *g_muxDeadlockCBArray = NULL;
|
|
|
|
|
|
/*
|
|
|
* Mutex deadlock detection time threshold, will print out task information
|
|
|
* that has not been scheduled within this time.
|
|
|
* The unit is tick.
|
|
|
*/
|
|
|
#define OS_MUX_DEADLOCK_CHECK_THRESHOLD 60000 //<2F><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>μ<EFBFBD><CEBC>黥<EFBFBD><E9BBA5><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>֮<EFBFBD><D6AE><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Сʱ<D0A1><CAB1><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>λΪ<CEBB><CEAA><EFBFBD><EFBFBD>//
|
|
|
|
|
|
UINT32 OsMuxDlockCheckInit(VOID) //<2F><><EFBFBD>ڷ<EFBFBD><DAB7><EFBFBD><EFBFBD>ڴ沢<DAB4><E6B2A2>ʼ<EFBFBD><CABC><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ƿ<EFBFBD><C6BF>б<EFBFBD>//
|
|
|
{
|
|
|
UINT32 index;
|
|
|
UINT32 size = (LOSCFG_BASE_CORE_TSK_LIMIT + 1) * sizeof(MuxDLinkCB);
|
|
|
|
|
|
/* system resident memory, don't free */
|
|
|
g_muxDeadlockCBArray = (MuxDLinkCB *)LOS_MemAlloc(m_aucSysMem1, size);
|
|
|
if (g_muxDeadlockCBArray == NULL) {
|
|
|
PRINT_ERR("%s: malloc failed!\n", __FUNCTION__);
|
|
|
return LOS_NOK;
|
|
|
}
|
|
|
|
|
|
for (index = 0; index < LOSCFG_BASE_CORE_TSK_LIMIT + 1; index++) {
|
|
|
g_muxDeadlockCBArray[index].lastAccessTime = 0;
|
|
|
LOS_ListInit(&g_muxDeadlockCBArray[index].muxListHead);
|
|
|
}
|
|
|
return LOS_OK;
|
|
|
}
|
|
|
|
|
|
VOID OsMuxDlockNodeInsert(UINT32 taskId, VOID *muxCB) //<2F><><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ָ<EFBFBD><D6B8><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ļ<EFBFBD><C4BB><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>в<EFBFBD><D0B2><EFBFBD>һ<EFBFBD><D2BB><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ڵ<EFBFBD>//
|
|
|
{
|
|
|
MuxDLinkNode *muxDLNode = NULL;
|
|
|
|
|
|
if ((taskId > LOSCFG_BASE_CORE_TSK_LIMIT) || (muxCB == NULL)) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
muxDLNode = (MuxDLinkNode *)LOS_MemAlloc(m_aucSysMem1, sizeof(MuxDLinkNode));
|
|
|
if (muxDLNode == NULL) {
|
|
|
return;
|
|
|
}
|
|
|
(VOID)memset_s(muxDLNode, sizeof(MuxDLinkNode), 0, sizeof(MuxDLinkNode));
|
|
|
muxDLNode->muxCB = muxCB;
|
|
|
|
|
|
LOS_ListTailInsert(&g_muxDeadlockCBArray[taskId].muxListHead, &muxDLNode->muxList);
|
|
|
}
|
|
|
|
|
|
VOID OsMuxDlockNodeDelete(UINT32 taskId, const VOID *muxCB) //<2F><><EFBFBD>ڴ<EFBFBD>ָ<EFBFBD><D6B8><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ļ<EFBFBD><C4BB><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ɾ<EFBFBD><C9BE>ָ<EFBFBD><D6B8><EFBFBD>Ļ<EFBFBD><C4BB><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ڵ<EFBFBD>//
|
|
|
{
|
|
|
MuxDLinkCB *muxDLCB = NULL;
|
|
|
LOS_DL_LIST *list = NULL;
|
|
|
MuxDLinkNode *muxDLNode = NULL;
|
|
|
|
|
|
if ((taskId > LOSCFG_BASE_CORE_TSK_LIMIT) || (muxCB == NULL)) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
muxDLCB = &g_muxDeadlockCBArray[taskId];
|
|
|
LOS_DL_LIST_FOR_EACH(list, &muxDLCB->muxListHead) {
|
|
|
muxDLNode = LOS_DL_LIST_ENTRY(list, MuxDLinkNode, muxList);
|
|
|
if (muxDLNode->muxCB == muxCB) {
|
|
|
LOS_ListDelete(&muxDLNode->muxList);
|
|
|
(VOID)LOS_MemFree(m_aucSysMem1, muxDLNode);
|
|
|
return;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
VOID OsTaskTimeUpdate(UINT32 taskId, UINT64 tickCount) //<2F><><EFBFBD>ڸ<EFBFBD><DAB8><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ʱ<EFBFBD><CAB1>//
|
|
|
{
|
|
|
if (taskId > LOSCFG_BASE_CORE_TSK_LIMIT) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
g_muxDeadlockCBArray[taskId].lastAccessTime = tickCount;
|
|
|
}
|
|
|
|
|
|
STATIC VOID OsDeadlockBackTrace(const LosTaskCB *taskCB) //<2F><><EFBFBD>ڴ<EFBFBD>ӡ<EFBFBD><D3A1><EFBFBD><EFBFBD><EFBFBD>ĺ<EFBFBD><C4BA><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ջ<EFBFBD><D5BB>Ϣ//
|
|
|
{
|
|
|
TaskContext *context = NULL;
|
|
|
|
|
|
PRINTK("*******backtrace begin*******\n");
|
|
|
context = (TaskContext *)taskCB->stackPointer;
|
|
|
ArchBackTraceWithSp(context);
|
|
|
PRINTK("********backtrace end********\n");
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
STATIC VOID OsMutexPendTaskList(LOS_DL_LIST *list) //<2F><><EFBFBD>ڴ<EFBFBD>ӡ<EFBFBD><D3A1><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ȴ<EFBFBD><C8B4><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>б<EFBFBD><D0B1><EFBFBD>Ϣ//
|
|
|
{
|
|
|
LOS_DL_LIST *listTmp = NULL;
|
|
|
LosTaskCB *pendedTask = NULL;
|
|
|
CHAR *name = NULL;
|
|
|
UINT32 index = 0;
|
|
|
UINT32 id, intSave;
|
|
|
|
|
|
SCHEDULER_LOCK(intSave);
|
|
|
if (LOS_ListEmpty(list) == TRUE) {
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
PRINTK("Pended Task: null\n");
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
LOS_DL_LIST_FOR_EACH(listTmp, list) {
|
|
|
pendedTask = OS_TCB_FROM_PENDLIST(listTmp);
|
|
|
name = pendedTask->taskName;
|
|
|
id = pendedTask->taskId;
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
if (index == 0) {
|
|
|
PRINTK("Pended task: %u. name:%-15s, id:0x%-5x\n", index, name, id);
|
|
|
} else {
|
|
|
PRINTK(" %u. name:%-15s, id:0x%-5x\n", index, name, id);
|
|
|
}
|
|
|
index++;
|
|
|
SCHEDULER_LOCK(intSave);
|
|
|
}
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
}
|
|
|
|
|
|
STATIC VOID OsTaskHoldMutexList(MuxDLinkCB *muxDLCB) //<2F><><EFBFBD>ڴ<EFBFBD>ӡ<EFBFBD><D3A1><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>еĻ<D0B5><C4BB><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ϣ//
|
|
|
{
|
|
|
UINT32 index = 0;
|
|
|
MuxDLinkNode *muxDLNode = NULL;
|
|
|
CHAR *ownerName = NULL;
|
|
|
LosMuxCB *muxCB = NULL;
|
|
|
LOS_DL_LIST *list = NULL;
|
|
|
LOS_DL_LIST *listTmp = NULL;
|
|
|
UINT32 count, intSave;
|
|
|
|
|
|
SCHEDULER_LOCK(intSave);
|
|
|
if (LOS_ListEmpty(&muxDLCB->muxListHead) == TRUE) {
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
PRINTK("null\n");
|
|
|
} else {
|
|
|
LOS_DL_LIST_FOR_EACH(list, &muxDLCB->muxListHead) {
|
|
|
muxDLNode = LOS_DL_LIST_ENTRY(list, MuxDLinkNode, muxList);
|
|
|
muxCB = (LosMuxCB *)muxDLNode->muxCB;
|
|
|
count = muxCB->muxCount;
|
|
|
ownerName = muxCB->owner->taskName;
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
PRINTK("<Mutex%u info>\n", index);
|
|
|
PRINTK("Ptr handle:%p\n", muxCB);
|
|
|
PRINTK("Owner:%s\n", ownerName);
|
|
|
PRINTK("Count:%u\n", count);
|
|
|
|
|
|
listTmp = &muxCB->muxList;
|
|
|
OsMutexPendTaskList(listTmp);
|
|
|
|
|
|
index++;
|
|
|
SCHEDULER_LOCK(intSave);
|
|
|
}
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
VOID OsMutexDlockCheck(VOID) //<2F><><EFBFBD>ڼ<EFBFBD><DABC>⻥<EFBFBD><E2BBA5><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ϣ//
|
|
|
{
|
|
|
UINT32 loop, intSave;
|
|
|
UINT32 taskId;
|
|
|
CHAR *name = NULL;
|
|
|
LosTaskCB *taskCB = NULL;
|
|
|
MuxDLinkCB *muxDLCB = NULL;
|
|
|
|
|
|
SCHEDULER_LOCK(intSave);
|
|
|
for (loop = 0; loop < g_taskMaxNum; loop++) {
|
|
|
taskCB = (LosTaskCB *)g_taskCBArray + loop;
|
|
|
if (taskCB->taskStatus & OS_TASK_STATUS_UNUSED) {
|
|
|
continue;
|
|
|
}
|
|
|
|
|
|
muxDLCB = &g_muxDeadlockCBArray[taskCB->taskId];
|
|
|
if ((LOS_TickCountGet() - muxDLCB->lastAccessTime) > OS_MUX_DEADLOCK_CHECK_THRESHOLD) {
|
|
|
name = taskCB->taskName;
|
|
|
taskId = taskCB->taskId;
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
PRINTK("Task_name:%s, ID:0x%x, holds the Mutexs below:\n", name, taskId);
|
|
|
OsTaskHoldMutexList(muxDLCB);
|
|
|
OsDeadlockBackTrace(taskCB);
|
|
|
PRINTK("\n");
|
|
|
SCHEDULER_LOCK(intSave);
|
|
|
}
|
|
|
}
|
|
|
SCHEDULER_UNLOCK(intSave);
|
|
|
}
|
|
|
|
|
|
#ifdef LOSCFG_SHELL //<2F><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ִ<EFBFBD>л<EFBFBD><D0BB><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>鲢<EFBFBD><E9B2A2><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ϣ//
|
|
|
UINT32 OsShellCmdMuxDeadlockCheck(UINT32 argc, const CHAR **argv)
|
|
|
{
|
|
|
if (argc > 0) {
|
|
|
PRINTK("\nUsage: dlock\n");
|
|
|
return OS_ERROR;
|
|
|
}
|
|
|
PRINTK("Start mutexs deadlock check: \n");
|
|
|
OsMutexDlockCheck();
|
|
|
PRINTK("-----------End-----------\n");
|
|
|
return LOS_OK;
|
|
|
}
|
|
|
SHELLCMD_ENTRY(deadlock_shellcmd, CMD_TYPE_EX, "dlock", 0, (CmdCallBackFunc)OsShellCmdMuxDeadlockCheck);
|
|
|
#endif /* LOSCFG_SHELL */
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
#if __cplusplus
|
|
|
}
|
|
|
#endif
|
|
|
#endif /* __cplusplus */
|