1 /*
2 * Copyright (C) 2016 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 #include "FQName.h"
18
19 #include <android-base/logging.h>
20 #include <android-base/parseint.h>
21 #include <android-base/strings.h>
22 #include <iostream>
23 #include <sstream>
24
25 namespace android {
26
FQName()27 FQName::FQName() : mIsIdentifier(false) {}
28
parse(const std::string & s,FQName * into)29 bool FQName::parse(const std::string& s, FQName* into) {
30 return into->setTo(s);
31 }
32
FQName(const std::string & package,const std::string & version,const std::string & name,const std::string & valueName)33 FQName::FQName(const std::string& package, const std::string& version, const std::string& name,
34 const std::string& valueName) {
35 size_t majorVer, minorVer;
36 CHECK(parseVersion(version, &majorVer, &minorVer));
37 CHECK(setTo(package, majorVer, minorVer, name, valueName)) << string();
38 }
39
setTo(const std::string & package,size_t majorVer,size_t minorVer,const std::string & name,const std::string & valueName)40 bool FQName::setTo(const std::string& package, size_t majorVer, size_t minorVer,
41 const std::string& name, const std::string& valueName) {
42 mPackage = package;
43 mMajor = majorVer;
44 mMinor = minorVer;
45 mName = name;
46 mValueName = valueName;
47
48 FQName other;
49 if (!parse(string(), &other)) return false;
50 if ((*this) != other) return false;
51 mIsIdentifier = other.isIdentifier();
52 return true;
53 }
54
isIdentifier() const55 bool FQName::isIdentifier() const {
56 return mIsIdentifier;
57 }
58
isFullyQualified() const59 bool FQName::isFullyQualified() const {
60 return !mPackage.empty() && !version().empty() && !mName.empty();
61 }
62
isValidValueName() const63 bool FQName::isValidValueName() const {
64 return mIsIdentifier
65 || (!mName.empty() && !mValueName.empty());
66 }
67
isInterfaceName() const68 bool FQName::isInterfaceName() const {
69 return !mName.empty() && mName[0] == 'I' && mName.find('.') == std::string::npos;
70 }
71
isIdentStart(char a)72 static inline bool isIdentStart(char a) {
73 return ('a' <= a && a <= 'z') || ('A' <= a && a <= 'Z') || a == '_';
74 }
isLeadingDigit(char a)75 static inline bool isLeadingDigit(char a) {
76 return '1' <= a && a <= '9';
77 }
isDigit(char a)78 static inline bool isDigit(char a) {
79 return '0' <= a && a <= '9';
80 }
isIdentBody(char a)81 static inline bool isIdentBody(char a) {
82 return isIdentStart(a) || isDigit(a);
83 }
84
85 // returns pointer to end of [a-zA-Z_][a-zA-Z0-9_]*
eatIdent(const char * l,const char * end)86 static const char* eatIdent(const char* l, const char* end) {
87 if (!(l < end && isIdentStart(*l++))) return nullptr;
88 while (l < end && isIdentBody(*l)) l++;
89 return l;
90 }
91
92 // returns pointer to end of <ident>(\.<ident>)*
eatPackage(const char * l,const char * end)93 static const char* eatPackage(const char* l, const char* end) {
94 if ((l = eatIdent(l, end)) == nullptr) return nullptr;
95
96 while (l < end && *l == '.') {
97 l++;
98 if ((l = eatIdent(l, end)) == nullptr) return nullptr;
99 }
100 return l;
101 }
102
103 // returns pointer to end of [1-9][0-9]*|0
eatNumber(const char * l,const char * end)104 static const char* eatNumber(const char* l, const char* end) {
105 if (!(l < end)) return nullptr;
106 if (*l == '0') return l + 1;
107 if (!isLeadingDigit(*l++)) return nullptr;
108 while (l < end && isDigit(*l)) l++;
109 return l;
110 }
111
setTo(const std::string & s)112 bool FQName::setTo(const std::string &s) {
113 clear();
114
115 if (s.empty()) return false;
116
117 const char* l = s.c_str();
118 const char* end = l + s.size();
119 // android.hardware.foo@10.12::IFoo.Type:MY_ENUM_VALUE
120 // S ES ES E S ES E
121 //
122 // S - start pointer
123 // E - end pointer
124
125 struct StartEnd {
126 const char* start = nullptr;
127 const char* end = nullptr;
128
129 std::string string() {
130 if (start == nullptr) return std::string();
131 return std::string(start, end - start);
132 }
133 };
134 StartEnd package, major, minor, name, type;
135
136 if (l < end && isIdentStart(*l)) {
137 package.start = l;
138 if ((package.end = l = eatPackage(l, end)) == nullptr) return false;
139 }
140 if (l < end && *l == '@') {
141 l++;
142
143 major.start = l;
144 if ((major.end = l = eatNumber(l, end)) == nullptr) return false;
145
146 if (!(l < end && *l++ == '.')) return false;
147
148 minor.start = l;
149 if ((minor.end = l = eatNumber(l, end)) == nullptr) return false;
150 }
151 if (l < end && *l == ':') {
152 l++;
153 if (l < end && *l == ':') {
154 l++;
155 name.start = l;
156 if ((name.end = l = eatPackage(l, end)) == nullptr) return false;
157 if (l < end && *l++ == ':') {
158 type.start = l;
159 if ((type.end = l = eatIdent(l, end)) == nullptr) return false;
160 }
161 } else {
162 type.start = l;
163 if ((type.end = l = eatIdent(l, end)) == nullptr) return false;
164 }
165 }
166
167 if (l < end) return false;
168
169 CHECK((major.start == nullptr) == (minor.start == nullptr));
170
171 // if we only parse a package, consider this to be a name
172 if (name.start == nullptr && major.start == nullptr) {
173 name.start = package.start;
174 name.end = package.end;
175 package.start = package.end = nullptr;
176 }
177
178 // failures after this goto fail to clear
179 mName = name.string();
180 mPackage = package.string();
181 mValueName = type.string();
182
183 if (major.start != nullptr) {
184 if (!parseVersion(major.string(), minor.string(), &mMajor, &mMinor)) goto fail;
185 } else if (mPackage.empty() && mValueName.empty() &&
186 name.end == eatIdent(name.start, name.end)) {
187 // major.start == nullptr
188 mIsIdentifier = true;
189 }
190
191 if (!mValueName.empty() && mName.empty()) goto fail;
192 if (!mPackage.empty() && version().empty()) goto fail;
193
194 return true;
195 fail:
196 clear();
197 return false;
198 }
199
getRelativeFQName(const FQName & relativeTo) const200 std::string FQName::getRelativeFQName(const FQName& relativeTo) const {
201 if (relativeTo.mPackage != mPackage) {
202 return string();
203 }
204
205 // Package is the same
206 std::string out;
207 if (relativeTo.version() != version()) {
208 out.append(atVersion());
209 if (!mName.empty() && !version().empty()) {
210 out.append("::");
211 }
212 }
213
214 if (!mName.empty()) {
215 out.append(mName);
216 if (!mValueName.empty()) {
217 out.append(":");
218 out.append(mValueName);
219 }
220 }
221
222 return out;
223 }
224
package() const225 const std::string& FQName::package() const {
226 return mPackage;
227 }
228
version() const229 std::string FQName::version() const {
230 if (!hasVersion()) {
231 return "";
232 }
233 return std::to_string(mMajor) + "." + std::to_string(mMinor);
234 }
235
sanitizedVersion() const236 std::string FQName::sanitizedVersion() const {
237 if (!hasVersion()) {
238 return "";
239 }
240 return "V" + std::to_string(mMajor) + "_" + std::to_string(mMinor);
241 }
242
atVersion() const243 std::string FQName::atVersion() const {
244 std::string v = version();
245 return v.empty() ? "" : ("@" + v);
246 }
247
clear()248 void FQName::clear() {
249 mIsIdentifier = false;
250 mPackage.clear();
251 clearVersion();
252 mName.clear();
253 mValueName.clear();
254 }
255
clearVersion(size_t * majorVer,size_t * minorVer)256 void FQName::clearVersion(size_t* majorVer, size_t* minorVer) {
257 *majorVer = *minorVer = 0;
258 }
259
parseVersion(const std::string & majorStr,const std::string & minorStr,size_t * majorVer,size_t * minorVer)260 bool FQName::parseVersion(const std::string& majorStr, const std::string& minorStr,
261 size_t* majorVer, size_t* minorVer) {
262 bool versionParseSuccess = ::android::base::ParseUint(majorStr, majorVer) &&
263 ::android::base::ParseUint(minorStr, minorVer);
264 if (!versionParseSuccess) {
265 LOG(ERROR) << "numbers in " << majorStr << "." << minorStr << " are out of range.";
266 }
267 return versionParseSuccess;
268 }
269
parseVersion(const std::string & v,size_t * majorVer,size_t * minorVer)270 bool FQName::parseVersion(const std::string& v, size_t* majorVer, size_t* minorVer) {
271 if (v.empty()) {
272 clearVersion(majorVer, minorVer);
273 return true;
274 }
275
276 std::vector<std::string> vs = base::Split(v, ".");
277 if (vs.size() != 2) return false;
278 return parseVersion(vs[0], vs[1], majorVer, minorVer);
279 }
280
setVersion(const std::string & v)281 bool FQName::setVersion(const std::string& v) {
282 return parseVersion(v, &mMajor, &mMinor);
283 }
284
clearVersion()285 void FQName::clearVersion() {
286 clearVersion(&mMajor, &mMinor);
287 }
288
parseVersion(const std::string & majorStr,const std::string & minorStr)289 bool FQName::parseVersion(const std::string& majorStr, const std::string& minorStr) {
290 return parseVersion(majorStr, minorStr, &mMajor, &mMinor);
291 }
292
name() const293 const std::string& FQName::name() const {
294 return mName;
295 }
296
names() const297 std::vector<std::string> FQName::names() const {
298 std::vector<std::string> res {};
299 std::istringstream ss(name());
300 std::string s;
301 while (std::getline(ss, s, '.')) {
302 res.push_back(s);
303 }
304 return res;
305 }
306
valueName() const307 const std::string& FQName::valueName() const {
308 return mValueName;
309 }
310
typeName() const311 FQName FQName::typeName() const {
312 return FQName(mPackage, version(), mName);
313 }
314
applyDefaults(const std::string & defaultPackage,const std::string & defaultVersion)315 void FQName::applyDefaults(
316 const std::string &defaultPackage,
317 const std::string &defaultVersion) {
318
319 // package without version is not allowed.
320 CHECK(mPackage.empty() || !version().empty());
321
322 if (mPackage.empty()) {
323 mPackage = defaultPackage;
324 }
325
326 if (version().empty()) {
327 CHECK(setVersion(defaultVersion));
328 }
329 }
330
string() const331 std::string FQName::string() const {
332 std::string out;
333 out.append(mPackage);
334 out.append(atVersion());
335 if (!mName.empty()) {
336 if (!mPackage.empty() || !version().empty()) {
337 out.append("::");
338 }
339 out.append(mName);
340
341 if (!mValueName.empty()) {
342 out.append(":");
343 out.append(mValueName);
344 }
345 }
346
347 return out;
348 }
349
operator <(const FQName & other) const350 bool FQName::operator<(const FQName &other) const {
351 return string() < other.string();
352 }
353
operator ==(const FQName & other) const354 bool FQName::operator==(const FQName &other) const {
355 return string() == other.string();
356 }
357
operator !=(const FQName & other) const358 bool FQName::operator!=(const FQName &other) const {
359 return !(*this == other);
360 }
361
getInterfaceName() const362 const std::string& FQName::getInterfaceName() const {
363 CHECK(isInterfaceName()) << mName;
364
365 return mName;
366 }
367
getInterfaceBaseName() const368 std::string FQName::getInterfaceBaseName() const {
369 // cut off the leading 'I'.
370 return getInterfaceName().substr(1);
371 }
372
getInterfaceHwName() const373 std::string FQName::getInterfaceHwName() const {
374 return "IHw" + getInterfaceBaseName();
375 }
376
getInterfaceProxyName() const377 std::string FQName::getInterfaceProxyName() const {
378 return "BpHw" + getInterfaceBaseName();
379 }
380
getInterfaceStubName() const381 std::string FQName::getInterfaceStubName() const {
382 return "BnHw" + getInterfaceBaseName();
383 }
384
getInterfacePassthroughName() const385 std::string FQName::getInterfacePassthroughName() const {
386 return "Bs" + getInterfaceBaseName();
387 }
388
getInterfaceProxyFqName() const389 FQName FQName::getInterfaceProxyFqName() const {
390 return FQName(package(), version(), getInterfaceProxyName());
391 }
392
getInterfaceStubFqName() const393 FQName FQName::getInterfaceStubFqName() const {
394 return FQName(package(), version(), getInterfaceStubName());
395 }
396
getInterfacePassthroughFqName() const397 FQName FQName::getInterfacePassthroughFqName() const {
398 return FQName(package(), version(), getInterfacePassthroughName());
399 }
400
getTypesForPackage() const401 FQName FQName::getTypesForPackage() const {
402 return FQName(package(), version(), "types");
403 }
404
getPackageAndVersion() const405 FQName FQName::getPackageAndVersion() const {
406 return FQName(package(), version(), "");
407 }
408
getTopLevelType() const409 FQName FQName::getTopLevelType() const {
410 auto idx = mName.find('.');
411
412 if (idx == std::string::npos) {
413 return *this;
414 }
415
416 return FQName(mPackage, version(), mName.substr(0, idx));
417 }
418
tokenName() const419 std::string FQName::tokenName() const {
420 std::vector<std::string> components = getPackageAndVersionComponents(true /* sanitized */);
421
422 if (!mName.empty()) {
423 std::vector<std::string> nameComponents = base::Split(mName, ".");
424
425 components.insert(components.end(), nameComponents.begin(), nameComponents.end());
426 }
427
428 return base::Join(components, "_");
429 }
430
cppNamespace() const431 std::string FQName::cppNamespace() const {
432 std::vector<std::string> components = getPackageAndVersionComponents(true /* sanitized */);
433 return "::" + base::Join(components, "::");
434 }
435
cppLocalName() const436 std::string FQName::cppLocalName() const {
437 std::vector<std::string> components = base::Split(mName, ".");
438
439 return base::Join(components, "::")
440 + (mValueName.empty() ? "" : ("::" + mValueName));
441 }
442
cppName() const443 std::string FQName::cppName() const {
444 std::string out = cppNamespace();
445
446 std::vector<std::string> components = base::Split(name(), ".");
447 out += "::";
448 out += base::Join(components, "::");
449 if (!mValueName.empty()) {
450 out += "::" + mValueName;
451 }
452
453 return out;
454 }
455
javaPackage() const456 std::string FQName::javaPackage() const {
457 std::vector<std::string> components = getPackageAndVersionComponents(true /* sanitized */);
458 return base::Join(components, ".");
459 }
460
javaName() const461 std::string FQName::javaName() const {
462 return javaPackage() + "." + name()
463 + (mValueName.empty() ? "" : ("." + mValueName));
464 }
465
getPackageComponents() const466 std::vector<std::string> FQName::getPackageComponents() const {
467 return base::Split(package(), ".");
468 }
469
getPackageAndVersionComponents(bool sanitized) const470 std::vector<std::string> FQName::getPackageAndVersionComponents(bool sanitized) const {
471 CHECK(hasVersion()) << string() << ": getPackageAndVersionComponents expects version.";
472
473 std::vector<std::string> components = getPackageComponents();
474 if (sanitized) {
475 components.push_back(sanitizedVersion());
476 } else {
477 components.push_back(version());
478 }
479 return components;
480 }
481
hasVersion() const482 bool FQName::hasVersion() const {
483 return mMajor > 0;
484 }
485
getVersion() const486 std::pair<size_t, size_t> FQName::getVersion() const {
487 return {mMajor, mMinor};
488 }
489
withVersion(size_t major,size_t minor) const490 FQName FQName::withVersion(size_t major, size_t minor) const {
491 FQName ret(*this);
492 ret.mMajor = major;
493 ret.mMinor = minor;
494 return ret;
495 }
496
getPackageMajorVersion() const497 size_t FQName::getPackageMajorVersion() const {
498 CHECK(hasVersion()) << "FQName: No version exists at getPackageMajorVersion(). "
499 << "Did you check hasVersion()?";
500 return mMajor;
501 }
502
getPackageMinorVersion() const503 size_t FQName::getPackageMinorVersion() const {
504 CHECK(hasVersion()) << "FQName: No version exists at getPackageMinorVersion(). "
505 << "Did you check hasVersion()?";
506 return mMinor;
507 }
508
endsWith(const FQName & other) const509 bool FQName::endsWith(const FQName &other) const {
510 std::string s1 = string();
511 std::string s2 = other.string();
512
513 size_t pos = s1.rfind(s2);
514 if (pos == std::string::npos || pos + s2.size() != s1.size()) {
515 return false;
516 }
517
518 // A match is only a match if it is preceded by a "boundary", i.e.
519 // we perform a component-wise match from the end.
520 // "az" is not a match for "android.hardware.foo@1.0::IFoo.bar.baz",
521 // "baz", "bar.baz", "IFoo.bar.baz", "@1.0::IFoo.bar.baz" are.
522 if (pos == 0) {
523 // matches "android.hardware.foo@1.0::IFoo.bar.baz"
524 return true;
525 }
526
527 if (s1[pos - 1] == '.') {
528 // matches "baz" and "bar.baz"
529 return true;
530 }
531
532 if (s1[pos - 1] == ':') {
533 // matches "IFoo.bar.baz"
534 return true;
535 }
536
537 if (s1[pos] == '@') {
538 // matches "@1.0::IFoo.bar.baz"
539 return true;
540 }
541
542 return false;
543 }
544
inPackage(const std::string & package) const545 bool FQName::inPackage(const std::string &package) const {
546 std::vector<std::string> components = getPackageComponents();
547 std::vector<std::string> inComponents = base::Split(package, ".");
548
549 if (inComponents.size() > components.size()) {
550 return false;
551 }
552
553 for (size_t i = 0; i < inComponents.size(); i++) {
554 if (inComponents[i] != components[i]) {
555 return false;
556 }
557 }
558
559 return true;
560 }
561
downRev() const562 FQName FQName::downRev() const {
563 FQName ret(*this);
564 CHECK(ret.mMinor > 0);
565 ret.mMinor--;
566 return ret;
567 }
568
upRev() const569 FQName FQName::upRev() const {
570 FQName ret(*this);
571 ret.mMinor++;
572 CHECK(ret.mMinor > 0);
573 return ret;
574 }
575
576 } // namespace android
577
578