diff options
author | ADAM David Alan Martin <adam.martin@10gen.com> | 2018-05-24 17:28:38 -0400 |
---|---|---|
committer | ADAM David Alan Martin <adam.martin@10gen.com> | 2018-05-24 17:40:11 -0400 |
commit | 49d89c0f4378787aa47d30ab253f431897b457f0 (patch) | |
tree | 9dd7fc62b996f59d431a1364c12fb7b6a4500955 | |
parent | a6bc80ed0712827f4793284c9d3a582260272e8b (diff) | |
download | mongo-49d89c0f4378787aa47d30ab253f431897b457f0.tar.gz |
SERVER-34944 Create a shim-override mechanism.
Modules for MongoDB need to be able to indicate their own overrides
of shimmed funcitons. These overrides must be able to access the
underlying registered shim. Further, the overrides must be installed
in the correct position in the initializer dependency graph.
(cherry picked from commit 1de64a4eb0a3655df083bc2160e5720d66f3c6fe)
-rw-r--r-- | src/mongo/base/shim.h | 66 | ||||
-rw-r--r-- | src/mongo/db/auth/authorization_manager_global.cpp | 5 | ||||
-rw-r--r-- | src/mongo/db/catalog/SConscript | 7 |
3 files changed, 70 insertions, 8 deletions
diff --git a/src/mongo/base/shim.h b/src/mongo/base/shim.h index a782109d24d..14de94df7c8 100644 --- a/src/mongo/base/shim.h +++ b/src/mongo/base/shim.h @@ -193,6 +193,15 @@ const bool checkShimsViaTUHook = false; } \ } + +/** + * Evaluates to a string which represents the `MONGO_INITIALIZER` step name in which this specific + * shim is registered. This can be useful to make sure that some initializers depend upon a shim's + * execution and presence in a binary. + */ +#define MONGO_SHIM_DEPENDENCY(...) MONGO_SHIM_EVIL_STRINGIFY_(__VA_ARGS__) +#define MONGO_SHIM_EVIL_STRINGIFY_(args) #args + /** * Define a shimmable function with name `SHIM_NAME`, returning a value of type `RETURN_TYPE`, with * any arguments. This shim definition macro should go in the associated C++ file to the header @@ -205,20 +214,25 @@ const bool checkShimsViaTUHook = false; namespace { \ namespace shim_namespace##LN { \ using ShimType = decltype(__VA_ARGS__); \ + ::mongo::Status initializerGroupStartup(::mongo::InitializerContext*) { \ + return Status::OK(); \ + } \ + ::mongo::GlobalInitializerRegisterer _mongoInitializerRegisterer( \ + std::string(MONGO_SHIM_DEPENDENCY(__VA_ARGS__)), \ + {}, \ + {MONGO_SHIM_DEPENDENTS}, \ + mongo::InitializerFunction(initializerGroupStartup)); \ } /*namespace shim_namespace*/ \ } /*namespace*/ \ shim_namespace##LN::ShimType::MongoShimImplGuts::LibTUHookTypeBase::LibTUHookTypeBase() = \ default; \ shim_namespace##LN::ShimType __VA_ARGS__{}; -#define MONGO_SHIM_EVIL_STRINGIFY_(args) #args - - /** * Define an implementation of a shimmable function with name `SHIM_NAME`. The compiler will check * supplied parameters for correctness. This shim registration macro should go in the associated * C++ implementation file to the header where a SHIM was defined. Such a file would be a mock - * implementation or a real implementation, for example + * implementation or a real implementation, for example. */ #define MONGO_REGISTER_SHIM(/*SHIM_NAME*/...) MONGO_REGISTER_SHIM_1(__LINE__, __VA_ARGS__) #define MONGO_REGISTER_SHIM_1(LN, ...) MONGO_REGISTER_SHIM_2(LN, __VA_ARGS__) @@ -240,9 +254,9 @@ const bool checkShimsViaTUHook = false; } \ \ const ::mongo::GlobalInitializerRegisterer registrationHook{ \ - std::string(MONGO_SHIM_EVIL_STRINGIFY_(__VA_ARGS__)), \ + std::string(MONGO_SHIM_DEPENDENCY(__VA_ARGS__) "_registration"), \ {}, \ - {MONGO_SHIM_DEPENDENTS}, \ + {MONGO_SHIM_DEPENDENCY(__VA_ARGS__), MONGO_SHIM_DEPENDENTS}, \ mongo::InitializerFunction(createInitializerRegistration)}; \ } /*namespace shim_namespace*/ \ } /*namespace*/ \ @@ -255,3 +269,43 @@ const bool checkShimsViaTUHook = false; and return value (using arrow \ notation). Then they write the \ body. */ + +/** + * Define an overriding implementation of a shimmable function with `SHIM_NAME`. The compiler will + * check the supplied parameters for correctness. This shim override macro should go in the + * associated C++ implementation file to the header where a SHIM was defined. Such a file + * specifying an override would be a C++ implementation used by a mongodb extension module. + * This creates a runtime dependency upon the original registration being linked in. + */ +#define MONGO_OVERRIDE_SHIM(/*SHIM_NAME*/...) MONGO_OVERRIDE_SHIM_1(__LINE__, __VA_ARGS__) +#define MONGO_OVERRIDE_SHIM_1(LN, ...) MONGO_OVERRIDE_SHIM_2(LN, __VA_ARGS__) +#define MONGO_OVERRIDE_SHIM_2(LN, ...) \ + namespace { \ + namespace shim_namespace##LN { \ + using ShimType = decltype(__VA_ARGS__); \ + \ + class OverrideImplementation final : public ShimType::MongoShimImplGuts { \ + /* Some compilers don't work well with the trailing `override` in this kind of \ + * function declaration. */ \ + ShimType::MongoShimImplGuts::function_type implementation; /* override */ \ + }; \ + \ + ::mongo::Status createInitializerOverride(::mongo::InitializerContext* const) { \ + static OverrideImplementation overrideImpl; \ + ShimType::storage::data = &overrideImpl; \ + return Status::OK(); \ + } \ + \ + const ::mongo::GlobalInitializerRegisterer overrideHook{ \ + std::string(MONGO_SHIM_DEPENDENCY(__VA_ARGS__) "_override"), \ + {MONGO_SHIM_DEPENDENCY( \ + __VA_ARGS__) "_registration"}, /* Override happens after first registration */ \ + {MONGO_SHIM_DEPENDENCY(__VA_ARGS__), /* Provides impl for this shim */ \ + MONGO_SHIM_DEPENDENTS}, /* Still a shim registration */ \ + mongo::InitializerFunction(createInitializerOverride)}; \ + } /*namespace shim_namespace*/ \ + } /*namespace*/ \ + \ + auto shim_namespace##LN::OverrideImplementation:: \ + implementation /* After this point someone just writes the signature's arguments and \ + return value (using arrow notation). Then they write the body. */ diff --git a/src/mongo/db/auth/authorization_manager_global.cpp b/src/mongo/db/auth/authorization_manager_global.cpp index b626c3b5573..3f8a8808d04 100644 --- a/src/mongo/db/auth/authorization_manager_global.cpp +++ b/src/mongo/db/auth/authorization_manager_global.cpp @@ -93,7 +93,10 @@ MONGO_EXPORT_STARTUP_SERVER_PARAMETER(startupAuthSchemaValidation, bool, true); GlobalInitializerRegisterer authorizationManagerInitializer( "CreateAuthorizationManager", - {"OIDGeneration", "EndStartupOptionStorage", "ServiceContext"}, + {MONGO_SHIM_DEPENDENCY(AuthorizationManager::create), + "OIDGeneration", + "EndStartupOptionStorage", + "ServiceContext"}, [](InitializerContext* context) { auto authzManager = AuthorizationManager::create(); authzManager->setAuthEnabled(serverGlobalParams.authState == diff --git a/src/mongo/db/catalog/SConscript b/src/mongo/db/catalog/SConscript index 259ea12ab77..5ced8ff384b 100644 --- a/src/mongo/db/catalog/SConscript +++ b/src/mongo/db/catalog/SConscript @@ -103,6 +103,7 @@ env.Library( "index_catalog.cpp", ], LIBDEPS=[ + '$BUILD_DIR/mongo/base', ], ) @@ -112,6 +113,7 @@ env.Library( "index_catalog_entry.cpp", ], LIBDEPS=[ + '$BUILD_DIR/mongo/base', ], ) @@ -158,9 +160,10 @@ env.CppUnitTest( env.Library( target='database_holder', source=[ - "database_holder.cpp", + 'database_holder.cpp', ], LIBDEPS=[ + '$BUILD_DIR/mongo/base', ], ) @@ -170,6 +173,7 @@ env.Library( 'collection_info_cache.cpp', ], LIBDEPS=[ + '$BUILD_DIR/mongo/base', ], ) @@ -179,6 +183,7 @@ env.Library( 'index_create.cpp', ], LIBDEPS=[ + '$BUILD_DIR/mongo/base', ], ) |