OSDN Git Service

* target.h (invalid_conversion, invalid_unary_op,
[pf3gnuchains/gcc-fork.git] / gcc / cp / cvt.c
index 5b153ef..de0792a 100644 (file)
@@ -1,44 +1,47 @@
 /* Language-level data type conversion for GNU C++.
    Copyright (C) 1987, 1988, 1992, 1993, 1994, 1995, 1996, 1997, 1998,
-   1999, 2000 Free Software Foundation, Inc.
+   1999, 2000, 2001, 2002, 2003, 2004, 2005 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
-the Free Software Foundation, 59 Temple Place - Suite 330,
-Boston, MA 02111-1307, USA.  */
+along with GCC; see the file COPYING.  If not, write to
+the Free Software Foundation, 51 Franklin Street, Fifth Floor,
+Boston, MA 02110-1301, 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"
 #include "convert.h"
 #include "toplev.h"
 #include "decl.h"
+#include "target.h"
 
-static tree cp_convert_to_pointer PARAMS ((tree, tree, int));
-static tree convert_to_pointer_force PARAMS ((tree, tree));
-static tree build_up_reference PARAMS ((tree, tree, int, tree));
-static void warn_ref_binding PARAMS ((tree, tree, tree));
+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
@@ -52,7 +55,7 @@ static void warn_ref_binding PARAMS ((tree, tree, tree));
    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.
@@ -72,30 +75,30 @@ static void warn_ref_binding PARAMS ((tree, tree, tree));
    but not static_cast).  */
 
 static tree
-cp_convert_to_pointer (type, expr, force)
-     tree type, expr;
-     int force;
+cp_convert_to_pointer (tree type, tree expr, bool force)
 {
-  register tree intype = TREE_TYPE (expr);
-  register enum tree_code form;
+  tree intype = TREE_TYPE (expr);
+  enum tree_code form;
   tree rval;
+  if (intype == error_mark_node)
+    return error_mark_node;
 
   if (IS_AGGR_TYPE (intype))
     {
       intype = complete_type (intype);
       if (!COMPLETE_TYPE_P (intype))
        {
-         cp_error ("can't convert from incomplete type `%T' to `%T'",
-                   intype, type);
+         error ("can't convert from incomplete type %qT to %qT",
+                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",
-                     expr, intype, type);
+           error ("conversion of %qE from %qT to %qT is ambiguous",
+                  expr, intype, type);
          return rval;
        }
     }
@@ -105,31 +108,17 @@ cp_convert_to_pointer (type, expr, force)
       && (TREE_CODE (TREE_TYPE (type)) == FUNCTION_TYPE
          || VOID_TYPE_P (TREE_TYPE (type))))
     {
-      /* Allow an implicit this pointer for pointer to member
-        functions.  */
-      if (TYPE_PTRMEMFUNC_P (intype))
-       {
-         tree fntype = TREE_TYPE (TYPE_PTRMEMFUNC_FN_TYPE (intype));
-         tree decl = maybe_dummy_object (TYPE_METHOD_BASETYPE (fntype), 0);
-         expr = build (OFFSET_REF, fntype, decl, expr);
-       }
-
-      if (TREE_CODE (expr) == OFFSET_REF
-         && TREE_CODE (TREE_TYPE (expr)) == METHOD_TYPE)
-       expr = resolve_offset_ref (expr);
-      if (TREE_CODE (TREE_TYPE (expr)) == METHOD_TYPE)
-       expr = build_addr_func (expr);
+      if (TYPE_PTRMEMFUNC_P (intype)
+         || TREE_CODE (intype) == METHOD_TYPE)
+       return convert_member_func_to_ptr (type, expr);
       if (TREE_CODE (TREE_TYPE (expr)) == POINTER_TYPE)
-       {
-         if (TREE_CODE (TREE_TYPE (TREE_TYPE (expr))) == METHOD_TYPE)
-           if (pedantic || warn_pmf2ptr)
-             cp_pedwarn ("converting from `%T' to `%T'", TREE_TYPE (expr),
-                         type);
-         return build1 (NOP_EXPR, type, expr);
-       }
+       return build_nop (type, expr);
       intype = TREE_TYPE (expr);
     }
 
+  if (expr == error_mark_node)
+    return error_mark_node;
+
   form = TREE_CODE (intype);
 
   if (POINTER_TYPE_P (intype))
@@ -141,152 +130,160 @@ cp_convert_to_pointer (type, expr, force)
          && 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);
            }
        }
 
-      if (TYPE_PTRMEM_P (type) && TYPE_PTRMEM_P (intype))
+      if (TYPE_PTRMEMFUNC_P (type))
        {
-         tree b1; 
-         tree b2;
-         tree binfo;
-         tree virt_binfo;
-         enum tree_code code;
+         error ("cannot convert %qE from type %qT to type %qT",
+                expr, intype, type);
+         return error_mark_node;
+       }
 
