Skip to content

Commit 5035178

Browse files
Fix masking of _First_used_block_idx in deque::shrink_to_fit
1 parent 73b5791 commit 5035178

File tree

2 files changed

+29
-2
lines changed
  • stl/inc
  • tests/std/tests/GH_002769_handle_deque_block_pointers

2 files changed

+29
-2
lines changed

stl/inc/deque

+4-2
Original file line numberDiff line numberDiff line change
@@ -997,7 +997,8 @@ public:
997997

998998
const auto _Mask = static_cast<size_type>(_Mapsize() - 1);
999999

1000-
const auto _First_used_block_idx = static_cast<size_type>(_Myoff() / _Block_size);
1000+
const auto _Unmasked_first_used_block_idx = static_cast<size_type>(_Myoff() / _Block_size);
1001+
const auto _First_used_block_idx = static_cast<size_type>(_Unmasked_first_used_block_idx & _Mask);
10011002

10021003
// (_Myoff() + _Mysize() - 1) is for the last element, i.e. the back() of the deque.
10031004
// Divide by _Block_size to get the unmasked index of the last used block.
@@ -1017,7 +1018,8 @@ public:
10171018
}
10181019
}
10191020

1020-
const auto _Used_block_count = static_cast<size_type>(_Unmasked_first_unused_block_idx - _First_used_block_idx);
1021+
const auto _Used_block_count =
1022+
static_cast<size_type>(_Unmasked_first_unused_block_idx - _Unmasked_first_used_block_idx);
10211023

10221024
size_type _New_block_count = _Minimum_map_size; // should be power of 2
10231025

tests/std/tests/GH_002769_handle_deque_block_pointers/test.cpp

+25
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44
#include <cassert>
55
#include <cstddef>
66
#include <cstdint>
7+
#include <cstdlib>
78
#include <deque>
89
#include <iterator>
910
#include <memory>
@@ -328,8 +329,32 @@ void test_inconsistent_difference_types() {
328329
assert(counter == 0);
329330
}
330331

332+
// Also test GH-4954: Endless loop in deque::shrink_to_fit()
333+
void test_gh_4954() {
334+
deque<int> qu;
335+
336+
int it = 0;
337+
while (it < 1024) {
338+
const auto numAlloc = static_cast<size_t>(rand() + 1);
339+
for (size_t i = 0; i < numAlloc; i++) {
340+
qu.push_back(0);
341+
}
342+
343+
auto numDealloc = static_cast<size_t>(rand() + 1);
344+
if (it % 100 == 0 || numDealloc > qu.size()) {
345+
numDealloc = qu.size();
346+
}
347+
for (size_t i = 0; i < numDealloc; i++) {
348+
qu.pop_front();
349+
}
350+
qu.shrink_to_fit();
351+
++it;
352+
}
353+
}
354+
331355
int main() {
332356
test_gh_2769();
333357
test_gh_3717();
358+
test_gh_4954();
334359
test_inconsistent_difference_types();
335360
}

0 commit comments

Comments
 (0)