diff --git a/ext/fiddle/extconf.rb b/ext/fiddle/extconf.rb index 8cc98fb8f620d9..053456d534d933 100644 --- a/ext/fiddle/extconf.rb +++ b/ext/fiddle/extconf.rb @@ -187,6 +187,7 @@ def enable_debug_build_flag(flags) end have_header 'sys/mman.h' +have_header 'link.h' if have_header "dlfcn.h" have_library "dl" @@ -196,8 +197,10 @@ def enable_debug_build_flag(flags) end have_func "dlerror" + have_func "dlinfo" + have_const("RTLD_DI_LINKMAP", "dlfcn.h") elsif have_header "windows.h" - %w{ LoadLibrary FreeLibrary GetProcAddress }.each do |func| + %w{ LoadLibrary FreeLibrary GetProcAddress GetModuleFileName }.each do |func| abort "missing function #{func}" unless have_func(func) end diff --git a/ext/fiddle/fiddle.h b/ext/fiddle/fiddle.h index bf97b59d538bc3..9de62a58cc8b9b 100644 --- a/ext/fiddle/fiddle.h +++ b/ext/fiddle/fiddle.h @@ -12,6 +12,10 @@ #include #endif +#if defined(HAVE_LINK_H) +# include +#endif + #if defined(HAVE_DLFCN_H) # include # /* some stranger systems may not define all of these */ diff --git a/ext/fiddle/handle.c b/ext/fiddle/handle.c index a4a32f1ecba635..76b90909d30a96 100644 --- a/ext/fiddle/handle.c +++ b/ext/fiddle/handle.c @@ -386,6 +386,48 @@ fiddle_handle_sym(void *handle, VALUE symbol) return PTR2NUM(func); } +/* + * call-seq: file_name + * + * Returns the file name of this handle. + */ +static VALUE +rb_fiddle_handle_file_name(VALUE self) +{ + struct dl_handle *fiddle_handle; + + TypedData_Get_Struct(self, struct dl_handle, &fiddle_handle_data_type, fiddle_handle); + +#if defined(HAVE_DLINFO) && defined(HAVE_CONST_RTLD_DI_LINKMAP) + { + struct link_map *lm = NULL; + int res = dlinfo(fiddle_handle->ptr, RTLD_DI_LINKMAP, &lm); + if (res == 0 && lm != NULL) { + return rb_str_new_cstr(lm->l_name); + } + else { +#if defined(HAVE_DLERROR) + rb_raise(rb_eFiddleDLError, "could not get handle file name: %s", dlerror()); +#else + rb_raise(rb_eFiddleDLError, "could not get handle file name"); +#endif + } + } +#elif defined(HAVE_GETMODULEFILENAME) + { + char filename[MAX_PATH]; + DWORD res = GetModuleFileName(fiddle_handle->ptr, filename, MAX_PATH); + if (res == 0) { + rb_raise(rb_eFiddleDLError, "could not get handle file name: %s", dlerror()); + } + return rb_str_new_cstr(filename); + } +#else + (void)fiddle_handle; + return Qnil; +#endif +} + void Init_fiddle_handle(void) { @@ -484,6 +526,7 @@ Init_fiddle_handle(void) rb_define_method(rb_cHandle, "close", rb_fiddle_handle_close, 0); rb_define_method(rb_cHandle, "sym", rb_fiddle_handle_sym, 1); rb_define_method(rb_cHandle, "[]", rb_fiddle_handle_sym, 1); + rb_define_method(rb_cHandle, "file_name", rb_fiddle_handle_file_name, 0); rb_define_method(rb_cHandle, "disable_close", rb_fiddle_handle_disable_close, 0); rb_define_method(rb_cHandle, "enable_close", rb_fiddle_handle_enable_close, 0); rb_define_method(rb_cHandle, "close_enabled?", rb_fiddle_handle_close_enabled_p, 0); diff --git a/test/fiddle/test_handle.rb b/test/fiddle/test_handle.rb index e89ad53e997055..5c151738ef7a17 100644 --- a/test/fiddle/test_handle.rb +++ b/test/fiddle/test_handle.rb @@ -112,6 +112,12 @@ def test_disable_close assert !handle.close_enabled?, 'close is enabled' end + def test_file_name + handle = Handle.new(LIBC_SO) + assert_kind_of String, handle.file_name + assert_equal File.basename(handle.file_name), File.basename(LIBC_SO) + end unless /darwin/ =~ RUBY_PLATFORM + def test_NEXT begin # Linux / Darwin