-
Notifications
You must be signed in to change notification settings - Fork 1
/
README
413 lines (268 loc) · 23.8 KB
/
README
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
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
General Information
===================
This is an implementation of the Run-Length Compressed Suffix Array (RLCSA) [1,2] and its incremental construction [2]. The implementation includes experimental support for LCP information [3], distribution-aware sampling [4], and document listing [5].
Copyright 2007 - 2014, Jouni Siren, unless otherwise noted. See LICENSE for further information.
Compiling
---------
The code should compile in both 32-bit and 64-bit environments. Uncomment SIZE_FLAGS in the makefile to use 64-bit integers in the 64-bit version.
Parallelism is supported by libstdc++ Parallel Mode and by MCSTL. Uncomment either version of PARALLEL_FLAGS to compile the parallel version of the library, and set MCSTL_ROOT if necessary. GCC 4.2 or newer is required for the MCSTL version.
Uncomment PSI_FLAGS to use a faster encoding for the run-length encoded bit vectors in .rlcsa.array. This increases the size somewhat. Uncomment LCP_FLAGS and SA_FLAGS to use a succinct bit vector instead of a gap encoded one to mark the sampled positions in the LCP array and the suffix array, respectively. This can increase the size of the samples, especially for sparse sampling. On the other hand, retrieving LCP values and locate() queries for single suffix array values can speed up significantly. LCP_FLAGS also uses a succinct vector instead of a run-length encoded one in PLCP.
32-bit integers limit the size of the collection to less than 4 gigabytes. The size of individual input files is limited to less than 2 gigabytes in both 32-bit and 64-bit versions.
Note that if 32-bit integers are used, then the bit-aligned arrays are limited to less than 512 megabytes (2^32 bits) in size. Hence if n is the collection size in characters and d is the sample rate, then (n / d) log ceil(n / d) must be less than 2^32. Otherwise the suffix array samples cannot be stored.
Index Construction
------------------
The naming conventions for files are:
base_name - the sequences
base_name.rlcsa.array - most of the CSA
base_name.rlcsa.sa_samples - suffix array samples for locate and display
base_name.rlcsa.parameters - some of the index parameters
base_name.rlcsa.docs - document listing structure
base_name.lcp_samples - sampled LCP array
base_name.plcp - run-length encoded PLCP array
base_name.sa - suffix array
A typical parameter file looks like:
RLCSA_BLOCK_SIZE = 32
SAMPLE_RATE = 128
SUPPORT_DISPLAY = 1
SUPPORT_LOCATE = 1
WEIGHTED_SAMPLES = 0
parallel_build is used to construct the index, as described in [2]. The program takes 2 to 4 parameters:
use '-n' as the first parameter to stop before merging the partial indexes
use '-f' as the first parameter to use an alternate algorithm (thesis: Fast)
a list of input files (a text file, one file name per line)
base name of the output
number of threads to use (optional)
The default parameters are 32 bytes for block size and 128 for sample rate. To modify these, one should create the parameter file for the output before running the construction program. Each input file should be a concatenation of non-empty C-style '\0'-terminated strings. The files must be smaller than 4 GB each.
build_rlcsa provides a simpler alternative for indexing one file. The program takes 1 or 2 parameters: base name of the input/output and an optional number of threads. The same assumptions and restrictions apply as for parallel_build.
merge_rlcsa is able to merge an index with a new index created by build_rlcsa. The program takes 2 or more: base name of the index to add to, and the base name(s) of the indices to add. The original indexed file for the first index need not exist, but it must still be present for the second. The first argument may optionally be "-THREADS" to use THREADS threads (for example, -3 for 3 threads).
Operations
----------
A number of operations have been implemented. The most important ones are the following:
pair_type count(const std::string& pattern) const
Returns the suffix array range corresponding to the matches of the pattern. The range is reported as a closed interval.
usint* locate(pair_type range, bool direct = false, bool steps = false) const
usint* locate(pair_type range, usint* data, bool direct = false, bool steps = false) const
usint locate(usint index, bool steps = false) const
These return the suffix array values at given range/position. The user is responsible for the allocated data. Optional parameters: bool direct = false (use direct locate implementation instead of the run-based optimizations) and bool steps = false (return the number of steps required to find a sample instead of the SA value).
uchar* display(usint sequence) const
uchar* display(usint sequence, pair_type range) const
uchar* display(usint sequence, pair_type range, uchar* data) const
These return a substring of the given sequence, as determined by the closed SA range 'range'. The user is responsible for freeing the allocated data.
uchar* display(usint position, usint len, usint context, usint& result_length) const
This is intended for displaying an occurrence of a pattern of length 'len' at SA position 'position' with 'context' extra characters on both sides. Parameter result_length will contain the actual length of the returned string.
uchar* readBWT() const
uchar* readBWT(pair_type range) const
Returns the BWT of the collection or a part of it. The user is responsible for the allocated string. Note that unlike the suffix array, the BWT includes all end markers.
pair_type getSequenceRange(usint number) const
Returns the sequence range for the given sequence.
pair_type getSequenceRangeForPosition(usint value) const
Returns the sequence range for the given text position (SA value).
usint getSequenceForPosition(usint value) const
Returns the sequence number for the given text position.
usint* getSequenceForPosition(usint* value, usint length) const
As above, but for multiple positions at once.
pair_type getRelativePosition(usint value) const
Converts text position to pair (sequence number, relative position).
Locate and display can only be used when the corresponding parameter (SUPPORT_LOCATE or SUPPORT_DISPLAY) has value 1 and the suffix array samples have been created during the construction. If both of the parameters are missing or have value 0, the suffix array samples will not be loaded into memory.
These operations are const and hence thread-safe.
There is also a low-level interface (sections SUPPORT FOR EXTERNAL MODULES: POSITIONS and SUPPORT FOR EXTERNAL MODULES: RANGES in rlcsa.h) for use with external modules. While some modules (adaptive_samples.h and GCSA/bwasearch.h) already use the interface, it is not considered stable and can change without warning.
Construction Interface
----------------------
Class RLCSABuilder provides other possibilities for index construction. The constructor takes four parameters:
block size for the Psi vectors in bytes
suffix array sample rate
buffer size for construction in bytes
number of threads to use
If suffix array sample rate is set to 0, the samples will not be created. The buffer size must be less than 4 gigabytes.
Function insertSequence is called to insert a new sequence into the collection. The parameters are:
sequence as a char array
length of the sequence (not including the trailing 0, if present)
should we free the memory used by the sequence
Function insertFromFile can be used to merge existing indexes into the collection. It takes the base name of the index as a parameter. The sequences and the index should both be available.
Function insertCollection can be used to index a new input file and merge it with the existing index. This approach is used in the alternate algorithm (Fast), and generally offers worse time/space trade-offs than the default option.
Function getRLCSA is used to finish the construction and get the final index. After the call, the builder no longer contains the index. The caller is responsible for freeing the index.
For example, the following inserts the sequences into the collection one at a time:
// Use a buffer of n megabytes.
RLCSABuilder builder(block_size, sample_rate, n * MEGABYTE, threads);
// For each sequence:
builder.insertSequence(sequence, length, false);
// If succesful, write the index to disk.
if(builder.isOk())
{
RLCSA* rlcsa = builder.getRLCSA();
rlcsa->writeTo(base_name);
delete rlcsa;
}
Incremental construction for multiple sequences
-----------------------------------------------
When there are multiple sequences in one increment, character '\0' is assumed to represent the end of sequence marker. Hence the sequences themselves cannot contain character '\0'. This is always the case when using RLCSABuilder to build the partial indexes.
If there is just one sequence in the increment, character '\0' is considered a normal character. This requires setting multiple_sequences = false in the RLCSA constructor. Note that RLCSABuilder cannot be used to merge these indexes, as it assumes character '\0' an end of sequence marker.
LCP Support
-----------
The implementation includes experimental support for two representations of the LCP array: run-length encoded PLCP array and the sampled LCP array. sample_lcp and build_plcp can be used to build the representations. lcp_test was used in the experiments reported in [3].
Distribution-Aware Samples
--------------------------
The implementation of distribution-aware sampling [4] should be considered very experimental. The implementation currently works only with a single sequence, and only optimizes the samples for either locate or display. The following assumes that the first 8 characters of each pattern contain the weight of that pattern.
1) Build a RLCSA with regular samples for the data file.
2) Use rlcsa_test -i8 -l -w to create a distribution file.
3) Use sampler_test build the index with optimal samples (default, requires about max(32n', 4n + 12n') bytes of memory, where n' is the number of text positions with a positive locate frequency) or greedy samples (e.g. option -g0.5 selects half of the samples greedily and the rest at regular intervals). Option -t# allows using multiple threads to find (almost) optimal samples, improving the sampling speed significantly.
4) Use rlcsa_test -i8 -l -d -g10000 to generate 10000 random patterns according to the pattern weights and search for them.
There is also some experimental support for adapting the samples to the query distribution. The samples are stored in a hash table, and several different heuristics are used to determine if the located position should be sampled. Use rlcsa_test -a with samples generated by sampler_test -g0 to test the adaptive samples. The following lines in the RLCSA parameter line control the use of the heuristics:
CANDIDATE_SAMPLES = 1
Use two hash tables instead of one, making it more likely that frequent text positions remain in the hash table.
HALF_GREEDY_SAMPLES = 1
Store half of the samples in the normal sample structure, guaranteeing worst-case performance and allowing display() queries in addition to locate().
SAMPLE_PROMOTE_RATE = r
SAMPLE_WINDOW_SIZE = w
Maintain a running average s of the number of steps required to find the sample in w previous queries. Sample the located position with probability x / (rs), where x is the distance to the sample used to locate the position.
Document Listing
----------------
The support for precomputed answers for document listing queries is very preliminary. The code for finding the bicliques used to build the grammar is not included in this package.
There are three phases in building the document listing structure:
document_graph base_name b \beta
This builds most of the structures and the graph used for building the grammar. Parameters b and \beta are explained in [5]. The finished structures are written to base_name.rlcsa.docs, while base_name.graph will contain the graph. Blocks containing only one document identifier are stored in base_name.singletons.
In the second phase, one should build the grammar. After this phase, the grammar rules should be found in files prefix-biclique-it-#.txt, where prefix is the chosen prefix and # is a number starting from 0. Document identifiers not included in any of the grammar rules should be found in files prefix-it-X, where X is the number of the last grammar file, and prefix.singletons (the singleton file built in the previous phase).
Finally,
document_graph base_name prefix
builds the grammar and encodes the blocks, storing the results in base_name.rlcsa.docs.
Use rlcsa_test -L to test document listing queries using the precomputed answers, or rlcsa_test -L -d to run the queries using the brute force solution.
Other Programs
--------------
rlcsa_test is a count/locate test program. It assumes that the pattern file is in Pizza & Chili format (-p) or contains one pattern per line. If the first m characters of each pattern contain a numerical weight, then parameters -im -gn can be used to generate n random patterns from the distribution specified by the weights. Parameter -W writes the actual patterns into a file, while -w writes the distribution of located positions for use with weighted sampling. Parameter -S uses a plain suffix array (built by build_sa) instead of RLCSA. Parameter -d does the locate/list query directly without resulting to run-length optimizations (locate) or the document listing structure (list). Parameter -o writes the patterns into a file, sorted by the occ/docc ratio in decreasing order.
display_test is a display test program. It extracts random substrings according to a distribution generated by rlcsa_test -w.
extract_sequence can be used to extract individual sequences from the index.
build_sa can be used to build a regular suffix array.
The rest of the programs have not been used recently. They might no longer work correctly.
Technical Information
=====================
A collection of sequences
-------------------------
The index contains a collection C of sequences T1, T2, ..., Tr. When constructing the index, each Ti is assumed to be followed by an implicit end of sequence marker $. The markers are assumed to be less than any character in the alphabet, and their mutual order is defined by sequence numbers. In some construction options, these end markers are represented explicitly by \0 characters.
RLCSA uses internally three kinds of ranges: BWT ranges, suffix array ranges, and text ranges.
BWT ranges are used internally for computing Psi and LF. The first r positions correspond to the suffixes starting with end markers.
Suffix array ranges are used in the query interface. Suffix array position i corresponds to BWT position i + r. Suffixes starting with end markers are not included in the suffix array, as the sequence order and hence the characters following the end markers are not well defined.
Text ranges are also used in the query interface. End markers are not included in the sequences. Every sequence is padded with empty characters, so that its length is a multiple of sample rate d. These padded sequences are implicitly concatenated. Text ranges corresponding to sequences do not include the padding.
Bit vector E is used to mark the last character of each sequence. The starting position of sequence k > 0 is d * ((E.select(k - 1) / d) + 1).
Suffix array samples
--------------------
For a given sample rate d, we store the positions of each sequence divisible by d. The sampled positions of suffix array are marked in a bit vector S, while the multipliers of d are stored in an array A in the same order. Another array B contains the inverse permutation of A.
When locating, we can use S.valueAfter(i - r) to get (j, k = S.rank(j) - 1) for the first sampled j >= i - r in the suffix array order. If j == i - r, we can get the suffix array value as d * A[k]. If i < r, we have reached the implicit end of sequence marker. In this case, the suffix array value is E.select(i) + 1 (this value should only be used to derive SA values for earlier positions). Note that i is BWT position, not SA position.
When displaying text starting from position i, j = B[i / d] gives us the sample used as a starting point. The sample is located in SA position k = S.select(j) corresponding to BWT position k + r.
Data formats
------------
.rlcsa.array
distribution of characters (CHARS * sizeof(usint) bytes)
RLEVector or NibbleVector for each character appearing in the text
DeltaVector E
sample rate d (sizeof(usint) bytes)
.rlcsa.sa_samples
DeltaVector or SuccinctVector S
array A (number_of_samples items of length(number_of_samples - 1) bits)
Any bit vector
universe size (sizeof(usint) bytes)
item count (sizeof(usint) bytes)
number of blocks (sizeof(usint) bytes)
block size in words (sizeof(usint) bytes)
block data (number_of_blocks * block_size words)
sample array (2 * (number_of_blocks + 1) items of length(size) bits)
Note that the samples are not stored for a succinct bit vector. Any bit vector must have at least one 1-bit.
Array
item count (sizeof(usint) bytes)
number of blocks (sizeof(usint) bytes)
block size in words (sizeof(usint) bytes)
block data (number_of_blocks * block_size words)
sample array (2 * (number_of_blocks + 1) items of length(size) bits)
MultiArray
flags (sizeof(usint) bytes)
SuccinctVector marking array borders
in FixedMultiArray:
an array of usints storing the elements (number of items from array borders, item bits from flags)
in DeltaMultiArray:
Array containing the items
LCP Information
---------------
The (P)LCP representations have been generalized to support multiple sequences. As the end markers are not included in the collection, the LCP values corresponding to the last characters of the sequences can be 1 or 0. The padding characters between the sequences are also assigned LCP values in the PLCP representation to ease its use. The sampled LCP array is used in a similar way as the SA samples in locate.
Data formats:
.lcp_samples
DeltaVector or SuccinctVector for the sampled positions
Array for the sampled values
.plcp
RLEVector or SuccinctVector
Array
item count (sizeof(usint) bytes)
number of blocks (sizeof(usint) bytes)
block size in words (sizeof(usint) bytes)
block data (number_of_blocks * block_size words)
file.write((char*)&(this->number_of_blocks), sizeof(this->number_of_blocks));
file.write((char*)&(this->block_size), sizeof(this->block_size));
file.write((char*)(this->array), this->block_size * this->number_of_blocks * sizeof(usint));
sample array (number_of_blocks + 1 items of length(items) bits)
Weighted Samples
----------------
RLCSA now experimentally supports weighted or distribution-aware samples. To use them, the index must be built with sampler_test. A weight file generated by rlcsa_test -w is required for construction. The suffix weights are assumed to represent a distribution of locate queries (or display queries, if parameter -b is used).
There are two options: optimal sampling and greedy sampling. Optimal sampling requires about 28n bytes of memory for a text of length n, and takes considerably more time than index construction. With option -g, some of the samples are selected greedily according to suffix weights, and the rest at regular intervals. In both cases, the sampler aims to select n / d samples. If there are less suffixes with positive weights, then only those suffixes are sampled.
To use weighted samples, MASSIVE_DATA_RLCSA must be set, as the largest integers used when selecting the optimal samples are roughly n^2 / 2 times the average suffix weight. Integer overflows might still occur, but as only the least significant bits of the integers are used, the results will generally be ok.
Parameter -w can be used to write just the sampled positions for use with another index. If the index uses LF instead of Psi, then the default is to sample for display, and parameter -b is used for locate.
Only one sequence is currently supported, and the weighted samples cannot be merged.
The current implementations uses the same samples for both locate and display. It would be preferable to be able to select them separately. For locate, bit vector S stores the sampled SA positions, and array A contains the sampled SA values. For display, bit vector S' stores the sampled text positions, and array B contains the sampled inverse SA values. A possible size optimization similar to the one used in standard sampling (where the values of A and B have been divided by d) would be to use
SA[i] = select(S', A[rank(S, i)]), SA^-1[j] = select(S, B[rank(S', j]).
The weighted samples can also be used in adaptive way in rlcsa_test (parameter -a). The initial samples are insterted into a hash table. When a new position is located, it is inserted into the hash table, overwriting any existing sample. This mechanism does not currently perform very well.
Data formats:
.rlcsa.sa_samples (samples)
text length (sizeof(usint) bytes)
item count (sizeof(usint) bytes)
samples as (i, SA[i]) pairs (number of samples * sizeof(pair_type) bytes)
.rlcsa.sa_samples (sampled positions)
sampled text positions (number_of_samples * sizeof(uint) bytes)
Document Listing
----------------
See [5] for the definition of the graph and for the names of the data structures. If there are d documents, nodes corresponding to documents are numbered as 0..d-1, while nodes corresponding to blocks are numbered starting from d.
In the encoding of the blocks, document identifiers are numbered as 0..d-1, value d indicates a block containing all possible documents, and the rule identifiers are numbered starting from d+1. Similarly, in the encoding of the grammar rules, value d indicates a rule expanding to all documents.
Data formats:
.graph
number of nodes (sizeof(uint) bytes)
number of edges (sizeof(uint) bytes)
for each node with multiple outgoing edges in an arbitrary order:
-1 * node identifier (sizeof(int) bytes)
for each outgoing edge in an arbitrary order:
destination node identifier (sizeof(int) bytes)
File containing the edges not encoded with any grammar rule (also .singletons)
for each node having outgoing edges in an arbitrary order:
"%u:", node identifier
for each outgoing edge in an arbitrary order:
" %u", destination node identifier
"\n"
Grammar rule files
for each grammar rule in an arbitrary order:
for each block containing the rule in sorted order:
"%u ", node identifier
"-"
for each document encoded by the rule in sorted order:
" %u", document identifier
"\n"
.rlcsa.docs
flags (sizeof(usint) bytes)
0x01 - are grammar rules encoded using RLE
DeltaVector B_L
SuccinctVector B_F
array F storing pointers to the parents of the first children (I items of length(I - 1) bits)
array N storing pointers to the leaf nodes following and the internal nodes (I items of length(L) bits)
The following are stored once the grammar has been generated:
SuccinctVector B_G
array G containing the rules; a sequence of the following
document identifier (length(d) bits)
the number of successive document identifiers including the first one, if using RLE (length(d) bits)
SuccinctVector B_A
array A containing document identifiers and rule identifiers (items of length(d + n_R) bits)
References
==========
[1] Veli Mäkinen, Gonzalo Navarro, Jouni Sirén, and Niko Välimäki: Storage and Retrieval of Highly Repetitive Sequence Collections.
Journal of Computational Biology 17(3):281-308, 2010.
[2] Jouni Sirén: Compressed Suffix Arrays for Massive Data.
In SPIRE 2009, Springer LNCS 5721, pp. 63-74, Saariselkä, Finland, August 25-27, 2009.
[3] Jouni Sirén: Sampled Longest Common Prefix Array.
In CPM 2010, Springer LNCS 6129, pp. 227-237, New York, USA, June 21-23, 2010.
[4] Paolo Ferragina, Jouni Sirén, and Rossano Venturini: Distribution-aware compressed full-text indexes.
Algorithmica 67(4):529-546, 2013.
[5] Travis Gagie, Kalle Karhu, Gonzalo Navarro, Simon J. Puglisi, and Jouni Sirén: Document Listing on Repetitive Collections.
In CPM 2013, Springer LNCS 7922, pp. 107-119, Bad Herrenalb, Germany, June 17-19, 2013.