plural.y 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383
  1. %{
  2. /* Expression parsing for plural form selection.
  3. Copyright (C) 2000-2001, 2003, 2005 Free Software Foundation, Inc.
  4. Written by Ulrich Drepper <drepper@cygnus.com>, 2000.
  5. This program is free software; you can redistribute it and/or modify it
  6. under the terms of the GNU Library General Public License as published
  7. by the Free Software Foundation; either version 2, or (at your option)
  8. any later version.
  9. This program is distributed in the hope that it will be useful,
  10. but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  12. Library General Public License for more details.
  13. You should have received a copy of the GNU Library General Public
  14. License along with this program; if not, write to the Free Software
  15. Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301,
  16. USA. */
  17. /* For bison < 2.0, the bison generated parser uses alloca. AIX 3 forces us
  18. to put this declaration at the beginning of the file. The declaration in
  19. bison's skeleton file comes too late. This must come before <config.h>
  20. because <config.h> may include arbitrary system headers.
  21. This can go away once the AM_INTL_SUBDIR macro requires bison >= 2.0. */
  22. #if defined _AIX && !defined __GNUC__
  23. #pragma alloca
  24. #endif
  25. #ifdef HAVE_CONFIG_H
  26. # include <config.h>
  27. #endif
  28. #include <stddef.h>
  29. #include <stdlib.h>
  30. #include <string.h>
  31. #include "plural-exp.h"
  32. /* The main function generated by the parser is called __gettextparse,
  33. but we want it to be called PLURAL_PARSE. */
  34. #ifndef _LIBC
  35. # define __gettextparse PLURAL_PARSE
  36. #endif
  37. #define YYLEX_PARAM &((struct parse_args *) arg)->cp
  38. #define YYPARSE_PARAM arg
  39. %}
  40. %pure_parser
  41. %expect 7
  42. %union {
  43. unsigned long int num;
  44. enum operator op;
  45. struct expression *exp;
  46. }
  47. %{
  48. /* Prototypes for local functions. */
  49. static int yylex (YYSTYPE *lval, const char **pexp);
  50. static void yyerror (const char *str);
  51. /* Allocation of expressions. */
  52. static struct expression *
  53. new_exp (int nargs, enum operator op, struct expression * const *args)
  54. {
  55. int i;
  56. struct expression *newp;
  57. /* If any of the argument could not be malloc'ed, just return NULL. */
  58. for (i = nargs - 1; i >= 0; i--)
  59. if (args[i] == NULL)
  60. goto fail;
  61. /* Allocate a new expression. */
  62. newp = (struct expression *) malloc (sizeof (*newp));
  63. if (newp != NULL)
  64. {
  65. newp->nargs = nargs;
  66. newp->operation = op;
  67. for (i = nargs - 1; i >= 0; i--)
  68. newp->val.args[i] = args[i];
  69. return newp;
  70. }
  71. fail:
  72. for (i = nargs - 1; i >= 0; i--)
  73. FREE_EXPRESSION (args[i]);
  74. return NULL;
  75. }
  76. static inline struct expression *
  77. new_exp_0 (enum operator op)
  78. {
  79. return new_exp (0, op, NULL);
  80. }
  81. static inline struct expression *
  82. new_exp_1 (enum operator op, struct expression *right)
  83. {
  84. struct expression *args[1];
  85. args[0] = right;
  86. return new_exp (1, op, args);
  87. }
  88. static struct expression *
  89. new_exp_2 (enum operator op, struct expression *left, struct expression *right)
  90. {
  91. struct expression *args[2];
  92. args[0] = left;
  93. args[1] = right;
  94. return new_exp (2, op, args);
  95. }
  96. static inline struct expression *
  97. new_exp_3 (enum operator op, struct expression *bexp,
  98. struct expression *tbranch, struct expression *fbranch)
  99. {
  100. struct expression *args[3];
  101. args[0] = bexp;
  102. args[1] = tbranch;
  103. args[2] = fbranch;
  104. return new_exp (3, op, args);
  105. }
  106. %}
  107. /* This declares that all operators have the same associativity and the
  108. precedence order as in C. See [Harbison, Steele: C, A Reference Manual].
  109. There is no unary minus and no bitwise operators.
  110. Operators with the same syntactic behaviour have been merged into a single
  111. token, to save space in the array generated by bison. */
  112. %right '?' /* ? */
  113. %left '|' /* || */
  114. %left '&' /* && */
  115. %left EQUOP2 /* == != */
  116. %left CMPOP2 /* < > <= >= */
  117. %left ADDOP2 /* + - */
  118. %left MULOP2 /* * / % */
  119. %right '!' /* ! */
  120. %token <op> EQUOP2 CMPOP2 ADDOP2 MULOP2
  121. %token <num> NUMBER
  122. %type <exp> exp
  123. %%
  124. start: exp
  125. {
  126. if ($1 == NULL)
  127. YYABORT;
  128. ((struct parse_args *) arg)->res = $1;
  129. }
  130. ;
  131. exp: exp '?' exp ':' exp
  132. {
  133. $$ = new_exp_3 (qmop, $1, $3, $5);
  134. }
  135. | exp '|' exp
  136. {
  137. $$ = new_exp_2 (lor, $1, $3);
  138. }
  139. | exp '&' exp
  140. {
  141. $$ = new_exp_2 (land, $1, $3);
  142. }
  143. | exp EQUOP2 exp
  144. {
  145. $$ = new_exp_2 ($2, $1, $3);
  146. }
  147. | exp CMPOP2 exp
  148. {
  149. $$ = new_exp_2 ($2, $1, $3);
  150. }
  151. | exp ADDOP2 exp
  152. {
  153. $$ = new_exp_2 ($2, $1, $3);
  154. }
  155. | exp MULOP2 exp
  156. {
  157. $$ = new_exp_2 ($2, $1, $3);
  158. }
  159. | '!' exp
  160. {
  161. $$ = new_exp_1 (lnot, $2);
  162. }
  163. | 'n'
  164. {
  165. $$ = new_exp_0 (var);
  166. }
  167. | NUMBER
  168. {
  169. if (($$ = new_exp_0 (num)) != NULL)
  170. $$->val.num = $1;
  171. }
  172. | '(' exp ')'
  173. {
  174. $$ = $2;
  175. }
  176. ;
  177. %%
  178. void
  179. internal_function
  180. FREE_EXPRESSION (struct expression *exp)
  181. {
  182. if (exp == NULL)
  183. return;
  184. /* Handle the recursive case. */
  185. switch (exp->nargs)
  186. {
  187. case 3:
  188. FREE_EXPRESSION (exp->val.args[2]);
  189. /* FALLTHROUGH */
  190. case 2:
  191. FREE_EXPRESSION (exp->val.args[1]);
  192. /* FALLTHROUGH */
  193. case 1:
  194. FREE_EXPRESSION (exp->val.args[0]);
  195. /* FALLTHROUGH */
  196. default:
  197. break;
  198. }
  199. free (exp);
  200. }
  201. static int
  202. yylex (YYSTYPE *lval, const char **pexp)
  203. {
  204. const char *exp = *pexp;
  205. int result;
  206. while (1)
  207. {
  208. if (exp[0] == '\0')
  209. {
  210. *pexp = exp;
  211. return YYEOF;
  212. }
  213. if (exp[0] != ' ' && exp[0] != '\t')
  214. break;
  215. ++exp;
  216. }
  217. result = *exp++;
  218. switch (result)
  219. {
  220. case '0': case '1': case '2': case '3': case '4':
  221. case '5': case '6': case '7': case '8': case '9':
  222. {
  223. unsigned long int n = result - '0';
  224. while (exp[0] >= '0' && exp[0] <= '9')
  225. {
  226. n *= 10;
  227. n += exp[0] - '0';
  228. ++exp;
  229. }
  230. lval->num = n;
  231. result = NUMBER;
  232. }
  233. break;
  234. case '=':
  235. if (exp[0] == '=')
  236. {
  237. ++exp;
  238. lval->op = equal;
  239. result = EQUOP2;
  240. }
  241. else
  242. result = YYERRCODE;
  243. break;
  244. case '!':
  245. if (exp[0] == '=')
  246. {
  247. ++exp;
  248. lval->op = not_equal;
  249. result = EQUOP2;
  250. }
  251. break;
  252. case '&':
  253. case '|':
  254. if (exp[0] == result)
  255. ++exp;
  256. else
  257. result = YYERRCODE;
  258. break;
  259. case '<':
  260. if (exp[0] == '=')
  261. {
  262. ++exp;
  263. lval->op = less_or_equal;
  264. }
  265. else
  266. lval->op = less_than;
  267. result = CMPOP2;
  268. break;
  269. case '>':
  270. if (exp[0] == '=')
  271. {
  272. ++exp;
  273. lval->op = greater_or_equal;
  274. }
  275. else
  276. lval->op = greater_than;
  277. result = CMPOP2;
  278. break;
  279. case '*':
  280. lval->op = mult;
  281. result = MULOP2;
  282. break;
  283. case '/':
  284. lval->op = divide;
  285. result = MULOP2;
  286. break;
  287. case '%':
  288. lval->op = module;
  289. result = MULOP2;
  290. break;
  291. case '+':
  292. lval->op = plus;
  293. result = ADDOP2;
  294. break;
  295. case '-':
  296. lval->op = minus;
  297. result = ADDOP2;
  298. break;
  299. case 'n':
  300. case '?':
  301. case ':':
  302. case '(':
  303. case ')':
  304. /* Nothing, just return the character. */
  305. break;
  306. case ';':
  307. case '\n':
  308. case '\0':
  309. /* Be safe and let the user call this function again. */
  310. --exp;
  311. result = YYEOF;
  312. break;
  313. default:
  314. result = YYERRCODE;
  315. #if YYDEBUG != 0
  316. --exp;
  317. #endif
  318. break;
  319. }
  320. *pexp = exp;
  321. return result;
  322. }
  323. static void
  324. yyerror (const char *str)
  325. {
  326. /* Do nothing. We don't print error messages here. */
  327. }