| /* |
| * Copyright (C) 2019 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 <aidl/metadata.h> |
| #include <android-base/logging.h> |
| #include <android-base/properties.h> |
| #include <android-base/strings.h> |
| #include <android/content/pm/IPackageManagerNative.h> |
| #include <binder/IServiceManager.h> |
| #include <gtest/gtest.h> |
| #include <hidl-util/FQName.h> |
| #include <hidl/metadata.h> |
| #include <vintf/VintfObject.h> |
| |
| #include <algorithm> |
| #include <cstddef> |
| #include <map> |
| #include <mutex> |
| #include <set> |
| #include <string> |
| #include <vector> |
| |
| #ifdef AIDL_USE_UNFROZEN |
| constexpr bool kAidlUseUnfrozen = true; |
| #else |
| constexpr bool kAidlUseUnfrozen = false; |
| #endif |
| |
| using namespace android; |
| |
| // clang-format off |
| static const std::set<std::string> kKnownMissingHidl = { |
| "[email protected]", // converted to AIDL, see b/170401743 |
| "[email protected]", |
| "[email protected]", // converted to AIDL, see b/205764761 |
| "[email protected]", // deprecated, see b/141930622 |
| "[email protected]", // deprecated, see b/37226359 |
| "[email protected]", // deprecated, see b/205764765 |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", // converted to AIDL, see b/177667419 |
| "[email protected]", // deprecated, see b/204935495 |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", // converted to AIDL, see b/264712385 |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", // converted to AIDL, see b/264712385 |
| "[email protected]", // converted to AIDL, see b/182976659 |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", // converted to AIDL, see b/170405615 |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", // converted to AIDL, see b/152416783 |
| "[email protected]", // converted to AIDL, see b/168730443 |
| "[email protected]", |
| "[email protected]", // converted to AIDL, see b/203490261 |
| "[email protected]", // converted to AIDL, see b/205758693 |
| "[email protected]", // converted to AIDL, see b/227536004 |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", // Camera converted to AIDL, b/196432585 |
| "[email protected]", // converted to AIDL, see b/227673974 |
| "[email protected]", |
| "[email protected]", // deprecated, see b/149050985, b/149050733 |
| "[email protected]", // converted to AIDL, see b/205760172 |
| "[email protected]", |
| "[email protected]", // converted to AIDL, b/200055138 |
| "[email protected]", |
| "[email protected]", // deprecated, see b/205760700 |
| "[email protected]", // converted to AIDL, b/205760843 |
| "[email protected]", // GNSS converted to AIDL, b/206670536 |
| "[email protected]", // GNSS converted to AIDL, b/206670536 |
| "[email protected]", // is sub-interface of gnss |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", // converted to AIDL, see b/205761012 |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", // converted to AIDL, see b/193240715 |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", // converted to Stable C, see b/205761028 |
| "[email protected]", // converted to AIDL, see b/177470478 |
| "[email protected]", // converted to AIDL, see b/177269435 |
| "[email protected]", // converted to AIDL, see b/205761620 |
| "[email protected]", // converted to AIDL, see b/205000342 |
| "[email protected]", |
| "[email protected]", // Replaced by AIDL KeyMint, see b/111446262 |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", // deprecated b/205761766 |
| "[email protected]", |
| "[email protected]", // converted to AIDL, see b/161428342 |
| "[email protected]", |
| "[email protected]", // converted to AIDL, see b/176107318 b/282160400 |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", // converted to AIDL |
| "[email protected]", // converted to AIDL |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", // see b/170699770 |
| "[email protected]", |
| "[email protected]", // Converted to AIDL (see b/205762943) |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", // migrated to AIDL see b/200993386 |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", // Converted to AIDL (see b/205044134) |
| "[email protected]", // Converted to AIDL (see b/194806512) |
| "[email protected]", // Converted to AIDL (see b/196235436) |
| "[email protected]", |
| "[email protected]", |
| "[email protected]", // Deprecated (see b/205764958) |
| "[email protected]", // Converted to AIDL (see b/205764585) |
| "[email protected]", // Converted to AIDL (see b/170260236) |
| "[email protected]", // Converted to AIDL (see b/205764502) |
| }; |
| // clang-format on |
| |
| struct VersionedAidlPackage { |
| std::string name; |
| size_t version; |
| int bugNum; |
| bool operator<(const VersionedAidlPackage& rhs) const { |
| return (name < rhs.name || (name == rhs.name && version < rhs.version)); |
| } |
| }; |
| |
| static const std::set<std::string> kPhoneOnlyAidl = { |
| "android.hardware.camera.provider", |
| }; |
| |
| static const std::set<std::string> kAutomotiveOnlyAidl = { |
| /** |
| * These types are only used in Android Automotive, so don't expect them |
| * on phones. |
| */ |
| "android.automotive.watchdog", |
| "android.frameworks.automotive.display", |
| "android.frameworks.automotive.powerpolicy", |
| "android.frameworks.automotive.powerpolicy.internal", |
| "android.frameworks.automotive.telemetry", |
| "android.hardware.automotive.audiocontrol", |
| "android.hardware.automotive.can", |
| "android.hardware.broadcastradio", |
| "android.hardware.automotive.occupant_awareness", |
| "android.hardware.automotive.remoteaccess", |
| "android.hardware.automotive.vehicle", |
| "android.hardware.automotive.ivn", |
| "android.hardware.macsec", |
| }; |
| |
| static const std::set<std::string> kTvOnlyAidl = { |
| /** |
| * These types are only used in Android TV, so don't expect them on other |
| * devices. |
| * TODO(b/266868403) This test should run on TV devices to enforce the same |
| * requirements |
| */ |
| "android.hardware.tv.hdmi.cec", "android.hardware.tv.hdmi.earc", |
| "android.hardware.tv.hdmi.connection", "android.hardware.tv.tuner", |
| "android.hardware.tv.input", |
| }; |
| |
| static const std::set<std::string> kRadioOnlyAidl = { |
| // Not all devices have radio capabilities |
| "android.hardware.radio.config", "android.hardware.radio.data", |
| "android.hardware.radio.messaging", "android.hardware.radio.modem", |
| "android.hardware.radio.network", "android.hardware.radio.sap", |
| "android.hardware.radio.sim", "android.hardware.radio.voice", |
| "android.hardware.radio.ims", "android.hardware.radio.ims.media", |
| "android.hardware.radio.satellite", |
| |
| }; |
| |
| /* |
| * Always missing AIDL packages that are not served on Cuttlefish. |
| * These are typically types-only packages. |
| */ |
| static const std::set<std::string> kAlwaysMissingAidl = { |
| // types-only packages, which never expect a default implementation |
| "android.frameworks.cameraservice.common", |
| "android.frameworks.cameraservice.device", |
| "android.hardware.audio.common", |
| "android.hardware.audio.core.sounddose", |
| "android.hardware.biometrics.common", |
| "android.hardware.camera.common", |
| "android.hardware.camera.device", |
| "android.hardware.camera.metadata", |
| "android.hardware.common", |
| "android.hardware.common.fmq", |
| "android.hardware.graphics.common", |
| "android.hardware.input.common", |
| "android.media.audio.common.types", |
| "android.hardware.radio", |
| "android.hardware.uwb.fira_android", |
| "android.hardware.wifi.common", |
| "android.hardware.keymaster", |
| "android.hardware.automotive.vehicle.property", |
| // not on Cuttlefish since it's needed only on systems using HIDL audio HAL |
| "android.hardware.audio.sounddose", |
| |
| // android.hardware.media.bufferpool2 is a HAL-less interface. |
| // It could be used for buffer recycling and caching by using the interface. |
| "android.hardware.media.bufferpool2", |
| |
| /** |
| * No implementation on cuttlefish for fastboot AIDL hal because it doesn't |
| * run during normal boot, only in recovery/fastboot mode. |
| */ |
| "android.hardware.fastboot", |
| /** |
| * No implementation for usb gadget HAL because cuttlefish doesn't |
| * support usb gadget configfs, and currently there is no |
| * plan to add this support. |
| * Context: (b/130076572, g/android-idl-discuss/c/0SaiY0p-vJw/) |
| */ |
| "android.hardware.usb.gadget", |
| }; |
| |
| /* |
| * These packages should have implementations but currently do not. |
| * These must be accompanied by a bug and expected to be here temporarily. |
| */ |
| static const std::vector<VersionedAidlPackage> kKnownMissingAidl = { |
| // Cuttlefish Identity Credential HAL implementation is currently |
| // stuck at version 3 while RKP support is being added. Will be |
| // updated soon. |
| {"android.hardware.identity.", 4, 266869317}, |
| {"android.hardware.identity.", 5, 266869317}, |
| |
| {"android.se.omapi.", 1, 266870904}, |
| {"android.hardware.soundtrigger3.", 2, 266941225}, |
| {"android.media.soundtrigger.", 2, 266941225}, |
| {"android.hardware.weaver.", 2, 262418065}, |
| |
| {"android.automotive.computepipe.registry.", 2, 273549907}, |
| {"android.automotive.computepipe.runner.", 2, 273549907}, |
| {"android.hardware.automotive.evs.", 2, 274162534}, |
| }; |
| |
| // android.hardware.foo.IFoo -> android.hardware.foo. |
| std::string getAidlPackage(const std::string& aidlType) { |
| size_t lastDot = aidlType.rfind('.'); |
| CHECK(lastDot != std::string::npos); |
| return aidlType.substr(0, lastDot + 1); |
| } |
| |
| static bool isAospAidlInterface(const std::string& name) { |
| return base::StartsWith(name, "android.") && |
| !base::StartsWith(name, "android.hardware.tests.") && |
| !base::StartsWith(name, "android.aidl.tests"); |
| } |
| |
| enum class DeviceType { |
| UNKNOWN, |
| AUTOMOTIVE, |
| TV, |
| WATCH, |
| PHONE, |
| }; |
| |
| static DeviceType getDeviceType() { |
| static DeviceType type = DeviceType::UNKNOWN; |
| if (type != DeviceType::UNKNOWN) return type; |
| |
| sp<IBinder> binder = |
| defaultServiceManager()->waitForService(String16("package_native")); |
| sp<content::pm::IPackageManagerNative> packageManager = |
| interface_cast<content::pm::IPackageManagerNative>(binder); |
| CHECK(packageManager != nullptr); |
| |
| bool hasFeature = false; |
| // PackageManager.FEATURE_AUTOMOTIVE |
| CHECK(packageManager |
| ->hasSystemFeature(String16("android.hardware.type.automotive"), 0, |
| &hasFeature) |
| .isOk()); |
| if (hasFeature) return DeviceType::AUTOMOTIVE; |
| |
| // PackageManager.FEATURE_LEANBACK |
| CHECK(packageManager |
| ->hasSystemFeature(String16("android.software.leanback"), 0, |
| &hasFeature) |
| .isOk()); |
| if (hasFeature) return DeviceType::TV; |
| |
| // PackageManager.FEATURE_WATCH |
| CHECK(packageManager |
| ->hasSystemFeature(String16("android.hardware.type.watch"), 0, |
| &hasFeature) |
| .isOk()); |
| if (hasFeature) return DeviceType::WATCH; |
| |
| return DeviceType::PHONE; |
| } |
| |
| static bool isMissingAidl(const std::string& packageName) { |
| static std::once_flag unionFlag; |
| static std::set<std::string> missingAidl = kAlwaysMissingAidl; |
| |
| std::call_once(unionFlag, [&]() { |
| const DeviceType type = getDeviceType(); |
| switch (type) { |
| case DeviceType::AUTOMOTIVE: |
| missingAidl.insert(kPhoneOnlyAidl.begin(), kPhoneOnlyAidl.end()); |
| missingAidl.insert(kTvOnlyAidl.begin(), kTvOnlyAidl.end()); |
| break; |
| case DeviceType::TV: |
| missingAidl.insert(kAutomotiveOnlyAidl.begin(), |
| kAutomotiveOnlyAidl.end()); |
| missingAidl.insert(kRadioOnlyAidl.begin(), kRadioOnlyAidl.end()); |
| break; |
| case DeviceType::WATCH: |
| missingAidl.insert(kAutomotiveOnlyAidl.begin(), |
| kAutomotiveOnlyAidl.end()); |
| missingAidl.insert(kPhoneOnlyAidl.begin(), kPhoneOnlyAidl.end()); |
| missingAidl.insert(kTvOnlyAidl.begin(), kTvOnlyAidl.end()); |
| break; |
| case DeviceType::PHONE: |
| missingAidl.insert(kAutomotiveOnlyAidl.begin(), |
| kAutomotiveOnlyAidl.end()); |
| missingAidl.insert(kTvOnlyAidl.begin(), kTvOnlyAidl.end()); |
| break; |
| case DeviceType::UNKNOWN: |
| CHECK(false) << "getDeviceType return UNKNOWN type."; |
| break; |
| } |
| }); |
| |
| return missingAidl.find(packageName) != missingAidl.end(); |
| } |
| |
| static std::vector<VersionedAidlPackage> allAidlManifestInterfaces() { |
| std::vector<VersionedAidlPackage> ret; |
| auto setInserter = [&](const vintf::ManifestInstance& i) -> bool { |
| if (i.format() != vintf::HalFormat::AIDL) { |
| return true; // continue |
| } |
| ret.push_back({i.package() + "." + i.interface(), i.version().minorVer, 0}); |
| return true; // continue |
| }; |
| vintf::VintfObject::GetDeviceHalManifest()->forEachInstance(setInserter); |
| vintf::VintfObject::GetFrameworkHalManifest()->forEachInstance(setInserter); |
| return ret; |
| } |
| |
| TEST(Hal, AllAidlInterfacesAreInAosp) { |
| if (!kAidlUseUnfrozen) GTEST_SKIP() << "Not valid in 'next' configuration"; |
| if (getDeviceType() != DeviceType::PHONE) |
| GTEST_SKIP() << "Test only supports phones right now"; |
| for (const auto& package : allAidlManifestInterfaces()) { |
| EXPECT_TRUE(isAospAidlInterface(package.name)) |
| << "This device should only have AOSP interfaces, not: " |
| << package.name; |
| } |
| } |
| |
| struct AidlPackageCheck { |
| bool hasRegistration; |
| bool knownMissing; |
| }; |
| |
| TEST(Hal, AidlInterfacesImplemented) { |
| if (!kAidlUseUnfrozen) GTEST_SKIP() << "Not valid in 'next' configuration"; |
| if (getDeviceType() != DeviceType::PHONE) |
| GTEST_SKIP() << "Test only supports phones right now"; |
| std::vector<VersionedAidlPackage> manifest = allAidlManifestInterfaces(); |
| std::vector<VersionedAidlPackage> thoughtMissing = kKnownMissingAidl; |
| |
| for (const auto& treePackage : AidlInterfaceMetadata::all()) { |
| ASSERT_FALSE(treePackage.types.empty()) << treePackage.name; |
| if (std::none_of(treePackage.types.begin(), treePackage.types.end(), |
| isAospAidlInterface) || |
| isMissingAidl(treePackage.name)) |
| continue; |
| if (treePackage.stability != "vintf") continue; |
| |
| // expect versions from 1 to latest version. If the package has development |
| // the latest version is the latest known version + 1. Each of these need |
| // to be checked for registration and knownMissing. |
| std::map<size_t, AidlPackageCheck> expectedVersions; |
| for (const auto version : treePackage.versions) { |
| expectedVersions[version] = {false, false}; |
| } |
| if (treePackage.has_development) { |
| size_t version = |
| treePackage.versions.empty() ? 1 : *treePackage.versions.rbegin() + 1; |
| expectedVersions[version] = {false, false}; |
| } |
| |
| // Check all types and versions defined by the package for registration. |
| // The package version is considered registered if any of those types are |
| // present in the manifest with the same version. |
| // The package version is considered known missing if it is found in |
| // thoughtMissing. |
| bool latestRegistered = false; |
| for (const std::string& type : treePackage.types) { |
| for (auto& [version, check] : expectedVersions) { |
| auto it = std::remove_if( |
| manifest.begin(), manifest.end(), |
| [&type, &ver = version](const VersionedAidlPackage& package) { |
| return package.name == type && package.version == ver; |
| }); |
| if (it != manifest.end()) { |
| manifest.erase(it, manifest.end()); |
| if (version == expectedVersions.rbegin()->first) { |
| latestRegistered = true; |
| } |
| check.hasRegistration = true; |
| } |
| it = std::remove_if( |
| thoughtMissing.begin(), thoughtMissing.end(), |
| [&type, &ver = version](const VersionedAidlPackage& package) { |
| return package.name == getAidlPackage(type) && |
| package.version == ver; |
| }); |
| if (it != thoughtMissing.end()) { |
| thoughtMissing.erase(it, thoughtMissing.end()); |
| check.knownMissing = true; |
| } |
| } |
| } |
| |
| if (!latestRegistered && !expectedVersions.rbegin()->second.knownMissing) { |
| ADD_FAILURE() << "The latest version (" |
| << expectedVersions.rbegin()->first |
| << ") of the module is not implemented: " |
| << treePackage.name |
| << " which declares the following types:\n " |
| << base::Join(treePackage.types, "\n "); |
| } |
| |
| for (const auto& [version, check] : expectedVersions) { |
| if (check.knownMissing) { |
| if (check.hasRegistration) { |
| ADD_FAILURE() << "Package in missing list, but available: " |
| << treePackage.name << " V" << version |
| << " which declares the following types:\n " |
| << base::Join(treePackage.types, "\n "); |
| } |
| |
| continue; |
| } |
| } |
| } |
| |
| for (const auto& package : thoughtMissing) { |
| ADD_FAILURE() << "Interface in missing list and cannot find it anywhere: " |
| << package.name << " V" << package.version; |
| } |
| |
| for (const auto& package : manifest) { |
| ADD_FAILURE() << "Can't find manifest entry in tree: " << package.name |
| << " version: " << package.version; |
| } |
| } |