Randomize type declaration order

This commit is contained in:
Eugen Wissner 2025-03-12 23:56:54 +01:00
parent c9a8ecdc0a
commit ac084be7f5
Signed by: belka
GPG Key ID: A27FDC1E8EE902C0
5 changed files with 48 additions and 28 deletions

View File

@ -21,16 +21,28 @@ namespace elna
{ {
namespace boot namespace boot
{ {
declaration_error::declaration_error(const std::string& identifier, const char *path, const struct position position) undeclared_error::undeclared_error(const std::string& identifier, const char *path, const struct position position)
: error(path, position), identifier(identifier) : error(path, position), identifier(identifier)
{ {
} }
std::string declaration_error::what() const std::string undeclared_error::what() const
{ {
return "Type '" + identifier + "' not declared"; return "Type '" + identifier + "' not declared";
} }
already_declared_error::already_declared_error(const std::string& identifier,
const char *path, const struct position position)
: error(path, position), identifier(identifier)
{
}
std::string already_declared_error::what() const
{
return "Symbol '" + identifier + "' has been already declared";
}
declaration_visitor::declaration_visitor(const char *path, std::shared_ptr<symbol_table> symbols) declaration_visitor::declaration_visitor(const char *path, std::shared_ptr<symbol_table> symbols)
: error_container(path), symbols(symbols) : error_container(path), symbols(symbols)
{ {
@ -40,7 +52,10 @@ namespace boot
{ {
for (type_definition *const type : program->types) for (type_definition *const type : program->types)
{ {
this->unresolved.insert({ type->identifier, std::make_shared<alias_type>() }); if (!this->unresolved.insert({ type->identifier, std::make_shared<alias_type>() }).second)
{
add_error<already_declared_error>(type->identifier, this->input_file, type->position());
}
} }
for (type_definition *const type : program->types) for (type_definition *const type : program->types)
{ {
@ -64,11 +79,14 @@ namespace boot
{ {
this->current_type = type(unresolved_alias->second); this->current_type = type(unresolved_alias->second);
} }
else if (auto from_symbol_table = this->symbols.lookup(type_expression->name))
{
this->current_type = from_symbol_table->is_type()->symbol;
}
else else
{ {
// TODO: add_error<undeclared_error>(type_expression->name, this->input_file, type_expression->position());
// add_error<declaration_error>(type_expression->name, this->input_file, type_expression->position()); this->current_type = type();
this->current_type = type(std::make_shared<primitive_type>(type_expression->name));
} }
} }
@ -91,8 +109,10 @@ namespace boot
for (auto& field : type_expression->fields) for (auto& field : type_expression->fields)
{ {
field.second->accept(this); field.second->accept(this);
if (!this->current_type.empty())
type_definition->fields.push_back({ field.first, type(this->current_type) }); {
type_definition->fields.push_back({ field.first, type(this->current_type) });
}
} }
this->current_type = type(type_definition); this->current_type = type(type_definition);
} }

View File

@ -15,8 +15,6 @@ You should have received a copy of the GNU General Public License
along with GCC; see the file COPYING3. If not see along with GCC; see the file COPYING3. If not see
<http://www.gnu.org/licenses/>. */ <http://www.gnu.org/licenses/>. */
#pragma once
#include "elna/boot/symbol.h" #include "elna/boot/symbol.h"
namespace elna namespace elna
@ -68,7 +66,7 @@ namespace boot
new (&alias) std::weak_ptr<alias_type>(other.alias); new (&alias) std::weak_ptr<alias_type>(other.alias);
break; break;
case type_tag::primitive: case type_tag::primitive:
new (&primitive) std::weak_ptr<primitive_type>(other.primitive); new (&primitive) std::shared_ptr<primitive_type>(other.primitive);
break; break;
case type_tag::record: case type_tag::record:
new (&record) std::shared_ptr<record_type>(other.record); new (&record) std::shared_ptr<record_type>(other.record);
@ -101,7 +99,7 @@ namespace boot
new (&alias) std::weak_ptr<alias_type>(std::move(other.alias)); new (&alias) std::weak_ptr<alias_type>(std::move(other.alias));
break; break;
case type_tag::primitive: case type_tag::primitive:
new (&primitive) std::weak_ptr<primitive_type>(std::move(other.primitive)); new (&primitive) std::shared_ptr<primitive_type>(std::move(other.primitive));
break; break;
case type_tag::record: case type_tag::record:
new (&record) std::shared_ptr<record_type>(std::move(other.record)); new (&record) std::shared_ptr<record_type>(std::move(other.record));
@ -149,6 +147,7 @@ namespace boot
case type_tag::alias: case type_tag::alias:
break; break;
case type_tag::primitive: case type_tag::primitive:
this->primitive.~shared_ptr<primitive_type>();
break; break;
case type_tag::record: case type_tag::record:
this->record.~shared_ptr<record_type>(); this->record.~shared_ptr<record_type>();
@ -183,7 +182,7 @@ namespace boot
{ {
if (tag == type_tag::primitive) if (tag == type_tag::primitive)
{ {
return this->primitive.lock(); return this->primitive;
} }
else else
{ {

View File

@ -754,17 +754,8 @@ namespace gcc
get_identifier(definition->identifier.c_str()), this->current_expression); get_identifier(definition->identifier.c_str()), this->current_expression);
auto result = this->symbols->enter(definition->identifier, this->current_expression); auto result = this->symbols->enter(definition->identifier, this->current_expression);
/* if (result) TREE_PUBLIC(definition_tree) = definition->exported;
{ */ TYPE_NAME(this->current_expression) = get_identifier(definition->identifier.c_str());
TREE_PUBLIC(definition_tree) = definition->exported;
TYPE_NAME(this->current_expression) = get_identifier(definition->identifier.c_str());
// }
/* else
{
error_at(get_location(&definition->position()),
"type '%s' already declared in this scope",
definition->identifier.c_str());
} */
this->current_expression = NULL_TREE; this->current_expression = NULL_TREE;
} }

View File

@ -28,14 +28,24 @@ namespace elna
{ {
namespace boot namespace boot
{ {
class declaration_error : public error class undeclared_error : public error
{ {
std::string identifier; std::string identifier;
public: public:
declaration_error(const std::string& identifier, const char *path, const struct position position); undeclared_error(const std::string& identifier, const char *path, const struct position position);
virtual std::string what() const override; std::string what() const override;
};
class already_declared_error : public error
{
std::string identifier;
public:
already_declared_error(const std::string& identifier, const char *path, const struct position position);
std::string what() const override;
}; };
class declaration_visitor final : public empty_visitor, public error_container class declaration_visitor final : public empty_visitor, public error_container

View File

@ -50,7 +50,7 @@ namespace boot
union union
{ {
std::weak_ptr<alias_type> alias; std::weak_ptr<alias_type> alias;
std::weak_ptr<primitive_type> primitive; std::shared_ptr<primitive_type> primitive;
std::shared_ptr<record_type> record; std::shared_ptr<record_type> record;
std::shared_ptr<union_type> _union; std::shared_ptr<union_type> _union;
std::shared_ptr<pointer_type> pointer; std::shared_ptr<pointer_type> pointer;