#include <algorithm>
#include <functional>
#include <utility>
#include "tail.h"
namespace marisa {
Tail::Tail() : buf_() {}
void Tail::build(const Vector<String> &keys,
Vector<UInt32> *offsets, int mode) {
switch (mode) {
case MARISA_BINARY_TAIL: {
build_binary_tail(keys, offsets);
return;
}
case MARISA_TEXT_TAIL: {
if (!build_text_tail(keys, offsets)) {
build_binary_tail(keys, offsets);
}
return;
}
default: {
MARISA_THROW(MARISA_PARAM_ERROR);
}
}
}
void Tail::mmap(Mapper *mapper, const char *filename,
long offset, int whence) {
if (mapper == NULL) {
MARISA_THROW(MARISA_PARAM_ERROR);
}
Mapper temp_mapper;
temp_mapper.open(filename, offset, whence);
map(temp_mapper);
temp_mapper.swap(mapper);
}
void Tail::map(const void *ptr, std::size_t size) {
Mapper mapper(ptr, size);
map(mapper);
}
void Tail::map(Mapper &mapper) {
Tail temp;
temp.buf_.map(mapper);
temp.swap(this);
}
void Tail::load(const char *filename, long offset, int whence) {
Reader reader;
reader.open(filename, offset, whence);
read(reader);
}
void Tail::fread(::FILE *file) {
Reader reader(file);
read(reader);
}
void Tail::read(int fd) {
Reader reader(fd);
read(reader);
}
void Tail::read(std::istream &stream) {
Reader reader(&stream);
read(reader);
}
void Tail::read(Reader &reader) {
Tail temp;
temp.buf_.read(reader);
temp.swap(this);
}
void Tail::save(const char *filename, bool trunc_flag,
long offset, int whence) const {
Writer writer;
writer.open(filename, trunc_flag, offset, whence);
write(writer);
}
void Tail::fwrite(::FILE *file) const {
Writer writer(file);
write(writer);
}
void Tail::write(int fd) const {
Writer writer(fd);
write(writer);
}
void Tail::write(std::ostream &stream) const {
Writer writer(&stream);
write(writer);
}
void Tail::write(Writer &writer) const {
buf_.write(writer);
}
void Tail::clear() {
Tail().swap(this);
}
void Tail::swap(Tail *rhs) {
buf_.swap(&rhs->buf_);
}
void Tail::build_binary_tail(const Vector<String> &keys,
Vector<UInt32> *offsets) {
if (keys.empty()) {
build_empty_tail(offsets);
return;
}
Vector<UInt8> buf;
buf.push_back('\0');
Vector<UInt32> temp_offsets;
temp_offsets.resize(keys.size() + 1);
for (std::size_t i = 0; i < keys.size(); ++i) {
temp_offsets[i] = (UInt32)buf.size();
for (std::size_t j = 0; j < keys[i].length(); ++j) {
buf.push_back(keys[i][j]);
}
}
temp_offsets.back() = (UInt32)buf.size();
buf.shrink();
if (offsets != NULL) {
temp_offsets.swap(offsets);
}
buf_.swap(&buf);
}
bool Tail::build_text_tail(const Vector<String> &keys,
Vector<UInt32> *offsets) {
if (keys.empty()) {
build_empty_tail(offsets);
return true;
}
typedef std::pair<RString, UInt32> KeyIdPair;
Vector<KeyIdPair> pairs;
pairs.resize(keys.size());
for (std::size_t i = 0; i < keys.size(); ++i) {
for (std::size_t j = 0; j < keys[i].length(); ++j) {
if (keys[i][j] == '\0') {
return false;
}
}
pairs[i].first = RString(keys[i]);
pairs[i].second = (UInt32)i;
}
std::sort(pairs.begin(), pairs.end(), std::greater<KeyIdPair>());
Vector<UInt8> buf;
buf.push_back('T');
Vector<UInt32> temp_offsets;
temp_offsets.resize(pairs.size(), 1);
const KeyIdPair dummy_key;
const KeyIdPair *last = &dummy_key;
for (std::size_t i = 0; i < pairs.size(); ++i) {
const KeyIdPair &cur = pairs[i];
std::size_t match = 0;
while ((match < cur.first.length()) && (match < last->first.length()) &&
last->first[match] == cur.first[match]) {
++match;
}
if ((match == cur.first.length()) && (last->first.length() != 0)) {
temp_offsets[cur.second] = (UInt32)(temp_offsets[last->second]
+ (last->first.length() - match));
} else {
temp_offsets[cur.second] = (UInt32)buf.size();
for (std::size_t j = 1; j <= cur.first.length(); ++j) {
buf.push_back(cur.first[cur.first.length() - j]);
}
buf.push_back('\0');
}
last = &cur;
}
buf.shrink();
if (offsets != NULL) {
temp_offsets.swap(offsets);
}
buf_.swap(&buf);
return true;
}
void Tail::build_empty_tail(Vector<UInt32> *offsets) {
buf_.clear();
if (offsets != NULL) {
offsets->clear();
}
}
} // namespace marisa