OLD | NEW |
1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file |
2 // for details. All rights reserved. Use of this source code is governed by a | 2 // for details. All rights reserved. Use of this source code is governed by a |
3 // BSD-style license that can be found in the LICENSE file. | 3 // BSD-style license that can be found in the LICENSE file. |
4 | 4 |
5 #include "vm/scanner.h" | 5 #include "vm/scanner.h" |
6 | 6 |
7 #include "platform/assert.h" | 7 #include "platform/assert.h" |
8 #include "vm/flags.h" | 8 #include "vm/flags.h" |
9 #include "vm/object.h" | 9 #include "vm/object.h" |
10 #include "vm/object_store.h" | 10 #include "vm/object_store.h" |
(...skipping 455 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
466 break; | 466 break; |
467 } | 467 } |
468 *value <<= 4; | 468 *value <<= 4; |
469 *value |= Utils::HexDigitToInt(c0_); | 469 *value |= Utils::HexDigitToInt(c0_); |
470 ReadChar(); | 470 ReadChar(); |
471 } | 471 } |
472 return true; | 472 return true; |
473 } | 473 } |
474 | 474 |
475 | 475 |
476 bool Scanner::ScanEscapedCodePoint(uint32_t* code_point) { | 476 void Scanner::ScanEscapedCodePoint(uint32_t* code_point) { |
477 ASSERT(c0_ == 'u' || c0_ == 'x'); | 477 ASSERT(c0_ == 'u' || c0_ == 'x'); |
478 bool is_valid; | 478 bool is_valid; |
479 if (c0_ == 'x') { | 479 if (c0_ == 'x') { |
480 is_valid = ScanHexDigits(2, code_point); | 480 is_valid = ScanHexDigits(2, code_point); |
481 } else if (c0_ == 'u' && LookaheadChar(1) != '{') { | 481 } else if (c0_ == 'u' && LookaheadChar(1) != '{') { |
482 is_valid = ScanHexDigits(4, code_point); | 482 is_valid = ScanHexDigits(4, code_point); |
483 } else { | 483 } else { |
484 ReadChar(); // Skip left curly bracket. | 484 ReadChar(); // Skip left curly bracket. |
485 is_valid = ScanHexDigits(1, 6, code_point); | 485 is_valid = ScanHexDigits(1, 6, code_point); |
486 if (is_valid) { | 486 if (is_valid) { |
487 if (c0_ != '}') { | 487 if (c0_ != '}') { |
488 ErrorMsg("expected '}' after character code"); | 488 ErrorMsg("expected '}' after character code"); |
489 return false; | 489 return; |
490 } | |
491 if (*code_point > 0x10FFFF) { | |
492 ErrorMsg("invalid code point"); | |
493 return false; | |
494 } | 490 } |
495 } | 491 } |
496 } | 492 } |
497 return is_valid; | 493 if (is_valid && |
| 494 ((*code_point > 0x10FFFF) || |
| 495 ((*code_point & 0xFFFFF800) == 0xD800))) { |
| 496 ErrorMsg("invalid code point"); |
| 497 } |
498 } | 498 } |
499 | 499 |
500 | 500 |
501 void Scanner::ScanLiteralStringChars(bool is_raw) { | 501 void Scanner::ScanLiteralStringChars(bool is_raw) { |
502 GrowableArray<uint32_t> string_chars(64); | 502 GrowableArray<uint32_t> string_chars(64); |
503 | 503 |
504 ASSERT(IsScanningString()); | 504 ASSERT(IsScanningString()); |
505 // We are at the first character of a string literal piece. A string literal | 505 // We are at the first character of a string literal piece. A string literal |
506 // can be broken up into multiple pieces by string interpolation. | 506 // can be broken up into multiple pieces by string interpolation. |
507 while (true) { | 507 while (true) { |
(...skipping 19 matching lines...) Expand all Loading... |
527 case 't': | 527 case 't': |
528 escape_char = '\t'; | 528 escape_char = '\t'; |
529 break; | 529 break; |
530 case 'b': | 530 case 'b': |
531 escape_char = '\b'; | 531 escape_char = '\b'; |
532 break; | 532 break; |
533 case 'v': | 533 case 'v': |
534 escape_char = '\v'; | 534 escape_char = '\v'; |
535 break; | 535 break; |
536 case 'u': | 536 case 'u': |
537 case 'x': | 537 case 'x': { |
538 if (!ScanEscapedCodePoint(&escape_char)) { | 538 ScanEscapedCodePoint(&escape_char); |
539 EndStringLiteral(); | |
540 return; | |
541 } | |
542 break; | 539 break; |
| 540 } |
543 default: | 541 default: |
544 if ((c0_ == '\0') || ((c0_ == '\n') && !string_is_multiline_)) { | 542 if ((c0_ == '\0') || ((c0_ == '\n') && !string_is_multiline_)) { |
545 ErrorMsg("unterminated string literal"); | 543 ErrorMsg("unterminated string literal"); |
546 EndStringLiteral(); | 544 EndStringLiteral(); |
547 return; | 545 return; |
548 } | 546 } |
549 escape_char = c0_; | 547 escape_char = c0_; |
550 break; | 548 break; |
551 } | 549 } |
552 string_chars.Add(escape_char); | 550 string_chars.Add(escape_char); |
553 } else if (c0_ == '$' && !is_raw) { | 551 } else if (c0_ == '$' && !is_raw) { |
554 // Scanned a string piece. | 552 // Scanned a string piece. |
555 ASSERT(string_chars.data() != NULL); | 553 ASSERT(string_chars.data() != NULL); |
556 // Strings are canonicalized: Allocate a symbol. | 554 // Strings are canonicalized: Allocate a symbol. |
557 current_token_.literal = &String::ZoneHandle( | 555 current_token_.literal = &String::ZoneHandle( |
558 String::NewSymbol(string_chars.data(), string_chars.length())); | 556 String::NewSymbol(string_chars.data(), string_chars.length())); |
559 current_token_.kind = Token::kSTRING; | 557 // Preserve error tokens. |
| 558 if (current_token_.kind != Token::kERROR) { |
| 559 current_token_.kind = Token::kSTRING; |
| 560 } |
560 return; | 561 return; |
561 } else if (c0_ == string_delimiter_) { | 562 } else if (c0_ == string_delimiter_) { |
562 // Check if we are at the end of the string literal. | 563 // Check if we are at the end of the string literal. |
563 if (!string_is_multiline_ || | 564 if (!string_is_multiline_ || |
564 ((LookaheadChar(1) == string_delimiter_) && | 565 ((LookaheadChar(1) == string_delimiter_) && |
565 (LookaheadChar(2) == string_delimiter_))) { | 566 (LookaheadChar(2) == string_delimiter_))) { |
566 if (string_is_multiline_) { | 567 if (string_is_multiline_) { |
567 ReadChar(); // Skip two string delimiters. | 568 ReadChar(); // Skip two string delimiters. |
568 ReadChar(); | 569 ReadChar(); |
569 } | 570 } |
570 Recognize(Token::kSTRING); | 571 // Preserve error tokens. |
571 ASSERT(string_chars.data() != NULL); | 572 if (current_token_.kind == Token::kERROR) { |
572 // Strings are canonicalized: Allocate a symbol. | 573 ReadChar(); |
573 current_token_.literal = &String::ZoneHandle( | 574 } else { |
574 String::NewSymbol(string_chars.data(), string_chars.length())); | 575 Recognize(Token::kSTRING); |
| 576 ASSERT(string_chars.data() != NULL); |
| 577 // Strings are canonicalized: Allocate a symbol. |
| 578 current_token_.literal = &String::ZoneHandle( |
| 579 String::NewSymbol(string_chars.data(), string_chars.length())); |
| 580 } |
575 EndStringLiteral(); | 581 EndStringLiteral(); |
576 return; | 582 return; |
577 } else { | 583 } else { |
578 string_chars.Add(string_delimiter_); | 584 string_chars.Add(string_delimiter_); |
579 } | 585 } |
580 } else { | 586 } else { |
581 string_chars.Add(c0_); | 587 string_chars.Add(c0_); |
582 } | 588 } |
583 ReadChar(); | 589 ReadChar(); |
584 } | 590 } |
(...skipping 329 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
914 "%c%"PRIxPTR, kPrivateKeySeparator, key_value); | 920 "%c%"PRIxPTR, kPrivateKeySeparator, key_value); |
915 const String& result = String::Handle(String::New(private_key)); | 921 const String& result = String::Handle(String::New(private_key)); |
916 return result.raw(); | 922 return result.raw(); |
917 } | 923 } |
918 | 924 |
919 | 925 |
920 void Scanner::InitOnce() { | 926 void Scanner::InitOnce() { |
921 } | 927 } |
922 | 928 |
923 } // namespace dart | 929 } // namespace dart |
OLD | NEW |