/* GNU gettext - internationalization aids Copyright (C) 1995-1999, 2000-2003 Free Software Foundation, Inc. This file was written by Peter Miller . Multibyte character handling by Bruno Haible . This program 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. This program 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 this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #ifdef HAVE_CONFIG_H # include "config.h" #endif /* Specification. */ #include "po-lex.h" #include #include #include #include #include #include #if HAVE_ICONV # include #endif #include "c-ctype.h" #include "linebreak.h" #include "vasprintf.h" #include "gettext.h" #include "po-charset.h" #include "xalloc.h" #include "exit.h" #include "error.h" #include "error-progname.h" #include "pos.h" #include "str-list.h" #include "po-gram-gen2.h" #define _(str) gettext(str) #if HAVE_ICONV # include "utf8-ucs4.h" #endif #if HAVE_DECL_GETC_UNLOCKED # undef getc # define getc getc_unlocked #endif /* Current position within the PO file. */ lex_pos_ty gram_pos; int gram_pos_column; /* Error handling during the parsing of a PO file. These functions can access gram_pos and gram_pos_column. */ #if !(__STDC__ && \ ((defined __STDC_VERSION__ && __STDC_VERSION__ >= 199901L && !defined __DECC) \ || (defined __GNUC__ && __GNUC__ >= 2 && !defined __APPLE_CC__))) /* CAUTION: If you change this function, you must also make identical changes to the macro of the same name in src/po-lex.h */ /* VARARGS1 */ void po_gram_error (const char *fmt, ...) { va_list ap; char *buffer; va_start (ap, fmt); if (vasprintf (&buffer, fmt, ap) < 0) error (EXIT_FAILURE, 0, _("memory exhausted")); va_end (ap); error_with_progname = false; error (0, 0, "%s:%lu:%d: %s", gram_pos.file_name, (unsigned long) gram_pos.line_number, gram_pos_column + 1, buffer); error_with_progname = true; free (buffer); /* Some messages need more than one line. Continuation lines are indicated by using "..." at the start of the string. We don't increment the error counter for these continuation lines. */ if (*fmt == '.') --error_message_count; else if (error_message_count >= gram_max_allowed_errors) error (EXIT_FAILURE, 0, _("too many errors, aborting")); } /* CAUTION: If you change this function, you must also make identical changes to the macro of the same name in src/po-lex.h */ /* VARARGS2 */ void po_gram_error_at_line (const lex_pos_ty *pp, const char *fmt, ...) { va_list ap; char *buffer; va_start (ap, fmt); if (vasprintf (&buffer, fmt, ap) < 0) error (EXIT_FAILURE, 0, _("memory exhausted")); va_end (ap); error_with_progname = false; error_at_line (0, 0, pp->file_name, pp->line_number, "%s", buffer); error_with_progname = true; free (buffer); /* Some messages need more than one line, or more than one location. Continuation lines are indicated by using "..." at the start of the string. We don't increment the error counter for these continuation lines. */ if (*fmt == '.') --error_message_count; else if (error_message_count >= gram_max_allowed_errors) error (EXIT_FAILURE, 0, _("too many errors, aborting")); } #endif /* The lowest level of PO file parsing converts bytes to multibyte characters. This is needed 1. for C compatibility: ISO C 99 section 5.1.1.2 says that the first translation phase maps bytes to characters. 2. to keep track of the current column, for the sake of precise error location. Emacs compile.el interprets the column in error messages by default as a screen column number, not as character number. 3. to avoid skipping backslash-newline in the midst of a multibyte character. If XY is a multibyte character, X \ newline Y is invalid. */ /* Multibyte character data type. */ /* Note this depends on po_lex_charset and po_lex_iconv, which get set while the file is being parsed. */ #define MBCHAR_BUF_SIZE 24 struct mbchar { size_t bytes; /* number of bytes of current character, > 0 */ #if HAVE_ICONV bool uc_valid; /* true if uc is a valid Unicode character */ unsigned int uc; /* if uc_valid: the current character */ #endif char buf[MBCHAR_BUF_SIZE]; /* room for the bytes */ }; /* We want to pass multibyte characters by reference automatically, therefore we use an array type. */ typedef struct mbchar mbchar_t[1]; /* A version of memcpy optimized for the case n <= 1. */ static inline void memcpy_small (void *dst, const void *src, size_t n) { if (n > 0) { char *q = (char *) dst; const char *p = (const char *) src; *q = *p; if (--n > 0) do *++q = *++p; while (--n > 0); } } /* EOF (not a real character) is represented with bytes = 0 and uc_valid = false. */ static inline bool mb_iseof (const mbchar_t mbc) { return (mbc->bytes == 0); } /* Access the current character. */ static inline const char * mb_ptr (const mbchar_t mbc) { return mbc->buf; } static inline size_t mb_len (const mbchar_t mbc) { return mbc->bytes; } /* Comparison of characters. */ static inline bool mb_iseq (const mbchar_t mbc, char sc) { /* Note: It is wrong to compare only mbc->uc, because when the encoding is SHIFT_JIS, mbc->buf[0] == '\\' corresponds to mbc->uc == 0x00A5, but we want to treat it as an escape character, although it looks like a Yen sign. */ #if HAVE_ICONV && 0 if (mbc->uc_valid) return (mbc->uc == sc); /* wrong! */ else #endif return (mbc->bytes == 1 && mbc->buf[0] == sc); } static inline bool mb_isnul (const mbchar_t mbc) { #if HAVE_ICONV if (mbc->uc_valid) return (mbc->uc == 0); else #endif return (mbc->bytes == 1 && mbc->buf[0] == 0); } static inline int mb_cmp (const mbchar_t mbc1, const mbchar_t mbc2) { #if HAVE_ICONV if (mbc1->uc_valid && mbc2->uc_valid) return (int) mbc1->uc - (int) mbc2->uc; else #endif return (mbc1->bytes == mbc2->bytes ? memcmp (mbc1->buf, mbc2->buf, mbc1->bytes) : mbc1->bytes < mbc2->bytes ? (memcmp (mbc1->buf, mbc2->buf, mbc1->bytes) > 0 ? 1 : -1) : (memcmp (mbc1->buf, mbc2->buf, mbc2->bytes) >= 0 ? 1 : -1)); } static inline bool mb_equal (const mbchar_t mbc1, const mbchar_t mbc2) { #if HAVE_ICONV if (mbc1->uc_valid && mbc2->uc_valid) return mbc1->uc == mbc2->uc; else #endif return (mbc1->bytes == mbc2->bytes && memcmp (mbc1->buf, mbc2->buf, mbc1->bytes) == 0); } /* , classification. */ static inline bool mb_isascii (const mbchar_t mbc) { #if HAVE_ICONV if (mbc->uc_valid) return (mbc->uc >= 0x0000 && mbc->uc <= 0x007F); else #endif return (mbc->bytes == 1 #if CHAR_MIN < 0x00 /* to avoid gcc warning */ && mbc->buf[0] >= 0x00 #endif #if CHAR_MAX > 0x7F /* to avoid gcc warning */ && mbc->buf[0] <= 0x7F #endif ); } /* Extra function. */ /* Unprintable characters appear as a small box of width 1. */ #define MB_UNPRINTABLE_WIDTH 1 static int mb_width (const mbchar_t mbc) { #if HAVE_ICONV if (mbc->uc_valid) { unsigned int uc = mbc->uc; const char *encoding = (po_lex_iconv != (iconv_t)(-1) ? po_lex_charset : ""); int w = uc_width (uc, encoding); /* For unprintable characters, arbitrarily return 0 for control characters (except tab) and MB_UNPRINTABLE_WIDTH otherwise. */ if (w >= 0) return w; if (uc >= 0x0000 && uc <= 0x001F) { if (uc == 0x0009) return 8 - (gram_pos_column & 7); return 0; } if ((uc >= 0x007F && uc <= 0x009F) || (uc >= 0x2028 && uc <= 0x2029)) return 0; return MB_UNPRINTABLE_WIDTH; } else #endif { if (mbc->bytes == 1) { if (mbc->buf[0] >= 0x00 && mbc->buf[0] <= 0x1F) { if (mbc->buf[0] == 0x09) return 8 - (gram_pos_column & 7); return 0; } if (mbc->buf[0] == 0x7F) return 0; } return MB_UNPRINTABLE_WIDTH; } } /* Output. */ static inline void mb_putc (const mbchar_t mbc, FILE *stream) { fwrite (mbc->buf, 1, mbc->bytes, stream); } /* Assignment. */ static inline void mb_setascii (mbchar_t mbc, char sc) { mbc->bytes = 1; #if HAVE_ICONV mbc->uc_valid = 1; mbc->uc = sc; #endif mbc->buf[0] = sc; } /* Copying a character. */ static inline void mb_copy (mbchar_t new, const mbchar_t old) { memcpy_small (&new->buf[0], &old->buf[0], old->bytes); new->bytes = old->bytes; #if HAVE_ICONV if ((new->uc_valid = old->uc_valid)) new->uc = old->uc; #endif } /* Multibyte character input. */ /* Number of characters that can be pushed back. We need 1 for lex_getc, plus 1 for lex_ungetc. */ #define NPUSHBACK 2 /* Data type of a multibyte character input stream. */ struct mbfile { FILE *fp; bool eof_seen; int have_pushback; unsigned int bufcount; char buf[MBCHAR_BUF_SIZE]; struct mbchar pushback[NPUSHBACK]; }; /* We want to pass multibyte streams by reference automatically, therefore we use an array type. */ typedef struct mbfile mbfile_t[1]; /* Whether invalid multibyte sequences in the input shall be signalled or silently tolerated. */ static bool signal_eilseq; static inline void mbfile_init (mbfile_t mbf, FILE *stream) { mbf->fp = stream; mbf->eof_seen = false; mbf->have_pushback = 0; mbf->bufcount = 0; } /* Read the next multibyte character from mbf and put it into mbc. If a read error occurs, errno is set and ferror (mbf->fp) becomes true. */ static void mbfile_getc (mbchar_t mbc, mbfile_t mbf) { size_t bytes; /* If EOF has already been seen, don't use getc. This matters if mbf->fp is connected to an interactive tty. */ if (mbf->eof_seen) goto eof; /* Return character pushed back, if there is one. */ if (mbf->have_pushback > 0) { mbf->have_pushback--; mb_copy (mbc, &mbf->pushback[mbf->have_pushback]); return; } /* Before using iconv, we need at least one byte. */ if (mbf->bufcount == 0) { int c = getc (mbf->fp); if (c == EOF) { mbf->eof_seen = true; goto eof; } mbf->buf[0] = (unsigned char) c; mbf->bufcount++; } #if HAVE_ICONV if (po_lex_iconv != (iconv_t)(-1)) { /* Use iconv on an increasing number of bytes. Read only as many bytes from mbf->fp as needed. This is needed to give reasonable interactive behaviour when mbf->fp is connected to an interactive tty. */ for (;;) { unsigned char scratchbuf[64]; const char *inptr = &mbf->buf[0]; size_t insize = mbf->bufcount; char *outptr = (char *) &scratchbuf[0]; size_t outsize = sizeof (scratchbuf); size_t res = iconv (po_lex_iconv, (ICONV_CONST char **) &inptr, &insize, &outptr, &outsize); /* We expect that a character has been produced if and only if some input bytes have been consumed. */ if ((insize < mbf->bufcount) != (outsize < sizeof (scratchbuf))) abort (); if (outsize == sizeof (scratchbuf)) { /* No character has been produced. Must be an error. */ if (res != (size_t)(-1)) abort (); if (errno == EILSEQ) { /* An invalid multibyte sequence was encountered. */ /* Return a single byte. */ if (signal_eilseq) po_gram_error (_("invalid multibyte sequence")); bytes = 1; mbc->uc_valid = false; break; } else if (errno == EINVAL) { /* An incomplete multibyte character. */ int c; if (mbf->bufcount == MBCHAR_BUF_SIZE) { /* An overlong incomplete multibyte sequence was encountered. */ /* Return a single byte. */ bytes = 1; mbc->uc_valid = false; break; } /* Read one more byte and retry iconv. */ c = getc (mbf->fp); if (c == EOF) { mbf->eof_seen = true; if (ferror (mbf->fp)) goto eof; if (signal_eilseq) po_gram_error (_("\ incomplete multibyte sequence at end of file")); bytes = mbf->bufcount; mbc->uc_valid = false; break; } mbf->buf[mbf->bufcount++] = (unsigned char) c; if (c == '\n') { if (signal_eilseq) po_gram_error (_("\ incomplete multibyte sequence at end of line")); bytes = mbf->bufcount - 1; mbc->uc_valid = false; break; } } else error (EXIT_FAILURE, errno, _("iconv failure")); } else { size_t outbytes = sizeof (scratchbuf) - outsize; bytes = mbf->bufcount - insize; /* We expect that one character has been produced. */ if (bytes == 0) abort (); if (outbytes == 0) abort (); /* Convert it from UTF-8 to UCS-4. */ if (u8_mbtouc (&mbc->uc, scratchbuf, outbytes) < outbytes) { /* scratchbuf contains an out-of-range Unicode character (> 0x10ffff). */ if (signal_eilseq) po_gram_error (_("invalid multibyte sequence")); mbc->uc_valid = false; break; } mbc->uc_valid = true; break; } } } else #endif { if (po_lex_weird_cjk /* Special handling of encodings with CJK structure. */ && (unsigned char) mbf->buf[0] >= 0x80) { if (mbf->bufcount == 1) { /* Read one more byte. */ int c = getc (mbf->fp); if (c == EOF) { if (ferror (mbf->fp)) { mbf->eof_seen = true; goto eof; } } else { mbf->buf[1] = (unsigned char) c; mbf->bufcount++; } } if (mbf->bufcount >= 2 && (unsigned char) mbf->buf[1] >= 0x30) /* Return a double byte. */ bytes = 2; else /* Return a single byte. */ bytes = 1; } else { /* Return a single byte. */ bytes = 1; } #if HAVE_ICONV mbc->uc_valid = false; #endif } /* Return the multibyte sequence mbf->buf[0..bytes-1]. */ memcpy_small (&mbc->buf[0], &mbf->buf[0], bytes); mbc->bytes = bytes; mbf->bufcount -= bytes; if (mbf->bufcount > 0) { /* It's not worth calling memmove() for so few bytes. */ unsigned int count = mbf->bufcount; char *p = &mbf->buf[0]; do { *p = *(p + bytes); p++; } while (--count > 0); } return; eof: /* An mbchar_t with bytes == 0 is used to indicate EOF. */ mbc->bytes = 0; #if HAVE_ICONV mbc->uc_valid = false; #endif return; } static void mbfile_ungetc (const mbchar_t mbc, mbfile_t mbf) { if (mbf->have_pushback >= NPUSHBACK) abort (); mb_copy (&mbf->pushback[mbf->have_pushback], mbc); mbf->have_pushback++; } /* Lexer variables. */ static mbfile_t mbf; unsigned int gram_max_allowed_errors = 20; static bool po_lex_obsolete; static bool pass_comments = false; bool pass_obsolete_entries = false; /* Prepare lexical analysis. */ void lex_start (FILE *fp, const char *real_filename, const char *logical_filename) { /* Ignore the logical_filename, because PO file entries already have their file names attached. But use real_filename for error messages. */ gram_pos.file_name = xstrdup (real_filename); mbfile_init (mbf, fp); gram_pos.line_number = 1; gram_pos_column = 0; signal_eilseq = true; po_lex_obsolete = false; po_lex_charset_init (); } /* Terminate lexical analysis. */ void lex_end () { mbf->fp = NULL; gram_pos.file_name = NULL; gram_pos.line_number = 0; gram_pos_column = 0; signal_eilseq = false; po_lex_obsolete = false; po_lex_charset_close (); } /* Read a single character, dealing with backslash-newline. Also keep track of the current line number and column number. */ static void lex_getc (mbchar_t mbc) { for (;;) { mbfile_getc (mbc, mbf); if (mb_iseof (mbc)) { if (ferror (mbf->fp)) { bomb: error (EXIT_FAILURE, errno, _("error while reading \"%s\""), gram_pos.file_name); } break; } if (mb_iseq (mbc, '\n')) { gram_pos.line_number++; gram_pos_column = 0; break; } gram_pos_column += mb_width (mbc); if (mb_iseq (mbc, '\\')) { mbchar_t mbc2; mbfile_getc (mbc2, mbf); if (mb_iseof (mbc2)) { if (ferror (mbf->fp)) goto bomb; break; } if (!mb_iseq (mbc2, '\n')) { mbfile_ungetc (mbc2, mbf); break; } gram_pos.line_number++; gram_pos_column = 0; } else break; } } static void lex_ungetc (const mbchar_t mbc) { if (!mb_iseof (mbc)) { if (mb_iseq (mbc, '\n')) /* Decrement the line number, but don't care about the column. */ gram_pos.line_number--; else /* Decrement the column number. Also works well enough for tabs. */ gram_pos_column -= mb_width (mbc); mbfile_ungetc (mbc, mbf); } } static int keyword_p (const char *s) { if (!strcmp (s, "domain")) return DOMAIN; if (!strcmp (s, "msgid")) return MSGID; if (!strcmp (s, "msgid_plural")) return MSGID_PLURAL; if (!strcmp (s, "msgstr")) return MSGSTR; po_gram_error_at_line (&gram_pos, _("keyword \"%s\" unknown"), s); return NAME; } static int control_sequence () { mbchar_t mbc; int val; int max; lex_getc (mbc); if (mb_len (mbc) == 1) switch (mb_ptr (mbc) [0]) { case 'n': return '\n'; case 't': return '\t'; case 'b': return '\b'; case 'r': return '\r'; case 'f': return '\f'; case 'v': return '\v'; case 'a': return '\a'; case '\\': case '"': return mb_ptr (mbc) [0]; case '0': case '1': case '2': case '3': case '4': case '5': case '6': case '7': val = 0; max = 0; for (;;) { char c = mb_ptr (mbc) [0]; /* Warning: not portable, can't depend on '0'..'7' ordering. */ val = val * 8 + (c - '0'); if (++max == 3) break; lex_getc (mbc); if (mb_len (mbc) == 1) switch (mb_ptr (mbc) [0]) { case '0': case '1': case '2': case '3': case '4': case '5': case '6': case '7': continue; default: break; } lex_ungetc (mbc); break; } return val; case 'x': lex_getc (mbc); if (mb_iseof (mbc) || mb_len (mbc) != 1 || !c_isxdigit (mb_ptr (mbc) [0])) break; val = 0; for (;;) { char c = mb_ptr (mbc) [0]; val *= 16; if (c_isdigit (c)) /* Warning: not portable, can't depend on '0'..'9' ordering */ val += c - '0'; else if (c_isupper (c)) /* Warning: not portable, can't depend on 'A'..'F' ordering */ val += c - 'A' + 10; else /* Warning: not portable, can't depend on 'a'..'f' ordering */ val += c - 'a' + 10; lex_getc (mbc); if (mb_len (mbc) == 1) switch (mb_ptr (mbc) [0]) { case '0': case '1': case '2': case '3': case '4': case '5': case '6': case '7': case '8': case '9': case 'a': case 'b': case 'c': case 'd': case 'e': case 'f': case 'A': case 'B': case 'C': case 'D': case 'E': case 'F': continue; default: break; } lex_ungetc (mbc); break; } return val; /* FIXME: \u and \U are not handled. */ } lex_ungetc (mbc); po_gram_error (_("invalid control sequence")); return ' '; } /* Return the next token in the PO file. The return codes are defined in "po-gram-gen2.h". Associated data is put in 'po_gram_lval'. */ int po_gram_lex () { static char *buf; static size_t bufmax; mbchar_t mbc; size_t bufpos; for (;;) { lex_getc (mbc); if (mb_iseof (mbc)) /* Yacc want this for end of file. */ return 0; if (mb_len (mbc) == 1) switch (mb_ptr (mbc) [0]) { case '\n': po_lex_obsolete = false; /* Ignore whitespace, not relevant for the grammar. */ break; case ' ': case '\t': case '\r': case '\f': case '\v': /* Ignore whitespace, not relevant for the grammar. */ break; case '#': lex_getc (mbc); if (mb_iseq (mbc, '~')) /* A pseudo-comment beginning with #~ is found. This is not a comment. It is the format for obsolete entries. We simply discard the "#~" prefix. The following characters are expected to be well formed. */ { po_lex_obsolete = true; break; } /* Accumulate comments into a buffer. If we have been asked to pass comments, generate a COMMENT token, otherwise discard it. */ signal_eilseq = false; if (pass_comments) { bufpos = 0; for (;;) { while (bufpos + mb_len (mbc) >= bufmax) { bufmax += 100; buf = xrealloc (buf, bufmax); } if (mb_iseof (mbc) || mb_iseq (mbc, '\n')) break; memcpy_small (&buf[bufpos], mb_ptr (mbc), mb_len (mbc)); bufpos += mb_len (mbc); lex_getc (mbc); } buf[bufpos] = '\0'; po_gram_lval.string.string = buf; po_gram_lval.string.pos = gram_pos; po_gram_lval.string.obsolete = po_lex_obsolete; po_lex_obsolete = false; signal_eilseq = true; return COMMENT; } else { /* We do this in separate loop because collecting large comments while they get not passed to the upper layers is not very effective. */ while (!mb_iseof (mbc) && !mb_iseq (mbc, '\n')) lex_getc (mbc); po_lex_obsolete = false; signal_eilseq = true; } break; case '"': /* Accumulate a string. */ bufpos = 0; for (;;) { lex_getc (mbc); while (bufpos + mb_len (mbc) >= bufmax) { bufmax += 100; buf = xrealloc (buf, bufmax); } if (mb_iseof (mbc)) { po_gram_error_at_line (&gram_pos, _("end-of-file within string")); break; } if (mb_iseq (mbc, '\n')) { po_gram_error_at_line (&gram_pos, _("end-of-line within string")); break; } if (mb_iseq (mbc, '"')) break; if (mb_iseq (mbc, '\\')) { buf[bufpos++] = control_sequence (); continue; } /* Add mbc to the accumulator. */ memcpy_small (&buf[bufpos], mb_ptr (mbc), mb_len (mbc)); bufpos += mb_len (mbc); } buf[bufpos] = '\0'; /* FIXME: Treatment of embedded \000 chars is incorrect. */ po_gram_lval.string.string = xstrdup (buf); po_gram_lval.string.pos = gram_pos; po_gram_lval.string.obsolete = po_lex_obsolete; return STRING; case 'a': case 'b': case 'c': case 'd': case 'e': case 'f': case 'g': case 'h': case 'i': case 'j': case 'k': case 'l': case 'm': case 'n': case 'o': case 'p': case 'q': case 'r': case 's': case 't': case 'u': case 'v': case 'w': case 'x': case 'y': case 'z': case 'A': case 'B': case 'C': case 'D': case 'E': case 'F': case 'G': case 'H': case 'I': case 'J': case 'K': case 'L': case 'M': case 'N': case 'O': case 'P': case 'Q': case 'R': case 'S': case 'T': case 'U': case 'V': case 'W': case 'X': case 'Y': case 'Z': case '_': case '$': bufpos = 0; for (;;) { char c = mb_ptr (mbc) [0]; if (bufpos + 1 >= bufmax) { bufmax += 100; buf = xrealloc (buf, bufmax); } buf[bufpos++] = c; lex_getc (mbc); if (mb_len (mbc) == 1) switch (mb_ptr (mbc) [0]) { default: break; case 'a': case 'b': case 'c': case 'd': case 'e': case 'f': case 'g': case 'h': case 'i': case 'j': case 'k': case 'l': case 'm': case 'n': case 'o': case 'p': case 'q': case 'r': case 's': case 't': case 'u': case 'v': case 'w': case 'x': case 'y': case 'z': case 'A': case 'B': case 'C': case 'D': case 'E': case 'F': case 'G': case 'H': case 'I': case 'J': case 'K': case 'L': case 'M': case 'N': case 'O': case 'P': case 'Q': case 'R': case 'S': case 'T': case 'U': case 'V': case 'W': case 'X': case 'Y': case 'Z': case '_': case '$': case '0': case '1': case '2': case '3': case '4': case '5': case '6': case '7': case '8': case '9': continue; } break; } lex_ungetc (mbc); buf[bufpos] = '\0'; { int k = keyword_p (buf); if (k == NAME) { po_gram_lval.string.string = xstrdup (buf); po_gram_lval.string.pos = gram_pos; po_gram_lval.string.obsolete = po_lex_obsolete; } else { po_gram_lval.pos.pos = gram_pos; po_gram_lval.pos.obsolete = po_lex_obsolete; } return k; } case '0': case '1': case '2': case '3': case '4': case '5': case '6': case '7': case '8': case '9': bufpos = 0; for (;;) { char c = mb_ptr (mbc) [0]; if (bufpos + 1 >= bufmax) { bufmax += 100; buf = xrealloc (buf, bufmax + 1); } buf[bufpos++] = c; lex_getc (mbc); if (mb_len (mbc) == 1) switch (mb_ptr (mbc) [0]) { default: break; case '0': case '1': case '2': case '3': case '4': case '5': case '6': case '7': case '8': case '9': continue; } break; } lex_ungetc (mbc); buf[bufpos] = '\0'; po_gram_lval.number.number = atol (buf); po_gram_lval.number.pos = gram_pos; po_gram_lval.number.obsolete = po_lex_obsolete; return NUMBER; case '[': po_gram_lval.pos.pos = gram_pos; po_gram_lval.pos.obsolete = po_lex_obsolete; return '['; case ']': po_gram_lval.pos.pos = gram_pos; po_gram_lval.pos.obsolete = po_lex_obsolete; return ']'; default: /* This will cause a syntax error. */ return JUNK; } else /* This will cause a syntax error. */ return JUNK; } } /* po_gram_lex() can return comments as COMMENT. Switch this on or off. */ void po_lex_pass_comments (bool flag) { pass_comments = flag; } /* po_gram_lex() can return obsolete entries as if they were normal entries. Switch this on or off. */ void po_lex_pass_obsolete_entries (bool flag) { pass_obsolete_entries = flag; }