2020-11-17 02:32:32 +01:00
|
|
|
#include "lang/validator.hpp"
|
2020-11-16 22:54:39 +01:00
|
|
|
|
|
|
|
#include <unordered_set>
|
|
|
|
#include <string>
|
|
|
|
|
|
|
|
namespace hex::lang {
|
|
|
|
|
|
|
|
Validator::Validator() {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Validator::validate(const std::vector<ASTNode*>& ast) {
|
|
|
|
|
|
|
|
std::unordered_set<std::string> typeNames;
|
|
|
|
|
|
|
|
for (const auto &node : ast) {
|
|
|
|
switch (node->getType()) {
|
|
|
|
case ASTNode::Type::VariableDecl:
|
|
|
|
{
|
|
|
|
// Check for duplicate variable names
|
|
|
|
auto varDeclNode = static_cast<ASTNodeVariableDecl*>(node);
|
|
|
|
if (!typeNames.insert(varDeclNode->getVariableName()).second)
|
|
|
|
return false;
|
2020-11-21 20:19:33 +01:00
|
|
|
|
|
|
|
if (varDeclNode->getArraySize() == 0 && !varDeclNode->getArraySizeVariable().has_value() ||
|
|
|
|
varDeclNode->getArraySize() != 0 && varDeclNode->getArraySizeVariable().has_value())
|
|
|
|
return false;
|
2020-11-16 22:54:39 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case ASTNode::Type::TypeDecl:
|
|
|
|
{
|
|
|
|
// Check for duplicate type names
|
|
|
|
auto typeDeclNode = static_cast<ASTNodeTypeDecl*>(node);
|
|
|
|
if (!typeNames.insert(typeDeclNode->getTypeName()).second)
|
|
|
|
return false;
|
2020-11-17 02:32:32 +01:00
|
|
|
|
|
|
|
if (typeDeclNode->getAssignedType() == Token::TypeToken::Type::CustomType && !typeNames.contains(typeDeclNode->getAssignedCustomTypeName()))
|
|
|
|
return false;
|
2020-11-16 22:54:39 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case ASTNode::Type::Struct:
|
|
|
|
{
|
|
|
|
// Check for duplicate type name
|
|
|
|
auto structNode = static_cast<ASTNodeStruct*>(node);
|
|
|
|
if (!typeNames.insert(structNode->getName()).second)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Check for duplicate member names
|
|
|
|
std::unordered_set<std::string> memberNames;
|
|
|
|
for (const auto &member : structNode->getNodes())
|
|
|
|
if (!memberNames.insert(static_cast<ASTNodeVariableDecl*>(member)->getVariableName()).second)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case ASTNode::Type::Enum:
|
|
|
|
{
|
|
|
|
// Check for duplicate type name
|
|
|
|
auto enumNode = static_cast<ASTNodeEnum*>(node);
|
|
|
|
if (!typeNames.insert(enumNode->getName()).second)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Check for duplicate constant names
|
|
|
|
std::unordered_set<std::string> constantNames;
|
|
|
|
for (const auto &[value, name] : enumNode->getValues())
|
|
|
|
if (!constantNames.insert(name).second)
|
|
|
|
return false;
|
|
|
|
}
|
2020-11-20 20:26:19 +01:00
|
|
|
case ASTNode::Type::Bitfield:
|
|
|
|
{
|
|
|
|
// Check for duplicate type name
|
|
|
|
auto bitfieldNode = static_cast<ASTNodeBitField*>(node);
|
|
|
|
if (!typeNames.insert(bitfieldNode->getName()).second)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
size_t bitfieldSize = 0;
|
|
|
|
|
|
|
|
// Check for duplicate constant names
|
|
|
|
std::unordered_set<std::string> flagNames;
|
|
|
|
for (const auto &[name, size] : bitfieldNode->getFields()) {
|
|
|
|
if (!flagNames.insert(name).second)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
bitfieldSize += size;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bitfieldSize > 64)
|
|
|
|
return false;
|
|
|
|
}
|
2020-11-16 22:54:39 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|