-         b1 = TYPE_OFFSET_BASETYPE (TREE_TYPE (type));
-         b2 = TYPE_OFFSET_BASETYPE (TREE_TYPE (intype));
-         binfo = get_binfo (b2, b1, 1);
+      return build_nop (type, expr);
+    }
+  else if (TYPE_PTRMEM_P (type) && TYPE_PTRMEM_P (intype))
+    {
+      tree b1;
+      tree b2;
+      tree binfo;
+      enum tree_code code = PLUS_EXPR;
+      base_kind bk;
+
+      b1 = TYPE_PTRMEM_CLASS_TYPE (type);
+      b2 = TYPE_PTRMEM_CLASS_TYPE (intype);
+      binfo = lookup_base (b1, b2, ba_check, &bk);
+      if (!binfo)
+       {
+         binfo = lookup_base (b2, b1, ba_check, &bk);
+         code = MINUS_EXPR;
+       }
+      if (binfo == error_mark_node)
+       return error_mark_node;
 
-         if (binfo == NULL_TREE)
-           {
-             binfo = get_binfo (b1, b2, 1);
-             code = MINUS_EXPR;
-           }
+      if (bk == bk_via_virtual)
+       {
+         if (force)
+           warning (0, "pointer to member cast from %qT to %qT is via"
+                    " virtual base", intype, type);
          else
-           code = PLUS_EXPR;
-
-         if (binfo == error_mark_node)
-           return error_mark_node;
-
-          virt_binfo = binfo_from_vbase (binfo);
-          if (virt_binfo)
            {
-             if (force)
-               cp_warning ("pointer to member cast via virtual base `%T' of `%T'",
-                           BINFO_TYPE (virt_binfo),
-                            BINFO_TYPE (BINFO_INHERITANCE_CHAIN (virt_binfo)));
-              else
-                {
-                 cp_error ("pointer to member cast via virtual base `%T' of `%T'",
-                           BINFO_TYPE (virt_binfo),
-                            BINFO_TYPE (BINFO_INHERITANCE_CHAIN (virt_binfo)));
-                 return error_mark_node;
-               }
-             /* This is a reinterpret cast, whose result is unspecified.
-                We choose to do nothing.  */
-             return build1 (NOP_EXPR, type, expr);
+             error ("pointer to member cast from %qT to %qT is"
+                    " via virtual base", intype, type);
+             return error_mark_node;
            }
-             
-         if (TREE_CODE (expr) == PTRMEM_CST)
-           expr = cplus_expand_constant (expr);
-
-         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'",
-                   expr, intype, 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);
        }
 
-      rval = build1 (NOP_EXPR, type, expr);
-      TREE_CONSTANT (rval) = TREE_CONSTANT (expr);
-      return rval;
+      if (TREE_CODE (expr) == PTRMEM_CST)
+       expr = cplus_expand_constant (expr);
+
+      if (binfo && !integer_zerop (BINFO_OFFSET (binfo)))
+       expr = size_binop (code,
+                          build_nop (sizetype, expr),
+                          BINFO_OFFSET (binfo));
+      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);
+    return build_ptrmemfunc (TYPE_PTRMEMFUNC_FN_TYPE (type), expr, 0,
+                            /*c_cast_p=*/false);
   else if (TYPE_PTRMEMFUNC_P (intype))
     {
-      cp_error ("cannot convert `%E' from type `%T' to type `%T'",
-               expr, intype, type);
+      if (!warn_pmf2ptr)
+       {
+         if (TREE_CODE (expr) == PTRMEM_CST)
+           return cp_convert_to_pointer (type,
+                                         PTRMEM_CST_MEMBER (expr),
+                                         force);
+         else if (TREE_CODE (expr) == OFFSET_REF)
+           {
+             tree object = TREE_OPERAND (expr, 0);
+             return get_member_function_from_ptrfunc (&object,
+                                                      TREE_OPERAND (expr, 1));
+           }
+       }
+      error ("cannot convert %qE from type %qT to type %qT",
+            expr, intype, type);
       return error_mark_node;
     }
 
-  my_friendly_assert (form != OFFSET_TYPE, 186);
-
   if (integer_zerop (expr))
     {
       if (TYPE_PTRMEMFUNC_P (type))
-       return build_ptrmemfunc (TYPE_PTRMEMFUNC_FN_TYPE (type), expr, 0);
+       return build_ptrmemfunc (TYPE_PTRMEMFUNC_FN_TYPE (type), expr, 0,
+                                /*c_cast_p=*/false);
 
       if (TYPE_PTRMEM_P (type))
-       /* A NULL pointer-to-member is represented by -1, not by
-          zero.  */
-       expr = build_int_2 (-1, -1);
+       {
+         /* A NULL pointer-to-member is represented by -1, not by
+            zero.  */
+         expr = build_int_cst (type, -1);
+         /* Fix up the representation of -1 if appropriate.  */
+         expr = force_fit_type (expr, 0, false, false);
+       }
       else
-       expr = build_int_2 (0, 0);
-      TREE_TYPE (expr) = type;
+       expr = build_int_cst (type, 0);
+
       return expr;
     }
+  else if (TYPE_PTR_TO_MEMBER_P (type) && INTEGRAL_CODE_P (form))
+    {
+      error ("invalid conversion from %qT to %qT", 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);
-      /* 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)))
-       /* There is supposed to be some integral type
-          that is the same width as a pointer.  */
-       abort ();
+      expr = cp_convert (c_common_type_for_size (POINTER_SIZE, 0), expr);
+      /* Modes may be different but sizes should be the same.  There
+        is supposed to be some integral type that is the same width
+        as a pointer.  */
+      gcc_assert (GET_MODE_SIZE (TYPE_MODE (TREE_TYPE (expr)))
+                 == GET_MODE_SIZE (TYPE_MODE (type)));
+
       return convert_to_pointer (type, expr);
     }
 
   if (type_unknown_p (expr))
-    return instantiate_type (type, expr, itf_complain);
+    return instantiate_type (type, expr, tf_error | tf_warning);
 
-  cp_error ("cannot convert `%E' from type `%T' to type `%T'",
-           expr, intype, type);
+  error ("cannot convert %qE from type %qT to type %qT",
+        expr, intype, type);
   return error_mark_node;
 }
 
