@@ -88,36 +88,86 @@ fn mmap(path: &Path) -> Option<Mmap> {
88
88
89
89
cfg_if:: cfg_if! {
90
90
if #[ cfg( windows) ] {
91
- // Windows uses COFF object files and currently doesn't implement
92
- // functionality to load a list of native libraries. This seems to work
93
- // well enough for the main executable but seems pretty likely to not
94
- // work for loaded DLLs. For now this seems sufficient, but we may have
95
- // to extend this over time.
96
- //
97
- // Note that the native_libraries loading here simply returns one
98
- // library encompassing the entire address space. This works naively
99
- // but likely indicates something about ASLR is busted. Let's try to
100
- // fix this over time if necessary!
91
+ use core:: mem:: MaybeUninit ;
92
+ use crate :: windows:: * ;
93
+ use std:: os:: windows:: prelude:: * ;
101
94
102
95
mod coff;
103
96
use self :: coff:: Object ;
104
97
105
98
fn native_libraries( ) -> Vec <Library > {
106
99
let mut ret = Vec :: new( ) ;
107
- if let Ok ( path) = std:: env:: current_exe( ) {
108
- let mut segments = Vec :: new( ) ;
109
- segments. push( LibrarySegment {
110
- stated_virtual_memory_address: 0 ,
111
- len: usize :: max_value( ) ,
112
- } ) ;
113
- ret. push( Library {
114
- name: path. into( ) ,
115
- segments,
116
- bias: 0 ,
117
- } ) ;
118
- }
100
+ unsafe { add_loaded_images( & mut ret) ; }
119
101
return ret;
120
102
}
103
+
104
+ unsafe fn add_loaded_images( ret: & mut Vec <Library >) {
105
+ let mut system_info = MaybeUninit :: zeroed( ) . assume_init( ) ;
106
+ GetSystemInfo ( & mut system_info) ;
107
+
108
+ let mut memory_basic_info = MaybeUninit :: zeroed( ) . assume_init( ) ;
109
+ let query_size = mem:: size_of_val( & memory_basic_info) ;
110
+
111
+ let start = system_info. lpMinimumApplicationAddress;
112
+ let end = system_info. lpMaximumApplicationAddress;
113
+ let mut base = start;
114
+ while base < end {
115
+ if VirtualQuery ( base, & mut memory_basic_info, query_size) == 0 {
116
+ break ;
117
+ }
118
+ if !base. is_null( ) {
119
+ let alloc_base = memory_basic_info. AllocationBase ;
120
+
121
+ // If this `alloc_base` is the same as our base address we
122
+ // queried, then we found a new library.
123
+ if alloc_base == base {
124
+ if let Some ( lib) = get_library( base, & memory_basic_info) {
125
+ ret. push( lib) ;
126
+ }
127
+
128
+ // ... otherwise see if the allocation base is the same as
129
+ // the preivous library, and if so we just discovered a new
130
+ // segment for the previous library.
131
+ } else if let Some ( last) = ret. last_mut( ) {
132
+ if last. segments[ 0 ] . stated_virtual_memory_address == alloc_base as usize {
133
+ last. segments. push( LibrarySegment {
134
+ stated_virtual_memory_address: base as usize ,
135
+ len: memory_basic_info. RegionSize ,
136
+ } ) ;
137
+ }
138
+ }
139
+ }
140
+
141
+ base = ( ( base as usize ) + memory_basic_info. RegionSize ) as LPVOID ;
142
+ }
143
+ }
144
+
145
+ unsafe fn get_library( base: LPVOID , info: & MEMORY_BASIC_INFORMATION ) -> Option <Library > {
146
+ let mut buf = Vec :: with_capacity( 128 ) ;
147
+ let module = base as HMODULE ;
148
+ loop {
149
+ let len = GetModuleFileNameW ( module, buf. as_mut_ptr( ) , buf. capacity( ) as u32 ) ;
150
+ if len == 0 {
151
+ return None ;
152
+ }
153
+ if len as usize == buf. capacity( ) && GetLastError ( ) == ERROR_INSUFFICIENT_BUFFER {
154
+ let new_cap = buf. capacity( ) * 2 ;
155
+ buf. reserve( new_cap) ;
156
+ } else {
157
+ buf. set_len( len as usize ) ;
158
+ break ;
159
+ }
160
+ }
161
+ let filename = OsString :: from_wide( & buf[ ..] ) ;
162
+ Some ( Library {
163
+ name: filename. into( ) ,
164
+ segments: vec![ LibrarySegment {
165
+ stated_virtual_memory_address: base as usize ,
166
+ len: info. RegionSize ,
167
+ } ] ,
168
+ bias: 0 ,
169
+ } )
170
+ }
121
171
} else if #[ cfg( target_os = "macos" ) ] {
122
172
// macOS uses the Mach-O file format and uses DYLD-specific APIs to
123
173
// load a list of native libraries that are part of the appplication.
0 commit comments