2018-04-01 14:49:10 +10:00
|
|
|
/*
|
2018-08-04 15:14:06 +10:00
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
2018-04-01 14:49:10 +10:00
|
|
|
*
|
|
|
|
* Copyright 2018 Danny Robson <danny@nerdcruft.net>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef CRUFT_UTIL_CPP_HPP
|
|
|
|
#define CRUFT_UTIL_CPP_HPP
|
|
|
|
|
|
|
|
#include "string.hpp"
|
|
|
|
#include "view.hpp"
|
|
|
|
|
2018-12-05 19:12:03 +11:00
|
|
|
#include <filesystem>
|
2018-04-01 14:49:10 +10:00
|
|
|
#include <string>
|
|
|
|
#include <map>
|
|
|
|
#include <stdexcept>
|
|
|
|
#include <stack>
|
|
|
|
|
2018-08-05 14:42:02 +10:00
|
|
|
namespace cruft::cpp {
|
2018-04-01 14:49:10 +10:00
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
struct context {
|
2018-12-05 19:12:03 +11:00
|
|
|
std::stack<std::filesystem::path> source;
|
2018-04-01 14:49:10 +10:00
|
|
|
std::map<std::string,std::string> defines;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
2018-05-07 13:28:31 +10:00
|
|
|
/// an abstract base that performs processing for a directive
|
2018-04-01 14:49:10 +10:00
|
|
|
class directive {
|
|
|
|
public:
|
|
|
|
virtual ~directive () = default;
|
|
|
|
|
2018-05-07 13:28:31 +10:00
|
|
|
/// handles a preprocessor directive by:
|
|
|
|
/// * perform some processing on any number of input lines,
|
|
|
|
/// * optionally writes to the supplied stream,
|
|
|
|
/// * and returns the line that should be consumed next
|
|
|
|
///
|
|
|
|
/// \param lines is a tokenised view over all the lines of the input
|
|
|
|
/// \return the next line that processing should continue from
|
2018-08-05 14:42:02 +10:00
|
|
|
virtual cruft::tokeniser<const char*>::iterator
|
2018-04-01 14:49:10 +10:00
|
|
|
process (std::ostream&,
|
|
|
|
context&,
|
2018-08-05 14:42:02 +10:00
|
|
|
cruft::view<cruft::tokeniser<const char*>::iterator> lines) const = 0;
|
2018-04-01 14:49:10 +10:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
class processor {
|
|
|
|
public:
|
|
|
|
processor ();
|
|
|
|
void add (std::string token, std::unique_ptr<directive>);
|
|
|
|
|
2018-12-05 19:12:03 +11:00
|
|
|
void process (std::ostream&, const std::filesystem::path&) const;
|
2018-04-01 14:49:10 +10:00
|
|
|
|
2018-12-05 19:12:03 +11:00
|
|
|
std::filesystem::path
|
|
|
|
resolve (const std::filesystem::path&) const;
|
2018-04-01 14:49:10 +10:00
|
|
|
|
2018-08-05 14:42:02 +10:00
|
|
|
cruft::tokeniser<const char*>::iterator
|
2018-04-01 14:49:10 +10:00
|
|
|
process (std::ostream&,
|
|
|
|
context&,
|
2018-08-05 14:42:02 +10:00
|
|
|
cruft::view<cruft::tokeniser<const char*>::iterator>) const;
|
2018-04-01 14:49:10 +10:00
|
|
|
|
|
|
|
private:
|
|
|
|
std::map<std::string, std::unique_ptr<directive>> m_directives;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
2018-05-07 13:28:31 +10:00
|
|
|
/// thrown when a processor encounters a directive it has not been
|
|
|
|
/// configured to handle
|
2018-04-01 14:49:10 +10:00
|
|
|
class unknown_directive : public std::runtime_error {
|
|
|
|
using runtime_error::runtime_error;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
2018-05-07 13:28:31 +10:00
|
|
|
/// silently ignores configured directive by advancing the input cursor
|
|
|
|
/// past the provided line without writing to the output stream.
|
|
|
|
class ignore : public directive {
|
2018-08-05 14:42:02 +10:00
|
|
|
cruft::tokeniser<const char*>::iterator
|
2018-04-01 14:49:10 +10:00
|
|
|
process (std::ostream&,
|
|
|
|
context&,
|
2018-08-05 14:42:02 +10:00
|
|
|
cruft::view<cruft::tokeniser<const char*>::iterator> lines) const override
|
2018-04-01 14:49:10 +10:00
|
|
|
{ return lines.begin ()++; }
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2018-05-07 13:28:31 +10:00
|
|
|
///------------------------------------------------------------------------
|
|
|
|
/// copies the supplied directive to the output stream without any
|
|
|
|
/// modification
|
|
|
|
class passthrough : public directive {
|
2018-04-01 14:49:10 +10:00
|
|
|
public:
|
|
|
|
passthrough (const std::string &name);
|
|
|
|
|
2018-08-05 14:42:02 +10:00
|
|
|
virtual cruft::tokeniser<const char*>::iterator
|
2018-04-01 14:49:10 +10:00
|
|
|
process (std::ostream&,
|
|
|
|
context&,
|
2018-08-05 14:42:02 +10:00
|
|
|
cruft::view<cruft::tokeniser<const char*>::iterator>) const override;
|
2018-04-01 14:49:10 +10:00
|
|
|
|
|
|
|
private:
|
|
|
|
const std::string m_name;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2018-05-07 13:28:31 +10:00
|
|
|
///------------------------------------------------------------------------
|
|
|
|
/// handles include directives by copying the contents of the referenced
|
|
|
|
/// path into the input stream
|
2018-04-01 14:49:10 +10:00
|
|
|
class include : public directive {
|
|
|
|
public:
|
|
|
|
include (processor &_parent);
|
|
|
|
|
2018-12-05 19:12:03 +11:00
|
|
|
void add (const std::filesystem::path&);
|
2018-04-01 14:49:10 +10:00
|
|
|
|
2018-08-05 14:42:02 +10:00
|
|
|
virtual cruft::tokeniser<const char*>::iterator
|
2018-04-01 14:49:10 +10:00
|
|
|
process (std::ostream&,
|
|
|
|
context&,
|
2018-08-05 14:42:02 +10:00
|
|
|
cruft::view<cruft::tokeniser<const char*>::iterator>) const override;
|
2018-04-01 14:49:10 +10:00
|
|
|
|
|
|
|
private:
|
|
|
|
processor &m_parent;
|
2018-12-05 19:12:03 +11:00
|
|
|
std::vector<std::filesystem::path> m_paths;
|
2018-04-01 14:49:10 +10:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif
|