@@ -295,18 +292,10 @@ cp_convert_to_pointer (type, expr, force)
    (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;
-    }
+  tree intype = TREE_TYPE (expr);
+  enum tree_code form = TREE_CODE (intype);
 
   if (form == POINTER_TYPE)
     {
@@ -319,38 +308,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_unique, NULL);
+         if (!binfo)
            {
-             cp_error ("type `%T' is ambiguous base of `%T'",
-                       TREE_TYPE (type),
-                       TREE_TYPE (intype));
-             return error_mark_node;
+             binfo = lookup_base (TREE_TYPE (type), TREE_TYPE (intype),
+                                  ba_unique, 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)
-               {
-                 cp_error ("type `%T' is ambiguous base of `%T'",
-                           TREE_TYPE (intype),
-                           TREE_TYPE (type));
-                 return error_mark_node;
-               }
-             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, 1);
+  return cp_convert_to_pointer (type, expr, true);
 }
 
 /* We are passing something to a function which requires a reference.
@@ -362,16 +346,13 @@ convert_to_pointer_force (type, expr)
      If DIRECT_BIND is set, DECL is the reference we're binding to.  */
 
 static tree
-build_up_reference (type, arg, flags, decl)
-     tree type, arg, decl;
-     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);
+  gcc_assert (TREE_CODE (type) == REFERENCE_TYPE);
 
   if ((flags & DIRECT_BIND) && ! real_lvalue_p (arg))
     {
@@ -379,29 +360,11 @@ build_up_reference (type, arg, flags, decl)
         here because it needs to live as long as DECL.  */
       tree targ = arg;
 
-      arg = build_decl (VAR_DECL, NULL_TREE, argtype);
-      DECL_ARTIFICIAL (arg) = 1;
-      TREE_USED (arg) = 1;
-      TREE_STATIC (arg) = TREE_STATIC (decl);
-
-      if (TREE_STATIC (decl))
-       {
-         /* Namespace-scope or local static; give it a mangled name.  */
-         tree name = mangle_ref_init_variable (decl);
-         DECL_NAME (arg) = name;
-         SET_DECL_ASSEMBLER_NAME (arg, name);
-         arg = pushdecl_top_level (arg);
-       }
-      else
-       {
-         /* Automatic; make sure we handle the cleanup properly.  */
-         maybe_push_cleanup_level (argtype);
-         arg = pushdecl (arg);
-       }
+      arg = make_temporary_var_for_ref_to_temp (decl, TREE_TYPE (arg));
 
       /* Process the initializer for the declaration.  */
       DECL_INITIAL (arg) = targ;
-      cp_finish_decl (arg, targ, NULL_TREE, 
+      cp_finish_decl (arg, targ, NULL_TREE,
                      LOOKUP_ONLYCONVERTING|DIRECT_BIND);
     }
   else if (!(flags & DIRECT_BIND) && ! lvalue_p (arg))
@@ -419,57 +382,51 @@ build_up_reference (type, arg, flags, decl)
       && 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));
-
-  /* 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);
-
-  /* And return the result.  */
-  return rval;
+  return build_nop (type, 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.  */
 
 static void
-warn_ref_binding (reftype, intype, decl)
-     tree reftype, intype, decl;
+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'";
+         msg = "initialization of volatile reference type %q#T from"
+           " rvalue of type %qT";
       else if (CP_TYPE_VOLATILE_P (ttl))
-         msg = "conversion to volatile reference type `%#T' from rvalue of type `%T'";
+         msg = "conversion to volatile reference type %q#T "
+           " from rvalue of type %qT";
       else if (decl)
-         msg = "initialization of non-const reference type `%#T' from rvalue of type `%T'";
+         msg = "initialization of non-const reference type %q#T from"
+           " rvalue of type %qT";
       else
-         msg = "conversion to non-const reference type `%#T' from rvalue of type `%T'";
+         msg = "conversion to non-const reference type %q#T from"
+           " rvalue of type %qT";
 
-      cp_pedwarn (msg, reftype, intype);
+      pedwarn (msg, reftype, intype);
     }
 }
 
@@ -481,42 +438,39 @@ warn_ref_binding (reftype, intype, decl)
    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);
+  tree type = TYPE_MAIN_VARIANT (TREE_TYPE (reftype));
+  tree intype;
   tree rval = NULL_TREE;
   tree rval_as_conversion = NULL_TREE;
-  int i;
+  bool can_convert_intype_to_type;
 
-  if (TREE_CODE (type) == FUNCTION_TYPE && intype == unknown_type_node)
-    {
-      expr = instantiate_type (type, expr, 
-                              (flags & LOOKUP_COMPLAIN)
-                              ? itf_complain : itf_none);
-      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);
 
-      intype = TREE_TYPE (expr);
-    }
+  if (expr == error_mark_node)
+    return error_mark_node;
 
-  if (TREE_CODE (intype) == REFERENCE_TYPE)
-    my_friendly_abort (364);
+  intype = TREE_TYPE (expr);
 
-  intype = TYPE_MAIN_VARIANT (intype);
+  gcc_assert (TREE_CODE (intype) != REFERENCE_TYPE);
 
-  i = comp_target_types (type, intype, 0);
+  intype = TYPE_MAIN_VARIANT (intype);
 
