/* * 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_cpup_pri.h" #include "los_base.h" #include "los_init.h" #include "los_process_pri.h" #include "los_swtmr.h" #ifdef LOSCFG_KERNEL_CPUP LITE_OS_SEC_BSS STATIC UINT16 cpupSwtmrID; LITE_OS_SEC_BSS STATIC UINT16 cpupInitFlg = 0; LITE_OS_SEC_BSS OsIrqCpupCB *g_irqCpup = NULL; LITE_OS_SEC_BSS STATIC UINT32 cpupMaxNum; LITE_OS_SEC_BSS STATIC UINT16 cpupHisPos = 0; /* current Sampling point of historyTime */ LITE_OS_SEC_BSS STATIC UINT64 cpuHistoryTime[OS_CPUP_HISTORY_RECORD_NUM + 1]; LITE_OS_SEC_BSS STATIC UINT32 runningTasks[LOSCFG_KERNEL_CORE_NUM]; LITE_OS_SEC_BSS STATIC UINT64 cpupStartCycles = 0; #ifdef LOSCFG_CPUP_INCLUDE_IRQ LITE_OS_SEC_BSS UINT64 timeInIrqSwitch[LOSCFG_KERNEL_CORE_NUM]; LITE_OS_SEC_BSS STATIC UINT64 cpupIntTimeStart[LOSCFG_KERNEL_CORE_NUM]; #endif #define INVALID_ID ((UINT32)-1) #define OS_CPUP_UNUSED 0x0U #define OS_CPUP_USED 0x1U #define HIGH_BITS 32 #define CPUP_PRE_POS(pos) (((pos) == 0) ? (OS_CPUP_HISTORY_RECORD_NUM - 1) : ((pos) - 1)) #define CPUP_POST_POS(pos) (((pos) == (OS_CPUP_HISTORY_RECORD_NUM - 1)) ? 0 : ((pos) + 1)) STATIC UINT64 OsGetCpuCycle(VOID) { UINT32 high; UINT32 low; UINT64 cycles; LOS_GetCpuCycle(&high, &low); cycles = ((UINT64)high << HIGH_BITS) + low; if (cpupStartCycles == 0) { cpupStartCycles = cycles; } /* * The cycles should keep growing, if the checking failed, * it mean LOS_GetCpuCycle has the problem which should be fixed. */ LOS_ASSERT(cycles >= cpupStartCycles); return (cycles - cpupStartCycles); } LITE_OS_SEC_TEXT_INIT VOID OsCpupGuard(VOID) { UINT16 prevPos; UINT32 loop; UINT32 runTaskID; UINT32 intSave; UINT64 cycle, cycleIncrement; LosTaskCB *taskCB = NULL; LosProcessCB *processCB = NULL; SCHEDULER_LOCK(intSave); cycle = OsGetCpuCycle(); prevPos = cpupHisPos; cpupHisPos = CPUP_POST_POS(cpupHisPos); cpuHistoryTime[prevPos] = cycle; #ifdef LOSCFG_CPUP_INCLUDE_IRQ for (loop = 0; loop < cpupMaxNum; loop++) { if (g_irqCpup[loop].status == OS_CPUP_UNUSED) { continue; } g_irqCpup[loop].cpup.historyTime[prevPos] = g_irqCpup[loop].cpup.allTime; } #endif for (loop = 0; loop < g_processMaxNum; loop++) { processCB = OS_PCB_FROM_PID(loop); if (processCB->processCpup == NULL) { continue; } processCB->processCpup->historyTime[prevPos] = processCB->processCpup->allTime; } for (loop = 0; loop < g_taskMaxNum; loop++) { taskCB = OS_TCB_FROM_TID(loop); taskCB->taskCpup.historyTime[prevPos] = taskCB->taskCpup.allTime; } for (loop = 0; loop < LOSCFG_KERNEL_CORE_NUM; loop++) { runTaskID = runningTasks[loop]; if (runTaskID == INVALID_ID) { continue; } taskCB = OS_TCB_FROM_TID(runTaskID); /* reacquire the cycle to prevent flip */ cycle = OsGetCpuCycle(); cycleIncrement = cycle - taskCB->taskCpup.startTime; #ifdef LOSCFG_CPUP_INCLUDE_IRQ cycleIncrement -= timeInIrqSwitch[loop]; #endif taskCB->taskCpup.historyTime[prevPos] += cycleIncrement; processCB = OS_PCB_FROM_PID(taskCB->processID); processCB->processCpup->historyTime[prevPos] += cycleIncrement; } SCHEDULER_UNLOCK(intSave); } LITE_OS_SEC_TEXT_INIT UINT32 OsCpupGuardCreator(VOID) { (VOID)LOS_SwtmrCreate(LOSCFG_BASE_CORE_TICK_PER_SECOND, LOS_SWTMR_MODE_PERIOD, (SWTMR_PROC_FUNC)OsCpupGuard, &cpupSwtmrID, 0); (VOID)LOS_SwtmrStart(cpupSwtmrID); return LOS_OK; } LOS_MODULE_INIT(OsCpupGuardCreator, LOS_INIT_LEVEL_KMOD_TASK); /* * Description: initialization of CPUP * Return : LOS_OK or Error Information */ LITE_OS_SEC_TEXT_INIT UINT32 OsCpupInit(VOID) { UINT16 loop; #ifdef LOSCFG_CPUP_INCLUDE_IRQ UINT32 size; cpupMaxNum = OS_HWI_MAX_NUM * LOSCFG_KERNEL_CORE_NUM; /* every process has only one record, and it won't operated at the same time */ size = cpupMaxNum * sizeof(OsIrqCpupCB); g_irqCpup = (OsIrqCpupCB *)LOS_MemAlloc(m_aucSysMem0, size); if (g_irqCpup == NULL) { PRINT_ERR("OsCpupInit error\n"); return LOS_ERRNO_CPUP_NO_MEMORY; } (VOID)memset_s(g_irqCpup, size, 0, size); #endif for (loop = 0; loop < LOSCFG_KERNEL_CORE_NUM; loop++) { runningTasks[loop] = INVALID_ID; } cpupInitFlg = 1; return LOS_OK; } LOS_MODULE_INIT(OsCpupInit, LOS_INIT_LEVEL_KMOD_EXTENDED); STATIC VOID OsResetCpup(OsCpupBase *cpup, UINT64 cycle) { UINT16 loop; cpup->startTime = cycle; cpup->allTime = cycle; for (loop = 0; loop < (OS_CPUP_HISTORY_RECORD_NUM + 1); loop++) { cpup->historyTime[loop] = cycle; } } LITE_OS_SEC_TEXT_INIT VOID LOS_CpupReset(VOID) { LosProcessCB *processCB = NULL; LosTaskCB *taskCB = NULL; UINT32 index; UINT64 cycle; UINT32 intSave; cpupInitFlg = 0; intSave = LOS_IntLock(); (VOID)LOS_SwtmrStop(cpupSwtmrID); cycle = OsGetCpuCycle(); for (index = 0; index < (OS_CPUP_HISTORY_RECORD_NUM + 1); index++) { cpuHistoryTime[index] = cycle; } for (index = 0; index < g_processMaxNum; index++) { processCB = OS_PCB_FROM_PID(index); if (processCB->processCpup == NULL) { continue; } OsResetCpup(processCB->processCpup, cycle); } for (index = 0; index < g_taskMaxNum; index++) { taskCB = OS_TCB_FROM_TID(index); OsResetCpup(&taskCB->taskCpup, cycle); } #ifdef LOSCFG_CPUP_INCLUDE_IRQ if (g_irqCpup != NULL) { for (index = 0; index < cpupMaxNum; index++) { OsResetCpup(&g_irqCpup[index].cpup, cycle); g_irqCpup[index].timeMax = 0; } for (index = 0; index < LOSCFG_KERNEL_CORE_NUM; index++) { timeInIrqSwitch[index] = 0; } } #endif (VOID)LOS_SwtmrStart(cpupSwtmrID); LOS_IntRestore(intSave); cpupInitFlg = 1; return; } VOID OsCpupCycleEndStart(UINT32 runTaskID, UINT32 newTaskID) { /* OsCurrTaskGet and OsCurrProcessGet are not allowed to be called. */ LosTaskCB *runTask = OS_TCB_FROM_TID(runTaskID); OsCpupBase *runTaskCpup = &runTask->taskCpup; OsCpupBase *newTaskCpup = (OsCpupBase *)&(OS_TCB_FROM_TID(newTaskID)->taskCpup); OsCpupBase *processCpup = OS_PCB_FROM_PID(runTask->processID)->processCpup; UINT64 cpuCycle, cycleIncrement; UINT16 cpuid = ArchCurrCpuid(); if (cpupInitFlg == 0) { return; } cpuCycle = OsGetCpuCycle(); if (runTaskCpup->startTime != 0) { cycleIncrement = cpuCycle - runTaskCpup->startTime; #ifdef LOSCFG_CPUP_INCLUDE_IRQ cycleIncrement -= timeInIrqSwitch[cpuid]; timeInIrqSwitch[cpuid] = 0; #endif runTaskCpup->allTime += cycleIncrement; if (processCpup != NULL) { processCpup->allTime += cycleIncrement; } runTaskCpup->startTime = 0; } newTaskCpup->startTime = cpuCycle; runningTasks[cpuid] = newTaskID; } LITE_OS_SEC_TEXT_MINOR STATIC VOID OsCpupGetPos(UINT16 mode, UINT16 *curPosPointer, UINT16 *prePosPointer) { UINT16 curPos; UINT16 tmpPos; UINT16 prePos; tmpPos = cpupHisPos; curPos = CPUP_PRE_POS(tmpPos); /* * The current position has nothing to do with the CPUP modes, * however, the previous position differs. */ switch (mode) { case CPUP_LAST_ONE_SECONDS: prePos = CPUP_PRE_POS(curPos); break; case CPUP_LAST_TEN_SECONDS: prePos = tmpPos; break; case CPUP_ALL_TIME: /* fall-through */ default: prePos = OS_CPUP_HISTORY_RECORD_NUM; break; } *curPosPointer = curPos; *prePosPointer = prePos; return; } STATIC INLINE UINT32 OsCalculateCpupUsage(const OsCpupBase *cpup, UINT16 pos, UINT16 prePos, UINT64 allCycle) { UINT32 usage = 0; UINT64 cpuCycle = cpup->historyTime[pos] - cpup->historyTime[prePos]; if (allCycle) { usage = (UINT32)((LOS_CPUP_SINGLE_CORE_PRECISION * cpuCycle) / allCycle); } return usage; } STATIC UINT32 OsHistorySysCpuUsageUnsafe(UINT16 mode) { UINT64 cpuAllCycle; UINT16 pos; UINT16 prePos; UINT32 idleProcessID; OsCpupBase *processCpup = NULL; if (cpupInitFlg == 0) { return LOS_ERRNO_CPUP_NO_INIT; } OsCpupGetPos(mode, &pos, &prePos); cpuAllCycle = cpuHistoryTime[pos] - cpuHistoryTime[prePos]; idleProcessID = OsGetIdleProcessID(); processCpup = OS_PCB_FROM_PID(idleProcessID)->processCpup; return (LOS_CPUP_PRECISION - OsCalculateCpupUsage(processCpup, pos, prePos, cpuAllCycle)); } LITE_OS_SEC_TEXT_MINOR UINT32 LOS_HistorySysCpuUsage(UINT16 mode) { UINT32 cpup; UINT32 intSave; /* get end time of current process */ SCHEDULER_LOCK(intSave); cpup = OsHistorySysCpuUsageUnsafe(mode); SCHEDULER_UNLOCK(intSave); return cpup; } STATIC UINT32 OsHistoryProcessCpuUsageUnsafe(UINT32 pid, UINT16 mode) { LosProcessCB *processCB = NULL; UINT64 cpuAllCycle; UINT16 pos, prePos; if (cpupInitFlg == 0) { return LOS_ERRNO_CPUP_NO_INIT; } if (OS_PID_CHECK_INVALID(pid)) { return LOS_ERRNO_CPUP_ID_INVALID; } processCB = OS_PCB_FROM_PID(pid); if (OsProcessIsUnused(processCB)) { return LOS_ERRNO_CPUP_NO_CREATED; } if (processCB->processCpup == NULL) { return LOS_ERRNO_CPUP_ID_INVALID; } OsCpupGetPos(mode, &pos, &prePos); cpuAllCycle = cpuHistoryTime[pos] - cpuHistoryTime[prePos]; return OsCalculateCpupUsage(processCB->processCpup, pos, prePos, cpuAllCycle); } LITE_OS_SEC_TEXT_MINOR UINT32 LOS_HistoryProcessCpuUsage(UINT32 pid, UINT16 mode) { UINT32 cpup; UINT32 intSave; SCHEDULER_LOCK(intSave); cpup = OsHistoryProcessCpuUsageUnsafe(pid, mode); SCHEDULER_UNLOCK(intSave); return cpup; } STATIC UINT32 OsHistoryTaskCpuUsageUnsafe(UINT32 tid, UINT16 mode) { LosTaskCB *taskCB = NULL; UINT64 cpuAllCycle; UINT16 pos, prePos; if (cpupInitFlg == 0) { return LOS_ERRNO_CPUP_NO_INIT; } if (OS_TID_CHECK_INVALID(tid)) { return LOS_ERRNO_CPUP_ID_INVALID; } taskCB = OS_TCB_FROM_TID(tid); if (OsTaskIsUnused(taskCB)) { return LOS_ERRNO_CPUP_NO_CREATED; } OsCpupGetPos(mode, &pos, &prePos); cpuAllCycle = cpuHistoryTime[pos] - cpuHistoryTime[prePos]; return OsCalculateCpupUsage(&taskCB->taskCpup, pos, prePos, cpuAllCycle); } LITE_OS_SEC_TEXT_MINOR UINT32 LOS_HistoryTaskCpuUsage(UINT32 tid, UINT16 mode) { UINT32 intSave; UINT32 cpup; SCHEDULER_LOCK(intSave); cpup = OsHistoryTaskCpuUsageUnsafe(tid, mode); SCHEDULER_UNLOCK(intSave); return cpup; } STATIC UINT32 OsCpupUsageParamCheckAndReset(CPUP_INFO_S *cpupInfo, UINT32 len, UINT32 number) { if (cpupInitFlg == 0) { return LOS_ERRNO_CPUP_NO_INIT; } if ((cpupInfo == NULL) || (len < (sizeof(CPUP_INFO_S) * number))) { return LOS_ERRNO_CPUP_PTR_ERR; } (VOID)memset_s(cpupInfo, len, 0, len); return LOS_OK; } LITE_OS_SEC_TEXT_MINOR UINT32 OsGetAllProcessCpuUsageUnsafe(UINT16 mode, CPUP_INFO_S *cpupInfo, UINT32 len) { LosProcessCB *processCB = NULL; UINT64 cpuAllCycle; UINT16 pos, prePos; UINT32 processID; UINT32 ret; ret = OsCpupUsageParamCheckAndReset(cpupInfo, len, g_processMaxNum); if (ret != LOS_OK) { return ret; } OsCpupGetPos(mode, &pos, &prePos); cpuAllCycle = cpuHistoryTime[pos] - cpuHistoryTime[prePos]; for (processID = 0; processID < g_processMaxNum; processID++) { processCB = OS_PCB_FROM_PID(processID); if (OsProcessIsUnused(processCB) || (processCB->processCpup == NULL)) { continue; } cpupInfo[processID].usage = OsCalculateCpupUsage(processCB->processCpup, pos, prePos, cpuAllCycle); cpupInfo[processID].status = OS_CPUP_USED; } return LOS_OK; } LITE_OS_SEC_TEXT_MINOR UINT32 LOS_GetAllProcessCpuUsage(UINT16 mode, CPUP_INFO_S *cpupInfo, UINT32 len) { UINT32 intSave; UINT32 ret; SCHEDULER_LOCK(intSave); ret = OsGetAllProcessCpuUsageUnsafe(mode, cpupInfo, len); SCHEDULER_UNLOCK(intSave); return ret; } LITE_OS_SEC_TEXT_MINOR UINT32 OsGetAllProcessAndTaskCpuUsageUnsafe(UINT16 mode, CPUP_INFO_S *cpupInfo, UINT32 len) { UINT64 cpuAllCycle; UINT16 pos, prePos; UINT32 taskID; UINT32 ret; LosTaskCB *taskCB = NULL; OsCpupBase *processCpupBase = NULL; CPUP_INFO_S *processCpup = cpupInfo; CPUP_INFO_S *taskCpup = (CPUP_INFO_S *)((UINTPTR)cpupInfo + sizeof(CPUP_INFO_S) * g_processMaxNum); ret = OsCpupUsageParamCheckAndReset(cpupInfo, len, g_taskMaxNum + g_processMaxNum); if (ret != LOS_OK) { return ret; } OsCpupGetPos(mode, &pos, &prePos); cpuAllCycle = cpuHistoryTime[pos] - cpuHistoryTime[prePos]; for (taskID = 0; taskID < g_taskMaxNum; taskID++) { taskCB = OS_TCB_FROM_TID(taskID); if (OsTaskIsUnused(taskCB)) { continue; } taskCpup[taskID].usage = OsCalculateCpupUsage(&taskCB->taskCpup, pos, prePos, cpuAllCycle); taskCpup[taskID].status = OS_CPUP_USED; if (processCpup[taskCB->processID].status == OS_CPUP_UNUSED) { processCpupBase = OS_PCB_FROM_PID(taskCB->processID)->processCpup; if (processCpupBase != NULL) { processCpup[taskCB->processID].usage = OsCalculateCpupUsage(processCpupBase, pos, prePos, cpuAllCycle); processCpup[taskCB->processID].status = OS_CPUP_USED; } } } return LOS_OK; } #ifdef LOSCFG_CPUP_INCLUDE_IRQ LITE_OS_SEC_TEXT_MINOR VOID OsCpupIrqStart(UINT16 cpuid) { UINT32 high; UINT32 low; LOS_GetCpuCycle(&high, &low); cpupIntTimeStart[cpuid] = ((UINT64)high << HIGH_BITS) + low; return; } LITE_OS_SEC_TEXT_MINOR VOID OsCpupIrqEnd(UINT16 cpuid, UINT32 intNum) { UINT32 high; UINT32 low; UINT64 intTimeEnd; UINT64 usedTime; LOS_GetCpuCycle(&high, &low); intTimeEnd = ((UINT64)high << HIGH_BITS) + low; OsIrqCpupCB *irqCb = &g_irqCpup[(intNum * LOSCFG_KERNEL_CORE_NUM) + cpuid]; irqCb->id = intNum; irqCb->status = OS_CPUP_USED; usedTime = intTimeEnd - cpupIntTimeStart[cpuid]; timeInIrqSwitch[cpuid] += usedTime; irqCb->cpup.allTime += usedTime; if (irqCb->count <= 100) { /* Take 100 samples */ irqCb->allTime += usedTime; irqCb->count++; } else { irqCb->allTime = 0; irqCb->count = 0; } if (usedTime > irqCb->timeMax) { irqCb->timeMax = usedTime; } return; } LITE_OS_SEC_TEXT_MINOR OsIrqCpupCB *OsGetIrqCpupArrayBase(VOID) { return g_irqCpup; } LITE_OS_SEC_TEXT_MINOR UINT32 OsGetAllIrqCpuUsageUnsafe(UINT16 mode, CPUP_INFO_S *cpupInfo, UINT32 len) { UINT16 pos, prePos; UINT64 cpuAllCycle; UINT32 loop; UINT32 ret; ret = OsCpupUsageParamCheckAndReset(cpupInfo, len, cpupMaxNum); if (ret != LOS_OK) { return ret; } OsCpupGetPos(mode, &pos, &prePos); cpuAllCycle = cpuHistoryTime[pos] - cpuHistoryTime[prePos]; for (loop = 0; loop < cpupMaxNum; loop++) { if (g_irqCpup[loop].status == OS_CPUP_UNUSED) { continue; } cpupInfo[loop].usage = OsCalculateCpupUsage(&g_irqCpup[loop].cpup, pos, prePos, cpuAllCycle); cpupInfo[loop].status = g_irqCpup[loop].status; } return LOS_OK; } LITE_OS_SEC_TEXT_MINOR UINT32 LOS_GetAllIrqCpuUsage(UINT16 mode, CPUP_INFO_S *cpupInfo, UINT32 len) { UINT32 intSave; UINT32 ret; SCHEDULER_LOCK(intSave); ret = OsGetAllIrqCpuUsageUnsafe(mode, cpupInfo, len); SCHEDULER_UNLOCK(intSave); return ret; } #endif #endif /* LOSCFG_KERNEL_CPUP */