| Adam Lesinski | ffa1686 | 2014-01-23 18:17:42 -0800 | [diff] [blame] | 1 | #include "aidl_language.h" |
| Christopher Wiley | f690be5 | 2015-09-14 15:19:10 -0700 | [diff] [blame] | 2 | |
| 3 | #include <iostream> |
| Adam Lesinski | ffa1686 | 2014-01-23 18:17:42 -0800 | [diff] [blame] | 4 | #include <stdio.h> |
| Adam Lesinski | ffa1686 | 2014-01-23 18:17:42 -0800 | [diff] [blame] | 5 | #include <stdlib.h> |
| Christopher Wiley | 4a2884b | 2015-10-07 11:27:45 -0700 | [diff] [blame] | 6 | #include <string.h> |
| Casey Dahlin | dd69181 | 2015-09-09 17:59:06 -0700 | [diff] [blame] | 7 | #include <string> |
| Christopher Wiley | f690be5 | 2015-09-14 15:19:10 -0700 | [diff] [blame] | 8 | |
| Elliott Hughes | 0a62067 | 2015-12-04 13:53:18 -0800 | [diff] [blame] | 9 | #include <android-base/strings.h> |
| Christopher Wiley | d76067c | 2015-10-19 17:00:13 -0700 | [diff] [blame] | 10 | |
| Christopher Wiley | f690be5 | 2015-09-14 15:19:10 -0700 | [diff] [blame] | 11 | #include "aidl_language_y.hpp" |
| Christopher Wiley | 4a2884b | 2015-10-07 11:27:45 -0700 | [diff] [blame] | 12 | #include "logging.h" |
| Adam Lesinski | ffa1686 | 2014-01-23 18:17:42 -0800 | [diff] [blame] | 13 | |
| Casey Dahlin | 07b9dde | 2015-09-10 19:13:49 -0700 | [diff] [blame] | 14 | #ifdef _WIN32 |
| 15 | int isatty(int fd) |
| 16 | { |
| 17 | return (fd == 0); |
| 18 | } |
| 19 | #endif |
| 20 | |
| Christopher Wiley | 4a2884b | 2015-10-07 11:27:45 -0700 | [diff] [blame] | 21 | using android::aidl::IoDelegate; |
| Christopher Wiley | d76067c | 2015-10-19 17:00:13 -0700 | [diff] [blame] | 22 | using android::base::Join; |
| Christopher Wiley | 8aa4d9f | 2015-11-16 19:10:45 -0800 | [diff] [blame] | 23 | using android::base::Split; |
| Casey Dahlin | dd69181 | 2015-09-09 17:59:06 -0700 | [diff] [blame] | 24 | using std::cerr; |
| 25 | using std::endl; |
| Christopher Wiley | 4a2884b | 2015-10-07 11:27:45 -0700 | [diff] [blame] | 26 | using std::string; |
| 27 | using std::unique_ptr; |
| Adam Lesinski | ffa1686 | 2014-01-23 18:17:42 -0800 | [diff] [blame] | 28 | |
| Casey Dahlin | dd69181 | 2015-09-09 17:59:06 -0700 | [diff] [blame] | 29 | void yylex_init(void **); |
| 30 | void yylex_destroy(void *); |
| 31 | void yyset_in(FILE *f, void *); |
| Casey Dahlin | e250749 | 2015-09-14 17:11:20 -0700 | [diff] [blame] | 32 | int yyparse(Parser*); |
| Christopher Wiley | 4a2884b | 2015-10-07 11:27:45 -0700 | [diff] [blame] | 33 | YY_BUFFER_STATE yy_scan_buffer(char *, size_t, void *); |
| Casey Dahlin | 89d4484 | 2015-09-24 18:45:54 -0700 | [diff] [blame] | 34 | void yy_delete_buffer(YY_BUFFER_STATE, void *); |
| Casey Dahlin | dd69181 | 2015-09-09 17:59:06 -0700 | [diff] [blame] | 35 | |
| Casey Dahlin | cdbbc8c | 2015-10-14 15:31:04 -0700 | [diff] [blame] | 36 | AidlToken::AidlToken(const std::string& text, const std::string& comments) |
| 37 | : text_(text), |
| 38 | comments_(comments) {} |
| Casey Dahlin | 98a544b | 2015-10-14 14:22:55 -0700 | [diff] [blame] | 39 | |
| Casey Dahlin | f2d23f7 | 2015-10-02 16:19:19 -0700 | [diff] [blame] | 40 | AidlType::AidlType(const std::string& name, unsigned line, |
| Casey Dahlin | f7a421c | 2015-10-05 17:24:28 -0700 | [diff] [blame] | 41 | const std::string& comments, bool is_array) |
| Casey Dahlin | f2d23f7 | 2015-10-02 16:19:19 -0700 | [diff] [blame] | 42 | : name_(name), |
| 43 | line_(line), |
| Casey Dahlin | f7a421c | 2015-10-05 17:24:28 -0700 | [diff] [blame] | 44 | is_array_(is_array), |
| Casey Dahlin | f2d23f7 | 2015-10-02 16:19:19 -0700 | [diff] [blame] | 45 | comments_(comments) {} |
| 46 | |
| Casey Dahlin | 98a544b | 2015-10-14 14:22:55 -0700 | [diff] [blame] | 47 | string AidlType::ToString() const { |
| 48 | return name_ + (is_array_ ? "[]" : ""); |
| Casey Dahlin | 70078e6 | 2015-09-30 17:01:30 -0700 | [diff] [blame] | 49 | } |
| 50 | |
| Casey Dahlin | 0ee3758 | 2015-09-30 16:31:55 -0700 | [diff] [blame] | 51 | AidlArgument::AidlArgument(AidlArgument::Direction direction, AidlType* type, |
| Casey Dahlin | 308f9d4 | 2015-10-05 18:48:42 -0700 | [diff] [blame] | 52 | std::string name, unsigned line) |
| Casey Dahlin | 0ee3758 | 2015-09-30 16:31:55 -0700 | [diff] [blame] | 53 | : type_(type), |
| Casey Dahlin | fd6fb48 | 2015-09-30 14:48:18 -0700 | [diff] [blame] | 54 | direction_(direction), |
| 55 | direction_specified_(true), |
| Casey Dahlin | 308f9d4 | 2015-10-05 18:48:42 -0700 | [diff] [blame] | 56 | name_(name), |
| 57 | line_(line) {} |
| Casey Dahlin | c378c99 | 2015-09-29 16:50:40 -0700 | [diff] [blame] | 58 | |
| Casey Dahlin | 308f9d4 | 2015-10-05 18:48:42 -0700 | [diff] [blame] | 59 | AidlArgument::AidlArgument(AidlType* type, std::string name, unsigned line) |
| Casey Dahlin | 0ee3758 | 2015-09-30 16:31:55 -0700 | [diff] [blame] | 60 | : type_(type), |
| Casey Dahlin | fd6fb48 | 2015-09-30 14:48:18 -0700 | [diff] [blame] | 61 | direction_(AidlArgument::IN_DIR), |
| 62 | direction_specified_(false), |
| Casey Dahlin | 308f9d4 | 2015-10-05 18:48:42 -0700 | [diff] [blame] | 63 | name_(name), |
| 64 | line_(line) {} |
| Casey Dahlin | c378c99 | 2015-09-29 16:50:40 -0700 | [diff] [blame] | 65 | |
| 66 | string AidlArgument::ToString() const { |
| 67 | string ret; |
| 68 | |
| 69 | if (direction_specified_) { |
| 70 | switch(direction_) { |
| 71 | case AidlArgument::IN_DIR: |
| 72 | ret += "in "; |
| 73 | break; |
| 74 | case AidlArgument::OUT_DIR: |
| 75 | ret += "out "; |
| 76 | break; |
| 77 | case AidlArgument::INOUT_DIR: |
| 78 | ret += "inout "; |
| 79 | break; |
| 80 | } |
| 81 | } |
| 82 | |
| Casey Dahlin | 70078e6 | 2015-09-30 17:01:30 -0700 | [diff] [blame] | 83 | ret += type_->ToString(); |
| Casey Dahlin | c378c99 | 2015-09-29 16:50:40 -0700 | [diff] [blame] | 84 | ret += " "; |
| Casey Dahlin | d127b50 | 2015-09-30 12:51:08 -0700 | [diff] [blame] | 85 | ret += name_; |
| Casey Dahlin | c378c99 | 2015-09-29 16:50:40 -0700 | [diff] [blame] | 86 | |
| 87 | return ret; |
| 88 | } |
| Casey Dahlin | bc7a50a | 2015-09-28 19:20:50 -0700 | [diff] [blame] | 89 | |
| Casey Dahlin | d40e2fe | 2015-11-24 14:06:52 -0800 | [diff] [blame] | 90 | AidlConstant::AidlConstant(std::string name, int32_t value) |
| 91 | : name_(name), |
| 92 | value_(value) {} |
| 93 | |
| Casey Dahlin | f4a9311 | 2015-10-05 16:58:09 -0700 | [diff] [blame] | 94 | AidlMethod::AidlMethod(bool oneway, AidlType* type, std::string name, |
| 95 | std::vector<std::unique_ptr<AidlArgument>>* args, |
| Casey Dahlin | fb7da2e | 2015-10-08 17:26:09 -0700 | [diff] [blame] | 96 | unsigned line, const std::string& comments, int id) |
| Casey Dahlin | f4a9311 | 2015-10-05 16:58:09 -0700 | [diff] [blame] | 97 | : oneway_(oneway), |
| 98 | comments_(comments), |
| 99 | type_(type), |
| 100 | name_(name), |
| 101 | line_(line), |
| 102 | arguments_(std::move(*args)), |
| 103 | id_(id) { |
| 104 | has_id_ = true; |
| 105 | delete args; |
| Christopher Wiley | ad33927 | 2015-10-05 19:11:58 -0700 | [diff] [blame] | 106 | for (const unique_ptr<AidlArgument>& a : arguments_) { |
| 107 | if (a->IsIn()) { in_arguments_.push_back(a.get()); } |
| 108 | if (a->IsOut()) { out_arguments_.push_back(a.get()); } |
| 109 | } |
| Casey Dahlin | f4a9311 | 2015-10-05 16:58:09 -0700 | [diff] [blame] | 110 | } |
| 111 | |
| 112 | AidlMethod::AidlMethod(bool oneway, AidlType* type, std::string name, |
| 113 | std::vector<std::unique_ptr<AidlArgument>>* args, |
| Casey Dahlin | fb7da2e | 2015-10-08 17:26:09 -0700 | [diff] [blame] | 114 | unsigned line, const std::string& comments) |
| Casey Dahlin | f4a9311 | 2015-10-05 16:58:09 -0700 | [diff] [blame] | 115 | : AidlMethod(oneway, type, name, args, line, comments, 0) { |
| 116 | has_id_ = false; |
| 117 | } |
| Casey Dahlin | f2d23f7 | 2015-10-02 16:19:19 -0700 | [diff] [blame] | 118 | |
| Christopher Wiley | 4a2884b | 2015-10-07 11:27:45 -0700 | [diff] [blame] | 119 | Parser::Parser(const IoDelegate& io_delegate) |
| 120 | : io_delegate_(io_delegate) { |
| 121 | yylex_init(&scanner_); |
| 122 | } |
| 123 | |
| Casey Dahlin | 2b2879b | 2015-10-13 16:59:44 -0700 | [diff] [blame] | 124 | AidlParcelable::AidlParcelable(AidlQualifiedName* name, unsigned line, |
| Christopher Wiley | 8aa4d9f | 2015-11-16 19:10:45 -0800 | [diff] [blame] | 125 | const std::vector<std::string>& package, |
| 126 | const std::string& cpp_header) |
| Casey Dahlin | 59401da | 2015-10-09 18:16:45 -0700 | [diff] [blame] | 127 | : name_(name), |
| 128 | line_(line), |
| Christopher Wiley | 8aa4d9f | 2015-11-16 19:10:45 -0800 | [diff] [blame] | 129 | package_(package), |
| 130 | cpp_header_(cpp_header) { |
| 131 | // Strip off quotation marks if we actually have a cpp header. |
| 132 | if (cpp_header_.length() >= 2) { |
| 133 | cpp_header_ = cpp_header_.substr(1, cpp_header_.length() - 2); |
| 134 | } |
| Casey Dahlin | 59401da | 2015-10-09 18:16:45 -0700 | [diff] [blame] | 135 | } |
| 136 | |
| Christopher Wiley | d76067c | 2015-10-19 17:00:13 -0700 | [diff] [blame] | 137 | std::string AidlParcelable::GetPackage() const { |
| 138 | return Join(package_, '.'); |
| 139 | } |
| 140 | |
| Casey Dahlin | c1f39b4 | 2015-11-24 10:34:34 -0800 | [diff] [blame] | 141 | std::string AidlParcelable::GetCanonicalName() const { |
| 142 | if (package_.empty()) { |
| 143 | return GetName(); |
| 144 | } |
| 145 | return GetPackage() + "." + GetName(); |
| 146 | } |
| 147 | |
| Casey Dahlin | fb7da2e | 2015-10-08 17:26:09 -0700 | [diff] [blame] | 148 | AidlInterface::AidlInterface(const std::string& name, unsigned line, |
| 149 | const std::string& comments, bool oneway, |
| Casey Dahlin | d40e2fe | 2015-11-24 14:06:52 -0800 | [diff] [blame] | 150 | std::vector<std::unique_ptr<AidlMember>>* members, |
| Christopher Wiley | d76067c | 2015-10-19 17:00:13 -0700 | [diff] [blame] | 151 | const std::vector<std::string>& package) |
| Casey Dahlin | fb7da2e | 2015-10-08 17:26:09 -0700 | [diff] [blame] | 152 | : name_(name), |
| 153 | comments_(comments), |
| 154 | line_(line), |
| 155 | oneway_(oneway), |
| Casey Dahlin | fb7da2e | 2015-10-08 17:26:09 -0700 | [diff] [blame] | 156 | package_(package) { |
| Casey Dahlin | d40e2fe | 2015-11-24 14:06:52 -0800 | [diff] [blame] | 157 | for (auto& member : *members) { |
| 158 | AidlMember* local = member.release(); |
| 159 | AidlMethod* method = local->AsMethod(); |
| 160 | AidlConstant* constant = local->AsConstant(); |
| 161 | |
| 162 | if (method) { |
| 163 | methods_.emplace_back(method); |
| 164 | } else if (constant) { |
| 165 | constants_.emplace_back(constant); |
| 166 | } else { |
| 167 | LOG(FATAL) << "Member is neither method nor constant!"; |
| 168 | } |
| 169 | } |
| 170 | |
| 171 | delete members; |
| Casey Dahlin | fb7da2e | 2015-10-08 17:26:09 -0700 | [diff] [blame] | 172 | } |
| 173 | |
| Christopher Wiley | d76067c | 2015-10-19 17:00:13 -0700 | [diff] [blame] | 174 | std::string AidlInterface::GetPackage() const { |
| 175 | return Join(package_, '.'); |
| 176 | } |
| 177 | |
| 178 | std::string AidlInterface::GetCanonicalName() const { |
| 179 | if (package_.empty()) { |
| 180 | return GetName(); |
| 181 | } |
| 182 | return GetPackage() + "." + GetName(); |
| 183 | } |
| 184 | |
| Casey Dahlin | c1f39b4 | 2015-11-24 10:34:34 -0800 | [diff] [blame] | 185 | AidlDocument::AidlDocument(AidlInterface* interface) |
| 186 | : interface_(interface) {} |
| 187 | |
| Casey Dahlin | 2b2879b | 2015-10-13 16:59:44 -0700 | [diff] [blame] | 188 | AidlQualifiedName::AidlQualifiedName(std::string term, |
| 189 | std::string comments) |
| 190 | : terms_({term}), |
| 191 | comments_(comments) { |
| Christopher Wiley | 8aa4d9f | 2015-11-16 19:10:45 -0800 | [diff] [blame] | 192 | if (term.find('.') != string::npos) { |
| 193 | terms_ = Split(term, "."); |
| 194 | for (const auto& term: terms_) { |
| 195 | if (term.empty()) { |
| 196 | LOG(FATAL) << "Malformed qualified identifier: '" << term << "'"; |
| 197 | } |
| 198 | } |
| 199 | } |
| Casey Dahlin | 2b2879b | 2015-10-13 16:59:44 -0700 | [diff] [blame] | 200 | } |
| 201 | |
| 202 | void AidlQualifiedName::AddTerm(std::string term) { |
| 203 | terms_.push_back(term); |
| 204 | } |
| 205 | |
| Casey Dahlin | 0edf342 | 2015-10-07 12:34:59 -0700 | [diff] [blame] | 206 | AidlImport::AidlImport(const std::string& from, |
| 207 | const std::string& needed_class, unsigned line) |
| 208 | : from_(from), |
| 209 | needed_class_(needed_class), |
| 210 | line_(line) {} |
| 211 | |
| Christopher Wiley | 4a2884b | 2015-10-07 11:27:45 -0700 | [diff] [blame] | 212 | Parser::~Parser() { |
| 213 | if (raw_buffer_) { |
| 214 | yy_delete_buffer(buffer_, scanner_); |
| 215 | raw_buffer_.reset(); |
| 216 | } |
| 217 | yylex_destroy(scanner_); |
| 218 | } |
| 219 | |
| 220 | bool Parser::ParseFile(const string& filename) { |
| 221 | // Make sure we can read the file first, before trashing previous state. |
| 222 | unique_ptr<string> new_buffer = io_delegate_.GetFileContents(filename); |
| 223 | if (!new_buffer) { |
| 224 | LOG(ERROR) << "Error while opening file for parsing: '" << filename << "'"; |
| 225 | return false; |
| 226 | } |
| 227 | |
| 228 | // Throw away old parsing state if we have any. |
| 229 | if (raw_buffer_) { |
| 230 | yy_delete_buffer(buffer_, scanner_); |
| 231 | raw_buffer_.reset(); |
| 232 | } |
| 233 | |
| 234 | raw_buffer_ = std::move(new_buffer); |
| 235 | // We're going to scan this buffer in place, and yacc demands we put two |
| 236 | // nulls at the end. |
| 237 | raw_buffer_->append(2u, '\0'); |
| 238 | filename_ = filename; |
| Christopher Wiley | d76067c | 2015-10-19 17:00:13 -0700 | [diff] [blame] | 239 | package_.reset(); |
| Christopher Wiley | 4a2884b | 2015-10-07 11:27:45 -0700 | [diff] [blame] | 240 | error_ = 0; |
| Casey Dahlin | c1f39b4 | 2015-11-24 10:34:34 -0800 | [diff] [blame] | 241 | document_.reset(); |
| Christopher Wiley | 4a2884b | 2015-10-07 11:27:45 -0700 | [diff] [blame] | 242 | |
| 243 | buffer_ = yy_scan_buffer(&(*raw_buffer_)[0], raw_buffer_->length(), scanner_); |
| 244 | |
| Casey Dahlin | c1f39b4 | 2015-11-24 10:34:34 -0800 | [diff] [blame] | 245 | if (yy::parser(this).parse() != 0 || error_ != 0) { |
| 246 | return false;} |
| Christopher Wiley | 4a2884b | 2015-10-07 11:27:45 -0700 | [diff] [blame] | 247 | |
| Casey Dahlin | c1f39b4 | 2015-11-24 10:34:34 -0800 | [diff] [blame] | 248 | if (document_.get() != nullptr) |
| 249 | return true; |
| 250 | |
| 251 | LOG(ERROR) << "Parser succeeded but yielded no document!"; |
| 252 | return false; |
| Christopher Wiley | 4a2884b | 2015-10-07 11:27:45 -0700 | [diff] [blame] | 253 | } |
| 254 | |
| Casey Dahlin | 98a544b | 2015-10-14 14:22:55 -0700 | [diff] [blame] | 255 | void Parser::ReportError(const string& err, unsigned line) { |
| 256 | cerr << filename_ << ":" << line << ": " << err << endl; |
| Casey Dahlin | dd69181 | 2015-09-09 17:59:06 -0700 | [diff] [blame] | 257 | error_ = 1; |
| 258 | } |
| 259 | |
| Christopher Wiley | 90be4e3 | 2015-10-20 14:55:25 -0700 | [diff] [blame] | 260 | std::vector<std::string> Parser::Package() const { |
| 261 | if (!package_) { |
| 262 | return {}; |
| 263 | } |
| 264 | return package_->GetTerms(); |
| 265 | } |
| 266 | |
| Casey Dahlin | 2b2879b | 2015-10-13 16:59:44 -0700 | [diff] [blame] | 267 | void Parser::AddImport(AidlQualifiedName* name, unsigned line) { |
| 268 | imports_.emplace_back(new AidlImport(this->FileName(), |
| 269 | name->GetDotName(), line)); |
| 270 | delete name; |
| Casey Dahlin | e250749 | 2015-09-14 17:11:20 -0700 | [diff] [blame] | 271 | } |