-  if (i <= 0 && (convtype & CONV_IMPLICIT) && IS_AGGR_TYPE (intype)
+  can_convert_intype_to_type = can_convert (type, intype);
+  if (!can_convert_intype_to_type
+      && (convtype & CONV_IMPLICIT) && IS_AGGR_TYPE (intype)
       && ! (flags & LOOKUP_NO_CONVERSION))
     {
       /* 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))
@@ -524,12 +478,12 @@ convert_to_reference (reftype, expr, convtype, flags, decl)
          expr = rval_as_conversion;
          rval_as_conversion = NULL_TREE;
          intype = type;
-         i = 1;
+         can_convert_intype_to_type = 1;
        }
     }
 
-  if (((convtype & CONV_STATIC) && i == -1)
-      || ((convtype & CONV_IMPLICIT) && i == 1))
+  if (((convtype & CONV_STATIC) && can_convert (intype, type))
+      || ((convtype & CONV_IMPLICIT) && can_convert_intype_to_type))
     {
       if (flags & LOOKUP_COMPLAIN)
        {
@@ -538,11 +492,11 @@ convert_to_reference (reftype, expr, convtype, flags, decl)
 
          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",
-                       ttr, reftype);
+           pedwarn ("conversion from %qT to %qT discards qualifiers",
+                    ttr, reftype);
        }
 
       return build_up_reference (reftype, expr, flags, decl);
@@ -555,13 +509,13 @@ convert_to_reference (reftype, expr, convtype, flags, decl)
         should be done directly (jason).  (int &)ri ---> *(int*)&ri */
 
       /* B* bp; A& ar = (A&)bp; is valid, but it's probably not what they
-         meant.  */
+        meant.  */
       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",
-                   intype, reftype);
-         
+         && (comptypes (TREE_TYPE (intype), type,
+                        COMPARE_BASE | COMPARE_DERIVED)))
+       warning (0, "casting %qT to %qT does not dereference pointer",
+                intype, reftype);
+
       rval = build_unary_op (ADDR_EXPR, expr, 0);
       if (rval != error_mark_node)
        rval = convert_force (build_pointer_type (TREE_TYPE (reftype)),
@@ -585,13 +539,8 @@ convert_to_reference (reftype, expr, convtype, flags, decl)
       return rval;
     }
 
-  my_friendly_assert (TREE_CODE (intype) != OFFSET_TYPE, 189);
-
   if (flags & LOOKUP_COMPLAIN)
-    cp_error ("cannot convert type `%T' to type `%T'", intype, reftype);
-
-  if (flags & LOOKUP_SPECULATIVELY)
-    return NULL_TREE;
+    error ("cannot convert type %qT to type %qT", intype, reftype);
 
   return error_mark_node;
 }
@@ -600,118 +549,46 @@ 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 (TREE_TYPE (val)) == REFERENCE_TYPE)
+    {
+      tree t = canonical_type_variant (TREE_TYPE (TREE_TYPE (val)));
+      tree ref = build1 (INDIRECT_REF, t, val);
+
+       /* We *must* set TREE_READONLY when dereferencing a pointer to const,
+         so that we get the proper error message if the result is used
+         to assign to.  Also, &* is supposed to be a no-op.  */
+      TREE_READONLY (ref) = CP_TYPE_CONST_P (t);
+      TREE_THIS_VOLATILE (ref) = CP_TYPE_VOLATILE_P (t);
+      TREE_SIDE_EFFECTS (ref)
+       = (TREE_THIS_VOLATILE (ref) || TREE_SIDE_EFFECTS (val));
+      REFERENCE_REF_P (ref) = 1;
+      val = ref;
+    }
 
-  if (TREE_CODE (type) == OFFSET_TYPE)
-    type = TREE_TYPE (type);
-  if (TREE_CODE (type) == REFERENCE_TYPE)
-    return build_indirect_ref (val, NULL);
   return val;
 }
 
-/* Implicitly convert the lvalue EXPR to another lvalue of type TOTYPE,
-   preserving cv-qualification.  */
+/* Really perform an lvalue-to-rvalue conversion, including copying an
+   argument of class type into a temporary.  */
 
 tree
-convert_lvalue (totype, expr)
-     tree totype, expr;
+force_rvalue (tree expr)
 {
-  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);
-}
-\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.  */
-
-tree
-convert_pointer_to_real (binfo, expr)
-     tree binfo, 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;
-    }
+  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;
-      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;
+    expr = decay_conversion (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;
-}
-
-/* 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.  */
-
-tree
-convert_pointer_to (binfo, expr)
-     tree binfo, expr;
-{
-  return convert_pointer_to_real (binfo, 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);
 }
@@ -721,21 +598,26 @@ 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);
+  tree e = expr;
+  enum tree_code code = TREE_CODE (type);
+  const char *invalid_conv_diag;
 
-  if (e == error_mark_node
-      || TREE_TYPE (e) == error_mark_node)
+  if (error_operand_p (e) || type == error_mark_node)
     return error_mark_node;
 
   complete_type (type);
   complete_type (TREE_TYPE (expr));
 
