forked from foudrer/Sux
-
Notifications
You must be signed in to change notification settings - Fork 0
/
rank9.cpp
67 lines (55 loc) · 1.93 KB
/
rank9.cpp
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
/*
* Sux: Succinct data structures
*
* Copyright (C) 2007-2013 Sebastiano Vigna
*
* This library is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as published by the Free
* Software Foundation; either version 3 of the License, or (at your option)
* any later version.
*
* This library is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License
* for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program; if not, see <http://www.gnu.org/licenses/>.
*
*/
#include <cassert>
#include <cstring>
#include "rank9.h"
rank9::rank9() {}
rank9::rank9( const uint64_t * const bits, const uint64_t num_bits ) {
this->bits = bits;
num_words = ( num_bits + 63 ) / 64;
num_counts = ( ( num_bits + 64 * 8 - 1 ) / ( 64 * 8 ) ) * 2;
// Init rank structure
counts = new uint64_t[ num_counts + 1 ]();
uint64_t c = 0;
uint64_t pos = 0;
for( uint64_t i = 0; i < num_words; i += 8, pos += 2 ) {
counts[ pos ] = c;
c += __builtin_popcountll( bits[ i ] );
for( int j = 1; j < 8; j++ ) {
counts[ pos + 1 ] |= ( c - counts[ pos ] ) << 9 * ( j - 1 );
if ( i + j < num_words ) c += __builtin_popcountll( bits[ i + j ] );
}
}
counts[ num_counts ] = c;
assert( c <= num_bits );
}
rank9::~rank9() {
delete [] counts;
}
uint64_t rank9::rank( const uint64_t k ) {
const uint64_t word = k / 64;
const uint64_t block = word / 4 & ~1;
const int offset = word % 8 - 1;
return counts[ block ] + ( counts[ block + 1 ] >> ( offset + ( offset >> sizeof offset * 8 - 4 & 0x8 ) ) * 9 & 0x1FF ) + __builtin_popcountll( bits[ word ] & ( ( 1ULL << k % 64 ) - 1 ) );
}
uint64_t rank9::bit_count() {
return num_counts * 64;
}
void rank9::print_counts() {}