0
0
Fork 0
mirror of https://github.com/matrix-construct/construct synced 2025-01-14 16:46:50 +01:00

ircd::mods: Improve MAPI header layout; various cleanup.

This commit is contained in:
Jason Volk 2018-10-25 13:03:07 -07:00
parent 30d938530a
commit d19edbda98
3 changed files with 131 additions and 73 deletions

View file

@ -15,66 +15,85 @@
namespace ircd::mapi namespace ircd::mapi
{ {
struct header; struct header;
struct metablock;
using magic_t = uint16_t; using magic_t = uint16_t;
using version_t = uint16_t; using version_t = uint16_t;
using metadata = std::map<string_view, string_view, std::less<>>; using meta_data = std::map<string_view, string_view, std::less<>>;
using init_function = std::function<void ()>; using init_func = std::function<void ()>;
using fini_function = std::function<void ()>; using fini_func = std::function<void ()>;
/// Used to communicate whether a module unload actually took place. dlclose() is allowed to return
/// success but the actual static destruction of the module's contents doesn't lie. (mods.cc)
extern bool static_destruction;
/// The name of the header variable in the module must match this string
const char *const header_symbol_name const char *const header_symbol_name
{ {
"IRCD_MODULE" "IRCD_MODULE"
}; };
}
constexpr const magic_t MAGIC /// The magic number at the front of the header
constexpr const ircd::mapi::magic_t
IRCD_MAPI_MAGIC
{ {
0x4D41 0x4D41
}; };
// Used to communicate whether a module unload actually took place. dlclose() is allowed to return /// The version number of this module's header.
// success but the actual static destruction of the module's contents doesn't lie. (mods.cc) constexpr const ircd::mapi::version_t
extern bool static_destruction; IRCD_MAPI_VERSION
} {
4
};
struct ircd::mapi::header struct ircd::mapi::header
{ {
magic_t magic; // The magic must match MAGIC const magic_t magic {IRCD_MAPI_MAGIC}; // The magic must match
version_t version; // Version indicator const version_t version {IRCD_MAPI_VERSION}; // Version indicator
int64_t timestamp; // Module's compile epoch const int32_t _reserved_ {0}; // MBZ
init_function init; // Executed after dlopen() const int64_t timestamp {RB_DATECODE}; // Module's compile epoch
fini_function fini; // Executed before dlclose() std::unique_ptr<metablock> meta; // Non-standard-layout header data
metadata meta; // Various key-value metadata mods::mod *self {nullptr}; // Point to mod instance once loaded
mods::mod *self; // Module instance once loaded
// get and set metadata // get and set metadata
auto &operator[](const string_view &s) const; const string_view &operator[](const string_view &s) const;
auto &operator[](const string_view &s); string_view &operator[](const string_view &s);
// become self // become self
operator const mods::mod &() const; operator const mods::mod &() const;
operator mods::mod &(); operator mods::mod &();
header(const string_view &desc = "<no description>", header(const string_view & = "<no description>",
init_function = {}, init_func = {},
fini_function = {}); fini_func = {});
header(header &&) = delete;
header(const header &) = delete;
~header() noexcept; ~header() noexcept;
}; };
inline struct ircd::mapi::metablock
ircd::mapi::header::header(const string_view &desc,
init_function init,
fini_function fini)
:magic(MAGIC)
,version(4)
,timestamp(RB_DATECODE)
,init{std::move(init)}
,fini{std::move(fini)}
,meta
{ {
{ "description", desc } init_func init; // Executed after dlopen()
fini_func fini; // Executed before dlclose()
meta_data meta; // Various key-value metadata
};
inline
ircd::mapi::header::header(const string_view &description,
init_func init,
fini_func fini)
:meta
{
new metablock
{
std::move(init), std::move(fini), meta_data
{
{ "description", description }
}
}
} }
,self{nullptr}
{ {
} }
@ -85,30 +104,15 @@ noexcept
static_destruction = true; static_destruction = true;
} }
inline ircd::mapi::header::operator static_assert
mods::mod &() (
{ std::is_standard_layout<ircd::mapi::header>(),
assert(self); "The MAPI header must be standard layout so the magic and version numbers"
return *self; " can be parsed from the shared object file by external applications."
} );
inline ircd::mapi::header::operator static_assert
const mods::mod &() (
const sizeof(ircd::mapi::header) == 2 + 2 + 4 + 8 + 8 + 8,
{ "The MAPI header size has changed on this platform."
assert(self); );
return *self;
}
inline auto &
ircd::mapi::header::operator[](const string_view &key)
{
return meta[key];
}
inline auto &
ircd::mapi::header::operator[](const string_view &key)
const
{
return meta.at(key);
}

