553 lines
17 KiB
C
553 lines
17 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.
|
|
*/
|
|
|
|
/**
|
|
* @defgroup los_vm_dump virtual memory dump operation
|
|
* @ingroup kernel
|
|
*/
|
|
|
|
#include "los_vm_dump.h"
|
|
#include "los_mmu_descriptor_v6.h"
|
|
#ifdef LOSCFG_FS_VFS
|
|
#include "fs/file.h"
|
|
#endif
|
|
#include "los_printf.h"
|
|
#include "los_vm_page.h"
|
|
#include "los_vm_phys.h"
|
|
#include "los_process_pri.h"
|
|
#include "los_atomic.h"
|
|
#include "los_vm_lock.h"
|
|
#include "los_memory_pri.h"
|
|
|
|
|
|
#ifdef LOSCFG_KERNEL_VM
|
|
|
|
#define FLAG_SIZE 4
|
|
#define FLAG_START 2
|
|
|
|
const CHAR *OsGetRegionNameOrFilePath(LosVmMapRegion *region)
|
|
{
|
|
struct file *filep = NULL;
|
|
if (region == NULL) {
|
|
return "";
|
|
#ifdef LOSCFG_FS_VFS
|
|
} else if (LOS_IsRegionFileValid(region)) {
|
|
filep = region->unTypeData.rf.file;
|
|
return filep->f_path;
|
|
#endif
|
|
} else if (region->regionFlags & VM_MAP_REGION_FLAG_HEAP) {
|
|
return "HEAP";
|
|
} else if (region->regionFlags & VM_MAP_REGION_FLAG_STACK) {
|
|
return "STACK";
|
|
} else if (region->regionFlags & VM_MAP_REGION_FLAG_TEXT) {
|
|
return "Text";
|
|
} else if (region->regionFlags & VM_MAP_REGION_FLAG_VDSO) {
|
|
return "VDSO";
|
|
} else if (region->regionFlags & VM_MAP_REGION_FLAG_MMAP) {
|
|
return "MMAP";
|
|
} else if (region->regionFlags & VM_MAP_REGION_FLAG_SHM) {
|
|
return "SHM";
|
|
} else {
|
|
return "";
|
|
}
|
|
return "";
|
|
}
|
|
|
|
INT32 OsRegionOverlapCheckUnlock(LosVmSpace *space, LosVmMapRegion *region)
|
|
{
|
|
LosVmMapRegion *regionTemp = NULL;
|
|
LosRbNode *pstRbNode = NULL;
|
|
LosRbNode *pstRbNodeNext = NULL;
|
|
|
|
/* search the region list */
|
|
RB_SCAN_SAFE(&space->regionRbTree, pstRbNode, pstRbNodeNext)
|
|
regionTemp = (LosVmMapRegion *)pstRbNode;
|
|
if (region->range.base == regionTemp->range.base && region->range.size == regionTemp->range.size) {
|
|
continue;
|
|
}
|
|
if (((region->range.base + region->range.size) > regionTemp->range.base) &&
|
|
(region->range.base < (regionTemp->range.base + regionTemp->range.size))) {
|
|
VM_ERR("overlap between regions:\n"
|
|
"flals:%#x base:%p size:%08x space:%p\n"
|
|
"flags:%#x base:%p size:%08x space:%p",
|
|
region->regionFlags, region->range.base, region->range.size, region->space,
|
|
regionTemp->regionFlags, regionTemp->range.base, regionTemp->range.size, regionTemp->space);
|
|
return -1;
|
|
}
|
|
RB_SCAN_SAFE_END(&space->regionRbTree, pstRbNode, pstRbNodeNext)
|
|
|
|
return 0;
|
|
}
|
|
|
|
UINT32 OsShellCmdProcessVmUsage(LosVmSpace *space)
|
|
{
|
|
LosVmMapRegion *region = NULL;
|
|
LosRbNode *pstRbNode = NULL;
|
|
LosRbNode *pstRbNodeNext = NULL;
|
|
UINT32 used = 0;
|
|
|
|
if (space == NULL) {
|
|
return 0;
|
|
}
|
|
|
|
if (space == LOS_GetKVmSpace()) {
|
|
OsShellCmdProcessPmUsage(space, NULL, &used);
|
|
} else {
|
|
RB_SCAN_SAFE(&space->regionRbTree, pstRbNode, pstRbNodeNext)
|
|
region = (LosVmMapRegion *)pstRbNode;
|
|
used += region->range.size;
|
|
RB_SCAN_SAFE_END(&space->regionRbTree, pstRbNode, pstRbNodeNext)
|
|
}
|
|
|
|
return used;
|
|
}
|
|
|
|
VOID OsKProcessPmUsage(LosVmSpace *kSpace, UINT32 *actualPm)
|
|
{
|
|
UINT32 memUsed;
|
|
UINT32 totalMem;
|
|
UINT32 freeMem;
|
|
UINT32 usedCount = 0;
|
|
UINT32 totalCount = 0;
|
|
LosVmSpace *space = NULL;
|
|
LOS_DL_LIST *spaceList = NULL;
|
|
UINT32 UProcessUsed = 0;
|
|
UINT32 pmTmp;
|
|
|
|
if (actualPm == NULL) {
|
|
return;
|
|
}
|
|
|
|
memUsed = LOS_MemTotalUsedGet(m_aucSysMem1);
|
|
totalMem = LOS_MemPoolSizeGet(m_aucSysMem1);
|
|
freeMem = totalMem - memUsed;
|
|
|
|
OsVmPhysUsedInfoGet(&usedCount, &totalCount);
|
|
/* Kernel resident memory, include default heap memory */
|
|
memUsed = SYS_MEM_SIZE_DEFAULT - (totalCount << PAGE_SHIFT);
|
|
|
|
spaceList = LOS_GetVmSpaceList();
|
|
LOS_DL_LIST_FOR_EACH_ENTRY(space, spaceList, LosVmSpace, node) {
|
|
if (space == LOS_GetKVmSpace()) {
|
|
continue;
|
|
}
|
|
OsUProcessPmUsage(space, NULL, &pmTmp);
|
|
UProcessUsed += pmTmp;
|
|
}
|
|
|
|
/* Kernel dynamic memory, include extended heap memory */
|
|
memUsed += ((usedCount << PAGE_SHIFT) - UProcessUsed);
|
|
/* Remaining heap memory */
|
|
memUsed -= freeMem;
|
|
|
|
*actualPm = memUsed;
|
|
}
|
|
|
|
VOID OsShellCmdProcessPmUsage(LosVmSpace *space, UINT32 *sharePm, UINT32 *actualPm)
|
|
{
|
|
if (space == NULL) {
|
|
return;
|
|
}
|
|
|
|
if ((sharePm == NULL) && (actualPm == NULL)) {
|
|
return;
|
|
}
|
|
|
|
if (space == LOS_GetKVmSpace()) {
|
|
OsKProcessPmUsage(space, actualPm);
|
|
} else {
|
|
OsUProcessPmUsage(space, sharePm, actualPm);
|
|
}
|
|
}
|
|
|
|
VOID OsUProcessPmUsage(LosVmSpace *space, UINT32 *sharePm, UINT32 *actualPm)
|
|
{
|
|
LosVmMapRegion *region = NULL;
|
|
LosRbNode *pstRbNode = NULL;
|
|
LosRbNode *pstRbNodeNext = NULL;
|
|
LosVmPage *page = NULL;
|
|
VADDR_T vaddr;
|
|
size_t size;
|
|
PADDR_T paddr;
|
|
STATUS_T ret;
|
|
INT32 shareRef;
|
|
|
|
if (sharePm != NULL) {
|
|
*sharePm = 0;
|
|
}
|
|
|
|
if (actualPm != NULL) {
|
|
*actualPm = 0;
|
|
}
|
|
|
|
RB_SCAN_SAFE(&space->regionRbTree, pstRbNode, pstRbNodeNext)
|
|
region = (LosVmMapRegion *)pstRbNode;
|
|
vaddr = region->range.base;
|
|
size = region->range.size;
|
|
for (; size > 0; vaddr += PAGE_SIZE, size -= PAGE_SIZE) {
|
|
ret = LOS_ArchMmuQuery(&space->archMmu, vaddr, &paddr, NULL);
|
|
if (ret < 0) {
|
|
continue;
|
|
}
|
|
page = LOS_VmPageGet(paddr);
|
|
if (page == NULL) {
|
|
continue;
|
|
}
|
|
|
|
shareRef = LOS_AtomicRead(&page->refCounts);
|
|
if (shareRef > 1) {
|
|
if (sharePm != NULL) {
|
|
*sharePm += PAGE_SIZE;
|
|
}
|
|
if (actualPm != NULL) {
|
|
*actualPm += PAGE_SIZE / shareRef;
|
|
}
|
|
} else {
|
|
if (actualPm != NULL) {
|
|
*actualPm += PAGE_SIZE;
|
|
}
|
|
}
|
|
}
|
|
RB_SCAN_SAFE_END(&oldVmSpace->regionRbTree, pstRbNode, pstRbNodeNext)
|
|
}
|
|
|
|
LosProcessCB *OsGetPIDByAspace(LosVmSpace *space)
|
|
{
|
|
UINT32 pid;
|
|
UINT32 intSave;
|
|
LosProcessCB *processCB = NULL;
|
|
|
|
SCHEDULER_LOCK(intSave);
|
|
for (pid = 0; pid < g_processMaxNum; ++pid) {
|
|
processCB = g_processCBArray + pid;
|
|
if (OsProcessIsUnused(processCB)) {
|
|
continue;
|
|
}
|
|
|
|
if (processCB->vmSpace == space) {
|
|
SCHEDULER_UNLOCK(intSave);
|
|
return processCB;
|
|
}
|
|
}
|
|
SCHEDULER_UNLOCK(intSave);
|
|
return NULL;
|
|
}
|
|
|
|
UINT32 OsCountRegionPages(LosVmSpace *space, LosVmMapRegion *region, UINT32 *pssPages)
|
|
{
|
|
UINT32 regionPages = 0;
|
|
PADDR_T paddr;
|
|
VADDR_T vaddr;
|
|
UINT32 ref;
|
|
STATUS_T status;
|
|
float pss = 0;
|
|
LosVmPage *page = NULL;
|
|
|
|
for (vaddr = region->range.base; vaddr < region->range.base + region->range.size; vaddr = vaddr + PAGE_SIZE) {
|
|
status = LOS_ArchMmuQuery(&space->archMmu, vaddr, &paddr, NULL);
|
|
if (status == LOS_OK) {
|
|
regionPages++;
|
|
if (pssPages == NULL) {
|
|
continue;
|
|
}
|
|
page = LOS_VmPageGet(paddr);
|
|
if (page != NULL) {
|
|
ref = LOS_AtomicRead(&page->refCounts);
|
|
pss += ((ref > 0) ? (1.0 / ref) : 1);
|
|
} else {
|
|
pss += 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (pssPages != NULL) {
|
|
*pssPages = (UINT32)(pss + 0.5);
|
|
}
|
|
|
|
return regionPages;
|
|
}
|
|
|
|
UINT32 OsCountAspacePages(LosVmSpace *space)
|
|
{
|
|
UINT32 spacePages = 0;
|
|
LosVmMapRegion *region = NULL;
|
|
LosRbNode *pstRbNode = NULL;
|
|
LosRbNode *pstRbNodeNext = NULL;
|
|
|
|
RB_SCAN_SAFE(&space->regionRbTree, pstRbNode, pstRbNodeNext)
|
|
region = (LosVmMapRegion *)pstRbNode;
|
|
spacePages += OsCountRegionPages(space, region, NULL);
|
|
RB_SCAN_SAFE_END(&space->regionRbTree, pstRbNode, pstRbNodeNext)
|
|
return spacePages;
|
|
}
|
|
|
|
CHAR *OsArchFlagsToStr(const UINT32 archFlags)
|
|
{
|
|
UINT32 index;
|
|
UINT32 cacheFlags = archFlags & VM_MAP_REGION_FLAG_CACHE_MASK;
|
|
UINT32 flagSize = FLAG_SIZE * BITMAP_BITS_PER_WORD * sizeof(CHAR);
|
|
CHAR *archMmuFlagsStr = (CHAR *)LOS_MemAlloc(m_aucSysMem0, flagSize);
|
|
if (archMmuFlagsStr == NULL) {
|
|
return NULL;
|
|
}
|
|
(VOID)memset_s(archMmuFlagsStr, flagSize, 0, flagSize);
|
|
switch (cacheFlags) {
|
|
case 0UL:
|
|
strcat_s(archMmuFlagsStr, flagSize, " CH\0");
|
|
break;
|
|
case 1UL:
|
|
strcat_s(archMmuFlagsStr, flagSize, " UC\0");
|
|
break;
|
|
case 2UL:
|
|
strcat_s(archMmuFlagsStr, flagSize, " UD\0");
|
|
break;
|
|
case 3UL:
|
|
strcat_s(archMmuFlagsStr, flagSize, " WC\0");
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
static const CHAR FLAGS[BITMAP_BITS_PER_WORD][FLAG_SIZE] = {
|
|
[0 ... (__builtin_ffsl(VM_MAP_REGION_FLAG_PERM_USER) - 2)] = "???\0",
|
|
[__builtin_ffsl(VM_MAP_REGION_FLAG_PERM_USER) - 1] = " US\0",
|
|
[__builtin_ffsl(VM_MAP_REGION_FLAG_PERM_READ) - 1] = " RD\0",
|
|
[__builtin_ffsl(VM_MAP_REGION_FLAG_PERM_WRITE) - 1] = " WR\0",
|
|
[__builtin_ffsl(VM_MAP_REGION_FLAG_PERM_EXECUTE) - 1] = " EX\0",
|
|
[__builtin_ffsl(VM_MAP_REGION_FLAG_NS) - 1] = " NS\0",
|
|
[__builtin_ffsl(VM_MAP_REGION_FLAG_INVALID) - 1] = " IN\0",
|
|
[__builtin_ffsl(VM_MAP_REGION_FLAG_INVALID) ... (BITMAP_BITS_PER_WORD - 1)] = "???\0",
|
|
};
|
|
|
|
for (index = FLAG_START; index < BITMAP_BITS_PER_WORD; index++) {
|
|
if (FLAGS[index][0] == '?') {
|
|
continue;
|
|
}
|
|
|
|
if (archFlags & (1UL << index)) {
|
|
UINT32 status = strcat_s(archMmuFlagsStr, flagSize, FLAGS[index]);
|
|
if (status != 0) {
|
|
PRINTK("error\n");
|
|
}
|
|
}
|
|
}
|
|
|
|
return archMmuFlagsStr;
|
|
}
|
|
|
|
VOID OsDumpRegion2(LosVmSpace *space, LosVmMapRegion *region)
|
|
{
|
|
UINT32 pssPages = 0;
|
|
UINT32 regionPages;
|
|
|
|
regionPages = OsCountRegionPages(space, region, &pssPages);
|
|
CHAR *flagsStr = OsArchFlagsToStr(region->regionFlags);
|
|
if (flagsStr == NULL) {
|
|
return;
|
|
}
|
|
PRINTK("\t %#010x %-32.32s %#010x %#010x %-15.15s %4d %4d\n",
|
|
region, OsGetRegionNameOrFilePath(region), region->range.base,
|
|
region->range.size, flagsStr, regionPages, pssPages);
|
|
(VOID)LOS_MemFree(m_aucSysMem0, flagsStr);
|
|
}
|
|
|
|
VOID OsDumpAspace(LosVmSpace *space)
|
|
{
|
|
LosVmMapRegion *region = NULL;
|
|
LosRbNode *pstRbNode = NULL;
|
|
LosRbNode *pstRbNodeNext = NULL;
|
|
UINT32 spacePages;
|
|
LosProcessCB *pcb = OsGetPIDByAspace(space);
|
|
|
|
if (pcb == NULL) {
|
|
return;
|
|
}
|
|
|
|
spacePages = OsCountAspacePages(space);
|
|
PRINTK("\r\n PID aspace name base size pages \n");
|
|
PRINTK(" ---- ------ ---- ---- ----- ----\n");
|
|
PRINTK(" %-4d %#010x %-10.10s %#010x %#010x %d\n", pcb->processID, space, pcb->processName,
|
|
space->base, space->size, spacePages);
|
|
PRINTK("\r\n\t region name base size mmu_flags pages pg/ref\n");
|
|
PRINTK("\t ------ ---- ---- ---- --------- ----- -----\n");
|
|
RB_SCAN_SAFE(&space->regionRbTree, pstRbNode, pstRbNodeNext)
|
|
region = (LosVmMapRegion *)pstRbNode;
|
|
if (region != NULL) {
|
|
OsDumpRegion2(space, region);
|
|
(VOID)OsRegionOverlapCheck(space, region);
|
|
} else {
|
|
PRINTK("region is NULL\n");
|
|
}
|
|
RB_SCAN_SAFE_END(&space->regionRbTree, pstRbNode, pstRbNodeNext)
|
|
return;
|
|
}
|
|
|
|
VOID OsDumpAllAspace(VOID)
|
|
{
|
|
LosVmSpace *space = NULL;
|
|
LOS_DL_LIST *aspaceList = LOS_GetVmSpaceList();
|
|
LOS_DL_LIST_FOR_EACH_ENTRY(space, aspaceList, LosVmSpace, node) {
|
|
(VOID)LOS_MuxAcquire(&space->regionMux);
|
|
OsDumpAspace(space);
|
|
(VOID)LOS_MuxRelease(&space->regionMux);
|
|
}
|
|
return;
|
|
}
|
|
|
|
STATUS_T OsRegionOverlapCheck(LosVmSpace *space, LosVmMapRegion *region)
|
|
{
|
|
int ret;
|
|
|
|
if (space == NULL || region == NULL) {
|
|
return -1;
|
|
}
|
|
|
|
(VOID)LOS_MuxAcquire(&space->regionMux);
|
|
ret = OsRegionOverlapCheckUnlock(space, region);
|
|
(VOID)LOS_MuxRelease(&space->regionMux);
|
|
return ret;
|
|
}
|
|
|
|
VOID OsDumpPte(VADDR_T vaddr)
|
|
{
|
|
UINT32 l1Index = vaddr >> MMU_DESCRIPTOR_L1_SMALL_SHIFT;
|
|
LosVmSpace *space = LOS_SpaceGet(vaddr);
|
|
UINT32 ttEntry;
|
|
LosVmPage *page = NULL;
|
|
PTE_T *l2Table = NULL;
|
|
UINT32 l2Index;
|
|
|
|
if (space == NULL) {
|
|
return;
|
|
}
|
|
|
|
ttEntry = space->archMmu.virtTtb[l1Index];
|
|
if (ttEntry) {
|
|
l2Table = LOS_PaddrToKVaddr(MMU_DESCRIPTOR_L1_PAGE_TABLE_ADDR(ttEntry));
|
|
l2Index = (vaddr % MMU_DESCRIPTOR_L1_SMALL_SIZE) >> PAGE_SHIFT;
|
|
if (l2Table == NULL) {
|
|
goto ERR;
|
|
}
|
|
page = LOS_VmPageGet(l2Table[l2Index] & ~(PAGE_SIZE - 1));
|
|
if (page == NULL) {
|
|
goto ERR;
|
|
}
|
|
PRINTK("vaddr %p, l1Index %d, ttEntry %p, l2Table %p, l2Index %d, pfn %p count %d\n",
|
|
vaddr, l1Index, ttEntry, l2Table, l2Index, l2Table[l2Index], LOS_AtomicRead(&page->refCounts));
|
|
} else {
|
|
PRINTK("vaddr %p, l1Index %d, ttEntry %p\n", vaddr, l1Index, ttEntry);
|
|
}
|
|
return;
|
|
ERR:
|
|
PRINTK("%s, error vaddr: %#x, l2Table: %#x, l2Index: %#x\n", __FUNCTION__, vaddr, l2Table, l2Index);
|
|
}
|
|
|
|
UINT32 OsVmPhySegPagesGet(LosVmPhysSeg *seg)
|
|
{
|
|
UINT32 intSave;
|
|
UINT32 flindex;
|
|
UINT32 segFreePages = 0;
|
|
|
|
LOS_SpinLockSave(&seg->freeListLock, &intSave);
|
|
for (flindex = 0; flindex < VM_LIST_ORDER_MAX; flindex++) {
|
|
segFreePages += ((1 << flindex) * seg->freeList[flindex].listCnt);
|
|
}
|
|
LOS_SpinUnlockRestore(&seg->freeListLock, intSave);
|
|
|
|
return segFreePages;
|
|
}
|
|
|
|
VOID OsVmPhysDump(VOID)
|
|
{
|
|
LosVmPhysSeg *seg = NULL;
|
|
UINT32 segFreePages;
|
|
UINT32 totalFreePages = 0;
|
|
UINT32 totalPages = 0;
|
|
UINT32 segIndex;
|
|
UINT32 intSave;
|
|
UINT32 flindex;
|
|
UINT32 listCount[VM_LIST_ORDER_MAX] = {0};
|
|
|
|
for (segIndex = 0; segIndex < g_vmPhysSegNum; segIndex++) {
|
|
seg = &g_vmPhysSeg[segIndex];
|
|
if (seg->size > 0) {
|
|
segFreePages = OsVmPhySegPagesGet(seg);
|
|
#ifdef LOSCFG_SHELL_CMD_DEBUG
|
|
PRINTK("\r\n phys_seg base size free_pages \n");
|
|
PRINTK(" -------- ------- ---------- --------- \n");
|
|
#endif
|
|
PRINTK(" 0x%08x 0x%08x 0x%08x %8u \n", seg, seg->start, seg->size, segFreePages);
|
|
totalFreePages += segFreePages;
|
|
totalPages += (seg->size >> PAGE_SHIFT);
|
|
|
|
LOS_SpinLockSave(&seg->freeListLock, &intSave);
|
|
for (flindex = 0; flindex < VM_LIST_ORDER_MAX; flindex++) {
|
|
listCount[flindex] = seg->freeList[flindex].listCnt;
|
|
}
|
|
LOS_SpinUnlockRestore(&seg->freeListLock, intSave);
|
|
for (flindex = 0; flindex < VM_LIST_ORDER_MAX; flindex++) {
|
|
PRINTK("order = %d, free_count = %d\n", flindex, listCount[flindex]);
|
|
}
|
|
|
|
PRINTK("active anon %d\n", seg->lruSize[VM_LRU_ACTIVE_ANON]);
|
|
PRINTK("inactive anon %d\n", seg->lruSize[VM_LRU_INACTIVE_ANON]);
|
|
PRINTK("active file %d\n", seg->lruSize[VM_LRU_ACTIVE_FILE]);
|
|
PRINTK("inactice file %d\n", seg->lruSize[VM_LRU_INACTIVE_FILE]);
|
|
}
|
|
}
|
|
PRINTK("\n\rpmm pages: total = %u, used = %u, free = %u\n",
|
|
totalPages, (totalPages - totalFreePages), totalFreePages);
|
|
}
|
|
|
|
VOID OsVmPhysUsedInfoGet(UINT32 *usedCount, UINT32 *totalCount)
|
|
{
|
|
UINT32 index;
|
|
UINT32 segFreePages;
|
|
LosVmPhysSeg *physSeg = NULL;
|
|
|
|
if (usedCount == NULL || totalCount == NULL) {
|
|
return;
|
|
}
|
|
*usedCount = 0;
|
|
*totalCount = 0;
|
|
|
|
for (index = 0; index < g_vmPhysSegNum; index++) {
|
|
physSeg = &g_vmPhysSeg[index];
|
|
if (physSeg->size > 0) {
|
|
*totalCount += physSeg->size >> PAGE_SHIFT;
|
|
segFreePages = OsVmPhySegPagesGet(physSeg);
|
|
*usedCount += (*totalCount - segFreePages);
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|