1 /*
2 * Copyright (C) 2019 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 #include "src/trace_processor/types/gfp_flags.h"
18
19 #include <array>
20
21 namespace perfetto {
22 namespace trace_processor {
23
24 namespace {
25
26 struct Flag {
27 uint64_t mask;
28 const char* flag_name;
29 };
30
31 using FlagArray = std::array<Flag, 37>;
32
33 constexpr FlagArray v3_4 = {
34 {{(((0x10u) | (0x40u) | (0x80u) | (0x20000u) | (0x02u) | (0x08u)) |
35 (0x4000u) | (0x10000u) | (0x1000u) | (0x200u) | (0x400000u)),
36 "GFP_TRANSHUGE"},
37 {((0x10u) | (0x40u) | (0x80u) | (0x20000u) | (0x02u) | (0x08u)),
38 "GFP_HIGHUSER_MOVABLE"},
39 {((0x10u) | (0x40u) | (0x80u) | (0x20000u) | (0x02u)), "GFP_HIGHUSER"},
40 {((0x10u) | (0x40u) | (0x80u) | (0x20000u)), "GFP_USER"},
41 {((0x10u) | (0x40u) | (0x80u) | (0x80000u)), "GFP_TEMPORARY"},
42 {((0x10u) | (0x40u) | (0x80u)), "GFP_KERNEL"},
43 {((0x10u) | (0x40u)), "GFP_NOFS"},
44 {((0x20u)), "GFP_ATOMIC"},
45 {((0x10u)), "GFP_NOIO"},
46 {(0x20u), "GFP_HIGH"},
47 {(0x10u), "GFP_WAIT"},
48 {(0x40u), "GFP_IO"},
49 {(0x100u), "GFP_COLD"},
50 {(0x200u), "GFP_NOWARN"},
51 {(0x400u), "GFP_REPEAT"},
52 {(0x800u), "GFP_NOFAIL"},
53 {(0x1000u), "GFP_NORETRY"},
54 {(0x4000u), "GFP_COMP"},
55 {(0x8000u), "GFP_ZERO"},
56 {(0x10000u), "GFP_NOMEMALLOC"},
57 {(0x20000u), "GFP_HARDWALL"},
58 {(0x40000u), "GFP_THISNODE"},
59 {(0x80000u), "GFP_RECLAIMABLE"},
60 {(0x08u), "GFP_MOVABLE"},
61 {(0), "GFP_NOTRACK"},
62 {(0x400000u), "GFP_NO_KSWAPD"},
63 {(0x800000u), "GFP_OTHER_NODE"},
64 {0, nullptr}}};
65
66 constexpr FlagArray v3_10 = {
67 {{(((0x10u) | (0x40u) | (0x80u) | (0x20000u) | (0x02u) | (0x08u)) |
68 (0x4000u) | (0x10000u) | (0x1000u) | (0x200u) | (0x400000u)),
69 "GFP_TRANSHUGE"},
70 {((0x10u) | (0x40u) | (0x80u) | (0x20000u) | (0x02u) | (0x08u)),
71 "GFP_HIGHUSER_MOVABLE"},
72 {((0x10u) | (0x40u) | (0x80u) | (0x20000u) | (0x02u)), "GFP_HIGHUSER"},
73 {((0x10u) | (0x40u) | (0x80u) | (0x20000u)), "GFP_USER"},
74 {((0x10u) | (0x40u) | (0x80u) | (0x80000u)), "GFP_TEMPORARY"},
75 {((0x10u) | (0x40u) | (0x80u)), "GFP_KERNEL"},
76 {((0x10u) | (0x40u)), "GFP_NOFS"},
77 {((0x20u)), "GFP_ATOMIC"},
78 {((0x10u)), "GFP_NOIO"},
79 {(0x20u), "GFP_HIGH"},
80 {(0x10u), "GFP_WAIT"},
81 {(0x40u), "GFP_IO"},
82 {(0x100u), "GFP_COLD"},
83 {(0x200u), "GFP_NOWARN"},
84 {(0x400u), "GFP_REPEAT"},
85 {(0x800u), "GFP_NOFAIL"},
86 {(0x1000u), "GFP_NORETRY"},
87 {(0x4000u), "GFP_COMP"},
88 {(0x8000u), "GFP_ZERO"},
89 {(0x10000u), "GFP_NOMEMALLOC"},
90 {(0x2000u), "GFP_MEMALLOC"},
91 {(0x20000u), "GFP_HARDWALL"},
92 {(0x40000u), "GFP_THISNODE"},
93 {(0x80000u), "GFP_RECLAIMABLE"},
94 {(0x100000u), "GFP_KMEMCG"},
95 {(0x08u), "GFP_MOVABLE"},
96 {(0x200000u), "GFP_NOTRACK"},
97 {(0x400000u), "GFP_NO_KSWAPD"},
98 {(0x800000u), "GFP_OTHER_NODE"},
99 {0, nullptr}}};
100
101 constexpr FlagArray v4_4 = {
102 {{(((((((0x400000u | 0x2000000u)) | (0x40u) | (0x80u) | (0x20000u)) |
103 (0x02u)) |
104 (0x08u)) |
105 (0x4000u) | (0x10000u) | (0x1000u) | (0x200u)) &
106 ~(0x2000000u)),
107 "GFP_TRANSHUGE"},
108 {(((((0x400000u | 0x2000000u)) | (0x40u) | (0x80u) | (0x20000u)) |
109 (0x02u)) |
110 (0x08u)),
111 "GFP_HIGHUSER_MOVABLE"},
112 {((((0x400000u | 0x2000000u)) | (0x40u) | (0x80u) | (0x20000u)) | (0x02u)),
113 "GFP_HIGHUSER"},
114 {(((0x400000u | 0x2000000u)) | (0x40u) | (0x80u) | (0x20000u)),
115 "GFP_USER"},
116 {(((0x400000u | 0x2000000u)) | (0x40u) | (0x80u) | (0x10u)),
117 "GFP_TEMPORARY"},
118 {(((0x400000u | 0x2000000u)) | (0x40u) | (0x80u)), "GFP_KERNEL"},
119 {(((0x400000u | 0x2000000u)) | (0x40u)), "GFP_NOFS"},
120 {((0x20u) | (0x80000u) | (0x2000000u)), "GFP_ATOMIC"},
121 {(((0x400000u | 0x2000000u))), "GFP_NOIO"},
122 {(0x20u), "GFP_HIGH"},
123 {(0x80000u), "GFP_ATOMIC"},
124 {(0x40u), "GFP_IO"},
125 {(0x100u), "GFP_COLD"},
126 {(0x200u), "GFP_NOWARN"},
127 {(0x400u), "GFP_REPEAT"},
128 {(0x800u), "GFP_NOFAIL"},
129 {(0x1000u), "GFP_NORETRY"},
130 {(0x4000u), "GFP_COMP"},
131 {(0x8000u), "GFP_ZERO"},
132 {(0x10000u), "GFP_NOMEMALLOC"},
133 {(0x2000u), "GFP_MEMALLOC"},
134 {(0x20000u), "GFP_HARDWALL"},
135 {(0x40000u), "GFP_THISNODE"},
136 {(0x10u), "GFP_RECLAIMABLE"},
137 {(0x08u), "GFP_MOVABLE"},
138 {(0x200000u), "GFP_NOTRACK"},
139 {(0x400000u), "GFP_DIRECT_RECLAIM"},
140 {(0x2000000u), "GFP_KSWAPD_RECLAIM"},
141 {(0x800000u), "GFP_OTHER_NODE"},
142 {0, nullptr}}};
143
144 constexpr FlagArray v4_14 = {
145 {{((((((((0x400000u | 0x1000000u)) | (0x40u) | (0x80u) | (0x20000u)) |
146 (0x02u)) |
147 (0x08u)) |
148 (0x4000u) | (0x10000u) | (0x200u)) &
149 ~((0x400000u | 0x1000000u))) |
150 (0x400000u)),
151 "GFP_TRANSHUGE"},
152 {(((((((0x400000u | 0x1000000u)) | (0x40u) | (0x80u) | (0x20000u)) |
153 (0x02u)) |
154 (0x08u)) |
155 (0x4000u) | (0x10000u) | (0x200u)) &
156 ~((0x400000u | 0x1000000u))),
157 "GFP_TRANSHUGE_LIGHT"},
158 {(((((0x400000u | 0x1000000u)) | (0x40u) | (0x80u) | (0x20000u)) |
159 (0x02u)) |
160 (0x08u)),
161 "GFP_HIGHUSER_MOVABLE"},
162 {((((0x400000u | 0x1000000u)) | (0x40u) | (0x80u) | (0x20000u)) | (0x02u)),
163 "GFP_HIGHUSER"},
164 {(((0x400000u | 0x1000000u)) | (0x40u) | (0x80u) | (0x20000u)),
165 "GFP_USER"},
166 {((((0x400000u | 0x1000000u)) | (0x40u) | (0x80u)) | (0x100000u)),
167 "GFP_KERNEL_ACCOUNT"},
168 {(((0x400000u | 0x1000000u)) | (0x40u) | (0x80u)), "GFP_KERNEL"},
169 {(((0x400000u | 0x1000000u)) | (0x40u)), "GFP_NOFS"},
170 {((0x20u) | (0x80000u) | (0x1000000u)), "GFP_ATOMIC"},
171 {(((0x400000u | 0x1000000u))), "GFP_NOIO"},
172 {((0x1000000u)), "GFP_NOWAIT"},
173 {(0x01u), "GFP_DMA"},
174 {(0x02u), "__GFP_HIGHMEM"},
175 {(0x04u), "GFP_DMA32"},
176 {(0x20u), "__GFP_HIGH"},
177 {(0x80000u), "__GFP_ATOMIC"},
178 {(0x40u), "__GFP_IO"},
179 {(0x80u), "__GFP_FS"},
180 {(0x100u), "__GFP_COLD"},
181 {(0x200u), "__GFP_NOWARN"},
182 {(0x400u), "__GFP_RETRY_MAYFAIL"},
183 {(0x800u), "__GFP_NOFAIL"},
184 {(0x1000u), "__GFP_NORETRY"},
185 {(0x4000u), "__GFP_COMP"},
186 {(0x8000u), "__GFP_ZERO"},
187 {(0x10000u), "__GFP_NOMEMALLOC"},
188 {(0x2000u), "__GFP_MEMALLOC"},
189 {(0x20000u), "__GFP_HARDWALL"},
190 {(0x40000u), "__GFP_THISNODE"},
191 {(0x10u), "__GFP_RECLAIMABLE"},
192 {(0x08u), "__GFP_MOVABLE"},
193 {(0x100000u), "__GFP_ACCOUNT"},
194 {(0x800000u), "__GFP_WRITE"},
195 {((0x400000u | 0x1000000u)), "__GFP_RECLAIM"},
196 {(0x400000u), "__GFP_DIRECT_RECLAIM"},
197 {(0x1000000u), "__GFP_KSWAPD_RECLAIM"},
198 {0, nullptr}}};
199
200 // Get the bitmask closest to the kernel version. For versions less than 3.4
201 // and greater than 4.14 this may end up being inaccurate.
GetBitmaskVersion(VersionNumber version=VersionNumber{4, 4})202 const FlagArray* GetBitmaskVersion(VersionNumber version = VersionNumber{4,
203 4}) {
204 if (version < VersionNumber{3, 10}) {
205 return &v3_4;
206 } else if (version >= VersionNumber{3, 10} && version < VersionNumber{4, 4}) {
207 return &v3_10;
208 } else if (version >= VersionNumber{4, 4} && version < VersionNumber{4, 14}) {
209 return &v4_4;
210 } else { // version >= 4.14
211 // TODO(hjd): Add newer kernel versions once we have access to them.
212 return &v4_14;
213 }
214 }
215 } // namespace
216
WriteGfpFlag(uint64_t value,base::Optional<VersionNumber> version,base::StringWriter * writer)217 void WriteGfpFlag(uint64_t value,
218 base::Optional<VersionNumber> version,
219 base::StringWriter* writer) {
220 // On all kernel versions if this flag is not set, return GFP_NOWAIT.
221 if (value == 0) {
222 writer->AppendString("GFP_NOWAIT");
223 return;
224 }
225
226 std::string result;
227 const FlagArray* bitmasks = version.has_value()
228 ? GetBitmaskVersion(version.value())
229 : GetBitmaskVersion();
230
231 // Based on trace_print_flags_seq() in the kernel.
232 size_t i = 0;
233 while (bitmasks->at(i).flag_name != nullptr) {
234 size_t current = i++;
235 uint64_t mask = bitmasks->at(current).mask;
236 const char* str = bitmasks->at(current).flag_name;
237
238 if ((value & mask) != mask)
239 continue;
240 value &= ~mask;
241
242 result += str;
243 result += "|";
244 }
245
246 // Add any leftover flags.
247 if (value) {
248 writer->AppendString(result.c_str(), result.size());
249 writer->AppendString("0x", 2);
250 writer->AppendHexInt(value);
251 } else {
252 writer->AppendString(result.c_str(), result.size() - 1);
253 }
254 }
255
256 } // namespace trace_processor
257 } // namespace perfetto
258