// 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.
#ifndef LIBBRILLO_BRILLO_STREAMS_MEMORY_CONTAINERS_H_
#define LIBBRILLO_BRILLO_STREAMS_MEMORY_CONTAINERS_H_
#include <string>
#include <vector>
#include <brillo/brillo_export.h>
#include <brillo/errors/error.h>
#include <brillo/streams/stream.h>
namespace brillo {
namespace data_container {
// MemoryStream class relies on helper classes defined below to support data
// storage in various types of containers.
// A particular implementation of container type (e.g. based on raw memory
// buffers, std::vector, std::string or others) need to implement the container
// interface provided by data_container::DataContainerInterface.
// Low-level functionality such as reading data from and writing data to the
// container, getting and changing the buffer size, and so on, must be provided.
// Not all methods must be provided. For example, for read-only containers, only
// read operations can be provided.
class BRILLO_EXPORT DataContainerInterface {
public:
DataContainerInterface() = default;
virtual ~DataContainerInterface() = default;
// Read the data from the container into |buffer|. Up to |size_to_read| bytes
// must be read at a time. The container can return fewer bytes. The actual
// size of data read is provided in |size_read|.
// If the read operation fails, the function must return false and provide
// additional information about the error in |error| object.
virtual bool Read(void* buffer,
size_t size_to_read,
size_t offset,
size_t* size_read,
ErrorPtr* error) = 0;
// Writes |size_to_write| bytes of data from |buffer| into the container.
// The container may accept fewer bytes of data. The actual size of data
// written is provided in |size_written|.
// If the read operation fails, the function must return false and provide
// additional information about the error in |error| object.
virtual bool Write(const void* buffer,
size_t size_to_write,
size_t offset,
size_t* size_written,
ErrorPtr* error) = 0;
// Resizes the container to the new size specified in |new_size|.
virtual bool Resize(size_t new_size, ErrorPtr* error) = 0;
// Returns the current size of the container.
virtual size_t GetSize() const = 0;
// Returns true if the container is read-only.
virtual bool IsReadOnly() const = 0;
private:
DISALLOW_COPY_AND_ASSIGN(DataContainerInterface);
};
// ContiguousBufferBase is a helper base class for memory containers that
// employ contiguous memory for all of their data. This class provides the
// default implementation for Read() and Write() functions and requires the
// implementations to provide GetBuffer() and/or ReadOnlyBuffer() functions.
class BRILLO_EXPORT ContiguousBufferBase : public DataContainerInterface {
public:
ContiguousBufferBase() = default;
// Implementation of DataContainerInterface::Read().
bool Read(void* buffer,
size_t size_to_read,
size_t offset,
size_t* size_read,
ErrorPtr* error) override;
// Implementation of DataContainerInterface::Write().
bool Write(const void* buffer,
size_t size_to_write,
size_t offset,
size_t* size_written,
ErrorPtr* error) override;
// Overload to provide the pointer to the read-only data for the container at
// the specified |offset|. In case of an error, this function must return
// nullptr and provide error details in |error| object if provided.
virtual const void* GetReadOnlyBuffer(size_t offset,
ErrorPtr* error) const = 0;
// Overload to provide the pointer to the read/write data for the container at
// the specified |offset|. In case of an error, this function must return
// nullptr and provide error details in |error| object if provided.
virtual void* GetBuffer(size_t offset, ErrorPtr* error) = 0;
protected:
// Wrapper around memcpy which can be mocked out in tests.
virtual void CopyMemoryBlock(void* dest, const void* src, size_t size) const;
private:
DISALLOW_COPY_AND_ASSIGN(ContiguousBufferBase);
};
// ContiguousReadOnlyBufferBase is a specialization of ContiguousBufferBase for
// read-only containers.
class BRILLO_EXPORT ContiguousReadOnlyBufferBase : public ContiguousBufferBase {
public:
ContiguousReadOnlyBufferBase() = default;
// Fails with an error "operation_not_supported" (Stream is read-only) error.
bool Write(const void* buffer,
size_t size_to_write,
size_t offset,
size_t* size_written,
ErrorPtr* error) override;
// Fails with an error "operation_not_supported" (Stream is read-only) error.
bool Resize(size_t new_size, ErrorPtr* error) override;
// Fails with an error "operation_not_supported" (Stream is read-only) error.
bool IsReadOnly() const override { return true; }
// Fails with an error "operation_not_supported" (Stream is read-only) error.
void* GetBuffer(size_t offset, ErrorPtr* error) override;
private:
DISALLOW_COPY_AND_ASSIGN(ContiguousReadOnlyBufferBase);
};
// ReadOnlyBuffer implements a read-only container based on raw memory block.
class BRILLO_EXPORT ReadOnlyBuffer : public ContiguousReadOnlyBufferBase {
public:
// Constructs the container based at the pointer to memory |buffer| and its
// |size|. The pointer to the memory must be valid throughout life-time of
// the stream using this container.
ReadOnlyBuffer(const void* buffer, size_t size)
: buffer_(buffer), size_(size) {}
// Returns the pointer to data at |offset|.
const void* GetReadOnlyBuffer(size_t offset,
ErrorPtr* /* error */) const override {
return reinterpret_cast<const uint8_t*>(buffer_) + offset;
}
// Returns the size of the container.
size_t GetSize() const override { return size_; }
private:
// Raw memory pointer to the data block and its size.
const void* buffer_;
size_t size_;
DISALLOW_COPY_AND_ASSIGN(ReadOnlyBuffer);
};
// VectorPtr<T> is a read/write container based on a vector<T> pointer.
// This is a template class to allow usage of both vector<char> and
// vector<uint8_t> without duplicating the implementation.
template<typename T>
class VectorPtr : public ContiguousBufferBase {
public:
static_assert(sizeof(T) == 1, "Only char/byte is supported");
explicit VectorPtr(std::vector<T>* vector) : vector_ptr_(vector) {}
bool Resize(size_t new_size, ErrorPtr* /* error */) override {
vector_ptr_->resize(new_size);
return true;
}
size_t GetSize() const override { return vector_ptr_->size(); }
bool IsReadOnly() const override { return false; }
const void* GetReadOnlyBuffer(size_t offset,
ErrorPtr* /* error */) const override {
return reinterpret_cast<const uint8_t*>(vector_ptr_->data()) + offset;
}
void* GetBuffer(size_t offset, ErrorPtr* /* error */) override {
return reinterpret_cast<uint8_t*>(vector_ptr_->data()) + offset;
}
protected:
std::vector<T>* vector_ptr_;
private:
DISALLOW_COPY_AND_ASSIGN(VectorPtr);
};
// ReadOnlyVectorRef<T> is a read-only container based on a vector<T> reference.
// This is a template class to allow usage of both vector<char> and
// vector<uint8_t> without duplicating the implementation.
template<typename T>
class ReadOnlyVectorRef : public ContiguousReadOnlyBufferBase {
public:
static_assert(sizeof(T) == 1, "Only char/byte is supported");
explicit ReadOnlyVectorRef(const std::vector<T>& vector)
: vector_ref_(vector) {}
const void* GetReadOnlyBuffer(size_t offset,
ErrorPtr* /* error */) const override {
return reinterpret_cast<const uint8_t*>(vector_ref_.data()) + offset;
}
size_t GetSize() const override { return vector_ref_.size(); }
protected:
const std::vector<T>& vector_ref_;
private:
DISALLOW_COPY_AND_ASSIGN(ReadOnlyVectorRef);
};
// ReadOnlyVectorCopy<T> is a read-only container based on a copy of vector<T>.
// This container actually owns the data stored in the vector.
// This is a template class to allow usage of both vector<char> and
// vector<uint8_t> without duplicating the implementation.
template<typename T>
class ReadOnlyVectorCopy : public ContiguousReadOnlyBufferBase {
public:
static_assert(sizeof(T) == 1, "Only char/byte is supported");
explicit ReadOnlyVectorCopy(std::vector<T> vector)
: vector_copy_(std::move(vector)) {}
ReadOnlyVectorCopy(const T* buffer, size_t size)
: vector_copy_(buffer, buffer + size) {}
const void* GetReadOnlyBuffer(size_t offset,
ErrorPtr* /* error */) const override {
return reinterpret_cast<const uint8_t*>(vector_copy_.data()) + offset;
}
size_t GetSize() const override { return vector_copy_.size(); }
protected:
std::vector<T> vector_copy_;
private:
DISALLOW_COPY_AND_ASSIGN(ReadOnlyVectorCopy);
};
// ByteBuffer is a read/write container that manages the data and underlying
// storage.
class BRILLO_EXPORT ByteBuffer : public VectorPtr<uint8_t> {
public:
explicit ByteBuffer(size_t reserve_size);
~ByteBuffer() override;
private:
DISALLOW_COPY_AND_ASSIGN(ByteBuffer);
};
// StringPtr is a read/write container based on external std::string storage.
class BRILLO_EXPORT StringPtr : public ContiguousBufferBase {
public:
explicit StringPtr(std::string* string);
bool Resize(size_t new_size, ErrorPtr* error) override;
size_t GetSize() const override { return string_ptr_->size(); }
bool IsReadOnly() const override { return false; }
const void* GetReadOnlyBuffer(size_t offset, ErrorPtr* error) const override;
void* GetBuffer(size_t offset, ErrorPtr* error) override;
protected:
std::string* string_ptr_;
private:
DISALLOW_COPY_AND_ASSIGN(StringPtr);
};
// ReadOnlyStringRef is a read-only container based on external std::string.
class BRILLO_EXPORT ReadOnlyStringRef : public ContiguousReadOnlyBufferBase {
public:
explicit ReadOnlyStringRef(const std::string& string);
const void* GetReadOnlyBuffer(size_t offset, ErrorPtr* error) const override;
size_t GetSize() const override { return string_ref_.size(); }
protected:
const std::string& string_ref_;
private:
DISALLOW_COPY_AND_ASSIGN(ReadOnlyStringRef);
};
// ReadOnlyStringCopy is a read-only container based on a copy of a std::string.
// This container actually owns the data stored in the string.
class BRILLO_EXPORT ReadOnlyStringCopy : public ReadOnlyStringRef {
public:
explicit ReadOnlyStringCopy(std::string string);
protected:
std::string string_copy_;
private:
DISALLOW_COPY_AND_ASSIGN(ReadOnlyStringCopy);
};
} // namespace data_container
} // namespace brillo
#endif // LIBBRILLO_BRILLO_STREAMS_MEMORY_CONTAINERS_H_