OSDN Git Service

* g++.dg/abi/vague1.C: Use xfail, rather than embedded Tcl code.
[pf3gnuchains/gcc-fork.git] / gcc / cp / cvt.c
index bcc467a..c62edf0 100644 (file)
@@ -1,32 +1,35 @@
 /* Language-level data type conversion for GNU C++.
-   Copyright (C) 1987-1988, 1992-2000 Free Software Foundation, Inc.
+   Copyright (C) 1987, 1988, 1992, 1993, 1994, 1995, 1996, 1997, 1998,
+   1999, 2000, 2001, 2002, 2003 Free Software Foundation, Inc.
    Hacked by Michael Tiemann (tiemann@cygnus.com)
 
-This file is part of GNU CC.
+This file is part of GCC.
 
-GNU CC is free software; you can redistribute it and/or modify
+GCC 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,
+GCC 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
+along with GCC; see the file COPYING.  If not, write to
 the Free Software Foundation, 59 Temple Place - Suite 330,
 Boston, MA 02111-1307, USA.  */
 
 
-/* This file contains the functions for converting C expressions
+/* This file contains the functions for converting C++ expressions
    to different data types.  The only entry point is `convert'.
    Every language front end must have a `convert' function
    but what kind of conversions it does will depend on the language.  */
 
 #include "config.h"
 #include "system.h"
+#include "coretypes.h"
+#include "tm.h"
 #include "tree.h"
 #include "flags.h"
 #include "cp-tree.h"
@@ -34,9 +37,10 @@ Boston, MA 02111-1307, USA.  */
 #include "toplev.h"
 #include "decl.h"
 
-static tree cp_convert_to_pointer PROTO((tree, tree));
-static tree convert_to_pointer_force PROTO((tree, tree));
-static tree build_up_reference PROTO((tree, tree, int));
+static tree cp_convert_to_pointer (tree, tree, bool);
+static tree convert_to_pointer_force (tree, tree);
+static tree build_up_reference (tree, tree, int, tree);
+static void warn_ref_binding (tree, tree, tree);
 
 /* Change of width--truncation and extension of integers or reals--
    is represented with NOP_EXPR.  Proper functioning of many things
@@ -50,7 +54,7 @@ static tree build_up_reference PROTO((tree, tree, int));
    narrowing is always done with a NOP_EXPR:
      In convert.c, convert_to_integer.
      In c-typeck.c, build_binary_op_nodefault (boolean ops),
-        and truthvalue_conversion.
+        and c_common_truthvalue_conversion.
      In expr.c: expand_expr, for operands of a MULT_EXPR.
      In fold-const.c: fold.
      In tree.c: get_narrower and get_unwidened.
@@ -65,11 +69,12 @@ static tree build_up_reference PROTO((tree, tree, int));
      else if dealing with method pointers, delegate
      else convert blindly
    else if converting class, pass off to build_type_conversion
-   else try C-style pointer conversion  */
+   else try C-style pointer conversion.  If FORCE is true then allow
+   conversions via virtual bases (these are permitted by reinterpret_cast,
+   but not static_cast).  */
 
 static tree
