You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
149 lines
4.6 KiB
149 lines
4.6 KiB
// Copyright 2020 The Pigweed Authors
|
|
//
|
|
// 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
|
|
//
|
|
// https://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.
|
|
|
|
#define PW_LOG_MODULE_NAME "KVS"
|
|
#define PW_LOG_LEVEL PW_KVS_LOG_LEVEL
|
|
|
|
#include "pw_kvs/fake_flash_memory.h"
|
|
|
|
#include "pw_kvs_private/config.h"
|
|
#include "pw_log/log.h"
|
|
|
|
namespace pw::kvs {
|
|
|
|
Status FlashError::Check(std::span<FlashError> errors,
|
|
FlashMemory::Address address,
|
|
size_t size) {
|
|
for (auto& error : errors) {
|
|
if (Status status = error.Check(address, size); !status.ok()) {
|
|
return status;
|
|
}
|
|
}
|
|
|
|
return OkStatus();
|
|
}
|
|
|
|
Status FlashError::Check(FlashMemory::Address start_address, size_t size) {
|
|
// Check if the event overlaps with this address range.
|
|
if (begin_ != kAnyAddress &&
|
|
(start_address >= end_ || (start_address + size) <= begin_)) {
|
|
return OkStatus();
|
|
}
|
|
|
|
if (delay_ > 0u) {
|
|
delay_ -= 1;
|
|
return OkStatus();
|
|
}
|
|
|
|
if (remaining_ == 0u) {
|
|
return OkStatus();
|
|
}
|
|
|
|
if (remaining_ != kAlways) {
|
|
remaining_ -= 1;
|
|
}
|
|
|
|
return status_;
|
|
}
|
|
|
|
Status FakeFlashMemory::Erase(Address address, size_t num_sectors) {
|
|
if (address % sector_size_bytes() != 0) {
|
|
PW_LOG_ERROR(
|
|
"Attempted to erase sector at non-sector aligned boundary; address %x",
|
|
unsigned(address));
|
|
return Status::InvalidArgument();
|
|
}
|
|
const size_t sector_id = address / sector_size_bytes();
|
|
if (address / sector_size_bytes() + num_sectors > sector_count()) {
|
|
PW_LOG_ERROR(
|
|
"Tried to erase a sector at an address past flash end; "
|
|
"address: %x, sector implied: %u",
|
|
unsigned(address),
|
|
unsigned(sector_id));
|
|
return Status::OutOfRange();
|
|
}
|
|
|
|
std::memset(
|
|
&buffer_[address], int(kErasedValue), sector_size_bytes() * num_sectors);
|
|
return OkStatus();
|
|
}
|
|
|
|
StatusWithSize FakeFlashMemory::Read(Address address,
|
|
std::span<std::byte> output) {
|
|
if (address + output.size() >= sector_count() * size_bytes()) {
|
|
return StatusWithSize::OutOfRange();
|
|
}
|
|
|
|
// Check for injected read errors
|
|
Status status = FlashError::Check(read_errors_, address, output.size());
|
|
std::memcpy(output.data(), &buffer_[address], output.size());
|
|
return StatusWithSize(status, output.size());
|
|
}
|
|
|
|
StatusWithSize FakeFlashMemory::Write(Address address,
|
|
std::span<const std::byte> data) {
|
|
if (address % alignment_bytes() != 0 ||
|
|
data.size() % alignment_bytes() != 0) {
|
|
PW_LOG_ERROR("Unaligned write; address %x, size %u B, alignment %u",
|
|
unsigned(address),
|
|
unsigned(data.size()),
|
|
unsigned(alignment_bytes()));
|
|
return StatusWithSize::InvalidArgument();
|
|
}
|
|
|
|
if (data.size() > sector_size_bytes() - (address % sector_size_bytes())) {
|
|
PW_LOG_ERROR("Write crosses sector boundary; address %x, size %u B",
|
|
unsigned(address),
|
|
unsigned(data.size()));
|
|
return StatusWithSize::InvalidArgument();
|
|
}
|
|
|
|
if (address + data.size() > sector_count() * sector_size_bytes()) {
|
|
PW_LOG_ERROR(
|
|
"Write beyond end of memory; address %x, size %u B, max address %x",
|
|
unsigned(address),
|
|
unsigned(data.size()),
|
|
unsigned(sector_count() * sector_size_bytes()));
|
|
return StatusWithSize::OutOfRange();
|
|
}
|
|
|
|
// Check in erased state
|
|
for (unsigned i = 0; i < data.size(); i++) {
|
|
if (buffer_[address + i] != kErasedValue) {
|
|
PW_LOG_ERROR("Writing to previously written address: %x",
|
|
unsigned(address));
|
|
return StatusWithSize::Unknown();
|
|
}
|
|
}
|
|
|
|
// Check for any injected write errors
|
|
Status status = FlashError::Check(write_errors_, address, data.size());
|
|
std::memcpy(&buffer_[address], data.data(), data.size());
|
|
return StatusWithSize(status, data.size());
|
|
}
|
|
|
|
std::byte* FakeFlashMemory::FlashAddressToMcuAddress(Address address) const {
|
|
if (address > sector_count() * sector_size_bytes()) {
|
|
PW_LOG_ERROR(
|
|
"FlashAddressToMcuAddress beyond end of memory; address %x, max "
|
|
"address %x",
|
|
unsigned(address),
|
|
unsigned(sector_count() * sector_size_bytes()));
|
|
return nullptr;
|
|
}
|
|
return buffer_.data() + address;
|
|
}
|
|
|
|
} // namespace pw::kvs
|