xref: /aosp_15_r20/system/libvintf/utils.cpp (revision 70a7ec852fcefd15a4fb57f8f183a8b1c3aacb08)
1*70a7ec85SAndroid Build Coastguard Worker /*
2*70a7ec85SAndroid Build Coastguard Worker  * Copyright (C) 2021 The Android Open Source Project
3*70a7ec85SAndroid Build Coastguard Worker  *
4*70a7ec85SAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*70a7ec85SAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*70a7ec85SAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*70a7ec85SAndroid Build Coastguard Worker  *
8*70a7ec85SAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*70a7ec85SAndroid Build Coastguard Worker  *
10*70a7ec85SAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*70a7ec85SAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*70a7ec85SAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*70a7ec85SAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*70a7ec85SAndroid Build Coastguard Worker  * limitations under the License.
15*70a7ec85SAndroid Build Coastguard Worker  */
16*70a7ec85SAndroid Build Coastguard Worker #include "utils.h"
17*70a7ec85SAndroid Build Coastguard Worker 
18*70a7ec85SAndroid Build Coastguard Worker #include <sstream>
19*70a7ec85SAndroid Build Coastguard Worker 
20*70a7ec85SAndroid Build Coastguard Worker #include "parse_string.h"
21*70a7ec85SAndroid Build Coastguard Worker 
22*70a7ec85SAndroid Build Coastguard Worker namespace android::vintf::details {
23*70a7ec85SAndroid Build Coastguard Worker 
convertLegacyInstanceIntoFqInstance(const std::string & package,const Version & version,const std::string & interface,const std::string & instance,HalFormat format,std::string * appendedError)24*70a7ec85SAndroid Build Coastguard Worker std::optional<FqInstance> convertLegacyInstanceIntoFqInstance(
25*70a7ec85SAndroid Build Coastguard Worker     const std::string& package, const Version& version, const std::string& interface,
26*70a7ec85SAndroid Build Coastguard Worker     const std::string& instance, HalFormat format, std::string* appendedError) {
27*70a7ec85SAndroid Build Coastguard Worker     // Attempt to construct a good error message.
28*70a7ec85SAndroid Build Coastguard Worker     std::stringstream ss;
29*70a7ec85SAndroid Build Coastguard Worker     ss << "Invalid instance: '";
30*70a7ec85SAndroid Build Coastguard Worker     if (format == HalFormat::AIDL) {
31*70a7ec85SAndroid Build Coastguard Worker         ss << toAidlFqnameString(package, interface, instance) << " (@" << version.minorVer << ")";
32*70a7ec85SAndroid Build Coastguard Worker     } else {
33*70a7ec85SAndroid Build Coastguard Worker         ss << toFQNameString(package, version, interface, instance);
34*70a7ec85SAndroid Build Coastguard Worker     }
35*70a7ec85SAndroid Build Coastguard Worker     ss << "'. ";
36*70a7ec85SAndroid Build Coastguard Worker 
37*70a7ec85SAndroid Build Coastguard Worker     // Attempt to guess the source of error.
38*70a7ec85SAndroid Build Coastguard Worker     bool foundError = false;
39*70a7ec85SAndroid Build Coastguard Worker     details::FQName fqName;
40*70a7ec85SAndroid Build Coastguard Worker     if (!fqName.setTo(package)) {
41*70a7ec85SAndroid Build Coastguard Worker         ss << "Package '" << package
42*70a7ec85SAndroid Build Coastguard Worker            << "' should have the format [a-zA-Z_][a-zA-Z0-9_]*(\\.[a-zA-Z_][a-zA-Z0-9_]*)*";
43*70a7ec85SAndroid Build Coastguard Worker         foundError = true;
44*70a7ec85SAndroid Build Coastguard Worker     }
45*70a7ec85SAndroid Build Coastguard Worker 
46*70a7ec85SAndroid Build Coastguard Worker     switch (format) {
47*70a7ec85SAndroid Build Coastguard Worker         case HalFormat::HIDL:
48*70a7ec85SAndroid Build Coastguard Worker         case HalFormat::AIDL: {
49*70a7ec85SAndroid Build Coastguard Worker             if (!fqName.setTo(interface) || !fqName.isInterfaceName()) {
50*70a7ec85SAndroid Build Coastguard Worker                 ss << "Interface '" << interface << "' should have the format I[a-zA-Z0-9_]*";
51*70a7ec85SAndroid Build Coastguard Worker                 foundError = true;
52*70a7ec85SAndroid Build Coastguard Worker             }
53*70a7ec85SAndroid Build Coastguard Worker         } break;
54*70a7ec85SAndroid Build Coastguard Worker         case HalFormat::NATIVE: {
55*70a7ec85SAndroid Build Coastguard Worker             if (!interface.empty() && (!fqName.setTo(interface) || !fqName.isInterfaceName())) {
56*70a7ec85SAndroid Build Coastguard Worker                 ss << "Interface '" << interface << "' should have the format I[a-zA-Z0-9_]*";
57*70a7ec85SAndroid Build Coastguard Worker                 foundError = true;
58*70a7ec85SAndroid Build Coastguard Worker             }
59*70a7ec85SAndroid Build Coastguard Worker         } break;
60*70a7ec85SAndroid Build Coastguard Worker     }
61*70a7ec85SAndroid Build Coastguard Worker 
62*70a7ec85SAndroid Build Coastguard Worker     if (foundError) {
63*70a7ec85SAndroid Build Coastguard Worker         if (appendedError != nullptr) {
64*70a7ec85SAndroid Build Coastguard Worker             *appendedError += ss.str() + "\n";
65*70a7ec85SAndroid Build Coastguard Worker         }
66*70a7ec85SAndroid Build Coastguard Worker         return std::nullopt;
67*70a7ec85SAndroid Build Coastguard Worker     }
68*70a7ec85SAndroid Build Coastguard Worker 
69*70a7ec85SAndroid Build Coastguard Worker     // AIDL HAL <fqname> never contains version
70*70a7ec85SAndroid Build Coastguard Worker     std::optional<FqInstance> parsed;
71*70a7ec85SAndroid Build Coastguard Worker     switch (format) {
72*70a7ec85SAndroid Build Coastguard Worker         case HalFormat::HIDL:
73*70a7ec85SAndroid Build Coastguard Worker         case HalFormat::NATIVE:
74*70a7ec85SAndroid Build Coastguard Worker             parsed = FqInstance::from(version.majorVer, version.minorVer, interface, instance);
75*70a7ec85SAndroid Build Coastguard Worker             break;
76*70a7ec85SAndroid Build Coastguard Worker         case HalFormat::AIDL:
77*70a7ec85SAndroid Build Coastguard Worker             // AIDL HAL <fqname> never contains version
78*70a7ec85SAndroid Build Coastguard Worker             parsed = FqInstance::from(interface, instance);
79*70a7ec85SAndroid Build Coastguard Worker             break;
80*70a7ec85SAndroid Build Coastguard Worker     }
81*70a7ec85SAndroid Build Coastguard Worker     if (!parsed.has_value()) {
82*70a7ec85SAndroid Build Coastguard Worker         if (appendedError != nullptr) {
83*70a7ec85SAndroid Build Coastguard Worker             std::string debugString = format == HalFormat::AIDL
84*70a7ec85SAndroid Build Coastguard Worker                                           ? toAidlFqnameString(package, interface, instance)
85*70a7ec85SAndroid Build Coastguard Worker                                           : toFQNameString(package, version, interface, instance);
86*70a7ec85SAndroid Build Coastguard Worker 
87*70a7ec85SAndroid Build Coastguard Worker             *appendedError += "Invalid FqInstance: " + debugString + "\n";
88*70a7ec85SAndroid Build Coastguard Worker         }
89*70a7ec85SAndroid Build Coastguard Worker     }
90*70a7ec85SAndroid Build Coastguard Worker 
91*70a7ec85SAndroid Build Coastguard Worker     return parsed;
92*70a7ec85SAndroid Build Coastguard Worker }
93*70a7ec85SAndroid Build Coastguard Worker 
isCoreHal(const std::string & halName)94*70a7ec85SAndroid Build Coastguard Worker bool isCoreHal(const std::string& halName) {
95*70a7ec85SAndroid Build Coastguard Worker     std::vector<std::string_view> allowedPrefixes{
96*70a7ec85SAndroid Build Coastguard Worker         "android.hardware.",
97*70a7ec85SAndroid Build Coastguard Worker         "android.frameworks.",
98*70a7ec85SAndroid Build Coastguard Worker         "android.system.",
99*70a7ec85SAndroid Build Coastguard Worker     };
100*70a7ec85SAndroid Build Coastguard Worker     for (auto allowedPrefix : allowedPrefixes) {
101*70a7ec85SAndroid Build Coastguard Worker         if (halName.substr(0, allowedPrefix.size()) == allowedPrefix) {
102*70a7ec85SAndroid Build Coastguard Worker             return true;
103*70a7ec85SAndroid Build Coastguard Worker         }
104*70a7ec85SAndroid Build Coastguard Worker     }
105*70a7ec85SAndroid Build Coastguard Worker     if (halName == "mapper") {
106*70a7ec85SAndroid Build Coastguard Worker         return true;
107*70a7ec85SAndroid Build Coastguard Worker     }
108*70a7ec85SAndroid Build Coastguard Worker     return false;
109*70a7ec85SAndroid Build Coastguard Worker }
110*70a7ec85SAndroid Build Coastguard Worker 
111*70a7ec85SAndroid Build Coastguard Worker }  // namespace android::vintf::details
112