/* * 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef AIDL_USE_UNFROZEN constexpr bool kAidlUseUnfrozen = true; #else constexpr bool kAidlUseUnfrozen = false; #endif using namespace android; // clang-format off static const std::set kKnownMissingHidl = { "android.frameworks.automotive.display@1.0", // converted to AIDL, see b/170401743 "android.frameworks.cameraservice.device@2.1", "android.frameworks.cameraservice.service@2.2", // converted to AIDL, see b/205764761 "android.frameworks.displayservice@1.0", // deprecated, see b/141930622 "android.frameworks.schedulerservice@1.0", // deprecated, see b/37226359 "android.frameworks.sensorservice@1.0", // deprecated, see b/205764765 "android.frameworks.vr.composer@1.0", "android.frameworks.vr.composer@2.0", "android.frameworks.stats@1.0", // converted to AIDL, see b/177667419 "android.hardware.atrace@1.0", // deprecated, see b/204935495 "android.hardware.audio@2.0", "android.hardware.audio@4.0", "android.hardware.audio@5.0", "android.hardware.audio@6.0", "android.hardware.audio@7.1", // converted to AIDL, see b/264712385 "android.hardware.audio.effect@2.0", "android.hardware.audio.effect@4.0", "android.hardware.audio.effect@5.0", "android.hardware.audio.effect@6.0", "android.hardware.audio.effect@7.0", // converted to AIDL, see b/264712385 "android.hardware.authsecret@1.0", // converted to AIDL, see b/182976659 "android.hardware.automotive.audiocontrol@1.0", "android.hardware.automotive.audiocontrol@2.0", "android.hardware.automotive.can@1.0", // converted to AIDL, see b/170405615 "android.hardware.automotive.evs@1.1", "android.hardware.automotive.sv@1.0", "android.hardware.automotive.vehicle@2.0", "android.hardware.biometrics.fingerprint@2.3", // converted to AIDL, see b/152416783 "android.hardware.biometrics.face@1.0", // converted to AIDL, see b/168730443 "android.hardware.bluetooth.a2dp@1.0", "android.hardware.bluetooth.audio@2.1", // converted to AIDL, see b/203490261 "android.hardware.bluetooth@1.1", // converted to AIDL, see b/205758693 "android.hardware.boot@1.2", // converted to AIDL, see b/227536004 "android.hardware.broadcastradio@1.1", "android.hardware.broadcastradio@2.0", "android.hardware.camera.provider@2.7", // Camera converted to AIDL, b/196432585 "android.hardware.cas@1.2", // converted to AIDL, see b/227673974 "android.hardware.cas.native@1.0", "android.hardware.configstore@1.1", // deprecated, see b/149050985, b/149050733 "android.hardware.confirmationui@1.0", // converted to AIDL, see b/205760172 "android.hardware.contexthub@1.2", "android.hardware.drm@1.4", // converted to AIDL, b/200055138 "android.hardware.fastboot@1.1", "android.hardware.dumpstate@1.1", // deprecated, see b/205760700 "android.hardware.gatekeeper@1.0", // converted to AIDL, b/205760843 "android.hardware.gnss@1.1", // GNSS converted to AIDL, b/206670536 "android.hardware.gnss@2.1", // GNSS converted to AIDL, b/206670536 "android.hardware.gnss.measurement_corrections@1.1", // is sub-interface of gnss "android.hardware.gnss.visibility_control@1.0", "android.hardware.graphics.allocator@2.0", "android.hardware.graphics.allocator@3.0", "android.hardware.graphics.allocator@4.0", // converted to AIDL, see b/205761012 "android.hardware.graphics.bufferqueue@1.0", "android.hardware.graphics.bufferqueue@2.0", "android.hardware.graphics.composer@2.4", // converted to AIDL, see b/193240715 "android.hardware.graphics.mapper@2.1", "android.hardware.graphics.mapper@3.0", "android.hardware.graphics.mapper@4.0", // converted to Stable C, see b/205761028 "android.hardware.health.storage@1.0", // converted to AIDL, see b/177470478 "android.hardware.health@2.1", // converted to AIDL, see b/177269435 "android.hardware.input.classifier@1.0", // converted to AIDL, see b/205761620 "android.hardware.ir@1.0", // converted to AIDL, see b/205000342 "android.hardware.keymaster@3.0", "android.hardware.keymaster@4.1", // Replaced by AIDL KeyMint, see b/111446262 "android.hardware.light@2.0", "android.hardware.media.bufferpool@1.0", "android.hardware.media.bufferpool@2.0", "android.hardware.media.omx@1.0", // deprecated b/205761766 "android.hardware.memtrack@1.0", "android.hardware.neuralnetworks@1.3", // converted to AIDL, see b/161428342 "android.hardware.nfc@1.2", "android.hardware.oemlock@1.0", // converted to AIDL, see b/176107318 b/282160400 "android.hardware.power@1.3", "android.hardware.power.stats@1.0", "android.hardware.radio@1.6", // converted to AIDL "android.hardware.radio.config@1.3", // converted to AIDL "android.hardware.radio.deprecated@1.0", "android.hardware.renderscript@1.0", "android.hardware.soundtrigger@2.3", "android.hardware.secure_element@1.2", "android.hardware.sensors@1.0", "android.hardware.sensors@2.1", "android.hardware.tetheroffload.config@1.0", "android.hardware.tetheroffload.control@1.1", // see b/170699770 "android.hardware.thermal@1.1", "android.hardware.thermal@2.0", // Converted to AIDL (see b/205762943) "android.hardware.tv.cec@1.1", "android.hardware.tv.input@1.0", "android.hardware.tv.tuner@1.1", "android.hardware.usb@1.3", // migrated to AIDL see b/200993386 "android.hardware.usb.gadget@1.2", "android.hardware.vibrator@1.3", "android.hardware.vr@1.0", "android.hardware.weaver@1.0", "android.hardware.wifi@1.6", // Converted to AIDL (see b/205044134) "android.hardware.wifi.hostapd@1.3", // Converted to AIDL (see b/194806512) "android.hardware.wifi.supplicant@1.4", // Converted to AIDL (see b/196235436) "android.hidl.base@1.0", "android.hidl.memory.token@1.0", "android.hidl.memory@1.0", // Deprecated (see b/205764958) "android.system.net.netd@1.1", // Converted to AIDL (see b/205764585) "android.system.suspend@1.0", // Converted to AIDL (see b/170260236) "android.system.wifi.keystore@1.0", // 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 kPhoneOnlyAidl = { "android.hardware.camera.provider", }; static const std::set 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 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 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 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 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 binder = defaultServiceManager()->waitForService(String16("package_native")); sp packageManager = interface_cast(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 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 allAidlManifestInterfaces() { std::vector 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 manifest = allAidlManifestInterfaces(); std::vector 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 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; } }