source: src/CodeGen/FixMain.cc@ 8d182b1

Last change on this file since 8d182b1 was 61efa42, checked in by Andrew Beach <ajbeach@…>, 23 months ago

Took the special main linkage code out of FunctionDecl and put it into a new pass. There is also a lot of related clean-up done.

  • Property mode set to 100644
File size: 4.7 KB
RevLine 
[13de47bc]1//
2// Cforall Version 1.0.0 Copyright (C) 2015 University of Waterloo
3//
4// The contents of this file are covered under the licence agreement in the
5// file "LICENCE" distributed with Cforall.
6//
[61efa42]7// FixMain.cc -- Tools to change a Cforall main into a C main.
[13de47bc]8//
9// Author : Thierry Delisle
10// Created On : Thr Jan 12 14:11:09 2017
[bf2438c]11// Last Modified By :
12// Last Modified On :
[13de47bc]13// Update Count : 0
14//
[7cc2c8d]15
[bf2438c]16#include "FixMain.h"
[13de47bc]17
[bf2438c]18#include <cassert> // for assert, assertf
19#include <fstream> // for operator<<, basic_ostream::operator<<
20#include <list> // for list
21#include <string> // for operator<<
[13de47bc]22
[0c577f7]23#include "AST/Decl.hpp"
[bef4f1a]24#include "AST/Pass.hpp"
[0c577f7]25#include "AST/Type.hpp"
[bef4f1a]26#include "AST/Vector.hpp"
[bf2438c]27#include "Common/SemanticError.h" // for SemanticError
[1619acd]28#include "CodeGen/GenType.h" // for GenType
[16ba4a6f]29#include "SymTab/Mangler.h"
[13de47bc]30
31namespace CodeGen {
[8e48fca4]32
33namespace {
34
[61efa42]35struct FindMainCore final {
[bef4f1a]36 ast::FunctionDecl const * main_declaration = nullptr;
37
38 void previsit( ast::FunctionDecl const * decl ) {
[61efa42]39 if ( isMain( decl ) ) {
[bef4f1a]40 if ( main_declaration ) {
41 SemanticError( decl, "Multiple definition of main routine\n" );
42 }
43 main_declaration = decl;
44 }
45 }
46};
47
48std::string genTypeAt( const ast::vector<ast::Type> & types, size_t at ) {
49 return genType( types[at], "", Options( false, false, false, false ) );
50}
51
52ast::ObjectDecl * makeIntObj(){
53 return new ast::ObjectDecl( CodeLocation(), "",
54 new ast::BasicType( ast::BasicType::SignedInt ) );
[f42fc13]55}
56
[bef4f1a]57ast::ObjectDecl * makeCharStarStarObj() {
58 return new ast::ObjectDecl( CodeLocation(), "",
59 new ast::PointerType(
60 new ast::PointerType(
61 new ast::BasicType( ast::BasicType::Char ) ) ) );
[f42fc13]62}
63
[bef4f1a]64std::string getMangledNameOfMain(
65 ast::vector<ast::DeclWithType> && params, ast::ArgumentFlag isVarArgs ) {
66 ast::ptr<ast::FunctionDecl> decl = new ast::FunctionDecl(
67 CodeLocation(),
68 "main",
69 ast::vector<ast::TypeDecl>(),
70 ast::vector<ast::DeclWithType>(),
71 std::move( params ),
72 { makeIntObj() },
73 nullptr,
74 ast::Storage::Classes(),
75 ast::Linkage::Spec(),
76 ast::vector<ast::Attribute>(),
77 ast::Function::Specs(),
78 isVarArgs
79 );
80 return Mangle::mangle( decl.get() );
[f42fc13]81}
82
[bef4f1a]83std::string getMangledNameOf0ParameterMain() {
84 return getMangledNameOfMain( {}, ast::VariableArgs );
[f42fc13]85}
86
[bef4f1a]87std::string getMangledNameOf2ParameterMain() {
88 return getMangledNameOfMain( {
89 makeIntObj(),
90 makeCharStarStarObj(),
91 }, ast::FixedArgs );
[f42fc13]92}
93
[0c577f7]94bool is_main( const std::string & mangled_name ) {
95 // This breaks if you move it out of the function.
96 static const std::string mangled_mains[] = {
[bef4f1a]97 getMangledNameOf0ParameterMain(),
98 getMangledNameOf2ParameterMain(),
99 //getMangledNameOf3ParameterMain(),
[0c577f7]100 };
101
102 for ( auto main_name : mangled_mains ) {
103 if ( main_name == mangled_name ) return true;
104 }
105 return false;
106}
107
[61efa42]108struct FixLinkageCore final {
109 ast::Linkage::Spec const spec;
110 FixLinkageCore( ast::Linkage::Spec spec ) : spec( spec ) {}
111
112 ast::FunctionDecl const * previsit( ast::FunctionDecl const * decl ) {
113 if ( decl->name != "main" ) return decl;
114 return ast::mutate_field( decl, &ast::FunctionDecl::linkage, spec );
115 }
116};
117
[f42fc13]118} // namespace
119
[61efa42]120bool isMain( const ast::FunctionDecl * decl ) {
[0c577f7]121 if ( std::string("main") != decl->name ) {
122 return false;
[f42fc13]123 }
[0c577f7]124 return is_main( Mangle::mangle( decl, Mangle::Type ) );
[f42fc13]125}
126
[61efa42]127void fixMainLinkage( ast::TranslationUnit & translationUnit,
128 bool replace_main ) {
129 ast::Linkage::Spec const spec =
130 ( replace_main ) ? ast::Linkage::Cforall : ast::Linkage::C;
131 ast::Pass<FixLinkageCore>::run( translationUnit, spec );
132}
133
134void fixMainInvoke( ast::TranslationUnit & translationUnit,
[bef4f1a]135 std::ostream &os, const char * bootloader_filename ) {
136
[61efa42]137 ast::Pass<FindMainCore> main_finder;
[bef4f1a]138 ast::accept_all( translationUnit, main_finder );
139 if ( nullptr == main_finder.core.main_declaration ) return;
140
141 ast::FunctionDecl * main_declaration =
142 ast::mutate( main_finder.core.main_declaration );
143
144 main_declaration->mangleName = Mangle::mangle( main_declaration );
145
146 os << "static inline int invoke_main(int argc, char* argv[], char* envp[]) { (void)argc; (void)argv; (void)envp; return ";
147 os << main_declaration->scopedMangleName() << "(";
148 const auto& params = main_declaration->type->params;
149 switch ( params.size() ) {
150 case 3: os << "(" << genTypeAt(params, 0) << ")argc, (" << genTypeAt(params, 1) << ")argv, (" << genTypeAt(params, 2) << ")envp"; break;
151 case 2: os << "(" << genTypeAt(params, 0) << ")argc, (" << genTypeAt(params, 1) << ")argv"; break;
152 case 0: break;
153 default : assert(false);
154 }
155 os << "); }\n";
156
157 std::ifstream bootloader( bootloader_filename, std::ios::in );
158 assertf( bootloader.is_open(), "cannot open bootloader.c\n" );
159 os << bootloader.rdbuf();
160}
161
162} // namespace CodeGen
Note: See TracBrowser for help on using the repository browser.