#ifndef _sys_Module_h #define _sys_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 #include #include namespace qpid { namespace sys { #if USE_APR #include typedef apr_dso_handle_t* dso_handle_t; #else typedef void* dso_handle_t; #endif 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(); }; } } using namespace qpid::sys; 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) { std::cout << "Error while destroying module: " << e.what() << std::endl; } destroy = 0; handle = 0; ptr = 0; } // APR ================================================================ #if USE_APR #include #include template void Module::load(const std::string& name) { CHECK_APR_SUCCESS(apr_dso_load(&handle, name.c_str(), APRPool::get())); } template void Module::unload() { CHECK_APR_SUCCESS(apr_dso_unload(handle)); } template void* Module::getSymbol(const std::string& name) { apr_dso_handle_sym_t symbol; CHECK_APR_SUCCESS(apr_dso_sym(&symbol, handle, name.c_str())); return (void*) symbol; } // POSIX================================================================ #else #include 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 //if USE_APR #endif //ifndef _sys_Module_h