/* ** ** Copyright 2017, 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 <stddef.h> #include <stdint.h> #include <iomanip> #include <iostream> #include <string> #include <android/hardware/confirmationui/support/msg_formatting.h> #include <gtest/gtest.h> using android::hardware::confirmationui::support::Message; using android::hardware::confirmationui::support::WriteStream; using android::hardware::confirmationui::support::ReadStream; using android::hardware::confirmationui::support::PromptUserConfirmationMsg; using android::hardware::confirmationui::support::write; using ::android::hardware::confirmationui::V1_0::UIOption; using ::android::hardware::keymaster::V4_0::HardwareAuthToken; using ::android::hardware::keymaster::V4_0::HardwareAuthenticatorType; using ::android::hardware::hidl_string; using ::android::hardware::hidl_vec; #ifdef DEBUG_MSG_FORMATTING namespace { char nibble2hex[16] = {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f'}; std::ostream& hexdump(std::ostream& out, const uint8_t* data, size_t size) { for (size_t i = 0; i < size; ++i) { uint8_t byte = data[i]; out << (nibble2hex[0x0F & (byte >> 4)]); out << (nibble2hex[0x0F & byte]); switch (i & 0xf) { case 0xf: out << "\n"; break; case 7: out << " "; break; default: out << " "; break; } } return out; } } // namespace #endif TEST(MsgFormattingTest, FeatureTest) { uint8_t buffer[0x1000]; WriteStream out(buffer); out = unalign(out); out += 4; auto begin = out.pos(); out = write( PromptUserConfirmationMsg(), out, hidl_string("Do you?"), hidl_vec<uint8_t>{0x01, 0x02, 0x03}, hidl_string("en"), hidl_vec<UIOption>{UIOption::AccessibilityInverted, UIOption::AccessibilityMagnified}); ReadStream in(buffer); in = unalign(in); in += 4; hidl_string prompt; hidl_vec<uint8_t> extra; hidl_string locale; hidl_vec<UIOption> uiOpts; bool command_matches; std::tie(in, command_matches, prompt, extra, locale, uiOpts) = read(PromptUserConfirmationMsg(), in); ASSERT_TRUE(in); ASSERT_TRUE(command_matches); ASSERT_EQ(hidl_string("Do you?"), prompt); ASSERT_EQ((hidl_vec<uint8_t>{0x01, 0x02, 0x03}), extra); ASSERT_EQ(hidl_string("en"), locale); ASSERT_EQ( (hidl_vec<UIOption>{UIOption::AccessibilityInverted, UIOption::AccessibilityMagnified}), uiOpts); #ifdef DEBUG_MSG_FORMATTING hexdump(std::cout, buffer, 100) << std::endl; #endif // The following assertions check that the hidl_[vec|string] types are in fact read in place, // and no copying occurs. Copying results in heap allocation which we intend to avoid. ASSERT_EQ(8, const_cast<uint8_t*>(reinterpret_cast<const uint8_t*>(prompt.c_str())) - begin); ASSERT_EQ(20, extra.data() - begin); ASSERT_EQ(27, const_cast<uint8_t*>(reinterpret_cast<const uint8_t*>(locale.c_str())) - begin); ASSERT_EQ(40, reinterpret_cast<uint8_t*>(uiOpts.data()) - begin); } TEST(MsgFormattingTest, HardwareAuthTokenTest) { uint8_t buffer[0x1000]; HardwareAuthToken expected, actual; expected.authenticatorId = 0xa1a3a4a5a6a7a8; expected.authenticatorType = HardwareAuthenticatorType::NONE; expected.challenge = 0xb1b2b3b4b5b6b7b8; expected.userId = 0x1122334455667788; expected.timestamp = 0xf1f2f3f4f5f6f7f8; expected.mac = hidl_vec<uint8_t>{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32}; WriteStream out(buffer); out = write(Message<HardwareAuthToken>(), out, expected); ReadStream in(buffer); std::tie(in, actual) = read(Message<HardwareAuthToken>(), in); ASSERT_EQ(expected, actual); }