source: src/SymTab/FixFunction.cc@ 1b0184b

Last change on this file since 1b0184b was b2ecd48, checked in by Andrew Beach <ajbeach@…>, 2 years ago

Changes related to invariant checking scoping, it is not ready by these are unlikely to change.

  • Property mode set to 100644
File size: 6.3 KB
RevLine 
[0dd3a2f]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//
[40e636a]7// FixFunction.cc --
[0dd3a2f]8//
9// Author : Richard C. Bilson
10// Created On : Sun May 17 16:19:49 2015
[1931bb01]11// Last Modified By : Andrew Beach
12// Last Modified On : Tue Jul 12 14:28:00 2022
13// Update Count : 7
[0dd3a2f]14//
15
[51b73452]16#include "FixFunction.h"
[30f9072]17
18#include <list> // for list
19
[c1ed2ee]20#include "AST/Decl.hpp"
21#include "AST/Pass.hpp"
22#include "AST/Type.hpp"
[9feb34b]23#include "Common/utility.h" // for copy
[30f9072]24#include "SynTree/Declaration.h" // for FunctionDecl, ObjectDecl, Declarati...
[c3acf0aa]25#include "SynTree/Expression.h" // for Expression
[30f9072]26#include "SynTree/Type.h" // for ArrayType, PointerType, Type, Basic...
[51b73452]27
28namespace SymTab {
[c1ed2ee]29 class FixFunction_old : public WithShortCircuiting {
30 typedef Mutator Parent;
31 public:
32 FixFunction_old() : isVoid( false ) {}
[51b73452]33
[c1ed2ee]34 void premutate(FunctionDecl *functionDecl);
35 DeclarationWithType* postmutate(FunctionDecl *functionDecl);
[21b7161]36
[c1ed2ee]37 Type * postmutate(ArrayType * arrayType);
38
39 void premutate(ArrayType * arrayType);
40 void premutate(VoidType * voidType);
41 void premutate(BasicType * basicType);
42 void premutate(PointerType * pointerType);
43 void premutate(StructInstType * aggregateUseType);
44 void premutate(UnionInstType * aggregateUseType);
45 void premutate(EnumInstType * aggregateUseType);
46 void premutate(TraitInstType * aggregateUseType);
47 void premutate(TypeInstType * aggregateUseType);
48 void premutate(TupleType * tupleType);
49 void premutate(VarArgsType * varArgsType);
50 void premutate(ZeroType * zeroType);
51 void premutate(OneType * oneType);
52
53 bool isVoid;
54 };
55
56 DeclarationWithType * FixFunction_old::postmutate(FunctionDecl *functionDecl) {
[8b11840]57 // can't delete function type because it may contain assertions, so transfer ownership to new object
[21b7161]58 ObjectDecl *pointer = new ObjectDecl( functionDecl->name, functionDecl->get_storageClasses(), functionDecl->linkage, nullptr, new PointerType( Type::Qualifiers(), functionDecl->type ), nullptr, functionDecl->attributes );
[d53772d]59 pointer->location = functionDecl->location;
[21b7161]60 functionDecl->attributes.clear();
[8b11840]61 functionDecl->type = nullptr;
[0dd3a2f]62 delete functionDecl;
63 return pointer;
64 }
[51b73452]65
[4bda2cf]66 // xxx - this passes on void[], e.g.
67 // void foo(void [10]);
68 // does not cause an error
69
[c1ed2ee]70 Type * FixFunction_old::postmutate(ArrayType *arrayType) {
[40e636a]71 // need to recursively mutate the base type in order for multi-dimensional arrays to work.
[21b7161]72 PointerType *pointerType = new PointerType( arrayType->get_qualifiers(), arrayType->base, arrayType->dimension, arrayType->isVarLen, arrayType->isStatic );
[d53772d]73 pointerType->location = arrayType->location;
[21b7161]74 arrayType->base = nullptr;
75 arrayType->dimension = nullptr;
[0dd3a2f]76 delete arrayType;
77 return pointerType;
78 }
[51b73452]79
[c1ed2ee]80 void FixFunction_old::premutate(VoidType *) {
[21b7161]81 isVoid = true;
[89e6ffc]82 }
83
[c1ed2ee]84 void FixFunction_old::premutate(FunctionDecl *) { visit_children = false; }
85 void FixFunction_old::premutate(ArrayType *) { visit_children = false; }
86 void FixFunction_old::premutate(BasicType *) { visit_children = false; }
87 void FixFunction_old::premutate(PointerType *) { visit_children = false; }
88 void FixFunction_old::premutate(StructInstType *) { visit_children = false; }
89 void FixFunction_old::premutate(UnionInstType *) { visit_children = false; }
90 void FixFunction_old::premutate(EnumInstType *) { visit_children = false; }
91 void FixFunction_old::premutate(TraitInstType *) { visit_children = false; }
92 void FixFunction_old::premutate(TypeInstType *) { visit_children = false; }
93 void FixFunction_old::premutate(TupleType *) { visit_children = false; }
94 void FixFunction_old::premutate(VarArgsType *) { visit_children = false; }
95 void FixFunction_old::premutate(ZeroType *) { visit_children = false; }
96 void FixFunction_old::premutate(OneType *) { visit_children = false; }
[4bda2cf]97
98 bool fixFunction( DeclarationWithType *& dwt ) {
[c1ed2ee]99 PassVisitor<FixFunction_old> fixer;
[4bda2cf]100 dwt = dwt->acceptMutator( fixer );
101 return fixer.pass.isVoid;
102 }
[c1ed2ee]103
104namespace {
105 struct FixFunction_new final : public ast::WithShortCircuiting {
106 bool isVoid = false;
107
[c408483]108 void previsit( const ast::FunctionDecl * ) { visit_children = false; }
[c1ed2ee]109
[c408483]110 const ast::DeclWithType * postvisit( const ast::FunctionDecl * func ) {
[b2ecd48]111 // Cannot handle cases with asserions.
112 assert( func->assertions.empty() );
113 return new ast::ObjectDecl{
114 func->location, func->name, new ast::PointerType( func->type ), nullptr,
[c1ed2ee]115 func->storage, func->linkage, nullptr, copy( func->attributes ) };
116 }
117
[c408483]118 void previsit( const ast::ArrayType * ) { visit_children = false; }
[c1ed2ee]119
[c408483]120 const ast::Type * postvisit( const ast::ArrayType * array ) {
[b2ecd48]121 return new ast::PointerType{
122 array->base, array->dimension, array->isVarLen, array->isStatic,
[c1ed2ee]123 array->qualifiers };
124 }
125
[1931bb01]126 void previsit( const ast::FunctionType * ) { visit_children = false; }
127
128 const ast::Type * postvisit( const ast::FunctionType * type ) {
129 return new ast::PointerType( type );
130 }
131
[c408483]132 void previsit( const ast::VoidType * ) { isVoid = true; }
133
134 void previsit( const ast::BasicType * ) { visit_children = false; }
135 void previsit( const ast::PointerType * ) { visit_children = false; }
136 void previsit( const ast::StructInstType * ) { visit_children = false; }
137 void previsit( const ast::UnionInstType * ) { visit_children = false; }
138 void previsit( const ast::EnumInstType * ) { visit_children = false; }
139 void previsit( const ast::TraitInstType * ) { visit_children = false; }
140 void previsit( const ast::TypeInstType * ) { visit_children = false; }
141 void previsit( const ast::TupleType * ) { visit_children = false; }
142 void previsit( const ast::VarArgsType * ) { visit_children = false; }
143 void previsit( const ast::ZeroType * ) { visit_children = false; }
144 void previsit( const ast::OneType * ) { visit_children = false; }
[c1ed2ee]145 };
146} // anonymous namespace
147
148const ast::DeclWithType * fixFunction( const ast::DeclWithType * dwt, bool & isVoid ) {
149 ast::Pass< FixFunction_new > fixer;
150 dwt = dwt->accept( fixer );
[7ff3e522]151 isVoid |= fixer.core.isVoid;
[c1ed2ee]152 return dwt;
153}
154
[1931bb01]155const ast::Type * fixFunction( const ast::Type * type, bool & isVoid ) {
156 ast::Pass< FixFunction_new > fixer;
157 type = type->accept( fixer );
158 isVoid |= fixer.core.isVoid;
159 return type;
160}
161
[51b73452]162} // namespace SymTab
[0dd3a2f]163
164// Local Variables: //
165// tab-width: 4 //
166// mode: c++ //
167// compile-command: "make install" //
168// End: //
Note: See TracBrowser for help on using the repository browser.