mirror of
https://github.com/XRPLF/rippled.git
synced 2026-06-05 17:56:49 +00:00
156 lines
4.4 KiB
C++
156 lines
4.4 KiB
C++
#include <xrpl/protocol/SField.h>
|
|
|
|
#include <xrpl/beast/utility/instrumentation.h>
|
|
|
|
#include <string>
|
|
#include <unordered_map>
|
|
|
|
namespace xrpl {
|
|
|
|
// Storage for static const members.
|
|
SField::IsSigning const SField::kNotSigning;
|
|
int SField::num = 0;
|
|
std::unordered_map<int, SField const*> SField::knownCodeToField;
|
|
std::unordered_map<std::string, SField const*> SField::knownNameToField;
|
|
|
|
// Give only this translation unit permission to construct SFields
|
|
struct SField::PrivateAccessTagT
|
|
{
|
|
explicit PrivateAccessTagT() = default;
|
|
};
|
|
|
|
static SField::PrivateAccessTagT access;
|
|
|
|
template <class T>
|
|
template <class... Args>
|
|
TypedField<T>::TypedField(PrivateAccessTagT pat, Args&&... args)
|
|
: SField(pat, std::forward<Args>(args)...)
|
|
{
|
|
}
|
|
|
|
// Construct all compile-time SFields, and register them in the knownCodeToField
|
|
// and knownNameToField databases:
|
|
|
|
// Use macros for most SField construction to enforce naming conventions.
|
|
#pragma push_macro("UNTYPED_SFIELD")
|
|
#undef UNTYPED_SFIELD
|
|
#pragma push_macro("TYPED_SFIELD")
|
|
#undef TYPED_SFIELD
|
|
|
|
#define UNTYPED_SFIELD(sfName, stiSuffix, fieldValue, ...) \
|
|
SField const sfName( \
|
|
access, \
|
|
STI_##stiSuffix, \
|
|
fieldValue, \
|
|
std::string_view(#sfName).substr(2).data(), \
|
|
##__VA_ARGS__);
|
|
#define TYPED_SFIELD(sfName, stiSuffix, fieldValue, ...) \
|
|
SF_##stiSuffix const sfName( \
|
|
access, \
|
|
STI_##stiSuffix, \
|
|
fieldValue, \
|
|
std::string_view(#sfName).substr(2).data(), \
|
|
##__VA_ARGS__);
|
|
|
|
// SFields which, for historical reasons, do not follow naming conventions.
|
|
SField const sfInvalid(access, -1, "");
|
|
SField const sfGeneric(access, 0, "Generic");
|
|
// The following two fields aren't used anywhere, but they break tests/have
|
|
// downstream effects.
|
|
SField const kSfHash(access, STI_UINT256, 257, "hash");
|
|
SField const kSfIndex(access, STI_UINT256, 258, "index");
|
|
|
|
#include <xrpl/protocol/detail/sfields.macro>
|
|
|
|
#undef TYPED_SFIELD
|
|
#pragma pop_macro("TYPED_SFIELD")
|
|
#undef UNTYPED_SFIELD
|
|
#pragma pop_macro("UNTYPED_SFIELD")
|
|
|
|
SField::SField(
|
|
PrivateAccessTagT,
|
|
SerializedTypeID tid,
|
|
int fv,
|
|
char const* fn,
|
|
int meta,
|
|
IsSigning signing)
|
|
: fieldCodeMem(fieldCode(tid, fv))
|
|
, fieldType(tid)
|
|
, fieldValue(fv)
|
|
, fieldName(fn)
|
|
, fieldMeta(meta)
|
|
, fieldNum(++num)
|
|
, signingField(signing)
|
|
, jsonName(fieldName.c_str())
|
|
{
|
|
XRPL_ASSERT(
|
|
!knownCodeToField.contains(fieldCodeMem),
|
|
"xrpl::SField::SField(tid,fv,fn,meta,signing) : fieldCode is unique");
|
|
XRPL_ASSERT(
|
|
!knownNameToField.contains(fieldName),
|
|
"xrpl::SField::SField(tid,fv,fn,meta,signing) : fieldName is unique");
|
|
knownCodeToField[fieldCodeMem] = this;
|
|
knownNameToField[fieldName] = this;
|
|
}
|
|
|
|
SField::SField(PrivateAccessTagT, int fc, char const* fn)
|
|
: fieldCodeMem(fc)
|
|
, fieldType(STI_UNKNOWN)
|
|
, fieldValue(0)
|
|
, fieldName(fn)
|
|
, fieldMeta(kSmdNever)
|
|
, fieldNum(++num)
|
|
, signingField(IsSigning::Yes)
|
|
, jsonName(fieldName.c_str())
|
|
{
|
|
XRPL_ASSERT(
|
|
!knownCodeToField.contains(fieldCodeMem),
|
|
"xrpl::SField::SField(fc,fn) : fieldCode is unique");
|
|
XRPL_ASSERT(
|
|
!knownNameToField.contains(fieldName), "xrpl::SField::SField(fc,fn) : fieldName is unique");
|
|
knownCodeToField[fieldCodeMem] = this;
|
|
knownNameToField[fieldName] = this;
|
|
}
|
|
|
|
SField const&
|
|
SField::getField(int code)
|
|
{
|
|
auto it = knownCodeToField.find(code);
|
|
|
|
if (it != knownCodeToField.end())
|
|
{
|
|
return *(it->second);
|
|
}
|
|
return sfInvalid;
|
|
}
|
|
|
|
int
|
|
SField::compare(SField const& f1, SField const& f2)
|
|
{
|
|
// -1 = f1 comes before f2, 0 = illegal combination, 1 = f1 comes after f2
|
|
if ((f1.fieldCodeMem <= 0) || (f2.fieldCodeMem <= 0))
|
|
return 0;
|
|
|
|
if (f1.fieldCodeMem < f2.fieldCodeMem)
|
|
return -1;
|
|
|
|
if (f2.fieldCodeMem < f1.fieldCodeMem)
|
|
return 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
SField const&
|
|
SField::getField(std::string const& fieldName)
|
|
{
|
|
auto it = knownNameToField.find(fieldName);
|
|
|
|
if (it != knownNameToField.end())
|
|
{
|
|
return *(it->second);
|
|
}
|
|
return sfInvalid;
|
|
}
|
|
|
|
} // namespace xrpl
|