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_ciSTATIC BOOL g_startTrace = FALSE;
393d8536b4Sopenharmony_ciSTATIC SchedTraceInfo *g_traceRingBuf = NULL;
403d8536b4Sopenharmony_ciSTATIC UINT32 g_schedCount = 0;
413d8536b4Sopenharmony_ciSTATIC SchedTraceRecordCB g_recordCB = NULL;
423d8536b4Sopenharmony_ciSTATIC SchedTraceShowCB g_showCB = NULL;
433d8536b4Sopenharmony_ci
443d8536b4Sopenharmony_ciSTATIC VOID DefaultShowFormat(SchedTraceInfo *buf, UINT32 count)
453d8536b4Sopenharmony_ci{
463d8536b4Sopenharmony_ci    INT32 i;
473d8536b4Sopenharmony_ci    UINT32 cycle = count / TRACE_NUM;
483d8536b4Sopenharmony_ci    UINT32 point = count % TRACE_NUM;
493d8536b4Sopenharmony_ci
503d8536b4Sopenharmony_ci    PRINTK ("sched %u time, last %u is:\r\n", count, TRACE_NUM);
513d8536b4Sopenharmony_ci    PRINTK("RunTaskID     RunTaskName      NewTaskID    NewTaskName  \r\n");
523d8536b4Sopenharmony_ci    if (cycle > 0) {
533d8536b4Sopenharmony_ci        for (i = point; i < TRACE_NUM; i++) {
543d8536b4Sopenharmony_ci            PRINTK("%4u %20s,  %4u %20s\n",
553d8536b4Sopenharmony_ci                   g_traceRingBuf[i].runTaskID, g_traceRingBuf[i].runTaskName,
563d8536b4Sopenharmony_ci                   g_traceRingBuf[i].newTaskID, g_traceRingBuf[i].newTaskName);
573d8536b4Sopenharmony_ci        }
583d8536b4Sopenharmony_ci    }
593d8536b4Sopenharmony_ci
603d8536b4Sopenharmony_ci    for (i = 0; i < point; i++) {
613d8536b4Sopenharmony_ci        PRINTK("%4u %20s,  %4u %20s\n",
623d8536b4Sopenharmony_ci               g_traceRingBuf[i].runTaskID, g_traceRingBuf[i].runTaskName,
633d8536b4Sopenharmony_ci               g_traceRingBuf[i].newTaskID, g_traceRingBuf[i].newTaskName);
643d8536b4Sopenharmony_ci    }
653d8536b4Sopenharmony_ci
663d8536b4Sopenharmony_ci    PRINTK("\r\n");
673d8536b4Sopenharmony_ci}
683d8536b4Sopenharmony_ci
693d8536b4Sopenharmony_ciSTATIC VOID DefaultRecordHandle(LosTaskCB *newTask, LosTaskCB *runTask)
703d8536b4Sopenharmony_ci{
713d8536b4Sopenharmony_ci    UINT32 point = g_schedCount % TRACE_NUM;
723d8536b4Sopenharmony_ci
733d8536b4Sopenharmony_ci    g_traceRingBuf[point].newTaskID = newTask->taskID;
743d8536b4Sopenharmony_ci    (VOID)memcpy_s(g_traceRingBuf[point].newTaskName, LOS_TASK_NAMELEN, newTask->taskName, LOS_TASK_NAMELEN);
753d8536b4Sopenharmony_ci    g_traceRingBuf[point].runTaskID = runTask->taskID;
763d8536b4Sopenharmony_ci    (VOID)memcpy_s(g_traceRingBuf[point].runTaskName, LOS_TASK_NAMELEN, runTask->taskName, LOS_TASK_NAMELEN);
773d8536b4Sopenharmony_ci
783d8536b4Sopenharmony_ci    g_schedCount++;
793d8536b4Sopenharmony_ci}
803d8536b4Sopenharmony_ci
813d8536b4Sopenharmony_ciSTATIC VOID ShowFormat(SchedTraceInfo *buf, UINT32 count)
823d8536b4Sopenharmony_ci{
833d8536b4Sopenharmony_ci    if (count == 0) {
843d8536b4Sopenharmony_ci        PRINT_ERR("none shed happened\n");
853d8536b4Sopenharmony_ci        return;
863d8536b4Sopenharmony_ci    }
873d8536b4Sopenharmony_ci
883d8536b4Sopenharmony_ci    if (g_showCB != NULL) {
893d8536b4Sopenharmony_ci        g_showCB(buf, count);
903d8536b4Sopenharmony_ci    }
913d8536b4Sopenharmony_ci}
923d8536b4Sopenharmony_ci
933d8536b4Sopenharmony_ciVOID OsSchedTraceRecord(LosTaskCB *newTask, LosTaskCB *runTask)
943d8536b4Sopenharmony_ci{
953d8536b4Sopenharmony_ci    if (g_startTrace == FALSE) {
963d8536b4Sopenharmony_ci        return;
973d8536b4Sopenharmony_ci    }
983d8536b4Sopenharmony_ci
993d8536b4Sopenharmony_ci    if (g_recordCB != NULL) {
1003d8536b4Sopenharmony_ci        g_recordCB(newTask, runTask);
1013d8536b4Sopenharmony_ci    }
1023d8536b4Sopenharmony_ci}
1033d8536b4Sopenharmony_ci
1043d8536b4Sopenharmony_ciVOID LOS_SchedTraceStart(VOID)
1053d8536b4Sopenharmony_ci{
1063d8536b4Sopenharmony_ci    if (g_recordCB == NULL) {
1073d8536b4Sopenharmony_ci        g_recordCB = DefaultRecordHandle;
1083d8536b4Sopenharmony_ci    }
1093d8536b4Sopenharmony_ci
1103d8536b4Sopenharmony_ci    g_traceRingBuf = (SchedTraceInfo *)LOS_MemAlloc(OS_SYS_MEM_ADDR, TRACE_NUM * sizeof(SchedTraceInfo));
1113d8536b4Sopenharmony_ci    if (g_traceRingBuf == NULL) {
1123d8536b4Sopenharmony_ci        PRINT_ERR("alloc failed for dump\n");
1133d8536b4Sopenharmony_ci        return;
1143d8536b4Sopenharmony_ci    }
1153d8536b4Sopenharmony_ci    (VOID)memset_s(g_traceRingBuf, TRACE_NUM * sizeof(SchedTraceInfo), 0, TRACE_NUM * sizeof(SchedTraceInfo));
1163d8536b4Sopenharmony_ci
1173d8536b4Sopenharmony_ci    g_startTrace = TRUE;
1183d8536b4Sopenharmony_ci}
1193d8536b4Sopenharmony_ci
1203d8536b4Sopenharmony_ciVOID LOS_SchedTraceStop(VOID)
1213d8536b4Sopenharmony_ci{
1223d8536b4Sopenharmony_ci    if (g_showCB == NULL) {
1233d8536b4Sopenharmony_ci        g_showCB = DefaultShowFormat;
1243d8536b4Sopenharmony_ci    }
1253d8536b4Sopenharmony_ci    g_startTrace = FALSE;
1263d8536b4Sopenharmony_ci    ShowFormat(g_traceRingBuf, g_schedCount);
1273d8536b4Sopenharmony_ci    g_schedCount = 0;
1283d8536b4Sopenharmony_ci
1293d8536b4Sopenharmony_ci    if (g_traceRingBuf != NULL) {
1303d8536b4Sopenharmony_ci        (VOID)LOS_MemFree(OS_SYS_MEM_ADDR, g_traceRingBuf);
1313d8536b4Sopenharmony_ci        g_traceRingBuf = NULL;
1323d8536b4Sopenharmony_ci    }
1333d8536b4Sopenharmony_ci
1343d8536b4Sopenharmony_ci    return;
1353d8536b4Sopenharmony_ci}
1363d8536b4Sopenharmony_ci
1373d8536b4Sopenharmony_ciVOID LOS_SchedTraceHandleRegister(SchedTraceRecordCB recordCB, SchedTraceShowCB showCB)
1383d8536b4Sopenharmony_ci{
1393d8536b4Sopenharmony_ci    g_recordCB = recordCB;
1403d8536b4Sopenharmony_ci    g_showCB = showCB;
1413d8536b4Sopenharmony_ci}
1423d8536b4Sopenharmony_ci
1433d8536b4Sopenharmony_ciUINT32 OsShellCmdSchedTrace(INT32 argc, const CHAR **argv)
1443d8536b4Sopenharmony_ci{
1453d8536b4Sopenharmony_ci    if (argc != 1) {
1463d8536b4Sopenharmony_ci        PRINT_ERR("\nUsage: st -h\n");
1473d8536b4Sopenharmony_ci        return LOS_NOK;
1483d8536b4Sopenharmony_ci    }
1493d8536b4Sopenharmony_ci
1503d8536b4Sopenharmony_ci    if (strcmp(argv[0], "-s") == 0) {
1513d8536b4Sopenharmony_ci        LOS_SchedTraceStart();
1523d8536b4Sopenharmony_ci    } else if (strcmp(argv[0], "-e") == 0) {
1533d8536b4Sopenharmony_ci        LOS_SchedTraceStop();
1543d8536b4Sopenharmony_ci    } else if (strcmp(argv[0], "-h") == 0) {
1553d8536b4Sopenharmony_ci        PRINTK("\nUsage: \nst -s , SchedTrace start\n");
1563d8536b4Sopenharmony_ci        PRINTK("st -e , SchedTrace end and show\n");
1573d8536b4Sopenharmony_ci    } else {
1583d8536b4Sopenharmony_ci        PRINTK("\nUsage: st -h\n");
1593d8536b4Sopenharmony_ci        return OS_ERROR;
1603d8536b4Sopenharmony_ci    }
1613d8536b4Sopenharmony_ci
1623d8536b4Sopenharmony_ci    return LOS_OK;
1633d8536b4Sopenharmony_ci}
1643d8536b4Sopenharmony_ci#endif /* LOSCFG_STACK_DUMP == 1 */
165