-cp_convert_to_pointer (type, expr)
-     tree type, expr;
+cp_convert_to_pointer (tree type, tree expr, bool force)
 {
   register tree intype = TREE_TYPE (expr);
   register enum tree_code form;
@@ -78,18 +83,18 @@ cp_convert_to_pointer (type, expr)
   if (IS_AGGR_TYPE (intype))
     {
       intype = complete_type (intype);
-      if (TYPE_SIZE (intype) == NULL_TREE)
+      if (!COMPLETE_TYPE_P (intype))
        {
-         cp_error ("can't convert from incomplete type `%T' to `%T'",
+         error ("can't convert from incomplete type `%T' to `%T'",
                    intype, type);
          return error_mark_node;
        }
 
-      rval = build_type_conversion (type, expr, 1);
+      rval = build_type_conversion (type, expr);
       if (rval)
        {
          if (rval == error_mark_node)
-           cp_error ("conversion of `%E' from `%T' to `%T' is ambiguous",
+           error ("conversion of `%E' from `%T' to `%T' is ambiguous",
                      expr, intype, type);
          return rval;
        }
@@ -98,7 +103,7 @@ cp_convert_to_pointer (type, expr)
   /* Handle anachronistic conversions from (::*)() to cv void* or (*)().  */
   if (TREE_CODE (type) == POINTER_TYPE
       && (TREE_CODE (TREE_TYPE (type)) == FUNCTION_TYPE
-         || TYPE_MAIN_VARIANT (TREE_TYPE (type)) == void_type_node))
+         || VOID_TYPE_P (TREE_TYPE (type))))
     {
       /* Allow an implicit this pointer for pointer to member
         functions.  */
@@ -118,13 +123,16 @@ cp_convert_to_pointer (type, expr)
        {
          if (TREE_CODE (TREE_TYPE (TREE_TYPE (expr))) == METHOD_TYPE)
            if (pedantic || warn_pmf2ptr)
-             cp_pedwarn ("converting from `%T' to `%T'", TREE_TYPE (expr),
+             pedwarn ("converting from `%T' to `%T'", TREE_TYPE (expr),
                          type);
          return build1 (NOP_EXPR, type, expr);
        }
       intype = TREE_TYPE (expr);
     }
 
+  if (expr == error_mark_node)
+    return error_mark_node;
+
   form = TREE_CODE (intype);
 
   if (POINTER_TYPE_P (intype))
@@ -136,44 +144,37 @@ cp_convert_to_pointer (type, expr)
          && TREE_CODE (TREE_TYPE (type)) == RECORD_TYPE
          && IS_AGGR_TYPE (TREE_TYPE (type))
          && IS_AGGR_TYPE (TREE_TYPE (intype))
-         && TREE_CODE (TREE_TYPE (intype)) == RECORD_TYPE
-         /* If EXPR is NULL, then we don't need to do any arithmetic
-            to convert it:
-
-              [conv.ptr]
-
-              The null pointer value is converted to the null pointer
-              value of the destination type.  */
-         && !integer_zerop (expr))
+         && TREE_CODE (TREE_TYPE (intype)) == RECORD_TYPE)
        {
          enum tree_code code = PLUS_EXPR;
-         tree binfo = get_binfo (TREE_TYPE (type), TREE_TYPE (intype), 1);
-         if (binfo == error_mark_node)
-           return error_mark_node;
-         if (binfo == NULL_TREE)
+         tree binfo;
+         tree intype_class;
+         tree type_class;
+         bool same_p;
+
+         intype_class = TREE_TYPE (intype);
+         type_class = TREE_TYPE (type);
+
+         same_p = same_type_p (TYPE_MAIN_VARIANT (intype_class), 
+                               TYPE_MAIN_VARIANT (type_class));
+         binfo = NULL_TREE;
+         /* Try derived to base conversion.  */
+         if (!same_p)
+           binfo = lookup_base (intype_class, type_class, ba_check, NULL);
+         if (!same_p && !binfo)
            {
-             binfo = get_binfo (TREE_TYPE (intype), TREE_TYPE (type), 1);
-             if (binfo == error_mark_node)
-               return error_mark_node;
+             /* Try base to derived conversion.  */
+             binfo = lookup_base (type_class, intype_class, ba_check, NULL);
              code = MINUS_EXPR;
            }
-         if (binfo)
+         if (binfo == error_mark_node)
+           return error_mark_node;
+         if (binfo || same_p)
            {
-             if (TYPE_USES_VIRTUAL_BASECLASSES (TREE_TYPE (type))
-                 || TYPE_USES_VIRTUAL_BASECLASSES (TREE_TYPE (intype))
-                 || ! BINFO_OFFSET_ZEROP (binfo))
-               {
-                 /* Need to get the path we took.  */
-                 tree path;
-
-                 if (code == PLUS_EXPR)
-                   get_base_distance (TREE_TYPE (type), TREE_TYPE (intype),
-                                      0, &path);
-                 else
-                   get_base_distance (TREE_TYPE (intype), TREE_TYPE (type),
-                                      0, &path);
-                 return build_vbase_path (code, type, expr, path, 0);
-               }
+             if (binfo)
+               expr = build_base_path (code, expr, binfo, 0);
+             /* Add any qualifier conversions.  */
+             return build_nop (type, expr);
            }
        }
 
@@ -182,52 +183,57 @@ cp_convert_to_pointer (type, expr)
          tree b1; 
          tree b2;
          tree binfo;
-         enum tree_code code;
+         enum tree_code code = PLUS_EXPR;
+         base_kind bk;
 
          b1 = TYPE_OFFSET_BASETYPE (TREE_TYPE (type));
          b2 = TYPE_OFFSET_BASETYPE (TREE_TYPE (intype));
-         binfo = get_binfo (b2, b1, 1);
-
-         if (binfo == NULL_TREE)
+         binfo = lookup_base (b1, b2, ba_check, &bk);
+         if (!binfo)
            {
-             binfo = get_binfo (b1, b2, 1);
+             binfo = lookup_base (b2, b1, ba_check, &bk);
              code = MINUS_EXPR;
            }
-         else
-           code = PLUS_EXPR;
-
          if (binfo == error_mark_node)
            return error_mark_node;
 
-         if (binfo_from_vbase (binfo))
+          if (bk == bk_via_virtual)
            {
-             cp_error ("conversion to `%T' from pointer to member of virtual base `%T'",
-                       type, intype);
-             return error_mark_node;
+             if (force)
+               warning ("pointer to member cast from `%T' to `%T' is via virtual base",
+                           TREE_TYPE (intype), TREE_TYPE (type));
+              else
+                {
+                 error ("pointer to member cast from `%T' to `%T' is via virtual base",
+                           TREE_TYPE (intype), TREE_TYPE (type));
+                 return error_mark_node;
+               }
+             /* This is a reinterpret cast, whose result is unspecified.
+                We choose to do nothing.  */
+             return build1 (NOP_EXPR, type, expr);
            }
              
          if (TREE_CODE (expr) == PTRMEM_CST)
            expr = cplus_expand_constant (expr);
 
-         if (binfo && ! TREE_VIA_VIRTUAL (binfo))
-           expr = size_binop (code, expr, BINFO_OFFSET (binfo));
+         if (binfo)
+           expr = size_binop (code, convert (sizetype, expr),
+                              BINFO_OFFSET (binfo));
        }
       else if (TYPE_PTRMEMFUNC_P (type))
        {
-         cp_error ("cannot convert `%E' from type `%T' to type `%T'",
+         error ("cannot convert `%E' from type `%T' to type `%T'",
                    expr, intype, type);
          return error_mark_node;
        }
 
-      rval = build1 (NOP_EXPR, type, expr);
-      TREE_CONSTANT (rval) = TREE_CONSTANT (expr);
-      return rval;
+      return build_nop (type, expr);
     }
   else if (TYPE_PTRMEMFUNC_P (type) && TYPE_PTRMEMFUNC_P (intype))
     return build_ptrmemfunc (TYPE_PTRMEMFUNC_FN_TYPE (type), expr, 0);
   else if (TYPE_PTRMEMFUNC_P (intype))
     {
-      cp_error ("cannot convert `%E' from type `%T' to type `%T'",
+      error ("cannot convert `%E' from type `%T' to type `%T'",
                expr, intype, type);
       return error_mark_node;
     }
@@ -238,16 +244,30 @@ cp_convert_to_pointer (type, expr)
     {
       if (TYPE_PTRMEMFUNC_P (type))
        return build_ptrmemfunc (TYPE_PTRMEMFUNC_FN_TYPE (type), expr, 0);
-      expr = build_int_2 (0, 0);
+
+      if (TYPE_PTRMEM_P (type))
+       /* A NULL pointer-to-member is represented by -1, not by
+          zero.  */
+       expr = build_int_2 (-1, -1);
+      else
+       expr = build_int_2 (0, 0);
       TREE_TYPE (expr) = type;
+      /* Fix up the representation of -1 if appropriate.  */
+      force_fit_type (expr, 0);
       return expr;
     }
+  else if ((TYPE_PTRMEM_P (type) || TYPE_PTRMEMFUNC_P (type))
+          && INTEGRAL_CODE_P (form))
+    {
+      error ("invalid conversion from '%T' to '%T'", intype, type);
+      return error_mark_node;
+    }
 
   if (INTEGRAL_CODE_P (form))
     {
       if (TYPE_PRECISION (intype) == POINTER_SIZE)
        return build1 (CONVERT_EXPR, type, expr);
-      expr = cp_convert (type_for_size (POINTER_SIZE, 0), expr);
+      expr = cp_convert (c_common_type_for_size (POINTER_SIZE, 0), expr);
       /* Modes may be different but sizes should be the same.  */
       if (GET_MODE_SIZE (TYPE_MODE (TREE_TYPE (expr)))
          != GET_MODE_SIZE (TYPE_MODE (type)))
@@ -258,9 +278,9 @@ cp_convert_to_pointer (type, expr)
     }
 
   if (type_unknown_p (expr))
-    return instantiate_type (type, expr, 1);
+    return instantiate_type (type, expr, tf_error | tf_warning);
 
-  cp_error ("cannot convert `%E' from type `%T' to type `%T'",
+  error ("cannot convert `%E' from type `%T' to type `%T'",
            expr, intype, type);
   return error_mark_node;
 }
@@ -270,19 +290,11 @@ cp_convert_to_pointer (type, expr)
    (such as conversion from sub-type to private super-type).  */
 
 static tree
-convert_to_pointer_force (type, expr)
-     tree type, expr;
+convert_to_pointer_force (tree type, tree expr)
 {
   register tree intype = TREE_TYPE (expr);
   register enum tree_code form = TREE_CODE (intype);
   
-  if (integer_zerop (expr))
-    {
-      expr = build_int_2 (0, 0);
-      TREE_TYPE (expr) = type;
-      return expr;
-    }
-
   if (form == POINTER_TYPE)
     {
       intype = TYPE_MAIN_VARIANT (intype);
@@ -294,34 +306,33 @@ convert_to_pointer_force (type, expr)
          && TREE_CODE (TREE_TYPE (intype)) == RECORD_TYPE)
        {
          enum tree_code code = PLUS_EXPR;
-         tree path;
-         int distance = get_base_distance (TREE_TYPE (type),
-                                           TREE_TYPE (intype), 0, &path);
-         if (distance == -2)
+         tree binfo;
+
+         binfo = lookup_base (TREE_TYPE (intype), TREE_TYPE (type),
+                              ba_ignore, NULL);
+         if (!binfo)
            {
-           ambig:
-             cp_error ("type `%T' is ambiguous baseclass of `%s'",
-                       TREE_TYPE (type),
-                       TYPE_NAME_STRING (TREE_TYPE (intype)));
-             return error_mark_node;
+             binfo = lookup_base (TREE_TYPE (type), TREE_TYPE (intype),
+                                  ba_ignore, NULL);
+             code = MINUS_EXPR;
            }
-         if (distance == -1)
+         if (binfo == error_mark_node)
+           return error_mark_node;
+         if (binfo)
            {
-             distance = get_base_distance (TREE_TYPE (intype),
-                                           TREE_TYPE (type), 0, &path);
-             if (distance == -2)
-               goto ambig;
-             if (distance < 0)
-               /* Doesn't need any special help from us.  */
-               return build1 (NOP_EXPR, type, expr);
-
-             code = MINUS_EXPR;
+             expr = build_base_path (code, expr, binfo, 0);
+              if (expr == error_mark_node)
+                 return error_mark_node;
+             /* Add any qualifier conversions.  */
+             if (!same_type_p (TREE_TYPE (TREE_TYPE (expr)),
+                               TREE_TYPE (type)))
+               expr = build_nop (type, expr);
+             return expr;
            }
-         return build_vbase_path (code, type, expr, path, 0);
        }
     }
 
-  return cp_convert_to_pointer (type, expr);
+  return cp_convert_to_pointer (type, expr, true);
 }
 
 /* We are passing something to a function which requires a reference.
@@ -329,31 +340,25 @@ convert_to_pointer_force (type, expr)
    value we have to begin with is in ARG.
 
    FLAGS controls how we manage access checking.
-   DIRECT_BIND in FLAGS controls how any temporaries are generated.  */
+   DIRECT_BIND in FLAGS controls how any temporaries are generated.
+     If DIRECT_BIND is set, DECL is the reference we're binding to.  */
 
 static tree
