LCOV - code coverage report
Current view: top level - soltools/cpp - _eval.c (source / functions) Hit Total Coverage
Test: commit 10e77ab3ff6f4314137acd6e2702a6e5c1ce1fae Lines: 90 301 29.9 %
Date: 2014-11-03 Functions: 3 4 75.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
       2             : /*
       3             :  * This file is part of the LibreOffice project.
       4             :  *
       5             :  * This Source Code Form is subject to the terms of the Mozilla Public
       6             :  * License, v. 2.0. If a copy of the MPL was not distributed with this
       7             :  * file, You can obtain one at http://mozilla.org/MPL/2.0/.
       8             :  *
       9             :  * This file incorporates work covered by the following license notice:
      10             :  *
      11             :  *   Licensed to the Apache Software Foundation (ASF) under one or more
      12             :  *   contributor license agreements. See the NOTICE file distributed
      13             :  *   with this work for additional information regarding copyright
      14             :  *   ownership. The ASF licenses this file to you under the Apache
      15             :  *   License, Version 2.0 (the "License"); you may not use this file
      16             :  *   except in compliance with the License. You may obtain a copy of
      17             :  *   the License at http://www.apache.org/licenses/LICENSE-2.0 .
      18             :  */
      19             : 
      20             : #include "cpp.h"
      21             : 
      22             : #define NSTAK   32
      23             : #define SGN 0
      24             : #define UNS 1
      25             : #define UND 2
      26             : 
      27             : #define UNSMARK 0x1000
      28             : 
      29             : struct value
      30             : {
      31             :     long val;
      32             :     int type;
      33             : };
      34             : 
      35             : /* conversion types */
      36             : #define RELAT   1
      37             : #define ARITH   2
      38             : #define LOGIC   3
      39             : #define SPCL    4
      40             : #define SHIFT   5
      41             : #define UNARY   6
      42             : 
      43             : /* operator priority, arity, and conversion type, indexed by tokentype */
      44             : struct pri
      45             : {
      46             :     char pri;
      47             :     char arity;
      48             :     char ctype;
      49             : }   priority[] =
      50             : 
      51             : {
      52             :     {
      53             :         0, 0, 0
      54             :     },                                  /* END */
      55             :     {
      56             :         0, 0, 0
      57             :     },                                  /* UNCLASS */
      58             :     {
      59             :         0, 0, 0
      60             :     },                                  /* NAME */
      61             :     {
      62             :         0, 0, 0
      63             :     },                                  /* NUMBER */
      64             :     {
      65             :         0, 0, 0
      66             :     },                                  /* STRING */
      67             :     {
      68             :         0, 0, 0
      69             :     },                                  /* CCON */
      70             :     {
      71             :         0, 0, 0
      72             :     },                                  /* NL */
      73             :     {
      74             :         0, 0, 0
      75             :     },                                  /* WS */
      76             :     {
      77             :         0, 0, 0
      78             :     },                                  /* DSHARP */
      79             :     {
      80             :         11, 2, RELAT
      81             :     },                                  /* EQ */
      82             :     {
      83             :         11, 2, RELAT
      84             :     },                                  /* NEQ */
      85             :     {
      86             :         12, 2, RELAT
      87             :     },                                  /* LEQ */
      88             :     {
      89             :         12, 2, RELAT
      90             :     },                                  /* GEQ */
      91             :     {
      92             :         13, 2, SHIFT
      93             :     },                                  /* LSH */
      94             :     {
      95             :         13, 2, SHIFT
      96             :     },                                  /* RSH */
      97             :     {
      98             :         7, 2, LOGIC
      99             :     },                                  /* LAND */
     100             :     {
     101             :         6, 2, LOGIC
     102             :     },                                  /* LOR */
     103             :     {
     104             :         0, 0, 0
     105             :     },                                  /* PPLUS */
     106             :     {
     107             :         0, 0, 0
     108             :     },                                  /* MMINUS */
     109             :     {
     110             :         0, 0, 0
     111             :     },                                  /* ARROW */
     112             :     {
     113             :         0, 0, 0
     114             :     },                                  /* SBRA */
     115             :     {
     116             :         0, 0, 0
     117             :     },                                  /* SKET */
     118             :     {
     119             :         3, 0, 0
     120             :     },                                  /* LP */
     121             :     {
     122             :         3, 0, 0
     123             :     },                                  /* RP */
     124             :     {
     125             :         0, 0, 0
     126             :     },                                  /* DOT */
     127             :     {
     128             :         10, 2, ARITH
     129             :     },                                  /* AND */
     130             :     {
     131             :         15, 2, ARITH
     132             :     },                                  /* STAR */
     133             :     {
     134             :         14, 2, ARITH
     135             :     },                                  /* PLUS */
     136             :     {
     137             :         14, 2, ARITH
     138             :     },                                  /* MINUS */
     139             :     {
     140             :         16, 1, UNARY
     141             :     },                                  /* TILDE */
     142             :     {
     143             :         16, 1, UNARY
     144             :     },                                  /* NOT */
     145             :     {
     146             :         15, 2, ARITH
     147             :     },                                  /* SLASH */
     148             :     {
     149             :         15, 2, ARITH
     150             :     },                                  /* PCT */
     151             :     {
     152             :         12, 2, RELAT
     153             :     },                                  /* LT */
     154             :     {
     155             :         12, 2, RELAT
     156             :     },                                  /* GT */
     157             :     {
     158             :         9, 2, ARITH
     159             :     },                                  /* CIRC */
     160             :     {
     161             :         8, 2, ARITH
     162             :     },                                  /* OR */
     163             :     {
     164             :         5, 2, SPCL
     165             :     },                                  /* QUEST */
     166             :     {
     167             :         5, 2, SPCL
     168             :     },                                  /* COLON */
     169             :     {
     170             :         0, 0, 0
     171             :     },                                  /* ASGN */
     172             :     {
     173             :         4, 2, 0
     174             :     },                                  /* COMMA */
     175             :     {
     176             :         0, 0, 0
     177             :     },                                  /* SHARP */
     178             :     {
     179             :         0, 0, 0
     180             :     },                                  /* SEMIC */
     181             :     {
     182             :         0, 0, 0
     183             :     },                                  /* CBRA */
     184             :     {
     185             :         0, 0, 0
     186             :     },                                  /* CKET */
     187             :     {
     188             :         0, 0, 0
     189             :     },                                  /* ASPLUS */
     190             :     {
     191             :         0, 0, 0
     192             :     },                                  /* ASMINUS */
     193             :     {
     194             :         0, 0, 0
     195             :     },                                  /* ASSTAR */
     196             :     {
     197             :         0, 0, 0
     198             :     },                                  /* ASSLASH */
     199             :     {
     200             :         0, 0, 0
     201             :     },                                  /* ASPCT */
     202             :     {
     203             :         0, 0, 0
     204             :     },                                  /* ASCIRC */
     205             :     {
     206             :         0, 0, 0
     207             :     },                                  /* ASLSH */
     208             :     {
     209             :         0, 0, 0
     210             :     },                                  /* ASRSH */
     211             :     {
     212             :         0, 0, 0
     213             :     },                                  /* ASOR */
     214             :     {
     215             :         0, 0, 0
     216             :     },                                  /* ASAND */
     217             :     {
     218             :         0, 0, 0
     219             :     },                                  /* ELLIPS */
     220             :     {
     221             :         0, 0, 0
     222             :     },                                  /* DSHARP1 */
     223             :     {
     224             :         0, 0, 0
     225             :     },                                  /* NAME1 */
     226             :     {
     227             :         0, 0, 0
     228             :     },                                  /* NAME2 */
     229             :     {
     230             :         16, 1, UNARY
     231             :     },                                  /* DEFINED */
     232             :     {
     233             :         16, 0, UNARY
     234             :     },                                  /* UMINUS */
     235             :     {
     236             :         16, 1, UNARY
     237             :     },                                  /* ARCHITECTURE */
     238             : };
     239             : 
     240             : int evalop(struct pri);
     241             : struct value tokval(Token *);
     242             : struct value vals[NSTAK], *vp;
     243             : enum toktype ops[NSTAK], *op;
     244             : 
     245             : /*
     246             :  * Evaluate an #if #elif #ifdef #ifndef line.  trp->tp points to the keyword.
     247             :  */
     248             : long
     249        2518 :     eval(Tokenrow * trp, int kw)
     250             : {
     251             :     Token *tp;
     252             :     Nlist *np;
     253             :     size_t  ntok;
     254             :     int rnd;
     255             : 
     256        2518 :     trp->tp++;
     257        2518 :     if (kw == KIFDEF || kw == KIFNDEF)
     258             :     {
     259        1694 :         if (trp->lp - trp->bp != 4 || trp->tp->type != NAME)
     260             :         {
     261           0 :             error(ERROR, "Syntax error in #ifdef/#ifndef");
     262           0 :             return 0;
     263             :         }
     264        1694 :         np = lookup(trp->tp, 0);
     265        1694 :         return (kw == KIFDEF) == (np && np->flag & (ISDEFINED | ISMAC));
     266             :     }
     267         824 :     ntok = trp->tp - trp->bp;
     268         824 :     kwdefined->val = KDEFINED;          /* activate special meaning of
     269             :                                          * defined */
     270         824 :     expandrow(trp, "<if>");
     271         824 :     kwdefined->val = NAME;
     272         824 :     vp = vals;
     273         824 :     op = ops;
     274         824 :     *op++ = END;
     275        3920 :     for (rnd = 0, tp = trp->bp + ntok; tp < trp->lp; tp++)
     276             :     {
     277        3096 :         switch (tp->type)
     278             :         {
     279             :             case WS:
     280             :             case NL:
     281         824 :                 continue;
     282             : 
     283             :                 /* nilary */
     284             :             case NAME:
     285             :             case NAME1:
     286             :             case NAME2:
     287             :             case NUMBER:
     288             :             case CCON:
     289             :             case STRING:
     290         900 :                 if (rnd)
     291           0 :                     goto syntax;
     292         900 :                 *vp++ = tokval(tp);
     293         900 :                 rnd = 1;
     294         900 :                 continue;
     295             : 
     296             :                 /* unary */
     297             :             case DEFINED:
     298             :             case TILDE:
     299             :             case NOT:
     300         968 :                 if (rnd)
     301           0 :                     goto syntax;
     302         968 :                 *op++ = tp->type;
     303         968 :                 continue;
     304             : 
     305             :                 /* unary-binary */
     306             :             case PLUS:
     307             :             case MINUS:
     308             :             case STAR:
     309             :             case AND:
     310           0 :                 if (rnd == 0)
     311             :                 {
     312           0 :                     if (tp->type == MINUS)
     313           0 :                         *op++ = UMINUS;
     314           0 :                     if (tp->type == STAR || tp->type == AND)
     315             :                     {
     316           0 :                         error(ERROR, "Illegal operator * or & in #if/#elif");
     317           0 :                         return 0;
     318             :                     }
     319           0 :                     continue;
     320             :                 }
     321             :                 /* flow through */
     322             : 
     323             :                 /* plain binary */
     324             :             case EQ:
     325             :             case NEQ:
     326             :             case LEQ:
     327             :             case GEQ:
     328             :             case LSH:
     329             :             case RSH:
     330             :             case LAND:
     331             :             case LOR:
     332             :             case SLASH:
     333             :             case PCT:
     334             :             case LT:
     335             :             case GT:
     336             :             case CIRC:
     337             :             case OR:
     338             :             case QUEST:
     339             :             case COLON:
     340             :             case COMMA:
     341          76 :                 if (rnd == 0)
     342           0 :                     goto syntax;
     343          76 :                 if (evalop(priority[tp->type]) != 0)
     344           0 :                     return 0;
     345          76 :                 *op++ = tp->type;
     346          76 :                 rnd = 0;
     347          76 :                 continue;
     348             : 
     349             :             case LP:
     350         164 :                 if (rnd)
     351           0 :                     goto syntax;
     352         164 :                 *op++ = LP;
     353         164 :                 continue;
     354             : 
     355             :             case RP:
     356         164 :                 if (!rnd)
     357           0 :                     goto syntax;
     358         164 :                 if (evalop(priority[RP]) != 0)
     359           0 :                     return 0;
     360         164 :                 if (op <= ops || op[-1] != LP)
     361             :                 {
     362             :                     goto syntax;
     363             :                 }
     364         164 :                 op--;
     365         164 :                 continue;
     366             : 
     367             :             case SHARP:
     368           0 :                 if ((tp + 1) < trp->lp)
     369             :                 {
     370           0 :                     np = lookup(tp + 1, 0);
     371           0 :                     if (np && (np->val == KMACHINE))
     372             :                     {
     373           0 :                         tp++;
     374           0 :                         if (rnd)
     375           0 :                             goto syntax;
     376           0 :                         *op++ = ARCHITECTURE;
     377           0 :                         continue;
     378             :                     }
     379             :                 }
     380             :                 /* fall through */
     381             : 
     382             :             default:
     383           0 :                 error(ERROR, "Bad operator (%t) in #if/#elif", tp);
     384           0 :                 return 0;
     385             :         }
     386             :     }
     387         824 :     if (rnd == 0)
     388           0 :         goto syntax;
     389         824 :     if (evalop(priority[END]) != 0)
     390           0 :         return 0;
     391         824 :     if (op != &ops[1] || vp != &vals[1])
     392             :     {
     393           0 :         error(ERROR, "Botch in #if/#elif");
     394           0 :         return 0;
     395             :     }
     396         824 :     if (vals[0].type == UND)
     397           0 :         error(ERROR, "Undefined expression value");
     398         824 :     return vals[0].val;
     399             : syntax:
     400           0 :     error(ERROR, "Syntax error in #if/#elif");
     401           0 :     return 0;
     402             : }
     403             : 
     404             : int
     405        1064 :     evalop(struct pri pri)
     406             : {
     407             :     struct value v1;
     408        1064 :     struct value v2 = { 0, UND };
     409             :     long rv1, rv2;
     410             :     int rtype, oper;
     411             : 
     412        1064 :     rv2 = 0;
     413        1064 :     rtype = 0;
     414        3172 :     while (pri.pri < priority[op[-1]].pri)
     415             :     {
     416        1044 :         oper = *--op;
     417        1044 :         if (priority[oper].arity == 2)
     418             :         {
     419          76 :             v2 = *--vp;
     420          76 :             rv2 = v2.val;
     421             :         }
     422        1044 :         v1 = *--vp;
     423        1044 :         rv1 = v1.val;
     424             : /*lint -e574 -e644 */
     425        1044 :         switch (priority[oper].ctype)
     426             :         {
     427             :             case 0:
     428             :             default:
     429           0 :                 error(WARNING, "Syntax error in #if/#endif");
     430           0 :                 return 1;
     431             :             case ARITH:
     432             :             case RELAT:
     433           0 :                 if (v1.type == UNS || v2.type == UNS)
     434           0 :                     rtype = UNS;
     435             :                 else
     436           0 :                     rtype = SGN;
     437           0 :                 if (v1.type == UND || v2.type == UND)
     438           0 :                     rtype = UND;
     439           0 :                 if (priority[oper].ctype == RELAT && rtype == UNS)
     440             :                 {
     441           0 :                     oper |= UNSMARK;
     442           0 :                     rtype = SGN;
     443             :                 }
     444           0 :                 break;
     445             :             case SHIFT:
     446           0 :                 if (v1.type == UND || v2.type == UND)
     447           0 :                     rtype = UND;
     448             :                 else
     449           0 :                     rtype = v1.type;
     450           0 :                 if (rtype == UNS)
     451           0 :                     oper |= UNSMARK;
     452           0 :                 break;
     453             :             case UNARY:
     454         968 :                 rtype = v1.type;
     455         968 :                 break;
     456             :             case LOGIC:
     457             :             case SPCL:
     458          76 :                 break;
     459             :         }
     460        1044 :         switch (oper)
     461             :         {
     462             :             case EQ:
     463             :             case EQ | UNSMARK:
     464           0 :                 rv1 = rv1 == rv2;
     465           0 :                 break;
     466             :             case NEQ:
     467             :             case NEQ | UNSMARK:
     468           0 :                 rv1 = rv1 != rv2;
     469           0 :                 break;
     470             :             case LEQ:
     471           0 :                 rv1 = rv1 <= rv2;
     472           0 :                 break;
     473             :             case GEQ:
     474           0 :                 rv1 = rv1 >= rv2;
     475           0 :                 break;
     476             :             case LT:
     477           0 :                 rv1 = rv1 < rv2;
     478           0 :                 break;
     479             :             case GT:
     480           0 :                 rv1 = rv1 > rv2;
     481           0 :                 break;
     482             :             case LEQ | UNSMARK:
     483           0 :                 rv1 = (unsigned long)rv1 <= (unsigned long)rv2;
     484           0 :                 break;
     485             :             case GEQ | UNSMARK:
     486           0 :                 rv1 = (unsigned long)rv1 >= (unsigned long)rv2;
     487           0 :                 break;
     488             :             case LT | UNSMARK:
     489           0 :                 rv1 = (unsigned long)rv1 < (unsigned long)rv2;
     490           0 :                 break;
     491             :             case GT | UNSMARK:
     492           0 :                 rv1 = (unsigned long)rv1 > (unsigned long)rv2;
     493           0 :                 break;
     494             :             case LSH:
     495           0 :                 rv1 <<= rv2;
     496           0 :                 break;
     497             :             case LSH | UNSMARK:
     498           0 :                 rv1 = (unsigned long) rv1 << rv2;
     499           0 :                 break;
     500             :             case RSH:
     501           0 :                 rv1 >>= rv2;
     502           0 :                 break;
     503             :             case RSH | UNSMARK:
     504           0 :                 rv1 = (unsigned long) rv1 >> rv2;
     505           0 :                 break;
     506             :             case LAND:
     507          62 :                 rtype = UND;
     508          62 :                 if (v1.type == UND)
     509           0 :                     break;
     510          62 :                 if (rv1 != 0)
     511             :                 {
     512          50 :                     if (v2.type == UND)
     513           0 :                         break;
     514          50 :                     rv1 = rv2 != 0;
     515             :                 }
     516             :                 else
     517          12 :                     rv1 = 0;
     518          62 :                 rtype = SGN;
     519          62 :                 break;
     520             :             case LOR:
     521          14 :                 rtype = UND;
     522          14 :                 if (v1.type == UND)
     523           0 :                     break;
     524          14 :                 if (rv1 == 0)
     525             :                 {
     526          10 :                     if (v2.type == UND)
     527           0 :                         break;
     528          10 :                     rv1 = rv2 != 0;
     529             :                 }
     530             :                 else
     531           4 :                     rv1 = 1;
     532          14 :                 rtype = SGN;
     533          14 :                 break;
     534             :             case AND:
     535           0 :                 rv1 &= rv2;
     536           0 :                 break;
     537             :             case STAR:
     538           0 :                 rv1 *= rv2;
     539           0 :                 break;
     540             :             case PLUS:
     541           0 :                 rv1 += rv2;
     542           0 :                 break;
     543             :             case MINUS:
     544           0 :                 rv1 -= rv2;
     545           0 :                 break;
     546             :             case UMINUS:
     547           0 :                 if (v1.type == UND)
     548           0 :                     rtype = UND;
     549           0 :                 rv1 = -rv1;
     550           0 :                 break;
     551             :             case OR:
     552           0 :                 rv1 |= rv2;
     553           0 :                 break;
     554             :             case CIRC:
     555           0 :                 rv1 ^= rv2;
     556           0 :                 break;
     557             :             case TILDE:
     558           0 :                 rv1 = ~rv1;
     559           0 :                 break;
     560             :             case NOT:
     561          68 :                 rv1 = !rv1;
     562          68 :                 if (rtype != UND)
     563          68 :                     rtype = SGN;
     564          68 :                 break;
     565             :             case SLASH:
     566           0 :                 if (rv2 == 0)
     567             :                 {
     568           0 :                     rtype = UND;
     569           0 :                     break;
     570             :                 }
     571           0 :                 if (rtype == UNS)
     572           0 :                     rv1 /= (unsigned long) rv2;
     573             :                 else
     574           0 :                     rv1 /= rv2;
     575           0 :                 break;
     576             :             case PCT:
     577           0 :                 if (rv2 == 0)
     578             :                 {
     579           0 :                     rtype = UND;
     580           0 :                     break;
     581             :                 }
     582           0 :                 if (rtype == UNS)
     583           0 :                     rv1 %= (unsigned long) rv2;
     584             :                 else
     585           0 :                     rv1 %= rv2;
     586           0 :                 break;
     587             :             case COLON:
     588           0 :                 if (op[-1] != QUEST)
     589           0 :                     error(ERROR, "Bad ?: in #if/endif");
     590             :                 else
     591             :                 {
     592           0 :                     op--;
     593           0 :                     if ((--vp)->val == 0)
     594           0 :                         v1 = v2;
     595           0 :                     rtype = v1.type;
     596           0 :                     rv1 = v1.val;
     597             :                 }
     598           0 :                 break;
     599             : 
     600             :             case DEFINED:
     601             :             case ARCHITECTURE:
     602         900 :                 break;
     603             : 
     604             :             default:
     605           0 :                 error(ERROR, "Eval botch (unknown operator)");
     606           0 :                 return 1;
     607             :         }
     608             : /*lint +e574 +e644 */
     609        1044 :         v1.val = rv1;
     610        1044 :         v1.type = rtype;
     611        1044 :         *vp++ = v1;
     612             :     }
     613        1064 :     return 0;
     614             : }
     615             : 
     616             : struct value
     617         900 :     tokval(Token * tp)
     618             : {
     619             :     struct value v;
     620             :     Nlist *np;
     621             :     int i, base;
     622             :     unsigned long n;
     623             :     uchar *p, c;
     624             : 
     625         900 :     v.type = SGN;
     626         900 :     v.val = 0;
     627         900 :     switch (tp->type)
     628             :     {
     629             : 
     630             :         case NAME:
     631           0 :             v.val = 0;
     632           0 :             break;
     633             : 
     634             :         case NAME1:
     635         900 :             if ((np = lookup(tp, 0)) != NULL && np->flag & (ISDEFINED | ISMAC))
     636          80 :                 v.val = 1;
     637         900 :             break;
     638             : 
     639             :         case NAME2:
     640           0 :             if ((np = lookup(tp, 0)) != NULL && np->flag & (ISARCHITECTURE))
     641           0 :                 v.val = 1;
     642           0 :             break;
     643             : 
     644             :         case NUMBER:
     645           0 :             n = 0;
     646           0 :             base = 10;
     647           0 :             p = tp->t;
     648           0 :             c = p[tp->len];
     649           0 :             p[tp->len] = '\0';
     650           0 :             if (*p == '0')
     651             :             {
     652           0 :                 base = 8;
     653           0 :                 if (p[1] == 'x' || p[1] == 'X')
     654             :                 {
     655           0 :                     base = 16;
     656           0 :                     p++;
     657             :                 }
     658           0 :                 p++;
     659             :             }
     660           0 :             for (;; p++)
     661             :             {
     662           0 :                 if ((i = digit(*p)) < 0)
     663           0 :                     break;
     664           0 :                 if (i >= base)
     665           0 :                     error(WARNING,
     666             :                           "Bad digit in number %t", tp);
     667           0 :                 n *= base;
     668           0 :                 n += i;
     669           0 :             }
     670           0 :             if (n >= 0x80000000 && base != 10)
     671           0 :                 v.type = UNS;
     672           0 :             for (; *p; p++)
     673             :             {
     674           0 :                 if (*p == 'u' || *p == 'U')
     675           0 :                     v.type = UNS;
     676             :                 else
     677           0 :                     if (*p == 'l' || *p == 'L')
     678             :                         ;
     679             :                     else
     680             :                     {
     681           0 :                         error(ERROR,
     682             :                               "Bad number %t in #if/#elif", tp);
     683           0 :                         break;
     684             :                     }
     685             :             }
     686           0 :             v.val = n;
     687           0 :             tp->t[tp->len] = c;
     688           0 :             break;
     689             : 
     690             :         case CCON:
     691           0 :             n = 0;
     692           0 :             p = tp->t;
     693           0 :             if (*p == 'L')
     694             :             {
     695           0 :                 p += 1;
     696           0 :                 error(WARNING, "Wide char constant value undefined");
     697             :             }
     698           0 :             p += 1;
     699           0 :             if (*p == '\\')
     700             :             {
     701           0 :                 p += 1;
     702           0 :                 if ((i = digit(*p)) >= 0 && i <= 7)
     703             :                 {
     704           0 :                     n = i;
     705           0 :                     p += 1;
     706           0 :                     if ((i = digit(*p)) >= 0 && i <= 7)
     707             :                     {
     708           0 :                         p += 1;
     709           0 :                         n <<= 3;
     710           0 :                         n += i;
     711           0 :                         if ((i = digit(*p)) >= 0 && i <= 7)
     712             :                         {
     713           0 :                             p += 1;
     714           0 :                             n <<= 3;
     715           0 :                             n += i;
     716             :                         }
     717             :                     }
     718             :                 }
     719             :                 else
     720           0 :                     if (*p == 'x')
     721             :                     {
     722           0 :                         p += 1;
     723           0 :                         while ((i = digit(*p)) >= 0 && i <= 15)
     724             :                         {
     725           0 :                             p += 1;
     726           0 :                             n <<= 4;
     727           0 :                             n += i;
     728             :                         }
     729             :                     }
     730             :                     else
     731             :                     {
     732             :                         static const char cvcon[] = "b\bf\fn\nr\rt\tv\v''\"\"??\\\\";
     733             :                         static size_t cvlen = sizeof(cvcon) - 1;
     734             : 
     735             :                         size_t j;
     736           0 :                         for (j = 0; j < cvlen; j += 2)
     737             :                         {
     738           0 :                             if (*p == cvcon[j])
     739             :                             {
     740           0 :                                 n = cvcon[j + 1];
     741           0 :                                 break;
     742             :                             }
     743             :                         }
     744           0 :                         p += 1;
     745           0 :                         if (j >= cvlen)
     746           0 :                             error(WARNING,
     747             :                                "Undefined escape in character constant");
     748             :                     }
     749             :             }
     750             :             else
     751           0 :                 if (*p == '\'')
     752           0 :                     error(ERROR, "Empty character constant");
     753             :                 else
     754           0 :                     n = *p++;
     755           0 :             if (*p != '\'')
     756           0 :                 error(WARNING, "Multibyte character constant undefined");
     757             :             else
     758           0 :                 if (n > 127)
     759           0 :                     error(WARNING, "Character constant taken as not signed");
     760           0 :             v.val = n;
     761           0 :             break;
     762             : 
     763             :         case STRING:
     764           0 :             error(ERROR, "String in #if/#elif");
     765           0 :             break;
     766             :     }
     767         900 :     return v;
     768             : }
     769             : 
     770             : int
     771           0 :     digit(int i)
     772             : {
     773           0 :     if ('0' <= i && i <= '9')
     774           0 :         i -= '0';
     775             :     else
     776           0 :         if ('a' <= i && i <= 'f')
     777           0 :             i -= 'a' - 10;
     778             :         else
     779           0 :             if ('A' <= i && i <= 'F')
     780           0 :                 i -= 'A' - 10;
     781             :             else
     782           0 :                 i = -1;
     783           0 :     return i;
     784             : }
     785             : 
     786             : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */

Generated by: LCOV version 1.10