| // Copyright (C) 2015 The Android Open Source Project | 
 | // | 
 | // Licensed under the Apache License, Version 2.0 (the "License"); | 
 | // you may not use this file except in compliance with the License. | 
 | // You may obtain a copy of the License at | 
 | // | 
 | //      http://www.apache.org/licenses/LICENSE-2.0 | 
 | // | 
 | // Unless required by applicable law or agreed to in writing, software | 
 | // distributed under the License is distributed on an "AS IS" BASIS, | 
 | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
 | // See the License for the specific language governing permissions and | 
 | // limitations under the License. | 
 |  | 
 | #include "tokenizer.h" | 
 |  | 
 | namespace init { | 
 |  | 
 | Tokenizer::Tokenizer(const std::string& data) | 
 |     : data_(data), eof_(false), pos_(0), tok_start_(0) { | 
 |   current_.type = TOK_START; | 
 |  | 
 |   if (data.size() > 0) { | 
 |     cur_char_ = data[0]; | 
 |   } else { | 
 |     eof_ = true; | 
 |     cur_char_ = '\0'; | 
 |   } | 
 | } | 
 |  | 
 | Tokenizer::~Tokenizer() {} | 
 |  | 
 | const Tokenizer::Token& Tokenizer::current() { | 
 |   return current_; | 
 | } | 
 |  | 
 | bool Tokenizer::Next() { | 
 |   while (!eof_) { | 
 |     AdvWhiteSpace(); | 
 |  | 
 |     // Check for comments. | 
 |     if (cur_char_ == '#') { | 
 |       AdvChar(); | 
 |       // Skip rest of line | 
 |       while (!eof_ && cur_char_ != '\n') { | 
 |         AdvChar(); | 
 |       } | 
 |     } | 
 |  | 
 |     if (eof_) { | 
 |       break; | 
 |     } | 
 |  | 
 |     if (cur_char_ == '\0') { | 
 |       AdvChar(); | 
 |     } else if (cur_char_ == '\n') { | 
 |       current_.type = TOK_NEWLINE; | 
 |       current_.text.clear(); | 
 |       AdvChar(); | 
 |       return true; | 
 |     } else if (cur_char_ == '\\') { | 
 |       AdvChar();  // skip backslash | 
 |       // This is line continuation so | 
 |       // do not generated TOK_NEWLINE at | 
 |       // the next \n. | 
 |       AdvUntil('\n'); | 
 |       AdvChar();  // skip \n | 
 |     } else if (cur_char_ == '\"') { | 
 |       AdvChar(); | 
 |       StartText(); | 
 |       // Grab everything until the next quote. | 
 |       AdvUntil('\"'); | 
 |       EndText(); | 
 |       AdvChar();  // skip quote. | 
 |       return true; | 
 |     } else { | 
 |       StartText(); | 
 |       AdvText(); | 
 |       EndText(); | 
 |       return true; | 
 |     } | 
 |   } | 
 |   current_.type = TOK_END; | 
 |   current_.text.clear(); | 
 |   return false; | 
 | } | 
 |  | 
 | void Tokenizer::AdvChar() { | 
 |   pos_++; | 
 |   if (pos_ < data_.size()) { | 
 |     cur_char_ = data_[pos_]; | 
 |   } else { | 
 |     eof_ = true; | 
 |     cur_char_ = '\0'; | 
 |   } | 
 | } | 
 |  | 
 | void Tokenizer::AdvWhiteSpace() { | 
 |   while (cur_char_ == '\t' || cur_char_ == '\r' || cur_char_ == ' ') { | 
 |     AdvChar(); | 
 |   } | 
 | } | 
 |  | 
 | void Tokenizer::AdvUntil(char x) { | 
 |   while (!eof_ && cur_char_ != x) { | 
 |     AdvChar(); | 
 |   } | 
 | } | 
 |  | 
 | void Tokenizer::AdvText() { | 
 |   while (cur_char_ != '\t' && cur_char_ != '\r' && cur_char_ != '\0' && | 
 |          cur_char_ != ' ' && cur_char_ != '\n' && cur_char_ != '#') { | 
 |     AdvChar(); | 
 |   } | 
 | } | 
 |  | 
 | void Tokenizer::StartText() { | 
 |   current_.text.clear(); | 
 |   tok_start_ = pos_; | 
 |   current_.type = TOK_TEXT; | 
 | } | 
 |  | 
 | void Tokenizer::EndText() { | 
 |   if (pos_ != tok_start_) { | 
 |     current_.text.append(data_, tok_start_, pos_ - tok_start_); | 
 |   } | 
 | } | 
 |  | 
 | }  // namespace init |