/* * Copyright (C) 2016 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. */ #if defined(HAS_GTEST) #include <gtest/gtest.h> #else #include "gtest_stubs.h" #endif #include <string.h> #include <nvram/core/nvram_manager.h> #include <nvram/core/persistence.h> #include "fake_storage.h" namespace nvram { namespace { class NvramManagerTest : public testing::Test { protected: NvramManagerTest() { storage::Clear(); } static void SetupHeader(uint32_t header_version, uint32_t index) { NvramHeader header; header.version = header_version; ASSERT_TRUE(header.allocated_indices.Resize(1)); header.allocated_indices[0] = index; ASSERT_EQ(storage::Status::kSuccess, persistence::StoreHeader(header)); } static void ReadAndCompareSpaceData(NvramManager* nvram, uint32_t index, const void* expected_contents, size_t expected_size) { ReadSpaceRequest read_space_request; read_space_request.index = index; ReadSpaceResponse read_space_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram->ReadSpace(read_space_request, &read_space_response)); ASSERT_EQ(expected_size, read_space_response.buffer.size()); EXPECT_EQ(0, memcmp(read_space_response.buffer.data(), expected_contents, expected_size)); } static uint32_t GetControlsMask(const Vector<nvram_control_t>& controls) { uint32_t mask = 0; for (nvram_control_t control : controls) { mask |= (1 << control); } return mask; } }; TEST_F(NvramManagerTest, Init_FromScratch) { NvramManager nvram; GetSpaceInfoRequest get_space_info_request; get_space_info_request.index = 1; GetSpaceInfoResponse get_space_info_response; EXPECT_EQ( NV_RESULT_SPACE_DOES_NOT_EXIST, nvram.GetSpaceInfo(get_space_info_request, &get_space_info_response)); } TEST_F(NvramManagerTest, Init_TrailingStorageBytes) { // Set up a pre-existing space and add some trailing bytes. NvramSpace space; ASSERT_TRUE(space.contents.Resize(10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(1, space)); Blob space_blob; ASSERT_EQ(storage::Status::kSuccess, storage::LoadSpace(1, &space_blob)); ASSERT_TRUE(space_blob.Resize(space_blob.size() + 10)); ASSERT_EQ(storage::Status::kSuccess, storage::StoreSpace(1, space_blob)); // Produce a matching header and append some trailing bytes. NvramHeader header; header.version = NvramHeader::kVersion; ASSERT_TRUE(header.allocated_indices.Resize(1)); header.allocated_indices[0] = 1; ASSERT_EQ(storage::Status::kSuccess, persistence::StoreHeader(header)); Blob header_blob; ASSERT_EQ(storage::Status::kSuccess, storage::LoadHeader(&header_blob)); ASSERT_TRUE(header_blob.Resize(header_blob.size() + 10)); ASSERT_EQ(storage::Status::kSuccess, storage::StoreHeader(header_blob)); // Initialize the |NvramManager| and check that the header and space blobs get // loaded successfully. NvramManager nvram; GetInfoRequest get_info_request; GetInfoResponse get_info_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.GetInfo(get_info_request, &get_info_response)); ASSERT_EQ(1U, get_info_response.space_list.size()); EXPECT_EQ(1U, get_info_response.space_list[0]); GetSpaceInfoRequest get_space_info_request; get_space_info_request.index = 1; GetSpaceInfoResponse get_space_info_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.GetSpaceInfo(get_space_info_request, &get_space_info_response)); EXPECT_EQ(10U, get_space_info_response.size); } TEST_F(NvramManagerTest, Init_SpacesPresent) { // Set up two pre-existing spaces. NvramSpace space; ASSERT_TRUE(space.contents.Resize(10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(1, space)); ASSERT_TRUE(space.contents.Resize(20)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(2, space)); // Indicate 3 present spaces in the header, including one that doesn't have // space data in storage. NvramHeader header; header.version = NvramHeader::kVersion; ASSERT_TRUE(header.allocated_indices.Resize(3)); header.allocated_indices[0] = 1; header.allocated_indices[1] = 2; header.allocated_indices[2] = 3; header.provisional_index.Activate() = 4; ASSERT_EQ(storage::Status::kSuccess, persistence::StoreHeader(header)); NvramManager nvram; // Check that the spaces are correctly recovered. GetSpaceInfoRequest get_space_info_request; get_space_info_request.index = 1; GetSpaceInfoResponse get_space_info_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.GetSpaceInfo(get_space_info_request, &get_space_info_response)); EXPECT_EQ(10u, get_space_info_response.size); get_space_info_request.index = 2; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.GetSpaceInfo(get_space_info_request, &get_space_info_response)); EXPECT_EQ(20u, get_space_info_response.size); get_space_info_request.index = 3; EXPECT_EQ( NV_RESULT_INTERNAL_ERROR, nvram.GetSpaceInfo(get_space_info_request, &get_space_info_response)); get_space_info_request.index = 4; EXPECT_EQ( NV_RESULT_SPACE_DOES_NOT_EXIST, nvram.GetSpaceInfo(get_space_info_request, &get_space_info_response)); } TEST_F(NvramManagerTest, Init_BadSpacePresent) { // Set up a good and a bad NVRAM space. NvramSpace space; ASSERT_TRUE(space.contents.Resize(10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(1, space)); const uint8_t kBadSpaceData[] = {0xba, 0xad}; Blob bad_space_blob; ASSERT_TRUE(bad_space_blob.Assign(kBadSpaceData, sizeof(kBadSpaceData))); ASSERT_EQ(storage::Status::kSuccess, storage::StoreSpace(2, bad_space_blob)); NvramHeader header; header.version = NvramHeader::kVersion; ASSERT_TRUE(header.allocated_indices.Resize(2)); header.allocated_indices[0] = 1; header.allocated_indices[1] = 2; ASSERT_EQ(storage::Status::kSuccess, persistence::StoreHeader(header)); NvramManager nvram; // The bad index will fail requests. GetSpaceInfoRequest get_space_info_request; get_space_info_request.index = 2; GetSpaceInfoResponse get_space_info_response; nvram_result_t result = nvram.GetSpaceInfo(get_space_info_request, &get_space_info_response); EXPECT_NE(NV_RESULT_SUCCESS, result); EXPECT_NE(NV_RESULT_SPACE_DOES_NOT_EXIST, result); // A request to get info for the good index should succeed. get_space_info_request.index = 1; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.GetSpaceInfo(get_space_info_request, &get_space_info_response)); EXPECT_EQ(10u, get_space_info_response.size); } TEST_F(NvramManagerTest, Init_NewerStorageVersion) { // Set up an NVRAM space. NvramSpace space; ASSERT_TRUE(space.contents.Resize(10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(1, space)); SetupHeader(NvramHeader::kVersion + 1, 1); NvramManager nvram; // Requests should fail due to version mismatch. GetSpaceInfoRequest get_space_info_request; get_space_info_request.index = 1; GetSpaceInfoResponse get_space_info_response; EXPECT_EQ( NV_RESULT_INTERNAL_ERROR, nvram.GetSpaceInfo(get_space_info_request, &get_space_info_response)); } TEST_F(NvramManagerTest, Init_StorageObjectTypeMismatch) { // Set up an NVRAM space. NvramSpace space; ASSERT_TRUE(space.contents.Resize(10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(1, space)); // Copy the space blob to the header storage. Blob space_blob; ASSERT_EQ(storage::Status::kSuccess, storage::LoadSpace(1, &space_blob)); ASSERT_EQ(storage::Status::kSuccess, storage::StoreHeader(space_blob)); NvramManager nvram; // Initialization should detect that the header storage object doesn't look // like a header, so initialization should fail. GetInfoRequest get_info_request; GetInfoResponse get_info_response; EXPECT_EQ(NV_RESULT_INTERNAL_ERROR, nvram.GetInfo(get_info_request, &get_info_response)); } TEST_F(NvramManagerTest, CreateSpace_Success) { NvramManager nvram; // Make a call to CreateSpace, which should succeed. CreateSpaceRequest create_space_request; create_space_request.index = 1; create_space_request.size = 32; ASSERT_TRUE(create_space_request.controls.Resize(5)); create_space_request.controls[0] = NV_CONTROL_BOOT_WRITE_LOCK; create_space_request.controls[1] = NV_CONTROL_BOOT_READ_LOCK; create_space_request.controls[2] = NV_CONTROL_WRITE_AUTHORIZATION; create_space_request.controls[3] = NV_CONTROL_READ_AUTHORIZATION; create_space_request.controls[4] = NV_CONTROL_WRITE_EXTEND; CreateSpaceResponse create_space_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.CreateSpace(create_space_request, &create_space_response)); // GetSpaceInfo should reflect the space parameters set during creation. GetSpaceInfoRequest get_space_info_request; get_space_info_request.index = 1; GetSpaceInfoResponse get_space_info_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.GetSpaceInfo(get_space_info_request, &get_space_info_response)); EXPECT_EQ(32u, get_space_info_response.size); EXPECT_EQ(GetControlsMask(create_space_request.controls), GetControlsMask(get_space_info_response.controls)); EXPECT_EQ(false, get_space_info_response.read_locked); EXPECT_EQ(false, get_space_info_response.write_locked); } TEST_F(NvramManagerTest, CreateSpace_Existing) { // Set up an NVRAM space. NvramSpace space; ASSERT_TRUE(space.contents.Resize(10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(1, space)); SetupHeader(NvramHeader::kVersion, 1); NvramManager nvram; // A request to create another space with the same index should fail. CreateSpaceRequest create_space_request; create_space_request.index = 1; create_space_request.size = 16; CreateSpaceResponse create_space_response; EXPECT_EQ(NV_RESULT_SPACE_ALREADY_EXISTS, nvram.CreateSpace(create_space_request, &create_space_response)); } TEST_F(NvramManagerTest, CreateSpace_TooLarge) { NvramManager nvram; // A request to create a space with a too large content size should fail. CreateSpaceRequest create_space_request; create_space_request.index = 1; create_space_request.size = 16384; CreateSpaceResponse create_space_response; EXPECT_EQ(NV_RESULT_INVALID_PARAMETER, nvram.CreateSpace(create_space_request, &create_space_response)); } TEST_F(NvramManagerTest, CreateSpace_AuthTooLarge) { NvramManager nvram; // A request to create a space with a too large authorization value size // should fail. CreateSpaceRequest create_space_request; create_space_request.index = 1; ASSERT_TRUE(create_space_request.authorization_value.Resize(256)); CreateSpaceResponse create_space_response; EXPECT_EQ(NV_RESULT_INVALID_PARAMETER, nvram.CreateSpace(create_space_request, &create_space_response)); } TEST_F(NvramManagerTest, CreateSpace_BadControl) { NvramManager nvram; // A request to create a space with an unknown control value should fail. CreateSpaceRequest create_space_request; create_space_request.index = 1; create_space_request.size = 16; ASSERT_TRUE(create_space_request.controls.Resize(2)); create_space_request.controls[0] = NV_CONTROL_BOOT_WRITE_LOCK; create_space_request.controls[1] = 17; CreateSpaceResponse create_space_response; EXPECT_EQ(NV_RESULT_INVALID_PARAMETER, nvram.CreateSpace(create_space_request, &create_space_response)); } TEST_F(NvramManagerTest, CreateSpace_ControlWriteLockExclusive) { NvramManager nvram; // Spaces may not be created with conflicting write lock modes. CreateSpaceRequest create_space_request; create_space_request.index = 1; create_space_request.size = 16; ASSERT_TRUE(create_space_request.controls.Resize(2)); create_space_request.controls[0] = NV_CONTROL_BOOT_WRITE_LOCK; create_space_request.controls[1] = NV_CONTROL_PERSISTENT_WRITE_LOCK; CreateSpaceResponse create_space_response; EXPECT_EQ(NV_RESULT_INVALID_PARAMETER, nvram.CreateSpace(create_space_request, &create_space_response)); } TEST_F(NvramManagerTest, CreateSpace_WriteExtendSpaceSize) { NvramManager nvram; // Write-extend spaces must match SHA256 hash size, i.e. 32 bytes. CreateSpaceRequest create_space_request; create_space_request.index = 1; create_space_request.size = 16; ASSERT_TRUE(create_space_request.controls.Resize(1)); create_space_request.controls[0] = NV_CONTROL_WRITE_EXTEND; CreateSpaceResponse create_space_response; EXPECT_EQ(NV_RESULT_INVALID_PARAMETER, nvram.CreateSpace(create_space_request, &create_space_response)); } TEST_F(NvramManagerTest, CreateSpace_HeaderWriteError) { // If the header fails to get written to storage, the creation request should // fail. storage::SetHeaderWriteError(true); NvramManager nvram; CreateSpaceRequest create_space_request; create_space_request.index = 1; create_space_request.size = 16; CreateSpaceResponse create_space_response; EXPECT_EQ(NV_RESULT_INTERNAL_ERROR, nvram.CreateSpace(create_space_request, &create_space_response)); // The space shouldn't be present. GetInfoRequest get_info_request; GetInfoResponse get_info_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.GetInfo(get_info_request, &get_info_response)); EXPECT_EQ(0U, get_info_response.space_list.size()); // Creation of the space after clearing the error should work. storage::SetHeaderWriteError(false); EXPECT_EQ(NV_RESULT_SUCCESS, nvram.CreateSpace(create_space_request, &create_space_response)); // The space should be reported as allocated now. EXPECT_EQ(NV_RESULT_SUCCESS, nvram.GetInfo(get_info_request, &get_info_response)); ASSERT_EQ(1U, get_info_response.space_list.size()); EXPECT_EQ(1U, get_info_response.space_list[0]); } TEST_F(NvramManagerTest, CreateSpace_SpaceWriteError) { storage::SetSpaceWriteError(1, true); NvramManager nvram; // A request to create another space with the same index should fail. CreateSpaceRequest create_space_request; create_space_request.index = 1; create_space_request.size = 16; CreateSpaceResponse create_space_response; EXPECT_EQ(NV_RESULT_INTERNAL_ERROR, nvram.CreateSpace(create_space_request, &create_space_response)); // Reloading the state after a crash should not show any traces of the space. storage::SetSpaceWriteError(1, false); NvramManager nvram2; // The space shouldn't exist in the space list. GetInfoRequest get_info_request; GetInfoResponse get_info_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram2.GetInfo(get_info_request, &get_info_response)); EXPECT_EQ(0U, get_info_response.space_list.size()); // The space info request should indicate the space doesn't exist. GetSpaceInfoRequest get_space_info_request; get_space_info_request.index = 1; GetSpaceInfoResponse get_space_info_response; EXPECT_EQ( NV_RESULT_SPACE_DOES_NOT_EXIST, nvram2.GetSpaceInfo(get_space_info_request, &get_space_info_response)); } TEST_F(NvramManagerTest, DeleteSpace_SpaceAbsent) { NvramManager nvram; // Attempt to delete a non-existing space. DeleteSpaceRequest delete_space_request; delete_space_request.index = 42; DeleteSpaceResponse delete_space_response; EXPECT_EQ(NV_RESULT_SPACE_DOES_NOT_EXIST, nvram.DeleteSpace(delete_space_request, &delete_space_response)); } TEST_F(NvramManagerTest, DeleteSpace_Success) { // Set up an NVRAM space. NvramSpace space; ASSERT_TRUE(space.contents.Resize(10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(42, space)); SetupHeader(NvramHeader::kVersion, 42); NvramManager nvram; // Successful deletion. DeleteSpaceRequest delete_space_request; delete_space_request.index = 42; DeleteSpaceResponse delete_space_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.DeleteSpace(delete_space_request, &delete_space_response)); } TEST_F(NvramManagerTest, DeleteSpace_AuthorizationFailure) { // Set up an NVRAM space. NvramSpace space; space.controls = (1 << NV_CONTROL_WRITE_AUTHORIZATION); const char kAuthorizationValue[] = "secret"; ASSERT_TRUE(space.authorization_value.Assign(kAuthorizationValue, sizeof(kAuthorizationValue))); ASSERT_TRUE(space.contents.Resize(10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(42, space)); SetupHeader(NvramHeader::kVersion, 42); NvramManager nvram; // Deletion should fail if correct secret is not provided. DeleteSpaceRequest delete_space_request; delete_space_request.index = 42; DeleteSpaceResponse delete_space_response; EXPECT_EQ(NV_RESULT_ACCESS_DENIED, nvram.DeleteSpace(delete_space_request, &delete_space_response)); } TEST_F(NvramManagerTest, DeleteSpace_HalfDeleted) { // Set up an NVRAM space. NvramSpace space; ASSERT_TRUE(space.contents.Resize(10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(42, space)); SetupHeader(NvramHeader::kVersion, 42); // Hold on to the space data. Blob space_data; ASSERT_EQ(storage::Status::kSuccess, storage::LoadSpace(42, &space_data)); NvramManager nvram; // Delete the space. DeleteSpaceRequest delete_space_request; delete_space_request.index = 42; DeleteSpaceResponse delete_space_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.DeleteSpace(delete_space_request, &delete_space_response)); // Put the space data back into place to simulate a half-completed deletion. ASSERT_EQ(storage::Status::kSuccess, storage::StoreSpace(42, space_data)); // The space should remain deleted after re-initialization. NvramManager nvram2; GetSpaceInfoRequest get_space_info_request; get_space_info_request.index = 42; GetSpaceInfoResponse get_space_info_response; EXPECT_EQ( NV_RESULT_SPACE_DOES_NOT_EXIST, nvram2.GetSpaceInfo(get_space_info_request, &get_space_info_response)); // Re-creation of a space with the same index should work. CreateSpaceRequest create_space_request; create_space_request.index = 42; create_space_request.size = 32; ASSERT_TRUE(create_space_request.controls.Resize(1)); create_space_request.controls[0] = NV_CONTROL_BOOT_WRITE_LOCK; CreateSpaceResponse create_space_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram2.CreateSpace(create_space_request, &create_space_response)); } TEST_F(NvramManagerTest, DeleteSpace_SpaceDeleteError) { // Set up an NVRAM space. NvramSpace space; ASSERT_TRUE(space.contents.Resize(10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(42, space)); SetupHeader(NvramHeader::kVersion, 42); // Make space deletion fail. storage::SetSpaceWriteError(42, true); NvramManager nvram; // Attempt to delete the space. DeleteSpaceRequest delete_space_request; delete_space_request.index = 42; DeleteSpaceResponse delete_space_response; EXPECT_EQ(NV_RESULT_INTERNAL_ERROR, nvram.DeleteSpace(delete_space_request, &delete_space_response)); // The space should remain present. GetSpaceInfoRequest get_space_info_request; get_space_info_request.index = 42; GetSpaceInfoResponse get_space_info_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.GetSpaceInfo(get_space_info_request, &get_space_info_response)); EXPECT_EQ(10U, get_space_info_response.size); // Starting up from scratch shouldn't destroy the space either. storage::SetSpaceWriteError(42, false); NvramManager nvram2; GetSpaceInfoResponse get_space_info_response_2; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.GetSpaceInfo(get_space_info_request, &get_space_info_response_2)); EXPECT_EQ(10U, get_space_info_response_2.size); } TEST_F(NvramManagerTest, DeleteSpace_HeaderWriteError) { // Set up an NVRAM space. NvramSpace space; ASSERT_TRUE(space.contents.Resize(10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(42, space)); SetupHeader(NvramHeader::kVersion, 42); // Header write on deletion will fail. storage::SetHeaderWriteError(true); NvramManager nvram; // Attempt to delete the space. DeleteSpaceRequest delete_space_request; delete_space_request.index = 42; DeleteSpaceResponse delete_space_response; EXPECT_EQ(NV_RESULT_INTERNAL_ERROR, nvram.DeleteSpace(delete_space_request, &delete_space_response)); // The space should remain present. GetSpaceInfoRequest get_space_info_request; get_space_info_request.index = 42; GetSpaceInfoResponse get_space_info_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.GetSpaceInfo(get_space_info_request, &get_space_info_response)); EXPECT_EQ(10U, get_space_info_response.size); // Starting up from scratch shouldn't destroy the space either. storage::SetSpaceWriteError(42, false); NvramManager nvram2; GetSpaceInfoResponse get_space_info_response_2; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.GetSpaceInfo(get_space_info_request, &get_space_info_response_2)); EXPECT_EQ(10U, get_space_info_response_2.size); } TEST_F(NvramManagerTest, DisableCreate_Success) { NvramManager nvram; // Issue a successful disable create request. DisableCreateRequest disable_create_request; DisableCreateResponse disable_create_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.DisableCreate(disable_create_request, &disable_create_response)); // Make sure space creation request fail afterwards. CreateSpaceRequest create_space_request; create_space_request.index = 1; create_space_request.size = 32; ASSERT_TRUE(create_space_request.controls.Resize(1)); create_space_request.controls[0] = NV_CONTROL_BOOT_WRITE_LOCK; CreateSpaceResponse create_space_response; EXPECT_EQ(NV_RESULT_OPERATION_DISABLED, nvram.CreateSpace(create_space_request, &create_space_response)); // Redundant requests to disable creation are OK. EXPECT_EQ(NV_RESULT_SUCCESS, nvram.DisableCreate(disable_create_request, &disable_create_response)); // Space creation should remain disabled even after a reboot. NvramManager nvram2; EXPECT_EQ(NV_RESULT_OPERATION_DISABLED, nvram2.CreateSpace(create_space_request, &create_space_response)); } TEST_F(NvramManagerTest, DisableCreate_WriteError) { // Make header writes fail. storage::SetHeaderWriteError(true); NvramManager nvram; // The disable request should fail. DisableCreateRequest disable_create_request; DisableCreateResponse disable_create_response; EXPECT_EQ( NV_RESULT_INTERNAL_ERROR, nvram.DisableCreate(disable_create_request, &disable_create_response)); // We should still be able to create spaces after clearing the error. storage::SetHeaderWriteError(false); CreateSpaceRequest create_space_request; create_space_request.index = 1; create_space_request.size = 32; ASSERT_TRUE(create_space_request.controls.Resize(1)); create_space_request.controls[0] = NV_CONTROL_BOOT_WRITE_LOCK; CreateSpaceResponse create_space_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.CreateSpace(create_space_request, &create_space_response)); } TEST_F(NvramManagerTest, WriteSpace_SpaceAbsent) { NvramManager nvram; // Attempt to write a non-existing space. WriteSpaceRequest write_space_request; write_space_request.index = 17; ASSERT_TRUE(write_space_request.buffer.Assign("0123456789", 10)); WriteSpaceResponse write_space_response; EXPECT_EQ(NV_RESULT_SPACE_DOES_NOT_EXIST, nvram.WriteSpace(write_space_request, &write_space_response)); } TEST_F(NvramManagerTest, WriteSpace_Success) { // Set up an NVRAM space. NvramSpace space; ASSERT_TRUE(space.contents.Resize(10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(17, space)); SetupHeader(NvramHeader::kVersion, 17); NvramManager nvram; // Write the space. WriteSpaceRequest write_space_request; write_space_request.index = 17; ASSERT_TRUE(write_space_request.buffer.Assign("0123456789", 10)); WriteSpaceResponse write_space_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.WriteSpace(write_space_request, &write_space_response)); // Read back the space and compare contents. ReadAndCompareSpaceData(&nvram, 17, "0123456789", 10); // The data should persist even after a reboot. NvramManager nvram2; ReadAndCompareSpaceData(&nvram2, 17, "0123456789", 10); } TEST_F(NvramManagerTest, WriteSpace_ExcessData) { // Set up an NVRAM space. NvramSpace space; ASSERT_TRUE(space.contents.Resize(10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(17, space)); SetupHeader(NvramHeader::kVersion, 17); NvramManager nvram; // Write the space. WriteSpaceRequest write_space_request; write_space_request.index = 17; ASSERT_TRUE(write_space_request.buffer.Assign("0123456789abcdef", 16)); WriteSpaceResponse write_space_response; EXPECT_EQ(NV_RESULT_INVALID_PARAMETER, nvram.WriteSpace(write_space_request, &write_space_response)); } TEST_F(NvramManagerTest, WriteSpace_ShortData) { // Set up an NVRAM space. NvramSpace space; ASSERT_TRUE(space.contents.Resize(10)); memset(space.contents.data(), 'X', space.contents.size()); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(17, space)); SetupHeader(NvramHeader::kVersion, 17); NvramManager nvram; // Write the space. WriteSpaceRequest write_space_request; write_space_request.index = 17; ASSERT_TRUE(write_space_request.buffer.Assign("01234", 5)); WriteSpaceResponse write_space_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.WriteSpace(write_space_request, &write_space_response)); // Read back the space data and verify that the missing content bytes have // been set to 0. const uint8_t kExpectedContents[] = {'0', '1', '2', '3', '4', 0, 0, 0, 0, 0}; ReadAndCompareSpaceData(&nvram, 17, kExpectedContents, 10); } TEST_F(NvramManagerTest, WriteSpace_WriteExtend) { // Set up an NVRAM space. NvramSpace space; space.controls = (1 << NV_CONTROL_WRITE_EXTEND); ASSERT_TRUE(space.contents.Resize(32)); memset(space.contents.data(), 0, space.contents.size()); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(17, space)); SetupHeader(NvramHeader::kVersion, 17); NvramManager nvram; // Write the space. WriteSpaceRequest write_space_request; write_space_request.index = 17; ASSERT_TRUE(write_space_request.buffer.Assign("data", 4)); WriteSpaceResponse write_space_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.WriteSpace(write_space_request, &write_space_response)); // Read back the space data and verify the hash. const uint8_t kExpectedContents[] = { 0xee, 0x84, 0x52, 0x88, 0xbb, 0x60, 0x7e, 0x02, 0xfd, 0xfb, 0x31, 0x95, 0x3a, 0x77, 0x23, 0xcf, 0x67, 0xea, 0x6e, 0x2d, 0xd7, 0xdb, 0x8c, 0xb4, 0xe4, 0xd2, 0xfd, 0xb4, 0x76, 0x7a, 0x67, 0x89, }; ReadAndCompareSpaceData(&nvram, 17, kExpectedContents, 32); } TEST_F(NvramManagerTest, WriteSpace_WriteExtendShortSpace) { // Set up an NVRAM space. NvramSpace space; space.controls = (1 << NV_CONTROL_WRITE_EXTEND); ASSERT_TRUE(space.contents.Resize(16)); memset(space.contents.data(), 0, space.contents.size()); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(17, space)); SetupHeader(NvramHeader::kVersion, 17); NvramManager nvram; // Write the space. WriteSpaceRequest write_space_request; write_space_request.index = 17; ASSERT_TRUE(write_space_request.buffer.Assign("data", 4)); WriteSpaceResponse write_space_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.WriteSpace(write_space_request, &write_space_response)); // Read back the space data and verify the truncated hash. const uint8_t kExpectedContents[] = { 0x24, 0x2a, 0xbb, 0x36, 0x10, 0x37, 0x92, 0x3f, 0x7d, 0x7d, 0x92, 0x3a, 0x16, 0x65, 0xd2, 0xa2, }; ReadAndCompareSpaceData(&nvram, 17, kExpectedContents, 16); } TEST_F(NvramManagerTest, WriteSpace_WriteExtendLongSpace) { // Set up an NVRAM space. NvramSpace space; space.controls = (1 << NV_CONTROL_WRITE_EXTEND); ASSERT_TRUE(space.contents.Resize(33)); memset(space.contents.data(), 'X', space.contents.size()); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(17, space)); SetupHeader(NvramHeader::kVersion, 17); NvramManager nvram; // Write the space. WriteSpaceRequest write_space_request; write_space_request.index = 17; ASSERT_TRUE(write_space_request.buffer.Assign("data", 4)); WriteSpaceResponse write_space_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.WriteSpace(write_space_request, &write_space_response)); // Read back the space data and verify the hash and trailing 0 bytes. const uint8_t kExpectedContents[] = { 0x99, 0xb8, 0x5f, 0xd0, 0xf7, 0x9b, 0x17, 0x2e, 0x0e, 0x58, 0x3d, 0x3c, 0x9a, 0x29, 0xa3, 0xaf, 0x0a, 0x4c, 0x68, 0x97, 0x72, 0x8c, 0x0c, 0xa4, 0x37, 0xad, 0x39, 0xf3, 0x8c, 0x6e, 0x64, 0xd7, 0x00, }; ReadAndCompareSpaceData(&nvram, 17, kExpectedContents, 33); } TEST_F(NvramManagerTest, WriteSpace_AuthorizationFailure) { // Set up an NVRAM space. NvramSpace space; space.controls = (1 << NV_CONTROL_WRITE_AUTHORIZATION); ASSERT_TRUE(space.contents.Resize(10)); const char kAuthorizationValue[] = "secret"; ASSERT_TRUE(space.authorization_value.Assign(kAuthorizationValue, sizeof(kAuthorizationValue))); ASSERT_TRUE(space.contents.Assign("0123456789", 10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(17, space)); SetupHeader(NvramHeader::kVersion, 17); NvramManager nvram; // Attempt a write with the wrong authorization value. WriteSpaceRequest write_space_request; write_space_request.index = 17; ASSERT_TRUE(write_space_request.buffer.Assign("data", 4)); WriteSpaceResponse write_space_response; EXPECT_EQ(NV_RESULT_ACCESS_DENIED, nvram.WriteSpace(write_space_request, &write_space_response)); // The previous data should remain effective. ReadAndCompareSpaceData(&nvram, 17, "0123456789", 10); } TEST_F(NvramManagerTest, WriteSpace_WriteError) { // Set up an NVRAM space. NvramSpace space; ASSERT_TRUE(space.contents.Assign("0123456789", 10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(17, space)); SetupHeader(NvramHeader::kVersion, 17); NvramManager nvram; storage::SetSpaceWriteError(17, true); // Attempt a write, which should fail. WriteSpaceRequest write_space_request; write_space_request.index = 17; ASSERT_TRUE(write_space_request.buffer.Assign("data", 4)); WriteSpaceResponse write_space_response; EXPECT_EQ(NV_RESULT_INTERNAL_ERROR, nvram.WriteSpace(write_space_request, &write_space_response)); // The previous data should remain effective. ReadAndCompareSpaceData(&nvram, 17, "0123456789", 10); } TEST_F(NvramManagerTest, ReadSpace_SpaceAbsent) { NvramManager nvram; // Attempt a read from a space that doesn't exist. ReadSpaceRequest read_space_request; read_space_request.index = 17; ReadSpaceResponse read_space_response; EXPECT_EQ(NV_RESULT_SPACE_DOES_NOT_EXIST, nvram.ReadSpace(read_space_request, &read_space_response)); } TEST_F(NvramManagerTest, ReadSpace_AuthorizationFailure) { // Set up an NVRAM space. NvramSpace space; space.controls = (1 << NV_CONTROL_READ_AUTHORIZATION); ASSERT_TRUE(space.contents.Resize(10)); const char kAuthorizationValue[] = "secret"; ASSERT_TRUE(space.authorization_value.Assign(kAuthorizationValue, sizeof(kAuthorizationValue))); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(17, space)); SetupHeader(NvramHeader::kVersion, 17); NvramManager nvram; // Attempt a read from the space. ReadSpaceRequest read_space_request; read_space_request.index = 17; ReadSpaceResponse read_space_response; EXPECT_EQ(NV_RESULT_ACCESS_DENIED, nvram.ReadSpace(read_space_request, &read_space_response)); EXPECT_EQ(0U, read_space_response.buffer.size()); } TEST_F(NvramManagerTest, LockSpaceWrite_SpaceAbsent) { NvramManager nvram; // Attempt to lock a space that doesn't exist. LockSpaceWriteRequest lock_space_write_request; lock_space_write_request.index = 17; LockSpaceWriteResponse lock_space_write_response; EXPECT_EQ(NV_RESULT_SPACE_DOES_NOT_EXIST, nvram.LockSpaceWrite(lock_space_write_request, &lock_space_write_response)); } TEST_F(NvramManagerTest, LockSpaceWrite_AuthorizationFailure) { // Set up an NVRAM space. NvramSpace space; space.controls = (1 << NV_CONTROL_PERSISTENT_WRITE_LOCK) | (1 << NV_CONTROL_WRITE_AUTHORIZATION); ASSERT_TRUE(space.contents.Resize(10)); const char kAuthorizationValue[] = "secret"; ASSERT_TRUE(space.authorization_value.Assign(kAuthorizationValue, sizeof(kAuthorizationValue))); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(17, space)); SetupHeader(NvramHeader::kVersion, 17); NvramManager nvram; // Attempt to lock a space without valid authentication. LockSpaceWriteRequest lock_space_write_request; lock_space_write_request.index = 17; LockSpaceWriteResponse lock_space_write_response; EXPECT_EQ(NV_RESULT_ACCESS_DENIED, nvram.LockSpaceWrite(lock_space_write_request, &lock_space_write_response)); } TEST_F(NvramManagerTest, LockSpaceWrite_SuccessPersistent) { // Set up an NVRAM space. NvramSpace space; space.controls = (1 << NV_CONTROL_PERSISTENT_WRITE_LOCK); ASSERT_TRUE(space.contents.Resize(10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(17, space)); SetupHeader(NvramHeader::kVersion, 17); NvramManager nvram; // Lock the space. LockSpaceWriteRequest lock_space_write_request; lock_space_write_request.index = 17; LockSpaceWriteResponse lock_space_write_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.LockSpaceWrite(lock_space_write_request, &lock_space_write_response)); // Writing should fail now. WriteSpaceRequest write_space_request; write_space_request.index = 17; ASSERT_TRUE(write_space_request.buffer.Assign("data", 4)); WriteSpaceResponse write_space_response; EXPECT_EQ(NV_RESULT_OPERATION_DISABLED, nvram.WriteSpace(write_space_request, &write_space_response)); // The lock should be persistent, so writing should fail after reboot. NvramManager nvram2; EXPECT_EQ(NV_RESULT_OPERATION_DISABLED, nvram2.WriteSpace(write_space_request, &write_space_response)); } TEST_F(NvramManagerTest, LockSpaceWrite_SuccessBoot) { // Set up an NVRAM space. NvramSpace space; space.controls = (1 << NV_CONTROL_BOOT_WRITE_LOCK); ASSERT_TRUE(space.contents.Assign("01234567890", 10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(17, space)); SetupHeader(NvramHeader::kVersion, 17); NvramManager nvram; // Lock the space. LockSpaceWriteRequest lock_space_write_request; lock_space_write_request.index = 17; LockSpaceWriteResponse lock_space_write_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.LockSpaceWrite(lock_space_write_request, &lock_space_write_response)); // Writing should fail now. WriteSpaceRequest write_space_request; write_space_request.index = 17; ASSERT_TRUE(write_space_request.buffer.Assign("newcontent", 10)); WriteSpaceResponse write_space_response; EXPECT_EQ(NV_RESULT_OPERATION_DISABLED, nvram.WriteSpace(write_space_request, &write_space_response)); // We configured a per-boot lock, so writing should succeed after reboot. NvramManager nvram2; EXPECT_EQ(NV_RESULT_SUCCESS, nvram2.WriteSpace(write_space_request, &write_space_response)); ReadAndCompareSpaceData(&nvram2, 17, "newcontent", 10); } TEST_F(NvramManagerTest, LockSpaceWrite_NotLockable) { // Set up an NVRAM space. NvramSpace space; ASSERT_TRUE(space.contents.Resize(10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(17, space)); SetupHeader(NvramHeader::kVersion, 17); NvramManager nvram; // Attempt to lock a space without valid authentication. LockSpaceWriteRequest lock_space_write_request; lock_space_write_request.index = 17; LockSpaceWriteResponse lock_space_write_response; EXPECT_EQ(NV_RESULT_INVALID_PARAMETER, nvram.LockSpaceWrite(lock_space_write_request, &lock_space_write_response)); } TEST_F(NvramManagerTest, LockSpaceRead_SpaceAbsent) { NvramManager nvram; // Attempt to lock a non-existing space. LockSpaceReadRequest lock_space_read_request; lock_space_read_request.index = 17; LockSpaceReadResponse lock_space_read_response; EXPECT_EQ( NV_RESULT_SPACE_DOES_NOT_EXIST, nvram.LockSpaceRead(lock_space_read_request, &lock_space_read_response)); } TEST_F(NvramManagerTest, LockSpaceRead_AuthorizationFailure) { // Set up an NVRAM space. NvramSpace space; space.controls = (1 << NV_CONTROL_BOOT_READ_LOCK) | (1 << NV_CONTROL_READ_AUTHORIZATION); ASSERT_TRUE(space.contents.Resize(10)); const char kAuthorizationValue[] = "secret"; ASSERT_TRUE(space.authorization_value.Assign(kAuthorizationValue, sizeof(kAuthorizationValue))); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(17, space)); SetupHeader(NvramHeader::kVersion, 17); NvramManager nvram; // Attempt to lock a space without valid authorization. LockSpaceReadRequest lock_space_read_request; lock_space_read_request.index = 17; LockSpaceReadResponse lock_space_read_response; EXPECT_EQ( NV_RESULT_ACCESS_DENIED, nvram.LockSpaceRead(lock_space_read_request, &lock_space_read_response)); } TEST_F(NvramManagerTest, LockSpaceRead_Success) { // Set up an NVRAM space. NvramSpace space; space.controls = (1 << NV_CONTROL_BOOT_READ_LOCK); ASSERT_TRUE(space.contents.Assign("0123456789", 10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(17, space)); SetupHeader(NvramHeader::kVersion, 17); NvramManager nvram; // Lock the space. LockSpaceReadRequest lock_space_read_request; lock_space_read_request.index = 17; LockSpaceReadResponse lock_space_read_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.LockSpaceRead(lock_space_read_request, &lock_space_read_response)); // Read requests should fail now. ReadSpaceRequest read_space_request; read_space_request.index = 17; ReadSpaceResponse read_space_response; EXPECT_EQ(NV_RESULT_OPERATION_DISABLED, nvram.ReadSpace(read_space_request, &read_space_response)); EXPECT_EQ(0U, read_space_response.buffer.size()); // This is a non-persistent lock, so reads should work again after a reboot. NvramManager nvram2; ReadAndCompareSpaceData(&nvram2, 17, "0123456789", 10); } TEST_F(NvramManagerTest, LockSpaceRead_NotLockable) { // Set up an NVRAM space. NvramSpace space; ASSERT_TRUE(space.contents.Resize(10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(17, space)); SetupHeader(NvramHeader::kVersion, 17); NvramManager nvram; // Attempt to lock a space without valid authorization. LockSpaceReadRequest lock_space_read_request; lock_space_read_request.index = 17; LockSpaceReadResponse lock_space_read_response; EXPECT_EQ( NV_RESULT_INVALID_PARAMETER, nvram.LockSpaceRead(lock_space_read_request, &lock_space_read_response)); } TEST_F(NvramManagerTest, WipeStorage_Success) { // Set up an NVRAM space. NvramSpace space; ASSERT_TRUE(space.contents.Resize(10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(17, space)); SetupHeader(NvramHeader::kVersion, 17); // Check that the space is visible. NvramManager nvram; GetSpaceInfoRequest get_space_info_request; get_space_info_request.index = 17; GetSpaceInfoResponse get_space_info_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.GetSpaceInfo(get_space_info_request, &get_space_info_response)); EXPECT_EQ(10U, get_space_info_response.size); // Request a wipe. WipeStorageRequest wipe_storage_request; WipeStorageResponse wipe_storage_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.WipeStorage(wipe_storage_request, &wipe_storage_response)); // The space should no longer be declared. GetInfoRequest get_info_request; GetInfoResponse get_info_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.GetInfo(get_info_request, &get_info_response)); EXPECT_EQ(0U, get_info_response.space_list.size()); // Accessing the space should fail. EXPECT_EQ( NV_RESULT_SPACE_DOES_NOT_EXIST, nvram.GetSpaceInfo(get_space_info_request, &get_space_info_response)); } TEST_F(NvramManagerTest, WipeStorage_Abort) { // Set up two pre-existing spaces and a matching header. NvramSpace space; ASSERT_TRUE(space.contents.Resize(10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(1, space)); ASSERT_TRUE(space.contents.Resize(20)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(2, space)); NvramHeader header; header.version = NvramHeader::kVersion; ASSERT_TRUE(header.allocated_indices.Resize(2)); header.allocated_indices[0] = 1; header.allocated_indices[1] = 2; ASSERT_EQ(storage::Status::kSuccess, persistence::StoreHeader(header)); // Check that the spaces are visible. NvramManager nvram; GetInfoRequest get_info_request; GetInfoResponse get_info_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.GetInfo(get_info_request, &get_info_response)); EXPECT_EQ(2U, get_info_response.space_list.size()); int space_mask = 0; for (size_t i = 0; i < get_info_response.space_list.size(); ++i) { space_mask |= (1 << get_info_response.space_list[i]); } EXPECT_EQ(0x6, space_mask); // Set things up so the deletion request for the second space fails. storage::SetSpaceWriteError(2, true); // The wipe request should fail now. WipeStorageRequest wipe_storage_request; WipeStorageResponse wipe_storage_response; EXPECT_EQ(NV_RESULT_INTERNAL_ERROR, nvram.WipeStorage(wipe_storage_request, &wipe_storage_response)); // New wipe attempt with a fresh instance after clearing the error. storage::SetSpaceWriteError(2, false); NvramManager nvram2; EXPECT_EQ(NV_RESULT_SUCCESS, nvram2.WipeStorage(wipe_storage_request, &wipe_storage_response)); // No spaces should remain. EXPECT_EQ(NV_RESULT_SUCCESS, nvram2.GetInfo(get_info_request, &get_info_response)); EXPECT_EQ(0U, get_info_response.space_list.size()); } TEST_F(NvramManagerTest, WipeStorage_Disable) { // Set up an NVRAM space. NvramSpace space; ASSERT_TRUE(space.contents.Resize(10)); ASSERT_EQ(storage::Status::kSuccess, persistence::StoreSpace(17, space)); SetupHeader(NvramHeader::kVersion, 17); NvramManager nvram; // Disable wiping. DisableWipeRequest disable_wipe_request; DisableWipeResponse disable_wipe_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.DisableWipe(disable_wipe_request, &disable_wipe_response)); // A wipe request should fail. WipeStorageRequest wipe_storage_request; WipeStorageResponse wipe_storage_response; EXPECT_EQ(NV_RESULT_OPERATION_DISABLED, nvram.WipeStorage(wipe_storage_request, &wipe_storage_response)); // The space should remain declared. GetInfoRequest get_info_request; GetInfoResponse get_info_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.GetInfo(get_info_request, &get_info_response)); ASSERT_EQ(1U, get_info_response.space_list.size()); EXPECT_EQ(17U, get_info_response.space_list[0]); // The space data should remain present. GetSpaceInfoRequest get_space_info_request; get_space_info_request.index = 17; GetSpaceInfoResponse get_space_info_response; EXPECT_EQ(NV_RESULT_SUCCESS, nvram.GetSpaceInfo(get_space_info_request, &get_space_info_response)); EXPECT_EQ(10U, get_space_info_response.size); } } // namespace } // namespace nvram