597 lines
19 KiB
C
Raw Normal View History

2024-10-29 10:49:46 +01:00
///\file
/******************************************************************************
The MIT License(MIT)
Embedded Template Library.
https://github.com/ETLCPP/etl
https://www.etlcpp.com
Copyright(c) 2021 John Wellbelove
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files(the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and / or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions :
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.
******************************************************************************/
#ifndef ETL_MEM_CAST_INCLUDED
#define ETL_MEM_CAST_INCLUDED
#include "platform.h"
#include "memory.h"
#include "static_assert.h"
#include "largest.h"
#include "utility.h"
#include "placement_new.h"
#include "exception.h"
#include "error_handler.h"
#include "file_error_numbers.h"
#include "binary.h"
#include <stdint.h>
#include <string.h>
namespace etl
{
//***************************************************************************
/// The base class for array_wrapper exceptions.
//***************************************************************************
class mem_cast_exception : public etl::exception
{
public:
mem_cast_exception(string_type reason_, string_type file_name_, numeric_type line_number_)
: exception(reason_, file_name_, line_number_)
{
}
};
//***************************************************************************
/// The exception thrown when the type size is too large.
//***************************************************************************
class mem_cast_size_exception : public etl::mem_cast_exception
{
public:
mem_cast_size_exception(string_type file_name_, numeric_type line_number_)
: mem_cast_exception(ETL_ERROR_TEXT("mem_cast:size", ETL_MEM_CAST_FILE_ID"A"), file_name_, line_number_)
{
}
};
//***************************************************************************
/// The exception thrown when the pointer is null.
//***************************************************************************
class mem_cast_nullptr_exception : public etl::mem_cast_exception
{
public:
mem_cast_nullptr_exception(string_type file_name_, numeric_type line_number_)
: mem_cast_exception(ETL_ERROR_TEXT("mem_cast:null pointer", ETL_MEM_CAST_FILE_ID"B"), file_name_, line_number_)
{
}
};
//*****************************************************************************
/// mem_cast
//*****************************************************************************
template <size_t Size_, size_t Alignment_>
class mem_cast
{
public:
static ETL_CONSTANT size_t Size = Size_;
static ETL_CONSTANT size_t Alignment = Alignment_;
ETL_STATIC_ASSERT((Alignment == 1) || etl::is_power_of_2<Alignment>::value, "Alignment must be a power of 2");
//***********************************
/// Default constructor
//***********************************
mem_cast()
: buffer()
{
}
//***********************************
/// Copy constructor
//***********************************
template <size_t Other_Size, size_t Other_Alignment>
mem_cast(const mem_cast<Other_Size, Other_Alignment>& other)
{
ETL_STATIC_ASSERT(Size >= Other_Size, "Other size is too large");
memcpy(buffer, other.buffer, Size_);
}
//***********************************
/// Assignment operator
//***********************************
template <size_t Other_Size, size_t Other_Alignment>
mem_cast& operator =(const mem_cast<Other_Size, Other_Alignment>& rhs)
{
ETL_STATIC_ASSERT(Size >= Other_Size, "RHS size is too large");
memcpy(buffer, rhs.buffer, Size_);
return *this;
}
//***********************************
/// Assign from value
//***********************************
template <typename T>
void assign(const T& value)
{
ETL_STATIC_ASSERT(Size >= sizeof(T), "Type size is too large");
::new (static_cast<void*>(buffer)) T(value);
}
//***********************************
/// Assign from value at offset
//***********************************
template <typename T>
void assign_at_offset(size_t offset, const T& value)
{
char* p = static_cast<char*>(buffer) + offset;
ETL_ASSERT(sizeof(T) <= (Size - offset), ETL_ERROR(etl::mem_cast_size_exception));
::new (p) T(value);
}
//***********************************
/// Assign from value at offset
//***********************************
template <typename T, size_t Offset>
void assign_at_offset(const T& value)
{
char* p = static_cast<char*>(buffer) + Offset;
ETL_STATIC_ASSERT(sizeof(T) <= (Size - Offset), "Type size is too large");
::new (p) T(value);
}
#if ETL_USING_CPP11
//***********************************
/// Emplace from parameters
//***********************************
template <typename T, typename... TArgs>
void emplace(TArgs... args)
{
ETL_STATIC_ASSERT(Size >= sizeof(T), "Type size is too large");
::new (static_cast<void*>(buffer)) T(etl::forward<TArgs>(args)...);
}
//***********************************
/// Emplace from parameters at offset
//***********************************
template <typename T, typename... TArgs>
void emplace_at_offset(size_t offset, TArgs... args)
{
char* p = static_cast<char*>(buffer) + offset;
ETL_ASSERT(sizeof(T) <= (Size - offset), ETL_ERROR(etl::mem_cast_size_exception));
::new (p) T(etl::forward<TArgs>(args)...);
}
//***********************************
/// Emplace from parameters at offset
//***********************************
template <typename T, size_t Offset, typename... TArgs>
void emplace_at_offset(TArgs... args)
{
char* p = static_cast<char*>(buffer) + Offset;
ETL_STATIC_ASSERT(sizeof(T) <= (Size - Offset), "Type size is too large");
::new (p) T(etl::forward<TArgs>(args)...);
}
#endif
//***********************************
/// Get a reference to T
//***********************************
template <typename T>
ETL_NODISCARD T& ref()
{
ETL_STATIC_ASSERT(sizeof(T) <= Size, "Size of T is too large");
return *static_cast<T*>(buffer);
}
//***********************************
/// Get a const reference to T
//***********************************
template <typename T>
ETL_NODISCARD const T& ref() const
{
ETL_STATIC_ASSERT(sizeof(T) <= Size, "Size of T is too large");
return *static_cast<const T*>(buffer);
}
//***********************************
/// Get a reference to T at offset (dynamic)
//***********************************
template <typename T>
ETL_NODISCARD T& ref_at_offset(size_t offset)
{
ETL_ASSERT(sizeof(T) <= (Size - offset), ETL_ERROR(etl::mem_cast_size_exception));
char* p = buffer + offset;
return *static_cast<T*>(p);
}
//***********************************
/// Get a const reference to T at offset (dynamic)
//***********************************
template <typename T>
ETL_NODISCARD const T& ref_at_offset(size_t offset) const
{
ETL_ASSERT(sizeof(T) <= (Size - offset), ETL_ERROR(etl::mem_cast_size_exception));
char* p = buffer + offset;
return *static_cast<const T*>(p);
}
//***********************************
/// Get a reference to T at offset (static)
//***********************************
template <typename T, size_t Offset>
ETL_NODISCARD T& ref_at_offset()
{
ETL_STATIC_ASSERT(sizeof(T) <= (Size - Offset), "Size of T is too large");
char* p = buffer + Offset;
return *static_cast<T*>(p);
}
//***********************************
/// Get a const reference to T at offset (static)
//***********************************
template <typename T, size_t Offset>
ETL_NODISCARD const T& ref_at_offset() const
{
ETL_STATIC_ASSERT(sizeof(T) <= (Size - Offset), "Size of T is too large");
char* p = buffer + Offset;
return *static_cast<const T*>(p);
}
//***********************************
/// Get the size of the buffer
//***********************************
ETL_NODISCARD static ETL_CONSTEXPR size_t size()
{
return Size;
}
//***********************************
/// Get the alignment of the buffer
//***********************************
ETL_NODISCARD static ETL_CONSTEXPR size_t alignment()
{
return Alignment;
}
//***********************************
/// Get a pointer to the internal buffer
//***********************************
ETL_NODISCARD char* data()
{
return buffer;
}
//***********************************
/// Get a const pointer to the internal buffer
//***********************************
ETL_NODISCARD const char* data() const
{
return buffer;
}
private:
/// The internal buffer
etl::uninitialized_buffer<Size, 1U, Alignment> buffer;
};
template <size_t Size_, size_t Alignment_>
ETL_CONSTANT size_t mem_cast<Size_, Alignment_>::Size;
template <size_t Size_, size_t Alignment_>
ETL_CONSTANT size_t mem_cast<Size_, Alignment_>::Alignment;
//*****************************************************************************
/// mem_cast_ptr
//*****************************************************************************
class mem_cast_ptr
{
public:
static ETL_CONSTANT size_t Undefined_Size = etl::integral_limits<size_t>::max;
//***********************************
/// Default constructor
//***********************************
mem_cast_ptr()
: pbuffer(ETL_NULLPTR)
, buffer_size(Undefined_Size)
{
}
//***********************************
/// Construct with pointer to buffer and optional size
//***********************************
mem_cast_ptr(char* pbuffer_, size_t buffer_size_ = Undefined_Size)
: pbuffer(pbuffer_)
, buffer_size(buffer_size_)
{
}
//***********************************
/// Copy construct
//***********************************
mem_cast_ptr(const mem_cast_ptr& other)
: pbuffer(other.pbuffer)
, buffer_size(other.buffer_size)
{
}
//***********************************
/// Assignment operator
//***********************************
mem_cast_ptr& operator =(const mem_cast_ptr& rhs)
{
pbuffer = rhs.pbuffer;
buffer_size = rhs.buffer_size;
return *this;
}
//***********************************
/// Assign from value
//***********************************
template <typename T>
void assign(const T& value)
{
ETL_ASSERT((pbuffer != ETL_NULLPTR), ETL_ERROR(etl::mem_cast_nullptr_exception));
ETL_ASSERT(sizeof(T) <= buffer_size, ETL_ERROR(etl::mem_cast_size_exception));
::new (pbuffer) T(value);
}
//***********************************
/// Assign from value at offset
//***********************************
template <typename T>
void assign_at_offset(size_t offset, const T& value)
{
ETL_ASSERT((pbuffer != ETL_NULLPTR), ETL_ERROR(etl::mem_cast_nullptr_exception));
char* p = pbuffer + offset;
ETL_ASSERT(sizeof(T) <= (buffer_size - offset), ETL_ERROR(etl::mem_cast_size_exception));
::new (p) T(value);
}
//***********************************
/// Assign from value at offset
//***********************************
template <typename T, size_t Offset>
void assign_at_offset(const T& value)
{
ETL_ASSERT((pbuffer != ETL_NULLPTR), ETL_ERROR(etl::mem_cast_nullptr_exception));
char* p = pbuffer + Offset;
ETL_ASSERT(sizeof(T) <= (buffer_size - Offset), ETL_ERROR(etl::mem_cast_size_exception));
::new (p) T(value);
}
#if ETL_USING_CPP11
//***********************************
/// Emplace from parameters
//***********************************
template <typename T, typename... TArgs>
void emplace(TArgs... args)
{
ETL_ASSERT((pbuffer != ETL_NULLPTR), ETL_ERROR(etl::mem_cast_nullptr_exception));
ETL_ASSERT(sizeof(T) <= buffer_size, ETL_ERROR(etl::mem_cast_size_exception));
::new (pbuffer) T(etl::forward<TArgs>(args)...);
}
//***********************************
/// Emplace from parameters at offset
//***********************************
template <typename T, typename... TArgs>
void emplace_at_offset(size_t offset, TArgs... args)
{
ETL_ASSERT((pbuffer != ETL_NULLPTR), ETL_ERROR(etl::mem_cast_nullptr_exception));
char* p = pbuffer + offset;
ETL_ASSERT(sizeof(T) <= (buffer_size - offset), ETL_ERROR(etl::mem_cast_size_exception));
::new (p) T(etl::forward<TArgs>(args)...);
}
//***********************************
/// Emplace from parameters at offset
//***********************************
template <typename T, size_t Offset, typename... TArgs>
void emplace_at_offset(TArgs... args)
{
ETL_ASSERT((pbuffer != ETL_NULLPTR), ETL_ERROR(etl::mem_cast_nullptr_exception));
char* p = pbuffer + Offset;
ETL_ASSERT(sizeof(T) <= (buffer_size - Offset), ETL_ERROR(etl::mem_cast_size_exception));
::new (p) T(etl::forward<TArgs>(args)...);
}
#endif
//***********************************
/// Get a reference to T
//***********************************
template <typename T>
ETL_NODISCARD T& ref()
{
ETL_ASSERT((pbuffer != ETL_NULLPTR), ETL_ERROR(etl::mem_cast_nullptr_exception));
ETL_ASSERT(sizeof(T) <= buffer_size, ETL_ERROR(etl::mem_cast_size_exception));
return *reinterpret_cast<T*>(pbuffer);
}
//***********************************
/// Get a const reference to T
//***********************************
template <typename T>
ETL_NODISCARD const T& ref() const
{
ETL_ASSERT((pbuffer != ETL_NULLPTR), ETL_ERROR(etl::mem_cast_nullptr_exception));
ETL_ASSERT(sizeof(T) <= buffer_size, ETL_ERROR(etl::mem_cast_size_exception));
return *reinterpret_cast<const T*>(pbuffer);
}
//***********************************
/// Get a reference to T at offset (dynamic)
//***********************************
template <typename T>
ETL_NODISCARD T& ref_at_offset(size_t offset)
{
ETL_ASSERT((pbuffer != ETL_NULLPTR), ETL_ERROR(etl::mem_cast_nullptr_exception));
char* p = pbuffer + offset;
ETL_ASSERT(sizeof(T) <= (buffer_size - offset), ETL_ERROR(etl::mem_cast_size_exception));
return *reinterpret_cast<T*>(p);
}
//***********************************
/// Get a const reference to T at offset (dynamic)
//***********************************
template <typename T>
ETL_NODISCARD const T& ref_at_offset(size_t offset) const
{
ETL_ASSERT((pbuffer != ETL_NULLPTR), ETL_ERROR(etl::mem_cast_nullptr_exception));
char* p = pbuffer + offset;
ETL_ASSERT(sizeof(T) <= (buffer_size - offset), ETL_ERROR(etl::mem_cast_size_exception));
return *reinterpret_cast<const T*>(p);
}
//***********************************
/// Get a reference to T at offset (static)
//***********************************
template <typename T, size_t Offset>
ETL_NODISCARD T& ref_at_offset()
{
ETL_ASSERT((pbuffer != ETL_NULLPTR), ETL_ERROR(etl::mem_cast_nullptr_exception));
char* p = pbuffer + Offset;
ETL_ASSERT(sizeof(T) <= (buffer_size - Offset), ETL_ERROR(etl::mem_cast_size_exception));
return *reinterpret_cast<T*>(p);
}
//***********************************
/// Get a const reference to T at offset (static)
//***********************************
template <typename T, size_t Offset>
ETL_NODISCARD const T& ref_at_offset() const
{
ETL_ASSERT((pbuffer != ETL_NULLPTR), ETL_ERROR(etl::mem_cast_nullptr_exception));
char* p = pbuffer + Offset;
ETL_ASSERT(sizeof(T) <= (buffer_size - Offset), ETL_ERROR(etl::mem_cast_size_exception));
return *reinterpret_cast<const T*>(p);
}
//***********************************
/// Get the size of the buffer
//***********************************
ETL_NODISCARD size_t size() const
{
return buffer_size;
}
//***********************************
/// Get the alignment of the buffer
//***********************************
ETL_NODISCARD size_t alignment() const
{
typedef typename etl::smallest_uint_for_bits<sizeof(uintptr_t)* CHAR_BIT>::type type;
const type p = reinterpret_cast<type>(pbuffer);
return size_t(1U) << etl::count_trailing_zeros(p);
}
//***********************************
/// Set the pointer to the external buffer
//***********************************
void data(char* pbuffer_, size_t buffer_size_ = Undefined_Size)
{
pbuffer = pbuffer_;
buffer_size = buffer_size_;
}
//***********************************
/// Get a pointer to the external buffer
//***********************************
ETL_NODISCARD char* data()
{
return pbuffer;
}
//***********************************
/// Get const a pointer to the external buffer
//***********************************
ETL_NODISCARD const char* data() const
{
return pbuffer;
}
private:
/// Pointer to the buffer
char* pbuffer;
size_t buffer_size;
};
//*****************************************************************************
/// mem_cast_var
/// mem_cast from a variadic list of types
//*****************************************************************************
#if ETL_USING_CPP11 && !defined(ETL_MEM_CAST_FORCE_CPP03_IMPLEMENTATION)
template <typename... TTypes>
using mem_cast_types = etl::mem_cast<etl::largest<TTypes...>::size, etl::largest<TTypes...>::alignment>;
#else
template <typename T1, typename T2 = char, typename T3 = char, typename T4 = char,
typename T5 = char, typename T6 = char, typename T7 = char, typename T8 = char,
typename T9 = char, typename T10 = char, typename T11 = char, typename T12 = char,
typename T13 = char, typename T14 = char, typename T15 = char, typename T16 = char>
struct mem_cast_types : public etl::mem_cast<etl::largest<T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, T12, T13, T14, T15, T16>::size,
etl::largest<T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, T12, T13, T14, T15, T16>::alignment>
{
};
#endif
}
#endif