source: src/SynTree/AddressExpr.cc @ 8a6cf7e

ADTaaron-thesisarm-ehast-experimentalcleanup-dtorsdeferred_resndemanglerenumforall-pointer-decayjacob/cs343-translationjenkins-sandboxnew-astnew-ast-unique-exprnew-envno_listpersistent-indexerpthread-emulationqualifiedEnumresolv-newwith_gc
Last change on this file since 8a6cf7e was c5e3208, checked in by Rob Schluntz <rschlunt@…>, 7 years ago

Properly assign types to AddressExpr?

  • Property mode set to 100644
File size: 2.1 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//
[89231bc]7// AddressExpr.cc --
[0dd3a2f]8//
9// Author           : Richard C. Bilson
10// Created On       : Sun May 17 23:54:44 2015
[89231bc]11// Last Modified By : Rob Schluntz
12// Last Modified On : Tue Apr 26 12:35:13 2016
[a08ba92]13// Update Count     : 6
[0dd3a2f]14//
[51b7345]15
16#include "Expression.h"
17#include "Type.h"
[d3b7937]18#include "Common/utility.h"
[51b7345]19
[c5e3208]20// Address expressions are typed based on the following inference rules:
21//    E : lvalue T  &..& (n references)
22//   &E :        T *&..& (n references)
23//
24//    E : T  &..&        (m references)
25//   &E : T *&..&        (m-1 references)
26//
27// That is, lvalues becomes
28
29namespace {
30        Type * addrType( Type * type ) {
31                if ( ReferenceType * refType = dynamic_cast< ReferenceType * >( type ) ) {
32                        return new ReferenceType( refType->get_qualifiers(), addrType( refType->get_base() ) );
33                } else {
34                        return new PointerType( Type::Qualifiers(), type->clone() );
35                }
36        }
37}
38
[0dd3a2f]39AddressExpr::AddressExpr( Expression *arg, Expression *_aname ) : Expression( _aname ), arg( arg ) {
[906e24d]40        if ( arg->has_result() ) {
[c5e3208]41                if ( arg->get_result()->get_lvalue() ) {
42                        // lvalue, retains all layers of reference and gains a pointer inside the references
43                        set_result( addrType( arg->get_result() ) );
[ce8c12f]44                } else {
[c5e3208]45                        // taking address of non-lvalue -- must be a reference, loses one layer of reference
46                        ReferenceType * refType = safe_dynamic_cast< ReferenceType * >( arg->get_result() );
47                        set_result( addrType( refType->get_base() ) );
[ce8c12f]48                }
[c5e3208]49                // result of & is never an lvalue
50                get_result()->set_lvalue( false );
[906e24d]51        }
[51b7345]52}
53
[0dd3a2f]54AddressExpr::AddressExpr( const AddressExpr &other ) : Expression( other ), arg( maybeClone( other.arg ) ) {
[51b7345]55}
56
[0dd3a2f]57AddressExpr::~AddressExpr() {
[a08ba92]58        delete arg;
[51b7345]59}
60
[0dd3a2f]61void AddressExpr::print( std::ostream &os, int indent ) const {
[89231bc]62        os << "Address of:" << std::endl;
[a08ba92]63        if ( arg ) {
[89231bc]64                os << std::string( indent+2, ' ' );
[906e24d]65                arg->print( os, indent+2 );
[a08ba92]66        } // if
[51b7345]67}
68
[0dd3a2f]69// Local Variables: //
70// tab-width: 4 //
71// mode: c++ //
72// compile-command: "make install" //
73// End: //
Note: See TracBrowser for help on using the repository browser.