|  | /* | 
|  | * Copyright (C) 2014 The Android Open Source Project | 
|  | * | 
|  | * Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | * you may not use this file except in compliance with the License. | 
|  | * You may obtain a copy of the License at | 
|  | * | 
|  | *      http://www.apache.org/licenses/LICENSE-2.0 | 
|  | * | 
|  | * Unless required by applicable law or agreed to in writing, software | 
|  | * distributed under the License is distributed on an "AS IS" BASIS, | 
|  | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | * See the License for the specific language governing permissions and | 
|  | * limitations under the License. | 
|  | */ | 
|  |  | 
|  | #include "malloc_info.h" | 
|  |  | 
|  | #include <errno.h> | 
|  | #include "private/bionic_macros.h" | 
|  |  | 
|  | class __LIBC_HIDDEN__ Elem { | 
|  | public: | 
|  | // name must be valid throughout lifetime of the object. | 
|  | explicit Elem(FILE* fp, const char* name, | 
|  | const char* attr_fmt = nullptr, ...) { | 
|  | this->fp = fp; | 
|  | this->name = name; | 
|  |  | 
|  | fprintf(fp, "<%s", name); | 
|  | if (attr_fmt != nullptr) { | 
|  | va_list args; | 
|  | va_start(args, attr_fmt); | 
|  | fputc(' ', fp); | 
|  | vfprintf(fp, attr_fmt, args); | 
|  | va_end(args); | 
|  | } | 
|  | fputc('>', fp); | 
|  | } | 
|  |  | 
|  | ~Elem() noexcept { | 
|  | fprintf(fp, "</%s>", name); | 
|  | } | 
|  |  | 
|  | void contents(const char* fmt, ...) { | 
|  | va_list args; | 
|  | va_start(args, fmt); | 
|  | vfprintf(fp, fmt, args); | 
|  | va_end(args); | 
|  | } | 
|  |  | 
|  | private: | 
|  | FILE* fp; | 
|  | const char* name; | 
|  |  | 
|  | BIONIC_DISALLOW_IMPLICIT_CONSTRUCTORS(Elem); | 
|  | }; | 
|  |  | 
|  | int malloc_info(int options, FILE* fp) { | 
|  | if (options != 0) { | 
|  | errno = EINVAL; | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | Elem root(fp, "malloc", "version=\"jemalloc-1\""); | 
|  |  | 
|  | // Dump all of the large allocations in the arenas. | 
|  | for (size_t i = 0; i < __mallinfo_narenas(); i++) { | 
|  | struct mallinfo mi = __mallinfo_arena_info(i); | 
|  | if (mi.hblkhd != 0) { | 
|  | Elem arena_elem(fp, "heap", "nr=\"%d\"", i); | 
|  | { | 
|  | Elem(fp, "allocated-large").contents("%zu", mi.ordblks); | 
|  | Elem(fp, "allocated-huge").contents("%zu", mi.uordblks); | 
|  | Elem(fp, "allocated-bins").contents("%zu", mi.fsmblks); | 
|  |  | 
|  | size_t total = 0; | 
|  | for (size_t j = 0; j < __mallinfo_nbins(); j++) { | 
|  | struct mallinfo mi = __mallinfo_bin_info(i, j); | 
|  | if (mi.ordblks != 0) { | 
|  | Elem bin_elem(fp, "bin", "nr=\"%d\"", j); | 
|  | Elem(fp, "allocated").contents("%zu", mi.ordblks); | 
|  | Elem(fp, "nmalloc").contents("%zu", mi.uordblks); | 
|  | Elem(fp, "ndalloc").contents("%zu", mi.fordblks); | 
|  | total += mi.ordblks; | 
|  | } | 
|  | } | 
|  | Elem(fp, "bins-total").contents("%zu", total); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } |