1*38fd1498Szrj /* Tree-dumping functionality for intermediate representation. 2*38fd1498Szrj Copyright (C) 1999-2018 Free Software Foundation, Inc. 3*38fd1498Szrj Written by Mark Mitchell <mark@codesourcery.com> 4*38fd1498Szrj 5*38fd1498Szrj This file is part of GCC. 6*38fd1498Szrj 7*38fd1498Szrj GCC is free software; you can redistribute it and/or modify it under 8*38fd1498Szrj the terms of the GNU General Public License as published by the Free 9*38fd1498Szrj Software Foundation; either version 3, or (at your option) any later 10*38fd1498Szrj version. 11*38fd1498Szrj 12*38fd1498Szrj GCC is distributed in the hope that it will be useful, but WITHOUT ANY 13*38fd1498Szrj WARRANTY; without even the implied warranty of MERCHANTABILITY or 14*38fd1498Szrj FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License 15*38fd1498Szrj for more details. 16*38fd1498Szrj 17*38fd1498Szrj You should have received a copy of the GNU General Public License 18*38fd1498Szrj along with GCC; see the file COPYING3. If not see 19*38fd1498Szrj <http://www.gnu.org/licenses/>. */ 20*38fd1498Szrj 21*38fd1498Szrj #include "config.h" 22*38fd1498Szrj #include "system.h" 23*38fd1498Szrj #include "coretypes.h" 24*38fd1498Szrj #include "tree.h" 25*38fd1498Szrj #include "tree-pretty-print.h" 26*38fd1498Szrj #include "tree-dump.h" 27*38fd1498Szrj #include "langhooks.h" 28*38fd1498Szrj #include "tree-iterator.h" 29*38fd1498Szrj 30*38fd1498Szrj static unsigned int queue (dump_info_p, const_tree, int); 31*38fd1498Szrj static void dump_index (dump_info_p, unsigned int); 32*38fd1498Szrj static void dequeue_and_dump (dump_info_p); 33*38fd1498Szrj static void dump_new_line (dump_info_p); 34*38fd1498Szrj static void dump_maybe_newline (dump_info_p); 35*38fd1498Szrj 36*38fd1498Szrj /* Add T to the end of the queue of nodes to dump. Returns the index 37*38fd1498Szrj assigned to T. */ 38*38fd1498Szrj 39*38fd1498Szrj static unsigned int 40*38fd1498Szrj queue (dump_info_p di, const_tree t, int flags) 41*38fd1498Szrj { 42*38fd1498Szrj dump_queue_p dq; 43*38fd1498Szrj dump_node_info_p dni; 44*38fd1498Szrj unsigned int index; 45*38fd1498Szrj 46*38fd1498Szrj /* Assign the next available index to T. */ 47*38fd1498Szrj index = ++di->index; 48*38fd1498Szrj 49*38fd1498Szrj /* Obtain a new queue node. */ 50*38fd1498Szrj if (di->free_list) 51*38fd1498Szrj { 52*38fd1498Szrj dq = di->free_list; 53*38fd1498Szrj di->free_list = dq->next; 54*38fd1498Szrj } 55*38fd1498Szrj else 56*38fd1498Szrj dq = XNEW (struct dump_queue); 57*38fd1498Szrj 58*38fd1498Szrj /* Create a new entry in the splay-tree. */ 59*38fd1498Szrj dni = XNEW (struct dump_node_info); 60*38fd1498Szrj dni->index = index; 61*38fd1498Szrj dni->binfo_p = ((flags & DUMP_BINFO) != 0); 62*38fd1498Szrj dq->node = splay_tree_insert (di->nodes, (splay_tree_key) t, 63*38fd1498Szrj (splay_tree_value) dni); 64*38fd1498Szrj 65*38fd1498Szrj /* Add it to the end of the queue. */ 66*38fd1498Szrj dq->next = 0; 67*38fd1498Szrj if (!di->queue_end) 68*38fd1498Szrj di->queue = dq; 69*38fd1498Szrj else 70*38fd1498Szrj di->queue_end->next = dq; 71*38fd1498Szrj di->queue_end = dq; 72*38fd1498Szrj 73*38fd1498Szrj /* Return the index. */ 74*38fd1498Szrj return index; 75*38fd1498Szrj } 76*38fd1498Szrj 77*38fd1498Szrj static void 78*38fd1498Szrj dump_index (dump_info_p di, unsigned int index) 79*38fd1498Szrj { 80*38fd1498Szrj fprintf (di->stream, "@%-6u ", index); 81*38fd1498Szrj di->column += 8; 82*38fd1498Szrj } 83*38fd1498Szrj 84*38fd1498Szrj /* If T has not already been output, queue it for subsequent output. 85*38fd1498Szrj FIELD is a string to print before printing the index. Then, the 86*38fd1498Szrj index of T is printed. */ 87*38fd1498Szrj 88*38fd1498Szrj void 89*38fd1498Szrj queue_and_dump_index (dump_info_p di, const char *field, const_tree t, int flags) 90*38fd1498Szrj { 91*38fd1498Szrj unsigned int index; 92*38fd1498Szrj splay_tree_node n; 93*38fd1498Szrj 94*38fd1498Szrj /* If there's no node, just return. This makes for fewer checks in 95*38fd1498Szrj our callers. */ 96*38fd1498Szrj if (!t) 97*38fd1498Szrj return; 98*38fd1498Szrj 99*38fd1498Szrj /* See if we've already queued or dumped this node. */ 100*38fd1498Szrj n = splay_tree_lookup (di->nodes, (splay_tree_key) t); 101*38fd1498Szrj if (n) 102*38fd1498Szrj index = ((dump_node_info_p) n->value)->index; 103*38fd1498Szrj else 104*38fd1498Szrj /* If we haven't, add it to the queue. */ 105*38fd1498Szrj index = queue (di, t, flags); 106*38fd1498Szrj 107*38fd1498Szrj /* Print the index of the node. */ 108*38fd1498Szrj dump_maybe_newline (di); 109*38fd1498Szrj fprintf (di->stream, "%-4s: ", field); 110*38fd1498Szrj di->column += 6; 111*38fd1498Szrj dump_index (di, index); 112*38fd1498Szrj } 113*38fd1498Szrj 114*38fd1498Szrj /* Dump the type of T. */ 115*38fd1498Szrj 116*38fd1498Szrj void 117*38fd1498Szrj queue_and_dump_type (dump_info_p di, const_tree t) 118*38fd1498Szrj { 119*38fd1498Szrj queue_and_dump_index (di, "type", TREE_TYPE (t), DUMP_NONE); 120*38fd1498Szrj } 121*38fd1498Szrj 122*38fd1498Szrj /* Dump column control */ 123*38fd1498Szrj #define SOL_COLUMN 25 /* Start of line column. */ 124*38fd1498Szrj #define EOL_COLUMN 55 /* End of line column. */ 125*38fd1498Szrj #define COLUMN_ALIGNMENT 15 /* Alignment. */ 126*38fd1498Szrj 127*38fd1498Szrj /* Insert a new line in the dump output, and indent to an appropriate 128*38fd1498Szrj place to start printing more fields. */ 129*38fd1498Szrj 130*38fd1498Szrj static void 131*38fd1498Szrj dump_new_line (dump_info_p di) 132*38fd1498Szrj { 133*38fd1498Szrj fprintf (di->stream, "\n%*s", SOL_COLUMN, ""); 134*38fd1498Szrj di->column = SOL_COLUMN; 135*38fd1498Szrj } 136*38fd1498Szrj 137*38fd1498Szrj /* If necessary, insert a new line. */ 138*38fd1498Szrj 139*38fd1498Szrj static void 140*38fd1498Szrj dump_maybe_newline (dump_info_p di) 141*38fd1498Szrj { 142*38fd1498Szrj int extra; 143*38fd1498Szrj 144*38fd1498Szrj /* See if we need a new line. */ 145*38fd1498Szrj if (di->column > EOL_COLUMN) 146*38fd1498Szrj dump_new_line (di); 147*38fd1498Szrj /* See if we need any padding. */ 148*38fd1498Szrj else if ((extra = (di->column - SOL_COLUMN) % COLUMN_ALIGNMENT) != 0) 149*38fd1498Szrj { 150*38fd1498Szrj fprintf (di->stream, "%*s", COLUMN_ALIGNMENT - extra, ""); 151*38fd1498Szrj di->column += COLUMN_ALIGNMENT - extra; 152*38fd1498Szrj } 153*38fd1498Szrj } 154*38fd1498Szrj 155*38fd1498Szrj /* Dump pointer PTR using FIELD to identify it. */ 156*38fd1498Szrj 157*38fd1498Szrj void 158*38fd1498Szrj dump_pointer (dump_info_p di, const char *field, void *ptr) 159*38fd1498Szrj { 160*38fd1498Szrj dump_maybe_newline (di); 161*38fd1498Szrj fprintf (di->stream, "%-4s: %-8" HOST_WIDE_INT_PRINT "x ", field, 162*38fd1498Szrj (unsigned HOST_WIDE_INT) (uintptr_t) ptr); 163*38fd1498Szrj di->column += 15; 164*38fd1498Szrj } 165*38fd1498Szrj 166*38fd1498Szrj /* Dump integer I using FIELD to identify it. */ 167*38fd1498Szrj 168*38fd1498Szrj void 169*38fd1498Szrj dump_int (dump_info_p di, const char *field, int i) 170*38fd1498Szrj { 171*38fd1498Szrj dump_maybe_newline (di); 172*38fd1498Szrj fprintf (di->stream, "%-4s: %-7d ", field, i); 173*38fd1498Szrj di->column += 14; 174*38fd1498Szrj } 175*38fd1498Szrj 176*38fd1498Szrj /* Dump the floating point value R, using FIELD to identify it. */ 177*38fd1498Szrj 178*38fd1498Szrj static void 179*38fd1498Szrj dump_real (dump_info_p di, const char *field, const REAL_VALUE_TYPE *r) 180*38fd1498Szrj { 181*38fd1498Szrj char buf[32]; 182*38fd1498Szrj real_to_decimal (buf, r, sizeof (buf), 0, true); 183*38fd1498Szrj dump_maybe_newline (di); 184*38fd1498Szrj fprintf (di->stream, "%-4s: %s ", field, buf); 185*38fd1498Szrj di->column += strlen (buf) + 7; 186*38fd1498Szrj } 187*38fd1498Szrj 188*38fd1498Szrj /* Dump the fixed-point value F, using FIELD to identify it. */ 189*38fd1498Szrj 190*38fd1498Szrj static void 191*38fd1498Szrj dump_fixed (dump_info_p di, const char *field, const FIXED_VALUE_TYPE *f) 192*38fd1498Szrj { 193*38fd1498Szrj char buf[32]; 194*38fd1498Szrj fixed_to_decimal (buf, f, sizeof (buf)); 195*38fd1498Szrj dump_maybe_newline (di); 196*38fd1498Szrj fprintf (di->stream, "%-4s: %s ", field, buf); 197*38fd1498Szrj di->column += strlen (buf) + 7; 198*38fd1498Szrj } 199*38fd1498Szrj 200*38fd1498Szrj 201*38fd1498Szrj /* Dump the string S. */ 202*38fd1498Szrj 203*38fd1498Szrj void 204*38fd1498Szrj dump_string (dump_info_p di, const char *string) 205*38fd1498Szrj { 206*38fd1498Szrj dump_maybe_newline (di); 207*38fd1498Szrj fprintf (di->stream, "%-13s ", string); 208*38fd1498Szrj if (strlen (string) > 13) 209*38fd1498Szrj di->column += strlen (string) + 1; 210*38fd1498Szrj else 211*38fd1498Szrj di->column += 14; 212*38fd1498Szrj } 213*38fd1498Szrj 214*38fd1498Szrj /* Dump the string field S. */ 215*38fd1498Szrj 216*38fd1498Szrj void 217*38fd1498Szrj dump_string_field (dump_info_p di, const char *field, const char *string) 218*38fd1498Szrj { 219*38fd1498Szrj dump_maybe_newline (di); 220*38fd1498Szrj fprintf (di->stream, "%-4s: %-7s ", field, string); 221*38fd1498Szrj if (strlen (string) > 7) 222*38fd1498Szrj di->column += 6 + strlen (string) + 1; 223*38fd1498Szrj else 224*38fd1498Szrj di->column += 14; 225*38fd1498Szrj } 226*38fd1498Szrj 227*38fd1498Szrj /* Dump the next node in the queue. */ 228*38fd1498Szrj 229*38fd1498Szrj static void 230*38fd1498Szrj dequeue_and_dump (dump_info_p di) 231*38fd1498Szrj { 232*38fd1498Szrj dump_queue_p dq; 233*38fd1498Szrj splay_tree_node stn; 234*38fd1498Szrj dump_node_info_p dni; 235*38fd1498Szrj tree t; 236*38fd1498Szrj unsigned int index; 237*38fd1498Szrj enum tree_code code; 238*38fd1498Szrj enum tree_code_class code_class; 239*38fd1498Szrj const char* code_name; 240*38fd1498Szrj 241*38fd1498Szrj /* Get the next node from the queue. */ 242*38fd1498Szrj dq = di->queue; 243*38fd1498Szrj stn = dq->node; 244*38fd1498Szrj t = (tree) stn->key; 245*38fd1498Szrj dni = (dump_node_info_p) stn->value; 246*38fd1498Szrj index = dni->index; 247*38fd1498Szrj 248*38fd1498Szrj /* Remove the node from the queue, and put it on the free list. */ 249*38fd1498Szrj di->queue = dq->next; 250*38fd1498Szrj if (!di->queue) 251*38fd1498Szrj di->queue_end = 0; 252*38fd1498Szrj dq->next = di->free_list; 253*38fd1498Szrj di->free_list = dq; 254*38fd1498Szrj 255*38fd1498Szrj /* Print the node index. */ 256*38fd1498Szrj dump_index (di, index); 257*38fd1498Szrj /* And the type of node this is. */ 258*38fd1498Szrj if (dni->binfo_p) 259*38fd1498Szrj code_name = "binfo"; 260*38fd1498Szrj else 261*38fd1498Szrj code_name = get_tree_code_name (TREE_CODE (t)); 262*38fd1498Szrj fprintf (di->stream, "%-16s ", code_name); 263*38fd1498Szrj di->column = 25; 264*38fd1498Szrj 265*38fd1498Szrj /* Figure out what kind of node this is. */ 266*38fd1498Szrj code = TREE_CODE (t); 267*38fd1498Szrj code_class = TREE_CODE_CLASS (code); 268*38fd1498Szrj 269*38fd1498Szrj /* Although BINFOs are TREE_VECs, we dump them specially so as to be 270*38fd1498Szrj more informative. */ 271*38fd1498Szrj if (dni->binfo_p) 272*38fd1498Szrj { 273*38fd1498Szrj unsigned ix; 274*38fd1498Szrj tree base; 275*38fd1498Szrj vec<tree, va_gc> *accesses = BINFO_BASE_ACCESSES (t); 276*38fd1498Szrj 277*38fd1498Szrj dump_child ("type", BINFO_TYPE (t)); 278*38fd1498Szrj 279*38fd1498Szrj if (BINFO_VIRTUAL_P (t)) 280*38fd1498Szrj dump_string_field (di, "spec", "virt"); 281*38fd1498Szrj 282*38fd1498Szrj dump_int (di, "bases", BINFO_N_BASE_BINFOS (t)); 283*38fd1498Szrj for (ix = 0; BINFO_BASE_ITERATE (t, ix, base); ix++) 284*38fd1498Szrj { 285*38fd1498Szrj tree access = (accesses ? (*accesses)[ix] : access_public_node); 286*38fd1498Szrj const char *string = NULL; 287*38fd1498Szrj 288*38fd1498Szrj if (access == access_public_node) 289*38fd1498Szrj string = "pub"; 290*38fd1498Szrj else if (access == access_protected_node) 291*38fd1498Szrj string = "prot"; 292*38fd1498Szrj else if (access == access_private_node) 293*38fd1498Szrj string = "priv"; 294*38fd1498Szrj else 295*38fd1498Szrj gcc_unreachable (); 296*38fd1498Szrj 297*38fd1498Szrj dump_string_field (di, "accs", string); 298*38fd1498Szrj queue_and_dump_index (di, "binf", base, DUMP_BINFO); 299*38fd1498Szrj } 300*38fd1498Szrj 301*38fd1498Szrj goto done; 302*38fd1498Szrj } 303*38fd1498Szrj 304*38fd1498Szrj /* We can knock off a bunch of expression nodes in exactly the same 305*38fd1498Szrj way. */ 306*38fd1498Szrj if (IS_EXPR_CODE_CLASS (code_class)) 307*38fd1498Szrj { 308*38fd1498Szrj /* If we're dumping children, dump them now. */ 309*38fd1498Szrj queue_and_dump_type (di, t); 310*38fd1498Szrj 311*38fd1498Szrj switch (code_class) 312*38fd1498Szrj { 313*38fd1498Szrj case tcc_unary: 314*38fd1498Szrj dump_child ("op 0", TREE_OPERAND (t, 0)); 315*38fd1498Szrj break; 316*38fd1498Szrj 317*38fd1498Szrj case tcc_binary: 318*38fd1498Szrj case tcc_comparison: 319*38fd1498Szrj dump_child ("op 0", TREE_OPERAND (t, 0)); 320*38fd1498Szrj dump_child ("op 1", TREE_OPERAND (t, 1)); 321*38fd1498Szrj break; 322*38fd1498Szrj 323*38fd1498Szrj case tcc_expression: 324*38fd1498Szrj case tcc_reference: 325*38fd1498Szrj case tcc_statement: 326*38fd1498Szrj case tcc_vl_exp: 327*38fd1498Szrj /* These nodes are handled explicitly below. */ 328*38fd1498Szrj break; 329*38fd1498Szrj 330*38fd1498Szrj default: 331*38fd1498Szrj gcc_unreachable (); 332*38fd1498Szrj } 333*38fd1498Szrj } 334*38fd1498Szrj else if (DECL_P (t)) 335*38fd1498Szrj { 336*38fd1498Szrj expanded_location xloc; 337*38fd1498Szrj /* All declarations have names. */ 338*38fd1498Szrj if (DECL_NAME (t)) 339*38fd1498Szrj dump_child ("name", DECL_NAME (t)); 340*38fd1498Szrj if (HAS_DECL_ASSEMBLER_NAME_P (t) 341*38fd1498Szrj && DECL_ASSEMBLER_NAME_SET_P (t) 342*38fd1498Szrj && DECL_ASSEMBLER_NAME (t) != DECL_NAME (t)) 343*38fd1498Szrj dump_child ("mngl", DECL_ASSEMBLER_NAME (t)); 344*38fd1498Szrj if (DECL_ABSTRACT_ORIGIN (t)) 345*38fd1498Szrj dump_child ("orig", DECL_ABSTRACT_ORIGIN (t)); 346*38fd1498Szrj /* And types. */ 347*38fd1498Szrj queue_and_dump_type (di, t); 348*38fd1498Szrj dump_child ("scpe", DECL_CONTEXT (t)); 349*38fd1498Szrj /* And a source position. */ 350*38fd1498Szrj xloc = expand_location (DECL_SOURCE_LOCATION (t)); 351*38fd1498Szrj if (xloc.file) 352*38fd1498Szrj { 353*38fd1498Szrj const char *filename = lbasename (xloc.file); 354*38fd1498Szrj 355*38fd1498Szrj dump_maybe_newline (di); 356*38fd1498Szrj fprintf (di->stream, "srcp: %s:%-6d ", filename, 357*38fd1498Szrj xloc.line); 358*38fd1498Szrj di->column += 6 + strlen (filename) + 8; 359*38fd1498Szrj } 360*38fd1498Szrj /* And any declaration can be compiler-generated. */ 361*38fd1498Szrj if (CODE_CONTAINS_STRUCT (TREE_CODE (t), TS_DECL_COMMON) 362*38fd1498Szrj && DECL_ARTIFICIAL (t)) 363*38fd1498Szrj dump_string_field (di, "note", "artificial"); 364*38fd1498Szrj if (DECL_CHAIN (t) && !dump_flag (di, TDF_SLIM, NULL)) 365*38fd1498Szrj dump_child ("chain", DECL_CHAIN (t)); 366*38fd1498Szrj } 367*38fd1498Szrj else if (code_class == tcc_type) 368*38fd1498Szrj { 369*38fd1498Szrj /* All types have qualifiers. */ 370*38fd1498Szrj int quals = lang_hooks.tree_dump.type_quals (t); 371*38fd1498Szrj 372*38fd1498Szrj if (quals != TYPE_UNQUALIFIED) 373*38fd1498Szrj { 374*38fd1498Szrj fprintf (di->stream, "qual: %c%c%c ", 375*38fd1498Szrj (quals & TYPE_QUAL_CONST) ? 'c' : ' ', 376*38fd1498Szrj (quals & TYPE_QUAL_VOLATILE) ? 'v' : ' ', 377*38fd1498Szrj (quals & TYPE_QUAL_RESTRICT) ? 'r' : ' '); 378*38fd1498Szrj di->column += 14; 379*38fd1498Szrj } 380*38fd1498Szrj 381*38fd1498Szrj /* All types have associated declarations. */ 382*38fd1498Szrj dump_child ("name", TYPE_NAME (t)); 383*38fd1498Szrj 384*38fd1498Szrj /* All types have a main variant. */ 385*38fd1498Szrj if (TYPE_MAIN_VARIANT (t) != t) 386*38fd1498Szrj dump_child ("unql", TYPE_MAIN_VARIANT (t)); 387*38fd1498Szrj 388*38fd1498Szrj /* And sizes. */ 389*38fd1498Szrj dump_child ("size", TYPE_SIZE (t)); 390*38fd1498Szrj 391*38fd1498Szrj /* All types have alignments. */ 392*38fd1498Szrj dump_int (di, "algn", TYPE_ALIGN (t)); 393*38fd1498Szrj } 394*38fd1498Szrj else if (code_class == tcc_constant) 395*38fd1498Szrj /* All constants can have types. */ 396*38fd1498Szrj queue_and_dump_type (di, t); 397*38fd1498Szrj 398*38fd1498Szrj /* Give the language-specific code a chance to print something. If 399*38fd1498Szrj it's completely taken care of things, don't bother printing 400*38fd1498Szrj anything more ourselves. */ 401*38fd1498Szrj if (lang_hooks.tree_dump.dump_tree (di, t)) 402*38fd1498Szrj goto done; 403*38fd1498Szrj 404*38fd1498Szrj /* Now handle the various kinds of nodes. */ 405*38fd1498Szrj switch (code) 406*38fd1498Szrj { 407*38fd1498Szrj int i; 408*38fd1498Szrj 409*38fd1498Szrj case IDENTIFIER_NODE: 410*38fd1498Szrj dump_string_field (di, "strg", IDENTIFIER_POINTER (t)); 411*38fd1498Szrj dump_int (di, "lngt", IDENTIFIER_LENGTH (t)); 412*38fd1498Szrj break; 413*38fd1498Szrj 414*38fd1498Szrj case TREE_LIST: 415*38fd1498Szrj dump_child ("purp", TREE_PURPOSE (t)); 416*38fd1498Szrj dump_child ("valu", TREE_VALUE (t)); 417*38fd1498Szrj dump_child ("chan", TREE_CHAIN (t)); 418*38fd1498Szrj break; 419*38fd1498Szrj 420*38fd1498Szrj case STATEMENT_LIST: 421*38fd1498Szrj { 422*38fd1498Szrj tree_stmt_iterator it; 423*38fd1498Szrj for (i = 0, it = tsi_start (t); !tsi_end_p (it); tsi_next (&it), i++) 424*38fd1498Szrj { 425*38fd1498Szrj char buffer[32]; 426*38fd1498Szrj sprintf (buffer, "%u", i); 427*38fd1498Szrj dump_child (buffer, tsi_stmt (it)); 428*38fd1498Szrj } 429*38fd1498Szrj } 430*38fd1498Szrj break; 431*38fd1498Szrj 432*38fd1498Szrj case TREE_VEC: 433*38fd1498Szrj dump_int (di, "lngt", TREE_VEC_LENGTH (t)); 434*38fd1498Szrj for (i = 0; i < TREE_VEC_LENGTH (t); ++i) 435*38fd1498Szrj { 436*38fd1498Szrj char buffer[32]; 437*38fd1498Szrj sprintf (buffer, "%u", i); 438*38fd1498Szrj dump_child (buffer, TREE_VEC_ELT (t, i)); 439*38fd1498Szrj } 440*38fd1498Szrj break; 441*38fd1498Szrj 442*38fd1498Szrj case INTEGER_TYPE: 443*38fd1498Szrj case ENUMERAL_TYPE: 444*38fd1498Szrj dump_int (di, "prec", TYPE_PRECISION (t)); 445*38fd1498Szrj dump_string_field (di, "sign", TYPE_UNSIGNED (t) ? "unsigned": "signed"); 446*38fd1498Szrj dump_child ("min", TYPE_MIN_VALUE (t)); 447*38fd1498Szrj dump_child ("max", TYPE_MAX_VALUE (t)); 448*38fd1498Szrj 449*38fd1498Szrj if (code == ENUMERAL_TYPE) 450*38fd1498Szrj dump_child ("csts", TYPE_VALUES (t)); 451*38fd1498Szrj break; 452*38fd1498Szrj 453*38fd1498Szrj case REAL_TYPE: 454*38fd1498Szrj dump_int (di, "prec", TYPE_PRECISION (t)); 455*38fd1498Szrj break; 456*38fd1498Szrj 457*38fd1498Szrj case FIXED_POINT_TYPE: 458*38fd1498Szrj dump_int (di, "prec", TYPE_PRECISION (t)); 459*38fd1498Szrj dump_string_field (di, "sign", TYPE_UNSIGNED (t) ? "unsigned": "signed"); 460*38fd1498Szrj dump_string_field (di, "saturating", 461*38fd1498Szrj TYPE_SATURATING (t) ? "saturating": "non-saturating"); 462*38fd1498Szrj break; 463*38fd1498Szrj 464*38fd1498Szrj case POINTER_TYPE: 465*38fd1498Szrj dump_child ("ptd", TREE_TYPE (t)); 466*38fd1498Szrj break; 467*38fd1498Szrj 468*38fd1498Szrj case REFERENCE_TYPE: 469*38fd1498Szrj dump_child ("refd", TREE_TYPE (t)); 470*38fd1498Szrj break; 471*38fd1498Szrj 472*38fd1498Szrj case METHOD_TYPE: 473*38fd1498Szrj dump_child ("clas", TYPE_METHOD_BASETYPE (t)); 474*38fd1498Szrj /* Fall through. */ 475*38fd1498Szrj 476*38fd1498Szrj case FUNCTION_TYPE: 477*38fd1498Szrj dump_child ("retn", TREE_TYPE (t)); 478*38fd1498Szrj dump_child ("prms", TYPE_ARG_TYPES (t)); 479*38fd1498Szrj break; 480*38fd1498Szrj 481*38fd1498Szrj case ARRAY_TYPE: 482*38fd1498Szrj dump_child ("elts", TREE_TYPE (t)); 483*38fd1498Szrj dump_child ("domn", TYPE_DOMAIN (t)); 484*38fd1498Szrj break; 485*38fd1498Szrj 486*38fd1498Szrj case RECORD_TYPE: 487*38fd1498Szrj case UNION_TYPE: 488*38fd1498Szrj if (TREE_CODE (t) == RECORD_TYPE) 489*38fd1498Szrj dump_string_field (di, "tag", "struct"); 490*38fd1498Szrj else 491*38fd1498Szrj dump_string_field (di, "tag", "union"); 492*38fd1498Szrj 493*38fd1498Szrj dump_child ("flds", TYPE_FIELDS (t)); 494*38fd1498Szrj queue_and_dump_index (di, "binf", TYPE_BINFO (t), 495*38fd1498Szrj DUMP_BINFO); 496*38fd1498Szrj break; 497*38fd1498Szrj 498*38fd1498Szrj case CONST_DECL: 499*38fd1498Szrj dump_child ("cnst", DECL_INITIAL (t)); 500*38fd1498Szrj break; 501*38fd1498Szrj 502*38fd1498Szrj case DEBUG_EXPR_DECL: 503*38fd1498Szrj dump_int (di, "-uid", DEBUG_TEMP_UID (t)); 504*38fd1498Szrj /* Fall through. */ 505*38fd1498Szrj 506*38fd1498Szrj case VAR_DECL: 507*38fd1498Szrj case PARM_DECL: 508*38fd1498Szrj case FIELD_DECL: 509*38fd1498Szrj case RESULT_DECL: 510*38fd1498Szrj if (TREE_CODE (t) == PARM_DECL) 511*38fd1498Szrj dump_child ("argt", DECL_ARG_TYPE (t)); 512*38fd1498Szrj else 513*38fd1498Szrj dump_child ("init", DECL_INITIAL (t)); 514*38fd1498Szrj dump_child ("size", DECL_SIZE (t)); 515*38fd1498Szrj dump_int (di, "algn", DECL_ALIGN (t)); 516*38fd1498Szrj 517*38fd1498Szrj if (TREE_CODE (t) == FIELD_DECL) 518*38fd1498Szrj { 519*38fd1498Szrj if (DECL_FIELD_OFFSET (t)) 520*38fd1498Szrj dump_child ("bpos", bit_position (t)); 521*38fd1498Szrj } 522*38fd1498Szrj else if (VAR_P (t) || TREE_CODE (t) == PARM_DECL) 523*38fd1498Szrj { 524*38fd1498Szrj dump_int (di, "used", TREE_USED (t)); 525*38fd1498Szrj if (DECL_REGISTER (t)) 526*38fd1498Szrj dump_string_field (di, "spec", "register"); 527*38fd1498Szrj } 528*38fd1498Szrj break; 529*38fd1498Szrj 530*38fd1498Szrj case FUNCTION_DECL: 531*38fd1498Szrj dump_child ("args", DECL_ARGUMENTS (t)); 532*38fd1498Szrj if (DECL_EXTERNAL (t)) 533*38fd1498Szrj dump_string_field (di, "body", "undefined"); 534*38fd1498Szrj if (TREE_PUBLIC (t)) 535*38fd1498Szrj dump_string_field (di, "link", "extern"); 536*38fd1498Szrj else 537*38fd1498Szrj dump_string_field (di, "link", "static"); 538*38fd1498Szrj if (DECL_SAVED_TREE (t) && !dump_flag (di, TDF_SLIM, t)) 539*38fd1498Szrj dump_child ("body", DECL_SAVED_TREE (t)); 540*38fd1498Szrj break; 541*38fd1498Szrj 542*38fd1498Szrj case INTEGER_CST: 543*38fd1498Szrj fprintf (di->stream, "int: "); 544*38fd1498Szrj print_decs (wi::to_wide (t), di->stream); 545*38fd1498Szrj break; 546*38fd1498Szrj 547*38fd1498Szrj case STRING_CST: 548*38fd1498Szrj fprintf (di->stream, "strg: %-7s ", TREE_STRING_POINTER (t)); 549*38fd1498Szrj dump_int (di, "lngt", TREE_STRING_LENGTH (t)); 550*38fd1498Szrj break; 551*38fd1498Szrj 552*38fd1498Szrj case REAL_CST: 553*38fd1498Szrj dump_real (di, "valu", TREE_REAL_CST_PTR (t)); 554*38fd1498Szrj break; 555*38fd1498Szrj 556*38fd1498Szrj case FIXED_CST: 557*38fd1498Szrj dump_fixed (di, "valu", TREE_FIXED_CST_PTR (t)); 558*38fd1498Szrj break; 559*38fd1498Szrj 560*38fd1498Szrj case TRUTH_NOT_EXPR: 561*38fd1498Szrj case ADDR_EXPR: 562*38fd1498Szrj case INDIRECT_REF: 563*38fd1498Szrj case CLEANUP_POINT_EXPR: 564*38fd1498Szrj case SAVE_EXPR: 565*38fd1498Szrj case REALPART_EXPR: 566*38fd1498Szrj case IMAGPART_EXPR: 567*38fd1498Szrj /* These nodes are unary, but do not have code class `1'. */ 568*38fd1498Szrj dump_child ("op 0", TREE_OPERAND (t, 0)); 569*38fd1498Szrj break; 570*38fd1498Szrj 571*38fd1498Szrj case TRUTH_ANDIF_EXPR: 572*38fd1498Szrj case TRUTH_ORIF_EXPR: 573*38fd1498Szrj case INIT_EXPR: 574*38fd1498Szrj case MODIFY_EXPR: 575*38fd1498Szrj case COMPOUND_EXPR: 576*38fd1498Szrj case PREDECREMENT_EXPR: 577*38fd1498Szrj case PREINCREMENT_EXPR: 578*38fd1498Szrj case POSTDECREMENT_EXPR: 579*38fd1498Szrj case POSTINCREMENT_EXPR: 580*38fd1498Szrj /* These nodes are binary, but do not have code class `2'. */ 581*38fd1498Szrj dump_child ("op 0", TREE_OPERAND (t, 0)); 582*38fd1498Szrj dump_child ("op 1", TREE_OPERAND (t, 1)); 583*38fd1498Szrj break; 584*38fd1498Szrj 585*38fd1498Szrj case COMPONENT_REF: 586*38fd1498Szrj case BIT_FIELD_REF: 587*38fd1498Szrj dump_child ("op 0", TREE_OPERAND (t, 0)); 588*38fd1498Szrj dump_child ("op 1", TREE_OPERAND (t, 1)); 589*38fd1498Szrj dump_child ("op 2", TREE_OPERAND (t, 2)); 590*38fd1498Szrj break; 591*38fd1498Szrj 592*38fd1498Szrj case ARRAY_REF: 593*38fd1498Szrj case ARRAY_RANGE_REF: 594*38fd1498Szrj dump_child ("op 0", TREE_OPERAND (t, 0)); 595*38fd1498Szrj dump_child ("op 1", TREE_OPERAND (t, 1)); 596*38fd1498Szrj dump_child ("op 2", TREE_OPERAND (t, 2)); 597*38fd1498Szrj dump_child ("op 3", TREE_OPERAND (t, 3)); 598*38fd1498Szrj break; 599*38fd1498Szrj 600*38fd1498Szrj case COND_EXPR: 601*38fd1498Szrj dump_child ("op 0", TREE_OPERAND (t, 0)); 602*38fd1498Szrj dump_child ("op 1", TREE_OPERAND (t, 1)); 603*38fd1498Szrj dump_child ("op 2", TREE_OPERAND (t, 2)); 604*38fd1498Szrj break; 605*38fd1498Szrj 606*38fd1498Szrj case TRY_FINALLY_EXPR: 607*38fd1498Szrj dump_child ("op 0", TREE_OPERAND (t, 0)); 608*38fd1498Szrj dump_child ("op 1", TREE_OPERAND (t, 1)); 609*38fd1498Szrj break; 610*38fd1498Szrj 611*38fd1498Szrj case CALL_EXPR: 612*38fd1498Szrj { 613*38fd1498Szrj int i = 0; 614*38fd1498Szrj tree arg; 615*38fd1498Szrj call_expr_arg_iterator iter; 616*38fd1498Szrj dump_child ("fn", CALL_EXPR_FN (t)); 617*38fd1498Szrj FOR_EACH_CALL_EXPR_ARG (arg, iter, t) 618*38fd1498Szrj { 619*38fd1498Szrj char buffer[32]; 620*38fd1498Szrj sprintf (buffer, "%u", i); 621*38fd1498Szrj dump_child (buffer, arg); 622*38fd1498Szrj i++; 623*38fd1498Szrj } 624*38fd1498Szrj } 625*38fd1498Szrj break; 626*38fd1498Szrj 627*38fd1498Szrj case CONSTRUCTOR: 628*38fd1498Szrj { 629*38fd1498Szrj unsigned HOST_WIDE_INT cnt; 630*38fd1498Szrj tree index, value; 631*38fd1498Szrj dump_int (di, "lngt", CONSTRUCTOR_NELTS (t)); 632*38fd1498Szrj FOR_EACH_CONSTRUCTOR_ELT (CONSTRUCTOR_ELTS (t), cnt, index, value) 633*38fd1498Szrj { 634*38fd1498Szrj dump_child ("idx", index); 635*38fd1498Szrj dump_child ("val", value); 636*38fd1498Szrj } 637*38fd1498Szrj } 638*38fd1498Szrj break; 639*38fd1498Szrj 640*38fd1498Szrj case BIND_EXPR: 641*38fd1498Szrj dump_child ("vars", TREE_OPERAND (t, 0)); 642*38fd1498Szrj dump_child ("body", TREE_OPERAND (t, 1)); 643*38fd1498Szrj break; 644*38fd1498Szrj 645*38fd1498Szrj case LOOP_EXPR: 646*38fd1498Szrj dump_child ("body", TREE_OPERAND (t, 0)); 647*38fd1498Szrj break; 648*38fd1498Szrj 649*38fd1498Szrj case EXIT_EXPR: 650*38fd1498Szrj dump_child ("cond", TREE_OPERAND (t, 0)); 651*38fd1498Szrj break; 652*38fd1498Szrj 653*38fd1498Szrj case RETURN_EXPR: 654*38fd1498Szrj dump_child ("expr", TREE_OPERAND (t, 0)); 655*38fd1498Szrj break; 656*38fd1498Szrj 657*38fd1498Szrj case TARGET_EXPR: 658*38fd1498Szrj dump_child ("decl", TREE_OPERAND (t, 0)); 659*38fd1498Szrj dump_child ("init", TREE_OPERAND (t, 1)); 660*38fd1498Szrj dump_child ("clnp", TREE_OPERAND (t, 2)); 661*38fd1498Szrj /* There really are two possible places the initializer can be. 662*38fd1498Szrj After RTL expansion, the second operand is moved to the 663*38fd1498Szrj position of the fourth operand, and the second operand 664*38fd1498Szrj becomes NULL. */ 665*38fd1498Szrj dump_child ("init", TREE_OPERAND (t, 3)); 666*38fd1498Szrj break; 667*38fd1498Szrj 668*38fd1498Szrj case CASE_LABEL_EXPR: 669*38fd1498Szrj dump_child ("name", CASE_LABEL (t)); 670*38fd1498Szrj if (CASE_LOW (t)) 671*38fd1498Szrj { 672*38fd1498Szrj dump_child ("low ", CASE_LOW (t)); 673*38fd1498Szrj if (CASE_HIGH (t)) 674*38fd1498Szrj dump_child ("high", CASE_HIGH (t)); 675*38fd1498Szrj } 676*38fd1498Szrj break; 677*38fd1498Szrj case LABEL_EXPR: 678*38fd1498Szrj dump_child ("name", TREE_OPERAND (t,0)); 679*38fd1498Szrj break; 680*38fd1498Szrj case GOTO_EXPR: 681*38fd1498Szrj dump_child ("labl", TREE_OPERAND (t, 0)); 682*38fd1498Szrj break; 683*38fd1498Szrj case SWITCH_EXPR: 684*38fd1498Szrj dump_child ("cond", TREE_OPERAND (t, 0)); 685*38fd1498Szrj dump_child ("body", TREE_OPERAND (t, 1)); 686*38fd1498Szrj if (TREE_OPERAND (t, 2)) 687*38fd1498Szrj { 688*38fd1498Szrj dump_child ("labl", TREE_OPERAND (t,2)); 689*38fd1498Szrj } 690*38fd1498Szrj break; 691*38fd1498Szrj case OMP_CLAUSE: 692*38fd1498Szrj { 693*38fd1498Szrj int i; 694*38fd1498Szrj fprintf (di->stream, "%s\n", omp_clause_code_name[OMP_CLAUSE_CODE (t)]); 695*38fd1498Szrj for (i = 0; i < omp_clause_num_ops[OMP_CLAUSE_CODE (t)]; i++) 696*38fd1498Szrj dump_child ("op: ", OMP_CLAUSE_OPERAND (t, i)); 697*38fd1498Szrj } 698*38fd1498Szrj break; 699*38fd1498Szrj default: 700*38fd1498Szrj /* There are no additional fields to print. */ 701*38fd1498Szrj break; 702*38fd1498Szrj } 703*38fd1498Szrj 704*38fd1498Szrj done: 705*38fd1498Szrj if (dump_flag (di, TDF_ADDRESS, NULL)) 706*38fd1498Szrj dump_pointer (di, "addr", (void *)t); 707*38fd1498Szrj 708*38fd1498Szrj /* Terminate the line. */ 709*38fd1498Szrj fprintf (di->stream, "\n"); 710*38fd1498Szrj } 711*38fd1498Szrj 712*38fd1498Szrj /* Return nonzero if FLAG has been specified for the dump, and NODE 713*38fd1498Szrj is not the root node of the dump. */ 714*38fd1498Szrj 715*38fd1498Szrj int dump_flag (dump_info_p di, dump_flags_t flag, const_tree node) 716*38fd1498Szrj { 717*38fd1498Szrj return (di->flags & flag) && (node != di->node); 718*38fd1498Szrj } 719*38fd1498Szrj 720*38fd1498Szrj /* Dump T, and all its children, on STREAM. */ 721*38fd1498Szrj 722*38fd1498Szrj void 723*38fd1498Szrj dump_node (const_tree t, dump_flags_t flags, FILE *stream) 724*38fd1498Szrj { 725*38fd1498Szrj struct dump_info di; 726*38fd1498Szrj dump_queue_p dq; 727*38fd1498Szrj dump_queue_p next_dq; 728*38fd1498Szrj 729*38fd1498Szrj /* Initialize the dump-information structure. */ 730*38fd1498Szrj di.stream = stream; 731*38fd1498Szrj di.index = 0; 732*38fd1498Szrj di.column = 0; 733*38fd1498Szrj di.queue = 0; 734*38fd1498Szrj di.queue_end = 0; 735*38fd1498Szrj di.free_list = 0; 736*38fd1498Szrj di.flags = flags; 737*38fd1498Szrj di.node = t; 738*38fd1498Szrj di.nodes = splay_tree_new (splay_tree_compare_pointers, 0, 739*38fd1498Szrj (splay_tree_delete_value_fn) 740*38fd1498Szrj (void (*) (void)) free); 741*38fd1498Szrj 742*38fd1498Szrj /* Queue up the first node. */ 743*38fd1498Szrj queue (&di, t, DUMP_NONE); 744*38fd1498Szrj 745*38fd1498Szrj /* Until the queue is empty, keep dumping nodes. */ 746*38fd1498Szrj while (di.queue) 747*38fd1498Szrj dequeue_and_dump (&di); 748*38fd1498Szrj 749*38fd1498Szrj /* Now, clean up. */ 750*38fd1498Szrj for (dq = di.free_list; dq; dq = next_dq) 751*38fd1498Szrj { 752*38fd1498Szrj next_dq = dq->next; 753*38fd1498Szrj free (dq); 754*38fd1498Szrj } 755*38fd1498Szrj splay_tree_delete (di.nodes); 756*38fd1498Szrj } 757