| /* |
| * Copyright (C) 2018 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 "fastboot.h" |
| |
| #include <android-base/logging.h> |
| #include <gtest/gtest.h> |
| |
| TEST(FastBoot, ParseOsPatchLevel) { |
| FastBootTool fb; |
| boot_img_hdr_v1 hdr; |
| |
| hdr = {}; |
| fb.ParseOsPatchLevel(&hdr, "2018-01-05"); |
| ASSERT_EQ(2018U, 2000U + ((hdr.os_version >> 4) & 0x7f)); |
| ASSERT_EQ(1U, ((hdr.os_version >> 0) & 0xf)); |
| |
| EXPECT_DEATH(fb.ParseOsPatchLevel(&hdr, "2018"), "should be YYYY-MM-DD"); |
| EXPECT_DEATH(fb.ParseOsPatchLevel(&hdr, "2018-01"), "should be YYYY-MM-DD"); |
| EXPECT_DEATH(fb.ParseOsPatchLevel(&hdr, "2128-01-05"), "year out of range"); |
| EXPECT_DEATH(fb.ParseOsPatchLevel(&hdr, "2018-13-05"), "month out of range"); |
| } |
| |
| TEST(FastBoot, ParseOsVersion) { |
| FastBootTool fb; |
| boot_img_hdr_v1 hdr; |
| |
| hdr = {}; |
| fb.ParseOsVersion(&hdr, "1.2.3"); |
| ASSERT_EQ(1U, ((hdr.os_version >> 25) & 0x7f)); |
| ASSERT_EQ(2U, ((hdr.os_version >> 18) & 0x7f)); |
| ASSERT_EQ(3U, ((hdr.os_version >> 11) & 0x7f)); |
| |
| fb.ParseOsVersion(&hdr, "1.2"); |
| ASSERT_EQ(1U, ((hdr.os_version >> 25) & 0x7f)); |
| ASSERT_EQ(2U, ((hdr.os_version >> 18) & 0x7f)); |
| ASSERT_EQ(0U, ((hdr.os_version >> 11) & 0x7f)); |
| |
| fb.ParseOsVersion(&hdr, "1"); |
| ASSERT_EQ(1U, ((hdr.os_version >> 25) & 0x7f)); |
| ASSERT_EQ(0U, ((hdr.os_version >> 18) & 0x7f)); |
| ASSERT_EQ(0U, ((hdr.os_version >> 11) & 0x7f)); |
| |
| EXPECT_DEATH(fb.ParseOsVersion(&hdr, ""), "bad OS version"); |
| EXPECT_DEATH(fb.ParseOsVersion(&hdr, "1.2.3.4"), "bad OS version"); |
| EXPECT_DEATH(fb.ParseOsVersion(&hdr, "128.2.3"), "bad OS version"); |
| EXPECT_DEATH(fb.ParseOsVersion(&hdr, "1.128.3"), "bad OS version"); |
| EXPECT_DEATH(fb.ParseOsVersion(&hdr, "1.2.128"), "bad OS version"); |
| } |
| |
| extern bool ParseRequirementLine(const std::string& line, std::string* name, std::string* product, |
| bool* invert, std::vector<std::string>* options); |
| |
| static void ParseRequirementLineTest(const std::string& line, const std::string& expected_name, |
| const std::string& expected_product, bool expected_invert, |
| const std::vector<std::string>& expected_options) { |
| std::string name; |
| std::string product; |
| bool invert; |
| std::vector<std::string> options; |
| |
| EXPECT_TRUE(ParseRequirementLine(line, &name, &product, &invert, &options)) << line; |
| |
| EXPECT_EQ(expected_name, name) << line; |
| EXPECT_EQ(expected_product, product) << line; |
| EXPECT_EQ(expected_invert, invert) << line; |
| EXPECT_EQ(expected_options, options) << line; |
| } |
| |
| TEST(FastBoot, ParseRequirementLineSuccesses) { |
| // Examples provided in the code + slight variations. |
| ParseRequirementLineTest("require product=alpha", "product", "", false, {"alpha"}); |
| ParseRequirementLineTest("require product=alpha|beta|gamma", "product", "", false, |
| {"alpha", "beta", "gamma"}); |
| ParseRequirementLineTest("require version-bootloader=1234", "version-bootloader", "", false, |
| {"1234"}); |
| ParseRequirementLineTest("require-for-product:gamma version-bootloader=istanbul", |
| "version-bootloader", "gamma", false, {"istanbul"}); |
| ParseRequirementLineTest("require-for-product:gamma version-bootloader=istanbul|constantinople", |
| "version-bootloader", "gamma", false, {"istanbul", "constantinople"}); |
| ParseRequirementLineTest("require partition-exists=vendor", "partition-exists", "", false, |
| {"vendor"}); |
| ParseRequirementLineTest("reject product=alpha", "product", "", true, {"alpha"}); |
| ParseRequirementLineTest("reject product=alpha|beta|gamma", "product", "", true, |
| {"alpha", "beta", "gamma"}); |
| |
| // Without any prefix, assume 'require' |
| ParseRequirementLineTest("product=alpha|beta|gamma", "product", "", false, |
| {"alpha", "beta", "gamma"}); |
| // Including if the variable name is otherwise a prefix keyword |
| ParseRequirementLineTest("require = alpha", "require", "", false, {"alpha"}); |
| ParseRequirementLineTest("reject = alpha", "reject", "", false, {"alpha"}); |
| ParseRequirementLineTest("require-for-product:gamma = alpha", "require-for-product:gamma", "", |
| false, {"alpha"}); |
| |
| // Extra spaces are allowed. |
| ParseRequirementLineTest("require product=alpha|beta|gamma", "product", "", false, |
| {"alpha", "beta", "gamma"}); |
| ParseRequirementLineTest("require product =alpha|beta|gamma", "product", "", false, |
| {"alpha", "beta", "gamma"}); |
| ParseRequirementLineTest("require product= alpha|beta|gamma", "product", "", false, |
| {"alpha", "beta", "gamma"}); |
| ParseRequirementLineTest("require product = alpha|beta|gamma", "product", "", false, |
| {"alpha", "beta", "gamma"}); |
| ParseRequirementLineTest("require product=alpha |beta|gamma", "product", "", false, |
| {"alpha", "beta", "gamma"}); |
| ParseRequirementLineTest("require product=alpha| beta|gamma", "product", "", false, |
| {"alpha", "beta", "gamma"}); |
| ParseRequirementLineTest("require product=alpha | beta|gamma", "product", "", false, |
| {"alpha", "beta", "gamma"}); |
| ParseRequirementLineTest("require product=alpha|beta|gamma ", "product", "", false, |
| {"alpha", "beta", "gamma"}); |
| ParseRequirementLineTest("product = alpha | beta | gamma ", "product", "", false, |
| {"alpha", "beta", "gamma"}); |
| ParseRequirementLineTest("require-for-product: gamma version-bootloader=istanbul", |
| "version-bootloader", "gamma", false, {"istanbul"}); |
| |
| // Extraneous ending | is okay, implies accepting an empty string. |
| ParseRequirementLineTest("require product=alpha|", "product", "", false, {"alpha", ""}); |
| ParseRequirementLineTest("require product=alpha|beta|gamma|", "product", "", false, |
| {"alpha", "beta", "gamma", ""}); |
| |
| // Accept empty options, double ||, etc, implies accepting an empty string. |
| ParseRequirementLineTest("require product=alpha||beta| |gamma", "product", "", false, |
| {"alpha", "", "beta", "", "gamma"}); |
| ParseRequirementLineTest("require product=alpha||beta|gamma", "product", "", false, |
| {"alpha", "", "beta", "gamma"}); |
| ParseRequirementLineTest("require product=alpha|beta| |gamma", "product", "", false, |
| {"alpha", "beta", "", "gamma"}); |
| ParseRequirementLineTest("require product=alpha||", "product", "", false, {"alpha", "", ""}); |
| ParseRequirementLineTest("require product=alpha|| ", "product", "", false, {"alpha", "", ""}); |
| ParseRequirementLineTest("require product=alpha| ", "product", "", false, {"alpha", ""}); |
| ParseRequirementLineTest("require product=alpha|beta| ", "product", "", false, |
| {"alpha", "beta", ""}); |
| |
| // No option string is also treating as accepting an empty string. |
| ParseRequirementLineTest("require =", "require", "", false, {""}); |
| ParseRequirementLineTest("require = |", "require", "", false, {"", ""}); |
| ParseRequirementLineTest("reject =", "reject", "", false, {""}); |
| ParseRequirementLineTest("reject = |", "reject", "", false, {"", ""}); |
| ParseRequirementLineTest("require-for-product: =", "require-for-product:", "", false, {""}); |
| ParseRequirementLineTest("require-for-product: = | ", "require-for-product:", "", false, |
| {"", ""}); |
| ParseRequirementLineTest("require product=", "product", "", false, {""}); |
| ParseRequirementLineTest("require product = ", "product", "", false, {""}); |
| ParseRequirementLineTest("require product = | ", "product", "", false, {"", ""}); |
| ParseRequirementLineTest("reject product=", "product", "", true, {""}); |
| ParseRequirementLineTest("reject product = ", "product", "", true, {""}); |
| ParseRequirementLineTest("reject product = | ", "product", "", true, {"", ""}); |
| ParseRequirementLineTest("require-for-product:gamma product=", "product", "gamma", false, {""}); |
| ParseRequirementLineTest("require-for-product:gamma product = ", "product", "gamma", false, |
| {""}); |
| ParseRequirementLineTest("require-for-product:gamma product = |", "product", "gamma", false, |
| {"", ""}); |
| |
| // Check for board -> product substitution. |
| ParseRequirementLineTest("require board=alpha", "product", "", false, {"alpha"}); |
| ParseRequirementLineTest("board=alpha", "product", "", false, {"alpha"}); |
| } |
| |
| static void ParseRequirementLineTestMalformed(const std::string& line) { |
| std::string name; |
| std::string product; |
| bool invert; |
| std::vector<std::string> options; |
| |
| EXPECT_FALSE(ParseRequirementLine(line, &name, &product, &invert, &options)) << line; |
| } |
| |
| TEST(FastBoot, ParseRequirementLineMalformed) { |
| ParseRequirementLineTestMalformed("nothing"); |
| ParseRequirementLineTestMalformed(""); |
| ParseRequirementLineTestMalformed("="); |
| ParseRequirementLineTestMalformed("|"); |
| |
| ParseRequirementLineTestMalformed("require"); |
| ParseRequirementLineTestMalformed("require "); |
| ParseRequirementLineTestMalformed("reject"); |
| ParseRequirementLineTestMalformed("reject "); |
| ParseRequirementLineTestMalformed("require-for-product:"); |
| ParseRequirementLineTestMalformed("require-for-product: "); |
| |
| ParseRequirementLineTestMalformed("require product"); |
| ParseRequirementLineTestMalformed("reject product"); |
| |
| ParseRequirementLineTestMalformed("require-for-product:gamma"); |
| ParseRequirementLineTestMalformed("require-for-product:gamma product"); |
| |
| // No spaces allowed before between require-for-product and :. |
| ParseRequirementLineTestMalformed("require-for-product :"); |
| } |
| |
| static void ParseNetworkSerialTest(const std::string& description, const std::string& serial, |
| const std::string& expected_address, |
| const Socket::Protocol expected_protocol, |
| const int expected_port) { |
| const Result<NetworkSerial, FastbootError> parsed = ParseNetworkSerial(serial); |
| |
| ASSERT_RESULT_OK(parsed) << description; |
| |
| const NetworkSerial network_serial = parsed.value(); |
| EXPECT_EQ(network_serial.address, expected_address) << description; |
| EXPECT_EQ(network_serial.protocol, expected_protocol) << description; |
| EXPECT_EQ(network_serial.port, expected_port) << description; |
| } |
| |
| static void ParseNetworkSerialNegativeTest(const std::string& description, |
| const std::string& serial, |
| const FastbootError::Type expected_error) { |
| const Result<NetworkSerial, FastbootError> parsed = ParseNetworkSerial(serial); |
| |
| EXPECT_FALSE(parsed.ok()) << description; |
| EXPECT_EQ(parsed.error().code(), expected_error) << description; |
| } |
| |
| TEST(FastBoot, ParseNetworkSerial) { |
| ParseNetworkSerialTest("tcp IPv4 parsed", "tcp:192.168.1.0", "192.168.1.0", |
| Socket::Protocol::kTcp, 5554); |
| |
| ParseNetworkSerialTest("udp IPv4 parsed", "udp:192.168.1.0", "192.168.1.0", |
| Socket::Protocol::kUdp, 5554); |
| |
| ParseNetworkSerialTest("port parsed", "udp:192.168.1.0:9999", "192.168.1.0", |
| Socket::Protocol::kUdp, 9999); |
| |
| ParseNetworkSerialTest("IPv6 parsed", "tcp:2001:db8:3333:4444:5555:6666:7777:8888", |
| "2001:db8:3333:4444:5555:6666:7777:8888", Socket::Protocol::kTcp, 5554); |
| |
| ParseNetworkSerialTest("empty IPv6 parsed", "tcp:::", "::", Socket::Protocol::kTcp, 5554); |
| |
| ParseNetworkSerialNegativeTest("wrong prefix", "tcpa:192.168.1.0", |
| FastbootError::Type::NETWORK_SERIAL_WRONG_PREFIX); |
| |
| ParseNetworkSerialNegativeTest("no prefix", "192.168.1.0", |
| FastbootError::Type::NETWORK_SERIAL_WRONG_PREFIX); |
| |
| ParseNetworkSerialNegativeTest("wrong port", "tcp:192.168.1.0:-1", |
| FastbootError::Type::NETWORK_SERIAL_WRONG_ADDRESS); |
| } |
| |
| int main(int argc, char* argv[]) { |
| ::testing::InitGoogleTest(&argc, argv); |
| android::base::InitLogging(argv); |
| android::base::SetMinimumLogSeverity(android::base::VERBOSE); |
| |
| return RUN_ALL_TESTS(); |
| } |