-build_up_reference (type, arg, flags)
-     tree type, arg;
-     int flags;
+build_up_reference (tree type, tree arg, int flags, tree decl)
 {
   tree rval;
   tree argtype = TREE_TYPE (arg);
   tree target_type = TREE_TYPE (type);
-  tree stmt_expr = NULL_TREE;
 
   my_friendly_assert (TREE_CODE (type) == REFERENCE_TYPE, 187);
 
   if ((flags & DIRECT_BIND) && ! real_lvalue_p (arg))
     {
-      /* Create a new temporary variable.  */
+      /* Create a new temporary variable.  We can't just use a TARGET_EXPR
+        here because it needs to live as long as DECL.  */
       tree targ = arg;
-      if (toplevel_bindings_p ())
-       arg = get_temp_name (argtype, 1);
-      else
-       {
-         arg = pushdecl (build_decl (VAR_DECL, NULL_TREE, argtype));
-         DECL_ARTIFICIAL (arg) = 1;
-       }
+
+      arg = make_temporary_var_for_ref_to_temp (decl, TREE_TYPE (arg));
 
       /* Process the initializer for the declaration.  */
       DECL_INITIAL (arg) = targ;
@@ -363,7 +368,7 @@ build_up_reference (type, arg, flags)
   else if (!(flags & DIRECT_BIND) && ! lvalue_p (arg))
     return get_target_expr (arg);
 
-  /* If we had a way to wrap this up, and say, if we ever needed it's
+  /* If we had a way to wrap this up, and say, if we ever needed its
      address, transform all occurrences of the register, into a memory
      reference we could win better.  */
   rval = build_unary_op (ADDR_EXPR, arg, 1);
@@ -375,27 +380,48 @@ build_up_reference (type, arg, flags)
       && IS_AGGR_TYPE (argtype)
       && IS_AGGR_TYPE (target_type))
     {
-      /* We go through get_binfo for the access control.  */
-      tree binfo = get_binfo (target_type, argtype, 1);
+      /* We go through lookup_base for the access control.  */
+      tree binfo = lookup_base (argtype, target_type, ba_check, NULL);
       if (binfo == error_mark_node)
        return error_mark_node;
       if (binfo == NULL_TREE)
        return error_not_base_type (target_type, argtype);
-      rval = convert_pointer_to_real (binfo, rval);
+      rval = build_base_path (PLUS_EXPR, rval, binfo, 1);
     }
   else
     rval
       = convert_to_pointer_force (build_pointer_type (target_type), rval);
-  rval = build1 (NOP_EXPR, type, rval);
-  TREE_CONSTANT (rval) = TREE_CONSTANT (TREE_OPERAND (rval, 0));
+  return build_nop (type, rval);
+}
 
