LCOV - code coverage report
Current view: top level - src/test - allocator_tests.cpp (source / functions) Coverage Total Hit
Test: total_coverage.info Lines: 98.6 % 144 142
Test Date: 2025-01-19 05:08:01 Functions: 100.0 % 9 9
Branches: 51.8 % 610 316

             Branch data     Line data    Source code
       1                 :             : // Copyright (c) 2012-2021 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                 :             : #include <common/system.h>
       6                 :             : #include <support/lockedpool.h>
       7                 :             : 
       8                 :             : #include <limits>
       9                 :             : #include <memory>
      10                 :             : #include <stdexcept>
      11                 :             : #include <utility>
      12                 :             : #include <vector>
      13                 :             : 
      14                 :             : #include <boost/test/unit_test.hpp>
      15                 :             : 
      16                 :             : BOOST_AUTO_TEST_SUITE(allocator_tests)
      17                 :             : 
      18   [ +  -  +  -  :           7 : BOOST_AUTO_TEST_CASE(arena_tests)
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
                      - ]
      19                 :             : {
      20                 :             :     // Fake memory base address for testing
      21                 :             :     // without actually using memory.
      22                 :           1 :     void *synth_base = reinterpret_cast<void*>(0x08000000);
      23                 :           1 :     const size_t synth_size = 1024*1024;
      24                 :           1 :     Arena b(synth_base, synth_size, 16);
      25         [ +  - ]:           1 :     void *chunk = b.alloc(1000);
      26                 :             : #ifdef ARENA_DEBUG
      27                 :             :     b.walk();
      28                 :             : #endif
      29   [ +  -  +  -  :           2 :     BOOST_CHECK(chunk != nullptr);
                   +  - ]
      30   [ +  -  +  -  :           2 :     BOOST_CHECK(b.stats().used == 1008); // Aligned to 16
             +  -  +  - ]
      31   [ +  -  +  -  :           2 :     BOOST_CHECK(b.stats().total == synth_size); // Nothing has disappeared?
             +  -  +  - ]
      32         [ +  - ]:           1 :     b.free(chunk);
      33                 :             : #ifdef ARENA_DEBUG
      34                 :             :     b.walk();
      35                 :             : #endif
      36   [ +  -  +  -  :           2 :     BOOST_CHECK(b.stats().used == 0);
             +  -  +  - ]
      37   [ +  -  +  -  :           2 :     BOOST_CHECK(b.stats().free == synth_size);
             +  -  -  + ]
      38                 :           1 :     try { // Test exception on double-free
      39         [ -  + ]:           1 :         b.free(chunk);
      40   [ #  #  #  # ]:           0 :         BOOST_CHECK(0);
      41         [ -  + ]:           1 :     } catch(std::runtime_error &)
      42                 :             :     {
      43                 :           1 :     }
      44                 :             : 
      45         [ +  - ]:           1 :     void *a0 = b.alloc(128);
      46         [ +  - ]:           1 :     void *a1 = b.alloc(256);
      47         [ +  - ]:           1 :     void *a2 = b.alloc(512);
      48   [ +  -  +  -  :           2 :     BOOST_CHECK(b.stats().used == 896);
             +  -  +  - ]
      49   [ +  -  +  -  :           2 :     BOOST_CHECK(b.stats().total == synth_size);
             +  -  +  - ]
      50                 :             : #ifdef ARENA_DEBUG
      51                 :             :     b.walk();
      52                 :             : #endif
      53         [ +  - ]:           1 :     b.free(a0);
      54                 :             : #ifdef ARENA_DEBUG
      55                 :             :     b.walk();
      56                 :             : #endif
      57   [ +  -  +  -  :           2 :     BOOST_CHECK(b.stats().used == 768);
             +  -  +  - ]
      58         [ +  - ]:           1 :     b.free(a1);
      59   [ +  -  +  -  :           2 :     BOOST_CHECK(b.stats().used == 512);
             +  -  +  - ]
      60         [ +  - ]:           1 :     void *a3 = b.alloc(128);
      61                 :             : #ifdef ARENA_DEBUG
      62                 :             :     b.walk();
      63                 :             : #endif
      64   [ +  -  +  -  :           2 :     BOOST_CHECK(b.stats().used == 640);
             +  -  +  - ]
      65         [ +  - ]:           1 :     b.free(a2);
      66   [ +  -  +  -  :           2 :     BOOST_CHECK(b.stats().used == 128);
             +  -  +  - ]
      67         [ +  - ]:           1 :     b.free(a3);
      68   [ +  -  +  -  :           2 :     BOOST_CHECK(b.stats().used == 0);
             +  -  +  - ]
      69   [ +  -  +  -  :           1 :     BOOST_CHECK_EQUAL(b.stats().chunks_used, 0U);
                   +  - ]
      70   [ +  -  +  -  :           2 :     BOOST_CHECK(b.stats().total == synth_size);
             +  -  +  - ]
      71   [ +  -  +  -  :           2 :     BOOST_CHECK(b.stats().free == synth_size);
             +  -  +  - ]
      72   [ +  -  +  -  :           1 :     BOOST_CHECK_EQUAL(b.stats().chunks_free, 1U);
                   +  - ]
      73                 :             : 
      74                 :           1 :     std::vector<void*> addr;
      75   [ +  -  +  -  :           2 :     BOOST_CHECK(b.alloc(0) == nullptr); // allocating 0 always returns nullptr
             +  -  +  - ]
      76                 :             : #ifdef ARENA_DEBUG
      77                 :             :     b.walk();
      78                 :             : #endif
      79                 :             :     // Sweeping allocate all memory
      80         [ +  - ]:           1 :     addr.reserve(2048);
      81         [ +  + ]:        1025 :     for (int x=0; x<1024; ++x)
      82   [ +  -  +  - ]:        1024 :         addr.push_back(b.alloc(1024));
      83   [ +  -  +  -  :           2 :     BOOST_CHECK(b.stats().free == 0);
          +  -  +  -  +  
                      - ]
      84   [ +  -  +  -  :           2 :     BOOST_CHECK(b.alloc(1024) == nullptr); // memory is full, this must return nullptr
             +  -  +  - ]
      85   [ +  -  +  -  :           2 :     BOOST_CHECK(b.alloc(0) == nullptr);
                   +  - ]
      86         [ +  + ]:        1025 :     for (int x=0; x<1024; ++x)
      87         [ +  - ]:        1024 :         b.free(addr[x]);
      88         [ +  - ]:           1 :     addr.clear();
      89   [ +  -  +  -  :           2 :     BOOST_CHECK(b.stats().total == synth_size);
             +  -  +  - ]
      90   [ +  -  +  -  :           2 :     BOOST_CHECK(b.stats().free == synth_size);
                   +  - ]
      91                 :             : 
      92                 :             :     // Now in the other direction...
      93         [ +  + ]:        1025 :     for (int x=0; x<1024; ++x)
      94   [ +  -  +  - ]:        1024 :         addr.push_back(b.alloc(1024));
      95         [ +  + ]:        1025 :     for (int x=0; x<1024; ++x)
      96         [ +  - ]:        1024 :         b.free(addr[1023-x]);
      97         [ +  - ]:           1 :     addr.clear();
      98                 :             : 
      99                 :             :     // Now allocate in smaller unequal chunks, then deallocate haphazardly
     100                 :             :     // Not all the chunks will succeed allocating, but freeing nullptr is
     101                 :             :     // allowed so that is no problem.
     102                 :        2049 :     for (int x=0; x<2048; ++x)
     103   [ +  -  +  -  :        2049 :         addr.push_back(b.alloc(x+1));
                   +  + ]
     104         [ +  + ]:        2049 :     for (int x=0; x<2048; ++x)
     105         [ +  - ]:        2048 :         b.free(addr[((x*23)%2048)^242]);
     106         [ +  - ]:           1 :     addr.clear();
     107                 :             : 
     108                 :             :     // Go entirely wild: free and alloc interleaved,
     109                 :             :     // generate targets and sizes using pseudo-randomness.
     110         [ +  + ]:        2049 :     for (int x=0; x<2048; ++x)
     111         [ +  - ]:        2048 :         addr.push_back(nullptr);
     112                 :             :     uint32_t s = 0x12345678;
     113         [ +  + ]:        5001 :     for (int x=0; x<5000; ++x) {
     114         [ +  + ]:        5000 :         int idx = s & (addr.size()-1);
     115         [ +  + ]:        5000 :         if (s & 0x80000000) {
     116         [ +  - ]:        2458 :             b.free(addr[idx]);
     117                 :        2458 :             addr[idx] = nullptr;
     118         [ +  + ]:        2542 :         } else if(!addr[idx]) {
     119         [ +  - ]:        1741 :             addr[idx] = b.alloc((s >> 16) & 2047);
     120                 :             :         }
     121                 :        5000 :         bool lsb = s & 1;
     122                 :        5000 :         s >>= 1;
     123         [ +  + ]:        5000 :         if (lsb)
     124                 :        2458 :             s ^= 0xf00f00f0; // LFSR period 0xf7ffffe0
     125                 :             :     }
     126         [ +  + ]:        2049 :     for (void *ptr: addr)
     127         [ +  - ]:        2048 :         b.free(ptr);
     128         [ +  - ]:           1 :     addr.clear();
     129                 :             : 
     130   [ +  -  +  -  :           2 :     BOOST_CHECK(b.stats().total == synth_size);
             +  -  +  - ]
     131   [ +  -  +  -  :           2 :     BOOST_CHECK(b.stats().free == synth_size);
                   +  - ]
     132                 :           1 : }
     133                 :             : 
     134                 :             : /** Mock LockedPageAllocator for testing */
     135                 :             : class TestLockedPageAllocator: public LockedPageAllocator
     136                 :             : {
     137                 :             : public:
     138                 :           1 :     TestLockedPageAllocator(int count_in, int lockedcount_in): count(count_in), lockedcount(lockedcount_in) {}
     139                 :           4 :     void* AllocateLocked(size_t len, bool *lockingSuccess) override
     140                 :             :     {
     141                 :           4 :         *lockingSuccess = false;
     142         [ +  + ]:           4 :         if (count > 0) {
     143                 :           3 :             --count;
     144                 :             : 
     145         [ +  + ]:           3 :             if (lockedcount > 0) {
     146                 :           1 :                 --lockedcount;
     147                 :           1 :                 *lockingSuccess = true;
     148                 :             :             }
     149                 :             : 
     150                 :           3 :             return reinterpret_cast<void*>(uint64_t{static_cast<uint64_t>(0x08000000) + (count << 24)}); // Fake address, do not actually use this memory
     151                 :             :         }
     152                 :             :         return nullptr;
     153                 :             :     }
     154                 :           3 :     void FreeLocked(void* addr, size_t len) override
     155                 :             :     {
     156                 :           3 :     }
     157                 :           1 :     size_t GetLimit() override
     158                 :             :     {
     159                 :           1 :         return std::numeric_limits<size_t>::max();
     160                 :             :     }
     161                 :             : private:
     162                 :             :     int count;
     163                 :             :     int lockedcount;
     164                 :             : };
     165                 :             : 
     166   [ +  -  +  -  :           7 : BOOST_AUTO_TEST_CASE(lockedpool_tests_mock)
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
                      - ]
     167                 :             : {
     168                 :             :     // Test over three virtual arenas, of which one will succeed being locked
     169                 :           1 :     std::unique_ptr<LockedPageAllocator> x = std::make_unique<TestLockedPageAllocator>(3, 1);
     170         [ +  - ]:           1 :     LockedPool pool(std::move(x));
     171   [ +  -  +  -  :           2 :     BOOST_CHECK(pool.stats().total == 0);
             +  -  +  - ]
     172   [ +  -  +  -  :           2 :     BOOST_CHECK(pool.stats().locked == 0);
             +  -  +  - ]
     173                 :             : 
     174                 :             :     // Ensure unreasonable requests are refused without allocating anything
     175         [ +  - ]:           1 :     void *invalid_toosmall = pool.alloc(0);
     176   [ +  -  +  -  :           2 :     BOOST_CHECK(invalid_toosmall == nullptr);
                   +  - ]
     177   [ +  -  +  -  :           2 :     BOOST_CHECK(pool.stats().used == 0);
             +  -  +  - ]
     178   [ +  -  +  -  :           2 :     BOOST_CHECK(pool.stats().free == 0);
             +  -  +  - ]
     179         [ +  - ]:           1 :     void *invalid_toobig = pool.alloc(LockedPool::ARENA_SIZE+1);
     180   [ +  -  +  -  :           2 :     BOOST_CHECK(invalid_toobig == nullptr);
                   +  - ]
     181   [ +  -  +  -  :           2 :     BOOST_CHECK(pool.stats().used == 0);
             +  -  +  - ]
     182   [ +  -  +  -  :           2 :     BOOST_CHECK(pool.stats().free == 0);
             +  -  +  - ]
     183                 :             : 
     184         [ +  - ]:           1 :     void *a0 = pool.alloc(LockedPool::ARENA_SIZE / 2);
     185   [ +  -  +  -  :           2 :     BOOST_CHECK(a0);
                   +  - ]
     186   [ +  -  +  -  :           2 :     BOOST_CHECK(pool.stats().locked == LockedPool::ARENA_SIZE);
             +  -  +  - ]
     187         [ +  - ]:           1 :     void *a1 = pool.alloc(LockedPool::ARENA_SIZE / 2);
     188   [ +  -  +  -  :           2 :     BOOST_CHECK(a1);
                   +  - ]
     189         [ +  - ]:           1 :     void *a2 = pool.alloc(LockedPool::ARENA_SIZE / 2);
     190   [ +  -  +  -  :           2 :     BOOST_CHECK(a2);
                   +  - ]
     191         [ +  - ]:           1 :     void *a3 = pool.alloc(LockedPool::ARENA_SIZE / 2);
     192   [ +  -  +  -  :           2 :     BOOST_CHECK(a3);
                   +  - ]
     193         [ +  - ]:           1 :     void *a4 = pool.alloc(LockedPool::ARENA_SIZE / 2);
     194   [ +  -  +  -  :           2 :     BOOST_CHECK(a4);
                   +  - ]
     195         [ +  - ]:           1 :     void *a5 = pool.alloc(LockedPool::ARENA_SIZE / 2);
     196   [ +  -  +  -  :           2 :     BOOST_CHECK(a5);
                   +  - ]
     197                 :             :     // We've passed a count of three arenas, so this allocation should fail
     198         [ +  - ]:           1 :     void *a6 = pool.alloc(16);
     199   [ +  -  +  -  :           2 :     BOOST_CHECK(!a6);
                   +  - ]
     200                 :             : 
     201         [ +  - ]:           1 :     pool.free(a0);
     202         [ +  - ]:           1 :     pool.free(a2);
     203         [ +  - ]:           1 :     pool.free(a4);
     204         [ +  - ]:           1 :     pool.free(a1);
     205         [ +  - ]:           1 :     pool.free(a3);
     206         [ +  - ]:           1 :     pool.free(a5);
     207   [ +  -  +  -  :           2 :     BOOST_CHECK(pool.stats().total == 3*LockedPool::ARENA_SIZE);
             +  -  +  - ]
     208   [ +  -  +  -  :           2 :     BOOST_CHECK(pool.stats().locked == LockedPool::ARENA_SIZE);
             +  -  +  - ]
     209   [ +  -  +  -  :           2 :     BOOST_CHECK(pool.stats().used == 0);
                   +  - ]
     210                 :           1 : }
     211                 :             : 
     212                 :             : // These tests used the live LockedPoolManager object, this is also used
     213                 :             : // by other tests so the conditions are somewhat less controllable and thus the
     214                 :             : // tests are somewhat more error-prone.
     215   [ +  -  +  -  :           7 : BOOST_AUTO_TEST_CASE(lockedpool_tests_live)
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
                      - ]
     216                 :             : {
     217                 :           1 :     LockedPoolManager &pool = LockedPoolManager::Instance();
     218                 :           1 :     LockedPool::Stats initial = pool.stats();
     219                 :             : 
     220                 :           1 :     void *a0 = pool.alloc(16);
     221         [ +  - ]:           2 :     BOOST_CHECK(a0);
     222                 :             :     // Test reading and writing the allocated memory
     223                 :           1 :     *((uint32_t*)a0) = 0x1234;
     224         [ +  - ]:           2 :     BOOST_CHECK(*((uint32_t*)a0) == 0x1234);
     225                 :             : 
     226                 :           1 :     pool.free(a0);
     227                 :           1 :     try { // Test exception on double-free
     228         [ -  + ]:           1 :         pool.free(a0);
     229   [ #  #  #  # ]:           0 :         BOOST_CHECK(0);
     230         [ -  + ]:           1 :     } catch(std::runtime_error &)
     231                 :             :     {
     232                 :           1 :     }
     233                 :             :     // If more than one new arena was allocated for the above tests, something is wrong
     234   [ +  -  +  - ]:           2 :     BOOST_CHECK(pool.stats().total <= (initial.total + LockedPool::ARENA_SIZE));
     235                 :             :     // Usage must be back to where it started
     236   [ +  -  +  - ]:           2 :     BOOST_CHECK(pool.stats().used == initial.used);
     237                 :           1 : }
     238                 :             : 
     239                 :             : BOOST_AUTO_TEST_SUITE_END()
        

Generated by: LCOV version 2.0-1