-  e = decl_constant_value (e);
+  if ((invalid_conv_diag
+       = targetm.invalid_conversion (TREE_TYPE (expr), type)))
+    {
+      error (invalid_conv_diag);
+      return error_mark_node;
+    }
+
+  e = integral_constant_value (e);
 
   if (IS_AGGR_TYPE (type) && (convtype & CONV_FORCE_TEMP)
       /* Some internal structures (vtable_entry_type, sigtbl_ptr_type)
@@ -748,17 +630,32 @@ ocp_convert (type, expr, convtype, flags)
       if (same_type_p (type, TREE_TYPE (e)))
        /* The call to fold will not always remove the NOP_EXPR as
           might be expected, since if one of the types is a typedef;
-          the comparsion in fold is just equality of pointers, not a
+          the comparison in fold is just equality of pointers, not a
           call to comptypes.  We don't call fold in this case because
           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.  */
+        conversion.  */
       else if (TREE_CODE (type) == COMPLEX_TYPE)
-        return fold (convert_to_complex (type, e));
+       return fold_if_not_in_template (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.  */
+         gcc_assert (same_type_p (TYPE_MAIN_VARIANT (TREE_TYPE (e)),
+                                  TYPE_MAIN_VARIANT (type)));
+         TREE_TYPE (e) = TREE_TYPE (TARGET_EXPR_SLOT (e)) = type;
+         return e;
+       }
       else
-       return fold (build1 (NOP_EXPR, type, e));
+       {
+         /* We shouldn't be treating objects of ADDRESSABLE type as
+            rvalues.  */
+         gcc_assert (!TREE_ADDRESSABLE (type));
+         return fold_if_not_in_template (build_nop (type, e));
+       }
     }
 
   if (code == VOID_TYPE && (convtype & CONV_STATIC))
@@ -767,26 +664,19 @@ ocp_convert (type, expr, convtype, flags)
       return e;
     }
 
-  /* Just convert to the type of the member.  */
-  if (code == OFFSET_TYPE)
-    {
-      type = TREE_TYPE (type);
-      code = TREE_CODE (type);
-    }
-
-  if (TREE_CODE (e) == OFFSET_REF)
-    e = resolve_offset_ref (e);
-
   if (INTEGRAL_CODE_P (code))
     {
       tree intype = TREE_TYPE (e);
       /* enum = enum, enum = int, enum = float, (enum)pointer are all
-         errors.  */
+        errors.  */
       if (TREE_CODE (type) == ENUMERAL_TYPE
-         && ((ARITHMETIC_TYPE_P (intype) && ! (convtype & CONV_STATIC))
-             || (TREE_CODE (intype) == POINTER_TYPE)))
+         && (((INTEGRAL_OR_ENUMERATION_TYPE_P (intype)
+               || TREE_CODE (intype) == REAL_TYPE)
+              && ! (convtype & CONV_STATIC))
+             || TREE_CODE (intype) == POINTER_TYPE))
        {
-         cp_pedwarn ("conversion from `%#T' to `%#T'", intype, type);
+         if (flags & LOOKUP_COMPLAIN)
+           pedwarn ("conversion from %q#T to %q#T", intype, type);
 
          if (flag_pedantic_errors)
            return error_mark_node;
@@ -794,54 +684,52 @@ 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);
-         if (flags & LOOKUP_SPECULATIVELY)
-           return NULL_TREE;
+           error ("%q#T used where a %qT was expected", intype, type);
          return error_mark_node;
        }
       if (code == BOOLEAN_TYPE)
+       return cp_truthvalue_conversion (e);
+
+      return fold_if_not_in_template (convert_to_integer (type, e));
+    }
+  if (POINTER_TYPE_P (type) || TYPE_PTR_TO_MEMBER_P (type))
+    return fold_if_not_in_template (cp_convert_to_pointer (type, e, false));
+  if (code == VECTOR_TYPE)
+    {
+      tree in_vtype = TREE_TYPE (e);
+      if (IS_AGGR_TYPE (in_vtype))
        {
-         tree fn = NULL_TREE;
-
-         /* Common Ada/Pascal programmer's mistake.  We always warn
-             about this since it is so bad.  */
-         if (TREE_CODE (expr) == FUNCTION_DECL)
-           fn = expr;
-         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 cp_truthvalue_conversion (e);
+         tree ret_val;
+         ret_val = build_type_conversion (type, e);
+         if (ret_val)
+           return ret_val;
+         if (flags & LOOKUP_COMPLAIN)
+           error ("%q#T used where a %qT was expected", in_vtype, type);
+         return error_mark_node;
        }
-      return fold (convert_to_integer (type, e));
+      return fold_if_not_in_template (convert_to_vector (type, e));
     }
-  if (code == POINTER_TYPE || code == REFERENCE_TYPE
-      || TYPE_PTRMEMFUNC_P (type))
-    return fold (cp_convert_to_pointer (type, e, 0));
-  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 ("%q#T used where a floating point value was expected",
                        TREE_TYPE (e));
        }
       if (code == REAL_TYPE)
-       return fold (convert_to_real (type, e));
+       return fold_if_not_in_template (convert_to_real (type, e));
       else if (code == COMPLEX_TYPE)
-       return fold (convert_to_complex (type, e));
+       return fold_if_not_in_template (convert_to_complex (type, e));
     }
 
   /* New C++ semantics:  since assignment is now based on
@@ -873,27 +761,17 @@ ocp_convert (type, expr, convtype, flags)
           the target with the temp (see [dcl.init]).  */
        ctor = build_user_type_conversion (type, ctor, flags);
       else
-       ctor = build_method_call (NULL_TREE, 
-                                 complete_ctor_identifier,
-                                 build_tree_list (NULL_TREE, ctor),
-                                 TYPE_BINFO (type), flags);
+       ctor = build_special_member_call (NULL_TREE,
+                                         complete_ctor_identifier,
+                                         build_tree_list (NULL_TREE, ctor),
+                                         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",
-             TREE_TYPE (expr), type);
-  if (flags & LOOKUP_SPECULATIVELY)
-    return NULL_TREE;
+    error ("conversion from %qT to non-scalar type %qT requested",
+          TREE_TYPE (expr), type);
   return error_mark_node;
 }
 