-  /* If we created and initialized a new temporary variable, add the
-     representation of that initialization to the RVAL.  */
-  if (stmt_expr)
-    rval = build (COMPOUND_EXPR, TREE_TYPE (rval), stmt_expr, rval);
+/* Subroutine of convert_to_reference. REFTYPE is the target reference type.
+   INTYPE is the original rvalue type and DECL is an optional _DECL node
+   for diagnostics.
+   
+   [dcl.init.ref] says that if an rvalue is used to
+   initialize a reference, then the reference must be to a
+   non-volatile const type.  */
 
-  /* And return the result.  */
-  return rval;
+static void
+warn_ref_binding (tree reftype, tree intype, tree decl)
+{
+  tree ttl = TREE_TYPE (reftype);
+  
+  if (!CP_TYPE_CONST_NON_VOLATILE_P (ttl))
+    {
+      const char *msg;
+
+      if (CP_TYPE_VOLATILE_P (ttl) && decl)
+         msg = "initialization of volatile reference type `%#T' from rvalue of type `%T'";
+      else if (CP_TYPE_VOLATILE_P (ttl))
+         msg = "conversion to volatile reference type `%#T' from rvalue of type `%T'";
+      else if (decl)
+         msg = "initialization of non-const reference type `%#T' from rvalue of type `%T'";
+      else
+         msg = "conversion to non-const reference type `%#T' from rvalue of type `%T'";
+
+      pedwarn (msg, reftype, intype);
+    }
 }
 
 /* For C++: Only need to do one-level references, but cannot
@@ -406,29 +432,29 @@ build_up_reference (type, arg, flags)
    we know it's an initialization.  */
 
 tree
