#ifndef _sys_posix_Module_h #define _sys_posix_Module_h /* * * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you 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 "qpid/QpidError.h" #include "qpid/log/Statement.h" #include #include #include namespace qpid { namespace sys { typedef void* dso_handle_t; template class Module : private boost::noncopyable { typedef T* create_t(); typedef void destroy_t(T*); dso_handle_t handle; destroy_t* destroy; T* ptr; void load(const std::string& name); void unload(); void* getSymbol(const std::string& name); public: Module(const std::string& name); T* operator->(); T* get(); ~Module() throw(); }; template Module::Module(const std::string& module) : destroy(0), ptr(0) { load(module); //TODO: need a better strategy for symbol names to allow multiple //modules to be loaded without clashes... //Note: need the double cast to avoid errors in casting from void* to function pointer with -pedantic create_t* create = reinterpret_cast(reinterpret_cast(getSymbol("create"))); destroy = reinterpret_cast(reinterpret_cast(getSymbol("destroy"))); ptr = create(); } template T* Module::operator->() { return ptr; } template T* Module::get() { return ptr; } template Module::~Module() throw() { try { if (handle && ptr) { destroy(ptr); } if (handle) unload(); } catch (std::exception& e) { QPID_LOG(error, "Error while destroying module: " << e.what()); } destroy = 0; handle = 0; ptr = 0; } template void Module::load(const std::string& name) { ::dlerror(); handle = ::dlopen(name.c_str(), RTLD_NOW); const char* error = ::dlerror(); if (error) { THROW_QPID_ERROR(INTERNAL_ERROR, error); } } template void Module::unload() { dlerror(); dlclose(handle); const char* error = dlerror(); if (error) { THROW_QPID_ERROR(INTERNAL_ERROR, error); } } template void* Module::getSymbol(const std::string& name) { dlerror(); void* sym = dlsym(handle, name.c_str()); const char* error = dlerror(); if (error) { THROW_QPID_ERROR(INTERNAL_ERROR, error); } return sym; } }} #endif //ifndef _sys_posix_Module_h