mirror of
https://github.com/ChaiScript/ChaiScript.git
synced 2025-12-06 16:57:04 +08:00
Using gcc 13.3.0 I get many warnings like the following:
```
In file included from .../include/c++/13.3.0/bits/stl_iterator.h:85,
from .../include/c++/13.3.0/bits/stl_algobase.h:67,
from .../include/c++/13.3.0/bits/stl_tree.h:63,
from .../include/c++/13.3.0/map:62,
from .../ChaiScript/static_libs/../include/chaiscript/chaiscript_stdlib.hpp:10,
from .../ChaiScript/static_libs/chaiscript_stdlib.cpp:1:
.../include/c++/13.3.0/bits/stl_construct.h: In instantiation of ‘constexpr decltype (::new(void*(0)) _Tp) std::construct_at(_Tp*, _Args&& ...) [with _Tp = chaiscript::Boxed_Value::Data; _Args = {chaiscript::Type_Info, chaiscript::detail::Any, bool, std::nullptr_t, bool&}; decltype (::new(void*(0)) _Tp) = chaiscript::Boxed_Value::Data*]’:
.../include/c++/13.3.0/bits/stl_construct.h:115:21: required from ‘constexpr void std::_Construct(_Tp*, _Args&& ...) [with _Tp = chaiscript::Boxed_Value::Data; _Args = {chaiscript::Type_Info, chaiscript::detail::Any, bool, std::nullptr_t, bool&}]’
.../include/c++/13.3.0/bits/alloc_traits.h:661:19: required from ‘static constexpr void std::allocator_traits<std::allocator<void> >::construct(allocator_type&, _Up*, _Args&& ...) [with _Up = chaiscript::Boxed_Value::Data; _Args = {chaiscript::Type_Info, chaiscript::detail::Any, bool, std::nullptr_t, bool&}; allocator_type = std::allocator<void>]’
.../include/c++/13.3.0/bits/shared_ptr_base.h:604:39: required from ‘std::_Sp_counted_ptr_inplace<_Tp, _Alloc, _Lp>::_Sp_counted_ptr_inplace(_Alloc, _Args&& ...) [with _Args = {chaiscript::Type_Info, chaiscript::detail::Any, bool, std::nullptr_t, bool&}; _Tp = chaiscript::Boxed_Value::Data; _Alloc = std::allocator<void>; __gnu_cxx::_Lock_policy _Lp = __gnu_cxx::_S_atomic]’
.../include/c++/13.3.0/bits/shared_ptr_base.h:971:16: required from ‘std::__shared_count<_Lp>::__shared_count(_Tp*&, std::_Sp_alloc_shared_tag<_Alloc>, _Args&& ...) [with _Tp = chaiscript::Boxed_Value::Data; _Alloc = std::allocator<void>; _Args = {chaiscript::Type_Info, chaiscript::detail::Any, bool, std::nullptr_t, bool&}; __gnu_cxx::_Lock_policy _Lp = __gnu_cxx::_S_atomic]’
.../include/c++/13.3.0/bits/shared_ptr_base.h:1712:14: required from ‘std::__shared_ptr<_Tp, _Lp>::__shared_ptr(std::_Sp_alloc_shared_tag<_Tp>, _Args&& ...) [with _Alloc = std::allocator<void>; _Args = {chaiscript::Type_Info, chaiscript::detail::Any, bool, std::nullptr_t, bool&}; _Tp = chaiscript::Boxed_Value::Data; __gnu_cxx::_Lock_policy _Lp = __gnu_cxx::_S_atomic]’
.../include/c++/13.3.0/bits/shared_ptr.h:464:59: required from ‘std::shared_ptr<_Tp>::shared_ptr(std::_Sp_alloc_shared_tag<_Tp>, _Args&& ...) [with _Alloc = std::allocator<void>; _Args = {chaiscript::Type_Info, chaiscript::detail::Any, bool, std::nullptr_t, bool&}; _Tp = chaiscript::Boxed_Value::Data]’
.../include/c++/13.3.0/bits/shared_ptr.h:1009:14: required from ‘std::shared_ptr<std::_NonArray<_Tp> > std::make_shared(_Args&& ...) [with _Tp = chaiscript::Boxed_Value::Data; _Args = {chaiscript::Type_Info, chaiscript::detail::Any, bool, std::nullptr_t, bool&}; _NonArray<_Tp> = chaiscript::Boxed_Value::Data]’
.../ChaiScript/static_libs/../include/chaiscript/language/../dispatchkit/boxed_value.hpp:74:38: required from here
.../include/c++/13.3.0/bits/stl_construct.h:95:14: warning: noexcept-expression evaluates to ‘false’ because of a call to ‘chaiscript::Boxed_Value::Data::Data(const chaiscript::Type_Info&, chaiscript::detail::Any, bool, const void*, bool)’ [-Wnoexcept]
95 | noexcept(noexcept(::new((void*)0) _Tp(std::declval<_Args>()...)))
| ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
In file included from .../ChaiScript/static_libs/../include/chaiscript/language/chaiscript_common.hpp:21,
from .../ChaiScript/static_libs/../include/chaiscript/chaiscript_stdlib.hpp:17:
.../ChaiScript/static_libs/../include/chaiscript/language/../dispatchkit/boxed_value.hpp:34:7: note: but ‘chaiscript::Boxed_Value::Data::Data(const chaiscript::Type_Info&, chaiscript::detail::Any, bool, const void*, bool)’ does not throw; perhaps it should be declared ‘noexcept’
34 | Data(const Type_Info &ti, chaiscript::detail::Any to, bool is_ref, const void *t_void_ptr, bool t_return_value)
| ^~~~
/
```
Fix this by adding a noexcept like the warning suggests.
359 lines
12 KiB
C++
359 lines
12 KiB
C++
// This file is distributed under the BSD License.
|
|
// See "license.txt" for details.
|
|
// Copyright 2009-2012, Jonathan Turner (jonathan@emptycrate.com)
|
|
// Copyright 2009-2018, Jason Turner (jason@emptycrate.com)
|
|
// http://www.chaiscript.com
|
|
|
|
// This is an open source non-commercial project. Dear PVS-Studio, please check it.
|
|
// PVS-Studio Static Code Analyzer for C, C++ and C#: http://www.viva64.com
|
|
|
|
#ifndef CHAISCRIPT_BOXED_VALUE_HPP_
|
|
#define CHAISCRIPT_BOXED_VALUE_HPP_
|
|
|
|
#include <map>
|
|
#include <memory>
|
|
#include <type_traits>
|
|
|
|
#include "../chaiscript_defines.hpp"
|
|
#include "any.hpp"
|
|
#include "type_info.hpp"
|
|
|
|
namespace chaiscript {
|
|
/// \brief A wrapper for holding any valid C++ type. All types in ChaiScript are Boxed_Value objects
|
|
/// \sa chaiscript::boxed_cast
|
|
class Boxed_Value {
|
|
public:
|
|
/// used for explicitly creating a "void" object
|
|
struct Void_Type {
|
|
};
|
|
|
|
private:
|
|
/// structure which holds the internal state of a Boxed_Value
|
|
/// \todo Get rid of Any and merge it with this, reducing an allocation in the process
|
|
struct Data {
|
|
Data(const Type_Info &ti, chaiscript::detail::Any to, bool is_ref, const void *t_void_ptr, bool t_return_value) noexcept
|
|
: m_type_info(ti)
|
|
, m_obj(std::move(to))
|
|
, m_data_ptr(ti.is_const() ? nullptr : const_cast<void *>(t_void_ptr))
|
|
, m_const_data_ptr(t_void_ptr)
|
|
, m_is_ref(is_ref)
|
|
, m_return_value(t_return_value) {
|
|
}
|
|
|
|
Data &operator=(const Data &rhs) {
|
|
m_type_info = rhs.m_type_info;
|
|
m_obj = rhs.m_obj;
|
|
m_is_ref = rhs.m_is_ref;
|
|
m_data_ptr = rhs.m_data_ptr;
|
|
m_const_data_ptr = rhs.m_const_data_ptr;
|
|
m_return_value = rhs.m_return_value;
|
|
|
|
if (rhs.m_attrs) {
|
|
m_attrs = std::make_unique<std::map<std::string, std::shared_ptr<Data>>>(*rhs.m_attrs);
|
|
}
|
|
|
|
return *this;
|
|
}
|
|
|
|
Data(const Data &) = delete;
|
|
|
|
Data(Data &&) = default;
|
|
Data &operator=(Data &&rhs) = default;
|
|
|
|
Type_Info m_type_info;
|
|
chaiscript::detail::Any m_obj;
|
|
void *m_data_ptr;
|
|
const void *m_const_data_ptr;
|
|
std::unique_ptr<std::map<std::string, std::shared_ptr<Data>>> m_attrs;
|
|
bool m_is_ref;
|
|
bool m_return_value;
|
|
};
|
|
|
|
struct Object_Data {
|
|
static auto get(Boxed_Value::Void_Type, bool t_return_value) {
|
|
return std::make_shared<Data>(detail::Get_Type_Info<void>::get(), chaiscript::detail::Any(), false, nullptr, t_return_value);
|
|
}
|
|
|
|
template<typename T>
|
|
static auto get(const std::shared_ptr<T> *obj, bool t_return_value) {
|
|
return get(*obj, t_return_value);
|
|
}
|
|
|
|
template<typename T>
|
|
static auto get(const std::shared_ptr<T> &obj, bool t_return_value) {
|
|
return std::make_shared<Data>(detail::Get_Type_Info<T>::get(), chaiscript::detail::Any(obj), false, obj.get(), t_return_value);
|
|
}
|
|
|
|
template<typename T>
|
|
static auto get(std::shared_ptr<T> &&obj, bool t_return_value) {
|
|
auto ptr = obj.get();
|
|
return std::make_shared<Data>(detail::Get_Type_Info<T>::get(), chaiscript::detail::Any(std::move(obj)), false, ptr, t_return_value);
|
|
}
|
|
|
|
template<typename T>
|
|
static auto get(T *t, bool t_return_value) {
|
|
return get(std::ref(*t), t_return_value);
|
|
}
|
|
|
|
template<typename T>
|
|
static auto get(const T *t, bool t_return_value) {
|
|
return get(std::cref(*t), t_return_value);
|
|
}
|
|
|
|
template<typename T>
|
|
static auto get(std::reference_wrapper<T> obj, bool t_return_value) {
|
|
auto p = &obj.get();
|
|
return std::make_shared<Data>(detail::Get_Type_Info<T>::get(), chaiscript::detail::Any(std::move(obj)), true, p, t_return_value);
|
|
}
|
|
|
|
template<typename T>
|
|
static auto get(std::unique_ptr<T> &&obj, bool t_return_value) {
|
|
auto ptr = obj.get();
|
|
return std::make_shared<Data>(detail::Get_Type_Info<T>::get(),
|
|
chaiscript::detail::Any(std::make_shared<std::unique_ptr<T>>(std::move(obj))),
|
|
true,
|
|
ptr,
|
|
t_return_value);
|
|
}
|
|
|
|
template<typename T>
|
|
static auto get(T t, bool t_return_value) {
|
|
auto p = std::make_shared<T>(std::move(t));
|
|
auto ptr = p.get();
|
|
return std::make_shared<Data>(detail::Get_Type_Info<T>::get(), chaiscript::detail::Any(std::move(p)), false, ptr, t_return_value);
|
|
}
|
|
|
|
static std::shared_ptr<Data> get() { return std::make_shared<Data>(Type_Info(), chaiscript::detail::Any(), false, nullptr, false); }
|
|
};
|
|
|
|
public:
|
|
/// Basic Boxed_Value constructor
|
|
template<typename T, typename = std::enable_if_t<!std::is_same_v<Boxed_Value, std::decay_t<T>>>>
|
|
explicit Boxed_Value(T &&t, bool t_return_value = false)
|
|
: m_data(Object_Data::get(std::forward<T>(t), t_return_value)) {
|
|
}
|
|
|
|
/// Unknown-type constructor
|
|
Boxed_Value() = default;
|
|
|
|
Boxed_Value(Boxed_Value &&) = default;
|
|
Boxed_Value &operator=(Boxed_Value &&) = default;
|
|
Boxed_Value(const Boxed_Value &) = default;
|
|
Boxed_Value &operator=(const Boxed_Value &) = default;
|
|
|
|
void swap(Boxed_Value &rhs) noexcept { std::swap(m_data, rhs.m_data); }
|
|
|
|
/// Copy the values stored in rhs.m_data to m_data.
|
|
/// m_data pointers are not shared in this case
|
|
Boxed_Value assign(const Boxed_Value &rhs) noexcept {
|
|
(*m_data) = (*rhs.m_data);
|
|
return *this;
|
|
}
|
|
|
|
const Type_Info &get_type_info() const noexcept { return m_data->m_type_info; }
|
|
|
|
/// return true if the object is uninitialized
|
|
bool is_undef() const noexcept { return m_data->m_type_info.is_undef(); }
|
|
|
|
bool is_const() const noexcept { return m_data->m_type_info.is_const(); }
|
|
|
|
bool is_type(const Type_Info &ti) const noexcept { return m_data->m_type_info.bare_equal(ti); }
|
|
|
|
template<typename T>
|
|
auto pointer_sentinel(std::shared_ptr<T> &ptr) const noexcept {
|
|
struct Sentinel {
|
|
Sentinel(std::shared_ptr<T> &t_ptr, Data &data)
|
|
: m_ptr(t_ptr)
|
|
, m_data(data) {
|
|
}
|
|
|
|
~Sentinel() {
|
|
// save new pointer data
|
|
const auto ptr_ = m_ptr.get().get();
|
|
m_data.get().m_data_ptr = ptr_;
|
|
m_data.get().m_const_data_ptr = ptr_;
|
|
}
|
|
|
|
Sentinel &operator=(Sentinel &&s) = default;
|
|
Sentinel(Sentinel &&s) = default;
|
|
|
|
operator std::shared_ptr<T> &() const noexcept { return m_ptr.get(); }
|
|
|
|
Sentinel &operator=(const Sentinel &) = delete;
|
|
Sentinel(Sentinel &) = delete;
|
|
|
|
std::reference_wrapper<std::shared_ptr<T>> m_ptr;
|
|
std::reference_wrapper<Data> m_data;
|
|
};
|
|
|
|
return Sentinel(ptr, *(m_data.get()));
|
|
}
|
|
|
|
bool is_null() const noexcept { return (m_data->m_data_ptr == nullptr && m_data->m_const_data_ptr == nullptr); }
|
|
|
|
const chaiscript::detail::Any &get() const noexcept { return m_data->m_obj; }
|
|
|
|
bool is_ref() const noexcept { return m_data->m_is_ref; }
|
|
|
|
bool is_return_value() const noexcept { return m_data->m_return_value; }
|
|
|
|
void reset_return_value() const noexcept { m_data->m_return_value = false; }
|
|
|
|
bool is_pointer() const noexcept { return !is_ref(); }
|
|
|
|
void *get_ptr() const noexcept { return m_data->m_data_ptr; }
|
|
|
|
const void *get_const_ptr() const noexcept { return m_data->m_const_data_ptr; }
|
|
|
|
Boxed_Value get_attr(const std::string &t_name) {
|
|
if (!m_data->m_attrs) {
|
|
m_data->m_attrs = std::make_unique<std::map<std::string, std::shared_ptr<Data>>>();
|
|
}
|
|
|
|
auto &attr = (*m_data->m_attrs)[t_name];
|
|
if (attr) {
|
|
return Boxed_Value(attr, Internal_Construction());
|
|
} else {
|
|
Boxed_Value bv; // default construct a new one
|
|
attr = bv.m_data;
|
|
return bv;
|
|
}
|
|
}
|
|
|
|
Boxed_Value ©_attrs(const Boxed_Value &t_obj) {
|
|
if (t_obj.m_data->m_attrs) {
|
|
m_data->m_attrs = std::make_unique<std::map<std::string, std::shared_ptr<Data>>>(*t_obj.m_data->m_attrs);
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
Boxed_Value &clone_attrs(const Boxed_Value &t_obj) {
|
|
copy_attrs(t_obj);
|
|
reset_return_value();
|
|
return *this;
|
|
}
|
|
|
|
/// \returns true if the two Boxed_Values share the same internal type
|
|
static bool type_match(const Boxed_Value &l, const Boxed_Value &r) noexcept { return l.get_type_info() == r.get_type_info(); }
|
|
|
|
private:
|
|
// necessary to avoid hitting the templated && constructor of Boxed_Value
|
|
struct Internal_Construction {
|
|
};
|
|
|
|
Boxed_Value(std::shared_ptr<Data> t_data, Internal_Construction)
|
|
: m_data(std::move(t_data)) {
|
|
}
|
|
|
|
std::shared_ptr<Data> m_data = Object_Data::get();
|
|
};
|
|
|
|
/// @brief Creates a Boxed_Value. If the object passed in is a value type, it is copied. If it is a pointer, std::shared_ptr, or
|
|
/// std::reference_type
|
|
/// a copy is not made.
|
|
/// @param t The value to box
|
|
///
|
|
/// Example:
|
|
///
|
|
/// ~~~{.cpp}
|
|
/// int i;
|
|
/// chaiscript::ChaiScript chai;
|
|
/// chai.add(chaiscript::var(i), "i");
|
|
/// chai.add(chaiscript::var(&i), "ip");
|
|
/// ~~~
|
|
///
|
|
/// @sa @ref adding_objects
|
|
template<typename T>
|
|
Boxed_Value var(T &&t) {
|
|
return Boxed_Value(std::forward<T>(t));
|
|
}
|
|
|
|
namespace detail {
|
|
/// \brief Takes a value, copies it and returns a Boxed_Value object that is immutable
|
|
/// \param[in] t Value to copy and make const
|
|
/// \returns Immutable Boxed_Value
|
|
/// \sa Boxed_Value::is_const
|
|
template<typename T>
|
|
Boxed_Value const_var_impl(const T &t) {
|
|
return Boxed_Value(std::make_shared<typename std::add_const<T>::type>(t));
|
|
}
|
|
|
|
/// \brief Takes a pointer to a value, adds const to the pointed to type and returns an immutable Boxed_Value.
|
|
/// Does not copy the pointed to value.
|
|
/// \param[in] t Pointer to make immutable
|
|
/// \returns Immutable Boxed_Value
|
|
/// \sa Boxed_Value::is_const
|
|
template<typename T>
|
|
Boxed_Value const_var_impl(T *t) {
|
|
return Boxed_Value(const_cast<typename std::add_const<T>::type *>(t));
|
|
}
|
|
|
|
/// \brief Takes a std::shared_ptr to a value, adds const to the pointed to type and returns an immutable Boxed_Value.
|
|
/// Does not copy the pointed to value.
|
|
/// \param[in] t Pointer to make immutable
|
|
/// \returns Immutable Boxed_Value
|
|
/// \sa Boxed_Value::is_const
|
|
template<typename T>
|
|
Boxed_Value const_var_impl(const std::shared_ptr<T> &t) {
|
|
return Boxed_Value(std::const_pointer_cast<typename std::add_const<T>::type>(t));
|
|
}
|
|
|
|
/// \brief Takes a std::reference_wrapper value, adds const to the referenced type and returns an immutable Boxed_Value.
|
|
/// Does not copy the referenced value.
|
|
/// \param[in] t Reference object to make immutable
|
|
/// \returns Immutable Boxed_Value
|
|
/// \sa Boxed_Value::is_const
|
|
template<typename T>
|
|
Boxed_Value const_var_impl(const std::reference_wrapper<T> &t) {
|
|
return Boxed_Value(std::cref(t.get()));
|
|
}
|
|
} // namespace detail
|
|
|
|
/// \brief Takes an object and returns an immutable Boxed_Value. If the object is a std::reference or pointer type
|
|
/// the value is not copied. If it is an object type, it is copied.
|
|
/// \param[in] t Object to make immutable
|
|
/// \returns Immutable Boxed_Value
|
|
/// \sa chaiscript::Boxed_Value::is_const
|
|
/// \sa chaiscript::var
|
|
///
|
|
/// Example:
|
|
/// \code
|
|
/// enum Colors
|
|
/// {
|
|
/// Blue,
|
|
/// Green,
|
|
/// Red
|
|
/// };
|
|
/// chaiscript::ChaiScript chai
|
|
/// chai.add(chaiscript::const_var(Blue), "Blue"); // add immutable constant
|
|
/// chai.add(chaiscript::const_var(Red), "Red");
|
|
/// chai.add(chaiscript::const_var(Green), "Green");
|
|
/// \endcode
|
|
///
|
|
/// \todo support C++11 strongly typed enums
|
|
/// \sa \ref adding_objects
|
|
template<typename T>
|
|
Boxed_Value const_var(const T &t) {
|
|
return detail::const_var_impl(t);
|
|
}
|
|
|
|
inline Boxed_Value void_var() {
|
|
static const auto v = Boxed_Value(Boxed_Value::Void_Type());
|
|
return v;
|
|
}
|
|
|
|
inline Boxed_Value const_var(bool b) {
|
|
static const auto t = detail::const_var_impl(true);
|
|
static const auto f = detail::const_var_impl(false);
|
|
|
|
if (b) {
|
|
return t;
|
|
} else {
|
|
return f;
|
|
}
|
|
}
|
|
|
|
} // namespace chaiscript
|
|
|
|
#endif
|