// Copyright 2005-2024 Google LLC
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the 'License');
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an 'AS IS' BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
//
|
|
// See www.openfst.org for extensive documentation on this weighted
|
|
// finite-state transducer library.
|
|
//
|
|
// Functions and classes to project an FST on to its domain or range.
|
|
|
|
#ifndef FST_PROJECT_H_
|
|
#define FST_PROJECT_H_
|
|
|
|
#include <cstdint>
|
|
|
|
#include <fst/arc-map.h>
|
|
#include <fst/arc.h>
|
|
#include <fst/cache.h>
|
|
#include <fst/float-weight.h>
|
|
#include <fst/fst.h>
|
|
#include <fst/impl-to-fst.h>
|
|
#include <fst/mutable-fst.h>
|
|
#include <fst/properties.h>
|
|
|
|
namespace fst {
|
|
|
|
// This specifies whether to project on input or output.
|
|
enum class ProjectType { INPUT = 1, OUTPUT = 2 };
|
|
OPENFST_DEPRECATED("Use `ProjectType::INPUT` instead.")
|
|
inline constexpr ProjectType PROJECT_INPUT = ProjectType::INPUT;
|
|
OPENFST_DEPRECATED("Use `ProjectType::OUTPUT` instead.")
|
|
inline constexpr ProjectType PROJECT_OUTPUT = ProjectType::OUTPUT;
|
|
|
|
// Mapper to implement projection per arc.
|
|
template <class A>
|
|
class ProjectMapper {
|
|
public:
|
|
using FromArc = A;
|
|
using ToArc = A;
|
|
|
|
constexpr explicit ProjectMapper(ProjectType project_type)
|
|
: project_type_(project_type) {}
|
|
|
|
ToArc operator()(const FromArc &arc) const {
|
|
const auto label =
|
|
project_type_ == ProjectType::INPUT ? arc.ilabel : arc.olabel;
|
|
return ToArc(label, label, arc.weight, arc.nextstate);
|
|
}
|
|
|
|
constexpr MapFinalAction FinalAction() const { return MAP_NO_SUPERFINAL; }
|
|
|
|
constexpr MapSymbolsAction InputSymbolsAction() const {
|
|
return project_type_ == ProjectType::INPUT ? MAP_COPY_SYMBOLS
|
|
: MAP_CLEAR_SYMBOLS;
|
|
}
|
|
|
|
constexpr MapSymbolsAction OutputSymbolsAction() const {
|
|
return project_type_ == ProjectType::OUTPUT ? MAP_COPY_SYMBOLS
|
|
: MAP_CLEAR_SYMBOLS;
|
|
}
|
|
|
|
constexpr uint64_t Properties(uint64_t props) const {
|
|
return ProjectProperties(props, project_type_ == ProjectType::INPUT);
|
|
}
|
|
|
|
private:
|
|
const ProjectType project_type_;
|
|
};
|
|
|
|
// Projects an FST onto its domain or range by either copying each arcs' input
|
|
// label to the output label or vice versa.
|
|
//
|
|
// Complexity:
|
|
//
|
|
// Time: O(V + E)
|
|
// Space: O(1)
|
|
//
|
|
// where V is the number of states and E is the number of arcs.
|
|
template <class Arc>
|
|
inline void Project(const Fst<Arc> &ifst, MutableFst<Arc> *ofst,
|
|
ProjectType project_type) {
|
|
ArcMap(ifst, ofst, ProjectMapper<Arc>(project_type));
|
|
switch (project_type) {
|
|
case ProjectType::INPUT:
|
|
ofst->SetOutputSymbols(ifst.InputSymbols());
|
|
return;
|
|
case ProjectType::OUTPUT:
|
|
ofst->SetInputSymbols(ifst.OutputSymbols());
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Destructive variant of the above.
|
|
template <class Arc>
|
|
inline void Project(MutableFst<Arc> *fst, ProjectType project_type) {
|
|
ArcMap(fst, ProjectMapper<Arc>(project_type));
|
|
switch (project_type) {
|
|
case ProjectType::INPUT:
|
|
fst->SetOutputSymbols(fst->InputSymbols());
|
|
return;
|
|
case ProjectType::OUTPUT:
|
|
fst->SetInputSymbols(fst->OutputSymbols());
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Projects an FST onto its domain or range by either copying each arc's input
|
|
// label to the output label or vice versa. This version is a delayed FST.
|
|
//
|
|
// Complexity:
|
|
//
|
|
// Time: O(v + e)
|
|
// Space: O(1)
|
|
//
|
|
// where v is the number of states visited and e is the number of arcs visited.
|
|
// Constant time and to visit an input state or arc is assumed and exclusive of
|
|
// caching.
|
|
template <class A>
|
|
class ProjectFst : public ArcMapFst<A, A, ProjectMapper<A>> {
|
|
public:
|
|
using FromArc = A;
|
|
using ToArc = A;
|
|
|
|
using Impl = internal::ArcMapFstImpl<A, A, ProjectMapper<A>>;
|
|
|
|
ProjectFst(const Fst<A> &fst, ProjectType project_type)
|
|
: ArcMapFst<A, A, ProjectMapper<A>>(fst, ProjectMapper<A>(project_type)) {
|
|
if (project_type == ProjectType::INPUT) {
|
|
GetMutableImpl()->SetOutputSymbols(fst.InputSymbols());
|
|
}
|
|
if (project_type == ProjectType::OUTPUT) {
|
|
GetMutableImpl()->SetInputSymbols(fst.OutputSymbols());
|
|
}
|
|
}
|
|
|
|
// See Fst<>::Copy() for doc.
|
|
ProjectFst(const ProjectFst &fst, bool safe = false)
|
|
: ArcMapFst<A, A, ProjectMapper<A>>(fst, safe) {}
|
|
|
|
// Gets a copy of this ProjectFst. See Fst<>::Copy() for further doc.
|
|
ProjectFst *Copy(bool safe = false) const override {
|
|
return new ProjectFst(*this, safe);
|
|
}
|
|
|
|
private:
|
|
using ImplToFst<Impl>::GetMutableImpl;
|
|
};
|
|
|
|
// Specialization for ProjectFst.
|
|
template <class A>
|
|
class StateIterator<ProjectFst<A>>
|
|
: public StateIterator<ArcMapFst<A, A, ProjectMapper<A>>> {
|
|
public:
|
|
explicit StateIterator(const ProjectFst<A> &fst)
|
|
: StateIterator<ArcMapFst<A, A, ProjectMapper<A>>>(fst) {}
|
|
};
|
|
|
|
// Specialization for ProjectFst.
|
|
template <class A>
|
|
class ArcIterator<ProjectFst<A>>
|
|
: public ArcIterator<ArcMapFst<A, A, ProjectMapper<A>>> {
|
|
public:
|
|
using StateId = typename A::StateId;
|
|
|
|
ArcIterator(const ProjectFst<A> &fst, StateId s)
|
|
: ArcIterator<ArcMapFst<A, A, ProjectMapper<A>>>(fst, s) {}
|
|
};
|
|
|
|
// Useful alias when using StdArc.
|
|
using StdProjectFst = ProjectFst<StdArc>;
|
|
|
|
} // namespace fst
|
|
|
|
#endif // FST_PROJECT_H_
|