diff options
author | neil <neil@138bc75d-0d04-0410-961f-82ee72b054a4> | 2001-08-05 17:31:25 +0000 |
---|---|---|
committer | neil <neil@138bc75d-0d04-0410-961f-82ee72b054a4> | 2001-08-05 17:31:25 +0000 |
commit | 1ea7ed2119257b80d0327b5c3d979cebd87aa894 (patch) | |
tree | 37a35ef3c394621edd2c84f886dcc1310e94c38d /gcc/cpplex.c | |
parent | 9c0e6d90f7b28384e458f3844e0eb12adce16452 (diff) | |
download | gcc-1ea7ed2119257b80d0327b5c3d979cebd87aa894.tar.gz |
PR preprocessor/3081
* c-lex.c (map): New.
(cb_file_change): Update map and use it.
(cb_def_pragma, cb_define, cb_undef): Use map and line.
(c_lex): Update to use map.
* cpperror.c (print_location): Move to using logical line numbers.
* cppfiles.c (stack_include_file): Update for new _cpp_do_file_change.
(cpp_make_system_header): Similarly.
(_cpp_execute_include): Stop line numbering hacks. Store the
line we will return to.
* cpphash.h (CPP_BUF_LINE): Remove.
(struct cpp_buffer): Remove lineno and pseudo_newlines.
Add map and return_to_line.
(_cpp_do_file_change): Update.
* cppinit.c (cpp_start_read): Update line kludge.
* cpplex.c (handle_newline): Don't update lineno and pseudo_newlines.
(trigraph_ok): Use logical line numbers for diagnostics.
(skip_block_comment): Likewise.
(skip_whitespace): Likewise.
(skip_line_comment): Use pfile->line instead.
(_cpp_lex_token): Update to use logical line numbering exclusively.
Handle BOL locally. Accept new lines in directives, but keep
pfile->line decremented. Diagnostics use logical lines. Update
directive handling.
* cpplib.c (SEEN_EOL): New.
(skip_rest_of_line, check_eol): Use it.
(end_directive): Increase line number when accepting the newline
at the end of a directive.
(run_directive): Simplify.
(do_line): Bad LC_LEAVEs become LC_RENAMEs. Update.
(_cpp_do_file_change): Update to take buffer line number as an
argument, and store the current map in the cpp_reader. Remove
line number kludges.
(_cpp_do__Pragma): Restore output position after a _Pragma.
(cpp_push_buffer): Don't set output line or lineno.
(_cpp_pop_buffer): Transfer more info from a faked buffer.
Remove line kludge. Set output_line.
* cppmacro.c (builtin_macro): Update handling of __LINE__.
(parse_arg): Use logical lines.
(save_lookahead_token): Save EOFs too now.
* cppmain.c (struct printer): Fix comments.
(printer_init): Simplify, let caller do errors.
(scan_translation_unit, check_multiline_token, dump_macro): Update.
(maybe_print_line): Simplify.
(print_line): Don't print a linemarker if -P.
(cb_define, cb_undef, cb_def_pragma, cb_ident, cb_include): Update.
(cb_file_change): Simplify.
* line-map.h (LAST_SOURCE_LINE): Fix.
(CURRENT_LINE_MAP): New.
* gcc.dg/cpp/19951025-1.c: Revert.
* gcc.dg/cpp/directiv.c: We no longer process directives that
interrupt macro arguments.
git-svn-id: svn+ssh://gcc.gnu.org/svn/gcc/trunk@44650 138bc75d-0d04-0410-961f-82ee72b054a4
Diffstat (limited to 'gcc/cpplex.c')
-rw-r--r-- | gcc/cpplex.c | 128 |
1 files changed, 65 insertions, 63 deletions
diff --git a/gcc/cpplex.c b/gcc/cpplex.c index e0002beac45..f9c4bb9ab0c 100644 --- a/gcc/cpplex.c +++ b/gcc/cpplex.c @@ -132,11 +132,8 @@ handle_newline (pfile, newline_char) cppchar_t next = EOF; pfile->line++; - pfile->pseudo_newlines++; - buffer = pfile->buffer; buffer->col_adjust = 0; - buffer->lineno++; buffer->line_base = buffer->cur; /* Handle CR-LF and LF-CR combinations, get the next character. */ @@ -173,15 +170,16 @@ trigraph_ok (pfile, from_char) if (CPP_OPTION (pfile, warn_trigraphs) && !pfile->state.lexing_comment) { cpp_buffer *buffer = pfile->buffer; + if (accept) - cpp_warning_with_line (pfile, buffer->lineno, CPP_BUF_COL (buffer) - 2, + cpp_warning_with_line (pfile, pfile->line, CPP_BUF_COL (buffer) - 2, "trigraph ??%c converted to %c", (int) from_char, (int) _cpp_trigraph_map[from_char]); else if (buffer->cur != buffer->last_Wtrigraphs) { buffer->last_Wtrigraphs = buffer->cur; - cpp_warning_with_line (pfile, buffer->lineno, + cpp_warning_with_line (pfile, pfile->line, CPP_BUF_COL (buffer) - 2, "trigraph ??%c ignored", (int) from_char); } @@ -344,8 +342,8 @@ skip_block_comment (pfile) { prevc = c, c = *buffer->cur++; if (c != '/') - cpp_warning_with_line (pfile, CPP_BUF_LINE (buffer), - CPP_BUF_COL (buffer), + cpp_warning_with_line (pfile, pfile->line, + CPP_BUF_COL (buffer) - 2, "\"/*\" within comment"); } goto next_char; @@ -373,7 +371,7 @@ skip_line_comment (pfile) cpp_reader *pfile; { cpp_buffer *buffer = pfile->buffer; - unsigned int orig_lineno = buffer->lineno; + unsigned int orig_line = pfile->line; cppchar_t c; pfile->state.lexing_comment = 1; @@ -391,7 +389,7 @@ skip_line_comment (pfile) pfile->state.lexing_comment = 0; buffer->read_ahead = c; /* Leave any newline for caller. */ - return orig_lineno != buffer->lineno; + return orig_line != pfile->line; } /* pfile->buffer->cur is one beyond the \t character. Update @@ -437,7 +435,7 @@ skip_whitespace (pfile, c) } } else if (pfile->state.in_directive && CPP_PEDANTIC (pfile)) - cpp_pedwarn_with_line (pfile, CPP_BUF_LINE (buffer), + cpp_pedwarn_with_line (pfile, pfile->line, CPP_BUF_COL (buffer), "%s in preprocessing directive", c == '\f' ? "form feed" : "vertical tab"); @@ -865,17 +863,16 @@ _cpp_lex_token (pfile, result) cppchar_t c; cpp_buffer *buffer; const unsigned char *comment_start; - unsigned char bol; + int bol; - skip: - bol = pfile->state.next_bol; - done_directive: + next_token: buffer = pfile->buffer; - pfile->state.next_bol = 0; result->flags = buffer->saved_flags; buffer->saved_flags = 0; + bol = (buffer->cur <= buffer->line_base + 1 + && pfile->lexer_pos.output_line == pfile->line); next_char: - pfile->lexer_pos.line = buffer->lineno; + pfile->lexer_pos.line = pfile->line; result->line = pfile->line; next_char2: pfile->lexer_pos.col = CPP_BUF_COLUMN (buffer, buffer->cur); @@ -893,22 +890,29 @@ _cpp_lex_token (pfile, result) switch (c) { case EOF: - if (!pfile->state.in_directive) + /* To prevent bogus diagnostics, only pop the buffer when + in-progress directives and arguments have been taken care of. + Decrement the line to terminate an in-progress directive. */ + if (pfile->state.in_directive) + pfile->line--; + else if (! pfile->state.parsing_args) { unsigned char ret = pfile->buffer->return_at_eof; /* Non-empty files should end in a newline. Don't warn for command line and _Pragma buffers. */ - if (pfile->lexer_pos.col != 0 && !buffer->from_stage3) - cpp_pedwarn (pfile, "no newline at end of file"); - _cpp_pop_buffer (pfile); - if (pfile->buffer && !ret) + if (pfile->lexer_pos.col != 0) { - bol = 1; - goto done_directive; + /* Account for the missing \n. */ + pfile->line++; + if (!buffer->from_stage3) + cpp_pedwarn (pfile, "no newline at end of file"); } + + _cpp_pop_buffer (pfile); + if (pfile->buffer && !ret) + goto next_token; } - pfile->state.next_bol = 1; result->type = CPP_EOF; return; @@ -918,36 +922,41 @@ _cpp_lex_token (pfile, result) goto next_char2; case '\n': case '\r': - if (!pfile->state.in_directive) + if (pfile->state.in_directive) { - handle_newline (pfile, c); - if (!pfile->state.parsing_args) - pfile->pseudo_newlines = 0; - bol = 1; - pfile->lexer_pos.output_line = buffer->lineno; - /* This is a new line, so clear any white space flag. - Newlines in arguments are white space (6.10.3.10); - parse_arg takes care of that. */ - result->flags &= ~(PREV_WHITE | AVOID_LPASTE); - goto next_char; + result->type = CPP_EOF; + if (pfile->state.parsing_args) + buffer->read_ahead = c; + else + { + handle_newline (pfile, c); + /* Decrementing pfile->line allows directives to + recognise that the newline has been seen, and also + means that diagnostics don't point to the next line. */ + pfile->lexer_pos.output_line = pfile->line--; + } + return; } - /* Don't let directives spill over to the next line. */ - buffer->read_ahead = c; - pfile->state.next_bol = 1; - result->type = CPP_EOF; - /* Don't break; pfile->state.skipping might be true. */ - return; + handle_newline (pfile, c); + /* This is a new line, so clear any white space flag. Newlines + in arguments are white space (6.10.3.10); parse_arg takes + care of that. */ + result->flags &= ~(PREV_WHITE | AVOID_LPASTE); + bol = 1; + if (pfile->state.parsing_args != 2) + pfile->lexer_pos.output_line = pfile->line; + goto next_char; case '?': case '\\': /* These could start an escaped newline, or '?' a trigraph. Let skip_escaped_newlines do all the work. */ { - unsigned int lineno = buffer->lineno; + unsigned int line = pfile->line; c = skip_escaped_newlines (buffer, c); - if (lineno != buffer->lineno) + if (line != pfile->line) /* We had at least one escaped newline of some sort, and the next character is in buffer->read_ahead. Update the token's line and column. */ @@ -1026,9 +1035,7 @@ _cpp_lex_token (pfile, result) if (c == '*') { if (skip_block_comment (pfile)) - cpp_error_with_line (pfile, pfile->lexer_pos.line, - pfile->lexer_pos.col, - "unterminated comment"); + cpp_error (pfile, "unterminated comment"); } else { @@ -1212,26 +1219,21 @@ _cpp_lex_token (pfile, result) macro invocation, and proceed to process the directive. */ if (pfile->state.parsing_args) { + pfile->lexer_pos.output_line = pfile->line; if (pfile->state.parsing_args == 2) - cpp_error (pfile, - "directives may not be used inside a macro argument"); - - /* Put a '#' in lookahead, return CPP_EOF for parse_arg. */ - buffer->extra_char = buffer->read_ahead; - buffer->read_ahead = '#'; - pfile->state.next_bol = 1; - result->type = CPP_EOF; - - /* Get whitespace right - newline_in_args sets it. */ - if (pfile->lexer_pos.col == 1) - result->flags &= ~(PREV_WHITE | AVOID_LPASTE); + { + cpp_error (pfile, + "directives may not be used inside a macro argument"); + result->type = CPP_EOF; + } } - else + /* in_directive can be true inside a _Pragma. */ + else if (!pfile->state.in_directive) { - /* This is the hash introducing a directive. */ + /* This is the hash introducing a directive. If the return + value is false, it is an assembler #. */ if (_cpp_handle_directive (pfile, result->flags & PREV_WHITE)) - goto done_directive; /* bol still 1. */ - /* This is in fact an assembler #. */ + goto next_token; } break; @@ -1283,7 +1285,7 @@ _cpp_lex_token (pfile, result) } if (!pfile->state.in_directive && pfile->state.skipping) - goto skip; + goto next_char; /* If not in a directive, this token invalidates controlling macros. */ if (!pfile->state.in_directive) |