2 * File macho_module.c - processing of Mach-O files
3 * Originally based on elf_module.c
5 * Copyright (C) 1996, Eric Youngdale.
6 * 1999-2007 Eric Pouech
7 * 2009 Ken Thomases, CodeWeavers Inc.
9 * This library is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU Lesser General Public
11 * License as published by the Free Software Foundation; either
12 * version 2.1 of the License, or (at your option) any later version.
14 * This library is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 * Lesser General Public License for more details.
19 * You should have received a copy of the GNU Lesser General Public
20 * License along with this library; if not, write to the Free Software
21 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
25 #include "wine/port.h"
28 #define WIN32_NO_STATUS
29 #include "dbghelp_private.h"
31 #ifdef HAVE_MACH_O_LOADER_H
36 #ifdef HAVE_SYS_STAT_H
37 # include <sys/stat.h>
39 #ifdef HAVE_SYS_MMAN_H
40 # include <sys/mman.h>
43 #include <mach-o/fat.h>
44 #include <mach-o/loader.h>
45 #include <mach-o/nlist.h>
47 #ifdef HAVE_MACH_O_DYLD_IMAGES_H
48 #include <mach-o/dyld_images.h>
50 struct dyld_image_info
{
51 const struct mach_header
*imageLoadAddress
;
52 const char *imageFilePath
;
53 uintptr_t imageFileModDate
;
56 struct dyld_all_image_infos
{
58 uint32_t infoArrayCount
;
59 const struct dyld_image_info
*infoArray
;
61 int processDetachedFromSharedRegion
;
66 #include "wine/library.h"
67 #include "wine/debug.h"
69 #ifdef WORDS_BIGENDIAN
70 #define swap_ulong_be_to_host(n) (n)
72 #define swap_ulong_be_to_host(n) (RtlUlongByteSwap(n))
75 WINE_DEFAULT_DEBUG_CHANNEL(dbghelp_macho
);
78 struct macho_module_info
80 unsigned long load_addr
;
81 unsigned short in_use
: 1,
85 #define MACHO_INFO_DEBUG_HEADER 0x0001
86 #define MACHO_INFO_MODULE 0x0002
87 #define MACHO_INFO_NAME 0x0004
91 unsigned flags
; /* IN one (or several) of the MACHO_INFO constants */
92 unsigned long dbg_hdr_addr
; /* OUT address of debug header (if MACHO_INFO_DEBUG_HEADER is set) */
93 struct module
* module
; /* OUT loaded module (if MACHO_INFO_MODULE is set) */
94 const WCHAR
* module_name
; /* OUT found module name (if MACHO_INFO_NAME is set) */
97 /* structure holding information while handling a Mach-O image */
98 #define BITS_PER_ULONG (sizeof(ULONG) * 8)
99 #define ULONGS_FOR_BITS(nbits) (((nbits) + BITS_PER_ULONG - 1) / BITS_PER_ULONG)
100 struct macho_file_map
102 /* A copy of the Mach-O header for an individual architecture. */
103 struct mach_header mach_header
;
105 /* The mapped load commands. */
106 const struct load_command
* load_commands
;
108 /* The portion of the file which is this architecture. mach_header was
109 * read from arch_offset. */
110 unsigned arch_offset
;
113 /* The range of address space covered by all segments. */
117 /* Map of which sections contain code. Sections are accessed using 1-based
118 * index. Bit 0 of this bitset indicates if the bitset has been initialized. */
119 RTL_BITMAP sect_is_code
;
120 ULONG sect_is_code_buff
[ULONGS_FOR_BITS(MAX_SECT
+ 1)];
126 static void macho_unmap_file(struct macho_file_map
* fmap
);
128 /******************************************************************
131 * For a range (offset & length) of a single architecture within
132 * a Mach-O file, calculate the page-aligned range of the whole file
133 * that encompasses it. For a fat binary, the architecture will
134 * itself be offset within the file, so take that into account.
136 static void macho_calc_range(const struct macho_file_map
* fmap
, unsigned offset
,
137 unsigned len
, unsigned* out_aligned_offset
,
138 unsigned* out_aligned_end
, unsigned* out_aligned_len
,
139 unsigned* out_misalign
)
141 unsigned pagemask
= sysconf( _SC_PAGESIZE
) - 1;
142 unsigned file_offset
, misalign
;
144 file_offset
= fmap
->arch_offset
+ offset
;
145 misalign
= file_offset
& pagemask
;
146 *out_aligned_offset
= file_offset
- misalign
;
147 *out_aligned_end
= (file_offset
+ len
+ pagemask
) & ~pagemask
;
149 *out_aligned_len
= *out_aligned_end
- *out_aligned_offset
;
151 *out_misalign
= misalign
;
154 /******************************************************************
157 * Maps a range (offset, length in bytes) from a Mach-O file into memory
159 static const char* macho_map_range(const struct macho_file_map
* fmap
, unsigned offset
, unsigned len
)
161 unsigned misalign
, aligned_offset
, aligned_map_end
, map_size
;
162 const void* aligned_ptr
;
164 TRACE("(%p/%d, 0x%08x, 0x%08x)\n", fmap
, fmap
->fd
, offset
, len
);
166 macho_calc_range(fmap
, offset
, len
, &aligned_offset
, &aligned_map_end
,
167 &map_size
, &misalign
);
169 aligned_ptr
= mmap(NULL
, map_size
, PROT_READ
, MAP_PRIVATE
, fmap
->fd
, aligned_offset
);
171 TRACE("Mapped (0x%08x - 0x%08x) to %p\n", aligned_offset
, aligned_map_end
, aligned_ptr
);
173 if (aligned_ptr
== MAP_FAILED
) return MACHO_NO_MAP
;
174 return (const char*)aligned_ptr
+ misalign
;
177 /******************************************************************
180 * Unmaps a range (offset, length in bytes) of a Mach-O file from memory
182 static void macho_unmap_range(const void** mapped
, const struct macho_file_map
* fmap
,
183 unsigned offset
, unsigned len
)
185 TRACE("(%p, %p/%d, 0x%08x, 0x%08x)\n", mapped
, fmap
, fmap
->fd
, offset
, len
);
187 if (mapped
&& *mapped
!= MACHO_NO_MAP
)
189 unsigned misalign
, aligned_offset
, aligned_map_end
, map_size
;
192 macho_calc_range(fmap
, offset
, len
, &aligned_offset
, &aligned_map_end
,
193 &map_size
, &misalign
);
195 aligned_ptr
= (char*)*mapped
- misalign
;
196 if (munmap(aligned_ptr
, map_size
) < 0)
197 WARN("Couldn't unmap the range\n");
198 TRACE("Unmapped (0x%08x - 0x%08x) from %p - %p\n", aligned_offset
, aligned_map_end
, aligned_ptr
, (char*)aligned_ptr
+ map_size
);
199 *mapped
= MACHO_NO_MAP
;
203 /******************************************************************
206 * Maps two ranges (offset, length in bytes) from a Mach-O file
207 * into memory. If the two ranges overlap, use one mmap so that
208 * the munmap doesn't fragment the mapping.
210 static BOOL
macho_map_ranges(const struct macho_file_map
* fmap
,
211 unsigned offset1
, unsigned len1
,
212 unsigned offset2
, unsigned len2
,
213 const void** mapped1
, const void** mapped2
)
215 unsigned aligned_offset1
, aligned_map_end1
;
216 unsigned aligned_offset2
, aligned_map_end2
;
218 TRACE("(%p/%d, 0x%08x, 0x%08x, 0x%08x, 0x%08x, %p, %p)\n", fmap
, fmap
->fd
,
219 offset1
, len1
, offset2
, len2
, mapped1
, mapped2
);
221 macho_calc_range(fmap
, offset1
, len1
, &aligned_offset1
, &aligned_map_end1
, NULL
, NULL
);
222 macho_calc_range(fmap
, offset2
, len2
, &aligned_offset2
, &aligned_map_end2
, NULL
, NULL
);
224 if (aligned_map_end1
< aligned_offset2
|| aligned_map_end2
< aligned_offset1
)
226 *mapped1
= macho_map_range(fmap
, offset1
, len1
);
227 if (*mapped1
!= MACHO_NO_MAP
)
229 *mapped2
= macho_map_range(fmap
, offset2
, len2
);
230 if (*mapped2
== MACHO_NO_MAP
)
231 macho_unmap_range(mapped1
, fmap
, offset1
, len1
);
236 if (offset1
< offset2
)
238 *mapped1
= macho_map_range(fmap
, offset1
, offset2
+ len2
- offset1
);
239 if (*mapped1
!= MACHO_NO_MAP
)
240 *mapped2
= (const char*)*mapped1
+ offset2
- offset1
;
244 *mapped2
= macho_map_range(fmap
, offset2
, offset1
+ len1
- offset2
);
245 if (*mapped2
!= MACHO_NO_MAP
)
246 *mapped1
= (const char*)*mapped2
+ offset1
- offset2
;
250 TRACE(" => %p, %p\n", *mapped1
, *mapped2
);
252 return (*mapped1
!= MACHO_NO_MAP
) && (*mapped2
!= MACHO_NO_MAP
);
255 /******************************************************************
258 * Unmaps two ranges (offset, length in bytes) of a Mach-O file
259 * from memory. Use for ranges which were mapped by
262 static void macho_unmap_ranges(const struct macho_file_map
* fmap
,
263 unsigned offset1
, unsigned len1
,
264 unsigned offset2
, unsigned len2
,
265 const void** mapped1
, const void** mapped2
)
267 unsigned aligned_offset1
, aligned_map_end1
;
268 unsigned aligned_offset2
, aligned_map_end2
;
270 TRACE("(%p/%d, 0x%08x, 0x%08x, 0x%08x, 0x%08x, %p/%p, %p/%p)\n", fmap
, fmap
->fd
,
271 offset1
, len1
, offset2
, len2
, mapped1
, *mapped1
, mapped2
, *mapped2
);
273 macho_calc_range(fmap
, offset1
, len1
, &aligned_offset1
, &aligned_map_end1
, NULL
, NULL
);
274 macho_calc_range(fmap
, offset2
, len2
, &aligned_offset2
, &aligned_map_end2
, NULL
, NULL
);
276 if (aligned_map_end1
< aligned_offset2
|| aligned_map_end2
< aligned_offset1
)
278 macho_unmap_range(mapped1
, fmap
, offset1
, len1
);
279 macho_unmap_range(mapped2
, fmap
, offset2
, len2
);
283 if (offset1
< offset2
)
285 macho_unmap_range(mapped1
, fmap
, offset1
, offset2
+ len2
- offset1
);
286 *mapped2
= MACHO_NO_MAP
;
290 macho_unmap_range(mapped2
, fmap
, offset2
, offset1
+ len1
- offset2
);
291 *mapped1
= MACHO_NO_MAP
;
296 /******************************************************************
297 * macho_map_load_commands
299 * Maps the load commands from a Mach-O file into memory
301 static const struct load_command
* macho_map_load_commands(struct macho_file_map
* fmap
)
303 if (fmap
->load_commands
== MACHO_NO_MAP
)
305 fmap
->load_commands
= (const struct load_command
*) macho_map_range(
306 fmap
, sizeof(fmap
->mach_header
), fmap
->mach_header
.sizeofcmds
);
307 TRACE("Mapped load commands: %p\n", fmap
->load_commands
);
310 return fmap
->load_commands
;
313 /******************************************************************
314 * macho_unmap_load_commands
316 * Unmaps the load commands of a Mach-O file from memory
318 static void macho_unmap_load_commands(struct macho_file_map
* fmap
)
320 if (fmap
->load_commands
!= MACHO_NO_MAP
)
322 TRACE("Unmapping load commands: %p\n", fmap
->load_commands
);
323 macho_unmap_range((const void**)&fmap
->load_commands
, fmap
,
324 sizeof(fmap
->mach_header
), fmap
->mach_header
.sizeofcmds
);
328 /******************************************************************
329 * macho_next_load_command
331 * Advance to the next load command
333 static const struct load_command
* macho_next_load_command(const struct load_command
* lc
)
335 return (const struct load_command
*)((const char*)lc
+ lc
->cmdsize
);
338 /******************************************************************
339 * macho_enum_load_commands
341 * Enumerates the load commands for a Mach-O file, selecting by
342 * the command type, calling a callback for each. If the callback
343 * returns <0, that indicates an error. If it returns >0, that means
344 * it's not interested in getting any more load commands.
345 * If this function returns <0, that's an error produced by the
346 * callback. If >=0, that's the count of load commands successfully
349 static int macho_enum_load_commands(struct macho_file_map
* fmap
, unsigned cmd
,
350 int (*cb
)(struct macho_file_map
*, const struct load_command
*, void*),
353 const struct load_command
* lc
;
357 TRACE("(%p/%d, %u, %p, %p)\n", fmap
, fmap
->fd
, cmd
, cb
, user
);
359 if ((lc
= macho_map_load_commands(fmap
)) == MACHO_NO_MAP
) return -1;
361 TRACE("%d total commands\n", fmap
->mach_header
.ncmds
);
363 for (i
= 0; i
< fmap
->mach_header
.ncmds
; i
++, lc
= macho_next_load_command(lc
))
367 if (cmd
&& cmd
!= lc
->cmd
) continue;
370 result
= cb(fmap
, lc
, user
);
371 TRACE("load_command[%d] (%p), cmd %u; callback => %d\n", i
, lc
, lc
->cmd
, result
);
372 if (result
) return (result
< 0) ? result
: count
;
378 /******************************************************************
379 * macho_accum_segs_range
381 * Callback for macho_enum_load_commands. Accumulates the address
382 * range covered by the segments of a Mach-O file. All commands
383 * are expected to be of LC_SEGMENT type.
385 static int macho_accum_segs_range(struct macho_file_map
* fmap
,
386 const struct load_command
* lc
, void* user
)
388 const struct segment_command
* sc
= (const struct segment_command
*)lc
;
389 unsigned tmp
, page_mask
= sysconf( _SC_PAGESIZE
) - 1;
391 TRACE("(%p/%d, %p, %p) before: 0x%08x - 0x%08x\n", fmap
, fmap
->fd
, lc
, user
,
392 (unsigned)fmap
->segs_start
, (unsigned)fmap
->segs_size
);
393 TRACE("Segment command vm: 0x%08x - 0x%08x\n", (unsigned)sc
->vmaddr
,
394 (unsigned)sc
->vmaddr
+ sc
->vmsize
);
396 if (!strncmp(sc
->segname
, "WINE_", 5))
398 TRACE("Ignoring special Wine segment %s\n", debugstr_an(sc
->segname
, sizeof(sc
->segname
)));
401 if (!strncmp(sc
->segname
, "__PAGEZERO", 10))
403 TRACE("Ignoring __PAGEZERO segment\n");
407 /* If this segment starts before previously-known earliest, record
409 if (sc
->vmaddr
< fmap
->segs_start
)
410 fmap
->segs_start
= sc
->vmaddr
;
412 /* If this segment extends beyond previously-known furthest, record
414 tmp
= (sc
->vmaddr
+ sc
->vmsize
+ page_mask
) & ~page_mask
;
415 if (fmap
->segs_size
< tmp
) fmap
->segs_size
= tmp
;
417 TRACE("after: 0x%08x - 0x%08x\n", (unsigned)fmap
->segs_start
, (unsigned)fmap
->segs_size
);
422 /******************************************************************
425 * Maps a Mach-O file into memory (and checks it's a real Mach-O file)
427 static BOOL
macho_map_file(const WCHAR
* filenameW
, struct macho_file_map
* fmap
)
429 struct fat_header fat_header
;
436 TRACE("(%s, %p)\n", debugstr_w(filenameW
), fmap
);
439 fmap
->load_commands
= MACHO_NO_MAP
;
440 RtlInitializeBitMap(&fmap
->sect_is_code
, fmap
->sect_is_code_buff
, MAX_SECT
+ 1);
442 len
= WideCharToMultiByte(CP_UNIXCP
, 0, filenameW
, -1, NULL
, 0, NULL
, NULL
);
443 if (!(filename
= HeapAlloc(GetProcessHeap(), 0, len
)))
445 WARN("failed to allocate filename buffer\n");
448 WideCharToMultiByte(CP_UNIXCP
, 0, filenameW
, -1, filename
, len
, NULL
, NULL
);
450 /* check that the file exists */
451 if (stat(filename
, &statbuf
) == -1 || S_ISDIR(statbuf
.st_mode
))
453 TRACE("stat() failed or %s is directory: %s\n", debugstr_a(filename
), strerror(errno
));
457 /* Now open the file, so that we can mmap() it. */
458 if ((fmap
->fd
= open(filename
, O_RDONLY
)) == -1)
460 TRACE("failed to open file %s: %d\n", debugstr_a(filename
), errno
);
464 if (read(fmap
->fd
, &fat_header
, sizeof(fat_header
)) != sizeof(fat_header
))
466 TRACE("failed to read fat header: %d\n", errno
);
469 TRACE("... got possible fat header\n");
471 /* Fat header is always in big-endian order. */
472 if (swap_ulong_be_to_host(fat_header
.magic
) == FAT_MAGIC
)
474 int narch
= swap_ulong_be_to_host(fat_header
.nfat_arch
);
475 for (i
= 0; i
< narch
; i
++)
477 struct fat_arch fat_arch
;
478 if (read(fmap
->fd
, &fat_arch
, sizeof(fat_arch
)) != sizeof(fat_arch
))
480 if (swap_ulong_be_to_host(fat_arch
.cputype
) == CPU_TYPE_X86
)
482 fmap
->arch_offset
= swap_ulong_be_to_host(fat_arch
.offset
);
483 fmap
->arch_size
= swap_ulong_be_to_host(fat_arch
.size
);
487 if (i
>= narch
) goto done
;
488 TRACE("... found x86 arch\n");
492 fmap
->arch_offset
= 0;
493 fmap
->arch_size
= statbuf
.st_size
;
494 TRACE("... not a fat header\n");
497 /* Individual architecture (standalone or within a fat file) is in its native byte order. */
498 lseek(fmap
->fd
, fmap
->arch_offset
, SEEK_SET
);
499 if (read(fmap
->fd
, &fmap
->mach_header
, sizeof(fmap
->mach_header
)) != sizeof(fmap
->mach_header
))
501 TRACE("... got possible Mach header\n");
502 /* and check for a Mach-O header */
503 if (fmap
->mach_header
.magic
!= MH_MAGIC
||
504 fmap
->mach_header
.cputype
!= CPU_TYPE_X86
) goto done
;
505 /* Make sure the file type is one of the ones we expect. */
506 switch (fmap
->mach_header
.filetype
)
516 TRACE("... verified Mach x86 header\n");
519 fmap
->segs_start
= ~0L;
521 if (macho_enum_load_commands(fmap
, LC_SEGMENT
, macho_accum_segs_range
, NULL
) < 0)
524 fmap
->segs_size
-= fmap
->segs_start
;
525 TRACE("segs_start: 0x%08x, segs_size: 0x%08x\n", (unsigned)fmap
->segs_start
,
526 (unsigned)fmap
->segs_size
);
531 macho_unmap_file(fmap
);
532 HeapFree(GetProcessHeap(), 0, filename
);
536 /******************************************************************
539 * Unmaps a Mach-O file from memory (previously mapped with macho_map_file)
541 static void macho_unmap_file(struct macho_file_map
* fmap
)
543 TRACE("(%p/%d)\n", fmap
, fmap
->fd
);
546 macho_unmap_load_commands(fmap
);
552 /******************************************************************
553 * macho_fill_sect_is_code
555 * Callback for macho_enum_load_commands. Determines which segments
556 * of a Mach-O file contain code. All commands are expected to be
557 * of LC_SEGMENT type.
559 static int macho_fill_sect_is_code(struct macho_file_map
* fmap
,
560 const struct load_command
* lc
, void* user
)
562 const struct segment_command
* sc
= (const struct segment_command
*)lc
;
563 const struct section
* sections
;
567 TRACE("(%p/%d, %p, %p/%d) scanning %u sections\n", fmap
, fmap
->fd
, lc
,
568 cursect
, *cursect
, sc
->nsects
);
570 sections
= (const struct section
*)(sc
+ 1);
571 for (i
= 0; i
< sc
->nsects
; i
++)
573 if (*cursect
> MAX_SECT
) return -1;
576 if (!(sections
[i
].flags
& SECTION_TYPE
) &&
577 (sections
[i
].flags
& (S_ATTR_PURE_INSTRUCTIONS
|S_ATTR_SOME_INSTRUCTIONS
)))
578 RtlSetBits(&fmap
->sect_is_code
, *cursect
, 1);
580 RtlClearBits(&fmap
->sect_is_code
, *cursect
, 1);
581 TRACE("Section %d (%d of this segment) is%s code\n", *cursect
, i
,
582 (RtlAreBitsSet(&fmap
->sect_is_code
, *cursect
, 1) ? "" : " not"));
588 /******************************************************************
591 * Checks if a section, identified by sectidx which is a 1-based
592 * index into the sections of all segments, in order of load
593 * commands, contains code.
595 static BOOL
macho_sect_is_code(struct macho_file_map
* fmap
, unsigned char sectidx
)
597 TRACE("(%p/%d, %u)\n", fmap
, fmap
->fd
, sectidx
);
599 if (!RtlAreBitsSet(&fmap
->sect_is_code
, 0, 1))
602 if (macho_enum_load_commands(fmap
, LC_SEGMENT
, macho_fill_sect_is_code
, &cursect
) < 0)
603 WARN("Couldn't load sect_is_code map\n");
604 RtlSetBits(&fmap
->sect_is_code
, 0, 1);
607 return RtlAreBitsSet(&fmap
->sect_is_code
, sectidx
, 1);
612 struct hash_table_elt ht_elt
;
613 struct symt_compiland
* compiland
;
615 unsigned char is_code
:1,
621 struct macho_debug_info
623 struct macho_file_map
* fmap
;
624 struct module
* module
;
626 struct hash_table ht_symtab
;
629 /******************************************************************
632 * Callback for stabs_parse. Collect symbol definitions.
634 static void macho_stabs_def_cb(struct module
* module
, unsigned long load_offset
,
635 const char* name
, unsigned long offset
,
636 BOOL is_public
, BOOL is_global
, unsigned char sectidx
,
637 struct symt_compiland
* compiland
, void* user
)
639 struct macho_debug_info
* mdi
= user
;
640 struct symtab_elt
* ste
;
642 TRACE("(%p, 0x%08lx, %s, 0x%08lx, %d, %d, %u, %p, %p/%p/%d)\n", module
, load_offset
,
643 debugstr_a(name
), offset
, is_public
, is_global
, sectidx
,
644 compiland
, mdi
, mdi
->fmap
, mdi
->fmap
->fd
);
646 /* Defer the creation of new non-debugging symbols until after we've
647 * finished parsing the stabs. */
648 ste
= pool_alloc(&mdi
->pool
, sizeof(*ste
));
649 ste
->ht_elt
.name
= pool_strdup(&mdi
->pool
, name
);
650 ste
->compiland
= compiland
;
651 ste
->addr
= load_offset
+ offset
;
652 ste
->is_code
= !!macho_sect_is_code(mdi
->fmap
, sectidx
);
653 ste
->is_public
= !!is_public
;
654 ste
->is_global
= !!is_global
;
656 hash_table_add(&mdi
->ht_symtab
, &ste
->ht_elt
);
659 /******************************************************************
662 * Callback for macho_enum_load_commands. Processes the LC_SYMTAB
663 * load commands from the Mach-O file.
665 static int macho_parse_symtab(struct macho_file_map
* fmap
,
666 const struct load_command
* lc
, void* user
)
668 const struct symtab_command
* sc
= (const struct symtab_command
*)lc
;
669 struct macho_debug_info
* mdi
= user
;
670 const struct nlist
* stab
;
674 TRACE("(%p/%d, %p, %p) %u syms at 0x%08x, strings 0x%08x - 0x%08x\n", fmap
, fmap
->fd
, lc
,
675 user
, sc
->nsyms
, sc
->symoff
, sc
->stroff
, sc
->stroff
+ sc
->strsize
);
677 if (!macho_map_ranges(fmap
, sc
->symoff
, sc
->nsyms
* sizeof(struct nlist
),
678 sc
->stroff
, sc
->strsize
, (const void**)&stab
, (const void**)&stabstr
))
681 if (!stabs_parse(mdi
->module
,
682 mdi
->module
->format_info
[DFI_MACHO
]->u
.macho_info
->load_addr
- fmap
->segs_start
,
683 stab
, sc
->nsyms
* sizeof(struct nlist
),
684 stabstr
, sc
->strsize
, macho_stabs_def_cb
, mdi
))
687 macho_unmap_ranges(fmap
, sc
->symoff
, sc
->nsyms
* sizeof(struct nlist
),
688 sc
->stroff
, sc
->strsize
, (const void**)&stab
, (const void**)&stabstr
);
693 /******************************************************************
696 * Integrate the non-debugging symbols we've gathered into the
697 * symbols that were generated during stabs parsing.
699 static void macho_finish_stabs(struct module
* module
, struct hash_table
* ht_symtab
)
701 struct hash_table_iter hti_ours
;
702 struct symtab_elt
* ste
;
703 BOOL adjusted
= FALSE
;
705 TRACE("(%p, %p)\n", module
, ht_symtab
);
707 /* For each of our non-debugging symbols, see if it can provide some
708 * missing details to one of the module's known symbols. */
709 hash_table_iter_init(ht_symtab
, &hti_ours
, NULL
);
710 while ((ste
= hash_table_iter_up(&hti_ours
)))
712 struct hash_table_iter hti_modules
;
715 struct symt_function
* func
;
716 struct symt_data
* data
;
718 hash_table_iter_init(&module
->ht_symbols
, &hti_modules
, ste
->ht_elt
.name
);
719 while ((ptr
= hash_table_iter_up(&hti_modules
)))
721 sym
= GET_ENTRY(ptr
, struct symt_ht
, hash_elt
);
723 if (strcmp(sym
->hash_elt
.name
, ste
->ht_elt
.name
))
726 switch (sym
->symt
.tag
)
729 func
= (struct symt_function
*)sym
;
730 if (func
->address
== module
->format_info
[DFI_MACHO
]->u
.macho_info
->load_addr
)
732 TRACE("Adjusting function %p/%s!%s from 0x%08lx to 0x%08lx\n", func
,
733 debugstr_w(module
->module
.ModuleName
), sym
->hash_elt
.name
,
734 func
->address
, ste
->addr
);
735 func
->address
= ste
->addr
;
738 if (func
->address
== ste
->addr
)
742 data
= (struct symt_data
*)sym
;
746 case DataIsFileStatic
:
747 if (data
->u
.var
.offset
== module
->format_info
[DFI_MACHO
]->u
.macho_info
->load_addr
)
749 TRACE("Adjusting data symbol %p/%s!%s from 0x%08lx to 0x%08lx\n",
750 data
, debugstr_w(module
->module
.ModuleName
), sym
->hash_elt
.name
,
751 data
->u
.var
.offset
, ste
->addr
);
752 data
->u
.var
.offset
= ste
->addr
;
755 if (data
->u
.var
.offset
== ste
->addr
)
757 enum DataKind new_kind
;
759 new_kind
= ste
->is_global
? DataIsGlobal
: DataIsFileStatic
;
760 if (data
->kind
!= new_kind
)
762 WARN("Changing kind for %p/%s!%s from %d to %d\n", sym
,
763 debugstr_w(module
->module
.ModuleName
), sym
->hash_elt
.name
,
764 (int)data
->kind
, (int)new_kind
);
765 data
->kind
= new_kind
;
775 TRACE("Ignoring tag %u\n", sym
->symt
.tag
);
783 /* since we may have changed some addresses, mark the module to be resorted */
784 module
->sortlist_valid
= FALSE
;
787 /* Mark any of our non-debugging symbols which fall on an already-used
788 * address as "used". This allows us to skip them in the next loop,
789 * below. We do this in separate loops because symt_new_* marks the
790 * list as needing sorting and symt_find_nearest sorts if needed,
791 * causing thrashing. */
792 if (!(dbghelp_options
& SYMOPT_PUBLICS_ONLY
))
794 hash_table_iter_init(ht_symtab
, &hti_ours
, NULL
);
795 while ((ste
= hash_table_iter_up(&hti_ours
)))
800 if (ste
->used
) continue;
802 sym
= symt_find_nearest(module
, ste
->addr
);
804 symt_get_address(&sym
->symt
, &addr
);
805 if (sym
&& ste
->addr
== addr
)
812 /* If neither symbol has a correct size (ours never does), we
813 * consider them both to be markers. No warning is needed in
815 * Also, we check that we don't have two symbols, one local, the other
816 * global, which is legal.
818 symt_get_info(module
, &sym
->symt
, TI_GET_LENGTH
, &size
);
819 symt_get_info(module
, &sym
->symt
, TI_GET_DATAKIND
, &kind
);
820 if (size
&& kind
== (ste
->is_global
? DataIsGlobal
: DataIsFileStatic
))
821 FIXME("Duplicate in %s: %s<%08lx> %s<%s-%s>\n",
822 debugstr_w(module
->module
.ModuleName
),
823 ste
->ht_elt
.name
, ste
->addr
,
825 wine_dbgstr_longlong(addr
), wine_dbgstr_longlong(size
));
830 /* For any of our remaining non-debugging symbols which have no match
831 * among the module's known symbols, add them as new symbols. */
832 hash_table_iter_init(ht_symtab
, &hti_ours
, NULL
);
833 while ((ste
= hash_table_iter_up(&hti_ours
)))
835 if (!(dbghelp_options
& SYMOPT_PUBLICS_ONLY
) && !ste
->used
)
839 symt_new_function(module
, ste
->compiland
, ste
->ht_elt
.name
,
846 loc
.kind
= loc_absolute
;
848 loc
.offset
= ste
->addr
;
849 symt_new_global_variable(module
, ste
->compiland
, ste
->ht_elt
.name
,
850 !ste
->is_global
, loc
, 0, NULL
);
856 if (ste
->is_public
&& !(dbghelp_options
& SYMOPT_NO_PUBLICS
))
858 symt_new_public(module
, ste
->compiland
, ste
->ht_elt
.name
, ste
->addr
, 0);
863 /******************************************************************
864 * macho_load_debug_info_from_map
866 * Loads the symbolic information from a Mach-O module.
868 * FALSE if the file doesn't contain symbolic info (or this info
869 * cannot be read or parsed)
872 static BOOL
macho_load_debug_info_from_map(struct module
* module
,
873 struct macho_file_map
* fmap
)
876 struct macho_debug_info mdi
;
879 TRACE("(%p, %p/%d)\n", module
, fmap
, fmap
->fd
);
881 module
->module
.SymType
= SymExport
;
885 pool_init(&mdi
.pool
, 65536);
886 hash_table_init(&mdi
.pool
, &mdi
.ht_symtab
, 256);
887 result
= macho_enum_load_commands(fmap
, LC_SYMTAB
, macho_parse_symtab
, &mdi
);
891 WARN("Couldn't correctly read stabs\n");
893 macho_finish_stabs(module
, &mdi
.ht_symtab
);
895 pool_destroy(&mdi
.pool
);
899 /******************************************************************
900 * macho_load_debug_info
902 * Loads Mach-O debugging information from the module image file.
904 BOOL
macho_load_debug_info(struct module
* module
, struct macho_file_map
* fmap
)
907 struct macho_file_map my_fmap
;
909 TRACE("(%p, %p/%d)\n", module
, fmap
, fmap
? fmap
->fd
: -1);
911 if (module
->type
!= DMT_MACHO
|| !module
->format_info
[DFI_MACHO
]->u
.macho_info
)
913 ERR("Bad Mach-O module '%s'\n", debugstr_w(module
->module
.LoadedImageName
));
920 ret
= macho_map_file(module
->module
.LoadedImageName
, fmap
);
923 ret
= macho_load_debug_info_from_map(module
, fmap
);
925 if (fmap
== &my_fmap
) macho_unmap_file(fmap
);
929 /******************************************************************
930 * macho_fetch_file_info
932 * Gathers some more information for a Mach-O module from a given file
934 BOOL
macho_fetch_file_info(const WCHAR
* name
, DWORD_PTR
* base
,
935 DWORD
* size
, DWORD
* checksum
)
937 struct macho_file_map fmap
;
939 TRACE("(%s, %p, %p, %p)\n", debugstr_w(name
), base
, size
, checksum
);
941 if (!macho_map_file(name
, &fmap
)) return FALSE
;
942 if (base
) *base
= fmap
.segs_start
;
943 *size
= fmap
.segs_size
;
944 *checksum
= calc_crc32(fmap
.fd
);
945 macho_unmap_file(&fmap
);
949 /******************************************************************
952 * Loads the information for Mach-O module stored in 'filename'.
953 * The module has been loaded at 'load_addr' address.
955 * FALSE if the file cannot be found/opened or if the file doesn't
956 * contain symbolic info (or this info cannot be read or parsed)
959 static BOOL
macho_load_file(struct process
* pcs
, const WCHAR
* filename
,
960 unsigned long load_addr
, struct macho_info
* macho_info
)
963 struct macho_file_map fmap
;
965 TRACE("(%p/%p, %s, 0x%08lx, %p/0x%08x)\n", pcs
, pcs
->handle
, debugstr_w(filename
),
966 load_addr
, macho_info
, macho_info
->flags
);
968 if (!macho_map_file(filename
, &fmap
)) return FALSE
;
970 /* Find the dynamic loader's table of images loaded into the process.
972 if (macho_info
->flags
& MACHO_INFO_DEBUG_HEADER
)
974 PROCESS_BASIC_INFORMATION pbi
;
979 /* Get address of PEB */
980 status
= NtQueryInformationProcess(pcs
->handle
, ProcessBasicInformation
,
981 &pbi
, sizeof(pbi
), NULL
);
982 if (status
== STATUS_SUCCESS
)
984 ULONG_PTR dyld_image_info
;
986 /* Read dyld image info address from PEB */
987 if (ReadProcessMemory(pcs
->handle
, &pbi
.PebBaseAddress
->Reserved
[0],
988 &dyld_image_info
, sizeof(dyld_image_info
), NULL
))
990 TRACE("got dyld_image_info 0x%08x from PEB %p MacDyldImageInfo %p\n",
991 dyld_image_info
, pbi
.PebBaseAddress
, &pbi
.PebBaseAddress
->Reserved
);
992 macho_info
->dbg_hdr_addr
= dyld_image_info
;
997 #ifndef __LP64__ /* No reading the symtab with nlist(3) in LP64 */
1000 static void* dyld_all_image_infos_addr
;
1002 /* Our next best guess is that dyld was loaded at its base address
1003 and we can find the dyld image infos address by looking up its symbol. */
1004 if (!dyld_all_image_infos_addr
)
1007 memset(nl
, 0, sizeof(nl
));
1008 nl
[0].n_un
.n_name
= (char*)"_dyld_all_image_infos";
1009 if (!nlist("/usr/lib/dyld", nl
))
1010 dyld_all_image_infos_addr
= (void*)nl
[0].n_value
;
1013 if (dyld_all_image_infos_addr
)
1015 TRACE("got dyld_image_info %p from /usr/lib/dyld symbol table\n",
1016 dyld_all_image_infos_addr
);
1017 macho_info
->dbg_hdr_addr
= (unsigned long)dyld_all_image_infos_addr
;
1024 if (macho_info
->flags
& MACHO_INFO_MODULE
)
1026 struct macho_module_info
*macho_module_info
;
1027 struct module_format
* modfmt
=
1028 HeapAlloc(GetProcessHeap(), 0, sizeof(struct module_format
) + sizeof(struct macho_module_info
));
1029 if (!modfmt
) goto leave
;
1031 load_addr
= fmap
.segs_start
;
1032 macho_info
->module
= module_new(pcs
, filename
, DMT_MACHO
, FALSE
, load_addr
,
1033 fmap
.segs_size
, 0, calc_crc32(fmap
.fd
));
1034 if (!macho_info
->module
)
1036 HeapFree(GetProcessHeap(), 0, modfmt
);
1039 macho_module_info
= (void*)(modfmt
+ 1);
1040 macho_info
->module
->format_info
[DFI_MACHO
] = modfmt
;
1042 modfmt
->module
= macho_info
->module
;
1043 modfmt
->remove
= NULL
;
1044 modfmt
->loc_compute
= NULL
;
1045 modfmt
->u
.macho_info
= macho_module_info
;
1047 macho_module_info
->load_addr
= load_addr
;
1049 if (dbghelp_options
& SYMOPT_DEFERRED_LOADS
)
1050 macho_info
->module
->module
.SymType
= SymDeferred
;
1051 else if (!macho_load_debug_info(macho_info
->module
, &fmap
))
1054 macho_info
->module
->format_info
[DFI_MACHO
]->u
.macho_info
->in_use
= 1;
1055 macho_info
->module
->format_info
[DFI_MACHO
]->u
.macho_info
->is_loader
= 0;
1056 TRACE("module = %p\n", macho_info
->module
);
1059 if (macho_info
->flags
& MACHO_INFO_NAME
)
1062 ptr
= HeapAlloc(GetProcessHeap(), 0, (lstrlenW(filename
) + 1) * sizeof(WCHAR
));
1065 strcpyW(ptr
, filename
);
1066 macho_info
->module_name
= ptr
;
1069 TRACE("module_name = %p %s\n", macho_info
->module_name
, debugstr_w(macho_info
->module_name
));
1072 macho_unmap_file(&fmap
);
1074 TRACE(" => %d\n", ret
);
1078 /******************************************************************
1079 * macho_load_file_from_path
1080 * Tries to load a Mach-O file from a set of paths (separated by ':')
1082 static BOOL
macho_load_file_from_path(struct process
* pcs
,
1083 const WCHAR
* filename
,
1084 unsigned long load_addr
,
1086 struct macho_info
* macho_info
)
1090 WCHAR
* pathW
= NULL
;
1093 TRACE("(%p/%p, %s, 0x%08lx, %s, %p)\n", pcs
, pcs
->handle
, debugstr_w(filename
), load_addr
,
1094 debugstr_a(path
), macho_info
);
1096 if (!path
) return FALSE
;
1098 len
= MultiByteToWideChar(CP_UNIXCP
, 0, path
, -1, NULL
, 0);
1099 pathW
= HeapAlloc(GetProcessHeap(), 0, len
* sizeof(WCHAR
));
1100 if (!pathW
) return FALSE
;
1101 MultiByteToWideChar(CP_UNIXCP
, 0, path
, -1, pathW
, len
);
1103 for (s
= pathW
; s
&& *s
; s
= (t
) ? (t
+1) : NULL
)
1105 t
= strchrW(s
, ':');
1107 fn
= HeapAlloc(GetProcessHeap(), 0, (lstrlenW(filename
) + 1 + lstrlenW(s
) + 1) * sizeof(WCHAR
));
1110 strcatW(fn
, S_SlashW
);
1111 strcatW(fn
, filename
);
1112 ret
= macho_load_file(pcs
, fn
, load_addr
, macho_info
);
1113 HeapFree(GetProcessHeap(), 0, fn
);
1115 s
= (t
) ? (t
+1) : NULL
;
1118 TRACE(" => %d\n", ret
);
1119 HeapFree(GetProcessHeap(), 0, pathW
);
1123 /******************************************************************
1124 * macho_load_file_from_dll_path
1126 * Tries to load a Mach-O file from the dll path
1128 static BOOL
macho_load_file_from_dll_path(struct process
* pcs
,
1129 const WCHAR
* filename
,
1130 unsigned long load_addr
,
1131 struct macho_info
* macho_info
)
1134 unsigned int index
= 0;
1137 TRACE("(%p/%p, %s, 0x%08lx, %p)\n", pcs
, pcs
->handle
, debugstr_w(filename
), load_addr
,
1140 while (!ret
&& (path
= wine_dll_enum_load_path( index
++ )))
1145 len
= MultiByteToWideChar(CP_UNIXCP
, 0, path
, -1, NULL
, 0);
1147 name
= HeapAlloc( GetProcessHeap(), 0,
1148 (len
+ lstrlenW(filename
) + 2) * sizeof(WCHAR
) );
1151 MultiByteToWideChar(CP_UNIXCP
, 0, path
, -1, name
, len
);
1152 strcatW( name
, S_SlashW
);
1153 strcatW( name
, filename
);
1154 ret
= macho_load_file(pcs
, name
, load_addr
, macho_info
);
1155 HeapFree( GetProcessHeap(), 0, name
);
1157 TRACE(" => %d\n", ret
);
1161 /******************************************************************
1162 * macho_search_and_load_file
1164 * Lookup a file in standard Mach-O locations, and if found, load it
1166 static BOOL
macho_search_and_load_file(struct process
* pcs
, const WCHAR
* filename
,
1167 unsigned long load_addr
,
1168 struct macho_info
* macho_info
)
1171 struct module
* module
;
1172 static const WCHAR S_libstdcPPW
[] = {'l','i','b','s','t','d','c','+','+','\0'};
1175 TRACE("(%p/%p, %s, 0x%08lx, %p)\n", pcs
, pcs
->handle
, debugstr_w(filename
), load_addr
,
1178 if (filename
== NULL
|| *filename
== '\0') return FALSE
;
1179 if ((module
= module_is_already_loaded(pcs
, filename
)))
1181 macho_info
->module
= module
;
1182 module
->format_info
[DFI_MACHO
]->u
.macho_info
->in_use
= 1;
1183 return module
->module
.SymType
;
1186 if (strstrW(filename
, S_libstdcPPW
)) return FALSE
; /* We know we can't do it */
1188 /* If has no directories, try LD_LIBRARY_PATH first. */
1189 if (!strchrW(filename
, '/'))
1191 ret
= macho_load_file_from_path(pcs
, filename
, load_addr
,
1192 getenv("PATH"), macho_info
);
1194 /* Try DYLD_LIBRARY_PATH, with just the filename (no directories). */
1197 if ((p
= strrchrW(filename
, '/'))) p
++;
1199 ret
= macho_load_file_from_path(pcs
, p
, load_addr
,
1200 getenv("DYLD_LIBRARY_PATH"), macho_info
);
1202 /* Try the path as given. */
1204 ret
= macho_load_file(pcs
, filename
, load_addr
, macho_info
);
1205 /* Try DYLD_FALLBACK_LIBRARY_PATH, with just the filename (no directories). */
1208 ret
= macho_load_file_from_path(pcs
, p
, load_addr
,
1209 getenv("DYLD_FALLBACK_LIBRARY_PATH"), macho_info
);
1211 if (!ret
&& !strchrW(filename
, '/'))
1212 ret
= macho_load_file_from_dll_path(pcs
, filename
, load_addr
, macho_info
);
1217 /******************************************************************
1218 * macho_enum_modules_internal
1220 * Enumerate Mach-O modules from a running process
1222 static BOOL
macho_enum_modules_internal(const struct process
* pcs
,
1223 const WCHAR
* main_name
,
1224 enum_modules_cb cb
, void* user
)
1226 struct dyld_all_image_infos image_infos
;
1227 struct dyld_image_info
* info_array
= NULL
;
1231 WCHAR bufstrW
[MAX_PATH
];
1234 TRACE("(%p/%p, %s, %p, %p)\n", pcs
, pcs
->handle
, debugstr_w(main_name
), cb
,
1237 if (!pcs
->dbg_hdr_addr
||
1238 !ReadProcessMemory(pcs
->handle
, (void*)pcs
->dbg_hdr_addr
,
1239 &image_infos
, sizeof(image_infos
), NULL
) ||
1240 !image_infos
.infoArray
)
1242 TRACE("Process has %u image infos at %p\n", image_infos
.infoArrayCount
, image_infos
.infoArray
);
1244 len
= image_infos
.infoArrayCount
* sizeof(info_array
[0]);
1245 info_array
= HeapAlloc(GetProcessHeap(), 0, len
);
1247 !ReadProcessMemory(pcs
->handle
, image_infos
.infoArray
,
1248 info_array
, len
, NULL
))
1250 TRACE("... read image infos\n");
1252 for (i
= 0; i
< image_infos
.infoArrayCount
; i
++)
1254 if (info_array
[i
].imageFilePath
!= NULL
&&
1255 ReadProcessMemory(pcs
->handle
, info_array
[i
].imageFilePath
, bufstr
, sizeof(bufstr
), NULL
))
1257 bufstr
[sizeof(bufstr
) - 1] = '\0';
1258 TRACE("[%d] image file %s\n", i
, debugstr_a(bufstr
));
1259 MultiByteToWideChar(CP_UNIXCP
, 0, bufstr
, -1, bufstrW
, sizeof(bufstrW
) / sizeof(WCHAR
));
1260 if (main_name
&& !bufstrW
[0]) strcpyW(bufstrW
, main_name
);
1261 if (!cb(bufstrW
, (unsigned long)info_array
[i
].imageLoadAddress
, user
)) break;
1267 HeapFree(GetProcessHeap(), 0, info_array
);
1273 struct process
* pcs
;
1274 struct macho_info macho_info
;
1277 static BOOL
macho_enum_sync_cb(const WCHAR
* name
, unsigned long addr
, void* user
)
1279 struct macho_sync
* ms
= user
;
1281 TRACE("(%s, 0x%08lx, %p)\n", debugstr_w(name
), addr
, user
);
1282 macho_search_and_load_file(ms
->pcs
, name
, addr
, &ms
->macho_info
);
1286 /******************************************************************
1287 * macho_synchronize_module_list
1289 * Rescans the debuggee's modules list and synchronizes it with
1290 * the one from 'pcs', ie:
1291 * - if a module is in debuggee and not in pcs, it's loaded into pcs
1292 * - if a module is in pcs and not in debuggee, it's unloaded from pcs
1294 BOOL
macho_synchronize_module_list(struct process
* pcs
)
1296 struct module
* module
;
1297 struct macho_sync ms
;
1299 TRACE("(%p/%p)\n", pcs
, pcs
->handle
);
1301 for (module
= pcs
->lmodules
; module
; module
= module
->next
)
1303 if (module
->type
== DMT_MACHO
&& !module
->is_virtual
)
1304 module
->format_info
[DFI_MACHO
]->u
.macho_info
->in_use
= 0;
1308 ms
.macho_info
.flags
= MACHO_INFO_MODULE
;
1309 if (!macho_enum_modules_internal(pcs
, NULL
, macho_enum_sync_cb
, &ms
))
1312 module
= pcs
->lmodules
;
1315 if (module
->type
== DMT_MACHO
&& !module
->is_virtual
&&
1316 !module
->format_info
[DFI_MACHO
]->u
.macho_info
->in_use
&&
1317 !module
->format_info
[DFI_MACHO
]->u
.macho_info
->is_loader
)
1319 module_remove(pcs
, module
);
1320 /* restart all over */
1321 module
= pcs
->lmodules
;
1323 else module
= module
->next
;
1328 /******************************************************************
1329 * macho_search_loader
1331 * Lookup in a running Mach-O process the loader, and sets its Mach-O link
1332 * address (for accessing the list of loaded images) in pcs.
1333 * If flags is MACHO_INFO_MODULE, the module for the loader is also
1334 * added as a module into pcs.
1336 static BOOL
macho_search_loader(struct process
* pcs
, struct macho_info
* macho_info
)
1338 return macho_search_and_load_file(pcs
, get_wine_loader_name(), 0, macho_info
);
1341 /******************************************************************
1342 * macho_read_wine_loader_dbg_info
1344 * Try to find a decent wine executable which could have loaded the debuggee
1346 BOOL
macho_read_wine_loader_dbg_info(struct process
* pcs
)
1348 struct macho_info macho_info
;
1350 TRACE("(%p/%p)\n", pcs
, pcs
->handle
);
1351 macho_info
.flags
= MACHO_INFO_DEBUG_HEADER
| MACHO_INFO_MODULE
;
1352 if (!macho_search_loader(pcs
, &macho_info
)) return FALSE
;
1353 macho_info
.module
->format_info
[DFI_MACHO
]->u
.macho_info
->is_loader
= 1;
1354 module_set_module(macho_info
.module
, S_WineLoaderW
);
1355 return (pcs
->dbg_hdr_addr
= macho_info
.dbg_hdr_addr
) != 0;
1358 /******************************************************************
1359 * macho_enum_modules
1361 * Enumerates the Mach-O loaded modules from a running target (hProc)
1362 * This function doesn't require that someone has called SymInitialize
1363 * on this very process.
1365 BOOL
macho_enum_modules(HANDLE hProc
, enum_modules_cb cb
, void* user
)
1368 struct macho_info macho_info
;
1371 TRACE("(%p, %p, %p)\n", hProc
, cb
, user
);
1372 memset(&pcs
, 0, sizeof(pcs
));
1374 macho_info
.flags
= MACHO_INFO_DEBUG_HEADER
| MACHO_INFO_NAME
;
1375 if (!macho_search_loader(&pcs
, &macho_info
)) return FALSE
;
1376 pcs
.dbg_hdr_addr
= macho_info
.dbg_hdr_addr
;
1377 ret
= macho_enum_modules_internal(&pcs
, macho_info
.module_name
, cb
, user
);
1378 HeapFree(GetProcessHeap(), 0, (char*)macho_info
.module_name
);
1384 struct process
* pcs
;
1385 struct macho_info macho_info
;
1390 /******************************************************************
1393 * Callback for macho_load_module, used to walk the list of loaded
1396 static BOOL
macho_load_cb(const WCHAR
* name
, unsigned long addr
, void* user
)
1398 struct macho_load
* ml
= user
;
1401 TRACE("(%s, 0x%08lx, %p)\n", debugstr_w(name
), addr
, user
);
1403 /* memcmp is needed for matches when bufstr contains also version information
1404 * ml->name: libc.so, name: libc.so.6.0
1406 p
= strrchrW(name
, '/');
1408 if (!memcmp(p
, ml
->name
, lstrlenW(ml
->name
) * sizeof(WCHAR
)))
1410 ml
->ret
= macho_search_and_load_file(ml
->pcs
, name
, addr
, &ml
->macho_info
);
1416 /******************************************************************
1419 * Loads a Mach-O module and stores it in process' module list.
1420 * Also, find module real name and load address from
1421 * the real loaded modules list in pcs address space.
1423 struct module
* macho_load_module(struct process
* pcs
, const WCHAR
* name
, unsigned long addr
)
1425 struct macho_load ml
;
1427 TRACE("(%p/%p, %s, 0x%08lx)\n", pcs
, pcs
->handle
, debugstr_w(name
), addr
);
1429 ml
.macho_info
.flags
= MACHO_INFO_MODULE
;
1432 if (pcs
->dbg_hdr_addr
) /* we're debugging a live target */
1435 /* do only the lookup from the filename, not the path (as we lookup module
1436 * name in the process' loaded module list)
1438 ml
.name
= strrchrW(name
, '/');
1439 if (!ml
.name
++) ml
.name
= name
;
1442 if (!macho_enum_modules_internal(pcs
, NULL
, macho_load_cb
, &ml
))
1448 ml
.ret
= macho_search_and_load_file(pcs
, ml
.name
, addr
, &ml
.macho_info
);
1450 if (!ml
.ret
) return NULL
;
1451 assert(ml
.macho_info
.module
);
1452 return ml
.macho_info
.module
;
1455 #else /* HAVE_MACH_O_LOADER_H */
1457 BOOL
macho_synchronize_module_list(struct process
* pcs
)
1462 BOOL
macho_fetch_file_info(const WCHAR
* name
, DWORD_PTR
* base
,
1463 DWORD
* size
, DWORD
* checksum
)
1468 BOOL
macho_read_wine_loader_dbg_info(struct process
* pcs
)
1473 BOOL
macho_enum_modules(HANDLE hProc
, enum_modules_cb cb
, void* user
)
1478 struct module
* macho_load_module(struct process
* pcs
, const WCHAR
* name
, unsigned long addr
)
1483 BOOL
macho_load_debug_info(struct module
* module
, struct macho_file_map
* fmap
)
1487 #endif /* HAVE_MACH_O_LOADER_H */