diff --git a/parse.y b/parse.y index b2bbafe123..abedadb7d7 100644 --- a/parse.y +++ b/parse.y @@ -836,7 +836,13 @@ static void token_info_pop(struct parser_params*, const char *token, const rb_co %type f_kwrest f_label f_arg_asgn call_op call_op2 reswords relop dot_or_colon %token END_OF_INPUT 0 "end-of-input" %token '.' +/* escaped chars, should be ignored otherwise */ %token '\\' "backslash" +%token ' ' "escaped space" +%token '\t' "escaped horizontal tab" +%token '\f' "escaped form feed" +%token '\r' "escaped carriage return" +%token '\13' "escaped vertical tab" %token tUPLUS RUBY_TOKEN(UPLUS) "unary+" %token tUMINUS RUBY_TOKEN(UMINUS) "unary-" %token tPOW RUBY_TOKEN(POW) "**" @@ -8095,6 +8101,7 @@ parser_yylex(struct parser_params *p) dispatch_scan_event(p, tSP); goto retry; /* skip \\n */ } + if (ISSPACE(c)) return c; pushback(p, c); return '\\'; diff --git a/test/ruby/test_parse.rb b/test/ruby/test_parse.rb index e843c66242..cd1c5a3ecc 100644 --- a/test/ruby/test_parse.rb +++ b/test/ruby/test_parse.rb @@ -1129,6 +1129,27 @@ x = __ENCODING__ end end + def test_whitespace_warning + assert_raise_with_message(SyntaxError, /backslash/) do + eval("\\foo") + end + assert_raise_with_message(SyntaxError, /escaped space/) do + eval("\\ ") + end + assert_raise_with_message(SyntaxError, /escaped horizontal tab/) do + eval("\\\t") + end + assert_raise_with_message(SyntaxError, /escaped form feed/) do + eval("\\\f") + end + assert_raise_with_message(SyntaxError, /escaped carriage return/) do + assert_warn(/middle of line/) {eval("\\\r")} + end + assert_raise_with_message(SyntaxError, /escaped vertical tab/) do + eval("\\\v") + end + end + =begin def test_past_scope_variable assert_warning(/past scope/) {catch {|tag| eval("BEGIN{throw tag}; tap {a = 1}; a")}}