llvm-capstone/flang/lib/parser/indirection.h
peter klausler 0ba1a14be2 [flang] Impose a directory structure. Move files around. Introduce
an intermediate "parser" namespace.

Original-commit: flang-compiler/f18@690b6f0d1e
Reviewed-on: https://github.com/flang-compiler/f18/pull/4
Tree-same-pre-rewrite: false
2018-02-07 12:04:42 -08:00

61 lines
1.6 KiB
C++

#ifndef FORTRAN_INDIRECTION_H_
#define FORTRAN_INDIRECTION_H_
// Defines a smart pointer template class that's rather like std::unique_ptr<>
// but further restricted, like a C++ reference, to be non-null when constructed
// or assigned. Users need not check whether these pointers are null.
// Intended to be as invisible as possible.
#include "idioms.h"
#include <ostream>
#include <utility>
namespace Fortran {
namespace parser {
template<typename A> class Indirection {
public:
using element_type = A;
Indirection() = delete;
Indirection(A *&p) : p_{p} {
CHECK(p_ && "assigning null pointer to Indirection");
p = nullptr;
}
Indirection(A *&&p) : p_{p} {
CHECK(p_ && "assigning null pointer to Indirection");
p = nullptr;
}
Indirection(A &&p) : p_{new A(std::move(p))} {}
template<typename... ARGS>
Indirection(ARGS &&... args) : p_{new A(std::forward<ARGS>(args)...)} {}
Indirection(Indirection &&that) {
CHECK(that.p_ && "constructing Indirection from null Indirection");
p_ = that.p_;
that.p_ = nullptr;
}
~Indirection() {
delete p_;
p_ = nullptr;
}
Indirection &operator=(Indirection &&that) {
CHECK(that.p_ && "assigning null Indirection to Indirection");
auto tmp = p_;
p_ = that.p_;
that.p_ = tmp;
return *this;
}
A &operator*() const { return *p_; }
A *operator->() { return p_; }
private:
A *p_{nullptr};
};
template<typename A>
std::ostream &operator<<(std::ostream &o, const Indirection<A> &x) {
return o << *x;
}
} // namespace parser
} // namespace Fortran
#endif // FORTRAN_INDIRECTION_H_