@@ -901,7 +779,7 @@ ocp_convert (type, expr, convtype, flags)
    no lvalue-rvalue and similar conversions happen [expr.static.cast/4,
    stmt.expr/1, expr.comma/1].  This permits dereferencing an incomplete type
    in a void context. The C++ standard does not define what an `access' to an
-   object is, but there is reason to beleive that it is the lvalue to rvalue
+   object is, but there is reason to believe that it is the lvalue to rvalue
    conversion -- if it were not, `*&*p = 1' would violate [expr]/4 in that it
    accesses `*p' not to calculate the value to be stored. But, dcl.type.cv/8
    indicates that volatile semantics should be the same between C and C++
@@ -911,124 +789,168 @@ ocp_convert (type, expr, convtype, flags)
    volatile references we do not do this interpretation, because that would
    make it impossible to ignore the reference return value from functions. We
    issue warnings in the confusing cases.
-   
+
    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 
+  if (expr == error_mark_node
       || TREE_TYPE (expr) == error_mark_node)
     return error_mark_node;
   if (!TREE_TYPE (expr))
     return expr;
+  if (invalid_nonstatic_memfn_p (expr))
+    return error_mark_node;
+  if (TREE_CODE (expr) == PSEUDO_DTOR_EXPR)
+    {
+      error ("pseudo-destructor is not called");
+      return error_mark_node;
+    }
   if (VOID_TYPE_P (TREE_TYPE (expr)))
     return expr;
   switch (TREE_CODE (expr))
     {
     case COND_EXPR:
       {
-        /* The two parts of a cond expr might be separate lvalues.  */
-        tree op1 = TREE_OPERAND (expr,1);
-        tree op2 = TREE_OPERAND (expr,2);
-        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);
-        break;
+       /* The two parts of a cond expr might be separate lvalues.  */
+       tree op1 = TREE_OPERAND (expr,1);
+       tree op2 = TREE_OPERAND (expr,2);
+       tree new_op1 = convert_to_void
+         (op1, (implicit && !TREE_SIDE_EFFECTS (op2)
+                ? "second operand of conditional" : NULL));
+       tree new_op2 = convert_to_void
+         (op2, (implicit && !TREE_SIDE_EFFECTS (op1)
+                ? "third operand of conditional" : NULL));
+
+       expr = build3 (COND_EXPR, TREE_TYPE (new_op1),
+                      TREE_OPERAND (expr, 0), new_op1, new_op2);
+       break;
       }
-    
+
     case COMPOUND_EXPR:
       {
-        /* The second part of a compound expr contains the value.  */
-        tree op1 = TREE_OPERAND (expr,1);
-        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);
-        break;
+       /* The second part of a compound expr contains the value.  */
+       tree op1 = TREE_OPERAND (expr,1);
+       tree new_op1 = convert_to_void
+         (op1, (implicit && !TREE_NO_WARNING (expr)
+                ? "right-hand operand of comma" : NULL));
+
+       if (new_op1 != op1)
+         {
+           tree t = build2 (COMPOUND_EXPR, TREE_TYPE (new_op1),
+                            TREE_OPERAND (expr, 0), new_op1);
+           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() */
+
+    case CALL_EXPR:   /* We have a special meaning for volatile void fn().  */
       break;
-    
+
     case INDIRECT_REF:
       {
-        tree type = TREE_TYPE (expr);
-        int is_reference = TREE_CODE (TREE_TYPE (TREE_OPERAND (expr, 0)))
-                           == REFERENCE_TYPE;
-        int is_volatile = TYPE_VOLATILE (type);
-        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",
-                      type, implicit ? implicit : "void context");
-        else if (is_reference && is_volatile)
-          cp_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)
-          expr = TREE_OPERAND (expr, 0);
-      
-        break;
+       tree type = TREE_TYPE (expr);
+       int is_reference = TREE_CODE (TREE_TYPE (TREE_OPERAND (expr, 0)))
+                          == REFERENCE_TYPE;
+       int is_volatile = TYPE_VOLATILE (type);
+       int is_complete = COMPLETE_TYPE_P (complete_type (type));
+
+       if (is_volatile && !is_complete)
+         warning (0, "object of incomplete type %qT will not be accessed in %s",
+                  type, implicit ? implicit : "void context");
+       else if (is_reference && is_volatile)
+         warning (0, "object of type %qT will not be accessed in %s",
+                  TREE_TYPE (TREE_OPERAND (expr, 0)),
+                  implicit ? implicit : "void context");
+       if (is_reference || !is_volatile || !is_complete)
+         expr = TREE_OPERAND (expr, 0);
+
+       break;
       }
-    
+
     case VAR_DECL:
       {
-        /* External variables might be incomplete.  */
-        tree type = TREE_TYPE (expr);
-        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",
-                      expr, type, implicit ? implicit : "void context");
-        break;
+       /* External variables might be incomplete.  */
+       tree type = TREE_TYPE (expr);
+       int is_complete = COMPLETE_TYPE_P (complete_type (type));
+
+       if (TYPE_VOLATILE (type) && !is_complete)
+         warning (0, "object %qE of incomplete type %qT will not be accessed in %s",
+                  expr, type, implicit ? implicit : "void context");
+       break;
       }
 
-    case OFFSET_REF:
-      expr = resolve_offset_ref (expr);
-      break;
-
     default:;
     }
   {
     tree probe = expr;
-  
+
     if (TREE_CODE (probe) == ADDR_EXPR)
       probe = TREE_OPERAND (expr, 0);
     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.  */
-       cp_pedwarn ("%s cannot resolve address of overloaded function",
+       pedwarn ("%s cannot resolve address of overloaded function",
                    implicit ? implicit : "void cast");
+       expr = void_zero_node;
       }
     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'",
+      warning (0, "%s is a reference, not call, to function %qE",
                  implicit, expr);
   }
-  
+
   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
-         and expand_expr_stmt -- inconsistently too.  For the moment
-         leave implicit void conversions unadorned so that expand_expr_stmt
-         has a chance of detecting some of the cases.  */
-      if (!implicit)
-        expr = build1 (CONVERT_EXPR, void_type_node, expr);
+      if (implicit && warn_unused_value && !TREE_NO_WARNING (expr))
+       {
+         /* The middle end does not warn about expressions that have
+            been explicitly cast to void, so we must do so here.  */
+         if (!TREE_SIDE_EFFECTS (expr))
+           warning (0, "%s has no effect", implicit);
+         else
+           {
+             tree e;
+             enum tree_code code;
+             enum tree_code_class class;
+
+             e = expr;
+             /* We might like to warn about (say) "(int) f()", as the
+                cast has no effect, but the compiler itself will
+                generate implicit conversions under some
+                circumstances.  (For example a block copy will be
+                turned into a call to "__builtin_memcpy", with a
+                conversion of the return value to an appropriate
+                type.)  So, to avoid false positives, we strip
+                conversions.  Do not use STRIP_NOPs because it will
+                not strip conversions to "void", as that is not a
+                mode-preserving conversion.  */
+             while (TREE_CODE (e) == NOP_EXPR)
+               e = TREE_OPERAND (e, 0);
+
+             code = TREE_CODE (e);
+             class = TREE_CODE_CLASS (code);
+             if (class == tcc_comparison
+                  || class == tcc_unary
+                  || (class == tcc_binary
+                      && !(code == MODIFY_EXPR
+                           || code == INIT_EXPR
+                           || code == PREDECREMENT_EXPR
+                           || code == PREINCREMENT_EXPR
+                           || code == POSTDECREMENT_EXPR
+                           || code == POSTINCREMENT_EXPR)))
+               warning (0, "value computed is not used");
+           }
+       }
+      expr = build1 (CONVERT_EXPR, void_type_node, expr);
     }
   return expr;
 }
@@ -1051,8 +973,7 @@ convert_to_void (expr, implicit)
    do a little bit more work.  */
 
 tree
-convert (type, expr)
-     tree type, expr;
+convert (tree type, tree expr)
 {
   tree intype;
 
@@ -1062,10 +983,7 @@ convert (type, expr)
   intype = TREE_TYPE (expr);
 
   if (POINTER_TYPE_P (type) && POINTER_TYPE_P (intype))
-    {
-      expr = decl_constant_value (expr);
-      return fold (build1 (NOP_EXPR, type, expr));
-    }
+    return fold_if_not_in_template (build_nop (type, expr));
 
   return ocp_convert (type, expr, CONV_OLD_CONVERT,
                      LOOKUP_NORMAL|LOOKUP_NO_CONVERSION);
@@ -1076,22 +994,18 @@ 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);
+  tree e = expr;
+  enum tree_code code = TREE_CODE (type);
 
   if (code == REFERENCE_TYPE)
-    return fold (convert_to_reference (type, e, CONV_C_CAST, LOOKUP_COMPLAIN,
-                                      NULL_TREE));
-  else if (TREE_CODE (TREE_TYPE (e)) == REFERENCE_TYPE)
-    e = convert_from_reference (e);
+    return (fold_if_not_in_template
+           (convert_to_reference (type, e, CONV_C_CAST, LOOKUP_COMPLAIN,
+                                  NULL_TREE)));
 
   if (code == POINTER_TYPE)
-    return fold (convert_to_pointer_force (type, e));
+    return fold_if_not_in_template (convert_to_pointer_force (type, e));
 
   /* From typeck.c convert_for_assignment */
   if (((TREE_CODE (TREE_TYPE (e)) == POINTER_TYPE && TREE_CODE (e) == ADDR_EXPR
@@ -1100,10 +1014,9 @@ convert_force (type, expr, convtype)
        || integer_zerop (e)
        || TYPE_PTRMEMFUNC_P (TREE_TYPE (e)))
       && TYPE_PTRMEMFUNC_P (type))
-    {
-      /* compatible pointer to member functions.  */
-      return build_ptrmemfunc (TYPE_PTRMEMFUNC_FN_TYPE (type), e, 1);
-    }
+    /* compatible pointer to member functions.  */
+    return build_ptrmemfunc (TYPE_PTRMEMFUNC_FN_TYPE (type), e, 1,
+                            /*c_cast_p=*/1);
 
   return ocp_convert (type, e, CONV_C_CAST|convtype, LOOKUP_NORMAL);
 }
@@ -1114,51 +1027,36 @@ 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;
   tree winner = NULL_TREE;
 
-  if (expr == null_node 
-      && (desires & WANT_INT) 
+  if (expr == null_node
+      && (desires & WANT_INT)
       && !(desires & WANT_NULL))
-    cp_warning ("converting NULL to non-pointer type");
-    
-  if (TREE_CODE (expr) == OFFSET_REF)
-    expr = resolve_offset_ref (expr);
-  expr = convert_from_reference (expr);
+    warning (0, "converting NULL to non-pointer type");
+
   basetype = TREE_TYPE (expr);
 
   if (basetype == error_mark_node)
@@ -1172,6 +1070,7 @@ build_expr_type_conversion (desires, expr, complain)
          return expr;
        /* else fall through...  */
 
+      case VECTOR_TYPE:
       case BOOLEAN_TYPE:
        return (desires & WANT_INT) ? expr : NULL_TREE;
       case ENUMERAL_TYPE:
@@ -1180,19 +1079,20 @@ build_expr_type_conversion (desires, expr, complain)
        return (desires & WANT_FLOAT) ? expr : NULL_TREE;
       case POINTER_TYPE:
        return (desires & WANT_POINTER) ? expr : NULL_TREE;
-       
+
       case FUNCTION_TYPE:
       case ARRAY_TYPE:
-       return (desires & WANT_POINTER) ? default_conversion (expr)
-                                       : NULL_TREE;
+       return (desires & WANT_POINTER) ? decay_conversion (expr)
+                                       : NULL_TREE;
       default:
        return NULL_TREE;
       }
 
   /* The code for conversions from class type is currently only used for
      delete expressions.  Other expressions are handled by build_new_op.  */
-
-  if (! TYPE_HAS_CONVERSION (basetype))
+  if (!complete_type_or_else (basetype, expr))
+    return error_mark_node;
+  if (!TYPE_HAS_CONVERSION (basetype))
     return NULL_TREE;
 
   for (conv = lookup_conversions (basetype); conv; conv = TREE_CHAIN (conv))
@@ -1204,9 +1104,7 @@ build_expr_type_conversion (desires, expr, complain)
       if (winner && winner == cand)
        continue;
 
-      candidate = TREE_TYPE (TREE_TYPE (cand));
-      if (TREE_CODE (candidate) == REFERENCE_TYPE)
-       candidate = TREE_TYPE (candidate);
+      candidate = non_reference (TREE_TYPE (TREE_TYPE (cand)));
 
       switch (TREE_CODE (candidate))
        {
@@ -1230,10 +1128,10 @@ build_expr_type_conversion (desires, expr, complain)
            {
              if (complain)
                {
-                 cp_error ("ambiguous default type conversion from `%T'",
-                           basetype);
-                 cp_error ("  candidate conversions include `%D' and `%D'",
-                           winner, cand);
+                 error ("ambiguous default type conversion from %qT",
+                        basetype);
+                 error ("  candidate conversions include %qD and %qD",
+                        winner, cand);
                }
              return error_mark_node;
            }
