/* Tree-dumping functionality for intermediate representation. Copyright (C) 1999, 2000, 2001, 2002 Free Software Foundation, Inc. Written by Mark Mitchell This file is part of GNU CC. GNU CC is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. GNU CC is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with GNU CC; see the file COPYING. If not, write to the Free Software Foundation, 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #include "config.h" #include "system.h" #include "tree.h" #include "cp-tree.h" #include "tree-dump.h" static void dump_access PARAMS ((dump_info_p, tree)); static void dump_op PARAMS ((dump_info_p, tree)); /* Dump a representation of the accessibility information associated with T. */ static void dump_access (di, t) dump_info_p di; tree t; { if (TREE_PROTECTED(t)) dump_string (di, "protected"); else if (TREE_PRIVATE(t)) dump_string (di, "private"); else dump_string (di, "public"); } /* Dump a representation of the specific operator for an overloaded operator associated with node t. */ static void dump_op (di, t) dump_info_p di; tree t; { switch (DECL_OVERLOADED_OPERATOR_P (t)) { case NEW_EXPR: dump_string (di, "new"); break; case VEC_NEW_EXPR: dump_string (di, "vecnew"); break; case DELETE_EXPR: dump_string (di, "delete"); break; case VEC_DELETE_EXPR: dump_string (di, "vecdelete"); break; case CONVERT_EXPR: dump_string (di, "pos"); break; case NEGATE_EXPR: dump_string (di, "neg"); break; case ADDR_EXPR: dump_string (di, "addr"); break; case INDIRECT_REF: dump_string(di, "deref"); break; case BIT_NOT_EXPR: dump_string(di, "not"); break; case TRUTH_NOT_EXPR: dump_string(di, "lnot"); break; case PREINCREMENT_EXPR: dump_string(di, "preinc"); break; case PREDECREMENT_EXPR: dump_string(di, "predec"); break; case PLUS_EXPR: if (DECL_ASSIGNMENT_OPERATOR_P (t)) dump_string (di, "plusassign"); else dump_string(di, "plus"); break; case MINUS_EXPR: if (DECL_ASSIGNMENT_OPERATOR_P (t)) dump_string (di, "minusassign"); else dump_string(di, "minus"); break; case MULT_EXPR: if (DECL_ASSIGNMENT_OPERATOR_P (t)) dump_string (di, "multassign"); else dump_string (di, "mult"); break; case TRUNC_DIV_EXPR: if (DECL_ASSIGNMENT_OPERATOR_P (t)) dump_string (di, "divassign"); else dump_string (di, "div"); break; case TRUNC_MOD_EXPR: if (DECL_ASSIGNMENT_OPERATOR_P (t)) dump_string (di, "modassign"); else dump_string (di, "mod"); break; case BIT_AND_EXPR: if (DECL_ASSIGNMENT_OPERATOR_P (t)) dump_string (di, "andassign"); else dump_string (di, "and"); break; case BIT_IOR_EXPR: if (DECL_ASSIGNMENT_OPERATOR_P (t)) dump_string (di, "orassign"); else dump_string (di, "or"); break; case BIT_XOR_EXPR: if (DECL_ASSIGNMENT_OPERATOR_P (t)) dump_string (di, "xorassign"); else dump_string (di, "xor"); break; case LSHIFT_EXPR: if (DECL_ASSIGNMENT_OPERATOR_P (t)) dump_string (di, "lshiftassign"); else dump_string (di, "lshift"); break; case RSHIFT_EXPR: if (DECL_ASSIGNMENT_OPERATOR_P (t)) dump_string (di, "rshiftassign"); else dump_string (di, "rshift"); break; case EQ_EXPR: dump_string (di, "eq"); break; case NE_EXPR: dump_string (di, "ne"); break; case LT_EXPR: dump_string (di, "lt"); break; case GT_EXPR: dump_string (di, "gt"); break; case LE_EXPR: dump_string (di, "le"); break; case GE_EXPR: dump_string (di, "ge"); break; case TRUTH_ANDIF_EXPR: dump_string (di, "land"); break; case TRUTH_ORIF_EXPR: dump_string (di, "lor"); break; case COMPOUND_EXPR: dump_string (di, "compound"); break; case MEMBER_REF: dump_string (di, "memref"); break; case COMPONENT_REF: dump_string (di, "ref"); break; case ARRAY_REF: dump_string (di, "subs"); break; case POSTINCREMENT_EXPR: dump_string (di, "postinc"); break; case POSTDECREMENT_EXPR: dump_string (di, "postdec"); break; case CALL_EXPR: dump_string (di, "call"); break; case NOP_EXPR: if (DECL_ASSIGNMENT_OPERATOR_P (t)) dump_string (di, "assign"); break; default: break; } } int cp_dump_tree (dump_info, t) void *dump_info; tree t; { enum tree_code code; dump_info_p di = (dump_info_p) dump_info; /* Figure out what kind of node this is. */ code = TREE_CODE (t); if (DECL_P (t)) { if (DECL_LANG_SPECIFIC (t) && DECL_LANGUAGE (t) != lang_cplusplus) dump_string (di, language_to_string (DECL_LANGUAGE (t), 0)); } switch (code) { case IDENTIFIER_NODE: if (IDENTIFIER_OPNAME_P (t)) { dump_string (di, "operator"); return 1; } else if (IDENTIFIER_TYPENAME_P (t)) { dump_child ("tynm", TREE_TYPE (t)); return 1; } else if (t == anonymous_namespace_name) { dump_string (di, "unnamed"); return 1; } break; case POINTER_TYPE: if (TYPE_PTRMEM_P (t)) { dump_string (di, "ptrmem"); dump_child ("ptd", TYPE_PTRMEM_POINTED_TO_TYPE (t)); dump_child ("cls", TYPE_PTRMEM_CLASS_TYPE (t)); return 1; } break; case RECORD_TYPE: case UNION_TYPE: if (TYPE_PTRMEMFUNC_P (t)) { dump_string (di, "ptrmem"); dump_child ("ptd", TYPE_PTRMEM_POINTED_TO_TYPE (t)); dump_child ("cls", TYPE_PTRMEM_CLASS_TYPE (t)); return 1; } dump_child ("vfld", TYPE_VFIELD (t)); if (CLASSTYPE_TEMPLATE_SPECIALIZATION(t)) dump_string(di, "spec"); if (!dump_flag (di, TDF_SLIM, t)) { int i; for (i = 0; i < CLASSTYPE_N_BASECLASSES (t); ++i) { tree base_binfo = BINFO_BASETYPE (TYPE_BINFO (t), i); dump_child ("base", BINFO_TYPE (base_binfo)); if (TREE_VIA_VIRTUAL (base_binfo)) dump_string (di, "virtual"); dump_access (di, base_binfo); } } break; case FIELD_DECL: dump_access (di, t); if (DECL_MUTABLE_P (t)) dump_string(di, "mutable"); break; case VAR_DECL: if (TREE_CODE (CP_DECL_CONTEXT (t)) == RECORD_TYPE) dump_access (di, t); if (TREE_STATIC (t) && !TREE_PUBLIC (t)) dump_string (di, "static"); break; case FUNCTION_DECL: if (!DECL_THUNK_P (t)) { if (DECL_OVERLOADED_OPERATOR_P (t)) { dump_string (di, "operator"); dump_op (di, t); } if (DECL_FUNCTION_MEMBER_P (t)) { dump_string (di, "member"); dump_access (di, t); } if (DECL_PURE_VIRTUAL_P (t)) dump_string (di, "pure"); if (DECL_VIRTUAL_P (t)) dump_string (di, "virtual"); if (DECL_CONSTRUCTOR_P (t)) dump_string (di, "constructor"); if (DECL_DESTRUCTOR_P (t)) dump_string (di, "destructor"); if (DECL_CONV_FN_P (t)) dump_string (di, "conversion"); if (DECL_GLOBAL_CTOR_P (t) || DECL_GLOBAL_DTOR_P (t)) { if (DECL_GLOBAL_CTOR_P (t)) dump_string (di, "global init"); if (DECL_GLOBAL_DTOR_P (t)) dump_string (di, "global fini"); dump_int (di, "prio", GLOBAL_INIT_PRIORITY (t)); } if (DECL_FRIEND_PSEUDO_TEMPLATE_INSTANTIATION (t)) dump_string (di, "pseudo tmpl"); } else { dump_string (di, "thunk"); dump_int (di, "dlta", THUNK_DELTA (t)); dump_child ("vcll", THUNK_VCALL_OFFSET (t)); dump_child ("fn", DECL_INITIAL (t)); } break; case NAMESPACE_DECL: if (DECL_NAMESPACE_ALIAS (t)) dump_child ("alis", DECL_NAMESPACE_ALIAS (t)); else if (!dump_flag (di, TDF_SLIM, t)) dump_child ("dcls", cp_namespace_decls (t)); break; case TEMPLATE_DECL: dump_child ("rslt", DECL_TEMPLATE_RESULT (t)); dump_child ("inst", DECL_TEMPLATE_INSTANTIATIONS (t)); dump_child ("spcs", DECL_TEMPLATE_SPECIALIZATIONS (t)); dump_child ("prms", DECL_TEMPLATE_PARMS (t)); break; case OVERLOAD: dump_child ("crnt", OVL_CURRENT (t)); dump_child ("chan", OVL_CHAIN (t)); break; case TRY_BLOCK: dump_stmt (di, t); if (CLEANUP_P (t)) dump_string (di, "cleanup"); dump_child ("body", TRY_STMTS (t)); dump_child ("hdlr", TRY_HANDLERS (t)); dump_next_stmt (di, t); break; case EH_SPEC_BLOCK: dump_stmt (di, t); dump_child ("body", EH_SPEC_STMTS (t)); dump_child ("raises", EH_SPEC_RAISES (t)); dump_next_stmt (di, t); break; case PTRMEM_CST: dump_child ("clas", PTRMEM_CST_CLASS (t)); dump_child ("mbr", PTRMEM_CST_MEMBER (t)); break; case THROW_EXPR: /* These nodes are unary, but do not have code class `1'. */ dump_child ("op 0", TREE_OPERAND (t, 0)); break; case AGGR_INIT_EXPR: dump_int (di, "ctor", AGGR_INIT_VIA_CTOR_P (t)); dump_child ("fn", TREE_OPERAND (t, 0)); dump_child ("args", TREE_OPERAND (t, 1)); dump_child ("decl", TREE_OPERAND (t, 2)); break; case CTOR_STMT: dump_stmt (di, t); if (CTOR_BEGIN_P (t)) dump_string (di, "begn"); else dump_string (di, "end"); dump_next_stmt (di, t); break; case HANDLER: dump_stmt (di, t); dump_child ("parm", HANDLER_PARMS (t)); dump_child ("body", HANDLER_BODY (t)); dump_next_stmt (di, t); break; case MUST_NOT_THROW_EXPR: dump_stmt (di, t); dump_child ("body", TREE_OPERAND (t, 0)); dump_next_stmt (di, t); break; case SUBOBJECT: dump_stmt (di, t); dump_child ("clnp", TREE_OPERAND (t, 0)); dump_next_stmt (di, t); break; case USING_STMT: dump_stmt (di, t); dump_child ("nmsp", USING_STMT_NAMESPACE (t)); dump_next_stmt (di, t); break; default: break; } return 0; }