2013-02-21 18:48:23 +04:00
|
|
|
#pragma once
|
|
|
|
|
2013-05-24 21:02:49 +04:00
|
|
|
#include "moses/FF/FFState.h"
|
2013-02-21 18:48:23 +04:00
|
|
|
#include "moses/ChartHypothesis.h"
|
|
|
|
#include "moses/ChartManager.h"
|
|
|
|
|
|
|
|
namespace Moses
|
|
|
|
{
|
|
|
|
|
|
|
|
class LanguageModelChartState : public FFState
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
float m_prefixScore;
|
|
|
|
FFState* m_lmRightContext;
|
|
|
|
|
|
|
|
Phrase m_contextPrefix, m_contextSuffix;
|
|
|
|
|
|
|
|
size_t m_numTargetTerminals; // This isn't really correct except for the surviving hypothesis
|
|
|
|
|
|
|
|
const ChartHypothesis &m_hypo;
|
|
|
|
|
2013-05-29 21:16:15 +04:00
|
|
|
/** Construct the prefix string of up to specified size
|
2013-02-21 18:48:23 +04:00
|
|
|
* \param ret prefix string
|
|
|
|
* \param size maximum size (typically max lm context window)
|
|
|
|
*/
|
2013-05-29 21:16:15 +04:00
|
|
|
size_t CalcPrefix(const ChartHypothesis &hypo, int featureID, Phrase &ret, size_t size) const {
|
2013-02-21 18:48:23 +04:00
|
|
|
const TargetPhrase &target = hypo.GetCurrTargetPhrase();
|
|
|
|
const AlignmentInfo::NonTermIndexMap &nonTermIndexMap =
|
2013-05-29 21:16:15 +04:00
|
|
|
target.GetAlignNonTerm().GetNonTermIndexMap();
|
|
|
|
|
2013-02-21 18:48:23 +04:00
|
|
|
// loop over the rule that is being applied
|
|
|
|
for (size_t pos = 0; pos < target.GetSize(); ++pos) {
|
|
|
|
const Word &word = target.GetWord(pos);
|
|
|
|
|
|
|
|
// for non-terminals, retrieve it from underlying hypothesis
|
|
|
|
if (word.IsNonTerminal()) {
|
|
|
|
size_t nonTermInd = nonTermIndexMap[pos];
|
|
|
|
const ChartHypothesis *prevHypo = hypo.GetPrevHypo(nonTermInd);
|
|
|
|
size = static_cast<const LanguageModelChartState*>(prevHypo->GetFFState(featureID))->CalcPrefix(*prevHypo, featureID, ret, size);
|
|
|
|
}
|
|
|
|
// for words, add word
|
|
|
|
else {
|
|
|
|
ret.AddWord(target.GetWord(pos));
|
|
|
|
size--;
|
|
|
|
}
|
|
|
|
|
|
|
|
// finish when maximum length reached
|
|
|
|
if (size==0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
2013-05-29 21:16:15 +04:00
|
|
|
/** Construct the suffix phrase of up to specified size
|
2013-02-21 18:48:23 +04:00
|
|
|
* will always be called after the construction of prefix phrase
|
|
|
|
* \param ret suffix phrase
|
|
|
|
* \param size maximum size of suffix
|
|
|
|
*/
|
2013-05-29 21:16:15 +04:00
|
|
|
size_t CalcSuffix(const ChartHypothesis &hypo, int featureID, Phrase &ret, size_t size) const {
|
2013-11-23 00:27:46 +04:00
|
|
|
UTIL_THROW_IF2(m_contextPrefix.GetSize() > m_numTargetTerminals, "Error");
|
2013-02-21 18:48:23 +04:00
|
|
|
|
|
|
|
// special handling for small hypotheses
|
|
|
|
// does the prefix match the entire hypothesis string? -> just copy prefix
|
|
|
|
if (m_contextPrefix.GetSize() == m_numTargetTerminals) {
|
|
|
|
size_t maxCount = std::min(m_contextPrefix.GetSize(), size);
|
|
|
|
size_t pos= m_contextPrefix.GetSize() - 1;
|
|
|
|
|
|
|
|
for (size_t ind = 0; ind < maxCount; ++ind) {
|
|
|
|
const Word &word = m_contextPrefix.GetWord(pos);
|
|
|
|
ret.PrependWord(word);
|
|
|
|
--pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
size -= maxCount;
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
// construct suffix analogous to prefix
|
|
|
|
else {
|
|
|
|
const TargetPhrase& target = hypo.GetCurrTargetPhrase();
|
|
|
|
const AlignmentInfo::NonTermIndexMap &nonTermIndexMap =
|
2013-05-29 21:16:15 +04:00
|
|
|
target.GetAlignNonTerm().GetNonTermIndexMap();
|
2013-02-21 18:48:23 +04:00
|
|
|
for (int pos = (int) target.GetSize() - 1; pos >= 0 ; --pos) {
|
|
|
|
const Word &word = target.GetWord(pos);
|
|
|
|
|
|
|
|
if (word.IsNonTerminal()) {
|
|
|
|
size_t nonTermInd = nonTermIndexMap[pos];
|
|
|
|
const ChartHypothesis *prevHypo = hypo.GetPrevHypo(nonTermInd);
|
|
|
|
size = static_cast<const LanguageModelChartState*>(prevHypo->GetFFState(featureID))->CalcSuffix(*prevHypo, featureID, ret, size);
|
2013-05-29 21:16:15 +04:00
|
|
|
} else {
|
2013-02-21 18:48:23 +04:00
|
|
|
ret.PrependWord(hypo.GetCurrTargetPhrase().GetWord(pos));
|
|
|
|
size--;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (size==0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
LanguageModelChartState(const ChartHypothesis &hypo, int featureID, size_t order)
|
2013-05-29 21:16:15 +04:00
|
|
|
:m_lmRightContext(NULL)
|
|
|
|
,m_contextPrefix(order - 1)
|
|
|
|
,m_contextSuffix( order - 1)
|
|
|
|
,m_hypo(hypo) {
|
2013-02-21 18:48:23 +04:00
|
|
|
m_numTargetTerminals = hypo.GetCurrTargetPhrase().GetNumTerminals();
|
|
|
|
|
|
|
|
for (std::vector<const ChartHypothesis*>::const_iterator i = hypo.GetPrevHypos().begin(); i != hypo.GetPrevHypos().end(); ++i) {
|
|
|
|
// keep count of words (= length of generated string)
|
|
|
|
m_numTargetTerminals += static_cast<const LanguageModelChartState*>((*i)->GetFFState(featureID))->GetNumTargetTerminals();
|
|
|
|
}
|
|
|
|
|
|
|
|
CalcPrefix(hypo, featureID, m_contextPrefix, order - 1);
|
|
|
|
CalcSuffix(hypo, featureID, m_contextSuffix, order - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
~LanguageModelChartState() {
|
|
|
|
delete m_lmRightContext;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Set(float prefixScore, FFState *rightState) {
|
|
|
|
m_prefixScore = prefixScore;
|
|
|
|
m_lmRightContext = rightState;
|
|
|
|
}
|
|
|
|
|
2013-05-29 21:16:15 +04:00
|
|
|
float GetPrefixScore() const {
|
|
|
|
return m_prefixScore;
|
|
|
|
}
|
|
|
|
FFState* GetRightContext() const {
|
|
|
|
return m_lmRightContext;
|
|
|
|
}
|
2013-02-21 18:48:23 +04:00
|
|
|
|
|
|
|
size_t GetNumTargetTerminals() const {
|
|
|
|
return m_numTargetTerminals;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Phrase &GetPrefix() const {
|
|
|
|
return m_contextPrefix;
|
|
|
|
}
|
|
|
|
const Phrase &GetSuffix() const {
|
|
|
|
return m_contextSuffix;
|
|
|
|
}
|
|
|
|
|
2015-10-16 15:53:33 +03:00
|
|
|
size_t hash() const {
|
|
|
|
size_t ret;
|
2015-10-14 17:21:37 +03:00
|
|
|
|
2015-10-16 15:53:33 +03:00
|
|
|
// prefix
|
|
|
|
ret = m_hypo.GetCurrSourceRange().GetStartPos() > 0;
|
2015-10-14 17:21:37 +03:00
|
|
|
if (m_hypo.GetCurrSourceRange().GetStartPos() > 0) { // not for "<s> ..."
|
2015-10-16 15:53:33 +03:00
|
|
|
size_t hash = hash_value(GetPrefix());
|
|
|
|
boost::hash_combine(ret, hash);
|
2015-10-14 17:21:37 +03:00
|
|
|
}
|
|
|
|
|
2015-10-16 15:53:33 +03:00
|
|
|
// suffix
|
|
|
|
size_t inputSize = m_hypo.GetManager().GetSource().GetSize();
|
2015-10-14 17:21:37 +03:00
|
|
|
boost::hash_combine(ret, m_hypo.GetCurrSourceRange().GetEndPos() < inputSize - 1);
|
|
|
|
if (m_hypo.GetCurrSourceRange().GetEndPos() < inputSize - 1) { // not for "... </s>"
|
2015-10-16 15:53:33 +03:00
|
|
|
size_t hash = m_lmRightContext->hash();
|
|
|
|
boost::hash_combine(ret, hash);
|
2015-10-14 17:21:37 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
2015-10-08 19:14:08 +03:00
|
|
|
}
|
2015-10-16 15:53:33 +03:00
|
|
|
virtual bool operator==(const FFState& o) const {
|
|
|
|
const LanguageModelChartState &other =
|
2015-10-18 15:30:02 +03:00
|
|
|
static_cast<const LanguageModelChartState &>( o );
|
2015-10-16 15:53:33 +03:00
|
|
|
|
|
|
|
// prefix
|
|
|
|
if (m_hypo.GetCurrSourceRange().GetStartPos() > 0) { // not for "<s> ..."
|
|
|
|
bool ret = GetPrefix() == other.GetPrefix();
|
|
|
|
if (ret == false)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// suffix
|
|
|
|
size_t inputSize = m_hypo.GetManager().GetSource().GetSize();
|
|
|
|
if (m_hypo.GetCurrSourceRange().GetEndPos() < inputSize - 1) { // not for "... </s>"
|
|
|
|
bool ret = (*other.GetRightContext()) == (*m_lmRightContext);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
return true;
|
2015-10-08 20:14:52 +03:00
|
|
|
}
|
2015-10-08 19:14:08 +03:00
|
|
|
|
2013-02-21 18:48:23 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|