jscript: Add the Romanian translation.
[wine/testsucceed.git] / dlls / dbghelp / macho_module.c
blob44fa4f2da1c9822b552973b4527a5b50e678f98d
1 /*
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
24 #include "config.h"
25 #include "wine/port.h"
27 #include "dbghelp_private.h"
29 #ifdef __MACH__
31 #include <assert.h>
32 #include <stdarg.h>
33 #ifdef HAVE_SYS_STAT_H
34 # include <sys/stat.h>
35 #endif
36 #ifdef HAVE_SYS_MMAN_H
37 # include <sys/mman.h>
38 #endif
40 #include <mach-o/fat.h>
41 #include <mach-o/loader.h>
42 #include <mach-o/nlist.h>
44 #ifdef HAVE_MACH_O_DYLD_IMAGES_H
45 #include <mach-o/dyld_images.h>
46 #else
47 struct dyld_image_info {
48 const struct mach_header *imageLoadAddress;
49 const char *imageFilePath;
50 uintptr_t imageFileModDate;
53 struct dyld_all_image_infos {
54 uint32_t version;
55 uint32_t infoArrayCount;
56 const struct dyld_image_info *infoArray;
57 void* notification;
58 int processDetachedFromSharedRegion;
60 #endif
62 #include "winternl.h"
63 #include "wine/library.h"
64 #include "wine/debug.h"
66 #ifdef WORDS_BIGENDIAN
67 #define swap_ulong_be_to_host(n) (n)
68 #else
69 #define swap_ulong_be_to_host(n) (RtlUlongByteSwap(n))
70 #endif
72 WINE_DEFAULT_DEBUG_CHANNEL(dbghelp_macho);
75 struct macho_module_info
77 unsigned long load_addr;
78 unsigned short in_use : 1,
79 is_loader : 1;
82 #define MACHO_INFO_DEBUG_HEADER 0x0001
83 #define MACHO_INFO_MODULE 0x0002
84 #define MACHO_INFO_NAME 0x0004
86 struct macho_info
88 unsigned flags; /* IN one (or several) of the MACHO_INFO constants */
89 unsigned long dbg_hdr_addr; /* OUT address of debug header (if MACHO_INFO_DEBUG_HEADER is set) */
90 struct module* module; /* OUT loaded module (if MACHO_INFO_MODULE is set) */
91 const WCHAR* module_name; /* OUT found module name (if MACHO_INFO_NAME is set) */
94 /* structure holding information while handling a Mach-O image */
95 #define BITS_PER_ULONG (sizeof(ULONG) * 8)
96 #define ULONGS_FOR_BITS(nbits) (((nbits) + BITS_PER_ULONG - 1) / BITS_PER_ULONG)
97 struct macho_file_map
99 /* A copy of the Mach-O header for an individual architecture. */
100 struct mach_header mach_header;
102 /* The mapped load commands. */
103 const struct load_command* load_commands;
105 /* The portion of the file which is this architecture. mach_header was
106 * read from arch_offset. */
107 unsigned arch_offset;
108 unsigned arch_size;
110 /* The range of address space covered by all segments. */
111 size_t segs_start;
112 size_t segs_size;
114 /* Map of which sections contain code. Sections are accessed using 1-based
115 * index. Bit 0 of this bitset indicates if the bitset has been initialized. */
116 RTL_BITMAP sect_is_code;
117 ULONG sect_is_code_buff[ULONGS_FOR_BITS(MAX_SECT + 1)];
119 /* The file. */
120 int fd;
123 static void macho_unmap_file(struct macho_file_map* fmap);
125 /******************************************************************
126 * macho_calc_range
128 * For a range (offset & length) of a single architecture within
129 * a Mach-O file, calculate the page-aligned range of the whole file
130 * that encompasses it. For a fat binary, the architecture will
131 * itself be offset within the file, so take that into account.
133 static void macho_calc_range(const struct macho_file_map* fmap, unsigned offset,
134 unsigned len, unsigned* out_aligned_offset,
135 unsigned* out_aligned_end, unsigned* out_aligned_len,
136 unsigned* out_misalign)
138 unsigned pagemask = getpagesize() - 1;
139 unsigned file_offset, misalign;
141 file_offset = fmap->arch_offset + offset;
142 misalign = file_offset & pagemask;
143 *out_aligned_offset = file_offset - misalign;
144 *out_aligned_end = (file_offset + len + pagemask) & ~pagemask;
145 if (out_aligned_len)
146 *out_aligned_len = *out_aligned_end - *out_aligned_offset;
147 if (out_misalign)
148 *out_misalign = misalign;
151 /******************************************************************
152 * macho_map_range
154 * Maps a range (offset, length in bytes) from a Mach-O file into memory
156 static const char* macho_map_range(const struct macho_file_map* fmap, unsigned offset, unsigned len)
158 unsigned misalign, aligned_offset, aligned_map_end, map_size;
159 const void* aligned_ptr;
161 TRACE("(%p/%d, 0x%08x, 0x%08x)\n", fmap, fmap->fd, offset, len);
163 macho_calc_range(fmap, offset, len, &aligned_offset, &aligned_map_end,
164 &map_size, &misalign);
166 aligned_ptr = mmap(NULL, map_size, PROT_READ, MAP_PRIVATE, fmap->fd, aligned_offset);
168 TRACE("Mapped (0x%08x - 0x%08x) to %p\n", aligned_offset, aligned_map_end, aligned_ptr);
170 if (aligned_ptr == MAP_FAILED) return MACHO_NO_MAP;
171 return (const char*)aligned_ptr + misalign;
174 /******************************************************************
175 * macho_unmap_range
177 * Unmaps a range (offset, length in bytes) of a Mach-O file from memory
179 static void macho_unmap_range(const void** mapped, const struct macho_file_map* fmap,
180 unsigned offset, unsigned len)
182 TRACE("(%p, %p/%d, 0x%08x, 0x%08x)\n", mapped, fmap, fmap->fd, offset, len);
184 if (mapped && *mapped != MACHO_NO_MAP)
186 unsigned misalign, aligned_offset, aligned_map_end, map_size;
187 void* aligned_ptr;
189 macho_calc_range(fmap, offset, len, &aligned_offset, &aligned_map_end,
190 &map_size, &misalign);
192 aligned_ptr = (char*)*mapped - misalign;
193 if (munmap(aligned_ptr, map_size) < 0)
194 WARN("Couldn't unmap the range\n");
195 TRACE("Unmapped (0x%08x - 0x%08x) from %p - %p\n", aligned_offset, aligned_map_end, aligned_ptr, (char*)aligned_ptr + map_size);
196 *mapped = MACHO_NO_MAP;
200 /******************************************************************
201 * macho_map_ranges
203 * Maps two ranges (offset, length in bytes) from a Mach-O file
204 * into memory. If the two ranges overlap, use one mmap so that
205 * the munmap doesn't fragment the mapping.
207 static BOOL macho_map_ranges(const struct macho_file_map* fmap,
208 unsigned offset1, unsigned len1,
209 unsigned offset2, unsigned len2,
210 const void** mapped1, const void** mapped2)
212 unsigned aligned_offset1, aligned_map_end1;
213 unsigned aligned_offset2, aligned_map_end2;
215 TRACE("(%p/%d, 0x%08x, 0x%08x, 0x%08x, 0x%08x, %p, %p)\n", fmap, fmap->fd,
216 offset1, len1, offset2, len2, mapped1, mapped2);
218 macho_calc_range(fmap, offset1, len1, &aligned_offset1, &aligned_map_end1, NULL, NULL);
219 macho_calc_range(fmap, offset2, len2, &aligned_offset2, &aligned_map_end2, NULL, NULL);
221 if (aligned_map_end1 < aligned_offset2 || aligned_map_end2 < aligned_offset1)
223 *mapped1 = macho_map_range(fmap, offset1, len1);
224 if (*mapped1 != MACHO_NO_MAP)
226 *mapped2 = macho_map_range(fmap, offset2, len2);
227 if (*mapped2 == MACHO_NO_MAP)
228 macho_unmap_range(mapped1, fmap, offset1, len1);
231 else
233 if (offset1 < offset2)
235 *mapped1 = macho_map_range(fmap, offset1, offset2 + len2 - offset1);
236 if (*mapped1 != MACHO_NO_MAP)
237 *mapped2 = (const char*)*mapped1 + offset2 - offset1;
239 else
241 *mapped2 = macho_map_range(fmap, offset2, offset1 + len1 - offset2);
242 if (*mapped2 != MACHO_NO_MAP)
243 *mapped1 = (const char*)*mapped2 + offset1 - offset2;
247 TRACE(" => %p, %p\n", *mapped1, *mapped2);
249 return (*mapped1 != MACHO_NO_MAP) && (*mapped2 != MACHO_NO_MAP);
252 /******************************************************************
253 * macho_unmap_ranges
255 * Unmaps two ranges (offset, length in bytes) of a Mach-O file
256 * from memory. Use for ranges which were mapped by
257 * macho_map_ranges.
259 static void macho_unmap_ranges(const struct macho_file_map* fmap,
260 unsigned offset1, unsigned len1,
261 unsigned offset2, unsigned len2,
262 const void** mapped1, const void** mapped2)
264 unsigned aligned_offset1, aligned_map_end1;
265 unsigned aligned_offset2, aligned_map_end2;
267 TRACE("(%p/%d, 0x%08x, 0x%08x, 0x%08x, 0x%08x, %p/%p, %p/%p)\n", fmap, fmap->fd,
268 offset1, len1, offset2, len2, mapped1, *mapped1, mapped2, *mapped2);
270 macho_calc_range(fmap, offset1, len1, &aligned_offset1, &aligned_map_end1, NULL, NULL);
271 macho_calc_range(fmap, offset2, len2, &aligned_offset2, &aligned_map_end2, NULL, NULL);
273 if (aligned_map_end1 < aligned_offset2 || aligned_map_end2 < aligned_offset1)
275 macho_unmap_range(mapped1, fmap, offset1, len1);
276 macho_unmap_range(mapped2, fmap, offset2, len2);
278 else
280 if (offset1 < offset2)
282 macho_unmap_range(mapped1, fmap, offset1, offset2 + len2 - offset1);
283 *mapped2 = MACHO_NO_MAP;
285 else
287 macho_unmap_range(mapped2, fmap, offset2, offset1 + len1 - offset2);
288 *mapped1 = MACHO_NO_MAP;
293 /******************************************************************
294 * macho_map_load_commands
296 * Maps the load commands from a Mach-O file into memory
298 static const struct load_command* macho_map_load_commands(struct macho_file_map* fmap)
300 if (fmap->load_commands == MACHO_NO_MAP)
302 fmap->load_commands = (const struct load_command*) macho_map_range(
303 fmap, sizeof(fmap->mach_header), fmap->mach_header.sizeofcmds);
304 TRACE("Mapped load commands: %p\n", fmap->load_commands);
307 return fmap->load_commands;
310 /******************************************************************
311 * macho_unmap_load_commands
313 * Unmaps the load commands of a Mach-O file from memory
315 static void macho_unmap_load_commands(struct macho_file_map* fmap)
317 if (fmap->load_commands != MACHO_NO_MAP)
319 TRACE("Unmapping load commands: %p\n", fmap->load_commands);
320 macho_unmap_range((const void**)&fmap->load_commands, fmap,
321 sizeof(fmap->mach_header), fmap->mach_header.sizeofcmds);
325 /******************************************************************
326 * macho_next_load_command
328 * Advance to the next load command
330 static const struct load_command* macho_next_load_command(const struct load_command* lc)
332 return (const struct load_command*)((const char*)lc + lc->cmdsize);
335 /******************************************************************
336 * macho_enum_load_commands
338 * Enumerates the load commands for a Mach-O file, selecting by
339 * the command type, calling a callback for each. If the callback
340 * returns <0, that indicates an error. If it returns >0, that means
341 * it's not interested in getting any more load commands.
342 * If this function returns <0, that's an error produced by the
343 * callback. If >=0, that's the count of load commands successfully
344 * processed.
346 static int macho_enum_load_commands(struct macho_file_map* fmap, unsigned cmd,
347 int (*cb)(struct macho_file_map*, const struct load_command*, void*),
348 void* user)
350 const struct load_command* lc;
351 int i;
352 int count = 0;
354 TRACE("(%p/%d, %u, %p, %p)\n", fmap, fmap->fd, cmd, cb, user);
356 if ((lc = macho_map_load_commands(fmap)) == MACHO_NO_MAP) return -1;
358 TRACE("%d total commands\n", fmap->mach_header.ncmds);
360 for (i = 0; i < fmap->mach_header.ncmds; i++, lc = macho_next_load_command(lc))
362 int result;
364 if (cmd && cmd != lc->cmd) continue;
365 count++;
367 result = cb(fmap, lc, user);
368 TRACE("load_command[%d] (%p), cmd %u; callback => %d\n", i, lc, lc->cmd, result);
369 if (result) return (result < 0) ? result : count;
372 return count;
375 /******************************************************************
376 * macho_accum_segs_range
378 * Callback for macho_enum_load_commands. Accumulates the address
379 * range covered by the segments of a Mach-O file. All commands
380 * are expected to be of LC_SEGMENT type.
382 static int macho_accum_segs_range(struct macho_file_map* fmap,
383 const struct load_command* lc, void* user)
385 const struct segment_command* sc = (const struct segment_command*)lc;
386 unsigned tmp, page_mask = getpagesize() - 1;
388 TRACE("(%p/%d, %p, %p) before: 0x%08x - 0x%08x\n", fmap, fmap->fd, lc, user,
389 (unsigned)fmap->segs_start, (unsigned)fmap->segs_size);
390 TRACE("Segment command vm: 0x%08x - 0x%08x\n", (unsigned)sc->vmaddr,
391 (unsigned)sc->vmaddr + sc->vmsize);
393 if (!strncmp(sc->segname, "WINE_", 5))
395 TRACE("Ignoring special Wine segment %s\n", debugstr_an(sc->segname, sizeof(sc->segname)));
396 return 0;
399 /* If this segment starts before previously-known earliest, record
400 * new earliest. */
401 if (sc->vmaddr < fmap->segs_start)
402 fmap->segs_start = sc->vmaddr;
404 /* If this segment extends beyond previously-known furthest, record
405 * new furthest. */
406 tmp = (sc->vmaddr + sc->vmsize + page_mask) & ~page_mask;
407 if (fmap->segs_size < tmp) fmap->segs_size = tmp;
409 TRACE("after: 0x%08x - 0x%08x\n", (unsigned)fmap->segs_start, (unsigned)fmap->segs_size);
411 return 0;
414 /******************************************************************
415 * macho_map_file
417 * Maps a Mach-O file into memory (and checks it's a real Mach-O file)
419 static BOOL macho_map_file(const WCHAR* filenameW, struct macho_file_map* fmap)
421 struct fat_header fat_header;
422 struct stat statbuf;
423 int i;
424 char* filename;
425 unsigned len;
426 BOOL ret = FALSE;
428 TRACE("(%s, %p)\n", debugstr_w(filenameW), fmap);
430 fmap->fd = -1;
431 fmap->load_commands = MACHO_NO_MAP;
432 RtlInitializeBitMap(&fmap->sect_is_code, fmap->sect_is_code_buff, MAX_SECT + 1);
434 len = WideCharToMultiByte(CP_UNIXCP, 0, filenameW, -1, NULL, 0, NULL, NULL);
435 if (!(filename = HeapAlloc(GetProcessHeap(), 0, len))) return FALSE;
436 WideCharToMultiByte(CP_UNIXCP, 0, filenameW, -1, filename, len, NULL, NULL);
438 /* check that the file exists */
439 if (stat(filename, &statbuf) == -1 || S_ISDIR(statbuf.st_mode)) goto done;
441 /* Now open the file, so that we can mmap() it. */
442 if ((fmap->fd = open(filename, O_RDONLY)) == -1) goto done;
444 if (read(fmap->fd, &fat_header, sizeof(fat_header)) != sizeof(fat_header))
445 goto done;
446 TRACE("... got possible fat header\n");
448 /* Fat header is always in big-endian order. */
449 if (swap_ulong_be_to_host(fat_header.magic) == FAT_MAGIC)
451 int narch = swap_ulong_be_to_host(fat_header.nfat_arch);
452 for (i = 0; i < narch; i++)
454 struct fat_arch fat_arch;
455 if (read(fmap->fd, &fat_arch, sizeof(fat_arch)) != sizeof(fat_arch))
456 goto done;
457 if (swap_ulong_be_to_host(fat_arch.cputype) == CPU_TYPE_X86)
459 fmap->arch_offset = swap_ulong_be_to_host(fat_arch.offset);
460 fmap->arch_size = swap_ulong_be_to_host(fat_arch.size);
461 break;
464 if (i >= narch) goto done;
465 TRACE("... found x86 arch\n");
467 else
469 fmap->arch_offset = 0;
470 fmap->arch_size = statbuf.st_size;
471 TRACE("... not a fat header\n");
474 /* Individual architecture (standalone or within a fat file) is in its native byte order. */
475 lseek(fmap->fd, fmap->arch_offset, SEEK_SET);
476 if (read(fmap->fd, &fmap->mach_header, sizeof(fmap->mach_header)) != sizeof(fmap->mach_header))
477 goto done;
478 TRACE("... got possible Mach header\n");
479 /* and check for a Mach-O header */
480 if (fmap->mach_header.magic != MH_MAGIC ||
481 fmap->mach_header.cputype != CPU_TYPE_X86) goto done;
482 /* Make sure the file type is one of the ones we expect. */
483 switch (fmap->mach_header.filetype)
485 case MH_EXECUTE:
486 case MH_DYLIB:
487 case MH_DYLINKER:
488 case MH_BUNDLE:
489 break;
490 default:
491 goto done;
493 TRACE("... verified Mach x86 header\n");
495 fmap->segs_size = 0;
496 fmap->segs_start = ~0L;
498 if (macho_enum_load_commands(fmap, LC_SEGMENT, macho_accum_segs_range, NULL) < 0)
499 goto done;
501 fmap->segs_size -= fmap->segs_start;
502 TRACE("segs_start: 0x%08x, segs_size: 0x%08x\n", (unsigned)fmap->segs_start,
503 (unsigned)fmap->segs_size);
505 ret = TRUE;
506 done:
507 if (!ret)
508 macho_unmap_file(fmap);
509 HeapFree(GetProcessHeap(), 0, filename);
510 return ret;
513 /******************************************************************
514 * macho_unmap_file
516 * Unmaps a Mach-O file from memory (previously mapped with macho_map_file)
518 static void macho_unmap_file(struct macho_file_map* fmap)
520 TRACE("(%p/%d)\n", fmap, fmap->fd);
521 if (fmap->fd != -1)
523 macho_unmap_load_commands(fmap);
524 close(fmap->fd);
525 fmap->fd = -1;
529 /******************************************************************
530 * macho_fill_sect_is_code
532 * Callback for macho_enum_load_commands. Determines which segments
533 * of a Mach-O file contain code. All commands are expected to be
534 * of LC_SEGMENT type.
536 static int macho_fill_sect_is_code(struct macho_file_map* fmap,
537 const struct load_command* lc, void* user)
539 const struct segment_command* sc = (const struct segment_command*)lc;
540 const struct section* sections;
541 int* cursect = user;
542 int i;
544 TRACE("(%p/%d, %p, %p/%d) scanning %u sections\n", fmap, fmap->fd, lc,
545 cursect, *cursect, sc->nsects);
547 sections = (const struct section*)(sc + 1);
548 for (i = 0; i < sc->nsects; i++)
550 if (*cursect > MAX_SECT) return -1;
551 (*cursect)++;
553 if (!(sections[i].flags & SECTION_TYPE) &&
554 (sections[i].flags & (S_ATTR_PURE_INSTRUCTIONS|S_ATTR_SOME_INSTRUCTIONS)))
555 RtlSetBits(&fmap->sect_is_code, *cursect, 1);
556 else
557 RtlClearBits(&fmap->sect_is_code, *cursect, 1);
558 TRACE("Section %d (%d of this segment) is%s code\n", *cursect, i,
559 (RtlAreBitsSet(&fmap->sect_is_code, *cursect, 1) ? "" : " not"));
562 return 0;
565 /******************************************************************
566 * macho_sect_is_code
568 * Checks if a section, identified by sectidx which is a 1-based
569 * index into the sections of all segments, in order of load
570 * commands, contains code.
572 static BOOL macho_sect_is_code(struct macho_file_map* fmap, unsigned char sectidx)
574 TRACE("(%p/%d, %u)\n", fmap, fmap->fd, sectidx);
576 if (!RtlAreBitsSet(&fmap->sect_is_code, 0, 1))
578 int cursect = 0;
579 if (macho_enum_load_commands(fmap, LC_SEGMENT, macho_fill_sect_is_code, &cursect) < 0)
580 WARN("Couldn't load sect_is_code map\n");
581 RtlSetBits(&fmap->sect_is_code, 0, 1);
584 return RtlAreBitsSet(&fmap->sect_is_code, sectidx, 1);
587 struct symtab_elt
589 struct hash_table_elt ht_elt;
590 struct symt_compiland* compiland;
591 unsigned long addr;
592 unsigned char is_code:1,
593 is_public:1,
594 is_global:1,
595 used:1;
598 struct macho_debug_info
600 struct macho_file_map* fmap;
601 struct module* module;
602 struct pool pool;
603 struct hash_table ht_symtab;
606 /******************************************************************
607 * macho_stabs_def_cb
609 * Callback for stabs_parse. Collect symbol definitions.
611 static void macho_stabs_def_cb(struct module* module, unsigned long load_offset,
612 const char* name, unsigned long offset,
613 BOOL is_public, BOOL is_global, unsigned char sectidx,
614 struct symt_compiland* compiland, void* user)
616 struct macho_debug_info* mdi = user;
617 struct symtab_elt* ste;
619 TRACE("(%p, 0x%08lx, %s, 0x%08lx, %d, %d, %u, %p, %p/%p/%d)\n", module, load_offset,
620 debugstr_a(name), offset, is_public, is_global, sectidx,
621 compiland, mdi, mdi->fmap, mdi->fmap->fd);
623 /* Defer the creation of new non-debugging symbols until after we've
624 * finished parsing the stabs. */
625 ste = pool_alloc(&mdi->pool, sizeof(*ste));
626 ste->ht_elt.name = pool_strdup(&mdi->pool, name);
627 ste->compiland = compiland;
628 ste->addr = load_offset + offset;
629 ste->is_code = !!macho_sect_is_code(mdi->fmap, sectidx);
630 ste->is_public = !!is_public;
631 ste->is_global = !!is_global;
632 ste->used = 0;
633 hash_table_add(&mdi->ht_symtab, &ste->ht_elt);
636 /******************************************************************
637 * macho_parse_symtab
639 * Callback for macho_enum_load_commands. Processes the LC_SYMTAB
640 * load commands from the Mach-O file.
642 static int macho_parse_symtab(struct macho_file_map* fmap,
643 const struct load_command* lc, void* user)
645 const struct symtab_command* sc = (const struct symtab_command*)lc;
646 struct macho_debug_info* mdi = user;
647 const struct nlist* stab;
648 const char* stabstr;
649 int ret = 0;
651 TRACE("(%p/%d, %p, %p) %u syms at 0x%08x, strings 0x%08x - 0x%08x\n", fmap, fmap->fd, lc,
652 user, sc->nsyms, sc->symoff, sc->stroff, sc->stroff + sc->strsize);
654 if (!macho_map_ranges(fmap, sc->symoff, sc->nsyms * sizeof(struct nlist),
655 sc->stroff, sc->strsize, (const void**)&stab, (const void**)&stabstr))
656 return 0;
658 if (!stabs_parse(mdi->module,
659 mdi->module->format_info[DFI_MACHO]->u.macho_info->load_addr - fmap->segs_start,
660 stab, sc->nsyms * sizeof(struct nlist),
661 stabstr, sc->strsize, macho_stabs_def_cb, mdi))
662 ret = -1;
664 macho_unmap_ranges(fmap, sc->symoff, sc->nsyms * sizeof(struct nlist),
665 sc->stroff, sc->strsize, (const void**)&stab, (const void**)&stabstr);
667 return ret;
670 /******************************************************************
671 * macho_finish_stabs
673 * Integrate the non-debugging symbols we've gathered into the
674 * symbols that were generated during stabs parsing.
676 static void macho_finish_stabs(struct module* module, struct hash_table* ht_symtab)
678 struct hash_table_iter hti_ours;
679 struct symtab_elt* ste;
680 BOOL adjusted = FALSE;
682 TRACE("(%p, %p)\n", module, ht_symtab);
684 /* For each of our non-debugging symbols, see if it can provide some
685 * missing details to one of the module's known symbols. */
686 hash_table_iter_init(ht_symtab, &hti_ours, NULL);
687 while ((ste = hash_table_iter_up(&hti_ours)))
689 struct hash_table_iter hti_modules;
690 void* ptr;
691 struct symt_ht* sym;
692 struct symt_function* func;
693 struct symt_data* data;
695 hash_table_iter_init(&module->ht_symbols, &hti_modules, ste->ht_elt.name);
696 while ((ptr = hash_table_iter_up(&hti_modules)))
698 sym = GET_ENTRY(ptr, struct symt_ht, hash_elt);
700 if (strcmp(sym->hash_elt.name, ste->ht_elt.name))
701 continue;
703 switch (sym->symt.tag)
705 case SymTagFunction:
706 func = (struct symt_function*)sym;
707 if (func->address == module->format_info[DFI_MACHO]->u.macho_info->load_addr)
709 TRACE("Adjusting function %p/%s!%s from 0x%08lx to 0x%08lx\n", func,
710 debugstr_w(module->module.ModuleName), sym->hash_elt.name,
711 func->address, ste->addr);
712 func->address = ste->addr;
713 adjusted = TRUE;
715 if (func->address == ste->addr)
716 ste->used = 1;
717 break;
718 case SymTagData:
719 data = (struct symt_data*)sym;
720 switch (data->kind)
722 case DataIsGlobal:
723 case DataIsFileStatic:
724 if (data->u.var.offset == module->format_info[DFI_MACHO]->u.macho_info->load_addr)
726 TRACE("Adjusting data symbol %p/%s!%s from 0x%08lx to 0x%08lx\n",
727 data, debugstr_w(module->module.ModuleName), sym->hash_elt.name,
728 data->u.var.offset, ste->addr);
729 data->u.var.offset = ste->addr;
730 adjusted = TRUE;
732 if (data->u.var.offset == ste->addr)
734 enum DataKind new_kind;
736 new_kind = ste->is_global ? DataIsGlobal : DataIsFileStatic;
737 if (data->kind != new_kind)
739 WARN("Changing kind for %p/%s!%s from %d to %d\n", sym,
740 debugstr_w(module->module.ModuleName), sym->hash_elt.name,
741 (int)data->kind, (int)new_kind);
742 data->kind = new_kind;
743 adjusted = TRUE;
745 ste->used = 1;
747 break;
748 default:;
750 break;
751 default:
752 TRACE("Ignoring tag %u\n", sym->symt.tag);
753 break;
758 if (adjusted)
760 /* since we may have changed some addresses, mark the module to be resorted */
761 module->sortlist_valid = FALSE;
764 /* Mark any of our non-debugging symbols which fall on an already-used
765 * address as "used". This allows us to skip them in the next loop,
766 * below. We do this in separate loops because symt_new_* marks the
767 * list as needing sorting and symt_find_nearest sorts if needed,
768 * causing thrashing. */
769 if (!(dbghelp_options & SYMOPT_PUBLICS_ONLY))
771 hash_table_iter_init(ht_symtab, &hti_ours, NULL);
772 while ((ste = hash_table_iter_up(&hti_ours)))
774 struct symt_ht* sym;
775 ULONG64 addr;
777 if (ste->used) continue;
779 sym = symt_find_nearest(module, ste->addr);
780 if (sym)
781 symt_get_info(module, &sym->symt, TI_GET_ADDRESS, &addr);
782 if (sym && ste->addr == addr)
784 ULONG64 size = 0;
785 DWORD kind = -1;
787 ste->used = 1;
789 /* If neither symbol has a correct size (ours never does), we
790 * consider them both to be markers. No warning is needed in
791 * that case.
792 * Also, we check that we don't have two symbols, one local, the other
793 * global, which is legal.
795 symt_get_info(module, &sym->symt, TI_GET_LENGTH, &size);
796 symt_get_info(module, &sym->symt, TI_GET_DATAKIND, &kind);
797 if (size && kind == (ste->is_global ? DataIsGlobal : DataIsFileStatic))
798 FIXME("Duplicate in %s: %s<%08lx> %s<%s-%s>\n",
799 debugstr_w(module->module.ModuleName),
800 ste->ht_elt.name, ste->addr,
801 sym->hash_elt.name,
802 wine_dbgstr_longlong(addr), wine_dbgstr_longlong(size));
807 /* For any of our remaining non-debugging symbols which have no match
808 * among the module's known symbols, add them as new symbols. */
809 hash_table_iter_init(ht_symtab, &hti_ours, NULL);
810 while ((ste = hash_table_iter_up(&hti_ours)))
812 if (!(dbghelp_options & SYMOPT_PUBLICS_ONLY) && !ste->used)
814 if (ste->is_code)
816 symt_new_function(module, ste->compiland, ste->ht_elt.name,
817 ste->addr, 0, NULL);
819 else
821 symt_new_global_variable(module, ste->compiland, ste->ht_elt.name,
822 !ste->is_global, ste->addr, 0, NULL);
825 ste->used = 1;
828 if (ste->is_public && !(dbghelp_options & SYMOPT_NO_PUBLICS))
830 symt_new_public(module, ste->compiland, ste->ht_elt.name, ste->addr, 0);
835 /******************************************************************
836 * macho_load_debug_info_from_map
838 * Loads the symbolic information from a Mach-O module.
839 * Returns
840 * FALSE if the file doesn't contain symbolic info (or this info
841 * cannot be read or parsed)
842 * TRUE on success
844 static BOOL macho_load_debug_info_from_map(struct module* module,
845 struct macho_file_map* fmap)
847 BOOL ret = FALSE;
848 struct macho_debug_info mdi;
849 int result;
851 TRACE("(%p, %p/%d)\n", module, fmap, fmap->fd);
853 module->module.SymType = SymExport;
855 mdi.fmap = fmap;
856 mdi.module = module;
857 pool_init(&mdi.pool, 65536);
858 hash_table_init(&mdi.pool, &mdi.ht_symtab, 256);
859 result = macho_enum_load_commands(fmap, LC_SYMTAB, macho_parse_symtab, &mdi);
860 if (result > 0)
861 ret = TRUE;
862 else if (result < 0)
863 WARN("Couldn't correctly read stabs\n");
865 macho_finish_stabs(module, &mdi.ht_symtab);
867 pool_destroy(&mdi.pool);
868 return ret;
871 /******************************************************************
872 * macho_load_debug_info
874 * Loads Mach-O debugging information from the module image file.
876 BOOL macho_load_debug_info(struct module* module, struct macho_file_map* fmap)
878 BOOL ret = TRUE;
879 struct macho_file_map my_fmap;
881 TRACE("(%p, %p/%d)\n", module, fmap, fmap ? fmap->fd : -1);
883 if (module->type != DMT_MACHO || !module->format_info[DFI_MACHO]->u.macho_info)
885 ERR("Bad Mach-O module '%s'\n", debugstr_w(module->module.LoadedImageName));
886 return FALSE;
889 if (!fmap)
891 fmap = &my_fmap;
892 ret = macho_map_file(module->module.LoadedImageName, fmap);
894 if (ret)
895 ret = macho_load_debug_info_from_map(module, fmap);
897 if (fmap == &my_fmap) macho_unmap_file(fmap);
898 return ret;
901 /******************************************************************
902 * macho_fetch_file_info
904 * Gathers some more information for a Mach-O module from a given file
906 BOOL macho_fetch_file_info(const WCHAR* name, DWORD* base,
907 DWORD* size, DWORD* checksum)
909 struct macho_file_map fmap;
911 TRACE("(%s, %p, %p, %p)\n", debugstr_w(name), base, size, checksum);
913 if (!macho_map_file(name, &fmap)) return FALSE;
914 if (base) *base = fmap.segs_start;
915 *size = fmap.segs_size;
916 *checksum = calc_crc32(fmap.fd);
917 macho_unmap_file(&fmap);
918 return TRUE;
921 /******************************************************************
922 * macho_load_file
924 * Loads the information for Mach-O module stored in 'filename'.
925 * The module has been loaded at 'load_addr' address.
926 * returns
927 * FALSE if the file cannot be found/opened or if the file doesn't
928 * contain symbolic info (or this info cannot be read or parsed)
929 * TRUE on success
931 static BOOL macho_load_file(struct process* pcs, const WCHAR* filename,
932 unsigned long load_addr, struct macho_info* macho_info)
934 BOOL ret = TRUE;
935 struct macho_file_map fmap;
937 TRACE("(%p/%p, %s, 0x%08lx, %p/0x%08x)\n", pcs, pcs->handle, debugstr_w(filename),
938 load_addr, macho_info, macho_info->flags);
940 if (!macho_map_file(filename, &fmap)) return FALSE;
942 /* Find the dynamic loader's table of images loaded into the process.
944 if (macho_info->flags & MACHO_INFO_DEBUG_HEADER)
946 static void* dyld_all_image_infos_addr;
948 /* This symbol should be in the same place in all processes. */
949 if (!dyld_all_image_infos_addr)
951 struct nlist nl[2];
952 memset(nl, 0, sizeof(nl));
953 nl[0].n_un.n_name = (char*)"_dyld_all_image_infos";
954 if (!nlist("/usr/lib/dyld", nl))
955 dyld_all_image_infos_addr = (void*)nl[0].n_value;
958 if (dyld_all_image_infos_addr)
959 macho_info->dbg_hdr_addr = (unsigned long)dyld_all_image_infos_addr;
960 else
961 ret = FALSE;
962 TRACE("dbg_hdr_addr = 0x%08lx\n", macho_info->dbg_hdr_addr);
965 if (macho_info->flags & MACHO_INFO_MODULE)
967 struct macho_module_info *macho_module_info;
968 struct module_format* modfmt =
969 HeapAlloc(GetProcessHeap(), 0, sizeof(struct module_format) + sizeof(struct macho_module_info));
970 if (!modfmt) goto leave;
971 macho_info->module = module_new(pcs, filename, DMT_MACHO, FALSE, load_addr,
972 fmap.segs_size, 0, calc_crc32(fmap.fd));
973 if (!macho_info->module)
975 HeapFree(GetProcessHeap(), 0, modfmt);
976 goto leave;
978 macho_module_info = (void*)(modfmt + 1);
979 macho_info->module->format_info[DFI_MACHO] = modfmt;
981 modfmt->module = macho_info->module;
982 modfmt->remove = NULL;
983 modfmt->loc_compute = NULL;
984 modfmt->u.macho_info = macho_module_info;
986 macho_module_info->load_addr = load_addr;
988 if (dbghelp_options & SYMOPT_DEFERRED_LOADS)
989 macho_info->module->module.SymType = SymDeferred;
990 else if (!macho_load_debug_info(macho_info->module, &fmap))
991 ret = FALSE;
993 macho_info->module->format_info[DFI_MACHO]->u.macho_info->in_use = 1;
994 macho_info->module->format_info[DFI_MACHO]->u.macho_info->is_loader = 0;
995 TRACE("module = %p\n", macho_info->module);
998 if (macho_info->flags & MACHO_INFO_NAME)
1000 WCHAR* ptr;
1001 ptr = HeapAlloc(GetProcessHeap(), 0, (lstrlenW(filename) + 1) * sizeof(WCHAR));
1002 if (ptr)
1004 strcpyW(ptr, filename);
1005 macho_info->module_name = ptr;
1007 else ret = FALSE;
1008 TRACE("module_name = %p %s\n", macho_info->module_name, debugstr_w(macho_info->module_name));
1010 leave:
1011 macho_unmap_file(&fmap);
1013 TRACE(" => %d\n", ret);
1014 return ret;
1017 /******************************************************************
1018 * macho_load_file_from_path
1019 * Tries to load a Mach-O file from a set of paths (separated by ':')
1021 static BOOL macho_load_file_from_path(HANDLE hProcess,
1022 const WCHAR* filename,
1023 unsigned long load_addr,
1024 const char* path,
1025 struct macho_info* macho_info)
1027 BOOL ret = FALSE;
1028 WCHAR *s, *t, *fn;
1029 WCHAR* pathW = NULL;
1030 unsigned len;
1032 TRACE("(%p, %s, 0x%08lx, %s, %p)\n", hProcess, debugstr_w(filename), load_addr,
1033 debugstr_a(path), macho_info);
1035 if (!path) return FALSE;
1037 len = MultiByteToWideChar(CP_UNIXCP, 0, path, -1, NULL, 0);
1038 pathW = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
1039 if (!pathW) return FALSE;
1040 MultiByteToWideChar(CP_UNIXCP, 0, path, -1, pathW, len);
1042 for (s = pathW; s && *s; s = (t) ? (t+1) : NULL)
1044 t = strchrW(s, ':');
1045 if (t) *t = '\0';
1046 fn = HeapAlloc(GetProcessHeap(), 0, (lstrlenW(filename) + 1 + lstrlenW(s) + 1) * sizeof(WCHAR));
1047 if (!fn) break;
1048 strcpyW(fn, s);
1049 strcatW(fn, S_SlashW);
1050 strcatW(fn, filename);
1051 ret = macho_load_file(hProcess, fn, load_addr, macho_info);
1052 HeapFree(GetProcessHeap(), 0, fn);
1053 if (ret) break;
1054 s = (t) ? (t+1) : NULL;
1057 TRACE(" => %d\n", ret);
1058 HeapFree(GetProcessHeap(), 0, pathW);
1059 return ret;
1062 /******************************************************************
1063 * macho_load_file_from_dll_path
1065 * Tries to load a Mach-O file from the dll path
1067 static BOOL macho_load_file_from_dll_path(HANDLE hProcess,
1068 const WCHAR* filename,
1069 unsigned long load_addr,
1070 struct macho_info* macho_info)
1072 BOOL ret = FALSE;
1073 unsigned int index = 0;
1074 const char *path;
1076 TRACE("(%p, %s, 0x%08lx, %p)\n", hProcess, debugstr_w(filename), load_addr,
1077 macho_info);
1079 while (!ret && (path = wine_dll_enum_load_path( index++ )))
1081 WCHAR *name;
1082 unsigned len;
1084 len = MultiByteToWideChar(CP_UNIXCP, 0, path, -1, NULL, 0);
1086 name = HeapAlloc( GetProcessHeap(), 0,
1087 (len + lstrlenW(filename) + 2) * sizeof(WCHAR) );
1089 if (!name) break;
1090 MultiByteToWideChar(CP_UNIXCP, 0, path, -1, name, len);
1091 strcatW( name, S_SlashW );
1092 strcatW( name, filename );
1093 ret = macho_load_file(hProcess, name, load_addr, macho_info);
1094 HeapFree( GetProcessHeap(), 0, name );
1096 TRACE(" => %d\n", ret);
1097 return ret;
1100 /******************************************************************
1101 * macho_search_and_load_file
1103 * Lookup a file in standard Mach-O locations, and if found, load it
1105 static BOOL macho_search_and_load_file(struct process* pcs, const WCHAR* filename,
1106 unsigned long load_addr,
1107 struct macho_info* macho_info)
1109 BOOL ret = FALSE;
1110 struct module* module;
1111 static WCHAR S_libstdcPPW[] = {'l','i','b','s','t','d','c','+','+','\0'};
1112 const WCHAR* p;
1114 TRACE("(%p/%p, %s, 0x%08lx, %p)\n", pcs, pcs->handle, debugstr_w(filename), load_addr,
1115 macho_info);
1117 if (filename == NULL || *filename == '\0') return FALSE;
1118 if ((module = module_is_already_loaded(pcs, filename)))
1120 macho_info->module = module;
1121 module->format_info[DFI_MACHO]->u.macho_info->in_use = 1;
1122 return module->module.SymType;
1125 if (strstrW(filename, S_libstdcPPW)) return FALSE; /* We know we can't do it */
1127 /* If has no directories, try LD_LIBRARY_PATH first. */
1128 if (!strchrW(filename, '/'))
1130 ret = macho_load_file_from_path(pcs, filename, load_addr,
1131 getenv("PATH"), macho_info);
1133 /* Try DYLD_LIBRARY_PATH, with just the filename (no directories). */
1134 if (!ret)
1136 if ((p = strrchrW(filename, '/'))) p++;
1137 else p = filename;
1138 ret = macho_load_file_from_path(pcs, p, load_addr,
1139 getenv("DYLD_LIBRARY_PATH"), macho_info);
1141 /* Try the path as given. */
1142 if (!ret)
1143 ret = macho_load_file(pcs, filename, load_addr, macho_info);
1144 /* Try DYLD_FALLBACK_LIBRARY_PATH, with just the filename (no directories). */
1145 if (!ret)
1147 ret = macho_load_file_from_path(pcs, p, load_addr,
1148 getenv("DYLD_FALLBACK_LIBRARY_PATH"), macho_info);
1150 if (!ret && !strchrW(filename, '/'))
1151 ret = macho_load_file_from_dll_path(pcs, filename, load_addr, macho_info);
1153 return ret;
1156 /******************************************************************
1157 * macho_enum_modules_internal
1159 * Enumerate Mach-O modules from a running process
1161 static BOOL macho_enum_modules_internal(const struct process* pcs,
1162 const WCHAR* main_name,
1163 enum_modules_cb cb, void* user)
1165 struct dyld_all_image_infos image_infos;
1166 struct dyld_image_info* info_array = NULL;
1167 unsigned long len;
1168 int i;
1169 char bufstr[256];
1170 WCHAR bufstrW[MAX_PATH];
1171 BOOL ret = FALSE;
1173 TRACE("(%p/%p, %s, %p, %p)\n", pcs, pcs->handle, debugstr_w(main_name), cb,
1174 user);
1176 if (!pcs->dbg_hdr_addr ||
1177 !ReadProcessMemory(pcs->handle, (void*)pcs->dbg_hdr_addr,
1178 &image_infos, sizeof(image_infos), NULL) ||
1179 !image_infos.infoArray)
1180 goto done;
1181 TRACE("Process has %u image infos at %p\n", image_infos.infoArrayCount, image_infos.infoArray);
1183 len = image_infos.infoArrayCount * sizeof(info_array[0]);
1184 info_array = HeapAlloc(GetProcessHeap(), 0, len);
1185 if (!info_array ||
1186 !ReadProcessMemory(pcs->handle, image_infos.infoArray,
1187 info_array, len, NULL))
1188 goto done;
1189 TRACE("... read image infos\n");
1191 for (i = 0; i < image_infos.infoArrayCount; i++)
1193 if (info_array[i].imageFilePath != NULL &&
1194 ReadProcessMemory(pcs->handle, info_array[i].imageFilePath, bufstr, sizeof(bufstr), NULL))
1196 bufstr[sizeof(bufstr) - 1] = '\0';
1197 TRACE("[%d] image file %s\n", i, debugstr_a(bufstr));
1198 MultiByteToWideChar(CP_UNIXCP, 0, bufstr, -1, bufstrW, sizeof(bufstrW) / sizeof(WCHAR));
1199 if (main_name && !bufstrW[0]) strcpyW(bufstrW, main_name);
1200 if (!cb(bufstrW, (unsigned long)info_array[i].imageLoadAddress, user)) break;
1204 ret = TRUE;
1205 done:
1206 HeapFree(GetProcessHeap(), 0, info_array);
1207 return ret;
1210 struct macho_sync
1212 struct process* pcs;
1213 struct macho_info macho_info;
1216 static BOOL macho_enum_sync_cb(const WCHAR* name, unsigned long addr, void* user)
1218 struct macho_sync* ms = user;
1220 TRACE("(%s, 0x%08lx, %p)\n", debugstr_w(name), addr, user);
1221 macho_search_and_load_file(ms->pcs, name, addr, &ms->macho_info);
1222 return TRUE;
1225 /******************************************************************
1226 * macho_synchronize_module_list
1228 * Rescans the debuggee's modules list and synchronizes it with
1229 * the one from 'pcs', ie:
1230 * - if a module is in debuggee and not in pcs, it's loaded into pcs
1231 * - if a module is in pcs and not in debuggee, it's unloaded from pcs
1233 BOOL macho_synchronize_module_list(struct process* pcs)
1235 struct module* module;
1236 struct macho_sync ms;
1238 TRACE("(%p/%p)\n", pcs, pcs->handle);
1240 for (module = pcs->lmodules; module; module = module->next)
1242 if (module->type == DMT_MACHO && !module->is_virtual)
1243 module->format_info[DFI_MACHO]->u.macho_info->in_use = 0;
1246 ms.pcs = pcs;
1247 ms.macho_info.flags = MACHO_INFO_MODULE;
1248 if (!macho_enum_modules_internal(pcs, NULL, macho_enum_sync_cb, &ms))
1249 return FALSE;
1251 module = pcs->lmodules;
1252 while (module)
1254 if (module->type == DMT_MACHO && !module->is_virtual &&
1255 !module->format_info[DFI_MACHO]->u.macho_info->in_use &&
1256 !module->format_info[DFI_MACHO]->u.macho_info->is_loader)
1258 module_remove(pcs, module);
1259 /* restart all over */
1260 module = pcs->lmodules;
1262 else module = module->next;
1264 return TRUE;
1267 /******************************************************************
1268 * macho_search_loader
1270 * Lookup in a running Mach-O process the loader, and sets its Mach-O link
1271 * address (for accessing the list of loaded images) in pcs.
1272 * If flags is MACHO_INFO_MODULE, the module for the loader is also
1273 * added as a module into pcs.
1275 static BOOL macho_search_loader(struct process* pcs, struct macho_info* macho_info)
1277 BOOL ret;
1278 const char* ptr;
1280 TRACE("(%p/%p, %p)\n", pcs, pcs->handle, macho_info);
1282 /* All binaries are loaded with WINELOADER (if run from tree) or by the
1283 * main executable
1285 if ((ptr = getenv("WINELOADER")))
1287 WCHAR tmp[MAX_PATH];
1288 MultiByteToWideChar(CP_UNIXCP, 0, ptr, -1, tmp, sizeof(tmp) / sizeof(WCHAR));
1289 ret = macho_search_and_load_file(pcs, tmp, 0, macho_info);
1291 else
1293 ret = macho_search_and_load_file(pcs, S_WineW, 0, macho_info);
1295 return ret;
1298 /******************************************************************
1299 * macho_read_wine_loader_dbg_info
1301 * Try to find a decent wine executable which could have loaded the debuggee
1303 BOOL macho_read_wine_loader_dbg_info(struct process* pcs)
1305 struct macho_info macho_info;
1307 TRACE("(%p/%p)\n", pcs, pcs->handle);
1308 macho_info.flags = MACHO_INFO_DEBUG_HEADER | MACHO_INFO_MODULE;
1309 if (!macho_search_loader(pcs, &macho_info)) return FALSE;
1310 macho_info.module->format_info[DFI_MACHO]->u.macho_info->is_loader = 1;
1311 module_set_module(macho_info.module, S_WineLoaderW);
1312 return (pcs->dbg_hdr_addr = macho_info.dbg_hdr_addr) != 0;
1315 /******************************************************************
1316 * macho_enum_modules
1318 * Enumerates the Mach-O loaded modules from a running target (hProc)
1319 * This function doesn't require that someone has called SymInitialize
1320 * on this very process.
1322 BOOL macho_enum_modules(HANDLE hProc, enum_modules_cb cb, void* user)
1324 struct process pcs;
1325 struct macho_info macho_info;
1326 BOOL ret;
1328 TRACE("(%p, %p, %p)\n", hProc, cb, user);
1329 memset(&pcs, 0, sizeof(pcs));
1330 pcs.handle = hProc;
1331 macho_info.flags = MACHO_INFO_DEBUG_HEADER | MACHO_INFO_NAME;
1332 if (!macho_search_loader(&pcs, &macho_info)) return FALSE;
1333 pcs.dbg_hdr_addr = macho_info.dbg_hdr_addr;
1334 ret = macho_enum_modules_internal(&pcs, macho_info.module_name, cb, user);
1335 HeapFree(GetProcessHeap(), 0, (char*)macho_info.module_name);
1336 return ret;
1339 struct macho_load
1341 struct process* pcs;
1342 struct macho_info macho_info;
1343 const WCHAR* name;
1344 BOOL ret;
1347 /******************************************************************
1348 * macho_load_cb
1350 * Callback for macho_load_module, used to walk the list of loaded
1351 * modules.
1353 static BOOL macho_load_cb(const WCHAR* name, unsigned long addr, void* user)
1355 struct macho_load* ml = user;
1356 const WCHAR* p;
1358 TRACE("(%s, 0x%08lx, %p)\n", debugstr_w(name), addr, user);
1360 /* memcmp is needed for matches when bufstr contains also version information
1361 * ml->name: libc.so, name: libc.so.6.0
1363 p = strrchrW(name, '/');
1364 if (!p++) p = name;
1365 if (!memcmp(p, ml->name, lstrlenW(ml->name) * sizeof(WCHAR)))
1367 ml->ret = macho_search_and_load_file(ml->pcs, name, addr, &ml->macho_info);
1368 return FALSE;
1370 return TRUE;
1373 /******************************************************************
1374 * macho_load_module
1376 * Loads a Mach-O module and stores it in process' module list.
1377 * Also, find module real name and load address from
1378 * the real loaded modules list in pcs address space.
1380 struct module* macho_load_module(struct process* pcs, const WCHAR* name, unsigned long addr)
1382 struct macho_load ml;
1384 TRACE("(%p/%p, %s, 0x%08lx)\n", pcs, pcs->handle, debugstr_w(name), addr);
1386 ml.macho_info.flags = MACHO_INFO_MODULE;
1387 ml.ret = FALSE;
1389 if (pcs->dbg_hdr_addr) /* we're debugging a live target */
1391 ml.pcs = pcs;
1392 /* do only the lookup from the filename, not the path (as we lookup module
1393 * name in the process' loaded module list)
1395 ml.name = strrchrW(name, '/');
1396 if (!ml.name++) ml.name = name;
1397 ml.ret = FALSE;
1399 if (!macho_enum_modules_internal(pcs, NULL, macho_load_cb, &ml))
1400 return NULL;
1402 else if (addr)
1404 ml.name = name;
1405 ml.ret = macho_search_and_load_file(pcs, ml.name, addr, &ml.macho_info);
1407 if (!ml.ret) return NULL;
1408 assert(ml.macho_info.module);
1409 return ml.macho_info.module;
1412 #else /* !__MACH__ */
1414 BOOL macho_synchronize_module_list(struct process* pcs)
1416 return FALSE;
1419 BOOL macho_fetch_file_info(const WCHAR* name, DWORD* base,
1420 DWORD* size, DWORD* checksum)
1422 return FALSE;
1425 BOOL macho_read_wine_loader_dbg_info(struct process* pcs)
1427 return FALSE;
1430 BOOL macho_enum_modules(HANDLE hProc, enum_modules_cb cb, void* user)
1432 return FALSE;
1435 struct module* macho_load_module(struct process* pcs, const WCHAR* name, unsigned long addr)
1437 return NULL;
1440 BOOL macho_load_debug_info(struct module* module, struct macho_file_map* fmap)
1442 return FALSE;
1444 #endif /* __MACH__ */