source: trunk/src/core/library.cc @ 403

Last change on this file since 403 was 403, checked in by epyon, 10 years ago
  • got rid of exceptions
  • assert enhancements
  • lots of minor cleanup
File size: 4.0 KB
RevLine 
[395]1// Copyright (C) 2012-2015 ChaosForge Ltd
2// http://chaosforge.org/
3//
4// This file is part of Nova libraries.
5// For conditions of distribution and use, see copying.txt file in root folder.
6
[319]7#include "nv/core/library.hh"
[4]8
9#if NV_PLATFORM == NV_WINDOWS
10#   define WIN32_LEAN_AND_MEAN
11#   include <windows.h>
12#   define NV_LIB_EXT ".dll"
[402]13#   define NV_LIB_OPEN( name ) static_cast<void*>( LoadLibraryEx( name, NULL, LOAD_WITH_ALTERED_SEARCH_PATH ) )
14#   define NV_LIB_GET( handle, name ) reinterpret_cast<void*>( GetProcAddress( static_cast<HMODULE>( handle ), name ) )
15#   define NV_LIB_CLOSE( handle ) ( FreeLibrary( static_cast<HMODULE>( handle ) ) != 0 )
[113]16#elif NV_PLATFORM == NV_LINUX || NV_PLATFORM == NV_APPLE
[4]17#   include <dlfcn.h>
18#   define NV_LIB_EXT ".so"
19#   define NV_LIB_OPEN( name ) dlopen( name, RTLD_LAZY | RTLD_GLOBAL)
[402]20#   define NV_LIB_GET( handle, name ) dlsym( static_cast<void*>( handle ), name )
21#   define NV_LIB_CLOSE( handle ) ( dlclose( static_cast<void*>( handle ) ) == 0 )
[4]22#elif NV_PLATFORM == NV_APPLE
23#   include "macUtils.h"
24#   include <dlfcn.h>
25#   define NV_LIB_EXT ".dylib"
26#   define NV_LIB_OPEN( name ) mac_loadExeBundle( name )
27#   define NV_LIB_GET( handle, name ) mac_getBundleSym( handle, name )
[402]28#   define NV_LIB_CLOSE( handle ) ( mac_unloadExeBundle( handle ) == 0 )
[4]29#endif
30
[319]31#include "nv/core/logging.hh"
[4]32
33using namespace nv;
34
[109]35library::library()
[121]36    : m_handle( nullptr ), m_name()
[4]37{
38}
39
[399]40void library::open( string_view name )
[4]41{
[380]42        m_name.assign( name.data(), name.size() );
[166]43        if ( !open() )
44        {
45                m_handle = nullptr;
[403]46                NV_LOG_CRITICAL( "library \"", name, "\" : failed to load!" );
47                NV_ABORT( "Can't load library!" );
[166]48        }
[4]49}
50
[399]51bool nv::library::try_open( string_view name )
[166]52{
[380]53        m_name.assign( name.data(), name.size() );
[166]54        if ( !open() )
55        {
56                m_handle = nullptr;
57                return false;
58        }
59        return true;
60}
61
[399]62string_view library::get_name() const
[4]63{
[399]64    return string_view( m_name );
[4]65}
66
[166]67bool library::open( )
[4]68{
69    if ( m_handle != NULL )
70    {
[166]71        return true;
[4]72    }
[403]73    NV_LOG_NOTICE( "library \"", string_view( m_name ), "\" : loading..." );
[4]74
[380]75        std::string name = m_name;
[399]76        string_view ext( NV_LIB_EXT );
[4]77
[380]78        if ( name.length() < ext.length() || name.substr( name.length() - ext.length(), ext.length() ) != ext )
[4]79    {
[380]80        name.append( ext.data(), ext.length() );
[4]81    }
82
[402]83    m_handle = NV_LIB_OPEN( name.c_str() );
[4]84
85    if ( m_handle == NULL )
86    {
[403]87                NV_LOG_NOTICE( "library \"", string_view( name ), "\" : failed to open!" );
[166]88                return false;
[4]89    }
[403]90    NV_LOG_NOTICE( "library \"", string_view( name ), "\" : loaded." );
[166]91        return true;
[4]92}
93
[399]94void* library::get( string_view symbol )
[4]95{
[402]96        void* result = NV_LIB_GET( m_handle, symbol.data() );
[4]97    if ( !result )
98    {
[403]99                NV_LOG_CRITICAL( "library \"", string_view( m_name ), "\" : can't find symbol \"", symbol, "\"" );
100                NV_ABORT( "Library symbol load failed!" );
[4]101    }
102        return result;
103}
104
[399]105void* nv::library::try_get( string_view symbol )
[166]106{
[402]107        return NV_LIB_GET( m_handle, symbol.data() );
[166]108}
109
[109]110bool library::is_open() const
111{
112        return m_handle != nullptr;
113}
[4]114
115void library::close()
116{
[402]117    if ( ! NV_LIB_CLOSE( m_handle ) )
[4]118    {
[403]119        NV_LOG_ERROR( "library \"", string_view( m_name ), "\" : can't close library!" );
[4]120    }
[402]121    m_handle = nullptr;
[4]122}
123
124library::~library()
125{
[402]126    if ( m_handle != nullptr )
[4]127    {
128        close();
129    }
130}
131
[380]132std::string library::get_error()
[4]133{
134#if NV_PLATFORM == NV_WINDOWS
135    // We do hate WinAPI for code like this, don't we?
[402]136    LPTSTR buffer = nullptr;
[4]137    FormatMessage( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS,
[402]138        NULL, GetLastError(), MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), reinterpret_cast<LPTSTR>( &buffer ), 0, NULL );
139    std::string msg( reinterpret_cast<char*>( buffer ) );
[4]140    LocalFree( buffer );
141    return msg;
142#elif NV_PLATFORM == NV_LINUX || NV_PLATFORM == NV_APPLE
[402]143    return std::string( dlerror() );
[4]144#else
[402]145    return std::string("");
[4]146#endif
147}
Note: See TracBrowser for help on using the repository browser.