github.com/ronhuafeng/gofrontend@v0.0.0-20220715151246-ff23266b8bc5/go/lex.cc (about)

     1  // lex.cc -- Go frontend lexer.
     2  
     3  // Copyright 2009 The Go Authors. All rights reserved.
     4  // Use of this source code is governed by a BSD-style
     5  // license that can be found in the LICENSE file.
     6  
     7  #include "go-system.h"
     8  #include "go-diagnostics.h"
     9  
    10  #include "lex.h"
    11  
    12  // Manage mapping from keywords to the Keyword codes.
    13  
    14  class Keywords
    15  {
    16   public:
    17    // The structure which maps keywords to codes.
    18    struct Mapping
    19    {
    20      // Keyword string.
    21      const char* keystring;
    22      // Keyword code.
    23      Keyword keycode;
    24    };
    25  
    26    // Return the parsecode corresponding to KEYSTRING, or
    27    // KEYWORD_INVALID if it is not a keyword.
    28    Keyword
    29    keyword_to_code(const char* keyword, size_t len) const;
    30  
    31    // Return the string for a keyword.
    32    const char*
    33    keyword_to_string(Keyword) const;
    34  
    35   private:
    36    static const Mapping mapping_[];
    37    static const int count_;
    38  };
    39  
    40  // Mapping from keyword string to keyword code.  This array must be
    41  // kept in sorted order, and the order must match the Keyword enum.
    42  // Strings are looked up using bsearch.
    43  
    44  const Keywords::Mapping
    45  Keywords::mapping_[] =
    46  {
    47    { NULL,	   KEYWORD_INVALID },
    48    { "__asm__",	   KEYWORD_ASM },
    49    { "break",	   KEYWORD_BREAK },
    50    { "case",	   KEYWORD_CASE },
    51    { "chan",	   KEYWORD_CHAN },
    52    { "const",	   KEYWORD_CONST },
    53    { "continue",	   KEYWORD_CONTINUE },
    54    { "default",	   KEYWORD_DEFAULT },
    55    { "defer",	   KEYWORD_DEFER },
    56    { "else",	   KEYWORD_ELSE },
    57    { "fallthrough", KEYWORD_FALLTHROUGH },
    58    { "for",	   KEYWORD_FOR },
    59    { "func",	   KEYWORD_FUNC },
    60    { "go",	   KEYWORD_GO },
    61    { "goto",	   KEYWORD_GOTO },
    62    { "if",	   KEYWORD_IF },
    63    { "import",	   KEYWORD_IMPORT },
    64    { "interface",   KEYWORD_INTERFACE },
    65    { "map",	   KEYWORD_MAP },
    66    { "package",	   KEYWORD_PACKAGE },
    67    { "range",	   KEYWORD_RANGE },
    68    { "return",	   KEYWORD_RETURN },
    69    { "select",	   KEYWORD_SELECT },
    70    { "struct",	   KEYWORD_STRUCT },
    71    { "switch",	   KEYWORD_SWITCH },
    72    { "type",	   KEYWORD_TYPE },
    73    { "var",	   KEYWORD_VAR }
    74  };
    75  
    76  // Number of entries in the map.
    77  
    78  const int Keywords::count_ =
    79    sizeof(Keywords::mapping_) / sizeof(Keywords::mapping_[0]);
    80  
    81  // Comparison function passed to bsearch.
    82  
    83  extern "C"
    84  {
    85  
    86  struct Keywords_search_key
    87  {
    88    const char* str;
    89    size_t len;
    90  };
    91  
    92  static int
    93  keyword_compare(const void* keyv, const void* mapv)
    94  {
    95    const Keywords_search_key* key =
    96      static_cast<const Keywords_search_key*>(keyv);
    97    const Keywords::Mapping* map =
    98      static_cast<const Keywords::Mapping*>(mapv);
    99    if (map->keystring == NULL)
   100      return 1;
   101    int i = strncmp(key->str, map->keystring, key->len);
   102    if (i != 0)
   103      return i;
   104    if (map->keystring[key->len] != '\0')
   105      return -1;
   106    return 0;
   107  }
   108  
   109  } // End extern "C".
   110  
   111  // Convert a string to a keyword code.  Return KEYWORD_INVALID if the
   112  // string is not a keyword.
   113  
   114  Keyword
   115  Keywords::keyword_to_code(const char* keyword, size_t len) const
   116  {
   117    Keywords_search_key key;
   118    key.str = keyword;
   119    key.len = len;
   120    void* mapv = bsearch(&key,
   121                         this->mapping_,
   122                         this->count_,
   123                         sizeof(this->mapping_[0]),
   124                         keyword_compare);
   125    if (mapv == NULL)
   126      return KEYWORD_INVALID;
   127    Mapping* map = static_cast<Mapping*>(mapv);
   128    return map->keycode;
   129  }
   130  
   131  // Convert a keyword code to a string.
   132  
   133  const char*
   134  Keywords::keyword_to_string(Keyword code) const
   135  {
   136    go_assert(code > KEYWORD_INVALID && code < this->count_);
   137    const Mapping* map = &this->mapping_[code];
   138    go_assert(map->keycode == code);
   139    return map->keystring;
   140  }
   141  
   142  // There is one instance of the Keywords class.
   143  
   144  static Keywords keywords;
   145  
   146  // Class Token.
   147  
   148  // Make a general token.
   149  
   150  Token::Token(Classification classification, Location location)
   151    : classification_(classification), location_(location)
   152  {
   153  }
   154  
   155  // Destroy a token.
   156  
   157  Token::~Token()
   158  {
   159    this->clear();
   160  }
   161  
   162  // Clear a token--release memory.
   163  
   164  void
   165  Token::clear()
   166  {
   167    if (this->classification_ == TOKEN_INTEGER
   168        || this->classification_ == TOKEN_CHARACTER)
   169      mpz_clear(this->u_.integer_value);
   170    else if (this->classification_ == TOKEN_FLOAT
   171  	   || this->classification_ == TOKEN_IMAGINARY)
   172      mpfr_clear(this->u_.float_value);
   173  }
   174  
   175  // Construct a token.
   176  
   177  Token::Token(const Token& tok)
   178    : classification_(tok.classification_), location_(tok.location_)
   179  {
   180    switch (this->classification_)
   181      {
   182      case TOKEN_INVALID:
   183      case TOKEN_EOF:
   184        break;
   185      case TOKEN_KEYWORD:
   186        this->u_.keyword = tok.u_.keyword;
   187        break;
   188      case TOKEN_IDENTIFIER:
   189      case TOKEN_STRING:
   190        this->u_.string_value = tok.u_.string_value;
   191        break;
   192      case TOKEN_OPERATOR:
   193        this->u_.op = tok.u_.op;
   194        break;
   195      case TOKEN_CHARACTER:
   196      case TOKEN_INTEGER:
   197        mpz_init_set(this->u_.integer_value, tok.u_.integer_value);
   198        break;
   199      case TOKEN_FLOAT:
   200      case TOKEN_IMAGINARY:
   201        mpfr_init_set(this->u_.float_value, tok.u_.float_value, MPFR_RNDN);
   202        break;
   203      default:
   204        go_unreachable();
   205      }
   206  }
   207  
   208  // Assign to a token.
   209  
   210  Token&
   211  Token::operator=(const Token& tok)
   212  {
   213    this->clear();
   214    this->classification_ = tok.classification_;
   215    this->location_ = tok.location_;
   216    switch (tok.classification_)
   217      {
   218      case TOKEN_INVALID:
   219      case TOKEN_EOF:
   220        break;
   221      case TOKEN_KEYWORD:
   222        this->u_.keyword = tok.u_.keyword;
   223        break;
   224      case TOKEN_IDENTIFIER:
   225        this->u_.identifier_value.name = tok.u_.identifier_value.name;
   226        this->u_.identifier_value.is_exported =
   227  	tok.u_.identifier_value.is_exported;
   228        break;
   229      case TOKEN_STRING:
   230        this->u_.string_value = tok.u_.string_value;
   231        break;
   232      case TOKEN_OPERATOR:
   233        this->u_.op = tok.u_.op;
   234        break;
   235      case TOKEN_CHARACTER:
   236      case TOKEN_INTEGER:
   237        mpz_init_set(this->u_.integer_value, tok.u_.integer_value);
   238        break;
   239      case TOKEN_FLOAT:
   240      case TOKEN_IMAGINARY:
   241        mpfr_init_set(this->u_.float_value, tok.u_.float_value, MPFR_RNDN);
   242        break;
   243      default:
   244        go_unreachable();
   245      }
   246    return *this;
   247  }
   248  
   249  // Print the token for debugging.
   250  
   251  void
   252  Token::print(FILE* file) const
   253  {
   254    switch (this->classification_)
   255      {
   256      case TOKEN_INVALID:
   257        fprintf(file, "invalid");
   258        break;
   259      case TOKEN_EOF:
   260        fprintf(file, "EOF");
   261        break;
   262      case TOKEN_KEYWORD:
   263        fprintf(file, "keyword %s", keywords.keyword_to_string(this->u_.keyword));
   264        break;
   265      case TOKEN_IDENTIFIER:
   266        fprintf(file, "identifier \"%s\"", this->u_.string_value->c_str());
   267        break;
   268      case TOKEN_STRING:
   269        fprintf(file, "quoted string \"%s\"", this->u_.string_value->c_str());
   270        break;
   271      case TOKEN_CHARACTER:
   272        fprintf(file, "character ");
   273        mpz_out_str(file, 10, this->u_.integer_value);
   274        break;
   275      case TOKEN_INTEGER:
   276        fprintf(file, "integer ");
   277        mpz_out_str(file, 10, this->u_.integer_value);
   278        break;
   279      case TOKEN_FLOAT:
   280        fprintf(file, "float ");
   281        mpfr_out_str(file, 10, 0, this->u_.float_value, MPFR_RNDN);
   282        break;
   283      case TOKEN_IMAGINARY:
   284        fprintf(file, "imaginary ");
   285        mpfr_out_str(file, 10, 0, this->u_.float_value, MPFR_RNDN);
   286        break;
   287      case TOKEN_OPERATOR:
   288        fprintf(file, "operator ");
   289        switch (this->u_.op)
   290  	{
   291  	case OPERATOR_INVALID:
   292  	  fprintf(file, "invalid");
   293  	  break;
   294  	case OPERATOR_OROR:
   295  	  fprintf(file, "||");
   296  	  break;
   297  	case OPERATOR_ANDAND:
   298  	  fprintf(file, "&&");
   299  	  break;
   300  	case OPERATOR_EQEQ:
   301  	  fprintf(file, "==");
   302  	  break;
   303  	case OPERATOR_NOTEQ:
   304  	  fprintf(file, "!=");
   305  	  break;
   306  	case OPERATOR_LT:
   307  	  fprintf(file, "<");
   308  	  break;
   309  	case OPERATOR_LE:
   310  	  fprintf(file, "<=");
   311  	  break;
   312  	case OPERATOR_GT:
   313  	  fprintf(file, ">");
   314  	  break;
   315  	case OPERATOR_GE:
   316  	  fprintf(file, ">=");
   317  	  break;
   318  	case OPERATOR_PLUS:
   319  	  fprintf(file, "+");
   320  	  break;
   321  	case OPERATOR_MINUS:
   322  	  fprintf(file, "-");
   323  	  break;
   324  	case OPERATOR_OR:
   325  	  fprintf(file, "|");
   326  	  break;
   327  	case OPERATOR_XOR:
   328  	  fprintf(file, "^");
   329  	  break;
   330  	case OPERATOR_MULT:
   331  	  fprintf(file, "*");
   332  	  break;
   333  	case OPERATOR_DIV:
   334  	  fprintf(file, "/");
   335  	  break;
   336  	case OPERATOR_MOD:
   337  	  fprintf(file, "%%");
   338  	  break;
   339  	case OPERATOR_LSHIFT:
   340  	  fprintf(file, "<<");
   341  	  break;
   342  	case OPERATOR_RSHIFT:
   343  	  fprintf(file, ">>");
   344  	  break;
   345  	case OPERATOR_AND:
   346  	  fprintf(file, "&");
   347  	  break;
   348  	case OPERATOR_BITCLEAR:
   349  	  fprintf(file, "&^");
   350  	  break;
   351  	case OPERATOR_NOT:
   352  	  fprintf(file, "!");
   353  	  break;
   354  	case OPERATOR_CHANOP:
   355  	  fprintf(file, "<-");
   356  	  break;
   357  	case OPERATOR_EQ:
   358  	  fprintf(file, "=");
   359  	  break;
   360  	case OPERATOR_PLUSEQ:
   361  	  fprintf(file, "+=");
   362  	  break;
   363  	case OPERATOR_MINUSEQ:
   364  	  fprintf(file, "-=");
   365  	  break;
   366  	case OPERATOR_OREQ:
   367  	  fprintf(file, "|=");
   368  	  break;
   369  	case OPERATOR_XOREQ:
   370  	  fprintf(file, "^=");
   371  	  break;
   372  	case OPERATOR_MULTEQ:
   373  	  fprintf(file, "*=");
   374  	  break;
   375  	case OPERATOR_DIVEQ:
   376  	  fprintf(file, "/=");
   377  	  break;
   378  	case OPERATOR_MODEQ:
   379  	  fprintf(file, "%%=");
   380  	  break;
   381  	case OPERATOR_LSHIFTEQ:
   382  	  fprintf(file, "<<=");
   383  	  break;
   384  	case OPERATOR_RSHIFTEQ:
   385  	  fprintf(file, ">>=");
   386  	  break;
   387  	case OPERATOR_ANDEQ:
   388  	  fprintf(file, "&=");
   389  	  break;
   390  	case OPERATOR_BITCLEAREQ:
   391  	  fprintf(file, "&^=");
   392  	  break;
   393  	case OPERATOR_PLUSPLUS:
   394  	  fprintf(file, "++");
   395  	  break;
   396  	case OPERATOR_MINUSMINUS:
   397  	  fprintf(file, "--");
   398  	  break;
   399  	case OPERATOR_COLON:
   400  	  fprintf(file, ":");
   401  	  break;
   402  	case OPERATOR_COLONEQ:
   403  	  fprintf(file, ":=");
   404  	  break;
   405  	case OPERATOR_SEMICOLON:
   406  	  fprintf(file, ";");
   407  	  break;
   408  	case OPERATOR_DOT:
   409  	  fprintf(file, ".");
   410  	  break;
   411  	case OPERATOR_COMMA:
   412  	  fprintf(file, ",");
   413  	  break;
   414  	case OPERATOR_LPAREN:
   415  	  fprintf(file, "(");
   416  	  break;
   417  	case OPERATOR_RPAREN:
   418  	  fprintf(file, ")");
   419  	  break;
   420  	case OPERATOR_LCURLY:
   421  	  fprintf(file, "{");
   422  	  break;
   423  	case OPERATOR_RCURLY:
   424  	  fprintf(file, "}");
   425  	  break;
   426  	case OPERATOR_LSQUARE:
   427  	  fprintf(file, "[");
   428  	  break;
   429  	case OPERATOR_RSQUARE:
   430  	  fprintf(file, "]");
   431  	  break;
   432  	default:
   433  	  go_unreachable();
   434  	}
   435        break;
   436      default:
   437        go_unreachable();
   438      }
   439  }
   440  
   441  // Class Lex.
   442  
   443  Lex::Lex(const char* input_file_name, FILE* input_file, Linemap* linemap)
   444    : input_file_name_(input_file_name), input_file_(input_file),
   445      linemap_(linemap), linebuf_(NULL), linebufsize_(120), linesize_(0),
   446      lineoff_(0), lineno_(0), add_semi_at_eol_(false), pragmas_(0),
   447      extern_(), linknames_(NULL)
   448  {
   449    this->linebuf_ = new char[this->linebufsize_];
   450    this->linemap_->start_file(input_file_name, 0);
   451  }
   452  
   453  Lex::~Lex()
   454  {
   455    delete[] this->linebuf_;
   456  }
   457  
   458  // Read a new line from the file.
   459  
   460  ssize_t
   461  Lex::get_line()
   462  {
   463    char* buf = this->linebuf_;
   464    size_t size = this->linebufsize_;
   465  
   466    FILE* file = this->input_file_;
   467    size_t cur = 0;
   468    while (true)
   469      {
   470        int c = getc(file);
   471        if (c == EOF)
   472  	{
   473  	  if (cur == 0)
   474  	    return -1;
   475  	  break;
   476  	}
   477        if (cur + 1 >= size)
   478  	{
   479  	  size_t ns = 2 * size + 1;
   480  	  if (ns < size || static_cast<ssize_t>(ns) < 0)
   481  	    go_error_at(this->location(), "out of memory");
   482  	  char* nb = new char[ns];
   483  	  memcpy(nb, buf, cur);
   484  	  delete[] buf;
   485  	  buf = nb;
   486  	  size = ns;
   487  	}
   488        buf[cur] = c;
   489        ++cur;
   490  
   491        if (c == '\n')
   492  	break;
   493      }
   494  
   495    buf[cur] = '\0';
   496  
   497    this->linebuf_ = buf;
   498    this->linebufsize_ = size;
   499  
   500    return cur;
   501  }
   502  
   503  // See if we need to read a new line.  Return true if there is a new
   504  // line, false if we are at EOF.
   505  
   506  bool
   507  Lex::require_line()
   508  {
   509    if (this->lineoff_ < this->linesize_)
   510      return true;
   511  
   512    ssize_t got = this->get_line();
   513    if (got < 0)
   514      return false;
   515    ++this->lineno_;
   516    this->linesize_= got;
   517    this->lineoff_ = 0;
   518  
   519    this->linemap_->start_line(this->lineno_, this->linesize_);
   520  
   521    return true;
   522  }
   523  
   524  // Get the current location.
   525  
   526  Location
   527  Lex::location() const
   528  {
   529    return this->linemap_->get_location(this->lineoff_ + 1);
   530  }
   531  
   532  // Get a location slightly before the current one.  This is used for
   533  // slightly more efficient handling of operator tokens.
   534  
   535  Location
   536  Lex::earlier_location(int chars) const
   537  {
   538    return this->linemap_->get_location(this->lineoff_ + 1 - chars);
   539  }
   540  
   541  // Get the next token.
   542  
   543  Token
   544  Lex::next_token()
   545  {
   546    bool saw_cpp_comment = false;
   547    while (true)
   548      {
   549        if (!this->require_line())
   550  	{
   551  	  bool add_semi_at_eol = this->add_semi_at_eol_;
   552  	  this->add_semi_at_eol_ = false;
   553  	  if (add_semi_at_eol)
   554  	    return this->make_operator(OPERATOR_SEMICOLON, 1);
   555  	  return this->make_eof_token();
   556  	}
   557  
   558        if (!saw_cpp_comment)
   559  	this->extern_.clear();
   560        saw_cpp_comment = false;
   561  
   562        const char* p = this->linebuf_ + this->lineoff_;
   563        const char* pend = this->linebuf_ + this->linesize_;
   564  
   565        while (p < pend)
   566  	{
   567  	  unsigned char cc = *p;
   568  	  switch (cc)
   569  	    {
   570  	    case ' ': case '\t': case '\r':
   571  	      ++p;
   572  	      // Skip whitespace quickly.
   573  	      while (*p == ' ' || *p == '\t' || *p == '\r')
   574  		++p;
   575  	      break;
   576  
   577  	    case '\n':
   578  	      {
   579  		++p;
   580  		bool add_semi_at_eol = this->add_semi_at_eol_;
   581  		this->add_semi_at_eol_ = false;
   582  		if (add_semi_at_eol)
   583  		  {
   584  		    this->lineoff_ = p - this->linebuf_;
   585  		    return this->make_operator(OPERATOR_SEMICOLON, 1);
   586  		  }
   587  	      }
   588  	      break;
   589  
   590  	    case '/':
   591  	      if (p[1] == '/')
   592  		{
   593  		  this->lineoff_ = p + 2 - this->linebuf_;
   594  		  this->skip_cpp_comment();
   595  		  p = pend;
   596  		  if (p[-1] == '\n' && this->add_semi_at_eol_)
   597  		    --p;
   598  		  saw_cpp_comment = true;
   599  		}
   600  	      else if (p[1] == '*')
   601  		{
   602  		  this->lineoff_ = p + 2 - this->linebuf_;
   603  		  Location location = this->location();
   604                    bool found_newline = false;
   605  		  if (!this->skip_c_comment(&found_newline))
   606  		    return Token::make_invalid_token(location);
   607                    if (found_newline && this->add_semi_at_eol_)
   608                      {
   609                        this->add_semi_at_eol_ = false;
   610                        return this->make_operator(OPERATOR_SEMICOLON, 1);
   611                      }
   612  		  p = this->linebuf_ + this->lineoff_;
   613  		  pend = this->linebuf_ + this->linesize_;
   614  		}
   615  	      else if (p[1] == '=')
   616  		{
   617  		  this->add_semi_at_eol_ = false;
   618  		  this->lineoff_ = p + 2 - this->linebuf_;
   619  		  return this->make_operator(OPERATOR_DIVEQ, 2);
   620  		}
   621  	      else
   622  		{
   623  		  this->add_semi_at_eol_ = false;
   624  		  this->lineoff_ = p + 1 - this->linebuf_;
   625  		  return this->make_operator(OPERATOR_DIV, 1);
   626  		}
   627  	      break;
   628  
   629  	    case 'A': case 'B': case 'C': case 'D': case 'E': case 'F':
   630  	    case 'G': case 'H': case 'I': case 'J': case 'K': case 'L':
   631  	    case 'M': case 'N': case 'O': case 'P': case 'Q': case 'R':
   632  	    case 'S': case 'T': case 'U': case 'V': case 'W': case 'X':
   633  	    case 'Y': case 'Z':
   634  	    case 'a': case 'b': case 'c': case 'd': case 'e': case 'f':
   635  	    case 'g': case 'h': case 'i': case 'j': case 'k': case 'l':
   636  	    case 'm': case 'n': case 'o': case 'p': case 'q': case 'r':
   637  	    case 's': case 't': case 'u': case 'v': case 'w': case 'x':
   638  	    case 'y': case 'z':
   639  	    case '_':
   640  	      this->lineoff_ = p - this->linebuf_;
   641  	      return this->gather_identifier();
   642  
   643  	    case '0': case '1': case '2': case '3': case '4':
   644  	    case '5': case '6': case '7': case '8': case '9':
   645  	      this->add_semi_at_eol_ = true;
   646  	      this->lineoff_ = p - this->linebuf_;
   647  	      return this->gather_number();
   648  
   649  	    case '\'':
   650  	      this->add_semi_at_eol_ = true;
   651  	      this->lineoff_ = p - this->linebuf_;
   652  	      return this->gather_character();
   653  
   654  	    case '"':
   655  	      this->add_semi_at_eol_ = true;
   656  	      this->lineoff_ = p - this->linebuf_;
   657  	      return this->gather_string();
   658  
   659  	    case '`':
   660  	      this->add_semi_at_eol_ = true;
   661  	      this->lineoff_ = p - this->linebuf_;
   662  	      return this->gather_raw_string();
   663  
   664  	    case '<':
   665  	    case '>':
   666  	    case '&':
   667  	      if (p + 2 < pend)
   668  		{
   669  		  this->add_semi_at_eol_ = false;
   670  		  Operator op = this->three_character_operator(cc, p[1], p[2]);
   671  		  if (op != OPERATOR_INVALID)
   672  		    {
   673  		      this->lineoff_ = p + 3 - this->linebuf_;
   674  		      return this->make_operator(op, 3);
   675  		    }
   676  		}
   677  	      // Fall through.
   678  	    case '|':
   679  	    case '=':
   680  	    case '!':
   681  	    case '+':
   682  	    case '-':
   683  	    case '^':
   684  	    case '*':
   685  	      // '/' handled above.
   686  	    case '%':
   687  	    case ':':
   688  	    case ';':
   689  	    case ',':
   690  	    case '(': case ')':
   691  	    case '{': case '}':
   692  	    case '[': case ']':
   693  	      {
   694  		this->add_semi_at_eol_ = false;
   695  		Operator op = this->two_character_operator(cc, p[1]);
   696  		int chars;
   697  		if (op != OPERATOR_INVALID)
   698  		  {
   699  		    ++p;
   700  		    chars = 2;
   701  		  }
   702  		else
   703  		  {
   704  		    op = this->one_character_operator(cc);
   705  		    chars = 1;
   706  		  }
   707  		this->lineoff_ = p + 1 - this->linebuf_;
   708  		return this->make_operator(op, chars);
   709  	      }
   710  
   711  	    case '.':
   712  	      if (p[1] >= '0' && p[1] <= '9')
   713  		{
   714  		  this->add_semi_at_eol_ = true;
   715  		  this->lineoff_ = p - this->linebuf_;
   716  		  return this->gather_number();
   717  		}
   718  	      if (p[1] == '.' && p[2] == '.')
   719  		{
   720  		  this->add_semi_at_eol_ = false;
   721  		  this->lineoff_ = p + 3 - this->linebuf_;
   722  		  return this->make_operator(OPERATOR_ELLIPSIS, 3);
   723  		}
   724  	      this->add_semi_at_eol_ = false;
   725  	      this->lineoff_ = p + 1 - this->linebuf_;
   726  	      return this->make_operator(OPERATOR_DOT, 1);
   727  
   728  	    default:
   729  	      {
   730  		unsigned int ci;
   731  		bool issued_error;
   732  		this->lineoff_ = p - this->linebuf_;
   733  		const char *pnext = this->advance_one_utf8_char(p, &ci,
   734  								&issued_error);
   735  
   736  		// Ignore byte order mark at start of file.
   737  		if (ci == 0xfeff)
   738  		  {
   739  		    p = pnext;
   740  		    break;
   741  		  }
   742  
   743  		if (Lex::is_unicode_letter(ci))
   744  		  return this->gather_identifier();
   745  
   746  		if (!issued_error && Lex::is_unicode_digit(ci))
   747  		  {
   748  		    go_error_at(this->location(),
   749  				"identifier cannot begin with digit");
   750  		    issued_error = true;
   751  		  }
   752  
   753  		if (!issued_error)
   754  		  go_error_at(this->location(),
   755  			      "invalid character 0x%x in input file",
   756  			      ci);
   757  
   758  		p = pend;
   759  
   760  		break;
   761  	      }
   762  	    }
   763  	}
   764  
   765        this->lineoff_ = p - this->linebuf_;
   766      }
   767  }
   768  
   769  // Fetch one UTF-8 character from a string.  Set *VALUE to the value.
   770  // Return the number of bytes read from the string.  Returns 0 if the
   771  // string does not point to a valid UTF-8 character.
   772  
   773  int
   774  Lex::fetch_char(const char* p, unsigned int* value)
   775  {
   776    unsigned char c = *p;
   777    if (c <= 0x7f)
   778      {
   779        *value = c;
   780        return 1;
   781      }
   782    else if ((c & 0xe0) == 0xc0
   783  	   && (p[1] & 0xc0) == 0x80)
   784      {
   785        *value = (((c & 0x1f) << 6)
   786  		+ (p[1] & 0x3f));
   787        if (*value <= 0x7f)
   788  	{
   789  	  *value = 0xfffd;
   790  	  return 0;
   791  	}
   792        return 2;
   793      }
   794    else if ((c & 0xf0) == 0xe0
   795  	   && (p[1] & 0xc0) == 0x80
   796  	   && (p[2] & 0xc0) == 0x80)
   797      {
   798        *value = (((c & 0xf) << 12)
   799  		+ ((p[1] & 0x3f) << 6)
   800  		+ (p[2] & 0x3f));
   801        if (*value <= 0x7ff)
   802  	{
   803  	  *value = 0xfffd;
   804  	  return 0;
   805  	}
   806        return 3;
   807      }
   808    else if ((c & 0xf8) == 0xf0
   809  	   && (p[1] & 0xc0) == 0x80
   810  	   && (p[2] & 0xc0) == 0x80
   811  	   && (p[3] & 0xc0) == 0x80)
   812      {
   813        *value = (((c & 0x7) << 18)
   814  		+ ((p[1] & 0x3f) << 12)
   815  		+ ((p[2] & 0x3f) << 6)
   816  		+ (p[3] & 0x3f));
   817        if (*value <= 0xffff)
   818  	{
   819  	  *value = 0xfffd;
   820  	  return 0;
   821  	}
   822        return 4;
   823      }
   824    else
   825      {
   826        /* Invalid encoding. Return the Unicode replacement
   827  	 character.  */
   828        *value = 0xfffd;
   829        return 0;
   830      }
   831  }
   832  
   833  // Advance one UTF-8 character.  Return the pointer beyond the
   834  // character.  Set *VALUE to the value.  Set *ISSUED_ERROR if an error
   835  // was issued.
   836  
   837  const char*
   838  Lex::advance_one_utf8_char(const char* p, unsigned int* value,
   839  			   bool* issued_error)
   840  {
   841    *issued_error = false;
   842  
   843    if (*p == '\0')
   844      {
   845        go_error_at(this->location(), "invalid NUL byte");
   846        *issued_error = true;
   847        *value = 0;
   848        return p + 1;
   849      }
   850  
   851    int adv = Lex::fetch_char(p, value);
   852    if (adv == 0)
   853      {
   854        go_error_at(this->location(), "invalid UTF-8 encoding");
   855        *issued_error = true;
   856        return p + 1;
   857      }
   858  
   859    // Warn about byte order mark, except at start of file.
   860    if (*value == 0xfeff && (this->lineno_ != 1 || this->lineoff_ != 0))
   861      {
   862        go_error_at(this->location(), "Unicode (UTF-8) BOM in middle of file");
   863        *issued_error = true;
   864      }
   865  
   866    return p + adv;
   867  }
   868  
   869  // Pick up an identifier.
   870  
   871  Token
   872  Lex::gather_identifier()
   873  {
   874    const char* pstart = this->linebuf_ + this->lineoff_;
   875    const char* p = pstart;
   876    const char* pend = this->linebuf_ + this->linesize_;
   877    bool is_first = true;
   878    bool is_exported = false;
   879    bool has_non_ascii_char = false;
   880    std::string buf;
   881    while (p < pend)
   882      {
   883        unsigned char cc = *p;
   884        if (cc <= 0x7f)
   885  	{
   886  	  if ((cc < 'A' || cc > 'Z')
   887  	      && (cc < 'a' || cc > 'z')
   888  	      && cc != '_'
   889  	      && (cc < '0' || cc > '9'))
   890  	    {
   891  	      // Check for an invalid character here, as we get better
   892  	      // error behaviour if we swallow them as part of the
   893  	      // identifier we are building.
   894  	      if ((cc >= ' ' && cc < 0x7f)
   895  		  || cc == '\t'
   896  		  || cc == '\r'
   897  		  || cc == '\n')
   898  		break;
   899  
   900  	      this->lineoff_ = p - this->linebuf_;
   901  	      go_error_at(this->location(),
   902  			  "invalid character 0x%x in identifier",
   903  			  cc);
   904  	      if (!has_non_ascii_char)
   905  		{
   906  		  buf.assign(pstart, p - pstart);
   907  		  has_non_ascii_char = true;
   908  		}
   909  	      if (!Lex::is_invalid_identifier(buf))
   910  		buf.append("$INVALID$");
   911  	    }
   912  	  ++p;
   913  	  if (is_first)
   914  	    {
   915  	      is_exported = cc >= 'A' && cc <= 'Z';
   916  	      is_first = false;
   917  	    }
   918  	  if (has_non_ascii_char)
   919  	    buf.push_back(cc);
   920  	}
   921        else
   922  	{
   923  	  unsigned int ci;
   924  	  bool issued_error;
   925  	  this->lineoff_ = p - this->linebuf_;
   926  	  const char* pnext = this->advance_one_utf8_char(p, &ci,
   927  							  &issued_error);
   928  	  bool is_invalid = false;
   929  	  if (!Lex::is_unicode_letter(ci) && !Lex::is_unicode_digit(ci))
   930  	    {
   931  	      // There is no valid place for a non-ASCII character
   932  	      // other than an identifier, so we get better error
   933  	      // handling behaviour if we swallow this character after
   934  	      // giving an error.
   935  	      if (!issued_error)
   936  		go_error_at(this->location(),
   937  			    "invalid character 0x%x in identifier",
   938  			    ci);
   939  	      is_invalid = true;
   940  	    }
   941  	  if (is_first)
   942  	    {
   943  	      is_exported = Lex::is_unicode_uppercase(ci);
   944  	      is_first = false;
   945  	    }
   946  	  if (!has_non_ascii_char)
   947  	    {
   948  	      buf.assign(pstart, p - pstart);
   949  	      has_non_ascii_char = true;
   950  	    }
   951  	  if (is_invalid && !Lex::is_invalid_identifier(buf))
   952  	    buf.append("$INVALID$");
   953  	  buf.append(p, pnext - p);
   954  	  p = pnext;
   955  	}
   956      }
   957    Location location = this->location();
   958    this->add_semi_at_eol_ = true;
   959    this->lineoff_ = p - this->linebuf_;
   960    if (has_non_ascii_char)
   961      return Token::make_identifier_token(buf, is_exported, location);
   962    else
   963      {
   964        Keyword code = keywords.keyword_to_code(pstart, p - pstart);
   965        if (code == KEYWORD_INVALID)
   966  	return Token::make_identifier_token(std::string(pstart, p - pstart),
   967  					    is_exported, location);
   968        else
   969  	{
   970  	  switch (code)
   971  	    {
   972  	    case KEYWORD_BREAK:
   973  	    case KEYWORD_CONTINUE:
   974  	    case KEYWORD_FALLTHROUGH:
   975  	    case KEYWORD_RETURN:
   976  	      break;
   977  	    default:
   978  	      this->add_semi_at_eol_ = false;
   979  	      break;
   980  	    }
   981  	  return Token::make_keyword_token(code, location);
   982  	}
   983      }
   984  }
   985  
   986  // Return whether C is a hex digit.
   987  
   988  bool
   989  Lex::is_hex_digit(char c)
   990  {
   991    return ((c >= '0' && c <= '9')
   992  	  || (c >= 'A' && c <= 'F')
   993  	  || (c >= 'a' && c <= 'f'));
   994  }
   995  
   996  // Return whether C is a valid digit in BASE.
   997  
   998  bool
   999  Lex::is_base_digit(int base, char c)
  1000  {
  1001    switch (base)
  1002      {
  1003      case 2:
  1004        return c == '0' || c == '1';
  1005      case 8:
  1006        return c >= '0' && c <= '7';
  1007      case 10:
  1008        return c >= '0' && c <= '9';
  1009      case 16:
  1010        return Lex::is_hex_digit(c);
  1011      default:
  1012        go_unreachable();
  1013      }
  1014  }
  1015  
  1016  // not a hex value
  1017  #define NHV 100
  1018  
  1019  // for use by Lex::hex_val
  1020  static const unsigned char hex_value_lookup_table[256] =
  1021  {
  1022    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, // NUL SOH STX ETX EOT ENQ ACK BEL
  1023    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //  BS  HT  LF  VT  FF  CR  SO  SI
  1024    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, // DLE DC1 DC2 DC3 DC4 NAK SYN ETB
  1025    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, // CAN  EM SUB ESC  FS  GS  RS  US
  1026    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //  SP   !   "   #   $   %   &   '
  1027    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //   (   )   *   +   ,   -   .   /
  1028      0,   1,   2,   3,   4,   5,   6,   7, //   0   1   2   3   4   5   6   7
  1029      8,   9, NHV, NHV, NHV, NHV, NHV, NHV, //   8   9   :   ;   <   =   >   ?
  1030    NHV,  10,  11,  12,  13,  14,  15, NHV, //   @   A   B   C   D   E   F   G
  1031    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //   H   I   J   K   L   M   N   O
  1032    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //   P   Q   R   S   T   U   V   W
  1033    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //   X   Y   Z   [   \   ]   ^   _
  1034    NHV,  10,  11,  12,  13,  14,  15, NHV, //   `   a   b   c   d   e   f   g
  1035    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //   h   i   j   k   l   m   n   o
  1036    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //   p   q   r   s   t   u   v   w
  1037    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //   x   y   z   {   |   }   ~
  1038    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //
  1039    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //
  1040    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //
  1041    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //
  1042    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //
  1043    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //
  1044    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //
  1045    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //
  1046    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //
  1047    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //
  1048    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //
  1049    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //
  1050    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //
  1051    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //
  1052    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV, //
  1053    NHV, NHV, NHV, NHV, NHV, NHV, NHV, NHV //
  1054  };
  1055  
  1056  unsigned
  1057  Lex::hex_val(char c)
  1058  {
  1059    return hex_value_lookup_table[static_cast<unsigned char>(c)];
  1060  }
  1061  
  1062  // Return whether an exponent could start at P, in base BASE.
  1063  
  1064  bool
  1065  Lex::could_be_exponent(int base, const char* p, const char* pend)
  1066  {
  1067    switch (base)
  1068      {
  1069      case 10:
  1070        if (*p != 'e' && *p != 'E')
  1071  	return false;
  1072        break;
  1073      case 16:
  1074        if (*p != 'p' && *p != 'P')
  1075  	return false;
  1076        break;
  1077      default:
  1078        go_unreachable();
  1079      }
  1080    ++p;
  1081    if (p >= pend)
  1082      return false;
  1083    if (*p == '+' || *p == '-')
  1084      {
  1085        ++p;
  1086        if (p >= pend)
  1087  	return false;
  1088      }
  1089    return *p >= '0' && *p <= '9';
  1090  }
  1091  
  1092  // Pick up a number.
  1093  
  1094  Token
  1095  Lex::gather_number()
  1096  {
  1097    const char* pstart = this->linebuf_ + this->lineoff_;
  1098    const char* p = pstart;
  1099    const char* pend = this->linebuf_ + this->linesize_;
  1100  
  1101    Location location = this->location();
  1102  
  1103    int base = 10;
  1104    std::string num;
  1105    if (*p == '0')
  1106      {
  1107        int basecheck;
  1108        int off;
  1109        if (p[1] == 'x' || p[1] == 'X')
  1110  	{
  1111  	  base = 16;
  1112  	  basecheck = 16;
  1113  	  off = 2;
  1114  	}
  1115        else if (p[1] == 'o' || p[1] == 'O')
  1116  	{
  1117  	  base = 8;
  1118  	  basecheck = 8;
  1119  	  off = 2;
  1120  	}
  1121        else if (p[1] == 'b' || p[1] == 'B')
  1122  	{
  1123  	  base = 2;
  1124  	  basecheck = 2;
  1125  	  off = 2;
  1126  	}
  1127        else
  1128  	{
  1129  	  // Old style octal literal.  May also be the start of a
  1130  	  // floating-point number (e.g., 09.2, 09e2) or an imaginary
  1131  	  // literal (e.g., 09i), so we have to accept decimal digits.
  1132  	  base = 8;
  1133  	  basecheck = 10;
  1134  	  off = 0;
  1135  	}
  1136  
  1137        p += off;
  1138        if (*p == '_' && Lex::is_base_digit(basecheck, p[1]))
  1139  	++p;
  1140  
  1141        while (Lex::is_base_digit(basecheck, *p))
  1142  	{
  1143  	  num.push_back(*p);
  1144  	  ++p;
  1145  	  if (*p == '_' && Lex::is_base_digit(basecheck, p[1]))
  1146  	    ++p;
  1147  	}
  1148  
  1149        // We must see at least one valid digit, except for a case like
  1150        // 0x.0p1.
  1151        if (num.length() == 0 && (base != 16 || *p != '.'))
  1152  	{
  1153  	  go_error_at(this->location(), "invalid numeric literal");
  1154  	  this->lineoff_ = p - this->linebuf_;
  1155  	  mpz_t val;
  1156  	  mpz_init_set_ui(val, 0);
  1157  	  Token ret = Token::make_integer_token(val, location);
  1158  	  mpz_clear(val);
  1159  	  return ret;
  1160  	}
  1161  
  1162        bool is_float = false;
  1163        // A number that looks like an old-style octal literal might
  1164        // actually be the beginning of a floating-point or imaginary
  1165        // literal, in which case the value is decimal digits.  Handle
  1166        // that case below by treating the leading '0' as decimal.
  1167        if (off == 0
  1168  	  && (*p == '.' || *p == 'i' || Lex::could_be_exponent(10, p, pend)))
  1169  	{
  1170  	  is_float = true;
  1171  	  base = 10;
  1172  	}
  1173        else if (base == 16
  1174  	       && (*p == '.' || Lex::could_be_exponent(16, p, pend)))
  1175  	is_float = true;
  1176  
  1177        if (!is_float)
  1178  	{
  1179  	  mpz_t val;
  1180  	  int r = mpz_init_set_str(val, num.c_str(), base);
  1181            if (r != 0)
  1182              {
  1183  	      const char *errword;
  1184  	      switch (base)
  1185  		{
  1186  		case 2:
  1187  		  errword = "binary";
  1188  		  break;
  1189  		case 8:
  1190  		  errword = "octal";
  1191  		  break;
  1192  		case 16:
  1193  		  errword = "hex";
  1194  		  break;
  1195  		default:
  1196  		  go_unreachable();
  1197  		}
  1198  	      go_error_at(this->location(), "invalid %s literal", errword);
  1199              }
  1200  
  1201  	  bool is_imaginary = *p == 'i';
  1202  	  if (is_imaginary)
  1203  	    ++p;
  1204  
  1205  	  this->lineoff_ = p - this->linebuf_;
  1206  
  1207  	  if (*p == 'e' || *p == 'E' || *p == 'p' || *p == 'P')
  1208  	    {
  1209  	      go_error_at(location,
  1210  			  "invalid prefix for floating constant");
  1211  	      this->skip_exponent();
  1212  	    }
  1213  
  1214  	  if (!is_imaginary)
  1215  	    {
  1216  	      Token ret = Token::make_integer_token(val, location);
  1217  	      mpz_clear(val);
  1218  	      return ret;
  1219  	    }
  1220  	  else
  1221  	    {
  1222  	      mpfr_t ival;
  1223  	      mpfr_init_set_z(ival, val, MPFR_RNDN);
  1224  	      mpz_clear(val);
  1225  	      Token ret = Token::make_imaginary_token(ival, location);
  1226  	      mpfr_clear(ival);
  1227  	      return ret;
  1228  	    }
  1229  	}
  1230      }
  1231  
  1232    while (p < pend)
  1233      {
  1234        if (*p == '_' && p[1] >= '0' && p[1] <= '9')
  1235  	++p;
  1236        else if (*p < '0' || *p > '9')
  1237  	break;
  1238        num.push_back(*p);
  1239        ++p;
  1240      }
  1241  
  1242    if (*p != '.' && *p != 'i' && !Lex::could_be_exponent(base, p, pend))
  1243      {
  1244        mpz_t val;
  1245        int r = mpz_init_set_str(val, num.c_str(), 10);
  1246        go_assert(r == 0);
  1247  
  1248        this->lineoff_ = p - this->linebuf_;
  1249  
  1250        if (*p == 'e' || *p == 'E' || *p == 'p' || *p == 'P')
  1251  	{
  1252  	  go_error_at(location,
  1253  		      "invalid prefix for floating constant");
  1254  	  this->skip_exponent();
  1255  	}
  1256  
  1257        Token ret = Token::make_integer_token(val, location);
  1258        mpz_clear(val);
  1259        return ret;
  1260      }
  1261  
  1262    if (*p != 'i')
  1263      {
  1264        bool dot = *p == '.';
  1265  
  1266        num.push_back(*p);
  1267        ++p;
  1268  
  1269        if (!dot)
  1270  	{
  1271  	  if (*p == '+' || *p == '-')
  1272  	    {
  1273  	      num.push_back(*p);
  1274  	      ++p;
  1275  	    }
  1276  	}
  1277  
  1278        bool first = true;
  1279        while (p < pend)
  1280  	{
  1281  	  if (!first && *p == '_' && Lex::is_base_digit(base, p[1]))
  1282  	    ++p;
  1283  	  else if (!Lex::is_base_digit(base, *p))
  1284  	    break;
  1285  	  num.push_back(*p);
  1286  	  ++p;
  1287  	  first = false;
  1288  	}
  1289  
  1290        if (dot && Lex::could_be_exponent(base, p, pend))
  1291  	{
  1292  	  num.push_back(*p);
  1293  	  ++p;
  1294  	  if (*p == '+' || *p == '-')
  1295  	    {
  1296  	      num.push_back(*p);
  1297  	      ++p;
  1298  	    }
  1299  	  first = true;
  1300  	  while (p < pend)
  1301  	    {
  1302  	      if (!first && *p == '_' && p[1] >= '0' && p[1] <= '9')
  1303  		++p;
  1304  	      else if (*p < '0' || *p > '9')
  1305  		break;
  1306  	      num.push_back(*p);
  1307  	      ++p;
  1308  	      first = false;
  1309  	    }
  1310  	}
  1311        else if (dot && base == 16)
  1312  	{
  1313  	  go_error_at(this->location(),
  1314  		      "invalid hex floating-point literal with no exponent");
  1315  	  num.append("p0");
  1316  	}
  1317      }
  1318  
  1319    mpfr_clear_overflow();
  1320    mpfr_t val;
  1321    int r = mpfr_init_set_str(val, num.c_str(), base, MPFR_RNDN);
  1322    go_assert(r == 0);
  1323    if (mpfr_overflow_p())
  1324      go_error_at(this->location(),
  1325  		"floating-point exponent too large to represent");
  1326  
  1327    bool is_imaginary = *p == 'i';
  1328    if (is_imaginary)
  1329      ++p;
  1330  
  1331    this->lineoff_ = p - this->linebuf_;
  1332  
  1333    if (*p == 'e' || *p == 'E' || *p == 'p' || *p == 'P')
  1334      {
  1335        go_error_at(location,
  1336  		  "invalid prefix for floating constant");
  1337        this->skip_exponent();
  1338      }
  1339  
  1340    if (is_imaginary)
  1341      {
  1342        Token ret = Token::make_imaginary_token(val, location);
  1343        mpfr_clear(val);
  1344        return ret;
  1345      }
  1346    else
  1347      {
  1348        Token ret = Token::make_float_token(val, location);
  1349        mpfr_clear(val);
  1350        return ret;
  1351      }
  1352  }
  1353  
  1354  // Skip an exponent after reporting an error.
  1355  
  1356  void
  1357  Lex::skip_exponent()
  1358  {
  1359    const char* p = this->linebuf_ + this->lineoff_;
  1360    const char* pend = this->linebuf_ + this->linesize_;
  1361    if (*p != 'e' && *p != 'E' && *p != 'p' && *p != 'P')
  1362      return;
  1363    ++p;
  1364    if (*p == '+' || *p == '-')
  1365      ++p;
  1366    while (p < pend)
  1367      {
  1368        if ((*p < '0' || *p > '9') && *p != '_')
  1369  	break;
  1370        ++p;
  1371      }
  1372    this->lineoff_ = p - this->linebuf_;
  1373  }
  1374  
  1375  // Advance one character, possibly escaped.  Return the pointer beyond
  1376  // the character.  Set *VALUE to the character.  Set *IS_CHARACTER if
  1377  // this is a character (e.g., 'a' or '\u1234') rather than a byte
  1378  // value (e.g., '\001').
  1379  
  1380  const char*
  1381  Lex::advance_one_char(const char* p, bool is_single_quote, unsigned int* value,
  1382  		      bool* is_character)
  1383  {
  1384    *value = 0;
  1385    *is_character = true;
  1386    if (*p != '\\')
  1387      {
  1388        bool issued_error;
  1389        const char* ret = this->advance_one_utf8_char(p, value, &issued_error);
  1390        if (is_single_quote
  1391  	  && (*value == '\'' || *value == '\n')
  1392  	  && !issued_error)
  1393  	go_error_at(this->location(), "invalid character literal");
  1394        return ret;
  1395      }
  1396    else
  1397      {
  1398        ++p;
  1399        switch (*p)
  1400  	{
  1401  	case '0': case '1': case '2': case '3':
  1402  	case '4': case '5': case '6': case '7':
  1403  	  *is_character = false;
  1404  	  if (p[1] >= '0' && p[1] <= '7'
  1405  	      && p[2] >= '0' && p[2] <= '7')
  1406  	    {
  1407  	      *value = ((Lex::octal_value(p[0]) << 6)
  1408  			+ (Lex::octal_value(p[1]) << 3)
  1409  			+ Lex::octal_value(p[2]));
  1410  	      if (*value > 255)
  1411  		{
  1412  		  go_error_at(this->location(), "invalid octal constant");
  1413  		  *value = 255;
  1414  		}
  1415  	      return p + 3;
  1416  	    }
  1417  	      go_error_at(this->location(), "invalid octal character");
  1418  	  return (p[1] >= '0' && p[1] <= '7'
  1419  		  ? p + 2
  1420  		  : p + 1);
  1421  
  1422  	case 'x':
  1423  	  *is_character = false;
  1424  	  if (Lex::is_hex_digit(p[1]) && Lex::is_hex_digit(p[2]))
  1425  	    {
  1426  	      *value = (Lex::hex_val(p[1]) << 4) + Lex::hex_val(p[2]);
  1427  	      return p + 3;
  1428  	    }
  1429  	  go_error_at(this->location(), "invalid hex character");
  1430  	  return (Lex::is_hex_digit(p[1])
  1431  		  ? p + 2
  1432  		  : p + 1);
  1433  
  1434  	case 'a':
  1435  	  *value = '\a';
  1436  	  return p + 1;
  1437  	case 'b':
  1438  	  *value = '\b';
  1439  	  return p + 1;
  1440  	case 'f':
  1441  	  *value = '\f';
  1442  	  return p + 1;
  1443  	case 'n':
  1444  	  *value = '\n';
  1445  	  return p + 1;
  1446  	case 'r':
  1447  	  *value = '\r';
  1448  	  return p + 1;
  1449  	case 't':
  1450  	  *value = '\t';
  1451  	  return p + 1;
  1452  	case 'v':
  1453  	  *value = '\v';
  1454  	  return p + 1;
  1455  	case '\\':
  1456  	  *value = '\\';
  1457  	  return p + 1;
  1458  	case '\'':
  1459  	  if (!is_single_quote)
  1460  	    go_error_at(this->location(), "invalid quoted character");
  1461  	  *value = '\'';
  1462  	  return p + 1;
  1463  	case '"':
  1464  	  if (is_single_quote)
  1465  	    go_error_at(this->location(), "invalid quoted character");
  1466  	  *value = '"';
  1467  	  return p + 1;
  1468  
  1469  	case 'u':
  1470  	  if (Lex::is_hex_digit(p[1]) && Lex::is_hex_digit(p[2])
  1471  	      && Lex::is_hex_digit(p[3]) && Lex::is_hex_digit(p[4]))
  1472  	    {
  1473  	      *value = ((Lex::hex_val(p[1]) << 12)
  1474  			+ (Lex::hex_val(p[2]) << 8)
  1475  			+ (Lex::hex_val(p[3]) << 4)
  1476  			+ Lex::hex_val(p[4]));
  1477  	      if (*value >= 0xd800 && *value < 0xe000)
  1478  		{
  1479  		  go_error_at(this->location(),
  1480  			      "invalid unicode code point 0x%x",
  1481  			      *value);
  1482  		  // Use the replacement character.
  1483  		  *value = 0xfffd;
  1484  		}
  1485  	      return p + 5;
  1486  	    }
  1487  	  go_error_at(this->location(), "invalid little unicode code point");
  1488  	  return p + 1;
  1489  
  1490  	case 'U':
  1491  	  if (Lex::is_hex_digit(p[1]) && Lex::is_hex_digit(p[2])
  1492  	      && Lex::is_hex_digit(p[3]) && Lex::is_hex_digit(p[4])
  1493  	      && Lex::is_hex_digit(p[5]) && Lex::is_hex_digit(p[6])
  1494  	      && Lex::is_hex_digit(p[7]) && Lex::is_hex_digit(p[8]))
  1495  	    {
  1496  	      *value = ((Lex::hex_val(p[1]) << 28)
  1497  			+ (Lex::hex_val(p[2]) << 24)
  1498  			+ (Lex::hex_val(p[3]) << 20)
  1499  			+ (Lex::hex_val(p[4]) << 16)
  1500  			+ (Lex::hex_val(p[5]) << 12)
  1501  			+ (Lex::hex_val(p[6]) << 8)
  1502  			+ (Lex::hex_val(p[7]) << 4)
  1503  			+ Lex::hex_val(p[8]));
  1504  	      if (*value > 0x10ffff
  1505  		  || (*value >= 0xd800 && *value < 0xe000))
  1506  		{
  1507  		  go_error_at(this->location(),
  1508  			      "invalid unicode code point 0x%x",
  1509  			      *value);
  1510  		  // Use the replacement character.
  1511  		  *value = 0xfffd;
  1512  		}
  1513  	      return p + 9;
  1514  	    }
  1515  	  go_error_at(this->location(), "invalid big unicode code point");
  1516  	  return p + 1;
  1517  
  1518  	default:
  1519  	  go_error_at(this->location(), "invalid character after %<\\%>");
  1520  	  *value = *p;
  1521  	  return p + 1;
  1522  	}
  1523      }
  1524  }
  1525  
  1526  // Append V to STR.  IS_CHARACTER is true for a character which should
  1527  // be stored in UTF-8, false for a general byte value which should be
  1528  // stored directly.
  1529  
  1530  void
  1531  Lex::append_char(unsigned int v, bool is_character, std::string* str,
  1532  		 Location location)
  1533  {
  1534    char buf[4];
  1535    size_t len;
  1536    if (v <= 0x7f || !is_character)
  1537      {
  1538        buf[0] = v;
  1539        len = 1;
  1540      }
  1541    else if (v <= 0x7ff)
  1542      {
  1543        buf[0] = 0xc0 + (v >> 6);
  1544        buf[1] = 0x80 + (v & 0x3f);
  1545        len = 2;
  1546      }
  1547    else
  1548      {
  1549        if (v > 0x10ffff)
  1550  	{
  1551  	  go_warning_at(location, 0,
  1552  			"unicode code point 0x%x out of range in string", v);
  1553  	  // Turn it into the "replacement character".
  1554  	  v = 0xfffd;
  1555  	}
  1556        if (v >= 0xd800 && v < 0xe000)
  1557  	{
  1558  	  go_warning_at(location, 0,
  1559  			"unicode code point 0x%x is invalid surrogate pair", v);
  1560  	  v = 0xfffd;
  1561  	}
  1562        if (v <= 0xffff)
  1563  	{
  1564  	  buf[0] = 0xe0 + (v >> 12);
  1565  	  buf[1] = 0x80 + ((v >> 6) & 0x3f);
  1566  	  buf[2] = 0x80 + (v & 0x3f);
  1567  	  len = 3;
  1568  	}
  1569        else
  1570  	{
  1571  	  buf[0] = 0xf0 + (v >> 18);
  1572  	  buf[1] = 0x80 + ((v >> 12) & 0x3f);
  1573  	  buf[2] = 0x80 + ((v >> 6) & 0x3f);
  1574  	  buf[3] = 0x80 + (v & 0x3f);
  1575  	  len = 4;
  1576  	}
  1577      }
  1578    str->append(buf, len);
  1579  }
  1580  
  1581  // Pick up a character literal.
  1582  
  1583  Token
  1584  Lex::gather_character()
  1585  {
  1586    ++this->lineoff_;
  1587    const char* pstart = this->linebuf_ + this->lineoff_;
  1588    const char* p = pstart;
  1589  
  1590    unsigned int value;
  1591    bool is_character;
  1592    p = this->advance_one_char(p, true, &value, &is_character);
  1593  
  1594    if (*p != '\'')
  1595      {
  1596        go_error_at(this->location(), "unterminated character constant");
  1597        this->lineoff_ = p - this->linebuf_;
  1598        return this->make_invalid_token();
  1599      }
  1600  
  1601    mpz_t val;
  1602    mpz_init_set_ui(val, value);
  1603  
  1604    Location location = this->location();
  1605    this->lineoff_ = p + 1 - this->linebuf_;
  1606    Token ret = Token::make_character_token(val, location);
  1607    mpz_clear(val);
  1608    return ret;
  1609  }
  1610  
  1611  // Pick up a quoted string.
  1612  
  1613  Token
  1614  Lex::gather_string()
  1615  {
  1616    const char* pstart = this->linebuf_ + this->lineoff_ + 1;
  1617    const char* p = pstart;
  1618    const char* pend = this->linebuf_ + this->linesize_;
  1619  
  1620    std::string value;
  1621    while (*p != '"')
  1622      {
  1623        Location loc = this->location();
  1624        unsigned int c;
  1625        bool is_character;
  1626        this->lineoff_ = p - this->linebuf_;
  1627        p = this->advance_one_char(p, false, &c, &is_character);
  1628        if (p >= pend)
  1629  	{
  1630  	  go_error_at(this->location(), "unterminated string");
  1631  	  --p;
  1632  	  break;
  1633  	}
  1634        Lex::append_char(c, is_character, &value, loc);
  1635      }
  1636  
  1637    Location location = this->location();
  1638    this->lineoff_ = p + 1 - this->linebuf_;
  1639    return Token::make_string_token(value, location);
  1640  }
  1641  
  1642  // Pick up a raw string.
  1643  
  1644  Token
  1645  Lex::gather_raw_string()
  1646  {
  1647    const char* p = this->linebuf_ + this->lineoff_ + 1;
  1648    const char* pend = this->linebuf_ + this->linesize_;
  1649    Location location = this->location();
  1650  
  1651    std::string value;
  1652    while (true)
  1653      {
  1654        while (p < pend)
  1655  	{
  1656  	  if (*p == '`')
  1657  	    {
  1658  	      this->lineoff_ = p + 1 - this->linebuf_;
  1659  	      return Token::make_string_token(value, location);
  1660  	    }
  1661  	  Location loc = this->location();
  1662  	  unsigned int c;
  1663  	  bool issued_error;
  1664  	  this->lineoff_ = p - this->linebuf_;
  1665  	  p = this->advance_one_utf8_char(p, &c, &issued_error);
  1666  	  // "Carriage return characters ('\r') inside raw string literals
  1667  	  // are discarded from the raw string value."
  1668  	  if (c != '\r')
  1669  	      Lex::append_char(c, true, &value, loc);
  1670  	}
  1671        this->lineoff_ = p - this->linebuf_;
  1672        if (!this->require_line())
  1673  	{
  1674  	  go_error_at(location, "unterminated raw string");
  1675  	  return Token::make_string_token(value, location);
  1676  	}
  1677        p = this->linebuf_ + this->lineoff_;
  1678        pend = this->linebuf_ + this->linesize_;
  1679      }
  1680  }
  1681  
  1682  // If C1 C2 C3 are a three character operator, return the code.
  1683  
  1684  Operator
  1685  Lex::three_character_operator(char c1, char c2, char c3)
  1686  {
  1687    if (c3 == '=')
  1688      {
  1689        if (c1 == '<' && c2 == '<')
  1690  	return OPERATOR_LSHIFTEQ;
  1691        else if (c1 == '>' && c2 == '>')
  1692  	return OPERATOR_RSHIFTEQ;
  1693        else if (c1 == '&' && c2 == '^')
  1694  	return OPERATOR_BITCLEAREQ;
  1695      }
  1696    return OPERATOR_INVALID;
  1697  }
  1698  
  1699  // If C1 C2 are a two character operator, return the code.
  1700  
  1701  Operator
  1702  Lex::two_character_operator(char c1, char c2)
  1703  {
  1704    switch (c1)
  1705      {
  1706      case '|':
  1707        if (c2 == '|')
  1708  	return OPERATOR_OROR;
  1709        else if (c2 == '=')
  1710  	return OPERATOR_OREQ;
  1711        break;
  1712      case '&':
  1713        if (c2 == '&')
  1714  	return OPERATOR_ANDAND;
  1715        else if (c2 == '^')
  1716  	return OPERATOR_BITCLEAR;
  1717        else if (c2 == '=')
  1718  	return OPERATOR_ANDEQ;
  1719        break;
  1720      case '^':
  1721        if (c2 == '=')
  1722  	return OPERATOR_XOREQ;
  1723        break;
  1724      case '=':
  1725        if (c2 == '=')
  1726  	return OPERATOR_EQEQ;
  1727        break;
  1728      case '!':
  1729        if (c2 == '=')
  1730  	return OPERATOR_NOTEQ;
  1731        break;
  1732      case '<':
  1733        if (c2 == '=')
  1734  	return OPERATOR_LE;
  1735        else if (c2 == '<')
  1736  	return OPERATOR_LSHIFT;
  1737        else if (c2 == '-')
  1738  	return OPERATOR_CHANOP;
  1739        break;
  1740      case '>':
  1741        if (c2 == '=')
  1742  	return OPERATOR_GE;
  1743        else if (c2 == '>')
  1744  	return OPERATOR_RSHIFT;
  1745        break;
  1746      case '*':
  1747        if (c2 == '=')
  1748  	return OPERATOR_MULTEQ;
  1749        break;
  1750      case '/':
  1751        if (c2 == '=')
  1752  	return OPERATOR_DIVEQ;
  1753        break;
  1754      case '%':
  1755        if (c2 == '=')
  1756  	return OPERATOR_MODEQ;
  1757        break;
  1758      case '+':
  1759        if (c2 == '+')
  1760  	{
  1761  	  this->add_semi_at_eol_ = true;
  1762  	  return OPERATOR_PLUSPLUS;
  1763  	}
  1764        else if (c2 == '=')
  1765  	return OPERATOR_PLUSEQ;
  1766        break;
  1767      case '-':
  1768        if (c2 == '-')
  1769  	{
  1770  	  this->add_semi_at_eol_ = true;
  1771  	  return OPERATOR_MINUSMINUS;
  1772  	}
  1773        else if (c2 == '=')
  1774  	return OPERATOR_MINUSEQ;
  1775        break;
  1776      case ':':
  1777        if (c2 == '=')
  1778  	return OPERATOR_COLONEQ;
  1779        break;
  1780      default:
  1781        break;
  1782      }
  1783    return OPERATOR_INVALID;
  1784  }
  1785  
  1786  // If character C is an operator, return the code.
  1787  
  1788  Operator
  1789  Lex::one_character_operator(char c)
  1790  {
  1791    switch (c)
  1792      {
  1793      case '<':
  1794        return OPERATOR_LT;
  1795      case '>':
  1796        return OPERATOR_GT;
  1797      case '+':
  1798        return OPERATOR_PLUS;
  1799      case '-':
  1800        return OPERATOR_MINUS;
  1801      case '|':
  1802        return OPERATOR_OR;
  1803      case '^':
  1804        return OPERATOR_XOR;
  1805      case '*':
  1806        return OPERATOR_MULT;
  1807      case '/':
  1808        return OPERATOR_DIV;
  1809      case '%':
  1810        return OPERATOR_MOD;
  1811      case '&':
  1812        return OPERATOR_AND;
  1813      case '!':
  1814        return OPERATOR_NOT;
  1815      case '=':
  1816        return OPERATOR_EQ;
  1817      case ':':
  1818        return OPERATOR_COLON;
  1819      case ';':
  1820        return OPERATOR_SEMICOLON;
  1821      case '.':
  1822        return OPERATOR_DOT;
  1823      case ',':
  1824        return OPERATOR_COMMA;
  1825      case '(':
  1826        return OPERATOR_LPAREN;
  1827      case ')':
  1828        this->add_semi_at_eol_ = true;
  1829        return OPERATOR_RPAREN;
  1830      case '{':
  1831        return OPERATOR_LCURLY;
  1832      case '}':
  1833        this->add_semi_at_eol_ = true;
  1834        return OPERATOR_RCURLY;
  1835      case '[':
  1836        return OPERATOR_LSQUARE;
  1837      case ']':
  1838        this->add_semi_at_eol_ = true;
  1839        return OPERATOR_RSQUARE;
  1840      default:
  1841        return OPERATOR_INVALID;
  1842      }
  1843  }
  1844  
  1845  // Skip a C-style comment.
  1846  
  1847  bool
  1848  Lex::skip_c_comment(bool* found_newline)
  1849  {
  1850    while (true)
  1851      {
  1852        if (!this->require_line())
  1853  	{
  1854  	  go_error_at(this->location(), "unterminated comment");
  1855  	  return false;
  1856  	}
  1857  
  1858        const char* p = this->linebuf_ + this->lineoff_;
  1859        const char* pend = this->linebuf_ + this->linesize_;
  1860  
  1861        while (p < pend)
  1862  	{
  1863  	  if (p[0] == '*' && p + 1 < pend && p[1] == '/')
  1864  	    {
  1865  	      this->lineoff_ = p + 2 - this->linebuf_;
  1866  	      return true;
  1867  	    }
  1868  
  1869            if (p[0] == '\n')
  1870              *found_newline = true;
  1871  
  1872  	  this->lineoff_ = p - this->linebuf_;
  1873  	  unsigned int c;
  1874  	  bool issued_error;
  1875  	  p = this->advance_one_utf8_char(p, &c, &issued_error);
  1876  	}
  1877  
  1878        this->lineoff_ = p - this->linebuf_;
  1879      }
  1880  }
  1881  
  1882  // Skip a C++-style comment.
  1883  
  1884  void
  1885  Lex::skip_cpp_comment()
  1886  {
  1887    // Ensure that if EXTERN_ is set, it means that we just saw a
  1888    // //extern comment.
  1889    this->extern_.clear();
  1890  
  1891    Location loc = this->location();
  1892    size_t lineoff = this->lineoff_;
  1893  
  1894    const char* p = this->linebuf_ + lineoff;
  1895    const char* pend = this->linebuf_ + this->linesize_;
  1896  
  1897    const char* pcheck = p;
  1898    bool saw_error = false;
  1899    while (pcheck < pend)
  1900      {
  1901        this->lineoff_ = pcheck - this->linebuf_;
  1902        unsigned int c;
  1903        bool issued_error;
  1904        pcheck = this->advance_one_utf8_char(pcheck, &c, &issued_error);
  1905        if (issued_error)
  1906  	saw_error = true;
  1907      }
  1908  
  1909    if (saw_error)
  1910      return;
  1911  
  1912    // Recognize various magic comments at the start of a line, preceded
  1913    // only by spaces or tabs.
  1914  
  1915    // "- 2" for the "//" at the start of the comment.
  1916    for (const char* psp = this->linebuf_; psp < p - 2; psp++)
  1917      if (*psp != ' ' && *psp != '\t')
  1918        return;
  1919  
  1920    while (pend > p
  1921  	 && (pend[-1] == ' ' || pend[-1] == '\t'
  1922  	     || pend[-1] == '\r' || pend[-1] == '\n'))
  1923      --pend;
  1924  
  1925    // A C++ comment at the start of the line of the form
  1926    //   //line FILE:LINENO
  1927    // is interpreted as setting the file name and line number of the
  1928    // next source line.
  1929    if (pend - p > 5 && memcmp(p, "line ", 5) == 0)
  1930      {
  1931        p += 5;
  1932        while (p < pend && *p == ' ')
  1933  	++p;
  1934        const char* pcolon = static_cast<const char*>(memchr(p, ':', pend - p));
  1935        if (pcolon != NULL
  1936  	  && pcolon[1] >= '0'
  1937  	  && pcolon[1] <= '9')
  1938  	{
  1939  	  char* plend;
  1940  	  long lineno = strtol(pcolon + 1, &plend, 10);
  1941  	  if (plend > pcolon + 1
  1942  	      && (plend == pend
  1943  		  || *plend < '0'
  1944  		  || *plend > '9')
  1945  	      && lineno > 0
  1946  	      && lineno < 0x7fffffff)
  1947  	    {
  1948  	      unsigned int filelen = pcolon - p;
  1949  	      char* file = new char[filelen + 1];
  1950  	      memcpy(file, p, filelen);
  1951  	      file[filelen] = '\0';
  1952  
  1953                this->linemap_->start_file(file, lineno);
  1954  	      this->lineno_ = lineno - 1;
  1955  
  1956  	      p = plend;
  1957  	    }
  1958  	}
  1959        return;
  1960      }
  1961  
  1962    // As a special gccgo extension, a C++ comment at the start of the
  1963    // line of the form
  1964    //   //extern NAME
  1965    // which immediately precedes a function declaration means that the
  1966    // external name of the function declaration is NAME.  This is
  1967    // normally used to permit Go code to call a C function.
  1968    if (pend - p > 7 && memcmp(p, "extern ", 7) == 0)
  1969      {
  1970        p += 7;
  1971        while (p < pend && (*p == ' ' || *p == '\t'))
  1972  	++p;
  1973        if (pend > p)
  1974  	this->extern_ = std::string(p, pend - p);
  1975        return;
  1976      }
  1977  
  1978    // All other special comments start with "go:".
  1979  
  1980    if (pend - p < 4 || memcmp(p, "go:", 3) != 0)
  1981      return;
  1982  
  1983    const char *ps = p + 3;
  1984    while (ps < pend && *ps != ' ' && *ps != '\t')
  1985      ++ps;
  1986    std::string verb = std::string(p, ps - p);
  1987  
  1988    if (verb == "go:linkname")
  1989      {
  1990        // As in the gc compiler, set the external link name for a Go symbol.
  1991        std::string go_name;
  1992        std::string ext_name;
  1993        bool is_exported = false;
  1994        if (ps < pend)
  1995  	{
  1996  	  while (ps < pend && (*ps == ' ' || *ps == '\t'))
  1997  	    ++ps;
  1998  	  if (ps < pend)
  1999  	    {
  2000  	      const char* pg = ps;
  2001  
  2002  	      unsigned int c;
  2003  	      bool issued_error;
  2004  	      ps = this->advance_one_utf8_char(ps, &c, &issued_error);
  2005  	      is_exported = Lex::is_unicode_uppercase(c);
  2006  
  2007  	      while (ps < pend && *ps != ' ' && *ps != '\t')
  2008  		++ps;
  2009  	      if (ps <= pend)
  2010  		go_name = std::string(pg, ps - pg);
  2011  	      while (ps < pend && (*ps == ' ' || *ps == '\t'))
  2012  		++ps;
  2013  	    }
  2014  	  if (ps < pend)
  2015  	    {
  2016  	      const char* pc = ps;
  2017  	      while (ps < pend && *ps != ' ' && *ps != '\t')
  2018  		++ps;
  2019  	      if (ps <= pend)
  2020  		ext_name = std::string(pc, ps - pc);
  2021  	    }
  2022  	  if (ps != pend)
  2023  	    {
  2024  	      go_name.clear();
  2025  	      ext_name.clear();
  2026  	    }
  2027  	}
  2028        if (go_name.empty())
  2029  	go_error_at(loc, "usage: %<//go:linkname%> localname [linkname]");
  2030        else
  2031  	{
  2032  	  if (this->linknames_ == NULL)
  2033  	    this->linknames_ = new Linknames();
  2034  	  (*this->linknames_)[go_name] = Linkname(ext_name, is_exported, loc);
  2035  	}
  2036      }
  2037    else if (verb == "go:embed")
  2038      this->gather_embed(ps, pend);
  2039    else if (verb == "go:nointerface")
  2040      {
  2041        // For field tracking analysis: a //go:nointerface comment means
  2042        // that the next interface method should not be stored in the
  2043        // type descriptor.  This permits it to be discarded if it is
  2044        // not needed.
  2045        this->pragmas_ |= GOPRAGMA_NOINTERFACE;
  2046      }
  2047    else if (verb == "go:noescape")
  2048      {
  2049        // Applies to the next function declaration.  Any arguments do
  2050        // not escape.
  2051        // FIXME: Not implemented.
  2052        this->pragmas_ |= GOPRAGMA_NOESCAPE;
  2053      }
  2054    else if (verb == "go:nosplit")
  2055      {
  2056        // Applies to the next function.  Do not split the stack when
  2057        // entering the function.
  2058        this->pragmas_ |= GOPRAGMA_NOSPLIT;
  2059      }
  2060    else if (verb == "go:noinline")
  2061      {
  2062        // Applies to the next function.  Do not inline the function.
  2063        this->pragmas_ |= GOPRAGMA_NOINLINE;
  2064      }
  2065    else if (verb == "go:notinheap")
  2066      {
  2067        // Applies to the next type.  The type does not live in the heap.
  2068        this->pragmas_ |= GOPRAGMA_NOTINHEAP;
  2069      }
  2070    else if (verb == "go:systemstack")
  2071      {
  2072        // Applies to the next function.  It must run on the system stack.
  2073        // FIXME: Should only work when compiling the runtime package.
  2074        // FIXME: Not implemented.
  2075        this->pragmas_ |= GOPRAGMA_SYSTEMSTACK;
  2076      }
  2077    else if (verb == "go:nowritebarrier")
  2078      {
  2079        // Applies to the next function.  If the function needs to use
  2080        // any write barriers, it should emit an error instead.
  2081        // FIXME: Should only work when compiling the runtime package.
  2082        this->pragmas_ |= GOPRAGMA_NOWRITEBARRIER;
  2083      }
  2084    else if (verb == "go:nowritebarrierrec")
  2085      {
  2086        // Applies to the next function.  If the function, or any
  2087        // function that it calls, needs to use any write barriers, it
  2088        // should emit an error instead.
  2089        // FIXME: Should only work when compiling the runtime package.
  2090        this->pragmas_ |= GOPRAGMA_NOWRITEBARRIERREC;
  2091      }
  2092    else if (verb == "go:yeswritebarrierrec")
  2093      {
  2094        // Applies to the next function.  Disables go:nowritebarrierrec
  2095        // when looking at callees; write barriers are permitted here.
  2096        // FIXME: Should only work when compiling the runtime package.
  2097        this->pragmas_ |= GOPRAGMA_YESWRITEBARRIERREC;
  2098      }
  2099    else if (verb == "go:cgo_unsafe_args")
  2100      {
  2101        // Applies to the next function.  Taking the address of any
  2102        // argument implies taking the address of all arguments.
  2103        // FIXME: Not implemented.
  2104        this->pragmas_ |= GOPRAGMA_CGOUNSAFEARGS;
  2105      }
  2106    else if (verb == "go:uintptrescapes")
  2107      {
  2108        // Applies to the next function.  If an argument is a pointer
  2109        // converted to uintptr, then the pointer escapes.
  2110        // FIXME: Not implemented.
  2111        this->pragmas_ |= GOPRAGMA_UINTPTRESCAPES;
  2112      }
  2113  }
  2114  
  2115  // Read a go:embed directive.  This is a series of space-separated
  2116  // patterns.  Each pattern may be a quoted or backquoted string.
  2117  
  2118  void
  2119  Lex::gather_embed(const char *p, const char *pend)
  2120  {
  2121    while (true)
  2122      {
  2123        // Skip spaces to find the start of the next pattern.  We do a
  2124        // fast skip of space and tab, but we also permit and skip
  2125        // Unicode space characters.
  2126        while (p < pend && (*p == ' ' || *p == '\t'))
  2127  	++p;
  2128        if (p >= pend)
  2129  	break;
  2130        unsigned int c;
  2131        bool issued_error;
  2132        const char *pnext = this->advance_one_utf8_char(p, &c, &issued_error);
  2133        if (issued_error)
  2134  	return;
  2135        if (Lex::is_unicode_space(c))
  2136  	{
  2137  	  p = pnext;
  2138  	  continue;
  2139  	}
  2140  
  2141        // Here P points to the start of the next pattern, PNEXT points
  2142        // to the second character in the pattern, and C is the first
  2143        // character in that pattern (the character to which P points).
  2144  
  2145        if (c == '"' || c == '`')
  2146  	{
  2147  	  Location loc = this->location();
  2148  	  const unsigned char quote = c;
  2149  	  std::string value;
  2150  	  p = pnext;
  2151  	  while (p < pend && *p != quote)
  2152  	    {
  2153  	      bool is_character;
  2154  	      if (quote == '"')
  2155  		p = this->advance_one_char(p, false, &c, &is_character);
  2156  	      else
  2157  		{
  2158  		  p = this->advance_one_utf8_char(p, &c, &issued_error);
  2159  		  if (issued_error)
  2160  		    return;
  2161  		  // "Carriage return characters ('\r') inside raw string
  2162  		  // literals are discarded from the raw string value."
  2163  		  if (c == '\r')
  2164  		    continue;
  2165  		  is_character = true;
  2166  		}
  2167  	      Lex::append_char(c, is_character, &value, loc);
  2168  	    }
  2169  	  if (p >= pend)
  2170  	    {
  2171  	      // Note that within a go:embed directive we do not
  2172  	      // permit raw strings to cross multiple lines.
  2173  	      go_error_at(loc, "unterminated string");
  2174  	      return;
  2175  	    }
  2176  	  this->embeds_.push_back(value);
  2177  	  ++p;
  2178  	}
  2179        else
  2180  	{
  2181  	  const char *start = p;
  2182  	  p = pnext;
  2183  	  while (p < pend)
  2184  	    {
  2185  	      c = *p;
  2186  	      if (c == ' ' || c == '\t')
  2187  		break;
  2188  	      if (c > ' ' && c <= 0x7f)
  2189  		{
  2190  		  // ASCII non-space character.
  2191  		  ++p;
  2192  		  continue;
  2193  		}
  2194  	      pnext = this->advance_one_utf8_char(p, &c, &issued_error);
  2195  	      if (issued_error)
  2196  		return;
  2197  	      if (Lex::is_unicode_space(c))
  2198  		break;
  2199  	      p = pnext;
  2200  	    }
  2201  
  2202  	  this->embeds_.push_back(std::string(start, p - start));
  2203  	}
  2204      }
  2205  }
  2206  
  2207  // The Unicode tables use this struct.
  2208  
  2209  struct Unicode_range
  2210  {
  2211    // The low end of the range.
  2212    unsigned int low;
  2213    // The high end of the range.
  2214    unsigned int high;
  2215    // The stride.  This entries represents low, low + stride, low + 2 *
  2216    // stride, etc., up to high.
  2217    unsigned int stride;
  2218  };
  2219  
  2220  // A table of whitespace characters--Unicode code points classified as
  2221  // "Space", "C" locale whitespace characters, the "next line" control
  2222  // character (0085), the line separator (2028), the paragraph
  2223  // separator (2029), and the "zero-width non-break space" (feff).
  2224  
  2225  static const Unicode_range unicode_space[] =
  2226  {
  2227    { 0x0009, 0x000d, 1 },
  2228    { 0x0020, 0x0020, 1 },
  2229    { 0x0085, 0x0085, 1 },
  2230    { 0x00a0, 0x00a0, 1 },
  2231    { 0x1680, 0x1680, 1 },
  2232    { 0x180e, 0x180e, 1 },
  2233    { 0x2000, 0x200a, 1 },
  2234    { 0x2028, 0x2029, 1 },
  2235    { 0x202f, 0x202f, 1 },
  2236    { 0x205f, 0x205f, 1 },
  2237    { 0x3000, 0x3000, 1 },
  2238    { 0xfeff, 0xfeff, 1 },
  2239  };
  2240  
  2241  // A table of Unicode digits--Unicode code points classified as
  2242  // "Digit".
  2243  
  2244  static const Unicode_range unicode_digits[] =
  2245  {
  2246    { 0x0030, 0x0039, 1},
  2247    { 0x0660, 0x0669, 1},
  2248    { 0x06f0, 0x06f9, 1},
  2249    { 0x07c0, 0x07c9, 1},
  2250    { 0x0966, 0x096f, 1},
  2251    { 0x09e6, 0x09ef, 1},
  2252    { 0x0a66, 0x0a6f, 1},
  2253    { 0x0ae6, 0x0aef, 1},
  2254    { 0x0b66, 0x0b6f, 1},
  2255    { 0x0be6, 0x0bef, 1},
  2256    { 0x0c66, 0x0c6f, 1},
  2257    { 0x0ce6, 0x0cef, 1},
  2258    { 0x0d66, 0x0d6f, 1},
  2259    { 0x0e50, 0x0e59, 1},
  2260    { 0x0ed0, 0x0ed9, 1},
  2261    { 0x0f20, 0x0f29, 1},
  2262    { 0x1040, 0x1049, 1},
  2263    { 0x17e0, 0x17e9, 1},
  2264    { 0x1810, 0x1819, 1},
  2265    { 0x1946, 0x194f, 1},
  2266    { 0x19d0, 0x19d9, 1},
  2267    { 0x1b50, 0x1b59, 1},
  2268    { 0xff10, 0xff19, 1},
  2269    { 0x104a0, 0x104a9, 1},
  2270    { 0x1d7ce, 0x1d7ff, 1},
  2271  };
  2272  
  2273  // A table of Unicode letters--Unicode code points classified as
  2274  // "Letter".
  2275  
  2276  static const Unicode_range unicode_letters[] =
  2277  {
  2278    { 0x0041, 0x005a, 1},
  2279    { 0x0061, 0x007a, 1},
  2280    { 0x00aa, 0x00b5, 11},
  2281    { 0x00ba, 0x00c0, 6},
  2282    { 0x00c1, 0x00d6, 1},
  2283    { 0x00d8, 0x00f6, 1},
  2284    { 0x00f8, 0x02c1, 1},
  2285    { 0x02c6, 0x02d1, 1},
  2286    { 0x02e0, 0x02e4, 1},
  2287    { 0x02ec, 0x02ee, 2},
  2288    { 0x0370, 0x0374, 1},
  2289    { 0x0376, 0x0377, 1},
  2290    { 0x037a, 0x037d, 1},
  2291    { 0x037f, 0x0386, 7},
  2292    { 0x0388, 0x038a, 1},
  2293    { 0x038c, 0x038e, 2},
  2294    { 0x038f, 0x03a1, 1},
  2295    { 0x03a3, 0x03f5, 1},
  2296    { 0x03f7, 0x0481, 1},
  2297    { 0x048a, 0x052f, 1},
  2298    { 0x0531, 0x0556, 1},
  2299    { 0x0559, 0x0561, 8},
  2300    { 0x0562, 0x0587, 1},
  2301    { 0x05d0, 0x05ea, 1},
  2302    { 0x05f0, 0x05f2, 1},
  2303    { 0x0620, 0x064a, 1},
  2304    { 0x066e, 0x066f, 1},
  2305    { 0x0671, 0x06d3, 1},
  2306    { 0x06d5, 0x06e5, 16},
  2307    { 0x06e6, 0x06ee, 8},
  2308    { 0x06ef, 0x06fa, 11},
  2309    { 0x06fb, 0x06fc, 1},
  2310    { 0x06ff, 0x0710, 17},
  2311    { 0x0712, 0x072f, 1},
  2312    { 0x074d, 0x07a5, 1},
  2313    { 0x07b1, 0x07ca, 25},
  2314    { 0x07cb, 0x07ea, 1},
  2315    { 0x07f4, 0x07f5, 1},
  2316    { 0x07fa, 0x0800, 6},
  2317    { 0x0801, 0x0815, 1},
  2318    { 0x081a, 0x0824, 10},
  2319    { 0x0828, 0x0840, 24},
  2320    { 0x0841, 0x0858, 1},
  2321    { 0x08a0, 0x08b4, 1},
  2322    { 0x0904, 0x0939, 1},
  2323    { 0x093d, 0x0950, 19},
  2324    { 0x0958, 0x0961, 1},
  2325    { 0x0971, 0x0980, 1},
  2326    { 0x0985, 0x098c, 1},
  2327    { 0x098f, 0x0990, 1},
  2328    { 0x0993, 0x09a8, 1},
  2329    { 0x09aa, 0x09b0, 1},
  2330    { 0x09b2, 0x09b6, 4},
  2331    { 0x09b7, 0x09b9, 1},
  2332    { 0x09bd, 0x09ce, 17},
  2333    { 0x09dc, 0x09dd, 1},
  2334    { 0x09df, 0x09e1, 1},
  2335    { 0x09f0, 0x09f1, 1},
  2336    { 0x0a05, 0x0a0a, 1},
  2337    { 0x0a0f, 0x0a10, 1},
  2338    { 0x0a13, 0x0a28, 1},
  2339    { 0x0a2a, 0x0a30, 1},
  2340    { 0x0a32, 0x0a33, 1},
  2341    { 0x0a35, 0x0a36, 1},
  2342    { 0x0a38, 0x0a39, 1},
  2343    { 0x0a59, 0x0a5c, 1},
  2344    { 0x0a5e, 0x0a72, 20},
  2345    { 0x0a73, 0x0a74, 1},
  2346    { 0x0a85, 0x0a8d, 1},
  2347    { 0x0a8f, 0x0a91, 1},
  2348    { 0x0a93, 0x0aa8, 1},
  2349    { 0x0aaa, 0x0ab0, 1},
  2350    { 0x0ab2, 0x0ab3, 1},
  2351    { 0x0ab5, 0x0ab9, 1},
  2352    { 0x0abd, 0x0ad0, 19},
  2353    { 0x0ae0, 0x0ae1, 1},
  2354    { 0x0af9, 0x0b05, 12},
  2355    { 0x0b06, 0x0b0c, 1},
  2356    { 0x0b0f, 0x0b10, 1},
  2357    { 0x0b13, 0x0b28, 1},
  2358    { 0x0b2a, 0x0b30, 1},
  2359    { 0x0b32, 0x0b33, 1},
  2360    { 0x0b35, 0x0b39, 1},
  2361    { 0x0b3d, 0x0b5c, 31},
  2362    { 0x0b5d, 0x0b5f, 2},
  2363    { 0x0b60, 0x0b61, 1},
  2364    { 0x0b71, 0x0b83, 18},
  2365    { 0x0b85, 0x0b8a, 1},
  2366    { 0x0b8e, 0x0b90, 1},
  2367    { 0x0b92, 0x0b95, 1},
  2368    { 0x0b99, 0x0b9a, 1},
  2369    { 0x0b9c, 0x0b9e, 2},
  2370    { 0x0b9f, 0x0ba3, 4},
  2371    { 0x0ba4, 0x0ba8, 4},
  2372    { 0x0ba9, 0x0baa, 1},
  2373    { 0x0bae, 0x0bb9, 1},
  2374    { 0x0bd0, 0x0c05, 53},
  2375    { 0x0c06, 0x0c0c, 1},
  2376    { 0x0c0e, 0x0c10, 1},
  2377    { 0x0c12, 0x0c28, 1},
  2378    { 0x0c2a, 0x0c39, 1},
  2379    { 0x0c3d, 0x0c58, 27},
  2380    { 0x0c59, 0x0c5a, 1},
  2381    { 0x0c60, 0x0c61, 1},
  2382    { 0x0c85, 0x0c8c, 1},
  2383    { 0x0c8e, 0x0c90, 1},
  2384    { 0x0c92, 0x0ca8, 1},
  2385    { 0x0caa, 0x0cb3, 1},
  2386    { 0x0cb5, 0x0cb9, 1},
  2387    { 0x0cbd, 0x0cde, 33},
  2388    { 0x0ce0, 0x0ce1, 1},
  2389    { 0x0cf1, 0x0cf2, 1},
  2390    { 0x0d05, 0x0d0c, 1},
  2391    { 0x0d0e, 0x0d10, 1},
  2392    { 0x0d12, 0x0d3a, 1},
  2393    { 0x0d3d, 0x0d5f, 17},
  2394    { 0x0d60, 0x0d61, 1},
  2395    { 0x0d7a, 0x0d7f, 1},
  2396    { 0x0d85, 0x0d96, 1},
  2397    { 0x0d9a, 0x0db1, 1},
  2398    { 0x0db3, 0x0dbb, 1},
  2399    { 0x0dbd, 0x0dc0, 3},
  2400    { 0x0dc1, 0x0dc6, 1},
  2401    { 0x0e01, 0x0e30, 1},
  2402    { 0x0e32, 0x0e33, 1},
  2403    { 0x0e40, 0x0e46, 1},
  2404    { 0x0e81, 0x0e82, 1},
  2405    { 0x0e84, 0x0e87, 3},
  2406    { 0x0e88, 0x0e8a, 2},
  2407    { 0x0e8d, 0x0e94, 7},
  2408    { 0x0e95, 0x0e97, 1},
  2409    { 0x0e99, 0x0e9f, 1},
  2410    { 0x0ea1, 0x0ea3, 1},
  2411    { 0x0ea5, 0x0ea7, 2},
  2412    { 0x0eaa, 0x0eab, 1},
  2413    { 0x0ead, 0x0eb0, 1},
  2414    { 0x0eb2, 0x0eb3, 1},
  2415    { 0x0ebd, 0x0ec0, 3},
  2416    { 0x0ec1, 0x0ec4, 1},
  2417    { 0x0ec6, 0x0edc, 22},
  2418    { 0x0edd, 0x0edf, 1},
  2419    { 0x0f00, 0x0f40, 64},
  2420    { 0x0f41, 0x0f47, 1},
  2421    { 0x0f49, 0x0f6c, 1},
  2422    { 0x0f88, 0x0f8c, 1},
  2423    { 0x1000, 0x102a, 1},
  2424    { 0x103f, 0x1050, 17},
  2425    { 0x1051, 0x1055, 1},
  2426    { 0x105a, 0x105d, 1},
  2427    { 0x1061, 0x1065, 4},
  2428    { 0x1066, 0x106e, 8},
  2429    { 0x106f, 0x1070, 1},
  2430    { 0x1075, 0x1081, 1},
  2431    { 0x108e, 0x10a0, 18},
  2432    { 0x10a1, 0x10c5, 1},
  2433    { 0x10c7, 0x10cd, 6},
  2434    { 0x10d0, 0x10fa, 1},
  2435    { 0x10fc, 0x1248, 1},
  2436    { 0x124a, 0x124d, 1},
  2437    { 0x1250, 0x1256, 1},
  2438    { 0x1258, 0x125a, 2},
  2439    { 0x125b, 0x125d, 1},
  2440    { 0x1260, 0x1288, 1},
  2441    { 0x128a, 0x128d, 1},
  2442    { 0x1290, 0x12b0, 1},
  2443    { 0x12b2, 0x12b5, 1},
  2444    { 0x12b8, 0x12be, 1},
  2445    { 0x12c0, 0x12c2, 2},
  2446    { 0x12c3, 0x12c5, 1},
  2447    { 0x12c8, 0x12d6, 1},
  2448    { 0x12d8, 0x1310, 1},
  2449    { 0x1312, 0x1315, 1},
  2450    { 0x1318, 0x135a, 1},
  2451    { 0x1380, 0x138f, 1},
  2452    { 0x13a0, 0x13f5, 1},
  2453    { 0x13f8, 0x13fd, 1},
  2454    { 0x1401, 0x166c, 1},
  2455    { 0x166f, 0x167f, 1},
  2456    { 0x1681, 0x169a, 1},
  2457    { 0x16a0, 0x16ea, 1},
  2458    { 0x16f1, 0x16f8, 1},
  2459    { 0x1700, 0x170c, 1},
  2460    { 0x170e, 0x1711, 1},
  2461    { 0x1720, 0x1731, 1},
  2462    { 0x1740, 0x1751, 1},
  2463    { 0x1760, 0x176c, 1},
  2464    { 0x176e, 0x1770, 1},
  2465    { 0x1780, 0x17b3, 1},
  2466    { 0x17d7, 0x17dc, 5},
  2467    { 0x1820, 0x1877, 1},
  2468    { 0x1880, 0x18a8, 1},
  2469    { 0x18aa, 0x18b0, 6},
  2470    { 0x18b1, 0x18f5, 1},
  2471    { 0x1900, 0x191e, 1},
  2472    { 0x1950, 0x196d, 1},
  2473    { 0x1970, 0x1974, 1},
  2474    { 0x1980, 0x19ab, 1},
  2475    { 0x19b0, 0x19c9, 1},
  2476    { 0x1a00, 0x1a16, 1},
  2477    { 0x1a20, 0x1a54, 1},
  2478    { 0x1aa7, 0x1b05, 94},
  2479    { 0x1b06, 0x1b33, 1},
  2480    { 0x1b45, 0x1b4b, 1},
  2481    { 0x1b83, 0x1ba0, 1},
  2482    { 0x1bae, 0x1baf, 1},
  2483    { 0x1bba, 0x1be5, 1},
  2484    { 0x1c00, 0x1c23, 1},
  2485    { 0x1c4d, 0x1c4f, 1},
  2486    { 0x1c5a, 0x1c7d, 1},
  2487    { 0x1ce9, 0x1cec, 1},
  2488    { 0x1cee, 0x1cf1, 1},
  2489    { 0x1cf5, 0x1cf6, 1},
  2490    { 0x1d00, 0x1dbf, 1},
  2491    { 0x1e00, 0x1f15, 1},
  2492    { 0x1f18, 0x1f1d, 1},
  2493    { 0x1f20, 0x1f45, 1},
  2494    { 0x1f48, 0x1f4d, 1},
  2495    { 0x1f50, 0x1f57, 1},
  2496    { 0x1f59, 0x1f5f, 2},
  2497    { 0x1f60, 0x1f7d, 1},
  2498    { 0x1f80, 0x1fb4, 1},
  2499    { 0x1fb6, 0x1fbc, 1},
  2500    { 0x1fbe, 0x1fc2, 4},
  2501    { 0x1fc3, 0x1fc4, 1},
  2502    { 0x1fc6, 0x1fcc, 1},
  2503    { 0x1fd0, 0x1fd3, 1},
  2504    { 0x1fd6, 0x1fdb, 1},
  2505    { 0x1fe0, 0x1fec, 1},
  2506    { 0x1ff2, 0x1ff4, 1},
  2507    { 0x1ff6, 0x1ffc, 1},
  2508    { 0x2071, 0x207f, 14},
  2509    { 0x2090, 0x209c, 1},
  2510    { 0x2102, 0x2107, 5},
  2511    { 0x210a, 0x2113, 1},
  2512    { 0x2115, 0x2119, 4},
  2513    { 0x211a, 0x211d, 1},
  2514    { 0x2124, 0x212a, 2},
  2515    { 0x212b, 0x212d, 1},
  2516    { 0x212f, 0x2139, 1},
  2517    { 0x213c, 0x213f, 1},
  2518    { 0x2145, 0x2149, 1},
  2519    { 0x214e, 0x2183, 53},
  2520    { 0x2184, 0x2c00, 2684},
  2521    { 0x2c01, 0x2c2e, 1},
  2522    { 0x2c30, 0x2c5e, 1},
  2523    { 0x2c60, 0x2ce4, 1},
  2524    { 0x2ceb, 0x2cee, 1},
  2525    { 0x2cf2, 0x2cf3, 1},
  2526    { 0x2d00, 0x2d25, 1},
  2527    { 0x2d27, 0x2d2d, 6},
  2528    { 0x2d30, 0x2d67, 1},
  2529    { 0x2d6f, 0x2d80, 17},
  2530    { 0x2d81, 0x2d96, 1},
  2531    { 0x2da0, 0x2da6, 1},
  2532    { 0x2da8, 0x2dae, 1},
  2533    { 0x2db0, 0x2db6, 1},
  2534    { 0x2db8, 0x2dbe, 1},
  2535    { 0x2dc0, 0x2dc6, 1},
  2536    { 0x2dc8, 0x2dce, 1},
  2537    { 0x2dd0, 0x2dd6, 1},
  2538    { 0x2dd8, 0x2dde, 1},
  2539    { 0x2e2f, 0x3005, 470},
  2540    { 0x3006, 0x3031, 43},
  2541    { 0x3032, 0x3035, 1},
  2542    { 0x303b, 0x303c, 1},
  2543    { 0x3041, 0x3096, 1},
  2544    { 0x309d, 0x309f, 1},
  2545    { 0x30a1, 0x30fa, 1},
  2546    { 0x30fc, 0x30ff, 1},
  2547    { 0x3105, 0x312d, 1},
  2548    { 0x3131, 0x318e, 1},
  2549    { 0x31a0, 0x31ba, 1},
  2550    { 0x31f0, 0x31ff, 1},
  2551    { 0x3400, 0x4db5, 1},
  2552    { 0x4e00, 0x9fd5, 1},
  2553    { 0xa000, 0xa48c, 1},
  2554    { 0xa4d0, 0xa4fd, 1},
  2555    { 0xa500, 0xa60c, 1},
  2556    { 0xa610, 0xa61f, 1},
  2557    { 0xa62a, 0xa62b, 1},
  2558    { 0xa640, 0xa66e, 1},
  2559    { 0xa67f, 0xa69d, 1},
  2560    { 0xa6a0, 0xa6e5, 1},
  2561    { 0xa717, 0xa71f, 1},
  2562    { 0xa722, 0xa788, 1},
  2563    { 0xa78b, 0xa7ad, 1},
  2564    { 0xa7b0, 0xa7b7, 1},
  2565    { 0xa7f7, 0xa801, 1},
  2566    { 0xa803, 0xa805, 1},
  2567    { 0xa807, 0xa80a, 1},
  2568    { 0xa80c, 0xa822, 1},
  2569    { 0xa840, 0xa873, 1},
  2570    { 0xa882, 0xa8b3, 1},
  2571    { 0xa8f2, 0xa8f7, 1},
  2572    { 0xa8fb, 0xa8fd, 2},
  2573    { 0xa90a, 0xa925, 1},
  2574    { 0xa930, 0xa946, 1},
  2575    { 0xa960, 0xa97c, 1},
  2576    { 0xa984, 0xa9b2, 1},
  2577    { 0xa9cf, 0xa9e0, 17},
  2578    { 0xa9e1, 0xa9e4, 1},
  2579    { 0xa9e6, 0xa9ef, 1},
  2580    { 0xa9fa, 0xa9fe, 1},
  2581    { 0xaa00, 0xaa28, 1},
  2582    { 0xaa40, 0xaa42, 1},
  2583    { 0xaa44, 0xaa4b, 1},
  2584    { 0xaa60, 0xaa76, 1},
  2585    { 0xaa7a, 0xaa7e, 4},
  2586    { 0xaa7f, 0xaaaf, 1},
  2587    { 0xaab1, 0xaab5, 4},
  2588    { 0xaab6, 0xaab9, 3},
  2589    { 0xaaba, 0xaabd, 1},
  2590    { 0xaac0, 0xaac2, 2},
  2591    { 0xaadb, 0xaadd, 1},
  2592    { 0xaae0, 0xaaea, 1},
  2593    { 0xaaf2, 0xaaf4, 1},
  2594    { 0xab01, 0xab06, 1},
  2595    { 0xab09, 0xab0e, 1},
  2596    { 0xab11, 0xab16, 1},
  2597    { 0xab20, 0xab26, 1},
  2598    { 0xab28, 0xab2e, 1},
  2599    { 0xab30, 0xab5a, 1},
  2600    { 0xab5c, 0xab65, 1},
  2601    { 0xab70, 0xabe2, 1},
  2602    { 0xac00, 0xd7a3, 1},
  2603    { 0xd7b0, 0xd7c6, 1},
  2604    { 0xd7cb, 0xd7fb, 1},
  2605    { 0xf900, 0xfa6d, 1},
  2606    { 0xfa70, 0xfad9, 1},
  2607    { 0xfb00, 0xfb06, 1},
  2608    { 0xfb13, 0xfb17, 1},
  2609    { 0xfb1d, 0xfb1f, 2},
  2610    { 0xfb20, 0xfb28, 1},
  2611    { 0xfb2a, 0xfb36, 1},
  2612    { 0xfb38, 0xfb3c, 1},
  2613    { 0xfb3e, 0xfb40, 2},
  2614    { 0xfb41, 0xfb43, 2},
  2615    { 0xfb44, 0xfb46, 2},
  2616    { 0xfb47, 0xfbb1, 1},
  2617    { 0xfbd3, 0xfd3d, 1},
  2618    { 0xfd50, 0xfd8f, 1},
  2619    { 0xfd92, 0xfdc7, 1},
  2620    { 0xfdf0, 0xfdfb, 1},
  2621    { 0xfe70, 0xfe74, 1},
  2622    { 0xfe76, 0xfefc, 1},
  2623    { 0xff21, 0xff3a, 1},
  2624    { 0xff41, 0xff5a, 1},
  2625    { 0xff66, 0xffbe, 1},
  2626    { 0xffc2, 0xffc7, 1},
  2627    { 0xffca, 0xffcf, 1},
  2628    { 0xffd2, 0xffd7, 1},
  2629    { 0xffda, 0xffdc, 1},
  2630    { 0x10000, 0x1000b, 1},
  2631    { 0x1000d, 0x10026, 1},
  2632    { 0x10028, 0x1003a, 1},
  2633    { 0x1003c, 0x1003d, 1},
  2634    { 0x1003f, 0x1004d, 1},
  2635    { 0x10050, 0x1005d, 1},
  2636    { 0x10080, 0x100fa, 1},
  2637    { 0x10280, 0x1029c, 1},
  2638    { 0x102a0, 0x102d0, 1},
  2639    { 0x10300, 0x1031f, 1},
  2640    { 0x10330, 0x10340, 1},
  2641    { 0x10342, 0x10349, 1},
  2642    { 0x10350, 0x10375, 1},
  2643    { 0x10380, 0x1039d, 1},
  2644    { 0x103a0, 0x103c3, 1},
  2645    { 0x103c8, 0x103cf, 1},
  2646    { 0x10400, 0x1049d, 1},
  2647    { 0x10500, 0x10527, 1},
  2648    { 0x10530, 0x10563, 1},
  2649    { 0x10600, 0x10736, 1},
  2650    { 0x10740, 0x10755, 1},
  2651    { 0x10760, 0x10767, 1},
  2652    { 0x10800, 0x10805, 1},
  2653    { 0x10808, 0x1080a, 2},
  2654    { 0x1080b, 0x10835, 1},
  2655    { 0x10837, 0x10838, 1},
  2656    { 0x1083c, 0x1083f, 3},
  2657    { 0x10840, 0x10855, 1},
  2658    { 0x10860, 0x10876, 1},
  2659    { 0x10880, 0x1089e, 1},
  2660    { 0x108e0, 0x108f2, 1},
  2661    { 0x108f4, 0x108f5, 1},
  2662    { 0x10900, 0x10915, 1},
  2663    { 0x10920, 0x10939, 1},
  2664    { 0x10980, 0x109b7, 1},
  2665    { 0x109be, 0x109bf, 1},
  2666    { 0x10a00, 0x10a10, 16},
  2667    { 0x10a11, 0x10a13, 1},
  2668    { 0x10a15, 0x10a17, 1},
  2669    { 0x10a19, 0x10a33, 1},
  2670    { 0x10a60, 0x10a7c, 1},
  2671    { 0x10a80, 0x10a9c, 1},
  2672    { 0x10ac0, 0x10ac7, 1},
  2673    { 0x10ac9, 0x10ae4, 1},
  2674    { 0x10b00, 0x10b35, 1},
  2675    { 0x10b40, 0x10b55, 1},
  2676    { 0x10b60, 0x10b72, 1},
  2677    { 0x10b80, 0x10b91, 1},
  2678    { 0x10c00, 0x10c48, 1},
  2679    { 0x10c80, 0x10cb2, 1},
  2680    { 0x10cc0, 0x10cf2, 1},
  2681    { 0x11003, 0x11037, 1},
  2682    { 0x11083, 0x110af, 1},
  2683    { 0x110d0, 0x110e8, 1},
  2684    { 0x11103, 0x11126, 1},
  2685    { 0x11150, 0x11172, 1},
  2686    { 0x11176, 0x11183, 13},
  2687    { 0x11184, 0x111b2, 1},
  2688    { 0x111c1, 0x111c4, 1},
  2689    { 0x111da, 0x111dc, 2},
  2690    { 0x11200, 0x11211, 1},
  2691    { 0x11213, 0x1122b, 1},
  2692    { 0x11280, 0x11286, 1},
  2693    { 0x11288, 0x1128a, 2},
  2694    { 0x1128b, 0x1128d, 1},
  2695    { 0x1128f, 0x1129d, 1},
  2696    { 0x1129f, 0x112a8, 1},
  2697    { 0x112b0, 0x112de, 1},
  2698    { 0x11305, 0x1130c, 1},
  2699    { 0x1130f, 0x11310, 1},
  2700    { 0x11313, 0x11328, 1},
  2701    { 0x1132a, 0x11330, 1},
  2702    { 0x11332, 0x11333, 1},
  2703    { 0x11335, 0x11339, 1},
  2704    { 0x1133d, 0x11350, 19},
  2705    { 0x1135d, 0x11361, 1},
  2706    { 0x11480, 0x114af, 1},
  2707    { 0x114c4, 0x114c5, 1},
  2708    { 0x114c7, 0x11580, 185},
  2709    { 0x11581, 0x115ae, 1},
  2710    { 0x115d8, 0x115db, 1},
  2711    { 0x11600, 0x1162f, 1},
  2712    { 0x11644, 0x11680, 60},
  2713    { 0x11681, 0x116aa, 1},
  2714    { 0x11700, 0x11719, 1},
  2715    { 0x118a0, 0x118df, 1},
  2716    { 0x118ff, 0x11ac0, 449},
  2717    { 0x11ac1, 0x11af8, 1},
  2718    { 0x12000, 0x12399, 1},
  2719    { 0x12480, 0x12543, 1},
  2720    { 0x13000, 0x1342e, 1},
  2721    { 0x14400, 0x14646, 1},
  2722    { 0x16800, 0x16a38, 1},
  2723    { 0x16a40, 0x16a5e, 1},
  2724    { 0x16ad0, 0x16aed, 1},
  2725    { 0x16b00, 0x16b2f, 1},
  2726    { 0x16b40, 0x16b43, 1},
  2727    { 0x16b63, 0x16b77, 1},
  2728    { 0x16b7d, 0x16b8f, 1},
  2729    { 0x16f00, 0x16f44, 1},
  2730    { 0x16f50, 0x16f93, 67},
  2731    { 0x16f94, 0x16f9f, 1},
  2732    { 0x1b000, 0x1b001, 1},
  2733    { 0x1bc00, 0x1bc6a, 1},
  2734    { 0x1bc70, 0x1bc7c, 1},
  2735    { 0x1bc80, 0x1bc88, 1},
  2736    { 0x1bc90, 0x1bc99, 1},
  2737    { 0x1d400, 0x1d454, 1},
  2738    { 0x1d456, 0x1d49c, 1},
  2739    { 0x1d49e, 0x1d49f, 1},
  2740    { 0x1d4a2, 0x1d4a5, 3},
  2741    { 0x1d4a6, 0x1d4a9, 3},
  2742    { 0x1d4aa, 0x1d4ac, 1},
  2743    { 0x1d4ae, 0x1d4b9, 1},
  2744    { 0x1d4bb, 0x1d4bd, 2},
  2745    { 0x1d4be, 0x1d4c3, 1},
  2746    { 0x1d4c5, 0x1d505, 1},
  2747    { 0x1d507, 0x1d50a, 1},
  2748    { 0x1d50d, 0x1d514, 1},
  2749    { 0x1d516, 0x1d51c, 1},
  2750    { 0x1d51e, 0x1d539, 1},
  2751    { 0x1d53b, 0x1d53e, 1},
  2752    { 0x1d540, 0x1d544, 1},
  2753    { 0x1d546, 0x1d54a, 4},
  2754    { 0x1d54b, 0x1d550, 1},
  2755    { 0x1d552, 0x1d6a5, 1},
  2756    { 0x1d6a8, 0x1d6c0, 1},
  2757    { 0x1d6c2, 0x1d6da, 1},
  2758    { 0x1d6dc, 0x1d6fa, 1},
  2759    { 0x1d6fc, 0x1d714, 1},
  2760    { 0x1d716, 0x1d734, 1},
  2761    { 0x1d736, 0x1d74e, 1},
  2762    { 0x1d750, 0x1d76e, 1},
  2763    { 0x1d770, 0x1d788, 1},
  2764    { 0x1d78a, 0x1d7a8, 1},
  2765    { 0x1d7aa, 0x1d7c2, 1},
  2766    { 0x1d7c4, 0x1d7cb, 1},
  2767    { 0x1e800, 0x1e8c4, 1},
  2768    { 0x1ee00, 0x1ee03, 1},
  2769    { 0x1ee05, 0x1ee1f, 1},
  2770    { 0x1ee21, 0x1ee22, 1},
  2771    { 0x1ee24, 0x1ee27, 3},
  2772    { 0x1ee29, 0x1ee32, 1},
  2773    { 0x1ee34, 0x1ee37, 1},
  2774    { 0x1ee39, 0x1ee3b, 2},
  2775    { 0x1ee42, 0x1ee47, 5},
  2776    { 0x1ee49, 0x1ee4d, 2},
  2777    { 0x1ee4e, 0x1ee4f, 1},
  2778    { 0x1ee51, 0x1ee52, 1},
  2779    { 0x1ee54, 0x1ee57, 3},
  2780    { 0x1ee59, 0x1ee61, 2},
  2781    { 0x1ee62, 0x1ee64, 2},
  2782    { 0x1ee67, 0x1ee6a, 1},
  2783    { 0x1ee6c, 0x1ee72, 1},
  2784    { 0x1ee74, 0x1ee77, 1},
  2785    { 0x1ee79, 0x1ee7c, 1},
  2786    { 0x1ee7e, 0x1ee80, 2},
  2787    { 0x1ee81, 0x1ee89, 1},
  2788    { 0x1ee8b, 0x1ee9b, 1},
  2789    { 0x1eea1, 0x1eea3, 1},
  2790    { 0x1eea5, 0x1eea9, 1},
  2791    { 0x1eeab, 0x1eebb, 1},
  2792    { 0x20000, 0x2a6d6, 1},
  2793    { 0x2a700, 0x2b734, 1},
  2794    { 0x2b740, 0x2b81d, 1},
  2795    { 0x2b820, 0x2cea1, 1},
  2796    { 0x2f800, 0x2fa1d, 1},
  2797  };
  2798  
  2799  // A table of Unicode uppercase letters--Unicode code points
  2800  // classified as "Letter, uppercase".
  2801  
  2802  static const Unicode_range unicode_uppercase_letters[] =
  2803  {
  2804    { 0x0041, 0x005a, 1},
  2805    { 0x00c0, 0x00d6, 1},
  2806    { 0x00d8, 0x00de, 1},
  2807    { 0x0100, 0x0136, 2},
  2808    { 0x0139, 0x0147, 2},
  2809    { 0x014a, 0x0178, 2},
  2810    { 0x0179, 0x017d, 2},
  2811    { 0x0181, 0x0182, 1},
  2812    { 0x0184, 0x0186, 2},
  2813    { 0x0187, 0x0189, 2},
  2814    { 0x018a, 0x018b, 1},
  2815    { 0x018e, 0x0191, 1},
  2816    { 0x0193, 0x0194, 1},
  2817    { 0x0196, 0x0198, 1},
  2818    { 0x019c, 0x019d, 1},
  2819    { 0x019f, 0x01a0, 1},
  2820    { 0x01a2, 0x01a6, 2},
  2821    { 0x01a7, 0x01a9, 2},
  2822    { 0x01ac, 0x01ae, 2},
  2823    { 0x01af, 0x01b1, 2},
  2824    { 0x01b2, 0x01b3, 1},
  2825    { 0x01b5, 0x01b7, 2},
  2826    { 0x01b8, 0x01bc, 4},
  2827    { 0x01c4, 0x01cd, 3},
  2828    { 0x01cf, 0x01db, 2},
  2829    { 0x01de, 0x01ee, 2},
  2830    { 0x01f1, 0x01f4, 3},
  2831    { 0x01f6, 0x01f8, 1},
  2832    { 0x01fa, 0x0232, 2},
  2833    { 0x023a, 0x023b, 1},
  2834    { 0x023d, 0x023e, 1},
  2835    { 0x0241, 0x0243, 2},
  2836    { 0x0244, 0x0246, 1},
  2837    { 0x0248, 0x024e, 2},
  2838    { 0x0370, 0x0372, 2},
  2839    { 0x0376, 0x037f, 9},
  2840    { 0x0386, 0x0388, 2},
  2841    { 0x0389, 0x038a, 1},
  2842    { 0x038c, 0x038e, 2},
  2843    { 0x038f, 0x0391, 2},
  2844    { 0x0392, 0x03a1, 1},
  2845    { 0x03a3, 0x03ab, 1},
  2846    { 0x03cf, 0x03d2, 3},
  2847    { 0x03d3, 0x03d4, 1},
  2848    { 0x03d8, 0x03ee, 2},
  2849    { 0x03f4, 0x03f7, 3},
  2850    { 0x03f9, 0x03fa, 1},
  2851    { 0x03fd, 0x042f, 1},
  2852    { 0x0460, 0x0480, 2},
  2853    { 0x048a, 0x04c0, 2},
  2854    { 0x04c1, 0x04cd, 2},
  2855    { 0x04d0, 0x052e, 2},
  2856    { 0x0531, 0x0556, 1},
  2857    { 0x10a0, 0x10c5, 1},
  2858    { 0x10c7, 0x10cd, 6},
  2859    { 0x1e00, 0x1e94, 2},
  2860    { 0x1e9e, 0x1efe, 2},
  2861    { 0x1f08, 0x1f0f, 1},
  2862    { 0x1f18, 0x1f1d, 1},
  2863    { 0x1f28, 0x1f2f, 1},
  2864    { 0x1f38, 0x1f3f, 1},
  2865    { 0x1f48, 0x1f4d, 1},
  2866    { 0x1f59, 0x1f5f, 2},
  2867    { 0x1f68, 0x1f6f, 1},
  2868    { 0x1fb8, 0x1fbb, 1},
  2869    { 0x1fc8, 0x1fcb, 1},
  2870    { 0x1fd8, 0x1fdb, 1},
  2871    { 0x1fe8, 0x1fec, 1},
  2872    { 0x1ff8, 0x1ffb, 1},
  2873    { 0x2102, 0x2107, 5},
  2874    { 0x210b, 0x210d, 1},
  2875    { 0x2110, 0x2112, 1},
  2876    { 0x2115, 0x2119, 4},
  2877    { 0x211a, 0x211d, 1},
  2878    { 0x2124, 0x212a, 2},
  2879    { 0x212b, 0x212d, 1},
  2880    { 0x2130, 0x2133, 1},
  2881    { 0x213e, 0x213f, 1},
  2882    { 0x2145, 0x2183, 62},
  2883    { 0x2c00, 0x2c2e, 1},
  2884    { 0x2c60, 0x2c62, 2},
  2885    { 0x2c63, 0x2c64, 1},
  2886    { 0x2c67, 0x2c6d, 2},
  2887    { 0x2c6e, 0x2c70, 1},
  2888    { 0x2c72, 0x2c75, 3},
  2889    { 0x2c7e, 0x2c80, 1},
  2890    { 0x2c82, 0x2ce2, 2},
  2891    { 0x2ceb, 0x2ced, 2},
  2892    { 0x2cf2, 0xa640, 31054},
  2893    { 0xa642, 0xa66c, 2},
  2894    { 0xa680, 0xa69a, 2},
  2895    { 0xa722, 0xa72e, 2},
  2896    { 0xa732, 0xa76e, 2},
  2897    { 0xa779, 0xa77d, 2},
  2898    { 0xa77e, 0xa786, 2},
  2899    { 0xa78b, 0xa78d, 2},
  2900    { 0xa790, 0xa792, 2},
  2901    { 0xa796, 0xa7aa, 2},
  2902    { 0xa7ab, 0xa7ad, 1},
  2903    { 0xa7b0, 0xa7b1, 1},
  2904    { 0xff21, 0xff3a, 1},
  2905    { 0x10400, 0x10427, 1},
  2906    { 0x118a0, 0x118bf, 1},
  2907    { 0x1d400, 0x1d419, 1},
  2908    { 0x1d434, 0x1d44d, 1},
  2909    { 0x1d468, 0x1d481, 1},
  2910    { 0x1d49c, 0x1d49e, 2},
  2911    { 0x1d49f, 0x1d4a5, 3},
  2912    { 0x1d4a6, 0x1d4a9, 3},
  2913    { 0x1d4aa, 0x1d4ac, 1},
  2914    { 0x1d4ae, 0x1d4b5, 1},
  2915    { 0x1d4d0, 0x1d4e9, 1},
  2916    { 0x1d504, 0x1d505, 1},
  2917    { 0x1d507, 0x1d50a, 1},
  2918    { 0x1d50d, 0x1d514, 1},
  2919    { 0x1d516, 0x1d51c, 1},
  2920    { 0x1d538, 0x1d539, 1},
  2921    { 0x1d53b, 0x1d53e, 1},
  2922    { 0x1d540, 0x1d544, 1},
  2923    { 0x1d546, 0x1d54a, 4},
  2924    { 0x1d54b, 0x1d550, 1},
  2925    { 0x1d56c, 0x1d585, 1},
  2926    { 0x1d5a0, 0x1d5b9, 1},
  2927    { 0x1d5d4, 0x1d5ed, 1},
  2928    { 0x1d608, 0x1d621, 1},
  2929    { 0x1d63c, 0x1d655, 1},
  2930    { 0x1d670, 0x1d689, 1},
  2931    { 0x1d6a8, 0x1d6c0, 1},
  2932    { 0x1d6e2, 0x1d6fa, 1},
  2933    { 0x1d71c, 0x1d734, 1},
  2934    { 0x1d756, 0x1d76e, 1},
  2935    { 0x1d790, 0x1d7a8, 1},
  2936    { 0x1d7ca, 0x1d7ca, 1},
  2937  };
  2938  
  2939  // Return true if C is in RANGES.
  2940  
  2941  bool
  2942  Lex::is_in_unicode_range(unsigned int c, const Unicode_range* ranges,
  2943  			 size_t range_size)
  2944  {
  2945    if (c < 0x100)
  2946      {
  2947        // The common case is a small value, and we know that it will be
  2948        // in the first few entries of the table.  Do a linear scan
  2949        // rather than a binary search.
  2950        for (size_t i = 0; i < range_size; ++i)
  2951  	{
  2952  	  const Unicode_range* p = &ranges[i];
  2953  	  if (c <= p->high)
  2954  	    {
  2955  	      if (c < p->low)
  2956  		return false;
  2957  	      return (c - p->low) % p->stride == 0;
  2958  	    }
  2959  	}
  2960        return false;
  2961      }
  2962    else
  2963      {
  2964        size_t lo = 0;
  2965        size_t hi = range_size;
  2966        while (lo < hi)
  2967  	{
  2968  	  size_t mid = lo + (hi - lo) / 2;
  2969  	  const Unicode_range* p = &ranges[mid];
  2970  	  if (c < p->low)
  2971  	    hi = mid;
  2972  	  else if (c > p->high)
  2973  	    lo = mid + 1;
  2974  	  else
  2975  	    return (c - p->low) % p->stride == 0;
  2976  	}
  2977        return false;
  2978      }
  2979  }
  2980  
  2981  // Return whether C is a space character.
  2982  
  2983  bool
  2984  Lex::is_unicode_space(unsigned int c)
  2985  {
  2986    return Lex::is_in_unicode_range(c, unicode_space,
  2987  				  ARRAY_SIZE(unicode_space));
  2988  }
  2989  
  2990  // Return whether C is a Unicode digit--a Unicode code point
  2991  // classified as "Digit".
  2992  
  2993  bool
  2994  Lex::is_unicode_digit(unsigned int c)
  2995  {
  2996    return Lex::is_in_unicode_range(c, unicode_digits,
  2997  				  ARRAY_SIZE(unicode_digits));
  2998  }
  2999  
  3000  // Return whether C is a Unicode letter--a Unicode code point
  3001  // classified as "Letter".
  3002  
  3003  bool
  3004  Lex::is_unicode_letter(unsigned int c)
  3005  {
  3006    return Lex::is_in_unicode_range(c, unicode_letters,
  3007  				  ARRAY_SIZE(unicode_letters));
  3008  }
  3009  
  3010  // Return whether C is a Unicode uppercase letter.  a Unicode code
  3011  // point classified as "Letter, uppercase".
  3012  
  3013  bool
  3014  Lex::is_unicode_uppercase(unsigned int c)
  3015  {
  3016    return Lex::is_in_unicode_range(c, unicode_uppercase_letters,
  3017  				  ARRAY_SIZE(unicode_uppercase_letters));
  3018  }
  3019  
  3020  // Return whether the identifier NAME should be exported.  NAME is a
  3021  // mangled name which includes only ASCII characters.
  3022  
  3023  bool
  3024  Lex::is_exported_mangled_name(const std::string& name)
  3025  {
  3026    unsigned char c = name[0];
  3027    if (c != '.')
  3028      return c >= 'A' && c <= 'Z';
  3029    else
  3030      {
  3031        const char* p = name.data();
  3032        size_t len = name.length();
  3033        if (len < 4 || p[1] != '.' || (p[2] != 'u' && p[2] != 'U'))
  3034  	return false;
  3035        unsigned int ci = 0;
  3036        size_t want = (p[2] == 'u' ? 4 : 8);
  3037        if (len < want + 3)
  3038  	return false;
  3039        for (size_t i = 3; i < want; ++i)
  3040  	{
  3041  	  c = p[i];
  3042  	  if (!Lex::is_hex_digit(c))
  3043  	    return false;
  3044  	  ci <<= 4;
  3045  	  ci |= Lex::hex_val(c);
  3046  	}
  3047        return Lex::is_unicode_uppercase(ci);
  3048      }
  3049  }
  3050  
  3051  // Return whether the identifier NAME should be exported.  NAME is a
  3052  // an unmangled utf-8 string and may contain non-ASCII characters.
  3053  
  3054  bool
  3055  Lex::is_exported_name(const std::string& name)
  3056  {
  3057    unsigned int uchar;
  3058    if (Lex::fetch_char(name.c_str(), &uchar) != 0)
  3059      return Lex::is_unicode_letter(uchar) && Lex::is_unicode_uppercase(uchar);
  3060    return false;
  3061  }
  3062  
  3063  // Return whether the identifier NAME contains an invalid character.
  3064  // This is based on how we handle invalid characters in
  3065  // gather_identifier.
  3066  
  3067  bool
  3068  Lex::is_invalid_identifier(const std::string& name)
  3069  {
  3070    return name.find("$INVALID$") != std::string::npos;
  3071  }