1 /* 2 * Copyright (C) 2020 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 #ifndef INCLUDE_PERFETTO_PROTOZERO_MESSAGE_ARENA_H_ 18 #define INCLUDE_PERFETTO_PROTOZERO_MESSAGE_ARENA_H_ 19 20 #include <stdint.h> 21 22 #include <list> 23 #include <type_traits> 24 25 #include "perfetto/base/export.h" 26 #include "perfetto/base/logging.h" 27 #include "perfetto/protozero/message.h" 28 29 namespace protozero { 30 31 class Message; 32 33 // Object allocator for fixed-sized protozero::Message objects. 34 // It's a simple bump-pointer allocator which leverages the stack-alike 35 // usage pattern of protozero nested messages. It avoids hitting the system 36 // allocator in most cases, by reusing the same block, and falls back on 37 // allocating new blocks only when using deeply nested messages (which are 38 // extremely rare). 39 // This is used by RootMessage<T> to handle the storage for root-level messages. 40 class PERFETTO_EXPORT MessageArena { 41 public: 42 MessageArena(); 43 ~MessageArena(); 44 45 // Strictly no copies or moves as this is used to hand out pointers. 46 MessageArena(const MessageArena&) = delete; 47 MessageArena& operator=(const MessageArena&) = delete; 48 MessageArena(MessageArena&&) = delete; 49 MessageArena& operator=(MessageArena&&) = delete; 50 51 // Allocates a new Message object. 52 Message* NewMessage(); 53 54 // Deletes the last message allocated. The |msg| argument is used only for 55 // DCHECKs, it MUST be the pointer obtained by the last NewMessage() call. DeleteLastMessage(Message * msg)56 void DeleteLastMessage(Message* msg) { 57 PERFETTO_DCHECK(!blocks_.empty() && blocks_.back().entries > 0); 58 PERFETTO_DCHECK(&blocks_.back().storage[blocks_.back().entries - 1] == 59 static_cast<void*>(msg)); 60 DeleteLastMessageInternal(); 61 } 62 63 // Resets the state of the arena, clearing up all but one block. This is used 64 // to avoid leaking outstanding unfinished sub-messages while recycling the 65 // RootMessage object (this is extremely rare due to the RAII scoped handles 66 // but could happen if some client does some overly clever std::move() trick). Reset()67 void Reset() { 68 PERFETTO_DCHECK(!blocks_.empty()); 69 blocks_.resize(1); 70 auto& block = blocks_.back(); 71 block.entries = 0; 72 PERFETTO_ASAN_POISON(block.storage, sizeof(block.storage)); 73 } 74 75 private: 76 void DeleteLastMessageInternal(); 77 78 struct Block { 79 static constexpr size_t kCapacity = 16; 80 BlockBlock81 Block() { PERFETTO_ASAN_POISON(storage, sizeof(storage)); } 82 83 std::aligned_storage<sizeof(Message), alignof(Message)>::type 84 storage[kCapacity]; 85 uint32_t entries = 0; // # Message entries used (<= kCapacity). 86 }; 87 88 // blocks are used to hand out pointers and must not be moved. Hence why 89 // std::list rather than std::vector. 90 std::list<Block> blocks_; 91 }; 92 93 } // namespace protozero 94 95 #endif // INCLUDE_PERFETTO_PROTOZERO_MESSAGE_ARENA_H_ 96