Annotation of GNUtools/cc/stmt.c, revision 1.1.1.1

1.1       root        1: /* Expands front end tree to back end RTL for GNU C-Compiler
                      2:    Copyright (C) 1987, 1988, 1989, 1992, 1993 Free Software Foundation, Inc.
                      3: 
                      4: This file is part of GNU CC.
                      5: 
                      6: GNU CC is free software; you can redistribute it and/or modify
                      7: it under the terms of the GNU General Public License as published by
                      8: the Free Software Foundation; either version 2, or (at your option)
                      9: any later version.
                     10: 
                     11: GNU CC is distributed in the hope that it will be useful,
                     12: but WITHOUT ANY WARRANTY; without even the implied warranty of
                     13: MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
                     14: GNU General Public License for more details.
                     15: 
                     16: You should have received a copy of the GNU General Public License
                     17: along with GNU CC; see the file COPYING.  If not, write to
                     18: the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.  */
                     19: 
                     20: 
                     21: /* This file handles the generation of rtl code from tree structure
                     22:    above the level of expressions, using subroutines in exp*.c and emit-rtl.c.
                     23:    It also creates the rtl expressions for parameters and auto variables
                     24:    and has full responsibility for allocating stack slots.
                     25: 
                     26:    The functions whose names start with `expand_' are called by the
                     27:    parser to generate RTL instructions for various kinds of constructs.
                     28: 
                     29:    Some control and binding constructs require calling several such
                     30:    functions at different times.  For example, a simple if-then
                     31:    is expanded by calling `expand_start_cond' (with the condition-expression
                     32:    as argument) before parsing the then-clause and calling `expand_end_cond'
                     33:    after parsing the then-clause.  */
                     34: 
                     35: #include "config.h"
                     36: 
                     37: #include <stdio.h>
                     38: #include <ctype.h>
                     39: 
                     40: #include "rtl.h"
                     41: #include "tree.h"
                     42: #include "flags.h"
                     43: #include "function.h"
                     44: #include "insn-flags.h"
                     45: #include "insn-config.h"
                     46: #include "insn-codes.h"
                     47: #include "expr.h"
                     48: #include "hard-reg-set.h"
                     49: #include "obstack.h"
                     50: #include "loop.h"
                     51: #include "recog.h"
                     52: #include "machmode.h"
                     53: 
                     54: #include "bytecode.h"
                     55: #include "bc-typecd.h"
                     56: #include "bc-opcode.h"
                     57: #include "bc-optab.h"
                     58: #include "bc-emit.h"
                     59: 
                     60: #define obstack_chunk_alloc xmalloc
                     61: #define obstack_chunk_free free
                     62: struct obstack stmt_obstack;
                     63: 
                     64: /* Filename and line number of last line-number note,
                     65:    whether we actually emitted it or not.  */
                     66: char *emit_filename;
                     67: int emit_lineno;
                     68: 
                     69: /* Nonzero if within a ({...}) grouping, in which case we must
                     70:    always compute a value for each expr-stmt in case it is the last one.  */
                     71: 
                     72: int expr_stmts_for_value;
                     73: 
                     74: /* Each time we expand an expression-statement,
                     75:    record the expr's type and its RTL value here.  */
                     76: 
                     77: static tree last_expr_type;
                     78: static rtx last_expr_value;
                     79: 
                     80: /* Each time we expand the end of a binding contour (in `expand_end_bindings')
                     81:    and we emit a new NOTE_INSN_BLOCK_END note, we save a pointer to it here.
                     82:    This is used by the `remember_end_note' function to record the endpoint
                     83:    of each generated block in its associated BLOCK node.  */
                     84: 
                     85: static rtx last_block_end_note;
                     86: 
                     87: /* Number of binding contours started so far in this function.  */
                     88: 
                     89: int block_start_count;
                     90: 
                     91: /* Nonzero if function being compiled needs to
                     92:    return the address of where it has put a structure value.  */
                     93: 
                     94: extern int current_function_returns_pcc_struct;
                     95: 
                     96: /* Label that will go on parm cleanup code, if any.
                     97:    Jumping to this label runs cleanup code for parameters, if
                     98:    such code must be run.  Following this code is the logical return label.  */
                     99: 
                    100: extern rtx cleanup_label;
                    101: 
                    102: /* Label that will go on function epilogue.
                    103:    Jumping to this label serves as a "return" instruction
                    104:    on machines which require execution of the epilogue on all returns.  */
                    105: 
                    106: extern rtx return_label;
                    107: 
                    108: /* List (chain of EXPR_LISTs) of pseudo-regs of SAVE_EXPRs.
                    109:    So we can mark them all live at the end of the function, if nonopt.  */
                    110: extern rtx save_expr_regs;
                    111: 
                    112: /* Offset to end of allocated area of stack frame.
                    113:    If stack grows down, this is the address of the last stack slot allocated.
                    114:    If stack grows up, this is the address for the next slot.  */
                    115: extern int frame_offset;
                    116: 
                    117: /* Label to jump back to for tail recursion, or 0 if we have
                    118:    not yet needed one for this function.  */
                    119: extern rtx tail_recursion_label;
                    120: 
                    121: /* Place after which to insert the tail_recursion_label if we need one.  */
                    122: extern rtx tail_recursion_reentry;
                    123: 
                    124: /* Location at which to save the argument pointer if it will need to be
                    125:    referenced.  There are two cases where this is done: if nonlocal gotos
                    126:    exist, or if vars whose is an offset from the argument pointer will be
                    127:    needed by inner routines.  */
                    128: 
                    129: extern rtx arg_pointer_save_area;
                    130: 
                    131: /* Chain of all RTL_EXPRs that have insns in them.  */
                    132: extern tree rtl_expr_chain;
                    133: 
                    134: #if 0  /* Turned off because 0 seems to work just as well.  */
                    135: /* Cleanup lists are required for binding levels regardless of whether
                    136:    that binding level has cleanups or not.  This node serves as the
                    137:    cleanup list whenever an empty list is required.  */
                    138: static tree empty_cleanup_list;
                    139: #endif
                    140: 
                    141: /* Functions and data structures for expanding case statements.  */
                    142: 
                    143: /* Case label structure, used to hold info on labels within case
                    144:    statements.  We handle "range" labels; for a single-value label
                    145:    as in C, the high and low limits are the same.
                    146: 
                    147:    A chain of case nodes is initially maintained via the RIGHT fields
                    148:    in the nodes.  Nodes with higher case values are later in the list.
                    149: 
                    150:    Switch statements can be output in one of two forms.  A branch table
                    151:    is used if there are more than a few labels and the labels are dense
                    152:    within the range between the smallest and largest case value.  If a
                    153:    branch table is used, no further manipulations are done with the case
                    154:    node chain.
                    155: 
                    156:    The alternative to the use of a branch table is to generate a series
                    157:    of compare and jump insns.  When that is done, we use the LEFT, RIGHT,
                    158:    and PARENT fields to hold a binary tree.  Initially the tree is
                    159:    totally unbalanced, with everything on the right.  We balance the tree
                    160:    with nodes on the left having lower case values than the parent
                    161:    and nodes on the right having higher values.  We then output the tree
                    162:    in order.  */
                    163: 
                    164: struct case_node
                    165: {
                    166:   struct case_node     *left;  /* Left son in binary tree */
                    167:   struct case_node     *right; /* Right son in binary tree; also node chain */
                    168:   struct case_node     *parent; /* Parent of node in binary tree */
                    169:   tree                 low;    /* Lowest index value for this label */
                    170:   tree                 high;   /* Highest index value for this label */
                    171:   tree                 code_label; /* Label to jump to when node matches */
                    172: };
                    173: 
                    174: typedef struct case_node case_node;
                    175: typedef struct case_node *case_node_ptr;
                    176: 
                    177: /* These are used by estimate_case_costs and balance_case_nodes.  */
                    178: 
                    179: /* This must be a signed type, and non-ANSI compilers lack signed char.  */
                    180: static short *cost_table;
                    181: static int use_cost_table;
                    182: 
                    183: static int estimate_case_costs ();
                    184: static void balance_case_nodes ();
                    185: static void emit_case_nodes ();
                    186: static void group_case_nodes ();
                    187: static void emit_jump_if_reachable ();
                    188: 
                    189: static int warn_if_unused_value ();
                    190: static void expand_goto_internal ();
                    191: static void bc_expand_goto_internal ();
                    192: static int expand_fixup ();
                    193: static void bc_expand_fixup ();
                    194: void fixup_gotos ();
                    195: static void bc_fixup_gotos ();
                    196: void free_temp_slots ();
                    197: static void expand_cleanups ();
                    198: static void expand_null_return_1 ();
                    199: static int tail_recursion_args ();
                    200: static void do_jump_if_equal ();
                    201: int bc_expand_exit_loop_if_false ();
                    202: void bc_expand_start_cond ();
                    203: void bc_expand_end_cond ();
                    204: void bc_expand_start_else ();
                    205: void bc_expand_end_bindings ();
                    206: void bc_expand_start_case ();
                    207: void bc_check_for_full_enumeration_handling ();
                    208: void bc_expand_end_case ();
                    209: void bc_expand_decl ();
                    210: 
                    211: extern rtx bc_allocate_local ();
                    212: extern rtx bc_allocate_variable_array ();
                    213: 
                    214: /* Stack of control and binding constructs we are currently inside.
                    215: 
                    216:    These constructs begin when you call `expand_start_WHATEVER'
                    217:    and end when you call `expand_end_WHATEVER'.  This stack records
                    218:    info about how the construct began that tells the end-function
                    219:    what to do.  It also may provide information about the construct
                    220:    to alter the behavior of other constructs within the body.
                    221:    For example, they may affect the behavior of C `break' and `continue'.
                    222: 
                    223:    Each construct gets one `struct nesting' object.
                    224:    All of these objects are chained through the `all' field.
                    225:    `nesting_stack' points to the first object (innermost construct).
                    226:    The position of an entry on `nesting_stack' is in its `depth' field.
                    227: 
                    228:    Each type of construct has its own individual stack.
                    229:    For example, loops have `loop_stack'.  Each object points to the
                    230:    next object of the same type through the `next' field.
                    231: 
                    232:    Some constructs are visible to `break' exit-statements and others
                    233:    are not.  Which constructs are visible depends on the language.
                    234:    Therefore, the data structure allows each construct to be visible
                    235:    or not, according to the args given when the construct is started.
                    236:    The construct is visible if the `exit_label' field is non-null.
                    237:    In that case, the value should be a CODE_LABEL rtx.  */
                    238: 
                    239: struct nesting
                    240: {
                    241:   struct nesting *all;
                    242:   struct nesting *next;
                    243:   int depth;
                    244:   rtx exit_label;
                    245:   union
                    246:     {
                    247:       /* For conds (if-then and if-then-else statements).  */
                    248:       struct
                    249:        {
                    250:          /* Label for the end of the if construct.
                    251:             There is none if EXITFLAG was not set
                    252:             and no `else' has been seen yet.  */
                    253:          rtx endif_label;
                    254:          /* Label for the end of this alternative.
                    255:             This may be the end of the if or the next else/elseif. */
                    256:          rtx next_label;
                    257:        } cond;
                    258:       /* For loops.  */
                    259:       struct
                    260:        {
                    261:          /* Label at the top of the loop; place to loop back to.  */
                    262:          rtx start_label;
                    263:          /* Label at the end of the whole construct.  */
                    264:          rtx end_label;
                    265:          /* Label for `continue' statement to jump to;
                    266:             this is in front of the stepper of the loop.  */
                    267:          rtx continue_label;
                    268:        } loop;
                    269:       /* For variable binding contours.  */
                    270:       struct
                    271:        {
                    272:          /* Sequence number of this binding contour within the function,
                    273:             in order of entry.  */
                    274:          int block_start_count;
                    275:          /* Nonzero => value to restore stack to on exit.  Complemented by
                    276:             bc_stack_level (see below) when generating bytecodes. */
                    277:          rtx stack_level;
                    278:          /* The NOTE that starts this contour.
                    279:             Used by expand_goto to check whether the destination
                    280:             is within each contour or not.  */
                    281:          rtx first_insn;
                    282:          /* Innermost containing binding contour that has a stack level.  */
                    283:          struct nesting *innermost_stack_block;
                    284:          /* List of cleanups to be run on exit from this contour.
                    285:             This is a list of expressions to be evaluated.
                    286:             The TREE_PURPOSE of each link is the ..._DECL node
                    287:             which the cleanup pertains to.  */
                    288:          tree cleanups;
                    289:          /* List of cleanup-lists of blocks containing this block,
                    290:             as they were at the locus where this block appears.
                    291:             There is an element for each containing block,
                    292:             ordered innermost containing block first.
                    293:             The tail of this list can be 0 (was empty_cleanup_list),
                    294:             if all remaining elements would be empty lists.
                    295:             The element's TREE_VALUE is the cleanup-list of that block,
                    296:             which may be null.  */
                    297:          tree outer_cleanups;
                    298:          /* Chain of labels defined inside this binding contour.
                    299:             For contours that have stack levels or cleanups.  */
                    300:          struct label_chain *label_chain;
                    301:          /* Number of function calls seen, as of start of this block.  */
                    302:          int function_call_count;
                    303:          /* Bytecode specific: stack level to restore stack to on exit.  */
                    304:          int bc_stack_level;
                    305:        } block;
                    306:       /* For switch (C) or case (Pascal) statements,
                    307:         and also for dummies (see `expand_start_case_dummy').  */
                    308:       struct
                    309:        {
                    310:          /* The insn after which the case dispatch should finally
                    311:             be emitted.  Zero for a dummy.  */
                    312:          rtx start;
                    313:          /* For bytecodes, the case table is in-lined right in the code.
                    314:             A label is needed for skipping over this block. It is only
                    315:             used when generating bytecodes. */
                    316:          rtx skip_label;
                    317:          /* A list of case labels, kept in ascending order by value
                    318:             as the list is built.
                    319:             During expand_end_case, this list may be rearranged into a
                    320:             nearly balanced binary tree.  */
                    321:          struct case_node *case_list;
                    322:          /* Label to jump to if no case matches.  */
                    323:          tree default_label;
                    324:          /* The expression to be dispatched on.  */
                    325:          tree index_expr;
                    326:          /* Type that INDEX_EXPR should be converted to.  */
                    327:          tree nominal_type;
                    328:          /* Number of range exprs in case statement.  */
                    329:          int num_ranges;
                    330:          /* Name of this kind of statement, for warnings.  */
                    331:          char *printname;
                    332:          /* Nonzero if a case label has been seen in this case stmt.  */
                    333:          char seenlabel;
                    334:        } case_stmt;
                    335:       /* For exception contours.  */
                    336:       struct
                    337:        {
                    338:          /* List of exceptions raised.  This is a TREE_LIST
                    339:             of whatever you want.  */
                    340:          tree raised;
                    341:          /* List of exceptions caught.  This is also a TREE_LIST
                    342:             of whatever you want.  As a special case, it has the
                    343:             value `void_type_node' if it handles default exceptions.  */
                    344:          tree handled;
                    345: 
                    346:          /* First insn of TRY block, in case resumptive model is needed.  */
                    347:          rtx first_insn;
                    348:          /* Label for the catch clauses.  */
                    349:          rtx except_label;
                    350:          /* Label for unhandled exceptions.  */
                    351:          rtx unhandled_label;
                    352:          /* Label at the end of whole construct.  */
                    353:          rtx after_label;
                    354:          /* Label which "escapes" the exception construct.
                    355:             Like EXIT_LABEL for BREAK construct, but for exceptions.  */
                    356:          rtx escape_label;
                    357:        } except_stmt;
                    358:     } data;
                    359: };
                    360: 
                    361: /* Chain of all pending binding contours.  */
                    362: struct nesting *block_stack;
                    363: 
                    364: /* If any new stacks are added here, add them to POPSTACKS too.  */
                    365: 
                    366: /* Chain of all pending binding contours that restore stack levels
                    367:    or have cleanups.  */
                    368: struct nesting *stack_block_stack;
                    369: 
                    370: /* Chain of all pending conditional statements.  */
                    371: struct nesting *cond_stack;
                    372: 
                    373: /* Chain of all pending loops.  */
                    374: struct nesting *loop_stack;
                    375: 
                    376: /* Chain of all pending case or switch statements.  */
                    377: struct nesting *case_stack;
                    378: 
                    379: /* Chain of all pending exception contours.  */
                    380: struct nesting *except_stack;
                    381: 
                    382: /* Separate chain including all of the above,
                    383:    chained through the `all' field.  */
                    384: struct nesting *nesting_stack;
                    385: 
                    386: /* Number of entries on nesting_stack now.  */
                    387: int nesting_depth;
                    388: 
                    389: /* Allocate and return a new `struct nesting'.  */
                    390: 
                    391: #define ALLOC_NESTING() \
                    392:  (struct nesting *) obstack_alloc (&stmt_obstack, sizeof (struct nesting))
                    393: 
                    394: /* Pop the nesting stack element by element until we pop off
                    395:    the element which is at the top of STACK.
                    396:    Update all the other stacks, popping off elements from them
                    397:    as we pop them from nesting_stack.  */
                    398: 
                    399: #define POPSTACK(STACK)                                        \
                    400: do { struct nesting *target = STACK;                   \
                    401:      struct nesting *this;                             \
                    402:      do { this = nesting_stack;                                \
                    403:          if (loop_stack == this)                       \
                    404:            loop_stack = loop_stack->next;              \
                    405:          if (cond_stack == this)                       \
                    406:            cond_stack = cond_stack->next;              \
                    407:          if (block_stack == this)                      \
                    408:            block_stack = block_stack->next;            \
                    409:          if (stack_block_stack == this)                \
                    410:            stack_block_stack = stack_block_stack->next; \
                    411:          if (case_stack == this)                       \
                    412:            case_stack = case_stack->next;              \
                    413:          if (except_stack == this)                     \
                    414:            except_stack = except_stack->next;          \
                    415:          nesting_depth = nesting_stack->depth - 1;     \
                    416:          nesting_stack = this->all;                    \
                    417:          obstack_free (&stmt_obstack, this); }         \
                    418:      while (this != target); } while (0)
                    419: 
                    420: /* In some cases it is impossible to generate code for a forward goto
                    421:    until the label definition is seen.  This happens when it may be necessary
                    422:    for the goto to reset the stack pointer: we don't yet know how to do that.
                    423:    So expand_goto puts an entry on this fixup list.
                    424:    Each time a binding contour that resets the stack is exited,
                    425:    we check each fixup.
                    426:    If the target label has now been defined, we can insert the proper code.  */
                    427: 
                    428: struct goto_fixup
                    429: {
                    430:   /* Points to following fixup.  */
                    431:   struct goto_fixup *next;
                    432:   /* Points to the insn before the jump insn.
                    433:      If more code must be inserted, it goes after this insn.  */
                    434:   rtx before_jump;
                    435:   /* The LABEL_DECL that this jump is jumping to, or 0
                    436:      for break, continue or return.  */
                    437:   tree target;
                    438:   /* The BLOCK for the place where this goto was found.  */
                    439:   tree context;
                    440:   /* The CODE_LABEL rtx that this is jumping to.  */
                    441:   rtx target_rtl;
                    442:   /* Number of binding contours started in current function
                    443:      before the label reference.  */
                    444:   int block_start_count;
                    445:   /* The outermost stack level that should be restored for this jump.
                    446:      Each time a binding contour that resets the stack is exited,
                    447:      if the target label is *not* yet defined, this slot is updated.  */
                    448:   rtx stack_level;
                    449:   /* List of lists of cleanup expressions to be run by this goto.
                    450:      There is one element for each block that this goto is within.
                    451:      The tail of this list can be 0 (was empty_cleanup_list),
                    452:      if all remaining elements would be empty.
                    453:      The TREE_VALUE contains the cleanup list of that block as of the
                    454:      time this goto was seen.
                    455:      The TREE_ADDRESSABLE flag is 1 for a block that has been exited.  */
                    456:   tree cleanup_list_list;
                    457: 
                    458:   /* Bytecode specific members follow */
                    459: 
                    460:   /* The label that this jump is jumping to, or 0 for break, continue
                    461:      or return.  */
                    462:   struct bc_label *bc_target;
                    463: 
                    464:   /* The label we use for the fixup patch */
                    465:   struct bc_label *label;
                    466: 
                    467:   /* True (non-0) if fixup has been handled */
                    468:   int bc_handled:1;
                    469: 
                    470:   /* Like stack_level above, except refers to the interpreter stack */
                    471:   int bc_stack_level;
                    472: };
                    473: 
                    474: static struct goto_fixup *goto_fixup_chain;
                    475: 
                    476: /* Within any binding contour that must restore a stack level,
                    477:    all labels are recorded with a chain of these structures.  */
                    478: 
                    479: struct label_chain
                    480: {
                    481:   /* Points to following fixup.  */
                    482:   struct label_chain *next;
                    483:   tree label;
                    484: };
                    485: 
                    486: void
                    487: init_stmt ()
                    488: {
                    489:   gcc_obstack_init (&stmt_obstack);
                    490: #if 0
                    491:   empty_cleanup_list = build_tree_list (NULL_TREE, NULL_TREE);
                    492: #endif
                    493: }
                    494: 
                    495: void
                    496: init_stmt_for_function ()
                    497: {
                    498:   /* We are not currently within any block, conditional, loop or case.  */
                    499:   block_stack = 0;
                    500:   loop_stack = 0;
                    501:   case_stack = 0;
                    502:   cond_stack = 0;
                    503:   nesting_stack = 0;
                    504:   nesting_depth = 0;
                    505: 
                    506:   block_start_count = 0;
                    507: 
                    508:   /* No gotos have been expanded yet.  */
                    509:   goto_fixup_chain = 0;
                    510: 
                    511:   /* We are not processing a ({...}) grouping.  */
                    512:   expr_stmts_for_value = 0;
                    513:   last_expr_type = 0;
                    514: }
                    515: 
                    516: void
                    517: save_stmt_status (p)
                    518:      struct function *p;
                    519: {
                    520:   p->block_stack = block_stack;
                    521:   p->stack_block_stack = stack_block_stack;
                    522:   p->cond_stack = cond_stack;
                    523:   p->loop_stack = loop_stack;
                    524:   p->case_stack = case_stack;
                    525:   p->nesting_stack = nesting_stack;
                    526:   p->nesting_depth = nesting_depth;
                    527:   p->block_start_count = block_start_count;
                    528:   p->last_expr_type = last_expr_type;
                    529:   p->last_expr_value = last_expr_value;
                    530:   p->expr_stmts_for_value = expr_stmts_for_value;
                    531:   p->emit_filename = emit_filename;
                    532:   p->emit_lineno = emit_lineno;
                    533:   p->goto_fixup_chain = goto_fixup_chain;
                    534: }
                    535: 
                    536: void
                    537: restore_stmt_status (p)
                    538:      struct function *p;
                    539: {
                    540:   block_stack = p->block_stack;
                    541:   stack_block_stack = p->stack_block_stack;
                    542:   cond_stack = p->cond_stack;
                    543:   loop_stack = p->loop_stack;
                    544:   case_stack = p->case_stack;
                    545:   nesting_stack = p->nesting_stack;
                    546:   nesting_depth = p->nesting_depth;
                    547:   block_start_count = p->block_start_count;
                    548:   last_expr_type = p->last_expr_type;
                    549:   last_expr_value = p->last_expr_value;
                    550:   expr_stmts_for_value = p->expr_stmts_for_value;
                    551:   emit_filename = p->emit_filename;
                    552:   emit_lineno = p->emit_lineno;
                    553:   goto_fixup_chain = p->goto_fixup_chain;
                    554: }
                    555: 
                    556: /* Emit a no-op instruction.  */
                    557: 
                    558: void
                    559: emit_nop ()
                    560: {
                    561:   rtx last_insn;
                    562: 
                    563:   if (!output_bytecode)
                    564:     {
                    565:       last_insn = get_last_insn ();
                    566:       if (!optimize
                    567:          && (GET_CODE (last_insn) == CODE_LABEL
                    568:              || prev_real_insn (last_insn) == 0))
                    569:        emit_insn (gen_nop ());
                    570:     }
                    571: }
                    572: 
                    573: /* Return the rtx-label that corresponds to a LABEL_DECL,
                    574:    creating it if necessary.  */
                    575: 
                    576: rtx
                    577: label_rtx (label)
                    578:      tree label;
                    579: {
                    580:   if (TREE_CODE (label) != LABEL_DECL)
                    581:     abort ();
                    582: 
                    583:   if (DECL_RTL (label))
                    584:     return DECL_RTL (label);
                    585: 
                    586:   return DECL_RTL (label) = gen_label_rtx ();
                    587: }
                    588: 
                    589: /* Add an unconditional jump to LABEL as the next sequential instruction.  */
                    590: 
                    591: void
                    592: emit_jump (label)
                    593:      rtx label;
                    594: {
                    595:   do_pending_stack_adjust ();
                    596:   emit_jump_insn (gen_jump (label));
                    597:   emit_barrier ();
                    598: }
                    599: 
                    600: /* Emit code to jump to the address
                    601:    specified by the pointer expression EXP.  */
                    602: 
                    603: void
                    604: expand_computed_goto (exp)
                    605:      tree exp;
                    606: {
                    607:   if (output_bytecode)
                    608:     {
                    609:       bc_expand_expr (exp);
                    610:       bc_emit_instruction (jumpP);
                    611:     }
                    612:   else
                    613:     {
                    614:       rtx x = expand_expr (exp, NULL_RTX, VOIDmode, 0);
                    615:       emit_queue ();
                    616:       emit_indirect_jump (x);
                    617:     }
                    618: }
                    619: 
                    620: /* Handle goto statements and the labels that they can go to.  */
                    621: 
                    622: /* Specify the location in the RTL code of a label LABEL,
                    623:    which is a LABEL_DECL tree node.
                    624: 
                    625:    This is used for the kind of label that the user can jump to with a
                    626:    goto statement, and for alternatives of a switch or case statement.
                    627:    RTL labels generated for loops and conditionals don't go through here;
                    628:    they are generated directly at the RTL level, by other functions below.
                    629: 
                    630:    Note that this has nothing to do with defining label *names*.
                    631:    Languages vary in how they do that and what that even means.  */
                    632: 
                    633: void
                    634: expand_label (label)
                    635:      tree label;
                    636: {
                    637:   struct label_chain *p;
                    638: 
                    639:   if (output_bytecode)
                    640:     {
                    641:       if (! DECL_RTL (label))
                    642:        DECL_RTL (label) = bc_gen_rtx ((char *) 0, 0, bc_get_bytecode_label ());
                    643:       if (! bc_emit_bytecode_labeldef (BYTECODE_BC_LABEL (DECL_RTL (label))))
                    644:        error ("multiply defined label");
                    645:       return;
                    646:     }
                    647: 
                    648:   do_pending_stack_adjust ();
                    649:   emit_label (label_rtx (label));
                    650:   if (DECL_NAME (label))
                    651:     LABEL_NAME (DECL_RTL (label)) = IDENTIFIER_POINTER (DECL_NAME (label));
                    652: 
                    653:   if (stack_block_stack != 0)
                    654:     {
                    655:       p = (struct label_chain *) oballoc (sizeof (struct label_chain));
                    656:       p->next = stack_block_stack->data.block.label_chain;
                    657:       stack_block_stack->data.block.label_chain = p;
                    658:       p->label = label;
                    659:     }
                    660: }
                    661: 
                    662: /* Declare that LABEL (a LABEL_DECL) may be used for nonlocal gotos
                    663:    from nested functions.  */
                    664: 
                    665: void
                    666: declare_nonlocal_label (label)
                    667:      tree label;
                    668: {
                    669:   nonlocal_labels = tree_cons (NULL_TREE, label, nonlocal_labels);
                    670:   LABEL_PRESERVE_P (label_rtx (label)) = 1;
                    671:   if (nonlocal_goto_handler_slot == 0)
                    672:     {
                    673:       nonlocal_goto_handler_slot
                    674:        = assign_stack_local (Pmode, GET_MODE_SIZE (Pmode), 0);
                    675:       emit_stack_save (SAVE_NONLOCAL,
                    676:                       &nonlocal_goto_stack_level,
                    677:                       PREV_INSN (tail_recursion_reentry));
                    678:     }
                    679: }
                    680: 
                    681: /* Generate RTL code for a `goto' statement with target label LABEL.
                    682:    LABEL should be a LABEL_DECL tree node that was or will later be
                    683:    defined with `expand_label'.  */
                    684: 
                    685: void
                    686: expand_goto (label)
                    687:      tree label;
                    688: {
                    689:   tree context;
                    690: 
                    691:   if (output_bytecode)
                    692:     {
                    693:       expand_goto_internal (label, label_rtx (label), NULL_RTX);
                    694:       return;
                    695:     }
                    696: 
                    697:   /* Check for a nonlocal goto to a containing function.  */
                    698:   context = decl_function_context (label);
                    699:   if (context != 0 && context != current_function_decl)
                    700:     {
                    701:       struct function *p = find_function_data (context);
                    702:       rtx label_ref = gen_rtx (LABEL_REF, Pmode, label_rtx (label));
                    703:       rtx temp;
                    704: 
                    705:       p->has_nonlocal_label = 1;
                    706:       current_function_has_nonlocal_goto = 1;
                    707:       LABEL_REF_NONLOCAL_P (label_ref) = 1;
                    708: 
                    709:       /* Copy the rtl for the slots so that they won't be shared in
                    710:         case the virtual stack vars register gets instantiated differently
                    711:         in the parent than in the child.  */
                    712: 
                    713: #if HAVE_nonlocal_goto
                    714:       if (HAVE_nonlocal_goto)
                    715:        emit_insn (gen_nonlocal_goto (lookup_static_chain (label),
                    716:                                      copy_rtx (p->nonlocal_goto_handler_slot),
                    717:                                      copy_rtx (p->nonlocal_goto_stack_level),
                    718:                                      label_ref));
                    719:       else
                    720: #endif
                    721:        {
                    722:          rtx addr;
                    723: 
                    724:          /* Restore frame pointer for containing function.
                    725:             This sets the actual hard register used for the frame pointer
                    726:             to the location of the function's incoming static chain info.
                    727:             The non-local goto handler will then adjust it to contain the
                    728:             proper value and reload the argument pointer, if needed.  */
                    729:          emit_move_insn (hard_frame_pointer_rtx, lookup_static_chain (label));
                    730: 
                    731:          /* We have now loaded the frame pointer hardware register with
                    732:             the address of that corresponds to the start of the virtual
                    733:             stack vars.  So replace virtual_stack_vars_rtx in all
                    734:             addresses we use with stack_pointer_rtx.  */
                    735: 
                    736:          /* Get addr of containing function's current nonlocal goto handler,
                    737:             which will do any cleanups and then jump to the label.  */
                    738:          addr = copy_rtx (p->nonlocal_goto_handler_slot);
                    739:          temp = copy_to_reg (replace_rtx (addr, virtual_stack_vars_rtx,
                    740:                                           hard_frame_pointer_rtx));
                    741:          
                    742:          /* Restore the stack pointer.  Note this uses fp just restored.  */
                    743:          addr = p->nonlocal_goto_stack_level;
                    744:          if (addr)
                    745:            addr = replace_rtx (copy_rtx (addr),
                    746:                                virtual_stack_vars_rtx,
                    747:                                hard_frame_pointer_rtx);
                    748: 
                    749:          emit_stack_restore (SAVE_NONLOCAL, addr, NULL_RTX);
                    750: 
                    751:          /* Put in the static chain register the nonlocal label address.  */
                    752:          emit_move_insn (static_chain_rtx, label_ref);
                    753:          /* USE of hard_frame_pointer_rtx added for consistency; not clear if
                    754:             really needed.  */
                    755:          emit_insn (gen_rtx (USE, VOIDmode, hard_frame_pointer_rtx));
                    756:          emit_insn (gen_rtx (USE, VOIDmode, stack_pointer_rtx));
                    757:          emit_insn (gen_rtx (USE, VOIDmode, static_chain_rtx));
                    758:          emit_indirect_jump (temp);
                    759:        }
                    760:      }
                    761:   else
                    762:     expand_goto_internal (label, label_rtx (label), NULL_RTX);
                    763: }
                    764: 
                    765: /* Generate RTL code for a `goto' statement with target label BODY.
                    766:    LABEL should be a LABEL_REF.
                    767:    LAST_INSN, if non-0, is the rtx we should consider as the last
                    768:    insn emitted (for the purposes of cleaning up a return).  */
                    769: 
                    770: static void
                    771: expand_goto_internal (body, label, last_insn)
                    772:      tree body;
                    773:      rtx label;
                    774:      rtx last_insn;
                    775: {
                    776:   struct nesting *block;
                    777:   rtx stack_level = 0;
                    778: 
                    779:   /* NOTICE!  If a bytecode instruction other than `jump' is needed,
                    780:      then the caller has to call bc_expand_goto_internal()
                    781:      directly. This is rather an exceptional case, and there aren't
                    782:      that many places where this is necessary. */
                    783:   if (output_bytecode)
                    784:     {
                    785:       expand_goto_internal (body, label, last_insn);
                    786:       return;
                    787:     }
                    788: 
                    789:   if (GET_CODE (label) != CODE_LABEL)
                    790:     abort ();
                    791: 
                    792:   /* If label has already been defined, we can tell now
                    793:      whether and how we must alter the stack level.  */
                    794: 
                    795:   if (PREV_INSN (label) != 0)
                    796:     {
                    797:       /* Find the innermost pending block that contains the label.
                    798:         (Check containment by comparing insn-uids.)
                    799:         Then restore the outermost stack level within that block,
                    800:         and do cleanups of all blocks contained in it.  */
                    801:       for (block = block_stack; block; block = block->next)
                    802:        {
                    803:          if (INSN_UID (block->data.block.first_insn) < INSN_UID (label))
                    804:            break;
                    805:          if (block->data.block.stack_level != 0)
                    806:            stack_level = block->data.block.stack_level;
                    807:          /* Execute the cleanups for blocks we are exiting.  */
                    808:          if (block->data.block.cleanups != 0)
                    809:            {
                    810:              expand_cleanups (block->data.block.cleanups, NULL_TREE);
                    811:              do_pending_stack_adjust ();
                    812:            }
                    813:        }
                    814: 
                    815:       if (stack_level)
                    816:        {
                    817:          /* Ensure stack adjust isn't done by emit_jump, as this would clobber
                    818:             the stack pointer.  This one should be deleted as dead by flow. */
                    819:          clear_pending_stack_adjust ();
                    820:          do_pending_stack_adjust ();
                    821:          emit_stack_restore (SAVE_BLOCK, stack_level, NULL_RTX);
                    822:        }
                    823: 
                    824:       if (body != 0 && DECL_TOO_LATE (body))
                    825:        error ("jump to `%s' invalidly jumps into binding contour",
                    826:               IDENTIFIER_POINTER (DECL_NAME (body)));
                    827:     }
                    828:   /* Label not yet defined: may need to put this goto
                    829:      on the fixup list.  */
                    830:   else if (! expand_fixup (body, label, last_insn))
                    831:     {
                    832:       /* No fixup needed.  Record that the label is the target
                    833:         of at least one goto that has no fixup.  */
                    834:       if (body != 0)
                    835:        TREE_ADDRESSABLE (body) = 1;
                    836:     }
                    837: 
                    838:   emit_jump (label);
                    839: }
                    840: 
                    841: /* Generate a jump with OPCODE to the given bytecode LABEL which is
                    842:    found within BODY. */
                    843: static void
                    844: bc_expand_goto_internal (opcode, label, body)
                    845:      enum bytecode_opcode opcode;
                    846:      struct bc_label *label;
                    847:      tree body;
                    848: {
                    849:   struct nesting *block;
                    850:   int stack_level = -1;
                    851: 
                    852:   /* If the label is defined, adjust the stack as necessary.
                    853:      If it's not defined, we have to push the reference on the
                    854:      fixup list. */
                    855: 
                    856:   if (label->defined)
                    857:     {
                    858: 
                    859:       /* Find the innermost pending block that contains the label.
                    860:         (Check containment by comparing bytecode uids.)  Then restore the
                    861:         outermost stack level within that block.  */
                    862: 
                    863:       for (block = block_stack; block; block = block->next)
                    864:        {
                    865:          if (BYTECODE_BC_LABEL (block->data.block.first_insn)->uid < label->uid)
                    866:            break;
                    867:          if (block->data.block.bc_stack_level)
                    868:            stack_level = block->data.block.bc_stack_level;
                    869: 
                    870:          /* Execute the cleanups for blocks we are exiting.  */
                    871:          if (block->data.block.cleanups != 0)
                    872:            {
                    873:              expand_cleanups (block->data.block.cleanups, NULL_TREE);
                    874:              do_pending_stack_adjust ();
                    875:            }
                    876:        }
                    877: 
                    878:       /* Restore the stack level. If we need to adjust the stack, we
                    879:         must do so after the jump, since the jump may depend on
                    880:         what's on the stack.  Thus, any stack-modifying conditional
                    881:         jumps (these are the only ones that rely on what's on the
                    882:         stack) go into the fixup list. */
                    883: 
                    884:       if (stack_level >= 0
                    885:          && stack_depth != stack_level
                    886:          && opcode != jump)
                    887: 
                    888:        bc_expand_fixup (opcode, label, stack_level);
                    889:       else
                    890:        {
                    891:          if (stack_level >= 0)
                    892:            bc_adjust_stack (stack_depth - stack_level);
                    893: 
                    894:          if (body && DECL_BIT_FIELD (body))
                    895:            error ("jump to `%s' invalidly jumps into binding contour",
                    896:                   IDENTIFIER_POINTER (DECL_NAME (body)));
                    897:          
                    898:          /* Emit immediate jump */
                    899:          bc_emit_bytecode (opcode);
                    900:          bc_emit_bytecode_labelref (label);
                    901:          
                    902: #ifdef DEBUG_PRINT_CODE
                    903:          fputc ('\n', stderr);
                    904: #endif
                    905:        }
                    906:     }
                    907:   else
                    908:     /* Put goto in the fixup list */
                    909:     bc_expand_fixup (opcode, label, stack_level);
                    910: }
                    911: 
                    912: /* Generate if necessary a fixup for a goto
                    913:    whose target label in tree structure (if any) is TREE_LABEL
                    914:    and whose target in rtl is RTL_LABEL.
                    915: 
                    916:    If LAST_INSN is nonzero, we pretend that the jump appears
                    917:    after insn LAST_INSN instead of at the current point in the insn stream.
                    918: 
                    919:    The fixup will be used later to insert insns just before the goto.
                    920:    Those insns will restore the stack level as appropriate for the
                    921:    target label, and will (in the case of C++) also invoke any object
                    922:    destructors which have to be invoked when we exit the scopes which
                    923:    are exited by the goto.
                    924: 
                    925:    Value is nonzero if a fixup is made.  */
                    926: 
                    927: static int
                    928: expand_fixup (tree_label, rtl_label, last_insn)
                    929:      tree tree_label;
                    930:      rtx rtl_label;
                    931:      rtx last_insn;
                    932: {
                    933:   struct nesting *block, *end_block;
                    934: 
                    935:   /* See if we can recognize which block the label will be output in.
                    936:      This is possible in some very common cases.
                    937:      If we succeed, set END_BLOCK to that block.
                    938:      Otherwise, set it to 0.  */
                    939: 
                    940:   if (cond_stack
                    941:       && (rtl_label == cond_stack->data.cond.endif_label
                    942:          || rtl_label == cond_stack->data.cond.next_label))
                    943:     end_block = cond_stack;
                    944:   /* If we are in a loop, recognize certain labels which
                    945:      are likely targets.  This reduces the number of fixups
                    946:      we need to create.  */
                    947:   else if (loop_stack
                    948:       && (rtl_label == loop_stack->data.loop.start_label
                    949:          || rtl_label == loop_stack->data.loop.end_label
                    950:          || rtl_label == loop_stack->data.loop.continue_label))
                    951:     end_block = loop_stack;
                    952:   else
                    953:     end_block = 0;
                    954: 
                    955:   /* Now set END_BLOCK to the binding level to which we will return.  */
                    956: 
                    957:   if (end_block)
                    958:     {
                    959:       struct nesting *next_block = end_block->all;
                    960:       block = block_stack;
                    961: 
                    962:       /* First see if the END_BLOCK is inside the innermost binding level.
                    963:         If so, then no cleanups or stack levels are relevant.  */
                    964:       while (next_block && next_block != block)
                    965:        next_block = next_block->all;
                    966: 
                    967:       if (next_block)
                    968:        return 0;
                    969: 
                    970:       /* Otherwise, set END_BLOCK to the innermost binding level
                    971:         which is outside the relevant control-structure nesting.  */
                    972:       next_block = block_stack->next;
                    973:       for (block = block_stack; block != end_block; block = block->all)
                    974:        if (block == next_block)
                    975:          next_block = next_block->next;
                    976:       end_block = next_block;
                    977:     }
                    978: 
                    979:   /* Does any containing block have a stack level or cleanups?
                    980:      If not, no fixup is needed, and that is the normal case
                    981:      (the only case, for standard C).  */
                    982:   for (block = block_stack; block != end_block; block = block->next)
                    983:     if (block->data.block.stack_level != 0
                    984:        || block->data.block.cleanups != 0)
                    985:       break;
                    986: 
                    987:   if (block != end_block)
                    988:     {
                    989:       /* Ok, a fixup is needed.  Add a fixup to the list of such.  */
                    990:       struct goto_fixup *fixup
                    991:        = (struct goto_fixup *) oballoc (sizeof (struct goto_fixup));
                    992:       /* In case an old stack level is restored, make sure that comes
                    993:         after any pending stack adjust.  */
                    994:       /* ?? If the fixup isn't to come at the present position,
                    995:         doing the stack adjust here isn't useful.  Doing it with our
                    996:         settings at that location isn't useful either.  Let's hope
                    997:         someone does it!  */
                    998:       if (last_insn == 0)
                    999:        do_pending_stack_adjust ();
                   1000:       fixup->target = tree_label;
                   1001:       fixup->target_rtl = rtl_label;
                   1002: 
                   1003:       /* Create a BLOCK node and a corresponding matched set of
                   1004:         NOTE_INSN_BEGIN_BLOCK and NOTE_INSN_END_BLOCK notes at
                   1005:         this point.  The notes will encapsulate any and all fixup
                   1006:         code which we might later insert at this point in the insn
                   1007:         stream.  Also, the BLOCK node will be the parent (i.e. the
                   1008:         `SUPERBLOCK') of any other BLOCK nodes which we might create
                   1009:         later on when we are expanding the fixup code.  */
                   1010: 
                   1011:       {
                   1012:         register rtx original_before_jump
                   1013:           = last_insn ? last_insn : get_last_insn ();
                   1014: 
                   1015:         start_sequence ();
                   1016:         pushlevel (0);
                   1017:         fixup->before_jump = emit_note (NULL_PTR, NOTE_INSN_BLOCK_BEG);
                   1018:         last_block_end_note = emit_note (NULL_PTR, NOTE_INSN_BLOCK_END);
                   1019:         fixup->context = poplevel (1, 0, 0);  /* Create the BLOCK node now! */
                   1020:         end_sequence ();
                   1021:         emit_insns_after (fixup->before_jump, original_before_jump);
                   1022:       }
                   1023: 
                   1024:       fixup->block_start_count = block_start_count;
                   1025:       fixup->stack_level = 0;
                   1026:       fixup->cleanup_list_list
                   1027:        = (((block->data.block.outer_cleanups
                   1028: #if 0
                   1029:             && block->data.block.outer_cleanups != empty_cleanup_list
                   1030: #endif
                   1031:             )
                   1032:            || block->data.block.cleanups)
                   1033:           ? tree_cons (NULL_TREE, block->data.block.cleanups,
                   1034:                        block->data.block.outer_cleanups)
                   1035:           : 0);
                   1036:       fixup->next = goto_fixup_chain;
                   1037:       goto_fixup_chain = fixup;
                   1038:     }
                   1039: 
                   1040:   return block != 0;
                   1041: }
                   1042: 
                   1043: 
                   1044: /* Generate bytecode jump with OPCODE to a fixup routine that links to LABEL.
                   1045:    Make the fixup restore the stack level to STACK_LEVEL.  */
                   1046: 
                   1047: static void
                   1048: bc_expand_fixup (opcode, label, stack_level)
                   1049:      enum bytecode_opcode opcode;
                   1050:      struct bc_label *label;
                   1051:      int stack_level;
                   1052: {
                   1053:   struct goto_fixup *fixup
                   1054:     = (struct goto_fixup *) oballoc (sizeof (struct goto_fixup));
                   1055: 
                   1056:   fixup->label  = bc_get_bytecode_label ();
                   1057:   fixup->bc_target = label;
                   1058:   fixup->bc_stack_level = stack_level;
                   1059:   fixup->bc_handled = FALSE;
                   1060: 
                   1061:   fixup->next = goto_fixup_chain;
                   1062:   goto_fixup_chain = fixup;
                   1063: 
                   1064:   /* Insert a jump to the fixup code */
                   1065:   bc_emit_bytecode (opcode);
                   1066:   bc_emit_bytecode_labelref (fixup->label);
                   1067: 
                   1068: #ifdef DEBUG_PRINT_CODE
                   1069:   fputc ('\n', stderr);
                   1070: #endif
                   1071: }
                   1072: 
                   1073: 
                   1074: /* When exiting a binding contour, process all pending gotos requiring fixups.
                   1075:    THISBLOCK is the structure that describes the block being exited.
                   1076:    STACK_LEVEL is the rtx for the stack level to restore exiting this contour.
                   1077:    CLEANUP_LIST is a list of expressions to evaluate on exiting this contour.
                   1078:    FIRST_INSN is the insn that began this contour.
                   1079: 
                   1080:    Gotos that jump out of this contour must restore the
                   1081:    stack level and do the cleanups before actually jumping.
                   1082: 
                   1083:    DONT_JUMP_IN nonzero means report error there is a jump into this
                   1084:    contour from before the beginning of the contour.
                   1085:    This is also done if STACK_LEVEL is nonzero.  */
                   1086: 
                   1087: void
                   1088: fixup_gotos (thisblock, stack_level, cleanup_list, first_insn, dont_jump_in)
                   1089:      struct nesting *thisblock;
                   1090:      rtx stack_level;
                   1091:      tree cleanup_list;
                   1092:      rtx first_insn;
                   1093:      int dont_jump_in;
                   1094: {
                   1095:   register struct goto_fixup *f, *prev;
                   1096: 
                   1097:   if (output_bytecode)
                   1098:     {
                   1099:       bc_fixup_gotos (thisblock, stack_level, cleanup_list, first_insn, dont_jump_in);
                   1100:       return;
                   1101:     }
                   1102: 
                   1103:   /* F is the fixup we are considering; PREV is the previous one.  */
                   1104:   /* We run this loop in two passes so that cleanups of exited blocks
                   1105:      are run first, and blocks that are exited are marked so
                   1106:      afterwards.  */
                   1107: 
                   1108:   for (prev = 0, f = goto_fixup_chain; f; prev = f, f = f->next)
                   1109:     {
                   1110:       /* Test for a fixup that is inactive because it is already handled.  */
                   1111:       if (f->before_jump == 0)
                   1112:        {
                   1113:          /* Delete inactive fixup from the chain, if that is easy to do.  */
                   1114:          if (prev != 0)
                   1115:            prev->next = f->next;
                   1116:        }
                   1117:       /* Has this fixup's target label been defined?
                   1118:         If so, we can finalize it.  */
                   1119:       else if (PREV_INSN (f->target_rtl) != 0)
                   1120:        {
                   1121:          register rtx cleanup_insns;
                   1122: 
                   1123:          /* Get the first non-label after the label
                   1124:             this goto jumps to.  If that's before this scope begins,
                   1125:             we don't have a jump into the scope.  */
                   1126:          rtx after_label = f->target_rtl;
                   1127:          while (after_label != 0 && GET_CODE (after_label) == CODE_LABEL)
                   1128:            after_label = NEXT_INSN (after_label);
                   1129: 
                   1130:          /* If this fixup jumped into this contour from before the beginning
                   1131:             of this contour, report an error.  */
                   1132:          /* ??? Bug: this does not detect jumping in through intermediate
                   1133:             blocks that have stack levels or cleanups.
                   1134:             It detects only a problem with the innermost block
                   1135:             around the label.  */
                   1136:          if (f->target != 0
                   1137:              && (dont_jump_in || stack_level || cleanup_list)
                   1138:              /* If AFTER_LABEL is 0, it means the jump goes to the end
                   1139:                 of the rtl, which means it jumps into this scope.  */
                   1140:              && (after_label == 0
                   1141:                  || INSN_UID (first_insn) < INSN_UID (after_label))
                   1142:              && INSN_UID (first_insn) > INSN_UID (f->before_jump)
                   1143:              && ! DECL_REGISTER (f->target))
                   1144:            {
                   1145:              error_with_decl (f->target,
                   1146:                               "label `%s' used before containing binding contour");
                   1147:              /* Prevent multiple errors for one label.  */
                   1148:              DECL_REGISTER (f->target) = 1;
                   1149:            }
                   1150: 
                   1151:          /* We will expand the cleanups into a sequence of their own and
                   1152:             then later on we will attach this new sequence to the insn
                   1153:             stream just ahead of the actual jump insn.  */
                   1154: 
                   1155:          start_sequence ();
                   1156: 
                   1157:          /* Temporarily restore the lexical context where we will
                   1158:             logically be inserting the fixup code.  We do this for the
                   1159:             sake of getting the debugging information right.  */
                   1160: 
                   1161:          pushlevel (0);
                   1162:          set_block (f->context);
                   1163: 
                   1164:          /* Expand the cleanups for blocks this jump exits.  */
                   1165:          if (f->cleanup_list_list)
                   1166:            {
                   1167:              tree lists;
                   1168:              for (lists = f->cleanup_list_list; lists; lists = TREE_CHAIN (lists))
                   1169:                /* Marked elements correspond to blocks that have been closed.
                   1170:                   Do their cleanups.  */
                   1171:                if (TREE_ADDRESSABLE (lists)
                   1172:                    && TREE_VALUE (lists) != 0)
                   1173:                  {
                   1174:                    expand_cleanups (TREE_VALUE (lists), 0);
                   1175:                    /* Pop any pushes done in the cleanups,
                   1176:                       in case function is about to return.  */
                   1177:                    do_pending_stack_adjust ();
                   1178:                  }
                   1179:            }
                   1180: 
                   1181:          /* Restore stack level for the biggest contour that this
                   1182:             jump jumps out of.  */
                   1183:          if (f->stack_level)
                   1184:            emit_stack_restore (SAVE_BLOCK, f->stack_level, f->before_jump);
                   1185: 
                   1186:          /* Finish up the sequence containing the insns which implement the
                   1187:             necessary cleanups, and then attach that whole sequence to the
                   1188:             insn stream just ahead of the actual jump insn.  Attaching it
                   1189:             at that point insures that any cleanups which are in fact
                   1190:             implicit C++ object destructions (which must be executed upon
                   1191:             leaving the block) appear (to the debugger) to be taking place
                   1192:             in an area of the generated code where the object(s) being
                   1193:             destructed are still "in scope".  */
                   1194: 
                   1195:          cleanup_insns = get_insns ();
                   1196:          poplevel (1, 0, 0);
                   1197: 
                   1198:          end_sequence ();
                   1199:          emit_insns_after (cleanup_insns, f->before_jump);
                   1200: 
                   1201: 
                   1202:          f->before_jump = 0;
                   1203:        }
                   1204:     }
                   1205: 
                   1206:   /* Mark the cleanups of exited blocks so that they are executed
                   1207:      by the code above.  */
                   1208:   for (prev = 0, f = goto_fixup_chain; f; prev = f, f = f->next)
                   1209:     if (f->before_jump != 0
                   1210:        && PREV_INSN (f->target_rtl) == 0
                   1211:        /* Label has still not appeared.  If we are exiting a block with
                   1212:           a stack level to restore, that started before the fixup,
                   1213:           mark this stack level as needing restoration
                   1214:           when the fixup is later finalized.
                   1215:           Also mark the cleanup_list_list element for F
                   1216:           that corresponds to this block, so that ultimately
                   1217:           this block's cleanups will be executed by the code above.  */
                   1218:        && thisblock != 0
                   1219:        /* Note: if THISBLOCK == 0 and we have a label that hasn't appeared,
                   1220:           it means the label is undefined.  That's erroneous, but possible.  */
                   1221:        && (thisblock->data.block.block_start_count
                   1222:            <= f->block_start_count))
                   1223:       {
                   1224:        tree lists = f->cleanup_list_list;
                   1225:        for (; lists; lists = TREE_CHAIN (lists))
                   1226:          /* If the following elt. corresponds to our containing block
                   1227:             then the elt. must be for this block.  */
                   1228:          if (TREE_CHAIN (lists) == thisblock->data.block.outer_cleanups)
                   1229:            TREE_ADDRESSABLE (lists) = 1;
                   1230: 
                   1231:        if (stack_level)
                   1232:          f->stack_level = stack_level;
                   1233:       }
                   1234: }
                   1235: 
                   1236: 
                   1237: /* When exiting a binding contour, process all pending gotos requiring fixups.
                   1238:    Note: STACK_DEPTH is not altered.
                   1239: 
                   1240:    The arguments are currently not used in the bytecode compiler, but we may need
                   1241:    them one day for languages other than C.
                   1242: 
                   1243:    THISBLOCK is the structure that describes the block being exited.
                   1244:    STACK_LEVEL is the rtx for the stack level to restore exiting this contour.
                   1245:    CLEANUP_LIST is a list of expressions to evaluate on exiting this contour.
                   1246:    FIRST_INSN is the insn that began this contour.
                   1247: 
                   1248:    Gotos that jump out of this contour must restore the
                   1249:    stack level and do the cleanups before actually jumping.
                   1250: 
                   1251:    DONT_JUMP_IN nonzero means report error there is a jump into this
                   1252:    contour from before the beginning of the contour.
                   1253:    This is also done if STACK_LEVEL is nonzero.  */
                   1254: 
                   1255: static void
                   1256: bc_fixup_gotos (thisblock, stack_level, cleanup_list, first_insn, dont_jump_in)
                   1257:      struct nesting *thisblock;
                   1258:      int stack_level;
                   1259:      tree cleanup_list;
                   1260:      rtx first_insn;
                   1261:      int dont_jump_in;
                   1262: {
                   1263:   register struct goto_fixup *f, *prev;
                   1264:   int saved_stack_depth;
                   1265: 
                   1266:   /* F is the fixup we are considering; PREV is the previous one.  */
                   1267: 
                   1268:   for (prev = 0, f = goto_fixup_chain; f; prev = f, f = f->next)
                   1269:     {
                   1270:       /* Test for a fixup that is inactive because it is already handled.  */
                   1271:       if (f->before_jump == 0)
                   1272:        {
                   1273:          /* Delete inactive fixup from the chain, if that is easy to do.  */
                   1274:          if (prev)
                   1275:            prev->next = f->next;
                   1276:        }
                   1277: 
                   1278:       /* Emit code to restore the stack and continue */
                   1279:       bc_emit_bytecode_labeldef (f->label);
                   1280: 
                   1281:       /* Save stack_depth across call, since bc_adjust_stack () will alter
                   1282:          the perceived stack depth via the instructions generated. */
                   1283: 
                   1284:       if (f->bc_stack_level >= 0)
                   1285:        {
                   1286:          saved_stack_depth = stack_depth;
                   1287:          bc_adjust_stack (stack_depth - f->bc_stack_level);
                   1288:          stack_depth = saved_stack_depth;
                   1289:        }
                   1290: 
                   1291:       bc_emit_bytecode (jump);
                   1292:       bc_emit_bytecode_labelref (f->bc_target);
                   1293: 
                   1294: #ifdef DEBUG_PRINT_CODE
                   1295:   fputc ('\n', stderr);
                   1296: #endif
                   1297:     }
                   1298: 
                   1299:   goto_fixup_chain = NULL;
                   1300: }
                   1301: 
                   1302: /* Generate RTL for an asm statement (explicit assembler code).
                   1303:    BODY is a STRING_CST node containing the assembler code text,
                   1304:    or an ADDR_EXPR containing a STRING_CST.  */
                   1305: 
                   1306: void
                   1307: expand_asm (body)
                   1308:      tree body;
                   1309: {
                   1310:   if (output_bytecode)
                   1311:     {
                   1312:       error ("`asm' is illegal when generating bytecode");
                   1313:       return;
                   1314:     }
                   1315: 
                   1316:   if (TREE_CODE (body) == ADDR_EXPR)
                   1317:     body = TREE_OPERAND (body, 0);
                   1318: 
                   1319:   emit_insn (gen_rtx (ASM_INPUT, VOIDmode,
                   1320:                      TREE_STRING_POINTER (body)));
                   1321:   last_expr_type = 0;
                   1322: }
                   1323: 
                   1324: /* Generate RTL for an asm statement with arguments.
                   1325:    STRING is the instruction template.
                   1326:    OUTPUTS is a list of output arguments (lvalues); INPUTS a list of inputs.
                   1327:    Each output or input has an expression in the TREE_VALUE and
                   1328:    a constraint-string in the TREE_PURPOSE.
                   1329:    CLOBBERS is a list of STRING_CST nodes each naming a hard register
                   1330:    that is clobbered by this insn.
                   1331: 
                   1332:    Not all kinds of lvalue that may appear in OUTPUTS can be stored directly.
                   1333:    Some elements of OUTPUTS may be replaced with trees representing temporary
                   1334:    values.  The caller should copy those temporary values to the originally
                   1335:    specified lvalues.
                   1336: 
                   1337:    VOL nonzero means the insn is volatile; don't optimize it.  */
                   1338: 
                   1339: void
                   1340: expand_asm_operands (string, outputs, inputs, clobbers, vol, filename, line)
                   1341:      tree string, outputs, inputs, clobbers;
                   1342:      int vol;
                   1343:      char *filename;
                   1344:      int line;
                   1345: {
                   1346:   rtvec argvec, constraints;
                   1347:   rtx body;
                   1348:   int ninputs = list_length (inputs);
                   1349:   int noutputs = list_length (outputs);
                   1350:   int nclobbers;
                   1351:   tree tail;
                   1352:   register int i;
                   1353:   /* Vector of RTX's of evaluated output operands.  */
                   1354:   rtx *output_rtx = (rtx *) alloca (noutputs * sizeof (rtx));
                   1355:   /* The insn we have emitted.  */
                   1356:   rtx insn;
                   1357: 
                   1358:   if (output_bytecode)
                   1359:     {
                   1360:       error ("`asm' is illegal when generating bytecode");
                   1361:       return;
                   1362:     }
                   1363: 
                   1364:   /* Count the number of meaningful clobbered registers, ignoring what
                   1365:      we would ignore later.  */
                   1366:   nclobbers = 0;
                   1367:   for (tail = clobbers; tail; tail = TREE_CHAIN (tail))
                   1368:     {
                   1369:       char *regname = TREE_STRING_POINTER (TREE_VALUE (tail));
                   1370:       i = decode_reg_name (regname);
                   1371:       if (i >= 0 || i == -4)
                   1372:        ++nclobbers;
                   1373:     }
                   1374: 
                   1375:   last_expr_type = 0;
                   1376: 
                   1377:   for (i = 0, tail = outputs; tail; tail = TREE_CHAIN (tail), i++)
                   1378:     {
                   1379:       tree val = TREE_VALUE (tail);
                   1380:       tree val1;
                   1381:       int j;
                   1382:       int found_equal;
                   1383: 
                   1384:       /* If there's an erroneous arg, emit no insn.  */
                   1385:       if (TREE_TYPE (val) == error_mark_node)
                   1386:        return;
                   1387: 
                   1388:       /* Make sure constraint has `=' and does not have `+'.  */
                   1389: 
                   1390:       found_equal = 0;
                   1391:       for (j = 0; j < TREE_STRING_LENGTH (TREE_PURPOSE (tail)); j++)
                   1392:        {
                   1393:          if (TREE_STRING_POINTER (TREE_PURPOSE (tail))[j] == '+')
                   1394:            {
                   1395:              error ("output operand constraint contains `+'");
                   1396:              return;
                   1397:            }
                   1398:          if (TREE_STRING_POINTER (TREE_PURPOSE (tail))[j] == '=')
                   1399:            found_equal = 1;
                   1400:        }
                   1401:       if (! found_equal)
                   1402:        {
                   1403:          error ("output operand constraint lacks `='");
                   1404:          return;
                   1405:        }
                   1406: 
                   1407:       /* If an output operand is not a variable or indirect ref,
                   1408:         or a part of one,
                   1409:         create a SAVE_EXPR which is a pseudo-reg
                   1410:         to act as an intermediate temporary.
                   1411:         Make the asm insn write into that, then copy it to
                   1412:         the real output operand.  */
                   1413: 
                   1414:       while (TREE_CODE (val) == COMPONENT_REF
                   1415:             || TREE_CODE (val) == ARRAY_REF)
                   1416:        val = TREE_OPERAND (val, 0);
                   1417: 
                   1418:       if (TREE_CODE (val) != VAR_DECL
                   1419:          && TREE_CODE (val) != PARM_DECL
                   1420:          && TREE_CODE (val) != INDIRECT_REF)
                   1421:        {
                   1422:          TREE_VALUE (tail) = save_expr (TREE_VALUE (tail));
                   1423:          /* If it's a constant, print error now so don't crash later.  */
                   1424:          if (TREE_CODE (TREE_VALUE (tail)) != SAVE_EXPR)
                   1425:            {
                   1426:              error ("invalid output in `asm'");
                   1427:              return;
                   1428:            }
                   1429:        }
                   1430: 
                   1431:       output_rtx[i] = expand_expr (TREE_VALUE (tail), NULL_RTX, VOIDmode, 0);
                   1432:     }
                   1433: 
                   1434:   if (ninputs + noutputs > MAX_RECOG_OPERANDS)
                   1435:     {
                   1436:       error ("more than %d operands in `asm'", MAX_RECOG_OPERANDS);
                   1437:       return;
                   1438:     }
                   1439: 
                   1440:   /* Make vectors for the expression-rtx and constraint strings.  */
                   1441: 
                   1442:   argvec = rtvec_alloc (ninputs);
                   1443:   constraints = rtvec_alloc (ninputs);
                   1444: 
                   1445:   body = gen_rtx (ASM_OPERANDS, VOIDmode,
                   1446:                  TREE_STRING_POINTER (string), "", 0, argvec, constraints,
                   1447:                  filename, line);
                   1448:   MEM_VOLATILE_P (body) = vol;
                   1449: 
                   1450:   /* Eval the inputs and put them into ARGVEC.
                   1451:      Put their constraints into ASM_INPUTs and store in CONSTRAINTS.  */
                   1452: 
                   1453:   i = 0;
                   1454:   for (tail = inputs; tail; tail = TREE_CHAIN (tail))
                   1455:     {
                   1456:       int j;
                   1457: 
                   1458:       /* If there's an erroneous arg, emit no insn,
                   1459:         because the ASM_INPUT would get VOIDmode
                   1460:         and that could cause a crash in reload.  */
                   1461:       if (TREE_TYPE (TREE_VALUE (tail)) == error_mark_node)
                   1462:        return;
                   1463:       if (TREE_PURPOSE (tail) == NULL_TREE)
                   1464:        {
                   1465:          error ("hard register `%s' listed as input operand to `asm'",
                   1466:                 TREE_STRING_POINTER (TREE_VALUE (tail)) );
                   1467:          return;
                   1468:        }
                   1469: 
                   1470:       /* Make sure constraint has neither `=' nor `+'.  */
                   1471: 
                   1472:       for (j = 0; j < TREE_STRING_LENGTH (TREE_PURPOSE (tail)); j++)
                   1473:        if (TREE_STRING_POINTER (TREE_PURPOSE (tail))[j] == '='
                   1474:            || TREE_STRING_POINTER (TREE_PURPOSE (tail))[j] == '+')
                   1475:          {
                   1476:            error ("input operand constraint contains `%c'",
                   1477:                   TREE_STRING_POINTER (TREE_PURPOSE (tail))[j]);
                   1478:            return;
                   1479:          }
                   1480: 
                   1481:       XVECEXP (body, 3, i)      /* argvec */
                   1482:        = expand_expr (TREE_VALUE (tail), NULL_RTX, VOIDmode, 0);
                   1483:       XVECEXP (body, 4, i)      /* constraints */
                   1484:        = gen_rtx (ASM_INPUT, TYPE_MODE (TREE_TYPE (TREE_VALUE (tail))),
                   1485:                   TREE_STRING_POINTER (TREE_PURPOSE (tail)));
                   1486:       i++;
                   1487:     }
                   1488: 
                   1489:   /* Protect all the operands from the queue,
                   1490:      now that they have all been evaluated.  */
                   1491: 
                   1492:   for (i = 0; i < ninputs; i++)
                   1493:     XVECEXP (body, 3, i) = protect_from_queue (XVECEXP (body, 3, i), 0);
                   1494: 
                   1495:   for (i = 0; i < noutputs; i++)
                   1496:     output_rtx[i] = protect_from_queue (output_rtx[i], 1);
                   1497: 
                   1498:   /* Now, for each output, construct an rtx
                   1499:      (set OUTPUT (asm_operands INSN OUTPUTNUMBER OUTPUTCONSTRAINT
                   1500:                               ARGVEC CONSTRAINTS))
                   1501:      If there is more than one, put them inside a PARALLEL.  */
                   1502: 
                   1503:   if (noutputs == 1 && nclobbers == 0)
                   1504:     {
                   1505:       XSTR (body, 1) = TREE_STRING_POINTER (TREE_PURPOSE (outputs));
                   1506:       insn = emit_insn (gen_rtx (SET, VOIDmode, output_rtx[0], body));
                   1507:     }
                   1508:   else if (noutputs == 0 && nclobbers == 0)
                   1509:     {
                   1510:       /* No output operands: put in a raw ASM_OPERANDS rtx.  */
                   1511:       insn = emit_insn (body);
                   1512:     }
                   1513:   else
                   1514:     {
                   1515:       rtx obody = body;
                   1516:       int num = noutputs;
                   1517:       if (num == 0) num = 1;
                   1518:       body = gen_rtx (PARALLEL, VOIDmode, rtvec_alloc (num + nclobbers));
                   1519: 
                   1520:       /* For each output operand, store a SET.  */
                   1521: 
                   1522:       for (i = 0, tail = outputs; tail; tail = TREE_CHAIN (tail), i++)
                   1523:        {
                   1524:          XVECEXP (body, 0, i)
                   1525:            = gen_rtx (SET, VOIDmode,
                   1526:                       output_rtx[i],
                   1527:                       gen_rtx (ASM_OPERANDS, VOIDmode,
                   1528:                                TREE_STRING_POINTER (string),
                   1529:                                TREE_STRING_POINTER (TREE_PURPOSE (tail)),
                   1530:                                i, argvec, constraints,
                   1531:                                filename, line));
                   1532:          MEM_VOLATILE_P (SET_SRC (XVECEXP (body, 0, i))) = vol;
                   1533:        }
                   1534: 
                   1535:       /* If there are no outputs (but there are some clobbers)
                   1536:         store the bare ASM_OPERANDS into the PARALLEL.  */
                   1537: 
                   1538:       if (i == 0)
                   1539:        XVECEXP (body, 0, i++) = obody;
                   1540: 
                   1541:       /* Store (clobber REG) for each clobbered register specified.  */
                   1542: 
                   1543:       for (tail = clobbers; tail; tail = TREE_CHAIN (tail))
                   1544:        {
                   1545:          char *regname = TREE_STRING_POINTER (TREE_VALUE (tail));
                   1546:          int j = decode_reg_name (regname);
                   1547: 
                   1548:          if (j < 0)
                   1549:            {
                   1550:              if (j == -3)      /* `cc', which is not a register */
                   1551:                continue;
                   1552: 
                   1553:              if (j == -4)      /* `memory', don't cache memory across asm */
                   1554:                {
                   1555:                  XVECEXP (body, 0, i++)
                   1556:                    = gen_rtx (CLOBBER, VOIDmode,
                   1557:                               gen_rtx (MEM, QImode,
                   1558:                                        gen_rtx (SCRATCH, VOIDmode, 0)));
                   1559:                  continue;
                   1560:                }
                   1561: 
                   1562:              error ("unknown register name `%s' in `asm'", regname);
                   1563:              return;
                   1564:            }
                   1565: 
                   1566:          /* Use QImode since that's guaranteed to clobber just one reg.  */
                   1567:          XVECEXP (body, 0, i++)
                   1568:            = gen_rtx (CLOBBER, VOIDmode, gen_rtx (REG, QImode, j));
                   1569:        }
                   1570: 
                   1571:       insn = emit_insn (body);
                   1572:     }
                   1573: 
                   1574:   free_temp_slots ();
                   1575: }
                   1576: 
                   1577: /* Generate RTL to evaluate the expression EXP
                   1578:    and remember it in case this is the VALUE in a ({... VALUE; }) constr.  */
                   1579: 
                   1580: void
                   1581: expand_expr_stmt (exp)
                   1582:      tree exp;
                   1583: {
                   1584:   if (output_bytecode)
                   1585:     {
                   1586:       int org_stack_depth = stack_depth;
                   1587: 
                   1588:       bc_expand_expr (exp);
                   1589: 
                   1590:       /* Restore stack depth */
                   1591:       if (stack_depth < org_stack_depth)
                   1592:        abort ();
                   1593:       
                   1594:       bc_emit_instruction (drop);
                   1595: 
                   1596:       last_expr_type = TREE_TYPE (exp);
                   1597:       return;
                   1598:     }
                   1599: 
                   1600:   /* If -W, warn about statements with no side effects,
                   1601:      except for an explicit cast to void (e.g. for assert()), and
                   1602:      except inside a ({...}) where they may be useful.  */
                   1603:   if (expr_stmts_for_value == 0 && exp != error_mark_node)
                   1604:     {
                   1605:       if (! TREE_SIDE_EFFECTS (exp) && (extra_warnings || warn_unused)
                   1606:          && !(TREE_CODE (exp) == CONVERT_EXPR
                   1607:               && TREE_TYPE (exp) == void_type_node))
                   1608:        warning_with_file_and_line (emit_filename, emit_lineno,
                   1609:                                    "statement with no effect");
                   1610:       else if (warn_unused)
                   1611:        warn_if_unused_value (exp);
                   1612:     }
                   1613:   last_expr_type = TREE_TYPE (exp);
                   1614:   if (! flag_syntax_only)
                   1615:     last_expr_value = expand_expr (exp,
                   1616:                                   (expr_stmts_for_value
                   1617:                                    ? NULL_RTX : const0_rtx),
                   1618:                                   VOIDmode, 0);
                   1619: 
                   1620:   /* If all we do is reference a volatile value in memory,
                   1621:      copy it to a register to be sure it is actually touched.  */
                   1622:   if (last_expr_value != 0 && GET_CODE (last_expr_value) == MEM
                   1623:       && TREE_THIS_VOLATILE (exp))
                   1624:     {
                   1625:       if (TYPE_MODE (TREE_TYPE (exp)) == VOIDmode)
                   1626:        ;
                   1627:       else if (TYPE_MODE (TREE_TYPE (exp)) != BLKmode)
                   1628:        copy_to_reg (last_expr_value);
                   1629:       else
                   1630:        {
                   1631:          rtx lab = gen_label_rtx ();
                   1632:          
                   1633:          /* Compare the value with itself to reference it.  */
                   1634:          emit_cmp_insn (last_expr_value, last_expr_value, EQ,
                   1635:                         expand_expr (TYPE_SIZE (last_expr_type),
                   1636:                                      NULL_RTX, VOIDmode, 0),
                   1637:                         BLKmode, 0,
                   1638:                         TYPE_ALIGN (last_expr_type) / BITS_PER_UNIT);
                   1639:          emit_jump_insn ((*bcc_gen_fctn[(int) EQ]) (lab));
                   1640:          emit_label (lab);
                   1641:        }
                   1642:     }
                   1643: 
                   1644:   /* If this expression is part of a ({...}) and is in memory, we may have
                   1645:      to preserve temporaries.  */
                   1646:   preserve_temp_slots (last_expr_value);
                   1647: 
                   1648:   /* Free any temporaries used to evaluate this expression.  Any temporary
                   1649:      used as a result of this expression will already have been preserved
                   1650:      above.  */
                   1651:   free_temp_slots ();
                   1652: 
                   1653:   emit_queue ();
                   1654: }
                   1655: 
                   1656: /* Warn if EXP contains any computations whose results are not used.
                   1657:    Return 1 if a warning is printed; 0 otherwise.  */
                   1658: 
                   1659: static int
                   1660: warn_if_unused_value (exp)
                   1661:      tree exp;
                   1662: {
                   1663:   if (TREE_USED (exp))
                   1664:     return 0;
                   1665: 
                   1666:   switch (TREE_CODE (exp))
                   1667:     {
                   1668:     case PREINCREMENT_EXPR:
                   1669:     case POSTINCREMENT_EXPR:
                   1670:     case PREDECREMENT_EXPR:
                   1671:     case POSTDECREMENT_EXPR:
                   1672:     case MODIFY_EXPR:
                   1673:     case INIT_EXPR:
                   1674:     case TARGET_EXPR:
                   1675:     case CALL_EXPR:
                   1676:     case METHOD_CALL_EXPR:
                   1677:     case RTL_EXPR:
                   1678:     case WITH_CLEANUP_EXPR:
                   1679:     case EXIT_EXPR:
                   1680:       /* We don't warn about COND_EXPR because it may be a useful
                   1681:         construct if either arm contains a side effect.  */
                   1682:     case COND_EXPR:
                   1683:       return 0;
                   1684: 
                   1685:     case BIND_EXPR:
                   1686:       /* For a binding, warn if no side effect within it.  */
                   1687:       return warn_if_unused_value (TREE_OPERAND (exp, 1));
                   1688: 
                   1689:     case TRUTH_ORIF_EXPR:
                   1690:     case TRUTH_ANDIF_EXPR:
                   1691:       /* In && or ||, warn if 2nd operand has no side effect.  */
                   1692:       return warn_if_unused_value (TREE_OPERAND (exp, 1));
                   1693: 
                   1694:     case COMPOUND_EXPR:
                   1695:       if (warn_if_unused_value (TREE_OPERAND (exp, 0)))
                   1696:        return 1;
                   1697:       /* Let people do `(foo (), 0)' without a warning.  */
                   1698:       if (TREE_CONSTANT (TREE_OPERAND (exp, 1)))
                   1699:        return 0;
                   1700:       return warn_if_unused_value (TREE_OPERAND (exp, 1));
                   1701: 
                   1702:     case NOP_EXPR:
                   1703:     case CONVERT_EXPR:
                   1704:     case NON_LVALUE_EXPR:
                   1705:       /* Don't warn about values cast to void.  */
                   1706:       if (TREE_TYPE (exp) == void_type_node)
                   1707:        return 0;
                   1708:       /* Don't warn about conversions not explicit in the user's program.  */
                   1709:       if (TREE_NO_UNUSED_WARNING (exp))
                   1710:        return 0;
                   1711:       /* Assignment to a cast usually results in a cast of a modify.
                   1712:         Don't complain about that.  There can be an arbitrary number of
                   1713:         casts before the modify, so we must loop until we find the first
                   1714:         non-cast expression and then test to see if that is a modify.  */
                   1715:       {
                   1716:        tree tem = TREE_OPERAND (exp, 0);
                   1717: 
                   1718:        while (TREE_CODE (tem) == CONVERT_EXPR || TREE_CODE (tem) == NOP_EXPR)
                   1719:          tem = TREE_OPERAND (tem, 0);
                   1720: 
                   1721:        if (TREE_CODE (tem) == MODIFY_EXPR)
                   1722:          return 0;
                   1723:       }
                   1724:       /* ... fall through ... */
                   1725: 
                   1726:     default:
                   1727:       /* Referencing a volatile value is a side effect, so don't warn.  */
                   1728:       if ((TREE_CODE_CLASS (TREE_CODE (exp)) == 'd'
                   1729:           || TREE_CODE_CLASS (TREE_CODE (exp)) == 'r')
                   1730:          && TREE_THIS_VOLATILE (exp))
                   1731:        return 0;
                   1732:       warning_with_file_and_line (emit_filename, emit_lineno,
                   1733:                                  "value computed is not used");
                   1734:       return 1;
                   1735:     }
                   1736: }
                   1737: 
                   1738: /* Clear out the memory of the last expression evaluated.  */
                   1739: 
                   1740: void
                   1741: clear_last_expr ()
                   1742: {
                   1743:   last_expr_type = 0;
                   1744: }
                   1745: 
                   1746: /* Begin a statement which will return a value.
                   1747:    Return the RTL_EXPR for this statement expr.
                   1748:    The caller must save that value and pass it to expand_end_stmt_expr.  */
                   1749: 
                   1750: tree
                   1751: expand_start_stmt_expr ()
                   1752: {
                   1753:   int momentary;
                   1754:   tree t;
                   1755: 
                   1756:   /* When generating bytecode just note down the stack depth */
                   1757:   if (output_bytecode)
                   1758:     return (build_int_2 (stack_depth, 0));
                   1759: 
                   1760:   /* Make the RTL_EXPR node temporary, not momentary,
                   1761:      so that rtl_expr_chain doesn't become garbage.  */
                   1762:   momentary = suspend_momentary ();
                   1763:   t = make_node (RTL_EXPR);
                   1764:   resume_momentary (momentary);
                   1765:   start_sequence_for_rtl_expr (t);
                   1766:   NO_DEFER_POP;
                   1767:   expr_stmts_for_value++;
                   1768:   return t;
                   1769: }
                   1770: 
                   1771: /* Restore the previous state at the end of a statement that returns a value.
                   1772:    Returns a tree node representing the statement's value and the
                   1773:    insns to compute the value.
                   1774: 
                   1775:    The nodes of that expression have been freed by now, so we cannot use them.
                   1776:    But we don't want to do that anyway; the expression has already been
                   1777:    evaluated and now we just want to use the value.  So generate a RTL_EXPR
                   1778:    with the proper type and RTL value.
                   1779: 
                   1780:    If the last substatement was not an expression,
                   1781:    return something with type `void'.  */
                   1782: 
                   1783: tree
                   1784: expand_end_stmt_expr (t)
                   1785:      tree t;
                   1786: {
                   1787:   if (output_bytecode)
                   1788:     {
                   1789:       int i;
                   1790:       tree t;
                   1791:       
                   1792:       
                   1793:       /* At this point, all expressions have been evaluated in order.
                   1794:         However, all expression values have been popped when evaluated,
                   1795:         which means we have to recover the last expression value.  This is
                   1796:         the last value removed by means of a `drop' instruction.  Instead
                   1797:         of adding code to inhibit dropping the last expression value, it
                   1798:         is here recovered by undoing the `drop'.  Since `drop' is
                   1799:         equivalent to `adjustackSI [1]', it can be undone with `adjstackSI
                   1800:         [-1]'. */
                   1801:       
                   1802:       bc_adjust_stack (-1);
                   1803:       
                   1804:       if (!last_expr_type)
                   1805:        last_expr_type = void_type_node;
                   1806:       
                   1807:       t = make_node (RTL_EXPR);
                   1808:       TREE_TYPE (t) = last_expr_type;
                   1809:       RTL_EXPR_RTL (t) = NULL;
                   1810:       RTL_EXPR_SEQUENCE (t) = NULL;
                   1811:       
                   1812:       /* Don't consider deleting this expr or containing exprs at tree level.  */
                   1813:       TREE_THIS_VOLATILE (t) = 1;
                   1814:       
                   1815:       last_expr_type = 0;
                   1816:       return t;
                   1817:     }
                   1818: 
                   1819:   OK_DEFER_POP;
                   1820: 
                   1821:   if (last_expr_type == 0)
                   1822:     {
                   1823:       last_expr_type = void_type_node;
                   1824:       last_expr_value = const0_rtx;
                   1825:     }
                   1826:   else if (last_expr_value == 0)
                   1827:     /* There are some cases where this can happen, such as when the
                   1828:        statement is void type.  */
                   1829:     last_expr_value = const0_rtx;
                   1830:   else if (GET_CODE (last_expr_value) != REG && ! CONSTANT_P (last_expr_value))
                   1831:     /* Remove any possible QUEUED.  */
                   1832:     last_expr_value = protect_from_queue (last_expr_value, 0);
                   1833: 
                   1834:   emit_queue ();
                   1835: 
                   1836:   TREE_TYPE (t) = last_expr_type;
                   1837:   RTL_EXPR_RTL (t) = last_expr_value;
                   1838:   RTL_EXPR_SEQUENCE (t) = get_insns ();
                   1839: 
                   1840:   rtl_expr_chain = tree_cons (NULL_TREE, t, rtl_expr_chain);
                   1841: 
                   1842:   end_sequence ();
                   1843: 
                   1844:   /* Don't consider deleting this expr or containing exprs at tree level.  */
                   1845:   TREE_SIDE_EFFECTS (t) = 1;
                   1846:   /* Propagate volatility of the actual RTL expr.  */
                   1847:   TREE_THIS_VOLATILE (t) = volatile_refs_p (last_expr_value);
                   1848: 
                   1849:   last_expr_type = 0;
                   1850:   expr_stmts_for_value--;
                   1851: 
                   1852:   return t;
                   1853: }
                   1854: 
                   1855: /* The exception handling nesting looks like this:
                   1856: 
                   1857:                <-- Level N-1
                   1858:     {          <-- exception handler block
                   1859:                <-- Level N
                   1860:                <-- in an exception handler
                   1861:        {       <-- try block
                   1862:        :       <-- in a TRY block
                   1863:        :       <-- in an exception handler
                   1864:        :
                   1865:        }
                   1866: 
                   1867:        {       <-- except block
                   1868:        :       <-- in an except block
                   1869:        :       <-- in an exception handler
                   1870:        :
                   1871:        }
                   1872: 
                   1873:     }
                   1874: */
                   1875: 
                   1876: /* Return nonzero iff in a try block at level LEVEL.  */
                   1877: 
                   1878: int
                   1879: in_try_block (level)
                   1880:      int level;
                   1881: {
                   1882:   struct nesting *n = except_stack;
                   1883:   while (1)
                   1884:     {
                   1885:       while (n && n->data.except_stmt.after_label != 0)
                   1886:        n = n->next;
                   1887:       if (n == 0)
                   1888:        return 0;
                   1889:       if (level == 0)
                   1890:        return n != 0;
                   1891:       level--;
                   1892:       n = n->next;
                   1893:     }
                   1894: }
                   1895: 
                   1896: /* Return nonzero iff in an except block at level LEVEL.  */
                   1897: 
                   1898: int
                   1899: in_except_block (level)
                   1900:      int level;
                   1901: {
                   1902:   struct nesting *n = except_stack;
                   1903:   while (1)
                   1904:     {
                   1905:       while (n && n->data.except_stmt.after_label == 0)
                   1906:        n = n->next;
                   1907:       if (n == 0)
                   1908:        return 0;
                   1909:       if (level == 0)
                   1910:        return n != 0;
                   1911:       level--;
                   1912:       n = n->next;
                   1913:     }
                   1914: }
                   1915: 
                   1916: /* Return nonzero iff in an exception handler at level LEVEL.  */
                   1917: 
                   1918: int
                   1919: in_exception_handler (level)
                   1920:      int level;
                   1921: {
                   1922:   struct nesting *n = except_stack;
                   1923:   while (n && level--)
                   1924:     n = n->next;
                   1925:   return n != 0;
                   1926: }
                   1927: 
                   1928: /* Record the fact that the current exception nesting raises
                   1929:    exception EX.  If not in an exception handler, return 0.  */
                   1930: int
                   1931: expand_raise (ex)
                   1932:      tree ex;
                   1933: {
                   1934:   tree *raises_ptr;
                   1935: 
                   1936:   if (except_stack == 0)
                   1937:     return 0;
                   1938:   raises_ptr = &except_stack->data.except_stmt.raised;
                   1939:   if (! value_member (ex, *raises_ptr))
                   1940:     *raises_ptr = tree_cons (NULL_TREE, ex, *raises_ptr);
                   1941:   return 1;
                   1942: }
                   1943: 
                   1944: /* Generate RTL for the start of a try block.
                   1945: 
                   1946:    TRY_CLAUSE is the condition to test to enter the try block.  */
                   1947: 
                   1948: void
                   1949: expand_start_try (try_clause, exitflag, escapeflag)
                   1950:      tree try_clause;
                   1951:      int exitflag;
                   1952:      int escapeflag;
                   1953: {
                   1954:   struct nesting *thishandler = ALLOC_NESTING ();
                   1955: 
                   1956:   /* Make an entry on cond_stack for the cond we are entering.  */
                   1957: 
                   1958:   thishandler->next = except_stack;
                   1959:   thishandler->all = nesting_stack;
                   1960:   thishandler->depth = ++nesting_depth;
                   1961:   thishandler->data.except_stmt.raised = 0;
                   1962:   thishandler->data.except_stmt.handled = 0;
                   1963:   thishandler->data.except_stmt.first_insn = get_insns ();
                   1964:   thishandler->data.except_stmt.except_label = gen_label_rtx ();
                   1965:   thishandler->data.except_stmt.unhandled_label = 0;
                   1966:   thishandler->data.except_stmt.after_label = 0;
                   1967:   thishandler->data.except_stmt.escape_label
                   1968:     = escapeflag ? thishandler->data.except_stmt.except_label : 0;
                   1969:   thishandler->exit_label = exitflag ? gen_label_rtx () : 0;
                   1970:   except_stack = thishandler;
                   1971:   nesting_stack = thishandler;
                   1972: 
                   1973:   do_jump (try_clause, thishandler->data.except_stmt.except_label, NULL_RTX);
                   1974: }
                   1975: 
                   1976: /* End of a TRY block.  Nothing to do for now.  */
                   1977: 
                   1978: void
                   1979: expand_end_try ()
                   1980: {
                   1981:   except_stack->data.except_stmt.after_label = gen_label_rtx ();
                   1982:   expand_goto_internal (NULL_TREE, except_stack->data.except_stmt.after_label,
                   1983:                        NULL_RTX);
                   1984: }
                   1985: 
                   1986: /* Start an `except' nesting contour.
                   1987:    EXITFLAG says whether this contour should be able to `exit' something.
                   1988:    ESCAPEFLAG says whether this contour should be escapable.  */
                   1989: 
                   1990: void
                   1991: expand_start_except (exitflag, escapeflag)
                   1992:      int exitflag;
                   1993:      int escapeflag;
                   1994: {
                   1995:   if (exitflag)
                   1996:     {
                   1997:       struct nesting *n;
                   1998:       /* An `exit' from catch clauses goes out to next exit level,
                   1999:         if there is one.  Otherwise, it just goes to the end
                   2000:         of the construct.  */
                   2001:       for (n = except_stack->next; n; n = n->next)
                   2002:        if (n->exit_label != 0)
                   2003:          {
                   2004:            except_stack->exit_label = n->exit_label;
                   2005:            break;
                   2006:          }
                   2007:       if (n == 0)
                   2008:        except_stack->exit_label = except_stack->data.except_stmt.after_label;
                   2009:     }
                   2010:   if (escapeflag)
                   2011:     {
                   2012:       struct nesting *n;
                   2013:       /* An `escape' from catch clauses goes out to next escape level,
                   2014:         if there is one.  Otherwise, it just goes to the end
                   2015:         of the construct.  */
                   2016:       for (n = except_stack->next; n; n = n->next)
                   2017:        if (n->data.except_stmt.escape_label != 0)
                   2018:          {
                   2019:            except_stack->data.except_stmt.escape_label
                   2020:              = n->data.except_stmt.escape_label;
                   2021:            break;
                   2022:          }
                   2023:       if (n == 0)
                   2024:        except_stack->data.except_stmt.escape_label
                   2025:          = except_stack->data.except_stmt.after_label;
                   2026:     }
                   2027:   do_pending_stack_adjust ();
                   2028:   emit_label (except_stack->data.except_stmt.except_label);
                   2029: }
                   2030: 
                   2031: /* Generate code to `escape' from an exception contour.  This
                   2032:    is like `exiting', but does not conflict with constructs which
                   2033:    use `exit_label'.
                   2034: 
                   2035:    Return nonzero if this contour is escapable, otherwise
                   2036:    return zero, and language-specific code will emit the
                   2037:    appropriate error message.  */
                   2038: int
                   2039: expand_escape_except ()
                   2040: {
                   2041:   struct nesting *n;
                   2042:   last_expr_type = 0;
                   2043:   for (n = except_stack; n; n = n->next)
                   2044:     if (n->data.except_stmt.escape_label != 0)
                   2045:       {
                   2046:        expand_goto_internal (NULL_TREE,
                   2047:                              n->data.except_stmt.escape_label, NULL_RTX);
                   2048:        return 1;
                   2049:       }
                   2050: 
                   2051:   return 0;
                   2052: }
                   2053: 
                   2054: /* Finish processing and `except' contour.
                   2055:    Culls out all exceptions which might be raise but not
                   2056:    handled, and returns the list to the caller.
                   2057:    Language-specific code is responsible for dealing with these
                   2058:    exceptions.  */
                   2059: 
                   2060: tree
                   2061: expand_end_except ()
                   2062: {
                   2063:   struct nesting *n;
                   2064:   tree raised = NULL_TREE;
                   2065: 
                   2066:   do_pending_stack_adjust ();
                   2067:   emit_label (except_stack->data.except_stmt.after_label);
                   2068: 
                   2069:   n = except_stack->next;
                   2070:   if (n)
                   2071:     {
                   2072:       /* Propagate exceptions raised but not handled to next
                   2073:         highest level.  */
                   2074:       tree handled = except_stack->data.except_stmt.raised;
                   2075:       if (handled != void_type_node)
                   2076:        {
                   2077:          tree prev = NULL_TREE;
                   2078:          raised = except_stack->data.except_stmt.raised;
                   2079:          while (handled)
                   2080:            {
                   2081:              tree this_raise;
                   2082:              for (this_raise = raised, prev = 0; this_raise;
                   2083:                   this_raise = TREE_CHAIN (this_raise))
                   2084:                {
                   2085:                  if (value_member (TREE_VALUE (this_raise), handled))
                   2086:                    {
                   2087:                      if (prev)
                   2088:                        TREE_CHAIN (prev) = TREE_CHAIN (this_raise);
                   2089:                      else
                   2090:                        {
                   2091:                          raised = TREE_CHAIN (raised);
                   2092:                          if (raised == NULL_TREE)
                   2093:                            goto nada;
                   2094:                        }
                   2095:                    }
                   2096:                  else
                   2097:                    prev = this_raise;
                   2098:                }
                   2099:              handled = TREE_CHAIN (handled);
                   2100:            }
                   2101:          if (prev == NULL_TREE)
                   2102:            prev = raised;
                   2103:          if (prev)
                   2104:            TREE_CHAIN (prev) = n->data.except_stmt.raised;
                   2105:        nada:
                   2106:          n->data.except_stmt.raised = raised;
                   2107:        }
                   2108:     }
                   2109: 
                   2110:   POPSTACK (except_stack);
                   2111:   last_expr_type = 0;
                   2112:   return raised;
                   2113: }
                   2114: 
                   2115: /* Record that exception EX is caught by this exception handler.
                   2116:    Return nonzero if in exception handling construct, otherwise return 0.  */
                   2117: int
                   2118: expand_catch (ex)
                   2119:      tree ex;
                   2120: {
                   2121:   tree *raises_ptr;
                   2122: 
                   2123:   if (except_stack == 0)
                   2124:     return 0;
                   2125:   raises_ptr = &except_stack->data.except_stmt.handled;
                   2126:   if (*raises_ptr != void_type_node
                   2127:       && ex != NULL_TREE
                   2128:       && ! value_member (ex, *raises_ptr))
                   2129:     *raises_ptr = tree_cons (NULL_TREE, ex, *raises_ptr);
                   2130:   return 1;
                   2131: }
                   2132: 
                   2133: /* Record that this exception handler catches all exceptions.
                   2134:    Return nonzero if in exception handling construct, otherwise return 0.  */
                   2135: 
                   2136: int
                   2137: expand_catch_default ()
                   2138: {
                   2139:   if (except_stack == 0)
                   2140:     return 0;
                   2141:   except_stack->data.except_stmt.handled = void_type_node;
                   2142:   return 1;
                   2143: }
                   2144: 
                   2145: int
                   2146: expand_end_catch ()
                   2147: {
                   2148:   if (except_stack == 0 || except_stack->data.except_stmt.after_label == 0)
                   2149:     return 0;
                   2150:   expand_goto_internal (NULL_TREE, except_stack->data.except_stmt.after_label,
                   2151:                        NULL_RTX);
                   2152:   return 1;
                   2153: }
                   2154: 
                   2155: /* Generate RTL for the start of an if-then.  COND is the expression
                   2156:    whose truth should be tested.
                   2157: 
                   2158:    If EXITFLAG is nonzero, this conditional is visible to
                   2159:    `exit_something'.  */
                   2160: 
                   2161: void
                   2162: expand_start_cond (cond, exitflag)
                   2163:      tree cond;
                   2164:      int exitflag;
                   2165: {
                   2166:   struct nesting *thiscond = ALLOC_NESTING ();
                   2167: 
                   2168:   /* Make an entry on cond_stack for the cond we are entering.  */
                   2169: 
                   2170:   thiscond->next = cond_stack;
                   2171:   thiscond->all = nesting_stack;
                   2172:   thiscond->depth = ++nesting_depth;
                   2173:   thiscond->data.cond.next_label = gen_label_rtx ();
                   2174:   /* Before we encounter an `else', we don't need a separate exit label
                   2175:      unless there are supposed to be exit statements
                   2176:      to exit this conditional.  */
                   2177:   thiscond->exit_label = exitflag ? gen_label_rtx () : 0;
                   2178:   thiscond->data.cond.endif_label = thiscond->exit_label;
                   2179:   cond_stack = thiscond;
                   2180:   nesting_stack = thiscond;
                   2181: 
                   2182:   if (output_bytecode)
                   2183:     bc_expand_start_cond (cond, exitflag);
                   2184:   else
                   2185:     do_jump (cond, thiscond->data.cond.next_label, NULL_RTX);
                   2186: }
                   2187: 
                   2188: /* Generate RTL between then-clause and the elseif-clause
                   2189:    of an if-then-elseif-....  */
                   2190: 
                   2191: void
                   2192: expand_start_elseif (cond)
                   2193:      tree cond;
                   2194: {
                   2195:   if (cond_stack->data.cond.endif_label == 0)
                   2196:     cond_stack->data.cond.endif_label = gen_label_rtx ();
                   2197:   emit_jump (cond_stack->data.cond.endif_label);
                   2198:   emit_label (cond_stack->data.cond.next_label);
                   2199:   cond_stack->data.cond.next_label = gen_label_rtx ();
                   2200:   do_jump (cond, cond_stack->data.cond.next_label, NULL_RTX);
                   2201: }
                   2202: 
                   2203: /* Generate RTL between the then-clause and the else-clause
                   2204:    of an if-then-else.  */
                   2205: 
                   2206: void
                   2207: expand_start_else ()
                   2208: {
                   2209:   if (cond_stack->data.cond.endif_label == 0)
                   2210:     cond_stack->data.cond.endif_label = gen_label_rtx ();
                   2211: 
                   2212:   if (output_bytecode)
                   2213:     {
                   2214:       bc_expand_start_else ();
                   2215:       return;
                   2216:     }
                   2217: 
                   2218:   emit_jump (cond_stack->data.cond.endif_label);
                   2219:   emit_label (cond_stack->data.cond.next_label);
                   2220:   cond_stack->data.cond.next_label = 0;  /* No more _else or _elseif calls. */
                   2221: }
                   2222: 
                   2223: /* Generate RTL for the end of an if-then.
                   2224:    Pop the record for it off of cond_stack.  */
                   2225: 
                   2226: void
                   2227: expand_end_cond ()
                   2228: {
                   2229:   struct nesting *thiscond = cond_stack;
                   2230: 
                   2231:   if (output_bytecode)
                   2232:     bc_expand_end_cond ();
                   2233:   else
                   2234:     {
                   2235:       do_pending_stack_adjust ();
                   2236:       if (thiscond->data.cond.next_label)
                   2237:        emit_label (thiscond->data.cond.next_label);
                   2238:       if (thiscond->data.cond.endif_label)
                   2239:        emit_label (thiscond->data.cond.endif_label);
                   2240:     }
                   2241: 
                   2242:   POPSTACK (cond_stack);
                   2243:   last_expr_type = 0;
                   2244: }
                   2245: 
                   2246: 
                   2247: /* Generate code for the start of an if-then.  COND is the expression
                   2248:    whose truth is to be tested; if EXITFLAG is nonzero this conditional
                   2249:    is to be visible to exit_something.  It is assumed that the caller
                   2250:    has pushed the previous context on the cond stack. */
                   2251: void
                   2252: bc_expand_start_cond (cond, exitflag)
                   2253:      tree cond;
                   2254:      int exitflag;
                   2255: {
                   2256:   struct nesting *thiscond = cond_stack;
                   2257: 
                   2258:   thiscond->data.case_stmt.nominal_type = cond;
                   2259:   bc_expand_expr (cond);
                   2260:   bc_emit_bytecode (xjumpifnot);
                   2261:   bc_emit_bytecode_labelref (BYTECODE_BC_LABEL (thiscond->exit_label));
                   2262: 
                   2263: #ifdef DEBUG_PRINT_CODE
                   2264:   fputc ('\n', stderr);
                   2265: #endif
                   2266: }
                   2267: 
                   2268: /* Generate the label for the end of an if with
                   2269:    no else- clause.  */
                   2270: void
                   2271: bc_expand_end_cond ()
                   2272: {
                   2273:   struct nesting *thiscond = cond_stack;
                   2274: 
                   2275:   bc_emit_bytecode_labeldef (BYTECODE_BC_LABEL (thiscond->exit_label));
                   2276: }
                   2277: 
                   2278: /* Generate code for the start of the else- clause of
                   2279:    an if-then-else.  */
                   2280: void
                   2281: bc_expand_start_else ()
                   2282: {
                   2283:   struct nesting *thiscond = cond_stack;
                   2284: 
                   2285:   thiscond->data.cond.endif_label = thiscond->exit_label;
                   2286:   thiscond->exit_label = gen_label_rtx ();
                   2287:   bc_emit_bytecode (jump);
                   2288:   bc_emit_bytecode_labelref (BYTECODE_BC_LABEL (thiscond->exit_label));
                   2289: 
                   2290: #ifdef DEBUG_PRINT_CODE
                   2291:   fputc ('\n', stderr);
                   2292: #endif
                   2293: 
                   2294:   bc_emit_bytecode_labeldef (BYTECODE_BC_LABEL (thiscond->data.cond.endif_label));
                   2295: }
                   2296: 
                   2297: /* Generate RTL for the start of a loop.  EXIT_FLAG is nonzero if this
                   2298:    loop should be exited by `exit_something'.  This is a loop for which
                   2299:    `expand_continue' will jump to the top of the loop.
                   2300: 
                   2301:    Make an entry on loop_stack to record the labels associated with
                   2302:    this loop.  */
                   2303: 
                   2304: struct nesting *
                   2305: expand_start_loop (exit_flag)
                   2306:      int exit_flag;
                   2307: {
                   2308:   register struct nesting *thisloop = ALLOC_NESTING ();
                   2309: 
                   2310:   /* Make an entry on loop_stack for the loop we are entering.  */
                   2311: 
                   2312:   thisloop->next = loop_stack;
                   2313:   thisloop->all = nesting_stack;
                   2314:   thisloop->depth = ++nesting_depth;
                   2315:   thisloop->data.loop.start_label = gen_label_rtx ();
                   2316:   thisloop->data.loop.end_label = gen_label_rtx ();
                   2317:   thisloop->data.loop.continue_label = thisloop->data.loop.start_label;
                   2318:   thisloop->exit_label = exit_flag ? thisloop->data.loop.end_label : 0;
                   2319:   loop_stack = thisloop;
                   2320:   nesting_stack = thisloop;
                   2321: 
                   2322:   if (output_bytecode)
                   2323:     {
                   2324:       bc_emit_bytecode_labeldef (BYTECODE_BC_LABEL (thisloop->data.loop.start_label));
                   2325:       return thisloop;
                   2326:     }
                   2327: 
                   2328:   do_pending_stack_adjust ();
                   2329:   emit_queue ();
                   2330:   emit_note (NULL_PTR, NOTE_INSN_LOOP_BEG);
                   2331:   emit_label (thisloop->data.loop.start_label);
                   2332: 
                   2333:   return thisloop;
                   2334: }
                   2335: 
                   2336: /* Like expand_start_loop but for a loop where the continuation point
                   2337:    (for expand_continue_loop) will be specified explicitly.  */
                   2338: 
                   2339: struct nesting *
                   2340: expand_start_loop_continue_elsewhere (exit_flag)
                   2341:      int exit_flag;
                   2342: {
                   2343:   struct nesting *thisloop = expand_start_loop (exit_flag);
                   2344:   loop_stack->data.loop.continue_label = gen_label_rtx ();
                   2345:   return thisloop;
                   2346: }
                   2347: 
                   2348: /* Specify the continuation point for a loop started with
                   2349:    expand_start_loop_continue_elsewhere.
                   2350:    Use this at the point in the code to which a continue statement
                   2351:    should jump.  */
                   2352: 
                   2353: void
                   2354: expand_loop_continue_here ()
                   2355: {
                   2356:   if (output_bytecode)
                   2357:     {
                   2358:       bc_emit_bytecode_labeldef (BYTECODE_BC_LABEL (loop_stack->data.loop.continue_label));
                   2359:       return;
                   2360:     }
                   2361:   do_pending_stack_adjust ();
                   2362:   emit_note (NULL_PTR, NOTE_INSN_LOOP_CONT);
                   2363:   emit_label (loop_stack->data.loop.continue_label);
                   2364: }
                   2365: 
                   2366: /* End a loop.  */
                   2367: static void
                   2368: bc_expand_end_loop ()
                   2369: {
                   2370:   struct nesting *thisloop = loop_stack;
                   2371: 
                   2372:   bc_emit_bytecode (jump);
                   2373:   bc_emit_bytecode_labelref (BYTECODE_BC_LABEL (thisloop->data.loop.start_label));
                   2374: 
                   2375: #ifdef DEBUG_PRINT_CODE
                   2376:   fputc ('\n', stderr);
                   2377: #endif
                   2378: 
                   2379:   bc_emit_bytecode_labeldef (BYTECODE_BC_LABEL (thisloop->exit_label));
                   2380:   POPSTACK (loop_stack);
                   2381:   last_expr_type = 0;
                   2382: }
                   2383: 
                   2384: 
                   2385: /* Finish a loop.  Generate a jump back to the top and the loop-exit label.
                   2386:    Pop the block off of loop_stack.  */
                   2387: 
                   2388: void
                   2389: expand_end_loop ()
                   2390: {
                   2391:   register rtx insn;
                   2392:   register rtx start_label;
                   2393:   rtx last_test_insn = 0;
                   2394:   int num_insns = 0;
                   2395:     
                   2396:   if (output_bytecode)
                   2397:     {
                   2398:       bc_expand_end_loop ();
                   2399:       return;
                   2400:     }
                   2401: 
                   2402:   insn = get_last_insn ();
                   2403:   start_label = loop_stack->data.loop.start_label;
                   2404: 
                   2405:   /* Mark the continue-point at the top of the loop if none elsewhere.  */
                   2406:   if (start_label == loop_stack->data.loop.continue_label)
                   2407:     emit_note_before (NOTE_INSN_LOOP_CONT, start_label);
                   2408: 
                   2409:   do_pending_stack_adjust ();
                   2410: 
                   2411:   /* If optimizing, perhaps reorder the loop.  If the loop
                   2412:      starts with a conditional exit, roll that to the end
                   2413:      where it will optimize together with the jump back.
                   2414: 
                   2415:      We look for the last conditional branch to the exit that we encounter
                   2416:      before hitting 30 insns or a CALL_INSN.  If we see an unconditional
                   2417:      branch to the exit first, use it.
                   2418: 
                   2419:      We must also stop at NOTE_INSN_BLOCK_BEG and NOTE_INSN_BLOCK_END notes
                   2420:      because moving them is not valid.  */
                   2421: 
                   2422:   if (optimize
                   2423:       &&
                   2424:       ! (GET_CODE (insn) == JUMP_INSN
                   2425:         && GET_CODE (PATTERN (insn)) == SET
                   2426:         && SET_DEST (PATTERN (insn)) == pc_rtx
                   2427:         && GET_CODE (SET_SRC (PATTERN (insn))) == IF_THEN_ELSE))
                   2428:     {
                   2429:       /* Scan insns from the top of the loop looking for a qualified
                   2430:         conditional exit.  */
                   2431:       for (insn = NEXT_INSN (loop_stack->data.loop.start_label); insn;
                   2432:           insn = NEXT_INSN (insn))
                   2433:        {
                   2434:          if (GET_CODE (insn) == CALL_INSN || GET_CODE (insn) == CODE_LABEL)
                   2435:            break;
                   2436: 
                   2437:          if (GET_CODE (insn) == NOTE
                   2438:              && (NOTE_LINE_NUMBER (insn) == NOTE_INSN_BLOCK_BEG
                   2439:                  || NOTE_LINE_NUMBER (insn) == NOTE_INSN_BLOCK_END))
                   2440:            break;
                   2441: 
                   2442:          if (GET_CODE (insn) == JUMP_INSN || GET_CODE (insn) == INSN)
                   2443:            num_insns++;
                   2444: 
                   2445:          if (last_test_insn && num_insns > 30)
                   2446:            break;
                   2447: 
                   2448:          if (GET_CODE (insn) == JUMP_INSN && GET_CODE (PATTERN (insn)) == SET
                   2449:              && SET_DEST (PATTERN (insn)) == pc_rtx
                   2450:              && GET_CODE (SET_SRC (PATTERN (insn))) == IF_THEN_ELSE
                   2451:              && ((GET_CODE (XEXP (SET_SRC (PATTERN (insn)), 1)) == LABEL_REF
                   2452:                   && (XEXP (XEXP (SET_SRC (PATTERN (insn)), 1), 0)
                   2453:                       == loop_stack->data.loop.end_label))
                   2454:                  || (GET_CODE (XEXP (SET_SRC (PATTERN (insn)), 2)) == LABEL_REF
                   2455:                      && (XEXP (XEXP (SET_SRC (PATTERN (insn)), 2), 0)
                   2456:                          == loop_stack->data.loop.end_label))))
                   2457:            last_test_insn = insn;
                   2458: 
                   2459:          if (last_test_insn == 0 && GET_CODE (insn) == JUMP_INSN
                   2460:              && GET_CODE (PATTERN (insn)) == SET
                   2461:              && SET_DEST (PATTERN (insn)) == pc_rtx
                   2462:              && GET_CODE (SET_SRC (PATTERN (insn))) == LABEL_REF
                   2463:              && (XEXP (SET_SRC (PATTERN (insn)), 0)
                   2464:                  == loop_stack->data.loop.end_label))
                   2465:            /* Include BARRIER.  */
                   2466:            last_test_insn = NEXT_INSN (insn);
                   2467:        }
                   2468: 
                   2469:       if (last_test_insn != 0 && last_test_insn != get_last_insn ())
                   2470:        {
                   2471:          /* We found one.  Move everything from there up
                   2472:             to the end of the loop, and add a jump into the loop
                   2473:             to jump to there.  */
                   2474:          register rtx newstart_label = gen_label_rtx ();
                   2475:          register rtx start_move = start_label;
                   2476: 
                   2477:          /* If the start label is preceded by a NOTE_INSN_LOOP_CONT note,
                   2478:             then we want to move this note also.  */
                   2479:          if (GET_CODE (PREV_INSN (start_move)) == NOTE
                   2480:              && (NOTE_LINE_NUMBER (PREV_INSN (start_move))
                   2481:                  == NOTE_INSN_LOOP_CONT))
                   2482:            start_move = PREV_INSN (start_move);
                   2483: 
                   2484:          emit_label_after (newstart_label, PREV_INSN (start_move));
                   2485:          reorder_insns (start_move, last_test_insn, get_last_insn ());
                   2486:          emit_jump_insn_after (gen_jump (start_label),
                   2487:                                PREV_INSN (newstart_label));
                   2488:          emit_barrier_after (PREV_INSN (newstart_label));
                   2489:          start_label = newstart_label;
                   2490:        }
                   2491:     }
                   2492: 
                   2493:   emit_jump (start_label);
                   2494:   emit_note (NULL_PTR, NOTE_INSN_LOOP_END);
                   2495:   emit_label (loop_stack->data.loop.end_label);
                   2496: 
                   2497:   POPSTACK (loop_stack);
                   2498: 
                   2499:   last_expr_type = 0;
                   2500: }
                   2501: 
                   2502: /* Generate a jump to the current loop's continue-point.
                   2503:    This is usually the top of the loop, but may be specified
                   2504:    explicitly elsewhere.  If not currently inside a loop,
                   2505:    return 0 and do nothing; caller will print an error message.  */
                   2506: 
                   2507: int
                   2508: expand_continue_loop (whichloop)
                   2509:      struct nesting *whichloop;
                   2510: {
                   2511:   last_expr_type = 0;
                   2512:   if (whichloop == 0)
                   2513:     whichloop = loop_stack;
                   2514:   if (whichloop == 0)
                   2515:     return 0;
                   2516:   expand_goto_internal (NULL_TREE, whichloop->data.loop.continue_label,
                   2517:                        NULL_RTX);
                   2518:   return 1;
                   2519: }
                   2520: 
                   2521: /* Generate a jump to exit the current loop.  If not currently inside a loop,
                   2522:    return 0 and do nothing; caller will print an error message.  */
                   2523: 
                   2524: int
                   2525: expand_exit_loop (whichloop)
                   2526:      struct nesting *whichloop;
                   2527: {
                   2528:   last_expr_type = 0;
                   2529:   if (whichloop == 0)
                   2530:     whichloop = loop_stack;
                   2531:   if (whichloop == 0)
                   2532:     return 0;
                   2533:   expand_goto_internal (NULL_TREE, whichloop->data.loop.end_label, NULL_RTX);
                   2534:   return 1;
                   2535: }
                   2536: 
                   2537: /* Generate a conditional jump to exit the current loop if COND
                   2538:    evaluates to zero.  If not currently inside a loop,
                   2539:    return 0 and do nothing; caller will print an error message.  */
                   2540: 
                   2541: int
                   2542: expand_exit_loop_if_false (whichloop, cond)
                   2543:      struct nesting *whichloop;
                   2544:      tree cond;
                   2545: {
                   2546:   last_expr_type = 0;
                   2547:   if (whichloop == 0)
                   2548:     whichloop = loop_stack;
                   2549:   if (whichloop == 0)
                   2550:     return 0;
                   2551:   if (output_bytecode)
                   2552:     {
                   2553:       bc_expand_expr (cond);
                   2554:       bc_expand_goto_internal (xjumpifnot,
                   2555:                               BYTECODE_BC_LABEL (whichloop->exit_label),
                   2556:                               NULL_RTX);
                   2557:     }
                   2558:   else
                   2559:     do_jump (cond, whichloop->data.loop.end_label, NULL_RTX);
                   2560: 
                   2561:   return 1;
                   2562: }
                   2563: 
                   2564: /* Return non-zero if we should preserve sub-expressions as separate
                   2565:    pseudos.  We never do so if we aren't optimizing.  We always do so
                   2566:    if -fexpensive-optimizations.
                   2567: 
                   2568:    Otherwise, we only do so if we are in the "early" part of a loop.  I.e.,
                   2569:    the loop may still be a small one.  */
                   2570: 
                   2571: int
                   2572: preserve_subexpressions_p ()
                   2573: {
                   2574:   rtx insn;
                   2575: 
                   2576:   if (flag_expensive_optimizations)
                   2577:     return 1;
                   2578: 
                   2579:   if (optimize == 0 || loop_stack == 0)
                   2580:     return 0;
                   2581: 
                   2582:   insn = get_last_insn_anywhere ();
                   2583: 
                   2584:   return (insn
                   2585:          && (INSN_UID (insn) - INSN_UID (loop_stack->data.loop.start_label)
                   2586:              < n_non_fixed_regs * 3));
                   2587: 
                   2588: }
                   2589: 
                   2590: /* Generate a jump to exit the current loop, conditional, binding contour
                   2591:    or case statement.  Not all such constructs are visible to this function,
                   2592:    only those started with EXIT_FLAG nonzero.  Individual languages use
                   2593:    the EXIT_FLAG parameter to control which kinds of constructs you can
                   2594:    exit this way.
                   2595: 
                   2596:    If not currently inside anything that can be exited,
                   2597:    return 0 and do nothing; caller will print an error message.  */
                   2598: 
                   2599: int
                   2600: expand_exit_something ()
                   2601: {
                   2602:   struct nesting *n;
                   2603:   last_expr_type = 0;
                   2604:   for (n = nesting_stack; n; n = n->all)
                   2605:     if (n->exit_label != 0)
                   2606:       {
                   2607:        expand_goto_internal (NULL_TREE, n->exit_label, NULL_RTX);
                   2608:        return 1;
                   2609:       }
                   2610: 
                   2611:   return 0;
                   2612: }
                   2613: 
                   2614: /* Generate RTL to return from the current function, with no value.
                   2615:    (That is, we do not do anything about returning any value.)  */
                   2616: 
                   2617: void
                   2618: expand_null_return ()
                   2619: {
                   2620:   struct nesting *block = block_stack;
                   2621:   rtx last_insn = 0;
                   2622: 
                   2623:   if (output_bytecode)
                   2624:     {
                   2625:       bc_emit_instruction (ret);
                   2626:       return;
                   2627:     }
                   2628: 
                   2629:   /* Does any pending block have cleanups?  */
                   2630: 
                   2631:   while (block && block->data.block.cleanups == 0)
                   2632:     block = block->next;
                   2633: 
                   2634:   /* If yes, use a goto to return, since that runs cleanups.  */
                   2635: 
                   2636:   expand_null_return_1 (last_insn, block != 0);
                   2637: }
                   2638: 
                   2639: /* Generate RTL to return from the current function, with value VAL.  */
                   2640: 
                   2641: void
                   2642: expand_value_return (val)
                   2643:      rtx val;
                   2644: {
                   2645:   struct nesting *block = block_stack;
                   2646:   rtx last_insn = get_last_insn ();
                   2647:   rtx return_reg = DECL_RTL (DECL_RESULT (current_function_decl));
                   2648: 
                   2649:   /* Copy the value to the return location
                   2650:      unless it's already there.  */
                   2651: 
                   2652:   if (return_reg != val)
                   2653:     {
                   2654: #ifdef PROMOTE_FUNCTION_RETURN
                   2655:       enum machine_mode mode = DECL_MODE (DECL_RESULT (current_function_decl));
                   2656:       tree type = TREE_TYPE (DECL_RESULT (current_function_decl));
                   2657:       int unsignedp = TREE_UNSIGNED (type);
                   2658: 
                   2659:       if (TREE_CODE (type) == INTEGER_TYPE || TREE_CODE (type) == ENUMERAL_TYPE
                   2660:          || TREE_CODE (type) == BOOLEAN_TYPE || TREE_CODE (type) == CHAR_TYPE
                   2661:          || TREE_CODE (type) == REAL_TYPE || TREE_CODE (type) == POINTER_TYPE
                   2662:          || TREE_CODE (type) == OFFSET_TYPE)
                   2663:        {
                   2664:          PROMOTE_MODE (mode, unsignedp, type);
                   2665:        }
                   2666: 
                   2667:       if (GET_MODE (val) != VOIDmode && GET_MODE (val) != mode)
                   2668:        convert_move (return_reg, val, unsignedp);
                   2669:       else
                   2670: #endif
                   2671:        emit_move_insn (return_reg, val);
                   2672:     }
                   2673:   if (GET_CODE (return_reg) == REG
                   2674:       && REGNO (return_reg) < FIRST_PSEUDO_REGISTER)
                   2675:     emit_insn (gen_rtx (USE, VOIDmode, return_reg));
                   2676: 
                   2677:   /* Does any pending block have cleanups?  */
                   2678: 
                   2679:   while (block && block->data.block.cleanups == 0)
                   2680:     block = block->next;
                   2681: 
                   2682:   /* If yes, use a goto to return, since that runs cleanups.
                   2683:      Use LAST_INSN to put cleanups *before* the move insn emitted above.  */
                   2684: 
                   2685:   expand_null_return_1 (last_insn, block != 0);
                   2686: }
                   2687: 
                   2688: /* Output a return with no value.  If LAST_INSN is nonzero,
                   2689:    pretend that the return takes place after LAST_INSN.
                   2690:    If USE_GOTO is nonzero then don't use a return instruction;
                   2691:    go to the return label instead.  This causes any cleanups
                   2692:    of pending blocks to be executed normally.  */
                   2693: 
                   2694: static void
                   2695: expand_null_return_1 (last_insn, use_goto)
                   2696:      rtx last_insn;
                   2697:      int use_goto;
                   2698: {
                   2699:   rtx end_label = cleanup_label ? cleanup_label : return_label;
                   2700: 
                   2701:   clear_pending_stack_adjust ();
                   2702:   do_pending_stack_adjust ();
                   2703:   last_expr_type = 0;
                   2704: 
                   2705:   /* PCC-struct return always uses an epilogue.  */
                   2706:   if (current_function_returns_pcc_struct || use_goto)
                   2707:     {
                   2708:       if (end_label == 0)
                   2709:        end_label = return_label = gen_label_rtx ();
                   2710:       expand_goto_internal (NULL_TREE, end_label, last_insn);
                   2711:       return;
                   2712:     }
                   2713: 
                   2714:   /* Otherwise output a simple return-insn if one is available,
                   2715:      unless it won't do the job.  */
                   2716: #ifdef HAVE_return
                   2717:   if (HAVE_return && use_goto == 0 && cleanup_label == 0)
                   2718:     {
                   2719:       emit_jump_insn (gen_return ());
                   2720:       emit_barrier ();
                   2721:       return;
                   2722:     }
                   2723: #endif
                   2724: 
                   2725:   /* Otherwise jump to the epilogue.  */
                   2726:   expand_goto_internal (NULL_TREE, end_label, last_insn);
                   2727: }
                   2728: 
                   2729: /* Generate RTL to evaluate the expression RETVAL and return it
                   2730:    from the current function.  */
                   2731: 
                   2732: void
                   2733: expand_return (retval)
                   2734:      tree retval;
                   2735: {
                   2736:   /* If there are any cleanups to be performed, then they will
                   2737:      be inserted following LAST_INSN.  It is desirable
                   2738:      that the last_insn, for such purposes, should be the
                   2739:      last insn before computing the return value.  Otherwise, cleanups
                   2740:      which call functions can clobber the return value.  */
                   2741:   /* ??? rms: I think that is erroneous, because in C++ it would
                   2742:      run destructors on variables that might be used in the subsequent
                   2743:      computation of the return value.  */
                   2744:   rtx last_insn = 0;
                   2745:   register rtx val = 0;
                   2746:   register rtx op0;
                   2747:   tree retval_rhs;
                   2748:   int cleanups;
                   2749:   struct nesting *block;
                   2750: 
                   2751:   /* Bytecode returns are quite simple, just leave the result on the
                   2752:      arithmetic stack. */
                   2753:   if (output_bytecode)
                   2754:     {
                   2755:       bc_expand_expr (retval);
                   2756:       bc_emit_instruction (ret);
                   2757:       return;
                   2758:     }
                   2759:   
                   2760:   /* If function wants no value, give it none.  */
                   2761:   if (TREE_CODE (TREE_TYPE (TREE_TYPE (current_function_decl))) == VOID_TYPE)
                   2762:     {
                   2763:       expand_expr (retval, NULL_RTX, VOIDmode, 0);
                   2764:       emit_queue ();
                   2765:       expand_null_return ();
                   2766:       return;
                   2767:     }
                   2768: 
                   2769:   /* Are any cleanups needed?  E.g. C++ destructors to be run?  */
                   2770:   cleanups = any_pending_cleanups (1);
                   2771: 
                   2772:   if (TREE_CODE (retval) == RESULT_DECL)
                   2773:     retval_rhs = retval;
                   2774:   else if ((TREE_CODE (retval) == MODIFY_EXPR || TREE_CODE (retval) == INIT_EXPR)
                   2775:           && TREE_CODE (TREE_OPERAND (retval, 0)) == RESULT_DECL)
                   2776:     retval_rhs = TREE_OPERAND (retval, 1);
                   2777:   else if (TREE_TYPE (retval) == void_type_node)
                   2778:     /* Recognize tail-recursive call to void function.  */
                   2779:     retval_rhs = retval;
                   2780:   else
                   2781:     retval_rhs = NULL_TREE;
                   2782: 
                   2783:   /* Only use `last_insn' if there are cleanups which must be run.  */
                   2784:   if (cleanups || cleanup_label != 0)
                   2785:     last_insn = get_last_insn ();
                   2786: 
                   2787:   /* Distribute return down conditional expr if either of the sides
                   2788:      may involve tail recursion (see test below).  This enhances the number
                   2789:      of tail recursions we see.  Don't do this always since it can produce
                   2790:      sub-optimal code in some cases and we distribute assignments into
                   2791:      conditional expressions when it would help.  */
                   2792: 
                   2793:   if (optimize && retval_rhs != 0
                   2794:       && frame_offset == 0
                   2795:       && TREE_CODE (retval_rhs) == COND_EXPR
                   2796:       && (TREE_CODE (TREE_OPERAND (retval_rhs, 1)) == CALL_EXPR
                   2797:          || TREE_CODE (TREE_OPERAND (retval_rhs, 2)) == CALL_EXPR))
                   2798:     {
                   2799:       rtx label = gen_label_rtx ();
                   2800:       tree expr;
                   2801: 
                   2802:       do_jump (TREE_OPERAND (retval_rhs, 0), label, NULL_RTX);
                   2803:       expr = build (MODIFY_EXPR, TREE_TYPE (current_function_decl),
                   2804:                    DECL_RESULT (current_function_decl),
                   2805:                    TREE_OPERAND (retval_rhs, 1));
                   2806:       TREE_SIDE_EFFECTS (expr) = 1;
                   2807:       expand_return (expr);
                   2808:       emit_label (label);
                   2809: 
                   2810:       expr = build (MODIFY_EXPR, TREE_TYPE (current_function_decl),
                   2811:                    DECL_RESULT (current_function_decl),
                   2812:                    TREE_OPERAND (retval_rhs, 2));
                   2813:       TREE_SIDE_EFFECTS (expr) = 1;
                   2814:       expand_return (expr);
                   2815:       return;
                   2816:     }
                   2817: 
                   2818:   /* For tail-recursive call to current function,
                   2819:      just jump back to the beginning.
                   2820:      It's unsafe if any auto variable in this function
                   2821:      has its address taken; for simplicity,
                   2822:      require stack frame to be empty.  */
                   2823:   if (optimize && retval_rhs != 0
                   2824:       && frame_offset == 0
                   2825:       && TREE_CODE (retval_rhs) == CALL_EXPR
                   2826:       && TREE_CODE (TREE_OPERAND (retval_rhs, 0)) == ADDR_EXPR
                   2827:       && TREE_OPERAND (TREE_OPERAND (retval_rhs, 0), 0) == current_function_decl
                   2828:       /* Finish checking validity, and if valid emit code
                   2829:         to set the argument variables for the new call.  */
                   2830:       && tail_recursion_args (TREE_OPERAND (retval_rhs, 1),
                   2831:                              DECL_ARGUMENTS (current_function_decl)))
                   2832:     {
                   2833:       if (tail_recursion_label == 0)
                   2834:        {
                   2835:          tail_recursion_label = gen_label_rtx ();
                   2836:          emit_label_after (tail_recursion_label,
                   2837:                            tail_recursion_reentry);
                   2838:        }
                   2839:       emit_queue ();
                   2840:       expand_goto_internal (NULL_TREE, tail_recursion_label, last_insn);
                   2841:       emit_barrier ();
                   2842:       return;
                   2843:     }
                   2844: #ifdef HAVE_return
                   2845:   /* This optimization is safe if there are local cleanups
                   2846:      because expand_null_return takes care of them.
                   2847:      ??? I think it should also be safe when there is a cleanup label,
                   2848:      because expand_null_return takes care of them, too.
                   2849:      Any reason why not?  */
                   2850:   if (HAVE_return && cleanup_label == 0
                   2851:       && ! current_function_returns_pcc_struct
                   2852:       && BRANCH_COST <= 1)
                   2853:     {
                   2854:       /* If this is  return x == y;  then generate
                   2855:         if (x == y) return 1; else return 0;
                   2856:         if we can do it with explicit return insns and
                   2857:         branches are cheap.  */
                   2858:       if (retval_rhs)
                   2859:        switch (TREE_CODE (retval_rhs))
                   2860:          {
                   2861:          case EQ_EXPR:
                   2862:          case NE_EXPR:
                   2863:          case GT_EXPR:
                   2864:          case GE_EXPR:
                   2865:          case LT_EXPR:
                   2866:          case LE_EXPR:
                   2867:          case TRUTH_ANDIF_EXPR:
                   2868:          case TRUTH_ORIF_EXPR:
                   2869:          case TRUTH_AND_EXPR:
                   2870:          case TRUTH_OR_EXPR:
                   2871:          case TRUTH_NOT_EXPR:
                   2872:          case TRUTH_XOR_EXPR:
                   2873:            op0 = gen_label_rtx ();
                   2874:            jumpifnot (retval_rhs, op0);
                   2875:            expand_value_return (const1_rtx);
                   2876:            emit_label (op0);
                   2877:            expand_value_return (const0_rtx);
                   2878:            return;
                   2879:          }
                   2880:     }
                   2881: #endif /* HAVE_return */
                   2882: 
                   2883:   if (cleanups
                   2884:       && retval_rhs != 0
                   2885:       && TREE_TYPE (retval_rhs) != void_type_node
                   2886:       && GET_CODE (DECL_RTL (DECL_RESULT (current_function_decl))) == REG)
                   2887:     {
                   2888:       /* Calculate the return value into a pseudo reg.  */
                   2889:       val = expand_expr (retval_rhs, NULL_RTX, VOIDmode, 0);
                   2890:       emit_queue ();
                   2891:       /* All temporaries have now been used.  */
                   2892:       free_temp_slots ();
                   2893:       /* Return the calculated value, doing cleanups first.  */
                   2894:       expand_value_return (val);
                   2895:     }
                   2896:   else
                   2897:     {
                   2898:       /* No cleanups or no hard reg used;
                   2899:         calculate value into hard return reg.  */
                   2900:       expand_expr (retval, const0_rtx, VOIDmode, 0);
                   2901:       emit_queue ();
                   2902:       free_temp_slots ();
                   2903:       expand_value_return (DECL_RTL (DECL_RESULT (current_function_decl)));
                   2904:     }
                   2905: }
                   2906: 
                   2907: /* Return 1 if the end of the generated RTX is not a barrier.
                   2908:    This means code already compiled can drop through.  */
                   2909: 
                   2910: int
                   2911: drop_through_at_end_p ()
                   2912: {
                   2913:   rtx insn = get_last_insn ();
                   2914:   while (insn && GET_CODE (insn) == NOTE)
                   2915:     insn = PREV_INSN (insn);
                   2916:   return insn && GET_CODE (insn) != BARRIER;
                   2917: }
                   2918: 
                   2919: /* Emit code to alter this function's formal parms for a tail-recursive call.
                   2920:    ACTUALS is a list of actual parameter expressions (chain of TREE_LISTs).
                   2921:    FORMALS is the chain of decls of formals.
                   2922:    Return 1 if this can be done;
                   2923:    otherwise return 0 and do not emit any code.  */
                   2924: 
                   2925: static int
                   2926: tail_recursion_args (actuals, formals)
                   2927:      tree actuals, formals;
                   2928: {
                   2929:   register tree a = actuals, f = formals;
                   2930:   register int i;
                   2931:   register rtx *argvec;
                   2932: 
                   2933:   /* Check that number and types of actuals are compatible
                   2934:      with the formals.  This is not always true in valid C code.
                   2935:      Also check that no formal needs to be addressable
                   2936:      and that all formals are scalars.  */
                   2937: 
                   2938:   /* Also count the args.  */
                   2939: 
                   2940:   for (a = actuals, f = formals, i = 0; a && f; a = TREE_CHAIN (a), f = TREE_CHAIN (f), i++)
                   2941:     {
                   2942:       if (TREE_TYPE (TREE_VALUE (a)) != TREE_TYPE (f))
                   2943:        return 0;
                   2944:       if (GET_CODE (DECL_RTL (f)) != REG || DECL_MODE (f) == BLKmode)
                   2945:        return 0;
                   2946:     }
                   2947:   if (a != 0 || f != 0)
                   2948:     return 0;
                   2949: 
                   2950:   /* Compute all the actuals.  */
                   2951: 
                   2952:   argvec = (rtx *) alloca (i * sizeof (rtx));
                   2953: 
                   2954:   for (a = actuals, i = 0; a; a = TREE_CHAIN (a), i++)
                   2955:     argvec[i] = expand_expr (TREE_VALUE (a), NULL_RTX, VOIDmode, 0);
                   2956: 
                   2957:   /* Find which actual values refer to current values of previous formals.
                   2958:      Copy each of them now, before any formal is changed.  */
                   2959: 
                   2960:   for (a = actuals, i = 0; a; a = TREE_CHAIN (a), i++)
                   2961:     {
                   2962:       int copy = 0;
                   2963:       register int j;
                   2964:       for (f = formals, j = 0; j < i; f = TREE_CHAIN (f), j++)
                   2965:        if (reg_mentioned_p (DECL_RTL (f), argvec[i]))
                   2966:          { copy = 1; break; }
                   2967:       if (copy)
                   2968:        argvec[i] = copy_to_reg (argvec[i]);
                   2969:     }
                   2970: 
                   2971:   /* Store the values of the actuals into the formals.  */
                   2972: 
                   2973:   for (f = formals, a = actuals, i = 0; f;
                   2974:        f = TREE_CHAIN (f), a = TREE_CHAIN (a), i++)
                   2975:     {
                   2976:       if (GET_MODE (DECL_RTL (f)) == GET_MODE (argvec[i]))
                   2977:        emit_move_insn (DECL_RTL (f), argvec[i]);
                   2978:       else
                   2979:        convert_move (DECL_RTL (f), argvec[i],
                   2980:                      TREE_UNSIGNED (TREE_TYPE (TREE_VALUE (a))));
                   2981:     }
                   2982: 
                   2983:   free_temp_slots ();
                   2984:   return 1;
                   2985: }
                   2986: 
                   2987: /* Generate the RTL code for entering a binding contour.
                   2988:    The variables are declared one by one, by calls to `expand_decl'.
                   2989: 
                   2990:    EXIT_FLAG is nonzero if this construct should be visible to
                   2991:    `exit_something'.  */
                   2992: 
                   2993: void
                   2994: expand_start_bindings (exit_flag)
                   2995:      int exit_flag;
                   2996: {
                   2997:   struct nesting *thisblock = ALLOC_NESTING ();
                   2998:   rtx note;
                   2999: 
                   3000:   if (!output_bytecode)
                   3001:     note = emit_note (NULL_PTR, NOTE_INSN_BLOCK_BEG);
                   3002: 
                   3003:   /* Make an entry on block_stack for the block we are entering.  */
                   3004: 
                   3005:   thisblock->next = block_stack;
                   3006:   thisblock->all = nesting_stack;
                   3007:   thisblock->depth = ++nesting_depth;
                   3008:   thisblock->data.block.stack_level = 0;
                   3009:   thisblock->data.block.cleanups = 0;
                   3010:   thisblock->data.block.function_call_count = 0;
                   3011: #if 0
                   3012:   if (block_stack)
                   3013:     {
                   3014:       if (block_stack->data.block.cleanups == NULL_TREE
                   3015:          && (block_stack->data.block.outer_cleanups == NULL_TREE
                   3016:              || block_stack->data.block.outer_cleanups == empty_cleanup_list))
                   3017:        thisblock->data.block.outer_cleanups = empty_cleanup_list;
                   3018:       else
                   3019:        thisblock->data.block.outer_cleanups
                   3020:          = tree_cons (NULL_TREE, block_stack->data.block.cleanups,
                   3021:                       block_stack->data.block.outer_cleanups);
                   3022:     }
                   3023:   else
                   3024:     thisblock->data.block.outer_cleanups = 0;
                   3025: #endif
                   3026: #if 1
                   3027:   if (block_stack
                   3028:       && !(block_stack->data.block.cleanups == NULL_TREE
                   3029:           && block_stack->data.block.outer_cleanups == NULL_TREE))
                   3030:     thisblock->data.block.outer_cleanups
                   3031:       = tree_cons (NULL_TREE, block_stack->data.block.cleanups,
                   3032:                   block_stack->data.block.outer_cleanups);
                   3033:   else
                   3034:     thisblock->data.block.outer_cleanups = 0;
                   3035: #endif
                   3036:   thisblock->data.block.label_chain = 0;
                   3037:   thisblock->data.block.innermost_stack_block = stack_block_stack;
                   3038:   thisblock->data.block.first_insn = note;
                   3039:   thisblock->data.block.block_start_count = ++block_start_count;
                   3040:   thisblock->exit_label = exit_flag ? gen_label_rtx () : 0;
                   3041:   block_stack = thisblock;
                   3042:   nesting_stack = thisblock;
                   3043: 
                   3044:   if (!output_bytecode)
                   3045:     {
                   3046:       /* Make a new level for allocating stack slots.  */
                   3047:       push_temp_slots ();
                   3048:     }
                   3049: }
                   3050: 
                   3051: /* Given a pointer to a BLOCK node, save a pointer to the most recently
                   3052:    generated NOTE_INSN_BLOCK_END in the BLOCK_END_NOTE field of the given
                   3053:    BLOCK node.  */
                   3054: 
                   3055: void
                   3056: remember_end_note (block)
                   3057:      register tree block;
                   3058: {
                   3059:   BLOCK_END_NOTE (block) = last_block_end_note;
                   3060:   last_block_end_note = NULL_RTX;
                   3061: }
                   3062: 
                   3063: /* Generate RTL code to terminate a binding contour.
                   3064:    VARS is the chain of VAR_DECL nodes
                   3065:    for the variables bound in this contour.
                   3066:    MARK_ENDS is nonzero if we should put a note at the beginning
                   3067:    and end of this binding contour.
                   3068: 
                   3069:    DONT_JUMP_IN is nonzero if it is not valid to jump into this contour.
                   3070:    (That is true automatically if the contour has a saved stack level.)  */
                   3071: 
                   3072: void
                   3073: expand_end_bindings (vars, mark_ends, dont_jump_in)
                   3074:      tree vars;
                   3075:      int mark_ends;
                   3076:      int dont_jump_in;
                   3077: {
                   3078:   register struct nesting *thisblock = block_stack;
                   3079:   register tree decl;
                   3080: 
                   3081:   if (output_bytecode)
                   3082:     {
                   3083:       bc_expand_end_bindings (vars, mark_ends, dont_jump_in);
                   3084:       return;
                   3085:     }
                   3086: 
                   3087:   if (warn_unused)
                   3088:     for (decl = vars; decl; decl = TREE_CHAIN (decl))
                   3089:       if (! TREE_USED (decl) && TREE_CODE (decl) == VAR_DECL
                   3090:          && ! DECL_IN_SYSTEM_HEADER (decl))
                   3091:        warning_with_decl (decl, "unused variable `%s'");
                   3092: 
                   3093:   if (thisblock->exit_label)
                   3094:     {
                   3095:       do_pending_stack_adjust ();
                   3096:       emit_label (thisblock->exit_label);
                   3097:     }
                   3098: 
                   3099:   /* If necessary, make a handler for nonlocal gotos taking
                   3100:      place in the function calls in this block.  */
                   3101:   if (function_call_count != thisblock->data.block.function_call_count
                   3102:       && nonlocal_labels
                   3103:       /* Make handler for outermost block
                   3104:         if there were any nonlocal gotos to this function.  */
                   3105:       && (thisblock->next == 0 ? current_function_has_nonlocal_label
                   3106:          /* Make handler for inner block if it has something
                   3107:             special to do when you jump out of it.  */
                   3108:          : (thisblock->data.block.cleanups != 0
                   3109:             || thisblock->data.block.stack_level != 0)))
                   3110:     {
                   3111:       tree link;
                   3112:       rtx afterward = gen_label_rtx ();
                   3113:       rtx handler_label = gen_label_rtx ();
                   3114:       rtx save_receiver = gen_reg_rtx (Pmode);
                   3115:       rtx insns;
                   3116: 
                   3117:       /* Don't let jump_optimize delete the handler.  */
                   3118:       LABEL_PRESERVE_P (handler_label) = 1;
                   3119: 
                   3120:       /* Record the handler address in the stack slot for that purpose,
                   3121:         during this block, saving and restoring the outer value.  */
                   3122:       if (thisblock->next != 0)
                   3123:        {
                   3124:          emit_move_insn (nonlocal_goto_handler_slot, save_receiver);
                   3125: 
                   3126:          start_sequence ();
                   3127:          emit_move_insn (save_receiver, nonlocal_goto_handler_slot);
                   3128:          insns = get_insns ();
                   3129:          end_sequence ();
                   3130:          emit_insns_before (insns, thisblock->data.block.first_insn);
                   3131:        }
                   3132: 
                   3133:       start_sequence ();
                   3134:       emit_move_insn (nonlocal_goto_handler_slot,
                   3135:                      gen_rtx (LABEL_REF, Pmode, handler_label));
                   3136:       insns = get_insns ();
                   3137:       end_sequence ();
                   3138:       emit_insns_before (insns, thisblock->data.block.first_insn);
                   3139: 
                   3140:       /* Jump around the handler; it runs only when specially invoked.  */
                   3141:       emit_jump (afterward);
                   3142:       emit_label (handler_label);
                   3143: 
                   3144: #ifdef HAVE_nonlocal_goto
                   3145:       if (! HAVE_nonlocal_goto)
                   3146: #endif
                   3147:        /* First adjust our frame pointer to its actual value.  It was
                   3148:           previously set to the start of the virtual area corresponding to
                   3149:           the stacked variables when we branched here and now needs to be
                   3150:           adjusted to the actual hardware fp value.
                   3151: 
                   3152:           Assignments are to virtual registers are converted by
                   3153:           instantiate_virtual_regs into the corresponding assignment
                   3154:           to the underlying register (fp in this case) that makes
                   3155:           the original assignment true.
                   3156:           So the following insn will actually be
                   3157:           decrementing fp by STARTING_FRAME_OFFSET.  */
                   3158:        emit_move_insn (virtual_stack_vars_rtx, frame_pointer_rtx);
                   3159: 
                   3160: #if ARG_POINTER_REGNUM != HARD_FRAME_POINTER_REGNUM
                   3161:       if (fixed_regs[ARG_POINTER_REGNUM])
                   3162:        {
                   3163: #ifdef ELIMINABLE_REGS
                   3164:          /* If the argument pointer can be eliminated in favor of the
                   3165:             frame pointer, we don't need to restore it.  We assume here
                   3166:             that if such an elimination is present, it can always be used.
                   3167:             This is the case on all known machines; if we don't make this
                   3168:             assumption, we do unnecessary saving on many machines.  */
                   3169:          static struct elims {int from, to;} elim_regs[] = ELIMINABLE_REGS;
                   3170:          int i;
                   3171: 
                   3172:          for (i = 0; i < sizeof elim_regs / sizeof elim_regs[0]; i++)
                   3173:            if (elim_regs[i].from == ARG_POINTER_REGNUM
                   3174:                && elim_regs[i].to == HARD_FRAME_POINTER_REGNUM)
                   3175:              break;
                   3176: 
                   3177:          if (i == sizeof elim_regs / sizeof elim_regs [0])
                   3178: #endif
                   3179:            {
                   3180:              /* Now restore our arg pointer from the address at which it
                   3181:                 was saved in our stack frame.
                   3182:                 If there hasn't be space allocated for it yet, make
                   3183:                 some now.  */
                   3184:              if (arg_pointer_save_area == 0)
                   3185:                arg_pointer_save_area
                   3186:                  = assign_stack_local (Pmode, GET_MODE_SIZE (Pmode), 0);
                   3187:              emit_move_insn (virtual_incoming_args_rtx,
                   3188:                              /* We need a pseudo here, or else
                   3189:                                 instantiate_virtual_regs_1 complains.  */
                   3190:                              copy_to_reg (arg_pointer_save_area));
                   3191:            }
                   3192:        }
                   3193: #endif
                   3194: 
                   3195:       /* The handler expects the desired label address in the static chain
                   3196:         register.  It tests the address and does an appropriate jump
                   3197:         to whatever label is desired.  */
                   3198:       for (link = nonlocal_labels; link; link = TREE_CHAIN (link))
                   3199:        /* Skip any labels we shouldn't be able to jump to from here.  */
                   3200:        if (! DECL_TOO_LATE (TREE_VALUE (link)))
                   3201:          {
                   3202:            rtx not_this = gen_label_rtx ();
                   3203:            rtx this = gen_label_rtx ();
                   3204:            do_jump_if_equal (static_chain_rtx,
                   3205:                              gen_rtx (LABEL_REF, Pmode, DECL_RTL (TREE_VALUE (link))),
                   3206:                              this, 0);
                   3207:            emit_jump (not_this);
                   3208:            emit_label (this);
                   3209:            expand_goto (TREE_VALUE (link));
                   3210:            emit_label (not_this);
                   3211:          }
                   3212:       /* If label is not recognized, abort.  */
                   3213:       emit_library_call (gen_rtx (SYMBOL_REF, Pmode, "abort"), 0,
                   3214:                         VOIDmode, 0);
                   3215:       emit_label (afterward);
                   3216:     }
                   3217: 
                   3218:   /* Don't allow jumping into a block that has cleanups or a stack level.  */
                   3219:   if (dont_jump_in
                   3220:       || thisblock->data.block.stack_level != 0
                   3221:       || thisblock->data.block.cleanups != 0)
                   3222:     {
                   3223:       struct label_chain *chain;
                   3224: 
                   3225:       /* Any labels in this block are no longer valid to go to.
                   3226:         Mark them to cause an error message.  */
                   3227:       for (chain = thisblock->data.block.label_chain; chain; chain = chain->next)
                   3228:        {
                   3229:          DECL_TOO_LATE (chain->label) = 1;
                   3230:          /* If any goto without a fixup came to this label,
                   3231:             that must be an error, because gotos without fixups
                   3232:             come from outside all saved stack-levels and all cleanups.  */
                   3233:          if (TREE_ADDRESSABLE (chain->label))
                   3234:            error_with_decl (chain->label,
                   3235:                             "label `%s' used before containing binding contour");
                   3236:        }
                   3237:     }
                   3238: 
                   3239:   /* Restore stack level in effect before the block
                   3240:      (only if variable-size objects allocated).  */
                   3241:   /* Perform any cleanups associated with the block.  */
                   3242: 
                   3243:   if (thisblock->data.block.stack_level != 0
                   3244:       || thisblock->data.block.cleanups != 0)
                   3245:     {
                   3246:       /* Don't let cleanups affect ({...}) constructs.  */
                   3247:       int old_expr_stmts_for_value = expr_stmts_for_value;
                   3248:       rtx old_last_expr_value = last_expr_value;
                   3249:       tree old_last_expr_type = last_expr_type;
                   3250:       expr_stmts_for_value = 0;
                   3251: 
                   3252:       /* Do the cleanups.  */
                   3253:       expand_cleanups (thisblock->data.block.cleanups, NULL_TREE);
                   3254:       do_pending_stack_adjust ();
                   3255: 
                   3256:       expr_stmts_for_value = old_expr_stmts_for_value;
                   3257:       last_expr_value = old_last_expr_value;
                   3258:       last_expr_type = old_last_expr_type;
                   3259: 
                   3260:       /* Restore the stack level.  */
                   3261: 
                   3262:       if (thisblock->data.block.stack_level != 0)
                   3263:        {
                   3264:          emit_stack_restore (thisblock->next ? SAVE_BLOCK : SAVE_FUNCTION,
                   3265:                              thisblock->data.block.stack_level, NULL_RTX);
                   3266:          if (nonlocal_goto_handler_slot != 0)
                   3267:            emit_stack_save (SAVE_NONLOCAL, &nonlocal_goto_stack_level,
                   3268:                             NULL_RTX);
                   3269:        }
                   3270: 
                   3271:       /* Any gotos out of this block must also do these things.
                   3272:         Also report any gotos with fixups that came to labels in this
                   3273:         level.  */
                   3274:       fixup_gotos (thisblock,
                   3275:                   thisblock->data.block.stack_level,
                   3276:                   thisblock->data.block.cleanups,
                   3277:                   thisblock->data.block.first_insn,
                   3278:                   dont_jump_in);
                   3279:     }
                   3280: 
                   3281:   /* Mark the beginning and end of the scope if requested.
                   3282:      We do this now, after running cleanups on the variables
                   3283:      just going out of scope, so they are in scope for their cleanups.  */
                   3284: 
                   3285:   if (mark_ends)
                   3286:     last_block_end_note = emit_note (NULL_PTR, NOTE_INSN_BLOCK_END);
                   3287:   else
                   3288:     /* Get rid of the beginning-mark if we don't make an end-mark.  */
                   3289:     NOTE_LINE_NUMBER (thisblock->data.block.first_insn) = NOTE_INSN_DELETED;
                   3290: 
                   3291:   /* If doing stupid register allocation, make sure lives of all
                   3292:      register variables declared here extend thru end of scope.  */
                   3293: 
                   3294:   if (obey_regdecls)
                   3295:     for (decl = vars; decl; decl = TREE_CHAIN (decl))
                   3296:       {
                   3297:        rtx rtl = DECL_RTL (decl);
                   3298:        if (TREE_CODE (decl) == VAR_DECL && rtl != 0)
                   3299:          use_variable (rtl);
                   3300:       }
                   3301: 
                   3302:   /* Restore block_stack level for containing block.  */
                   3303: 
                   3304:   stack_block_stack = thisblock->data.block.innermost_stack_block;
                   3305:   POPSTACK (block_stack);
                   3306: 
                   3307:   /* Pop the stack slot nesting and free any slots at this level.  */
                   3308:   pop_temp_slots ();
                   3309: }
                   3310: 
                   3311: 
                   3312: /* End a binding contour.
                   3313:    VARS is the chain of VAR_DECL nodes for the variables bound
                   3314:    in this contour.  MARK_ENDS is nonzer if we should put a note
                   3315:    at the beginning and end of this binding contour.
                   3316:    DONT_JUMP_IN is nonzero if it is not valid to jump into this
                   3317:    contour.  */
                   3318: 
                   3319: void
                   3320: bc_expand_end_bindings (vars, mark_ends, dont_jump_in)
                   3321:      tree vars;
                   3322:      int mark_ends;
                   3323:      int dont_jump_in;
                   3324: {
                   3325:   struct nesting *thisbind = nesting_stack;
                   3326:   tree decl;
                   3327: 
                   3328:   if (warn_unused)
                   3329:     for (decl = vars; decl; decl = TREE_CHAIN (decl))
                   3330:       if (! TREE_USED (TREE_VALUE (decl)) && TREE_CODE (TREE_VALUE (decl)) == VAR_DECL)
                   3331:        warning_with_decl (decl, "unused variable `%s'");
                   3332: 
                   3333:   bc_emit_bytecode_labeldef (BYTECODE_BC_LABEL (thisbind->exit_label));
                   3334: 
                   3335:   /* Pop block/bindings off stack */
                   3336:   POPSTACK (nesting_stack);
                   3337:   POPSTACK (block_stack);
                   3338: }
                   3339: 
                   3340: /* Generate RTL for the automatic variable declaration DECL.
                   3341:    (Other kinds of declarations are simply ignored if seen here.)
                   3342:    CLEANUP is an expression to be executed at exit from this binding contour;
                   3343:    for example, in C++, it might call the destructor for this variable.
                   3344: 
                   3345:    If CLEANUP contains any SAVE_EXPRs, then you must preevaluate them
                   3346:    either before or after calling `expand_decl' but before compiling
                   3347:    any subsequent expressions.  This is because CLEANUP may be expanded
                   3348:    more than once, on different branches of execution.
                   3349:    For the same reason, CLEANUP may not contain a CALL_EXPR
                   3350:    except as its topmost node--else `preexpand_calls' would get confused.
                   3351: 
                   3352:    If CLEANUP is nonzero and DECL is zero, we record a cleanup
                   3353:    that is not associated with any particular variable.
                   3354: 
                   3355:    There is no special support here for C++ constructors.
                   3356:    They should be handled by the proper code in DECL_INITIAL.  */
                   3357: 
                   3358: void
                   3359: expand_decl (decl)
                   3360:      register tree decl;
                   3361: {
                   3362:   struct nesting *thisblock = block_stack;
                   3363:   tree type;
                   3364: 
                   3365:   if (output_bytecode)
                   3366:     {
                   3367:       bc_expand_decl (decl, 0);
                   3368:       return;
                   3369:     }
                   3370: 
                   3371:   type = TREE_TYPE (decl);
                   3372: 
                   3373:   /* Only automatic variables need any expansion done.
                   3374:      Static and external variables, and external functions,
                   3375:      will be handled by `assemble_variable' (called from finish_decl).
                   3376:      TYPE_DECL and CONST_DECL require nothing.
                   3377:      PARM_DECLs are handled in `assign_parms'.  */
                   3378: 
                   3379:   if (TREE_CODE (decl) != VAR_DECL)
                   3380:     return;
                   3381:   if (TREE_STATIC (decl) || DECL_EXTERNAL (decl))
                   3382:     return;
                   3383: 
                   3384:   /* Create the RTL representation for the variable.  */
                   3385: 
                   3386:   if (type == error_mark_node)
                   3387:     DECL_RTL (decl) = gen_rtx (MEM, BLKmode, const0_rtx);
                   3388:   else if (DECL_SIZE (decl) == 0)
                   3389:     /* Variable with incomplete type.  */
                   3390:     {
                   3391:       if (DECL_INITIAL (decl) == 0)
                   3392:        /* Error message was already done; now avoid a crash.  */
                   3393:        DECL_RTL (decl) = assign_stack_temp (DECL_MODE (decl), 0, 1);
                   3394:       else
                   3395:        /* An initializer is going to decide the size of this array.
                   3396:           Until we know the size, represent its address with a reg.  */
                   3397:        DECL_RTL (decl) = gen_rtx (MEM, BLKmode, gen_reg_rtx (Pmode));
                   3398:     }
                   3399:   else if (DECL_MODE (decl) != BLKmode
                   3400:           /* If -ffloat-store, don't put explicit float vars
                   3401:              into regs.  */
                   3402:           && !(flag_float_store
                   3403:                && TREE_CODE (type) == REAL_TYPE)
                   3404:           && ! TREE_THIS_VOLATILE (decl)
                   3405:           && ! TREE_ADDRESSABLE (decl)
                   3406:           && (DECL_REGISTER (decl) || ! obey_regdecls))
                   3407:     {
                   3408:       /* Automatic variable that can go in a register.  */
                   3409:       enum machine_mode reg_mode = DECL_MODE (decl);
                   3410:       int unsignedp = TREE_UNSIGNED (type);
                   3411: 
                   3412:       if (TREE_CODE (type) == INTEGER_TYPE || TREE_CODE (type) == ENUMERAL_TYPE
                   3413:          || TREE_CODE (type) == BOOLEAN_TYPE || TREE_CODE (type) == CHAR_TYPE
                   3414:          || TREE_CODE (type) == REAL_TYPE || TREE_CODE (type) == POINTER_TYPE
                   3415:          || TREE_CODE (type) == OFFSET_TYPE)
                   3416:        {
                   3417:          PROMOTE_MODE (reg_mode, unsignedp, type);
                   3418:        }
                   3419: 
                   3420:       if (TREE_CODE (type) == COMPLEX_TYPE)
                   3421:        {
                   3422:          rtx realpart, imagpart;
                   3423:          enum machine_mode partmode = TYPE_MODE (TREE_TYPE (type));
                   3424: 
                   3425:          /* For a complex type variable, make a CONCAT of two pseudos
                   3426:             so that the real and imaginary parts
                   3427:             can be allocated separately.  */
                   3428:          realpart = gen_reg_rtx (partmode);
                   3429:          REG_USERVAR_P (realpart) = 1;
                   3430:          imagpart = gen_reg_rtx (partmode);
                   3431:          REG_USERVAR_P (imagpart) = 1;
                   3432:          DECL_RTL (decl) = gen_rtx (CONCAT, reg_mode, realpart, imagpart);
                   3433:        }
                   3434:       else
                   3435:        {
                   3436:          DECL_RTL (decl) = gen_reg_rtx (reg_mode);
                   3437:          if (TREE_CODE (type) == POINTER_TYPE)
                   3438:            mark_reg_pointer (DECL_RTL (decl));
                   3439:          REG_USERVAR_P (DECL_RTL (decl)) = 1;
                   3440:        }
                   3441:     }
                   3442:   else if (TREE_CODE (DECL_SIZE (decl)) == INTEGER_CST)
                   3443:     {
                   3444:       /* Variable of fixed size that goes on the stack.  */
                   3445:       rtx oldaddr = 0;
                   3446:       rtx addr;
                   3447: 
                   3448:       /* If we previously made RTL for this decl, it must be an array
                   3449:         whose size was determined by the initializer.
                   3450:         The old address was a register; set that register now
                   3451:         to the proper address.  */
                   3452:       if (DECL_RTL (decl) != 0)
                   3453:        {
                   3454:          if (GET_CODE (DECL_RTL (decl)) != MEM
                   3455:              || GET_CODE (XEXP (DECL_RTL (decl), 0)) != REG)
                   3456:            abort ();
                   3457:          oldaddr = XEXP (DECL_RTL (decl), 0);
                   3458:        }
                   3459: 
                   3460:       DECL_RTL (decl)
                   3461:        = assign_stack_temp (DECL_MODE (decl),
                   3462:                             ((TREE_INT_CST_LOW (DECL_SIZE (decl))
                   3463:                               + BITS_PER_UNIT - 1)
                   3464:                              / BITS_PER_UNIT),
                   3465:                             1);
                   3466: 
                   3467:       /* Set alignment we actually gave this decl.  */
                   3468:       DECL_ALIGN (decl) = (DECL_MODE (decl) == BLKmode ? BIGGEST_ALIGNMENT
                   3469:                           : GET_MODE_BITSIZE (DECL_MODE (decl)));
                   3470: 
                   3471:       if (oldaddr)
                   3472:        {
                   3473:          addr = force_operand (XEXP (DECL_RTL (decl), 0), oldaddr);
                   3474:          if (addr != oldaddr)
                   3475:            emit_move_insn (oldaddr, addr);
                   3476:        }
                   3477: 
                   3478:       /* If this is a memory ref that contains aggregate components,
                   3479:         mark it as such for cse and loop optimize.  */
                   3480:       MEM_IN_STRUCT_P (DECL_RTL (decl))
                   3481:        = (TREE_CODE (TREE_TYPE (decl)) == ARRAY_TYPE
                   3482:           || TREE_CODE (TREE_TYPE (decl)) == RECORD_TYPE
                   3483:           || TREE_CODE (TREE_TYPE (decl)) == UNION_TYPE
                   3484:           || TREE_CODE (TREE_TYPE (decl)) == QUAL_UNION_TYPE);
                   3485: #if 0
                   3486:       /* If this is in memory because of -ffloat-store,
                   3487:         set the volatile bit, to prevent optimizations from
                   3488:         undoing the effects.  */
                   3489:       if (flag_float_store && TREE_CODE (type) == REAL_TYPE)
                   3490:        MEM_VOLATILE_P (DECL_RTL (decl)) = 1;
                   3491: #endif
                   3492:     }
                   3493:   else
                   3494:     /* Dynamic-size object: must push space on the stack.  */
                   3495:     {
                   3496:       rtx address, size;
                   3497: 
                   3498:       /* Record the stack pointer on entry to block, if have
                   3499:         not already done so.  */
                   3500:       if (thisblock->data.block.stack_level == 0)
                   3501:        {
                   3502:          do_pending_stack_adjust ();
                   3503:          emit_stack_save (thisblock->next ? SAVE_BLOCK : SAVE_FUNCTION,
                   3504:                           &thisblock->data.block.stack_level,
                   3505:                           thisblock->data.block.first_insn);
                   3506:          stack_block_stack = thisblock;
                   3507:        }
                   3508: 
                   3509:       /* Compute the variable's size, in bytes.  */
                   3510:       size = expand_expr (size_binop (CEIL_DIV_EXPR,
                   3511:                                      DECL_SIZE (decl),
                   3512:                                      size_int (BITS_PER_UNIT)),
                   3513:                          NULL_RTX, VOIDmode, 0);
                   3514:       free_temp_slots ();
                   3515: 
                   3516:       /* This is equivalent to calling alloca.  */
                   3517:       current_function_calls_alloca = 1;
                   3518: 
                   3519:       /* Allocate space on the stack for the variable.  */
                   3520:       address = allocate_dynamic_stack_space (size, NULL_RTX,
                   3521:                                              DECL_ALIGN (decl));
                   3522: 
                   3523:       if (nonlocal_goto_handler_slot != 0)
                   3524:        emit_stack_save (SAVE_NONLOCAL, &nonlocal_goto_stack_level, NULL_RTX);
                   3525: 
                   3526:       /* Reference the variable indirect through that rtx.  */
                   3527:       DECL_RTL (decl) = gen_rtx (MEM, DECL_MODE (decl), address);
                   3528: 
                   3529:       /* If this is a memory ref that contains aggregate components,
                   3530:         mark it as such for cse and loop optimize.  */
                   3531:       MEM_IN_STRUCT_P (DECL_RTL (decl))
                   3532:        = (TREE_CODE (TREE_TYPE (decl)) == ARRAY_TYPE
                   3533:           || TREE_CODE (TREE_TYPE (decl)) == RECORD_TYPE
                   3534:           || TREE_CODE (TREE_TYPE (decl)) == UNION_TYPE
                   3535:           || TREE_CODE (TREE_TYPE (decl)) == QUAL_UNION_TYPE);
                   3536: 
                   3537:       /* Indicate the alignment we actually gave this variable.  */
                   3538: #ifdef STACK_BOUNDARY
                   3539:       DECL_ALIGN (decl) = STACK_BOUNDARY;
                   3540: #else
                   3541:       DECL_ALIGN (decl) = BIGGEST_ALIGNMENT;
                   3542: #endif
                   3543:     }
                   3544: 
                   3545:   if (TREE_THIS_VOLATILE (decl))
                   3546:     MEM_VOLATILE_P (DECL_RTL (decl)) = 1;
                   3547: #if 0 /* A variable is not necessarily unchanging
                   3548:         just because it is const.  RTX_UNCHANGING_P
                   3549:         means no change in the function,
                   3550:         not merely no change in the variable's scope.
                   3551:         It is correct to set RTX_UNCHANGING_P if the variable's scope
                   3552:         is the whole function.  There's no convenient way to test that.  */
                   3553:   if (TREE_READONLY (decl))
                   3554:     RTX_UNCHANGING_P (DECL_RTL (decl)) = 1;
                   3555: #endif
                   3556: 
                   3557:   /* If doing stupid register allocation, make sure life of any
                   3558:      register variable starts here, at the start of its scope.  */
                   3559: 
                   3560:   if (obey_regdecls)
                   3561:     use_variable (DECL_RTL (decl));
                   3562: }
                   3563: 
                   3564: 
                   3565: /* Generate code for the automatic variable declaration DECL.  For
                   3566:    most variables this just means we give it a stack offset.  The
                   3567:    compiler sometimes emits cleanups without variables and we will
                   3568:    have to deal with those too.  */
                   3569: 
                   3570: void
                   3571: bc_expand_decl (decl, cleanup)
                   3572:      tree decl;
                   3573:      tree cleanup;
                   3574: {
                   3575:   tree type;
                   3576: 
                   3577:   if (!decl)
                   3578:     {
                   3579:       /* A cleanup with no variable.  */
                   3580:       if (!cleanup)
                   3581:        abort ();
                   3582: 
                   3583:       return;
                   3584:     }
                   3585: 
                   3586:   /* Only auto variables need any work.  */
                   3587:   if (TREE_CODE (decl) != VAR_DECL || TREE_STATIC (decl) || DECL_EXTERNAL (decl))
                   3588:     return;
                   3589: 
                   3590:   type = TREE_TYPE (decl);
                   3591: 
                   3592:   if (type == error_mark_node)
                   3593:     DECL_RTL (decl) = bc_gen_rtx ((char *) 0, 0, (struct bc_label *) 0);
                   3594: 
                   3595:   else if (DECL_SIZE (decl) == 0)
                   3596: 
                   3597:     /* Variable with incomplete type.  The stack offset herein will be
                   3598:        fixed later in expand_decl_init ().  */
                   3599:     DECL_RTL (decl) = bc_gen_rtx ((char *) 0, 0, (struct bc_label *) 0);
                   3600: 
                   3601:   else if (TREE_CONSTANT (DECL_SIZE (decl)))
                   3602:     {
                   3603:       DECL_RTL (decl) = bc_allocate_local (TREE_INT_CST_LOW (DECL_SIZE (decl)) / BITS_PER_UNIT,
                   3604:                                           DECL_ALIGN (decl));
                   3605:     }
                   3606:   else
                   3607:     DECL_RTL (decl) = bc_allocate_variable_array (DECL_SIZE (decl));
                   3608: }
                   3609: 
                   3610: /* Emit code to perform the initialization of a declaration DECL.  */
                   3611: 
                   3612: void
                   3613: expand_decl_init (decl)
                   3614:      tree decl;
                   3615: {
                   3616:   int was_used = TREE_USED (decl);
                   3617: 
                   3618:   /* If this is a CONST_DECL, we don't have to generate any code, but
                   3619:      if DECL_INITIAL is a constant, call expand_expr to force TREE_CST_RTL
                   3620:      to be set while in the obstack containing the constant.  If we don't
                   3621:      do this, we can lose if we have functions nested three deep and the middle
                   3622:      function makes a CONST_DECL whose DECL_INITIAL is a STRING_CST while
                   3623:      the innermost function is the first to expand that STRING_CST.  */
                   3624:   if (TREE_CODE (decl) == CONST_DECL)
                   3625:     {
                   3626:       if (DECL_INITIAL (decl) && TREE_CONSTANT (DECL_INITIAL (decl)))
                   3627:        expand_expr (DECL_INITIAL (decl), NULL_RTX, VOIDmode,
                   3628:                     EXPAND_INITIALIZER);
                   3629:       return;
                   3630:     }
                   3631: 
                   3632:   if (TREE_STATIC (decl))
                   3633:     return;
                   3634: 
                   3635:   /* Compute and store the initial value now.  */
                   3636: 
                   3637:   if (DECL_INITIAL (decl) == error_mark_node)
                   3638:     {
                   3639:       enum tree_code code = TREE_CODE (TREE_TYPE (decl));
                   3640:       if (code == INTEGER_TYPE || code == REAL_TYPE || code == ENUMERAL_TYPE
                   3641:          || code == POINTER_TYPE)
                   3642:        expand_assignment (decl, convert (TREE_TYPE (decl), integer_zero_node),
                   3643:                           0, 0);
                   3644:       emit_queue ();
                   3645:     }
                   3646:   else if (DECL_INITIAL (decl) && TREE_CODE (DECL_INITIAL (decl)) != TREE_LIST)
                   3647:     {
                   3648:       emit_line_note (DECL_SOURCE_FILE (decl), DECL_SOURCE_LINE (decl));
                   3649:       expand_assignment (decl, DECL_INITIAL (decl), 0, 0);
                   3650:       emit_queue ();
                   3651:     }
                   3652: 
                   3653:   /* Don't let the initialization count as "using" the variable.  */
                   3654:   TREE_USED (decl) = was_used;
                   3655: 
                   3656:   /* Free any temporaries we made while initializing the decl.  */
                   3657:   free_temp_slots ();
                   3658: }
                   3659: 
                   3660: /* Expand initialization for variable-sized types. Allocate array
                   3661:    using newlocalSI and set local variable, which is a pointer to the
                   3662:    storage. */
                   3663: 
                   3664: bc_expand_variable_local_init (decl)
                   3665:      tree decl;
                   3666: {
                   3667:   /* Evaluate size expression and coerce to SI */
                   3668:   bc_expand_expr (DECL_SIZE (decl));
                   3669: 
                   3670:   /* Type sizes are always (?) of TREE_CODE INTEGER_CST, so
                   3671:      no coercion is necessary (?) */
                   3672: 
                   3673: /*  emit_typecode_conversion (preferred_typecode (TYPE_MODE (DECL_SIZE (decl)),
                   3674:                                                TREE_UNSIGNED (DECL_SIZE (decl))), SIcode); */
                   3675: 
                   3676:   /* Emit code to allocate array */
                   3677:   bc_emit_instruction (newlocalSI);
                   3678: 
                   3679:   /* Store array pointer in local variable. This is the only instance
                   3680:      where we actually want the address of the pointer to the
                   3681:      variable-size block, rather than the pointer itself.  We avoid
                   3682:      using expand_address() since that would cause the pointer to be
                   3683:      pushed rather than its address. Hence the hard-coded reference;
                   3684:      notice also that the variable is always local (no global
                   3685:      variable-size type variables). */
                   3686: 
                   3687:   bc_load_localaddr (DECL_RTL (decl));
                   3688:   bc_emit_instruction (storeP);
                   3689: }
                   3690: 
                   3691: 
                   3692: /* Emit code to initialize a declaration.  */
                   3693: void
                   3694: bc_expand_decl_init (decl)
                   3695:      tree decl;
                   3696: {
                   3697:   int org_stack_depth;
                   3698: 
                   3699:   /* Statical initializers are handled elsewhere */
                   3700: 
                   3701:   if (TREE_STATIC (decl))
                   3702:     return;
                   3703: 
                   3704:   /* Memory original stack depth */
                   3705:   org_stack_depth = stack_depth;
                   3706: 
                   3707:   /* If the type is variable-size, we first create its space (we ASSUME
                   3708:      it CAN'T be static).  We do this regardless of whether there's an
                   3709:      initializer assignment or not. */
                   3710: 
                   3711:   if (TREE_CODE (DECL_SIZE (decl)) != INTEGER_CST)
                   3712:     bc_expand_variable_local_init (decl);
                   3713: 
                   3714:   /* Expand initializer assignment */
                   3715:   if (DECL_INITIAL (decl) == error_mark_node)
                   3716:     {
                   3717:       enum tree_code code = TREE_CODE (TREE_TYPE (decl));
                   3718: 
                   3719:       if (code == INTEGER_TYPE || code == REAL_TYPE || code == ENUMERAL_TYPE
                   3720:          || code == POINTER_TYPE)
                   3721: 
                   3722:        expand_assignment (TREE_TYPE (decl), decl, 0, 0);
                   3723:     }
                   3724:   else if (DECL_INITIAL (decl))
                   3725:     expand_assignment (TREE_TYPE (decl), decl, 0, 0);
                   3726: 
                   3727:   /* Restore stack depth */
                   3728:   if (org_stack_depth > stack_depth)
                   3729:     abort ();
                   3730: 
                   3731:   bc_adjust_stack (stack_depth - org_stack_depth);
                   3732: }
                   3733:  
                   3734: 
                   3735: /* CLEANUP is an expression to be executed at exit from this binding contour;
                   3736:    for example, in C++, it might call the destructor for this variable.
                   3737: 
                   3738:    If CLEANUP contains any SAVE_EXPRs, then you must preevaluate them
                   3739:    either before or after calling `expand_decl' but before compiling
                   3740:    any subsequent expressions.  This is because CLEANUP may be expanded
                   3741:    more than once, on different branches of execution.
                   3742:    For the same reason, CLEANUP may not contain a CALL_EXPR
                   3743:    except as its topmost node--else `preexpand_calls' would get confused.
                   3744: 
                   3745:    If CLEANUP is nonzero and DECL is zero, we record a cleanup
                   3746:    that is not associated with any particular variable.   */
                   3747: 
                   3748: int
                   3749: expand_decl_cleanup (decl, cleanup)
                   3750:      tree decl, cleanup;
                   3751: {
                   3752:   struct nesting *thisblock = block_stack;
                   3753: 
                   3754:   /* Error if we are not in any block.  */
                   3755:   if (thisblock == 0)
                   3756:     return 0;
                   3757: 
                   3758:   /* Record the cleanup if there is one.  */
                   3759: 
                   3760:   if (cleanup != 0)
                   3761:     {
                   3762:       thisblock->data.block.cleanups
                   3763:        = temp_tree_cons (decl, cleanup, thisblock->data.block.cleanups);
                   3764:       /* If this block has a cleanup, it belongs in stack_block_stack.  */
                   3765:       stack_block_stack = thisblock;
                   3766:     }
                   3767:   return 1;
                   3768: }
                   3769: 
                   3770: /* DECL is an anonymous union.  CLEANUP is a cleanup for DECL.
                   3771:    DECL_ELTS is the list of elements that belong to DECL's type.
                   3772:    In each, the TREE_VALUE is a VAR_DECL, and the TREE_PURPOSE a cleanup.  */
                   3773: 
                   3774: void
                   3775: expand_anon_union_decl (decl, cleanup, decl_elts)
                   3776:      tree decl, cleanup, decl_elts;
                   3777: {
                   3778:   struct nesting *thisblock = block_stack;
                   3779:   rtx x;
                   3780: 
                   3781:   expand_decl (decl, cleanup);
                   3782:   x = DECL_RTL (decl);
                   3783: 
                   3784:   while (decl_elts)
                   3785:     {
                   3786:       tree decl_elt = TREE_VALUE (decl_elts);
                   3787:       tree cleanup_elt = TREE_PURPOSE (decl_elts);
                   3788:       enum machine_mode mode = TYPE_MODE (TREE_TYPE (decl_elt));
                   3789: 
                   3790:       /* (SUBREG (MEM ...)) at RTL generation time is invalid, so we
                   3791:          instead create a new MEM rtx with the proper mode.  */
                   3792:       if (GET_CODE (x) == MEM)
                   3793:        {
                   3794:          if (mode == GET_MODE (x))
                   3795:            DECL_RTL (decl_elt) = x;
                   3796:          else
                   3797:            {
                   3798:              DECL_RTL (decl_elt) = gen_rtx (MEM, mode, copy_rtx (XEXP (x, 0)));
                   3799:              MEM_IN_STRUCT_P (DECL_RTL (decl_elt)) = MEM_IN_STRUCT_P (x);
                   3800:              RTX_UNCHANGING_P (DECL_RTL (decl_elt)) = RTX_UNCHANGING_P (x);
                   3801:            }
                   3802:        }
                   3803:       else if (GET_CODE (x) == REG)
                   3804:        {
                   3805:          if (mode == GET_MODE (x))
                   3806:            DECL_RTL (decl_elt) = x;
                   3807:          else
                   3808:            DECL_RTL (decl_elt) = gen_rtx (SUBREG, mode, x, 0);
                   3809:        }
                   3810:       else
                   3811:        abort ();
                   3812: 
                   3813:       /* Record the cleanup if there is one.  */
                   3814: 
                   3815:       if (cleanup != 0)
                   3816:        thisblock->data.block.cleanups
                   3817:          = temp_tree_cons (decl_elt, cleanup_elt,
                   3818:                            thisblock->data.block.cleanups);
                   3819: 
                   3820:       decl_elts = TREE_CHAIN (decl_elts);
                   3821:     }
                   3822: }
                   3823: 
                   3824: /* Expand a list of cleanups LIST.
                   3825:    Elements may be expressions or may be nested lists.
                   3826: 
                   3827:    If DONT_DO is nonnull, then any list-element
                   3828:    whose TREE_PURPOSE matches DONT_DO is omitted.
                   3829:    This is sometimes used to avoid a cleanup associated with
                   3830:    a value that is being returned out of the scope.  */
                   3831: 
                   3832: static void
                   3833: expand_cleanups (list, dont_do)
                   3834:      tree list;
                   3835:      tree dont_do;
                   3836: {
                   3837:   tree tail;
                   3838:   for (tail = list; tail; tail = TREE_CHAIN (tail))
                   3839:     if (dont_do == 0 || TREE_PURPOSE (tail) != dont_do)
                   3840:       {
                   3841:        if (TREE_CODE (TREE_VALUE (tail)) == TREE_LIST)
                   3842:          expand_cleanups (TREE_VALUE (tail), dont_do);
                   3843:        else
                   3844:          {
                   3845:            /* Cleanups may be run multiple times.  For example,
                   3846:               when exiting a binding contour, we expand the
                   3847:               cleanups associated with that contour.  When a goto
                   3848:               within that binding contour has a target outside that
                   3849:               contour, it will expand all cleanups from its scope to
                   3850:               the target.  Though the cleanups are expanded multiple
                   3851:               times, the control paths are non-overlapping so the
                   3852:               cleanups will not be executed twice.  */
                   3853:            expand_expr (TREE_VALUE (tail), const0_rtx, VOIDmode, 0);
                   3854:            free_temp_slots ();
                   3855:          }
                   3856:       }
                   3857: }
                   3858: 
                   3859: /* Move all cleanups from the current block_stack
                   3860:    to the containing block_stack, where they are assumed to
                   3861:    have been created.  If anything can cause a temporary to
                   3862:    be created, but not expanded for more than one level of
                   3863:    block_stacks, then this code will have to change.  */
                   3864: 
                   3865: void
                   3866: move_cleanups_up ()
                   3867: {
                   3868:   struct nesting *block = block_stack;
                   3869:   struct nesting *outer = block->next;
                   3870: 
                   3871:   outer->data.block.cleanups
                   3872:     = chainon (block->data.block.cleanups,
                   3873:               outer->data.block.cleanups);
                   3874:   block->data.block.cleanups = 0;
                   3875: }
                   3876: 
                   3877: tree
                   3878: last_cleanup_this_contour ()
                   3879: {
                   3880:   if (block_stack == 0)
                   3881:     return 0;
                   3882: 
                   3883:   return block_stack->data.block.cleanups;
                   3884: }
                   3885: 
                   3886: /* Return 1 if there are any pending cleanups at this point.
                   3887:    If THIS_CONTOUR is nonzero, check the current contour as well.
                   3888:    Otherwise, look only at the contours that enclose this one.  */
                   3889: 
                   3890: int
                   3891: any_pending_cleanups (this_contour)
                   3892:      int this_contour;
                   3893: {
                   3894:   struct nesting *block;
                   3895: 
                   3896:   if (block_stack == 0)
                   3897:     return 0;
                   3898: 
                   3899:   if (this_contour && block_stack->data.block.cleanups != NULL)
                   3900:     return 1;
                   3901:   if (block_stack->data.block.cleanups == 0
                   3902:       && (block_stack->data.block.outer_cleanups == 0
                   3903: #if 0
                   3904:          || block_stack->data.block.outer_cleanups == empty_cleanup_list
                   3905: #endif
                   3906:          ))
                   3907:     return 0;
                   3908: 
                   3909:   for (block = block_stack->next; block; block = block->next)
                   3910:     if (block->data.block.cleanups != 0)
                   3911:       return 1;
                   3912: 
                   3913:   return 0;
                   3914: }
                   3915: 
                   3916: /* Enter a case (Pascal) or switch (C) statement.
                   3917:    Push a block onto case_stack and nesting_stack
                   3918:    to accumulate the case-labels that are seen
                   3919:    and to record the labels generated for the statement.
                   3920: 
                   3921:    EXIT_FLAG is nonzero if `exit_something' should exit this case stmt.
                   3922:    Otherwise, this construct is transparent for `exit_something'.
                   3923: 
                   3924:    EXPR is the index-expression to be dispatched on.
                   3925:    TYPE is its nominal type.  We could simply convert EXPR to this type,
                   3926:    but instead we take short cuts.  */
                   3927: 
                   3928: void
                   3929: expand_start_case (exit_flag, expr, type, printname)
                   3930:      int exit_flag;
                   3931:      tree expr;
                   3932:      tree type;
                   3933:      char *printname;
                   3934: {
                   3935:   register struct nesting *thiscase = ALLOC_NESTING ();
                   3936: 
                   3937:   /* Make an entry on case_stack for the case we are entering.  */
                   3938: 
                   3939:   thiscase->next = case_stack;
                   3940:   thiscase->all = nesting_stack;
                   3941:   thiscase->depth = ++nesting_depth;
                   3942:   thiscase->exit_label = exit_flag ? gen_label_rtx () : 0;
                   3943:   thiscase->data.case_stmt.case_list = 0;
                   3944:   thiscase->data.case_stmt.index_expr = expr;
                   3945:   thiscase->data.case_stmt.nominal_type = type;
                   3946:   thiscase->data.case_stmt.default_label = 0;
                   3947:   thiscase->data.case_stmt.num_ranges = 0;
                   3948:   thiscase->data.case_stmt.printname = printname;
                   3949:   thiscase->data.case_stmt.seenlabel = 0;
                   3950:   case_stack = thiscase;
                   3951:   nesting_stack = thiscase;
                   3952: 
                   3953:   if (output_bytecode)
                   3954:     {
                   3955:       bc_expand_start_case (thiscase, expr, type, printname);
                   3956:       return;
                   3957:     }
                   3958: 
                   3959:   do_pending_stack_adjust ();
                   3960: 
                   3961:   /* Make sure case_stmt.start points to something that won't
                   3962:      need any transformation before expand_end_case.  */
                   3963:   if (GET_CODE (get_last_insn ()) != NOTE)
                   3964:     emit_note (NULL_PTR, NOTE_INSN_DELETED);
                   3965: 
                   3966:   thiscase->data.case_stmt.start = get_last_insn ();
                   3967: }
                   3968: 
                   3969: 
                   3970: /* Enter a case statement. It is assumed that the caller has pushed
                   3971:    the current context onto the case stack. */
                   3972: void
                   3973: bc_expand_start_case (thiscase, expr, type, printname)
                   3974:      struct nesting *thiscase;
                   3975:      tree expr;
                   3976:      tree type;
                   3977:      char *printname;
                   3978: {
                   3979:   bc_expand_expr (expr);
                   3980:   bc_expand_conversion (TREE_TYPE (expr), type);
                   3981: 
                   3982:   /* For cases, the skip is a place we jump to that's emitted after
                   3983:      the size of the jump table is known.  */
                   3984: 
                   3985:   thiscase->data.case_stmt.skip_label = gen_label_rtx ();
                   3986:   bc_emit_bytecode (jump);
                   3987:   bc_emit_bytecode_labelref (BYTECODE_BC_LABEL (thiscase->data.case_stmt.skip_label));
                   3988: 
                   3989: #ifdef DEBUG_PRINT_CODE
                   3990:   fputc ('\n', stderr);
                   3991: #endif
                   3992: }
                   3993: 
                   3994: 
                   3995: /* Start a "dummy case statement" within which case labels are invalid
                   3996:    and are not connected to any larger real case statement.
                   3997:    This can be used if you don't want to let a case statement jump
                   3998:    into the middle of certain kinds of constructs.  */
                   3999: 
                   4000: void
                   4001: expand_start_case_dummy ()
                   4002: {
                   4003:   register struct nesting *thiscase = ALLOC_NESTING ();
                   4004: 
                   4005:   /* Make an entry on case_stack for the dummy.  */
                   4006: 
                   4007:   thiscase->next = case_stack;
                   4008:   thiscase->all = nesting_stack;
                   4009:   thiscase->depth = ++nesting_depth;
                   4010:   thiscase->exit_label = 0;
                   4011:   thiscase->data.case_stmt.case_list = 0;
                   4012:   thiscase->data.case_stmt.start = 0;
                   4013:   thiscase->data.case_stmt.nominal_type = 0;
                   4014:   thiscase->data.case_stmt.default_label = 0;
                   4015:   thiscase->data.case_stmt.num_ranges = 0;
                   4016:   case_stack = thiscase;
                   4017:   nesting_stack = thiscase;
                   4018: }
                   4019: 
                   4020: /* End a dummy case statement.  */
                   4021: 
                   4022: void
                   4023: expand_end_case_dummy ()
                   4024: {
                   4025:   POPSTACK (case_stack);
                   4026: }
                   4027: 
                   4028: /* Return the data type of the index-expression
                   4029:    of the innermost case statement, or null if none.  */
                   4030: 
                   4031: tree
                   4032: case_index_expr_type ()
                   4033: {
                   4034:   if (case_stack)
                   4035:     return TREE_TYPE (case_stack->data.case_stmt.index_expr);
                   4036:   return 0;
                   4037: }
                   4038: 
                   4039: /* Accumulate one case or default label inside a case or switch statement.
                   4040:    VALUE is the value of the case (a null pointer, for a default label).
                   4041:    The function CONVERTER, when applied to arguments T and V,
                   4042:    converts the value V to the type T.
                   4043: 
                   4044:    If not currently inside a case or switch statement, return 1 and do
                   4045:    nothing.  The caller will print a language-specific error message.
                   4046:    If VALUE is a duplicate or overlaps, return 2 and do nothing
                   4047:    except store the (first) duplicate node in *DUPLICATE.
                   4048:    If VALUE is out of range, return 3 and do nothing.
                   4049:    If we are jumping into the scope of a cleaup or var-sized array, return 5.
                   4050:    Return 0 on success.
                   4051: 
                   4052:    Extended to handle range statements.  */
                   4053: 
                   4054: int
                   4055: pushcase (value, converter, label, duplicate)
                   4056:      register tree value;
                   4057:      tree (*converter) PROTO((tree, tree));
                   4058:      register tree label;
                   4059:      tree *duplicate;
                   4060: {
                   4061:   register struct case_node **l;
                   4062:   register struct case_node *n;
                   4063:   tree index_type;
                   4064:   tree nominal_type;
                   4065: 
                   4066:   if (output_bytecode)
                   4067:     return bc_pushcase (value, label);
                   4068: 
                   4069:   /* Fail if not inside a real case statement.  */
                   4070:   if (! (case_stack && case_stack->data.case_stmt.start))
                   4071:     return 1;
                   4072: 
                   4073:   if (stack_block_stack
                   4074:       && stack_block_stack->depth > case_stack->depth)
                   4075:     return 5;
                   4076: 
                   4077:   index_type = TREE_TYPE (case_stack->data.case_stmt.index_expr);
                   4078:   nominal_type = case_stack->data.case_stmt.nominal_type;
                   4079: 
                   4080:   /* If the index is erroneous, avoid more problems: pretend to succeed.  */
                   4081:   if (index_type == error_mark_node)
                   4082:     return 0;
                   4083: 
                   4084:   /* Convert VALUE to the type in which the comparisons are nominally done.  */
                   4085:   if (value != 0)
                   4086:     value = (*converter) (nominal_type, value);
                   4087: 
                   4088:   /* If this is the first label, warn if any insns have been emitted.  */
                   4089:   if (case_stack->data.case_stmt.seenlabel == 0)
                   4090:     {
                   4091:       rtx insn;
                   4092:       for (insn = case_stack->data.case_stmt.start;
                   4093:           insn;
                   4094:           insn = NEXT_INSN (insn))
                   4095:        {
                   4096:          if (GET_CODE (insn) == CODE_LABEL)
                   4097:            break;
                   4098:          if (GET_CODE (insn) != NOTE
                   4099:              && (GET_CODE (insn) != INSN || GET_CODE (PATTERN (insn)) != USE))
                   4100:            {
                   4101:              warning ("unreachable code at beginning of %s",
                   4102:                       case_stack->data.case_stmt.printname);
                   4103:              break;
                   4104:            }
                   4105:        }
                   4106:     }
                   4107:   case_stack->data.case_stmt.seenlabel = 1;
                   4108: 
                   4109:   /* Fail if this value is out of range for the actual type of the index
                   4110:      (which may be narrower than NOMINAL_TYPE).  */
                   4111:   if (value != 0 && ! int_fits_type_p (value, index_type))
                   4112:     return 3;
                   4113: 
                   4114:   /* Fail if this is a duplicate or overlaps another entry.  */
                   4115:   if (value == 0)
                   4116:     {
                   4117:       if (case_stack->data.case_stmt.default_label != 0)
                   4118:        {
                   4119:          *duplicate = case_stack->data.case_stmt.default_label;
                   4120:          return 2;
                   4121:        }
                   4122:       case_stack->data.case_stmt.default_label = label;
                   4123:     }
                   4124:   else
                   4125:     {
                   4126:       /* Find the elt in the chain before which to insert the new value,
                   4127:         to keep the chain sorted in increasing order.
                   4128:         But report an error if this element is a duplicate.  */
                   4129:       for (l = &case_stack->data.case_stmt.case_list;
                   4130:           /* Keep going past elements distinctly less than VALUE.  */
                   4131:           *l != 0 && tree_int_cst_lt ((*l)->high, value);
                   4132:           l = &(*l)->right)
                   4133:        ;
                   4134:       if (*l)
                   4135:        {
                   4136:          /* Element we will insert before must be distinctly greater;
                   4137:             overlap means error.  */
                   4138:          if (! tree_int_cst_lt (value, (*l)->low))
                   4139:            {
                   4140:              *duplicate = (*l)->code_label;
                   4141:              return 2;
                   4142:            }
                   4143:        }
                   4144: 
                   4145:       /* Add this label to the chain, and succeed.
                   4146:         Copy VALUE so it is on temporary rather than momentary
                   4147:         obstack and will thus survive till the end of the case statement.  */
                   4148:       n = (struct case_node *) oballoc (sizeof (struct case_node));
                   4149:       n->left = 0;
                   4150:       n->right = *l;
                   4151:       n->high = n->low = copy_node (value);
                   4152:       n->code_label = label;
                   4153:       *l = n;
                   4154:     }
                   4155: 
                   4156:   expand_label (label);
                   4157:   return 0;
                   4158: }
                   4159: 
                   4160: /* Like pushcase but this case applies to all values
                   4161:    between VALUE1 and VALUE2 (inclusive).
                   4162:    The return value is the same as that of pushcase
                   4163:    but there is one additional error code:
                   4164:    4 means the specified range was empty.  */
                   4165: 
                   4166: int
                   4167: pushcase_range (value1, value2, converter, label, duplicate)
                   4168:      register tree value1, value2;
                   4169:      tree (*converter) PROTO((tree, tree));
                   4170:      register tree label;
                   4171:      tree *duplicate;
                   4172: {
                   4173:   register struct case_node **l;
                   4174:   register struct case_node *n;
                   4175:   tree index_type;
                   4176:   tree nominal_type;
                   4177: 
                   4178:   /* Fail if not inside a real case statement.  */
                   4179:   if (! (case_stack && case_stack->data.case_stmt.start))
                   4180:     return 1;
                   4181: 
                   4182:   if (stack_block_stack
                   4183:       && stack_block_stack->depth > case_stack->depth)
                   4184:     return 5;
                   4185: 
                   4186:   index_type = TREE_TYPE (case_stack->data.case_stmt.index_expr);
                   4187:   nominal_type = case_stack->data.case_stmt.nominal_type;
                   4188: 
                   4189:   /* If the index is erroneous, avoid more problems: pretend to succeed.  */
                   4190:   if (index_type == error_mark_node)
                   4191:     return 0;
                   4192: 
                   4193:   /* If this is the first label, warn if any insns have been emitted.  */
                   4194:   if (case_stack->data.case_stmt.seenlabel == 0)
                   4195:     {
                   4196:       rtx insn;
                   4197:       for (insn = case_stack->data.case_stmt.start;
                   4198:           insn;
                   4199:           insn = NEXT_INSN (insn))
                   4200:        {
                   4201:          if (GET_CODE (insn) == CODE_LABEL)
                   4202:            break;
                   4203:          if (GET_CODE (insn) != NOTE
                   4204:              && (GET_CODE (insn) != INSN || GET_CODE (PATTERN (insn)) != USE))
                   4205:            {
                   4206:              warning ("unreachable code at beginning of %s",
                   4207:                       case_stack->data.case_stmt.printname);
                   4208:              break;
                   4209:            }
                   4210:        }
                   4211:     }
                   4212:   case_stack->data.case_stmt.seenlabel = 1;
                   4213: 
                   4214:   /* Convert VALUEs to type in which the comparisons are nominally done.  */
                   4215:   if (value1 == 0)  /* Negative infinity. */
                   4216:     value1 = TYPE_MIN_VALUE(index_type);
                   4217:   value1 = (*converter) (nominal_type, value1);
                   4218: 
                   4219:   if (value2 == 0)  /* Positive infinity. */
                   4220:     value2 = TYPE_MAX_VALUE(index_type);
                   4221:   value2 = (*converter) (nominal_type, value2);
                   4222: 
                   4223:   /* Fail if these values are out of range.  */
                   4224:   if (! int_fits_type_p (value1, index_type))
                   4225:     return 3;
                   4226: 
                   4227:   if (! int_fits_type_p (value2, index_type))
                   4228:     return 3;
                   4229: 
                   4230:   /* Fail if the range is empty.  */
                   4231:   if (tree_int_cst_lt (value2, value1))
                   4232:     return 4;
                   4233: 
                   4234:   /* If the bounds are equal, turn this into the one-value case.  */
                   4235:   if (tree_int_cst_equal (value1, value2))
                   4236:     return pushcase (value1, converter, label, duplicate);
                   4237: 
                   4238:   /* Find the elt in the chain before which to insert the new value,
                   4239:      to keep the chain sorted in increasing order.
                   4240:      But report an error if this element is a duplicate.  */
                   4241:   for (l = &case_stack->data.case_stmt.case_list;
                   4242:        /* Keep going past elements distinctly less than this range.  */
                   4243:        *l != 0 && tree_int_cst_lt ((*l)->high, value1);
                   4244:        l = &(*l)->right)
                   4245:     ;
                   4246:   if (*l)
                   4247:     {
                   4248:       /* Element we will insert before must be distinctly greater;
                   4249:         overlap means error.  */
                   4250:       if (! tree_int_cst_lt (value2, (*l)->low))
                   4251:        {
                   4252:          *duplicate = (*l)->code_label;
                   4253:          return 2;
                   4254:        }
                   4255:     }
                   4256: 
                   4257:   /* Add this label to the chain, and succeed.
                   4258:      Copy VALUE1, VALUE2 so they are on temporary rather than momentary
                   4259:      obstack and will thus survive till the end of the case statement.  */
                   4260: 
                   4261:   n = (struct case_node *) oballoc (sizeof (struct case_node));
                   4262:   n->left = 0;
                   4263:   n->right = *l;
                   4264:   n->low = copy_node (value1);
                   4265:   n->high = copy_node (value2);
                   4266:   n->code_label = label;
                   4267:   *l = n;
                   4268: 
                   4269:   expand_label (label);
                   4270: 
                   4271:   case_stack->data.case_stmt.num_ranges++;
                   4272: 
                   4273:   return 0;
                   4274: }
                   4275: 
                   4276: 
                   4277: /* Accumulate one case or default label; VALUE is the value of the
                   4278:    case, or nil for a default label.  If not currently inside a case,
                   4279:    return 1 and do nothing.  If VALUE is a duplicate or overlaps, return
                   4280:    2 and do nothing.  If VALUE is out of range, return 3 and do nothing.
                   4281:    Return 0 on success.  This function is a leftover from the earlier
                   4282:    bytecode compiler, which was based on gcc 1.37.  It should be
                   4283:    merged into pushcase. */
                   4284: 
                   4285: int
                   4286: bc_pushcase (value, label)
                   4287:      tree value;
                   4288:      tree label;
                   4289: {
                   4290:   struct nesting *thiscase = case_stack;
                   4291:   struct case_node *case_label, *new_label;
                   4292: 
                   4293:   if (! thiscase)
                   4294:     return 1;
                   4295: 
                   4296:   /* Fail if duplicate, overlap, or out of type range.  */
                   4297:   if (value)
                   4298:     {
                   4299:       value = convert (thiscase->data.case_stmt.nominal_type, value);
                   4300:       if (! int_fits_type_p (value, thiscase->data.case_stmt.nominal_type))
                   4301:        return 3;
                   4302: 
                   4303:       for (case_label = thiscase->data.case_stmt.case_list;
                   4304:           case_label->left; case_label = case_label->left)
                   4305:        if (! tree_int_cst_lt (case_label->left->high, value))
                   4306:          break;
                   4307: 
                   4308:       if (case_label != thiscase->data.case_stmt.case_list
                   4309:          && ! tree_int_cst_lt (case_label->high, value)
                   4310:          || case_label->left && ! tree_int_cst_lt (value, case_label->left->low))
                   4311:        return 2;
                   4312: 
                   4313:       new_label = (struct case_node *) oballoc (sizeof (struct case_node));
                   4314:       new_label->low = new_label->high = copy_node (value);
                   4315:       new_label->code_label = label;
                   4316:       new_label->left = case_label->left;
                   4317: 
                   4318:       case_label->left = new_label;
                   4319:       thiscase->data.case_stmt.num_ranges++;
                   4320:     }
                   4321:   else
                   4322:     {
                   4323:       if (thiscase->data.case_stmt.default_label)
                   4324:        return 2;
                   4325:       thiscase->data.case_stmt.default_label = label;
                   4326:     }
                   4327: 
                   4328:   expand_label (label);
                   4329:   return 0;
                   4330: }
                   4331: 
                   4332: /* Called when the index of a switch statement is an enumerated type
                   4333:    and there is no default label.
                   4334: 
                   4335:    Checks that all enumeration literals are covered by the case
                   4336:    expressions of a switch.  Also, warn if there are any extra
                   4337:    switch cases that are *not* elements of the enumerated type.
                   4338: 
                   4339:    If all enumeration literals were covered by the case expressions,
                   4340:    turn one of the expressions into the default expression since it should
                   4341:    not be possible to fall through such a switch.  */
                   4342: 
                   4343: void
                   4344: check_for_full_enumeration_handling (type)
                   4345:      tree type;
                   4346: {
                   4347:   register struct case_node *n;
                   4348:   register struct case_node **l;
                   4349:   register tree chain;
                   4350:   int all_values = 1;
                   4351: 
                   4352:   if (output_bytecode)
                   4353:     {
                   4354:       bc_check_for_full_enumeration_handling (type);
                   4355:       return;
                   4356:     }
                   4357: 
                   4358:   /* The time complexity of this loop is currently O(N * M), with
                   4359:      N being the number of members in the enumerated type, and
                   4360:      M being the number of case expressions in the switch. */
                   4361: 
                   4362:   for (chain = TYPE_VALUES (type);
                   4363:        chain;
                   4364:        chain = TREE_CHAIN (chain))
                   4365:     {
                   4366:       /* Find a match between enumeral and case expression, if possible.
                   4367:         Quit looking when we've gone too far (since case expressions
                   4368:         are kept sorted in ascending order).  Warn about enumerators not
                   4369:         handled in the switch statement case expression list. */
                   4370: 
                   4371:       for (n = case_stack->data.case_stmt.case_list;
                   4372:           n && tree_int_cst_lt (n->high, TREE_VALUE (chain));
                   4373:           n = n->right)
                   4374:        ;
                   4375: 
                   4376:       if (!n || tree_int_cst_lt (TREE_VALUE (chain), n->low))
                   4377:        {
                   4378:          if (warn_switch)
                   4379:            warning ("enumeration value `%s' not handled in switch",
                   4380:                     IDENTIFIER_POINTER (TREE_PURPOSE (chain)));
                   4381:          all_values = 0;
                   4382:        }
                   4383:     }
                   4384: 
                   4385:   /* Now we go the other way around; we warn if there are case
                   4386:      expressions that don't correspond to enumerators.  This can
                   4387:      occur since C and C++ don't enforce type-checking of
                   4388:      assignments to enumeration variables. */
                   4389: 
                   4390:   if (warn_switch)
                   4391:     for (n = case_stack->data.case_stmt.case_list; n; n = n->right)
                   4392:       {
                   4393:        for (chain = TYPE_VALUES (type);
                   4394:             chain && !tree_int_cst_equal (n->low, TREE_VALUE (chain));
                   4395:             chain = TREE_CHAIN (chain))
                   4396:          ;
                   4397: 
                   4398:        if (!chain)
                   4399:          {
                   4400:            if (TYPE_NAME (type) == 0)
                   4401:              warning ("case value `%d' not in enumerated type",
                   4402:                       TREE_INT_CST_LOW (n->low));
                   4403:            else
                   4404:              warning ("case value `%d' not in enumerated type `%s'",
                   4405:                       TREE_INT_CST_LOW (n->low),
                   4406:                       IDENTIFIER_POINTER ((TREE_CODE (TYPE_NAME (type))
                   4407:                                            == IDENTIFIER_NODE)
                   4408:                                           ? TYPE_NAME (type)
                   4409:                                           : DECL_NAME (TYPE_NAME (type))));
                   4410:          }
                   4411:        if (!tree_int_cst_equal (n->low, n->high))
                   4412:          {
                   4413:            for (chain = TYPE_VALUES (type);
                   4414:                 chain && !tree_int_cst_equal (n->high, TREE_VALUE (chain));
                   4415:                 chain = TREE_CHAIN (chain))
                   4416:              ;
                   4417: 
                   4418:            if (!chain)
                   4419:              {
                   4420:                if (TYPE_NAME (type) == 0)
                   4421:                  warning ("case value `%d' not in enumerated type",
                   4422:                           TREE_INT_CST_LOW (n->high));
                   4423:                else
                   4424:                  warning ("case value `%d' not in enumerated type `%s'",
                   4425:                           TREE_INT_CST_LOW (n->high),
                   4426:                           IDENTIFIER_POINTER ((TREE_CODE (TYPE_NAME (type))
                   4427:                                                == IDENTIFIER_NODE)
                   4428:                                               ? TYPE_NAME (type)
                   4429:                                               : DECL_NAME (TYPE_NAME (type))));
                   4430:              }
                   4431:          }
                   4432:       }
                   4433: 
                   4434: #if 0
                   4435:   /* ??? This optimization is disabled because it causes valid programs to
                   4436:      fail.  ANSI C does not guarantee that an expression with enum type
                   4437:      will have a value that is the same as one of the enumation literals.  */
                   4438: 
                   4439:   /* If all values were found as case labels, make one of them the default
                   4440:      label.  Thus, this switch will never fall through.  We arbitrarily pick
                   4441:      the last one to make the default since this is likely the most
                   4442:      efficient choice.  */
                   4443: 
                   4444:   if (all_values)
                   4445:     {
                   4446:       for (l = &case_stack->data.case_stmt.case_list;
                   4447:           (*l)->right != 0;
                   4448:           l = &(*l)->right)
                   4449:        ;
                   4450: 
                   4451:       case_stack->data.case_stmt.default_label = (*l)->code_label;
                   4452:       *l = 0;
                   4453:     }
                   4454: #endif /* 0 */
                   4455: }
                   4456: 
                   4457: 
                   4458: /* Check that all enumeration literals are covered by the case
                   4459:    expressions of a switch.  Also warn if there are any cases
                   4460:    that are not elements of the enumerated type.  */
                   4461: void
                   4462: bc_check_for_full_enumeration_handling (type)
                   4463:      tree type;
                   4464: {
                   4465:   struct nesting *thiscase = case_stack;
                   4466:   struct case_node *c;
                   4467:   tree e;
                   4468: 
                   4469:   /* Check for enums not handled.  */
                   4470:   for (e = TYPE_VALUES (type); e; e = TREE_CHAIN (e))
                   4471:     {
                   4472:       for (c = thiscase->data.case_stmt.case_list->left;
                   4473:           c && tree_int_cst_lt (c->high, TREE_VALUE (e));
                   4474:           c = c->left)
                   4475:        ;
                   4476:       if (! (c && tree_int_cst_equal (c->low, TREE_VALUE (e))))
                   4477:        warning ("enumerated value `%s' not handled in switch",
                   4478:                 IDENTIFIER_POINTER (TREE_PURPOSE (e)));
                   4479:     }
                   4480: 
                   4481:   /* Check for cases not in the enumeration.  */
                   4482:   for (c = thiscase->data.case_stmt.case_list->left; c; c = c->left)
                   4483:     {
                   4484:       for (e = TYPE_VALUES (type);
                   4485:           e && !tree_int_cst_equal (c->low, TREE_VALUE (e));
                   4486:           e = TREE_CHAIN (e))
                   4487:        ;
                   4488:       if (! e)
                   4489:        warning ("case value `%d' not in enumerated type `%s'",
                   4490:                 TREE_INT_CST_LOW (c->low),
                   4491:                 IDENTIFIER_POINTER (TREE_CODE (TYPE_NAME (type)) == IDENTIFIER_NODE
                   4492:                                     ? TYPE_NAME (type)
                   4493:                                     : DECL_NAME (TYPE_NAME (type))));
                   4494:     }
                   4495: }
                   4496: 
                   4497: /* Terminate a case (Pascal) or switch (C) statement
                   4498:    in which ORIG_INDEX is the expression to be tested.
                   4499:    Generate the code to test it and jump to the right place.  */
                   4500: 
                   4501: void
                   4502: expand_end_case (orig_index)
                   4503:      tree orig_index;
                   4504: {
                   4505:   tree minval, maxval, range, orig_minval;
                   4506:   rtx default_label = 0;
                   4507:   register struct case_node *n;
                   4508:   int count;
                   4509:   rtx index;
                   4510:   rtx table_label;
                   4511:   int ncases;
                   4512:   rtx *labelvec;
                   4513:   register int i;
                   4514:   rtx before_case;
                   4515:   register struct nesting *thiscase = case_stack;
                   4516:   tree index_expr;
                   4517:   int unsignedp;
                   4518: 
                   4519:   if (output_bytecode)
                   4520:     {
                   4521:       bc_expand_end_case (orig_index);
                   4522:       return;
                   4523:     }
                   4524: 
                   4525:   table_label = gen_label_rtx ();
                   4526:   index_expr = thiscase->data.case_stmt.index_expr;
                   4527:   unsignedp = TREE_UNSIGNED (TREE_TYPE (index_expr));
                   4528: 
                   4529:   do_pending_stack_adjust ();
                   4530: 
                   4531:   /* An ERROR_MARK occurs for various reasons including invalid data type.  */
                   4532:   if (TREE_TYPE (index_expr) != error_mark_node)
                   4533:     {
                   4534:       /* If switch expression was an enumerated type, check that all
                   4535:         enumeration literals are covered by the cases.
                   4536:         No sense trying this if there's a default case, however.  */
                   4537: 
                   4538:       if (!thiscase->data.case_stmt.default_label
                   4539:          && TREE_CODE (TREE_TYPE (orig_index)) == ENUMERAL_TYPE
                   4540:          && TREE_CODE (index_expr) != INTEGER_CST)
                   4541:        check_for_full_enumeration_handling (TREE_TYPE (orig_index));
                   4542: 
                   4543:       /* If this is the first label, warn if any insns have been emitted.  */
                   4544:       if (thiscase->data.case_stmt.seenlabel == 0)
                   4545:        {
                   4546:          rtx insn;
                   4547:          for (insn = get_last_insn ();
                   4548:               insn != case_stack->data.case_stmt.start;
                   4549:               insn = PREV_INSN (insn))
                   4550:            if (GET_CODE (insn) != NOTE
                   4551:                && (GET_CODE (insn) != INSN || GET_CODE (PATTERN (insn))!= USE))
                   4552:              {
                   4553:                warning ("unreachable code at beginning of %s",
                   4554:                         case_stack->data.case_stmt.printname);
                   4555:                break;
                   4556:              }
                   4557:        }
                   4558: 
                   4559:       /* If we don't have a default-label, create one here,
                   4560:         after the body of the switch.  */
                   4561:       if (thiscase->data.case_stmt.default_label == 0)
                   4562:        {
                   4563:          thiscase->data.case_stmt.default_label
                   4564:            = build_decl (LABEL_DECL, NULL_TREE, NULL_TREE);
                   4565:          expand_label (thiscase->data.case_stmt.default_label);
                   4566:        }
                   4567:       default_label = label_rtx (thiscase->data.case_stmt.default_label);
                   4568: 
                   4569:       before_case = get_last_insn ();
                   4570: 
                   4571:       /* Simplify the case-list before we count it.  */
                   4572:       group_case_nodes (thiscase->data.case_stmt.case_list);
                   4573: 
                   4574:       /* Get upper and lower bounds of case values.
                   4575:         Also convert all the case values to the index expr's data type.  */
                   4576: 
                   4577:       count = 0;
                   4578:       for (n = thiscase->data.case_stmt.case_list; n; n = n->right)
                   4579:        {
                   4580:          /* Check low and high label values are integers.  */
                   4581:          if (TREE_CODE (n->low) != INTEGER_CST)
                   4582:            abort ();
                   4583:          if (TREE_CODE (n->high) != INTEGER_CST)
                   4584:            abort ();
                   4585: 
                   4586:          n->low = convert (TREE_TYPE (index_expr), n->low);
                   4587:          n->high = convert (TREE_TYPE (index_expr), n->high);
                   4588: 
                   4589:          /* Count the elements and track the largest and smallest
                   4590:             of them (treating them as signed even if they are not).  */
                   4591:          if (count++ == 0)
                   4592:            {
                   4593:              minval = n->low;
                   4594:              maxval = n->high;
                   4595:            }
                   4596:          else
                   4597:            {
                   4598:              if (INT_CST_LT (n->low, minval))
                   4599:                minval = n->low;
                   4600:              if (INT_CST_LT (maxval, n->high))
                   4601:                maxval = n->high;
                   4602:            }
                   4603:          /* A range counts double, since it requires two compares.  */
                   4604:          if (! tree_int_cst_equal (n->low, n->high))
                   4605:            count++;
                   4606:        }
                   4607: 
                   4608:       orig_minval = minval;
                   4609: 
                   4610:       /* Compute span of values.  */
                   4611:       if (count != 0)
                   4612:        range = fold (build (MINUS_EXPR, TREE_TYPE (index_expr),
                   4613:                             maxval, minval));
                   4614: 
                   4615:       if (count == 0 || TREE_CODE (TREE_TYPE (index_expr)) == ERROR_MARK)
                   4616:        {
                   4617:          expand_expr (index_expr, const0_rtx, VOIDmode, 0);
                   4618:          emit_queue ();
                   4619:          emit_jump (default_label);
                   4620:        }
                   4621: 
                   4622:       /* If range of values is much bigger than number of values,
                   4623:         make a sequence of conditional branches instead of a dispatch.
                   4624:         If the switch-index is a constant, do it this way
                   4625:         because we can optimize it.  */
                   4626: 
                   4627: #ifndef CASE_VALUES_THRESHOLD
                   4628: #ifdef HAVE_casesi
                   4629: #define CASE_VALUES_THRESHOLD (HAVE_casesi ? 4 : 5)
                   4630: #else
                   4631:       /* If machine does not have a case insn that compares the
                   4632:         bounds, this means extra overhead for dispatch tables
                   4633:         which raises the threshold for using them.  */
                   4634: #define CASE_VALUES_THRESHOLD 5
                   4635: #endif /* HAVE_casesi */
                   4636: #endif /* CASE_VALUES_THRESHOLD */
                   4637: 
                   4638:       else if (TREE_INT_CST_HIGH (range) != 0
                   4639:               || count < CASE_VALUES_THRESHOLD
                   4640:               || ((unsigned HOST_WIDE_INT) (TREE_INT_CST_LOW (range))
                   4641:                   > 10 * count)
                   4642:               || TREE_CODE (index_expr) == INTEGER_CST
                   4643:               /* These will reduce to a constant.  */
                   4644:               || (TREE_CODE (index_expr) == CALL_EXPR
                   4645:                   && TREE_CODE (TREE_OPERAND (index_expr, 0)) == ADDR_EXPR
                   4646:                   && TREE_CODE (TREE_OPERAND (TREE_OPERAND (index_expr, 0), 0)) == FUNCTION_DECL
                   4647:                   && DECL_FUNCTION_CODE (TREE_OPERAND (TREE_OPERAND (index_expr, 0), 0)) == BUILT_IN_CLASSIFY_TYPE)
                   4648:               || (TREE_CODE (index_expr) == COMPOUND_EXPR
                   4649:                   && TREE_CODE (TREE_OPERAND (index_expr, 1)) == INTEGER_CST))
                   4650:        {
                   4651:          index = expand_expr (index_expr, NULL_RTX, VOIDmode, 0);
                   4652: 
                   4653:          /* If the index is a short or char that we do not have
                   4654:             an insn to handle comparisons directly, convert it to
                   4655:             a full integer now, rather than letting each comparison
                   4656:             generate the conversion.  */
                   4657: 
                   4658:          if (GET_MODE_CLASS (GET_MODE (index)) == MODE_INT
                   4659:              && (cmp_optab->handlers[(int) GET_MODE(index)].insn_code
                   4660:                  == CODE_FOR_nothing))
                   4661:            {
                   4662:              enum machine_mode wider_mode;
                   4663:              for (wider_mode = GET_MODE (index); wider_mode != VOIDmode;
                   4664:                   wider_mode = GET_MODE_WIDER_MODE (wider_mode))
                   4665:                if (cmp_optab->handlers[(int) wider_mode].insn_code
                   4666:                    != CODE_FOR_nothing)
                   4667:                  {
                   4668:                    index = convert_to_mode (wider_mode, index, unsignedp);
                   4669:                    break;
                   4670:                  }
                   4671:            }
                   4672: 
                   4673:          emit_queue ();
                   4674:          do_pending_stack_adjust ();
                   4675: 
                   4676:          index = protect_from_queue (index, 0);
                   4677:          if (GET_CODE (index) == MEM)
                   4678:            index = copy_to_reg (index);
                   4679:          if (GET_CODE (index) == CONST_INT
                   4680:              || TREE_CODE (index_expr) == INTEGER_CST)
                   4681:            {
                   4682:              /* Make a tree node with the proper constant value
                   4683:                 if we don't already have one.  */
                   4684:              if (TREE_CODE (index_expr) != INTEGER_CST)
                   4685:                {
                   4686:                  index_expr
                   4687:                    = build_int_2 (INTVAL (index),
                   4688:                                   !unsignedp && INTVAL (index) >= 0 ? 0 : -1);
                   4689:                  index_expr = convert (TREE_TYPE (index_expr), index_expr);
                   4690:                }
                   4691: 
                   4692:              /* For constant index expressions we need only
                   4693:                 issue a unconditional branch to the appropriate
                   4694:                 target code.  The job of removing any unreachable
                   4695:                 code is left to the optimisation phase if the
                   4696:                 "-O" option is specified.  */
                   4697:              for (n = thiscase->data.case_stmt.case_list;
                   4698:                   n;
                   4699:                   n = n->right)
                   4700:                {
                   4701:                  if (! tree_int_cst_lt (index_expr, n->low)
                   4702:                      && ! tree_int_cst_lt (n->high, index_expr))
                   4703:                    break;
                   4704:                }
                   4705:              if (n)
                   4706:                emit_jump (label_rtx (n->code_label));
                   4707:              else
                   4708:                emit_jump (default_label);
                   4709:            }
                   4710:          else
                   4711:            {
                   4712:              /* If the index expression is not constant we generate
                   4713:                 a binary decision tree to select the appropriate
                   4714:                 target code.  This is done as follows:
                   4715: 
                   4716:                 The list of cases is rearranged into a binary tree,
                   4717:                 nearly optimal assuming equal probability for each case.
                   4718: 
                   4719:                 The tree is transformed into RTL, eliminating
                   4720:                 redundant test conditions at the same time.
                   4721: 
                   4722:                 If program flow could reach the end of the
                   4723:                 decision tree an unconditional jump to the
                   4724:                 default code is emitted.  */
                   4725: 
                   4726:              use_cost_table
                   4727:                = (TREE_CODE (TREE_TYPE (orig_index)) != ENUMERAL_TYPE
                   4728:                   && estimate_case_costs (thiscase->data.case_stmt.case_list));
                   4729:              balance_case_nodes (&thiscase->data.case_stmt.case_list, 
                   4730:                                  NULL_PTR);
                   4731:              emit_case_nodes (index, thiscase->data.case_stmt.case_list,
                   4732:                               default_label, TREE_TYPE (index_expr));
                   4733:              emit_jump_if_reachable (default_label);
                   4734:            }
                   4735:        }
                   4736:       else
                   4737:        {
                   4738:          int win = 0;
                   4739: #ifdef HAVE_casesi
                   4740:          if (HAVE_casesi)
                   4741:            {
                   4742:              enum machine_mode index_mode = SImode;
                   4743:              int index_bits = GET_MODE_BITSIZE (index_mode);
                   4744: 
                   4745:              /* Convert the index to SImode.  */
                   4746:              if (GET_MODE_BITSIZE (TYPE_MODE (TREE_TYPE (index_expr)))
                   4747:                  > GET_MODE_BITSIZE (index_mode))
                   4748:                {
                   4749:                  enum machine_mode omode = TYPE_MODE (TREE_TYPE (index_expr));
                   4750:                  rtx rangertx = expand_expr (range, NULL_RTX, VOIDmode, 0);
                   4751: 
                   4752:                  /* We must handle the endpoints in the original mode.  */
                   4753:                  index_expr = build (MINUS_EXPR, TREE_TYPE (index_expr),
                   4754:                                      index_expr, minval);
                   4755:                  minval = integer_zero_node;
                   4756:                  index = expand_expr (index_expr, NULL_RTX, VOIDmode, 0);
                   4757:                  emit_cmp_insn (rangertx, index, LTU, NULL_RTX, omode, 1, 0);
                   4758:                  emit_jump_insn (gen_bltu (default_label));
                   4759:                  /* Now we can safely truncate.  */
                   4760:                  index = convert_to_mode (index_mode, index, 0);
                   4761:                }
                   4762:              else
                   4763:                {
                   4764:                  if (TYPE_MODE (TREE_TYPE (index_expr)) != index_mode)
                   4765:                    index_expr = convert (type_for_size (index_bits, 0),
                   4766:                                          index_expr);
                   4767:                  index = expand_expr (index_expr, NULL_RTX, VOIDmode, 0);
                   4768:                }
                   4769:              emit_queue ();
                   4770:              index = protect_from_queue (index, 0);
                   4771:              do_pending_stack_adjust ();
                   4772: 
                   4773:              emit_jump_insn (gen_casesi (index, expand_expr (minval, NULL_RTX,
                   4774:                                                              VOIDmode, 0),
                   4775:                                          expand_expr (range, NULL_RTX,
                   4776:                                                       VOIDmode, 0),
                   4777:                                          table_label, default_label));
                   4778:              win = 1;
                   4779:            }
                   4780: #endif
                   4781: #ifdef HAVE_tablejump
                   4782:          if (! win && HAVE_tablejump)
                   4783:            {
                   4784:              index_expr = convert (thiscase->data.case_stmt.nominal_type,
                   4785:                                    fold (build (MINUS_EXPR,
                   4786:                                                 TREE_TYPE (index_expr),
                   4787:                                                 index_expr, minval)));
                   4788:              index = expand_expr (index_expr, NULL_RTX, VOIDmode, 0);
                   4789:              emit_queue ();
                   4790:              index = protect_from_queue (index, 0);
                   4791:              do_pending_stack_adjust ();
                   4792: 
                   4793:              do_tablejump (index, TYPE_MODE (TREE_TYPE (index_expr)),
                   4794:                            expand_expr (range, NULL_RTX, VOIDmode, 0),
                   4795:                            table_label, default_label);
                   4796:              win = 1;
                   4797:            }
                   4798: #endif
                   4799:          if (! win)
                   4800:            abort ();
                   4801: 
                   4802:          /* Get table of labels to jump to, in order of case index.  */
                   4803: 
                   4804:          ncases = TREE_INT_CST_LOW (range) + 1;
                   4805:          labelvec = (rtx *) alloca (ncases * sizeof (rtx));
                   4806:          bzero (labelvec, ncases * sizeof (rtx));
                   4807: 
                   4808:          for (n = thiscase->data.case_stmt.case_list; n; n = n->right)
                   4809:            {
                   4810:              register HOST_WIDE_INT i
                   4811:                = TREE_INT_CST_LOW (n->low) - TREE_INT_CST_LOW (orig_minval);
                   4812: 
                   4813:              while (1)
                   4814:                {
                   4815:                  labelvec[i]
                   4816:                    = gen_rtx (LABEL_REF, Pmode, label_rtx (n->code_label));
                   4817:                  if (i + TREE_INT_CST_LOW (orig_minval)
                   4818:                      == TREE_INT_CST_LOW (n->high))
                   4819:                    break;
                   4820:                  i++;
                   4821:                }
                   4822:            }
                   4823: 
                   4824:          /* Fill in the gaps with the default.  */
                   4825:          for (i = 0; i < ncases; i++)
                   4826:            if (labelvec[i] == 0)
                   4827:              labelvec[i] = gen_rtx (LABEL_REF, Pmode, default_label);
                   4828: 
                   4829:          /* Output the table */
                   4830:          emit_label (table_label);
                   4831: 
                   4832:          /* This would be a lot nicer if CASE_VECTOR_PC_RELATIVE
                   4833:             were an expression, instead of an #ifdef/#ifndef.  */
                   4834:          if (
                   4835: #ifdef CASE_VECTOR_PC_RELATIVE
                   4836:              1 ||
                   4837: #endif
                   4838: #ifdef MACHO_PURE
                   4839:              MACHOPIC_PURE
                   4840: #else
                   4841:              flag_pic
                   4842: #endif
                   4843:              )
                   4844:            emit_jump_insn (gen_rtx (ADDR_DIFF_VEC, CASE_VECTOR_MODE,
                   4845:                                     gen_rtx (LABEL_REF, Pmode, table_label),
                   4846:                                     gen_rtvec_v (ncases, labelvec)));
                   4847:          else
                   4848:            emit_jump_insn (gen_rtx (ADDR_VEC, CASE_VECTOR_MODE,
                   4849:                                     gen_rtvec_v (ncases, labelvec)));
                   4850: 
                   4851:          /* If the case insn drops through the table,
                   4852:             after the table we must jump to the default-label.
                   4853:             Otherwise record no drop-through after the table.  */
                   4854: #ifdef CASE_DROPS_THROUGH
                   4855:          emit_jump (default_label);
                   4856: #else
                   4857:          emit_barrier ();
                   4858: #endif
                   4859:        }
                   4860: 
                   4861:       before_case = squeeze_notes (NEXT_INSN (before_case), get_last_insn ());
                   4862:       reorder_insns (before_case, get_last_insn (),
                   4863:                     thiscase->data.case_stmt.start);
                   4864:     }
                   4865:   if (thiscase->exit_label)
                   4866:     emit_label (thiscase->exit_label);
                   4867: 
                   4868:   POPSTACK (case_stack);
                   4869: 
                   4870:   free_temp_slots ();
                   4871: }
                   4872: 
                   4873: 
                   4874: /* Terminate a case statement.  EXPR is the original index
                   4875:    expression.  */
                   4876: void
                   4877: bc_expand_end_case (expr)
                   4878:      tree expr;
                   4879: {
                   4880:   struct nesting *thiscase = case_stack;
                   4881:   enum bytecode_opcode opcode;
                   4882:   struct bc_label *jump_label;
                   4883:   struct case_node *c;
                   4884: 
                   4885:   bc_emit_bytecode (jump);
                   4886:   bc_emit_bytecode_labelref (BYTECODE_BC_LABEL (thiscase->exit_label));
                   4887: 
                   4888: #ifdef DEBUG_PRINT_CODE
                   4889:   fputc ('\n', stderr);
                   4890: #endif
                   4891: 
                   4892:   /* Now that the size of the jump table is known, emit the actual
                   4893:      indexed jump instruction.  */
                   4894:   bc_emit_bytecode_labeldef (BYTECODE_BC_LABEL (thiscase->data.case_stmt.skip_label));
                   4895: 
                   4896:   opcode = TYPE_MODE (thiscase->data.case_stmt.nominal_type) == SImode
                   4897:     ? TREE_UNSIGNED (thiscase->data.case_stmt.nominal_type) ? caseSU : caseSI
                   4898:       : TREE_UNSIGNED (thiscase->data.case_stmt.nominal_type) ? caseDU : caseDI;
                   4899: 
                   4900:   bc_emit_bytecode (opcode);
                   4901: 
                   4902:   /* Now emit the case instructions literal arguments, in order.
                   4903:      In addition to the value on the stack, it uses:
                   4904:      1.  The address of the jump table.
                   4905:      2.  The size of the jump table.
                   4906:      3.  The default label.  */
                   4907: 
                   4908:   jump_label = bc_get_bytecode_label ();
                   4909:   bc_emit_bytecode_labelref (jump_label);
                   4910:   bc_emit_bytecode_const ((char *) &thiscase->data.case_stmt.num_ranges,
                   4911:                          sizeof thiscase->data.case_stmt.num_ranges);
                   4912: 
                   4913:   if (thiscase->data.case_stmt.default_label)
                   4914:     bc_emit_bytecode_labelref (BYTECODE_BC_LABEL (DECL_RTL (thiscase->data.case_stmt.default_label)));
                   4915:   else
                   4916:     bc_emit_bytecode_labelref (BYTECODE_BC_LABEL (thiscase->exit_label));
                   4917: 
                   4918:   /* Output the jump table.  */
                   4919: 
                   4920:   bc_align_bytecode (3 /* PTR_ALIGN */);
                   4921:   bc_emit_bytecode_labeldef (jump_label);
                   4922: 
                   4923:   if (TYPE_MODE (thiscase->data.case_stmt.nominal_type) == SImode)
                   4924:     for (c = thiscase->data.case_stmt.case_list->left; c; c = c->left)
                   4925:       {
                   4926:        opcode = TREE_INT_CST_LOW (c->low);
                   4927:        bc_emit_bytecode_const ((char *) &opcode, sizeof opcode);
                   4928: 
                   4929:        opcode = TREE_INT_CST_LOW (c->high);
                   4930:        bc_emit_bytecode_const ((char *) &opcode, sizeof opcode);
                   4931: 
                   4932:        bc_emit_bytecode_labelref (BYTECODE_BC_LABEL (DECL_RTL (c->code_label)));
                   4933:       }
                   4934:   else
                   4935:     if (TYPE_MODE (thiscase->data.case_stmt.nominal_type) == DImode)
                   4936:       for (c = thiscase->data.case_stmt.case_list->left; c; c = c->left)
                   4937:        {
                   4938:          bc_emit_bytecode_DI_const (c->low);
                   4939:          bc_emit_bytecode_DI_const (c->high);
                   4940: 
                   4941:          bc_emit_bytecode_labelref (BYTECODE_BC_LABEL (DECL_RTL (c->code_label)));
                   4942:        }
                   4943:     else
                   4944:       /* Bad mode */
                   4945:       abort ();
                   4946: 
                   4947:     
                   4948:   bc_emit_bytecode_labeldef (BYTECODE_BC_LABEL (thiscase->exit_label));
                   4949: 
                   4950:   /* Possibly issue enumeration warnings.  */
                   4951: 
                   4952:   if (!thiscase->data.case_stmt.default_label
                   4953:       && TREE_CODE (TREE_TYPE (expr)) == ENUMERAL_TYPE
                   4954:       && TREE_CODE (expr) != INTEGER_CST
                   4955:       && warn_switch)
                   4956:     check_for_full_enumeration_handling (TREE_TYPE (expr));
                   4957: 
                   4958: 
                   4959: #ifdef DEBUG_PRINT_CODE
                   4960:   fputc ('\n', stderr);
                   4961: #endif
                   4962: 
                   4963:   POPSTACK (case_stack);
                   4964: }
                   4965: 
                   4966: 
                   4967: /* Return unique bytecode ID. */
                   4968: int 
                   4969: bc_new_uid ()
                   4970: {
                   4971:   static int bc_uid = 0;
                   4972: 
                   4973:   return (++bc_uid);
                   4974: }
                   4975: 
                   4976: /* Generate code to jump to LABEL if OP1 and OP2 are equal.  */
                   4977: 
                   4978: static void
                   4979: do_jump_if_equal (op1, op2, label, unsignedp)
                   4980:      rtx op1, op2, label;
                   4981:      int unsignedp;
                   4982: {
                   4983:   if (GET_CODE (op1) == CONST_INT
                   4984:       && GET_CODE (op2) == CONST_INT)
                   4985:     {
                   4986:       if (INTVAL (op1) == INTVAL (op2))
                   4987:        emit_jump (label);
                   4988:     }
                   4989:   else
                   4990:     {
                   4991:       enum machine_mode mode = GET_MODE (op1);
                   4992:       if (mode == VOIDmode)
                   4993:        mode = GET_MODE (op2);
                   4994:       emit_cmp_insn (op1, op2, EQ, NULL_RTX, mode, unsignedp, 0);
                   4995:       emit_jump_insn (gen_beq (label));
                   4996:     }
                   4997: }
                   4998: 
                   4999: /* Not all case values are encountered equally.  This function
                   5000:    uses a heuristic to weight case labels, in cases where that
                   5001:    looks like a reasonable thing to do.
                   5002: 
                   5003:    Right now, all we try to guess is text, and we establish the
                   5004:    following weights:
                   5005: 
                   5006:        chars above space:      16
                   5007:        digits:                 16
                   5008:        default:                12
                   5009:        space, punct:           8
                   5010:        tab:                    4
                   5011:        newline:                2
                   5012:        other "\" chars:        1
                   5013:        remaining chars:        0
                   5014: 
                   5015:    If we find any cases in the switch that are not either -1 or in the range
                   5016:    of valid ASCII characters, or are control characters other than those
                   5017:    commonly used with "\", don't treat this switch scanning text.
                   5018: 
                   5019:    Return 1 if these nodes are suitable for cost estimation, otherwise
                   5020:    return 0.  */
                   5021: 
                   5022: static int
                   5023: estimate_case_costs (node)
                   5024:      case_node_ptr node;
                   5025: {
                   5026:   tree min_ascii = build_int_2 (-1, -1);
                   5027:   tree max_ascii = convert (TREE_TYPE (node->high), build_int_2 (127, 0));
                   5028:   case_node_ptr n;
                   5029:   int i;
                   5030: 
                   5031:   /* If we haven't already made the cost table, make it now.  Note that the
                   5032:      lower bound of the table is -1, not zero.  */
                   5033: 
                   5034:   if (cost_table == NULL)
                   5035:     {
                   5036:       cost_table = ((short *) xmalloc (129 * sizeof (short))) + 1;
                   5037:       bzero (cost_table - 1, 129 * sizeof (short));
                   5038: 
                   5039:       for (i = 0; i < 128; i++)
                   5040:        {
                   5041:          if (isalnum (i))
                   5042:            cost_table[i] = 16;
                   5043:          else if (ispunct (i))
                   5044:            cost_table[i] = 8;
                   5045:          else if (iscntrl (i))
                   5046:            cost_table[i] = -1;
                   5047:        }
                   5048: 
                   5049:       cost_table[' '] = 8;
                   5050:       cost_table['\t'] = 4;
                   5051:       cost_table['\0'] = 4;
                   5052:       cost_table['\n'] = 2;
                   5053:       cost_table['\f'] = 1;
                   5054:       cost_table['\v'] = 1;
                   5055:       cost_table['\b'] = 1;
                   5056:     }
                   5057: 
                   5058:   /* See if all the case expressions look like text.  It is text if the
                   5059:      constant is >= -1 and the highest constant is <= 127.  Do all comparisons
                   5060:      as signed arithmetic since we don't want to ever access cost_table with a
                   5061:      value less than -1.  Also check that none of the constants in a range
                   5062:      are strange control characters.  */
                   5063: 
                   5064:   for (n = node; n; n = n->right)
                   5065:     {
                   5066:       if ((INT_CST_LT (n->low, min_ascii)) || INT_CST_LT (max_ascii, n->high))
                   5067:        return 0;
                   5068: 
                   5069:       for (i = TREE_INT_CST_LOW (n->low); i <= TREE_INT_CST_LOW (n->high); i++)
                   5070:        if (cost_table[i] < 0)
                   5071:          return 0;
                   5072:     }
                   5073: 
                   5074:   /* All interesting values are within the range of interesting
                   5075:      ASCII characters.  */
                   5076:   return 1;
                   5077: }
                   5078: 
                   5079: /* Scan an ordered list of case nodes
                   5080:    combining those with consecutive values or ranges.
                   5081: 
                   5082:    Eg. three separate entries 1: 2: 3: become one entry 1..3:  */
                   5083: 
                   5084: static void
                   5085: group_case_nodes (head)
                   5086:      case_node_ptr head;
                   5087: {
                   5088:   case_node_ptr node = head;
                   5089: 
                   5090:   while (node)
                   5091:     {
                   5092:       rtx lb = next_real_insn (label_rtx (node->code_label));
                   5093:       case_node_ptr np = node;
                   5094: 
                   5095:       /* Try to group the successors of NODE with NODE.  */
                   5096:       while (((np = np->right) != 0)
                   5097:             /* Do they jump to the same place?  */
                   5098:             && next_real_insn (label_rtx (np->code_label)) == lb
                   5099:             /* Are their ranges consecutive?  */
                   5100:             && tree_int_cst_equal (np->low,
                   5101:                                    fold (build (PLUS_EXPR,
                   5102:                                                 TREE_TYPE (node->high),
                   5103:                                                 node->high,
                   5104:                                                 integer_one_node)))
                   5105:             /* An overflow is not consecutive.  */
                   5106:             && tree_int_cst_lt (node->high,
                   5107:                                 fold (build (PLUS_EXPR,
                   5108:                                              TREE_TYPE (node->high),
                   5109:                                              node->high,
                   5110:                                              integer_one_node))))
                   5111:        {
                   5112:          node->high = np->high;
                   5113:        }
                   5114:       /* NP is the first node after NODE which can't be grouped with it.
                   5115:         Delete the nodes in between, and move on to that node.  */
                   5116:       node->right = np;
                   5117:       node = np;
                   5118:     }
                   5119: }
                   5120: 
                   5121: /* Take an ordered list of case nodes
                   5122:    and transform them into a near optimal binary tree,
                   5123:    on the assumption that any target code selection value is as
                   5124:    likely as any other.
                   5125: 
                   5126:    The transformation is performed by splitting the ordered
                   5127:    list into two equal sections plus a pivot.  The parts are
                   5128:    then attached to the pivot as left and right branches.  Each
                   5129:    branch is is then transformed recursively.  */
                   5130: 
                   5131: static void
                   5132: balance_case_nodes (head, parent)
                   5133:      case_node_ptr *head;
                   5134:      case_node_ptr parent;
                   5135: {
                   5136:   register case_node_ptr np;
                   5137: 
                   5138:   np = *head;
                   5139:   if (np)
                   5140:     {
                   5141:       int cost = 0;
                   5142:       int i = 0;
                   5143:       int ranges = 0;
                   5144:       register case_node_ptr *npp;
                   5145:       case_node_ptr left;
                   5146: 
                   5147:       /* Count the number of entries on branch.  Also count the ranges.  */
                   5148: 
                   5149:       while (np)
                   5150:        {
                   5151:          if (!tree_int_cst_equal (np->low, np->high))
                   5152:            {
                   5153:              ranges++;
                   5154:              if (use_cost_table)
                   5155:                cost += cost_table[TREE_INT_CST_LOW (np->high)];
                   5156:            }
                   5157: 
                   5158:          if (use_cost_table)
                   5159:            cost += cost_table[TREE_INT_CST_LOW (np->low)];
                   5160: 
                   5161:          i++;
                   5162:          np = np->right;
                   5163:        }
                   5164: 
                   5165:       if (i > 2)
                   5166:        {
                   5167:          /* Split this list if it is long enough for that to help.  */
                   5168:          npp = head;
                   5169:          left = *npp;
                   5170:          if (use_cost_table)
                   5171:            {
                   5172:              /* Find the place in the list that bisects the list's total cost,
                   5173:                 Here I gets half the total cost.  */
                   5174:              int n_moved = 0;
                   5175:              i = (cost + 1) / 2;
                   5176:              while (1)
                   5177:                {
                   5178:                  /* Skip nodes while their cost does not reach that amount.  */
                   5179:                  if (!tree_int_cst_equal ((*npp)->low, (*npp)->high))
                   5180:                    i -= cost_table[TREE_INT_CST_LOW ((*npp)->high)];
                   5181:                  i -= cost_table[TREE_INT_CST_LOW ((*npp)->low)];
                   5182:                  if (i <= 0)
                   5183:                    break;
                   5184:                  npp = &(*npp)->right;
                   5185:                  n_moved += 1;
                   5186:                }
                   5187:              if (n_moved == 0)
                   5188:                {
                   5189:                  /* Leave this branch lopsided, but optimize left-hand
                   5190:                     side and fill in `parent' fields for right-hand side.  */
                   5191:                  np = *head;
                   5192:                  np->parent = parent;
                   5193:                  balance_case_nodes (&np->left, np);
                   5194:                  for (; np->right; np = np->right)
                   5195:                    np->right->parent = np;
                   5196:                  return;
                   5197:                }
                   5198:            }
                   5199:          /* If there are just three nodes, split at the middle one.  */
                   5200:          else if (i == 3)
                   5201:            npp = &(*npp)->right;
                   5202:          else
                   5203:            {
                   5204:              /* Find the place in the list that bisects the list's total cost,
                   5205:                 where ranges count as 2.
                   5206:                 Here I gets half the total cost.  */
                   5207:              i = (i + ranges + 1) / 2;
                   5208:              while (1)
                   5209:                {
                   5210:                  /* Skip nodes while their cost does not reach that amount.  */
                   5211:                  if (!tree_int_cst_equal ((*npp)->low, (*npp)->high))
                   5212:                    i--;
                   5213:                  i--;
                   5214:                  if (i <= 0)
                   5215:                    break;
                   5216:                  npp = &(*npp)->right;
                   5217:                }
                   5218:            }
                   5219:          *head = np = *npp;
                   5220:          *npp = 0;
                   5221:          np->parent = parent;
                   5222:          np->left = left;
                   5223: 
                   5224:          /* Optimize each of the two split parts.  */
                   5225:          balance_case_nodes (&np->left, np);
                   5226:          balance_case_nodes (&np->right, np);
                   5227:        }
                   5228:       else
                   5229:        {
                   5230:          /* Else leave this branch as one level,
                   5231:             but fill in `parent' fields.  */
                   5232:          np = *head;
                   5233:          np->parent = parent;
                   5234:          for (; np->right; np = np->right)
                   5235:            np->right->parent = np;
                   5236:        }
                   5237:     }
                   5238: }
                   5239: 
                   5240: /* Search the parent sections of the case node tree
                   5241:    to see if a test for the lower bound of NODE would be redundant.
                   5242:    INDEX_TYPE is the type of the index expression.
                   5243: 
                   5244:    The instructions to generate the case decision tree are
                   5245:    output in the same order as nodes are processed so it is
                   5246:    known that if a parent node checks the range of the current
                   5247:    node minus one that the current node is bounded at its lower
                   5248:    span.  Thus the test would be redundant.  */
                   5249: 
                   5250: static int
                   5251: node_has_low_bound (node, index_type)
                   5252:      case_node_ptr node;
                   5253:      tree index_type;
                   5254: {
                   5255:   tree low_minus_one;
                   5256:   case_node_ptr pnode;
                   5257: 
                   5258:   /* If the lower bound of this node is the lowest value in the index type,
                   5259:      we need not test it.  */
                   5260: 
                   5261:   if (tree_int_cst_equal (node->low, TYPE_MIN_VALUE (index_type)))
                   5262:     return 1;
                   5263: 
                   5264:   /* If this node has a left branch, the value at the left must be less
                   5265:      than that at this node, so it cannot be bounded at the bottom and
                   5266:      we need not bother testing any further.  */
                   5267: 
                   5268:   if (node->left)
                   5269:     return 0;
                   5270: 
                   5271:   low_minus_one = fold (build (MINUS_EXPR, TREE_TYPE (node->low),
                   5272:                               node->low, integer_one_node));
                   5273: 
                   5274:   /* If the subtraction above overflowed, we can't verify anything.
                   5275:      Otherwise, look for a parent that tests our value - 1.  */
                   5276: 
                   5277:   if (! tree_int_cst_lt (low_minus_one, node->low))
                   5278:     return 0;
                   5279: 
                   5280:   for (pnode = node->parent; pnode; pnode = pnode->parent)
                   5281:     if (tree_int_cst_equal (low_minus_one, pnode->high))
                   5282:       return 1;
                   5283: 
                   5284:   return 0;
                   5285: }
                   5286: 
                   5287: /* Search the parent sections of the case node tree
                   5288:    to see if a test for the upper bound of NODE would be redundant.
                   5289:    INDEX_TYPE is the type of the index expression.
                   5290: 
                   5291:    The instructions to generate the case decision tree are
                   5292:    output in the same order as nodes are processed so it is
                   5293:    known that if a parent node checks the range of the current
                   5294:    node plus one that the current node is bounded at its upper
                   5295:    span.  Thus the test would be redundant.  */
                   5296: 
                   5297: static int
                   5298: node_has_high_bound (node, index_type)
                   5299:      case_node_ptr node;
                   5300:      tree index_type;
                   5301: {
                   5302:   tree high_plus_one;
                   5303:   case_node_ptr pnode;
                   5304: 
                   5305:   /* If the upper bound of this node is the highest value in the type
                   5306:      of the index expression, we need not test against it.  */
                   5307: 
                   5308:   if (tree_int_cst_equal (node->high, TYPE_MAX_VALUE (index_type)))
                   5309:     return 1;
                   5310: 
                   5311:   /* If this node has a right branch, the value at the right must be greater
                   5312:      than that at this node, so it cannot be bounded at the top and
                   5313:      we need not bother testing any further.  */
                   5314: 
                   5315:   if (node->right)
                   5316:     return 0;
                   5317: 
                   5318:   high_plus_one = fold (build (PLUS_EXPR, TREE_TYPE (node->high),
                   5319:                               node->high, integer_one_node));
                   5320: 
                   5321:   /* If the addition above overflowed, we can't verify anything.
                   5322:      Otherwise, look for a parent that tests our value + 1.  */
                   5323: 
                   5324:   if (! tree_int_cst_lt (node->high, high_plus_one))
                   5325:     return 0;
                   5326: 
                   5327:   for (pnode = node->parent; pnode; pnode = pnode->parent)
                   5328:     if (tree_int_cst_equal (high_plus_one, pnode->low))
                   5329:       return 1;
                   5330: 
                   5331:   return 0;
                   5332: }
                   5333: 
                   5334: /* Search the parent sections of the
                   5335:    case node tree to see if both tests for the upper and lower
                   5336:    bounds of NODE would be redundant.  */
                   5337: 
                   5338: static int
                   5339: node_is_bounded (node, index_type)
                   5340:      case_node_ptr node;
                   5341:      tree index_type;
                   5342: {
                   5343:   return (node_has_low_bound (node, index_type)
                   5344:          && node_has_high_bound (node, index_type));
                   5345: }
                   5346: 
                   5347: /*  Emit an unconditional jump to LABEL unless it would be dead code.  */
                   5348: 
                   5349: static void
                   5350: emit_jump_if_reachable (label)
                   5351:      rtx label;
                   5352: {
                   5353:   if (GET_CODE (get_last_insn ()) != BARRIER)
                   5354:     emit_jump (label);
                   5355: }
                   5356: 
                   5357: /* Emit step-by-step code to select a case for the value of INDEX.
                   5358:    The thus generated decision tree follows the form of the
                   5359:    case-node binary tree NODE, whose nodes represent test conditions.
                   5360:    INDEX_TYPE is the type of the index of the switch.
                   5361: 
                   5362:    Care is taken to prune redundant tests from the decision tree
                   5363:    by detecting any boundary conditions already checked by
                   5364:    emitted rtx.  (See node_has_high_bound, node_has_low_bound
                   5365:    and node_is_bounded, above.)
                   5366: 
                   5367:    Where the test conditions can be shown to be redundant we emit
                   5368:    an unconditional jump to the target code.  As a further
                   5369:    optimization, the subordinates of a tree node are examined to
                   5370:    check for bounded nodes.  In this case conditional and/or
                   5371:    unconditional jumps as a result of the boundary check for the
                   5372:    current node are arranged to target the subordinates associated
                   5373:    code for out of bound conditions on the current node node.
                   5374: 
                   5375:    We can assume that when control reaches the code generated here,
                   5376:    the index value has already been compared with the parents
                   5377:    of this node, and determined to be on the same side of each parent
                   5378:    as this node is.  Thus, if this node tests for the value 51,
                   5379:    and a parent tested for 52, we don't need to consider
                   5380:    the possibility of a value greater than 51.  If another parent
                   5381:    tests for the value 50, then this node need not test anything.  */
                   5382: 
                   5383: static void
                   5384: emit_case_nodes (index, node, default_label, index_type)
                   5385:      rtx index;
                   5386:      case_node_ptr node;
                   5387:      rtx default_label;
                   5388:      tree index_type;
                   5389: {
                   5390:   /* If INDEX has an unsigned type, we must make unsigned branches.  */
                   5391:   int unsignedp = TREE_UNSIGNED (index_type);
                   5392:   typedef rtx rtx_function ();
                   5393:   rtx_function *gen_bgt_pat = unsignedp ? gen_bgtu : gen_bgt;
                   5394:   rtx_function *gen_bge_pat = unsignedp ? gen_bgeu : gen_bge;
                   5395:   rtx_function *gen_blt_pat = unsignedp ? gen_bltu : gen_blt;
                   5396:   rtx_function *gen_ble_pat = unsignedp ? gen_bleu : gen_ble;
                   5397:   enum machine_mode mode = GET_MODE (index);
                   5398: 
                   5399:   /* See if our parents have already tested everything for us.
                   5400:      If they have, emit an unconditional jump for this node.  */
                   5401:   if (node_is_bounded (node, index_type))
                   5402:     emit_jump (label_rtx (node->code_label));
                   5403: 
                   5404:   else if (tree_int_cst_equal (node->low, node->high))
                   5405:     {
                   5406:       /* Node is single valued.  First see if the index expression matches
                   5407:         this node and then check our children, if any. */
                   5408: 
                   5409:       do_jump_if_equal (index, expand_expr (node->low, NULL_RTX, VOIDmode, 0),
                   5410:                        label_rtx (node->code_label), unsignedp);
                   5411: 
                   5412:       if (node->right != 0 && node->left != 0)
                   5413:        {
                   5414:          /* This node has children on both sides.
                   5415:             Dispatch to one side or the other
                   5416:             by comparing the index value with this node's value.
                   5417:             If one subtree is bounded, check that one first,
                   5418:             so we can avoid real branches in the tree.  */
                   5419: 
                   5420:          if (node_is_bounded (node->right, index_type))
                   5421:            {
                   5422:              emit_cmp_insn (index, expand_expr (node->high, NULL_RTX,
                   5423:                                                 VOIDmode, 0),
                   5424:                             GT, NULL_RTX, mode, unsignedp, 0);
                   5425: 
                   5426:              emit_jump_insn ((*gen_bgt_pat) (label_rtx (node->right->code_label)));
                   5427:              emit_case_nodes (index, node->left, default_label, index_type);
                   5428:            }
                   5429: 
                   5430:          else if (node_is_bounded (node->left, index_type))
                   5431:            {
                   5432:              emit_cmp_insn (index, expand_expr (node->high, NULL_RTX,
                   5433:                                                 VOIDmode, 0),
                   5434:                             LT, NULL_RTX, mode, unsignedp, 0);
                   5435:              emit_jump_insn ((*gen_blt_pat) (label_rtx (node->left->code_label)));
                   5436:              emit_case_nodes (index, node->right, default_label, index_type);
                   5437:            }
                   5438: 
                   5439:          else
                   5440:            {
                   5441:              /* Neither node is bounded.  First distinguish the two sides;
                   5442:                 then emit the code for one side at a time.  */
                   5443: 
                   5444:              tree test_label
                   5445:                = build_decl (LABEL_DECL, NULL_TREE, NULL_TREE);
                   5446: 
                   5447:              /* See if the value is on the right.  */
                   5448:              emit_cmp_insn (index, expand_expr (node->high, NULL_RTX,
                   5449:                                                 VOIDmode, 0),
                   5450:                             GT, NULL_RTX, mode, unsignedp, 0);
                   5451:              emit_jump_insn ((*gen_bgt_pat) (label_rtx (test_label)));
                   5452: 
                   5453:              /* Value must be on the left.
                   5454:                 Handle the left-hand subtree.  */
                   5455:              emit_case_nodes (index, node->left, default_label, index_type);
                   5456:              /* If left-hand subtree does nothing,
                   5457:                 go to default.  */
                   5458:              emit_jump_if_reachable (default_label);
                   5459: 
                   5460:              /* Code branches here for the right-hand subtree.  */
                   5461:              expand_label (test_label);
                   5462:              emit_case_nodes (index, node->right, default_label, index_type);
                   5463:            }
                   5464:        }
                   5465: 
                   5466:       else if (node->right != 0 && node->left == 0)
                   5467:        {
                   5468:          /* Here we have a right child but no left so we issue conditional
                   5469:             branch to default and process the right child.
                   5470: 
                   5471:             Omit the conditional branch to default if we it avoid only one
                   5472:             right child; it costs too much space to save so little time.  */
                   5473: 
                   5474:          if (node->right->right || node->right->left
                   5475:              || !tree_int_cst_equal (node->right->low, node->right->high))
                   5476:            {
                   5477:              if (!node_has_low_bound (node, index_type))
                   5478:                {
                   5479:                  emit_cmp_insn (index, expand_expr (node->high, NULL_RTX,
                   5480:                                                     VOIDmode, 0),
                   5481:                                 LT, NULL_RTX, mode, unsignedp, 0);
                   5482:                  emit_jump_insn ((*gen_blt_pat) (default_label));
                   5483:                }
                   5484: 
                   5485:              emit_case_nodes (index, node->right, default_label, index_type);
                   5486:            }
                   5487:          else
                   5488:            /* We cannot process node->right normally
                   5489:               since we haven't ruled out the numbers less than
                   5490:               this node's value.  So handle node->right explicitly.  */
                   5491:            do_jump_if_equal (index,
                   5492:                              expand_expr (node->right->low, NULL_RTX,
                   5493:                                           VOIDmode, 0),
                   5494:                              label_rtx (node->right->code_label), unsignedp);
                   5495:        }
                   5496: 
                   5497:       else if (node->right == 0 && node->left != 0)
                   5498:        {
                   5499:          /* Just one subtree, on the left.  */
                   5500: 
                   5501: #if 0 /* The following code and comment were formerly part
                   5502:         of the condition here, but they didn't work
                   5503:         and I don't understand what the idea was.  -- rms.  */
                   5504:          /* If our "most probable entry" is less probable
                   5505:             than the default label, emit a jump to
                   5506:             the default label using condition codes
                   5507:             already lying around.  With no right branch,
                   5508:             a branch-greater-than will get us to the default
                   5509:             label correctly.  */
                   5510:          if (use_cost_table
                   5511:               && cost_table[TREE_INT_CST_LOW (node->high)] < 12)
                   5512:            ;
                   5513: #endif /* 0 */
                   5514:          if (node->left->left || node->left->right
                   5515:              || !tree_int_cst_equal (node->left->low, node->left->high))
                   5516:            {
                   5517:              if (!node_has_high_bound (node, index_type))
                   5518:                {
                   5519:                  emit_cmp_insn (index, expand_expr (node->high, NULL_RTX,
                   5520:                                                     VOIDmode, 0),
                   5521:                                 GT, NULL_RTX, mode, unsignedp, 0);
                   5522:                  emit_jump_insn ((*gen_bgt_pat) (default_label));
                   5523:                }
                   5524: 
                   5525:              emit_case_nodes (index, node->left, default_label, index_type);
                   5526:            }
                   5527:          else
                   5528:            /* We cannot process node->left normally
                   5529:               since we haven't ruled out the numbers less than
                   5530:               this node's value.  So handle node->left explicitly.  */
                   5531:            do_jump_if_equal (index,
                   5532:                              expand_expr (node->left->low, NULL_RTX,
                   5533:                                           VOIDmode, 0),
                   5534:                              label_rtx (node->left->code_label), unsignedp);
                   5535:        }
                   5536:     }
                   5537:   else
                   5538:     {
                   5539:       /* Node is a range.  These cases are very similar to those for a single
                   5540:         value, except that we do not start by testing whether this node
                   5541:         is the one to branch to.  */
                   5542: 
                   5543:       if (node->right != 0 && node->left != 0)
                   5544:        {
                   5545:          /* Node has subtrees on both sides.
                   5546:             If the right-hand subtree is bounded,
                   5547:             test for it first, since we can go straight there.
                   5548:             Otherwise, we need to make a branch in the control structure,
                   5549:             then handle the two subtrees.  */
                   5550:          tree test_label = 0;
                   5551: 
                   5552:          emit_cmp_insn (index, expand_expr (node->high, NULL_RTX,
                   5553:                                             VOIDmode, 0),
                   5554:                         GT, NULL_RTX, mode, unsignedp, 0);
                   5555: 
                   5556:          if (node_is_bounded (node->right, index_type))
                   5557:            /* Right hand node is fully bounded so we can eliminate any
                   5558:               testing and branch directly to the target code.  */
                   5559:            emit_jump_insn ((*gen_bgt_pat) (label_rtx (node->right->code_label)));
                   5560:          else
                   5561:            {
                   5562:              /* Right hand node requires testing.
                   5563:                 Branch to a label where we will handle it later.  */
                   5564: 
                   5565:              test_label = build_decl (LABEL_DECL, NULL_TREE, NULL_TREE);
                   5566:              emit_jump_insn ((*gen_bgt_pat) (label_rtx (test_label)));
                   5567:            }
                   5568: 
                   5569:          /* Value belongs to this node or to the left-hand subtree.  */
                   5570: 
                   5571:          emit_cmp_insn (index, expand_expr (node->low, NULL_RTX, VOIDmode, 0),
                   5572:                         GE, NULL_RTX, mode, unsignedp, 0);
                   5573:          emit_jump_insn ((*gen_bge_pat) (label_rtx (node->code_label)));
                   5574: 
                   5575:          /* Handle the left-hand subtree.  */
                   5576:          emit_case_nodes (index, node->left, default_label, index_type);
                   5577: 
                   5578:          /* If right node had to be handled later, do that now.  */
                   5579: 
                   5580:          if (test_label)
                   5581:            {
                   5582:              /* If the left-hand subtree fell through,
                   5583:                 don't let it fall into the right-hand subtree.  */
                   5584:              emit_jump_if_reachable (default_label);
                   5585: 
                   5586:              expand_label (test_label);
                   5587:              emit_case_nodes (index, node->right, default_label, index_type);
                   5588:            }
                   5589:        }
                   5590: 
                   5591:       else if (node->right != 0 && node->left == 0)
                   5592:        {
                   5593:          /* Deal with values to the left of this node,
                   5594:             if they are possible.  */
                   5595:          if (!node_has_low_bound (node, index_type))
                   5596:            {
                   5597:              emit_cmp_insn (index, expand_expr (node->low, NULL_RTX,
                   5598:                                                 VOIDmode, 0),
                   5599:                             LT, NULL_RTX, mode, unsignedp, 0);
                   5600:              emit_jump_insn ((*gen_blt_pat) (default_label));
                   5601:            }
                   5602: 
                   5603:          /* Value belongs to this node or to the right-hand subtree.  */
                   5604: 
                   5605:          emit_cmp_insn (index, expand_expr (node->high, NULL_RTX,
                   5606:                                             VOIDmode, 0),
                   5607:                         LE, NULL_RTX, mode, unsignedp, 0);
                   5608:          emit_jump_insn ((*gen_ble_pat) (label_rtx (node->code_label)));
                   5609: 
                   5610:          emit_case_nodes (index, node->right, default_label, index_type);
                   5611:        }
                   5612: 
                   5613:       else if (node->right == 0 && node->left != 0)
                   5614:        {
                   5615:          /* Deal with values to the right of this node,
                   5616:             if they are possible.  */
                   5617:          if (!node_has_high_bound (node, index_type))
                   5618:            {
                   5619:              emit_cmp_insn (index, expand_expr (node->high, NULL_RTX,
                   5620:                                                 VOIDmode, 0),
                   5621:                             GT, NULL_RTX, mode, unsignedp, 0);
                   5622:              emit_jump_insn ((*gen_bgt_pat) (default_label));
                   5623:            }
                   5624: 
                   5625:          /* Value belongs to this node or to the left-hand subtree.  */
                   5626: 
                   5627:          emit_cmp_insn (index, expand_expr (node->low, NULL_RTX, VOIDmode, 0),
                   5628:                         GE, NULL_RTX, mode, unsignedp, 0);
                   5629:          emit_jump_insn ((*gen_bge_pat) (label_rtx (node->code_label)));
                   5630: 
                   5631:          emit_case_nodes (index, node->left, default_label, index_type);
                   5632:        }
                   5633: 
                   5634:       else
                   5635:        {
                   5636:          /* Node has no children so we check low and high bounds to remove
                   5637:             redundant tests.  Only one of the bounds can exist,
                   5638:             since otherwise this node is bounded--a case tested already.  */
                   5639: 
                   5640:          if (!node_has_high_bound (node, index_type))
                   5641:            {
                   5642:              emit_cmp_insn (index, expand_expr (node->high, NULL_RTX,
                   5643:                                                 VOIDmode, 0),
                   5644:                             GT, NULL_RTX, mode, unsignedp, 0);
                   5645:              emit_jump_insn ((*gen_bgt_pat) (default_label));
                   5646:            }
                   5647: 
                   5648:          if (!node_has_low_bound (node, index_type))
                   5649:            {
                   5650:              emit_cmp_insn (index, expand_expr (node->low, NULL_RTX,
                   5651:                                                 VOIDmode, 0),
                   5652:                             LT, NULL_RTX, mode, unsignedp, 0);
                   5653:              emit_jump_insn ((*gen_blt_pat) (default_label));
                   5654:            }
                   5655: 
                   5656:          emit_jump (label_rtx (node->code_label));
                   5657:        }
                   5658:     }
                   5659: }
                   5660: 
                   5661: /* These routines are used by the loop unrolling code.  They copy BLOCK trees
                   5662:    so that the debugging info will be correct for the unrolled loop.  */
                   5663: 
                   5664: /* Indexed by block number, contains a pointer to the N'th block node.  */
                   5665: 
                   5666: static tree *block_vector;
                   5667: 
                   5668: void
                   5669: find_loop_tree_blocks ()
                   5670: {
                   5671:   tree block = DECL_INITIAL (current_function_decl);
                   5672: 
                   5673:   /* There first block is for the function body, and does not have
                   5674:      corresponding block notes.  Don't include it in the block vector.  */
                   5675:   block = BLOCK_SUBBLOCKS (block);
                   5676: 
                   5677:   block_vector = identify_blocks (block, get_insns ());
                   5678: }
                   5679: 
                   5680: void
                   5681: unroll_block_trees ()
                   5682: {
                   5683:   tree block = DECL_INITIAL (current_function_decl);
                   5684: 
                   5685:   reorder_blocks (block_vector, block, get_insns ());
                   5686: }
                   5687: 

unix.superglobalmegacorp.com

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