-convert_to_reference (reftype, expr, convtype, flags, decl)
-     tree reftype, expr;
-     int convtype, flags;
-     tree decl;
+convert_to_reference (tree reftype, tree expr, int convtype,
+                      int flags, tree decl)
 {
   register tree type = TYPE_MAIN_VARIANT (TREE_TYPE (reftype));
-  register tree intype = TREE_TYPE (expr);
+  register tree intype;
   tree rval = NULL_TREE;
   tree rval_as_conversion = NULL_TREE;
   int i;
 
-  if (TREE_CODE (type) == FUNCTION_TYPE && intype == unknown_type_node)
-    {
-      expr = instantiate_type (type, expr, 
-                              (flags & LOOKUP_COMPLAIN) != 0);
-      if (expr == error_mark_node)
-       return error_mark_node;
+  if (TREE_CODE (type) == FUNCTION_TYPE 
+      && TREE_TYPE (expr) == unknown_type_node)
+    expr = instantiate_type (type, expr, 
+                            (flags & LOOKUP_COMPLAIN)
+                            ? tf_error | tf_warning : tf_none);
+  else
+    expr = convert_from_reference (expr);
 
-      intype = TREE_TYPE (expr);
-    }
+  if (expr == error_mark_node)
+    return error_mark_node;
+
+  intype = TREE_TYPE (expr);
 
-  if (TREE_CODE (intype) == REFERENCE_TYPE)
-    my_friendly_abort (364);
+  my_friendly_assert (TREE_CODE (intype) != REFERENCE_TYPE, 364);
 
   intype = TYPE_MAIN_VARIANT (intype);
 
@@ -440,7 +466,7 @@ convert_to_reference (reftype, expr, convtype, flags, decl)
       /* Look for a user-defined conversion to lvalue that we can use.  */
 
       rval_as_conversion
-       = build_type_conversion (reftype, expr, 1);
+       = build_type_conversion (reftype, expr);
 
       if (rval_as_conversion && rval_as_conversion != error_mark_node
          && real_lvalue_p (rval_as_conversion))
@@ -460,33 +486,16 @@ convert_to_reference (reftype, expr, convtype, flags, decl)
          tree ttl = TREE_TYPE (reftype);
          tree ttr = lvalue_type (expr);
 
-         /* [dcl.init.ref] says that if an rvalue is used to
-            initialize a reference, then the reference must be to a
-            non-volatile const type.  */
-         if (! real_lvalue_p (expr)
-             && !CP_TYPE_CONST_NON_VOLATILE_P (ttl))
-           {
-             const char *msg;
-
-             if (CP_TYPE_VOLATILE_P (ttl) && decl)
-               msg = "initialization of volatile reference type `%#T'";
-             else if (CP_TYPE_VOLATILE_P (ttl))
-               msg = "conversion to volatile reference type `%#T'";
-             else if (decl)
-               msg = "initialization of non-const reference type `%#T'";
-             else
-               msg = "conversion to non-const reference type `%#T'";
-
-             cp_pedwarn (msg, reftype);
-             cp_pedwarn ("from rvalue of type `%T'", intype);
-           }
-         else if (! (convtype & CONV_CONST)
+         if (! real_lvalue_p (expr))
+           warn_ref_binding (reftype, intype, decl);
+         
+         if (! (convtype & CONV_CONST)
                   && !at_least_as_qualified_p (ttl, ttr))
-           cp_pedwarn ("conversion from `%T' to `%T' discards qualifiers",
+           pedwarn ("conversion from `%T' to `%T' discards qualifiers",
                        ttr, reftype);
        }
 
-      return build_up_reference (reftype, expr, flags);
+      return build_up_reference (reftype, expr, flags, decl);
     }
   else if ((convtype & CONV_REINTERPRET) && lvalue_p (expr))
     {
@@ -500,7 +509,7 @@ convert_to_reference (reftype, expr, convtype, flags, decl)
       if (TREE_CODE (intype) == POINTER_TYPE
          && (comptypes (TREE_TYPE (intype), type, 
                         COMPARE_BASE | COMPARE_RELAXED )))
-       cp_warning ("casting `%T' to `%T' does not dereference pointer",
+       warning ("casting `%T' to `%T' does not dereference pointer",
                    intype, reftype);
          
       rval = build_unary_op (ADDR_EXPR, expr, 0);
@@ -516,11 +525,8 @@ convert_to_reference (reftype, expr, convtype, flags, decl)
                                         "converting", 0, 0);
       if (rval == NULL_TREE || rval == error_mark_node)
        return rval;
-      rval = build_up_reference (reftype, rval, flags);
-
-      if (rval && ! CP_TYPE_CONST_P (TREE_TYPE (reftype)))
-       cp_pedwarn ("initializing non-const `%T' with `%T' will use a temporary",
-                   reftype, intype);
+      warn_ref_binding (reftype, intype, decl);
+      rval = build_up_reference (reftype, rval, flags, decl);
     }
 
   if (rval)
@@ -532,7 +538,7 @@ convert_to_reference (reftype, expr, convtype, flags, decl)
   my_friendly_assert (TREE_CODE (intype) != OFFSET_TYPE, 189);
 
   if (flags & LOOKUP_COMPLAIN)
-    cp_error ("cannot convert type `%T' to type `%T'", intype, reftype);
+    error ("cannot convert type `%T' to type `%T'", intype, reftype);
 
   if (flags & LOOKUP_SPECULATIVELY)
     return NULL_TREE;
@@ -544,112 +550,49 @@ convert_to_reference (reftype, expr, convtype, flags, decl)
    way down to its lowest form.  */
 
 tree
-convert_from_reference (val)
-     tree val;
+convert_from_reference (tree val)
 {
   tree type = TREE_TYPE (val);
 
   if (TREE_CODE (type) == OFFSET_TYPE)
     type = TREE_TYPE (type);
   if (TREE_CODE (type) == REFERENCE_TYPE)
-    return build_indirect_ref (val, NULL_PTR);
+    return build_indirect_ref (val, NULL);
   return val;
 }
-\f
-/* Call this when we know (for any reason) that expr is not, in fact,
-   zero.  This routine is like convert_pointer_to, but it pays
-   attention to which specific instance of what type we want to
-   convert to.  This routine should eventually become
-   convert_to_pointer after all references to convert_to_pointer
-   are removed.  */
+
+/* Implicitly convert the lvalue EXPR to another lvalue of type TOTYPE,
+   preserving cv-qualification.  */
 
 tree
-convert_pointer_to_real (binfo, expr)
-     tree binfo, expr;
+convert_lvalue (tree totype, tree expr)
 {
-  register tree intype = TREE_TYPE (expr);
-  tree ptr_type;
-  tree type, rval;
-
-  if (intype == error_mark_node)
-    return error_mark_node;
-
-  if (TREE_CODE (binfo) == TREE_VEC)
-    type = BINFO_TYPE (binfo);
-  else if (IS_AGGR_TYPE (binfo))
-    {
-      type = binfo;
-    }
-  else
-    {
-      type = binfo;
-      binfo = NULL_TREE;
-    }
-
-  ptr_type = cp_build_qualified_type (type,
-                                     CP_TYPE_QUALS (TREE_TYPE (intype)));
-  ptr_type = build_pointer_type (ptr_type);
-  if (same_type_p (ptr_type, TYPE_MAIN_VARIANT (intype)))
-    return expr;
-
-  my_friendly_assert (!integer_zerop (expr), 191);
-
-  intype = TYPE_MAIN_VARIANT (TREE_TYPE (intype));
-  if (TREE_CODE (type) == RECORD_TYPE
-      && TREE_CODE (intype) == RECORD_TYPE
-      && type != intype)
-    {
-      tree path;
-      int distance
-       = get_base_distance (binfo, intype, 0, &path);
-
-      /* This function shouldn't be called with unqualified arguments
-        but if it is, give them an error message that they can read.  */
-      if (distance < 0)
-       {
-         cp_error ("cannot convert a pointer of type `%T' to a pointer of type `%T'",
-                   intype, type);
-
-         if (distance == -2)
-           cp_error ("because `%T' is an ambiguous base class", type);
-         return error_mark_node;
-       }
-
-      return build_vbase_path (PLUS_EXPR, ptr_type, expr, path, 1);
-    }
-  rval = build1 (NOP_EXPR, ptr_type,
-                TREE_CODE (expr) == NOP_EXPR ? TREE_OPERAND (expr, 0) : expr);
-  TREE_CONSTANT (rval) = TREE_CONSTANT (expr);
-  return rval;
+  totype = cp_build_qualified_type (totype, TYPE_QUALS (TREE_TYPE (expr)));
+  totype = build_reference_type (totype);
+  expr = convert_to_reference (totype, expr, CONV_IMPLICIT, LOOKUP_NORMAL,
+                              NULL_TREE);
+  return convert_from_reference (expr);
 }
 
-/* Call this when we know (for any reason) that expr is
-   not, in fact, zero.  This routine gets a type out of the first
-   argument and uses it to search for the type to convert to.  If there
-   is more than one instance of that type in the expr, the conversion is
-   ambiguous.  This routine should eventually go away, and all
-   callers should use convert_to_pointer_real.  */
+/* Really perform an lvalue-to-rvalue conversion, including copying an
+   argument of class type into a temporary.  */
 
 tree
-convert_pointer_to (binfo, expr)
-     tree binfo, expr;
+force_rvalue (tree expr)
 {
-  tree type;
-
-  if (TREE_CODE (binfo) == TREE_VEC)
-    type = BINFO_TYPE (binfo);
-  else if (IS_AGGR_TYPE (binfo))
-    type = binfo;
+  if (IS_AGGR_TYPE (TREE_TYPE (expr)) && TREE_CODE (expr) != TARGET_EXPR)
+    expr = ocp_convert (TREE_TYPE (expr), expr,
+                       CONV_IMPLICIT|CONV_FORCE_TEMP, LOOKUP_NORMAL);
   else
-    type = binfo;
-  return convert_pointer_to_real (type, expr);
+    expr = decay_conversion (expr);
+
+  return expr;
 }
 \f
 /* C++ conversions, preference to static cast conversions.  */
 
 tree
-cp_convert (type, expr)
-     tree type, expr;
+cp_convert (tree type, tree expr)
 {
   return ocp_convert (type, expr, CONV_OLD_CONVERT, LOOKUP_NORMAL);
 }
@@ -659,9 +602,7 @@ cp_convert (type, expr)
    FLAGS indicates how we should behave.  */
 
 tree
-ocp_convert (type, expr, convtype, flags)
-     tree type, expr;
-     int convtype, flags;
+ocp_convert (tree type, tree expr, int convtype, int flags)
 {
   register tree e = expr;
   register enum tree_code code = TREE_CODE (type);
@@ -673,8 +614,7 @@ ocp_convert (type, expr, convtype, flags)
   complete_type (type);
   complete_type (TREE_TYPE (expr));
 
-  if (TREE_READONLY_DECL_P (e))
-    e = decl_constant_value (e);
+  e = decl_constant_value (e);
 
   if (IS_AGGR_TYPE (type) && (convtype & CONV_FORCE_TEMP)
       /* Some internal structures (vtable_entry_type, sigtbl_ptr_type)
@@ -692,6 +632,24 @@ ocp_convert (type, expr, convtype, flags)
           that can result in infinite recursion; fold will call
           convert, which will call ocp_convert, etc.  */
        return e;
+      /* For complex data types, we need to perform componentwise
+         conversion.  */
+      else if (TREE_CODE (type) == COMPLEX_TYPE)
+        return fold (convert_to_complex (type, e));
+      else if (TREE_CODE (e) == TARGET_EXPR)
+       {
+         /* Don't build a NOP_EXPR of class type.  Instead, change the
+            type of the temporary.  Only allow this for cv-qual changes,
+            though.  */
+         if (!same_type_p (TYPE_MAIN_VARIANT (TREE_TYPE (e)),
+                           TYPE_MAIN_VARIANT (type)))
+           abort ();
+         TREE_TYPE (e) = TREE_TYPE (TARGET_EXPR_SLOT (e)) = type;
+         return e;
+       }
+      else if (TREE_ADDRESSABLE (type))
+       /* We shouldn't be treating objects of ADDRESSABLE type as rvalues.  */
+       abort ();
       else
        return fold (build1 (NOP_EXPR, type, e));
     }
@@ -721,7 +679,7 @@ ocp_convert (type, expr, convtype, flags)
          && ((ARITHMETIC_TYPE_P (intype) && ! (convtype & CONV_STATIC))
              || (TREE_CODE (intype) == POINTER_TYPE)))
        {
-         cp_pedwarn ("conversion from `%#T' to `%#T'", intype, type);
+         pedwarn ("conversion from `%#T' to `%#T'", intype, type);
 
          if (flag_pedantic_errors)
            return error_mark_node;
@@ -729,11 +687,11 @@ ocp_convert (type, expr, convtype, flags)
       if (IS_AGGR_TYPE (intype))
        {
          tree rval;
-         rval = build_type_conversion (type, e, 1);
+         rval = build_type_conversion (type, e);
          if (rval)
            return rval;
          if (flags & LOOKUP_COMPLAIN)
-           cp_error ("`%#T' used where a `%T' was expected", intype, type);
+           error ("`%#T' used where a `%T' was expected", intype, type);
          if (flags & LOOKUP_SPECULATIVELY)
            return NULL_TREE;
          return error_mark_node;
@@ -749,26 +707,28 @@ ocp_convert (type, expr, convtype, flags)
          else if (TREE_CODE (expr) == ADDR_EXPR 
                   && TREE_CODE (TREE_OPERAND (expr, 0)) == FUNCTION_DECL)
            fn = TREE_OPERAND (expr, 0);
-         if (fn)
-           cp_warning ("the address of `%D', will always be `true'", fn);
-         return truthvalue_conversion (e);
+         if (fn && !DECL_WEAK (fn))
+           warning ("the address of `%D', will always be `true'", fn);
+         return cp_truthvalue_conversion (e);
        }
       return fold (convert_to_integer (type, e));
     }
   if (code == POINTER_TYPE || code == REFERENCE_TYPE
       || TYPE_PTRMEMFUNC_P (type))
-    return fold (cp_convert_to_pointer (type, e));
+    return fold (cp_convert_to_pointer (type, e, false));
+  if (code == VECTOR_TYPE)
+    return fold (convert_to_vector (type, e));
   if (code == REAL_TYPE || code == COMPLEX_TYPE)
     {
       if (IS_AGGR_TYPE (TREE_TYPE (e)))
        {
          tree rval;
-         rval = build_type_conversion (type, e, 1);
+         rval = build_type_conversion (type, e);
          if (rval)
            return rval;
          else
            if (flags & LOOKUP_COMPLAIN)
-             cp_error ("`%#T' used where a floating point value was expected",
+             error ("`%#T' used where a floating point value was expected",
                        TREE_TYPE (e));
        }
       if (code == REAL_TYPE)
@@ -805,24 +765,17 @@ ocp_convert (type, expr, convtype, flags)
           with a user-defined conversion sequence, then we direct-initialize
           the target with the temp (see [dcl.init]).  */
        ctor = build_user_type_conversion (type, ctor, flags);
-      if (ctor)
-       ctor = build_method_call (NULL_TREE, ctor_identifier,
-                                 build_tree_list (NULL_TREE, ctor),
-                                 TYPE_BINFO (type), flags);
+      else
+       ctor = build_special_member_call (NULL_TREE, 
+                                         complete_ctor_identifier,
+                                         build_tree_list (NULL_TREE, ctor),
+                                         TYPE_BINFO (type), flags);
       if (ctor)
        return build_cplus_new (type, ctor);
     }
 
-  /* If TYPE or TREE_TYPE (E) is not on the permanent_obstack,
-     then it won't be hashed and hence compare as not equal,
-     even when it is.  */
-  if (code == ARRAY_TYPE
-      && TREE_TYPE (TREE_TYPE (e)) == TREE_TYPE (type)
-      && index_type_equal (TYPE_DOMAIN (TREE_TYPE (e)), TYPE_DOMAIN (type)))
-    return e;
-
   if (flags & LOOKUP_COMPLAIN)
-    cp_error ("conversion from `%T' to non-scalar type `%T' requested",
+    error ("conversion from `%T' to non-scalar type `%T' requested",
              TREE_TYPE (expr), type);
   if (flags & LOOKUP_SPECULATIVELY)
     return NULL_TREE;
@@ -847,16 +800,14 @@ ocp_convert (type, expr, convtype, flags)
    IMPLICIT is tells us the context of an implicit void conversion.  */
 
 tree
-convert_to_void (expr, implicit)
-     tree expr;
-     const char *implicit;
+convert_to_void (tree expr, const char *implicit)
 {
   if (expr == error_mark_node 
       || TREE_TYPE (expr) == error_mark_node)
     return error_mark_node;
   if (!TREE_TYPE (expr))
     return expr;
-  if (same_type_p (TYPE_MAIN_VARIANT (TREE_TYPE (expr)), void_type_node))
+  if (VOID_TYPE_P (TREE_TYPE (expr)))
     return expr;
   switch (TREE_CODE (expr))
     {
@@ -868,10 +819,8 @@ convert_to_void (expr, implicit)
         tree new_op1 = convert_to_void (op1, implicit);
         tree new_op2 = convert_to_void (op2, implicit);
         
-        if (new_op1 != op1 || new_op2 != op2)
-          expr = build (COND_EXPR,
-                        implicit ? TREE_TYPE (expr) : void_type_node,
-                        TREE_OPERAND (expr, 0), new_op1, new_op2);
+       expr = build (COND_EXPR, TREE_TYPE (new_op1),
+                     TREE_OPERAND (expr, 0), new_op1, new_op2);
         break;
       }
     
@@ -882,14 +831,20 @@ convert_to_void (expr, implicit)
         tree new_op1 = convert_to_void (op1, implicit);
         
         if (new_op1 != op1)
-          expr = build (COMPOUND_EXPR, TREE_TYPE (new_op1),
-                        TREE_OPERAND (expr, 0), new_op1);
+         {
+           tree t = build (COMPOUND_EXPR, TREE_TYPE (new_op1),
+                           TREE_OPERAND (expr, 0), new_op1);
+           TREE_SIDE_EFFECTS (t) = TREE_SIDE_EFFECTS (expr);
+           TREE_NO_UNUSED_WARNING (t) = TREE_NO_UNUSED_WARNING (expr);
+           expr = t;
+         }
+
         break;
       }
     
     case NON_LVALUE_EXPR:
     case NOP_EXPR:
-      /* These have already decayed to rvalue. */
+      /* These have already decayed to rvalue.  */
       break;
     
     case CALL_EXPR:   /* we have a special meaning for volatile void fn() */
@@ -901,13 +856,13 @@ convert_to_void (expr, implicit)
         int is_reference = TREE_CODE (TREE_TYPE (TREE_OPERAND (expr, 0)))
                            == REFERENCE_TYPE;
         int is_volatile = TYPE_VOLATILE (type);
-        int is_complete = TYPE_SIZE (complete_type (type)) != NULL_TREE;
+        int is_complete = COMPLETE_TYPE_P (complete_type (type));
         
         if (is_volatile && !is_complete)
-          cp_warning ("object of incomplete type `%T' will not be accessed in %s",
+          warning ("object of incomplete type `%T' will not be accessed in %s",
                       type, implicit ? implicit : "void context");
         else if (is_reference && is_volatile)
-          cp_warning ("object of type `%T' will not be accessed in %s",
+          warning ("object of type `%T' will not be accessed in %s",
                       TREE_TYPE (TREE_OPERAND (expr, 0)),
                       implicit ? implicit : "void context");
         if (is_reference || !is_volatile || !is_complete)
@@ -920,14 +875,18 @@ convert_to_void (expr, implicit)
       {
         /* External variables might be incomplete.  */
         tree type = TREE_TYPE (expr);
-        int is_complete = TYPE_SIZE (complete_type (type)) != NULL_TREE;
+        int is_complete = COMPLETE_TYPE_P (complete_type (type));
         
         if (TYPE_VOLATILE (type) && !is_complete)
-          cp_warning ("object `%E' of incomplete type `%T' will not be accessed in %s",
+          warning ("object `%E' of incomplete type `%T' will not be accessed in %s",
                       expr, type, implicit ? implicit : "void context");
         break;
       }
-    
+
+    case OFFSET_REF:
+      expr = resolve_offset_ref (expr);
+      break;
+
     default:;
     }
   {
@@ -935,23 +894,20 @@ convert_to_void (expr, implicit)
   
     if (TREE_CODE (probe) == ADDR_EXPR)
       probe = TREE_OPERAND (expr, 0);
-    if (!is_overloaded_fn (probe))
-      ;/* OK */
-    else if (really_overloaded_fn (probe))
-        {
-          /* [over.over] enumerates the places where we can take the address
-             of an overloaded function, and this is not one of them.  */
-          cp_pedwarn ("%s has no context for overloaded function name `%E'",
-                      implicit ? implicit : "void cast", expr);
-        }
-    else if (implicit && probe == expr)
+    if (type_unknown_p (probe))
+      {
+       /* [over.over] enumerates the places where we can take the address
+          of an overloaded function, and this is not one of them.  */
+       pedwarn ("%s cannot resolve address of overloaded function",
+                   implicit ? implicit : "void cast");
+      }
+    else if (implicit && probe == expr && is_overloaded_fn (probe))
       /* Only warn when there is no &.  */
-      cp_warning ("%s is a reference, not call, to function `%E'",
-                    implicit, expr);
+      warning ("%s is a reference, not call, to function `%E'",
+                 implicit, expr);
   }
   
-  if (expr != error_mark_node
-      && !same_type_p (TYPE_MAIN_VARIANT (TREE_TYPE (expr)), void_type_node))
+  if (expr != error_mark_node && !VOID_TYPE_P (TREE_TYPE (expr)))
     {
       /* FIXME: This is where we should check for expressions with no
          effects.  At the moment we do that in both build_x_component_expr
@@ -976,14 +932,13 @@ convert_to_void (expr, implicit)
    conversions to/from basetypes may involve memory references
    (vbases) and adding or subtracting small values (multiple
    inheritance), but it calls convert from the constant folding code
-   on subtrees of already build trees after it has ripped them apart.
+   on subtrees of already built trees after it has ripped them apart.
 
    Also, if we ever support range variables, we'll probably also have to
    do a little bit more work.  */
 
 tree
-convert (type, expr)
-     tree type, expr;
+convert (tree type, tree expr)
 {
   tree intype;
 
@@ -994,8 +949,7 @@ convert (type, expr)
 
   if (POINTER_TYPE_P (type) && POINTER_TYPE_P (intype))
     {
-      if (TREE_READONLY_DECL_P (expr))
-       expr = decl_constant_value (expr);
+      expr = decl_constant_value (expr);
       return fold (build1 (NOP_EXPR, type, expr));
     }
 
@@ -1008,10 +962,7 @@ convert (type, expr)
    (such as conversion from sub-type to private super-type).  */
 
 tree
-convert_force (type, expr, convtype)
-     tree type;
-     tree expr;
-     int convtype;
+convert_force (tree type, tree expr, int convtype)
 {
   register tree e = expr;
   register enum tree_code code = TREE_CODE (type);
@@ -1046,38 +997,26 @@ convert_force (type, expr, convtype)
    allowed (references private members, etc).
    If no conversion exists, NULL_TREE is returned.
 
-   If (FOR_SURE & 1) is non-zero, then we allow this type conversion
-   to take place immediately.  Otherwise, we build a SAVE_EXPR
-   which can be evaluated if the results are ever needed.
-
-   Changes to this functions should be mirrored in user_harshness.
-
    FIXME: Ambiguity checking is wrong.  Should choose one by the implicit
    object parameter, or by the second standard conversion sequence if
    that doesn't do it.  This will probably wait for an overloading rewrite.
    (jason 8/9/95)  */
 
 tree
-build_type_conversion (xtype, expr, for_sure)
-     tree xtype, expr;
-     int for_sure;
+build_type_conversion (tree xtype, tree expr)
 {
   /* C++: check to see if we can convert this aggregate type
      into the required type.  */
-  return build_user_type_conversion
-    (xtype, expr, for_sure ? LOOKUP_NORMAL : 0);
+  return build_user_type_conversion (xtype, expr, LOOKUP_NORMAL);
 }
 
 /* Convert the given EXPR to one of a group of types suitable for use in an
    expression.  DESIRES is a combination of various WANT_* flags (q.v.)
-   which indicates which types are suitable.  If COMPLAIN is 1, complain
+   which indicates which types are suitable.  If COMPLAIN is true, complain
    about ambiguity; otherwise, the caller will deal with it.  */
 
 tree
-build_expr_type_conversion (desires, expr, complain)
-     int desires;
-     tree expr;
-     int complain;
+build_expr_type_conversion (int desires, tree expr, bool complain)
 {
   tree basetype = TREE_TYPE (expr);
   tree conv = NULL_TREE;
@@ -1086,7 +1025,7 @@ build_expr_type_conversion (desires, expr, complain)
   if (expr == null_node 
       && (desires & WANT_INT) 
       && !(desires & WANT_NULL))
-    cp_warning ("converting NULL to non-pointer type");
+    warning ("converting NULL to non-pointer type");
     
   if (TREE_CODE (expr) == OFFSET_REF)
     expr = resolve_offset_ref (expr);
@@ -1162,9 +1101,9 @@ build_expr_type_conversion (desires, expr, complain)
            {
              if (complain)
                {
-                 cp_error ("ambiguous default type conversion from `%T'",
+                 error ("ambiguous default type conversion from `%T'",
                            basetype);
-                 cp_error ("  candidate conversions include `%D' and `%D'",
+                 error ("  candidate conversions include `%D' and `%D'",
                            winner, cand);
                }
              return error_mark_node;
@@ -1188,15 +1127,14 @@ build_expr_type_conversion (desires, expr, complain)
 /* Implements integral promotion (4.1) and float->double promotion.  */
 
 tree
-type_promotes_to (type)
-     tree type;
+type_promotes_to (tree type)
 {
   int type_quals;
 
   if (type == error_mark_node)
     return error_mark_node;
 
-  type_quals = CP_TYPE_QUALS (type);
+  type_quals = cp_type_quals (type);
   type = TYPE_MAIN_VARIANT (type);
 
   /* bool always promotes to int (not unsigned), even if it's the same
@@ -1211,14 +1149,14 @@ type_promotes_to (type)
     {
       int precision = MAX (TYPE_PRECISION (type),
                           TYPE_PRECISION (integer_type_node));
-      tree totype = type_for_size (precision, 0);
+      tree totype = c_common_type_for_size (precision, 0);
       if (TREE_UNSIGNED (type)
          && ! int_fits_type_p (TYPE_MAX_VALUE (type), totype))
-       type = type_for_size (precision, 1);
+       type = c_common_type_for_size (precision, 1);
       else
        type = totype;
     }
-  else if (C_PROMOTING_INTEGER_TYPE_P (type))
+  else if (c_promoting_integer_type_p (type))
     {
       /* Retain unsignedness if really not getting bigger.  */
       if (TREE_UNSIGNED (type)
@@ -1243,9 +1181,7 @@ type_promotes_to (type)
    the conversion was impossible.  */
 
 tree 
-perform_qualification_conversions (type, expr)
-     tree type;
-     tree expr;
+perform_qualification_conversions (tree type, tree expr)
 {
   if (TREE_CODE (type) == POINTER_TYPE
       && TREE_CODE (TREE_TYPE (expr)) == POINTER_TYPE