2021-02-06 15:51:04 +02:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2021, Itamar S. <itamar8910@gmail.com>
|
|
|
|
*
|
2021-04-22 01:24:48 -07:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2021-02-06 15:51:04 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2021-03-13 10:37:23 +02:00
|
|
|
#include <AK/FlyString.h>
|
2021-08-06 11:15:57 +03:00
|
|
|
#include <AK/Function.h>
|
2021-02-06 15:51:04 +02:00
|
|
|
#include <AK/HashMap.h>
|
|
|
|
#include <AK/Optional.h>
|
|
|
|
#include <AK/String.h>
|
|
|
|
#include <AK/StringView.h>
|
|
|
|
#include <AK/Vector.h>
|
2021-08-06 10:29:57 +03:00
|
|
|
#include <LibCpp/Token.h>
|
2021-02-06 15:51:04 +02:00
|
|
|
|
|
|
|
namespace Cpp {
|
2021-06-01 23:54:55 +02:00
|
|
|
|
2021-02-06 15:51:04 +02:00
|
|
|
class Preprocessor {
|
|
|
|
|
|
|
|
public:
|
2021-03-13 10:37:23 +02:00
|
|
|
explicit Preprocessor(const String& filename, const StringView& program);
|
2021-08-06 10:29:57 +03:00
|
|
|
Vector<Token> process_and_lex();
|
2021-02-06 15:51:04 +02:00
|
|
|
Vector<StringView> included_paths() const { return m_included_paths; }
|
|
|
|
|
|
|
|
struct DefinedValue {
|
2021-08-06 12:23:20 +03:00
|
|
|
String value;
|
2021-03-13 10:37:23 +02:00
|
|
|
FlyString filename;
|
|
|
|
size_t line { 0 };
|
|
|
|
size_t column { 0 };
|
2021-02-06 15:51:04 +02:00
|
|
|
};
|
2021-03-12 17:04:08 +02:00
|
|
|
using Definitions = HashMap<StringView, DefinedValue>;
|
|
|
|
|
2021-08-06 12:23:20 +03:00
|
|
|
struct Substitution {
|
|
|
|
Token original_token;
|
|
|
|
DefinedValue defined_value;
|
|
|
|
};
|
|
|
|
|
|
|
|
Definitions const& definitions() const { return m_definitions; }
|
|
|
|
Vector<Substitution> const& substitutions() const { return m_substitutions; }
|
2021-02-06 15:51:04 +02:00
|
|
|
|
2021-03-17 21:18:08 +03:00
|
|
|
void set_ignore_unsupported_keywords(bool ignore) { m_options.ignore_unsupported_keywords = ignore; }
|
2021-05-21 15:41:38 +03:00
|
|
|
void set_keep_include_statements(bool keep) { m_options.keep_include_statements = keep; }
|
2021-03-17 21:18:08 +03:00
|
|
|
|
2021-08-06 11:15:57 +03:00
|
|
|
Function<Definitions(StringView)> definitions_in_header_callback { nullptr };
|
|
|
|
|
2021-03-12 17:04:08 +02:00
|
|
|
private:
|
2021-05-21 15:39:38 +03:00
|
|
|
using PreprocessorKeyword = StringView;
|
2021-08-06 10:29:57 +03:00
|
|
|
PreprocessorKeyword handle_preprocessor_line(StringView const&);
|
|
|
|
void handle_preprocessor_keyword(StringView const& keyword, GenericLexer& line_lexer);
|
2021-08-06 12:23:20 +03:00
|
|
|
void process_line(StringView const& line);
|
|
|
|
void do_substitution(Token const& replaced_token, DefinedValue const&);
|
2021-02-06 15:51:04 +02:00
|
|
|
|
2021-08-06 10:29:57 +03:00
|
|
|
String m_filename;
|
|
|
|
String m_program;
|
2021-02-06 15:51:04 +02:00
|
|
|
Vector<StringView> m_lines;
|
2021-08-06 12:23:20 +03:00
|
|
|
|
|
|
|
Vector<Token> m_tokens;
|
|
|
|
Definitions m_definitions;
|
|
|
|
Vector<Substitution> m_substitutions;
|
|
|
|
|
2021-02-06 15:51:04 +02:00
|
|
|
size_t m_line_index { 0 };
|
|
|
|
size_t m_current_depth { 0 };
|
|
|
|
Vector<size_t> m_depths_of_taken_branches;
|
|
|
|
Vector<size_t> m_depths_of_not_taken_branches;
|
|
|
|
|
|
|
|
enum class State {
|
|
|
|
Normal,
|
|
|
|
SkipIfBranch,
|
|
|
|
SkipElseBranch
|
|
|
|
};
|
|
|
|
State m_state { State::Normal };
|
|
|
|
|
|
|
|
Vector<StringView> m_included_paths;
|
2021-03-17 21:18:08 +03:00
|
|
|
|
|
|
|
struct Options {
|
|
|
|
bool ignore_unsupported_keywords { false };
|
2021-05-21 15:41:38 +03:00
|
|
|
bool keep_include_statements { false };
|
2021-03-17 21:18:08 +03:00
|
|
|
} m_options;
|
2021-02-06 15:51:04 +02:00
|
|
|
};
|
|
|
|
}
|