View file

@ -116,19 +116,18 @@ try
"Unexpected null header" "Unexpected null header"
}; };
if(header->magic != mapi::MAGIC) if(header->magic != IRCD_MAPI_MAGIC)
throw error throw error
{ {
"Bad magic [%04x] need: [%04x]", header->magic, mapi::MAGIC "Bad magic [%04x] need: [%04x]", header->magic, IRCD_MAPI_MAGIC
}; };
// Tell the module where to find us. // Tell the module where to find us.
header->self = this; header->self = this;
// Set some basic metadata // Set some basic metadata
auto &meta(header->meta); (*this)["name"] = name();
meta["name"] = name(); (*this)["location"] = location();
meta["location"] = location();
if(!loading.empty()) if(!loading.empty())
{ {
@ -214,8 +213,10 @@ ircd::mods::mod::unload()
}; };
// Call the user's unloading function here. // Call the user's unloading function here.
if(header->fini) assert(header);
header->fini(); assert(header->meta);
if(header->meta->fini)
header->meta->fini();
// Save the children! dlclose() does not like to be called recursively during static // Save the children! dlclose() does not like to be called recursively during static
// destruction of a module. The mod ctor recorded all of the modules loaded while this // destruction of a module. The mod ctor recorded all of the modules loaded while this
@ -256,6 +257,56 @@ ircd::mods::mod::unload()
return true; return true;
} }
ircd::string_view &
ircd::mods::mod::operator[](const string_view &key)
{
assert(header);
return header->operator[](key);
}
const ircd::string_view &
ircd::mods::mod::operator[](const string_view &key)
const
{
assert(header);
return header->operator[](key);
}
///////////////////////////////////////////////////////////////////////////////
//
// mods/mapi.h
//
ircd::mapi::header::operator
mods::mod &()
{
assert(self);
return *self;
}
ircd::mapi::header::operator
const mods::mod &()
const
{
assert(self);
return *self;
}
ircd::string_view &
ircd::mapi::header::operator[](const string_view &key)
{
assert(meta);
return meta->meta[key];
}
const ircd::string_view &
ircd::mapi::header::operator[](const string_view &key)
const
{
assert(meta);
return meta->meta.at(key);
}
/////////////////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////////////////
// //
// mods/mods.h (misc util) // mods/mods.h (misc util)
@ -594,8 +645,10 @@ try
// Call the user-supplied init function well after fully loading and // Call the user-supplied init function well after fully loading and
// construction of the module. This way the init function sees the module // construction of the module. This way the init function sees the module
// as loaded and can make shared_ptr references, etc. // as loaded and can make shared_ptr references, etc.
if(ret->header->init) assert(ret->header);
ret->header->init(); assert(ret->header->meta);
if(ret->header->meta->init)
ret->header->meta->init();
log::info log::info
{ {

View file

@ -54,9 +54,10 @@ struct ircd::mods::mod
mapi::header *header; mapi::header *header;
// Metadata // Metadata
auto &operator[](const std::string &s) const { return header->meta.operator[](s); } const string_view &operator[](const string_view &s) const;
auto &operator[](const std::string &s) { return header->meta.operator[](s); } string_view &operator[](const string_view &s);
// Convenience accessors
auto &name() const { return _name; } auto &name() const { return _name; }
auto &location() const { return _location; } auto &location() const { return _location; }
auto &version() const { return header->version; } auto &version() const { return header->version; }