|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110 |
- /**
- * Copyright 2021 Huawei Technologies Co., Ltd
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- #include <algorithm>
- #include <numeric>
- #include <set>
- #include <string>
- #include <vector>
-
- #include "minddata/dataset/plugin/plugin_loader.h"
- #include "minddata/dataset/plugin/shared_lib_util.h"
- #include "mindspore/core/utils/log_adapter.h"
-
- namespace mindspore {
- namespace dataset {
-
- PluginLoader *PluginLoader::GetInstance() noexcept {
- static PluginLoader pl;
- return &pl;
- }
-
- PluginLoader::~PluginLoader() {
- std::vector<std::string> keys;
- // get the keys from map, this is to avoid concurrent iteration and delete
- std::transform(plugins_.begin(), plugins_.end(), std::back_inserter(keys), [](const auto &p) { return p.first; });
- for (std::string &key : keys) {
- Status rc = UnloadPlugin(key);
- MSLOG_IF(ERROR, rc.IsError(), mindspore::NoExceptionType) << rc.ToString();
- }
- }
-
- // LoadPlugin() is NOT thread-safe. It is supposed to be called when Ops are being built. E.g. PluginOp should call this
- // within constructor instead of in its Compute() which is parallel.
- Status PluginLoader::LoadPlugin(const std::string &filename, plugin::PluginManagerBase **singleton_plugin) {
- RETURN_UNEXPECTED_IF_NULL(singleton_plugin);
- auto itr = plugins_.find(filename);
- // return ok if this module is already loaded
- if (itr != plugins_.end()) {
- *singleton_plugin = itr->second.first;
- return Status::OK();
- }
- // Open the .so file
- void *handle = SharedLibUtil::Load(filename);
- CHECK_FAIL_RETURN_UNEXPECTED(handle != nullptr, "fail to load:" + filename + ".\n" + SharedLibUtil::ErrMsg());
-
- // Load GetInstance function ptr from the so file, so needs to be compiled with -fPIC
- void *func_handle = SharedLibUtil::FindSym(handle, "GetInstance");
- CHECK_FAIL_RETURN_UNEXPECTED(func_handle != nullptr, "fail to find GetInstance()\n" + SharedLibUtil::ErrMsg());
-
- // cast the returned function ptr of type void* to the type of GetInstance
- plugin::PluginManagerBase *(*get_instance)(plugin::MindDataManagerBase *) =
- reinterpret_cast<plugin::PluginManagerBase *(*)(plugin::MindDataManagerBase *)>(func_handle);
- RETURN_UNEXPECTED_IF_NULL(get_instance);
-
- *singleton_plugin = get_instance(nullptr); // call function ptr to get instance
- RETURN_UNEXPECTED_IF_NULL(*singleton_plugin);
-
- std::string v1 = (*singleton_plugin)->GetPluginVersion(), v2(plugin::kSharedIncludeVersion);
-
- // Version check, if version are not the same, log the error and return fail
- if (v1 != v2) {
- std::string err_msg = "[Plugin Version Error] expected:" + v2 + ", received:" + v1 + " please recompile.";
- if (SharedLibUtil::Close(handle) != 0) err_msg += ("\ndlclose() error, err_msg:" + SharedLibUtil::ErrMsg() + ".");
- RETURN_STATUS_UNEXPECTED(err_msg);
- }
-
- const std::map<std::string, std::set<std::string>> module_names = (*singleton_plugin)->GetModuleNames();
- for (auto &p : module_names) {
- std::string msg = "Plugin " + p.first + " has module:";
- MS_LOG(DEBUG) << std::accumulate(p.second.begin(), p.second.end(), msg,
- [](const std::string &msg, const std::string &nm) { return msg + " " + nm; });
- }
-
- // save the name and handle
- plugins_.insert({filename, {*singleton_plugin, handle}});
- return Status::OK();
- }
-
- Status PluginLoader::UnloadPlugin(const std::string &filename) {
- auto itr = plugins_.find(filename);
- RETURN_OK_IF_TRUE(itr == plugins_.end()); // return true if this plugin was never loaded or already removed
-
- void *func_handle = SharedLibUtil::FindSym(itr->second.second, "DestroyInstance");
- CHECK_FAIL_RETURN_UNEXPECTED(func_handle != nullptr, "fail to find DestroyInstance()\n" + SharedLibUtil::ErrMsg());
-
- void (*destroy_instance)() = reinterpret_cast<void (*)()>(func_handle);
- RETURN_UNEXPECTED_IF_NULL(destroy_instance);
-
- destroy_instance();
- CHECK_FAIL_RETURN_UNEXPECTED(SharedLibUtil::Close(itr->second.second) == 0,
- "dlclose() error: " + SharedLibUtil::ErrMsg());
-
- plugins_.erase(filename);
- return Status::OK();
- }
-
- } // namespace dataset
- } // namespace mindspore
|