13d8536b4Sopenharmony_ci/*
23d8536b4Sopenharmony_ci * Copyright (c) 2022-2022 Huawei Device Co., Ltd. All rights reserved.
33d8536b4Sopenharmony_ci *
43d8536b4Sopenharmony_ci * Redistribution and use in source and binary forms, with or without modification,
53d8536b4Sopenharmony_ci * are permitted provided that the following conditions are met:
63d8536b4Sopenharmony_ci *
73d8536b4Sopenharmony_ci * 1. Redistributions of source code must retain the above copyright notice, this list of
83d8536b4Sopenharmony_ci *    conditions and the following disclaimer.
93d8536b4Sopenharmony_ci *
103d8536b4Sopenharmony_ci * 2. Redistributions in binary form must reproduce the above copyright notice, this list
113d8536b4Sopenharmony_ci *    of conditions and the following disclaimer in the documentation and/or other materials
123d8536b4Sopenharmony_ci *    provided with the distribution.
133d8536b4Sopenharmony_ci *
143d8536b4Sopenharmony_ci * 3. Neither the name of the copyright holder nor the names of its contributors may be used
153d8536b4Sopenharmony_ci *    to endorse or promote products derived from this software without specific prior written
163d8536b4Sopenharmony_ci *    permission.
173d8536b4Sopenharmony_ci *
183d8536b4Sopenharmony_ci * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
193d8536b4Sopenharmony_ci * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
203d8536b4Sopenharmony_ci * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
213d8536b4Sopenharmony_ci * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
223d8536b4Sopenharmony_ci * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
233d8536b4Sopenharmony_ci * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
243d8536b4Sopenharmony_ci * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
253d8536b4Sopenharmony_ci * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
263d8536b4Sopenharmony_ci * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
273d8536b4Sopenharmony_ci * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
283d8536b4Sopenharmony_ci * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
293d8536b4Sopenharmony_ci */
303d8536b4Sopenharmony_ci
313d8536b4Sopenharmony_ci#include "los_debugtools.h"
323d8536b4Sopenharmony_ci#include "securec.h"
333d8536b4Sopenharmony_ci#include "los_debug.h"
343d8536b4Sopenharmony_ci#include "los_memory.h"
353d8536b4Sopenharmony_ci#include "los_arch.h"
363d8536b4Sopenharmony_ci
373d8536b4Sopenharmony_ci#if (LOSCFG_DEBUG_TOOLS == 1)
383d8536b4Sopenharmony_citypedef struct {
393d8536b4Sopenharmony_ci    UINT32 waterLine;
403d8536b4Sopenharmony_ci    UINT32 taskSPTop;
413d8536b4Sopenharmony_ci    UINT32 taskSPLimit;
423d8536b4Sopenharmony_ci    UINTPTR taskSP;
433d8536b4Sopenharmony_ci} DumpInfo;
443d8536b4Sopenharmony_ci
453d8536b4Sopenharmony_ciSTATIC VOID ShowFormat(UINTPTR *buf, DumpInfo *info)
463d8536b4Sopenharmony_ci{
473d8536b4Sopenharmony_ci    INT32 i;
483d8536b4Sopenharmony_ci
493d8536b4Sopenharmony_ci    UINT32 len = info->waterLine / sizeof(UINTPTR);
503d8536b4Sopenharmony_ci    UINTPTR addr = (info->taskSPLimit - info->waterLine);
513d8536b4Sopenharmony_ci
523d8536b4Sopenharmony_ci    for (i = 0; i < len; i++) {
533d8536b4Sopenharmony_ci        if ((i % PRINT_PER_ROW) == 0) {
543d8536b4Sopenharmony_ci            PRINTK("\r\n 0x%08x: ", addr);
553d8536b4Sopenharmony_ci        }
563d8536b4Sopenharmony_ci        if (addr == info->taskSP) {
573d8536b4Sopenharmony_ci            PRINTK(" *%08x", buf[i]);
583d8536b4Sopenharmony_ci        } else {
593d8536b4Sopenharmony_ci            PRINTK("  %08x", buf[i]);
603d8536b4Sopenharmony_ci        }
613d8536b4Sopenharmony_ci        addr += sizeof(UINTPTR);
623d8536b4Sopenharmony_ci    }
633d8536b4Sopenharmony_ci
643d8536b4Sopenharmony_ci    PRINTK("\r\n");
653d8536b4Sopenharmony_ci}
663d8536b4Sopenharmony_ci
673d8536b4Sopenharmony_ciSTATIC INT32 DumpTaskInfo(UINT32 taskID, UINTPTR *buf, DumpInfo *info)
683d8536b4Sopenharmony_ci{
693d8536b4Sopenharmony_ci    errno_t ret;
703d8536b4Sopenharmony_ci    LosTaskCB *taskCB = OS_TCB_FROM_TID(taskID);
713d8536b4Sopenharmony_ci
723d8536b4Sopenharmony_ci    if (taskID == LOS_CurTaskIDGet()) {
733d8536b4Sopenharmony_ci        info->taskSP = ArchSpGet();
743d8536b4Sopenharmony_ci    } else {
753d8536b4Sopenharmony_ci        info->taskSP = (UINTPTR)taskCB->stackPointer;
763d8536b4Sopenharmony_ci    }
773d8536b4Sopenharmony_ci
783d8536b4Sopenharmony_ci    info->taskSPTop = taskCB->topOfStack;
793d8536b4Sopenharmony_ci    info->taskSPLimit = taskCB->topOfStack + taskCB->stackSize;
803d8536b4Sopenharmony_ci    if ((info->taskSP > info->taskSPLimit) || (info->taskSP < info->taskSPTop)) {
813d8536b4Sopenharmony_ci        return LOS_NOK;
823d8536b4Sopenharmony_ci    }
833d8536b4Sopenharmony_ci
843d8536b4Sopenharmony_ci    ret = memcpy_s(buf, info->waterLine, (const VOID *)(info->taskSPLimit - info->waterLine), info->waterLine);
853d8536b4Sopenharmony_ci    if (ret != EOK) {
863d8536b4Sopenharmony_ci        return LOS_NOK;
873d8536b4Sopenharmony_ci    }
883d8536b4Sopenharmony_ci
893d8536b4Sopenharmony_ci    return LOS_OK;
903d8536b4Sopenharmony_ci}
913d8536b4Sopenharmony_ci
923d8536b4Sopenharmony_ciVOID LOS_TaskStackDump(UINT32 taskID)
933d8536b4Sopenharmony_ci{
943d8536b4Sopenharmony_ci    UINTPTR *buf = NULL;
953d8536b4Sopenharmony_ci    DumpInfo info;
963d8536b4Sopenharmony_ci    UINT32 intSave;
973d8536b4Sopenharmony_ci    INT32 ret;
983d8536b4Sopenharmony_ci
993d8536b4Sopenharmony_ci    if (taskID > g_taskMaxNum) {
1003d8536b4Sopenharmony_ci        PRINT_ERR("error taskID %u\r\n", taskID);
1013d8536b4Sopenharmony_ci        return;
1023d8536b4Sopenharmony_ci    }
1033d8536b4Sopenharmony_ci
1043d8536b4Sopenharmony_ci    if (OS_INT_ACTIVE) {
1053d8536b4Sopenharmony_ci        PRINT_ERR("called during an interrupt.\r\n");
1063d8536b4Sopenharmony_ci        return;
1073d8536b4Sopenharmony_ci    }
1083d8536b4Sopenharmony_ci
1093d8536b4Sopenharmony_ci    intSave = LOS_IntLock();
1103d8536b4Sopenharmony_ci    info.waterLine = OsGetTaskWaterLine(taskID);
1113d8536b4Sopenharmony_ci    if (info.waterLine == OS_NULL_INT) {
1123d8536b4Sopenharmony_ci        LOS_IntRestore(intSave);
1133d8536b4Sopenharmony_ci        return;
1143d8536b4Sopenharmony_ci    }
1153d8536b4Sopenharmony_ci
1163d8536b4Sopenharmony_ci    buf = (UINTPTR *)LOS_MemAlloc(OS_SYS_MEM_ADDR, info.waterLine);
1173d8536b4Sopenharmony_ci    if (buf == NULL) {
1183d8536b4Sopenharmony_ci        LOS_IntRestore(intSave);
1193d8536b4Sopenharmony_ci        PRINT_ERR("alloc failed for dump\n");
1203d8536b4Sopenharmony_ci        return;
1213d8536b4Sopenharmony_ci    }
1223d8536b4Sopenharmony_ci    (VOID)memset_s(buf, info.waterLine, 0, info.waterLine);
1233d8536b4Sopenharmony_ci
1243d8536b4Sopenharmony_ci    ret = DumpTaskInfo(taskID, buf, &info);
1253d8536b4Sopenharmony_ci    if (ret != LOS_OK) {
1263d8536b4Sopenharmony_ci        LOS_IntRestore(intSave);
1273d8536b4Sopenharmony_ci        (VOID)LOS_MemFree(OS_SYS_MEM_ADDR, buf);
1283d8536b4Sopenharmony_ci        PRINT_ERR("SP 0x%x may error or memcpy_s failed, stack space from 0x%x to 0x%x\r\n", \
1293d8536b4Sopenharmony_ci                  info.taskSP, info.taskSPTop, info.taskSPLimit);
1303d8536b4Sopenharmony_ci        return;
1313d8536b4Sopenharmony_ci    }
1323d8536b4Sopenharmony_ci
1333d8536b4Sopenharmony_ci    LOS_IntRestore(intSave);
1343d8536b4Sopenharmony_ci    PRINTK("Task %u, SP 0x%x, WaterLine 0x%x", taskID, info.taskSP, info.waterLine);
1353d8536b4Sopenharmony_ci    ShowFormat(buf, &info);
1363d8536b4Sopenharmony_ci    (VOID)LOS_MemFree(OS_SYS_MEM_ADDR, buf);
1373d8536b4Sopenharmony_ci
1383d8536b4Sopenharmony_ci    return;
1393d8536b4Sopenharmony_ci}
1403d8536b4Sopenharmony_ci
1413d8536b4Sopenharmony_ciUINT32 OsShellCmdStackDump(INT32 argc, const CHAR **argv)
1423d8536b4Sopenharmony_ci{
1433d8536b4Sopenharmony_ci    UINT32 taskID;
1443d8536b4Sopenharmony_ci
1453d8536b4Sopenharmony_ci    if (argc != 1) {
1463d8536b4Sopenharmony_ci        PRINT_ERR("\nUsage: stack taskID\n");
1473d8536b4Sopenharmony_ci        return LOS_NOK;
1483d8536b4Sopenharmony_ci    }
1493d8536b4Sopenharmony_ci
1503d8536b4Sopenharmony_ci    taskID = (UINT32)atoi(argv[0]);
1513d8536b4Sopenharmony_ci
1523d8536b4Sopenharmony_ci    LOS_TaskStackDump(taskID);
1533d8536b4Sopenharmony_ci    return LOS_OK;
1543d8536b4Sopenharmony_ci}
1553d8536b4Sopenharmony_ci#endif /* LOSCFG_STACK_DUMP == 1 */
156