xref: /aosp_15_r20/external/pigweed/pw_kvs/flash_partition_test.cc (revision 61c4878ac05f98d0ceed94b57d316916de578985)
1*61c4878aSAndroid Build Coastguard Worker // Copyright 2020 The Pigweed Authors
2*61c4878aSAndroid Build Coastguard Worker //
3*61c4878aSAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License"); you may not
4*61c4878aSAndroid Build Coastguard Worker // use this file except in compliance with the License. You may obtain a copy of
5*61c4878aSAndroid Build Coastguard Worker // the License at
6*61c4878aSAndroid Build Coastguard Worker //
7*61c4878aSAndroid Build Coastguard Worker //     https://www.apache.org/licenses/LICENSE-2.0
8*61c4878aSAndroid Build Coastguard Worker //
9*61c4878aSAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
10*61c4878aSAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
11*61c4878aSAndroid Build Coastguard Worker // WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
12*61c4878aSAndroid Build Coastguard Worker // License for the specific language governing permissions and limitations under
13*61c4878aSAndroid Build Coastguard Worker // the License.
14*61c4878aSAndroid Build Coastguard Worker 
15*61c4878aSAndroid Build Coastguard Worker #include <algorithm>
16*61c4878aSAndroid Build Coastguard Worker 
17*61c4878aSAndroid Build Coastguard Worker #include "pw_kvs/flash_memory.h"
18*61c4878aSAndroid Build Coastguard Worker #include "pw_kvs/flash_test_partition.h"
19*61c4878aSAndroid Build Coastguard Worker #include "pw_kvs_private/config.h"
20*61c4878aSAndroid Build Coastguard Worker #include "pw_log/log.h"
21*61c4878aSAndroid Build Coastguard Worker #include "pw_span/span.h"
22*61c4878aSAndroid Build Coastguard Worker #include "pw_unit_test/framework.h"
23*61c4878aSAndroid Build Coastguard Worker 
24*61c4878aSAndroid Build Coastguard Worker namespace pw::kvs::PartitionTest {
25*61c4878aSAndroid Build Coastguard Worker namespace {
26*61c4878aSAndroid Build Coastguard Worker 
27*61c4878aSAndroid Build Coastguard Worker #if !defined(PW_FLASH_TEST_ITERATIONS) || (PW_FLASH_TEST_ITERATIONS <= 0)
28*61c4878aSAndroid Build Coastguard Worker #error PW_FLASH_TEST_ITERATIONS must be defined and > 0
29*61c4878aSAndroid Build Coastguard Worker #endif  // PW_FLASH_TEST_ITERATIONS
30*61c4878aSAndroid Build Coastguard Worker 
31*61c4878aSAndroid Build Coastguard Worker #if !defined(PW_FLASH_TEST_WRITE_SIZE) || (PW_FLASH_TEST_WRITE_SIZE <= 0)
32*61c4878aSAndroid Build Coastguard Worker #error PW_FLASH_TEST_WRITE_SIZE must be defined and > 0
33*61c4878aSAndroid Build Coastguard Worker #endif  // PW_FLASH_TEST_WRITE_SIZE
34*61c4878aSAndroid Build Coastguard Worker 
35*61c4878aSAndroid Build Coastguard Worker constexpr size_t kTestIterations = PW_FLASH_TEST_ITERATIONS;
36*61c4878aSAndroid Build Coastguard Worker constexpr size_t kTestWriteSize = PW_FLASH_TEST_WRITE_SIZE;
37*61c4878aSAndroid Build Coastguard Worker 
38*61c4878aSAndroid Build Coastguard Worker size_t error_count = 0;
39*61c4878aSAndroid Build Coastguard Worker 
WriteData(FlashPartition & partition,uint8_t fill_byte)40*61c4878aSAndroid Build Coastguard Worker void WriteData(FlashPartition& partition, uint8_t fill_byte) {
41*61c4878aSAndroid Build Coastguard Worker   uint8_t test_data[kMaxFlashAlignment];
42*61c4878aSAndroid Build Coastguard Worker   memset(test_data, fill_byte, sizeof(test_data));
43*61c4878aSAndroid Build Coastguard Worker 
44*61c4878aSAndroid Build Coastguard Worker   const size_t write_size =
45*61c4878aSAndroid Build Coastguard Worker       std::max(kTestWriteSize, partition.alignment_bytes());
46*61c4878aSAndroid Build Coastguard Worker 
47*61c4878aSAndroid Build Coastguard Worker   ASSERT_EQ(OkStatus(), partition.Erase(0, partition.sector_count()));
48*61c4878aSAndroid Build Coastguard Worker 
49*61c4878aSAndroid Build Coastguard Worker   const size_t chunks_per_sector = partition.sector_size_bytes() / write_size;
50*61c4878aSAndroid Build Coastguard Worker 
51*61c4878aSAndroid Build Coastguard Worker   // Fill partition sector by sector. Fill the sector with an integer number
52*61c4878aSAndroid Build Coastguard Worker   // of write_size-size chunks. If the sector is not evenly divisible by
53*61c4878aSAndroid Build Coastguard Worker   // write_size-size, the remainder is not written.
54*61c4878aSAndroid Build Coastguard Worker   for (size_t sector_index = 0; sector_index < partition.sector_count();
55*61c4878aSAndroid Build Coastguard Worker        sector_index++) {
56*61c4878aSAndroid Build Coastguard Worker     FlashPartition::Address address =
57*61c4878aSAndroid Build Coastguard Worker         sector_index * partition.sector_size_bytes();
58*61c4878aSAndroid Build Coastguard Worker 
59*61c4878aSAndroid Build Coastguard Worker     for (size_t chunk_index = 0; chunk_index < chunks_per_sector;
60*61c4878aSAndroid Build Coastguard Worker          chunk_index++) {
61*61c4878aSAndroid Build Coastguard Worker       StatusWithSize status =
62*61c4878aSAndroid Build Coastguard Worker           partition.Write(address, as_bytes(span(test_data, write_size)));
63*61c4878aSAndroid Build Coastguard Worker       ASSERT_EQ(OkStatus(), status.status());
64*61c4878aSAndroid Build Coastguard Worker       ASSERT_EQ(write_size, status.size());
65*61c4878aSAndroid Build Coastguard Worker       address += write_size;
66*61c4878aSAndroid Build Coastguard Worker     }
67*61c4878aSAndroid Build Coastguard Worker   }
68*61c4878aSAndroid Build Coastguard Worker 
69*61c4878aSAndroid Build Coastguard Worker   // Check the fill result. Use expect so the test doesn't bail on error.
70*61c4878aSAndroid Build Coastguard Worker   // Count the errors and print if any errors are found.
71*61c4878aSAndroid Build Coastguard Worker   for (size_t sector_index = 0; sector_index < partition.sector_count();
72*61c4878aSAndroid Build Coastguard Worker        sector_index++) {
73*61c4878aSAndroid Build Coastguard Worker     FlashPartition::Address address =
74*61c4878aSAndroid Build Coastguard Worker         sector_index * partition.sector_size_bytes();
75*61c4878aSAndroid Build Coastguard Worker 
76*61c4878aSAndroid Build Coastguard Worker     for (size_t chunk_index = 0; chunk_index < chunks_per_sector;
77*61c4878aSAndroid Build Coastguard Worker          chunk_index++) {
78*61c4878aSAndroid Build Coastguard Worker       memset(test_data, 0, sizeof(test_data));
79*61c4878aSAndroid Build Coastguard Worker       StatusWithSize status = partition.Read(address, write_size, test_data);
80*61c4878aSAndroid Build Coastguard Worker 
81*61c4878aSAndroid Build Coastguard Worker       EXPECT_EQ(OkStatus(), status.status());
82*61c4878aSAndroid Build Coastguard Worker       EXPECT_EQ(write_size, status.size());
83*61c4878aSAndroid Build Coastguard Worker       if (!status.ok() || (write_size != status.size())) {
84*61c4878aSAndroid Build Coastguard Worker         error_count++;
85*61c4878aSAndroid Build Coastguard Worker         PW_LOG_DEBUG("   Read Error [%s], %u of %u",
86*61c4878aSAndroid Build Coastguard Worker                      status.status().str(),
87*61c4878aSAndroid Build Coastguard Worker                      unsigned(status.size()),
88*61c4878aSAndroid Build Coastguard Worker                      unsigned(write_size));
89*61c4878aSAndroid Build Coastguard Worker         continue;
90*61c4878aSAndroid Build Coastguard Worker       }
91*61c4878aSAndroid Build Coastguard Worker 
92*61c4878aSAndroid Build Coastguard Worker       for (size_t i = 0; i < write_size; i++) {
93*61c4878aSAndroid Build Coastguard Worker         if (test_data[i] != fill_byte) {
94*61c4878aSAndroid Build Coastguard Worker           error_count++;
95*61c4878aSAndroid Build Coastguard Worker           PW_LOG_DEBUG(
96*61c4878aSAndroid Build Coastguard Worker               "   Error %u, Read compare @ address %x, got 0x%02x, "
97*61c4878aSAndroid Build Coastguard Worker               "expected 0x%02x",
98*61c4878aSAndroid Build Coastguard Worker               unsigned(error_count),
99*61c4878aSAndroid Build Coastguard Worker               unsigned(address + i),
100*61c4878aSAndroid Build Coastguard Worker               unsigned(test_data[i]),
101*61c4878aSAndroid Build Coastguard Worker               unsigned(fill_byte));
102*61c4878aSAndroid Build Coastguard Worker         }
103*61c4878aSAndroid Build Coastguard Worker       }
104*61c4878aSAndroid Build Coastguard Worker 
105*61c4878aSAndroid Build Coastguard Worker       address += write_size;
106*61c4878aSAndroid Build Coastguard Worker     }
107*61c4878aSAndroid Build Coastguard Worker   }
108*61c4878aSAndroid Build Coastguard Worker 
109*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(error_count, 0U);
110*61c4878aSAndroid Build Coastguard Worker   if (error_count != 0) {
111*61c4878aSAndroid Build Coastguard Worker     PW_LOG_ERROR("Partition test, fill '%c', %u errors found",
112*61c4878aSAndroid Build Coastguard Worker                  fill_byte,
113*61c4878aSAndroid Build Coastguard Worker                  unsigned(error_count));
114*61c4878aSAndroid Build Coastguard Worker   }
115*61c4878aSAndroid Build Coastguard Worker }
116*61c4878aSAndroid Build Coastguard Worker 
TEST(FlashPartitionTest,FillTest)117*61c4878aSAndroid Build Coastguard Worker TEST(FlashPartitionTest, FillTest) {
118*61c4878aSAndroid Build Coastguard Worker   FlashPartition& test_partition = FlashTestPartition();
119*61c4878aSAndroid Build Coastguard Worker 
120*61c4878aSAndroid Build Coastguard Worker   ASSERT_GE(kMaxFlashAlignment, test_partition.alignment_bytes());
121*61c4878aSAndroid Build Coastguard Worker 
122*61c4878aSAndroid Build Coastguard Worker   for (size_t i = 0; i < kTestIterations; i++) {
123*61c4878aSAndroid Build Coastguard Worker     PW_LOG_DEBUG("FillTest iteration %u, write '0'", unsigned(i));
124*61c4878aSAndroid Build Coastguard Worker     WriteData(test_partition, 0);
125*61c4878aSAndroid Build Coastguard Worker     PW_LOG_DEBUG("FillTest iteration %u, write '0xff'", unsigned(i));
126*61c4878aSAndroid Build Coastguard Worker     WriteData(test_partition, 0xff);
127*61c4878aSAndroid Build Coastguard Worker     PW_LOG_DEBUG("FillTest iteration %u, write '0x55'", unsigned(i));
128*61c4878aSAndroid Build Coastguard Worker     WriteData(test_partition, 0x55);
129*61c4878aSAndroid Build Coastguard Worker     PW_LOG_DEBUG("FillTest iteration %u, write '0xa3'", unsigned(i));
130*61c4878aSAndroid Build Coastguard Worker     WriteData(test_partition, 0xa3);
131*61c4878aSAndroid Build Coastguard Worker     PW_LOG_DEBUG("Completed iterations %u, Total errors %u",
132*61c4878aSAndroid Build Coastguard Worker                  unsigned(i),
133*61c4878aSAndroid Build Coastguard Worker                  unsigned(error_count));
134*61c4878aSAndroid Build Coastguard Worker   }
135*61c4878aSAndroid Build Coastguard Worker }
136*61c4878aSAndroid Build Coastguard Worker 
TEST(FlashPartitionTest,EraseTest)137*61c4878aSAndroid Build Coastguard Worker TEST(FlashPartitionTest, EraseTest) {
138*61c4878aSAndroid Build Coastguard Worker   FlashPartition& test_partition = FlashTestPartition();
139*61c4878aSAndroid Build Coastguard Worker 
140*61c4878aSAndroid Build Coastguard Worker   static const uint8_t fill_byte = 0x55;
141*61c4878aSAndroid Build Coastguard Worker   uint8_t test_data[kMaxFlashAlignment];
142*61c4878aSAndroid Build Coastguard Worker   memset(test_data, fill_byte, sizeof(test_data));
143*61c4878aSAndroid Build Coastguard Worker 
144*61c4878aSAndroid Build Coastguard Worker   ASSERT_GE(kMaxFlashAlignment, test_partition.alignment_bytes());
145*61c4878aSAndroid Build Coastguard Worker 
146*61c4878aSAndroid Build Coastguard Worker   const size_t block_size =
147*61c4878aSAndroid Build Coastguard Worker       std::min(sizeof(test_data), test_partition.sector_size_bytes());
148*61c4878aSAndroid Build Coastguard Worker   auto data_span = span(test_data, block_size);
149*61c4878aSAndroid Build Coastguard Worker 
150*61c4878aSAndroid Build Coastguard Worker   ASSERT_EQ(OkStatus(), test_partition.Erase(0, test_partition.sector_count()));
151*61c4878aSAndroid Build Coastguard Worker 
152*61c4878aSAndroid Build Coastguard Worker   // Write to the first page of each sector.
153*61c4878aSAndroid Build Coastguard Worker   for (size_t sector_index = 0; sector_index < test_partition.sector_count();
154*61c4878aSAndroid Build Coastguard Worker        sector_index++) {
155*61c4878aSAndroid Build Coastguard Worker     FlashPartition::Address address =
156*61c4878aSAndroid Build Coastguard Worker         sector_index * test_partition.sector_size_bytes();
157*61c4878aSAndroid Build Coastguard Worker 
158*61c4878aSAndroid Build Coastguard Worker     StatusWithSize status = test_partition.Write(address, as_bytes(data_span));
159*61c4878aSAndroid Build Coastguard Worker     ASSERT_EQ(OkStatus(), status.status());
160*61c4878aSAndroid Build Coastguard Worker     ASSERT_EQ(block_size, status.size());
161*61c4878aSAndroid Build Coastguard Worker   }
162*61c4878aSAndroid Build Coastguard Worker 
163*61c4878aSAndroid Build Coastguard Worker   // Preset the flag to make sure the check actually sets it.
164*61c4878aSAndroid Build Coastguard Worker   bool is_erased = true;
165*61c4878aSAndroid Build Coastguard Worker   ASSERT_EQ(OkStatus(), test_partition.IsErased(&is_erased));
166*61c4878aSAndroid Build Coastguard Worker   ASSERT_EQ(false, is_erased);
167*61c4878aSAndroid Build Coastguard Worker 
168*61c4878aSAndroid Build Coastguard Worker   ASSERT_EQ(OkStatus(), test_partition.Erase());
169*61c4878aSAndroid Build Coastguard Worker 
170*61c4878aSAndroid Build Coastguard Worker   // Preset the flag to make sure the check actually sets it.
171*61c4878aSAndroid Build Coastguard Worker   is_erased = false;
172*61c4878aSAndroid Build Coastguard Worker   ASSERT_EQ(OkStatus(), test_partition.IsErased(&is_erased));
173*61c4878aSAndroid Build Coastguard Worker   ASSERT_EQ(true, is_erased);
174*61c4878aSAndroid Build Coastguard Worker 
175*61c4878aSAndroid Build Coastguard Worker   // Read the first page of each sector and make sure it has been erased.
176*61c4878aSAndroid Build Coastguard Worker   for (size_t sector_index = 0; sector_index < test_partition.sector_count();
177*61c4878aSAndroid Build Coastguard Worker        sector_index++) {
178*61c4878aSAndroid Build Coastguard Worker     FlashPartition::Address address =
179*61c4878aSAndroid Build Coastguard Worker         sector_index * test_partition.sector_size_bytes();
180*61c4878aSAndroid Build Coastguard Worker 
181*61c4878aSAndroid Build Coastguard Worker     StatusWithSize status =
182*61c4878aSAndroid Build Coastguard Worker         test_partition.Read(address, data_span.size_bytes(), data_span.data());
183*61c4878aSAndroid Build Coastguard Worker     EXPECT_EQ(OkStatus(), status.status());
184*61c4878aSAndroid Build Coastguard Worker     EXPECT_EQ(data_span.size_bytes(), status.size());
185*61c4878aSAndroid Build Coastguard Worker 
186*61c4878aSAndroid Build Coastguard Worker     EXPECT_EQ(true, test_partition.AppearsErased(as_bytes(data_span)));
187*61c4878aSAndroid Build Coastguard Worker   }
188*61c4878aSAndroid Build Coastguard Worker }
189*61c4878aSAndroid Build Coastguard Worker 
TEST(FlashPartitionTest,AlignmentCheck)190*61c4878aSAndroid Build Coastguard Worker TEST(FlashPartitionTest, AlignmentCheck) {
191*61c4878aSAndroid Build Coastguard Worker   FlashPartition& test_partition = FlashTestPartition();
192*61c4878aSAndroid Build Coastguard Worker   const size_t alignment = test_partition.alignment_bytes();
193*61c4878aSAndroid Build Coastguard Worker   const size_t sector_size_bytes = test_partition.sector_size_bytes();
194*61c4878aSAndroid Build Coastguard Worker 
195*61c4878aSAndroid Build Coastguard Worker   EXPECT_LE(kTestWriteSize, kMaxFlashAlignment);
196*61c4878aSAndroid Build Coastguard Worker   EXPECT_GT(kTestWriteSize, 0u);
197*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(kMaxFlashAlignment % kTestWriteSize, 0U);
198*61c4878aSAndroid Build Coastguard Worker 
199*61c4878aSAndroid Build Coastguard Worker   EXPECT_LE(alignment, kMaxFlashAlignment);
200*61c4878aSAndroid Build Coastguard Worker   EXPECT_GT(alignment, 0u);
201*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(kMaxFlashAlignment % alignment, 0U);
202*61c4878aSAndroid Build Coastguard Worker 
203*61c4878aSAndroid Build Coastguard Worker   EXPECT_LE(kMaxFlashAlignment, sector_size_bytes);
204*61c4878aSAndroid Build Coastguard Worker   EXPECT_LE(sector_size_bytes % kMaxFlashAlignment, 0U);
205*61c4878aSAndroid Build Coastguard Worker }
206*61c4878aSAndroid Build Coastguard Worker 
207*61c4878aSAndroid Build Coastguard Worker #define TESTING_CHECK_FAILURES_IS_SUPPORTED 0
208*61c4878aSAndroid Build Coastguard Worker #if TESTING_CHECK_FAILURES_IS_SUPPORTED
209*61c4878aSAndroid Build Coastguard Worker // TODO(davidrogers): Ensure that this test triggers an assert.
TEST(FlashPartitionTest,BadWriteAddressAlignment)210*61c4878aSAndroid Build Coastguard Worker TEST(FlashPartitionTest, BadWriteAddressAlignment) {
211*61c4878aSAndroid Build Coastguard Worker   FlashPartition& test_partition = FlashTestPartition();
212*61c4878aSAndroid Build Coastguard Worker 
213*61c4878aSAndroid Build Coastguard Worker   // Can't get bad alignment with alignment of 1.
214*61c4878aSAndroid Build Coastguard Worker   if (test_partition.alignment_bytes() == 1) {
215*61c4878aSAndroid Build Coastguard Worker     return;
216*61c4878aSAndroid Build Coastguard Worker   }
217*61c4878aSAndroid Build Coastguard Worker 
218*61c4878aSAndroid Build Coastguard Worker   std::array<std::byte, kMaxFlashAlignment> source_data;
219*61c4878aSAndroid Build Coastguard Worker   test_partition.Write(1, source_data);
220*61c4878aSAndroid Build Coastguard Worker }
221*61c4878aSAndroid Build Coastguard Worker 
222*61c4878aSAndroid Build Coastguard Worker // TODO(davidrogers): Ensure that this test triggers an assert.
TEST(FlashPartitionTest,BadWriteSizeAlignment)223*61c4878aSAndroid Build Coastguard Worker TEST(FlashPartitionTest, BadWriteSizeAlignment) {
224*61c4878aSAndroid Build Coastguard Worker   FlashPartition& test_partition = FlashTestPartition();
225*61c4878aSAndroid Build Coastguard Worker 
226*61c4878aSAndroid Build Coastguard Worker   // Can't get bad alignment with alignment of 1.
227*61c4878aSAndroid Build Coastguard Worker   if (test_partition.alignment_bytes() == 1) {
228*61c4878aSAndroid Build Coastguard Worker     return;
229*61c4878aSAndroid Build Coastguard Worker   }
230*61c4878aSAndroid Build Coastguard Worker 
231*61c4878aSAndroid Build Coastguard Worker   std::array<std::byte, 1> source_data;
232*61c4878aSAndroid Build Coastguard Worker   test_partition.Write(0, source_data);
233*61c4878aSAndroid Build Coastguard Worker }
234*61c4878aSAndroid Build Coastguard Worker 
235*61c4878aSAndroid Build Coastguard Worker // TODO(davidrogers): Ensure that this test triggers an assert.
TEST(FlashPartitionTest,BadEraseAddressAlignment)236*61c4878aSAndroid Build Coastguard Worker TEST(FlashPartitionTest, BadEraseAddressAlignment) {
237*61c4878aSAndroid Build Coastguard Worker   FlashPartition& test_partition = FlashTestPartition();
238*61c4878aSAndroid Build Coastguard Worker 
239*61c4878aSAndroid Build Coastguard Worker   // Can't get bad alignment with sector size of 1.
240*61c4878aSAndroid Build Coastguard Worker   if (test_partition.sector_size_bytes() == 1) {
241*61c4878aSAndroid Build Coastguard Worker     return;
242*61c4878aSAndroid Build Coastguard Worker   }
243*61c4878aSAndroid Build Coastguard Worker 
244*61c4878aSAndroid Build Coastguard Worker   // Try Erase at address 1 for 1 sector.
245*61c4878aSAndroid Build Coastguard Worker   test_partition.Erase(1, 1);
246*61c4878aSAndroid Build Coastguard Worker }
247*61c4878aSAndroid Build Coastguard Worker 
248*61c4878aSAndroid Build Coastguard Worker #endif  // TESTING_CHECK_FAILURES_IS_SUPPORTED
249*61c4878aSAndroid Build Coastguard Worker 
TEST(FlashPartitionTest,IsErased)250*61c4878aSAndroid Build Coastguard Worker TEST(FlashPartitionTest, IsErased) {
251*61c4878aSAndroid Build Coastguard Worker   FlashPartition& test_partition = FlashTestPartition();
252*61c4878aSAndroid Build Coastguard Worker   const size_t write_size =
253*61c4878aSAndroid Build Coastguard Worker       std::max(kTestWriteSize, test_partition.alignment_bytes());
254*61c4878aSAndroid Build Coastguard Worker 
255*61c4878aSAndroid Build Coastguard Worker   // Make sure the partition is big enough to do this test.
256*61c4878aSAndroid Build Coastguard Worker   ASSERT_GE(test_partition.size_bytes(), 3 * kMaxFlashAlignment);
257*61c4878aSAndroid Build Coastguard Worker 
258*61c4878aSAndroid Build Coastguard Worker   ASSERT_EQ(OkStatus(), test_partition.Erase());
259*61c4878aSAndroid Build Coastguard Worker 
260*61c4878aSAndroid Build Coastguard Worker   bool is_erased = true;
261*61c4878aSAndroid Build Coastguard Worker   ASSERT_EQ(OkStatus(), test_partition.IsErased(&is_erased));
262*61c4878aSAndroid Build Coastguard Worker   ASSERT_EQ(true, is_erased);
263*61c4878aSAndroid Build Coastguard Worker 
264*61c4878aSAndroid Build Coastguard Worker   static const uint8_t fill_byte = 0x55;
265*61c4878aSAndroid Build Coastguard Worker   uint8_t test_data[kMaxFlashAlignment];
266*61c4878aSAndroid Build Coastguard Worker   memset(test_data, fill_byte, sizeof(test_data));
267*61c4878aSAndroid Build Coastguard Worker   auto data_span = span(test_data);
268*61c4878aSAndroid Build Coastguard Worker 
269*61c4878aSAndroid Build Coastguard Worker   // Write the chunk with fill byte.
270*61c4878aSAndroid Build Coastguard Worker   StatusWithSize status = test_partition.Write(write_size, as_bytes(data_span));
271*61c4878aSAndroid Build Coastguard Worker   ASSERT_EQ(OkStatus(), status.status());
272*61c4878aSAndroid Build Coastguard Worker   ASSERT_EQ(data_span.size_bytes(), status.size());
273*61c4878aSAndroid Build Coastguard Worker 
274*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(OkStatus(), test_partition.IsErased(&is_erased));
275*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(false, is_erased);
276*61c4878aSAndroid Build Coastguard Worker 
277*61c4878aSAndroid Build Coastguard Worker   // Check the chunk that was written.
278*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(OkStatus(),
279*61c4878aSAndroid Build Coastguard Worker             test_partition.IsRegionErased(
280*61c4878aSAndroid Build Coastguard Worker                 write_size, data_span.size_bytes(), &is_erased));
281*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(false, is_erased);
282*61c4878aSAndroid Build Coastguard Worker 
283*61c4878aSAndroid Build Coastguard Worker   // Check a region that starts erased but later has been written.
284*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(OkStatus(),
285*61c4878aSAndroid Build Coastguard Worker             test_partition.IsRegionErased(0, 2 * write_size, &is_erased));
286*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(false, is_erased);
287*61c4878aSAndroid Build Coastguard Worker 
288*61c4878aSAndroid Build Coastguard Worker   // Check erased for a region smaller than kMaxFlashAlignment. This has been a
289*61c4878aSAndroid Build Coastguard Worker   // bug in the past.
290*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(OkStatus(),
291*61c4878aSAndroid Build Coastguard Worker             test_partition.IsRegionErased(0, write_size, &is_erased));
292*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(true, is_erased);
293*61c4878aSAndroid Build Coastguard Worker }
294*61c4878aSAndroid Build Coastguard Worker 
TEST(FlashPartitionTest,EndOfWrittenData)295*61c4878aSAndroid Build Coastguard Worker TEST(FlashPartitionTest, EndOfWrittenData) {
296*61c4878aSAndroid Build Coastguard Worker   FlashPartition& test_partition = FlashTestPartition();
297*61c4878aSAndroid Build Coastguard Worker 
298*61c4878aSAndroid Build Coastguard Worker   // Make sure the partition is big enough to do this test.
299*61c4878aSAndroid Build Coastguard Worker   ASSERT_GE(test_partition.size_bytes(), 3 * kMaxFlashAlignment);
300*61c4878aSAndroid Build Coastguard Worker 
301*61c4878aSAndroid Build Coastguard Worker   ASSERT_EQ(OkStatus(), test_partition.Erase());
302*61c4878aSAndroid Build Coastguard Worker 
303*61c4878aSAndroid Build Coastguard Worker   StatusWithSize end_sws = test_partition.EndOfWrittenData();
304*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(OkStatus(), end_sws.status());
305*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(end_sws.size(), 0U);
306*61c4878aSAndroid Build Coastguard Worker 
307*61c4878aSAndroid Build Coastguard Worker   static const uint8_t fill_byte = 0x55;
308*61c4878aSAndroid Build Coastguard Worker   uint8_t test_data[kMaxFlashAlignment];
309*61c4878aSAndroid Build Coastguard Worker   memset(test_data, fill_byte, sizeof(test_data));
310*61c4878aSAndroid Build Coastguard Worker   auto data_span = span(test_data);
311*61c4878aSAndroid Build Coastguard Worker 
312*61c4878aSAndroid Build Coastguard Worker   // Write the chunk with fill byte at start of partition.
313*61c4878aSAndroid Build Coastguard Worker   StatusWithSize write_sws = test_partition.Write(0, as_bytes(data_span));
314*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(OkStatus(), write_sws.status());
315*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(data_span.size_bytes(), write_sws.size());
316*61c4878aSAndroid Build Coastguard Worker 
317*61c4878aSAndroid Build Coastguard Worker   end_sws = test_partition.EndOfWrittenData();
318*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(OkStatus(), end_sws.status());
319*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(end_sws.size(), data_span.size_bytes());
320*61c4878aSAndroid Build Coastguard Worker 
321*61c4878aSAndroid Build Coastguard Worker   // Write alignment number of bytes mid-partition.
322*61c4878aSAndroid Build Coastguard Worker   ASSERT_EQ(OkStatus(), test_partition.Erase());
323*61c4878aSAndroid Build Coastguard Worker   write_sws = test_partition.Write(
324*61c4878aSAndroid Build Coastguard Worker       kMaxFlashAlignment,
325*61c4878aSAndroid Build Coastguard Worker       as_bytes(data_span.first(test_partition.alignment_bytes())));
326*61c4878aSAndroid Build Coastguard Worker   ASSERT_EQ(OkStatus(), write_sws.status());
327*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(write_sws.size(), test_partition.alignment_bytes());
328*61c4878aSAndroid Build Coastguard Worker 
329*61c4878aSAndroid Build Coastguard Worker   end_sws = test_partition.EndOfWrittenData();
330*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(OkStatus(), end_sws.status());
331*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(kMaxFlashAlignment + test_partition.alignment_bytes(),
332*61c4878aSAndroid Build Coastguard Worker             end_sws.size());
333*61c4878aSAndroid Build Coastguard Worker 
334*61c4878aSAndroid Build Coastguard Worker   // Write full partition
335*61c4878aSAndroid Build Coastguard Worker   ASSERT_EQ(OkStatus(), test_partition.Erase());
336*61c4878aSAndroid Build Coastguard Worker   size_t remaining_bytes = test_partition.size_bytes();
337*61c4878aSAndroid Build Coastguard Worker   size_t offset = 0;
338*61c4878aSAndroid Build Coastguard Worker   while (remaining_bytes > 0) {
339*61c4878aSAndroid Build Coastguard Worker     size_t write_size = std::min(remaining_bytes, data_span.size_bytes());
340*61c4878aSAndroid Build Coastguard Worker     write_sws =
341*61c4878aSAndroid Build Coastguard Worker         test_partition.Write(offset, as_bytes(data_span.first(write_size)));
342*61c4878aSAndroid Build Coastguard Worker     EXPECT_EQ(OkStatus(), write_sws.status());
343*61c4878aSAndroid Build Coastguard Worker     EXPECT_EQ(write_size, write_sws.size());
344*61c4878aSAndroid Build Coastguard Worker     remaining_bytes -= write_size;
345*61c4878aSAndroid Build Coastguard Worker     offset += write_size;
346*61c4878aSAndroid Build Coastguard Worker   }
347*61c4878aSAndroid Build Coastguard Worker 
348*61c4878aSAndroid Build Coastguard Worker   end_sws = test_partition.EndOfWrittenData();
349*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(OkStatus(), end_sws.status());
350*61c4878aSAndroid Build Coastguard Worker   EXPECT_EQ(test_partition.size_bytes(), end_sws.size());
351*61c4878aSAndroid Build Coastguard Worker }
352*61c4878aSAndroid Build Coastguard Worker 
353*61c4878aSAndroid Build Coastguard Worker }  // namespace
354*61c4878aSAndroid Build Coastguard Worker }  // namespace pw::kvs::PartitionTest
355