// Copyright 2015 The Chromium OS Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include <brillo/streams/memory_containers.h>
#include <base/callback.h>
#include <brillo/streams/stream_errors.h>
namespace brillo {
namespace data_container {
namespace {
bool ErrorStreamReadOnly(const base::Location& location,
ErrorPtr* error) {
Error::AddTo(error,
location,
errors::stream::kDomain,
errors::stream::kOperationNotSupported,
"Stream is read-only");
return false;
}
} // anonymous namespace
void ContiguousBufferBase::CopyMemoryBlock(void* dest,
const void* src,
size_t size) const {
memcpy(dest, src, size);
}
bool ContiguousBufferBase::Read(void* buffer,
size_t size_to_read,
size_t offset,
size_t* size_read,
ErrorPtr* error) {
size_t buf_size = GetSize();
if (offset < buf_size) {
size_t remaining = buf_size - offset;
if (size_to_read >= remaining) {
size_to_read = remaining;
}
const void* src_buffer = GetReadOnlyBuffer(offset, error);
if (!src_buffer)
return false;
CopyMemoryBlock(buffer, src_buffer, size_to_read);
} else {
size_to_read = 0;
}
if (size_read)
*size_read = size_to_read;
return true;
}
bool ContiguousBufferBase::Write(const void* buffer,
size_t size_to_write,
size_t offset,
size_t* size_written,
ErrorPtr* error) {
if (size_to_write) {
size_t new_size = offset + size_to_write;
if (GetSize() < new_size && !Resize(new_size, error))
return false;
void* ptr = GetBuffer(offset, error);
if (!ptr)
return false;
CopyMemoryBlock(ptr, buffer, size_to_write);
if (size_written)
*size_written = size_to_write;
}
return true;
}
bool ContiguousReadOnlyBufferBase::Write(const void* /* buffer */,
size_t /* size_to_write */,
size_t /* offset */,
size_t* /* size_written */,
ErrorPtr* error) {
return ErrorStreamReadOnly(FROM_HERE, error);
}
bool ContiguousReadOnlyBufferBase::Resize(size_t /* new_size */,
ErrorPtr* error) {
return ErrorStreamReadOnly(FROM_HERE, error);
}
void* ContiguousReadOnlyBufferBase::GetBuffer(size_t /* offset */,
ErrorPtr* error) {
ErrorStreamReadOnly(FROM_HERE, error);
return nullptr;
}
ByteBuffer::ByteBuffer(size_t reserve_size)
: VectorPtr(new std::vector<uint8_t>()) {
vector_ptr_->reserve(reserve_size);
}
ByteBuffer::~ByteBuffer() {
delete vector_ptr_;
}
StringPtr::StringPtr(std::string* string) : string_ptr_(string) {}
bool StringPtr::Resize(size_t new_size, ErrorPtr* /* error */) {
string_ptr_->resize(new_size);
return true;
}
const void* StringPtr::GetReadOnlyBuffer(size_t offset,
ErrorPtr* /* error */) const {
return string_ptr_->data() + offset;
}
void* StringPtr::GetBuffer(size_t offset, ErrorPtr* /* error */) {
return &(*string_ptr_)[offset];
}
ReadOnlyStringRef::ReadOnlyStringRef(const std::string& string)
: string_ref_(string) {}
const void* ReadOnlyStringRef::GetReadOnlyBuffer(size_t offset,
ErrorPtr* /* error */) const {
return string_ref_.data() + offset;
}
ReadOnlyStringCopy::ReadOnlyStringCopy(std::string string)
: ReadOnlyStringRef(string_copy_), string_copy_(std::move(string)) {}
} // namespace data_container
} // namespace brillo