• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (c) 2023 Huawei Device Co., Ltd.
3  * Licensed under the Apache License, Version 2.0 (the "License");
4  * you may not use this file except in compliance with the License.
5  * You may obtain a copy of the License at
6  *
7  *     http://www.apache.org/licenses/LICENSE-2.0
8  *
9  * Unless required by applicable law or agreed to in writing, software
10  * distributed under the License is distributed on an "AS IS" BASIS,
11  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12  * See the License for the specific language governing permissions and
13  * limitations under the License.
14  */
15 
16 #include "meminfo.h"
17 
18 #include <fstream>
19 #include <sstream>
20 #include <v1_0/imemory_tracker_interface.h>
21 
22 #include "file_ex.h" // LoadStringFromFile
23 #include "hilog/log.h"
24 
25 #undef LOG_TAG
26 #define LOG_TAG "MemInfo"
27 
28 #undef LOG_DOMAIN
29 #define LOG_DOMAIN 0xD001799
30 
31 
32 namespace OHOS {
33 namespace MemInfo {
34 using namespace OHOS::HDI::Memorytracker::V1_0;
35 constexpr int PAGE_TO_KB = 4;
36 constexpr int BYTE_PER_KB = 1024;
37 
38 // get Rss from statm
GetRssByPid(const int pid)39 uint64_t GetRssByPid(const int pid)
40 {
41     uint64_t size = 0;
42     std::string statm;
43     std::string vss;
44     std::string rss;
45 
46     std::string statmPath = "/proc/" + std::to_string(pid) + "/statm";
47     // format like:
48     // 640 472 369 38 0 115 0
49     if (!OHOS::LoadStringFromFile(statmPath, statm)) {
50         HILOG_ERROR(LOG_CORE, "statm file error!");
51         return size;
52     }
53     std::istringstream isStatm(statm);
54     isStatm >> vss >> rss; // pages
55 
56     size = static_cast<uint64_t>(atoi(rss.c_str()) * PAGE_TO_KB);
57     return size;
58 }
59 
60 // get Pss from smaps_rollup
GetPssByPid(const int pid)61 uint64_t GetPssByPid(const int pid)
62 {
63     uint64_t size = 0;
64     std::string filename = "/proc/" + std::to_string(pid) + "/smaps_rollup";
65     std::ifstream in(filename);
66     if (!in) {
67         HILOG_ERROR(LOG_CORE, "File %{public}s not found.\n", filename.c_str());
68         return size;
69     }
70 
71     std::string content;
72     while (in.good() && getline(in, content)) {
73         std::string::size_type typePos = content.find(":");
74         if (typePos != content.npos) {
75             std::string type = content.substr(0, typePos);
76             if (type == "Pss") {
77                 std::string valueStr = content.substr(typePos + 1);
78                 const int base = 10;
79                 size = strtoull(valueStr.c_str(), nullptr, base);
80                 break;
81             }
82         }
83     }
84     in.close();
85     return size;
86 }
87 
88 // get SwapPss from smaps_rollup
GetSwapPssByPid(const int pid)89 uint64_t GetSwapPssByPid(const int pid)
90 {
91     uint64_t size = 0;
92     std::string filename = "/proc/" + std::to_string(pid) + "/smaps_rollup";
93     std::ifstream in(filename);
94     if (!in) {
95         HILOG_ERROR(LOG_CORE, "File %{public}s not found.\n", filename.c_str());
96         return size;
97     }
98 
99     std::string content;
100     while (in.good() && getline(in, content)) {
101         std::string::size_type typePos = content.find(":");
102         if (typePos != content.npos) {
103             std::string type = content.substr(0, typePos);
104             if (type == "SwapPss") {
105                 std::string valueStr = content.substr(typePos + 1);
106                 const int base = 10;
107                 size = strtoull(valueStr.c_str(), nullptr, base);
108                 break;
109             }
110         }
111     }
112     in.close();
113     return size;
114 }
115 
116 // get graphics memory from hdi
GetGraphicsMemory(const int pid,uint64_t & gl,uint64_t & graph)117 bool GetGraphicsMemory(const int pid, uint64_t &gl, uint64_t &graph)
118 {
119     bool ret = false;
120     sptr<IMemoryTrackerInterface> memtrack = IMemoryTrackerInterface::Get(true);
121     if (memtrack == nullptr) {
122         HILOG_ERROR(LOG_CORE, "memtrack service is null");
123         return ret;
124     }
125     const std::vector<std::pair<MemoryTrackerType, std::string>> MEMORY_TRACKER_TYPES = {
126         {MEMORY_TRACKER_TYPE_GL, "GL"}, {MEMORY_TRACKER_TYPE_GRAPH, "Graph"},
127         {MEMORY_TRACKER_TYPE_OTHER, "Other"}
128     };
129 
130     for (const auto &memTrackerType : MEMORY_TRACKER_TYPES) {
131         std::vector<MemoryRecord> records;
132         if (memtrack->GetDevMem(pid, memTrackerType.first, records) != HDF_SUCCESS) {
133             continue;
134         }
135         uint64_t value = 0;
136         for (const auto &record : records) {
137             if ((static_cast<uint32_t>(record.flags) & FLAG_UNMAPPED) == FLAG_UNMAPPED) {
138                 value = static_cast<uint64_t>(record.size / BYTE_PER_KB);
139                 break;
140             }
141         }
142         if (memTrackerType.first == MEMORY_TRACKER_TYPE_GL) {
143             gl = value;
144             ret = true;
145         } else if (memTrackerType.first == MEMORY_TRACKER_TYPE_GRAPH) {
146             graph = value;
147             ret = true;
148         }
149     }
150     return ret;
151 }
152 } /* namespace MemInfo */
153 } /* namespace OHOS */
154