/* * Copyright (C) 2022 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 #include #include #include "AllocParser.h" #include void AllocGetData(const std::string& line, AllocEntry* entry) { int op_prefix_pos = 0; char name[128]; // All lines have this format: // TID: ALLOCATION_TYPE POINTER // where // TID is the thread id of the thread doing the operation. // ALLOCATION_TYPE is one of malloc, calloc, memalign, realloc, free, thread_done // POINTER is the hex value of the actual pointer if (sscanf(line.c_str(), "%d: %127s %" SCNx64 " %n", &entry->tid, name, &entry->ptr, &op_prefix_pos) != 3) { errx(1, "File Error: Failed to process %s", line.c_str()); } std::string type(name); if (type == "thread_done") { entry->type = THREAD_DONE; } else { int args_offset = 0; const char* args_beg = &line[op_prefix_pos]; if (type == "malloc") { // Format: // TID: malloc POINTER SIZE_OF_ALLOCATION if (sscanf(args_beg, "%zu%n", &entry->size, &args_offset) != 1) { errx(1, "File Error: Failed to read malloc data %s", line.c_str()); } entry->type = MALLOC; } else if (type == "free") { // Format: // TID: free POINTER entry->type = FREE; } else if (type == "calloc") { // Format: // TID: calloc POINTER ITEM_COUNT ITEM_SIZE if (sscanf(args_beg, "%" SCNd64 " %zu%n", &entry->u.n_elements, &entry->size, &args_offset) != 2) { errx(1, "File Error: Failed to read calloc data %s", line.c_str()); } entry->type = CALLOC; } else if (type == "realloc") { // Format: // TID: realloc POINTER OLD_POINTER NEW_SIZE if (sscanf(args_beg, "%" SCNx64 " %zu%n", &entry->u.old_ptr, &entry->size, &args_offset) != 2) { errx(1, "File Error: Failed to read realloc data %s", line.c_str()); } entry->type = REALLOC; } else if (type == "memalign") { // Format: // TID: memalign POINTER ALIGNMENT SIZE if (sscanf(args_beg, "%" SCNd64 " %zu%n", &entry->u.align, &entry->size, &args_offset) != 2) { errx(1, "File Error: Failed to read memalign data %s", line.c_str()); } entry->type = MEMALIGN; } else { errx(1, "File Error: Unknown type %s", type.c_str()); } const char* timestamps_beg = &args_beg[args_offset]; // Timestamps come after the alloc args if present, for example, // TID: malloc POINTER SIZE_OF_ALLOCATION START_TIME END_TIME int n_match = sscanf(timestamps_beg, "%" SCNd64 " %" SCNd64, &entry->st, &entry->et); if (n_match != EOF && n_match != 2) { errx(1, "File Error: Failed to read timestamps %s", line.c_str()); } } }