forked from rethinkdb/rethinkdb
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsegmented_vector.hpp
90 lines (70 loc) · 2.34 KB
/
segmented_vector.hpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
// Copyright 2010-2012 RethinkDB, all rights reserved.
#ifndef CONTAINERS_SEGMENTED_VECTOR_HPP_
#define CONTAINERS_SEGMENTED_VECTOR_HPP_
#include <stdio.h>
#include "errors.hpp"
#include "logger.hpp"
#define ELEMENTS_PER_SEGMENT (1 << 14)
template<class element_t, size_t max_size>
class segmented_vector_t
{
private:
struct segment_t {
element_t elements[ELEMENTS_PER_SEGMENT];
} *segments[max_size / ELEMENTS_PER_SEGMENT];
size_t size;
public:
explicit segmented_vector_t(size_t _size = 0) : size(0) {
set_size(_size);
}
~segmented_vector_t() {
set_size(0);
}
public:
element_t &operator[](size_t i) {
return get(i);
}
const element_t &operator[](size_t i) const {
return const_get(i);
}
element_t &get(size_t i) {
return const_cast<element_t &>(const_get(i));
}
size_t get_size() const {
return size;
}
// Note: sometimes elements will be initialized before you ask the
// array to grow to that size (e.g. one hundred elements might be
// initialized even though the array might be of size 1).
void set_size(size_t new_size) {
rassert(new_size <= max_size);
size_t num_segs = size ? ((size - 1) / ELEMENTS_PER_SEGMENT) + 1 : 0;
size_t new_num_segs = new_size ? ((new_size - 1) / ELEMENTS_PER_SEGMENT) + 1 : 0;
if (num_segs > new_num_segs) {
for (size_t si = new_num_segs; si < num_segs; si ++) {
delete segments[si];
}
}
if (new_num_segs > num_segs) {
for (size_t si = num_segs; si < new_num_segs; si ++) {
segments[si] = new segment_t;
}
}
size = new_size;
}
// This form of set_size fills the newly allocated space with a value
void set_size(size_t new_size, element_t fill) {
size_t old_size = size;
set_size(new_size);
for (; old_size < new_size; old_size++) (*this)[old_size] = fill;
}
private:
const element_t &const_get(size_t i) const {
rassert(i < size, "i is %zu, size is %zu", i, size);
segment_t *segment = segments[i / ELEMENTS_PER_SEGMENT];
rassert(segment);
return segment->elements[i % ELEMENTS_PER_SEGMENT];
}
DISABLE_COPYING(segmented_vector_t);
};
#endif // CONTAINERS_SEGMENTED_VECTOR_HPP_