OSDN Git Service

PR c++/9704
[pf3gnuchains/gcc-fork.git] / gcc / cppexp.c
index 5853844..202b2d0 100644 (file)
@@ -20,6 +20,8 @@ Boston, MA 02111-1307, USA.  */
 
 #include "config.h"
 #include "system.h"
+#include "coretypes.h"
+#include "tm.h"
 #include "cpplib.h"
 #include "cpphash.h"
 
@@ -30,6 +32,7 @@ Boston, MA 02111-1307, USA.  */
 
 struct op
 {
+  const cpp_token *token;      /* The token forming op (for diagnostics).  */
   cpp_num value;               /* The value logically "right" of op.  */
   enum cpp_ttype op;
 };
@@ -64,6 +67,7 @@ static cpp_num eval_token PARAMS ((cpp_reader *, const cpp_token *));
 static struct op *reduce PARAMS ((cpp_reader *, struct op *, enum cpp_ttype));
 static unsigned int interpret_float_suffix PARAMS ((const uchar *, size_t));
 static unsigned int interpret_int_suffix PARAMS ((const uchar *, size_t));
+static void check_promotion PARAMS ((cpp_reader *, const struct op *));
 
 /* Token type abuse to create unary plus and minus operators.  */
 #define CPP_UPLUS (CPP_LAST_CPP_OP + 1)
@@ -268,13 +272,18 @@ cpp_classify_number (pfile, token)
          return CPP_N_INVALID;
        }
 
