source: src/CodeGen/FixMain.cc@ 5b7fea7

ADT ast-experimental enum pthread-emulation qualifiedEnum
Last change on this file since 5b7fea7 was 8e48fca4, checked in by Andrew Beach <ajbeach@…>, 4 years ago

Combined the code in FixMain so it is all done with one pass.

  • Property mode set to 100644
File size: 4.6 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//
[bf2438c]7// FixMain.cc --
[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
[13de47bc]16
[bf2438c]17#include "FixMain.h"
[13de47bc]18
[bf2438c]19#include <cassert> // for assert, assertf
20#include <fstream> // for operator<<, basic_ostream::operator<<
21#include <list> // for list
22#include <string> // for operator<<
[13de47bc]23
[0c577f7]24#include "AST/Decl.hpp"
25#include "AST/Type.hpp"
[f42fc13]26#include "Common/PassVisitor.h"
[bf2438c]27#include "Common/SemanticError.h" // for SemanticError
[1619acd]28#include "CodeGen/GenType.h" // for GenType
[bf2438c]29#include "SynTree/Declaration.h" // for FunctionDecl, operator<<
30#include "SynTree/Type.h" // for FunctionType
[16ba4a6f]31#include "SymTab/Mangler.h"
[13de47bc]32
33namespace CodeGen {
[8e48fca4]34
35namespace {
36
37struct FindMainCore {
38 FunctionDecl * main_signature = nullptr;
39
40 void previsit( FunctionDecl * decl ) {
41 if ( FixMain::isMain( decl ) ) {
42 if ( main_signature ) {
43 SemanticError( decl, "Multiple definition of main routine\n" );
44 }
45 main_signature = decl;
46 }
47 }
48};
49
50}
51
[13de47bc]52 bool FixMain::replace_main = false;
[bf2438c]53
[1619acd]54 template<typename container>
55 std::string genTypeAt(const container& p, size_t idx) {
56 return genType((*std::next(p.begin(), idx))->get_type(), "");
57 }
58
[8e48fca4]59 void FixMain::fix( std::list< Declaration * > & translationUnit,
60 std::ostream &os, const char* bootloader_filename ) {
61 PassVisitor< FindMainCore > main_finder;
62 acceptAll( translationUnit, main_finder );
63 FunctionDecl * main_signature = main_finder.pass.main_signature;
[13de47bc]64
65 if( main_signature ) {
66 os << "static inline int invoke_main(int argc, char* argv[], char* envp[]) { (void)argc; (void)argv; (void)envp; return ";
[8e48fca4]67 main_signature->mangleName = SymTab::Mangler::mangle(main_signature);
[13de47bc]68
69 os << main_signature->get_scopedMangleName() << "(";
[1619acd]70 const auto& params = main_signature->get_functionType()->get_parameters();
71 switch(params.size()) {
72 case 3: os << "(" << genTypeAt(params, 0) << ")argc, (" << genTypeAt(params, 1) << ")argv, (" << genTypeAt(params, 2) << ")envp"; break;
73 case 2: os << "(" << genTypeAt(params, 0) << ")argc, (" << genTypeAt(params, 1) << ")argv"; break;
[13de47bc]74 case 0: break;
75 default : assert(false);
76 }
77 os << "); }\n";
78
79 std::ifstream bootloader(bootloader_filename, std::ios::in);
80 assertf( bootloader.is_open(), "cannot open bootloader.c\n" );
81 os << bootloader.rdbuf();
82 }
83 }
[f42fc13]84
85namespace {
86
87ObjectDecl * signedIntObj() {
88 return new ObjectDecl(
89 "", Type::StorageClasses(), LinkageSpec::Cforall, 0,
90 new BasicType( Type::Qualifiers(), BasicType::SignedInt ), nullptr );
91}
92
93ObjectDecl * charStarObj() {
94 return new ObjectDecl(
95 "", Type::StorageClasses(), LinkageSpec::Cforall, 0,
96 new PointerType( Type::Qualifiers(),
97 new PointerType( Type::Qualifiers(),
98 new BasicType( Type::Qualifiers(), BasicType::Char ) ) ),
99 nullptr );
100}
101
102std::string create_mangled_main_function_name( FunctionType * function_type ) {
103 std::unique_ptr<FunctionDecl> decl( new FunctionDecl(
104 "main", Type::StorageClasses(), LinkageSpec::Cforall,
105 function_type, nullptr ) );
106 return SymTab::Mangler::mangle( decl.get() );
107}
108
109std::string mangled_0_argument_main() {
110 FunctionType* main_type = new FunctionType( Type::Qualifiers(), true );
111 main_type->get_returnVals().push_back( signedIntObj() );
112 return create_mangled_main_function_name( main_type );
113}
114
115std::string mangled_2_argument_main() {
116 FunctionType* main_type = new FunctionType( Type::Qualifiers(), false );
117 main_type->get_returnVals().push_back( signedIntObj() );
118 main_type->get_parameters().push_back( signedIntObj() );
119 main_type->get_parameters().push_back( charStarObj() );
120 return create_mangled_main_function_name( main_type );
121}
122
[0c577f7]123bool is_main( const std::string & mangled_name ) {
124 // This breaks if you move it out of the function.
125 static const std::string mangled_mains[] = {
126 mangled_0_argument_main(),
127 mangled_2_argument_main(),
128 //mangled_3_argument_main(),
129 };
130
131 for ( auto main_name : mangled_mains ) {
132 if ( main_name == mangled_name ) return true;
133 }
134 return false;
135}
136
[f42fc13]137} // namespace
138
139bool FixMain::isMain( FunctionDecl * decl ) {
140 if ( std::string("main") != decl->name ) {
141 return false;
142 }
[0c577f7]143 return is_main( SymTab::Mangler::mangle( decl, true, true ) );
144}
145
146bool FixMain::isMain( const ast::FunctionDecl * decl ) {
147 if ( std::string("main") != decl->name ) {
148 return false;
[f42fc13]149 }
[0c577f7]150 return is_main( Mangle::mangle( decl, Mangle::Type ) );
[f42fc13]151}
152
[bf2438c]153};
Note: See TracBrowser for help on using the repository browser.