2012-10-15 16:58:33 +04:00
|
|
|
#ifndef SEARCH_VERTEX__
|
|
|
|
#define SEARCH_VERTEX__
|
|
|
|
|
|
|
|
#include "lm/left.hh"
|
|
|
|
#include "search/types.hh"
|
|
|
|
|
|
|
|
#include <boost/unordered_set.hpp>
|
|
|
|
|
|
|
|
#include <queue>
|
|
|
|
#include <vector>
|
2015-03-28 15:48:20 +03:00
|
|
|
#include <cmath>
|
2012-10-15 16:58:33 +04:00
|
|
|
#include <stdint.h>
|
|
|
|
|
|
|
|
namespace search {
|
|
|
|
|
|
|
|
class ContextBase;
|
|
|
|
|
2013-02-14 17:11:53 +04:00
|
|
|
struct HypoState {
|
|
|
|
History history;
|
|
|
|
lm::ngram::ChartState state;
|
|
|
|
Score score;
|
|
|
|
};
|
|
|
|
|
2012-10-15 16:58:33 +04:00
|
|
|
class VertexNode {
|
|
|
|
public:
|
2013-02-14 17:11:53 +04:00
|
|
|
VertexNode() {}
|
|
|
|
|
|
|
|
void InitRoot() { hypos_.clear(); }
|
|
|
|
|
|
|
|
/* The steps of building a VertexNode:
|
|
|
|
* 1. Default construct.
|
|
|
|
* 2. AppendHypothesis at least once, possibly multiple times.
|
|
|
|
* 3. FinishAppending with the number of words on left and right guaranteed
|
|
|
|
* to be common.
|
|
|
|
* 4. If !Complete(), call BuildExtend to construct the extensions
|
|
|
|
*/
|
|
|
|
// Must default construct, call AppendHypothesis 1 or more times then do FinishedAppending.
|
|
|
|
void AppendHypothesis(const NBestComplete &best) {
|
|
|
|
assert(hypos_.empty() || !(hypos_.front().state == *best.state));
|
|
|
|
HypoState hypo;
|
|
|
|
hypo.history = best.history;
|
|
|
|
hypo.state = *best.state;
|
|
|
|
hypo.score = best.score;
|
|
|
|
hypos_.push_back(hypo);
|
|
|
|
}
|
|
|
|
void AppendHypothesis(const HypoState &hypo) {
|
|
|
|
hypos_.push_back(hypo);
|
2012-10-15 16:58:33 +04:00
|
|
|
}
|
|
|
|
|
2013-02-14 17:11:53 +04:00
|
|
|
// Sort hypotheses for the root.
|
|
|
|
void FinishRoot();
|
2012-10-15 16:58:33 +04:00
|
|
|
|
2013-02-14 17:11:53 +04:00
|
|
|
void FinishedAppending(const unsigned char common_left, const unsigned char common_right);
|
2012-10-15 16:58:33 +04:00
|
|
|
|
2013-02-14 17:11:53 +04:00
|
|
|
void BuildExtend();
|
2012-10-15 16:58:33 +04:00
|
|
|
|
2015-04-30 08:05:11 +03:00
|
|
|
// Should only happen to a root node when the entire vertex is empty.
|
2012-10-15 16:58:33 +04:00
|
|
|
bool Empty() const {
|
2013-02-14 17:11:53 +04:00
|
|
|
return hypos_.empty() && extend_.empty();
|
2012-10-15 16:58:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Complete() const {
|
2013-02-14 17:11:53 +04:00
|
|
|
// HACK: prevent root from being complete. TODO: allow root to be complete.
|
|
|
|
return hypos_.size() == 1 && extend_.empty();
|
2012-10-15 16:58:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
const lm::ngram::ChartState &State() const { return state_; }
|
|
|
|
bool RightFull() const { return right_full_; }
|
|
|
|
|
2013-02-14 17:11:53 +04:00
|
|
|
// Priority relative to other non-terminals. 0 is highest.
|
|
|
|
unsigned char Niceness() const { return niceness_; }
|
|
|
|
|
2012-10-15 16:58:33 +04:00
|
|
|
Score Bound() const {
|
|
|
|
return bound_;
|
|
|
|
}
|
|
|
|
|
2015-04-30 08:05:11 +03:00
|
|
|
// Will be invalid unless this is a leaf.
|
2013-02-14 17:11:53 +04:00
|
|
|
const History End() const {
|
|
|
|
assert(hypos_.size() == 1);
|
|
|
|
return hypos_.front().history;
|
|
|
|
}
|
2012-10-15 16:58:33 +04:00
|
|
|
|
2013-02-14 17:11:53 +04:00
|
|
|
VertexNode &operator[](size_t index) {
|
|
|
|
assert(!extend_.empty());
|
|
|
|
return extend_[index];
|
2012-10-15 16:58:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
size_t Size() const {
|
|
|
|
return extend_.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2013-02-14 17:11:53 +04:00
|
|
|
// Hypotheses to be split.
|
|
|
|
std::vector<HypoState> hypos_;
|
2012-11-15 22:04:07 +04:00
|
|
|
|
2013-02-14 17:11:53 +04:00
|
|
|
std::vector<VertexNode> extend_;
|
2012-10-15 16:58:33 +04:00
|
|
|
|
|
|
|
lm::ngram::ChartState state_;
|
|
|
|
bool right_full_;
|
|
|
|
|
2013-02-14 17:11:53 +04:00
|
|
|
unsigned char niceness_;
|
|
|
|
|
|
|
|
unsigned char policy_;
|
|
|
|
|
2012-10-15 16:58:33 +04:00
|
|
|
Score bound_;
|
|
|
|
};
|
|
|
|
|
|
|
|
class PartialVertex {
|
|
|
|
public:
|
|
|
|
PartialVertex() {}
|
|
|
|
|
2013-02-14 17:11:53 +04:00
|
|
|
explicit PartialVertex(VertexNode &back) : back_(&back), index_(0) {}
|
2012-10-15 16:58:33 +04:00
|
|
|
|
|
|
|
bool Empty() const { return back_->Empty(); }
|
|
|
|
|
|
|
|
bool Complete() const { return back_->Complete(); }
|
|
|
|
|
|
|
|
const lm::ngram::ChartState &State() const { return back_->State(); }
|
|
|
|
bool RightFull() const { return back_->RightFull(); }
|
|
|
|
|
2013-02-14 17:11:53 +04:00
|
|
|
Score Bound() const { return index_ ? (*back_)[index_].Bound() : back_->Bound(); }
|
2012-10-15 16:58:33 +04:00
|
|
|
|
2013-02-14 17:11:53 +04:00
|
|
|
unsigned char Niceness() const { return back_->Niceness(); }
|
2012-10-15 16:58:33 +04:00
|
|
|
|
2012-10-17 16:50:08 +04:00
|
|
|
// Split into continuation and alternative, rendering this the continuation.
|
|
|
|
bool Split(PartialVertex &alternative) {
|
2012-10-15 16:58:33 +04:00
|
|
|
assert(!Complete());
|
2013-02-14 17:11:53 +04:00
|
|
|
back_->BuildExtend();
|
2012-10-17 16:50:08 +04:00
|
|
|
bool ret;
|
2012-10-15 16:58:33 +04:00
|
|
|
if (index_ + 1 < back_->Size()) {
|
2012-10-17 16:50:08 +04:00
|
|
|
alternative.index_ = index_ + 1;
|
|
|
|
alternative.back_ = back_;
|
|
|
|
ret = true;
|
|
|
|
} else {
|
|
|
|
ret = false;
|
2012-10-15 16:58:33 +04:00
|
|
|
}
|
2012-10-17 16:50:08 +04:00
|
|
|
back_ = &((*back_)[index_]);
|
|
|
|
index_ = 0;
|
|
|
|
return ret;
|
2012-10-15 16:58:33 +04:00
|
|
|
}
|
|
|
|
|
2012-11-15 22:04:07 +04:00
|
|
|
const History End() const {
|
2012-10-18 21:54:38 +04:00
|
|
|
return back_->End();
|
2012-10-15 16:58:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2013-02-14 17:11:53 +04:00
|
|
|
VertexNode *back_;
|
2012-10-15 16:58:33 +04:00
|
|
|
unsigned int index_;
|
|
|
|
};
|
|
|
|
|
2012-11-15 22:04:07 +04:00
|
|
|
template <class Output> class VertexGenerator;
|
|
|
|
|
2012-10-15 16:58:33 +04:00
|
|
|
class Vertex {
|
|
|
|
public:
|
|
|
|
Vertex() {}
|
|
|
|
|
2013-02-14 17:11:53 +04:00
|
|
|
//PartialVertex RootFirst() const { return PartialVertex(right_); }
|
|
|
|
PartialVertex RootAlternate() { return PartialVertex(root_); }
|
|
|
|
//PartialVertex RootLast() const { return PartialVertex(left_); }
|
|
|
|
|
|
|
|
bool Empty() const {
|
|
|
|
return root_.Empty();
|
|
|
|
}
|
|
|
|
|
|
|
|
Score Bound() const {
|
|
|
|
return root_.Bound();
|
|
|
|
}
|
2012-10-15 16:58:33 +04:00
|
|
|
|
2013-02-14 17:11:53 +04:00
|
|
|
const History BestChild() {
|
|
|
|
// left_ and right_ are not set at the root.
|
|
|
|
PartialVertex top(RootAlternate());
|
2012-10-15 16:58:33 +04:00
|
|
|
if (top.Empty()) {
|
2012-11-15 22:04:07 +04:00
|
|
|
return History();
|
2012-10-15 16:58:33 +04:00
|
|
|
} else {
|
|
|
|
PartialVertex continuation;
|
|
|
|
while (!top.Complete()) {
|
|
|
|
top.Split(continuation);
|
|
|
|
}
|
2012-10-18 21:54:38 +04:00
|
|
|
return top.End();
|
2012-10-15 16:58:33 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2012-11-15 22:04:07 +04:00
|
|
|
template <class Output> friend class VertexGenerator;
|
|
|
|
template <class Output> friend class RootVertexGenerator;
|
2012-10-15 16:58:33 +04:00
|
|
|
VertexNode root_;
|
2013-02-14 17:11:53 +04:00
|
|
|
|
|
|
|
// These will not be set for the root vertex.
|
|
|
|
// Branches only on left state.
|
|
|
|
//VertexNode left_;
|
|
|
|
// Branches only on right state.
|
|
|
|
//VertexNode right_;
|
2012-10-15 16:58:33 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace search
|
|
|
|
#endif // SEARCH_VERTEX__
|