@@ -1244,9 +1142,7 @@ build_expr_type_conversion (desires, expr, complain)
 
   if (winner)
     {
-      tree type = TREE_TYPE (TREE_TYPE (winner));
-      if (TREE_CODE (type) == REFERENCE_TYPE)
-       type = TREE_TYPE (type);
+      tree type = non_reference (TREE_TYPE (TREE_TYPE (winner)));
       return build_user_type_conversion (type, expr, LOOKUP_NORMAL);
     }
 
@@ -1256,15 +1152,11 @@ 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 = TYPE_MAIN_VARIANT (type);
 
   /* bool always promotes to int (not unsigned), even if it's the same
@@ -1279,17 +1171,17 @@ type_promotes_to (type)
     {
       int precision = MAX (TYPE_PRECISION (type),
                           TYPE_PRECISION (integer_type_node));
-      tree totype = type_for_size (precision, 0);
-      if (TREE_UNSIGNED (type)
+      tree totype = c_common_type_for_size (precision, 0);
+      if (TYPE_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))
     {
       /* Retain unsignedness if really not getting bigger.  */
-      if (TREE_UNSIGNED (type)
+      if (TYPE_UNSIGNED (type)
          && TYPE_PRECISION (type) == TYPE_PRECISION (integer_type_node))
        type = unsigned_type_node;
       else
@@ -1298,7 +1190,7 @@ type_promotes_to (type)
   else if (type == float_type_node)
     type = double_type_node;
 
-  return cp_build_qualified_type (type, type_quals);
+  return type;
 }
 
 /* The routines below this point are carefully written to conform to
@@ -1310,15 +1202,23 @@ type_promotes_to (type)
    to TYPE.  Return the resulting expression, or error_mark_node if
    the conversion was impossible.  */
 
