/* Dependency graph analysis. Copyright (C) 2025 Free Software Foundation, Inc. GCC is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3, or (at your option) any later version. GCC is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with GCC; see the file COPYING3. If not see . */ #include "elna/boot/dependency.h" #include #include #include #include "elna/boot/driver.h" #include "elna/boot/semantic.h" #include "parser.hh" namespace elna::boot { source_path_error::source_path_error(const std::string& message, const char *path) : error(path, { 0, 0 }), message(message) { } std::string source_path_error::what() const { std::stringstream output; output << "Cannot open filename "; output << this->path; output << ": "; output << this->message; return output.str(); } dependency::dependency() { } dependency::dependency(error_list&& errors) : m_errors(std::move(errors)) { } bool dependency::has_errors() const { return !errors().empty(); } const error_list& dependency::errors() const { return m_errors; } dependency parse_source(std::istream& entry_point, const char *entry_path) { driver parse_driver{ entry_path }; lexer tokenizer(entry_point); yy::parser parser(tokenizer, parse_driver); if (parser()) { return dependency(std::move(parse_driver.errors())); } else { dependency outcome; std::swap(outcome.tree, parse_driver.tree); return outcome; } } dependency read_sources(const char *entry_path, std::shared_ptr info_table, std::shared_ptr imports) { std::ifstream entry_point{ entry_path, std::ios::in }; if (!entry_point) { error_list errors; errors.push_back(std::make_unique(strerror(errno), entry_path)); return dependency(std::move(errors)); } auto outcome = parse_source(entry_point, entry_path); declaration_visitor declaration_visitor(entry_path); outcome.tree->accept(&declaration_visitor); if (!declaration_visitor.errors().empty()) { return dependency(std::move(declaration_visitor.errors())); } outcome.unresolved = declaration_visitor.unresolved; outcome.bag = symbol_bag{ std::move(declaration_visitor.unresolved), info_table }; if (imports != nullptr) { outcome.bag.add_import(imports); } elna::boot::name_analysis_visitor name_analysis_visitor(entry_path, outcome.bag); outcome.tree->accept(&name_analysis_visitor); if (!name_analysis_visitor.errors().empty()) { return dependency(std::move(declaration_visitor.errors())); } return outcome; } }