Detect type aliasing cycles
This commit is contained in:
parent
0658d07e97
commit
f2e2da4a34
12
boot/ast.cc
12
boot/ast.cc
@ -264,18 +264,8 @@ namespace elna::boot
|
|||||||
delete m_body;
|
delete m_body;
|
||||||
}
|
}
|
||||||
|
|
||||||
return_declaration::return_declaration(std::shared_ptr<type_expression> type)
|
|
||||||
: type(type)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
return_declaration::return_declaration(std::monostate)
|
|
||||||
: no_return(true)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
procedure_type_expression::procedure_type_expression(const struct position position,
|
procedure_type_expression::procedure_type_expression(const struct position position,
|
||||||
return_declaration return_type)
|
return_declaration<std::shared_ptr<type_expression>> return_type)
|
||||||
: type_expression(position), return_type(return_type)
|
: type_expression(position), return_type(return_type)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
@ -144,7 +144,7 @@ along with GCC; see the file COPYING3. If not see
|
|||||||
%type <std::vector<elna::boot::statement *>> statements;
|
%type <std::vector<elna::boot::statement *>> statements;
|
||||||
%type <elna::boot::procedure_definition *> procedure_definition;
|
%type <elna::boot::procedure_definition *> procedure_definition;
|
||||||
%type <std::pair<std::vector<std::string>, std::shared_ptr<elna::boot::procedure_type_expression>>> procedure_heading;
|
%type <std::pair<std::vector<std::string>, std::shared_ptr<elna::boot::procedure_type_expression>>> procedure_heading;
|
||||||
%type <elna::boot::return_declaration> return_declaration;
|
%type <elna::boot::procedure_type_expression::return_t> return_declaration;
|
||||||
%type <std::vector<elna::boot::procedure_definition *>> procedure_definitions procedure_part;
|
%type <std::vector<elna::boot::procedure_definition *>> procedure_definitions procedure_part;
|
||||||
%type <elna::boot::type_definition *> type_definition;
|
%type <elna::boot::type_definition *> type_definition;
|
||||||
%type <std::vector<elna::boot::type_definition *>> type_definitions type_part;
|
%type <std::vector<elna::boot::type_definition *>> type_definitions type_part;
|
||||||
@ -197,8 +197,8 @@ identifier_definitions:
|
|||||||
| identifier_definition { $$.emplace_back(std::move($1)); }
|
| identifier_definition { $$.emplace_back(std::move($1)); }
|
||||||
return_declaration:
|
return_declaration:
|
||||||
/* proper procedure */ {}
|
/* proper procedure */ {}
|
||||||
| "->" "!" { $$ = boot::return_declaration(std::monostate{}); }
|
| "->" "!" { $$ = boot::procedure_type_expression::return_t(std::monostate{}); }
|
||||||
| "->" type_expression { $$ = boot::return_declaration($2); }
|
| "->" type_expression { $$ = boot::procedure_type_expression::return_t($2); }
|
||||||
procedure_heading:
|
procedure_heading:
|
||||||
"(" formal_parameters ")" return_declaration
|
"(" formal_parameters ")" return_declaration
|
||||||
{
|
{
|
||||||
|
@ -142,8 +142,28 @@ namespace elna::boot
|
|||||||
this->current_type = type(result_type);
|
this->current_type = type(result_type);
|
||||||
}
|
}
|
||||||
|
|
||||||
void declaration_visitor::visit(procedure_type_expression *)
|
void declaration_visitor::visit(procedure_type_expression *type_expression)
|
||||||
{
|
{
|
||||||
|
std::shared_ptr<procedure_type> result_type;
|
||||||
|
|
||||||
|
if (type_expression->return_type.no_return)
|
||||||
|
{
|
||||||
|
result_type = std::make_shared<procedure_type>(procedure_type::return_t(std::monostate{}));
|
||||||
|
}
|
||||||
|
else if (type_expression->return_type.proper_type != nullptr)
|
||||||
|
{
|
||||||
|
type_expression->return_type.proper_type->accept(this);
|
||||||
|
result_type = std::make_shared<procedure_type>(procedure_type::return_t(this->current_type));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
result_type = std::make_shared<procedure_type>(procedure_type::return_t());
|
||||||
|
}
|
||||||
|
for (auto& parameter : type_expression->parameters)
|
||||||
|
{
|
||||||
|
parameter->accept(this);
|
||||||
|
result_type->parameters.push_back(this->current_type);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void declaration_visitor::visit(variable_declaration *declaration)
|
void declaration_visitor::visit(variable_declaration *declaration)
|
||||||
@ -161,9 +181,9 @@ namespace elna::boot
|
|||||||
{
|
{
|
||||||
heading_parameter->accept(this);
|
heading_parameter->accept(this);
|
||||||
}
|
}
|
||||||
if (definition->heading().return_type.type != nullptr)
|
if (definition->heading().return_type.proper_type != nullptr)
|
||||||
{
|
{
|
||||||
definition->heading().return_type.type->accept(this);
|
definition->heading().return_type.proper_type->accept(this);
|
||||||
}
|
}
|
||||||
if (definition->body != nullptr)
|
if (definition->body != nullptr)
|
||||||
{
|
{
|
||||||
|
@ -135,6 +135,11 @@ namespace elna::boot
|
|||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool type::operator==(const std::nullptr_t&)
|
||||||
|
{
|
||||||
|
return empty();
|
||||||
|
}
|
||||||
|
|
||||||
type::~type()
|
type::~type()
|
||||||
{
|
{
|
||||||
switch (tag)
|
switch (tag)
|
||||||
@ -265,6 +270,11 @@ namespace elna::boot
|
|||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
procedure_type::procedure_type(return_t return_type)
|
||||||
|
: return_type(return_type)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
info::~info()
|
info::~info()
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
@ -22,6 +22,8 @@ along with GCC; see the file COPYING3. If not see
|
|||||||
#include "elna/gcc/elna-diagnostic.h"
|
#include "elna/gcc/elna-diagnostic.h"
|
||||||
#include "elna/gcc/elna1.h"
|
#include "elna/gcc/elna1.h"
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
|
|
||||||
#include "ggc.h"
|
#include "ggc.h"
|
||||||
#include "function.h"
|
#include "function.h"
|
||||||
#include "cgraph.h"
|
#include "cgraph.h"
|
||||||
@ -37,7 +39,7 @@ along with GCC; see the file COPYING3. If not see
|
|||||||
namespace elna::gcc
|
namespace elna::gcc
|
||||||
{
|
{
|
||||||
tree get_inner_alias(const boot::type& type, std::shared_ptr<symbol_table> symbols,
|
tree get_inner_alias(const boot::type& type, std::shared_ptr<symbol_table> symbols,
|
||||||
std::unordered_map<std::string, tree>& unresolved)
|
std::unordered_map<std::string, tree>& unresolved, std::vector<std::string>& path)
|
||||||
{
|
{
|
||||||
if (auto reference = type.get<boot::primitive_type>())
|
if (auto reference = type.get<boot::primitive_type>())
|
||||||
{
|
{
|
||||||
@ -57,7 +59,7 @@ namespace elna::gcc
|
|||||||
}
|
}
|
||||||
else if (auto reference = type.get<boot::pointer_type>())
|
else if (auto reference = type.get<boot::pointer_type>())
|
||||||
{
|
{
|
||||||
return build_global_pointer_type(get_inner_alias(reference->base, symbols, unresolved));
|
return build_global_pointer_type(get_inner_alias(reference->base, symbols, unresolved, path));
|
||||||
}
|
}
|
||||||
else if (auto reference = type.get<boot::array_type>())
|
else if (auto reference = type.get<boot::array_type>())
|
||||||
{
|
{
|
||||||
@ -65,19 +67,27 @@ namespace elna::gcc
|
|||||||
tree upper_bound = build_int_cst_type(integer_type_node, reference->size);
|
tree upper_bound = build_int_cst_type(integer_type_node, reference->size);
|
||||||
tree range_type = build_range_type(integer_type_node, lower_bound, upper_bound);
|
tree range_type = build_range_type(integer_type_node, lower_bound, upper_bound);
|
||||||
|
|
||||||
return build_array_type(get_inner_alias(reference->base, symbols, unresolved), range_type);
|
return build_array_type(get_inner_alias(reference->base, symbols, unresolved, path), range_type);
|
||||||
}
|
}
|
||||||
else if (auto reference = type.get<boot::alias_type>())
|
else if (auto reference = type.get<boot::alias_type>())
|
||||||
{
|
{
|
||||||
return handle_symbol(reference->name, reference, symbols, unresolved);
|
return handle_symbol(reference->name, reference, symbols, unresolved, path);
|
||||||
}
|
}
|
||||||
return error_mark_node;
|
return error_mark_node;
|
||||||
}
|
}
|
||||||
|
|
||||||
tree handle_symbol(const std::string& symbol_name, std::shared_ptr<boot::alias_type> reference,
|
tree handle_symbol(const std::string& symbol_name, std::shared_ptr<boot::alias_type> reference,
|
||||||
std::shared_ptr<symbol_table> symbols, std::unordered_map<std::string, tree>& unresolved)
|
std::shared_ptr<symbol_table> symbols, std::unordered_map<std::string, tree>& unresolved,
|
||||||
|
std::vector<std::string>& path)
|
||||||
{
|
{
|
||||||
auto looked_up = get_inner_alias(reference->reference, symbols, unresolved);
|
path.push_back(symbol_name);
|
||||||
|
std::vector<std::string>::const_iterator head_peace = std::cbegin(path);
|
||||||
|
|
||||||
|
if (std::find(std::next(head_peace), std::cend(path), *head_peace) != std::cend(path))
|
||||||
|
{
|
||||||
|
return error_mark_node;
|
||||||
|
}
|
||||||
|
auto looked_up = get_inner_alias(reference->reference, symbols, unresolved, path);
|
||||||
|
|
||||||
unresolved.insert({ symbol_name, looked_up });
|
unresolved.insert({ symbol_name, looked_up });
|
||||||
|
|
||||||
@ -97,7 +107,8 @@ namespace elna::gcc
|
|||||||
{
|
{
|
||||||
for (auto& [symbol_name, symbol_info] : declaration_visitor.unresolved)
|
for (auto& [symbol_name, symbol_info] : declaration_visitor.unresolved)
|
||||||
{
|
{
|
||||||
handle_symbol(symbol_name, symbol_info, symbols, unresolved);
|
std::vector<std::string> type_path;
|
||||||
|
handle_symbol(symbol_name, symbol_info, symbols, unresolved, type_path);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return std::move(declaration_visitor.errors());
|
return std::move(declaration_visitor.errors());
|
||||||
@ -803,9 +814,9 @@ namespace elna::gcc
|
|||||||
}
|
}
|
||||||
tree return_type = void_type_node;
|
tree return_type = void_type_node;
|
||||||
|
|
||||||
if (type.return_type.type != nullptr)
|
if (type.return_type.proper_type != nullptr)
|
||||||
{
|
{
|
||||||
type.return_type.type->accept(this);
|
type.return_type.proper_type->accept(this);
|
||||||
return_type = this->current_expression;
|
return_type = this->current_expression;
|
||||||
}
|
}
|
||||||
this->current_expression = NULL_TREE;
|
this->current_expression = NULL_TREE;
|
||||||
|
@ -21,7 +21,6 @@ along with GCC; see the file COPYING3. If not see
|
|||||||
#include <memory>
|
#include <memory>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include <variant>
|
|
||||||
#include "elna/boot/result.h"
|
#include "elna/boot/result.h"
|
||||||
|
|
||||||
namespace elna::boot
|
namespace elna::boot
|
||||||
@ -316,30 +315,19 @@ namespace elna::boot
|
|||||||
virtual ~constant_definition() override;
|
virtual ~constant_definition() override;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
|
||||||
* Tags a procedure type as never returning.
|
|
||||||
*/
|
|
||||||
struct return_declaration
|
|
||||||
{
|
|
||||||
return_declaration() = default;
|
|
||||||
explicit return_declaration(std::shared_ptr<type_expression> type);
|
|
||||||
explicit return_declaration(std::monostate);
|
|
||||||
|
|
||||||
std::shared_ptr<type_expression> type{ nullptr };
|
|
||||||
bool no_return{ false };
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Procedure type.
|
* Procedure type.
|
||||||
*/
|
*/
|
||||||
class procedure_type_expression : public type_expression
|
class procedure_type_expression : public type_expression
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
const return_declaration return_type;
|
using return_t = return_declaration<std::shared_ptr<type_expression>>;
|
||||||
|
|
||||||
|
const return_t return_type;
|
||||||
std::vector<std::shared_ptr<type_expression>> parameters;
|
std::vector<std::shared_ptr<type_expression>> parameters;
|
||||||
|
|
||||||
procedure_type_expression(const struct position position,
|
procedure_type_expression(const struct position position,
|
||||||
return_declaration return_type = return_declaration());
|
return_t return_type = return_t());
|
||||||
|
|
||||||
void accept(parser_visitor *visitor) override;
|
void accept(parser_visitor *visitor) override;
|
||||||
std::shared_ptr<procedure_type_expression> is_procedure() override;
|
std::shared_ptr<procedure_type_expression> is_procedure() override;
|
||||||
|
@ -21,6 +21,7 @@ along with GCC; see the file COPYING3. If not see
|
|||||||
#include <string>
|
#include <string>
|
||||||
#include <deque>
|
#include <deque>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
#include <variant>
|
||||||
|
|
||||||
namespace elna::boot
|
namespace elna::boot
|
||||||
{
|
{
|
||||||
@ -79,4 +80,27 @@ namespace elna::boot
|
|||||||
m_errors.emplace_back(std::move(new_error));
|
m_errors.emplace_back(std::move(new_error));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tags a procedure type as never returning.
|
||||||
|
*/
|
||||||
|
template<typename T>
|
||||||
|
struct return_declaration
|
||||||
|
{
|
||||||
|
return_declaration() = default;
|
||||||
|
|
||||||
|
explicit return_declaration(T type)
|
||||||
|
: proper_type(type)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
explicit return_declaration(std::monostate)
|
||||||
|
: no_return(true)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
T proper_type{};
|
||||||
|
bool no_return{ false };
|
||||||
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -63,7 +63,7 @@ namespace elna::boot
|
|||||||
void visit(type_definition *definition) override;
|
void visit(type_definition *definition) override;
|
||||||
void visit(record_type_expression *type_expression) override;
|
void visit(record_type_expression *type_expression) override;
|
||||||
void visit(union_type_expression *type_expression) override;
|
void visit(union_type_expression *type_expression) override;
|
||||||
void visit(procedure_type_expression *) override;
|
void visit(procedure_type_expression *type_expression) override;
|
||||||
|
|
||||||
void visit(variable_declaration *declaration) override;
|
void visit(variable_declaration *declaration) override;
|
||||||
void visit(constant_definition *) override;
|
void visit(constant_definition *) override;
|
||||||
|
@ -23,6 +23,8 @@ along with GCC; see the file COPYING3. If not see
|
|||||||
#include <memory>
|
#include <memory>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
|
#include "elna/boot/result.h"
|
||||||
|
|
||||||
namespace elna::boot
|
namespace elna::boot
|
||||||
{
|
{
|
||||||
class alias_type;
|
class alias_type;
|
||||||
@ -73,6 +75,8 @@ namespace elna::boot
|
|||||||
type(type&& other);
|
type(type&& other);
|
||||||
type& operator=(type&& other);
|
type& operator=(type&& other);
|
||||||
|
|
||||||
|
bool operator==(const std::nullptr_t&);
|
||||||
|
|
||||||
~type();
|
~type();
|
||||||
|
|
||||||
template<typename T>
|
template<typename T>
|
||||||
@ -123,6 +127,16 @@ namespace elna::boot
|
|||||||
std::vector<type_field> fields;
|
std::vector<type_field> fields;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct procedure_type
|
||||||
|
{
|
||||||
|
using return_t = return_declaration<type>;
|
||||||
|
|
||||||
|
std::vector<type> parameters;
|
||||||
|
const return_t return_type;
|
||||||
|
|
||||||
|
procedure_type(return_t return_type = return_t());
|
||||||
|
};
|
||||||
|
|
||||||
class type_info;
|
class type_info;
|
||||||
|
|
||||||
class info : public std::enable_shared_from_this<info>
|
class info : public std::enable_shared_from_this<info>
|
||||||
|
@ -36,7 +36,8 @@ namespace elna::gcc
|
|||||||
std::unique_ptr<boot::program>& ast, std::shared_ptr<boot::symbol_table> info_table,
|
std::unique_ptr<boot::program>& ast, std::shared_ptr<boot::symbol_table> info_table,
|
||||||
std::shared_ptr<symbol_table> symbols, std::unordered_map<std::string, tree>& unresolved);
|
std::shared_ptr<symbol_table> symbols, std::unordered_map<std::string, tree>& unresolved);
|
||||||
tree handle_symbol(const std::string& symbol_name, std::shared_ptr<boot::alias_type> reference,
|
tree handle_symbol(const std::string& symbol_name, std::shared_ptr<boot::alias_type> reference,
|
||||||
std::shared_ptr<symbol_table> symbols, std::unordered_map<std::string, tree>& unresolved);
|
std::shared_ptr<symbol_table> symbols, std::unordered_map<std::string, tree>& unresolved,
|
||||||
|
std::vector<std::string>& path);
|
||||||
|
|
||||||
|
|
||||||
class generic_visitor final : public boot::parser_visitor
|
class generic_visitor final : public boot::parser_visitor
|
||||||
|
Loading…
x
Reference in New Issue
Block a user