LCOV - code coverage report
Current view: top level - src/test/util - poolresourcetester.h (source / functions) Coverage Total Hit
Test: total_coverage.info Lines: 98.1 % 52 51
Test Date: 2025-01-19 05:08:01 Functions: 100.0 % 5 5
Branches: 68.9 % 90 62

             Branch data     Line data    Source code
       1                 :             : // Copyright (c) 2022 The Bitcoin Core developers
       2                 :             : // Distributed under the MIT software license, see the accompanying
       3                 :             : // file COPYING or http://www.opensource.org/licenses/mit-license.php.
       4                 :             : 
       5                 :             : #ifndef BITCOIN_TEST_UTIL_POOLRESOURCETESTER_H
       6                 :             : #define BITCOIN_TEST_UTIL_POOLRESOURCETESTER_H
       7                 :             : 
       8                 :             : #include <support/allocators/pool.h>
       9                 :             : 
      10                 :             : #include <algorithm>
      11                 :             : #include <cassert>
      12                 :             : #include <cstddef>
      13                 :             : #include <cstdint>
      14                 :             : #include <vector>
      15                 :             : 
      16                 :             : /**
      17                 :             :  * Helper to get access to private parts of PoolResource. Used in unit tests and in the fuzzer
      18                 :             :  */
      19                 :             : class PoolResourceTester
      20                 :             : {
      21                 :             :     struct PtrAndBytes {
      22                 :             :         uintptr_t ptr;
      23                 :             :         std::size_t size;
      24                 :             : 
      25                 :       11820 :         PtrAndBytes(const void* p, std::size_t s)
      26                 :       11820 :             : ptr(reinterpret_cast<uintptr_t>(p)), size(s)
      27                 :             :         {
      28                 :             :         }
      29                 :             : 
      30                 :             :         /**
      31                 :             :          * defines a sort ordering by the pointer value
      32                 :             :          */
      33                 :      432157 :         friend bool operator<(PtrAndBytes const& a, PtrAndBytes const& b)
      34                 :             :         {
      35   [ +  +  +  +  :      431880 :             return a.ptr < b.ptr;
          +  +  +  +  +  
          +  +  +  +  +  
          +  +  +  +  +  
             +  -  -  +  
                      + ]
      36                 :             :         }
      37                 :             :     };
      38                 :             : 
      39                 :             : public:
      40                 :             :     /**
      41                 :             :      * Extracts the number of elements per freelist
      42                 :             :      */
      43                 :             :     template <std::size_t MAX_BLOCK_SIZE_BYTES, std::size_t ALIGN_BYTES>
      44                 :          10 :     static std::vector<std::size_t> FreeListSizes(const PoolResource<MAX_BLOCK_SIZE_BYTES, ALIGN_BYTES>& resource)
      45                 :             :     {
      46                 :          10 :         auto sizes = std::vector<std::size_t>();
      47         [ +  + ]:          30 :         for (const auto* ptr : resource.m_free_lists) {
      48                 :          20 :             size_t size = 0;
      49         [ +  + ]:          27 :             while (ptr != nullptr) {
      50                 :           7 :                 ++size;
      51                 :           7 :                 ptr = ptr->m_next;
      52                 :             :             }
      53         [ +  - ]:          20 :             sizes.push_back(size);
      54                 :             :         }
      55                 :          10 :         return sizes;
      56                 :           0 :     }
      57                 :             : 
      58                 :             :     /**
      59                 :             :      * How many bytes are still available from the last allocated chunk
      60                 :             :      */
      61                 :             :     template <std::size_t MAX_BLOCK_SIZE_BYTES, std::size_t ALIGN_BYTES>
      62                 :          10 :     static std::size_t AvailableMemoryFromChunk(const PoolResource<MAX_BLOCK_SIZE_BYTES, ALIGN_BYTES>& resource)
      63                 :             :     {
      64   [ +  -  +  -  :          10 :         return resource.m_available_memory_end - resource.m_available_memory_it;
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
                      - ]
      65                 :             :     }
      66                 :             : 
      67                 :             :     /**
      68                 :             :      * Once all blocks are given back to the resource, tests that the freelists are consistent:
      69                 :             :      *
      70                 :             :      * * All data in the freelists must come from the chunks
      71                 :             :      * * Memory doesn't overlap
      72                 :             :      * * Each byte in the chunks can be accounted for in either the freelist or as available bytes.
      73                 :             :      */
      74                 :             :     template <std::size_t MAX_BLOCK_SIZE_BYTES, std::size_t ALIGN_BYTES>
      75                 :          14 :     static void CheckAllDataAccountedFor(const PoolResource<MAX_BLOCK_SIZE_BYTES, ALIGN_BYTES>& resource)
      76                 :             :     {
      77                 :             :         // collect all free blocks by iterating all freelists
      78                 :          14 :         std::vector<PtrAndBytes> free_blocks;
      79         [ +  + ]:         116 :         for (std::size_t freelist_idx = 0; freelist_idx < resource.m_free_lists.size(); ++freelist_idx) {
      80                 :         102 :             std::size_t bytes = freelist_idx * resource.ELEM_ALIGN_BYTES;
      81                 :         102 :             auto* ptr = resource.m_free_lists[freelist_idx];
      82         [ +  + ]:       11648 :             while (ptr != nullptr) {
      83         [ +  - ]:       11546 :                 free_blocks.emplace_back(ptr, bytes);
      84                 :       11546 :                 ptr = ptr->m_next;
      85                 :             :             }
      86                 :             :         }
      87                 :             :         // also add whatever has not yet been used for blocks
      88                 :          14 :         auto num_available_bytes = resource.m_available_memory_end - resource.m_available_memory_it;
      89         [ +  - ]:          14 :         if (num_available_bytes > 0) {
      90         [ +  - ]:          14 :             free_blocks.emplace_back(resource.m_available_memory_it, num_available_bytes);
      91                 :             :         }
      92                 :             : 
      93                 :             :         // collect all chunks
      94                 :          14 :         std::vector<PtrAndBytes> chunks;
      95   [ +  -  +  + ]:         274 :         for (const std::byte* ptr : resource.m_allocated_chunks) {
      96         [ +  - ]:         260 :             chunks.emplace_back(ptr, resource.ChunkSizeBytes());
      97                 :             :         }
      98                 :             : 
      99                 :             :         // now we have all the data from all freelists on the one hand side, and all chunks on the other hand side.
     100                 :             :         // To check if all of them match, sort by address and iterate.
     101                 :          14 :         std::sort(free_blocks.begin(), free_blocks.end());
     102                 :          14 :         std::sort(chunks.begin(), chunks.end());
     103                 :             : 
     104                 :          14 :         auto chunk_it = chunks.begin();
     105                 :          14 :         auto chunk_ptr_remaining = chunk_it->ptr;
     106                 :          14 :         auto chunk_size_remaining = chunk_it->size;
     107         [ +  + ]:       11574 :         for (const auto& free_block : free_blocks) {
     108         [ +  + ]:       11560 :             if (chunk_size_remaining == 0) {
     109         [ -  + ]:         246 :                 assert(chunk_it != chunks.end());
     110         [ -  + ]:         246 :                 ++chunk_it;
     111         [ -  + ]:         246 :                 assert(chunk_it != chunks.end());
     112                 :         246 :                 chunk_ptr_remaining = chunk_it->ptr;
     113                 :         246 :                 chunk_size_remaining = chunk_it->size;
     114                 :             :             }
     115         [ -  + ]:       11560 :             assert(free_block.ptr == chunk_ptr_remaining);                   // ensure addresses match
     116         [ -  + ]:       11560 :             assert(free_block.size <= chunk_size_remaining);                 // ensure no overflow
     117         [ -  + ]:       11560 :             assert((free_block.ptr & (resource.ELEM_ALIGN_BYTES - 1)) == 0); // ensure correct alignment
     118                 :       11560 :             chunk_ptr_remaining += free_block.size;
     119                 :       11560 :             chunk_size_remaining -= free_block.size;
     120                 :             :         }
     121                 :             :         // ensure we are at the end of the chunks
     122         [ -  + ]:          14 :         assert(chunk_ptr_remaining == chunk_it->ptr + chunk_it->size);
     123         [ -  + ]:          14 :         ++chunk_it;
     124         [ -  + ]:          14 :         assert(chunk_it == chunks.end());
     125         [ -  + ]:          14 :         assert(chunk_size_remaining == 0);
     126                 :          14 :     }
     127                 :             : };
     128                 :             : 
     129                 :             : #endif // BITCOIN_TEST_UTIL_POOLRESOURCETESTER_H
        

Generated by: LCOV version 2.0-1