Annotation of GNUtools/cc/cp-init.c, revision 1.1

1.1     ! root        1: /* Handle initialization things in C++.
        !             2:    Copyright (C) 1987, 1989, 1992, 1993 Free Software Foundation, Inc.
        !             3:    Contributed by Michael Tiemann ([email protected])
        !             4: 
        !             5: This file is part of GNU CC.
        !             6: 
        !             7: GNU CC is free software; you can redistribute it and/or modify
        !             8: it under the terms of the GNU General Public License as published by
        !             9: the Free Software Foundation; either version 2, or (at your option)
        !            10: any later version.
        !            11: 
        !            12: GNU CC is distributed in the hope that it will be useful,
        !            13: but WITHOUT ANY WARRANTY; without even the implied warranty of
        !            14: MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
        !            15: GNU General Public License for more details.
        !            16: 
        !            17: You should have received a copy of the GNU General Public License
        !            18: along with GNU CC; see the file COPYING.  If not, write to
        !            19: the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.  */
        !            20: 
        !            21: 
        !            22: /* High-level class interface. */
        !            23: 
        !            24: #include "config.h"
        !            25: #include "tree.h"
        !            26: #include "rtl.h"
        !            27: #include "cp-tree.h"
        !            28: #include "flags.h"
        !            29: 
        !            30: #undef NULL
        !            31: #define NULL 0
        !            32: 
        !            33: /* In C++, structures with well-defined constructors are initialized by
        !            34:    those constructors, unasked.  CURRENT_BASE_INIT_LIST
        !            35:    holds a list of stmts for a BASE_INIT term in the grammar.
        !            36:    This list has one element for each base class which must be
        !            37:    initialized.  The list elements are [basename, init], with
        !            38:    type basetype.  This allows the possibly anachronistic form
        !            39:    (assuming d : a, b, c) "d (int a) : c(a+5), b (a-4), a (a+3)"
        !            40:    where each successive term can be handed down the constructor
        !            41:    line.  Perhaps this was not intended.  */
        !            42: tree current_base_init_list, current_member_init_list;
        !            43: 
        !            44: void emit_base_init ();
        !            45: void check_base_init ();
        !            46: static void expand_aggr_vbase_init ();
        !            47: void expand_member_init ();
        !            48: void expand_aggr_init ();
        !            49: 
        !            50: static void expand_aggr_init_1 ();
        !            51: static void expand_recursive_init_1 ();
        !            52: static void expand_recursive_init ();
        !            53: tree expand_vec_init ();
        !            54: tree build_vec_delete ();
        !            55: 
        !            56: static void add_friend (), add_friends ();
        !            57: 
        !            58: /* Cache _builtin_new and _builtin_delete exprs.  */
        !            59: static tree BIN, BID;
        !            60: 
        !            61: /* Cache the identifier nodes for the two magic field of a new cookie.  */
        !            62: static tree nc_nelts_field_id;
        !            63: static tree nc_ptr_2comp_field_id;
        !            64: 
        !            65: static tree minus_one;
        !            66: 
        !            67: /* Set up local variable for this file.  MUST BE CALLED AFTER
        !            68:    INIT_DECL_PROCESSING.  */
        !            69: 
        !            70: tree BI_header_type, BI_header_size;
        !            71: 
        !            72: void init_init_processing ()
        !            73: {
        !            74:   tree op_id;
        !            75:   tree fields[1];
        !            76: 
        !            77:   /* Define implicit `operator new' and `operator delete' functions.  */
        !            78:   BIN = default_conversion (TREE_VALUE (IDENTIFIER_GLOBAL_VALUE (ansi_opname[(int) NEW_EXPR])));
        !            79:   TREE_USED (TREE_OPERAND (BIN, 0)) = 0;
        !            80:   BID = default_conversion (TREE_VALUE (IDENTIFIER_GLOBAL_VALUE (ansi_opname[(int) DELETE_EXPR])));
        !            81:   TREE_USED (TREE_OPERAND (BID, 0)) = 0;
        !            82:   minus_one = build_int_2 (-1, -1);
        !            83: 
        !            84:   op_id = ansi_opname[(int) NEW_EXPR];
        !            85:   IDENTIFIER_GLOBAL_VALUE (op_id) = BIN;
        !            86:   op_id = ansi_opname[(int) DELETE_EXPR];
        !            87:   IDENTIFIER_GLOBAL_VALUE (op_id) = BID;
        !            88: 
        !            89:   /* Define the structure that holds header information for
        !            90:      arrays allocated via operator new.  */
        !            91:   BI_header_type = make_lang_type (RECORD_TYPE);
        !            92:   nc_nelts_field_id = get_identifier ("nelts");
        !            93:   fields[0] = build_lang_field_decl (FIELD_DECL, nc_nelts_field_id, sizetype);
        !            94:   finish_builtin_type (BI_header_type, "__new_cookie", fields,
        !            95:                       0, double_type_node);
        !            96:   BI_header_size = size_in_bytes (BI_header_type);
        !            97: }
        !            98: 
        !            99: /* Recursive subroutine of emit_base_init.  For main type T,
        !           100:    recursively initialize the vfields of the base type PARENT.
        !           101:    RECURSE is non-zero when this function is being called
        !           102:    recursively.  */
        !           103: 
        !           104: static void
        !           105: init_vfields (t, parent, recurse)
        !           106:      tree t, parent;
        !           107:      int recurse;
        !           108: {
        !           109:   tree vfields;
        !           110: 
        !           111:   /* Initialize all the virtual function table fields that
        !           112:      do not come from virtual base classes.  */
        !           113:   vfields = CLASSTYPE_VFIELDS (parent);
        !           114:   while (vfields)
        !           115:     {
        !           116:       tree basetype = VF_DERIVED_VALUE (vfields)
        !           117:        ? TYPE_MAIN_VARIANT (VF_DERIVED_VALUE (vfields))
        !           118:          : VF_BASETYPE_VALUE (vfields);
        !           119: 
        !           120:       /* If the vtable installed by the constructor was not
        !           121:         the right one, fix that here.  */
        !           122:       if (TREE_ADDRESSABLE (vfields)
        !           123:          && CLASSTYPE_NEEDS_VIRTUAL_REINIT (basetype)
        !           124:          && (recurse > 0
        !           125:              || TYPE_HAS_CONSTRUCTOR (basetype)
        !           126:              /* BASE_INIT_LIST has already initialized the immediate basetypes.  */
        !           127:              || get_base_distance (basetype, t, 0, (tree *) 0) > 1))
        !           128:        {
        !           129:          tree binfo = binfo_value (basetype, t);
        !           130:          if ((recurse != 0 && (binfo != binfo_value (basetype, parent)))
        !           131:              || (recurse == 0
        !           132:                  && BINFO_VTABLE (binfo) != TYPE_BINFO_VTABLE (basetype)))
        !           133:            {
        !           134:              tree ptr = convert_pointer_to (binfo, current_class_decl);
        !           135:              expand_expr_stmt (build_virtual_init (TYPE_BINFO (t), binfo, ptr));
        !           136:            }
        !           137:          init_vfields (t, basetype, recurse+1);
        !           138:        }
        !           139:       vfields = TREE_CHAIN (vfields);
        !           140:     }
        !           141: }
        !           142: 
        !           143: /* 348 - 351 */
        !           144: /* Subroutine of emit_base_init.  */
        !           145: static void
        !           146: perform_member_init (member, name, init, explicit)
        !           147:      tree member, name, init;
        !           148:      int explicit;
        !           149: {
        !           150:   tree decl;
        !           151:   tree type = TREE_TYPE (member);
        !           152: 
        !           153:   if (TYPE_NEEDS_CONSTRUCTING (type))
        !           154:     {
        !           155:       /* Since `init' is already a TREE_LIST on the current_member_init_list,
        !           156:         only build it into one if we aren't already a list.  */
        !           157:       if (init != NULL_TREE && TREE_CODE (init) != TREE_LIST)
        !           158:        init = build_tree_list (NULL_TREE, init);
        !           159: 
        !           160:       decl = build_component_ref (C_C_D, name, 0, explicit);
        !           161: 
        !           162:       if (explicit
        !           163:          && TREE_CODE (type) == ARRAY_TYPE
        !           164:          && init != NULL_TREE
        !           165:          && TREE_CHAIN (init) == NULL_TREE
        !           166:          && TREE_CODE (TREE_TYPE (TREE_VALUE (init))) == ARRAY_TYPE)
        !           167:        {
        !           168:          /* Initialization of one array from another.  */
        !           169:          expand_vec_init (TREE_OPERAND (decl, 1), decl,
        !           170:                           array_type_nelts (type), TREE_VALUE (init), 1);
        !           171:        }
        !           172:       else
        !           173:        expand_aggr_init (decl, init, 0);
        !           174:     }
        !           175:   else
        !           176:     {
        !           177:       if (init == NULL_TREE)
        !           178:        {
        !           179:          if (explicit)
        !           180:            {
        !           181:              error ("incomplete initializer for member `%s' of class `%s' which has no constructor",
        !           182:                     IDENTIFIER_POINTER (name),
        !           183:                     TYPE_NAME_STRING (current_class_type));
        !           184:              init = error_mark_node;
        !           185:            }
        !           186:          /* member traversal: note it leaves init NULL */
        !           187:          else if (TREE_CODE (TREE_TYPE (member)) == REFERENCE_TYPE)
        !           188:            pedwarn ("uninitialized reference member `%s'",
        !           189:                     IDENTIFIER_POINTER (name));
        !           190:        }
        !           191:       else if (TREE_CODE (init) == TREE_LIST)
        !           192:        {
        !           193:          /* There was an explicit member initialization.  Do some
        !           194:             work in that case.  */
        !           195:          if (TREE_CHAIN (init))
        !           196:            {
        !           197:              warning ("initializer list treated as compound expression");
        !           198:              init = build_compound_expr (init);
        !           199:            }
        !           200:          else
        !           201:            init = TREE_VALUE (init);
        !           202:        }
        !           203: 
        !           204:       /* We only build this with a null init if we got it from the
        !           205:         current_member_init_list.  */
        !           206:       if (init || explicit)
        !           207:        {
        !           208:          decl = build_component_ref (C_C_D, name, 0, explicit);
        !           209:          expand_expr_stmt (build_modify_expr (decl, INIT_EXPR, init));
        !           210:        }
        !           211:     }
        !           212: 
        !           213:   if (flag_handle_exceptions == 2 && TYPE_NEEDS_DESTRUCTOR (type))
        !           214:     {
        !           215:       cplus_expand_start_try (1);
        !           216:       push_exception_cleanup (build_unary_op (ADDR_EXPR, decl, 0));
        !           217:     }
        !           218: }
        !           219: 
        !           220: /* Subroutine of emit_member_init.  */
        !           221: static tree
        !           222: sort_member_init (t)
        !           223:      tree t;
        !           224: {
        !           225:   tree x, member, name, field, init;
        !           226:   tree init_list = NULL_TREE;
        !           227:   tree fields_to_unmark = NULL_TREE;
        !           228:   int found;
        !           229: 
        !           230:   for (member = TYPE_FIELDS (t); member ; member = TREE_CHAIN (member))
        !           231:     {
        !           232:       found = 0;
        !           233:       for (x = current_member_init_list ; x ; x = TREE_CHAIN (x))
        !           234:        {
        !           235:          /* If we cleared this out, then pay no attention to it.  */
        !           236:          if (TREE_PURPOSE (x) == NULL_TREE)
        !           237:            continue;
        !           238:          name = TREE_PURPOSE (x);
        !           239: 
        !           240: #if 0
        !           241:          field = (TREE_CODE (name) == COMPONENT_REF
        !           242:                   ? TREE_OPERAND (name, 1) : IDENTIFIER_CLASS_VALUE (name));
        !           243: #else
        !           244:          /* Let's find out when this happens.  */
        !           245:          my_friendly_assert (TREE_CODE (name) != COMPONENT_REF, 348);
        !           246:          field = IDENTIFIER_CLASS_VALUE (name);
        !           247: #endif
        !           248: 
        !           249:          /* If one member shadows another, get the outermost one.  */
        !           250:          if (TREE_CODE (field) == TREE_LIST)
        !           251:            field = TREE_VALUE (field);
        !           252: 
        !           253:          if (field == member)
        !           254:            {
        !           255:              /* See if we already found an initializer for this field.  */
        !           256:              if (found)
        !           257:                {
        !           258:                  if (DECL_NAME (field))
        !           259:                    cp_error ("multiple initializations given for member `%D'",
        !           260:                              field);
        !           261:                  continue;
        !           262:                }
        !           263: 
        !           264:              init_list = chainon (init_list,
        !           265:                                   build_tree_list (name, TREE_VALUE (x)));
        !           266:              /* Make sure we won't try to work on this init again.  */
        !           267:              TREE_PURPOSE (x) = NULL_TREE;
        !           268:              found = 1;
        !           269:              break;
        !           270:            }
        !           271:        }
        !           272: 
        !           273:       /* If we didn't find MEMBER in the list, create a dummy entry
        !           274:         so the two lists (INIT_LIST and the list of members) will be
        !           275:         symmetrical.  */
        !           276:       if (! found)
        !           277:        init_list = chainon (init_list, build_tree_list (NULL_TREE, NULL_TREE));
        !           278:     }
        !           279: 
        !           280:   for (x = current_member_init_list ; x ; x = TREE_CHAIN (x))
        !           281:     {
        !           282:       if (TREE_PURPOSE (x))
        !           283:        {
        !           284:          name = TREE_PURPOSE (x);
        !           285:          init = TREE_VALUE (x);
        !           286:          /* XXX: this may need the COMPONENT_REF operand 0 check if
        !           287:             it turns out we actually get them.  */
        !           288:          field = IDENTIFIER_CLASS_VALUE (name);
        !           289: 
        !           290:          /* If one member shadows another, get the outermost one.  */
        !           291:          if (TREE_CODE (field) == TREE_LIST)
        !           292:            {
        !           293:              field = TREE_VALUE (field);
        !           294:              if (decl_type_context (field) != current_class_type)
        !           295:                cp_error ("field `%D' not in immediate context", field);
        !           296:            }
        !           297: 
        !           298: #if 0
        !           299:          /* It turns out if you have an anonymous union in the
        !           300:             class, a member from it can end up not being on the
        !           301:             list of fields (rather, the type is), and therefore
        !           302:             won't be seen by the for loop above.  */
        !           303: 
        !           304:          /* The code in this for loop is derived from a general loop
        !           305:             which had this check in it.  Theoretically, we've hit
        !           306:             every initialization for the list of members in T, so
        !           307:             we shouldn't have anything but these left in this list.  */
        !           308:          my_friendly_assert (DECL_FIELD_CONTEXT (field) != t, 351);
        !           309: #endif
        !           310: 
        !           311:          if (TREE_HAS_CONSTRUCTOR (field))
        !           312:            {
        !           313:              if (DECL_NAME (field))
        !           314:                error ("multiple initializations given for member `%s'",
        !           315:                       IDENTIFIER_POINTER (DECL_NAME (field)));
        !           316:              continue;
        !           317:            }
        !           318: 
        !           319:          TREE_HAS_CONSTRUCTOR (field) = 1;
        !           320:          fields_to_unmark = tree_cons (NULL_TREE, field, fields_to_unmark);
        !           321: 
        !           322:          perform_member_init (field, name, init, 1);
        !           323:          TREE_PURPOSE (x) = NULL_TREE;
        !           324:        }
        !           325:     }
        !           326: 
        !           327:   /* Unmark fields which are initialized for the base class.  */
        !           328:   while (fields_to_unmark)
        !           329:     {
        !           330:       TREE_HAS_CONSTRUCTOR (TREE_VALUE (fields_to_unmark)) = 0;
        !           331:       /* XXX is this a memory leak? */
        !           332:       fields_to_unmark = TREE_CHAIN (fields_to_unmark);
        !           333:     }
        !           334: 
        !           335:   return init_list;
        !           336: }
        !           337: 
        !           338: /* Perform whatever initialization have yet to be done on the
        !           339:    base class of the class variable.  These actions are in
        !           340:    the global variable CURRENT_BASE_INIT_LIST.  Such an
        !           341:    action could be NULL_TREE, meaning that the user has explicitly
        !           342:    called the base class constructor with no arguments.
        !           343: 
        !           344:    If there is a need for a call to a constructor, we
        !           345:    must surround that call with a pushlevel/poplevel pair,
        !           346:    since we are technically at the PARM level of scope.
        !           347: 
        !           348:    Argument IMMEDIATELY, if zero, forces a new sequence to be generated
        !           349:    to contain these new insns, so it can be emitted later.  This sequence
        !           350:    is saved in the global variable BASE_INIT_INSNS.  Otherwise, the insns
        !           351:    are emitted into the current sequence.
        !           352: 
        !           353:    Note that emit_base_init does *not* initialize virtual
        !           354:    base classes.  That is done specially, elsewhere.  */
        !           355:    
        !           356: void
        !           357: emit_base_init (t, immediately)
        !           358:      tree t;
        !           359:      int immediately;
        !           360: {
        !           361:   extern tree in_charge_identifier;
        !           362: 
        !           363:   tree member, decl, vbases;
        !           364:   tree init_list, member_init;
        !           365:   int pass, start;
        !           366:   tree t_binfo = TYPE_BINFO (t);
        !           367:   tree binfos = BINFO_BASETYPES (t_binfo);
        !           368:   int i, n_baseclasses = binfos ? TREE_VEC_LENGTH (binfos) : 0;
        !           369:   tree fields_to_unmark = NULL_TREE;
        !           370:   int have_init_list = 0, from_init_list;
        !           371: 
        !           372:   if (! immediately)
        !           373:     {
        !           374:       do_pending_stack_adjust ();
        !           375:       start_sequence ();
        !           376:     }
        !           377: 
        !           378:   if (write_symbols == NO_DEBUG)
        !           379:     /* As a matter of principle, `start_sequence' should do this.  */
        !           380:     emit_note (0, -1);
        !           381:   else
        !           382:     /* Always emit a line number note so we can step into constructors.  */
        !           383:     emit_line_note_force (DECL_SOURCE_FILE (current_function_decl),
        !           384:                          DECL_SOURCE_LINE (current_function_decl));
        !           385: 
        !           386:   /* In this case, we always need IN_CHARGE_NODE, because we have
        !           387:      to know whether to deallocate or not before exiting.  */
        !           388:   if (flag_handle_exceptions == 2
        !           389:       && lookup_name (in_charge_identifier, 0) == NULL_TREE)
        !           390:     {
        !           391:       tree in_charge_node = pushdecl (build_decl (VAR_DECL, in_charge_identifier,
        !           392:                                                  integer_type_node));
        !           393:       store_init_value (in_charge_node, build (EQ_EXPR, integer_type_node,
        !           394:                                               current_class_decl,
        !           395:                                               integer_zero_node));
        !           396:       expand_decl (in_charge_node);
        !           397:       expand_decl_init (in_charge_node);
        !           398:     }
        !           399: 
        !           400:   start = ! TYPE_USES_VIRTUAL_BASECLASSES (t);
        !           401:   for (pass = start; pass < 2; pass++)
        !           402:     {
        !           403:       tree vbase_init_list = NULL_TREE;
        !           404: 
        !           405:       for (init_list = current_base_init_list; init_list;
        !           406:           init_list = TREE_CHAIN (init_list))
        !           407:        {
        !           408:          tree basename = TREE_PURPOSE (init_list);
        !           409:          tree binfo;
        !           410:          tree init = TREE_VALUE (init_list);
        !           411: 
        !           412:          if (basename == NULL_TREE)
        !           413:            {
        !           414:              /* Initializer for single base class.  Must not
        !           415:                 use multiple inheritance or this is ambiguous.  */
        !           416:              switch (n_baseclasses)
        !           417:                {
        !           418:                case 0:
        !           419:                  error ("type `%s' does not have a base class to initialize",
        !           420:                         IDENTIFIER_POINTER (current_class_name));
        !           421:                  return;
        !           422:                case 1:
        !           423:                  break;
        !           424:                default:
        !           425:                  error ("unnamed initializer ambiguous for type `%s' which uses multiple inheritance", IDENTIFIER_POINTER (current_class_name));
        !           426:                  return;
        !           427:                }
        !           428:              binfo = TREE_VEC_ELT (binfos, 0);
        !           429:            }
        !           430:          else if (is_aggr_typedef (basename, 1))
        !           431:            {
        !           432:              binfo = binfo_or_else (IDENTIFIER_TYPE_VALUE (basename), t);
        !           433:              if (binfo == NULL_TREE)
        !           434:                continue;
        !           435: 
        !           436:              /* Virtual base classes are special cases.  Their initializers
        !           437:                 are recorded with this constructor, and they are used when
        !           438:                 this constructor is the top-level constructor called.  */
        !           439:              if (! TREE_VIA_VIRTUAL (binfo))
        !           440:                {
        !           441:                  /* Otherwise, if it is not an immediate base class, complain.  */
        !           442:                  for (i = n_baseclasses-1; i >= 0; i--)
        !           443:                    if (BINFO_TYPE (binfo) == BINFO_TYPE (TREE_VEC_ELT (binfos, i)))
        !           444:                      break;
        !           445:                  if (i < 0)
        !           446:                    {
        !           447:                      error ("type `%s' is not an immediate base class of type `%s'",
        !           448:                             IDENTIFIER_POINTER (basename),
        !           449:                             IDENTIFIER_POINTER (current_class_name));
        !           450:                      continue;
        !           451:                    }
        !           452:                }
        !           453:            }
        !           454:          else
        !           455:            continue;
        !           456: 
        !           457:          /* The base initialization list goes up to the first
        !           458:             base class which can actually use it.  */
        !           459: 
        !           460:          if (pass == start)
        !           461:            {
        !           462:              char *msgp = (! TYPE_HAS_CONSTRUCTOR (BINFO_TYPE (binfo)))
        !           463:                ? "cannot pass initialization up to class `%s'" : 0;
        !           464: 
        !           465:              while (! TYPE_HAS_CONSTRUCTOR (BINFO_TYPE (binfo))
        !           466:                     && BINFO_BASETYPES (binfo) != NULL_TREE
        !           467:                     && TREE_VEC_LENGTH (BINFO_BASETYPES (binfo)) == 1)
        !           468:                {
        !           469:                  /* ?? This should be fixed in RENO by forcing
        !           470:                     default constructors to exist.  */
        !           471:                  SET_BINFO_BASEINIT_MARKED (binfo);
        !           472:                  binfo = BINFO_BASETYPE (binfo, 0);
        !           473:                }
        !           474: 
        !           475:              /* We used to give an error if this wasn't true, saying that
        !           476:                 there's no constructor for the initialization of basename.
        !           477:                 This turned out to be incorrect---it should use the
        !           478:                 default constructor, since a user could try to initialize
        !           479:                 the class in a derived class's base initializer list.  */
        !           480:              if (TYPE_HAS_CONSTRUCTOR (BINFO_TYPE (binfo)))
        !           481:                {
        !           482:                  if (msgp)
        !           483:                    {
        !           484:                      if (pedantic)
        !           485:                        error_with_aggr_type (binfo, msgp);
        !           486:                      else
        !           487:                        msgp = NULL;
        !           488:                    }
        !           489:                }
        !           490: 
        !           491:              if (BINFO_BASEINIT_MARKED (binfo))
        !           492:                {
        !           493:                  msgp = "class `%s' initializer already specified";
        !           494:                  error (msgp, IDENTIFIER_POINTER (basename));
        !           495:                }
        !           496: 
        !           497:              if (msgp)
        !           498:                continue;
        !           499: 
        !           500:              SET_BINFO_BASEINIT_MARKED (binfo);
        !           501:              if (TREE_VIA_VIRTUAL (binfo))
        !           502:                {
        !           503:                  vbase_init_list = tree_cons (init, BINFO_TYPE (binfo),
        !           504:                                               vbase_init_list);
        !           505:                  continue;
        !           506:                }
        !           507:              if (pass == 0)
        !           508:                continue;
        !           509:            }
        !           510:          else if (TREE_VIA_VIRTUAL (binfo))
        !           511:            continue;
        !           512: 
        !           513:          member = convert_pointer_to (binfo, current_class_decl);
        !           514:          expand_aggr_init_1 (t_binfo, 0,
        !           515:                              build_indirect_ref (member, NULL_PTR), init,
        !           516:                              BINFO_OFFSET_ZEROP (binfo),
        !           517:                              LOOKUP_PROTECTED_OK|LOOKUP_COMPLAIN);
        !           518:          if (flag_handle_exceptions == 2 && TYPE_NEEDS_DESTRUCTOR (BINFO_TYPE (binfo)))
        !           519:            {
        !           520:              cplus_expand_start_try (1);
        !           521:              push_exception_cleanup (member);
        !           522:            }
        !           523:        }
        !           524: 
        !           525:       if (pass == 0)
        !           526:        {
        !           527:          tree first_arg = TREE_CHAIN (DECL_ARGUMENTS (current_function_decl));
        !           528:          tree vbases;
        !           529: 
        !           530:          if (DECL_NAME (current_function_decl) == NULL_TREE
        !           531:              && TREE_CHAIN (first_arg) != NULL_TREE)
        !           532:            {
        !           533:              /* If there are virtual baseclasses without initialization
        !           534:                 specified, and this is a default X(X&) constructor,
        !           535:                 build the initialization list so that each virtual baseclass
        !           536:                 of the new object is initialized from the virtual baseclass
        !           537:                 of the incoming arg.  */
        !           538:              tree init_arg = build_unary_op (ADDR_EXPR, TREE_CHAIN (first_arg), 0);
        !           539:              for (vbases = CLASSTYPE_VBASECLASSES (t);
        !           540:                   vbases; vbases = TREE_CHAIN (vbases))
        !           541:                {
        !           542:                  if (BINFO_BASEINIT_MARKED (vbases) == 0)
        !           543:                    {
        !           544:                      member = convert_pointer_to (vbases, init_arg);
        !           545:                      if (member == init_arg)
        !           546:                        member = TREE_CHAIN (first_arg);
        !           547:                      else
        !           548:                        TREE_TYPE (member) = build_reference_type (BINFO_TYPE (vbases));
        !           549:                      vbase_init_list = tree_cons (convert_from_reference (member),
        !           550:                                                   vbases, vbase_init_list);
        !           551:                      SET_BINFO_BASEINIT_MARKED (vbases);
        !           552:                    }
        !           553:                }
        !           554:            }
        !           555:          expand_start_cond (first_arg, 0);
        !           556:          expand_aggr_vbase_init (t_binfo, C_C_D, current_class_decl,
        !           557:                                  vbase_init_list);
        !           558:          expand_end_cond ();
        !           559:        }
        !           560:     }
        !           561:   current_base_init_list = NULL_TREE;
        !           562: 
        !           563:   /* Now, perform default initialization of all base classes which
        !           564:      have not yet been initialized, and unmark baseclasses which
        !           565:      have been initialized.  */
        !           566:   for (i = 0; i < n_baseclasses; i++)
        !           567:     {
        !           568:       tree base = current_class_decl;
        !           569:       tree base_binfo = TREE_VEC_ELT (binfos, i);
        !           570: 
        !           571:       if (TYPE_NEEDS_CONSTRUCTING (BINFO_TYPE (base_binfo)))
        !           572:        {
        !           573:          if (! TREE_VIA_VIRTUAL (base_binfo)
        !           574:              && ! BINFO_BASEINIT_MARKED (base_binfo))
        !           575:            {
        !           576:              tree ref;
        !           577: 
        !           578:              if (BINFO_OFFSET_ZEROP (base_binfo))
        !           579:                base = build1 (NOP_EXPR,
        !           580:                               TYPE_POINTER_TO (BINFO_TYPE (base_binfo)),
        !           581:                               current_class_decl);
        !           582:              else
        !           583:                base = build (PLUS_EXPR,
        !           584:                              TYPE_POINTER_TO (BINFO_TYPE (base_binfo)),
        !           585:                              current_class_decl, BINFO_OFFSET (base_binfo));
        !           586: 
        !           587:              ref = build_indirect_ref (base, NULL_PTR);
        !           588:              expand_aggr_init_1 (t_binfo, 0, ref, NULL_TREE,
        !           589:                                  BINFO_OFFSET_ZEROP (base_binfo),
        !           590:                                  LOOKUP_PROTECTED_OK|LOOKUP_COMPLAIN);
        !           591:              if (flag_handle_exceptions == 2
        !           592:                  && TYPE_NEEDS_DESTRUCTOR (BINFO_TYPE (base_binfo)))
        !           593:                {
        !           594:                  cplus_expand_start_try (1);
        !           595:                  push_exception_cleanup (base);
        !           596:                }
        !           597:            }
        !           598:        }
        !           599:       CLEAR_BINFO_BASEINIT_MARKED (base_binfo);
        !           600: 
        !           601:       if (! TYPE_USES_VIRTUAL_BASECLASSES (t))
        !           602:        {
        !           603:          while (! TYPE_HAS_CONSTRUCTOR (BINFO_TYPE (base_binfo))
        !           604:                 && BINFO_BASETYPES (base_binfo) != NULL_TREE
        !           605:                 && TREE_VEC_LENGTH (BINFO_BASETYPES (base_binfo)) == 1)
        !           606:            {
        !           607:              /* ?? This should be fixed in RENO by forcing
        !           608:                 default constructors to exist.  It is needed for symmetry
        !           609:                 with code above.  */
        !           610:              base_binfo = BINFO_BASETYPE (base_binfo, 0);
        !           611:              CLEAR_BINFO_BASEINIT_MARKED (base_binfo);
        !           612:            }
        !           613:        }
        !           614:     }
        !           615: 
        !           616:   /* Initialize all the virtual function table fields that
        !           617:      dp come from virtual base classes. */
        !           618:   if (TYPE_USES_VIRTUAL_BASECLASSES (t))
        !           619:     expand_expr_stmt (build_vbase_vtables_init (t_binfo, t_binfo,
        !           620:                                                C_C_D, current_class_decl, 0));
        !           621:   for (vbases = CLASSTYPE_VBASECLASSES (t); vbases; vbases = TREE_CHAIN (vbases))
        !           622:     CLEAR_BINFO_BASEINIT_MARKED (vbases);
        !           623: 
        !           624:   /* Initialize all the virtual function table fields that
        !           625:      do not come from virtual base classes.  */
        !           626:   init_vfields (t, t, 0);
        !           627: 
        !           628:   if (CLASSTYPE_NEEDS_VIRTUAL_REINIT (t))
        !           629:     expand_expr_stmt (build_virtual_init (TYPE_BINFO (t), t,
        !           630:                                          current_class_decl));
        !           631: 
        !           632:   if (current_member_init_list)
        !           633:     {
        !           634:       init_list = sort_member_init (t);
        !           635:       have_init_list = 1;
        !           636:     }
        !           637: 
        !           638:   for (member = TYPE_FIELDS (t); member; member = TREE_CHAIN (member))
        !           639:     {
        !           640:       tree init, name;
        !           641:       from_init_list = 0;
        !           642: 
        !           643:       /* See if we had a user-specified member initialization.  */
        !           644:       if (have_init_list)
        !           645:        {
        !           646:          if (TREE_PURPOSE (init_list))
        !           647:            {
        !           648:              name = TREE_PURPOSE (init_list);
        !           649:              init = TREE_VALUE (init_list);
        !           650:              from_init_list = 1;
        !           651: 
        !           652:              if (TREE_STATIC (member))
        !           653:                {
        !           654:                  error_with_aggr_type (DECL_FIELD_CONTEXT (member),
        !           655:                                        "field `%s::%s' is static; only point of initialization is its declaration",
        !           656:                                        IDENTIFIER_POINTER (TREE_PURPOSE (init_list)));
        !           657:                  continue;
        !           658:                }
        !           659: 
        !           660:              /* Also see if it's ever a COMPONENT_REF here.  If it is, we
        !           661:                 need to do `expand_assignment (name, init, 0, 0);' and
        !           662:                 a continue.  */
        !           663:              my_friendly_assert (TREE_CODE (name) != COMPONENT_REF, 349);
        !           664:            }
        !           665: 
        !           666:          init_list = TREE_CHAIN (init_list);
        !           667:        }
        !           668: 
        !           669:       if (! from_init_list)
        !           670:        {
        !           671:          /* member could be, for example, a CONST_DECL for an enumerated
        !           672:             tag; we don't want to try to initialize that, since it already
        !           673:             has a value.  */
        !           674:          if (TREE_CODE (member) != FIELD_DECL)
        !           675:            continue;
        !           676: 
        !           677:          name = DECL_NAME (member);
        !           678:          init = DECL_INITIAL (member);
        !           679:        }
        !           680: 
        !           681:       perform_member_init (member, name, init, from_init_list);
        !           682:     }
        !           683: 
        !           684:   current_member_init_list = NULL_TREE;
        !           685: 
        !           686:   /* It is possible for the initializers to need cleanups.
        !           687:      Expand those cleanups now that all the initialization
        !           688:      has been done.  */
        !           689:   expand_cleanups_to (NULL_TREE);
        !           690: 
        !           691:   if (! immediately)
        !           692:     {
        !           693:       extern rtx base_init_insns;
        !           694: 
        !           695:       do_pending_stack_adjust ();
        !           696:       my_friendly_assert (base_init_insns == 0, 207);
        !           697:       base_init_insns = get_insns ();
        !           698:       end_sequence ();
        !           699:     }
        !           700: 
        !           701:   /* All the implicit try blocks we built up will be zapped
        !           702:      when we come to a real binding contour boundary.  */
        !           703: }
        !           704: 
        !           705: /* Check that all fields are properly initialized after
        !           706:    an assignment to `this'.  */
        !           707: void
        !           708: check_base_init (t)
        !           709:      tree t;
        !           710: {
        !           711:   tree member;
        !           712:   for (member = TYPE_FIELDS (t); member; member = TREE_CHAIN (member))
        !           713:     if (DECL_NAME (member) && TREE_USED (member))
        !           714:       error ("field `%s' used before initialized (after assignment to `this')",
        !           715:             IDENTIFIER_POINTER (DECL_NAME (member)));
        !           716: }
        !           717: 
        !           718: /* This code sets up the virtual function tables appropriate for
        !           719:    the pointer DECL.  It is a one-ply initialization.
        !           720: 
        !           721:    BINFO is the exact type that DECL is supposed to be.  In
        !           722:    multiple inheritance, this might mean "C's A" if C : A, B.  */
        !           723: tree
        !           724: build_virtual_init (main_binfo, binfo, decl)
        !           725:      tree main_binfo, binfo;
        !           726:      tree decl;
        !           727: {
        !           728:   tree type;
        !           729:   tree vtbl, vtbl_ptr;
        !           730:   tree vtype, vtype_binfo;
        !           731: 
        !           732:   if (TREE_CODE (binfo) == TREE_VEC)
        !           733:     type = BINFO_TYPE (binfo);
        !           734:   else if (TREE_CODE (binfo) == RECORD_TYPE)
        !           735:     {
        !           736:       type = binfo;
        !           737:       binfo = TYPE_BINFO (type);
        !           738:     }
        !           739:   else
        !           740:     my_friendly_abort (46);
        !           741: 
        !           742:   vtype = DECL_CONTEXT (CLASSTYPE_VFIELD (type));
        !           743:   vtype_binfo = get_binfo (vtype, TREE_TYPE (TREE_TYPE (decl)), 0);
        !           744: #if 0
        !           745:   /* This code suggests that it's time to rewrite how we handle
        !           746:      replicated baseclasses in G++.  */
        !           747:   if (get_base_distance (vtype, TREE_TYPE (TREE_TYPE (decl)),
        !           748:                         0, (tree *) 0) == -2)
        !           749:     {
        !           750:       tree binfos = TYPE_BINFO_BASETYPES (TREE_TYPE (TREE_TYPE (decl)));
        !           751:       int i, n_baselinks = binfos ? TREE_VEC_LENGTH (binfos) : 0;
        !           752:       tree result = NULL_TREE;
        !           753: 
        !           754:       for (i = n_baselinks-1; i >= 0; i--)
        !           755:        {
        !           756:          tree base_binfo = TREE_VEC_ELT (binfos, i);
        !           757:          tree this_decl;
        !           758: 
        !           759:          if (get_base_distance (vtype, BINFO_TYPE (base_binfo), 0, 0) == -1)
        !           760:            continue;
        !           761: 
        !           762:          if (TREE_VIA_VIRTUAL (base_binfo))
        !           763:            this_decl = build_vbase_pointer (build_indirect_ref (decl, NULL_PTR), BINFO_TYPE (base_binfo));
        !           764:          else if (BINFO_OFFSET_ZEROP (base_binfo))
        !           765:            this_decl = build1 (NOP_EXPR, TYPE_POINTER_TO (BINFO_TYPE (base_binfo)),
        !           766:                                decl);
        !           767:          else
        !           768:            this_decl = build (PLUS_EXPR, TYPE_POINTER_TO (BINFO_TYPE (base_binfo)),
        !           769:                               decl, BINFO_OFFSET (base_binfo));
        !           770:          result = tree_cons (NULL_TREE, build_virtual_init (main_binfo, base_binfo, this_decl), result);
        !           771:        }
        !           772:       return build_compound_expr (result);
        !           773:     }
        !           774: #endif
        !           775: 
        !           776:     {
        !           777: #if 1
        !           778: #if 1
        !           779:       vtbl = BINFO_VTABLE (binfo_value (DECL_FIELD_CONTEXT (CLASSTYPE_VFIELD (type)), binfo));
        !           780: #else
        !           781:       /* The below does not work when we have to step through the
        !           782:         vfield, on our way down to the most base class for the
        !           783:         vfield. */
        !           784:       vtbl = BINFO_VTABLE (binfo_value (DECL_FIELD_CONTEXT (CLASSTYPE_VFIELD (type)),
        !           785:                                        BINFO_TYPE (main_binfo)));
        !           786: #endif
        !           787: #else
        !           788:       my_friendly_assert (BINFO_TYPE (main_binfo) == BINFO_TYPE (binfo), 208);
        !           789:       vtbl = BINFO_VTABLE (main_binfo);
        !           790: #endif /* 1 */
        !           791:       assemble_external (vtbl);
        !           792:       TREE_USED (vtbl) = 1;
        !           793:       vtbl = build1 (ADDR_EXPR, TYPE_POINTER_TO (TREE_TYPE (vtbl)), vtbl);
        !           794:     }
        !           795:   decl = convert_pointer_to_real (vtype_binfo, decl);
        !           796:   vtbl_ptr = build_vfield_ref (build_indirect_ref (decl, NULL_PTR), vtype);
        !           797:   if (vtbl_ptr == error_mark_node)
        !           798:     return error_mark_node;
        !           799: 
        !           800:   /* Have to convert VTBL since array sizes may be different.  */
        !           801:   return build_modify_expr (vtbl_ptr, NOP_EXPR,
        !           802:                            convert (TREE_TYPE (vtbl_ptr), vtbl));
        !           803: }
        !           804: 
        !           805: /* Subroutine of `expand_aggr_vbase_init'.
        !           806:    BINFO is the binfo of the type that is being initialized.
        !           807:    INIT_LIST is the list of initializers for the virtual baseclass.  */
        !           808: static void
        !           809: expand_aggr_vbase_init_1 (binfo, exp, addr, init_list)
        !           810:      tree binfo, exp, addr, init_list;
        !           811: {
        !           812:   tree init = value_member (BINFO_TYPE (binfo), init_list);
        !           813:   tree ref = build_indirect_ref (addr, NULL_PTR);
        !           814:   if (init)
        !           815:     init = TREE_PURPOSE (init);
        !           816:   /* Call constructors, but don't set up vtables.  */
        !           817:   expand_aggr_init_1 (binfo, exp, ref, init, 0,
        !           818:                      LOOKUP_PROTECTED_OK|LOOKUP_COMPLAIN|LOOKUP_SPECULATIVELY);
        !           819:   CLEAR_BINFO_VBASE_INIT_MARKED (binfo);
        !           820: }
        !           821: 
        !           822: /* Initialize this object's virtual base class pointers.  This must be
        !           823:    done only at the top-level of the object being constructed.
        !           824: 
        !           825:    INIT_LIST is list of initialization for constructor to perform.  */
        !           826: static void
        !           827: expand_aggr_vbase_init (binfo, exp, addr, init_list)
        !           828:      tree binfo;
        !           829:      tree exp;
        !           830:      tree addr;
        !           831:      tree init_list;
        !           832: {
        !           833:   tree type = BINFO_TYPE (binfo);
        !           834: 
        !           835:   if (TYPE_USES_VIRTUAL_BASECLASSES (type))
        !           836:     {
        !           837:       tree result = init_vbase_pointers (type, addr);
        !           838:       tree vbases;
        !           839: 
        !           840:       if (result)
        !           841:        expand_expr_stmt (build_compound_expr (result));
        !           842: 
        !           843:       /* Mark everything as having an initializer
        !           844:         (either explicit or default).  */
        !           845:       for (vbases = CLASSTYPE_VBASECLASSES (type);
        !           846:           vbases; vbases = TREE_CHAIN (vbases))
        !           847:        SET_BINFO_VBASE_INIT_MARKED (vbases);
        !           848: 
        !           849:       /* First, initialize baseclasses which could be baseclasses
        !           850:         for other virtual baseclasses.  */
        !           851:       for (vbases = CLASSTYPE_VBASECLASSES (type);
        !           852:           vbases; vbases = TREE_CHAIN (vbases))
        !           853:        /* Don't initialize twice.  */
        !           854:        if (BINFO_VBASE_INIT_MARKED (vbases))
        !           855:          {
        !           856:            tree tmp = result;
        !           857: 
        !           858:            while (BINFO_TYPE (vbases) != BINFO_TYPE (TREE_PURPOSE (tmp)))
        !           859:              tmp = TREE_CHAIN (tmp);
        !           860:            expand_aggr_vbase_init_1 (vbases, exp,
        !           861:                                      TREE_OPERAND (TREE_VALUE (tmp), 0),
        !           862:                                      init_list);
        !           863:          }
        !           864: 
        !           865:       /* Now initialize the baseclasses which don't have virtual baseclasses.  */
        !           866:       for (; result; result = TREE_CHAIN (result))
        !           867:        /* Don't initialize twice.  */
        !           868:        if (BINFO_VBASE_INIT_MARKED (TREE_PURPOSE (result)))
        !           869:          {
        !           870:            my_friendly_abort (47);
        !           871:            expand_aggr_vbase_init_1 (TREE_PURPOSE (result), exp,
        !           872:                                      TREE_OPERAND (TREE_VALUE (result), 0),
        !           873:                                      init_list);
        !           874:          }
        !           875:     }
        !           876: }
        !           877: 
        !           878: /* Subroutine to perform parser actions for member initialization.
        !           879:    S_ID is the scoped identifier.
        !           880:    NAME is the name of the member.
        !           881:    INIT is the initializer, or `void_type_node' if none.  */
        !           882: void
        !           883: do_member_init (s_id, name, init)
        !           884:      tree s_id, name, init;
        !           885: {
        !           886:   tree binfo, base;
        !           887: 
        !           888:   if (current_class_type == NULL_TREE
        !           889:       || ! is_aggr_typedef (s_id, 1))
        !           890:     return;
        !           891:   binfo = get_binfo (IDENTIFIER_TYPE_VALUE (s_id),
        !           892:                          current_class_type, 1);
        !           893:   if (binfo == error_mark_node)
        !           894:     return;
        !           895:   if (binfo == 0)
        !           896:     {
        !           897:       error_not_base_type (IDENTIFIER_TYPE_VALUE (s_id), current_class_type);
        !           898:       return;
        !           899:     }
        !           900: 
        !           901:   base = convert_pointer_to (binfo, current_class_decl);
        !           902:   expand_member_init (build_indirect_ref (base, NULL_PTR), name, init);
        !           903: }
        !           904: 
        !           905: /* Function to give error message if member initialization specification
        !           906:    is erroneous.  FIELD is the member we decided to initialize.
        !           907:    TYPE is the type for which the initialization is being performed.
        !           908:    FIELD must be a member of TYPE, or the base type from which FIELD
        !           909:    comes must not need a constructor.
        !           910:    
        !           911:    MEMBER_NAME is the name of the member.  */
        !           912: 
        !           913: static int
        !           914: member_init_ok_or_else (field, type, member_name)
        !           915:      tree field;
        !           916:      tree type;
        !           917:      char *member_name;
        !           918: {
        !           919:   if (field == error_mark_node)
        !           920:     return 0;
        !           921:   if (field == NULL_TREE)
        !           922:     {
        !           923:       cp_error ("class `%T' does not have any field named `%s'", type,
        !           924:                  member_name);
        !           925:       return 0;
        !           926:     }
        !           927:   if (DECL_CONTEXT (field) != type
        !           928:       && TYPE_NEEDS_CONSTRUCTOR (DECL_CONTEXT (field)))
        !           929:     {
        !           930:       error ("member `%s' comes from base class needing constructor",
        !           931:             member_name);
        !           932:       return 0;
        !           933:     }
        !           934:   return 1;
        !           935: }
        !           936: 
        !           937: /* If NAME is a viable field name for the aggregate DECL,
        !           938:    and PARMS is a viable parameter list, then expand an _EXPR
        !           939:    which describes this initialization.
        !           940: 
        !           941:    Note that we do not need to chase through the class's base classes
        !           942:    to look for NAME, because if it's in that list, it will be handled
        !           943:    by the constructor for that base class.
        !           944: 
        !           945:    We do not yet have a fixed-point finder to instantiate types
        !           946:    being fed to overloaded constructors.  If there is a unique
        !           947:    constructor, then argument types can be got from that one.
        !           948: 
        !           949:    If INIT is non-NULL, then it the initialization should
        !           950:    be placed in `current_base_init_list', where it will be processed
        !           951:    by `emit_base_init'.  */
        !           952: void
        !           953: expand_member_init (exp, name, init)
        !           954:      tree exp, name, init;
        !           955: {
        !           956:   extern tree ptr_type_node;   /* should be in tree.h */
        !           957: 
        !           958:   tree basetype = NULL_TREE, field;
        !           959:   tree parm;
        !           960:   tree rval, type;
        !           961:   tree actual_name;
        !           962: 
        !           963:   if (exp == NULL_TREE)
        !           964:     return;                    /* complain about this later */
        !           965: 
        !           966:   type = TYPE_MAIN_VARIANT (TREE_TYPE (exp));
        !           967: 
        !           968:   if (name == NULL_TREE && IS_AGGR_TYPE (type))
        !           969:     switch (CLASSTYPE_N_BASECLASSES (type))
        !           970:       {
        !           971:       case 0:
        !           972:        error ("base class initializer specified, but no base class to initialize");
        !           973:        return;
        !           974:       case 1:
        !           975:        basetype = TYPE_BINFO_BASETYPE (type, 0);
        !           976:        break;
        !           977:       default:
        !           978:        error ("initializer for unnamed base class ambiguous");
        !           979:        cp_error ("(type `%T' uses multiple inheritance)", type);
        !           980:        return;
        !           981:       }
        !           982: 
        !           983:   if (init)
        !           984:     {
        !           985:       /* The grammar should not allow fields which have names
        !           986:         that are TYPENAMEs.  Therefore, if the field has
        !           987:         a non-NULL TREE_TYPE, we may assume that this is an
        !           988:         attempt to initialize a base class member of the current
        !           989:         type.  Otherwise, it is an attempt to initialize a
        !           990:         member field.  */
        !           991: 
        !           992:       if (init == void_type_node)
        !           993:        init = NULL_TREE;
        !           994: 
        !           995:       if (name == NULL_TREE || IDENTIFIER_HAS_TYPE_VALUE (name))
        !           996:        {
        !           997:          tree base_init;
        !           998: 
        !           999:          if (name == NULL_TREE)
        !          1000:            {
        !          1001:              if (basetype)
        !          1002:                name = TYPE_IDENTIFIER (basetype);
        !          1003:              else
        !          1004:                {
        !          1005:                  error ("no base class to initialize");
        !          1006:                  return;
        !          1007:                }
        !          1008:            }
        !          1009:          else
        !          1010:            {
        !          1011:              basetype = IDENTIFIER_TYPE_VALUE (name);
        !          1012:              if (basetype != type
        !          1013:                  && ! binfo_member (basetype, TYPE_BINFO (type))
        !          1014:                  && ! binfo_member (basetype, CLASSTYPE_VBASECLASSES (type)))
        !          1015:                {
        !          1016:                  if (IDENTIFIER_CLASS_VALUE (name))
        !          1017:                    goto try_member;
        !          1018:                  if (TYPE_USES_VIRTUAL_BASECLASSES (type))
        !          1019:                    error ("type `%s' is not an immediate or virtual basetype for `%s'",
        !          1020:                           IDENTIFIER_POINTER (name),
        !          1021:                           TYPE_NAME_STRING (type));
        !          1022:                  else
        !          1023:                    error ("type `%s' is not an immediate basetype for `%s'",
        !          1024:                           IDENTIFIER_POINTER (name),
        !          1025:                           TYPE_NAME_STRING (type));
        !          1026:                  return;
        !          1027:                }
        !          1028:            }
        !          1029: 
        !          1030:          if (purpose_member (name, current_base_init_list))
        !          1031:            {
        !          1032:              error ("base class `%s' already initialized",
        !          1033:                     IDENTIFIER_POINTER (name));
        !          1034:              return;
        !          1035:            }
        !          1036: 
        !          1037:          base_init = build_tree_list (name, init);
        !          1038:          TREE_TYPE (base_init) = basetype;
        !          1039:          current_base_init_list = chainon (current_base_init_list, base_init);
        !          1040:        }
        !          1041:       else
        !          1042:        {
        !          1043:          tree member_init;
        !          1044: 
        !          1045:        try_member:
        !          1046:          field = lookup_field (type, name, 1, 0);
        !          1047: 
        !          1048:          if (! member_init_ok_or_else (field, type, IDENTIFIER_POINTER (name)))
        !          1049:            return;
        !          1050: 
        !          1051:          if (purpose_member (name, current_member_init_list))
        !          1052:            {
        !          1053:              error ("field `%s' already initialized", IDENTIFIER_POINTER (name));
        !          1054:              return;
        !          1055:            }
        !          1056: 
        !          1057:          member_init = build_tree_list (name, init);
        !          1058:          TREE_TYPE (member_init) = TREE_TYPE (field);
        !          1059:          current_member_init_list = chainon (current_member_init_list, member_init);
        !          1060:        }
        !          1061:       return;
        !          1062:     }
        !          1063:   else if (name == NULL_TREE)
        !          1064:     {
        !          1065:       compiler_error ("expand_member_init: name == NULL_TREE");
        !          1066:       return;
        !          1067:     }
        !          1068: 
        !          1069:   basetype = type;
        !          1070:   field = lookup_field (basetype, name, 0, 0);
        !          1071: 
        !          1072:   if (! member_init_ok_or_else (field, basetype, IDENTIFIER_POINTER (name)))
        !          1073:     return;
        !          1074: 
        !          1075:   /* now see if there is a constructor for this type
        !          1076:      which will take these args. */
        !          1077: 
        !          1078:   if (TYPE_HAS_CONSTRUCTOR (TREE_TYPE (field)))
        !          1079:     {
        !          1080:       tree parmtypes, fndecl;
        !          1081: 
        !          1082:       if (TREE_CODE (exp) == VAR_DECL || TREE_CODE (exp) == PARM_DECL)
        !          1083:        {
        !          1084:          /* just know that we've seen something for this node */
        !          1085:          DECL_INITIAL (exp) = error_mark_node;
        !          1086:          TREE_USED (exp) = 1;
        !          1087:        }
        !          1088:       type = TYPE_MAIN_VARIANT (TREE_TYPE (field));
        !          1089:       actual_name = TYPE_IDENTIFIER (type);
        !          1090:       parm = build_component_ref (exp, name, 0, 0);
        !          1091: 
        !          1092:       /* Now get to the constructor.  */
        !          1093:       fndecl = TREE_VEC_ELT (CLASSTYPE_METHOD_VEC (type), 0);
        !          1094:       /* Get past destructor, if any.  */
        !          1095:       if (TYPE_HAS_DESTRUCTOR (type))
        !          1096:        fndecl = DECL_CHAIN (fndecl);
        !          1097: 
        !          1098:       if (fndecl)
        !          1099:        my_friendly_assert (TREE_CODE (fndecl) == FUNCTION_DECL, 209);
        !          1100: 
        !          1101:       /* If the field is unique, we can use the parameter
        !          1102:         types to guide possible type instantiation.  */
        !          1103:       if (DECL_CHAIN (fndecl) == NULL_TREE)
        !          1104:        {
        !          1105:          /* There was a confusion here between
        !          1106:             FIELD and FNDECL.  The following code
        !          1107:             should be correct, but abort is here
        !          1108:             to make sure.  */
        !          1109:          my_friendly_abort (48);
        !          1110:          parmtypes = FUNCTION_ARG_CHAIN (fndecl);
        !          1111:        }
        !          1112:       else
        !          1113:        {
        !          1114:          parmtypes = NULL_TREE;
        !          1115:          fndecl = NULL_TREE;
        !          1116:        }
        !          1117: 
        !          1118:       init = convert_arguments (parm, parmtypes, NULL_TREE, fndecl, LOOKUP_NORMAL);
        !          1119:       if (init == NULL_TREE || TREE_TYPE (init) != error_mark_node)
        !          1120:        rval = build_method_call (NULL_TREE, actual_name, init, NULL_TREE, LOOKUP_NORMAL);
        !          1121:       else
        !          1122:        return;
        !          1123: 
        !          1124:       if (rval != error_mark_node)
        !          1125:        {
        !          1126:          /* Now, fill in the first parm with our guy */
        !          1127:          TREE_VALUE (TREE_OPERAND (rval, 1))
        !          1128:            = build_unary_op (ADDR_EXPR, parm, 0);
        !          1129:          TREE_TYPE (rval) = ptr_type_node;
        !          1130:          TREE_SIDE_EFFECTS (rval) = 1;
        !          1131:        }
        !          1132:     }
        !          1133:   else if (TYPE_NEEDS_CONSTRUCTING (TREE_TYPE (field)))
        !          1134:     {
        !          1135:       parm = build_component_ref (exp, name, 0, 0);
        !          1136:       expand_aggr_init (parm, NULL_TREE, 0);
        !          1137:       rval = error_mark_node;
        !          1138:     }
        !          1139: 
        !          1140:   /* Now initialize the member.  It does not have to
        !          1141:      be of aggregate type to receive initialization.  */
        !          1142:   if (rval != error_mark_node)
        !          1143:     expand_expr_stmt (rval);
        !          1144: }
        !          1145: 
        !          1146: /* This is like `expand_member_init', only it stores one aggregate
        !          1147:    value into another.
        !          1148: 
        !          1149:    INIT comes in two flavors: it is either a value which
        !          1150:    is to be stored in EXP, or it is a parameter list
        !          1151:    to go to a constructor, which will operate on EXP.
        !          1152:    If `init' is a CONSTRUCTOR, then we emit a warning message,
        !          1153:    explaining that such initializations are illegal.
        !          1154: 
        !          1155:    ALIAS_THIS is nonzero iff we are initializing something which is
        !          1156:    essentially an alias for C_C_D.  In this case, the base constructor
        !          1157:    may move it on us, and we must keep track of such deviations.
        !          1158: 
        !          1159:    If INIT resolves to a CALL_EXPR which happens to return
        !          1160:    something of the type we are looking for, then we know
        !          1161:    that we can safely use that call to perform the
        !          1162:    initialization.
        !          1163: 
        !          1164:    The virtual function table pointer cannot be set up here, because
        !          1165:    we do not really know its type.
        !          1166: 
        !          1167:    Virtual baseclass pointers are also set up here.
        !          1168: 
        !          1169:    This never calls operator=().
        !          1170: 
        !          1171:    When initializing, nothing is CONST.
        !          1172: 
        !          1173:    A default copy constructor may have to be used to perform the
        !          1174:    initialization.
        !          1175: 
        !          1176:    A constructor or a conversion operator may have to be used to
        !          1177:    perform the initialization, but not both, as it would be ambiguous.
        !          1178:    */
        !          1179: 
        !          1180: void
        !          1181: expand_aggr_init (exp, init, alias_this)
        !          1182:      tree exp, init;
        !          1183:      int alias_this;
        !          1184: {
        !          1185:   tree type = TREE_TYPE (exp);
        !          1186:   int was_const = TREE_READONLY (exp);
        !          1187: 
        !          1188:   if (init == error_mark_node)
        !          1189:     return;
        !          1190: 
        !          1191:   TREE_READONLY (exp) = 0;
        !          1192: 
        !          1193:   if (TREE_CODE (type) == ARRAY_TYPE)
        !          1194:     {
        !          1195:       /* Must arrange to initialize each element of EXP
        !          1196:         from elements of INIT.  */
        !          1197:       int was_const_elts = TYPE_READONLY (TREE_TYPE (type));
        !          1198:       tree itype = init ? TREE_TYPE (init) : NULL_TREE;
        !          1199:       if (was_const_elts)
        !          1200:        {
        !          1201:          tree atype = build_cplus_array_type (TYPE_MAIN_VARIANT (TREE_TYPE (type)),
        !          1202:                                               TYPE_DOMAIN (type));
        !          1203:          if (init && (TREE_TYPE (exp) == TREE_TYPE (init)))
        !          1204:            TREE_TYPE (init) = atype;
        !          1205:          TREE_TYPE (exp) = atype;
        !          1206:        }
        !          1207:       if (init && TREE_TYPE (init) == NULL_TREE)
        !          1208:        {
        !          1209:          /* Handle bad initializers like:
        !          1210:             class COMPLEX {
        !          1211:             public:
        !          1212:               double re, im;
        !          1213:               COMPLEX(double r = 0.0, double i = 0.0) {re = r; im = i;};
        !          1214:               ~COMPLEX() {};
        !          1215:             };
        !          1216: 
        !          1217:             int main(int argc, char **argv) {
        !          1218:               COMPLEX zees(1.0, 0.0)[10];
        !          1219:             }
        !          1220:          */
        !          1221:          error ("bad array initializer");
        !          1222:          return;
        !          1223:        }
        !          1224:       expand_vec_init (exp, exp, array_type_nelts (type), init,
        !          1225:                       init && comptypes (TREE_TYPE (init), TREE_TYPE (exp), 1));
        !          1226:       TREE_READONLY (exp) = was_const;
        !          1227:       TREE_TYPE (exp) = type;
        !          1228:       if (init) TREE_TYPE (init) = itype;
        !          1229:       return;
        !          1230:     }
        !          1231: 
        !          1232:   if (TREE_CODE (exp) == VAR_DECL || TREE_CODE (exp) == PARM_DECL)
        !          1233:     /* just know that we've seen something for this node */
        !          1234:     TREE_USED (exp) = 1;
        !          1235: 
        !          1236:   /* If initializing from a GNU C CONSTRUCTOR, consider the elts in the
        !          1237:      constructor as parameters to an implicit GNU C++ constructor.  */
        !          1238:   if (init && TREE_CODE (init) == CONSTRUCTOR
        !          1239:       && TYPE_HAS_CONSTRUCTOR (type)
        !          1240:       && TREE_TYPE (init) == type)
        !          1241:     init = CONSTRUCTOR_ELTS (init);
        !          1242:   expand_aggr_init_1 (TYPE_BINFO (type), exp, exp,
        !          1243:                      init, alias_this, LOOKUP_NORMAL);
        !          1244:   TREE_READONLY (exp) = was_const;
        !          1245: }
        !          1246: 
        !          1247: static void
        !          1248: expand_default_init (binfo, true_exp, exp, type, init, alias_this, flags)
        !          1249:      tree binfo;
        !          1250:      tree true_exp, exp;
        !          1251:      tree type;
        !          1252:      tree init;
        !          1253:      int alias_this;
        !          1254:      int flags;
        !          1255: {
        !          1256:   /* It fails because there may not be a constructor which takes
        !          1257:      its own type as the first (or only parameter), but which does
        !          1258:      take other types via a conversion.  So, if the thing initializing
        !          1259:      the expression is a unit element of type X, first try X(X&),
        !          1260:      followed by initialization by X.  If neither of these work
        !          1261:      out, then look hard.  */
        !          1262:   tree rval;
        !          1263:   tree parms;
        !          1264:   int xxref_init_possible;
        !          1265: 
        !          1266:   if (init == NULL_TREE || TREE_CODE (init) == TREE_LIST)
        !          1267:     {
        !          1268:       parms = init;
        !          1269:       if (parms) init = TREE_VALUE (parms);
        !          1270:     }
        !          1271:   else if (TREE_CODE (init) == INDIRECT_REF && TREE_HAS_CONSTRUCTOR (init))
        !          1272:     {
        !          1273:       rval = convert_for_initialization (exp, type, init, 0, 0, 0, 0);
        !          1274:       expand_expr_stmt (rval);
        !          1275:       return;
        !          1276:     }
        !          1277:   else
        !          1278:     parms = build_tree_list (NULL_TREE, init);
        !          1279: 
        !          1280:   if (TYPE_HAS_INIT_REF (type)
        !          1281:       || init == NULL_TREE
        !          1282:       || TREE_CHAIN (parms) != NULL_TREE)
        !          1283:     xxref_init_possible = 0;
        !          1284:   else
        !          1285:     {
        !          1286:       xxref_init_possible = LOOKUP_SPECULATIVELY;
        !          1287:       flags &= ~LOOKUP_COMPLAIN;
        !          1288:     }
        !          1289: 
        !          1290:   if (TYPE_USES_VIRTUAL_BASECLASSES (type))
        !          1291:     {
        !          1292:       if (true_exp == exp)
        !          1293:        parms = tree_cons (NULL_TREE, integer_one_node, parms);
        !          1294:       else
        !          1295:        parms = tree_cons (NULL_TREE, integer_zero_node, parms);
        !          1296:       flags |= LOOKUP_HAS_IN_CHARGE;
        !          1297:     }
        !          1298: 
        !          1299:   rval = build_method_call (exp, constructor_name_full (type),
        !          1300:                            parms, binfo, flags|xxref_init_possible);
        !          1301:   if (rval == NULL_TREE && xxref_init_possible)
        !          1302:     {
        !          1303:       /* It is an error to implement a default copy constructor if
        !          1304:         (see ARM 12.8 for details) ... one case being if another
        !          1305:         copy constructor already exists. */
        !          1306:       tree init_type = TREE_TYPE (init);
        !          1307:       if (TREE_CODE (init_type) == REFERENCE_TYPE)
        !          1308:        init_type = TREE_TYPE (init_type);
        !          1309:       if (TYPE_MAIN_VARIANT (init_type) == TYPE_MAIN_VARIANT (type)
        !          1310:          || (IS_AGGR_TYPE (init_type)
        !          1311:              && UNIQUELY_DERIVED_FROM_P (type, init_type)))
        !          1312:        {
        !          1313:          if (type == BINFO_TYPE (binfo)
        !          1314:              && TYPE_USES_VIRTUAL_BASECLASSES (type))
        !          1315:            {
        !          1316:              tree addr = build_unary_op (ADDR_EXPR, exp, 0);
        !          1317:              expand_aggr_vbase_init (binfo, exp, addr, NULL_TREE);
        !          1318: 
        !          1319:              expand_expr_stmt (build_vbase_vtables_init (binfo, binfo,
        !          1320:                                                          exp, addr, 1));
        !          1321:            }
        !          1322:          expand_expr_stmt (build_modify_expr (exp, INIT_EXPR, init));
        !          1323:          return;
        !          1324:        }
        !          1325:       else
        !          1326:        rval = build_method_call (exp, constructor_name_full (type), parms,
        !          1327:                                  binfo, flags);
        !          1328:     }
        !          1329: 
        !          1330:   /* Private, protected, or otherwise unavailable.  */
        !          1331:   if (rval == error_mark_node && (flags&LOOKUP_COMPLAIN))
        !          1332:     cp_error ("in base initialization for class `%T'", binfo);
        !          1333:   /* A valid initialization using constructor.  */
        !          1334:   else if (rval != error_mark_node && rval != NULL_TREE)
        !          1335:     {
        !          1336:       /* p. 222: if the base class assigns to `this', then that
        !          1337:         value is used in the derived class.  */
        !          1338:       if ((flag_this_is_variable & 1) && alias_this)
        !          1339:        {
        !          1340:          TREE_TYPE (rval) = TREE_TYPE (current_class_decl);
        !          1341:          expand_assignment (current_class_decl, rval, 0, 0);
        !          1342:        }
        !          1343:       else
        !          1344:        expand_expr_stmt (rval);
        !          1345:     }
        !          1346:   else if (parms && TREE_CHAIN (parms) == NULL_TREE)
        !          1347:     {
        !          1348:       /* If we are initializing one aggregate value
        !          1349:         from another, and though there are constructors,
        !          1350:         and none accept the initializer, just do a bitwise
        !          1351:         copy.
        !          1352: 
        !          1353:         The above sounds wrong, ``If a class has any copy
        !          1354:         constructor defined, the default copy constructor will
        !          1355:         not be generated.'' 12.8 Copying Class Objects  (mrs)
        !          1356: 
        !          1357:         @@ This should reject initializer which a constructor
        !          1358:         @@ rejected on visibility gounds, but there is
        !          1359:         @@ no way right now to recognize that case with
        !          1360:         @@ just `error_mark_node'.  */
        !          1361:       tree itype;
        !          1362:       init = TREE_VALUE (parms);
        !          1363:       itype = TREE_TYPE (init);
        !          1364:       if (TREE_CODE (itype) == REFERENCE_TYPE)
        !          1365:        {
        !          1366:          init = convert_from_reference (init);
        !          1367:          itype = TREE_TYPE (init);
        !          1368:        }
        !          1369:       itype = TYPE_MAIN_VARIANT (itype);
        !          1370: 
        !          1371:       /* This is currently how the default X(X&) constructor
        !          1372:         is implemented.  */
        !          1373:       if (comptypes (TYPE_MAIN_VARIANT (type), itype, 0))
        !          1374:        {
        !          1375: #if 0
        !          1376:          warning ("bitwise copy in initialization of type `%s'",
        !          1377:                   TYPE_NAME_STRING (type));
        !          1378: #endif
        !          1379:          rval = build (INIT_EXPR, type, exp, init);
        !          1380:          expand_expr_stmt (rval);
        !          1381:        }
        !          1382:       else
        !          1383:        {
        !          1384:          cp_error ("in base initialization for class `%T',", binfo);
        !          1385:          cp_error ("invalid initializer to constructor for type `%T'", type);
        !          1386:          return;
        !          1387:        }
        !          1388:     }
        !          1389:   else
        !          1390:     {
        !          1391:       if (init == NULL_TREE)
        !          1392:        my_friendly_assert (parms == NULL_TREE, 210);
        !          1393:       if (parms == NULL_TREE && TREE_VIA_VIRTUAL (binfo))
        !          1394:        cp_error ("virtual baseclass `%T' does not have default initializer", binfo);
        !          1395:       else
        !          1396:        {
        !          1397:          cp_error ("in base initialization for class `%T',", binfo);
        !          1398:          /* This will make an error message for us.  */
        !          1399:          build_method_call (exp, constructor_name_full (type), parms, binfo,
        !          1400:                             (TYPE_USES_VIRTUAL_BASECLASSES (type)
        !          1401:                              ? LOOKUP_NORMAL|LOOKUP_HAS_IN_CHARGE
        !          1402:                              : LOOKUP_NORMAL));
        !          1403:        }
        !          1404:       return;
        !          1405:     }
        !          1406:   /* Constructor has been called, but vtables may be for TYPE
        !          1407:      rather than for FOR_TYPE.  */
        !          1408: }
        !          1409: 
        !          1410: /* This function is responsible for initializing EXP with INIT
        !          1411:    (if any).
        !          1412: 
        !          1413:    BINFO is the binfo of the type for who we are performing the
        !          1414:    initialization.  For example, if W is a virtual base class of A and B,
        !          1415:    and C : A, B.
        !          1416:    If we are initializing B, then W must contain B's W vtable, whereas
        !          1417:    were we initializing C, W must contain C's W vtable.
        !          1418: 
        !          1419:    TRUE_EXP is nonzero if it is the true expression being initialized.
        !          1420:    In this case, it may be EXP, or may just contain EXP.  The reason we
        !          1421:    need this is because if EXP is a base element of TRUE_EXP, we
        !          1422:    don't necessarily know by looking at EXP where its virtual
        !          1423:    baseclass fields should really be pointing.  But we do know
        !          1424:    from TRUE_EXP.  In constructors, we don't know anything about
        !          1425:    the value being initialized.
        !          1426: 
        !          1427:    ALIAS_THIS serves the same purpose it serves for expand_aggr_init.
        !          1428: 
        !          1429:    FLAGS is just passes to `build_method_call'.  See that function for
        !          1430:    its description.  */
        !          1431: 
        !          1432: static void
        !          1433: expand_aggr_init_1 (binfo, true_exp, exp, init, alias_this, flags)
        !          1434:      tree binfo;
        !          1435:      tree true_exp, exp;
        !          1436:      tree init;
        !          1437:      int alias_this;
        !          1438:      int flags;
        !          1439: {
        !          1440:   tree type = TREE_TYPE (exp);
        !          1441:   tree init_type = NULL_TREE;
        !          1442:   tree rval;
        !          1443: 
        !          1444:   my_friendly_assert (init != error_mark_node && type != error_mark_node, 211);
        !          1445: 
        !          1446:   /* Use a function returning the desired type to initialize EXP for us.
        !          1447:      If the function is a constructor, and its first argument is
        !          1448:      NULL_TREE, know that it was meant for us--just slide exp on
        !          1449:      in and expand the constructor.  Constructors now come
        !          1450:      as TARGET_EXPRs.  */
        !          1451:   if (init)
        !          1452:     {
        !          1453:       tree init_list = NULL_TREE;
        !          1454: 
        !          1455:       if (TREE_CODE (init) == TREE_LIST)
        !          1456:        {
        !          1457:          init_list = init;
        !          1458:          if (TREE_CHAIN (init) == NULL_TREE)
        !          1459:            init = TREE_VALUE (init);
        !          1460:        }
        !          1461: 
        !          1462:       init_type = TREE_TYPE (init);
        !          1463: 
        !          1464:       if (TREE_CODE (init) != TREE_LIST)
        !          1465:        {
        !          1466:          if (TREE_CODE (init_type) == ERROR_MARK)
        !          1467:            return;
        !          1468: 
        !          1469: #if 0
        !          1470:          /* These lines are found troublesome 5/11/89.  */
        !          1471:          if (TREE_CODE (init_type) == REFERENCE_TYPE)
        !          1472:            init_type = TREE_TYPE (init_type);
        !          1473: #endif
        !          1474: 
        !          1475:          /* This happens when we use C++'s functional cast notation.
        !          1476:             If the types match, then just use the TARGET_EXPR
        !          1477:             directly.  Otherwise, we need to create the initializer
        !          1478:             separately from the object being initialized.  */
        !          1479:          if (TREE_CODE (init) == TARGET_EXPR)
        !          1480:            {
        !          1481:              if (init_type == type)
        !          1482:                {
        !          1483:                  if (TREE_CODE (exp) == VAR_DECL
        !          1484:                      || TREE_CODE (exp) == RESULT_DECL)
        !          1485:                    /* Unify the initialization targets.  */
        !          1486:                    DECL_RTL (TREE_OPERAND (init, 0)) = DECL_RTL (exp);
        !          1487:                  else
        !          1488:                    DECL_RTL (TREE_OPERAND (init, 0)) = expand_expr (exp, NULL_RTX, 0, 0);
        !          1489: 
        !          1490:                  expand_expr_stmt (init);
        !          1491:                  return;
        !          1492:                }
        !          1493:              else
        !          1494:                {
        !          1495:                  init = TREE_OPERAND (init, 1);
        !          1496:                  init = build (CALL_EXPR, init_type,
        !          1497:                                TREE_OPERAND (init, 0), TREE_OPERAND (init, 1), 0);
        !          1498:                  TREE_SIDE_EFFECTS (init) = 1;
        !          1499: #if 0
        !          1500:                  TREE_RAISES (init) = ??
        !          1501: #endif
        !          1502:                    if (init_list)
        !          1503:                      TREE_VALUE (init_list) = init;
        !          1504:                }
        !          1505:            }
        !          1506: 
        !          1507:          if (init_type == type && TREE_CODE (init) == CALL_EXPR
        !          1508: #if 0
        !          1509:              /* It is legal to directly initialize from a CALL_EXPR
        !          1510:                 without going through X(X&), apparently.  */
        !          1511:              && ! TYPE_GETS_INIT_REF (type)
        !          1512: #endif
        !          1513:              )
        !          1514:            {
        !          1515:              /* A CALL_EXPR is a legitimate form of initialization, so
        !          1516:                 we should not print this warning message.  */
        !          1517: #if 0
        !          1518:              /* Should have gone away due to 5/11/89 change.  */
        !          1519:              if (TREE_CODE (TREE_TYPE (init)) == REFERENCE_TYPE)
        !          1520:                init = convert_from_reference (init);
        !          1521: #endif
        !          1522:              expand_assignment (exp, init, 0, 0);
        !          1523:              if (exp == DECL_RESULT (current_function_decl))
        !          1524:                {
        !          1525:                  /* Failing this assertion means that the return value
        !          1526:                     from receives multiple initializations.  */
        !          1527:                  my_friendly_assert (DECL_INITIAL (exp) == NULL_TREE
        !          1528:                                      || DECL_INITIAL (exp) == error_mark_node,
        !          1529:                                      212);
        !          1530:                  DECL_INITIAL (exp) = init;
        !          1531:                }
        !          1532:              return;
        !          1533:            }
        !          1534:          else if (init_type == type
        !          1535:                   && TREE_CODE (init) == COND_EXPR)
        !          1536:            {
        !          1537:              /* Push value to be initialized into the cond, where possible.
        !          1538:                 Avoid spurious warning messages when initializing the
        !          1539:                 result of this function.  */
        !          1540:              TREE_OPERAND (init, 1)
        !          1541:                = build_modify_expr (exp, INIT_EXPR, TREE_OPERAND (init, 1));
        !          1542:              if (exp == DECL_RESULT (current_function_decl))
        !          1543:                DECL_INITIAL (exp) = NULL_TREE;
        !          1544:              TREE_OPERAND (init, 2)
        !          1545:                = build_modify_expr (exp, INIT_EXPR, TREE_OPERAND (init, 2));
        !          1546:              if (exp == DECL_RESULT (current_function_decl))
        !          1547:                DECL_INITIAL (exp) = init;
        !          1548:              TREE_SIDE_EFFECTS (init) = 1;
        !          1549:              expand_expr (init, const0_rtx, VOIDmode, 0);
        !          1550:              free_temp_slots ();
        !          1551:              return;
        !          1552:            }
        !          1553:        }
        !          1554: 
        !          1555:       /* We did not know what we were initializing before.  Now we do.  */
        !          1556:       if (TREE_CODE (init) == TARGET_EXPR)
        !          1557:        {
        !          1558:          tree tmp = TREE_OPERAND (TREE_OPERAND (init, 1), 1);
        !          1559: 
        !          1560:          if (TREE_CODE (TREE_VALUE (tmp)) == NOP_EXPR
        !          1561:              && TREE_OPERAND (TREE_VALUE (tmp), 0) == integer_zero_node)
        !          1562:            {
        !          1563:              /* In order for this to work for RESULT_DECLs, if their
        !          1564:                 type has a constructor, then they must be BLKmode
        !          1565:                 so that they will be meaningfully addressable.  */
        !          1566:              tree arg = build_unary_op (ADDR_EXPR, exp, 0);
        !          1567:              init = TREE_OPERAND (init, 1);
        !          1568:              init = build (CALL_EXPR, build_pointer_type (TREE_TYPE (init)),
        !          1569:                            TREE_OPERAND (init, 0), TREE_OPERAND (init, 1), 0);
        !          1570:              TREE_SIDE_EFFECTS (init) = 1;
        !          1571: #if 0
        !          1572:              TREE_RAISES (init) = ??
        !          1573: #endif
        !          1574:              TREE_VALUE (TREE_OPERAND (init, 1))
        !          1575:                = convert_pointer_to (TREE_TYPE (TREE_TYPE (TREE_VALUE (tmp))), arg);
        !          1576: 
        !          1577:              if (alias_this)
        !          1578:                {
        !          1579:                  expand_assignment (current_function_decl, init, 0, 0);
        !          1580:                  return;
        !          1581:                }
        !          1582:              if (exp == DECL_RESULT (current_function_decl))
        !          1583:                {
        !          1584:                  if (DECL_INITIAL (DECL_RESULT (current_function_decl)))
        !          1585:                    fatal ("return value from function receives multiple initializations");
        !          1586:                  DECL_INITIAL (exp) = init;
        !          1587:                }
        !          1588:              expand_expr_stmt (init);
        !          1589:              return;
        !          1590:            }
        !          1591:        }
        !          1592: 
        !          1593:       /* Handle this case: when calling a constructor: xyzzy foo(bar);
        !          1594:         which really means:  xyzzy foo = bar; Ugh!
        !          1595: 
        !          1596:         We can also be called with an initializer for an object
        !          1597:         which has virtual functions, but no constructors.  In that
        !          1598:         case, we perform the assignment first, then initialize
        !          1599:         the virtual function table pointer fields.  */
        !          1600: 
        !          1601:       if (! TYPE_NEEDS_CONSTRUCTING (type))
        !          1602:        {
        !          1603:          if (init_list && TREE_CHAIN (init_list))
        !          1604:            {
        !          1605:              warning ("initializer list being treated as compound expression");
        !          1606:              init = convert (TREE_TYPE (exp), build_compound_expr (init_list));
        !          1607:              if (init == error_mark_node)
        !          1608:                return;
        !          1609:            }
        !          1610:          if (TREE_CODE (exp) == VAR_DECL
        !          1611:              && TREE_CODE (init) == CONSTRUCTOR
        !          1612:              && TREE_HAS_CONSTRUCTOR (init)
        !          1613:              && flag_pic == 0)
        !          1614:            store_init_value (exp, init);
        !          1615:          else
        !          1616:            expand_assignment (exp, init, 0, 0);
        !          1617: 
        !          1618:          if (TYPE_VIRTUAL_P (type))
        !          1619:            expand_recursive_init (binfo, true_exp, exp, init, CLASSTYPE_BASE_INIT_LIST (type), alias_this);
        !          1620:          return;
        !          1621:        }
        !          1622: 
        !          1623:       /* See whether we can go through a type conversion operator.
        !          1624:         This wins over going through a non-existent constructor.  If
        !          1625:         there is a constructor, it is ambiguous.  */
        !          1626:       if (TREE_CODE (init) != TREE_LIST)
        !          1627:        {
        !          1628:          tree ttype = TREE_CODE (init_type) == REFERENCE_TYPE
        !          1629:            ? TREE_TYPE (init_type) : init_type;
        !          1630: 
        !          1631:          if (ttype != type && IS_AGGR_TYPE (ttype))
        !          1632:            {
        !          1633:              tree rval = build_type_conversion (CONVERT_EXPR, type, init, 0);
        !          1634: 
        !          1635:              if (rval)
        !          1636:                {
        !          1637:                  /* See if there is a constructor for``type'' that takes a
        !          1638:                     ``ttype''-typed object. */
        !          1639:                  tree parms = build_tree_list (NULL_TREE, init);
        !          1640:                  tree as_cons = NULL_TREE;
        !          1641:                  if (TYPE_HAS_CONSTRUCTOR (type))
        !          1642:                    as_cons = build_method_call (exp, constructor_name_full (type),
        !          1643:                                                 parms, binfo,
        !          1644:                                                 LOOKUP_SPECULATIVELY|LOOKUP_NO_CONVERSION);
        !          1645:                  if (as_cons != NULL_TREE && as_cons != error_mark_node)
        !          1646:                    {
        !          1647:                      tree name = TYPE_NAME (type);
        !          1648:                      if (TREE_CODE (name) == TYPE_DECL)
        !          1649:                        name = DECL_NAME (name);
        !          1650:                      /* ANSI C++ June 5 1992 WP 12.3.2.6.1 */
        !          1651:                      error ("ambiguity between conversion to `%s' and constructor",
        !          1652:                             IDENTIFIER_POINTER (name));
        !          1653:                    }
        !          1654:                  else
        !          1655:                    expand_assignment (exp, rval, 0, 0);
        !          1656:                  return;
        !          1657:                }
        !          1658:            }
        !          1659:        }
        !          1660:     }
        !          1661: 
        !          1662:   /* Handle default copy constructors here, does not matter if there is
        !          1663:      a constructor or not.  */
        !          1664:   if (type == init_type && IS_AGGR_TYPE (type)
        !          1665:       && init && TREE_CODE (init) != TREE_LIST)
        !          1666:     expand_default_init (binfo, true_exp, exp, type, init, alias_this, flags);
        !          1667:   /* Not sure why this is here... */
        !          1668:   else if (TYPE_HAS_CONSTRUCTOR (type))
        !          1669:     expand_default_init (binfo, true_exp, exp, type, init, alias_this, flags);
        !          1670:   else if (TREE_CODE (type) == ARRAY_TYPE)
        !          1671:     {
        !          1672:       if (TYPE_NEEDS_CONSTRUCTING (TREE_TYPE (type)))
        !          1673:        expand_vec_init (exp, exp, array_type_nelts (type), init, 0);
        !          1674:       else if (TYPE_VIRTUAL_P (TREE_TYPE (type)))
        !          1675:        sorry ("arrays of objects with virtual functions but no constructors");
        !          1676:     }
        !          1677:   else
        !          1678:     expand_recursive_init (binfo, true_exp, exp, init,
        !          1679:                           CLASSTYPE_BASE_INIT_LIST (type), alias_this);
        !          1680: }
        !          1681: 
        !          1682: /* A pointer which holds the initializer.  First call to
        !          1683:    expand_aggr_init gets this value pointed to, and sets it to init_null.  */
        !          1684: static tree *init_ptr, init_null;
        !          1685: 
        !          1686: /* Subroutine of expand_recursive_init:
        !          1687: 
        !          1688:    ADDR is the address of the expression being initialized.
        !          1689:    INIT_LIST is the cons-list of initializations to be performed.
        !          1690:    ALIAS_THIS is its same, lovable self.  */
        !          1691: static void
        !          1692: expand_recursive_init_1 (binfo, true_exp, addr, init_list, alias_this)
        !          1693:      tree binfo, true_exp, addr;
        !          1694:      tree init_list;
        !          1695:      int alias_this;
        !          1696: {
        !          1697:   while (init_list)
        !          1698:     {
        !          1699:       if (TREE_PURPOSE (init_list))
        !          1700:        {
        !          1701:          if (TREE_CODE (TREE_PURPOSE (init_list)) == FIELD_DECL)
        !          1702:            {
        !          1703:              tree member = TREE_PURPOSE (init_list);
        !          1704:              tree subexp = build_indirect_ref (convert_pointer_to (TREE_VALUE (init_list), addr), NULL_PTR);
        !          1705:              tree member_base = build (COMPONENT_REF, TREE_TYPE (member), subexp, member);
        !          1706:              if (IS_AGGR_TYPE (TREE_TYPE (member)))
        !          1707:                expand_aggr_init (member_base, DECL_INITIAL (member), 0);
        !          1708:              else if (TREE_CODE (TREE_TYPE (member)) == ARRAY_TYPE
        !          1709:                       && TYPE_NEEDS_CONSTRUCTING (TREE_TYPE (member)))
        !          1710:                {
        !          1711:                  member_base = save_expr (default_conversion (member_base));
        !          1712:                  expand_vec_init (member, member_base,
        !          1713:                                   array_type_nelts (TREE_TYPE (member)),
        !          1714:                                   DECL_INITIAL (member), 0);
        !          1715:                }
        !          1716:              else
        !          1717:                expand_expr_stmt (build_modify_expr (member_base, INIT_EXPR, DECL_INITIAL (member)));
        !          1718:            }
        !          1719:          else if (TREE_CODE (TREE_PURPOSE (init_list)) == TREE_LIST)
        !          1720:            {
        !          1721:              expand_recursive_init_1 (binfo, true_exp, addr, TREE_PURPOSE (init_list), alias_this);
        !          1722:              expand_recursive_init_1 (binfo, true_exp, addr, TREE_VALUE (init_list), alias_this);
        !          1723:            }
        !          1724:          else if (TREE_CODE (TREE_PURPOSE (init_list)) == ERROR_MARK)
        !          1725:            {
        !          1726:              /* Only initialize the virtual function tables if we
        !          1727:                 are initializing the ultimate users of those vtables.  */
        !          1728:              if (TREE_VALUE (init_list))
        !          1729:                {
        !          1730:                  /* We have to ensure that the second argment to
        !          1731:                     build_virtual_init is in binfo's hierarchy.  */
        !          1732:                  expand_expr_stmt (build_virtual_init (binfo,
        !          1733:                                                        get_binfo (TREE_VALUE (init_list), binfo, 0),
        !          1734:                                                        addr));
        !          1735:                  if (TREE_VALUE (init_list) == binfo
        !          1736:                      && TYPE_USES_VIRTUAL_BASECLASSES (BINFO_TYPE (binfo)))
        !          1737:                    expand_expr_stmt (build_vbase_vtables_init (binfo, binfo, true_exp, addr, 1));
        !          1738:                }
        !          1739:            }
        !          1740:          else
        !          1741:            my_friendly_abort (49);
        !          1742:        }
        !          1743:       else if (TREE_VALUE (init_list)
        !          1744:               && TREE_CODE (TREE_VALUE (init_list)) == TREE_VEC)
        !          1745:        {
        !          1746:          tree subexp = build_indirect_ref (convert_pointer_to (TREE_VALUE (init_list), addr), NULL_PTR);
        !          1747:          expand_aggr_init_1 (binfo, true_exp, subexp, *init_ptr,
        !          1748:                              alias_this && BINFO_OFFSET_ZEROP (TREE_VALUE (init_list)),
        !          1749:                              LOOKUP_PROTECTED_OK|LOOKUP_COMPLAIN);
        !          1750: 
        !          1751:          /* INIT_PTR is used up.  */
        !          1752:          init_ptr = &init_null;
        !          1753:        }
        !          1754:       else
        !          1755:        my_friendly_abort (50);
        !          1756:       init_list = TREE_CHAIN (init_list);
        !          1757:     }
        !          1758: }
        !          1759: 
        !          1760: /* Initialize EXP with INIT.  Type EXP does not have a constructor,
        !          1761:    but it has a baseclass with a constructor or a virtual function
        !          1762:    table which needs initializing.
        !          1763: 
        !          1764:    INIT_LIST is a cons-list describing what parts of EXP actually
        !          1765:    need to be initialized.  INIT is given to the *unique*, first
        !          1766:    constructor within INIT_LIST.  If there are multiple first
        !          1767:    constructors, such as with multiple inheritance, INIT must
        !          1768:    be zero or an ambiguity error is reported.
        !          1769: 
        !          1770:    ALIAS_THIS is passed from `expand_aggr_init'.  See comments
        !          1771:    there.  */
        !          1772: 
        !          1773: static void
        !          1774: expand_recursive_init (binfo, true_exp, exp, init, init_list, alias_this)
        !          1775:      tree binfo, true_exp, exp, init;
        !          1776:      tree init_list;
        !          1777:      int alias_this;
        !          1778: {
        !          1779:   tree *old_init_ptr = init_ptr;
        !          1780:   tree addr = build_unary_op (ADDR_EXPR, exp, 0);
        !          1781:   init_ptr = &init;
        !          1782: 
        !          1783:   if (true_exp == exp && TYPE_USES_VIRTUAL_BASECLASSES (BINFO_TYPE (binfo)))
        !          1784:     {
        !          1785:       expand_aggr_vbase_init (binfo, exp, addr, init_list);
        !          1786:       expand_expr_stmt (build_vbase_vtables_init (binfo, binfo, true_exp, addr, 1));
        !          1787:     }
        !          1788:   expand_recursive_init_1 (binfo, true_exp, addr, init_list, alias_this);
        !          1789: 
        !          1790:   if (*init_ptr)
        !          1791:     {
        !          1792:       tree type = TREE_TYPE (exp);
        !          1793: 
        !          1794:       if (TREE_CODE (type) == REFERENCE_TYPE)
        !          1795:        type = TREE_TYPE (type);
        !          1796:       if (IS_AGGR_TYPE (type))
        !          1797:        cp_error ("unexpected argument to constructor `%T'", type);
        !          1798:       else
        !          1799:        error ("unexpected argument to constructor");
        !          1800:     }
        !          1801:   init_ptr = old_init_ptr;
        !          1802: }
        !          1803: 
        !          1804: /* Report an error if NAME is not the name of a user-defined,
        !          1805:    aggregate type.  If OR_ELSE is nonzero, give an error message.  */
        !          1806: int
        !          1807: is_aggr_typedef (name, or_else)
        !          1808:      tree name;
        !          1809:      int or_else;
        !          1810: {
        !          1811:   tree type;
        !          1812: 
        !          1813:   if (name == error_mark_node)
        !          1814:     return 0;
        !          1815: 
        !          1816:   if (IDENTIFIER_HAS_TYPE_VALUE (name))
        !          1817:     type = IDENTIFIER_TYPE_VALUE (name);
        !          1818:   else if (IDENTIFIER_HAS_CLASS_TYPE_VALUE (name))
        !          1819:     type = IDENTIFIER_CLASS_TYPE_VALUE (name);
        !          1820:   else
        !          1821:     {
        !          1822:       if (or_else)
        !          1823:        cp_error ("`%T' fails to be an aggregate typedef", name);
        !          1824:       return 0;
        !          1825:     }
        !          1826: 
        !          1827:   if (! IS_AGGR_TYPE (type))
        !          1828:     {
        !          1829:       if (or_else)
        !          1830:        cp_error ("type `%T' is of non-aggregate type", type);
        !          1831:       return 0;
        !          1832:     }
        !          1833:   return 1;
        !          1834: }
        !          1835: 
        !          1836: /* This code could just as well go in `cp-class.c', but is placed here for
        !          1837:    modularity.  */
        !          1838: 
        !          1839: /* For an expression of the form CNAME :: NAME (PARMLIST), build
        !          1840:    the appropriate function call.  */
        !          1841: tree
        !          1842: build_member_call (cname, name, parmlist)
        !          1843:      tree cname, name, parmlist;
        !          1844: {
        !          1845:   tree type, t;
        !          1846:   tree method_name = name;
        !          1847:   int dtor = 0;
        !          1848:   int dont_use_this = 0;
        !          1849:   tree basetype_path, decl;
        !          1850: 
        !          1851:   if (TREE_CODE (method_name) == BIT_NOT_EXPR)
        !          1852:     {
        !          1853:       method_name = TREE_OPERAND (method_name, 0);
        !          1854:       dtor = 1;
        !          1855:     }
        !          1856: 
        !          1857:   if (TREE_CODE (cname) == SCOPE_REF)
        !          1858:     cname = resolve_scope_to_name (NULL_TREE, cname);
        !          1859: 
        !          1860:   if (cname == NULL_TREE || ! is_aggr_typedef (cname, 1))
        !          1861:     return error_mark_node;
        !          1862: 
        !          1863:   /* An operator we did not like.  */
        !          1864:   if (name == NULL_TREE)
        !          1865:     return error_mark_node;
        !          1866: 
        !          1867:   if (dtor)
        !          1868:     {
        !          1869: #if 0
        !          1870:       /* Everything can explicitly call a destructor; see 12.4 */
        !          1871:       if (! TYPE_HAS_DESTRUCTOR (IDENTIFIER_TYPE_VALUE (cname)))
        !          1872:        cp_error ("type `%#T' does not have a destructor", cname);
        !          1873:       else
        !          1874: #endif
        !          1875:       cp_error ("cannot call destructor `%T::~%T' without object",
        !          1876:                  IDENTIFIER_TYPE_VALUE (cname), method_name);
        !          1877:       return error_mark_node;
        !          1878:     }
        !          1879: 
        !          1880:   type = IDENTIFIER_TYPE_VALUE (cname);
        !          1881: 
        !          1882:   /* No object?  Then just fake one up, and let build_method_call
        !          1883:      figure out what to do.  */
        !          1884:   if (current_class_type == 0
        !          1885:       || get_base_distance (type, current_class_type, 0, &basetype_path) == -1)
        !          1886:     dont_use_this = 1;
        !          1887: 
        !          1888:   if (dont_use_this)
        !          1889:     {
        !          1890:       basetype_path = NULL_TREE;
        !          1891:       decl = build1 (NOP_EXPR, TYPE_POINTER_TO (type), error_mark_node);
        !          1892:     }
        !          1893:   else if (current_class_decl == 0)
        !          1894:     {
        !          1895:       dont_use_this = 1;
        !          1896:       decl = build1 (NOP_EXPR, TYPE_POINTER_TO (type), error_mark_node);
        !          1897:     }
        !          1898:   else
        !          1899:     {
        !          1900:       decl = current_class_decl;
        !          1901:       if (TREE_TYPE (decl) != type)
        !          1902:        decl = convert (TYPE_POINTER_TO (type), decl);
        !          1903:     }
        !          1904: 
        !          1905:   if (t = lookup_fnfields (TYPE_BINFO (type), method_name, 0))
        !          1906:     return build_method_call (decl, method_name, parmlist, basetype_path,
        !          1907:                              LOOKUP_NORMAL|LOOKUP_NONVIRTUAL);
        !          1908:   if (TREE_CODE (name) == IDENTIFIER_NODE
        !          1909:       && (t = lookup_field (TYPE_BINFO (type), name, 1, 0)))
        !          1910:     {
        !          1911:       if (t == error_mark_node)
        !          1912:        return error_mark_node;
        !          1913:       if (TREE_CODE (t) == FIELD_DECL)
        !          1914:        {
        !          1915:          if (dont_use_this)
        !          1916:            {
        !          1917:              error ("invalid use of non-static field `%s'",
        !          1918:                     IDENTIFIER_POINTER (name));
        !          1919:              return error_mark_node;
        !          1920:            }
        !          1921:          decl = build (COMPONENT_REF, TREE_TYPE (t), decl, t);
        !          1922:        }
        !          1923:       else if (TREE_CODE (t) == VAR_DECL)
        !          1924:        decl = t;
        !          1925:       else
        !          1926:        {
        !          1927:          error ("invalid use of member `%s::%s'",
        !          1928:                 IDENTIFIER_POINTER (cname), name);
        !          1929:          return error_mark_node;
        !          1930:        }
        !          1931:       if (TYPE_LANG_SPECIFIC (TREE_TYPE (decl))
        !          1932:          && TYPE_OVERLOADS_CALL_EXPR (TREE_TYPE (decl)))
        !          1933:        return build_opfncall (CALL_EXPR, LOOKUP_NORMAL, decl, parmlist, NULL_TREE);
        !          1934:       return build_function_call (decl, parmlist);
        !          1935:     }
        !          1936:   else
        !          1937:     {
        !          1938:       char *err_name;
        !          1939:       if (TREE_CODE (name) == IDENTIFIER_NODE)
        !          1940:        {
        !          1941:          if (IDENTIFIER_OPNAME_P (name))
        !          1942:            {
        !          1943:              char *op_name = operator_name_string (method_name);
        !          1944:              err_name = (char *)alloca (13 + strlen (op_name));
        !          1945:              sprintf (err_name, "operator %s", op_name);
        !          1946:            }
        !          1947:          else
        !          1948:            err_name = IDENTIFIER_POINTER (name);
        !          1949:        }
        !          1950:       else
        !          1951:        my_friendly_abort (51);
        !          1952: 
        !          1953:       error ("no method `%s::%s'", IDENTIFIER_POINTER (cname), err_name);
        !          1954:       return error_mark_node;
        !          1955:     }
        !          1956: }
        !          1957: 
        !          1958: /* Build a reference to a member of an aggregate.  This is not a
        !          1959:    C++ `&', but really something which can have its address taken,
        !          1960:    and then act as a pointer to member, for example CNAME :: FIELD
        !          1961:    can have its address taken by saying & CNAME :: FIELD.
        !          1962: 
        !          1963:    @@ Prints out lousy diagnostics for operator <typename>
        !          1964:    @@ fields.
        !          1965: 
        !          1966:    @@ This function should be rewritten and placed in cp-search.c.  */
        !          1967: tree
        !          1968: build_offset_ref (cname, name)
        !          1969:      tree cname, name;
        !          1970: {
        !          1971:   tree decl, type, fnfields, fields, t = error_mark_node;
        !          1972:   tree basetypes = NULL_TREE;
        !          1973:   int dtor = 0;
        !          1974: 
        !          1975:   if (TREE_CODE (cname) == SCOPE_REF)
        !          1976:     cname = resolve_scope_to_name (NULL_TREE, cname);
        !          1977: 
        !          1978:   if (cname == NULL_TREE || ! is_aggr_typedef (cname, 1))
        !          1979:     return error_mark_node;
        !          1980: 
        !          1981:   type = IDENTIFIER_TYPE_VALUE (cname);
        !          1982: 
        !          1983:   if (TREE_CODE (name) == BIT_NOT_EXPR)
        !          1984:     {
        !          1985:       dtor = 1;
        !          1986:       name = TREE_OPERAND (name, 0);
        !          1987:     }
        !          1988: 
        !          1989:   if (TYPE_SIZE (type) == 0)
        !          1990:     {
        !          1991:       t = IDENTIFIER_CLASS_VALUE (name);
        !          1992:       if (t == 0)
        !          1993:        {
        !          1994:          cp_error ("incomplete type `%T' does not have member `%D'", type,
        !          1995:                      name);
        !          1996:          return error_mark_node;
        !          1997:        }
        !          1998:       if (TREE_CODE (t) == TYPE_DECL)
        !          1999:        {
        !          2000:          cp_error ("member `%D' is just a type declaration", t);
        !          2001:          return error_mark_node;
        !          2002:        }
        !          2003:       if (TREE_CODE (t) == VAR_DECL || TREE_CODE (t) == CONST_DECL)
        !          2004:        {
        !          2005:          TREE_USED (t) = 1;
        !          2006:          return t;
        !          2007:        }
        !          2008:       if (TREE_CODE (t) == FIELD_DECL)
        !          2009:        sorry ("use of member in incomplete aggregate type");
        !          2010:       else if (TREE_CODE (t) == FUNCTION_DECL)
        !          2011:        sorry ("use of member function in incomplete aggregate type");
        !          2012:       else
        !          2013:        my_friendly_abort (52);
        !          2014:       return error_mark_node;
        !          2015:     }
        !          2016: 
        !          2017:   if (TREE_CODE (name) == TYPE_EXPR)
        !          2018:     /* Pass a TYPE_DECL to build_component_type_expr.  */
        !          2019:     return build_component_type_expr (TYPE_NAME (TREE_TYPE (cname)),
        !          2020:                                      name, NULL_TREE, 1);
        !          2021: 
        !          2022:   fnfields = lookup_fnfields (TYPE_BINFO (type), name, 1);
        !          2023:   fields = lookup_field (type, name, 0, 0);
        !          2024: 
        !          2025:   if (fields == error_mark_node || fnfields == error_mark_node)
        !          2026:     return error_mark_node;
        !          2027: 
        !          2028:   if (current_class_type == 0
        !          2029:       || get_base_distance (type, current_class_type, 0, &basetypes) == -1)
        !          2030:     {
        !          2031:       basetypes = TYPE_BINFO (type);
        !          2032:       decl = build1 (NOP_EXPR,
        !          2033:                     IDENTIFIER_TYPE_VALUE (cname),
        !          2034:                     error_mark_node);
        !          2035:     }
        !          2036:   else if (current_class_decl == 0)
        !          2037:     decl = build1 (NOP_EXPR, IDENTIFIER_TYPE_VALUE (cname),
        !          2038:                   error_mark_node);
        !          2039:   else
        !          2040:     decl = C_C_D;
        !          2041: 
        !          2042:   /* A lot of this logic is now handled in lookup_field and
        !          2043:      lookup_fnfield. */
        !          2044:   if (fnfields)
        !          2045:     {
        !          2046:       basetypes = TREE_PURPOSE (fnfields);
        !          2047: 
        !          2048:       /* Go from the TREE_BASELINK to the member function info.  */
        !          2049:       t = TREE_VALUE (fnfields);
        !          2050: 
        !          2051:       if (fields)
        !          2052:        {
        !          2053:          if (DECL_FIELD_CONTEXT (fields) == DECL_FIELD_CONTEXT (t))
        !          2054:            {
        !          2055:              error ("ambiguous member reference: member `%s' defined as both field and function",
        !          2056:                     IDENTIFIER_POINTER (name));
        !          2057:              return error_mark_node;
        !          2058:            }
        !          2059:          if (UNIQUELY_DERIVED_FROM_P (DECL_FIELD_CONTEXT (fields), DECL_FIELD_CONTEXT (t)))
        !          2060:            ;
        !          2061:          else if (UNIQUELY_DERIVED_FROM_P (DECL_FIELD_CONTEXT (t), DECL_FIELD_CONTEXT (fields)))
        !          2062:            t = fields;
        !          2063:          else
        !          2064:            {
        !          2065:              error ("ambiguous member reference: member `%s' derives from distinct classes in multiple inheritance lattice");
        !          2066:              return error_mark_node;
        !          2067:            }
        !          2068:        }
        !          2069: 
        !          2070:       if (t == TREE_VALUE (fnfields))
        !          2071:        {
        !          2072:          extern int flag_save_memoized_contexts;
        !          2073: 
        !          2074:          /* This does not handle visibility checking yet.  */
        !          2075:          if (DECL_CHAIN (t) == NULL_TREE || dtor)
        !          2076:            {
        !          2077:              enum visibility_type visibility;
        !          2078: 
        !          2079:              /* unique functions are handled easily.  */
        !          2080:            unique:
        !          2081:              visibility = compute_visibility (basetypes, t);
        !          2082:              if (visibility == visibility_protected)
        !          2083:                {
        !          2084:                  cp_error_at ("member function `%#D' is protected", t);
        !          2085:                  error ("in this context");
        !          2086:                  return error_mark_node;
        !          2087:                }
        !          2088:              if (visibility == visibility_private)
        !          2089:                {
        !          2090:                  cp_error_at ("member function `%#D' is private", t);
        !          2091:                  error ("in this context");
        !          2092:                  return error_mark_node;
        !          2093:                }
        !          2094:              assemble_external (t);
        !          2095:              return build (OFFSET_REF, TREE_TYPE (t), decl, t);
        !          2096:            }
        !          2097: 
        !          2098:          /* overloaded functions may need more work.  */
        !          2099:          if (cname == name)
        !          2100:            {
        !          2101:              if (TYPE_HAS_DESTRUCTOR (type)
        !          2102:                  && DECL_CHAIN (DECL_CHAIN (t)) == NULL_TREE)
        !          2103:                {
        !          2104:                  t = DECL_CHAIN (t);
        !          2105:                  goto unique;
        !          2106:                }
        !          2107:            }
        !          2108:          /* FNFIELDS is most likely allocated on the search_obstack,
        !          2109:             which will go away after this class scope.  If we need
        !          2110:             to save this value for later (either for memoization
        !          2111:             or for use as an initializer for a static variable), then
        !          2112:             do so here.
        !          2113: 
        !          2114:             ??? The smart thing to do for the case of saving initializers
        !          2115:             is to resolve them before we're done with this scope.  */
        !          2116:          if (!TREE_PERMANENT (fnfields)
        !          2117:              && ((flag_save_memoized_contexts && global_bindings_p ())
        !          2118:                  || ! allocation_temporary_p ()))
        !          2119:            fnfields = copy_list (fnfields);
        !          2120:          t = build_tree_list (error_mark_node, fnfields);
        !          2121:          TREE_TYPE (t) = build_offset_type (type, unknown_type_node);
        !          2122:          return t;
        !          2123:        }
        !          2124:     }
        !          2125: 
        !          2126:   /* Now that we know we are looking for a field, see if we
        !          2127:      have access to that field.  Lookup_field will give us the
        !          2128:      error message.  */
        !          2129: 
        !          2130:   t = lookup_field (basetypes, name, 1, 0);
        !          2131: 
        !          2132:   if (t == error_mark_node)
        !          2133:     return error_mark_node;
        !          2134: 
        !          2135:   if (t == NULL_TREE)
        !          2136:     {
        !          2137:       cp_error ("`%D' is not a member of type `%T'", name,
        !          2138:                  IDENTIFIER_TYPE_VALUE (cname));
        !          2139:       return error_mark_node;
        !          2140:     }
        !          2141: 
        !          2142:   if (TREE_CODE (t) == TYPE_DECL)
        !          2143:     {
        !          2144:       cp_error ("member `%D' is just a type declaration", t);
        !          2145:       return error_mark_node;
        !          2146:     }
        !          2147:   /* static class members and class-specific enum
        !          2148:      values can be returned without further ado.  */
        !          2149:   if (TREE_CODE (t) == VAR_DECL || TREE_CODE (t) == CONST_DECL)
        !          2150:     {
        !          2151:       assemble_external (t);
        !          2152:       TREE_USED (t) = 1;
        !          2153:       return t;
        !          2154:     }
        !          2155: 
        !          2156:   /* static class functions too.  */
        !          2157:   if (TREE_CODE (t) == FUNCTION_DECL && TREE_CODE (TREE_TYPE (t)) == FUNCTION_TYPE)
        !          2158:     my_friendly_abort (53);
        !          2159: 
        !          2160:   /* In member functions, the form `cname::name' is no longer
        !          2161:      equivalent to `this->cname::name'.  */
        !          2162:   return build (OFFSET_REF, build_offset_type (type, TREE_TYPE (t)), decl, t);
        !          2163: }
        !          2164: 
        !          2165: /* Given an object EXP and a member function reference MEMBER,
        !          2166:    return the address of the actual member function.  */
        !          2167: tree
        !          2168: get_member_function (exp_addr_ptr, exp, member)
        !          2169:      tree *exp_addr_ptr;
        !          2170:      tree exp, member;
        !          2171: {
        !          2172:   tree ctype = TREE_TYPE (exp);
        !          2173:   tree function = save_expr (build_unary_op (ADDR_EXPR, member, 0));
        !          2174: 
        !          2175:   if (TYPE_VIRTUAL_P (ctype)
        !          2176:       || (flag_all_virtual == 1 && TYPE_OVERLOADS_METHOD_CALL_EXPR (ctype)))
        !          2177:     {
        !          2178:       tree e0, e1, e3;
        !          2179:       tree exp_addr;
        !          2180: 
        !          2181:       /* Save away the unadulterated `this' pointer.  */
        !          2182:       exp_addr = save_expr (*exp_addr_ptr);
        !          2183: 
        !          2184:       /* Cast function to signed integer.  */
        !          2185:       e0 = build1 (NOP_EXPR, integer_type_node, function);
        !          2186: 
        !          2187: #ifdef VTABLE_USES_MASK
        !          2188:       /* If we are willing to limit the number of
        !          2189:         virtual functions a class may have to some
        !          2190:         *small* number, then if, for a function address,
        !          2191:         we are passed some small number, we know that
        !          2192:         it is a virtual function index, and work from there.  */
        !          2193:       e1 = build (BIT_AND_EXPR, integer_type_node, e0, vtbl_mask);
        !          2194: #else
        !          2195:       /* There is a hack here that takes advantage of
        !          2196:         twos complement arithmetic, and the fact that
        !          2197:         there are more than one UNITS to the WORD.
        !          2198:         If the high bit is set for the `function',
        !          2199:         then we pretend it is a virtual function,
        !          2200:         and the array indexing will knock this bit
        !          2201:         out the top, leaving a valid index.  */
        !          2202:       if (UNITS_PER_WORD <= 1)
        !          2203:        my_friendly_abort (54);
        !          2204: 
        !          2205:       e1 = build (GT_EXPR, integer_type_node, e0, integer_zero_node);
        !          2206:       e1 = build_compound_expr (tree_cons (NULL_TREE, exp_addr,
        !          2207:                                           build_tree_list (NULL_TREE, e1)));
        !          2208:       e1 = save_expr (e1);
        !          2209: #endif
        !          2210: 
        !          2211:       if (TREE_SIDE_EFFECTS (*exp_addr_ptr))
        !          2212:        {
        !          2213:          exp = build_indirect_ref (exp_addr, NULL_PTR);
        !          2214:          *exp_addr_ptr = exp_addr;
        !          2215:        }
        !          2216: 
        !          2217:       /* This is really hairy: if the function pointer is a pointer
        !          2218:         to a non-virtual member function, then we can't go mucking
        !          2219:         with the `this' pointer (any more than we already have to
        !          2220:         this point).  If it is a pointer to a virtual member function,
        !          2221:         then we have to adjust the `this' pointer according to
        !          2222:         what the virtual function table tells us.  */
        !          2223: 
        !          2224:       e3 = build_vfn_ref (exp_addr_ptr, exp, e0);
        !          2225:       my_friendly_assert (e3 != error_mark_node, 213);
        !          2226: 
        !          2227:       /* Change this pointer type from `void *' to the
        !          2228:         type it is really supposed to be.  */
        !          2229:       TREE_TYPE (e3) = TREE_TYPE (function);
        !          2230: 
        !          2231:       /* If non-virtual, use what we had originally.  Otherwise,
        !          2232:         use the value we get from the virtual function table.  */
        !          2233:       *exp_addr_ptr = build_conditional_expr (e1, exp_addr, *exp_addr_ptr);
        !          2234: 
        !          2235:       function = build_conditional_expr (e1, function, e3);
        !          2236:     }
        !          2237:   return build_indirect_ref (function, NULL_PTR);
        !          2238: }
        !          2239: 
        !          2240: /* If a OFFSET_REF made it through to here, then it did
        !          2241:    not have its address taken.  */
        !          2242: 
        !          2243: tree
        !          2244: resolve_offset_ref (exp)
        !          2245:      tree exp;
        !          2246: {
        !          2247:   tree type = TREE_TYPE (exp);
        !          2248:   tree base = NULL_TREE;
        !          2249:   tree member;
        !          2250:   tree basetype, addr;
        !          2251: 
        !          2252:   if (TREE_CODE (exp) == TREE_LIST)
        !          2253:     return build_unary_op (ADDR_EXPR, exp, 0);
        !          2254: 
        !          2255:   if (TREE_CODE (exp) != OFFSET_REF)
        !          2256:     {
        !          2257:       my_friendly_assert (TREE_CODE (type) == OFFSET_TYPE, 214);
        !          2258:       if (TYPE_OFFSET_BASETYPE (type) != current_class_type)
        !          2259:        {
        !          2260:          error ("object missing in use of pointer-to-member construct");
        !          2261:          return error_mark_node;
        !          2262:        }
        !          2263:       member = exp;
        !          2264:       type = TREE_TYPE (type);
        !          2265:       base = C_C_D;
        !          2266:     }
        !          2267:   else
        !          2268:     {
        !          2269:       member = TREE_OPERAND (exp, 1);
        !          2270:       base = TREE_OPERAND (exp, 0);
        !          2271:     }
        !          2272: 
        !          2273:   if ((TREE_CODE (member) == VAR_DECL
        !          2274:        && ! TYPE_PTRMEMFUNC_P (TREE_TYPE (member)))
        !          2275:       || TREE_CODE (TREE_TYPE (member)) == FUNCTION_TYPE)
        !          2276:     {
        !          2277:       /* These were static members.  */
        !          2278:       if (mark_addressable (member) == 0)
        !          2279:        return error_mark_node;
        !          2280:       return member;
        !          2281:     }
        !          2282: 
        !          2283:   /* Syntax error can cause a member which should
        !          2284:      have been seen as static to be grok'd as non-static.  */
        !          2285:   if (TREE_CODE (member) == FIELD_DECL && C_C_D == NULL_TREE)
        !          2286:     {
        !          2287:       if (TREE_ADDRESSABLE (member) == 0)
        !          2288:        {
        !          2289:          cp_error_at ("member `%D' is non-static in static member function context", member);
        !          2290:          error ("at this point in file");
        !          2291:          TREE_ADDRESSABLE (member) = 1;
        !          2292:        }
        !          2293:       return error_mark_node;
        !          2294:     }
        !          2295: 
        !          2296:   /* The first case is really just a reference to a member of `this'.  */
        !          2297:   if (TREE_CODE (member) == FIELD_DECL
        !          2298:       && (base == C_C_D
        !          2299:          || (TREE_CODE (base) == NOP_EXPR
        !          2300:              && TREE_OPERAND (base, 0) == error_mark_node)))
        !          2301:     {
        !          2302:       tree basetype_path;
        !          2303:       enum visibility_type visibility;
        !          2304: 
        !          2305:       basetype = DECL_CONTEXT (member);
        !          2306:       if (get_base_distance (basetype, current_class_type, 0, &basetype_path) < 0)
        !          2307:        {
        !          2308:          error_not_base_type (basetype, current_class_type);
        !          2309:          return error_mark_node;
        !          2310:        }
        !          2311:       addr = convert_pointer_to (basetype, current_class_decl);
        !          2312:       visibility = compute_visibility (basetype_path, member);
        !          2313:       if (visibility == visibility_public)
        !          2314:        return build (COMPONENT_REF, TREE_TYPE (member),
        !          2315:                      build_indirect_ref (addr, NULL_PTR), member);
        !          2316:       if (visibility == visibility_protected)
        !          2317:        {
        !          2318:          cp_error_at ("member `%D' is protected", member);
        !          2319:          error ("in this context");
        !          2320:          return error_mark_node;
        !          2321:        }
        !          2322:       if (visibility == visibility_private)
        !          2323:        {
        !          2324:          cp_error_at ("member `%D' is private", member);
        !          2325:          error ("in this context");
        !          2326:          return error_mark_node;
        !          2327:        }
        !          2328:       my_friendly_abort (55);
        !          2329:     }
        !          2330: 
        !          2331:   /* If this is a reference to a member function, then return
        !          2332:      the address of the member function (which may involve going
        !          2333:      through the object's vtable), otherwise, return an expression
        !          2334:      for the dereferenced pointer-to-member construct.  */
        !          2335:   addr = build_unary_op (ADDR_EXPR, base, 0);
        !          2336: 
        !          2337:   if (TREE_CODE (TREE_TYPE (member)) == METHOD_TYPE)
        !          2338:     {
        !          2339:       basetype = DECL_CLASS_CONTEXT (member);
        !          2340:       addr = convert_pointer_to (basetype, addr);
        !          2341:       return build_unary_op (ADDR_EXPR, get_member_function (&addr, build_indirect_ref (addr, NULL_PTR), member), 0);
        !          2342:     }
        !          2343:   else if (TREE_CODE (TREE_TYPE (member)) == OFFSET_TYPE)
        !          2344:     {
        !          2345:       basetype = TYPE_OFFSET_BASETYPE (TREE_TYPE (member));
        !          2346:       addr = convert_pointer_to (basetype, addr);
        !          2347:       member = convert (ptr_type_node, build_unary_op (ADDR_EXPR, member, 0));
        !          2348:       return build1 (INDIRECT_REF, type,
        !          2349:                     build (PLUS_EXPR, ptr_type_node, addr, member));
        !          2350:     }
        !          2351:   else if (TYPE_PTRMEMFUNC_P (TREE_TYPE (member)))
        !          2352:     {
        !          2353:       return get_member_function_from_ptrfunc (&addr, base, member);
        !          2354:     }
        !          2355:   my_friendly_abort (56);
        !          2356:   /* NOTREACHED */
        !          2357:   return NULL_TREE;
        !          2358: }
        !          2359: 
        !          2360: /* Return either DECL or its known constant value (if it has one).  */
        !          2361: 
        !          2362: tree
        !          2363: decl_constant_value (decl)
        !          2364:      tree decl;
        !          2365: {
        !          2366:   if (! TREE_THIS_VOLATILE (decl)
        !          2367: #if 0
        !          2368:       /* These may be necessary for C, but they break C++.  */
        !          2369:       ! TREE_PUBLIC (decl)
        !          2370:       /* Don't change a variable array bound or initial value to a constant
        !          2371:         in a place where a variable is invalid.  */
        !          2372:       && ! pedantic
        !          2373: #endif /* 0 */
        !          2374:       && DECL_INITIAL (decl) != 0
        !          2375:       && TREE_CODE (DECL_INITIAL (decl)) != ERROR_MARK
        !          2376:       /* This is invalid if initial value is not constant.
        !          2377:         If it has either a function call, a memory reference,
        !          2378:         or a variable, then re-evaluating it could give different results.  */
        !          2379:       && TREE_CONSTANT (DECL_INITIAL (decl))
        !          2380:       /* Check for cases where this is sub-optimal, even though valid.  */
        !          2381:       && TREE_CODE (DECL_INITIAL (decl)) != CONSTRUCTOR
        !          2382: #if 0
        !          2383:       /* We must allow this to work outside of functions so that
        !          2384:         static constants can be used for array sizes.  */
        !          2385:       && current_function_decl != 0
        !          2386:       && DECL_MODE (decl) != BLKmode
        !          2387: #endif
        !          2388:       )
        !          2389:     return DECL_INITIAL (decl);
        !          2390:   return decl;
        !          2391: }
        !          2392: 
        !          2393: /* Friend handling routines.  */
        !          2394: /* Friend data structures:
        !          2395: 
        !          2396:    Friend lists come from TYPE_DECL nodes.  Since all aggregate
        !          2397:    types are automatically typedef'd, these node are guaranteed
        !          2398:    to exist.
        !          2399: 
        !          2400:    The TREE_PURPOSE of a friend list is the name of the friend,
        !          2401:    and its TREE_VALUE is another list.
        !          2402: 
        !          2403:    The TREE_PURPOSE of that list is a type, which allows
        !          2404:    all functions of a given type to be friends.
        !          2405:    The TREE_VALUE of that list is an individual function
        !          2406:    which is a friend.
        !          2407: 
        !          2408:    Non-member friends will match only by their DECL.  Their
        !          2409:    member type is NULL_TREE, while the type of the inner
        !          2410:    list will either be of aggregate type or error_mark_node.  */
        !          2411: 
        !          2412: /* Tell if TYPE1 is a friend of TYPE2.  */
        !          2413: int
        !          2414: is_friend_type (type1, type2)
        !          2415:      tree type1, type2;
        !          2416: {
        !          2417:   register tree list;
        !          2418: 
        !          2419:   if (! type1 || ! type2)
        !          2420:     return 0;
        !          2421: 
        !          2422:   list = CLASSTYPE_FRIEND_CLASSES (TREE_TYPE (TYPE_NAME (type1)));
        !          2423:   while (list)
        !          2424:     {
        !          2425:       if (type2 == TREE_VALUE (list))
        !          2426:        return 1;
        !          2427:       list = TREE_CHAIN (list);
        !          2428:     }
        !          2429: 
        !          2430:   return 0;
        !          2431: }
        !          2432: 
        !          2433: /* Tell if this function specified by DECL can be a friend of type TYPE.
        !          2434:    Return nonzero if friend, zero otherwise.
        !          2435: 
        !          2436:    DECL can be zero if we are calling a constructor or accessing a
        !          2437:    member in global scope.  */
        !          2438: int
        !          2439: is_friend (type, decl)
        !          2440:      tree type, decl;
        !          2441: {
        !          2442:   tree ctype, list, name;
        !          2443: 
        !          2444:   if (decl == NULL_TREE)
        !          2445:     return 0;
        !          2446: 
        !          2447:   ctype = DECL_CLASS_CONTEXT (decl);
        !          2448: 
        !          2449:   if (ctype && is_friend_type (type, ctype))
        !          2450:     return 1;
        !          2451: 
        !          2452:   list = DECL_FRIENDLIST (TYPE_NAME (type));
        !          2453:   name = DECL_NAME (decl);
        !          2454:   while (list)
        !          2455:     {
        !          2456:       if (name == TREE_PURPOSE (list))
        !          2457:        {
        !          2458:          tree friends = TREE_VALUE (list);
        !          2459:          name = DECL_ASSEMBLER_NAME (decl);
        !          2460:          while (friends)
        !          2461:            {
        !          2462:              if (ctype == TREE_PURPOSE (friends))
        !          2463:                return 1;
        !          2464:              if (name == DECL_ASSEMBLER_NAME (TREE_VALUE (friends)))
        !          2465:                return 1;
        !          2466:              friends = TREE_CHAIN (friends);
        !          2467:            }
        !          2468:          return 0;
        !          2469:        }
        !          2470:       list = TREE_CHAIN (list);
        !          2471:     }
        !          2472:   return 0;
        !          2473: }
        !          2474: 
        !          2475: /* Add a new friend to the friends of the aggregate type TYPE.
        !          2476:    DECL is the FUNCTION_DECL of the friend being added.  */
        !          2477: static void
        !          2478: add_friend (type, decl)
        !          2479:      tree type, decl;
        !          2480: {
        !          2481:   tree typedecl = TYPE_NAME (type);
        !          2482:   tree list = DECL_FRIENDLIST (typedecl);
        !          2483:   tree name = DECL_NAME (decl);
        !          2484:   tree ctype = TREE_CODE (TREE_TYPE (decl)) == METHOD_TYPE
        !          2485:     ? DECL_CLASS_CONTEXT (decl) : error_mark_node;
        !          2486: 
        !          2487:   while (list)
        !          2488:     {
        !          2489:       if (name == TREE_PURPOSE (list))
        !          2490:        {
        !          2491:          tree friends = TREE_VALUE (list);
        !          2492:          while (friends)
        !          2493:            {
        !          2494:              if (decl == TREE_VALUE (friends))
        !          2495:                {
        !          2496:                  cp_pedwarn_at ("`%D' is already a friend of class `%T'",
        !          2497:                                   decl, type);
        !          2498:                  return;
        !          2499:                }
        !          2500:              friends = TREE_CHAIN (friends);
        !          2501:            }
        !          2502:          TREE_VALUE (list) = tree_cons (ctype, decl, TREE_VALUE (list));
        !          2503:          return;
        !          2504:        }
        !          2505:       list = TREE_CHAIN (list);
        !          2506:     }
        !          2507:   DECL_FRIENDLIST (typedecl)
        !          2508:     = tree_cons (DECL_NAME (decl), build_tree_list (error_mark_node, decl),
        !          2509:                 DECL_FRIENDLIST (typedecl));
        !          2510:   if (DECL_NAME (decl) == ansi_opname[(int) MODIFY_EXPR])
        !          2511:     {
        !          2512:       tree parmtypes = TYPE_ARG_TYPES (TREE_TYPE (decl));
        !          2513:       TYPE_HAS_ASSIGNMENT (TREE_TYPE (typedecl)) = 1;
        !          2514:       TYPE_GETS_ASSIGNMENT (TREE_TYPE (typedecl)) = 1;
        !          2515:       if (parmtypes && TREE_CHAIN (parmtypes))
        !          2516:        {
        !          2517:          tree parmtype = TREE_VALUE (TREE_CHAIN (parmtypes));
        !          2518:          if (TREE_CODE (parmtype) == REFERENCE_TYPE
        !          2519:              && TREE_TYPE (parmtypes) == TREE_TYPE (typedecl))
        !          2520:            {
        !          2521:              TYPE_HAS_ASSIGN_REF (TREE_TYPE (typedecl)) = 1;
        !          2522:              TYPE_GETS_ASSIGN_REF (TREE_TYPE (typedecl)) = 1;
        !          2523:            }
        !          2524:        }
        !          2525:     }
        !          2526: }
        !          2527: 
        !          2528: /* Declare that every member function NAME in FRIEND_TYPE
        !          2529:    (which may be NULL_TREE) is a friend of type TYPE.  */
        !          2530: static void
        !          2531: add_friends (type, name, friend_type)
        !          2532:      tree type, name, friend_type;
        !          2533: {
        !          2534:   tree typedecl = TYPE_NAME (type);
        !          2535:   tree list = DECL_FRIENDLIST (typedecl);
        !          2536: 
        !          2537:   while (list)
        !          2538:     {
        !          2539:       if (name == TREE_PURPOSE (list))
        !          2540:        {
        !          2541:          tree friends = TREE_VALUE (list);
        !          2542:          while (friends && TREE_PURPOSE (friends) != friend_type)
        !          2543:            friends = TREE_CHAIN (friends);
        !          2544:          if (friends)
        !          2545:            if (friend_type)
        !          2546:              warning ("method `%s::%s' is already a friend of class",
        !          2547:                       TYPE_NAME_STRING (friend_type),
        !          2548:                       IDENTIFIER_POINTER (name));
        !          2549:            else
        !          2550:              warning ("function `%s' is already a friend of class `%s'",
        !          2551:                       IDENTIFIER_POINTER (name),
        !          2552:                       IDENTIFIER_POINTER (DECL_NAME (typedecl)));
        !          2553:          else
        !          2554:            TREE_VALUE (list) = tree_cons (friend_type, NULL_TREE,
        !          2555:                                           TREE_VALUE (list));
        !          2556:          return;
        !          2557:        }
        !          2558:       list = TREE_CHAIN (list);
        !          2559:     }
        !          2560:   DECL_FRIENDLIST (typedecl) =
        !          2561:     tree_cons (name,
        !          2562:               build_tree_list (friend_type, NULL_TREE),
        !          2563:               DECL_FRIENDLIST (typedecl));
        !          2564:   if (! strncmp (IDENTIFIER_POINTER (name),
        !          2565:                 IDENTIFIER_POINTER (ansi_opname[(int) MODIFY_EXPR]),
        !          2566:                 strlen (IDENTIFIER_POINTER (ansi_opname[(int) MODIFY_EXPR]))))
        !          2567:     {
        !          2568:       TYPE_HAS_ASSIGNMENT (TREE_TYPE (typedecl)) = 1;
        !          2569:       TYPE_GETS_ASSIGNMENT (TREE_TYPE (typedecl)) = 1;
        !          2570:       sorry ("declaring \"friend operator =\" will not find \"operator = (X&)\" if it exists");
        !          2571:     }
        !          2572: }
        !          2573: 
        !          2574: /* Set up a cross reference so that type TYPE will make member function
        !          2575:    CTYPE::DECL a friend when CTYPE is finally defined.  For more than
        !          2576:    one, set up a cross reference so that functions with the name DECL
        !          2577:    and type CTYPE know that they are friends of TYPE.  */
        !          2578: static void
        !          2579: xref_friend (type, decl, ctype)
        !          2580:      tree type, decl, ctype;
        !          2581: {
        !          2582:   tree typedecl = TYPE_NAME (type);
        !          2583:   tree friend_decl = TYPE_NAME (ctype);
        !          2584:   tree t = tree_cons (NULL_TREE, ctype, DECL_UNDEFINED_FRIENDS (typedecl));
        !          2585: 
        !          2586:   DECL_UNDEFINED_FRIENDS (typedecl) = t;
        !          2587:   SET_DECL_WAITING_FRIENDS (friend_decl,
        !          2588:                            tree_cons (type, t,
        !          2589:                                       DECL_WAITING_FRIENDS (friend_decl)));
        !          2590:   TREE_TYPE (DECL_WAITING_FRIENDS (friend_decl)) = decl;
        !          2591: }
        !          2592: 
        !          2593: /* Make FRIEND_TYPE a friend class to TYPE.  If FRIEND_TYPE has already
        !          2594:    been defined, we make all of its member functions friends of
        !          2595:    TYPE.  If not, we make it a pending friend, which can later be added
        !          2596:    when its definition is seen.  If a type is defined, then its TYPE_DECL's
        !          2597:    DECL_UNDEFINED_FRIENDS contains a (possibly empty) list of friend
        !          2598:    classes that are not defined.  If a type has not yet been defined,
        !          2599:    then the DECL_WAITING_FRIENDS contains a list of types
        !          2600:    waiting to make it their friend.  Note that these two can both
        !          2601:    be in use at the same time!  */
        !          2602: void
        !          2603: make_friend_class (type, friend_type)
        !          2604:      tree type, friend_type;
        !          2605: {
        !          2606:   tree classes;
        !          2607: 
        !          2608:   if (type == friend_type)
        !          2609:     {
        !          2610:       warning ("class `%s' is implicitly friends with itself",
        !          2611:               TYPE_NAME_STRING (type));
        !          2612:       return;
        !          2613:     }
        !          2614: 
        !          2615:   GNU_xref_hier (TYPE_NAME_STRING (type),
        !          2616:                 TYPE_NAME_STRING (friend_type), 0, 0, 1);
        !          2617: 
        !          2618:   classes = CLASSTYPE_FRIEND_CLASSES (type);
        !          2619:   while (classes && TREE_VALUE (classes) != friend_type)
        !          2620:     classes = TREE_CHAIN (classes);
        !          2621:   if (classes)
        !          2622:     warning ("class `%s' is already friends with class `%s'",
        !          2623:             TYPE_NAME_STRING (TREE_VALUE (classes)), TYPE_NAME_STRING (type));
        !          2624:   else
        !          2625:     {
        !          2626:       CLASSTYPE_FRIEND_CLASSES (type)
        !          2627:        = tree_cons (NULL_TREE, friend_type, CLASSTYPE_FRIEND_CLASSES (type));
        !          2628:     }
        !          2629: }
        !          2630: 
        !          2631: /* Main friend processor.  This is large, and for modularity purposes,
        !          2632:    has been removed from grokdeclarator.  It returns `void_type_node'
        !          2633:    to indicate that something happened, though a FIELD_DECL is
        !          2634:    not returned.
        !          2635: 
        !          2636:    CTYPE is the class this friend belongs to.
        !          2637: 
        !          2638:    DECLARATOR is the name of the friend.
        !          2639: 
        !          2640:    DECL is the FUNCTION_DECL that the friend is.
        !          2641: 
        !          2642:    In case we are parsing a friend which is part of an inline
        !          2643:    definition, we will need to store PARM_DECL chain that comes
        !          2644:    with it into the DECL_ARGUMENTS slot of the FUNCTION_DECL.
        !          2645: 
        !          2646:    FLAGS is just used for `grokclassfn'.
        !          2647: 
        !          2648:    QUALS say what special qualifies should apply to the object
        !          2649:    pointed to by `this'.  */
        !          2650: tree
        !          2651: do_friend (ctype, declarator, decl, parmdecls, flags, quals)
        !          2652:      tree ctype, declarator, decl, parmdecls;
        !          2653:      enum overload_flags flags;
        !          2654:      tree quals;
        !          2655: {
        !          2656:   /* first, lets find out if what we are making a friend needs overloading */
        !          2657:   tree previous_decl;
        !          2658:   int was_c_linkage = 0;
        !          2659: 
        !          2660:   /* Every decl that gets here is a friend of something.  */
        !          2661:   DECL_FRIEND_P (decl) = 1;
        !          2662: 
        !          2663:   /* If we find something in scope, let see if it has extern "C" linkage.  */
        !          2664:   /* This code is pretty general and should be ripped out and reused
        !          2665:      as a separate function. */
        !          2666:   if (DECL_NAME (decl))
        !          2667:     {
        !          2668:       previous_decl = lookup_name (DECL_NAME (decl), 0);
        !          2669:       if (previous_decl && TREE_CODE (previous_decl) == TREE_LIST)
        !          2670:        {
        !          2671:          do
        !          2672:            {
        !          2673:              if (TREE_TYPE (TREE_VALUE (previous_decl)) == TREE_TYPE (decl))
        !          2674:                {
        !          2675:                  previous_decl = TREE_VALUE (previous_decl);
        !          2676:                  break;
        !          2677:                }
        !          2678:              previous_decl = TREE_CHAIN (previous_decl);
        !          2679:            }
        !          2680:          while (previous_decl);
        !          2681:        }
        !          2682: 
        !          2683:       /* It had extern "C" linkage, so don't overload this.  */
        !          2684:       if (previous_decl && TREE_CODE (previous_decl) == FUNCTION_DECL
        !          2685:          && TREE_TYPE (decl) == TREE_TYPE (previous_decl)
        !          2686:          && DECL_LANGUAGE (previous_decl) == lang_c)
        !          2687:        was_c_linkage = 1;
        !          2688:     }
        !          2689:          
        !          2690:   if (ctype)
        !          2691:     {
        !          2692:       tree cname = TYPE_NAME (ctype);
        !          2693:       if (TREE_CODE (cname) == TYPE_DECL)
        !          2694:        cname = DECL_NAME (cname);
        !          2695: 
        !          2696:       /* A method friend.  */
        !          2697:       if (TREE_CODE (decl) == FUNCTION_DECL)
        !          2698:        {
        !          2699:          if (flags == NO_SPECIAL && ctype && declarator == cname)
        !          2700:            DECL_CONSTRUCTOR_P (decl) = 1;
        !          2701: 
        !          2702:          /* This will set up DECL_ARGUMENTS for us.  */
        !          2703:          grokclassfn (ctype, cname, decl, flags, quals);
        !          2704:          if (TYPE_SIZE (ctype) != 0)
        !          2705:            check_classfn (ctype, cname, decl);
        !          2706: 
        !          2707:          if (TREE_TYPE (decl) != error_mark_node)
        !          2708:            {
        !          2709:              if (TYPE_SIZE (ctype))
        !          2710:                {
        !          2711:                  /* We don't call pushdecl here yet, or ever on this
        !          2712:                     actual FUNCTION_DECL.  We must preserve its TREE_CHAIN
        !          2713:                     until the end.  */
        !          2714:                  make_decl_rtl (decl, NULL_PTR, 1);
        !          2715:                  add_friend (current_class_type, decl);
        !          2716:                }
        !          2717:              else
        !          2718:                {
        !          2719:                  register char *classname
        !          2720:                    = IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (ctype)));
        !          2721: 
        !          2722:                  error ("member declared as friend before type `%s' defined",
        !          2723:                         classname);
        !          2724:                }
        !          2725:            }
        !          2726:        }
        !          2727:       else
        !          2728:        {
        !          2729:          /* Possibly a bunch of method friends.  */
        !          2730: 
        !          2731:          /* Get the class they belong to.  */
        !          2732:          tree ctype = IDENTIFIER_TYPE_VALUE (cname);
        !          2733: 
        !          2734:          /* This class is defined, use its methods now.  */
        !          2735:          if (TYPE_SIZE (ctype))
        !          2736:            {
        !          2737:              tree fields = lookup_fnfields (TYPE_BINFO (ctype), declarator, 0);
        !          2738:              if (fields)
        !          2739:                add_friends (current_class_type, declarator, ctype);
        !          2740:              else
        !          2741:                error ("method `%s' is not a member of class `%s'",
        !          2742:                       IDENTIFIER_POINTER (declarator),
        !          2743:                       IDENTIFIER_POINTER (cname));
        !          2744:            }
        !          2745:          else
        !          2746:            /* Note: DECLARATOR actually has more than one; in this
        !          2747:               case, we're making sure that fns with the name DECLARATOR
        !          2748:               and type CTYPE know they are friends of the current
        !          2749:               class type.  */
        !          2750:            xref_friend (current_class_type, declarator, ctype);
        !          2751:          decl = void_type_node;
        !          2752:        }
        !          2753:     }
        !          2754:   /* never overload C functions */
        !          2755:   else if (TREE_CODE (decl) == FUNCTION_DECL
        !          2756:           && ((IDENTIFIER_LENGTH (declarator) == 4
        !          2757:                && IDENTIFIER_POINTER (declarator)[0] == 'm'
        !          2758:                && ! strcmp (IDENTIFIER_POINTER (declarator), "main"))
        !          2759:               || (IDENTIFIER_LENGTH (declarator) > 10
        !          2760:                   && IDENTIFIER_POINTER (declarator)[0] == '_'
        !          2761:                   && IDENTIFIER_POINTER (declarator)[1] == '_'
        !          2762:                   && strncmp (IDENTIFIER_POINTER (declarator)+2,
        !          2763:                               "builtin_", 8) == 0)
        !          2764:               || was_c_linkage))
        !          2765:     {
        !          2766:       /* raw "main", and builtin functions never gets overloaded,
        !          2767:         but they can become friends.  */
        !          2768:       TREE_PUBLIC (decl) = 1;
        !          2769:       add_friend (current_class_type, decl);
        !          2770:       DECL_FRIEND_P (decl) = 1;
        !          2771:       if (IDENTIFIER_POINTER (declarator)[0] == '_')
        !          2772:        {
        !          2773:          if (! strcmp (IDENTIFIER_POINTER (declarator)+10, "new"))
        !          2774:            TREE_GETS_NEW (current_class_type) = 0;
        !          2775:          else if (! strcmp (IDENTIFIER_POINTER (declarator)+10, "delete"))
        !          2776:            TREE_GETS_DELETE (current_class_type) = 0;
        !          2777:        }
        !          2778:       decl = void_type_node;
        !          2779:     }
        !          2780:   /* A global friend.
        !          2781:      @@ or possibly a friend from a base class ?!?  */
        !          2782:   else if (TREE_CODE (decl) == FUNCTION_DECL)
        !          2783:     {
        !          2784:       /* Friends must all go through the overload machinery,
        !          2785:         even though they may not technically be overloaded.
        !          2786: 
        !          2787:         Note that because classes all wind up being top-level
        !          2788:         in their scope, their friend wind up in top-level scope as well.  */
        !          2789:       DECL_ASSEMBLER_NAME (decl)
        !          2790:        = build_decl_overload (declarator, TYPE_ARG_TYPES (TREE_TYPE (decl)),
        !          2791:                               TREE_CODE (TREE_TYPE (decl)) == METHOD_TYPE);
        !          2792:       DECL_ARGUMENTS (decl) = parmdecls;
        !          2793: 
        !          2794:       /* We can call pushdecl here, because the TREE_CHAIN of this
        !          2795:         FUNCTION_DECL is not needed for other purposes.  */
        !          2796:       decl = pushdecl_top_level (decl);
        !          2797: 
        !          2798:       make_decl_rtl (decl, NULL_PTR, 1);
        !          2799:       add_friend (current_class_type, decl);
        !          2800: 
        !          2801:       if (! TREE_OVERLOADED (declarator)
        !          2802:          && IDENTIFIER_GLOBAL_VALUE (declarator)
        !          2803:          && TREE_CODE (IDENTIFIER_GLOBAL_VALUE (declarator)) == FUNCTION_DECL)
        !          2804:        {
        !          2805:          error ("friend `%s' implicitly overloaded",
        !          2806:                 IDENTIFIER_POINTER (declarator));
        !          2807:          cp_error_at ("after declaration of non-overloaded `%D'", IDENTIFIER_GLOBAL_VALUE (declarator));
        !          2808:        }
        !          2809:       DECL_FRIEND_P (decl) = 1;
        !          2810:       DECL_OVERLOADED (decl) = 1;
        !          2811:       TREE_OVERLOADED (declarator) = 1;
        !          2812:       decl = push_overloaded_decl (decl, 1);
        !          2813:     }
        !          2814:   else
        !          2815:     {
        !          2816:       /* @@ Should be able to ingest later definitions of this function
        !          2817:         before use.  */
        !          2818:       tree decl = IDENTIFIER_GLOBAL_VALUE (declarator);
        !          2819:       if (decl == NULL_TREE)
        !          2820:        {
        !          2821:          warning ("implicitly declaring `%s' as struct",
        !          2822:                   IDENTIFIER_POINTER (declarator));
        !          2823:          decl = xref_tag (record_type_node, declarator, NULL_TREE);
        !          2824:          decl = TYPE_NAME (decl);
        !          2825:        }
        !          2826: 
        !          2827:       /* Allow abbreviated declarations of overloaded functions,
        !          2828:         but not if those functions are really class names.  */
        !          2829:       if (TREE_CODE (decl) == TREE_LIST && TREE_TYPE (TREE_PURPOSE (decl)))
        !          2830:        {
        !          2831:          warning ("`friend %s' archaic, use `friend class %s' instead",
        !          2832:                   IDENTIFIER_POINTER (declarator),
        !          2833:                   IDENTIFIER_POINTER (declarator));
        !          2834:          decl = TREE_TYPE (TREE_PURPOSE (decl));
        !          2835:        }
        !          2836: 
        !          2837:       if (TREE_CODE (decl) == TREE_LIST)
        !          2838:        add_friends (current_class_type, TREE_PURPOSE (decl), NULL_TREE);
        !          2839:       else
        !          2840:        make_friend_class (current_class_type, TREE_TYPE (decl));
        !          2841:       decl = void_type_node;
        !          2842:     }
        !          2843:   return decl;
        !          2844: }
        !          2845: 
        !          2846: /* TYPE has now been defined.  It may, however, have a number of things
        !          2847:    waiting make make it their friend.  We resolve these references
        !          2848:    here.  */
        !          2849: void
        !          2850: embrace_waiting_friends (type)
        !          2851:      tree type;
        !          2852: {
        !          2853:   tree decl = TYPE_NAME (type);
        !          2854:   tree waiters;
        !          2855: 
        !          2856:   if (TREE_CODE (decl) != TYPE_DECL)
        !          2857:     return;
        !          2858: 
        !          2859:   for (waiters = DECL_WAITING_FRIENDS (decl); waiters;
        !          2860:        waiters = TREE_CHAIN (waiters))
        !          2861:     {
        !          2862:       tree waiter = TREE_PURPOSE (waiters);
        !          2863:       tree waiter_prev = TREE_VALUE (waiters);
        !          2864:       tree decl = TREE_TYPE (waiters);
        !          2865:       tree name = decl ? (TREE_CODE (decl) == IDENTIFIER_NODE
        !          2866:                          ? decl : DECL_NAME (decl)) : NULL_TREE;
        !          2867:       if (name)
        !          2868:        {
        !          2869:          /* @@ There may be work to be done since we have not verified
        !          2870:             @@ consistency between original and friend declarations
        !          2871:             @@ of the functions waiting to become friends.  */
        !          2872:          tree field = lookup_fnfields (TYPE_BINFO (type), name, 0);
        !          2873:          if (field)
        !          2874:            if (decl == name)
        !          2875:              add_friends (waiter, name, type);
        !          2876:            else
        !          2877:              add_friend (waiter, decl);
        !          2878:          else
        !          2879:            error_with_file_and_line (DECL_SOURCE_FILE (TYPE_NAME (waiter)),
        !          2880:                                      DECL_SOURCE_LINE (TYPE_NAME (waiter)),
        !          2881:                                      "no method `%s' defined in class `%s' to be friend",
        !          2882:                                      IDENTIFIER_POINTER (DECL_NAME (TREE_TYPE (waiters))),
        !          2883:                                      TYPE_NAME_STRING (type));
        !          2884:        }
        !          2885:       else
        !          2886:        make_friend_class (type, waiter);
        !          2887: 
        !          2888:       if (TREE_CHAIN (waiter_prev))
        !          2889:        TREE_CHAIN (waiter_prev) = TREE_CHAIN (TREE_CHAIN (waiter_prev));
        !          2890:       else
        !          2891:        DECL_UNDEFINED_FRIENDS (TYPE_NAME (waiter)) = NULL_TREE;
        !          2892:     }
        !          2893: }
        !          2894: 
        !          2895: /* Common subroutines of build_new and build_vec_delete.  */
        !          2896: 
        !          2897: /* Common interface for calling "builtin" functions that are not
        !          2898:    really builtin.  */
        !          2899: 
        !          2900: tree
        !          2901: build_builtin_call (type, node, arglist)
        !          2902:      tree type;
        !          2903:      tree node;
        !          2904:      tree arglist;
        !          2905: {
        !          2906:   tree rval = build (CALL_EXPR, type, node, arglist, 0);
        !          2907:   TREE_SIDE_EFFECTS (rval) = 1;
        !          2908:   assemble_external (TREE_OPERAND (node, 0));
        !          2909:   TREE_USED (TREE_OPERAND (node, 0)) = 1;
        !          2910:   return rval;
        !          2911: }
        !          2912: 
        !          2913: /* Generate a C++ "new" expression. DECL is either a TREE_LIST
        !          2914:    (which needs to go through some sort of groktypename) or it
        !          2915:    is the name of the class we are newing. INIT is an initialization value.
        !          2916:    It is either an EXPRLIST, an EXPR_NO_COMMAS, or something in braces.
        !          2917:    If INIT is void_type_node, it means do *not* call a constructor
        !          2918:    for this instance.
        !          2919: 
        !          2920:    For types with constructors, the data returned is initialized
        !          2921:    by the appropriate constructor.
        !          2922: 
        !          2923:    Whether the type has a constructor or not, if it has a pointer
        !          2924:    to a virtual function table, then that pointer is set up
        !          2925:    here.
        !          2926: 
        !          2927:    Unless I am mistaken, a call to new () will return initialized
        !          2928:    data regardless of whether the constructor itself is private or
        !          2929:    not.
        !          2930: 
        !          2931:    Note that build_new does nothing to assure that any special
        !          2932:    alignment requirements of the type are met.  Rather, it leaves
        !          2933:    it up to malloc to do the right thing.  Otherwise, folding to
        !          2934:    the right alignment cal cause problems if the user tries to later
        !          2935:    free the memory returned by `new'.
        !          2936: 
        !          2937:    PLACEMENT is the `placement' list for user-defined operator new ().  */
        !          2938: 
        !          2939: tree
        !          2940: build_new (placement, decl, init, use_global_new)
        !          2941:      tree placement;
        !          2942:      tree decl, init;
        !          2943:      int use_global_new;
        !          2944: {
        !          2945:   tree type, true_type, size, rval;
        !          2946:   tree init1 = NULL_TREE, nelts;
        !          2947:   int has_call = 0, has_array = 0;
        !          2948: 
        !          2949:   tree pending_sizes = NULL_TREE;
        !          2950: 
        !          2951:   if (decl == error_mark_node)
        !          2952:     return error_mark_node;
        !          2953: 
        !          2954:   if (TREE_CODE (decl) == TREE_LIST)
        !          2955:     {
        !          2956:       tree absdcl = TREE_VALUE (decl);
        !          2957:       tree last_absdcl = NULL_TREE;
        !          2958:       int old_immediate_size_expand;
        !          2959: 
        !          2960:       if (current_function_decl
        !          2961:          && DECL_CONSTRUCTOR_P (current_function_decl))
        !          2962:        {
        !          2963:          old_immediate_size_expand = immediate_size_expand;
        !          2964:          immediate_size_expand = 0;
        !          2965:        }
        !          2966: 
        !          2967:       nelts = integer_one_node;
        !          2968: 
        !          2969:       if (absdcl && TREE_CODE (absdcl) == CALL_EXPR)
        !          2970:        {
        !          2971:          /* probably meant to be a call */
        !          2972:          has_call = 1;
        !          2973:          init1 = TREE_OPERAND (absdcl, 1);
        !          2974:          absdcl = TREE_OPERAND (absdcl, 0);
        !          2975:          TREE_VALUE (decl) = absdcl;
        !          2976:        }
        !          2977:       while (absdcl && TREE_CODE (absdcl) == INDIRECT_REF)
        !          2978:        {
        !          2979:          last_absdcl = absdcl;
        !          2980:          absdcl = TREE_OPERAND (absdcl, 0);
        !          2981:        }
        !          2982: 
        !          2983:       if (absdcl && TREE_CODE (absdcl) == ARRAY_REF)
        !          2984:        {
        !          2985:          /* probably meant to be a vec new */
        !          2986:          tree this_nelts;
        !          2987: 
        !          2988:          has_array = 1;
        !          2989:          this_nelts = TREE_OPERAND (absdcl, 1);
        !          2990:          if (this_nelts != error_mark_node)
        !          2991:            {
        !          2992:              if (this_nelts == NULL_TREE)
        !          2993:                error ("new of array type fails to specify size");
        !          2994:              else
        !          2995:                {
        !          2996:                  this_nelts = save_expr (this_nelts);
        !          2997:                  absdcl = TREE_OPERAND (absdcl, 0);
        !          2998:                  if (this_nelts == integer_zero_node)
        !          2999:                    {
        !          3000:                      warning ("zero size array reserves no space");
        !          3001:                      nelts = integer_zero_node;
        !          3002:                    }
        !          3003:                  else
        !          3004:                    nelts = build_binary_op (MULT_EXPR, nelts, this_nelts, 1);
        !          3005:                }
        !          3006:            }
        !          3007:          else
        !          3008:            nelts = integer_zero_node;
        !          3009:        }
        !          3010: 
        !          3011:       if (last_absdcl)
        !          3012:        TREE_OPERAND (last_absdcl, 0) = absdcl;
        !          3013:       else
        !          3014:        TREE_VALUE (decl) = absdcl;
        !          3015: 
        !          3016:       type = true_type = groktypename (decl);
        !          3017:       if (! type || type == error_mark_node
        !          3018:          || true_type == error_mark_node)
        !          3019:        return error_mark_node;
        !          3020: 
        !          3021:       /* ``A reference cannot be created by the new operator.  A reference
        !          3022:         is not an object (8.2.2, 8.4.3), so a pointer to it could not be
        !          3023:         returned by new.'' ARM 5.3.3 */
        !          3024:       if (TREE_CODE (type) == REFERENCE_TYPE)
        !          3025:        error ("new cannot be applied to a reference type");
        !          3026: 
        !          3027:       type = TYPE_MAIN_VARIANT (type);
        !          3028:       if (type == void_type_node)
        !          3029:        {
        !          3030:          error ("invalid type: `void []'");
        !          3031:          return error_mark_node;
        !          3032:        }
        !          3033:       if (current_function_decl
        !          3034:          && DECL_CONSTRUCTOR_P (current_function_decl))
        !          3035:        {
        !          3036:          pending_sizes = get_pending_sizes ();
        !          3037:          immediate_size_expand = old_immediate_size_expand;
        !          3038:        }
        !          3039:     }
        !          3040:   else if (TREE_CODE (decl) == IDENTIFIER_NODE)
        !          3041:     {
        !          3042:       if (IDENTIFIER_HAS_TYPE_VALUE (decl))
        !          3043:        {
        !          3044:          /* An aggregate type.  */
        !          3045:          type = IDENTIFIER_TYPE_VALUE (decl);
        !          3046:          decl = TYPE_NAME (type);
        !          3047:        }
        !          3048:       else
        !          3049:        {
        !          3050:          /* A builtin type.  */
        !          3051:          decl = lookup_name (decl, 1);
        !          3052:          my_friendly_assert (TREE_CODE (decl) == TYPE_DECL, 215);
        !          3053:          type = TREE_TYPE (decl);
        !          3054:        }
        !          3055:       true_type = type;
        !          3056:     }
        !          3057:   else if (TREE_CODE (decl) == TYPE_DECL)
        !          3058:     {
        !          3059:       type = TREE_TYPE (decl);
        !          3060:       true_type = type;
        !          3061:     }
        !          3062:   else
        !          3063:     {
        !          3064:       type = decl;
        !          3065:       true_type = type;
        !          3066:       decl = TYPE_NAME (type);
        !          3067:     }
        !          3068: 
        !          3069:   if (TYPE_SIZE (type) == 0)
        !          3070:     {
        !          3071:       if (type == void_type_node)
        !          3072:        error ("invalid type for new: `void'");
        !          3073:       else
        !          3074:        incomplete_type_error (0, type);
        !          3075:       return error_mark_node;
        !          3076:     }
        !          3077: 
        !          3078:   if (TYPE_LANG_SPECIFIC (type) && CLASSTYPE_ABSTRACT_VIRTUALS (type))
        !          3079:     {
        !          3080:       abstract_virtuals_error (NULL_TREE, type);
        !          3081:       return error_mark_node;
        !          3082:     }
        !          3083: 
        !          3084:   /* If our base type is an array, then make sure we know how many elements
        !          3085:      it has.  */
        !          3086:   while (TREE_CODE (type) == ARRAY_TYPE)
        !          3087:     {
        !          3088:       tree this_nelts = array_type_nelts_top (type);
        !          3089:       if (nelts == integer_one_node)
        !          3090:        {
        !          3091:          has_array = 1;
        !          3092:          nelts = this_nelts;
        !          3093:        }
        !          3094:       else
        !          3095:        {
        !          3096:          my_friendly_assert (has_array != 0, 216);
        !          3097:          nelts = build_binary_op (MULT_EXPR, nelts, this_nelts, 1);
        !          3098:        }
        !          3099:       type = TREE_TYPE (type);
        !          3100:     }
        !          3101:   if (has_array)
        !          3102:     size = fold (build_binary_op (MULT_EXPR, size_in_bytes (type), nelts, 1));
        !          3103:   else
        !          3104:     size = size_in_bytes (type);
        !          3105: 
        !          3106:   if (has_call)
        !          3107:     init = init1;
        !          3108: 
        !          3109:   /* Get to the target type of TRUE_TYPE, so we can decide whether
        !          3110:      any constructors need to be called or not.  */
        !          3111:   type = true_type;
        !          3112:   while (TREE_CODE (type) == ARRAY_TYPE)
        !          3113:     type = TREE_TYPE (type);
        !          3114: 
        !          3115:   /* Get a little extra space to store a couple of things before the new'ed
        !          3116:      array. */
        !          3117:   if (has_array && TYPE_NEEDS_DESTRUCTOR (true_type))
        !          3118:     {
        !          3119:       tree extra = BI_header_size;
        !          3120: 
        !          3121:       size = size_binop (PLUS_EXPR, size, extra);
        !          3122:     }
        !          3123: 
        !          3124:   /* Allocate the object. */
        !          3125:   if (TYPE_LANG_SPECIFIC (true_type)
        !          3126:       && (TREE_GETS_NEW (true_type) && !use_global_new))
        !          3127:     rval = build_opfncall (NEW_EXPR, LOOKUP_NORMAL,
        !          3128:                           TYPE_POINTER_TO (true_type), size, placement);
        !          3129:   else if (placement)
        !          3130:     {
        !          3131:       rval = build_opfncall (NEW_EXPR, LOOKUP_GLOBAL|LOOKUP_COMPLAIN,
        !          3132:                             ptr_type_node, size, placement);
        !          3133:       rval = convert (TYPE_POINTER_TO (true_type), rval);
        !          3134:     }
        !          3135:   else if (flag_this_is_variable > 0
        !          3136:           && TYPE_HAS_CONSTRUCTOR (true_type) && init != void_type_node)
        !          3137:     {
        !          3138:       if (init == NULL_TREE || TREE_CODE (init) == TREE_LIST)
        !          3139:        rval = NULL_TREE;
        !          3140:       else
        !          3141:        {
        !          3142:          error ("constructors take parameter lists");
        !          3143:          return error_mark_node;
        !          3144:        }
        !          3145:     }
        !          3146:   else
        !          3147:     {
        !          3148:       rval = build_builtin_call (build_pointer_type (true_type),
        !          3149:                                 BIN, build_tree_list (NULL_TREE, size));
        !          3150: #if 0
        !          3151:       /* See comment above as to why this is disabled.  */
        !          3152:       if (alignment)
        !          3153:        {
        !          3154:          rval = build (PLUS_EXPR, TYPE_POINTER_TO (true_type), rval,
        !          3155:                        alignment);
        !          3156:          rval = build (BIT_AND_EXPR, TYPE_POINTER_TO (true_type),
        !          3157:                        rval, build1 (BIT_NOT_EXPR, integer_type_node,
        !          3158:                                      alignment));
        !          3159:        }
        !          3160: #endif
        !          3161:       TREE_CALLS_NEW (rval) = 1;
        !          3162:       TREE_SIDE_EFFECTS (rval) = 1;
        !          3163:     }
        !          3164: 
        !          3165:   /* if rval is NULL_TREE I don't have to allocate it, but are we totally
        !          3166:      sure we have some extra bytes in that case for the BI_header_size
        !          3167:      cookies? And how does that interact with the code below? (mrs) */
        !          3168:   /* Finish up some magic for new'ed arrays */
        !          3169:   if (has_array && TYPE_NEEDS_DESTRUCTOR (true_type) && rval != NULL_TREE)
        !          3170:     {
        !          3171:       tree extra = BI_header_size;
        !          3172:       tree cookie, exp1;
        !          3173:       rval = convert (ptr_type_node, rval);    /* convert to void * first */
        !          3174:       rval = convert (string_type_node, rval); /* lets not add void* and ints */
        !          3175:       rval = save_expr (build_binary_op (PLUS_EXPR, rval, extra, 1));
        !          3176:       /* Store header info.  */
        !          3177:       cookie = build_indirect_ref (build (MINUS_EXPR, TYPE_POINTER_TO (BI_header_type),
        !          3178:                                          rval, extra), NULL_PTR);
        !          3179:       exp1 = build (MODIFY_EXPR, void_type_node,
        !          3180:                    build_component_ref (cookie, nc_nelts_field_id, 0, 0),
        !          3181:                    nelts);
        !          3182:       TREE_SIDE_EFFECTS (exp1) = 1;
        !          3183:       rval = convert (build_pointer_type (true_type), rval);
        !          3184:       TREE_CALLS_NEW (rval) = 1;
        !          3185:       TREE_SIDE_EFFECTS (rval) = 1;
        !          3186:       rval = build_compound_expr (tree_cons (NULL_TREE, exp1,
        !          3187:                                             build_tree_list (NULL_TREE, rval)));
        !          3188:     }
        !          3189: 
        !          3190:   /* We've figured out where the allocation is to go.
        !          3191:      If we're not eliding constructors, then if a constructor
        !          3192:      is defined, we must go through it.  */
        !          3193:   if (!has_array && (rval == NULL_TREE || !flag_elide_constructors)
        !          3194:       && TYPE_HAS_CONSTRUCTOR (true_type) && init != void_type_node)
        !          3195:     {
        !          3196:       tree newrval;
        !          3197:       /* Constructors are never virtual. If it has an initialization, we
        !          3198:         need to complain if we aren't allowed to use the ctor that took
        !          3199:         that argument.  */
        !          3200:       int flags = LOOKUP_NORMAL|LOOKUP_NONVIRTUAL|LOOKUP_COMPLAIN;
        !          3201: 
        !          3202:       /* If a copy constructor might work, set things up so that we can
        !          3203:         try that after this.  We deliberately don't clear LOOKUP_COMPLAIN
        !          3204:         any more, since that would make it impossible to rationally use
        !          3205:         the visibility of a constructor that matches perfectly.  */
        !          3206:       if (rval != NULL_TREE)
        !          3207:        flags |= LOOKUP_SPECULATIVELY;
        !          3208: 
        !          3209:       if (rval && TYPE_USES_VIRTUAL_BASECLASSES (true_type))
        !          3210:        {
        !          3211:          init = tree_cons (NULL_TREE, integer_one_node, init);
        !          3212:          flags |= LOOKUP_HAS_IN_CHARGE;
        !          3213:        }
        !          3214: 
        !          3215:       newrval = build_method_call (rval, constructor_name_full (true_type),
        !          3216:                                init, NULL_TREE, flags);
        !          3217:       if (newrval)
        !          3218:        {
        !          3219:          rval = newrval;
        !          3220:          TREE_HAS_CONSTRUCTOR (rval) = 1;
        !          3221:          goto done;
        !          3222:        }
        !          3223:       /* Didn't find the constructor, maybe it is a call to a copy constructor
        !          3224:         that we should implement. */
        !          3225:     }
        !          3226: 
        !          3227:   if (rval == error_mark_node)
        !          3228:     return error_mark_node;
        !          3229:   rval = save_expr (rval);
        !          3230:   TREE_HAS_CONSTRUCTOR (rval) = 1;
        !          3231: 
        !          3232:   /* Don't call any constructors or do any initialization.  */
        !          3233:   if (init == void_type_node)
        !          3234:     goto done;
        !          3235: 
        !          3236:   if (TYPE_NEEDS_CONSTRUCTING (type)
        !          3237:       || (has_call || init))
        !          3238:     {
        !          3239:       extern tree static_aggregates;
        !          3240: 
        !          3241:       if (current_function_decl == NULL_TREE)
        !          3242:        {
        !          3243:          /* In case of static initialization, SAVE_EXPR is good enough.  */
        !          3244:          init = copy_to_permanent (init);
        !          3245:          rval = copy_to_permanent (rval);
        !          3246:          static_aggregates = perm_tree_cons (init, rval, static_aggregates);
        !          3247:        }
        !          3248:       else
        !          3249:        {
        !          3250:          /* Have to wrap this in RTL_EXPR for two cases:
        !          3251:             in base or member initialization and if we
        !          3252:             are a branch of a ?: operator.  Since we
        !          3253:             can't easily know the latter, just do it always.  */
        !          3254:          tree xval = make_node (RTL_EXPR);
        !          3255: 
        !          3256:          TREE_TYPE (xval) = TREE_TYPE (rval);
        !          3257:          do_pending_stack_adjust ();
        !          3258:          start_sequence_for_rtl_expr (xval);
        !          3259: 
        !          3260:          /* As a matter of principle, `start_sequence' should do this.  */
        !          3261:          emit_note (0, -1);
        !          3262: 
        !          3263:          if (has_array)
        !          3264:            rval = expand_vec_init (decl, rval,
        !          3265:                                    build_binary_op (MINUS_EXPR, nelts, integer_one_node, 1),
        !          3266:                                    init, 0);
        !          3267:          else
        !          3268:            expand_aggr_init (build_indirect_ref (rval, NULL_PTR), init, 0);
        !          3269: 
        !          3270:          do_pending_stack_adjust ();
        !          3271: 
        !          3272:          TREE_SIDE_EFFECTS (xval) = 1;
        !          3273:          TREE_CALLS_NEW (xval) = 1;
        !          3274:          RTL_EXPR_SEQUENCE (xval) = get_insns ();
        !          3275:          end_sequence ();
        !          3276: 
        !          3277:          if (TREE_CODE (rval) == SAVE_EXPR)
        !          3278:            {
        !          3279:              /* Errors may cause this to not get evaluated.  */
        !          3280:              if (SAVE_EXPR_RTL (rval) == 0)
        !          3281:                SAVE_EXPR_RTL (rval) = const0_rtx;
        !          3282:              RTL_EXPR_RTL (xval) = SAVE_EXPR_RTL (rval);
        !          3283:            }
        !          3284:          else
        !          3285:            {
        !          3286:              my_friendly_assert (TREE_CODE (rval) == VAR_DECL, 217);
        !          3287:              RTL_EXPR_RTL (xval) = DECL_RTL (rval);
        !          3288:            }
        !          3289:          rval = xval;
        !          3290:        }
        !          3291:     }
        !          3292: #if 0
        !          3293:   /* It would seem that the above code handles this better than the code
        !          3294:      below. (mrs) */
        !          3295:   else if (has_call || init)
        !          3296:     {
        !          3297:       if (IS_AGGR_TYPE (type))
        !          3298:        {
        !          3299:          /*  default copy constructor may be missing from the below. (mrs) */
        !          3300:          cp_error ("no constructor for type `%T'", type);
        !          3301:          rval = error_mark_node;
        !          3302:        }
        !          3303:       else
        !          3304:        {
        !          3305:          /* New 2.0 interpretation: `new int (10)' means
        !          3306:             allocate an int, and initialize it with 10.  */
        !          3307: 
        !          3308:          init = build_c_cast (type, init);
        !          3309:          rval = build (COMPOUND_EXPR, TREE_TYPE (rval),
        !          3310:                        build_modify_expr (build_indirect_ref (rval, NULL_PTR),
        !          3311:                                           NOP_EXPR, init),
        !          3312:                        rval);
        !          3313:          TREE_SIDE_EFFECTS (rval) = 1;
        !          3314:        }
        !          3315:     }
        !          3316: #endif
        !          3317:  done:
        !          3318:   if (pending_sizes)
        !          3319:     rval = build_compound_expr (chainon (pending_sizes,
        !          3320:                                         build_tree_list (NULL_TREE, rval)));
        !          3321: 
        !          3322:   if (flag_gc)
        !          3323:     {
        !          3324:       extern tree gc_visible;
        !          3325:       tree objbits;
        !          3326:       tree update_expr;
        !          3327: 
        !          3328:       rval = save_expr (rval);
        !          3329:       /* We don't need a `headof' operation to do this because
        !          3330:         we know where the object starts.  */
        !          3331:       objbits = build1 (INDIRECT_REF, unsigned_type_node,
        !          3332:                        build (MINUS_EXPR, ptr_type_node,
        !          3333:                               rval, c_sizeof_nowarn (unsigned_type_node)));
        !          3334:       update_expr = build_modify_expr (objbits, BIT_IOR_EXPR, gc_visible);
        !          3335:       rval = build_compound_expr (tree_cons (NULL_TREE, rval,
        !          3336:                                             tree_cons (NULL_TREE, update_expr,
        !          3337:                                                        build_tree_list (NULL_TREE, rval))));
        !          3338:     }
        !          3339: 
        !          3340:   return save_expr (rval);
        !          3341: }
        !          3342: 
        !          3343: /* `expand_vec_init' performs initialization of a vector of aggregate
        !          3344:    types.
        !          3345: 
        !          3346:    DECL is passed only for error reporting, and provides line number
        !          3347:    and source file name information.
        !          3348:    BASE is the space where the vector will be.
        !          3349:    MAXINDEX is the maximum index of the array (one less than the
        !          3350:            number of elements).
        !          3351:    INIT is the (possibly NULL) initializer.
        !          3352: 
        !          3353:    FROM_ARRAY is 0 if we should init everything with INIT
        !          3354:    (i.e., every element initialized from INIT).
        !          3355:    FROM_ARRAY is 1 if we should index into INIT in parallel
        !          3356:    with initialization of DECL.
        !          3357:    FROM_ARRAY is 2 if we should index into INIT in parallel,
        !          3358:    but use assignment instead of initialization.  */
        !          3359: 
        !          3360: tree
        !          3361: expand_vec_init (decl, base, maxindex, init, from_array)
        !          3362:      tree decl, base, maxindex, init;
        !          3363:      int from_array;
        !          3364: {
        !          3365:   tree rval;
        !          3366:   tree iterator, base2 = NULL_TREE;
        !          3367:   tree type = TREE_TYPE (TREE_TYPE (base));
        !          3368:   tree size;
        !          3369: 
        !          3370:   maxindex = convert (integer_type_node, maxindex);
        !          3371:   if (maxindex == error_mark_node)
        !          3372:     return error_mark_node;
        !          3373: 
        !          3374:   if (current_function_decl == NULL_TREE)
        !          3375:     {
        !          3376:       rval = make_tree_vec (3);
        !          3377:       TREE_VEC_ELT (rval, 0) = base;
        !          3378:       TREE_VEC_ELT (rval, 1) = maxindex;
        !          3379:       TREE_VEC_ELT (rval, 2) = init;
        !          3380:       return rval;
        !          3381:     }
        !          3382: 
        !          3383:   size = size_in_bytes (type);
        !          3384: 
        !          3385:   /* Set to zero in case size is <= 0.  Optimizer will delete this if
        !          3386:      it is not needed.  */
        !          3387:   rval = get_temp_regvar (TYPE_POINTER_TO (type),
        !          3388:                          convert (TYPE_POINTER_TO (type), null_pointer_node));
        !          3389:   base = default_conversion (base);
        !          3390:   base = convert (TYPE_POINTER_TO (type), base);
        !          3391:   expand_assignment (rval, base, 0, 0);
        !          3392:   base = get_temp_regvar (TYPE_POINTER_TO (type), base);
        !          3393: 
        !          3394:   if (init != NULL_TREE
        !          3395:       && TREE_CODE (init) == CONSTRUCTOR
        !          3396:       && TREE_TYPE (init) == TREE_TYPE (decl))
        !          3397:     {
        !          3398:       /* Initialization of array from {...}.  */
        !          3399:       tree elts = CONSTRUCTOR_ELTS (init);
        !          3400:       tree baseref = build1 (INDIRECT_REF, type, base);
        !          3401:       tree baseinc = build (PLUS_EXPR, TYPE_POINTER_TO (type), base, size);
        !          3402:       int host_i = TREE_INT_CST_LOW (maxindex);
        !          3403: 
        !          3404:       if (IS_AGGR_TYPE (type))
        !          3405:        {
        !          3406:          while (elts)
        !          3407:            {
        !          3408:              host_i -= 1;
        !          3409:              expand_aggr_init (baseref, TREE_VALUE (elts), 0);
        !          3410: 
        !          3411:              expand_assignment (base, baseinc, 0, 0);
        !          3412:              elts = TREE_CHAIN (elts);
        !          3413:            }
        !          3414:          /* Initialize any elements by default if possible.  */
        !          3415:          if (host_i >= 0)
        !          3416:            {
        !          3417:              if (TYPE_NEEDS_CONSTRUCTING (type) == 0)
        !          3418:                {
        !          3419:                  if (obey_regdecls)
        !          3420:                    use_variable (DECL_RTL (base));
        !          3421:                  goto done_init;
        !          3422:                }
        !          3423: 
        !          3424:              iterator = get_temp_regvar (integer_type_node,
        !          3425:                                          build_int_2 (host_i, 0));
        !          3426:              init = NULL_TREE;
        !          3427:              goto init_by_default;
        !          3428:            }
        !          3429:        }
        !          3430:       else
        !          3431:        while (elts)
        !          3432:          {
        !          3433:            expand_assignment (baseref, TREE_VALUE (elts), 0, 0);
        !          3434: 
        !          3435:            expand_assignment (base, baseinc, 0, 0);
        !          3436:            elts = TREE_CHAIN (elts);
        !          3437:          }
        !          3438: 
        !          3439:       if (obey_regdecls)
        !          3440:        use_variable (DECL_RTL (base));
        !          3441:     }
        !          3442:   else
        !          3443:     {
        !          3444:       iterator = get_temp_regvar (integer_type_node, maxindex);
        !          3445: 
        !          3446:     init_by_default:
        !          3447: 
        !          3448:       /* If initializing one array from another,
        !          3449:         initialize element by element.  */
        !          3450:       if (from_array)
        !          3451:        {
        !          3452:          if (decl == NULL_TREE
        !          3453:              || (init && !comptypes (TREE_TYPE (init), TREE_TYPE (decl), 1)))
        !          3454:            {
        !          3455:              sorry ("initialization of array from dissimilar array type");
        !          3456:              return error_mark_node;
        !          3457:            }
        !          3458:          if (init)
        !          3459:            {
        !          3460:              base2 = default_conversion (init);
        !          3461:              base2 = get_temp_regvar (TYPE_POINTER_TO (type), base2);
        !          3462:            }
        !          3463:          else if (TYPE_LANG_SPECIFIC (type)
        !          3464:                   && TYPE_NEEDS_CONSTRUCTING (type)
        !          3465:                   && ! TYPE_HAS_DEFAULT_CONSTRUCTOR (type))
        !          3466:            {
        !          3467:              error ("initializer ends prematurely");
        !          3468:              return error_mark_node;
        !          3469:            }
        !          3470:        }
        !          3471: 
        !          3472:       expand_start_cond (build (GE_EXPR, integer_type_node,
        !          3473:                                iterator, integer_zero_node), 0);
        !          3474:       expand_start_loop_continue_elsewhere (1);
        !          3475: 
        !          3476:       if (from_array)
        !          3477:        {
        !          3478:          tree to = build1 (INDIRECT_REF, type, base);
        !          3479:          tree from;
        !          3480: 
        !          3481:          if (base2)
        !          3482:            from = build1 (INDIRECT_REF, type, base2);
        !          3483:          else
        !          3484:            from = NULL_TREE;
        !          3485: 
        !          3486:          if (from_array == 2)
        !          3487:            expand_expr_stmt (build_modify_expr (to, NOP_EXPR, from));
        !          3488:          else if (TYPE_NEEDS_CONSTRUCTING (type))
        !          3489:            expand_aggr_init (to, from, 0);
        !          3490:          else if (from)
        !          3491:            expand_assignment (to, from, 0, 0);
        !          3492:          else
        !          3493:            my_friendly_abort (57);
        !          3494:        }
        !          3495:       else if (TREE_CODE (type) == ARRAY_TYPE)
        !          3496:        {
        !          3497:          if (init != 0)
        !          3498:            sorry ("cannot initialize multi-dimensional array with initializer");
        !          3499:          expand_vec_init (decl, build1 (NOP_EXPR, TYPE_POINTER_TO (TREE_TYPE (type)), base),
        !          3500:                           array_type_nelts (type), 0, 0);
        !          3501:        }
        !          3502:       else
        !          3503:        expand_aggr_init (build1 (INDIRECT_REF, type, base), init, 0);
        !          3504: 
        !          3505:       expand_assignment (base,
        !          3506:                         build (PLUS_EXPR, TYPE_POINTER_TO (type), base, size),
        !          3507:                         0, 0);
        !          3508:       if (base2)
        !          3509:        expand_assignment (base2,
        !          3510:                           build (PLUS_EXPR, TYPE_POINTER_TO (type), base2, size), 0, 0);
        !          3511:       expand_loop_continue_here ();
        !          3512:       expand_exit_loop_if_false (0, build (NE_EXPR, integer_type_node,
        !          3513:                                           build (PREDECREMENT_EXPR, integer_type_node, iterator, integer_one_node), minus_one));
        !          3514: 
        !          3515:       if (obey_regdecls)
        !          3516:        {
        !          3517:          use_variable (DECL_RTL (base));
        !          3518:          if (base2)
        !          3519:            use_variable (DECL_RTL (base2));
        !          3520:        }
        !          3521:       expand_end_loop ();
        !          3522:       expand_end_cond ();
        !          3523:       if (obey_regdecls)
        !          3524:        use_variable (DECL_RTL (iterator));
        !          3525:     }
        !          3526:  done_init:
        !          3527: 
        !          3528:   if (obey_regdecls)
        !          3529:     use_variable (DECL_RTL (rval));
        !          3530:   return rval;
        !          3531: }
        !          3532: 
        !          3533: /* Free up storage of type TYPE, at address ADDR.
        !          3534: 
        !          3535:    TYPE is a POINTER_TYPE and can be ptr_type_node for no special type
        !          3536:    of pointer.
        !          3537: 
        !          3538:    VIRTUAL_SIZE is the amount of storage that was allocated, and is
        !          3539:    used as the second argument to operator delete.  It can include
        !          3540:    things like padding and magic size cookies.  It has virtual in it,
        !          3541:    because if you have a base pointer and you delete through a virtual
        !          3542:    destructor, it should be the size of the dynamic object, not the
        !          3543:    static object, see Free Store 12.5 ANSI C++ WP.
        !          3544: 
        !          3545:    This does not call any destructors.  */
        !          3546: tree
        !          3547: build_x_delete (type, addr, use_global_delete, virtual_size)
        !          3548:      tree type, addr;
        !          3549:      int use_global_delete;
        !          3550:      tree virtual_size;
        !          3551: {
        !          3552:   tree rval;
        !          3553: 
        !          3554:   if (!use_global_delete
        !          3555:       && TYPE_LANG_SPECIFIC (TREE_TYPE (type))
        !          3556:       && TREE_GETS_DELETE (TREE_TYPE (type)))
        !          3557:     rval = build_opfncall (DELETE_EXPR, LOOKUP_NORMAL, addr,
        !          3558:                           virtual_size, NULL_TREE);
        !          3559:   else
        !          3560:     rval = build_builtin_call (void_type_node, BID,
        !          3561:                               build_tree_list (NULL_TREE, addr));
        !          3562:   return rval;
        !          3563: }
        !          3564: 
        !          3565: /* Generate a call to a destructor. TYPE is the type to cast ADDR to.
        !          3566:    ADDR is an expression which yields the store to be destroyed.
        !          3567:    AUTO_DELETE is nonzero if a call to DELETE should be made or not.
        !          3568:    If in the program, (AUTO_DELETE & 2) is non-zero, we tear down the
        !          3569:    virtual baseclasses.
        !          3570:    If in the program, (AUTO_DELETE & 1) is non-zero, then we deallocate.
        !          3571: 
        !          3572:    FLAGS is the logical disjunction of zero or more LOOKUP_
        !          3573:    flags.  See cp-tree.h for more info.
        !          3574: 
        !          3575:    This function does not delete an object's virtual base classes.  */
        !          3576: tree
        !          3577: build_delete (type, addr, auto_delete, flags, use_global_delete)
        !          3578:      tree type, addr;
        !          3579:      tree auto_delete;
        !          3580:      int flags;
        !          3581:      int use_global_delete;
        !          3582: {
        !          3583:   tree function, parms;
        !          3584:   tree member;
        !          3585:   tree expr;
        !          3586:   tree ref;
        !          3587:   int ptr;
        !          3588: 
        !          3589:   if (addr == error_mark_node)
        !          3590:     return error_mark_node;
        !          3591: 
        !          3592:   /* Can happen when CURRENT_EXCEPTION_OBJECT gets its type
        !          3593:      set to `error_mark_node' before it gets properly cleaned up.  */
        !          3594:   if (type == error_mark_node)
        !          3595:     return error_mark_node;
        !          3596: 
        !          3597:   type = TYPE_MAIN_VARIANT (type);
        !          3598: 
        !          3599:   if (TREE_CODE (type) == POINTER_TYPE)
        !          3600:     {
        !          3601:       type = TREE_TYPE (type);
        !          3602:       if (TYPE_SIZE (type) == 0)
        !          3603:        {
        !          3604:          incomplete_type_error (0, type);
        !          3605:          return error_mark_node;
        !          3606:        }
        !          3607:       if (TREE_CODE (type) == ARRAY_TYPE)
        !          3608:        goto handle_array;
        !          3609:       if (! IS_AGGR_TYPE (type))
        !          3610:        {
        !          3611:          /* Call the builtin operator delete.  */
        !          3612:          return build_builtin_call (void_type_node, BID,
        !          3613:                                     build_tree_list (NULL_TREE, addr));
        !          3614:        }
        !          3615:       if (TREE_SIDE_EFFECTS (addr))
        !          3616:        addr = save_expr (addr);
        !          3617:       ref = build_indirect_ref (addr, NULL_PTR);
        !          3618:       ptr = 1;
        !          3619:     }
        !          3620:   else if (TREE_CODE (type) == ARRAY_TYPE)
        !          3621:     {
        !          3622:     handle_array:
        !          3623:       if (TREE_SIDE_EFFECTS (addr))
        !          3624:        addr = save_expr (addr);
        !          3625:       return build_vec_delete (addr, array_type_nelts (type),
        !          3626:                               c_sizeof_nowarn (TREE_TYPE (type)),
        !          3627:                               NULL_TREE, auto_delete, integer_two_node);
        !          3628:     }
        !          3629:   else
        !          3630:     {
        !          3631:       /* Don't check PROTECT here; leave that decision to the
        !          3632:         destructor.  If the destructor is visible, call it,
        !          3633:         else report error.  */
        !          3634:       addr = build_unary_op (ADDR_EXPR, addr, 0);
        !          3635:       if (TREE_SIDE_EFFECTS (addr))
        !          3636:        addr = save_expr (addr);
        !          3637: 
        !          3638:       if (TREE_CONSTANT (addr))
        !          3639:        addr = convert_pointer_to (type, addr);
        !          3640:       else
        !          3641:        addr = convert_force (build_pointer_type (type), addr);
        !          3642: 
        !          3643:       if (TREE_CODE (addr) == NOP_EXPR
        !          3644:          && TREE_OPERAND (addr, 0) == current_class_decl)
        !          3645:        ref = C_C_D;
        !          3646:       else
        !          3647:        ref = build_indirect_ref (addr, NULL_PTR);
        !          3648:       ptr = 0;
        !          3649:     }
        !          3650: 
        !          3651:   my_friendly_assert (IS_AGGR_TYPE (type), 220);
        !          3652: 
        !          3653:   if (! TYPE_NEEDS_DESTRUCTOR (type))
        !          3654:     {
        !          3655:       tree virtual_size;
        !          3656: 
        !          3657:       if (auto_delete == integer_zero_node)
        !          3658:        return void_zero_node;
        !          3659: 
        !          3660:       /* Pass the size of the object down to the operator delete() in
        !          3661:         addition to the ADDR.  */
        !          3662:       if (TREE_GETS_DELETE (type) && !use_global_delete)
        !          3663:        {
        !          3664:          /* This is probably wrong. It should be the size of the virtual
        !          3665:             object being deleted.  */
        !          3666:          tree virtual_size = c_sizeof_nowarn (type);
        !          3667:          return build_opfncall (DELETE_EXPR, LOOKUP_NORMAL, addr,
        !          3668:                                 virtual_size, NULL_TREE);
        !          3669:        }
        !          3670: 
        !          3671:       /* Call the builtin operator delete.  */
        !          3672:       return build_builtin_call (void_type_node, BID,
        !          3673:                                 build_tree_list (NULL_TREE, addr));
        !          3674:     }
        !          3675:   parms = build_tree_list (NULL_TREE, addr);
        !          3676: 
        !          3677:   /* Below, we will reverse the order in which these calls are made.
        !          3678:      If we have a destructor, then that destructor will take care
        !          3679:      of the base classes; otherwise, we must do that here.  */
        !          3680:   if (TYPE_HAS_DESTRUCTOR (type))
        !          3681:     {
        !          3682:       tree dtor = DECL_MAIN_VARIANT (TREE_VEC_ELT (CLASSTYPE_METHOD_VEC (type), 0));
        !          3683:       tree basetypes = TYPE_BINFO (type);
        !          3684: 
        !          3685:       if (flags & LOOKUP_PROTECT)
        !          3686:        {
        !          3687:          enum visibility_type visibility = compute_visibility (basetypes, dtor);
        !          3688: 
        !          3689:          if (visibility == visibility_private)
        !          3690:            {
        !          3691:              if (flags & LOOKUP_COMPLAIN)
        !          3692:                cp_error ("destructor for type `%T' is private in this scope", type);
        !          3693:              return error_mark_node;
        !          3694:            }
        !          3695:          else if (visibility == visibility_protected
        !          3696:                   && (flags & LOOKUP_PROTECTED_OK) == 0)
        !          3697:            {
        !          3698:              if (flags & LOOKUP_COMPLAIN)
        !          3699:                cp_error ("destructor for type `%T' is protected in this scope", type);
        !          3700:              return error_mark_node;
        !          3701:            }
        !          3702:        }
        !          3703: 
        !          3704:       /* Once we are in a destructor, try not going through
        !          3705:         the virtual function table to find the next destructor.  */
        !          3706:       if (DECL_VINDEX (dtor)
        !          3707:          && ! (flags & LOOKUP_NONVIRTUAL)
        !          3708:          && TREE_CODE (auto_delete) != PARM_DECL
        !          3709:          && (ptr == 1 || ! resolves_to_fixed_type_p (ref, 0)))
        !          3710:        {
        !          3711:          /* This destructor must be called via virtual function table.  */
        !          3712:          dtor = TREE_VEC_ELT (CLASSTYPE_METHOD_VEC (DECL_CONTEXT (dtor)), 0);
        !          3713:          expr = convert_pointer_to (DECL_CLASS_CONTEXT (dtor), TREE_VALUE (parms));
        !          3714:          if (expr != TREE_VALUE (parms))
        !          3715:            {
        !          3716:              expr = fold (expr);
        !          3717:              ref = build_indirect_ref (expr, NULL_PTR);
        !          3718:              TREE_VALUE (parms) = expr;
        !          3719:            }
        !          3720:          function = build_vfn_ref (&TREE_VALUE (parms), ref, DECL_VINDEX (dtor));
        !          3721:          if (function == error_mark_node)
        !          3722:            return error_mark_node;
        !          3723:          TREE_TYPE (function) = build_pointer_type (TREE_TYPE (dtor));
        !          3724:          TREE_CHAIN (parms) = build_tree_list (NULL_TREE, auto_delete);
        !          3725:          expr = build_function_call (function, parms);
        !          3726:          if (ptr && (flags & LOOKUP_DESTRUCTOR) == 0)
        !          3727:            {
        !          3728:              /* Handle the case where a virtual destructor is
        !          3729:                 being called on an item that is 0.
        !          3730: 
        !          3731:                 @@ Does this really need to be done?  */
        !          3732:              tree ifexp = build_binary_op(NE_EXPR, addr, integer_zero_node,1);
        !          3733: #if 0
        !          3734:              if (TREE_CODE (ref) == VAR_DECL
        !          3735:                  || TREE_CODE (ref) == COMPONENT_REF)
        !          3736:                warning ("losing in build_delete");
        !          3737: #endif
        !          3738:              expr = build (COND_EXPR, void_type_node,
        !          3739:                            ifexp, expr, void_zero_node);
        !          3740:            }
        !          3741:        }
        !          3742:       else
        !          3743:        {
        !          3744:          tree ifexp;
        !          3745: 
        !          3746:          if ((flags & LOOKUP_DESTRUCTOR)
        !          3747:              || TREE_CODE (ref) == VAR_DECL
        !          3748:              || TREE_CODE (ref) == PARM_DECL
        !          3749:              || TREE_CODE (ref) == COMPONENT_REF
        !          3750:              || TREE_CODE (ref) == ARRAY_REF)
        !          3751:            /* These can't be 0.  */
        !          3752:            ifexp = integer_one_node;
        !          3753:          else
        !          3754:            /* Handle the case where a non-virtual destructor is
        !          3755:               being called on an item that is 0.  */
        !          3756:            ifexp = build_binary_op (NE_EXPR, addr, integer_zero_node, 1);
        !          3757: 
        !          3758:          /* Used to mean that this destructor was known to be empty,
        !          3759:             but that's now obsolete.  */
        !          3760:          my_friendly_assert (DECL_INITIAL (dtor) != void_type_node, 221);
        !          3761: 
        !          3762:          TREE_CHAIN (parms) = build_tree_list (NULL_TREE, auto_delete);
        !          3763:          expr = build_function_call (dtor, parms);
        !          3764: 
        !          3765:          if (ifexp != integer_one_node)
        !          3766:            expr = build (COND_EXPR, void_type_node,
        !          3767:                          ifexp, expr, void_zero_node);
        !          3768:        }
        !          3769:       return expr;
        !          3770:     }
        !          3771:   else
        !          3772:     {
        !          3773:       /* This can get visibilities wrong.  */
        !          3774:       tree binfos = BINFO_BASETYPES (TYPE_BINFO (type));
        !          3775:       int i, n_baseclasses = binfos ? TREE_VEC_LENGTH (binfos) : 0;
        !          3776:       tree base_binfo = n_baseclasses > 0 ? TREE_VEC_ELT (binfos, 0) : NULL_TREE;
        !          3777:       tree exprstmt = NULL_TREE;
        !          3778:       tree parent_auto_delete = auto_delete;
        !          3779:       tree cond;
        !          3780: 
        !          3781:       /* If this type does not have a destructor, but does have
        !          3782:         operator delete, call the parent parent destructor (if any),
        !          3783:         but let this node do the deleting.  Otherwise, it is ok
        !          3784:         to let the parent destructor do the deleting.  */
        !          3785:       if (TREE_GETS_DELETE (type) && !use_global_delete)
        !          3786:        {
        !          3787:          parent_auto_delete = integer_zero_node;
        !          3788:          if (auto_delete == integer_zero_node)
        !          3789:            cond = NULL_TREE;
        !          3790:          else
        !          3791:            {
        !          3792:              tree virtual_size;
        !          3793: 
        !          3794:                /* This is probably wrong. It should be the size of the
        !          3795:                   virtual object being deleted.  */
        !          3796:              virtual_size = c_sizeof_nowarn (type);
        !          3797: 
        !          3798:              expr = build_opfncall (DELETE_EXPR, LOOKUP_NORMAL, addr,
        !          3799:                                     virtual_size, NULL_TREE);
        !          3800:              if (expr == error_mark_node)
        !          3801:                return error_mark_node;
        !          3802:              if (auto_delete != integer_one_node)
        !          3803:                cond = build (COND_EXPR, void_type_node,
        !          3804:                              build (BIT_AND_EXPR, integer_type_node,
        !          3805:                                     auto_delete, integer_one_node),
        !          3806:                              expr, void_zero_node);
        !          3807:              else
        !          3808:                cond = expr;
        !          3809:            }
        !          3810:        }
        !          3811:       else if (base_binfo == NULL_TREE
        !          3812:               || (TREE_VIA_VIRTUAL (base_binfo) == 0
        !          3813:                   && ! TYPE_NEEDS_DESTRUCTOR (BINFO_TYPE (base_binfo))))
        !          3814:        {
        !          3815:          tree virtual_size;
        !          3816: 
        !          3817:          /* This is probably wrong. It should be the size of the virtual
        !          3818:             object being deleted.  */
        !          3819:          virtual_size = c_sizeof_nowarn (type);
        !          3820: 
        !          3821:          cond = build (COND_EXPR, void_type_node,
        !          3822:                        build (BIT_AND_EXPR, integer_type_node, auto_delete, integer_one_node),
        !          3823:                        build_builtin_call (void_type_node, BID,
        !          3824:                                            build_tree_list (NULL_TREE, addr)),
        !          3825:                        void_zero_node);
        !          3826:        }
        !          3827:       else
        !          3828:        cond = NULL_TREE;
        !          3829: 
        !          3830:       if (cond)
        !          3831:        exprstmt = build_tree_list (NULL_TREE, cond);
        !          3832: 
        !          3833:       if (base_binfo
        !          3834:          && ! TREE_VIA_VIRTUAL (base_binfo)
        !          3835:          && TYPE_NEEDS_DESTRUCTOR (BINFO_TYPE (base_binfo)))
        !          3836:        {
        !          3837:          tree this_auto_delete;
        !          3838: 
        !          3839:          if (BINFO_OFFSET_ZEROP (base_binfo))
        !          3840:            this_auto_delete = parent_auto_delete;
        !          3841:          else
        !          3842:            this_auto_delete = integer_zero_node;
        !          3843: 
        !          3844:          expr = build_delete (TYPE_POINTER_TO (BINFO_TYPE (base_binfo)), addr,
        !          3845:                               this_auto_delete, flags|LOOKUP_PROTECTED_OK, 0);
        !          3846:          exprstmt = tree_cons (NULL_TREE, expr, exprstmt);
        !          3847:        }
        !          3848: 
        !          3849:       /* Take care of the remaining baseclasses.  */
        !          3850:       for (i = 1; i < n_baseclasses; i++)
        !          3851:        {
        !          3852:          base_binfo = TREE_VEC_ELT (binfos, i);
        !          3853:          if (! TYPE_NEEDS_DESTRUCTOR (BINFO_TYPE (base_binfo))
        !          3854:              || TREE_VIA_VIRTUAL (base_binfo))
        !          3855:            continue;
        !          3856: 
        !          3857:          /* May be zero offset if other baseclasses are virtual.  */
        !          3858:          expr = fold (build (PLUS_EXPR, TYPE_POINTER_TO (BINFO_TYPE (base_binfo)),
        !          3859:                              addr, BINFO_OFFSET (base_binfo)));
        !          3860: 
        !          3861:          expr = build_delete (TYPE_POINTER_TO (BINFO_TYPE (base_binfo)), expr,
        !          3862:                               integer_zero_node,
        !          3863:                               flags|LOOKUP_PROTECTED_OK, 0);
        !          3864: 
        !          3865:          exprstmt = tree_cons (NULL_TREE, expr, exprstmt);
        !          3866:        }
        !          3867: 
        !          3868:       for (member = TYPE_FIELDS (type); member; member = TREE_CHAIN (member))
        !          3869:        {
        !          3870:          if (TREE_CODE (member) != FIELD_DECL)
        !          3871:            continue;
        !          3872:          if (TYPE_NEEDS_DESTRUCTOR (TREE_TYPE (member)))
        !          3873:            {
        !          3874:              tree this_member = build_component_ref (ref, DECL_NAME (member), 0, 0);
        !          3875:              tree this_type = TREE_TYPE (member);
        !          3876:              expr = build_delete (this_type, this_member, integer_two_node, flags, 0);
        !          3877:              exprstmt = tree_cons (NULL_TREE, expr, exprstmt);
        !          3878:            }
        !          3879:        }
        !          3880: 
        !          3881:       if (exprstmt)
        !          3882:        return build_compound_expr (exprstmt);
        !          3883:       /* Virtual base classes make this function do nothing.  */
        !          3884:       return void_zero_node;
        !          3885:     }
        !          3886: }
        !          3887: 
        !          3888: /* For type TYPE, delete the virtual baseclass objects of DECL.  */
        !          3889: 
        !          3890: tree
        !          3891: build_vbase_delete (type, decl)
        !          3892:      tree type, decl;
        !          3893: {
        !          3894:   tree vbases = CLASSTYPE_VBASECLASSES (type);
        !          3895:   tree result = NULL_TREE;
        !          3896:   tree addr = build_unary_op (ADDR_EXPR, decl, 0);
        !          3897: 
        !          3898:   my_friendly_assert (addr != error_mark_node, 222);
        !          3899: 
        !          3900:   while (vbases)
        !          3901:     {
        !          3902:       tree this_addr = convert_force (TYPE_POINTER_TO (BINFO_TYPE (vbases)),
        !          3903:                                      addr);
        !          3904:       result = tree_cons (NULL_TREE,
        !          3905:                          build_delete (TREE_TYPE (this_addr), this_addr,
        !          3906:                                        integer_zero_node,
        !          3907:                                        LOOKUP_NORMAL|LOOKUP_DESTRUCTOR, 0),
        !          3908:                          result);
        !          3909:       vbases = TREE_CHAIN (vbases);
        !          3910:     }
        !          3911:   return build_compound_expr (nreverse (result));
        !          3912: }
        !          3913: 
        !          3914: /* Build a C++ vector delete expression.
        !          3915:    MAXINDEX is the number of elements to be deleted.
        !          3916:    ELT_SIZE is the nominal size of each element in the vector.
        !          3917:    BASE is the expression that should yield the store to be deleted.
        !          3918:    DTOR_DUMMY is a placeholder for a destructor.  The library function
        !          3919:    __builtin_vec_delete has a pointer to function in this position.
        !          3920:    This function expands (or synthesizes) these calls itself.
        !          3921:    AUTO_DELETE_VEC says whether the container (vector) should be deallocated.
        !          3922:    AUTO_DELETE say whether each item in the container should be deallocated.
        !          3923: 
        !          3924:    This also calls delete for virtual baseclasses of elements of the vector.
        !          3925: 
        !          3926:    Update: MAXINDEX is no longer needed.  The size can be extracted from the
        !          3927:    start of the vector for pointers, and from the type for arrays.  We still
        !          3928:    use MAXINDEX for arrays because it happens to already have one of the
        !          3929:    values we'd have to extract.  (We could use MAXINDEX with pointers to
        !          3930:    confirm the size, and trap if the numbers differ; not clear that it'd
        !          3931:    be worth bothering.)  */
        !          3932: tree
        !          3933: build_vec_delete (base, maxindex, elt_size, dtor_dummy, auto_delete_vec, auto_delete)
        !          3934:      tree base, maxindex, elt_size;
        !          3935:      tree dtor_dummy;
        !          3936:      tree auto_delete_vec, auto_delete;
        !          3937: {
        !          3938:   tree ptype = TREE_TYPE (base);
        !          3939:   tree type;
        !          3940:   tree virtual_size;
        !          3941:   /* Temporary variables used by the loop.  */
        !          3942:   tree tbase, size_exp, tbase_init;
        !          3943: 
        !          3944:   /* This is the body of the loop that implements the deletion of a
        !          3945:      single element, and moves temp variables to next elements.  */
        !          3946:   tree body;
        !          3947: 
        !          3948:   /* This is the LOOP_EXPR that governs the deletion of the elements.  */
        !          3949:   tree loop;
        !          3950: 
        !          3951:   /* This is the thing that governs what to do after the loop has run.  */
        !          3952:   tree deallocate_expr = 0;
        !          3953: 
        !          3954:   /* This is the BIND_EXPR which holds the outermost iterator of the
        !          3955:      loop.  It is convenient to set this variable up and test it before
        !          3956:      executing any other code in the loop.
        !          3957:      This is also the containing expression returned by this function.  */
        !          3958:   tree controller = NULL_TREE;
        !          3959: 
        !          3960:   /* This is the BLOCK to record the symbol binding for debugging.  */
        !          3961:   tree block;
        !          3962: 
        !          3963:   base = stabilize_reference (base);
        !          3964: 
        !          3965:   /* Since we can use base many times, save_expr it. */
        !          3966:   if (TREE_SIDE_EFFECTS (base))
        !          3967:     base = save_expr (base);
        !          3968: 
        !          3969:   if (TREE_CODE (ptype) == POINTER_TYPE)
        !          3970:     {
        !          3971:       /* Step back one from start of vector, and read dimension.  */
        !          3972:       tree cookie_addr = build (MINUS_EXPR, TYPE_POINTER_TO (BI_header_type),
        !          3973:                                base, BI_header_size);
        !          3974:       tree cookie = build_indirect_ref (cookie_addr, NULL_PTR);
        !          3975:       maxindex = build_component_ref (cookie, nc_nelts_field_id, 0, 0);
        !          3976:       do
        !          3977:        ptype = TREE_TYPE (ptype);
        !          3978:       while (TREE_CODE (ptype) == ARRAY_TYPE);
        !          3979:     }
        !          3980:   else if (TREE_CODE (ptype) == ARRAY_TYPE)
        !          3981:     {
        !          3982:       /* get the total number of things in the array, maxindex is a bad name */
        !          3983:       maxindex = array_type_nelts_total (ptype);
        !          3984:       while (TREE_CODE (ptype) == ARRAY_TYPE)
        !          3985:        ptype = TREE_TYPE (ptype);
        !          3986:       base = build_unary_op (ADDR_EXPR, base, 1);
        !          3987:     }
        !          3988:   else
        !          3989:     {
        !          3990:       error ("type to vector delete is neither pointer or array type");
        !          3991:       return error_mark_node;
        !          3992:     }
        !          3993:   type = ptype;
        !          3994:   ptype = TYPE_POINTER_TO (type);
        !          3995: 
        !          3996:   size_exp = size_in_bytes (type);
        !          3997: 
        !          3998:   if (! IS_AGGR_TYPE (type) || ! TYPE_NEEDS_DESTRUCTOR (type))
        !          3999:     {
        !          4000:       loop = integer_zero_node;
        !          4001:       goto no_destructor;
        !          4002:     }
        !          4003: 
        !          4004:   /* The below is short by BI_header_size */
        !          4005:   virtual_size = fold (size_binop (MULT_EXPR, size_exp, maxindex));
        !          4006: 
        !          4007:   tbase = build_decl (VAR_DECL, NULL_TREE, ptype);
        !          4008:   tbase_init = build_modify_expr (tbase, NOP_EXPR,
        !          4009:                                  fold (build (PLUS_EXPR, ptype,
        !          4010:                                               base,
        !          4011:                                               virtual_size)));
        !          4012:   DECL_REGISTER (tbase) = 1;
        !          4013:   controller = build (BIND_EXPR, void_type_node, tbase, 0, 0);
        !          4014:   TREE_SIDE_EFFECTS (controller) = 1;
        !          4015:   block = build_block (tbase, 0, 0, 0, 0);
        !          4016:   add_block_current_level (block);
        !          4017: 
        !          4018:   if (auto_delete != integer_zero_node
        !          4019:       && auto_delete != integer_two_node)
        !          4020:     {
        !          4021:       tree base_tbd = convert (ptype,
        !          4022:                               build_binary_op (MINUS_EXPR,
        !          4023:                                                convert (ptr_type_node, base),
        !          4024:                                                BI_header_size,
        !          4025:                                                1));
        !          4026:       /* This is the real size */
        !          4027:       virtual_size = size_binop (PLUS_EXPR, virtual_size, BI_header_size);
        !          4028:       body = build_tree_list (NULL_TREE,
        !          4029:                              build_x_delete (ptr_type_node, base_tbd, 0,
        !          4030:                                              virtual_size));
        !          4031:       body = build (COND_EXPR, void_type_node,
        !          4032:                    build (BIT_AND_EXPR, integer_type_node,
        !          4033:                           auto_delete, integer_one_node),
        !          4034:                    body, integer_zero_node);
        !          4035:     }
        !          4036:   else
        !          4037:     body = NULL_TREE;
        !          4038: 
        !          4039:   body = tree_cons (NULL_TREE,
        !          4040:                    build_delete (ptype, tbase, auto_delete,
        !          4041:                                  LOOKUP_NORMAL|LOOKUP_DESTRUCTOR, 0),
        !          4042:                    body);
        !          4043: 
        !          4044:   body = tree_cons (NULL_TREE,
        !          4045:                    build_modify_expr (tbase, NOP_EXPR, build (MINUS_EXPR, ptype, tbase, size_exp)),
        !          4046:                    body);
        !          4047: 
        !          4048:   body = tree_cons (NULL_TREE,
        !          4049:                    build (EXIT_EXPR, void_type_node,
        !          4050:                           build (EQ_EXPR, integer_type_node, base, tbase)),
        !          4051:                    body);
        !          4052: 
        !          4053:   loop = build (LOOP_EXPR, void_type_node, build_compound_expr (body));
        !          4054: 
        !          4055:   loop = tree_cons (NULL_TREE, tbase_init,
        !          4056:                    tree_cons (NULL_TREE, loop, NULL_TREE));
        !          4057:   loop = build_compound_expr (loop);
        !          4058: 
        !          4059:  no_destructor:
        !          4060:   /* If the delete flag is one, or anything else with the low bit set,
        !          4061:      delete the storage.  */
        !          4062:   if (auto_delete_vec == integer_zero_node
        !          4063:       || auto_delete_vec == integer_two_node)
        !          4064:     deallocate_expr = integer_zero_node;
        !          4065:   else
        !          4066:     {
        !          4067:       tree base_tbd;
        !          4068: 
        !          4069:       /* The below is short by BI_header_size */
        !          4070:       virtual_size = fold (size_binop (MULT_EXPR, size_exp, maxindex));
        !          4071: 
        !          4072:       if (loop == integer_zero_node)
        !          4073:        /* no header */
        !          4074:        base_tbd = base;
        !          4075:       else
        !          4076:        {
        !          4077:          base_tbd = convert (ptype,
        !          4078:                              build_binary_op (MINUS_EXPR,
        !          4079:                                               convert (string_type_node, base),
        !          4080:                                               BI_header_size,
        !          4081:                                               1));
        !          4082:          /* True size with header. */
        !          4083:          virtual_size = size_binop (PLUS_EXPR, virtual_size, BI_header_size);
        !          4084:        }
        !          4085:       deallocate_expr = build_x_delete (ptr_type_node, base_tbd, 1,
        !          4086:                                        virtual_size);
        !          4087:       if (auto_delete_vec != integer_one_node)
        !          4088:        deallocate_expr = build (COND_EXPR, void_type_node,
        !          4089:                                 build (BIT_AND_EXPR, integer_type_node,
        !          4090:                                        auto_delete_vec, integer_one_node),
        !          4091:                                 deallocate_expr, integer_zero_node);
        !          4092:     }
        !          4093: 
        !          4094:   if (loop && deallocate_expr != integer_zero_node)
        !          4095:     {
        !          4096:       body = tree_cons (NULL_TREE, loop,
        !          4097:                        tree_cons (NULL_TREE, deallocate_expr, NULL_TREE));
        !          4098:       body = build_compound_expr (body);
        !          4099:     }
        !          4100:   else
        !          4101:     body = loop;
        !          4102: 
        !          4103:   /* Outermost wrapper: If pointer is null, punt.  */
        !          4104:   body = build (COND_EXPR, void_type_node,
        !          4105:                build (NE_EXPR, integer_type_node, base, integer_zero_node),
        !          4106:                body, integer_zero_node);
        !          4107:   body = build1 (NOP_EXPR, void_type_node, body);
        !          4108: 
        !          4109:   if (controller)
        !          4110:     {
        !          4111:       TREE_OPERAND (controller, 1) = body;
        !          4112:       return controller;
        !          4113:     }
        !          4114:   else
        !          4115:     return convert (void_type_node, body);
        !          4116: }

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.