-tree 
-perform_qualification_conversions (type, expr)
-     tree type;
-     tree expr;
+tree
+perform_qualification_conversions (tree type, tree expr)
 {
-  if (TREE_CODE (type) == POINTER_TYPE
-      && TREE_CODE (TREE_TYPE (expr)) == POINTER_TYPE
-      && comp_ptr_ttypes (TREE_TYPE (type), TREE_TYPE (TREE_TYPE (expr))))
-    return build1 (NOP_EXPR, type, expr);
+  tree expr_type;
+
+  expr_type = TREE_TYPE (expr);
+
+  if (TYPE_PTR_P (type) && TYPE_PTR_P (expr_type)
+      && comp_ptr_ttypes (TREE_TYPE (type), TREE_TYPE (expr_type)))
+    return build_nop (type, expr);
+  else if (TYPE_PTR_TO_MEMBER_P (type)
+          && TYPE_PTR_TO_MEMBER_P (expr_type)
+          && same_type_p (TYPE_PTRMEM_CLASS_TYPE (type),
+                          TYPE_PTRMEM_CLASS_TYPE (expr_type))
+          && comp_ptr_ttypes (TYPE_PTRMEM_POINTED_TO_TYPE (type),
+                              TYPE_PTRMEM_POINTED_TO_TYPE (expr_type)))
+    return build_nop (type, expr);
   else
     return error_mark_node;
 }