blob: 5afc6011f3bd4b07a33eec8406968bb914ba036a [file] [log] [blame]
David Reissea2cba82009-03-30 21:35:00 +00001/*
2 * Licensed to the Apache Software Foundation (ASF) under one
3 * or more contributor license agreements. See the NOTICE file
4 * distributed with this work for additional information
5 * regarding copyright ownership. The ASF licenses this file
6 * to you under the Apache License, Version 2.0 (the
7 * "License"); you may not use this file except in compliance
8 * with the License. You may obtain a copy of the License at
Mark Sleee9ce01c2007-05-16 02:29:53 +00009 *
David Reissea2cba82009-03-30 21:35:00 +000010 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing,
13 * software distributed under the License is distributed on an
14 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15 * KIND, either express or implied. See the License for the
16 * specific language governing permissions and limitations
17 * under the License.
Mark Sleee9ce01c2007-05-16 02:29:53 +000018 */
19
20/**
Mark Slee31985722006-05-24 21:45:31 +000021 * Thrift scanner.
Mark Slee27ed6ec2007-08-16 01:26:31 +000022 *
Mark Slee31985722006-05-24 21:45:31 +000023 * Tokenizes a thrift definition file.
Mark Slee31985722006-05-24 21:45:31 +000024 */
Mark Sleef5377b32006-10-10 01:42:59 +000025
Mark Slee31985722006-05-24 21:45:31 +000026%{
27
Christian Lavoieaf65f1b2010-11-24 21:58:05 +000028/* This is redundant with some of the flags in Makefile.am, but it works
29 * when people override CXXFLAGS without being careful. The pragmas are
30 * the 'right' way to do it, but don't work on old-enough GCC (in particular
31 * the GCC that ship on Mac OS X 10.6.5, *counter* to what the GNU docs say)
32 *
33 * We should revert the Makefile.am changes once Apple ships a reasonable
34 * GCC.
35 */
Ben Craige9576752013-10-11 08:19:16 -050036#ifdef __GNUC__
Roger Meier3b771a12010-11-17 22:11:26 +000037#pragma GCC diagnostic ignored "-Wunused-function"
38#pragma GCC diagnostic ignored "-Wunused-label"
Ben Craige9576752013-10-11 08:19:16 -050039#endif
40
41#ifdef _MSC_VER
42//warning C4102: 'find_rule' : unreferenced label
43#pragma warning(disable:4102)
44//avoid isatty redefinition
45#define YY_NEVER_INTERACTIVE 1
46#endif
Roger Meier3b771a12010-11-17 22:11:26 +000047
Jens Geyer8cd3efe2013-09-16 22:17:52 +020048#include <cassert>
David Reiss82e6fc02009-03-26 23:32:36 +000049#include <string>
David Reissf1454162008-06-30 20:45:47 +000050#include <errno.h>
Roger Meier9212e792012-06-12 21:01:06 +000051#include <stdlib.h>
David Reissf1454162008-06-30 20:45:47 +000052
Ben Craige9576752013-10-11 08:19:16 -050053#ifdef _MSC_VER
54#include "windows/config.h"
Roger Meier57e6de42014-07-16 10:19:59 +020055#endif
Mark Slee31985722006-05-24 21:45:31 +000056#include "main.h"
David Reisscbd4bac2007-08-14 17:12:33 +000057#include "globals.h"
Mark Slee31985722006-05-24 21:45:31 +000058#include "parse/t_program.h"
59
Mark Sleef5377b32006-10-10 01:42:59 +000060/**
61 * Must be included AFTER parse/t_program.h, but I can't remember why anymore
62 * because I wrote this a while ago.
63 */
jfarrell4f54d132014-07-10 09:23:43 -040064#if defined(BISON_USE_PARSER_H_EXTENSION)
jfarrell92f24b22013-08-17 15:47:13 -040065#include "thrifty.h"
jfarrell4f54d132014-07-10 09:23:43 -040066#else
67#include "thrifty.hh"
68#endif
Mark Slee31985722006-05-24 21:45:31 +000069
Mark Sleef12865a2007-01-12 00:23:26 +000070void thrift_reserved_keyword(char* keyword) {
71 yyerror("Cannot use reserved language keyword: \"%s\"\n", keyword);
72 exit(1);
73}
74
David Reissf1454162008-06-30 20:45:47 +000075void integer_overflow(char* text) {
76 yyerror("This integer is too big: \"%s\"\n", text);
77 exit(1);
78}
79
Bryan Duxbury235f8b52011-08-19 18:27:47 +000080void unexpected_token(char* text) {
81 yyerror("Unexpected token in input: \"%s\"\n", text);
82 exit(1);
83}
84
Mark Slee31985722006-05-24 21:45:31 +000085%}
86
Mark Sleef5377b32006-10-10 01:42:59 +000087/**
88 * Provides the yylineno global, useful for debugging output
89 */
Mark Slee27ed6ec2007-08-16 01:26:31 +000090%option lex-compat
Mark Slee31985722006-05-24 21:45:31 +000091
Mark Slee27ed6ec2007-08-16 01:26:31 +000092/**
David Reiss4563acd2010-08-31 16:51:29 +000093 * Our inputs are all single files, so no need for yywrap
94 */
95%option noyywrap
96
97/**
Christian Lavoie77215d82010-11-07 19:42:48 +000098 * We don't use it, and it fires up warnings at -Wall
99 */
100%option nounput
101
102/**
Mark Sleef5377b32006-10-10 01:42:59 +0000103 * Helper definitions, comments, constants, and whatnot
104 */
105
Mark Sleebd588222007-11-21 08:43:35 +0000106intconstant ([+-]?[0-9]+)
107hexconstant ("0x"[0-9A-Fa-f]+)
108dubconstant ([+-]?[0-9]*(\.[0-9]+)?([eE][+-]?[0-9]+)?)
Carl Yeksigiande074082013-06-04 04:28:31 -0400109identifier ([a-zA-Z_](\.[a-zA-Z_0-9]|[a-zA-Z_0-9])*)
Mark Sleebd588222007-11-21 08:43:35 +0000110whitespace ([ \t\r\n]*)
111sillycomm ("/*""*"*"*/")
112multicomm ("/*"[^*]"/"*([^*/]|[^*]"/"|"*"[^/])*"*"*"*/")
113doctext ("/**"([^*/]|[^*]"/"|"*"[^/])*"*"*"*/")
114comment ("//"[^\n]*)
115unixcomment ("#"[^\n]*)
116symbol ([:;\,\{\}\(\)\=<>\[\]])
Carl Yeksigiande074082013-06-04 04:28:31 -0400117st_identifier ([a-zA-Z-](\.[a-zA-Z_0-9-]|[a-zA-Z_0-9-])*)
David Reiss82e6fc02009-03-26 23:32:36 +0000118literal_begin (['\"])
Mark Slee31985722006-05-24 21:45:31 +0000119
120%%
121
Mark Sleebd588222007-11-21 08:43:35 +0000122{whitespace} { /* do nothing */ }
123{sillycomm} { /* do nothing */ }
124{multicomm} { /* do nothing */ }
125{comment} { /* do nothing */ }
126{unixcomment} { /* do nothing */ }
Mark Slee31985722006-05-24 21:45:31 +0000127
Mark Sleebd588222007-11-21 08:43:35 +0000128{symbol} { return yytext[0]; }
Roger Meier0c3c8952011-08-22 21:38:16 +0000129"*" { return yytext[0]; }
Mark Slee9cb7c612006-09-01 22:17:45 +0000130
Bryan Duxbury6c928f32011-10-13 21:32:52 +0000131"false" { yylval.iconst=0; return tok_int_constant; }
132"true" { yylval.iconst=1; return tok_int_constant; }
133
Mark Sleebd588222007-11-21 08:43:35 +0000134"namespace" { return tok_namespace; }
135"cpp_namespace" { return tok_cpp_namespace; }
136"cpp_include" { return tok_cpp_include; }
137"cpp_type" { return tok_cpp_type; }
138"java_package" { return tok_java_package; }
139"cocoa_prefix" { return tok_cocoa_prefix; }
David Reiss7f42bcf2008-01-11 20:59:12 +0000140"csharp_namespace" { return tok_csharp_namespace; }
Jake Farrell7ae13e12011-10-18 14:35:26 +0000141"delphi_namespace" { return tok_delphi_namespace; }
Mark Sleebd588222007-11-21 08:43:35 +0000142"php_namespace" { return tok_php_namespace; }
143"py_module" { return tok_py_module; }
144"perl_package" { return tok_perl_package; }
145"ruby_namespace" { return tok_ruby_namespace; }
146"smalltalk_category" { return tok_smalltalk_category; }
David Reiss15457c92007-12-14 07:03:03 +0000147"smalltalk_prefix" { return tok_smalltalk_prefix; }
Mark Sleebd588222007-11-21 08:43:35 +0000148"xsd_all" { return tok_xsd_all; }
149"xsd_optional" { return tok_xsd_optional; }
150"xsd_nillable" { return tok_xsd_nillable; }
151"xsd_namespace" { return tok_xsd_namespace; }
152"xsd_attrs" { return tok_xsd_attrs; }
153"include" { return tok_include; }
154"void" { return tok_void; }
155"bool" { return tok_bool; }
156"byte" { return tok_byte; }
157"i16" { return tok_i16; }
158"i32" { return tok_i32; }
159"i64" { return tok_i64; }
160"double" { return tok_double; }
161"string" { return tok_string; }
162"binary" { return tok_binary; }
Jens Geyer0ca234f2013-06-04 22:01:47 +0200163"slist" {
164 pwarning(0, "\"slist\" is deprecated and will be removed in a future compiler version. This type should be replaced with \"string\".\n");
165 return tok_slist;
166}
Carl Yeksigianc3178522013-06-07 12:31:13 -0400167"senum" {
168 pwarning(0, "\"senum\" is deprecated and will be removed in a future compiler version. This type should be replaced with \"string\".\n");
169 return tok_senum;
170}
Mark Sleebd588222007-11-21 08:43:35 +0000171"map" { return tok_map; }
172"list" { return tok_list; }
173"set" { return tok_set; }
David Reisscecbed82009-03-24 20:02:22 +0000174"oneway" { return tok_oneway; }
Mark Sleebd588222007-11-21 08:43:35 +0000175"typedef" { return tok_typedef; }
176"struct" { return tok_struct; }
Bryan Duxburyab3666e2009-09-01 23:03:47 +0000177"union" { return tok_union; }
Mark Sleebd588222007-11-21 08:43:35 +0000178"exception" { return tok_xception; }
179"extends" { return tok_extends; }
180"throws" { return tok_throws; }
181"service" { return tok_service; }
182"enum" { return tok_enum; }
183"const" { return tok_const; }
184"required" { return tok_required; }
185"optional" { return tok_optional; }
David Reisscecbed82009-03-24 20:02:22 +0000186"async" {
187 pwarning(0, "\"async\" is deprecated. It is called \"oneway\" now.\n");
188 return tok_oneway;
189}
Jens Geyer885c6792014-05-02 21:31:55 +0200190"&" { return tok_reference; }
Mark Sleef0712dc2006-10-25 19:03:57 +0000191
Mark Slee52f643d2006-08-09 00:03:43 +0000192
Bryan Duxbury7f3285e2010-08-05 23:28:14 +0000193"BEGIN" { thrift_reserved_keyword(yytext); }
194"END" { thrift_reserved_keyword(yytext); }
195"__CLASS__" { thrift_reserved_keyword(yytext); }
196"__DIR__" { thrift_reserved_keyword(yytext); }
197"__FILE__" { thrift_reserved_keyword(yytext); }
198"__FUNCTION__" { thrift_reserved_keyword(yytext); }
199"__LINE__" { thrift_reserved_keyword(yytext); }
200"__METHOD__" { thrift_reserved_keyword(yytext); }
201"__NAMESPACE__" { thrift_reserved_keyword(yytext); }
Mark Sleebd588222007-11-21 08:43:35 +0000202"abstract" { thrift_reserved_keyword(yytext); }
Bryan Duxbury7f3285e2010-08-05 23:28:14 +0000203"alias" { thrift_reserved_keyword(yytext); }
Mark Sleebd588222007-11-21 08:43:35 +0000204"and" { thrift_reserved_keyword(yytext); }
Mark Sleec27fc312007-12-21 23:52:19 +0000205"args" { thrift_reserved_keyword(yytext); }
Mark Sleebd588222007-11-21 08:43:35 +0000206"as" { thrift_reserved_keyword(yytext); }
207"assert" { thrift_reserved_keyword(yytext); }
Bryan Duxbury7f3285e2010-08-05 23:28:14 +0000208"begin" { thrift_reserved_keyword(yytext); }
Mark Sleebd588222007-11-21 08:43:35 +0000209"break" { thrift_reserved_keyword(yytext); }
210"case" { thrift_reserved_keyword(yytext); }
Bryan Duxbury7f3285e2010-08-05 23:28:14 +0000211"catch" { thrift_reserved_keyword(yytext); }
Mark Sleebd588222007-11-21 08:43:35 +0000212"class" { thrift_reserved_keyword(yytext); }
Bryan Duxbury7f3285e2010-08-05 23:28:14 +0000213"clone" { thrift_reserved_keyword(yytext); }
Mark Sleebd588222007-11-21 08:43:35 +0000214"continue" { thrift_reserved_keyword(yytext); }
215"declare" { thrift_reserved_keyword(yytext); }
216"def" { thrift_reserved_keyword(yytext); }
217"default" { thrift_reserved_keyword(yytext); }
218"del" { thrift_reserved_keyword(yytext); }
219"delete" { thrift_reserved_keyword(yytext); }
220"do" { thrift_reserved_keyword(yytext); }
Bryan Duxbury7f3285e2010-08-05 23:28:14 +0000221"dynamic" { thrift_reserved_keyword(yytext); }
Mark Sleebd588222007-11-21 08:43:35 +0000222"elif" { thrift_reserved_keyword(yytext); }
223"else" { thrift_reserved_keyword(yytext); }
224"elseif" { thrift_reserved_keyword(yytext); }
Bryan Duxbury7f3285e2010-08-05 23:28:14 +0000225"elsif" { thrift_reserved_keyword(yytext); }
226"end" { thrift_reserved_keyword(yytext); }
227"enddeclare" { thrift_reserved_keyword(yytext); }
228"endfor" { thrift_reserved_keyword(yytext); }
229"endforeach" { thrift_reserved_keyword(yytext); }
230"endif" { thrift_reserved_keyword(yytext); }
231"endswitch" { thrift_reserved_keyword(yytext); }
232"endwhile" { thrift_reserved_keyword(yytext); }
233"ensure" { thrift_reserved_keyword(yytext); }
Mark Sleebd588222007-11-21 08:43:35 +0000234"except" { thrift_reserved_keyword(yytext); }
235"exec" { thrift_reserved_keyword(yytext); }
Mark Sleebd588222007-11-21 08:43:35 +0000236"finally" { thrift_reserved_keyword(yytext); }
237"float" { thrift_reserved_keyword(yytext); }
238"for" { thrift_reserved_keyword(yytext); }
239"foreach" { thrift_reserved_keyword(yytext); }
240"function" { thrift_reserved_keyword(yytext); }
241"global" { thrift_reserved_keyword(yytext); }
242"goto" { thrift_reserved_keyword(yytext); }
243"if" { thrift_reserved_keyword(yytext); }
244"implements" { thrift_reserved_keyword(yytext); }
245"import" { thrift_reserved_keyword(yytext); }
246"in" { thrift_reserved_keyword(yytext); }
247"inline" { thrift_reserved_keyword(yytext); }
248"instanceof" { thrift_reserved_keyword(yytext); }
249"interface" { thrift_reserved_keyword(yytext); }
250"is" { thrift_reserved_keyword(yytext); }
251"lambda" { thrift_reserved_keyword(yytext); }
Bryan Duxbury7f3285e2010-08-05 23:28:14 +0000252"module" { thrift_reserved_keyword(yytext); }
Mark Sleebd588222007-11-21 08:43:35 +0000253"native" { thrift_reserved_keyword(yytext); }
254"new" { thrift_reserved_keyword(yytext); }
Bryan Duxbury7f3285e2010-08-05 23:28:14 +0000255"next" { thrift_reserved_keyword(yytext); }
256"nil" { thrift_reserved_keyword(yytext); }
Mark Sleebd588222007-11-21 08:43:35 +0000257"not" { thrift_reserved_keyword(yytext); }
258"or" { thrift_reserved_keyword(yytext); }
259"pass" { thrift_reserved_keyword(yytext); }
260"public" { thrift_reserved_keyword(yytext); }
261"print" { thrift_reserved_keyword(yytext); }
262"private" { thrift_reserved_keyword(yytext); }
263"protected" { thrift_reserved_keyword(yytext); }
Bryan Duxbury7f3285e2010-08-05 23:28:14 +0000264"public" { thrift_reserved_keyword(yytext); }
Mark Sleebd588222007-11-21 08:43:35 +0000265"raise" { thrift_reserved_keyword(yytext); }
Bryan Duxbury7f3285e2010-08-05 23:28:14 +0000266"redo" { thrift_reserved_keyword(yytext); }
267"rescue" { thrift_reserved_keyword(yytext); }
268"retry" { thrift_reserved_keyword(yytext); }
Mark Sleef5a0b3d2009-08-13 19:21:40 +0000269"register" { thrift_reserved_keyword(yytext); }
Mark Sleebd588222007-11-21 08:43:35 +0000270"return" { thrift_reserved_keyword(yytext); }
Bryan Duxbury7f3285e2010-08-05 23:28:14 +0000271"self" { thrift_reserved_keyword(yytext); }
Mark Sleebd588222007-11-21 08:43:35 +0000272"sizeof" { thrift_reserved_keyword(yytext); }
273"static" { thrift_reserved_keyword(yytext); }
Bryan Duxbury7f3285e2010-08-05 23:28:14 +0000274"super" { thrift_reserved_keyword(yytext); }
Mark Sleebd588222007-11-21 08:43:35 +0000275"switch" { thrift_reserved_keyword(yytext); }
276"synchronized" { thrift_reserved_keyword(yytext); }
Bryan Duxbury7f3285e2010-08-05 23:28:14 +0000277"then" { thrift_reserved_keyword(yytext); }
Mark Sleebd588222007-11-21 08:43:35 +0000278"this" { thrift_reserved_keyword(yytext); }
279"throw" { thrift_reserved_keyword(yytext); }
280"transient" { thrift_reserved_keyword(yytext); }
Mark Sleebd588222007-11-21 08:43:35 +0000281"try" { thrift_reserved_keyword(yytext); }
Bryan Duxbury7f3285e2010-08-05 23:28:14 +0000282"undef" { thrift_reserved_keyword(yytext); }
283"union" { thrift_reserved_keyword(yytext); }
284"unless" { thrift_reserved_keyword(yytext); }
Mark Sleebd588222007-11-21 08:43:35 +0000285"unsigned" { thrift_reserved_keyword(yytext); }
Bryan Duxbury7f3285e2010-08-05 23:28:14 +0000286"until" { thrift_reserved_keyword(yytext); }
287"use" { thrift_reserved_keyword(yytext); }
Mark Sleebd588222007-11-21 08:43:35 +0000288"var" { thrift_reserved_keyword(yytext); }
289"virtual" { thrift_reserved_keyword(yytext); }
290"volatile" { thrift_reserved_keyword(yytext); }
Bryan Duxbury7f3285e2010-08-05 23:28:14 +0000291"when" { thrift_reserved_keyword(yytext); }
Mark Sleebd588222007-11-21 08:43:35 +0000292"while" { thrift_reserved_keyword(yytext); }
293"with" { thrift_reserved_keyword(yytext); }
Bryan Duxbury7f3285e2010-08-05 23:28:14 +0000294"xor" { thrift_reserved_keyword(yytext); }
Mark Sleebd588222007-11-21 08:43:35 +0000295"yield" { thrift_reserved_keyword(yytext); }
Mark Sleef12865a2007-01-12 00:23:26 +0000296
Mark Slee4f8da1d2006-10-12 02:47:27 +0000297{intconstant} {
David Reissf1454162008-06-30 20:45:47 +0000298 errno = 0;
299 yylval.iconst = strtoll(yytext, NULL, 10);
300 if (errno == ERANGE) {
301 integer_overflow(yytext);
302 }
Mark Slee4f8da1d2006-10-12 02:47:27 +0000303 return tok_int_constant;
304}
Mark Sleef5377b32006-10-10 01:42:59 +0000305
Mark Slee600cdb32006-11-29 22:06:42 +0000306{hexconstant} {
David Reissf1454162008-06-30 20:45:47 +0000307 errno = 0;
308 yylval.iconst = strtoll(yytext+2, NULL, 16);
309 if (errno == ERANGE) {
310 integer_overflow(yytext);
311 }
Mark Slee600cdb32006-11-29 22:06:42 +0000312 return tok_int_constant;
313}
314
Mark Slee30152872006-11-28 01:24:07 +0000315{dubconstant} {
316 yylval.dconst = atof(yytext);
317 return tok_dub_constant;
318}
319
Mark Slee4f8da1d2006-10-12 02:47:27 +0000320{identifier} {
321 yylval.id = strdup(yytext);
322 return tok_identifier;
323}
324
Mark Sleebd588222007-11-21 08:43:35 +0000325{st_identifier} {
326 yylval.id = strdup(yytext);
327 return tok_st_identifier;
328}
329
David Reiss82e6fc02009-03-26 23:32:36 +0000330{literal_begin} {
331 char mark = yytext[0];
332 std::string result;
333 for(;;)
334 {
335 int ch = yyinput();
336 switch (ch) {
337 case EOF:
338 yyerror("End of file while read string at %d\n", yylineno);
339 exit(1);
340 case '\n':
341 yyerror("End of line while read string at %d\n", yylineno - 1);
342 exit(1);
343 case '\\':
344 ch = yyinput();
345 switch (ch) {
346 case 'r':
347 result.push_back('\r');
348 continue;
349 case 'n':
350 result.push_back('\n');
351 continue;
352 case 't':
353 result.push_back('\t');
354 continue;
355 case '"':
356 result.push_back('"');
357 continue;
358 case '\'':
359 result.push_back('\'');
360 continue;
361 case '\\':
362 result.push_back('\\');
363 continue;
364 default:
365 yyerror("Bad escape character\n");
366 return -1;
367 }
368 break;
369 default:
370 if (ch == mark) {
371 yylval.id = strdup(result.c_str());
372 return tok_literal;
373 } else {
374 result.push_back(ch);
375 }
376 }
377 }
Mark Slee30152872006-11-28 01:24:07 +0000378}
379
Mark Slee31985722006-05-24 21:45:31 +0000380
ccheeverf53b5cf2007-02-05 20:33:11 +0000381{doctext} {
David Reisscbd4bac2007-08-14 17:12:33 +0000382 /* This does not show up in the parse tree. */
383 /* Rather, the parser will grab it out of the global. */
384 if (g_parse_mode == PROGRAM) {
385 clear_doctext();
386 g_doctext = strdup(yytext + 3);
Jens Geyer8cd3efe2013-09-16 22:17:52 +0200387 assert(strlen(g_doctext) >= 2);
388 g_doctext[strlen(g_doctext) - 2] = ' ';
389 g_doctext[strlen(g_doctext) - 1] = '\0';
David Reisscbd4bac2007-08-14 17:12:33 +0000390 g_doctext = clean_up_doctext(g_doctext);
391 g_doctext_lineno = yylineno;
Jens Geyer813749d2014-01-31 23:42:57 +0100392 if( (g_program_doctext_candidate == NULL) && (g_program_doctext_status == INVALID)){
Jens Geyere8379b52014-01-25 00:59:45 +0100393 g_program_doctext_candidate = strdup(g_doctext);
394 g_program_doctext_lineno = g_doctext_lineno;
395 g_program_doctext_status = STILL_CANDIDATE;
Jens Geyer813749d2014-01-31 23:42:57 +0100396 pdebug("%s","program doctext set to STILL_CANDIDATE");
Jens Geyere8379b52014-01-25 00:59:45 +0100397 }
David Reisscbd4bac2007-08-14 17:12:33 +0000398 }
ccheeverf53b5cf2007-02-05 20:33:11 +0000399}
400
Bryan Duxbury235f8b52011-08-19 18:27:47 +0000401. {
402 unexpected_token(yytext);
403}
404
ccheeverf53b5cf2007-02-05 20:33:11 +0000405
David Reissfb790d72010-09-02 16:41:45 +0000406. {
407 /* Catch-all to let us catch "*" in the parser. */
408 return (int) yytext[0];
409}
410
Mark Slee31985722006-05-24 21:45:31 +0000411%%
David Reiss4a054342009-03-26 23:32:27 +0000412
413/* vim: filetype=lex
414*/