diff options
author | Ben Gamari <bgamari.foss@gmail.com> | 2016-12-07 08:51:02 -0500 |
---|---|---|
committer | Ben Gamari <ben@smart-cactus.org> | 2016-12-07 09:55:30 -0500 |
commit | 83d69dca896c7df1f2a36268d5b45c9283985ebf (patch) | |
tree | 5803804671418d397504a62756bd2601731bbc0f | |
parent | d70d452a38bed3321bfc3c14074a6b3e1f30a090 (diff) | |
download | haskell-83d69dca896c7df1f2a36268d5b45c9283985ebf.tar.gz |
Don't barf() on failures in loadArchive()
This patch replaces calls to barf() in loadArchive() with proper
error handling.
Test Plan: GHC CI
Reviewers: rwbarton, erikd, hvr, austin, simonmar, bgamari
Reviewed By: bgamari
Subscribers: thomie
Tags: #ghc
Differential Revision: https://phabricator.haskell.org/D2652
GHC Trac Issues: #12388
-rw-r--r-- | includes/Rts.h | 6 | ||||
-rw-r--r-- | rts/LinkerInternals.h | 7 | ||||
-rw-r--r-- | rts/linker/LoadArchive.c | 471 |
3 files changed, 291 insertions, 193 deletions
diff --git a/includes/Rts.h b/includes/Rts.h index 0599df655c..fc010d4d97 100644 --- a/includes/Rts.h +++ b/includes/Rts.h @@ -267,13 +267,13 @@ TICK_VAR(2) Assertions and Debuggery -------------------------------------------------------------------------- */ -#define IF_RTSFLAGS(c,s) if (RtsFlags.c) { s; } +#define IF_RTSFLAGS(c,s) if (RtsFlags.c) { s; } doNothing() #ifdef DEBUG #if IN_STG_CODE -#define IF_DEBUG(c,s) if (RtsFlags[0].DebugFlags.c) { s; } +#define IF_DEBUG(c,s) if (RtsFlags[0].DebugFlags.c) { s; } doNothing() #else -#define IF_DEBUG(c,s) if (RtsFlags.DebugFlags.c) { s; } +#define IF_DEBUG(c,s) if (RtsFlags.DebugFlags.c) { s; } doNothing() #endif #else #define IF_DEBUG(c,s) doNothing() diff --git a/rts/LinkerInternals.h b/rts/LinkerInternals.h index da20e3b278..a09d079758 100644 --- a/rts/LinkerInternals.h +++ b/rts/LinkerInternals.h @@ -259,8 +259,6 @@ extern /*Str*/HashTable *symhash; #define USE_CONTIGUOUS_MMAP 0 #endif -#include "EndPrivate.h" - HsInt isAlreadyLoaded( pathchar *path ); HsInt loadOc( ObjectCode* oc ); ObjectCode* mkOc( pathchar *path, char *image, int imageSize, @@ -268,10 +266,6 @@ ObjectCode* mkOc( pathchar *path, char *image, int imageSize, int misalignment ); -#ifdef darwin_HOST_OS -int machoGetMisalignment( FILE * f ); -#endif /* darwin_HOST_OS */ - #if defined (mingw32_HOST_OS) /* We use myindex to calculate array addresses, rather than simply doing the normal subscript thing. That's because @@ -313,4 +307,5 @@ char *cstring_from_section_name( #define MAP_ANONYMOUS 0 #endif +#include "EndPrivate.h" #endif /* LINKERINTERNALS_H */ diff --git a/rts/linker/LoadArchive.c b/rts/linker/LoadArchive.c index f07eff7955..99d8fe4384 100644 --- a/rts/linker/LoadArchive.c +++ b/rts/linker/LoadArchive.c @@ -2,11 +2,11 @@ #include <stddef.h> #include <Rts.h> +#include "PathUtils.h" #include "sm/Storage.h" #include "sm/OSMem.h" #include "RtsUtils.h" -#include "PathUtils.h" #include "LinkerInternals.h" #include "linker/M32Alloc.h" @@ -22,22 +22,22 @@ #include <ctype.h> -static HsInt loadArchive_ (pathchar *path) -{ - ObjectCode* oc; - char *image; - int memberSize; - FILE *f; - int n; - size_t thisFileNameSize; - char *fileName; - size_t fileNameSize; - int isObject, isGnuIndex, isThin, isImportLib; - char tmp[20]; - char *gnuFileIndex; - int gnuFileIndexSize; +#define FAIL(...) do {\ + errorBelch("loadArchive: "__VA_ARGS__); \ + goto fail;\ +} while (0) + +#define DEBUG_LOG(...) IF_DEBUG(linker, debugBelch("loadArchive: " __VA_ARGS__)) + #if defined(darwin_HOST_OS) - int i; +/* Read 4 bytes and convert to host byte order */ +static uint32_t read4Bytes(const char buf[static 4]) +{ + return ntohl(*(uint32_t*)buf); +} + +static StgBool loadFatArchive(char tmp[static 20], FILE* f, pathchar* path) +{ uint32_t nfat_arch, nfat_offset, cputype, cpusubtype; #if defined(i386_HOST_ARCH) const uint32_t mycputype = CPU_TYPE_X86; @@ -54,15 +54,206 @@ static HsInt loadArchive_ (pathchar *path) #else #error Unknown Darwin architecture #endif + + nfat_arch = read4Bytes(tmp + 4); + DEBUG_LOG("found a fat archive containing %d architectures\n", nfat_arch); + nfat_offset = 0; + for (uint32_t i = 0; i < nfat_arch; i++) { + /* search for the right arch */ + int n = fread(tmp, 1, 12, f); + if (n != 12) { + errorBelch("Failed reading arch from `%s'", path); + return false; + } + cputype = read4Bytes(tmp); + cpusubtype = read4Bytes(tmp + 4); + if (cputype == mycputype && cpusubtype == mycpusubtype) { + DEBUG_LOG("found my archive in a fat archive\n"); + nfat_offset = read4Bytes(tmp + 8); + break; + } + } + if (nfat_offset == 0) { + errorBelch("Fat archive contains %d architectures, " + "but none of them are compatible with the host", + (int)nfat_arch); + return false; + } else { + /* Seek to the correct architecture */ + int n = fseek(f, nfat_offset, SEEK_SET); + if (n != 0) { + errorBelch("Failed to seek to arch in `%s'", path); + return false; + } + + /* Read the header */ + n = fread(tmp, 1, 8, f); + if (n != 8) { + errorBelch("Failed reading header from `%s'", path); + return false; + } + + /* Check the magic number */ + if (strncmp(tmp, "!<arch>\n", 8) != 0) { + errorBelch("couldn't find archive in `%s' at offset %d", path, + nfat_offset); + return false; + } + } + return true; +} +#endif + +static StgBool readThinArchiveMember(int n, int memberSize, pathchar* path, + char* fileName, char* image) +{ + StgBool has_succeeded = false; + FILE* member = NULL; + pathchar *pathCopy, *dirName, *memberPath, *objFileName; + memberPath = NULL; + /* Allocate and setup the dirname of the archive. We'll need + this to locate the thin member */ + pathCopy = pathdup(path); // Convert the char* to a pathchar* + dirName = pathdir(pathCopy); + /* Append the relative member name to the dirname. This should be + be the full path to the actual thin member. */ + int memberLen = pathlen(dirName) + 1 + strlen(fileName) + 1; + memberPath = stgMallocBytes(pathsize * memberLen, "loadArchive(file)"); + objFileName = mkPath(fileName); + pathprintf(memberPath, memberLen, WSTR("%" PATH_FMT "%" PATH_FMT), dirName, + objFileName); + stgFree(objFileName); + stgFree(dirName); + member = pathopen(memberPath, WSTR("rb")); + if (!member) { + errorBelch("loadObj: can't read thin archive `%s'", memberPath); + goto inner_fail; + } + n = fread(image, 1, memberSize, member); + if (n != memberSize) { + errorBelch("loadArchive: error whilst reading `%s'", fileName); + goto inner_fail; + } + has_succeeded = true; + +inner_fail: + fclose(member); + stgFree(memberPath); + stgFree(pathCopy); + return has_succeeded; +} + +static StgBool checkFatArchive(char magic[static 20], FILE* f, pathchar* path) +{ + StgBool success; + success = false; +#ifdef darwin_HOST_OS + /* Not a standard archive, look for a fat archive magic number: */ + if (read4Bytes(magic) == FAT_MAGIC) + success = loadFatArchive(magic, f, path); + else + errorBelch("loadArchive: Neither an archive, nor a fat archive: `%s'", + path); +#else + (void)magic; + (void)f; + errorBelch("loadArchive: Not an archive: `%s'", path); #endif - int misalignment = 0; + return success; +} - /* TODO: don't call barf() on error, instead return an error code, freeing - * all resources correctly. This function is pretty complex, so it needs - * to be refactored to make this practical. */ +/** + * Look up the filename in the GNU-variant index file pointed to by + * gnuFileIndex. + * @param fileName_ a pointer to a pointer to the file name to be looked up. + * The file name must have been allocated with `StgMallocBytes`, and will + * be reallocated on return; the old value is now _invalid_. + * @param gnuFileIndexSize The size of the index. + */ +static StgBool +lookupGNUArchiveIndex(int gnuFileIndexSize, char **fileName_, + char* gnuFileIndex, pathchar* path, size_t* thisFileNameSize, + size_t* fileNameSize) +{ + int n; + char *fileName = *fileName_; + if (isdigit(fileName[1])) { + int i; + for (n = 2; isdigit(fileName[n]); n++) + ; + + fileName[n] = '\0'; + n = atoi(fileName + 1); + if (gnuFileIndex == NULL) { + errorBelch("loadArchive: GNU-variant filename " + "without an index while reading from `%s'", + path); + return false; + } + if (n < 0 || n > gnuFileIndexSize) { + errorBelch("loadArchive: GNU-variant filename " + "offset %d out of range [0..%d] " + "while reading filename from `%s'", + n, gnuFileIndexSize, path); + return false; + } + if (n != 0 && gnuFileIndex[n - 1] != '\n') { + errorBelch("loadArchive: GNU-variant filename offset " + "%d invalid (range [0..%d]) while reading " + "filename from `%s'", + n, gnuFileIndexSize, path); + return false; + } + for (i = n; gnuFileIndex[i] != '\n'; i++) + ; + + size_t FileNameSize = i - n - 1; + if (FileNameSize >= *fileNameSize) { + /* Double it to avoid potentially continually + increasing it by 1 */ + *fileNameSize = FileNameSize * 2; + *fileName_ = fileName = stgReallocBytes(fileName, *fileNameSize, + "loadArchive(fileName)"); + } + memcpy(fileName, gnuFileIndex + n, FileNameSize); + fileName[FileNameSize] = '\0'; + *thisFileNameSize = FileNameSize; + } + /* Skip 32-bit symbol table ("/" + 15 blank characters) + and 64-bit symbol table ("/SYM64/" + 9 blank characters) */ + else if (0 == strncmp(fileName + 1, " ", 15) || + 0 == strncmp(fileName + 1, "SYM64/ ", 15)) { + fileName[0] = '\0'; + *thisFileNameSize = 0; + } + else { + errorBelch("loadArchive: invalid GNU-variant filename `%.16s'" + " while reading filename from `%s'", fileName, path); + return false; + } - IF_DEBUG(linker, debugBelch("loadArchive: start\n")); - IF_DEBUG(linker, debugBelch("loadArchive: Loading archive `%" PATH_FMT" '\n", path)); + return true; +} + +static HsInt loadArchive_ (pathchar *path) +{ + ObjectCode* oc = NULL; + char *image = NULL; + HsInt retcode = 0; + int memberSize; + FILE *f = NULL; + int n; + size_t thisFileNameSize = (size_t)-1; /* shut up bogus GCC warning */ + char *fileName; + size_t fileNameSize; + int isObject, isGnuIndex, isThin, isImportLib; + char tmp[20]; + char *gnuFileIndex; + int gnuFileIndexSize; + int misalignment = 0; + + DEBUG_LOG("start\n"); + DEBUG_LOG("Loading archive `%" PATH_FMT" '\n", path); /* Check that we haven't already loaded this archive. Ignore requests to load multiple times */ @@ -83,14 +274,14 @@ static HsInt loadArchive_ (pathchar *path) f = pathopen(path, WSTR("rb")); if (!f) - barf("loadObj: can't read `%" PATH_FMT "'", path); + FAIL("loadObj: can't read `%" PATH_FMT "'", path); /* Check if this is an archive by looking for the magic "!<arch>\n" - * string. Usually, if this fails, we barf and quit. On Darwin however, - * we may have a fat archive, which contains archives for more than - * one architecture. Fat archives start with the magic number 0xcafebabe, - * always stored big endian. If we find a fat_header, we scan through - * the fat_arch structs, searching through for one for our host + * string. Usually, if this fails, we belch an error and return. On + * Darwin however, we may have a fat archive, which contains archives for + * more than one architecture. Fat archives start with the magic number + * 0xcafebabe, always stored big endian. If we find a fat_header, we scan + * through the fat_arch structs, searching through for one for our host * architecture. If a matching struct is found, we read the offset * of our archive data (nfat_offset) and seek forward nfat_offset bytes * from the start of the file. @@ -105,8 +296,9 @@ static HsInt loadArchive_ (pathchar *path) */ n = fread ( tmp, 1, 8, f ); - if (n != 8) - barf("loadArchive: Failed reading header from `%" PATH_FMT "'", path); + if (n != 8) { + FAIL("Failed reading header from `%" PATH_FMT "'", path); + } if (strncmp(tmp, "!<arch>\n", 8) == 0) {} /* Check if this is a thin archive by looking for the magic string "!<thin>\n" * @@ -126,108 +318,70 @@ static HsInt loadArchive_ (pathchar *path) else if (strncmp(tmp, "!<thin>\n", 8) == 0) { isThin = 1; } -#if defined(darwin_HOST_OS) - /* Not a standard archive, look for a fat archive magic number: */ - else if (ntohl(*(uint32_t *)tmp) == FAT_MAGIC) { - nfat_arch = ntohl(*(uint32_t *)(tmp + 4)); - IF_DEBUG(linker, debugBelch("loadArchive: found a fat archive containing %d architectures\n", nfat_arch)); - nfat_offset = 0; - - for (i = 0; i < (int)nfat_arch; i++) { - /* search for the right arch */ - n = fread( tmp, 1, 20, f ); - if (n != 8) - barf("loadArchive: Failed reading arch from `%s'", path); - cputype = ntohl(*(uint32_t *)tmp); - cpusubtype = ntohl(*(uint32_t *)(tmp + 4)); - - if (cputype == mycputype && cpusubtype == mycpusubtype) { - IF_DEBUG(linker, debugBelch("loadArchive: found my archive in a fat archive\n")); - nfat_offset = ntohl(*(uint32_t *)(tmp + 8)); - break; - } - } - - if (nfat_offset == 0) { - barf ("loadArchive: searched %d architectures, but no host arch found", (int)nfat_arch); - } - else { - n = fseek( f, nfat_offset, SEEK_SET ); - if (n != 0) - barf("loadArchive: Failed to seek to arch in `%s'", path); - n = fread ( tmp, 1, 8, f ); - if (n != 8) - barf("loadArchive: Failed reading header from `%s'", path); - if (strncmp(tmp, "!<arch>\n", 8) != 0) { - barf("loadArchive: couldn't find archive in `%s' at offset %d", path, nfat_offset); - } - } - } else { - barf("loadArchive: Neither an archive, nor a fat archive: `%s'", path); + StgBool success = checkFatArchive(tmp, f, path); + if (!success) + goto fail; } -#else - else { - barf("loadArchive: Not an archive: `%" PATH_FMT "'", path); - } -#endif - - IF_DEBUG(linker, debugBelch("loadArchive: loading archive contents\n")); + DEBUG_LOG("loading archive contents\n"); while (1) { - IF_DEBUG(linker, debugBelch("loadArchive: reading at %ld\n", ftell(f))); + DEBUG_LOG("reading at %ld\n", ftell(f)); n = fread ( fileName, 1, 16, f ); if (n != 16) { if (feof(f)) { - IF_DEBUG(linker, debugBelch("loadArchive: EOF while reading from '%" PATH_FMT "'\n", path)); + DEBUG_LOG("EOF while reading from '%" PATH_FMT "'\n", path); break; } else { - barf("loadArchive: Failed reading file name from `%" PATH_FMT "'", path); + FAIL("Failed reading file name from `%" PATH_FMT "'", path); } } #if defined(darwin_HOST_OS) if (strncmp(fileName, "!<arch>\n", 8) == 0) { - IF_DEBUG(linker, debugBelch("loadArchive: found the start of another archive, breaking\n")); + DEBUG_LOG("found the start of another archive, breaking\n"); break; } #endif n = fread ( tmp, 1, 12, f ); if (n != 12) - barf("loadArchive: Failed reading mod time from `%" PATH_FMT "'", path); + FAIL("Failed reading mod time from `%" PATH_FMT "'", path); n = fread ( tmp, 1, 6, f ); if (n != 6) - barf("loadArchive: Failed reading owner from `%" PATH_FMT "'", path); + FAIL("Failed reading owner from `%" PATH_FMT "'", path); n = fread ( tmp, 1, 6, f ); if (n != 6) - barf("loadArchive: Failed reading group from `%" PATH_FMT "'", path); + FAIL("Failed reading group from `%" PATH_FMT "'", path); n = fread ( tmp, 1, 8, f ); if (n != 8) - barf("loadArchive: Failed reading mode from `%" PATH_FMT "'", path); + FAIL("Failed reading mode from `%" PATH_FMT "'", path); n = fread ( tmp, 1, 10, f ); if (n != 10) - barf("loadArchive: Failed reading size from `%" PATH_FMT "'", path); + FAIL("Failed reading size from `%" PATH_FMT "'", path); tmp[10] = '\0'; for (n = 0; isdigit(tmp[n]); n++); tmp[n] = '\0'; memberSize = atoi(tmp); - IF_DEBUG(linker, debugBelch("loadArchive: size of this archive member is %d\n", memberSize)); + DEBUG_LOG("size of this archive member is %d\n", memberSize); n = fread ( tmp, 1, 2, f ); if (n != 2) - barf("loadArchive: Failed reading magic from `%" PATH_FMT "'", path); + FAIL("Failed reading magic from `%" PATH_FMT "'", path); if (strncmp(tmp, "\x60\x0A", 2) != 0) - barf("loadArchive: Failed reading magic from `%" PATH_FMT "' at %ld. Got %c%c", + FAIL("Failed reading magic from `%" PATH_FMT "' at %ld. Got %c%c", path, ftell(f), tmp[0], tmp[1]); isGnuIndex = 0; /* Check for BSD-variant large filenames */ if (0 == strncmp(fileName, "#1/", 3)) { + size_t n = 0; fileName[16] = '\0'; if (isdigit(fileName[3])) { - for (n = 4; isdigit(fileName[n]); n++); + for (n = 4; isdigit(fileName[n]); n++) + ; + fileName[n] = '\0'; thisFileNameSize = atoi(fileName + 3); memberSize -= thisFileNameSize; @@ -235,22 +389,23 @@ static HsInt loadArchive_ (pathchar *path) /* Double it to avoid potentially continually increasing it by 1 */ fileNameSize = thisFileNameSize * 2; - fileName = stgReallocBytes(fileName, fileNameSize, "loadArchive(fileName)"); + fileName = stgReallocBytes(fileName, fileNameSize, + "loadArchive(fileName)"); } - n = fread ( fileName, 1, thisFileNameSize, f ); - if (n != (int)thisFileNameSize) { - barf("loadArchive: Failed reading filename from `%" PATH_FMT "'", - path); + n = fread(fileName, 1, thisFileNameSize, f); + if (n != thisFileNameSize) { + errorBelch("Failed reading filename from `%s'", path); + goto fail; } fileName[thisFileNameSize] = 0; - /* On OS X at least, thisFileNameSize is the size of the fileName field, not the length of the fileName itself. */ thisFileNameSize = strlen(fileName); - } - else { - barf("loadArchive: BSD-variant filename size not found while reading filename from `%s'", path); + } else { + errorBelch("BSD-variant filename size not found " + "while reading filename from `%s'", path); + goto fail; } } /* Check for GNU file index file */ @@ -261,41 +416,9 @@ static HsInt loadArchive_ (pathchar *path) } /* Check for a file in the GNU file index */ else if (fileName[0] == '/') { - if (isdigit(fileName[1])) { - int i; - - for (n = 2; isdigit(fileName[n]); n++); - fileName[n] = '\0'; - n = atoi(fileName + 1); - - if (gnuFileIndex == NULL) { - barf("loadArchive: GNU-variant filename without an index while reading from `%s'", path); - } - if (n < 0 || n > gnuFileIndexSize) { - barf("loadArchive: GNU-variant filename offset %d out of range [0..%d] while reading filename from `%s'", n, gnuFileIndexSize, path); - } - if (n != 0 && gnuFileIndex[n - 1] != '\n') { - barf("loadArchive: GNU-variant filename offset %d invalid (range [0..%d]) while reading filename from `%s'", n, gnuFileIndexSize, path); - } - for (i = n; gnuFileIndex[i] != '\n'; i++); - thisFileNameSize = i - n - 1; - if (thisFileNameSize >= fileNameSize) { - /* Double it to avoid potentially continually - increasing it by 1 */ - fileNameSize = thisFileNameSize * 2; - fileName = stgReallocBytes(fileName, fileNameSize, "loadArchive(fileName)"); - } - memcpy(fileName, gnuFileIndex + n, thisFileNameSize); - fileName[thisFileNameSize] = '\0'; - } - /* Skip 32-bit symbol table ("/" + 15 blank characters) - and 64-bit symbol table ("/SYM64/" + 9 blank characters) */ - else if (fileName[1] == ' ' || (0 == strncmp(&fileName[1], "SYM64/", 6))) { - fileName[0] = '\0'; - thisFileNameSize = 0; - } - else { - barf("loadArchive: invalid GNU-variant filename `%.16s' while reading filename from `%s'", fileName, path); + if (!lookupGNUArchiveIndex(gnuFileIndexSize, &fileName, + gnuFileIndex, path, &thisFileNameSize, &fileNameSize)) { + goto fail; } } /* Finally, the case where the filename field actually contains @@ -328,8 +451,7 @@ static HsInt loadArchive_ (pathchar *path) } } - IF_DEBUG(linker, - debugBelch("loadArchive: Found member file `%s'\n", fileName)); + DEBUG_LOG("Found member file `%s'\n", fileName); isObject = (thisFileNameSize >= 2 && strncmp(fileName + thisFileNameSize - 2, ".o" , 2) == 0) || (thisFileNameSize >= 4 && strncmp(fileName + thisFileNameSize - 4, ".p_o", 4) == 0); @@ -358,13 +480,13 @@ static HsInt loadArchive_ (pathchar *path) isImportLib = isImportLib || endsWithPath(path, WSTR(".dll.a")); #endif // windows - IF_DEBUG(linker, debugBelch("loadArchive: \tthisFileNameSize = %d\n", (int)thisFileNameSize)); - IF_DEBUG(linker, debugBelch("loadArchive: \tisObject = %d\n", isObject)); + DEBUG_LOG("\tthisFileNameSize = %d\n", (int)thisFileNameSize); + DEBUG_LOG("\tisObject = %d\n", isObject); if (isObject) { char *archiveMemberName; - IF_DEBUG(linker, debugBelch("loadArchive: Member is an object file...loading...\n")); + DEBUG_LOG("Member is an object file...loading...\n"); #if defined(mingw32_HOST_OS) // TODO: We would like to use allocateExec here, but allocateExec @@ -386,41 +508,16 @@ static HsInt loadArchive_ (pathchar *path) image = stgMallocBytes(memberSize, "loadArchive(image)"); #endif if (isThin) { - FILE *member; - pathchar *pathCopy, *dirName, *memberPath, *objFileName; - - /* Allocate and setup the dirname of the archive. We'll need - this to locate the thin member */ - pathCopy = pathdup(path); // Convert the char* to a pathchar* - dirName = pathdir(pathCopy); - - /* Append the relative member name to the dirname. This should be - be the full path to the actual thin member. */ - int memberLen = pathlen(dirName) + 1 + strlen(fileName) + 1; - memberPath = stgMallocBytes(pathsize * memberLen, "loadArchive(file)"); - objFileName = mkPath(fileName); - pathprintf(memberPath, memberLen, WSTR("%" PATH_FMT "%" PATH_FMT), dirName, objFileName); - stgFree(objFileName); - stgFree(dirName); - - member = pathopen(memberPath, WSTR("rb")); - if (!member) - barf("loadObj: can't read thin archive `%" PATH_FMT "'", memberPath); - - n = fread ( image, 1, memberSize, member ); - if (n != memberSize) { - barf("loadArchive: error whilst reading `%s'", fileName); + if (!readThinArchiveMember(n, memberSize, path, + fileName, image)) { + goto fail; } - - fclose(member); - stgFree(memberPath); - stgFree(pathCopy); } else { n = fread ( image, 1, memberSize, f ); if (n != memberSize) { - barf("loadArchive: error whilst reading `%" PATH_FMT "'", path); + FAIL("error whilst reading `%" PATH_FMT "'", path); } } @@ -457,9 +554,10 @@ static HsInt loadArchive_ (pathchar *path) } else if (isGnuIndex) { if (gnuFileIndex != NULL) { - barf("loadArchive: GNU-variant index found, but already have an index, while reading filename from `%s'", path); + FAIL("GNU-variant index found, but already have an index, \ +while reading filename from `%s'", path); } - IF_DEBUG(linker, debugBelch("loadArchive: Found GNU-variant file index\n")); + DEBUG_LOG("Found GNU-variant file index\n"); #if RTS_LINKER_USE_MMAP gnuFileIndex = mmapForLinker(memberSize + 1, MAP_ANONYMOUS, -1, 0); #else @@ -467,7 +565,7 @@ static HsInt loadArchive_ (pathchar *path) #endif n = fread ( gnuFileIndex, 1, memberSize, f ); if (n != memberSize) { - barf("loadArchive: error whilst reading `%" PATH_FMT "'", path); + FAIL("error whilst reading `%" PATH_FMT "'", path); } gnuFileIndex[memberSize] = '/'; gnuFileIndexSize = memberSize; @@ -475,48 +573,53 @@ static HsInt loadArchive_ (pathchar *path) else if (isImportLib) { #if defined(OBJFORMAT_PEi386) if (checkAndLoadImportLibrary(path, fileName, f)) { - IF_DEBUG(linker, debugBelch("loadArchive: Member is an import file section... Corresponding DLL has been loaded...\n")); + DEBUG_LOG("Member is an import file section... " + "Corresponding DLL has been loaded...\n"); } else { - IF_DEBUG(linker, debugBelch("loadArchive: Member is not a valid import file section... Skipping...\n")); + DEBUG_LOG("Member is not a valid import file section... " + "Skipping...\n"); n = fseek(f, memberSize, SEEK_CUR); if (n != 0) - barf("loadArchive: error whilst seeking by %d in `%" PATH_FMT "'", + FAIL("error whilst seeking by %d in `%" PATH_FMT "'", memberSize, path); } #endif } else { - IF_DEBUG(linker, debugBelch("loadArchive: '%s' does not appear to be an object file\n", fileName)); + DEBUG_LOG("'%s' does not appear to be an object file\n", fileName); if (!isThin || thisFileNameSize == 0) { n = fseek(f, memberSize, SEEK_CUR); if (n != 0) - barf("loadArchive: error whilst seeking by %d in `%" PATH_FMT "'", + FAIL("error whilst seeking by %d in `%" PATH_FMT "'", memberSize, path); } } /* .ar files are 2-byte aligned */ if (!(isThin && thisFileNameSize > 0) && memberSize % 2) { - IF_DEBUG(linker, debugBelch("loadArchive: trying to read one pad byte\n")); + DEBUG_LOG("trying to read one pad byte\n"); n = fread ( tmp, 1, 1, f ); if (n != 1) { if (feof(f)) { - IF_DEBUG(linker, debugBelch("loadArchive: found EOF while reading one pad byte\n")); + DEBUG_LOG("found EOF while reading one pad byte\n"); break; } else { - barf("loadArchive: Failed reading padding from `%" PATH_FMT "'", path); + FAIL("Failed reading padding from `%" PATH_FMT "'", path); } } - IF_DEBUG(linker, debugBelch("loadArchive: successfully read one pad byte\n")); + DEBUG_LOG("successfully read one pad byte\n"); } - IF_DEBUG(linker, debugBelch("loadArchive: reached end of archive loading while loop\n")); + DEBUG_LOG("reached end of archive loading while loop\n"); } + retcode = 1; +fail: + if (f != NULL) + fclose(f); - fclose(f); - - stgFree(fileName); + if (fileName != NULL) + stgFree(fileName); if (gnuFileIndex != NULL) { #if RTS_LINKER_USE_MMAP munmap(gnuFileIndex, gnuFileIndexSize + 1); @@ -528,8 +631,8 @@ static HsInt loadArchive_ (pathchar *path) if (RTS_LINKER_USE_MMAP) m32_allocator_flush(); - IF_DEBUG(linker, debugBelch("loadArchive: done\n")); - return 1; + DEBUG_LOG("done\n"); + return retcode; } HsInt loadArchive (pathchar *path) |