LCOV - code coverage report
Current view: top level - src/test/fuzz - vecdeque.cpp (source / functions) Coverage Total Hit
Test: fuzz_coverage.info Lines: 100.0 % 276 276
Test Date: 2025-01-22 04:09:46 Functions: 100.0 % 58 58
Branches: 69.0 % 562 388

             Branch data     Line data    Source code
       1                 :             : // Copyright (c) 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 <random.h>
       6                 :             : #include <span.h>
       7                 :             : #include <test/fuzz/util.h>
       8                 :             : #include <util/vecdeque.h>
       9                 :             : 
      10                 :             : #include <deque>
      11                 :             : #include <stdint.h>
      12                 :             : 
      13                 :             : namespace {
      14                 :             : 
      15                 :             : /** The maximum number of simultaneous buffers kept by the test. */
      16                 :             : static constexpr size_t MAX_BUFFERS{3};
      17                 :             : /** How many elements are kept in a buffer at most. */
      18                 :             : static constexpr size_t MAX_BUFFER_SIZE{48};
      19                 :             : /** How many operations are performed at most on the buffers in one test. */
      20                 :             : static constexpr size_t MAX_OPERATIONS{1024};
      21                 :             : 
      22                 :             : /** Perform a simulation fuzz test on VecDeque type T.
      23                 :             :  *
      24                 :             :  * T must be constructible from a uint64_t seed, comparable to other T, copyable, and movable.
      25                 :             :  */
      26                 :             : template<typename T, bool CheckNoneLeft>
      27                 :        3164 : void TestType(Span<const uint8_t> buffer, uint64_t rng_tweak)
      28                 :             : {
      29                 :        3164 :     FuzzedDataProvider provider(buffer.data(), buffer.size());
      30                 :             :     // Local RNG, only used for the seeds to initialize T objects with.
      31                 :        3164 :     InsecureRandomContext rng(provider.ConsumeIntegral<uint64_t>() ^ rng_tweak);
      32                 :             : 
      33                 :             :     // Real circular buffers.
      34                 :        3164 :     std::vector<VecDeque<T>> real;
      35         [ +  - ]:        3164 :     real.reserve(MAX_BUFFERS);
      36                 :             :     // Simulated circular buffers.
      37                 :        3164 :     std::vector<std::deque<T>> sim;
      38         [ +  - ]:        3164 :     sim.reserve(MAX_BUFFERS);
      39                 :             :     // Temporary object of type T.
      40                 :        3164 :     std::optional<T> tmp;
      41                 :             : 
      42                 :             :     // Compare a real and a simulated buffer.
      43         [ -  + ]:      570626 :     auto compare_fn = [](const VecDeque<T>& r, const std::deque<T>& s) {
      44   [ -  +  -  +  :      570626 :         assert(r.size() == s.size());
          -  +  -  +  -  
             +  -  +  -  
                      + ]
      45   [ -  +  -  +  :      570626 :         assert(r.empty() == s.empty());
          -  +  -  +  -  
             +  -  +  -  
                      + ]
      46   [ -  +  -  +  :      570626 :         assert(r.capacity() >= r.size());
          -  +  -  +  -  
             +  -  +  -  
                      + ]
      47   [ +  +  +  +  :      570626 :         if (s.size() == 0) return;
          +  +  +  +  +  
             +  +  +  +  
                      + ]
      48   [ -  +  -  +  :      335293 :         assert(r.front() == s.front());
          -  +  -  +  -  
             +  -  +  -  
                      + ]
      49   [ -  +  -  +  :      335293 :         assert(r.back() == s.back());
          -  +  -  +  -  
             +  -  +  -  
                      + ]
      50   [ +  +  +  +  :     7862603 :         for (size_t i = 0; i < s.size(); ++i) {
          +  +  +  +  +  
             +  +  +  +  
                      + ]
      51   [ -  +  -  +  :     7527310 :             assert(r[i] == s[i]);
          -  +  -  +  -  
             +  -  +  -  
                      + ]
      52                 :             :         }
      53                 :             :     };
      54                 :             : 
      55   [ +  +  +  + ]:     1574328 :     LIMITED_WHILE(provider.remaining_bytes(), MAX_OPERATIONS) {
      56                 :     1571164 :         int command = provider.ConsumeIntegral<uint8_t>() % 64;
      57   [ +  +  +  + ]:     1571164 :         unsigned idx = real.empty() ? 0 : provider.ConsumeIntegralInRange<unsigned>(0, real.size() - 1);
      58                 :     1571164 :         const size_t num_buffers = sim.size();
      59                 :             :         // Pick one operation based on value of command. Not all operations are always applicable.
      60                 :             :         // Loop through the applicable ones until command reaches 0 (which avoids the need to
      61                 :             :         // compute the number of applicable commands ahead of time).
      62                 :     1571164 :         const bool non_empty{num_buffers != 0};
      63                 :     1571164 :         const bool non_full{num_buffers < MAX_BUFFERS};
      64                 :     1571164 :         const bool partially_full{non_empty && non_full};
      65                 :     1571164 :         const bool multiple_exist{num_buffers > 1};
      66   [ +  +  +  + ]:     1571164 :         const bool existing_buffer_non_full{non_empty && sim[idx].size() < MAX_BUFFER_SIZE};
      67   [ +  -  +  + ]:     1993691 :         const bool existing_buffer_non_empty{non_empty && !sim[idx].empty()};
      68                 :             :         assert(non_full || non_empty);
      69                 :             :         while (true) {
      70   [ +  +  +  + ]:     3646622 :             if (non_full && command-- == 0) {
      71                 :             :                 /* Default construct. */
      72         [ +  - ]:       46550 :                 real.emplace_back();
      73         [ +  - ]:       46550 :                 sim.emplace_back();
      74                 :             :                 break;
      75                 :             :             }
      76   [ +  +  +  + ]:     3600072 :             if (non_empty && command-- == 0) {
      77                 :             :                 /* resize() */
      78                 :      164822 :                 compare_fn(real[idx], sim[idx]);
      79                 :      164822 :                 size_t new_size = provider.ConsumeIntegralInRange<size_t>(0, MAX_BUFFER_SIZE);
      80         [ +  - ]:      164822 :                 real[idx].resize(new_size);
      81         [ +  - ]:      164822 :                 sim[idx].resize(new_size);
      82         [ -  + ]:      164822 :                 assert(real[idx].size() == new_size);
      83                 :             :                 break;
      84                 :             :             }
      85   [ +  +  +  + ]:     3435250 :             if (non_empty && command-- == 0) {
      86                 :             :                 /* clear() */
      87                 :       84056 :                 compare_fn(real[idx], sim[idx]);
      88                 :       84056 :                 real[idx].clear();
      89                 :       84056 :                 sim[idx].clear();
      90         [ -  + ]:       84056 :                 assert(real[idx].empty());
      91                 :             :                 break;
      92                 :             :             }
      93   [ +  +  +  + ]:     3351194 :             if (non_empty && command-- == 0) {
      94                 :             :                 /* Copy construct default. */
      95                 :       86667 :                 compare_fn(real[idx], sim[idx]);
      96                 :       86667 :                 real[idx] = VecDeque<T>();
      97                 :       86667 :                 sim[idx].clear();
      98         [ -  + ]:       86667 :                 assert(real[idx].size() == 0);
      99                 :             :                 break;
     100                 :             :             }
     101   [ +  +  +  + ]:     3264527 :             if (non_empty && command-- == 0) {
     102                 :             :                 /* Destruct. */
     103                 :       65394 :                 compare_fn(real.back(), sim.back());
     104                 :       65394 :                 real.pop_back();
     105                 :       65394 :                 sim.pop_back();
     106                 :             :                 break;
     107                 :             :             }
     108   [ +  +  +  + ]:     3199133 :             if (partially_full && command-- == 0) {
     109                 :             :                 /* Copy construct. */
     110         [ +  - ]:        7203 :                 real.emplace_back(real[idx]);
     111         [ +  - ]:        7203 :                 sim.emplace_back(sim[idx]);
     112                 :             :                 break;
     113                 :             :             }
     114   [ +  +  +  + ]:     3191930 :             if (partially_full && command-- == 0) {
     115                 :             :                 /* Move construct. */
     116         [ +  - ]:       19117 :                 VecDeque<T> copy(real[idx]);
     117         [ +  - ]:       19117 :                 real.emplace_back(std::move(copy));
     118         [ +  - ]:       19117 :                 sim.emplace_back(sim[idx]);
     119                 :             :                 break;
     120                 :       19117 :             }
     121   [ +  +  +  + ]:     3172813 :             if (multiple_exist && command-- == 0) {
     122                 :             :                 /* swap() */
     123                 :       22386 :                 swap(real[idx], real[(idx + 1) % num_buffers]);
     124                 :       22386 :                 swap(sim[idx], sim[(idx + 1) % num_buffers]);
     125                 :             :                 break;
     126                 :             :             }
     127   [ +  +  +  + ]:     3150427 :             if (multiple_exist && command-- == 0) {
     128                 :             :                 /* Copy assign. */
     129                 :       83265 :                 compare_fn(real[idx], sim[idx]);
     130         [ +  - ]:       83265 :                 real[idx] = real[(idx + 1) % num_buffers];
     131         [ +  - ]:       83265 :                 sim[idx] = sim[(idx + 1) % num_buffers];
     132                 :             :                 break;
     133                 :             :             }
     134   [ +  +  +  + ]:     3067162 :             if (multiple_exist && command-- == 0) {
     135                 :             :                 /* Move assign. */
     136         [ +  - ]:       78946 :                 VecDeque<T> copy(real[(idx + 1) % num_buffers]);
     137                 :       78946 :                 compare_fn(real[idx], sim[idx]);
     138         [ +  - ]:       78946 :                 real[idx] = std::move(copy);
     139         [ +  - ]:       78946 :                 sim[idx] = sim[(idx + 1) % num_buffers];
     140                 :             :                 break;
     141                 :       78946 :             }
     142   [ +  +  +  + ]:     2988216 :             if (non_empty && command-- == 0) {
     143                 :             :                 /* Self swap() */
     144                 :     1571164 :                 swap(real[idx], real[idx]);
     145                 :             :                 break;
     146                 :             :             }
     147   [ +  +  +  + ]:     2961826 :             if (non_empty && command-- == 0) {
     148                 :             :                 /* Self-copy assign. */
     149         [ +  - ]:       34468 :                 real[idx] = real[idx];
     150                 :             :                 break;
     151                 :             :             }
     152   [ +  +  +  + ]:     2927358 :             if (non_empty && command-- == 0) {
     153                 :             :                 /* Self-move assign. */
     154                 :             :                 // Do not use std::move(real[idx]) here: -Wself-move correctly warns about that.
     155                 :     1571164 :                 real[idx] = static_cast<VecDeque<T>&&>(real[idx]);
     156                 :             :                 break;
     157                 :             :             }
     158   [ +  +  +  + ]:     2886212 :             if (non_empty && command-- == 0) {
     159                 :             :                 /* reserve() */
     160                 :       34944 :                 size_t res_size = provider.ConsumeIntegralInRange<size_t>(0, MAX_BUFFER_SIZE);
     161         [ +  + ]:       34944 :                 size_t old_cap = real[idx].capacity();
     162         [ +  + ]:       34944 :                 size_t old_size = real[idx].size();
     163         [ +  + ]:       34944 :                 real[idx].reserve(res_size);
     164         [ -  + ]:       34944 :                 assert(real[idx].size() == old_size);
     165   [ +  +  -  + ]:       58548 :                 assert(real[idx].capacity() == std::max(old_cap, res_size));
     166                 :             :                 break;
     167                 :             :             }
     168   [ +  +  +  + ]:     2851268 :             if (non_empty && command-- == 0) {
     169                 :             :                 /* shrink_to_fit() */
     170         [ +  - ]:       41531 :                 size_t old_size = real[idx].size();
     171         [ +  - ]:       41531 :                 real[idx].shrink_to_fit();
     172         [ -  + ]:       41531 :                 assert(real[idx].size() == old_size);
     173         [ -  + ]:       41531 :                 assert(real[idx].capacity() == old_size);
     174                 :             :                 break;
     175                 :             :             }
     176   [ +  +  +  + ]:     2809737 :             if (existing_buffer_non_full && command-- == 0) {
     177                 :             :                 /* push_back() (copying) */
     178   [ +  -  +  - ]:       61754 :                 tmp = T(rng.rand64());
     179         [ +  - ]:       61754 :                 size_t old_size = real[idx].size();
     180         [ +  - ]:       61754 :                 size_t old_cap = real[idx].capacity();
     181         [ +  - ]:       61754 :                 real[idx].push_back(*tmp);
     182         [ +  - ]:       61754 :                 sim[idx].push_back(*tmp);
     183         [ -  + ]:       61754 :                 assert(real[idx].size() == old_size + 1);
     184         [ +  + ]:       61754 :                 if (old_cap > old_size) {
     185         [ -  + ]:       39879 :                     assert(real[idx].capacity() == old_cap);
     186                 :             :                 } else {
     187         [ -  + ]:       21875 :                     assert(real[idx].capacity() > old_cap);
     188         [ -  + ]:       21875 :                     assert(real[idx].capacity() <= 2 * (old_cap + 1));
     189                 :             :                 }
     190                 :             :                 break;
     191                 :             :             }
     192   [ +  +  +  + ]:     2747983 :             if (existing_buffer_non_full && command-- == 0) {
     193                 :             :                 /* push_back() (moving) */
     194   [ +  -  +  - ]:       43092 :                 tmp = T(rng.rand64());
     195         [ +  - ]:       43092 :                 size_t old_size = real[idx].size();
     196         [ +  - ]:       43092 :                 size_t old_cap = real[idx].capacity();
     197         [ +  - ]:       43092 :                 sim[idx].push_back(*tmp);
     198         [ +  - ]:       43092 :                 real[idx].push_back(std::move(*tmp));
     199         [ -  + ]:       43092 :                 assert(real[idx].size() == old_size + 1);
     200         [ +  + ]:       43092 :                 if (old_cap > old_size) {
     201         [ -  + ]:       30730 :                     assert(real[idx].capacity() == old_cap);
     202                 :             :                 } else {
     203         [ -  + ]:       12362 :                     assert(real[idx].capacity() > old_cap);
     204         [ -  + ]:       12362 :                     assert(real[idx].capacity() <= 2 * (old_cap + 1));
     205                 :             :                 }
     206                 :             :                 break;
     207                 :             :             }
     208   [ +  +  +  + ]:     2704891 :             if (existing_buffer_non_full && command-- == 0) {
     209                 :             :                 /* emplace_back() */
     210                 :       74788 :                 uint64_t seed{rng.rand64()};
     211         [ +  - ]:       74788 :                 size_t old_size = real[idx].size();
     212         [ +  - ]:       74788 :                 size_t old_cap = real[idx].capacity();
     213         [ +  - ]:       74788 :                 sim[idx].emplace_back(seed);
     214         [ +  - ]:       74788 :                 real[idx].emplace_back(seed);
     215         [ -  + ]:       74788 :                 assert(real[idx].size() == old_size + 1);
     216         [ +  + ]:       74788 :                 if (old_cap > old_size) {
     217         [ -  + ]:       57162 :                     assert(real[idx].capacity() == old_cap);
     218                 :             :                 } else {
     219         [ -  + ]:       17626 :                     assert(real[idx].capacity() > old_cap);
     220         [ -  + ]:       17626 :                     assert(real[idx].capacity() <= 2 * (old_cap + 1));
     221                 :             :                 }
     222                 :             :                 break;
     223                 :             :             }
     224   [ +  +  +  + ]:     2630103 :             if (existing_buffer_non_full && command-- == 0) {
     225                 :             :                 /* push_front() (copying) */
     226   [ +  -  +  - ]:       64610 :                 tmp = T(rng.rand64());
     227         [ +  - ]:       64610 :                 size_t old_size = real[idx].size();
     228         [ +  - ]:       64610 :                 size_t old_cap = real[idx].capacity();
     229         [ +  - ]:       64610 :                 real[idx].push_front(*tmp);
     230         [ +  - ]:       64610 :                 sim[idx].push_front(*tmp);
     231         [ -  + ]:       64610 :                 assert(real[idx].size() == old_size + 1);
     232         [ +  + ]:       64610 :                 if (old_cap > old_size) {
     233         [ -  + ]:       45185 :                     assert(real[idx].capacity() == old_cap);
     234                 :             :                 } else {
     235         [ -  + ]:       19425 :                     assert(real[idx].capacity() > old_cap);
     236         [ -  + ]:       19425 :                     assert(real[idx].capacity() <= 2 * (old_cap + 1));
     237                 :             :                 }
     238                 :             :                 break;
     239                 :             :             }
     240   [ +  +  +  + ]:     2565493 :             if (existing_buffer_non_full && command-- == 0) {
     241                 :             :                 /* push_front() (moving) */
     242   [ +  -  +  - ]:       73605 :                 tmp = T(rng.rand64());
     243         [ +  - ]:       73605 :                 size_t old_size = real[idx].size();
     244         [ +  - ]:       73605 :                 size_t old_cap = real[idx].capacity();
     245         [ +  - ]:       73605 :                 sim[idx].push_front(*tmp);
     246         [ +  - ]:       73605 :                 real[idx].push_front(std::move(*tmp));
     247         [ -  + ]:       73605 :                 assert(real[idx].size() == old_size + 1);
     248         [ +  + ]:       73605 :                 if (old_cap > old_size) {
     249         [ -  + ]:       57176 :                     assert(real[idx].capacity() == old_cap);
     250                 :             :                 } else {
     251         [ -  + ]:       16429 :                     assert(real[idx].capacity() > old_cap);
     252         [ -  + ]:       16429 :                     assert(real[idx].capacity() <= 2 * (old_cap + 1));
     253                 :             :                 }
     254                 :             :                 break;
     255                 :             :             }
     256   [ +  +  +  + ]:     2491888 :             if (existing_buffer_non_full && command-- == 0) {
     257                 :             :                 /* emplace_front() */
     258                 :      118356 :                 uint64_t seed{rng.rand64()};
     259         [ +  - ]:      118356 :                 size_t old_size = real[idx].size();
     260         [ +  - ]:      118356 :                 size_t old_cap = real[idx].capacity();
     261         [ +  - ]:      118356 :                 sim[idx].emplace_front(seed);
     262         [ +  - ]:      118356 :                 real[idx].emplace_front(seed);
     263         [ -  + ]:      118356 :                 assert(real[idx].size() == old_size + 1);
     264         [ +  + ]:      118356 :                 if (old_cap > old_size) {
     265         [ -  + ]:       92764 :                     assert(real[idx].capacity() == old_cap);
     266                 :             :                 } else {
     267         [ -  + ]:       25592 :                     assert(real[idx].capacity() > old_cap);
     268         [ -  + ]:       25592 :                     assert(real[idx].capacity() <= 2 * (old_cap + 1));
     269                 :             :                 }
     270                 :             :                 break;
     271                 :             :             }
     272   [ +  +  +  + ]:     2373532 :             if (existing_buffer_non_empty && command-- == 0) {
     273                 :             :                 /* front() [modifying] */
     274   [ +  -  +  - ]:       33957 :                 tmp = T(rng.rand64());
     275                 :       33957 :                 size_t old_size = real[idx].size();
     276         [ -  + ]:       33957 :                 assert(sim[idx].front() == real[idx].front());
     277                 :       33957 :                 sim[idx].front() = *tmp;
     278                 :       33957 :                 real[idx].front() = std::move(*tmp);
     279         [ -  + ]:       33957 :                 assert(real[idx].size() == old_size);
     280                 :             :                 break;
     281                 :             :             }
     282   [ +  +  +  + ]:     2339575 :             if (existing_buffer_non_empty && command-- == 0) {
     283                 :             :                 /* back() [modifying] */
     284   [ +  -  +  - ]:       25543 :                 tmp = T(rng.rand64());
     285                 :       25543 :                 size_t old_size = real[idx].size();
     286         [ -  + ]:       25543 :                 assert(sim[idx].back() == real[idx].back());
     287                 :       25543 :                 sim[idx].back() = *tmp;
     288                 :       25543 :                 real[idx].back() = *tmp;
     289         [ -  + ]:       25543 :                 assert(real[idx].size() == old_size);
     290                 :             :                 break;
     291                 :             :             }
     292   [ +  +  +  + ]:     2314032 :             if (existing_buffer_non_empty && command-- == 0) {
     293                 :             :                 /* operator[] [modifying] */
     294   [ +  -  +  - ]:       38850 :                 tmp = T(rng.rand64());
     295                 :       55500 :                 size_t pos = provider.ConsumeIntegralInRange<size_t>(0, sim[idx].size() - 1);
     296                 :       38850 :                 size_t old_size = real[idx].size();
     297         [ -  + ]:       38850 :                 assert(sim[idx][pos] == real[idx][pos]);
     298                 :       38850 :                 sim[idx][pos] = *tmp;
     299                 :       38850 :                 real[idx][pos] = std::move(*tmp);
     300         [ -  + ]:       38850 :                 assert(real[idx].size() == old_size);
     301                 :             :                 break;
     302                 :             :             }
     303   [ +  +  +  + ]:     2275182 :             if (existing_buffer_non_empty && command-- == 0) {
     304                 :             :                 /* pop_front() */
     305         [ -  + ]:      166488 :                 assert(sim[idx].front() == real[idx].front());
     306                 :      166488 :                 size_t old_size = real[idx].size();
     307                 :      166488 :                 sim[idx].pop_front();
     308         [ +  - ]:      166488 :                 real[idx].pop_front();
     309         [ -  + ]:      166488 :                 assert(real[idx].size() == old_size - 1);
     310                 :             :                 break;
     311                 :             :             }
     312   [ +  +  +  +  :     2990127 :             if (existing_buffer_non_empty && command-- == 0) {
                   +  + ]
     313                 :             :                 /* pop_back() */
     314         [ -  + ]:       33236 :                 assert(sim[idx].back() == real[idx].back());
     315                 :       33236 :                 size_t old_size = real[idx].size();
     316                 :       33236 :                 sim[idx].pop_back();
     317         [ +  - ]:       33236 :                 real[idx].pop_back();
     318         [ -  + ]:       33236 :                 assert(real[idx].size() == old_size - 1);
     319                 :             :                 break;
     320                 :             :             }
     321                 :             :         }
     322                 :             :     }
     323                 :             : 
     324                 :             :     /* Fully compare the final state. */
     325         [ +  + ]:       10640 :     for (unsigned i = 0; i < sim.size(); ++i) {
     326                 :             :         // Make sure const getters work.
     327                 :        7476 :         const VecDeque<T>& realbuf = real[i];
     328                 :        7476 :         const std::deque<T>& simbuf = sim[i];
     329                 :        7476 :         compare_fn(realbuf, simbuf);
     330         [ +  + ]:       27776 :         for (unsigned j = 0; j < sim.size(); ++j) {
     331         [ -  + ]:       20300 :             assert((realbuf == real[j]) == (simbuf == sim[j]));
     332         [ -  + ]:       20300 :             assert(((realbuf <=> real[j]) >= 0) == (simbuf >= sim[j]));
     333         [ -  + ]:       20300 :             assert(((realbuf <=> real[j]) <= 0) == (simbuf <= sim[j]));
     334                 :             :         }
     335                 :             :         // Clear out the buffers so we can check below that no objects exist anymore.
     336                 :        7476 :         sim[i].clear();
     337                 :        7476 :         real[i].clear();
     338                 :             :     }
     339                 :             : 
     340                 :             :     if constexpr (CheckNoneLeft) {
     341                 :        1356 :         tmp = std::nullopt;
     342                 :        1356 :         T::CheckNoneExist();
     343                 :             :     }
     344                 :        3164 : }
     345                 :             : 
     346                 :             : /** Data structure with built-in tracking of all existing objects. */
     347                 :             : template<size_t Size>
     348                 :             : class TrackedObj
     349                 :             : {
     350                 :             :     static_assert(Size > 0);
     351                 :             : 
     352                 :             :     /* Data type for map that actually stores the object data.
     353                 :             :      *
     354                 :             :      * The key is a pointer to the TrackedObj, the value is the uint64_t it was initialized with.
     355                 :             :      * Default-constructed and moved-from objects hold an std::nullopt.
     356                 :             :      */
     357                 :             :     using track_map_type = std::map<const TrackedObj<Size>*, std::optional<uint64_t>>;
     358                 :             : 
     359                 :             : private:
     360                 :             : 
     361                 :             :     /** Actual map. */
     362                 :             :     static inline track_map_type g_tracker;
     363                 :             : 
     364                 :             :     /** Iterators into the tracker map for this object.
     365                 :             :      *
     366                 :             :      * This is an array of size Size, all holding the same value, to give the object configurable
     367                 :             :      * size. The value is g_tracker.end() if this object is not fully initialized. */
     368                 :             :     typename track_map_type::iterator m_track_entry[Size];
     369                 :             : 
     370                 :    24904425 :     void Check() const
     371                 :             :     {
     372                 :    24904425 :         auto it = g_tracker.find(this);
     373         [ +  + ]:   199235400 :         for (size_t i = 0; i < Size; ++i) {
     374         [ -  + ]:   174330975 :             assert(m_track_entry[i] == it);
     375                 :             :         }
     376                 :    24904425 :     }
     377                 :             : 
     378                 :             :     /** Create entry for this object in g_tracker and populate m_track_entry. */
     379                 :     6040536 :     void Register()
     380                 :             :     {
     381         [ +  - ]:     6040536 :         auto [it, inserted] = g_tracker.emplace(this, std::nullopt);
     382         [ +  - ]:     6040536 :         assert(inserted);
     383         [ +  + ]:    48324288 :         for (size_t i = 0; i < Size; ++i) {
     384                 :    42283752 :             m_track_entry[i] = it;
     385                 :             :         }
     386                 :     6040536 :     }
     387                 :             : 
     388                 :     6040536 :     void Deregister()
     389                 :             :     {
     390         [ -  + ]:     6040536 :         Check();
     391         [ -  + ]:     6040536 :         assert(m_track_entry[0] != g_tracker.end());
     392                 :     6040536 :         g_tracker.erase(m_track_entry[0]);
     393         [ +  + ]:    48324288 :         for (size_t i = 0; i < Size; ++i) {
     394                 :    42283752 :             m_track_entry[i] = g_tracker.end();
     395                 :             :         }
     396                 :     6040536 :     }
     397                 :             : 
     398                 :             :     /** Get value corresponding to this object in g_tracker. */
     399                 :     7091523 :     std::optional<uint64_t>& Deref()
     400                 :             :     {
     401         [ -  + ]:     7091523 :         Check();
     402         [ -  + ]:     7091523 :         assert(m_track_entry[0] != g_tracker.end());
     403                 :     7091523 :         return m_track_entry[0]->second;
     404                 :             :     }
     405                 :             : 
     406                 :             :     /** Get value corresponding to this object in g_tracker. */
     407                 :    11772366 :     const std::optional<uint64_t>& Deref() const
     408                 :             :     {
     409         [ -  + ]:    11772366 :         Check();
     410         [ -  + ]:    11772366 :         assert(m_track_entry[0] != g_tracker.end());
     411                 :    11772366 :         return m_track_entry[0]->second;
     412                 :             :     }
     413                 :             : 
     414                 :             : public:
     415   [ +  +  +  +  :     6040536 :     ~TrackedObj() { Deregister(); }
          +  +  +  +  +  
          +  +  +  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
          +  -  +  -  +  
          -  +  -  +  -  
                   +  - ]
     416         [ +  + ]:    13821312 :     TrackedObj() { Register(); }
     417                 :             : 
     418                 :      311871 :     TrackedObj(uint64_t value)
     419         [ +  + ]:     2494968 :     {
     420                 :      311871 :         Register();
     421                 :      311871 :         Deref() = value;
     422                 :      311871 :     }
     423                 :             : 
     424                 :     3195528 :     TrackedObj(const TrackedObj& other)
     425         [ +  + ]:    25564224 :     {
     426                 :     3195528 :         Register();
     427                 :     3195528 :         Deref() = other.Deref();
     428                 :     3195528 :     }
     429                 :             : 
     430                 :      805473 :     TrackedObj(TrackedObj&& other)
     431         [ +  + ]:     6443784 :     {
     432                 :      805473 :         Register();
     433                 :      805473 :         Deref() = other.Deref();
     434         [ +  + ]:      805473 :         other.Deref() = std::nullopt;
     435                 :      805473 :     }
     436                 :             : 
     437                 :      638802 :     TrackedObj& operator=(const TrackedObj& other)
     438                 :             :     {
     439         [ +  - ]:      638802 :         if (this == &other) return *this;
     440                 :      638802 :         Deref() = other.Deref();
     441                 :      638802 :         return *this;
     442                 :             :     }
     443                 :             : 
     444                 :      176301 :     TrackedObj& operator=(TrackedObj&& other)
     445                 :             :     {
     446         [ +  - ]:      176301 :         if (this == &other) return *this;
     447                 :      176301 :         Deref() = other.Deref();
     448         [ +  - ]:      176301 :         other.Deref() = std::nullopt;
     449                 :             :         return *this;
     450                 :             :     }
     451                 :             : 
     452                 :     3746988 :     friend bool operator==(const TrackedObj& a, const TrackedObj& b)
     453                 :             :     {
     454         [ +  + ]:     3746988 :         return a.Deref() == b.Deref();
     455                 :             :     }
     456                 :             : 
     457                 :      221772 :     friend std::strong_ordering operator<=>(const TrackedObj& a, const TrackedObj& b)
     458                 :             :     {
     459                 :             :         // Libc++ 15 & 16 do not support std::optional<T>::operator<=> yet. See
     460                 :             :         // https://reviews.llvm.org/D146392.
     461   [ +  +  +  + ]:      221772 :         if (!a.Deref().has_value() || !b.Deref().has_value()) {
     462   [ +  +  +  + ]:      160488 :             return a.Deref().has_value() <=> b.Deref().has_value();
     463                 :             :         }
     464   [ +  +  +  + ]:       61284 :         return *a.Deref() <=> *b.Deref();
     465                 :             :     }
     466                 :             : 
     467         [ -  + ]:        1356 :     static void CheckNoneExist()
     468                 :             :     {
     469         [ -  + ]:        1356 :         assert(g_tracker.empty());
     470                 :        1356 :     }
     471                 :             : };
     472                 :             : 
     473                 :             : } // namespace
     474                 :             : 
     475         [ +  - ]:         866 : FUZZ_TARGET(vecdeque)
     476                 :             : {
     477                 :             :     // Run the test with simple uints (which satisfy all the trivial properties).
     478                 :         452 :     static_assert(std::is_trivially_copyable_v<uint32_t>);
     479                 :         452 :     static_assert(std::is_trivially_destructible_v<uint64_t>);
     480                 :         452 :     TestType<uint8_t, false>(buffer, 1);
     481                 :         452 :     TestType<uint16_t, false>(buffer, 2);
     482                 :         452 :     TestType<uint32_t, false>(buffer, 3);
     483                 :         452 :     TestType<uint64_t, false>(buffer, 4);
     484                 :             : 
     485                 :             :     // Run the test with TrackedObjs (which do not).
     486                 :         452 :     static_assert(!std::is_trivially_copyable_v<TrackedObj<3>>);
     487                 :         452 :     static_assert(!std::is_trivially_destructible_v<TrackedObj<17>>);
     488                 :         452 :     TestType<TrackedObj<1>, true>(buffer, 5);
     489                 :         452 :     TestType<TrackedObj<3>, true>(buffer, 6);
     490                 :         452 :     TestType<TrackedObj<17>, true>(buffer, 7);
     491                 :         452 : }
        

Generated by: LCOV version 2.0-1