source: src/Validate/Autogen.cpp @ c34a1a4

ADTast-experimental
Last change on this file since c34a1a4 was fb4dc28, checked in by Andrew Beach <ajbeach@…>, 19 months ago

Moved new ast code out of one of the old files. The new file may have to change if SymTab? is removed entirely, but for now at least, there is a lot less template code in headers.

  • Property mode set to 100644
File size: 27.4 KB
Line 
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//
7// Autogen.cpp -- Generate automatic routines for data types.
8//
9// Author           : Andrew Beach
10// Created On       : Thu Dec  2 13:44:00 2021
11// Last Modified By : Andrew Beach
12// Last Modified On : Tue Sep 20 16:00:00 2022
13// Update Count     : 2
14//
15
16#include "Autogen.hpp"
17
18#include <algorithm>               // for count_if
19#include <cassert>                 // for strict_dynamic_cast, assert, assertf
20#include <iterator>                // for back_insert_iterator, back_inserter
21#include <list>                    // for list, _List_iterator, list<>::iter...
22#include <set>                     // for set, _Rb_tree_const_iterator
23#include <utility>                 // for pair
24#include <vector>                  // for vector
25
26#include "AST/Attribute.hpp"
27#include "AST/Create.hpp"
28#include "AST/Decl.hpp"
29#include "AST/DeclReplacer.hpp"
30#include "AST/Expr.hpp"
31#include "AST/Inspect.hpp"
32#include "AST/Pass.hpp"
33#include "AST/Stmt.hpp"
34#include "AST/SymbolTable.hpp"
35#include "CodeGen/OperatorTable.h" // for isCtorDtor, isCtorDtorAssign
36#include "Common/ScopedMap.h"      // for ScopedMap<>::const_iterator, Scope...
37#include "Common/utility.h"        // for cloneAll, operator+
38#include "GenPoly/ScopedSet.h"     // for ScopedSet, ScopedSet<>::iterator
39#include "InitTweak/GenInit.h"     // for fixReturnStatements
40#include "InitTweak/InitTweak.h"   // for isAssignment, isCopyConstructor
41#include "SymTab/GenImplicitCall.hpp"  // for genImplicitCall
42#include "SymTab/Mangler.h"        // for Mangler
43#include "CompilationState.h"
44
45// TODO: The other new ast function should be moved over to this file.
46#include "SymTab/Autogen.h"
47
48namespace Validate {
49
50namespace {
51
52// --------------------------------------------------------------------------
53struct AutogenerateRoutines_new final :
54                public ast::WithDeclsToAdd<>,
55                public ast::WithShortCircuiting {
56        void previsit( const ast::EnumDecl * enumDecl );
57        void previsit( const ast::StructDecl * structDecl );
58        void previsit( const ast::UnionDecl * structDecl );
59        void previsit( const ast::TypeDecl * typeDecl );
60        void previsit( const ast::TraitDecl * traitDecl );
61        void previsit( const ast::FunctionDecl * functionDecl );
62        void postvisit( const ast::FunctionDecl * functionDecl );
63
64private:
65        // Current level of nested functions.
66        unsigned int functionNesting = 0;
67};
68
69// --------------------------------------------------------------------------
70/// Class used to generate functions for a particular declaration.
71/// Note it isn't really stored, it is just a class for organization and to
72/// help pass around some of the common arguments.
73class FuncGenerator {
74public:
75        std::list<ast::ptr<ast::Decl>> forwards;
76        std::list<ast::ptr<ast::Decl>> definitions;
77
78        FuncGenerator( const ast::Type * type, unsigned int functionNesting ) :
79                type( type ), functionNesting( functionNesting )
80        {}
81
82        /// Generate functions (and forward decls.) and append them to the list.
83        void generateAndAppendFunctions( std::list<ast::ptr<ast::Decl>> & );
84
85        virtual bool shouldAutogen() const = 0;
86protected:
87        const ast::Type * type;
88        unsigned int functionNesting;
89        ast::Linkage::Spec proto_linkage = ast::Linkage::AutoGen;
90
91        // Internal helpers:
92        void genStandardFuncs();
93        void produceDecl( const ast::FunctionDecl * decl );
94        void produceForwardDecl( const ast::FunctionDecl * decl );
95
96        const CodeLocation& getLocation() const { return getDecl()->location; }
97        ast::FunctionDecl * genProto( const std::string& name,
98                std::vector<ast::ptr<ast::DeclWithType>>&& params,
99                std::vector<ast::ptr<ast::DeclWithType>>&& returns ) const;
100
101        ast::ObjectDecl * dstParam() const;
102        ast::ObjectDecl * srcParam() const;
103        ast::FunctionDecl * genCtorProto() const;
104        ast::FunctionDecl * genCopyProto() const;
105        ast::FunctionDecl * genDtorProto() const;
106        ast::FunctionDecl * genAssignProto() const;
107        ast::FunctionDecl * genFieldCtorProto( unsigned int fields ) const;
108
109        // Internal Hooks:
110        virtual void genFuncBody( ast::FunctionDecl * decl ) = 0;
111        virtual void genFieldCtors() = 0;
112        virtual bool isConcurrentType() const { return false; }
113        virtual const ast::Decl * getDecl() const = 0;
114};
115
116class StructFuncGenerator final : public FuncGenerator {
117        const ast::StructDecl * decl;
118public:
119        StructFuncGenerator( const ast::StructDecl * decl,
120                        const ast::StructInstType * type,
121                        unsigned int functionNesting ) :
122                FuncGenerator( type, functionNesting ), decl( decl )
123        {}
124
125        // Built-ins do not use autogeneration.
126        bool shouldAutogen() const final { return !decl->linkage.is_builtin && !structHasFlexibleArray(decl); }
127private:
128        void genFuncBody( ast::FunctionDecl * decl ) final;
129        void genFieldCtors() final;
130        bool isConcurrentType() const final {
131                return decl->is_thread() || decl->is_monitor();
132        }
133        virtual const ast::Decl * getDecl() const final { return decl; }
134
135        /// Generates a single struct member operation.
136        /// (constructor call, destructor call, assignment call)
137        // This is managed because it uses another helper that returns a ast::ptr.
138        ast::ptr<ast::Stmt> makeMemberOp(
139                const CodeLocation& location,
140                const ast::ObjectDecl * dstParam, const ast::Expr * src,
141                const ast::ObjectDecl * field, ast::FunctionDecl * func,
142                SymTab::LoopDirection direction );
143
144        /// Generates the body of a struct function by iterating the struct members
145        /// (via parameters). Generates default constructor, copy constructor,
146        /// copy assignment, and destructor bodies. No field constructor bodies.
147        template<typename Iterator>
148        void makeFunctionBody( Iterator member, Iterator end,
149                        ast::FunctionDecl * func, SymTab::LoopDirection direction );
150
151        /// Generate the body of a constructor which takes parameters that match
152        /// fields. (With arguments for one to all of the fields.)
153        template<typename Iterator>
154        void makeFieldCtorBody( Iterator member, Iterator end,
155                        ast::FunctionDecl * func );
156};
157
158class UnionFuncGenerator final : public FuncGenerator {
159        const ast::UnionDecl * decl;
160public:
161        UnionFuncGenerator( const ast::UnionDecl * decl,
162                        const ast::UnionInstType * type,
163                        unsigned int functionNesting ) :
164                FuncGenerator( type, functionNesting ), decl( decl )
165        {}
166
167        // Built-ins do not use autogeneration.
168        bool shouldAutogen() const final { return !decl->linkage.is_builtin; }
169private:
170        void genFuncBody( ast::FunctionDecl * decl ) final;
171        void genFieldCtors() final;
172        const ast::Decl * getDecl() const final { return decl; }
173
174        /// Generate a single union assignment expression (using memcpy).
175        ast::ExprStmt * makeAssignOp( const CodeLocation& location,
176                const ast::ObjectDecl * dstParam, const ast::ObjectDecl * srcParam );
177};
178
179class EnumFuncGenerator final : public FuncGenerator {
180        const ast::EnumDecl * decl;
181public:
182        EnumFuncGenerator( const ast::EnumDecl * decl,
183                        const ast::EnumInstType * type,
184                        unsigned int functionNesting ) :
185                FuncGenerator( type, functionNesting ), decl( decl )
186        {
187                // TODO: These functions are somewhere between instrinsic and autogen,
188                // could possibly use a new linkage type. For now we just make the
189                // basic ones intrinsic to code-gen them as C assignments.
190                const auto & real_type = decl->base;
191                const auto & basic = real_type.as<ast::BasicType>();
192                if(!real_type || (basic && basic->isInteger())) proto_linkage = ast::Linkage::Intrinsic;
193        }
194
195        bool shouldAutogen() const final { return true; }
196private:
197        void genFuncBody( ast::FunctionDecl * decl ) final;
198        void genFieldCtors() final;
199        const ast::Decl * getDecl() const final { return decl; }
200};
201
202class TypeFuncGenerator final : public FuncGenerator {
203        const ast::TypeDecl * decl;
204public:
205        TypeFuncGenerator( const ast::TypeDecl * decl,
206                        ast::TypeInstType * type,
207                        unsigned int functionNesting ) :
208                FuncGenerator( type, functionNesting ), decl( decl )
209        {}
210
211        bool shouldAutogen() const final { return true; }
212        void genFieldCtors() final;
213private:
214        void genFuncBody( ast::FunctionDecl * decl ) final;
215        const ast::Decl * getDecl() const final { return decl; }
216};
217
218// --------------------------------------------------------------------------
219const std::vector<ast::ptr<ast::TypeDecl>>& getGenericParams( const ast::Type * t ) {
220        if ( auto inst = dynamic_cast< const ast::StructInstType * >( t ) ) {
221                return inst->base->params;
222        } else if ( auto inst = dynamic_cast< const ast::UnionInstType * >( t ) ) {
223                return inst->base->params;
224        }
225        static std::vector<ast::ptr<ast::TypeDecl>> const empty;
226        return empty;
227}
228
229/// Changes the node inside a pointer so that it has the unused attribute.
230void addUnusedAttribute( ast::ptr<ast::DeclWithType> & declPtr ) {
231        ast::DeclWithType * decl = declPtr.get_and_mutate();
232        decl->attributes.push_back( new ast::Attribute( "unused" ) );
233}
234
235// --------------------------------------------------------------------------
236void AutogenerateRoutines_new::previsit( const ast::EnumDecl * enumDecl ) {
237        // Must visit children (enum constants) to add them to the symbol table.
238        if ( !enumDecl->body ) return;
239
240        // if ( auto enumBaseType = enumDecl->base ) {
241        //      if ( auto enumBaseTypeAsStructInst = dynamic_cast<const ast::StructInstType *>(enumBaseType.get()) ) {
242        //              const ast::StructDecl * structDecl = enumBaseTypeAsStructInst->base.get();
243        //              this->previsit( structDecl );
244        //      }
245        // }
246
247        ast::EnumInstType enumInst( enumDecl->name );
248        enumInst.base = enumDecl;
249        EnumFuncGenerator gen( enumDecl, &enumInst, functionNesting );
250        gen.generateAndAppendFunctions( declsToAddAfter );
251}
252
253void AutogenerateRoutines_new::previsit( const ast::StructDecl * structDecl ) {
254        visit_children = false;
255        if ( !structDecl->body ) return;
256
257        ast::StructInstType structInst( structDecl->name );
258        structInst.base = structDecl;
259        for ( const ast::TypeDecl * typeDecl : structDecl->params ) {
260                structInst.params.push_back( new ast::TypeExpr(
261                        typeDecl->location,
262                        new ast::TypeInstType( typeDecl )
263                ) );
264        }
265        StructFuncGenerator gen( structDecl, &structInst, functionNesting );
266        gen.generateAndAppendFunctions( declsToAddAfter );
267}
268
269void AutogenerateRoutines_new::previsit( const ast::UnionDecl * unionDecl ) {
270        visit_children = false;
271        if ( !unionDecl->body ) return;
272
273        ast::UnionInstType unionInst( unionDecl->name );
274        unionInst.base = unionDecl;
275        for ( const ast::TypeDecl * typeDecl : unionDecl->params ) {
276                unionInst.params.push_back( new ast::TypeExpr(
277                        unionDecl->location,
278                        new ast::TypeInstType( typeDecl )
279                ) );
280        }
281        UnionFuncGenerator gen( unionDecl, &unionInst, functionNesting );
282        gen.generateAndAppendFunctions( declsToAddAfter );
283}
284
285/// Generate ctor/dtors/assign for typedecls, e.g., otype T = int *;
286void AutogenerateRoutines_new::previsit( const ast::TypeDecl * typeDecl ) {
287        if ( !typeDecl->base ) return;
288
289        ast::TypeInstType refType( typeDecl->name, typeDecl );
290        TypeFuncGenerator gen( typeDecl, &refType, functionNesting );
291        gen.generateAndAppendFunctions( declsToAddAfter );
292}
293
294void AutogenerateRoutines_new::previsit( const ast::TraitDecl * ) {
295        // Ensure that we don't add assignment ops for types defined as part of the trait
296        visit_children = false;
297}
298
299void AutogenerateRoutines_new::previsit( const ast::FunctionDecl * ) {
300        // Track whether we're currently in a function.
301        // Can ignore function type idiosyncrasies, because function type can never
302        // declare a new type.
303        functionNesting += 1;
304}
305
306void AutogenerateRoutines_new::postvisit( const ast::FunctionDecl * ) {
307        functionNesting -= 1;
308}
309
310void FuncGenerator::generateAndAppendFunctions(
311                std::list<ast::ptr<ast::Decl>> & decls ) {
312        if ( !shouldAutogen() ) return;
313
314        // Generate the functions (they go into forwards and definitions).
315        genStandardFuncs();
316        genFieldCtors();
317
318        // Now export the lists contents.
319        decls.splice( decls.end(), forwards );
320        decls.splice( decls.end(), definitions );
321}
322
323void FuncGenerator::produceDecl( const ast::FunctionDecl * decl ) {
324        assert( nullptr != decl->stmts );
325
326        definitions.push_back( decl );
327}
328
329/// Make a forward declaration of the decl and add it to forwards.
330void FuncGenerator::produceForwardDecl( const ast::FunctionDecl * decl ) {
331        if (0 != functionNesting) return;
332        ast::FunctionDecl * fwd =
333                ( decl->stmts ) ? ast::asForward( decl ) : ast::deepCopy( decl ) ;
334        fwd->fixUniqueId();
335        forwards.push_back( fwd );
336}
337
338/// Generates a basic prototype function declaration.
339ast::FunctionDecl * FuncGenerator::genProto( const std::string& name,
340                std::vector<ast::ptr<ast::DeclWithType>>&& params,
341                std::vector<ast::ptr<ast::DeclWithType>>&& returns ) const {
342
343        // Handle generic prameters and assertions, if any.
344        auto const & old_type_params = getGenericParams( type );
345        std::vector<ast::ptr<ast::TypeDecl>> type_params;
346        std::vector<ast::ptr<ast::DeclWithType>> assertions;
347        for ( auto & old_param : old_type_params ) {
348                ast::TypeDecl * decl = ast::deepCopy( old_param );
349                for ( auto assertion : decl->assertions ) {
350                        assertions.push_back( assertion );
351                }
352                decl->assertions.clear();
353                type_params.push_back( decl );
354        }
355        // TODO: The values in params and returns still may point at the old
356        // generic params, that does not appear to be an issue but perhaps it
357        // should be addressed.
358
359        ast::FunctionDecl * decl = new ast::FunctionDecl(
360                // Auto-generated routines use the type declaration's location.
361                getLocation(),
362                name,
363                std::move( type_params ),
364                std::move( assertions ),
365                std::move( params ),
366                std::move( returns ),
367                // Only a prototype, no body.
368                nullptr,
369                // Use static storage if we are at the top level.
370                (0 < functionNesting) ? ast::Storage::Classes() : ast::Storage::Static,
371                proto_linkage,
372                std::vector<ast::ptr<ast::Attribute>>(),
373                // Auto-generated routines are inline to avoid conflicts.
374                ast::Function::Specs( ast::Function::Inline ) );
375        decl->fixUniqueId();
376        return decl;
377}
378
379ast::ObjectDecl * FuncGenerator::dstParam() const {
380        return new ast::ObjectDecl( getLocation(), "_dst",
381                new ast::ReferenceType( ast::deepCopy( type ) ) );
382}
383
384ast::ObjectDecl * FuncGenerator::srcParam() const {
385        return new ast::ObjectDecl( getLocation(), "_src",
386                ast::deepCopy( type ) );
387}
388
389/// Use the current type T to create `void ?{}(T & _dst)`.
390ast::FunctionDecl * FuncGenerator::genCtorProto() const {
391        return genProto( "?{}", { dstParam() }, {} );
392}
393
394/// Use the current type T to create `void ?{}(T & _dst, T _src)`.
395ast::FunctionDecl * FuncGenerator::genCopyProto() const {
396        return genProto( "?{}", { dstParam(), srcParam() }, {} );
397}
398
399/// Use the current type T to create `void ?{}(T & _dst)`.
400ast::FunctionDecl * FuncGenerator::genDtorProto() const {
401        // The destructor must be mutex on a concurrent type.
402        auto dst = dstParam();
403        if ( isConcurrentType() ) {
404                add_qualifiers( dst->type, ast::CV::Qualifiers( ast::CV::Mutex ) );
405        }
406        return genProto( "^?{}", { dst }, {} );
407}
408
409/// Use the current type T to create `T ?{}(T & _dst, T _src)`.
410ast::FunctionDecl * FuncGenerator::genAssignProto() const {
411        // Only the name is different, so just reuse the generation function.
412        auto retval = srcParam();
413        retval->name = "_ret";
414        return genProto( "?=?", { dstParam(), srcParam() }, { retval } );
415}
416
417// This one can return null if the last field is an unnamed bitfield.
418ast::FunctionDecl * FuncGenerator::genFieldCtorProto(
419                unsigned int fields ) const {
420        assert( 0 < fields );
421        auto aggr = strict_dynamic_cast<const ast::AggregateDecl *>( getDecl() );
422
423        std::vector<ast::ptr<ast::DeclWithType>> params = { dstParam() };
424        for ( unsigned int index = 0 ; index < fields ; ++index ) {
425                auto member = aggr->members[index].strict_as<ast::DeclWithType>();
426                if ( ast::isUnnamedBitfield(
427                                dynamic_cast<const ast::ObjectDecl *>( member ) ) ) {
428                        if ( index == fields - 1 ) {
429                                return nullptr;
430                        }
431                        continue;
432                }
433
434                auto * paramType = ast::deepCopy( member->get_type() );
435                paramType->attributes.clear();
436                ast::ObjectDecl * param = new ast::ObjectDecl(
437                        getLocation(), member->name, paramType );
438                for ( auto & attr : member->attributes ) {
439                        if ( attr->isValidOnFuncParam() ) {
440                                param->attributes.push_back( attr );
441                        }
442                }
443                params.emplace_back( param );
444        }
445        return genProto( "?{}", std::move( params ), {} );
446}
447
448void appendReturnThis( ast::FunctionDecl * decl ) {
449        assert( 1 <= decl->params.size() );
450        assert( 1 == decl->returns.size() );
451        assert( decl->stmts );
452
453        const CodeLocation& location = (decl->stmts->kids.empty())
454                ? decl->stmts->location : decl->stmts->kids.back()->location;
455        const ast::DeclWithType * thisParam = decl->params.front();
456        decl->stmts.get_and_mutate()->push_back(
457                new ast::ReturnStmt( location,
458                        new ast::VariableExpr( location, thisParam )
459                )
460        );
461}
462
463void FuncGenerator::genStandardFuncs() {
464        // The order here determines the order that these functions are generated.
465        // Assignment should come last since it uses copy constructor in return.
466        ast::FunctionDecl *(FuncGenerator::*standardProtos[4])() const = {
467                        &FuncGenerator::genCtorProto, &FuncGenerator::genCopyProto,
468                        &FuncGenerator::genDtorProto, &FuncGenerator::genAssignProto };
469        for ( auto & generator : standardProtos ) {
470                ast::FunctionDecl * decl = (this->*generator)();
471                produceForwardDecl( decl );
472                genFuncBody( decl );
473                if ( CodeGen::isAssignment( decl->name ) ) {
474                        appendReturnThis( decl );
475                }
476                produceDecl( decl );
477        }
478}
479
480void StructFuncGenerator::genFieldCtors() {
481        // The field constructors are only generated if the default constructor
482        // and copy constructor are both generated, since the need both.
483        unsigned numCtors = std::count_if( definitions.begin(), definitions.end(),
484                [](const ast::Decl * decl){ return CodeGen::isConstructor( decl->name ); }
485        );
486        if ( 2 != numCtors ) return;
487
488        for ( unsigned int num = 1 ; num <= decl->members.size() ; ++num ) {
489                ast::FunctionDecl * ctor = genFieldCtorProto( num );
490                if ( nullptr == ctor ) {
491                        continue;
492                }
493                produceForwardDecl( ctor );
494                makeFieldCtorBody( decl->members.begin(), decl->members.end(), ctor );
495                produceDecl( ctor );
496        }
497}
498
499void StructFuncGenerator::genFuncBody( ast::FunctionDecl * functionDecl ) {
500        // Generate appropriate calls to member constructors and assignment.
501        // Destructor needs to do everything in reverse,
502        // so pass "forward" based on whether the function is a destructor
503        if ( CodeGen::isDestructor( functionDecl->name ) ) {
504                makeFunctionBody( decl->members.rbegin(), decl->members.rend(),
505                        functionDecl, SymTab::LoopBackward );
506        } else {
507                makeFunctionBody( decl->members.begin(), decl->members.end(),
508                        functionDecl, SymTab::LoopForward );
509        }
510}
511
512ast::ptr<ast::Stmt> StructFuncGenerator::makeMemberOp(
513                const CodeLocation& location, const ast::ObjectDecl * dstParam,
514                const ast::Expr * src, const ast::ObjectDecl * field,
515                ast::FunctionDecl * func, SymTab::LoopDirection direction ) {
516        InitTweak::InitExpander_new srcParam( src );
517        // Assign to destination.
518        ast::Expr * dstSelect = new ast::MemberExpr(
519                location,
520                field,
521                new ast::CastExpr(
522                        location,
523                        new ast::VariableExpr( location, dstParam ),
524                        dstParam->type.strict_as<ast::ReferenceType>()->base
525                )
526        );
527        return genImplicitCall(
528                srcParam, dstSelect, location, func->name,
529                field, direction
530        );
531}
532
533template<typename Iterator>
534void StructFuncGenerator::makeFunctionBody( Iterator current, Iterator end,
535                ast::FunctionDecl * func, SymTab::LoopDirection direction ) {
536        // Trying to get the best code location. Should probably use a helper or
537        // just figure out what that would be given where this is called.
538        assert( nullptr == func->stmts );
539        const CodeLocation& location = func->location;
540
541        ast::CompoundStmt * stmts = new ast::CompoundStmt( location );
542
543        for ( ; current != end ; ++current ) {
544                const ast::ptr<ast::Decl> & member = *current;
545                auto field = member.as<ast::ObjectDecl>();
546                if ( nullptr == field ) {
547                        continue;
548                }
549
550                // Don't assign to constant members (but do construct/destruct them).
551                if ( CodeGen::isAssignment( func->name ) ) {
552                        // For array types we need to strip off the array layers.
553                        const ast::Type * type = field->get_type();
554                        while ( auto at = dynamic_cast<const ast::ArrayType *>( type ) ) {
555                                type = at->base;
556                        }
557                        if ( type->is_const() ) {
558                                continue;
559                        }
560                }
561
562                assert( !func->params.empty() );
563                const ast::ObjectDecl * dstParam =
564                        func->params.front().strict_as<ast::ObjectDecl>();
565                const ast::ObjectDecl * srcParam = nullptr;
566                if ( 2 == func->params.size() ) {
567                        srcParam = func->params.back().strict_as<ast::ObjectDecl>();
568                }
569
570                ast::Expr * srcSelect = (srcParam) ? new ast::MemberExpr(
571                        location, field, new ast::VariableExpr( location, srcParam )
572                ) : nullptr;
573                ast::ptr<ast::Stmt> stmt =
574                        makeMemberOp( location, dstParam, srcSelect, field, func, direction );
575
576                if ( nullptr != stmt ) {
577                        stmts->kids.push_back( stmt );
578                }
579        }
580
581        func->stmts = stmts;
582}
583
584template<typename Iterator>
585void StructFuncGenerator::makeFieldCtorBody( Iterator current, Iterator end,
586                ast::FunctionDecl * func ) {
587        const CodeLocation& location = func->location;
588        auto & params = func->params;
589        // Need at least the constructed parameter and one field parameter.
590        assert( 2 <= params.size() );
591
592        ast::CompoundStmt * stmts = new ast::CompoundStmt( location );
593
594        auto dstParam = params.front().strict_as<ast::ObjectDecl>();
595        // Skip over the 'this' parameter.
596        for ( auto param = params.begin() + 1 ; current != end ; ++current ) {
597                const ast::ptr<ast::Decl> & member = *current;
598                ast::ptr<ast::Stmt> stmt = nullptr;
599                auto field = member.as<ast::ObjectDecl>();
600                // Not sure why it could be null.
601                // Don't make a function for a parameter that is an unnamed bitfield.
602                if ( nullptr == field || ast::isUnnamedBitfield( field ) ) {
603                        continue;
604                // Matching Parameter: Initialize the field by copy.
605                } else if ( params.end() != param ) {
606                        const ast::Expr *srcSelect = new ast::VariableExpr(
607                                func->location, param->get() );
608                        stmt = makeMemberOp( location, dstParam, srcSelect, field, func, SymTab::LoopForward );
609                        ++param;
610                // No Matching Parameter: Initialize the field by default constructor.
611                } else {
612                        stmt = makeMemberOp( location, dstParam, nullptr, field, func, SymTab::LoopForward );
613                }
614
615                if ( nullptr != stmt ) {
616                        stmts->kids.push_back( stmt );
617                }
618        }
619        func->stmts = stmts;
620}
621
622void UnionFuncGenerator::genFieldCtors() {
623        // Field constructors are only generated if default and copy constructor
624        // are generated, since they need access to both
625        unsigned numCtors = std::count_if( definitions.begin(), definitions.end(),
626                []( const ast::Decl * d ){ return CodeGen::isConstructor( d->name ); }
627        );
628        if ( 2 != numCtors ) {
629                return;
630        }
631
632        // Create a constructor which takes the first member type as a
633        // parameter. For example for `union A { int x; char y; };` generate
634        // a function with signature `void ?{}(A *, int)`. This mimics C's
635        // behaviour which initializes the first member of the union.
636
637        // Still, there must be some members.
638        if ( !decl->members.empty() ) {
639                ast::FunctionDecl * ctor = genFieldCtorProto( 1 );
640                if ( nullptr == ctor ) {
641                        return;
642                }
643                produceForwardDecl( ctor );
644                auto params = ctor->params;
645                auto dstParam = params.front().strict_as<ast::ObjectDecl>();
646                auto srcParam = params.back().strict_as<ast::ObjectDecl>();
647                ctor->stmts = new ast::CompoundStmt( getLocation(),
648                        { makeAssignOp( getLocation(), dstParam, srcParam ) }
649                );
650                produceDecl( ctor );
651        }
652}
653
654void UnionFuncGenerator::genFuncBody( ast::FunctionDecl * functionDecl ) {
655        const CodeLocation& location = functionDecl->location;
656        auto & params = functionDecl->params;
657        if ( InitTweak::isCopyConstructor( functionDecl )
658                        || InitTweak::isAssignment( functionDecl ) ) {
659                assert( 2 == params.size() );
660                auto dstParam = params.front().strict_as<ast::ObjectDecl>();
661                auto srcParam = params.back().strict_as<ast::ObjectDecl>();
662                functionDecl->stmts = new ast::CompoundStmt( location,
663                        { makeAssignOp( location, dstParam, srcParam ) }
664                );
665        } else {
666                assert( 1 == params.size() );
667                // Default constructor and destructor is empty.
668                functionDecl->stmts = new ast::CompoundStmt( location );
669                // Add unused attribute to parameter to silence warnings.
670                addUnusedAttribute( params.front() );
671
672                // Just an extra step to make the forward and declaration match.
673                if ( forwards.empty() ) return;
674                ast::FunctionDecl * fwd = strict_dynamic_cast<ast::FunctionDecl *>(
675                        forwards.back().get_and_mutate() );
676                addUnusedAttribute( fwd->params.front() );
677        }
678}
679
680ast::ExprStmt * UnionFuncGenerator::makeAssignOp( const CodeLocation& location,
681                const ast::ObjectDecl * dstParam, const ast::ObjectDecl * srcParam ) {
682        return new ast::ExprStmt( location, new ast::UntypedExpr(
683                location,
684                new ast::NameExpr( location, "__builtin_memcpy" ),
685                {
686                        new ast::AddressExpr( location,
687                                new ast::VariableExpr( location, dstParam ) ),
688                        new ast::AddressExpr( location,
689                                new ast::VariableExpr( location, srcParam ) ),
690                        new ast::SizeofExpr( location, srcParam->type ),
691                } ) );
692}
693
694void EnumFuncGenerator::genFieldCtors() {
695        // Enumerations to not have field constructors.
696}
697
698void EnumFuncGenerator::genFuncBody( ast::FunctionDecl * functionDecl ) {
699        const CodeLocation& location = functionDecl->location;
700        auto & params = functionDecl->params;
701        if ( InitTweak::isCopyConstructor( functionDecl )
702                        || InitTweak::isAssignment( functionDecl ) ) {
703                assert( 2 == params.size() );
704                auto dstParam = params.front().strict_as<ast::ObjectDecl>();
705                auto srcParam = params.back().strict_as<ast::ObjectDecl>();
706
707                /* This looks like a recursive call, but code-gen will turn it into
708                 * a C-style assignment.
709                 *
710                 * This is still before function pointer type conversion,
711                 * so this will have to do it manually.
712                 *
713                 * It will also reference the parent function declaration, creating
714                 * a cycle for references. This also means that the ref-counts are
715                 * now non-zero and the declaration will be deleted if it ever
716                 * returns to zero.
717                 */
718                auto callExpr = new ast::ApplicationExpr( location,
719                        ast::VariableExpr::functionPointer( location, functionDecl ),
720                        {
721                                new ast::VariableExpr( location, dstParam ),
722                                new ast::VariableExpr( location, srcParam ),
723                        }
724                );
725                functionDecl->stmts = new ast::CompoundStmt( location,
726                        { new ast::ExprStmt( location, callExpr ) }
727                );
728        } else {
729                assert( 1 == params.size() );
730                // Default constructor and destructor is empty.
731                functionDecl->stmts = new ast::CompoundStmt( location );
732                // Just add unused attribute to parameter to silence warnings.
733                addUnusedAttribute( params.front() );
734
735                // Just an extra step to make the forward and declaration match.
736                if ( forwards.empty() ) return;
737                ast::FunctionDecl * fwd = strict_dynamic_cast<ast::FunctionDecl *>(
738                        forwards.back().get_and_mutate() );
739                addUnusedAttribute( fwd->params.front() );
740        }
741}
742
743void TypeFuncGenerator::genFieldCtors() {
744        // Opaque types do not have field constructors.
745}
746
747void TypeFuncGenerator::genFuncBody( ast::FunctionDecl * functionDecl ) {
748        const CodeLocation& location = functionDecl->location;
749        auto & params = functionDecl->type->params;
750        assertf( 1 == params.size() || 2 == params.size(),
751                "Incorrect number of parameters in autogenerated typedecl function: %zd",
752                params.size() );
753        auto dstParam = params.front().strict_as<ast::ObjectDecl>();
754        auto srcParam = (2 == params.size())
755                ? params.back().strict_as<ast::ObjectDecl>() : nullptr;
756        // Generate appropriate calls to member constructor and assignment.
757        ast::UntypedExpr * expr = new ast::UntypedExpr( location,
758                new ast::NameExpr( location, functionDecl->name )
759        );
760        expr->args.push_back( new ast::CastExpr( location,
761                new ast::VariableExpr( location, dstParam ),
762                new ast::ReferenceType( decl->base )
763        ) );
764        if ( srcParam ) {
765                expr->args.push_back( new ast::CastExpr( location,
766                        new ast::VariableExpr( location, srcParam ),
767                        decl->base
768                ) );
769        }
770        functionDecl->stmts = new ast::CompoundStmt( location,
771                { new ast::ExprStmt( location, expr ) }
772        );
773}
774
775} // namespace
776
777void autogenerateRoutines( ast::TranslationUnit & translationUnit ) {
778        ast::Pass<AutogenerateRoutines_new>::run( translationUnit );
779}
780
781} // Validate
782
783// Local Variables: //
784// tab-width: 4 //
785// mode: c++ //
786// compile-command: "make install" //
787// End: //
Note: See TracBrowser for help on using the repository browser.