-      /* Traditional C only accepted the 'L' suffix.  */
-      if (result != CPP_N_SMALL && result != CPP_N_MEDIUM
-         && CPP_WTRADITIONAL (pfile)
-         && ! cpp_sys_macro_p (pfile))
-       cpp_error (pfile, DL_WARNING,
-                  "traditional C rejects the \"%.*s\" suffix",
-                  (int) (limit - str), str);
+      /* Traditional C only accepted the 'L' suffix.
+         Suppress warning about 'LL' with -Wno-long-long.  */
+      if (CPP_WTRADITIONAL (pfile) && ! cpp_sys_macro_p (pfile))
+       {
+         int u_or_i = (result & (CPP_N_UNSIGNED|CPP_N_IMAGINARY));
+         int large = (result & CPP_N_WIDTH) == CPP_N_LARGE;
+
+         if (u_or_i || (large && CPP_OPTION (pfile, warn_long_long)))
+           cpp_error (pfile, DL_WARNING,
+                      "traditional C rejects the \"%.*s\" suffix",
+                      (int) (limit - str), str);
+       }
 
       if ((result & CPP_N_WIDTH) == CPP_N_LARGE
          && ! CPP_OPTION (pfile, c99)
@@ -376,7 +385,7 @@ cpp_interpret_integer (pfile, token, type)
                   "integer constant is too large for its type");
       /* If too big to be signed, consider it unsigned.  Only warn for
         decimal numbers.  Traditional numbers were always signed (but
-        we still honour an explicit U suffix); but we only have
+        we still honor an explicit U suffix); but we only have
         traditional semantics in directives.  */
       else if (!result.unsignedp
               && !(CPP_OPTION (pfile, traditional)
@@ -499,6 +508,8 @@ parse_defined (pfile)
        cpp_error (pfile, DL_WARNING,
                   "this use of \"defined\" may not be portable");
 
+      _cpp_mark_macro_used (node);
+
       /* A possible controlling macro of the form #if !defined ().
         _cpp_parse_expr checks there was no other junk on the line.  */
       pfile->mi_ind_cmacro = node;
@@ -622,7 +633,7 @@ already on the stack.
 The remaining cases are '(' and ')'.  We handle '(' by skipping the
 reduction phase completely.  ')' is given lower priority than
 everything else, including '(', effectively forcing a reduction of the
-parenthesised expression.  If there is a matching '(', the routine
+parenthesized expression.  If there is a matching '(', the routine
 reduce() exits immediately.  If the normal exit route sees a ')', then
 there cannot have been a matching '(' and an error message is output.
 
@@ -630,9 +641,11 @@ The parser assumes all shifted operators require a left operand unless
 the flag NO_L_OPERAND is set.  These semantics are automatic; any
 extra semantics need to be handled with operator-specific code.  */
 
-/* Flags.  */
+/* Flags.  If CHECK_PROMOTION, we warn if the effective sign of an
+   operand changes because of integer promotions.  */
 #define NO_L_OPERAND   (1 << 0)
 #define LEFT_ASSOC     (1 << 1)
+#define CHECK_PROMOTION        (1 << 2)
 
 /* Operator to priority map.  Must be in the same order as the first
    N entries of enum cpp_ttype.  */
@@ -644,35 +657,35 @@ static const struct operator
 {
   /* EQ */             {0, 0}, /* Shouldn't happen.  */
   /* NOT */            {16, NO_L_OPERAND},
-  /* GREATER */                {12, LEFT_ASSOC},
-  /* LESS */           {12, LEFT_ASSOC},
-  /* PLUS */           {14, LEFT_ASSOC},
-  /* MINUS */          {14, LEFT_ASSOC},
-  /* MULT */           {15, LEFT_ASSOC},
-  /* DIV */            {15, LEFT_ASSOC},
-  /* MOD */            {15, LEFT_ASSOC},
-  /* AND */            {9, LEFT_ASSOC},
-  /* OR */             {7, LEFT_ASSOC},
-  /* XOR */            {8, LEFT_ASSOC},
+  /* GREATER */                {12, LEFT_ASSOC | CHECK_PROMOTION},
+  /* LESS */           {12, LEFT_ASSOC | CHECK_PROMOTION},
+  /* PLUS */           {14, LEFT_ASSOC | CHECK_PROMOTION},
+  /* MINUS */          {14, LEFT_ASSOC | CHECK_PROMOTION},
+  /* MULT */           {15, LEFT_ASSOC | CHECK_PROMOTION},
+  /* DIV */            {15, LEFT_ASSOC | CHECK_PROMOTION},
+  /* MOD */            {15, LEFT_ASSOC | CHECK_PROMOTION},
+  /* AND */            {9, LEFT_ASSOC | CHECK_PROMOTION},
+  /* OR */             {7, LEFT_ASSOC | CHECK_PROMOTION},
+  /* XOR */            {8, LEFT_ASSOC | CHECK_PROMOTION},
   /* RSHIFT */         {13, LEFT_ASSOC},
   /* LSHIFT */         {13, LEFT_ASSOC},
 
-  /* MIN */            {10, LEFT_ASSOC},
-  /* MAX */            {10, LEFT_ASSOC},
+  /* MIN */            {10, LEFT_ASSOC | CHECK_PROMOTION},
+  /* MAX */            {10, LEFT_ASSOC | CHECK_PROMOTION},
 
   /* COMPL */          {16, NO_L_OPERAND},
   /* AND_AND */                {6, LEFT_ASSOC},
   /* OR_OR */          {5, LEFT_ASSOC},
   /* QUERY */          {3, 0},
-  /* COLON */          {4, LEFT_ASSOC},
+  /* COLON */          {4, LEFT_ASSOC | CHECK_PROMOTION},
   /* COMMA */          {2, LEFT_ASSOC},
   /* OPEN_PAREN */     {1, NO_L_OPERAND},
   /* CLOSE_PAREN */    {0, 0},
   /* EOF */            {0, 0},
   /* EQ_EQ */          {11, LEFT_ASSOC},
   /* NOT_EQ */         {11, LEFT_ASSOC},
-  /* GREATER_EQ */     {12, LEFT_ASSOC},
-  /* LESS_EQ */                {12, LEFT_ASSOC},
+  /* GREATER_EQ */     {12, LEFT_ASSOC | CHECK_PROMOTION},
+  /* LESS_EQ */                {12, LEFT_ASSOC | CHECK_PROMOTION},
   /* UPLUS */          {16, NO_L_OPERAND},
   /* UMINUS */         {16, NO_L_OPERAND}
 };
@@ -693,7 +706,6 @@ _cpp_parse_expr (pfile)
      cpp_reader *pfile;
 {
   struct op *top = pfile->op_stack;
-  const cpp_token *token = NULL, *prev_token;
   unsigned int lex_count;
   bool saw_leading_not, want_value = true;
 
@@ -711,10 +723,9 @@ _cpp_parse_expr (pfile)
     {
       struct op op;
 
-      prev_token = token;
-      token = cpp_get_token (pfile);
       lex_count++;
-      op.op = token->type;
+      op.token = cpp_get_token (pfile);
+      op.op = op.token->type;
 
       switch (op.op)
        {
@@ -726,9 +737,9 @@ _cpp_parse_expr (pfile)
        case CPP_HASH:
          if (!want_value)
            SYNTAX_ERROR2 ("missing binary operator before token \"%s\"",
-                          cpp_token_as_text (pfile, token));
+                          cpp_token_as_text (pfile, op.token));
          want_value = false;
-         top->value = eval_token (pfile, token);
+         top->value = eval_token (pfile, op.token);
          continue;
 
        case CPP_NOT:
@@ -743,15 +754,16 @@ _cpp_parse_expr (pfile)
            op.op = CPP_UMINUS;
          break;
        case CPP_OTHER:
-         if (ISGRAPH (token->val.c))
-           SYNTAX_ERROR2 ("invalid character '%c' in #if", token->val.c);
+         if (ISGRAPH (op.token->val.c))
+           SYNTAX_ERROR2 ("invalid character '%c' in #if", op.token->val.c);
          else
-           SYNTAX_ERROR2 ("invalid character '\\%03o' in #if", token->val.c);
+           SYNTAX_ERROR2 ("invalid character '\\%03o' in #if",
+                          op.token->val.c);
 
        default:
          if ((int) op.op <= (int) CPP_EQ || (int) op.op >= (int) CPP_PLUS_EQ)
            SYNTAX_ERROR2 ("token \"%s\" is not valid in preprocessor expressions",
-                          cpp_token_as_text (pfile, token));
+                          cpp_token_as_text (pfile, op.token));
          break;
        }
 
@@ -760,11 +772,11 @@ _cpp_parse_expr (pfile)
        {
          if (!want_value)
            SYNTAX_ERROR2 ("missing binary operator before token \"%s\"",
-                          cpp_token_as_text (pfile, token));
+                          cpp_token_as_text (pfile, op.token));
        }
       else if (want_value)
        {
-         /* Ordering here is subtle and intended to favour the
+         /* Ordering here is subtle and intended to favor the
             missing parenthesis diagnostics over alternatives.  */
          if (op.op == CPP_CLOSE_PAREN)
            {
@@ -775,7 +787,7 @@ _cpp_parse_expr (pfile)
            SYNTAX_ERROR ("#if with no expression");
          if (top->op != CPP_EOF && top->op != CPP_OPEN_PAREN)
            SYNTAX_ERROR2 ("operator '%s' has no right operand",
-                          cpp_token_as_text (pfile, prev_token));
+                          cpp_token_as_text (pfile, top->token));
        }
 
       top = reduce (pfile, top, op.op);
@@ -816,6 +828,7 @@ _cpp_parse_expr (pfile)
        top = _cpp_expand_op_stack (pfile);
 
       top->op = op.op;
+      top->token = op.token;
     }
 
   /* The controlling macro expression is only valid if we called lex 3
@@ -860,6 +873,10 @@ reduce (pfile, top, op)
   prio = optab[op].prio - ((optab[op].flags & LEFT_ASSOC) != 0);
   while (prio < optab[top->op].prio)
     {
+      if (CPP_OPTION (pfile, warn_num_sign_change)
+         && optab[top->op].flags & CHECK_PROMOTION)
+       check_promotion (pfile, top);
+
       switch (top->op)
        {
        case CPP_UPLUS:
@@ -994,6 +1011,29 @@ _cpp_expand_op_stack (pfile)
   return pfile->op_stack + old_size;
 }
 
+/* Emits a warning if the effective sign of either operand of OP
+   changes because of integer promotions.  */
+static void
+check_promotion (pfile, op)
+     cpp_reader *pfile;
+     const struct op *op;
+{
+  if (op->value.unsignedp == op[-1].value.unsignedp)
+    return;
+
+  if (op->value.unsignedp)
+    {
+      if (!num_positive (op[-1].value, CPP_OPTION (pfile, precision)))
+       cpp_error (pfile, DL_WARNING,
+                  "the left operand of \"%s\" changes sign when promoted",
+                  cpp_token_as_text (pfile, op->token));
+    }
+  else if (!num_positive (op->value, CPP_OPTION (pfile, precision)))
+    cpp_error (pfile, DL_WARNING,
+              "the right operand of \"%s\" changes sign when promoted",
+              cpp_token_as_text (pfile, op->token));
+}
+
 /* Clears the unused high order bits of the number pointed to by PNUM.  */
 static cpp_num
 num_trim (num, precision)
@@ -1408,6 +1448,7 @@ num_part_mul (lhs, rhs)
 
   result.high += HIGH_PART (middle[0]);
   result.high += HIGH_PART (middle[1]);
+  result.unsignedp = 1;
 
   return result;
 }
@@ -1513,7 +1554,7 @@ num_div_op (pfile, lhs, rhs, op)
       return lhs;
     }
 
-  /* First non-zero bit of RHS is bit I.  Do naive division by
+  /* First nonzero bit of RHS is bit I.  Do naive division by
      shifting the RHS fully left, and subtracting from LHS if LHS is
      at least as big, and then repeating but with one less shift.
      This is not very efficient, but is easy to understand.  */