diff options
| author | Dan Gohman <djg@cray.com> | 2007-07-18 16:29:46 +0000 | 
|---|---|---|
| committer | Dan Gohman <djg@cray.com> | 2007-07-18 16:29:46 +0000 | 
| commit | f17a25c88b892d30c2b41ba7ecdfbdfb2b4be9cc (patch) | |
| tree | ebb79ea1ee5e3bc1fdf38541a811a8b804f0679a /lib/System | |
| download | external_llvm-f17a25c88b892d30c2b41ba7ecdfbdfb2b4be9cc.zip external_llvm-f17a25c88b892d30c2b41ba7ecdfbdfb2b4be9cc.tar.gz external_llvm-f17a25c88b892d30c2b41ba7ecdfbdfb2b4be9cc.tar.bz2 | |
It's not necessary to do rounding for alloca operations when the requested
alignment is equal to the stack alignment.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@40004 91177308-0d34-0410-b5e6-96231b3b80d8
Diffstat (limited to 'lib/System')
40 files changed, 9867 insertions, 0 deletions
| diff --git a/lib/System/Alarm.cpp b/lib/System/Alarm.cpp new file mode 100644 index 0000000..e0b7f72 --- /dev/null +++ b/lib/System/Alarm.cpp @@ -0,0 +1,35 @@ +//===- Alarm.cpp - Alarm Generation Support ---------------------*- C++ -*-===// +// +//                     The LLVM Compiler Infrastructure +// +// This file was developed by the Reid Spencer and is distributed under the  +// University of Illinois Open Source License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +// This file implements the Alarm functionality  +// +//===----------------------------------------------------------------------===// + +#include "llvm/System/Alarm.h" +#include "llvm/Config/config.h" + +namespace llvm { +using namespace sys; + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only TRULY operating system +//===          independent code. +//===----------------------------------------------------------------------===// + +} + +// Include the platform-specific parts of this class. +#ifdef LLVM_ON_UNIX +#include "Unix/Alarm.inc" +#endif +#ifdef LLVM_ON_WIN32 +#include "Win32/Alarm.inc" +#endif + +DEFINING_FILE_FOR(SystemAlarm) diff --git a/lib/System/Disassembler.cpp b/lib/System/Disassembler.cpp new file mode 100644 index 0000000..4aec698 --- /dev/null +++ b/lib/System/Disassembler.cpp @@ -0,0 +1,75 @@ +//===- lib/System/Disassembler.cpp ------------------------------*- C++ -*-===// +// +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Anton Korobeynikov and is distributed under the +// University of Illinois Open Source License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +// This file implements the necessary glue to call external disassembler +// libraries. +// +//===----------------------------------------------------------------------===// + +#include "llvm/Config/config.h" +#include "llvm/System/Disassembler.h" + +#include <cassert> +#include <iomanip> +#include <string> +#include <sstream> + +#if USE_UDIS86 +#include <udis86.h> +#endif + +using namespace llvm; + +bool llvm::sys::hasDisassembler(void)  +{ +#if defined (__i386__) || defined (__amd64__) || defined (__x86_64__) +  // We have option to enable udis86 library. +  return true; +#else +  return false; +#endif +} + +std::string llvm::sys::disassembleBuffer(uint8_t* start, size_t length, +                                         uint64_t pc) { +  std::stringstream res; + +#if defined (__i386__) || defined (__amd64__) || defined (__x86_64__) +  unsigned bits; +# if defined(__i386__) +  bits = 32; +# else +  bits = 64; +# endif +   +# if USE_UDIS86 +  ud_t ud_obj; +    +  ud_init(&ud_obj); +  ud_set_input_buffer(&ud_obj, start, length); +  ud_set_mode(&ud_obj, bits); +  ud_set_pc(&ud_obj, pc); +  ud_set_syntax(&ud_obj, UD_SYN_ATT); +   +  res << std::setbase(16) +      << std::setw(bits/4); +   +  while (ud_disassemble(&ud_obj)) { +    res << ud_insn_off(&ud_obj) << ":\t" << ud_insn_asm(&ud_obj) << "\n"; +  } +# else +  res << "No disassembler available. See configure help for options.\n"; +# endif +   +#else +  res << "No disassembler available. See configure help for options.\n"; +#endif + +  return res.str(); +} diff --git a/lib/System/DynamicLibrary.cpp b/lib/System/DynamicLibrary.cpp new file mode 100644 index 0000000..8119348 --- /dev/null +++ b/lib/System/DynamicLibrary.cpp @@ -0,0 +1,209 @@ +//===-- DynamicLibrary.cpp - Runtime link/load libraries --------*- C++ -*-===// +// +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Reid Spencer and is distributed under the +// University of Illinois Open Source License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +//  This header file implements the operating system DynamicLibrary concept. +// +//===----------------------------------------------------------------------===// + +#include "llvm/System/DynamicLibrary.h" +#include "llvm/Config/config.h" +#include <map> + +// Collection of symbol name/value pairs to be searched prior to any libraries. +static std::map<std::string, void *> g_symbols; + +void llvm::sys::DynamicLibrary::AddSymbol(const char* symbolName, +                                          void *symbolValue) { +  g_symbols[symbolName] = symbolValue; +} + +// It is not possible to use ltdl.c on VC++ builds as the terms of its LGPL +// license and special exception would cause all of LLVM to be placed under +// the LGPL.  This is because the exception applies only when libtool is +// used, and obviously libtool is not used with Visual Studio.  An entirely +// separate implementation is provided in win32/DynamicLibrary.cpp. + +#ifdef LLVM_ON_WIN32 + +#include "Win32/DynamicLibrary.inc" + +#else + +#include "ltdl.h" +#include <cassert> +using namespace llvm; +using namespace llvm::sys; + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only TRULY operating system +//===          independent code. +//===----------------------------------------------------------------------===// + +static inline void check_ltdl_initialization() { +  static bool did_initialize_ltdl = false; +  if (!did_initialize_ltdl) { +    int Err = lt_dlinit(); +    Err = Err; // Silence warning. +    assert(0 == Err && "Can't init the ltdl library"); +    did_initialize_ltdl = true; +  } +} + +static std::vector<lt_dlhandle> OpenedHandles; + +DynamicLibrary::DynamicLibrary() : handle(0) { +  check_ltdl_initialization(); + +  lt_dlhandle a_handle = lt_dlopen(0); + +  assert(a_handle == 0 || "Can't open program as dynamic library"); + +  handle = a_handle; +  OpenedHandles.push_back(a_handle); +} + +/* +DynamicLibrary::DynamicLibrary(const char*filename) : handle(0) { +  check_ltdl_initialization(); + +  lt_dlhandle a_handle = lt_dlopen(filename); + +  if (a_handle == 0) +    a_handle = lt_dlopenext(filename); + +  if (a_handle == 0) +    throw std::string("Can't open :") + filename + ": " + lt_dlerror(); + +  handle = a_handle; +  OpenedHandles.push_back(a_handle); +} +*/ + +DynamicLibrary::~DynamicLibrary() { +  lt_dlhandle a_handle = (lt_dlhandle) handle; +  if (a_handle) { +    lt_dlclose(a_handle); + +    for (std::vector<lt_dlhandle>::iterator I = OpenedHandles.begin(), +         E = OpenedHandles.end(); I != E; ++I) { +      if (*I == a_handle) { +        // Note: don't use the swap/pop_back trick here. Order is important. +        OpenedHandles.erase(I); +        return; +      } +    } +  } +} + +bool DynamicLibrary::LoadLibraryPermanently(const char *Filename, +                                            std::string *ErrMsg) { +  check_ltdl_initialization(); +  lt_dlhandle a_handle = lt_dlopen(Filename); + +  if (a_handle == 0) +    a_handle = lt_dlopenext(Filename); + +  if (a_handle == 0) { +    if (ErrMsg) +      *ErrMsg = std::string("Can't open :") + +          (Filename ? Filename : "<current process>") + ": " + lt_dlerror(); +    return true; +  } + +  lt_dlmakeresident(a_handle); + +  OpenedHandles.push_back(a_handle); +  return false; +} + +void* DynamicLibrary::SearchForAddressOfSymbol(const char* symbolName) { +  check_ltdl_initialization(); + +  // First check symbols added via AddSymbol(). +  std::map<std::string, void *>::iterator I = g_symbols.find(symbolName); +  if (I != g_symbols.end()) +    return I->second; + +  // Now search the libraries. +  for (std::vector<lt_dlhandle>::iterator I = OpenedHandles.begin(), +       E = OpenedHandles.end(); I != E; ++I) { +    lt_ptr ptr = lt_dlsym(*I, symbolName); +    if (ptr) +      return ptr; +  } + +  // If this is darwin, it has some funky issues, try to solve them here.  Some +  // important symbols are marked 'private external' which doesn't allow +  // SearchForAddressOfSymbol to find them.  As such, we special case them here, +  // there is only a small handful of them. + +#ifdef __APPLE__ +#define EXPLICIT_SYMBOL(SYM) \ +   extern void *SYM; if (!strcmp(symbolName, #SYM)) return &SYM +  { +    EXPLICIT_SYMBOL(__ashldi3); +    EXPLICIT_SYMBOL(__ashrdi3); +    EXPLICIT_SYMBOL(__cmpdi2); +    EXPLICIT_SYMBOL(__divdi3); +    EXPLICIT_SYMBOL(__eprintf); +    EXPLICIT_SYMBOL(__fixdfdi); +    EXPLICIT_SYMBOL(__fixsfdi); +    EXPLICIT_SYMBOL(__fixunsdfdi); +    EXPLICIT_SYMBOL(__fixunssfdi); +    EXPLICIT_SYMBOL(__floatdidf); +    EXPLICIT_SYMBOL(__floatdisf); +    EXPLICIT_SYMBOL(__lshrdi3); +    EXPLICIT_SYMBOL(__moddi3); +    EXPLICIT_SYMBOL(__udivdi3); +    EXPLICIT_SYMBOL(__umoddi3); +  } +#undef EXPLICIT_SYMBOL +#endif + +// This macro returns the address of a well-known, explicit symbol +#define EXPLICIT_SYMBOL(SYM) \ +   if (!strcmp(symbolName, #SYM)) return &SYM + +// On linux we have a weird situation. The stderr/out/in symbols are both +// macros and global variables because of standards requirements. So, we  +// boldly use the EXPLICIT_SYMBOL macro without checking for a #define first. +#if defined(__linux__) +  { +    EXPLICIT_SYMBOL(stderr); +    EXPLICIT_SYMBOL(stdout); +    EXPLICIT_SYMBOL(stdin); +  } +#else +  // For everything else, we want to check to make sure the symbol isn't defined +  // as a macro before using EXPLICIT_SYMBOL. +  { +#ifndef stdin +    EXPLICIT_SYMBOL(stdin); +#endif +#ifndef stdout +    EXPLICIT_SYMBOL(stdout); +#endif +#ifndef stderr +    EXPLICIT_SYMBOL(stderr); +#endif +  } +#endif +#undef EXPLICIT_SYMBOL + +  return 0; +} + +void *DynamicLibrary::GetAddressOfSymbol(const char *symbolName) { +  assert(handle != 0 && "Invalid DynamicLibrary handle"); +  return lt_dlsym((lt_dlhandle) handle, symbolName); +} + +#endif // LLVM_ON_WIN32 + +DEFINING_FILE_FOR(SystemDynamicLibrary) diff --git a/lib/System/IncludeFile.cpp b/lib/System/IncludeFile.cpp new file mode 100644 index 0000000..2ba9595 --- /dev/null +++ b/lib/System/IncludeFile.cpp @@ -0,0 +1,20 @@ +//===- lib/System/IncludeFile.cpp - Ensure Linking Of Implementation -----===// +// +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Reid Spencer and is distributed under the  +// University of Illinois Open Source License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +// This file implements the IncludeFile constructor. +// +//===----------------------------------------------------------------------===// + +#include "llvm/System/IncludeFile.h" + +using namespace llvm; + +// This constructor is used to ensure linking of other modules. See the +// llvm/System/IncludeFile.h header for details.  +IncludeFile::IncludeFile(void*) {} diff --git a/lib/System/LICENSE.TXT b/lib/System/LICENSE.TXT new file mode 100644 index 0000000..f569da2 --- /dev/null +++ b/lib/System/LICENSE.TXT @@ -0,0 +1,6 @@ +LLVM System Interface Library +------------------------------------------------------------------------------- +The LLVM System Interface Library is licensed under the Illinois Open Source  +License and has the following additional copyright: + +Copyright (C) 2004 eXtensible Systems, Inc. diff --git a/lib/System/Makefile b/lib/System/Makefile new file mode 100644 index 0000000..85de736 --- /dev/null +++ b/lib/System/Makefile @@ -0,0 +1,19 @@ +##===- lib/System/Makefile ---------------------------------*- Makefile -*-===## +#  +#                     The LLVM Compiler Infrastructure +# +# This file was developed by Reid Spencer and is distributed under the  +# University of Illinois Open Source License. See LICENSE.TXT for details. +#  +##===----------------------------------------------------------------------===## + +LEVEL = ../.. +LIBRARYNAME = LLVMSystem +BUILD_ARCHIVE = 1 + +EXTRA_DIST = Unix Win32 README.txt + +include $(LEVEL)/Makefile.common + +CompileCommonOpts := $(filter-out -pedantic,$(CompileCommonOpts)) +CompileCommonOpts := $(filter-out -Wno-long-long,$(CompileCommonOpts)) diff --git a/lib/System/MappedFile.cpp b/lib/System/MappedFile.cpp new file mode 100644 index 0000000..41b5946 --- /dev/null +++ b/lib/System/MappedFile.cpp @@ -0,0 +1,35 @@ +//===- MappedFile.cpp - MappedFile Support ----------------------*- C++ -*-===// +// +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Reid Spencer and is distributed under the +// University of Illinois Open Source License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +// This file implements the mapped file concept. +// +//===----------------------------------------------------------------------===// + +#include "llvm/System/MappedFile.h" +#include "llvm/Config/config.h" + +namespace llvm { +using namespace sys; + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only TRULY operating system +//===          independent code. +//===----------------------------------------------------------------------===// + +} + +// Include the platform-specific parts of this class. +#ifdef LLVM_ON_UNIX +#include "Unix/MappedFile.inc" +#endif +#ifdef LLVM_ON_WIN32 +#include "Win32/MappedFile.inc" +#endif + +DEFINING_FILE_FOR(SystemMappedFile) diff --git a/lib/System/Memory.cpp b/lib/System/Memory.cpp new file mode 100644 index 0000000..3788abe --- /dev/null +++ b/lib/System/Memory.cpp @@ -0,0 +1,37 @@ +//===- Memory.cpp - Memory Handling Support ---------------------*- C++ -*-===// +// +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Reid Spencer and is distributed under the +// University of Illinois Open Source License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +// This file defines some helpful functions for allocating memory and dealing +// with memory mapped files +// +//===----------------------------------------------------------------------===// + +#include "llvm/System/Memory.h" +#include "llvm/Config/config.h" +#include "llvm/System/IncludeFile.h" + +namespace llvm { +using namespace sys; + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only TRULY operating system +//===          independent code. +//===----------------------------------------------------------------------===// + +} + +// Include the platform-specific parts of this class. +#ifdef LLVM_ON_UNIX +#include "Unix/Memory.inc" +#endif +#ifdef LLVM_ON_WIN32 +#include "Win32/Memory.inc" +#endif + +DEFINING_FILE_FOR(SystemMemory) diff --git a/lib/System/Mutex.cpp b/lib/System/Mutex.cpp new file mode 100644 index 0000000..a0fd417 --- /dev/null +++ b/lib/System/Mutex.cpp @@ -0,0 +1,162 @@ +//===- Mutex.cpp - Mutual Exclusion Lock ------------------------*- C++ -*-===// +// +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Reid Spencer and is distributed under the +// University of Illinois Open Source License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +// This file implements the llvm::sys::Mutex class. +// +//===----------------------------------------------------------------------===// + +#include "llvm/Config/config.h" +#include "llvm/System/Mutex.h" +#include "llvm/System/IncludeFile.h" + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only TRULY operating system +//===          independent code. +//===----------------------------------------------------------------------===// + +#if !defined(ENABLE_THREADS) || ENABLE_THREADS == 0 +// Define all methods as no-ops if threading is explicitly disabled +namespace llvm { +using namespace sys; +Mutex::Mutex( bool recursive) { } +Mutex::~Mutex() { } +bool Mutex::acquire() { return true; } +bool Mutex::release() { return true; } +bool Mutex::tryacquire() { return true; } +} +#else + +#if defined(HAVE_PTHREAD_H) && defined(HAVE_PTHREAD_MUTEX_LOCK) + +#include <cassert> +#include <pthread.h> +#include <stdlib.h> + +namespace llvm { +using namespace sys; + + +// This variable is useful for situations where the pthread library has been +// compiled with weak linkage for its interface symbols. This allows the +// threading support to be turned off by simply not linking against -lpthread. +// In that situation, the value of pthread_mutex_init will be 0 and +// consequently pthread_enabled will be false. In such situations, all the +// pthread operations become no-ops and the functions all return false. If +// pthread_mutex_init does have an address, then mutex support is enabled. +// Note: all LLVM tools will link against -lpthread if its available since it +//       is configured into the LIBS variable. +// Note: this line of code generates a warning if pthread_mutex_init is not +//       declared with weak linkage. It's safe to ignore the warning. +static const bool pthread_enabled = true; + +// Construct a Mutex using pthread calls +Mutex::Mutex( bool recursive) +  : data_(0) +{ +  if (pthread_enabled) +  { +    // Declare the pthread_mutex data structures +    pthread_mutex_t* mutex = +      static_cast<pthread_mutex_t*>(malloc(sizeof(pthread_mutex_t))); +    pthread_mutexattr_t attr; + +    // Initialize the mutex attributes +    int errorcode = pthread_mutexattr_init(&attr); +    assert(errorcode == 0); + +    // Initialize the mutex as a recursive mutex, if requested, or normal +    // otherwise. +    int kind = ( recursive  ? PTHREAD_MUTEX_RECURSIVE : PTHREAD_MUTEX_NORMAL ); +    errorcode = pthread_mutexattr_settype(&attr, kind); +    assert(errorcode == 0); + +#if !defined(__FreeBSD__) && !defined(__OpenBSD__) && !defined(__NetBSD__) +    // Make it a process local mutex +    errorcode = pthread_mutexattr_setpshared(&attr, PTHREAD_PROCESS_PRIVATE); +#endif + +    // Initialize the mutex +    errorcode = pthread_mutex_init(mutex, &attr); +    assert(errorcode == 0); + +    // Destroy the attributes +    errorcode = pthread_mutexattr_destroy(&attr); +    assert(errorcode == 0); + +    // Assign the data member +    data_ = mutex; +  } +} + +// Destruct a Mutex +Mutex::~Mutex() +{ +  if (pthread_enabled) +  { +    pthread_mutex_t* mutex = reinterpret_cast<pthread_mutex_t*>(data_); +    assert(mutex != 0); +    pthread_mutex_destroy(mutex); +    assert(mutex != 0); +  } +} + +bool +Mutex::acquire() +{ +  if (pthread_enabled) +  { +    pthread_mutex_t* mutex = reinterpret_cast<pthread_mutex_t*>(data_); +    assert(mutex != 0); + +    int errorcode = pthread_mutex_lock(mutex); +    return errorcode == 0; +  } +  return false; +} + +bool +Mutex::release() +{ +  if (pthread_enabled) +  { +    pthread_mutex_t* mutex = reinterpret_cast<pthread_mutex_t*>(data_); +    assert(mutex != 0); + +    int errorcode = pthread_mutex_unlock(mutex); +    return errorcode == 0; +  } +  return false; +} + +bool +Mutex::tryacquire() +{ +  if (pthread_enabled) +  { +    pthread_mutex_t* mutex = reinterpret_cast<pthread_mutex_t*>(data_); +    assert(mutex != 0); + +    int errorcode = pthread_mutex_trylock(mutex); +    return errorcode == 0; +  } +  return false; +} + +} + +#elif defined(LLVM_ON_UNIX) +#include "Unix/Mutex.inc" +#elif defined( LLVM_ON_WIN32) +#include "Win32/Mutex.inc" +#else +#warning Neither LLVM_ON_UNIX nor LLVM_ON_WIN32 was set in System/Mutex.cpp +#endif +#endif + +DEFINING_FILE_FOR(SystemMutex) diff --git a/lib/System/Path.cpp b/lib/System/Path.cpp new file mode 100644 index 0000000..caf5789 --- /dev/null +++ b/lib/System/Path.cpp @@ -0,0 +1,184 @@ +//===-- Path.cpp - Implement OS Path Concept --------------------*- C++ -*-===// +// +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Reid Spencer and is distributed under the +// University of Illinois Open Source License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +//  This header file implements the operating system Path concept. +// +//===----------------------------------------------------------------------===// + +#include "llvm/System/Path.h" +#include "llvm/Config/config.h" +#include <cassert> +#include <ostream> +using namespace llvm; +using namespace sys; + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only TRULY operating system +//===          independent code. +//===----------------------------------------------------------------------===// + +std::ostream& llvm::operator<<(std::ostream &strm, const sys::Path &aPath) { +  strm << aPath.toString(); +  return strm; +} + +Path +Path::GetLLVMConfigDir() { +  Path result; +#ifdef LLVM_ETCDIR +  if (result.set(LLVM_ETCDIR)) +    return result; +#endif +  return GetLLVMDefaultConfigDir(); +} + +LLVMFileType +sys::IdentifyFileType(const char*magic, unsigned length) { +  assert(magic && "Invalid magic number string"); +  assert(length >=4 && "Invalid magic number length"); +  switch (magic[0]) { +    case 'B': +      if (magic[1] == 'C' && magic[2] == (char)0xC0 && magic[3] == (char)0xDE) +        return Bitcode_FileType; +      break; +    case '!': +      if (length >= 8) +        if (memcmp(magic,"!<arch>\n",8) == 0) +          return Archive_FileType; +      break; +       +    case '\177': +      if (magic[1] == 'E' && magic[2] == 'L' && magic[3] == 'F') { +        if (length >= 18 && magic[17] == 0) +          switch (magic[16]) { +            default: break; +            case 1: return ELF_Relocatable_FileType; +            case 2: return ELF_Executable_FileType; +            case 3: return ELF_SharedObject_FileType; +            case 4: return ELF_Core_FileType; +          } +      } +      break; + +    case 0xCA: +      // This is complicated by an overlap with Java class files.  +      // See the Mach-O section in /usr/share/file/magic for details. +      if (magic[1] == char(0xFE) && magic[2] == char(0xBA) &&  +          magic[3] == char(0xBE)) { +        return Mach_O_DynamicallyLinkedSharedLib_FileType; +         +        // FIXME: How does this work? +        if (length >= 14 && magic[13] == 0) +          switch (magic[12]) { +            default: break; +            case 1: return Mach_O_Object_FileType; +            case 2: return Mach_O_Executable_FileType; +            case 3: return Mach_O_FixedVirtualMemorySharedLib_FileType; +            case 4: return Mach_O_Core_FileType; +            case 5: return Mach_O_PreloadExectuable_FileType; +            case 6: return Mach_O_DynamicallyLinkedSharedLib_FileType; +            case 7: return Mach_O_DynamicLinker_FileType; +            case 8: return Mach_O_Bundle_FileType; +            case 9: return Mach_O_DynamicallyLinkedSharedLibStub_FileType; +          } +      } +      break; + +    case 0xF0: // PowerPC Windows +    case 0x83: // Alpha 32-bit +    case 0x84: // Alpha 64-bit +    case 0x66: // MPS R4000 Windows +    case 0x50: // mc68K +    case 0x4c: // 80386 Windows +      if (magic[1] == 0x01) +        return COFF_FileType; + +    case 0x90: // PA-RISC Windows +    case 0x68: // mc68K Windows +      if (magic[1] == 0x02) +        return COFF_FileType; +      break; + +    default: +      break; +  } +  return Unknown_FileType; +} + +bool +Path::isArchive() const { +  if (canRead()) +    return hasMagicNumber("!<arch>\012"); +  return false; +} + +bool +Path::isDynamicLibrary() const { +  if (canRead()) { +    std::string Magic; +    if (getMagicNumber(Magic, 64)) +      switch (IdentifyFileType(Magic.c_str(), Magic.length())) { +        default: return false; +        case Mach_O_FixedVirtualMemorySharedLib_FileType: +        case Mach_O_DynamicallyLinkedSharedLib_FileType: +        case Mach_O_DynamicallyLinkedSharedLibStub_FileType: +        case ELF_SharedObject_FileType: +        case COFF_FileType:  return true; +      } +  } +  return false; +} + +Path +Path::FindLibrary(std::string& name) { +  std::vector<sys::Path> LibPaths; +  GetSystemLibraryPaths(LibPaths); +  for (unsigned i = 0; i < LibPaths.size(); ++i) { +    sys::Path FullPath(LibPaths[i]); +    FullPath.appendComponent("lib" + name + LTDL_SHLIB_EXT); +    if (FullPath.isDynamicLibrary()) +      return FullPath; +    FullPath.eraseSuffix(); +    FullPath.appendSuffix("a"); +    if (FullPath.isArchive()) +      return FullPath; +  } +  return sys::Path(); +} + +std::string Path::GetDLLSuffix() { +  return LTDL_SHLIB_EXT; +} + +bool +Path::isBitcodeFile() const { +  std::string actualMagic; +  if (!getMagicNumber(actualMagic, 4)) +    return false; +  return actualMagic == "BC\xC0\xDE"; +} + +bool Path::hasMagicNumber(const std::string &Magic) const { +  std::string actualMagic; +  if (getMagicNumber(actualMagic, Magic.size())) +    return Magic == actualMagic; +  return false; +} + + + +// Include the truly platform-specific parts of this class. +#if defined(LLVM_ON_UNIX) +#include "Unix/Path.inc" +#endif +#if defined(LLVM_ON_WIN32) +#include "Win32/Path.inc" +#endif + +DEFINING_FILE_FOR(SystemPath) diff --git a/lib/System/Process.cpp b/lib/System/Process.cpp new file mode 100644 index 0000000..d69f78e --- /dev/null +++ b/lib/System/Process.cpp @@ -0,0 +1,35 @@ +//===-- Process.cpp - Implement OS Process Concept --------------*- C++ -*-===// +// +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Reid Spencer and is distributed under the +// University of Illinois Open Source License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +//  This header file implements the operating system Process concept. +// +//===----------------------------------------------------------------------===// + +#include "llvm/System/Process.h" +#include "llvm/Config/config.h" + +namespace llvm { +using namespace sys; + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only TRULY operating system +//===          independent code. +//===----------------------------------------------------------------------===// + +} + +// Include the platform-specific parts of this class. +#ifdef LLVM_ON_UNIX +#include "Unix/Process.inc" +#endif +#ifdef LLVM_ON_WIN32 +#include "Win32/Process.inc" +#endif + +DEFINING_FILE_FOR(SystemProcess) diff --git a/lib/System/Program.cpp b/lib/System/Program.cpp new file mode 100644 index 0000000..e91a3d5 --- /dev/null +++ b/lib/System/Program.cpp @@ -0,0 +1,35 @@ +//===-- Program.cpp - Implement OS Program Concept --------------*- C++ -*-===// +// +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Reid Spencer and is distributed under the +// University of Illinois Open Source License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +//  This header file implements the operating system Program concept. +// +//===----------------------------------------------------------------------===// + +#include "llvm/System/Program.h" +#include "llvm/Config/config.h" + +namespace llvm { +using namespace sys; + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only TRULY operating system +//===          independent code. +//===----------------------------------------------------------------------===// + +} + +// Include the platform-specific parts of this class. +#ifdef LLVM_ON_UNIX +#include "Unix/Program.inc" +#endif +#ifdef LLVM_ON_WIN32 +#include "Win32/Program.inc" +#endif + +DEFINING_FILE_FOR(SystemProgram) diff --git a/lib/System/README.txt b/lib/System/README.txt new file mode 100644 index 0000000..eacb200 --- /dev/null +++ b/lib/System/README.txt @@ -0,0 +1,43 @@ +Design Of lib/System +==================== + +The software in this directory is designed to completely shield LLVM from any +and all operating system specific functionality. It is not intended to be a +complete operating system wrapper (such as ACE), but only to provide the +functionality necessary to support LLVM. + +The software located here, of necessity, has very specific and stringent design +rules. Violation of these rules means that cracks in the shield could form and +the primary goal of the library is defeated. By consistently using this library, +LLVM becomes more easily ported to new platforms since the only thing requiring  +porting is this library. + +Complete documentation for the library can be found in the file: +  llvm/docs/SystemLibrary.html  +or at this URL: +  http://llvm.org/docs/SystemLibrary.html + +While we recommend that you read the more detailed documentation, for the  +impatient, here's a high level summary of the library's requirements. + + 1. No system header files are to be exposed through the interface. + 2. Std C++ and Std C header files are okay to be exposed through the interface. + 3. No exposed system-specific functions. + 4. No exposed system-specific data. + 5. Data in lib/System classes must use only simple C++ intrinsic types. + 6. Errors are handled by returning "true" and setting an optional std::string + 7. Library must not throw any exceptions, period. + 8. Interface functions must not have throw() specifications. + 9. No duplicate function impementations are permitted within an operating +    system class. + +To accomplish these requirements, the library has numerous design criteria that  +must be satisfied. Here's a high level summary of the library's design criteria: + + 1. No unused functionality (only what LLVM needs) + 2. High-Level Interfaces + 3. Use Opaque Classes + 4. Common Implementations</a></li> + 5. Multiple Implementations</a></li> + 6. Minimize Memory Allocation</a></li> + 7. No Virtual Methods diff --git a/lib/System/Signals.cpp b/lib/System/Signals.cpp new file mode 100644 index 0000000..229dd14 --- /dev/null +++ b/lib/System/Signals.cpp @@ -0,0 +1,36 @@ +//===- Signals.cpp - Signal Handling support --------------------*- C++ -*-===// +// +//                     The LLVM Compiler Infrastructure +// +// This file was developed by the LLVM research group and is distributed under +// the University of Illinois Open Source License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +// This file defines some helpful functions for dealing with the possibility of +// Unix signals occuring while your program is running. +// +//===----------------------------------------------------------------------===// + +#include "llvm/System/Signals.h" +#include "llvm/Config/config.h" + +namespace llvm { +using namespace sys; + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only TRULY operating system +//===          independent code. +//===----------------------------------------------------------------------===// + +} + +// Include the platform-specific parts of this class. +#ifdef LLVM_ON_UNIX +#include "Unix/Signals.inc" +#endif +#ifdef LLVM_ON_WIN32 +#include "Win32/Signals.inc" +#endif + +DEFINING_FILE_FOR(SystemSignals) diff --git a/lib/System/TimeValue.cpp b/lib/System/TimeValue.cpp new file mode 100644 index 0000000..8ecfd22 --- /dev/null +++ b/lib/System/TimeValue.cpp @@ -0,0 +1,59 @@ +//===-- TimeValue.cpp - Implement OS TimeValue Concept ----------*- C++ -*-===// +// +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Reid Spencer and is distributed under the +// University of Illinois Open Source License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +//  This file implements the operating system TimeValue concept. +// +//===----------------------------------------------------------------------===// + +#include "llvm/System/TimeValue.h" +#include "llvm/Config/config.h" + +namespace llvm { +using namespace sys; + +const TimeValue TimeValue::MinTime       = TimeValue ( INT64_MIN,0 ); +const TimeValue TimeValue::MaxTime       = TimeValue ( INT64_MAX,0 ); +const TimeValue TimeValue::ZeroTime      = TimeValue ( 0,0 ); +const TimeValue TimeValue::PosixZeroTime = TimeValue ( -946684800,0 ); +const TimeValue TimeValue::Win32ZeroTime = TimeValue ( -12591158400ULL,0 ); + +void +TimeValue::normalize( void ) { +  if ( nanos_ >= NANOSECONDS_PER_SECOND ) { +    do { +      seconds_++; +      nanos_ -= NANOSECONDS_PER_SECOND; +    } while ( nanos_ >= NANOSECONDS_PER_SECOND ); +  } else if (nanos_ <= -NANOSECONDS_PER_SECOND ) { +    do { +      seconds_--; +      nanos_ += NANOSECONDS_PER_SECOND; +    } while (nanos_ <= -NANOSECONDS_PER_SECOND); +  } + +  if (seconds_ >= 1 && nanos_ < 0) { +    seconds_--; +    nanos_ += NANOSECONDS_PER_SECOND; +  } else if (seconds_ < 0 && nanos_ > 0) { +    seconds_++; +    nanos_ -= NANOSECONDS_PER_SECOND; +  } +} + +} + +/// Include the platform specific portion of TimeValue class +#ifdef LLVM_ON_UNIX +#include "Unix/TimeValue.inc" +#endif +#ifdef LLVM_ON_WIN32 +#include "Win32/TimeValue.inc" +#endif + +DEFINING_FILE_FOR(SystemTimeValue) diff --git a/lib/System/Unix/Alarm.inc b/lib/System/Unix/Alarm.inc new file mode 100644 index 0000000..1480802 --- /dev/null +++ b/lib/System/Unix/Alarm.inc @@ -0,0 +1,68 @@ +//===-- Alarm.inc - Implement Unix Alarm Support --------------------------===// +// +//                     The LLVM Compiler Infrastructure +// +// This file was developed by the Reid Spencer and is distributed under the +// University of Illinois Open Source License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +// This file implements the UNIX Alarm support. +// +//===----------------------------------------------------------------------===// + +#include <signal.h> +#include <unistd.h> +#include <cassert> +using namespace llvm; + +/// AlarmCancelled - This flag is set by the SIGINT signal handler if the +/// user presses CTRL-C. +static volatile bool AlarmCancelled = false; + +/// AlarmTriggered - This flag is set by the SIGALRM signal handler if the  +/// alarm was triggered. +static volatile bool AlarmTriggered = false; + +/// NestedSOI - Sanity check.  Alarms cannot be nested or run in parallel.   +/// This ensures that they never do. +static bool NestedSOI = false; + +static RETSIGTYPE SigIntHandler(int Sig) { +  AlarmCancelled = true; +  signal(SIGINT, SigIntHandler); +} + +static RETSIGTYPE SigAlarmHandler(int Sig) { +  AlarmTriggered = true; +} + +static void (*OldSigIntHandler) (int); + +void sys::SetupAlarm(unsigned seconds) { +  assert(!NestedSOI && "sys::SetupAlarm calls cannot be nested!"); +  NestedSOI = true; +  AlarmCancelled = false; +  AlarmTriggered = false; +  ::signal(SIGALRM, SigAlarmHandler); +  OldSigIntHandler = ::signal(SIGINT, SigIntHandler); +  ::alarm(seconds); +} + +void sys::TerminateAlarm() { +  assert(NestedSOI && "sys::TerminateAlarm called without sys::SetupAlarm!"); +  ::alarm(0); +  ::signal(SIGALRM, SIG_DFL); +  ::signal(SIGINT, OldSigIntHandler); +  AlarmCancelled = false; +  AlarmTriggered = false; +  NestedSOI = false; +} + +int sys::AlarmStatus() { +  if (AlarmCancelled) +    return -1; +  if (AlarmTriggered) +    return 1; +  return 0; +} diff --git a/lib/System/Unix/MappedFile.inc b/lib/System/Unix/MappedFile.inc new file mode 100644 index 0000000..91b92ec --- /dev/null +++ b/lib/System/Unix/MappedFile.inc @@ -0,0 +1,154 @@ +//===- Unix/MappedFile.cpp - Unix MappedFile Implementation -----*- C++ -*-===// +//  +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Reid Spencer and is distributed under the  +// University of Illinois Open Source License. See LICENSE.TXT for details. +//  +//===----------------------------------------------------------------------===// +// +// This file provides the generic Unix implementation of the MappedFile concept. +// +//===----------------------------------------------------------------------===// + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only generic UNIX code that +//===          is guaranteed to work on *all* UNIX variants. +//===----------------------------------------------------------------------===// + +#include "Unix.h" +#include "llvm/System/Process.h" + +#ifdef HAVE_FCNTL_H +#include <fcntl.h> +#endif + +#ifdef HAVE_SYS_MMAN_H +#include <sys/mman.h> +#endif + +#ifdef HAVE_SYS_STAT_H +#include <sys/stat.h> +#endif + +namespace llvm { +using namespace sys; + +struct sys::MappedFileInfo { +  int   FD; +  off_t Size; +}; + +bool MappedFile::initialize(std::string* ErrMsg) { +  int mode = 0; +  if (options_ & READ_ACCESS)  +    if (options_ & WRITE_ACCESS) +      mode = O_RDWR; +    else +      mode = O_RDONLY; +  else if (options_ & WRITE_ACCESS) +    mode = O_WRONLY; + +  int FD = ::open(path_.c_str(), mode); +  if (FD < 0) { +    MakeErrMsg(ErrMsg, "can't open file '" + path_.toString() + "'"); +    return true; +  }  +  const FileStatus *Status = path_.getFileStatus(false, ErrMsg); +  if (!Status) { +    ::close(FD); +    return true; +  } +  info_ = new MappedFileInfo; +  info_->FD = FD; +  info_->Size = Status->getSize(); +  return false; +} + +void MappedFile::terminate() { +  assert(info_ && "MappedFile not initialized"); +  ::close(info_->FD); +  delete info_; +  info_ = 0; +} + +void MappedFile::unmap() { +  assert(info_ && "MappedFile not initialized"); +  if (isMapped()) { +    if (options_ & WRITE_ACCESS) +      ::msync(base_, info_->Size, MS_SYNC); +    ::munmap(base_, info_->Size); +    base_ = 0;  // Mark this as non-mapped. +  } +} + +void* MappedFile::map(std::string* ErrMsg) { +  assert(info_ && "MappedFile not initialized"); +  if (!isMapped()) { +    int prot = PROT_NONE; +    int flags = 0; +#ifdef MAP_FILE +    flags |= MAP_FILE; +#endif +    if (options_ == 0) { +      prot = PROT_READ; +      flags = MAP_PRIVATE; +    } else { +      if (options_ & READ_ACCESS) +        prot |= PROT_READ; +      if (options_ & WRITE_ACCESS) +        prot |= PROT_WRITE; +      if (options_ & EXEC_ACCESS) +        prot |= PROT_EXEC; +      if (options_ & SHARED_MAPPING) +        flags |= MAP_SHARED; +      else +        flags |= MAP_PRIVATE; +    } +    size_t map_size = ((info_->Size / Process::GetPageSize())+1) * +      Process::GetPageSize(); + +    base_ = ::mmap(0, map_size, prot, flags, info_->FD, 0); +    if (base_ == MAP_FAILED) { +      MakeErrMsg(ErrMsg, "Can't map file:" + path_.toString()); +      return 0; +    } +  } +  return base_; +} + +size_t MappedFile::size() const { +  assert(info_ && "MappedFile not initialized"); +  return info_->Size; +} + +bool MappedFile::size(size_t new_size, std::string* ErrMsg) { +  assert(info_ && "MappedFile not initialized"); + +  // Take the mapping out of memory +  this->unmap(); + +  // Adjust the current size to a page boundary +  size_t cur_size = ((info_->Size / Process::GetPageSize())+1) * +    Process::GetPageSize(); + +  // Adjust the new_size to a page boundary +  new_size = ((new_size / Process::GetPageSize())+1) * +    Process::GetPageSize(); + +  // If the file needs to be extended +  if (new_size > cur_size) { +    // Ensure we can allocate at least the idodes necessary to handle the +    // file size requested.  +    if ((off_t)-1 == ::lseek(info_->FD, new_size, SEEK_SET)) +      return MakeErrMsg(ErrMsg, "Can't lseek: "); +    if (-1 == ::write(info_->FD, "\0", 1)) +      return MakeErrMsg(ErrMsg, "Can't write: "); +  } + +  // Put the mapping back into memory. +  return this->map(ErrMsg); +} + +} + diff --git a/lib/System/Unix/Memory.inc b/lib/System/Unix/Memory.inc new file mode 100644 index 0000000..d040416 --- /dev/null +++ b/lib/System/Unix/Memory.inc @@ -0,0 +1,76 @@ +//===- Unix/Memory.cpp - Generic UNIX System Configuration ------*- C++ -*-===// +//  +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Reid Spencer and is distributed under the  +// University of Illinois Open Source License. See LICENSE.TXT for details. +//  +//===----------------------------------------------------------------------===// +// +// This file defines some functions for various memory management utilities. +// +//===----------------------------------------------------------------------===// + +#include "Unix.h" +#include "llvm/System/Process.h" + +#ifdef HAVE_SYS_MMAN_H +#include <sys/mman.h> +#endif + +/// AllocateRWXMemory - Allocate a slab of memory with read/write/execute +/// permissions.  This is typically used for JIT applications where we want +/// to emit code to the memory then jump to it.  Getting this type of memory +/// is very OS specific. +/// +llvm::sys::MemoryBlock  +llvm::sys::Memory::AllocateRWX(unsigned NumBytes, const MemoryBlock* NearBlock, +                               std::string *ErrMsg) { +  if (NumBytes == 0) return MemoryBlock(); + +  long pageSize = Process::GetPageSize(); +  unsigned NumPages = (NumBytes+pageSize-1)/pageSize; + +  int fd = -1; +#ifdef NEED_DEV_ZERO_FOR_MMAP +  static int zero_fd = open("/dev/zero", O_RDWR); +  if (zero_fd == -1) { +    MakeErrMsg(ErrMsg, "Can't open /dev/zero device"); +    return MemoryBlock(); +  } +  fd = zero_fd; +#endif + +  int flags = MAP_PRIVATE | +#ifdef HAVE_MMAP_ANONYMOUS +  MAP_ANONYMOUS +#else +  MAP_ANON +#endif +  ; + +  void* start = NearBlock ? (unsigned char*)NearBlock->base() +  +                            NearBlock->size() : 0; + +  void *pa = ::mmap(start, pageSize*NumPages, PROT_READ|PROT_WRITE|PROT_EXEC, +                    flags, fd, 0); +  if (pa == MAP_FAILED) { +    if (NearBlock) //Try again without a near hint +      return AllocateRWX(NumBytes, 0); + +    MakeErrMsg(ErrMsg, "Can't allocate RWX Memory"); +    return MemoryBlock(); +  } +  MemoryBlock result; +  result.Address = pa; +  result.Size = NumPages*pageSize; +  return result; +} + +bool llvm::sys::Memory::ReleaseRWX(MemoryBlock &M, std::string *ErrMsg) { +  if (M.Address == 0 || M.Size == 0) return false; +  if (0 != ::munmap(M.Address, M.Size)) +    return MakeErrMsg(ErrMsg, "Can't release RWX Memory"); +  return false; +} + diff --git a/lib/System/Unix/Mutex.inc b/lib/System/Unix/Mutex.inc new file mode 100644 index 0000000..d0984a4 --- /dev/null +++ b/lib/System/Unix/Mutex.inc @@ -0,0 +1,49 @@ +//===- llvm/System/Unix/Mutex.inc - Unix Mutex Implementation ---*- C++ -*-===// +//  +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Reid Spencer and is distributed under the  +// University of Illinois Open Source License. See LICENSE.TXT for details. +//  +//===----------------------------------------------------------------------===// +// +// This file implements the Unix specific (non-pthread) Mutex class. +// +//===----------------------------------------------------------------------===// + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only generic UNIX code that +//===          is guaranteed to work on *all* UNIX variants. +//===----------------------------------------------------------------------===// + +namespace llvm +{ +using namespace sys; + +Mutex::Mutex( bool recursive) +{ +} + +Mutex::~Mutex() +{ +} + +bool  +Mutex::acquire() +{ +  return true; +} + +bool  +Mutex::release() +{ +  return true; +} + +bool  +Mutex::tryacquire( void ) +{ +  return true; +} + +} diff --git a/lib/System/Unix/Path.inc b/lib/System/Unix/Path.inc new file mode 100644 index 0000000..e4916ba --- /dev/null +++ b/lib/System/Unix/Path.inc @@ -0,0 +1,721 @@ +//===- llvm/System/Unix/Path.cpp - Unix Path Implementation -----*- C++ -*-===// +//  +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Reid Spencer and is distributed under the  +// University of Illinois Open Source License. See LICENSE.TXT for details. +//  +//===----------------------------------------------------------------------===// +// +// This file implements the Unix specific portion of the Path class. +// +//===----------------------------------------------------------------------===// + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only generic UNIX code that +//===          is guaranteed to work on *all* UNIX variants. +//===----------------------------------------------------------------------===// + +#include "llvm/Config/alloca.h" +#include "Unix.h" +#if HAVE_SYS_STAT_H +#include <sys/stat.h> +#endif +#if HAVE_FCNTL_H +#include <fcntl.h> +#endif +#if HAVE_UTIME_H +#include <utime.h> +#endif +#if HAVE_TIME_H +#include <time.h> +#endif +#if HAVE_DIRENT_H +# include <dirent.h> +# define NAMLEN(dirent) strlen((dirent)->d_name) +#else +# define dirent direct +# define NAMLEN(dirent) (dirent)->d_namlen +# if HAVE_SYS_NDIR_H +#  include <sys/ndir.h> +# endif +# if HAVE_SYS_DIR_H +#  include <sys/dir.h> +# endif +# if HAVE_NDIR_H +#  include <ndir.h> +# endif +#endif + +// Put in a hack for Cygwin which falsely reports that the mkdtemp function +// is available when it is not. +#ifdef __CYGWIN__ +# undef HAVE_MKDTEMP +#endif + +namespace { +inline bool lastIsSlash(const std::string& path) { +  return !path.empty() && path[path.length() - 1] == '/'; +} + +} + +namespace llvm { +using namespace sys; + +bool  +Path::isValid() const { +  // Check some obvious things +  if (path.empty())  +    return false; +  else if (path.length() >= MAXPATHLEN) +    return false; + +  // Check that the characters are ascii chars +  size_t len = path.length(); +  unsigned i = 0; +  while (i < len && isascii(path[i]))  +    ++i; +  return i >= len;  +} + +bool  +Path::isAbsolute() const { +  if (path.empty()) +    return false; +  return path[0] == '/'; +}  +Path +Path::GetRootDirectory() { +  Path result; +  result.set("/"); +  return result; +} + +Path +Path::GetTemporaryDirectory(std::string* ErrMsg ) { +#if defined(HAVE_MKDTEMP) +  // The best way is with mkdtemp but that's not available on many systems,  +  // Linux and FreeBSD have it. Others probably won't. +  char pathname[MAXPATHLEN]; +  strcpy(pathname,"/tmp/llvm_XXXXXX"); +  if (0 == mkdtemp(pathname)) { +    MakeErrMsg(ErrMsg,  +      std::string(pathname) + ": can't create temporary directory"); +    return Path(); +  } +  Path result; +  result.set(pathname); +  assert(result.isValid() && "mkdtemp didn't create a valid pathname!"); +  return result; +#elif defined(HAVE_MKSTEMP) +  // If no mkdtemp is available, mkstemp can be used to create a temporary file +  // which is then removed and created as a directory. We prefer this over +  // mktemp because of mktemp's inherent security and threading risks. We still +  // have a slight race condition from the time the temporary file is created to +  // the time it is re-created as a directoy.  +  char pathname[MAXPATHLEN]; +  strcpy(pathname, "/tmp/llvm_XXXXXX"); +  int fd = 0; +  if (-1 == (fd = mkstemp(pathname))) { +    MakeErrMsg(ErrMsg,  +      std::string(pathname) + ": can't create temporary directory"); +    return Path(); +  } +  ::close(fd); +  ::unlink(pathname); // start race condition, ignore errors +  if (-1 == ::mkdir(pathname, S_IRWXU)) { // end race condition +    MakeErrMsg(ErrMsg,  +      std::string(pathname) + ": can't create temporary directory"); +    return Path(); +  } +  Path result; +  result.set(pathname); +  assert(result.isValid() && "mkstemp didn't create a valid pathname!"); +  return result; +#elif defined(HAVE_MKTEMP) +  // If a system doesn't have mkdtemp(3) or mkstemp(3) but it does have +  // mktemp(3) then we'll assume that system (e.g. AIX) has a reasonable +  // implementation of mktemp(3) and doesn't follow BSD 4.3's lead of replacing +  // the XXXXXX with the pid of the process and a letter. That leads to only +  // twenty six temporary files that can be generated. +  char pathname[MAXPATHLEN]; +  strcpy(pathname, "/tmp/llvm_XXXXXX"); +  char *TmpName = ::mktemp(pathname); +  if (TmpName == 0) { +    MakeErrMsg(ErrMsg,  +      std::string(TmpName) + ": can't create unique directory name"); +    return Path(); +  } +  if (-1 == ::mkdir(TmpName, S_IRWXU)) { +    MakeErrMsg(ErrMsg,  +        std::string(TmpName) + ": can't create temporary directory"); +    return Path(); +  } +  Path result; +  result.set(TmpName); +  assert(result.isValid() && "mktemp didn't create a valid pathname!"); +  return result; +#else +  // This is the worst case implementation. tempnam(3) leaks memory unless its +  // on an SVID2 (or later) system. On BSD 4.3 it leaks. tmpnam(3) has thread +  // issues. The mktemp(3) function doesn't have enough variability in the +  // temporary name generated. So, we provide our own implementation that  +  // increments an integer from a random number seeded by the current time. This +  // should be sufficiently unique that we don't have many collisions between +  // processes. Generally LLVM processes don't run very long and don't use very +  // many temporary files so this shouldn't be a big issue for LLVM. +  static time_t num = ::time(0); +  char pathname[MAXPATHLEN]; +  do { +    num++; +    sprintf(pathname, "/tmp/llvm_%010u", unsigned(num)); +  } while ( 0 == access(pathname, F_OK ) ); +  if (-1 == ::mkdir(pathname, S_IRWXU)) { +    MakeErrMsg(ErrMsg,  +      std::string(pathname) + ": can't create temporary directory"); +    return Path(); +  } +  Path result; +  result.set(pathname); +  assert(result.isValid() && "mkstemp didn't create a valid pathname!"); +  return result; +#endif +} + +static void getPathList(const char*path, std::vector<sys::Path>& Paths) { +  const char* at = path; +  const char* delim = strchr(at, ':'); +  Path tmpPath; +  while( delim != 0 ) { +    std::string tmp(at, size_t(delim-at)); +    if (tmpPath.set(tmp)) +      if (tmpPath.canRead()) +        Paths.push_back(tmpPath); +    at = delim + 1; +    delim = strchr(at, ':'); +  } +  if (*at != 0) +    if (tmpPath.set(std::string(at))) +      if (tmpPath.canRead()) +        Paths.push_back(tmpPath); + +} + +void  +Path::GetSystemLibraryPaths(std::vector<sys::Path>& Paths) { +#ifdef LTDL_SHLIBPATH_VAR +  char* env_var = getenv(LTDL_SHLIBPATH_VAR); +  if (env_var != 0) { +    getPathList(env_var,Paths); +  } +#endif +  // FIXME: Should this look at LD_LIBRARY_PATH too? +  Paths.push_back(sys::Path("/usr/local/lib/")); +  Paths.push_back(sys::Path("/usr/X11R6/lib/")); +  Paths.push_back(sys::Path("/usr/lib/")); +  Paths.push_back(sys::Path("/lib/")); +} + +void +Path::GetBitcodeLibraryPaths(std::vector<sys::Path>& Paths) { +  char * env_var = getenv("LLVM_LIB_SEARCH_PATH"); +  if (env_var != 0) { +    getPathList(env_var,Paths); +  } +#ifdef LLVM_LIBDIR +  { +    Path tmpPath; +    if (tmpPath.set(LLVM_LIBDIR)) +      if (tmpPath.canRead()) +        Paths.push_back(tmpPath); +  } +#endif +  GetSystemLibraryPaths(Paths); +} + +Path  +Path::GetLLVMDefaultConfigDir() { +  return Path("/etc/llvm/"); +} + +Path +Path::GetUserHomeDirectory() { +  const char* home = getenv("HOME"); +  if (home) { +    Path result; +    if (result.set(home)) +      return result; +  } +  return GetRootDirectory(); +} + + +std::string +Path::getBasename() const { +  // Find the last slash +  size_t slash = path.rfind('/'); +  if (slash == std::string::npos) +    slash = 0; +  else +    slash++; + +  size_t dot = path.rfind('.'); +  if (dot == std::string::npos || dot < slash) +    return path.substr(slash); +  else +    return path.substr(slash, dot - slash); +} + +bool Path::getMagicNumber(std::string& Magic, unsigned len) const { +  assert(len < 1024 && "Request for magic string too long"); +  char* buf = (char*) alloca(1 + len); +  int fd = ::open(path.c_str(), O_RDONLY); +  if (fd < 0) +    return false; +  ssize_t bytes_read = ::read(fd, buf, len); +  ::close(fd); +  if (ssize_t(len) != bytes_read) { +    Magic.clear(); +    return false; +  } +  Magic.assign(buf,len); +  return true; +} + +bool +Path::exists() const { +  return 0 == access(path.c_str(), F_OK ); +} + +bool +Path::canRead() const { +  return 0 == access(path.c_str(), F_OK | R_OK ); +} + +bool +Path::canWrite() const { +  return 0 == access(path.c_str(), F_OK | W_OK ); +} + +bool +Path::canExecute() const { +  if (0 != access(path.c_str(), R_OK | X_OK )) +    return false; +  struct stat buf; +  if (0 != stat(path.c_str(), &buf)) +    return false; +  if (!S_ISREG(buf.st_mode)) +    return false; +  return true; +} + +std::string  +Path::getLast() const { +  // Find the last slash +  size_t pos = path.rfind('/'); + +  // Handle the corner cases +  if (pos == std::string::npos) +    return path; + +  // If the last character is a slash +  if (pos == path.length()-1) { +    // Find the second to last slash +    size_t pos2 = path.rfind('/', pos-1); +    if (pos2 == std::string::npos) +      return path.substr(0,pos); +    else +      return path.substr(pos2+1,pos-pos2-1); +  } +  // Return everything after the last slash +  return path.substr(pos+1); +} + +const FileStatus * +PathWithStatus::getFileStatus(bool update, std::string *ErrStr) const { +  if (!fsIsValid || update) { +    struct stat buf; +    if (0 != stat(path.c_str(), &buf)) { +      MakeErrMsg(ErrStr, path + ": can't get status of file"); +      return 0; +    } +    status.fileSize = buf.st_size; +    status.modTime.fromEpochTime(buf.st_mtime); +    status.mode = buf.st_mode; +    status.user = buf.st_uid; +    status.group = buf.st_gid; +    status.uniqueID = uint64_t(buf.st_ino); +    status.isDir  = S_ISDIR(buf.st_mode); +    status.isFile = S_ISREG(buf.st_mode); +    fsIsValid = true; +  } +  return &status; +} + +static bool AddPermissionBits(const Path &File, int bits) { +  // Get the umask value from the operating system.  We want to use it +  // when changing the file's permissions. Since calling umask() sets +  // the umask and returns its old value, we must call it a second +  // time to reset it to the user's preference. +  int mask = umask(0777); // The arg. to umask is arbitrary. +  umask(mask);            // Restore the umask. + +  // Get the file's current mode. +  struct stat buf; +  if (0 != stat(File.toString().c_str(), &buf))  +    return false; +  // Change the file to have whichever permissions bits from 'bits' +  // that the umask would not disable. +  if ((chmod(File.c_str(), (buf.st_mode | (bits & ~mask)))) == -1) +      return false; +  return true; +} + +bool Path::makeReadableOnDisk(std::string* ErrMsg) { +  if (!AddPermissionBits(*this, 0444))  +    return MakeErrMsg(ErrMsg, path + ": can't make file readable"); +  return false; +} + +bool Path::makeWriteableOnDisk(std::string* ErrMsg) { +  if (!AddPermissionBits(*this, 0222)) +    return MakeErrMsg(ErrMsg, path + ": can't make file writable"); +  return false; +} + +bool Path::makeExecutableOnDisk(std::string* ErrMsg) { +  if (!AddPermissionBits(*this, 0111)) +    return MakeErrMsg(ErrMsg, path + ": can't make file executable"); +  return false; +} + +bool +Path::getDirectoryContents(std::set<Path>& result, std::string* ErrMsg) const { +  DIR* direntries = ::opendir(path.c_str()); +  if (direntries == 0) +    return MakeErrMsg(ErrMsg, path + ": can't open directory"); + +  std::string dirPath = path; +  if (!lastIsSlash(dirPath)) +    dirPath += '/'; + +  result.clear(); +  struct dirent* de = ::readdir(direntries); +  for ( ; de != 0; de = ::readdir(direntries)) { +    if (de->d_name[0] != '.') { +      Path aPath(dirPath + (const char*)de->d_name); +      struct stat st; +      if (0 != lstat(aPath.path.c_str(), &st)) { +        if (S_ISLNK(st.st_mode)) +          continue; // dangling symlink -- ignore +        return MakeErrMsg(ErrMsg,  +                          aPath.path +  ": can't determine file object type"); +      } +      result.insert(aPath); +    } +  } +   +  closedir(direntries); +  return false; +} + +bool +Path::set(const std::string& a_path) { +  if (a_path.empty()) +    return false; +  std::string save(path); +  path = a_path; +  if (!isValid()) { +    path = save; +    return false; +  } +  return true; +} + +bool +Path::appendComponent(const std::string& name) { +  if (name.empty()) +    return false; +  std::string save(path); +  if (!lastIsSlash(path)) +    path += '/'; +  path += name; +  if (!isValid()) { +    path = save; +    return false; +  } +  return true; +} + +bool +Path::eraseComponent() { +  size_t slashpos = path.rfind('/',path.size()); +  if (slashpos == 0 || slashpos == std::string::npos) { +    path.erase(); +    return true; +  } +  if (slashpos == path.size() - 1) +    slashpos = path.rfind('/',slashpos-1); +  if (slashpos == std::string::npos) { +    path.erase(); +    return true; +  } +  path.erase(slashpos); +  return true; +} + +bool +Path::appendSuffix(const std::string& suffix) { +  std::string save(path); +  path.append("."); +  path.append(suffix); +  if (!isValid()) { +    path = save; +    return false; +  } +  return true; +} + +bool +Path::eraseSuffix() { +  std::string save = path; +  size_t dotpos = path.rfind('.',path.size()); +  size_t slashpos = path.rfind('/',path.size()); +  if (dotpos != std::string::npos) { +    if (slashpos == std::string::npos || dotpos > slashpos+1) { +      path.erase(dotpos, path.size()-dotpos); +      return true; +    } +  } +  if (!isValid()) +    path = save; +  return false; +} + +bool +Path::createDirectoryOnDisk( bool create_parents, std::string* ErrMsg ) { +  // Get a writeable copy of the path name +  char pathname[MAXPATHLEN]; +  path.copy(pathname,MAXPATHLEN); + +  // Null-terminate the last component +  int lastchar = path.length() - 1 ;  +  if (pathname[lastchar] == '/')  +    pathname[lastchar] = 0; +  else  +    pathname[lastchar+1] = 0; + +  // If we're supposed to create intermediate directories +  if ( create_parents ) { +    // Find the end of the initial name component +    char * next = strchr(pathname,'/'); +    if ( pathname[0] == '/')  +      next = strchr(&pathname[1],'/'); + +    // Loop through the directory components until we're done  +    while ( next != 0 ) { +      *next = 0; +      if (0 != access(pathname, F_OK | R_OK | W_OK)) +        if (0 != mkdir(pathname, S_IRWXU | S_IRWXG)) { +          return MakeErrMsg(ErrMsg,  +                            std::string(pathname) + ": can't create directory"); +        } +      char* save = next; +      next = strchr(next+1,'/'); +      *save = '/'; +    } +  }  + +  if (0 != access(pathname, F_OK | R_OK)) +    if (0 != mkdir(pathname, S_IRWXU | S_IRWXG)) { +      return MakeErrMsg(ErrMsg,  +                        std::string(pathname) + ": can't create directory"); +    } +  return false; +} + +bool +Path::createFileOnDisk(std::string* ErrMsg) { +  // Create the file +  int fd = ::creat(path.c_str(), S_IRUSR | S_IWUSR); +  if (fd < 0) +    return MakeErrMsg(ErrMsg, path + ": can't create file"); +  ::close(fd); +  return false; +} + +bool +Path::createTemporaryFileOnDisk(bool reuse_current, std::string* ErrMsg) { +  // Make this into a unique file name +  if (makeUnique( reuse_current, ErrMsg )) +    return true; + +  // create the file +  int fd = ::open(path.c_str(), O_WRONLY|O_CREAT|O_TRUNC, 0666); +  if (fd < 0)  +    return MakeErrMsg(ErrMsg, path + ": can't create temporary file"); +  ::close(fd); +  return false; +} + +bool +Path::eraseFromDisk(bool remove_contents, std::string *ErrStr) const { +  // Get the status so we can determin if its a file or directory +  struct stat buf; +  if (0 != stat(path.c_str(), &buf)) { +    MakeErrMsg(ErrStr, path + ": can't get status of file"); +    return true; +  } + +  // Note: this check catches strange situations. In all cases, LLVM should  +  // only be involved in the creation and deletion of regular files.  This  +  // check ensures that what we're trying to erase is a regular file. It  +  // effectively prevents LLVM from erasing things like /dev/null, any block  +  // special file, or other things that aren't "regular" files.  +  if (S_ISREG(buf.st_mode)) { +    if (unlink(path.c_str()) != 0) +      return MakeErrMsg(ErrStr, path + ": can't destroy file"); +    return false; +  } +   +  if (!S_ISDIR(buf.st_mode)) { +    if (ErrStr) *ErrStr = "not a file or directory"; +    return true; +  } + +  if (remove_contents) { +    // Recursively descend the directory to remove its contents. +    std::string cmd = "/bin/rm -rf " + path; +    system(cmd.c_str()); +    return false; +  } + +  // Otherwise, try to just remove the one directory. +  char pathname[MAXPATHLEN]; +  path.copy(pathname, MAXPATHLEN); +  int lastchar = path.length() - 1 ;  +  if (pathname[lastchar] == '/')  +    pathname[lastchar] = 0; +  else +    pathname[lastchar+1] = 0; +     +  if (rmdir(pathname) != 0) +    return MakeErrMsg(ErrStr,  +      std::string(pathname) + ": can't erase directory"); +  return false; +} + +bool +Path::renamePathOnDisk(const Path& newName, std::string* ErrMsg) { +  if (0 != ::rename(path.c_str(), newName.c_str())) +    return MakeErrMsg(ErrMsg, std::string("can't rename '") + path + "' as '" +  +               newName.toString() + "' "); +  return false; +} + +bool  +Path::setStatusInfoOnDisk(const FileStatus &si, std::string *ErrStr) const { +  struct utimbuf utb; +  utb.actime = si.modTime.toPosixTime(); +  utb.modtime = utb.actime; +  if (0 != ::utime(path.c_str(),&utb)) +    return MakeErrMsg(ErrStr, path + ": can't set file modification time"); +  if (0 != ::chmod(path.c_str(),si.mode)) +    return MakeErrMsg(ErrStr, path + ": can't set mode"); +  return false; +} + +bool  +sys::CopyFile(const sys::Path &Dest, const sys::Path &Src, std::string* ErrMsg){ +  int inFile = -1; +  int outFile = -1; +  inFile = ::open(Src.c_str(), O_RDONLY); +  if (inFile == -1) +    return MakeErrMsg(ErrMsg, Src.toString() +  +      ": can't open source file to copy"); + +  outFile = ::open(Dest.c_str(), O_WRONLY|O_CREAT, 0666); +  if (outFile == -1) { +    ::close(inFile); +    return MakeErrMsg(ErrMsg, Dest.toString() + +      ": can't create destination file for copy"); +  } + +  char Buffer[16*1024]; +  while (ssize_t Amt = ::read(inFile, Buffer, 16*1024)) { +    if (Amt == -1) { +      if (errno != EINTR && errno != EAGAIN) { +        ::close(inFile); +        ::close(outFile); +        return MakeErrMsg(ErrMsg, Src.toString()+": can't read source file: "); +      } +    } else { +      char *BufPtr = Buffer; +      while (Amt) { +        ssize_t AmtWritten = ::write(outFile, BufPtr, Amt); +        if (AmtWritten == -1) { +          if (errno != EINTR && errno != EAGAIN) { +            ::close(inFile); +            ::close(outFile); +            return MakeErrMsg(ErrMsg, Dest.toString() +  +              ": can't write destination file: "); +          } +        } else { +          Amt -= AmtWritten; +          BufPtr += AmtWritten; +        } +      } +    } +  } +  ::close(inFile); +  ::close(outFile); +  return false; +} + +bool  +Path::makeUnique(bool reuse_current, std::string* ErrMsg) { +  if (reuse_current && !exists()) +    return false; // File doesn't exist already, just use it! + +  // Append an XXXXXX pattern to the end of the file for use with mkstemp,  +  // mktemp or our own implementation. +  char *FNBuffer = (char*) alloca(path.size()+8); +  path.copy(FNBuffer,path.size()); +  strcpy(FNBuffer+path.size(), "-XXXXXX"); + +#if defined(HAVE_MKSTEMP) +  int TempFD; +  if ((TempFD = mkstemp(FNBuffer)) == -1) +    return MakeErrMsg(ErrMsg, path + ": can't make unique filename"); + +  // We don't need to hold the temp file descriptor... we will trust that no one +  // will overwrite/delete the file before we can open it again. +  close(TempFD); + +  // Save the name +  path = FNBuffer; +#elif defined(HAVE_MKTEMP) +  // If we don't have mkstemp, use the old and obsolete mktemp function. +  if (mktemp(FNBuffer) == 0) +    return MakeErrMsg(ErrMsg, path + ": can't make unique filename"); + +  // Save the name +  path = FNBuffer; +#else +  // Okay, looks like we have to do it all by our lonesome. +  static unsigned FCounter = 0; +  unsigned offset = path.size() + 1; +  while ( FCounter < 999999 && exists()) { +    sprintf(FNBuffer+offset,"%06u",++FCounter); +    path = FNBuffer; +  } +  if (FCounter > 999999) +    return MakeErrMsg(ErrMsg,  +      path + ": can't make unique filename: too many files"); +#endif +  return false; +} + +} // end llvm namespace + diff --git a/lib/System/Unix/Process.inc b/lib/System/Unix/Process.inc new file mode 100644 index 0000000..6b9b2b3 --- /dev/null +++ b/lib/System/Unix/Process.inc @@ -0,0 +1,185 @@ +//===- Unix/Process.cpp - Unix Process Implementation --------- -*- C++ -*-===// +//  +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Reid Spencer and is distributed under the  +// University of Illinois Open Source License. See LICENSE.TXT for details. +//  +//===----------------------------------------------------------------------===// +// +// This file provides the generic Unix implementation of the Process class. +// +//===----------------------------------------------------------------------===// + +#include "Unix.h" +#ifdef HAVE_SYS_TIME_H +#include <sys/time.h> +#endif +#ifdef HAVE_SYS_RESOURCE_H +#include <sys/resource.h> +#endif +#ifdef HAVE_MALLOC_H +#include <malloc.h> +#endif +#ifdef HAVE_MALLOC_MALLOC_H +#include <malloc/malloc.h> +#endif + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only generic UNIX code that +//===          is guaranteed to work on *all* UNIX variants. +//===----------------------------------------------------------------------===// + +using namespace llvm; +using namespace sys; + +unsigned  +Process::GetPageSize()  +{ +#if defined(HAVE_GETPAGESIZE) +  static const int page_size = ::getpagesize(); +#elif defined(HAVE_SYSCONF) +  static long page_size = ::sysconf(_SC_PAGE_SIZE); +#else +#warning Cannot get the page size on this machine +#endif +  return static_cast<unsigned>(page_size); +} + +size_t Process::GetMallocUsage() { +#if defined(HAVE_MALLINFO) +  struct mallinfo mi; +  mi = ::mallinfo(); +  return mi.uordblks; +#elif defined(HAVE_MALLOC_ZONE_STATISTICS) && defined(HAVE_MALLOC_MALLOC_H) +  malloc_statistics_t Stats; +  malloc_zone_statistics(malloc_default_zone(), &Stats); +  return Stats.size_in_use;   // darwin +#elif defined(HAVE_SBRK) +  // Note this is only an approximation and more closely resembles +  // the value returned by mallinfo in the arena field. +  static char *StartOfMemory = reinterpret_cast<char*>(::sbrk(0)); +  char *EndOfMemory = (char*)sbrk(0); +  if (EndOfMemory != ((char*)-1) && StartOfMemory != ((char*)-1)) +    return EndOfMemory - StartOfMemory; +  else +    return 0; +#else +#warning Cannot get malloc info on this platform +  return 0; +#endif +} + +size_t +Process::GetTotalMemoryUsage() +{ +#if defined(HAVE_MALLINFO) +  struct mallinfo mi = ::mallinfo(); +  return mi.uordblks + mi.hblkhd; +#elif defined(HAVE_MALLOC_ZONE_STATISTICS) && defined(HAVE_MALLOC_MALLOC_H) +  malloc_statistics_t Stats; +  malloc_zone_statistics(malloc_default_zone(), &Stats); +  return Stats.size_allocated;   // darwin +#elif defined(HAVE_GETRUSAGE) +  struct rusage usage; +  ::getrusage(RUSAGE_SELF, &usage); +  return usage.ru_maxrss; +#else +#warning Cannot get total memory size on this platform +  return 0; +#endif +} + +void +Process::GetTimeUsage(TimeValue& elapsed, TimeValue& user_time,  +                      TimeValue& sys_time) +{ +  elapsed = TimeValue::now(); +#if defined(HAVE_GETRUSAGE) +  struct rusage usage; +  ::getrusage(RUSAGE_SELF, &usage); +  user_time = TimeValue(  +    static_cast<TimeValue::SecondsType>( usage.ru_utime.tv_sec ),  +    static_cast<TimeValue::NanoSecondsType>( usage.ru_utime.tv_usec *  +      TimeValue::NANOSECONDS_PER_MICROSECOND ) ); +  sys_time = TimeValue(  +    static_cast<TimeValue::SecondsType>( usage.ru_stime.tv_sec ),  +    static_cast<TimeValue::NanoSecondsType>( usage.ru_stime.tv_usec *  +      TimeValue::NANOSECONDS_PER_MICROSECOND ) ); +#else +#warning Cannot get usage times on this platform +  user_time.seconds(0); +  user_time.microseconds(0); +  sys_time.seconds(0); +  sys_time.microseconds(0); +#endif +} + +int Process::GetCurrentUserId() { +  return getuid(); +} + +int Process::GetCurrentGroupId() { +  return getgid(); +} + +#ifdef HAVE_MACH_MACH_H +#include <mach/mach.h> +#endif + +// Some LLVM programs such as bugpoint produce core files as a normal part of +// their operation. To prevent the disk from filling up, this function +// does what's necessary to prevent their generation. +void Process::PreventCoreFiles() { +#if HAVE_SETRLIMIT +  struct rlimit rlim; +  rlim.rlim_cur = rlim.rlim_max = 0; +  setrlimit(RLIMIT_CORE, &rlim); +#endif + +#ifdef HAVE_MACH_MACH_H +  // Disable crash reporting on Mac OS/X. + +  // get information about the original set of exception ports for the task +  mach_msg_type_number_t Count = 0; +  exception_mask_t OriginalMasks[EXC_TYPES_COUNT]; +  exception_port_t OriginalPorts[EXC_TYPES_COUNT]; +  exception_behavior_t OriginalBehaviors[EXC_TYPES_COUNT]; +  thread_state_flavor_t OriginalFlavors[EXC_TYPES_COUNT]; +  kern_return_t err =  +    task_get_exception_ports(mach_task_self(), EXC_MASK_ALL, OriginalMasks, +                             &Count, OriginalPorts, OriginalBehaviors, +                             OriginalFlavors); +  if (err == KERN_SUCCESS) { +    // replace each with MACH_PORT_NULL. +    for (unsigned i = 0; i != Count; ++i) +      task_set_exception_ports(mach_task_self(), OriginalMasks[i],  +                               MACH_PORT_NULL, OriginalBehaviors[i], +                               OriginalFlavors[i]); +  } +#endif +} + +bool Process::StandardInIsUserInput() { +#if HAVE_ISATTY +  return isatty(0); +#endif +  // If we don't have isatty, just return false. +  return false; +} + +bool Process::StandardOutIsDisplayed() { +#if HAVE_ISATTY +  return isatty(1); +#endif +  // If we don't have isatty, just return false. +  return false; +} + +bool Process::StandardErrIsDisplayed() { +#if HAVE_ISATTY +  return isatty(2); +#endif +  // If we don't have isatty, just return false. +  return false; +} diff --git a/lib/System/Unix/Program.inc b/lib/System/Unix/Program.inc new file mode 100644 index 0000000..896e809 --- /dev/null +++ b/lib/System/Unix/Program.inc @@ -0,0 +1,286 @@ +//===- llvm/System/Unix/Program.cpp -----------------------------*- C++ -*-===// +//  +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Reid Spencer and is distributed under the  +// University of Illinois Open Source License. See LICENSE.TXT for details. +//  +//===----------------------------------------------------------------------===// +// +// This file implements the Unix specific portion of the Program class. +// +//===----------------------------------------------------------------------===// + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only generic UNIX code that +//===          is guaranteed to work on *all* UNIX variants. +//===----------------------------------------------------------------------===// + +#include <llvm/Config/config.h> +#include "Unix.h" +#include <iostream> +#if HAVE_SYS_STAT_H +#include <sys/stat.h> +#endif +#if HAVE_SYS_RESOURCE_H +#include <sys/resource.h> +#endif +#if HAVE_SIGNAL_H +#include <signal.h> +#endif +#if HAVE_FCNTL_H +#include <fcntl.h> +#endif + +namespace llvm { +using namespace sys; + +// This function just uses the PATH environment variable to find the program. +Path +Program::FindProgramByName(const std::string& progName) { + +  // Check some degenerate cases +  if (progName.length() == 0) // no program +    return Path(); +  Path temp; +  if (!temp.set(progName)) // invalid name +    return Path(); +  // FIXME: have to check for absolute filename - we cannot assume anything +  // about "." being in $PATH +  if (temp.canExecute()) // already executable as is +    return temp; + +  // At this point, the file name is valid and its not executable +  +  // Get the path. If its empty, we can't do anything to find it. +  const char *PathStr = getenv("PATH"); +  if (PathStr == 0)  +    return Path(); + +  // Now we have a colon separated list of directories to search; try them. +  unsigned PathLen = strlen(PathStr); +  while (PathLen) { +    // Find the first colon... +    const char *Colon = std::find(PathStr, PathStr+PathLen, ':'); + +    // Check to see if this first directory contains the executable... +    Path FilePath; +    if (FilePath.set(std::string(PathStr,Colon))) { +      FilePath.appendComponent(progName); +      if (FilePath.canExecute()) +        return FilePath;                    // Found the executable! +    } + +    // Nope it wasn't in this directory, check the next path in the list! +    PathLen -= Colon-PathStr; +    PathStr = Colon; + +    // Advance past duplicate colons +    while (*PathStr == ':') { +      PathStr++; +      PathLen--; +    } +  } +  return Path(); +} + +static bool RedirectFD(const std::string &File, int FD, std::string* ErrMsg) { +  if (File.empty()) return false;  // Noop + +  // Open the file +  int InFD = open(File.c_str(), FD == 0 ? O_RDONLY : O_WRONLY|O_CREAT, 0666); +  if (InFD == -1) { +    MakeErrMsg(ErrMsg, "Cannot open file '" + File + "' for " +              + (FD == 0 ? "input" : "output") + "!\n"); +    return true; +  } + +  // Install it as the requested FD +  if (-1 == dup2(InFD, FD)) { +    MakeErrMsg(ErrMsg, "Cannot dup2"); +    return true; +  } +  close(InFD);      // Close the original FD +  return false; +} + +static bool Timeout = false; +static void TimeOutHandler(int Sig) { +  Timeout = true; +} + +static void SetMemoryLimits (unsigned size) +{ +#if HAVE_SYS_RESOURCE_H +  struct rlimit r; +  __typeof__ (r.rlim_cur) limit = (__typeof__ (r.rlim_cur)) (size) * 1048576; + +  // Heap size +  getrlimit (RLIMIT_DATA, &r); +  r.rlim_cur = limit; +  setrlimit (RLIMIT_DATA, &r); +#ifdef RLIMIT_RSS +  // Resident set size. +  getrlimit (RLIMIT_RSS, &r); +  r.rlim_cur = limit; +  setrlimit (RLIMIT_RSS, &r); +#endif +#ifdef RLIMIT_AS  // e.g. NetBSD doesn't have it. +  // Virtual memory. +  getrlimit (RLIMIT_AS, &r); +  r.rlim_cur = limit; +  setrlimit (RLIMIT_AS, &r); +#endif +#endif +} + +int  +Program::ExecuteAndWait(const Path& path,  +                        const char** args, +                        const char** envp, +                        const Path** redirects, +                        unsigned secondsToWait, +                        unsigned memoryLimit, +                        std::string* ErrMsg)  +{ +  if (!path.canExecute()) { +    if (ErrMsg) +      *ErrMsg = path.toString() + " is not executable"; +    return -1; +  } + +#ifdef HAVE_SYS_WAIT_H +  // Create a child process. +  int child = fork(); +  switch (child) { +    // An error occured:  Return to the caller. +    case -1: +      MakeErrMsg(ErrMsg, "Couldn't fork"); +      return -1; + +    // Child process: Execute the program. +    case 0: { +      // Redirect file descriptors... +      if (redirects) { +        if (redirects[0]) { +          if (redirects[0]->isEmpty()) { +            if (RedirectFD("/dev/null",0,ErrMsg)) { return -1; } +          } else { +            if (RedirectFD(redirects[0]->toString(), 0,ErrMsg)) { return -1; } +          } +        } +        if (redirects[1]) { +          if (redirects[1]->isEmpty()) { +            if (RedirectFD("/dev/null",1,ErrMsg)) { return -1; } +          } else { +            if (RedirectFD(redirects[1]->toString(),1,ErrMsg)) { return -1; } +          } +        } +        if (redirects[1] && redirects[2] &&  +            *(redirects[1]) != *(redirects[2])) { +          if (redirects[2]->isEmpty()) { +            if (RedirectFD("/dev/null",2,ErrMsg)) { return -1; } +          } else { +            if (RedirectFD(redirects[2]->toString(), 2,ErrMsg)) { return -1; } +          } +        } else if (-1 == dup2(1,2)) { +          MakeErrMsg(ErrMsg, "Can't redirect"); +          return -1; +        } +      } + +      // Set memory limits +      if (memoryLimit!=0) { +        SetMemoryLimits(memoryLimit); +      } +       +      // Execute! +      if (envp != 0) +        execve (path.c_str(), (char** const)args, (char**)envp); +      else +        execv (path.c_str(), (char** const)args); +      // If the execve() failed, we should exit and let the parent pick up +      // our non-zero exit status. +      exit (errno); +    } + +    // Parent process: Break out of the switch to do our processing. +    default: +      break; +  } + +  // Make sure stderr and stdout have been flushed +  std::cerr << std::flush; +  std::cout << std::flush; +  fsync(1); +  fsync(2); + +  struct sigaction Act, Old; + +  // Install a timeout handler. +  if (secondsToWait) { +    Timeout = false; +    Act.sa_sigaction = 0; +    Act.sa_handler = TimeOutHandler; +    sigemptyset(&Act.sa_mask); +    Act.sa_flags = 0; +    sigaction(SIGALRM, &Act, &Old); +    alarm(secondsToWait); +  } + +  // Parent process: Wait for the child process to terminate. +  int status; +  while (wait(&status) != child) +    if (secondsToWait && errno == EINTR) { +      // Kill the child. +      kill(child, SIGKILL); +         +      // Turn off the alarm and restore the signal handler +      alarm(0); +      sigaction(SIGALRM, &Old, 0); + +      // Wait for child to die +      if (wait(&status) != child) +        MakeErrMsg(ErrMsg, "Child timed out but wouldn't die"); +         +      return -1;   // Timeout detected +    } else { +      MakeErrMsg(ErrMsg, "Error waiting for child process"); +      return -1; +    } + +  // We exited normally without timeout, so turn off the timer. +  if (secondsToWait) { +    alarm(0); +    sigaction(SIGALRM, &Old, 0); +  } + +  // Return the proper exit status. 0=success, >0 is programs' exit status, +  // <0 means a signal was returned, -9999999 means the program dumped core. +  int result = 0; +  if (WIFEXITED(status)) +    result = WEXITSTATUS(status); +  else if (WIFSIGNALED(status)) +    result = 0 - WTERMSIG(status); +#ifdef WCOREDUMP +  else if (WCOREDUMP(status)) +    result |= 0x01000000; +#endif +  return result; +#else +  return -99; +#endif +     +} + +bool Program::ChangeStdinToBinary(){ +  // Do nothing, as Unix doesn't differentiate between text and binary. +  return false; +} + +bool Program::ChangeStdoutToBinary(){ +  // Do nothing, as Unix doesn't differentiate between text and binary. +  return false; +} + +} diff --git a/lib/System/Unix/README.txt b/lib/System/Unix/README.txt new file mode 100644 index 0000000..b3bace4 --- /dev/null +++ b/lib/System/Unix/README.txt @@ -0,0 +1,16 @@ +llvm/lib/System/Unix README +=========================== + +This directory provides implementations of the lib/System classes that +are common to two or more variants of UNIX. For example, the directory  +structure underneath this directory could look like this: + +Unix           - only code that is truly generic to all UNIX platforms +  Posix        - code that is specific to Posix variants of UNIX +  SUS          - code that is specific to the Single Unix Specification  +  SysV         - code that is specific to System V variants of UNIX + +As a rule, only those directories actually needing to be created should be +created. Also, further subdirectories could be created to reflect versions of +the various standards. For example, under SUS there could be v1, v2, and v3 +subdirectories to reflect the three major versions of SUS. diff --git a/lib/System/Unix/SUS/Process.cpp b/lib/System/Unix/SUS/Process.cpp new file mode 100644 index 0000000..fb462b4 --- /dev/null +++ b/lib/System/Unix/SUS/Process.cpp @@ -0,0 +1,30 @@ +//===- Unix/SUS/Process.cpp - Linux Process Implementation ---- -*- C++ -*-===// +// +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Reid Spencer and is distributed under the +// University of Illinois Open Source License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +// This file provides the Linux specific implementation of the Process class. +// +//===----------------------------------------------------------------------===// + +#include <unistd.h> + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only code specific to the +//===          SUS (Single Unix Specification). +//===----------------------------------------------------------------------===// + +namespace llvm { +using namespace sys; + +unsigned +Process::GetPageSize() { +  static const long page_size = sysconf(_SC_PAGE_SIZE); +  return static_cast<unsigned>(page_size); +} + +} diff --git a/lib/System/Unix/Signals.inc b/lib/System/Unix/Signals.inc new file mode 100644 index 0000000..d1493a2 --- /dev/null +++ b/lib/System/Unix/Signals.inc @@ -0,0 +1,201 @@ +//===- Signals.cpp - Generic Unix Signals Implementation -----*- C++ -*-===// +//  +//                     The LLVM Compiler Infrastructure +// +// This file was developed by the LLVM research group and is distributed under +// the University of Illinois Open Source License. See LICENSE.TXT for details. +//  +//===----------------------------------------------------------------------===// +// +// This file defines some helpful functions for dealing with the possibility of +// Unix signals occuring while your program is running. +// +//===----------------------------------------------------------------------===// + +#include "Unix.h" +#include <vector> +#include <algorithm> +#if HAVE_EXECINFO_H +# include <execinfo.h>         // For backtrace(). +#endif +#if HAVE_SIGNAL_H +#include <signal.h> +#endif +#if HAVE_SYS_STAT_H +#include <sys/stat.h> +#endif +using namespace llvm; + +namespace { + +bool StackTraceRequested = false;  + +/// InterruptFunction - The function to call if ctrl-c is pressed. +void (*InterruptFunction)() = 0; + +std::vector<sys::Path> *FilesToRemove = 0 ; +std::vector<sys::Path> *DirectoriesToRemove = 0; + +// IntSigs - Signals that may interrupt the program at any time. +const int IntSigs[] = { +  SIGHUP, SIGINT, SIGQUIT, SIGPIPE, SIGTERM, SIGUSR1, SIGUSR2 +}; +const int *IntSigsEnd = IntSigs + sizeof(IntSigs)/sizeof(IntSigs[0]); + +// KillSigs - Signals that are synchronous with the program that will cause it +// to die. +const int KillSigs[] = { +  SIGILL, SIGTRAP, SIGABRT, SIGFPE, SIGBUS, SIGSEGV, SIGSYS, SIGXCPU, SIGXFSZ +#ifdef SIGEMT +  , SIGEMT +#endif +}; +const int *KillSigsEnd = KillSigs + sizeof(KillSigs)/sizeof(KillSigs[0]); + +#ifdef HAVE_BACKTRACE +void* StackTrace[256]; +#endif + +// PrintStackTrace - In the case of a program crash or fault, print out a stack +// trace so that the user has an indication of why and where we died. +// +// On glibc systems we have the 'backtrace' function, which works nicely, but +// doesn't demangle symbols.  In order to backtrace symbols, we fork and exec a +// 'c++filt' process to do the demangling.  This seems like the simplest and +// most robust solution when we can't allocate memory (such as in a signal +// handler).  If we can't find 'c++filt', we fallback to printing mangled names. +// +void PrintStackTrace() { +#ifdef HAVE_BACKTRACE +  // Use backtrace() to output a backtrace on Linux systems with glibc. +  int depth = backtrace(StackTrace, sizeof(StackTrace)/sizeof(StackTrace[0])); +   +  // Create a one-way unix pipe.  The backtracing process writes to PipeFDs[1], +  // the c++filt process reads from PipeFDs[0]. +  int PipeFDs[2]; +  if (pipe(PipeFDs)) { +    backtrace_symbols_fd(StackTrace, depth, STDERR_FILENO); +    return; +  } + +  switch (pid_t ChildPID = fork()) { +  case -1:        // Error forking, print mangled stack trace +    close(PipeFDs[0]); +    close(PipeFDs[1]); +    backtrace_symbols_fd(StackTrace, depth, STDERR_FILENO); +    return; +  default:        // backtracing process +    close(PipeFDs[0]);  // Close the reader side. + +    // Print the mangled backtrace into the pipe. +    backtrace_symbols_fd(StackTrace, depth, PipeFDs[1]); +    close(PipeFDs[1]);   // We are done writing. +    while (waitpid(ChildPID, 0, 0) == -1) +      if (errno != EINTR) break; +    return; + +  case 0:         // c++filt process +    close(PipeFDs[1]);    // Close the writer side. +    dup2(PipeFDs[0], 0);  // Read from standard input +    close(PipeFDs[0]);    // Close the old descriptor +    dup2(2, 1);           // Revector stdout -> stderr + +    // Try to run c++filt or gc++filt.  If neither is found, call back on 'cat' +    // to print the mangled stack trace.  If we can't find cat, just exit. +    execlp("c++filt", "c++filt", (char*)NULL); +    execlp("gc++filt", "gc++filt", (char*)NULL); +    execlp("cat", "cat", (char*)NULL); +    execlp("/bin/cat", "cat", (char*)NULL); +    exit(0); +  } +#endif +} + +// SignalHandler - The signal handler that runs... +RETSIGTYPE SignalHandler(int Sig) { +  if (FilesToRemove != 0) +    while (!FilesToRemove->empty()) { +      FilesToRemove->back().eraseFromDisk(true); +      FilesToRemove->pop_back(); +    } + +  if (DirectoriesToRemove != 0) +    while (!DirectoriesToRemove->empty()) { +      DirectoriesToRemove->back().eraseFromDisk(true); +      DirectoriesToRemove->pop_back(); +    } + +  if (std::find(IntSigs, IntSigsEnd, Sig) != IntSigsEnd) { +    if (InterruptFunction) { +      void (*IF)() = InterruptFunction; +      InterruptFunction = 0; +      IF();        // run the interrupt function. +      return; +    } else { +      exit(1);   // If this is an interrupt signal, exit the program +    } +  } + +  // Otherwise if it is a fault (like SEGV) output the stacktrace to +  // STDERR (if we can) and reissue the signal to die... +  if (StackTraceRequested) +    PrintStackTrace(); +  signal(Sig, SIG_DFL); +} + +// Just call signal +void RegisterHandler(int Signal) {  +  signal(Signal, SignalHandler);  +} + +} + + +void sys::SetInterruptFunction(void (*IF)()) { +  InterruptFunction = IF; +  RegisterHandler(SIGINT); +} + +// RemoveFileOnSignal - The public API +bool sys::RemoveFileOnSignal(const sys::Path &Filename, std::string* ErrMsg) { +  if (FilesToRemove == 0) +    FilesToRemove = new std::vector<sys::Path>; + +  FilesToRemove->push_back(Filename); + +  std::for_each(IntSigs, IntSigsEnd, RegisterHandler); +  std::for_each(KillSigs, KillSigsEnd, RegisterHandler); +  return false; +} + +// RemoveDirectoryOnSignal - The public API +bool sys::RemoveDirectoryOnSignal(const sys::Path& path, std::string* ErrMsg) { +  // Not a directory? +  struct stat buf; +  if (0 != stat(path.c_str(), &buf)) { +    MakeErrMsg(ErrMsg, path.toString() + ": can't get status of file"); +    return true; +  } + +  if (!S_ISDIR(buf.st_mode)) { +    if (ErrMsg) +      *ErrMsg = path.toString() + " is not a directory"; +    return true; +  } + +  if (DirectoriesToRemove == 0) +    DirectoriesToRemove = new std::vector<sys::Path>; + +  DirectoriesToRemove->push_back(path); + +  std::for_each(IntSigs, IntSigsEnd, RegisterHandler); +  std::for_each(KillSigs, KillSigsEnd, RegisterHandler); +  return false; +} + +/// PrintStackTraceOnErrorSignal - When an error signal (such as SIBABRT or +/// SIGSEGV) is delivered to the process, print a stack trace and then exit. +void sys::PrintStackTraceOnErrorSignal() { +  StackTraceRequested = true; +  std::for_each(KillSigs, KillSigsEnd, RegisterHandler); +} diff --git a/lib/System/Unix/TimeValue.inc b/lib/System/Unix/TimeValue.inc new file mode 100644 index 0000000..77fc9ab --- /dev/null +++ b/lib/System/Unix/TimeValue.inc @@ -0,0 +1,56 @@ +//===- Unix/TimeValue.cpp - Unix TimeValue Implementation -------*- C++ -*-===// +//  +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Reid Spencer and is distributed under the  +// University of Illinois Open Source License. See LICENSE.TXT for details. +//  +//===----------------------------------------------------------------------===// +// +// This file implements the Unix specific portion of the TimeValue class. +// +//===----------------------------------------------------------------------===// + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only generic UNIX code that +//===          is guaranteed to work on *all* UNIX variants. +//===----------------------------------------------------------------------===// + +#include "Unix.h" + +namespace llvm { +  using namespace sys; + +std::string TimeValue::toString() const { +  char buffer[32]; + +  time_t ourTime = time_t(this->toEpochTime()); +#ifdef __hpux +// note that the following line needs -D_REENTRANT on HP-UX to be picked up  +  asctime_r(localtime(&ourTime), buffer); +#else +  ::asctime_r(::localtime(&ourTime), buffer); +#endif + +  std::string result(buffer); +  return result.substr(0,24); +} + +TimeValue TimeValue::now() { +  struct timeval the_time; +  timerclear(&the_time); +  if (0 != ::gettimeofday(&the_time,0)) { +    // This is *really* unlikely to occur because the only gettimeofday +    // errors concern the timezone parameter which we're passing in as 0. +    // In the unlikely case it does happen, just return MinTime, no error +    // message needed.  +    return MinTime; +  } + +  return TimeValue( +    static_cast<TimeValue::SecondsType>( the_time.tv_sec ),  +    static_cast<TimeValue::NanoSecondsType>( the_time.tv_usec *  +      NANOSECONDS_PER_MICROSECOND ) ); +} + +} diff --git a/lib/System/Unix/Unix.h b/lib/System/Unix/Unix.h new file mode 100644 index 0000000..c38c652 --- /dev/null +++ b/lib/System/Unix/Unix.h @@ -0,0 +1,101 @@ +//===- llvm/System/Unix/Unix.h - Common Unix Include File -------*- C++ -*-===// +// +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Reid Spencer and is distributed under the +// University of Illinois Open Source License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +// This file defines things specific to Unix implementations. +// +//===----------------------------------------------------------------------===// + +#ifndef LLVM_SYSTEM_UNIX_UNIX_H +#define LLVM_SYSTEM_UNIX_UNIX_H + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only generic UNIX code that +//===          is guaranteed to work on all UNIX variants. +//===----------------------------------------------------------------------===// + +#include "llvm/Config/config.h"     // Get autoconf configuration settings +#include <cstdlib> +#include <cstdio> +#include <cstring> +#include <cerrno> +#include <string> +#include <algorithm> + +#ifdef HAVE_UNISTD_H +#include <unistd.h> +#endif + +#ifdef HAVE_SYS_TYPES_H +#include <sys/types.h> +#endif + +#ifdef HAVE_SYS_PARAM_H +#include <sys/param.h> +#endif + +#ifdef HAVE_ASSERT_H +#include <assert.h> +#endif + +#ifdef TIME_WITH_SYS_TIME +# include <sys/time.h> +# include <time.h> +#else +# ifdef HAVE_SYS_TIME_H +#  include <sys/time.h> +# else +#  include <time.h> +# endif +#endif + +#ifdef HAVE_SYS_WAIT_H +# include <sys/wait.h> +#endif + +#ifndef WEXITSTATUS +# define WEXITSTATUS(stat_val) ((unsigned)(stat_val) >> 8) +#endif + +#ifndef WIFEXITED +# define WIFEXITED(stat_val) (((stat_val) & 255) == 0) +#endif + +/// This function builds an error message into \p ErrMsg using the \p prefix +/// string and the Unix error number given by \p errnum. If errnum is -1, the +/// default then the value of errno is used. +/// @brief Make an error message +inline bool MakeErrMsg( +  std::string* ErrMsg, const std::string& prefix, int errnum = -1) { +  if (!ErrMsg) +    return true; +  char buffer[MAXPATHLEN]; +  buffer[0] = 0; +  if (errnum == -1) +    errnum = errno; +#ifdef HAVE_STRERROR_R +  // strerror_r is thread-safe. +  if (errnum) +    strerror_r(errnum,buffer,MAXPATHLEN-1); +#elif HAVE_STRERROR +  // Copy the thread un-safe result of strerror into +  // the buffer as fast as possible to minimize impact +  // of collision of strerror in multiple threads. +  if (errnum) +    strncpy(buffer,strerror(errnum),MAXPATHLEN-1); +  buffer[MAXPATHLEN-1] = 0; +#else +  // Strange that this system doesn't even have strerror +  // but, oh well, just use a generic message +  sprintf(buffer, "Error #%d", errnum); +#endif +  *ErrMsg = prefix + buffer; +  return true; +} + +#endif diff --git a/lib/System/Win32/Alarm.inc b/lib/System/Win32/Alarm.inc new file mode 100644 index 0000000..e4ac512 --- /dev/null +++ b/lib/System/Win32/Alarm.inc @@ -0,0 +1,36 @@ +//===-- Alarm.inc - Implement Win32 Alarm Support -------------------------===// +// +//                     The LLVM Compiler Infrastructure +// +// This file was developed by the Reid Spencer and is distributed under the +// University of Illinois Open Source License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +// This file implements the Win32 Alarm support. +// +//===----------------------------------------------------------------------===// + +#include <cassert> +using namespace llvm; + +/// NestedSOI - Sanity check.  Alarms cannot be nested or run in parallel.   +/// This ensures that they never do. +static bool NestedSOI = false; + +void sys::SetupAlarm(unsigned seconds) { +  assert(!NestedSOI && "sys::SetupAlarm calls cannot be nested!"); +  NestedSOI = true; +  // FIXME: Implement for Win32 +} + +void sys::TerminateAlarm() { +  assert(NestedSOI && "sys::TerminateAlarm called without sys::SetupAlarm!"); +  // FIXME: Implement for Win32 +  NestedSOI = false; +} + +int sys::AlarmStatus() { +  // FIXME: Implement for Win32 +  return 0; +} diff --git a/lib/System/Win32/DynamicLibrary.inc b/lib/System/Win32/DynamicLibrary.inc new file mode 100644 index 0000000..251131e --- /dev/null +++ b/lib/System/Win32/DynamicLibrary.inc @@ -0,0 +1,169 @@ +//===- Win32/DynamicLibrary.cpp - Win32 DL Implementation -------*- C++ -*-===// +//  +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Jeff Cohen and is distributed under the  +// University of Illinois Open Source License. See LICENSE.TXT for details. +//  +//===----------------------------------------------------------------------===// +// +// This file provides the Win32 specific implementation of DynamicLibrary. +// +//===----------------------------------------------------------------------===// + +#include "Win32.h" + +#ifdef __MINGW32__ + #include <imagehlp.h> +#else + #include <dbghelp.h> +#endif + +#ifdef __MINGW32__ + #if (HAVE_LIBIMAGEHLP != 1) +  #error "libimagehlp.a should be present" + #endif +#else + #pragma comment(lib, "dbghelp.lib") +#endif + +namespace llvm { +using namespace sys; + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only Win32 specific code  +//===          and must not be UNIX code. +//===----------------------------------------------------------------------===// + +static std::vector<HMODULE> OpenedHandles; + +extern "C" { +  static BOOL CALLBACK ELM_Callback(PSTR  ModuleName, +                                    ULONG ModuleBase, +                                    ULONG ModuleSize, +                                    PVOID UserContext) +  { +    // Ignore VC++ runtimes prior to 7.1.  Somehow some of them get loaded +    // into the process. +    if (stricmp(ModuleName, "msvci70") != 0 && +        stricmp(ModuleName, "msvcirt") != 0 && +        stricmp(ModuleName, "msvcp50") != 0 && +        stricmp(ModuleName, "msvcp60") != 0 && +        stricmp(ModuleName, "msvcp70") != 0 && +        stricmp(ModuleName, "msvcr70") != 0 && +#ifndef __MINGW32__ +        // Mingw32 uses msvcrt.dll by default. Don't ignore it. +        // Otherwise, user should be aware, what he's doing :) +        stricmp(ModuleName, "msvcrt") != 0 && +#endif         +        stricmp(ModuleName, "msvcrt20") != 0 && +        stricmp(ModuleName, "msvcrt40") != 0) { +      OpenedHandles.push_back((HMODULE)ModuleBase); +    } +    return TRUE; +  } +} + +DynamicLibrary::DynamicLibrary() : handle(0) { +  handle = GetModuleHandle(NULL); +  OpenedHandles.push_back((HMODULE)handle); +} + +DynamicLibrary::~DynamicLibrary() { +  if (handle == 0) +    return; + +  // GetModuleHandle() does not increment the ref count, so we must not free +  // the handle to the executable. +  if (handle != GetModuleHandle(NULL)) +    FreeLibrary((HMODULE)handle); +  handle = 0; + +  for (std::vector<HMODULE>::iterator I = OpenedHandles.begin(), +       E = OpenedHandles.end(); I != E; ++I) { +    if (*I == handle) { +      // Note: don't use the swap/pop_back trick here. Order is important. +      OpenedHandles.erase(I); +    } +  } +} + +// Stack probing routines are in the support library (e.g. libgcc), but we don't +// have dynamic linking on windows. Provide a hook. +#if defined(__MINGW32__) || defined (_MSC_VER) +  #define EXPLICIT_SYMBOL(SYM)                    \ +    if (!strcmp(symbolName, #SYM)) return (void*)&SYM +  #define EXPLICIT_SYMBOL2(SYMFROM, SYMTO)        \ +    if (!strcmp(symbolName, #SYMFROM)) return (void*)&SYMTO +  #define EXPLICIT_SYMBOL_DEF(SYM)                \ +    extern "C" { extern void *SYM; } + +  #if defined(__MINGW32__) +    EXPLICIT_SYMBOL_DEF(_alloca); +  #elif defined(_MSC_VER) +    EXPLICIT_SYMBOL_DEF(_alloca_probe); +  #endif +#endif +  +bool DynamicLibrary::LoadLibraryPermanently(const char *filename, +                                            std::string *ErrMsg) { +  if (filename) { +    HMODULE a_handle = LoadLibrary(filename); + +    if (a_handle == 0) +      return MakeErrMsg(ErrMsg, std::string(filename) + ": Can't open : "); + +    OpenedHandles.push_back(a_handle); +  } else { +    // When no file is specified, enumerate all DLLs and EXEs in the +    // process. +    EnumerateLoadedModules(GetCurrentProcess(), ELM_Callback, 0); +  } + +  // Because we don't remember the handle, we will never free it; hence, +  // it is loaded permanently. +  return false; +} + +void* DynamicLibrary::SearchForAddressOfSymbol(const char* symbolName) { +  // First check symbols added via AddSymbol(). +  std::map<std::string, void *>::iterator I = g_symbols.find(symbolName); +  if (I != g_symbols.end()) +    return I->second; + +  // Now search the libraries. +  for (std::vector<HMODULE>::iterator I = OpenedHandles.begin(), +       E = OpenedHandles.end(); I != E; ++I) { +    FARPROC ptr = GetProcAddress((HMODULE)*I, symbolName); +    if (ptr) +      return (void *) ptr; +  } + +#if defined(__MINGW32__) +  { +    EXPLICIT_SYMBOL(_alloca); +    EXPLICIT_SYMBOL2(alloca, _alloca); +#undef EXPLICIT_SYMBOL +#undef EXPLICIT_SYMBOL2 +#undef EXPLICIT_SYMBOL_DEF     +  } +#elif defined(_MSC_VER) +  { +    EXPLICIT_SYMBOL2(alloca, _alloca_probe); +    EXPLICIT_SYMBOL2(_alloca, _alloca_probe); +#undef EXPLICIT_SYMBOL +#undef EXPLICIT_SYMBOL2 +#undef EXPLICIT_SYMBOL_DEF     +  }   +#endif + +  return 0; +} + +void *DynamicLibrary::GetAddressOfSymbol(const char *symbolName) { +  assert(handle != 0 && "Invalid DynamicLibrary handle"); +  return (void *) GetProcAddress((HMODULE)handle, symbolName); +} + +} + diff --git a/lib/System/Win32/MappedFile.inc b/lib/System/Win32/MappedFile.inc new file mode 100644 index 0000000..877ff52 --- /dev/null +++ b/lib/System/Win32/MappedFile.inc @@ -0,0 +1,140 @@ +//===- Win32/MappedFile.cpp - Win32 MappedFile Implementation ---*- C++ -*-===// +//  +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Jeff Cohen and is distributed under the  +// University of Illinois Open Source License. See LICENSE.TXT for details. +//  +//===----------------------------------------------------------------------===// +// +// This file provides the Win32 implementation of the MappedFile concept. +// +//===----------------------------------------------------------------------===// + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only Win32 code. +//===----------------------------------------------------------------------===// + +#include "Win32.h" +#include "llvm/System/Process.h" + +namespace llvm { +using namespace sys; + +struct sys::MappedFileInfo { +  HANDLE hFile; +  HANDLE hMapping; +  size_t size; +}; + +bool MappedFile::initialize(std::string* ErrMsg) { +  assert(!info_); +  info_ = new MappedFileInfo; +  info_->hFile = INVALID_HANDLE_VALUE; +  info_->hMapping = NULL; + +  DWORD mode = options_ & WRITE_ACCESS ? GENERIC_WRITE : GENERIC_READ; +  DWORD disposition = options_ & WRITE_ACCESS ? OPEN_ALWAYS : OPEN_EXISTING; +  DWORD share = options_ & WRITE_ACCESS ? FILE_SHARE_WRITE : FILE_SHARE_READ; +  share = options_ & SHARED_MAPPING ? share : 0; +  info_->hFile = CreateFile(path_.c_str(), mode, share, NULL, disposition, +                            FILE_ATTRIBUTE_NORMAL, NULL); +  if (info_->hFile == INVALID_HANDLE_VALUE) { +    delete info_; +    info_ = NULL; +    return MakeErrMsg(ErrMsg, +      std::string("Can't open file: ") + path_.toString()); +  } + +  LARGE_INTEGER size; +  if (!GetFileSizeEx(info_->hFile, &size) || +      (info_->size = size_t(size.QuadPart), info_->size != size.QuadPart)) { +    CloseHandle(info_->hFile); +    delete info_; +    info_ = NULL; +    return MakeErrMsg(ErrMsg,  +      std::string("Can't get size of file: ") + path_.toString()); +  } + +  return false; +} + +void MappedFile::terminate() { +  unmap(); +  if (info_->hFile != INVALID_HANDLE_VALUE) +    CloseHandle(info_->hFile); +  delete info_; +  info_ = NULL; +} + +void MappedFile::unmap() { +  assert(info_ && "MappedFile not initialized"); +  if (isMapped()) { +    UnmapViewOfFile(base_); +    base_ = NULL; +  } +  if (info_->hMapping != INVALID_HANDLE_VALUE) { +    CloseHandle(info_->hMapping); +    info_->hMapping = NULL; +  } +} + +void* MappedFile::map(std::string* ErrMsg) { +  if (!isMapped()) { +    DWORD prot = PAGE_READONLY; +    if (options_ & EXEC_ACCESS) +      prot = SEC_IMAGE; +    else if (options_ & WRITE_ACCESS) +      prot = PAGE_READWRITE; +    info_->hMapping = CreateFileMapping(info_->hFile, NULL, prot, 0, 0, NULL); +    if (info_->hMapping == NULL) { +      MakeErrMsg(ErrMsg, std::string("Can't map file: ") + path_.toString()); +      return 0; +    } + +    prot = (options_ & WRITE_ACCESS) ? FILE_MAP_WRITE : FILE_MAP_READ; +    base_ = MapViewOfFileEx(info_->hMapping, prot, 0, 0, 0, NULL); +    if (base_ == NULL) { +      CloseHandle(info_->hMapping); +      info_->hMapping = NULL; +      MakeErrMsg(ErrMsg, std::string("Can't map file: ") + path_.toString()); +      return 0; +    } +  } +  return base_; +} + +size_t MappedFile::size() const { +  assert(info_ && "MappedFile not initialized"); +  return info_->size; +} + +bool MappedFile::size(size_t new_size, std::string* ErrMsg) { +  assert(info_ && "MappedFile not initialized"); + +  // Take the mapping out of memory. +  unmap(); + +  // Adjust the new_size to a page boundary. +  size_t pagesizem1 = Process::GetPageSize() - 1; +  new_size = (new_size + pagesizem1) & ~pagesizem1; + +  // If the file needs to be extended, do so. +  if (new_size > info_->size) { +    LARGE_INTEGER eof; +    eof.QuadPart = new_size; +    if (!SetFilePointerEx(info_->hFile, eof, NULL, FILE_BEGIN)) +      return MakeErrMsg(ErrMsg,  +        std::string("Can't set end of file: ") + path_.toString()); +    if (!SetEndOfFile(info_->hFile)) +      return MakeErrMsg(ErrMsg,  +        std::string("Can't set end of file: ") + path_.toString()); +    info_->size = new_size; +  } + +  // Remap the file. +  return map(ErrMsg); +} + +} + diff --git a/lib/System/Win32/Memory.inc b/lib/System/Win32/Memory.inc new file mode 100644 index 0000000..739d530 --- /dev/null +++ b/lib/System/Win32/Memory.inc @@ -0,0 +1,57 @@ +//===- Win32/Memory.cpp - Win32 Memory Implementation -----------*- C++ -*-===// +//  +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Jeff Cohen and is distributed under the  +// University of Illinois Open Source License. See LICENSE.TXT for details. +//  +//===----------------------------------------------------------------------===// +// +// This file provides the Win32 specific implementation of various Memory +// management utilities +// +//===----------------------------------------------------------------------===// + +#include "Win32.h" +#include "llvm/System/Process.h" + +namespace llvm { +using namespace sys; + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only Win32 specific code  +//===          and must not be UNIX code +//===----------------------------------------------------------------------===// + +MemoryBlock Memory::AllocateRWX(unsigned NumBytes, +                                const MemoryBlock *NearBlock, +                                std::string *ErrMsg) { +  if (NumBytes == 0) return MemoryBlock(); + +  static const long pageSize = Process::GetPageSize(); +  unsigned NumPages = (NumBytes+pageSize-1)/pageSize; + +  //FIXME: support NearBlock if ever needed on Win64. + +  void *pa = VirtualAlloc(NULL, NumPages*pageSize, MEM_COMMIT, +                  PAGE_EXECUTE_READWRITE); +  if (pa == NULL) { +    MakeErrMsg(ErrMsg, "Can't allocate RWX Memory: "); +    return MemoryBlock(); +  } + +  MemoryBlock result; +  result.Address = pa; +  result.Size = NumPages*pageSize; +  return result; +} + +bool Memory::ReleaseRWX(MemoryBlock &M, std::string *ErrMsg) { +  if (M.Address == 0 || M.Size == 0) return false; +  if (!VirtualFree(M.Address, 0, MEM_RELEASE)) +    return MakeErrMsg(ErrMsg, "Can't release RWX Memory: "); +  return false; +} + +} + diff --git a/lib/System/Win32/Mutex.inc b/lib/System/Win32/Mutex.inc new file mode 100644 index 0000000..0a4bc4b --- /dev/null +++ b/lib/System/Win32/Mutex.inc @@ -0,0 +1,58 @@ +//===- llvm/System/Win32/Mutex.inc - Win32 Mutex Implementation -*- C++ -*-===// +//  +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Jeff Cohen and is distributed under the  +// University of Illinois Open Source License. See LICENSE.TXT for details. +//  +//===----------------------------------------------------------------------===// +// +// This file implements the Win32 specific (non-pthread) Mutex class. +// +//===----------------------------------------------------------------------===// + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only generic Win32 code that +//===          is guaranteed to work on *all* Win32 variants. +//===----------------------------------------------------------------------===// + +#include "Win32.h" +#include "llvm/System/Mutex.h" + +namespace llvm { +using namespace sys; + +Mutex::Mutex(bool /*recursive*/) +{ +  data_ = new CRITICAL_SECTION; +  InitializeCriticalSection((LPCRITICAL_SECTION)data_); +} + +Mutex::~Mutex() +{ +  DeleteCriticalSection((LPCRITICAL_SECTION)data_); +  delete (LPCRITICAL_SECTION)data_; +  data_ = 0; +} + +bool  +Mutex::acquire() +{ +  EnterCriticalSection((LPCRITICAL_SECTION)data_); +  return true; +} + +bool  +Mutex::release() +{ +  LeaveCriticalSection((LPCRITICAL_SECTION)data_); +  return true; +} + +bool  +Mutex::tryacquire() +{ +  return TryEnterCriticalSection((LPCRITICAL_SECTION)data_); +} + +} diff --git a/lib/System/Win32/Path.inc b/lib/System/Win32/Path.inc new file mode 100644 index 0000000..994bc67 --- /dev/null +++ b/lib/System/Win32/Path.inc @@ -0,0 +1,772 @@ +//===- llvm/System/Linux/Path.cpp - Linux Path Implementation ---*- C++ -*-===// +// +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Reid Spencer and is distributed under the +// University of Illinois Open Source License. See LICENSE.TXT for details. +// +// Modified by Henrik Bach to comply with at least MinGW. +// Ported to Win32 by Jeff Cohen. +// +//===----------------------------------------------------------------------===// +// +// This file provides the Win32 specific implementation of the Path class. +// +//===----------------------------------------------------------------------===// + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only generic Win32 code that +//===          is guaranteed to work on *all* Win32 variants. +//===----------------------------------------------------------------------===// + +#include "Win32.h" +#include <malloc.h> + +// We need to undo a macro defined in Windows.h, otherwise we won't compile: +#undef CopyFile + +// Windows happily accepts either forward or backward slashes, though any path +// returned by a Win32 API will have backward slashes.  As LLVM code basically +// assumes forward slashes are used, backward slashs are converted where they +// can be introduced into a path. +// +// Another invariant is that a path ends with a slash if and only if the path +// is a root directory.  Any other use of a trailing slash is stripped.  Unlike +// in Unix, Windows has a rather complicated notion of a root path and this +// invariant helps simply the code. + +static void FlipBackSlashes(std::string& s) { +  for (size_t i = 0; i < s.size(); i++) +    if (s[i] == '\\') +      s[i] = '/'; +} + +namespace llvm { +namespace sys { + +bool +Path::isValid() const { +  if (path.empty()) +    return false; + +  // If there is a colon, it must be the second character, preceded by a letter +  // and followed by something. +  size_t len = path.size(); +  size_t pos = path.rfind(':',len); +  size_t rootslash = 0; +  if (pos != std::string::npos) { +    if (pos != 1 || !isalpha(path[0]) || len < 3) +      return false; +      rootslash = 2; +  } + +  // Look for a UNC path, and if found adjust our notion of the root slash. +  if (len > 3 && path[0] == '/' && path[1] == '/') { +    rootslash = path.find('/', 2); +    if (rootslash == std::string::npos) +      rootslash = 0; +  } + +  // Check for illegal characters. +  if (path.find_first_of("\\<>\"|\001\002\003\004\005\006\007\010\011\012" +                         "\013\014\015\016\017\020\021\022\023\024\025\026" +                         "\027\030\031\032\033\034\035\036\037") +      != std::string::npos) +    return false; + +  // Remove trailing slash, unless it's a root slash. +  if (len > rootslash+1 && path[len-1] == '/') +    path.erase(--len); + +  // Check each component for legality. +  for (pos = 0; pos < len; ++pos) { +    // A component may not end in a space. +    if (path[pos] == ' ') { +      if (path[pos+1] == '/' || path[pos+1] == '\0') +        return false; +    } + +    // A component may not end in a period. +    if (path[pos] == '.') { +      if (path[pos+1] == '/' || path[pos+1] == '\0') { +        // Unless it is the pseudo-directory "."... +        if (pos == 0 || path[pos-1] == '/' || path[pos-1] == ':') +          return true; +        // or "..". +        if (pos > 0 && path[pos-1] == '.') { +          if (pos == 1 || path[pos-2] == '/' || path[pos-2] == ':') +            return true; +        } +        return false; +      } +    } +  } + +  return true; +} + +bool  +Path::isAbsolute() const { +  switch (path.length()) { +    case 0: +      return false; +    case 1: +    case 2: +      return path[0] == '/'; +    default: +      return path[0] == '/' || (path[1] == ':' && path[2] == '/'); +  } +}  + +static Path *TempDirectory = NULL; + +Path +Path::GetTemporaryDirectory(std::string* ErrMsg) { +  if (TempDirectory) +    return *TempDirectory; + +  char pathname[MAX_PATH]; +  if (!GetTempPath(MAX_PATH, pathname)) { +    if (ErrMsg) +      *ErrMsg = "Can't determine temporary directory"; +    return Path(); +  } + +  Path result; +  result.set(pathname); + +  // Append a subdirectory passed on our process id so multiple LLVMs don't +  // step on each other's toes. +#ifdef __MINGW32__ +  // Mingw's Win32 header files are broken. +  sprintf(pathname, "LLVM_%u", unsigned(GetCurrentProcessId())); +#else +  sprintf(pathname, "LLVM_%u", GetCurrentProcessId()); +#endif +  result.appendComponent(pathname); + +  // If there's a directory left over from a previous LLVM execution that +  // happened to have the same process id, get rid of it. +  result.eraseFromDisk(true); + +  // And finally (re-)create the empty directory. +  result.createDirectoryOnDisk(false); +  TempDirectory = new Path(result); +  return *TempDirectory; +} + +// FIXME: the following set of functions don't map to Windows very well. +Path +Path::GetRootDirectory() { +  Path result; +  result.set("C:/"); +  return result; +} + +static void getPathList(const char*path, std::vector<sys::Path>& Paths) { +  const char* at = path; +  const char* delim = strchr(at, ';'); +  Path tmpPath; +  while (delim != 0) { +    std::string tmp(at, size_t(delim-at)); +    if (tmpPath.set(tmp)) +      if (tmpPath.canRead()) +        Paths.push_back(tmpPath); +    at = delim + 1; +    delim = strchr(at, ';'); +  } + +  if (*at != 0) +    if (tmpPath.set(std::string(at))) +      if (tmpPath.canRead()) +        Paths.push_back(tmpPath); +} + +void +Path::GetSystemLibraryPaths(std::vector<sys::Path>& Paths) { +  Paths.push_back(sys::Path("C:/WINDOWS/SYSTEM32")); +  Paths.push_back(sys::Path("C:/WINDOWS")); +} + +void +Path::GetBitcodeLibraryPaths(std::vector<sys::Path>& Paths) { +  char * env_var = getenv("LLVM_LIB_SEARCH_PATH"); +  if (env_var != 0) { +    getPathList(env_var,Paths); +  } +#ifdef LLVM_LIBDIR +  { +    Path tmpPath; +    if (tmpPath.set(LLVM_LIBDIR)) +      if (tmpPath.canRead()) +        Paths.push_back(tmpPath); +  } +#endif +  GetSystemLibraryPaths(Paths); +} + +Path +Path::GetLLVMDefaultConfigDir() { +  // TODO: this isn't going to fly on Windows +  return Path("/etc/llvm"); +} + +Path +Path::GetUserHomeDirectory() { +  // TODO: Typical Windows setup doesn't define HOME. +  const char* home = getenv("HOME"); +  if (home) { +    Path result; +    if (result.set(home)) +      return result; +  } +  return GetRootDirectory(); +} +// FIXME: the above set of functions don't map to Windows very well. + + +bool +Path::isRootDirectory() const { +  size_t len = path.size(); +  return len > 0 && path[len-1] == '/'; +} + +std::string +Path::getBasename() const { +  // Find the last slash +  size_t slash = path.rfind('/'); +  if (slash == std::string::npos) +    slash = 0; +  else +    slash++; + +  size_t dot = path.rfind('.'); +  if (dot == std::string::npos || dot < slash) +    return path.substr(slash); +  else +    return path.substr(slash, dot - slash); +} + +bool +Path::exists() const { +  DWORD attr = GetFileAttributes(path.c_str()); +  return attr != INVALID_FILE_ATTRIBUTES; +} + +bool +Path::canRead() const { +  // FIXME: take security attributes into account. +  DWORD attr = GetFileAttributes(path.c_str()); +  return attr != INVALID_FILE_ATTRIBUTES; +} + +bool +Path::canWrite() const { +  // FIXME: take security attributes into account. +  DWORD attr = GetFileAttributes(path.c_str()); +  return (attr != INVALID_FILE_ATTRIBUTES) && !(attr & FILE_ATTRIBUTE_READONLY); +} + +bool +Path::canExecute() const { +  // FIXME: take security attributes into account. +  DWORD attr = GetFileAttributes(path.c_str()); +  return attr != INVALID_FILE_ATTRIBUTES; +} + +std::string +Path::getLast() const { +  // Find the last slash +  size_t pos = path.rfind('/'); + +  // Handle the corner cases +  if (pos == std::string::npos) +    return path; + +  // If the last character is a slash, we have a root directory +  if (pos == path.length()-1) +    return path; + +  // Return everything after the last slash +  return path.substr(pos+1); +} + +const FileStatus * +PathWithStatus::getFileStatus(bool update, std::string *ErrStr) const { +  if (!fsIsValid || update) { +    WIN32_FILE_ATTRIBUTE_DATA fi; +    if (!GetFileAttributesEx(path.c_str(), GetFileExInfoStandard, &fi)) { +      MakeErrMsg(ErrStr, "getStatusInfo():" + std::string(path) + +                      ": Can't get status: "); +      return 0; +    } + +    status.fileSize = fi.nFileSizeHigh; +    status.fileSize <<= sizeof(fi.nFileSizeHigh)*8; +    status.fileSize += fi.nFileSizeLow; + +    status.mode = fi.dwFileAttributes & FILE_ATTRIBUTE_READONLY ? 0555 : 0777; +    status.user = 9999;    // Not applicable to Windows, so... +    status.group = 9999;   // Not applicable to Windows, so... + +    // FIXME: this is only unique if the file is accessed by the same file path. +    // How do we do this for C:\dir\file and ..\dir\file ? Unix has inode +    // numbers, but the concept doesn't exist in Windows. +    status.uniqueID = 0; +    for (unsigned i = 0; i < path.length(); ++i) +      status.uniqueID += path[i]; + +    __int64 ft = *reinterpret_cast<__int64*>(&fi.ftLastWriteTime); +    status.modTime.fromWin32Time(ft); + +    status.isDir = fi.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY; +    fsIsValid = true; +  } +  return &status; +} + +bool Path::makeReadableOnDisk(std::string* ErrMsg) { +  // All files are readable on Windows (ignoring security attributes). +  return false; +} + +bool Path::makeWriteableOnDisk(std::string* ErrMsg) { +  DWORD attr = GetFileAttributes(path.c_str()); + +  // If it doesn't exist, we're done. +  if (attr == INVALID_FILE_ATTRIBUTES) +    return false; + +  if (attr & FILE_ATTRIBUTE_READONLY) { +    if (!SetFileAttributes(path.c_str(), attr & ~FILE_ATTRIBUTE_READONLY)) { +      MakeErrMsg(ErrMsg, std::string(path) + ": Can't make file writable: "); +      return true; +    } +  } +  return false; +} + +bool Path::makeExecutableOnDisk(std::string* ErrMsg) { +  // All files are executable on Windows (ignoring security attributes). +  return false; +} + +bool +Path::getDirectoryContents(std::set<Path>& result, std::string* ErrMsg) const { +  WIN32_FILE_ATTRIBUTE_DATA fi; +  if (!GetFileAttributesEx(path.c_str(), GetFileExInfoStandard, &fi)) { +    MakeErrMsg(ErrMsg, path + ": can't get status of file"); +    return true; +  } +     +  if (!(fi.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)) { +    if (ErrMsg) +      *ErrMsg = path + ": not a directory"; +    return true; +  } + +  result.clear(); +  WIN32_FIND_DATA fd; +  std::string searchpath = path; +  if (path.size() == 0 || searchpath[path.size()-1] == '/') +    searchpath += "*"; +  else +    searchpath += "/*"; + +  HANDLE h = FindFirstFile(searchpath.c_str(), &fd); +  if (h == INVALID_HANDLE_VALUE) { +    if (GetLastError() == ERROR_FILE_NOT_FOUND) +      return true; // not really an error, now is it? +    MakeErrMsg(ErrMsg, path + ": Can't read directory: "); +    return true; +  } + +  do { +    if (fd.cFileName[0] == '.') +      continue; +    Path aPath(path); +    aPath.appendComponent(&fd.cFileName[0]); +    result.insert(aPath); +  } while (FindNextFile(h, &fd)); + +  DWORD err = GetLastError(); +  FindClose(h); +  if (err != ERROR_NO_MORE_FILES) { +    SetLastError(err); +    MakeErrMsg(ErrMsg, path + ": Can't read directory: "); +    return true; +  } +  return false; +} + +bool +Path::set(const std::string& a_path) { +  if (a_path.size() == 0) +    return false; +  std::string save(path); +  path = a_path; +  FlipBackSlashes(path); +  if (!isValid()) { +    path = save; +    return false; +  } +  return true; +} + +bool +Path::appendComponent(const std::string& name) { +  if (name.empty()) +    return false; +  std::string save(path); +  if (!path.empty()) { +    size_t last = path.size() - 1; +    if (path[last] != '/') +      path += '/'; +  } +  path += name; +  if (!isValid()) { +    path = save; +    return false; +  } +  return true; +} + +bool +Path::eraseComponent() { +  size_t slashpos = path.rfind('/',path.size()); +  if (slashpos == path.size() - 1 || slashpos == std::string::npos) +    return false; +  std::string save(path); +  path.erase(slashpos); +  if (!isValid()) { +    path = save; +    return false; +  } +  return true; +} + +bool +Path::appendSuffix(const std::string& suffix) { +  std::string save(path); +  path.append("."); +  path.append(suffix); +  if (!isValid()) { +    path = save; +    return false; +  } +  return true; +} + +bool +Path::eraseSuffix() { +  size_t dotpos = path.rfind('.',path.size()); +  size_t slashpos = path.rfind('/',path.size()); +  if (dotpos != std::string::npos) { +    if (slashpos == std::string::npos || dotpos > slashpos+1) { +      std::string save(path); +      path.erase(dotpos, path.size()-dotpos); +      if (!isValid()) { +        path = save; +        return false; +      } +      return true; +    } +  } +  return false; +} + +inline bool PathMsg(std::string* ErrMsg, const char* pathname, const char*msg) { +  if (ErrMsg) +    *ErrMsg = std::string(pathname) + ": " + std::string(msg); +  return true; +} + +bool +Path::createDirectoryOnDisk(bool create_parents, std::string* ErrMsg) { +  // Get a writeable copy of the path name +  size_t len = path.length(); +  char *pathname = reinterpret_cast<char *>(_alloca(len+2)); +  path.copy(pathname, len); +  pathname[len] = 0; + +  // Make sure it ends with a slash. +  if (len == 0 || pathname[len - 1] != '/') { +    pathname[len] = '/'; +    pathname[++len] = 0; +  } + +  // Determine starting point for initial / search. +  char *next = pathname; +  if (pathname[0] == '/' && pathname[1] == '/') { +    // Skip host name. +    next = strchr(pathname+2, '/'); +    if (next == NULL) +      return PathMsg(ErrMsg, pathname, "badly formed remote directory"); + +    // Skip share name. +    next = strchr(next+1, '/'); +    if (next == NULL) +      return PathMsg(ErrMsg, pathname,"badly formed remote directory"); + +    next++; +    if (*next == 0) +      return PathMsg(ErrMsg, pathname, "badly formed remote directory"); + +  } else { +    if (pathname[1] == ':') +      next += 2;    // skip drive letter +    if (*next == '/') +      next++;       // skip root directory +  } + +  // If we're supposed to create intermediate directories +  if (create_parents) { +    // Loop through the directory components until we're done +    while (*next) { +      next = strchr(next, '/'); +      *next = 0; +      if (!CreateDirectory(pathname, NULL)) +          return MakeErrMsg(ErrMsg,  +            std::string(pathname) + ": Can't create directory: "); +      *next++ = '/'; +    } +  } else { +    // Drop trailing slash. +    pathname[len-1] = 0; +    if (!CreateDirectory(pathname, NULL)) { +      return MakeErrMsg(ErrMsg, std::string(pathname) + ": Can't create directory: "); +    } +  } +  return false; +} + +bool +Path::createFileOnDisk(std::string* ErrMsg) { +  // Create the file +  HANDLE h = CreateFile(path.c_str(), GENERIC_WRITE, 0, NULL, CREATE_NEW, +                        FILE_ATTRIBUTE_NORMAL, NULL); +  if (h == INVALID_HANDLE_VALUE) +    return MakeErrMsg(ErrMsg, path + ": Can't create file: "); + +  CloseHandle(h); +  return false; +} + +bool +Path::eraseFromDisk(bool remove_contents, std::string *ErrStr) const { +  WIN32_FILE_ATTRIBUTE_DATA fi; +  if (!GetFileAttributesEx(path.c_str(), GetFileExInfoStandard, &fi)) +    return true; +     +  if (fi.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) { +    // If it doesn't exist, we're done. +    if (!exists()) +      return false; + +    char *pathname = reinterpret_cast<char *>(_alloca(path.length()+3)); +    int lastchar = path.length() - 1 ; +    path.copy(pathname, lastchar+1); + +    // Make path end with '/*'. +    if (pathname[lastchar] != '/') +      pathname[++lastchar] = '/'; +    pathname[lastchar+1] = '*'; +    pathname[lastchar+2] = 0; + +    if (remove_contents) { +      WIN32_FIND_DATA fd; +      HANDLE h = FindFirstFile(pathname, &fd); + +      // It's a bad idea to alter the contents of a directory while enumerating +      // its contents. So build a list of its contents first, then destroy them. + +      if (h != INVALID_HANDLE_VALUE) { +        std::vector<Path> list; + +        do { +          if (strcmp(fd.cFileName, ".") == 0) +            continue; +          if (strcmp(fd.cFileName, "..") == 0) +            continue; + +          Path aPath(path); +          aPath.appendComponent(&fd.cFileName[0]); +          list.push_back(aPath); +        } while (FindNextFile(h, &fd)); + +        DWORD err = GetLastError(); +        FindClose(h); +        if (err != ERROR_NO_MORE_FILES) { +          SetLastError(err); +          return MakeErrMsg(ErrStr, path + ": Can't read directory: "); +        } + +        for (std::vector<Path>::iterator I = list.begin(); I != list.end(); +             ++I) { +          Path &aPath = *I; +          aPath.eraseFromDisk(true); +        } +      } else { +        if (GetLastError() != ERROR_FILE_NOT_FOUND) +          return MakeErrMsg(ErrStr, path + ": Can't read directory: "); +      } +    } + +    pathname[lastchar] = 0; +    if (!RemoveDirectory(pathname)) +      return MakeErrMsg(ErrStr,  +        std::string(pathname) + ": Can't destroy directory: "); +    return false; +  } else { +    // Read-only files cannot be deleted on Windows.  Must remove the read-only +    // attribute first. +    if (fi.dwFileAttributes & FILE_ATTRIBUTE_READONLY) { +      if (!SetFileAttributes(path.c_str(), +                             fi.dwFileAttributes & ~FILE_ATTRIBUTE_READONLY)) +        return MakeErrMsg(ErrStr, path + ": Can't destroy file: "); +    } + +    if (!DeleteFile(path.c_str())) +      return MakeErrMsg(ErrStr, path + ": Can't destroy file: "); +    return false; +  } +} + +bool Path::getMagicNumber(std::string& Magic, unsigned len) const { +  assert(len < 1024 && "Request for magic string too long"); +  char* buf = (char*) alloca(1 + len); + +  HANDLE h = CreateFile(path.c_str(), +                        GENERIC_READ, +                        FILE_SHARE_READ, +                        NULL, +                        OPEN_EXISTING, +                        FILE_ATTRIBUTE_NORMAL, +                        NULL); +  if (h == INVALID_HANDLE_VALUE) +    return false; + +  DWORD nRead = 0; +  BOOL ret = ReadFile(h, buf, len, &nRead, NULL); +  CloseHandle(h); + +  if (!ret || nRead != len) +    return false; + +  buf[len] = '\0'; +  Magic = buf; +  return true; +} + +bool +Path::renamePathOnDisk(const Path& newName, std::string* ErrMsg) { +  if (!MoveFileEx(path.c_str(), newName.c_str(), MOVEFILE_REPLACE_EXISTING)) +    return MakeErrMsg(ErrMsg, "Can't move '" + path + "' to '" + newName.path  +        + "': "); +  return true; +} + +bool +Path::setStatusInfoOnDisk(const FileStatus &si, std::string *ErrMsg) const { +  // FIXME: should work on directories also. +  if (!si.isFile) { +    return true; +  } +   +  HANDLE h = CreateFile(path.c_str(), +                        FILE_READ_ATTRIBUTES | FILE_WRITE_ATTRIBUTES, +                        FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, +                        NULL, +                        OPEN_EXISTING, +                        FILE_ATTRIBUTE_NORMAL, +                        NULL); +  if (h == INVALID_HANDLE_VALUE) +    return true; + +  BY_HANDLE_FILE_INFORMATION bhfi; +  if (!GetFileInformationByHandle(h, &bhfi)) { +    DWORD err = GetLastError(); +    CloseHandle(h); +    SetLastError(err); +    return MakeErrMsg(ErrMsg, path + ": GetFileInformationByHandle: "); +  } + +  FILETIME ft; +  (uint64_t&)ft = si.modTime.toWin32Time(); +  BOOL ret = SetFileTime(h, NULL, &ft, &ft); +  DWORD err = GetLastError(); +  CloseHandle(h); +  if (!ret) { +    SetLastError(err); +    return MakeErrMsg(ErrMsg, path + ": SetFileTime: "); +  } + +  // Best we can do with Unix permission bits is to interpret the owner +  // writable bit. +  if (si.mode & 0200) { +    if (bhfi.dwFileAttributes & FILE_ATTRIBUTE_READONLY) { +      if (!SetFileAttributes(path.c_str(), +              bhfi.dwFileAttributes & ~FILE_ATTRIBUTE_READONLY)) +        return MakeErrMsg(ErrMsg, path + ": SetFileAttributes: "); +    } +  } else { +    if (!(bhfi.dwFileAttributes & FILE_ATTRIBUTE_READONLY)) { +      if (!SetFileAttributes(path.c_str(), +              bhfi.dwFileAttributes | FILE_ATTRIBUTE_READONLY)) +        return MakeErrMsg(ErrMsg, path + ": SetFileAttributes: "); +    } +  } + +  return false; +} + +bool +CopyFile(const sys::Path &Dest, const sys::Path &Src, std::string* ErrMsg) { +  // Can't use CopyFile macro defined in Windows.h because it would mess up the +  // above line.  We use the expansion it would have in a non-UNICODE build. +  if (!::CopyFileA(Src.c_str(), Dest.c_str(), false)) +    return MakeErrMsg(ErrMsg, "Can't copy '" + Src.toString() + +               "' to '" + Dest.toString() + "': "); +  return false; +} + +bool +Path::makeUnique(bool reuse_current, std::string* ErrMsg) { +  if (reuse_current && !exists()) +    return false; // File doesn't exist already, just use it! + +  // Reserve space for -XXXXXX at the end. +  char *FNBuffer = (char*) alloca(path.size()+8); +  unsigned offset = path.size(); +  path.copy(FNBuffer, offset); + +  // Find a numeric suffix that isn't used by an existing file.  Assume there +  // won't be more than 1 million files with the same prefix.  Probably a safe +  // bet. +  static unsigned FCounter = 0; +  do { +    sprintf(FNBuffer+offset, "-%06u", FCounter); +    if (++FCounter > 999999) +      FCounter = 0; +    path = FNBuffer; +  } while (exists()); +  return false; +} + +bool +Path::createTemporaryFileOnDisk(bool reuse_current, std::string* ErrMsg) { +  // Make this into a unique file name +  makeUnique(reuse_current, ErrMsg); + +  // Now go and create it +  HANDLE h = CreateFile(path.c_str(), GENERIC_WRITE, 0, NULL, CREATE_NEW, +                        FILE_ATTRIBUTE_NORMAL, NULL); +  if (h == INVALID_HANDLE_VALUE) +    return MakeErrMsg(ErrMsg, path + ": can't create file"); + +  CloseHandle(h); +  return false; +} + +} +} diff --git a/lib/System/Win32/Process.inc b/lib/System/Win32/Process.inc new file mode 100644 index 0000000..d5e80b4 --- /dev/null +++ b/lib/System/Win32/Process.inc @@ -0,0 +1,134 @@ +//===- Win32/Process.cpp - Win32 Process Implementation ------- -*- C++ -*-===// +//  +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Jeff Cohen and is distributed under the  +// University of Illinois Open Source License. See LICENSE.TXT for details. +//  +//===----------------------------------------------------------------------===// +// +// This file provides the Win32 specific implementation of the Process class. +// +//===----------------------------------------------------------------------===// + +#include "Win32.h" +#include <psapi.h> +#include <malloc.h> +#include <io.h> + +#ifdef __MINGW32__ + #if (HAVE_LIBPSAPI != 1) +  #error "libpsapi.a should be present" + #endif +#else + #pragma comment(lib, "psapi.lib") +#endif + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only Win32 specific code  +//===          and must not be UNIX code +//===----------------------------------------------------------------------===// + +#ifdef __MINGW32__ +// This ban should be lifted when MinGW 1.0+ has defined this value. +#  define _HEAPOK (-2) +#endif + +namespace llvm { +using namespace sys; + +// This function retrieves the page size using GetSystemInfo and is present +// solely so it can be called once in Process::GetPageSize to initialize the +// static variable PageSize. +inline unsigned GetPageSizeOnce() { +  // NOTE: A 32-bit application running under WOW64 is supposed to use +  // GetNativeSystemInfo.  However, this interface is not present prior +  // to Windows XP so to use it requires dynamic linking.  It is not clear +  // how this affects the reported page size, if at all.  One could argue +  // that LLVM ought to run as 64-bits on a 64-bit system, anyway. +  SYSTEM_INFO info; +  GetSystemInfo(&info); +  return static_cast<unsigned>(info.dwPageSize); +} + +unsigned  +Process::GetPageSize() { +  static const unsigned PageSize = GetPageSizeOnce(); +  return PageSize; +} + +size_t  +Process::GetMallocUsage() +{ +  _HEAPINFO hinfo; +  hinfo._pentry = NULL; + +  size_t size = 0; + +  while (_heapwalk(&hinfo) == _HEAPOK) +    size += hinfo._size; + +  return size; +} + +size_t +Process::GetTotalMemoryUsage() +{ +  PROCESS_MEMORY_COUNTERS pmc; +  GetProcessMemoryInfo(GetCurrentProcess(), &pmc, sizeof(pmc)); +  return pmc.PagefileUsage; +} + +void +Process::GetTimeUsage( +  TimeValue& elapsed, TimeValue& user_time, TimeValue& sys_time) +{ +  elapsed = TimeValue::now(); + +  uint64_t ProcCreate, ProcExit, KernelTime, UserTime; +  GetProcessTimes(GetCurrentProcess(), (FILETIME*)&ProcCreate,  +                  (FILETIME*)&ProcExit, (FILETIME*)&KernelTime, +                  (FILETIME*)&UserTime); + +  // FILETIME's are # of 100 nanosecond ticks (1/10th of a microsecond) +  user_time.seconds( UserTime / 10000000 ); +  user_time.nanoseconds( unsigned(UserTime % 10000000) * 100 ); +  sys_time.seconds( KernelTime / 10000000 ); +  sys_time.nanoseconds( unsigned(KernelTime % 10000000) * 100 ); +} + +int Process::GetCurrentUserId() +{ +  return 65536; +} + +int Process::GetCurrentGroupId() +{ +  return 65536; +} + +// Some LLVM programs such as bugpoint produce core files as a normal part of +// their operation. To prevent the disk from filling up, this configuration item +// does what's necessary to prevent their generation. +void Process::PreventCoreFiles() { +  // Windows doesn't do core files, but it does do modal pop-up message +  // boxes.  As this method is used by bugpoint, preventing these pop-ups +  // is the moral equivalent of suppressing core files. +  SetErrorMode(SEM_FAILCRITICALERRORS | +               SEM_NOGPFAULTERRORBOX | +               SEM_NOOPENFILEERRORBOX); +} + +bool Process::StandardInIsUserInput() { +  return GetFileType((HANDLE)_get_osfhandle(0)) == FILE_TYPE_CHAR; +} + +bool Process::StandardOutIsDisplayed() { +  return GetFileType((HANDLE)_get_osfhandle(1)) == FILE_TYPE_CHAR; +} + +bool Process::StandardErrIsDisplayed() { +  return GetFileType((HANDLE)_get_osfhandle(2)) == FILE_TYPE_CHAR; +} + +} diff --git a/lib/System/Win32/Program.inc b/lib/System/Win32/Program.inc new file mode 100644 index 0000000..60e9e41 --- /dev/null +++ b/lib/System/Win32/Program.inc @@ -0,0 +1,284 @@ +//===- Win32/Program.cpp - Win32 Program Implementation ------- -*- C++ -*-===// +//  +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Jeff Cohen and is distributed under the  +// University of Illinois Open Source License. See LICENSE.TXT for details. +//  +//===----------------------------------------------------------------------===// +// +// This file provides the Win32 specific implementation of the Program class. +// +//===----------------------------------------------------------------------===// + +#include "Win32.h" +#include <cstdio> +#include <malloc.h> +#include <io.h> +#include <fcntl.h> + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only Win32 specific code  +//===          and must not be UNIX code +//===----------------------------------------------------------------------===// + +namespace llvm { +using namespace sys; + +// This function just uses the PATH environment variable to find the program. +Path +Program::FindProgramByName(const std::string& progName) { + +  // Check some degenerate cases +  if (progName.length() == 0) // no program +    return Path(); +  Path temp; +  if (!temp.set(progName)) // invalid name +    return Path(); +  if (temp.canExecute()) // already executable as is +    return temp; + +  // At this point, the file name is valid and its not executable. +  // Let Windows search for it. +  char buffer[MAX_PATH]; +  char *dummy = NULL; +  DWORD len = SearchPath(NULL, progName.c_str(), ".exe", MAX_PATH, +                         buffer, &dummy); + +  // See if it wasn't found. +  if (len == 0) +    return Path(); + +  // See if we got the entire path. +  if (len < MAX_PATH) +    return Path(buffer); + +  // Buffer was too small; grow and retry. +  while (true) { +    char *b = reinterpret_cast<char *>(_alloca(len+1)); +    DWORD len2 = SearchPath(NULL, progName.c_str(), ".exe", len+1, b, &dummy); + +    // It is unlikely the search failed, but it's always possible some file +    // was added or removed since the last search, so be paranoid... +    if (len2 == 0) +      return Path(); +    else if (len2 <= len) +      return Path(b); + +    len = len2; +  } +} + +static HANDLE RedirectIO(const Path *path, int fd, std::string* ErrMsg) { +  HANDLE h; +  if (path == 0) { +    DuplicateHandle(GetCurrentProcess(), (HANDLE)_get_osfhandle(fd), +                    GetCurrentProcess(), &h, +                    0, TRUE, DUPLICATE_SAME_ACCESS); +    return h; +  } + +  const char *fname = path->toString().c_str(); +  if (*fname == 0) +    fname = "NUL"; + +  SECURITY_ATTRIBUTES sa; +  sa.nLength = sizeof(sa); +  sa.lpSecurityDescriptor = 0; +  sa.bInheritHandle = TRUE; + +  h = CreateFile(fname, fd ? GENERIC_WRITE : GENERIC_READ, FILE_SHARE_READ, +                 &sa, fd == 0 ? OPEN_EXISTING : CREATE_ALWAYS, +                 FILE_ATTRIBUTE_NORMAL, NULL); +  if (h == INVALID_HANDLE_VALUE) { +    MakeErrMsg(ErrMsg, std::string(fname) + ": Can't open file for " + +        (fd ? "input: " : "output: ")); +  } + +  return h; +} + +#ifdef __MINGW32__ +  // Due to unknown reason, mingw32's w32api doesn't have this declaration. +  extern "C" +  BOOL WINAPI SetInformationJobObject(HANDLE hJob, +                                      JOBOBJECTINFOCLASS JobObjectInfoClass, +                                      LPVOID lpJobObjectInfo, +                                      DWORD cbJobObjectInfoLength); +#endif +   +int  +Program::ExecuteAndWait(const Path& path,  +                        const char** args, +                        const char** envp, +                        const Path** redirects, +                        unsigned secondsToWait, +                        unsigned memoryLimit, +                        std::string* ErrMsg) { +  if (!path.canExecute()) { +    if (ErrMsg) +      *ErrMsg = "program not executable"; +    return -1; +  } + +  // Windows wants a command line, not an array of args, to pass to the new +  // process.  We have to concatenate them all, while quoting the args that +  // have embedded spaces. + +  // First, determine the length of the command line. +  unsigned len = 0; +  for (unsigned i = 0; args[i]; i++) { +    len += strlen(args[i]) + 1; +    if (strchr(args[i], ' ')) +      len += 2; +  } + +  // Now build the command line. +  char *command = reinterpret_cast<char *>(_alloca(len)); +  char *p = command; + +  for (unsigned i = 0; args[i]; i++) { +    const char *arg = args[i]; +    size_t len = strlen(arg); +    bool needsQuoting = strchr(arg, ' ') != 0; +    if (needsQuoting) +      *p++ = '"'; +    memcpy(p, arg, len); +    p += len; +    if (needsQuoting) +      *p++ = '"'; +    *p++ = ' '; +  } + +  *p = 0; + +  // Create a child process. +  STARTUPINFO si; +  memset(&si, 0, sizeof(si)); +  si.cb = sizeof(si); +  si.hStdInput = INVALID_HANDLE_VALUE; +  si.hStdOutput = INVALID_HANDLE_VALUE; +  si.hStdError = INVALID_HANDLE_VALUE; + +  if (redirects) { +    si.dwFlags = STARTF_USESTDHANDLES; +     +    si.hStdInput = RedirectIO(redirects[0], 0, ErrMsg); +    if (si.hStdInput == INVALID_HANDLE_VALUE) { +      MakeErrMsg(ErrMsg, "can't redirect stdin"); +      return -1; +    } +    si.hStdOutput = RedirectIO(redirects[1], 1, ErrMsg); +    if (si.hStdOutput == INVALID_HANDLE_VALUE) { +      CloseHandle(si.hStdInput); +      MakeErrMsg(ErrMsg, "can't redirect stdout"); +      return -1; +    } +    if (redirects[1] && redirects[2] && *(redirects[1]) != *(redirects[2])) { +      si.hStdError = RedirectIO(redirects[2], 2, ErrMsg); +      if (si.hStdError == INVALID_HANDLE_VALUE) { +        CloseHandle(si.hStdInput); +        CloseHandle(si.hStdOutput); +        MakeErrMsg(ErrMsg, "can't redirect stderr"); +        return -1; +      } +    } else { +      DuplicateHandle(GetCurrentProcess(), si.hStdOutput, +                      GetCurrentProcess(), &si.hStdError, +                      0, TRUE, DUPLICATE_SAME_ACCESS); +    } +  } +   +  PROCESS_INFORMATION pi; +  memset(&pi, 0, sizeof(pi)); + +  fflush(stdout); +  fflush(stderr); +  BOOL rc = CreateProcess(path.c_str(), command, NULL, NULL, FALSE, 0, +                          envp, NULL, &si, &pi); +  DWORD err = GetLastError(); + +  // Regardless of whether the process got created or not, we are done with +  // the handles we created for it to inherit. +  CloseHandle(si.hStdInput); +  CloseHandle(si.hStdOutput); +  CloseHandle(si.hStdError); + +  // Now return an error if the process didn't get created. +  if (!rc) +  { +    SetLastError(err); +    MakeErrMsg(ErrMsg, std::string("Couldn't execute program '") +  +               path.toString() + "'"); +    return -1; +  } + +  // Make sure these get closed no matter what. +  AutoHandle hProcess(pi.hProcess); +  AutoHandle hThread(pi.hThread); + +  // Assign the process to a job if a memory limit is defined. +  AutoHandle hJob(0); +  if (memoryLimit != 0) { +    hJob = CreateJobObject(0, 0); +    bool success = false; +    if (hJob != 0) { +      JOBOBJECT_EXTENDED_LIMIT_INFORMATION jeli; +      memset(&jeli, 0, sizeof(jeli)); +      jeli.BasicLimitInformation.LimitFlags = JOB_OBJECT_LIMIT_PROCESS_MEMORY; +      jeli.ProcessMemoryLimit = uintptr_t(memoryLimit) * 1048576; +      if (SetInformationJobObject(hJob, JobObjectExtendedLimitInformation, +                                  &jeli, sizeof(jeli))) { +        if (AssignProcessToJobObject(hJob, pi.hProcess)) +          success = true; +      } +    } +    if (!success) { +      SetLastError(GetLastError()); +      MakeErrMsg(ErrMsg, std::string("Unable to set memory limit")); +      TerminateProcess(pi.hProcess, 1); +      WaitForSingleObject(pi.hProcess, INFINITE); +      return -1; +    } +  } + +  // Wait for it to terminate. +  DWORD millisecondsToWait = INFINITE; +  if (secondsToWait > 0) +    millisecondsToWait = secondsToWait * 1000; + +  if (WaitForSingleObject(pi.hProcess, millisecondsToWait) == WAIT_TIMEOUT) { +    if (!TerminateProcess(pi.hProcess, 1)) { +      MakeErrMsg(ErrMsg, std::string("Failed to terminate timed-out program '") +          + path.toString() + "'"); +      return -1; +    } +    WaitForSingleObject(pi.hProcess, INFINITE); +  } +   +  // Get its exit status. +  DWORD status; +  rc = GetExitCodeProcess(pi.hProcess, &status); +  err = GetLastError(); + +  if (!rc) { +    SetLastError(err); +    MakeErrMsg(ErrMsg, std::string("Failed getting status for program '") +  +               path.toString() + "'"); +    return -1; +  } + +  return status; +} + +bool Program::ChangeStdinToBinary(){ +  int result = _setmode( _fileno(stdin), _O_BINARY ); +  return result == -1; +} + +bool Program::ChangeStdoutToBinary(){ +  int result = _setmode( _fileno(stdout), _O_BINARY ); +  return result == -1; +} + +} diff --git a/lib/System/Win32/Signals.inc b/lib/System/Win32/Signals.inc new file mode 100644 index 0000000..7da0c75 --- /dev/null +++ b/lib/System/Win32/Signals.inc @@ -0,0 +1,287 @@ +//===- Win32/Signals.cpp - Win32 Signals Implementation ---------*- C++ -*-===// +//  +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Jeff Cohen and is distributed under the  +// University of Illinois Open Source License. See LICENSE.TXT for details. +//  +//===----------------------------------------------------------------------===// +// +// This file provides the Win32 specific implementation of the Signals class. +// +//===----------------------------------------------------------------------===// + +#include "Win32.h" +#include <stdio.h> +#include <vector> + +#ifdef __MINGW32__ + #include <imagehlp.h> +#else + #include <dbghelp.h> +#endif +#include <psapi.h> + +#ifdef __MINGW32__ + #if ((HAVE_LIBIMAGEHLP != 1) || (HAVE_LIBPSAPI != 1)) +  #error "libimagehlp.a & libpsapi.a should be present" + #endif +#else + #pragma comment(lib, "psapi.lib") + #pragma comment(lib, "dbghelp.lib") +#endif + +// Forward declare. +static LONG WINAPI LLVMUnhandledExceptionFilter(LPEXCEPTION_POINTERS ep); +static BOOL WINAPI LLVMConsoleCtrlHandler(DWORD dwCtrlType); + +// InterruptFunction - The function to call if ctrl-c is pressed. +static void (*InterruptFunction)() = 0; + +static std::vector<llvm::sys::Path> *FilesToRemove = NULL; +static std::vector<llvm::sys::Path> *DirectoriesToRemove = NULL; +static bool RegisteredUnhandledExceptionFilter = false; +static bool CleanupExecuted = false; +static PTOP_LEVEL_EXCEPTION_FILTER OldFilter = NULL; + +// Windows creates a new thread to execute the console handler when an event +// (such as CTRL/C) occurs.  This causes concurrency issues with the above +// globals which this critical section addresses. +static CRITICAL_SECTION CriticalSection; + +namespace llvm { + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only Win32 specific code  +//===          and must not be UNIX code +//===----------------------------------------------------------------------===// + + +static void RegisterHandler() {  +  if (RegisteredUnhandledExceptionFilter) { +    EnterCriticalSection(&CriticalSection); +    return; +  } + +  // Now's the time to create the critical section.  This is the first time +  // through here, and there's only one thread. +  InitializeCriticalSection(&CriticalSection); + +  // Enter it immediately.  Now if someone hits CTRL/C, the console handler +  // can't proceed until the globals are updated. +  EnterCriticalSection(&CriticalSection); + +  RegisteredUnhandledExceptionFilter = true; +  OldFilter = SetUnhandledExceptionFilter(LLVMUnhandledExceptionFilter); +  SetConsoleCtrlHandler(LLVMConsoleCtrlHandler, TRUE); + +  // IMPORTANT NOTE: Caller must call LeaveCriticalSection(&CriticalSection) or +  // else multi-threading problems will ensue. +} + +// RemoveFileOnSignal - The public API +bool sys::RemoveFileOnSignal(const sys::Path &Filename, std::string* ErrMsg) { +  RegisterHandler(); + +  if (CleanupExecuted) { +    if (ErrMsg) +      *ErrMsg = "Process terminating -- cannot register for removal"; +    return true; +  } + +  if (FilesToRemove == NULL) +    FilesToRemove = new std::vector<sys::Path>; + +  FilesToRemove->push_back(Filename); + +  LeaveCriticalSection(&CriticalSection); +  return false; +} + +// RemoveDirectoryOnSignal - The public API +bool sys::RemoveDirectoryOnSignal(const sys::Path& path, std::string* ErrMsg) { +  // Not a directory? +  WIN32_FILE_ATTRIBUTE_DATA fi; +  if (!GetFileAttributesEx(path.c_str(), GetFileExInfoStandard, &fi)) { +    MakeErrMsg(ErrMsg, path.toString() + ": can't get status of file"); +    return true; +  } +     +  if (!(fi.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)) { +    if (ErrMsg) +      *ErrMsg = path.toString() + ": not a directory"; +    return true; +  } + +  RegisterHandler(); + +  if (CleanupExecuted) { +    if (ErrMsg) +      *ErrMsg = "Process terminating -- cannot register for removal"; +    return true; +  } + +  if (DirectoriesToRemove == NULL) +    DirectoriesToRemove = new std::vector<sys::Path>; +  DirectoriesToRemove->push_back(path); + +  LeaveCriticalSection(&CriticalSection); +  return false; +} + +/// PrintStackTraceOnErrorSignal - When an error signal (such as SIBABRT or +/// SIGSEGV) is delivered to the process, print a stack trace and then exit. +void sys::PrintStackTraceOnErrorSignal() { +  RegisterHandler(); +  LeaveCriticalSection(&CriticalSection); +} + + +void sys::SetInterruptFunction(void (*IF)()) { +  RegisterHandler(); +  InterruptFunction = IF; +  LeaveCriticalSection(&CriticalSection); +} +} + +static void Cleanup() { +  EnterCriticalSection(&CriticalSection); + +  // Prevent other thread from registering new files and directories for +  // removal, should we be executing because of the console handler callback. +  CleanupExecuted = true; + +  // FIXME: open files cannot be deleted. + +  if (FilesToRemove != NULL) +    while (!FilesToRemove->empty()) { +      try { +        FilesToRemove->back().eraseFromDisk(); +      } catch (...) { +      } +      FilesToRemove->pop_back(); +    } + +  if (DirectoriesToRemove != NULL) +    while (!DirectoriesToRemove->empty()) { +      try { +        DirectoriesToRemove->back().eraseFromDisk(true); +      } catch (...) { +      } +      DirectoriesToRemove->pop_back(); +    } + +  LeaveCriticalSection(&CriticalSection); +} + +static LONG WINAPI LLVMUnhandledExceptionFilter(LPEXCEPTION_POINTERS ep) { +  try { +    Cleanup(); + +    // Initialize the STACKFRAME structure. +    STACKFRAME StackFrame; +    memset(&StackFrame, 0, sizeof(StackFrame)); + +    StackFrame.AddrPC.Offset = ep->ContextRecord->Eip; +    StackFrame.AddrPC.Mode = AddrModeFlat; +    StackFrame.AddrStack.Offset = ep->ContextRecord->Esp; +    StackFrame.AddrStack.Mode = AddrModeFlat; +    StackFrame.AddrFrame.Offset = ep->ContextRecord->Ebp; +    StackFrame.AddrFrame.Mode = AddrModeFlat; + +    HANDLE hProcess = GetCurrentProcess(); +    HANDLE hThread = GetCurrentThread(); + +    // Initialize the symbol handler. +    SymSetOptions(SYMOPT_DEFERRED_LOADS|SYMOPT_LOAD_LINES); +    SymInitialize(hProcess, NULL, TRUE); + +    while (true) { +      if (!StackWalk(IMAGE_FILE_MACHINE_I386, hProcess, hThread, &StackFrame, +                     ep->ContextRecord, NULL, SymFunctionTableAccess, +                     SymGetModuleBase, NULL)) { +        break; +      } + +      if (StackFrame.AddrFrame.Offset == 0) +        break; + +      // Print the PC in hexadecimal. +      DWORD PC = StackFrame.AddrPC.Offset; +      fprintf(stderr, "%08X", PC); + +      // Print the parameters.  Assume there are four. +      fprintf(stderr, " (0x%08X 0x%08X 0x%08X 0x%08X)", StackFrame.Params[0], +              StackFrame.Params[1], StackFrame.Params[2], StackFrame.Params[3]); + +      // Verify the PC belongs to a module in this process. +      if (!SymGetModuleBase(hProcess, PC)) { +        fputs(" <unknown module>\n", stderr); +        continue; +      } + +      // Print the symbol name. +      char buffer[512]; +      IMAGEHLP_SYMBOL *symbol = reinterpret_cast<IMAGEHLP_SYMBOL *>(buffer); +      memset(symbol, 0, sizeof(IMAGEHLP_SYMBOL)); +      symbol->SizeOfStruct = sizeof(IMAGEHLP_SYMBOL); +      symbol->MaxNameLength = 512 - sizeof(IMAGEHLP_SYMBOL); + +      DWORD dwDisp; +      if (!SymGetSymFromAddr(hProcess, PC, &dwDisp, symbol)) { +        fputc('\n', stderr); +        continue; +      } + +      buffer[511] = 0; +      if (dwDisp > 0) +        fprintf(stderr, ", %s()+%04d bytes(s)", symbol->Name, dwDisp); +      else +        fprintf(stderr, ", %s", symbol->Name); + +      // Print the source file and line number information. +      IMAGEHLP_LINE line; +      memset(&line, 0, sizeof(line)); +      line.SizeOfStruct = sizeof(line); +      if (SymGetLineFromAddr(hProcess, PC, &dwDisp, &line)) { +        fprintf(stderr, ", %s, line %d", line.FileName, line.LineNumber); +        if (dwDisp > 0) +          fprintf(stderr, "+%04d byte(s)", dwDisp); +      } + +      fputc('\n', stderr); +    } +  } catch (...) { +      assert(!"Crashed in LLVMUnhandledExceptionFilter"); +  } + +  // Allow dialog box to pop up allowing choice to start debugger. +  if (OldFilter) +    return (*OldFilter)(ep); +  else +    return EXCEPTION_CONTINUE_SEARCH; +} + +static BOOL WINAPI LLVMConsoleCtrlHandler(DWORD dwCtrlType) { +  // We are running in our very own thread, courtesy of Windows. +  EnterCriticalSection(&CriticalSection); +  Cleanup(); + +  // If an interrupt function has been set, go and run one it; otherwise, +  // the process dies. +  void (*IF)() = InterruptFunction; +  InterruptFunction = 0;      // Don't run it on another CTRL-C. + +  if (IF) { +    // Note: if the interrupt function throws an exception, there is nothing +    // to catch it in this thread so it will kill the process. +    IF();                     // Run it now. +    LeaveCriticalSection(&CriticalSection); +    return TRUE;              // Don't kill the process. +  } + +  // Allow normal processing to take place; i.e., the process dies. +  LeaveCriticalSection(&CriticalSection); +  return FALSE; +} + diff --git a/lib/System/Win32/TimeValue.inc b/lib/System/Win32/TimeValue.inc new file mode 100644 index 0000000..ce2eb8d --- /dev/null +++ b/lib/System/Win32/TimeValue.inc @@ -0,0 +1,51 @@ +//===- Win32/TimeValue.cpp - Win32 TimeValue Implementation -----*- C++ -*-===// +//  +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Jeff Cohen and is distributed under the  +// University of Illinois Open Source License. See LICENSE.TXT for details. +//  +//===----------------------------------------------------------------------===// +// +// This file provides the Win32 implementation of the TimeValue class. +// +//===----------------------------------------------------------------------===// + +#include "Win32.h" +#include <time.h> + +namespace llvm { +using namespace sys; + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only Win32 specific code. +//===----------------------------------------------------------------------===// + +TimeValue TimeValue::now() { +  uint64_t ft; +  GetSystemTimeAsFileTime(reinterpret_cast<FILETIME *>(&ft)); + +  TimeValue t(0, 0); +  t.fromWin32Time(ft); +  return t; +} + +std::string TimeValue::toString() const { +#ifdef __MINGW32__ +  // This ban may be lifted by either: +  // (i) a future MinGW version other than 1.0 inherents the __time64_t type, or +  // (ii) configure tests for either the time_t or __time64_t type. +  time_t ourTime = time_t(this->toEpochTime()); +  struct tm *lt = ::localtime(&ourTime); +#else +  __time64_t ourTime = this->toEpochTime(); +  struct tm *lt = ::_localtime64(&ourTime); +#endif + +  char buffer[25]; +  strftime(buffer, 25, "%a %b %d %H:%M:%S %Y", lt); +  return std::string(buffer); +} + + +} diff --git a/lib/System/Win32/Win32.h b/lib/System/Win32/Win32.h new file mode 100644 index 0000000..71f0be5 --- /dev/null +++ b/lib/System/Win32/Win32.h @@ -0,0 +1,57 @@ +//===- Win32/Win32.h - Common Win32 Include File ----------------*- C++ -*-===// +// +//                     The LLVM Compiler Infrastructure +// +// This file was developed by Jeff Cohen and is distributed under the +// University of Illinois Open Source License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// +// +// This file defines things specific to Win32 implementations. +// +//===----------------------------------------------------------------------===// + +//===----------------------------------------------------------------------===// +//=== WARNING: Implementation here must contain only generic Win32 code that +//===          is guaranteed to work on *all* Win32 variants. +//===----------------------------------------------------------------------===// + +// Require at least Windows 2000 API. +#define _WIN32_WINNT 0x0500 + +#include "llvm/Config/config.h"     // Get autoconf configuration settings +#include "windows.h" +#include <cassert> +#include <string> + +inline bool MakeErrMsg(std::string* ErrMsg, const std::string& prefix) { +  if (!ErrMsg) +    return true; +  char *buffer = NULL; +  FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER|FORMAT_MESSAGE_FROM_SYSTEM, +      NULL, GetLastError(), 0, (LPSTR)&buffer, 1, NULL); +  *ErrMsg = prefix + buffer; +  LocalFree(buffer); +  return true; +} + +class AutoHandle { +  HANDLE handle; + +public: +  AutoHandle(HANDLE h) : handle(h) {} + +  ~AutoHandle() { +    if (handle) +      CloseHandle(handle); +  } + +  operator HANDLE() { +    return handle; +  } + +  AutoHandle &operator=(HANDLE h) { +    handle = h; +    return *this; +  } +}; diff --git a/lib/System/ltdl.c b/lib/System/ltdl.c new file mode 100644 index 0000000..67631be --- /dev/null +++ b/lib/System/ltdl.c @@ -0,0 +1,4523 @@ +/* ltdl.c -- system independent dlopen wrapper +   Copyright (C) 1998, 1999, 2000, 2004, 2005  Free Software Foundation, Inc. +   Originally by Thomas Tanner <tanner@ffii.org> +   This file is part of GNU Libtool. + +This library is free software; you can redistribute it and/or +modify it under the terms of the GNU Lesser General Public +License as published by the Free Software Foundation; either +version 2 of the License, or (at your option) any later version. + +As a special exception to the GNU Lesser General Public License, +if you distribute this file as part of a program or library that +is built using GNU libtool, you may include it under the same +distribution terms that you use for the rest of that program. + +This library is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU +Lesser General Public License for more details. + +You should have received a copy of the GNU Lesser General Public +License along with this library; if not, write to the Free Software +Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA +02110-1301  USA + +*/ + +#include "llvm/Config/config.h" + +#if HAVE_CONFIG_H +#  include <config.h> +#endif + +#if HAVE_UNISTD_H +#  include <unistd.h> +#endif + +#if HAVE_STDIO_H +#  include <stdio.h> +#endif + +/* Include the header defining malloc.  On K&R C compilers, +   that's <malloc.h>, on ANSI C and ISO C compilers, that's <stdlib.h>.  */ +#if HAVE_STDLIB_H +#  include <stdlib.h> +#else +#  if HAVE_MALLOC_H +#    include <malloc.h> +#  endif +#endif + +#if HAVE_STRING_H +#  include <string.h> +#else +#  if HAVE_STRINGS_H +#    include <strings.h> +#  endif +#endif + +#if HAVE_CTYPE_H +#  include <ctype.h> +#endif + +#if HAVE_MEMORY_H +#  include <memory.h> +#endif + +#if HAVE_ERRNO_H +#  include <errno.h> +#endif + + +#ifndef __WINDOWS__ +#  ifdef __WIN32__ +#    define __WINDOWS__ +#  endif +#endif + + +#undef LT_USE_POSIX_DIRENT +#ifdef HAVE_CLOSEDIR +#  ifdef HAVE_OPENDIR +#    ifdef HAVE_READDIR +#      ifdef HAVE_DIRENT_H +#        define LT_USE_POSIX_DIRENT +#      endif /* HAVE_DIRENT_H */ +#    endif /* HAVE_READDIR */ +#  endif /* HAVE_OPENDIR */ +#endif /* HAVE_CLOSEDIR */ + + +#undef LT_USE_WINDOWS_DIRENT_EMULATION +#ifndef LT_USE_POSIX_DIRENT +#  ifdef __WINDOWS__ +#    define LT_USE_WINDOWS_DIRENT_EMULATION +#  endif /* __WINDOWS__ */ +#endif /* LT_USE_POSIX_DIRENT */ + + +#ifdef LT_USE_POSIX_DIRENT +#  include <dirent.h> +#  define LT_D_NAMLEN(dirent) (strlen((dirent)->d_name)) +#else +#  ifdef LT_USE_WINDOWS_DIRENT_EMULATION +#    define LT_D_NAMLEN(dirent) (strlen((dirent)->d_name)) +#  else +#    define dirent direct +#    define LT_D_NAMLEN(dirent) ((dirent)->d_namlen) +#    if HAVE_SYS_NDIR_H +#      include <sys/ndir.h> +#    endif +#    if HAVE_SYS_DIR_H +#      include <sys/dir.h> +#    endif +#    if HAVE_NDIR_H +#      include <ndir.h> +#    endif +#  endif +#endif + +#if HAVE_ARGZ_H +#  include <argz.h> +#endif + +#if HAVE_ASSERT_H +#  include <assert.h> +#else +#  define assert(arg)	((void) 0) +#endif + +#include "ltdl.h" + +#if WITH_DMALLOC +#  include <dmalloc.h> +#endif + + + + +/* --- WINDOWS SUPPORT --- */ + + +#ifdef DLL_EXPORT +#  define LT_GLOBAL_DATA	__declspec(dllexport) +#else +#  define LT_GLOBAL_DATA +#endif + +/* fopen() mode flags for reading a text file */ +#undef	LT_READTEXT_MODE +#ifdef __WINDOWS__ +#  define LT_READTEXT_MODE "rt" +#else +#  define LT_READTEXT_MODE "r" +#endif + +#ifdef LT_USE_WINDOWS_DIRENT_EMULATION + +#include <windows.h> + +#define dirent lt_dirent +#define DIR lt_DIR + +struct dirent +{ +  char d_name[2048]; +  int  d_namlen; +}; + +typedef struct _DIR +{ +  HANDLE hSearch; +  WIN32_FIND_DATA Win32FindData; +  BOOL firsttime; +  struct dirent file_info; +} DIR; + +#endif /* LT_USE_WINDOWS_DIRENT_EMULATION */ + + +/* --- MANIFEST CONSTANTS --- */ + + +/* Standard libltdl search path environment variable name  */ +#undef  LTDL_SEARCHPATH_VAR +#define LTDL_SEARCHPATH_VAR	"LTDL_LIBRARY_PATH" + +/* Standard libtool archive file extension.  */ +#undef  LTDL_ARCHIVE_EXT +#define LTDL_ARCHIVE_EXT	".la" + +/* max. filename length */ +#ifndef LT_FILENAME_MAX +#  define LT_FILENAME_MAX	1024 +#endif + +/* This is the maximum symbol size that won't require malloc/free */ +#undef	LT_SYMBOL_LENGTH +#define LT_SYMBOL_LENGTH	128 + +/* This accounts for the _LTX_ separator */ +#undef	LT_SYMBOL_OVERHEAD +#define LT_SYMBOL_OVERHEAD	5 + + + + +/* --- MEMORY HANDLING --- */ + + +/* These are the functions used internally.  In addition to making +   use of the associated function pointers above, they also perform +   error handling.  */ +static char   *lt_estrdup	LT_PARAMS((const char *str)); +static lt_ptr lt_emalloc	LT_PARAMS((size_t size)); +static lt_ptr lt_erealloc	LT_PARAMS((lt_ptr addr, size_t size)); + +/* static lt_ptr rpl_realloc	LT_PARAMS((lt_ptr ptr, size_t size)); */ +#define rpl_realloc realloc + +/* These are the pointers that can be changed by the caller:  */ +LT_GLOBAL_DATA lt_ptr (*lt_dlmalloc)	LT_PARAMS((size_t size)) + 			= (lt_ptr (*) LT_PARAMS((size_t))) malloc; +LT_GLOBAL_DATA lt_ptr (*lt_dlrealloc)	LT_PARAMS((lt_ptr ptr, size_t size)) + 			= (lt_ptr (*) LT_PARAMS((lt_ptr, size_t))) rpl_realloc; +LT_GLOBAL_DATA void   (*lt_dlfree)	LT_PARAMS((lt_ptr ptr)) + 			= (void (*) LT_PARAMS((lt_ptr))) free; + +/* The following macros reduce the amount of typing needed to cast +   assigned memory.  */ +#if WITH_DMALLOC + +#define LT_DLMALLOC(tp, n)	((tp *) xmalloc ((n) * sizeof(tp))) +#define LT_DLREALLOC(tp, p, n)	((tp *) xrealloc ((p), (n) * sizeof(tp))) +#define LT_DLFREE(p)						\ +	LT_STMT_START { if (p) (p) = (xfree (p), (lt_ptr) 0); } LT_STMT_END + +#define LT_EMALLOC(tp, n)	((tp *) xmalloc ((n) * sizeof(tp))) +#define LT_EREALLOC(tp, p, n)	((tp *) xrealloc ((p), (n) * sizeof(tp))) + +#else + +#define LT_DLMALLOC(tp, n)	((tp *) lt_dlmalloc ((n) * sizeof(tp))) +#define LT_DLREALLOC(tp, p, n)	((tp *) lt_dlrealloc ((p), (n) * sizeof(tp))) +#define LT_DLFREE(p)						\ +	LT_STMT_START { if (p) (p) = (lt_dlfree (p), (lt_ptr) 0); } LT_STMT_END + +#define LT_EMALLOC(tp, n)	((tp *) lt_emalloc ((n) * sizeof(tp))) +#define LT_EREALLOC(tp, p, n)	((tp *) lt_erealloc ((p), (n) * sizeof(tp))) + +#endif + +#define LT_DLMEM_REASSIGN(p, q)			LT_STMT_START {	\ +	if ((p) != (q)) { if (p) lt_dlfree (p); (p) = (q); (q) = 0; }	\ +						} LT_STMT_END + + +/* --- REPLACEMENT FUNCTIONS --- */ + + +#undef strdup +#define strdup rpl_strdup + +static char *strdup LT_PARAMS((const char *str)); + +static char * +strdup(str) +     const char *str; +{ +  char *tmp = 0; + +  if (str) +    { +      tmp = LT_DLMALLOC (char, 1+ strlen (str)); +      if (tmp) +	{ +	  strcpy(tmp, str); +	} +    } + +  return tmp; +} + + +#if ! HAVE_STRCMP + +#undef strcmp +#define strcmp rpl_strcmp + +static int strcmp LT_PARAMS((const char *str1, const char *str2)); + +static int +strcmp (str1, str2) +     const char *str1; +     const char *str2; +{ +  if (str1 == str2) +    return 0; +  if (str1 == 0) +    return -1; +  if (str2 == 0) +    return 1; + +  for (;*str1 && *str2; ++str1, ++str2) +    { +      if (*str1 != *str2) +	break; +    } + +  return (int)(*str1 - *str2); +} +#endif + + +#if ! HAVE_STRCHR + +#  if HAVE_INDEX +#    define strchr index +#  else +#    define strchr rpl_strchr + +static const char *strchr LT_PARAMS((const char *str, int ch)); + +static const char* +strchr(str, ch) +     const char *str; +     int ch; +{ +  const char *p; + +  for (p = str; *p != (char)ch && *p != LT_EOS_CHAR; ++p) +    /*NOWORK*/; + +  return (*p == (char)ch) ? p : 0; +} + +#  endif +#endif /* !HAVE_STRCHR */ + + +#if ! HAVE_STRRCHR + +#  if HAVE_RINDEX +#    define strrchr rindex +#  else +#    define strrchr rpl_strrchr + +static const char *strrchr LT_PARAMS((const char *str, int ch)); + +static const char* +strrchr(str, ch) +     const char *str; +     int ch; +{ +  const char *p, *q = 0; + +  for (p = str; *p != LT_EOS_CHAR; ++p) +    { +      if (*p == (char) ch) +	{ +	  q = p; +	} +    } + +  return q; +} + +# endif +#endif + +/* NOTE:  Neither bcopy nor the memcpy implementation below can +          reliably handle copying in overlapping areas of memory.  Use +          memmove (for which there is a fallback implmentation below) +	  if you need that behaviour.  */ +#if ! HAVE_MEMCPY + +#  if HAVE_BCOPY +#    define memcpy(dest, src, size)	bcopy (src, dest, size) +#  else +#    define memcpy rpl_memcpy + +static lt_ptr memcpy LT_PARAMS((lt_ptr dest, const lt_ptr src, size_t size)); + +static lt_ptr +memcpy (dest, src, size) +     lt_ptr dest; +     const lt_ptr src; +     size_t size; +{ +  const char *	s = src; +  char *	d = dest; +  size_t	i = 0; + +  for (i = 0; i < size; ++i) +    { +      d[i] = s[i]; +    } + +  return dest; +} + +#  endif /* !HAVE_BCOPY */ +#endif   /* !HAVE_MEMCPY */ + +#if ! HAVE_MEMMOVE +#  define memmove rpl_memmove + +static lt_ptr memmove LT_PARAMS((lt_ptr dest, const lt_ptr src, size_t size)); + +static lt_ptr +memmove (dest, src, size) +     lt_ptr dest; +     const lt_ptr src; +     size_t size; +{ +  const char *	s = src; +  char *	d = dest; +  size_t	i; + +  if (d < s) +    for (i = 0; i < size; ++i) +      { +	d[i] = s[i]; +      } +  else if (d > s && size > 0) +    for (i = size -1; ; --i) +      { +	d[i] = s[i]; +	if (i == 0) +	  break; +      } + +  return dest; +} + +#endif /* !HAVE_MEMMOVE */ + +#ifdef LT_USE_WINDOWS_DIRENT_EMULATION + +static void closedir LT_PARAMS((DIR *entry)); + +static void +closedir(entry) +  DIR *entry; +{ +  assert(entry != (DIR *) NULL); +  FindClose(entry->hSearch); +  lt_dlfree((lt_ptr)entry); +} + + +static DIR * opendir LT_PARAMS((const char *path)); + +static DIR* +opendir (path) +  const char *path; +{ +  char file_specification[LT_FILENAME_MAX]; +  DIR *entry; + +  assert(path != (char *) NULL); +  /* allow space for: path + '\\' '\\' '*' '.' '*' + '\0' */ +  (void) strncpy (file_specification, path, LT_FILENAME_MAX-6); +  file_specification[LT_FILENAME_MAX-6] = LT_EOS_CHAR; +  (void) strcat(file_specification,"\\"); +  entry = LT_DLMALLOC (DIR,sizeof(DIR)); +  if (entry != (DIR *) 0) +    { +      entry->firsttime = TRUE; +      entry->hSearch = FindFirstFile(file_specification,&entry->Win32FindData); +    } +  if (entry->hSearch == INVALID_HANDLE_VALUE) +    { +      (void) strcat(file_specification,"\\*.*"); +      entry->hSearch = FindFirstFile(file_specification,&entry->Win32FindData); +      if (entry->hSearch == INVALID_HANDLE_VALUE) +        { +          LT_DLFREE (entry); +          return (DIR *) 0; +        } +    } +  return(entry); +} + + +static struct dirent *readdir LT_PARAMS((DIR *entry)); + +static struct dirent *readdir(entry) +  DIR *entry; +{ +  int +    status; + +  if (entry == (DIR *) 0) +    return((struct dirent *) 0); +  if (!entry->firsttime) +    { +      status = FindNextFile(entry->hSearch,&entry->Win32FindData); +      if (status == 0) +        return((struct dirent *) 0); +    } +  entry->firsttime = FALSE; +  (void) strncpy(entry->file_info.d_name,entry->Win32FindData.cFileName, +    LT_FILENAME_MAX-1); +  entry->file_info.d_name[LT_FILENAME_MAX - 1] = LT_EOS_CHAR; +  entry->file_info.d_namlen = strlen(entry->file_info.d_name); +  return(&entry->file_info); +} + +#endif /* LT_USE_WINDOWS_DIRENT_EMULATION */ + +/* According to Alexandre Oliva <oliva@lsd.ic.unicamp.br>, +    ``realloc is not entirely portable'' +   In any case we want to use the allocator supplied by the user without +   burdening them with an lt_dlrealloc function pointer to maintain. +   Instead implement our own version (with known boundary conditions) +   using lt_dlmalloc and lt_dlfree. */ + +/* #undef realloc +   #define realloc rpl_realloc +*/ +#if 0 +  /* You can't (re)define realloc unless you also (re)define malloc. +     Right now, this code uses the size of the *destination* to decide +     how much to copy.  That's not right, but you can't know the size +     of the source unless you know enough about, or wrote malloc.  So +     this code is disabled... */ + +static lt_ptr +realloc (ptr, size) +     lt_ptr ptr; +     size_t size; +{ +  if (size == 0) +    { +      /* For zero or less bytes, free the original memory */ +      if (ptr != 0) +	{ +	  lt_dlfree (ptr); +	} + +      return (lt_ptr) 0; +    } +  else if (ptr == 0) +    { +      /* Allow reallocation of a NULL pointer.  */ +      return lt_dlmalloc (size); +    } +  else +    { +      /* Allocate a new block, copy and free the old block.  */ +      lt_ptr mem = lt_dlmalloc (size); + +      if (mem) +	{ +	  memcpy (mem, ptr, size); +	  lt_dlfree (ptr); +	} + +      /* Note that the contents of PTR are not damaged if there is +	 insufficient memory to realloc.  */ +      return mem; +    } +} +#endif + + +#if ! HAVE_ARGZ_APPEND +#  define argz_append rpl_argz_append + +static error_t argz_append LT_PARAMS((char **pargz, size_t *pargz_len, +					const char *buf, size_t buf_len)); + +static error_t +argz_append (pargz, pargz_len, buf, buf_len) +     char **pargz; +     size_t *pargz_len; +     const char *buf; +     size_t buf_len; +{ +  size_t argz_len; +  char  *argz; + +  assert (pargz); +  assert (pargz_len); +  assert ((*pargz && *pargz_len) || (!*pargz && !*pargz_len)); + +  /* If nothing needs to be appended, no more work is required.  */ +  if (buf_len == 0) +    return 0; + +  /* Ensure there is enough room to append BUF_LEN.  */ +  argz_len = *pargz_len + buf_len; +  argz = LT_DLREALLOC (char, *pargz, argz_len); +  if (!argz) +    return ENOMEM; + +  /* Copy characters from BUF after terminating '\0' in ARGZ.  */ +  memcpy (argz + *pargz_len, buf, buf_len); + +  /* Assign new values.  */ +  *pargz = argz; +  *pargz_len = argz_len; + +  return 0; +} +#endif /* !HAVE_ARGZ_APPEND */ + + +#if ! HAVE_ARGZ_CREATE_SEP +#  define argz_create_sep rpl_argz_create_sep + +static error_t argz_create_sep LT_PARAMS((const char *str, int delim, +					    char **pargz, size_t *pargz_len)); + +static error_t +argz_create_sep (str, delim, pargz, pargz_len) +     const char *str; +     int delim; +     char **pargz; +     size_t *pargz_len; +{ +  size_t argz_len; +  char *argz = 0; + +  assert (str); +  assert (pargz); +  assert (pargz_len); + +  /* Make a copy of STR, but replacing each occurrence of +     DELIM with '\0'.  */ +  argz_len = 1+ LT_STRLEN (str); +  if (argz_len) +    { +      const char *p; +      char *q; + +      argz = LT_DLMALLOC (char, argz_len); +      if (!argz) +	return ENOMEM; + +      for (p = str, q = argz; *p != LT_EOS_CHAR; ++p) +	{ +	  if (*p == delim) +	    { +	      /* Ignore leading delimiters, and fold consecutive +		 delimiters in STR into a single '\0' in ARGZ.  */ +	      if ((q > argz) && (q[-1] != LT_EOS_CHAR)) +		*q++ = LT_EOS_CHAR; +	      else +		--argz_len; +	    } +	  else +	    *q++ = *p; +	} +      /* Copy terminating LT_EOS_CHAR.  */ +      *q = *p; +    } + +  /* If ARGZ_LEN has shrunk to nothing, release ARGZ's memory.  */ +  if (!argz_len) +    LT_DLFREE (argz); + +  /* Assign new values.  */ +  *pargz = argz; +  *pargz_len = argz_len; + +  return 0; +} +#endif /* !HAVE_ARGZ_CREATE_SEP */ + + +#if ! HAVE_ARGZ_INSERT +#  define argz_insert rpl_argz_insert + +static error_t argz_insert LT_PARAMS((char **pargz, size_t *pargz_len, +					char *before, const char *entry)); + +static error_t +argz_insert (pargz, pargz_len, before, entry) +     char **pargz; +     size_t *pargz_len; +     char *before; +     const char *entry; +{ +  assert (pargz); +  assert (pargz_len); +  assert (entry && *entry); + +  /* No BEFORE address indicates ENTRY should be inserted after the +     current last element.  */ +  if (!before) +    return argz_append (pargz, pargz_len, entry, 1+ LT_STRLEN (entry)); + +  /* This probably indicates a programmer error, but to preserve +     semantics, scan back to the start of an entry if BEFORE points +     into the middle of it.  */ +  while ((before > *pargz) && (before[-1] != LT_EOS_CHAR)) +    --before; + +  { +    size_t entry_len	= 1+ LT_STRLEN (entry); +    size_t argz_len	= *pargz_len + entry_len; +    size_t offset	= before - *pargz; +    char   *argz	= LT_DLREALLOC (char, *pargz, argz_len); + +    if (!argz) +      return ENOMEM; + +    /* Make BEFORE point to the equivalent offset in ARGZ that it +       used to have in *PARGZ incase realloc() moved the block.  */ +    before = argz + offset; + +    /* Move the ARGZ entries starting at BEFORE up into the new +       space at the end -- making room to copy ENTRY into the +       resulting gap.  */ +    memmove (before + entry_len, before, *pargz_len - offset); +    memcpy  (before, entry, entry_len); + +    /* Assign new values.  */ +    *pargz = argz; +    *pargz_len = argz_len; +  } + +  return 0; +} +#endif /* !HAVE_ARGZ_INSERT */ + + +#if ! HAVE_ARGZ_NEXT +#  define argz_next rpl_argz_next + +static char *argz_next LT_PARAMS((char *argz, size_t argz_len, +				    const char *entry)); + +static char * +argz_next (argz, argz_len, entry) +     char *argz; +     size_t argz_len; +     const char *entry; +{ +  assert ((argz && argz_len) || (!argz && !argz_len)); + +  if (entry) +    { +      /* Either ARGZ/ARGZ_LEN is empty, or ENTRY points into an address +	 within the ARGZ vector.  */ +      assert ((!argz && !argz_len) +	      || ((argz <= entry) && (entry < (argz + argz_len)))); + +      /* Move to the char immediately after the terminating +	 '\0' of ENTRY.  */ +      entry = 1+ strchr (entry, LT_EOS_CHAR); + +      /* Return either the new ENTRY, or else NULL if ARGZ is +	 exhausted.  */ +      return (entry >= argz + argz_len) ? 0 : (char *) entry; +    } +  else +    { +      /* This should probably be flagged as a programmer error, +	 since starting an argz_next loop with the iterator set +	 to ARGZ is safer.  To preserve semantics, handle the NULL +	 case by returning the start of ARGZ (if any).  */ +      if (argz_len > 0) +	return argz; +      else +	return 0; +    } +} +#endif /* !HAVE_ARGZ_NEXT */ + + + +#if ! HAVE_ARGZ_STRINGIFY +#  define argz_stringify rpl_argz_stringify + +static void argz_stringify LT_PARAMS((char *argz, size_t argz_len, +				       int sep)); + +static void +argz_stringify (argz, argz_len, sep) +     char *argz; +     size_t argz_len; +     int sep; +{ +  assert ((argz && argz_len) || (!argz && !argz_len)); + +  if (sep) +    { +      --argz_len;		/* don't stringify the terminating EOS */ +      while (--argz_len > 0) +	{ +	  if (argz[argz_len] == LT_EOS_CHAR) +	    argz[argz_len] = sep; +	} +    } +} +#endif /* !HAVE_ARGZ_STRINGIFY */ + + + + +/* --- TYPE DEFINITIONS -- */ + + +/* This type is used for the array of caller data sets in each handler. */ +typedef struct { +  lt_dlcaller_id	key; +  lt_ptr		data; +} lt_caller_data; + + + + +/* --- OPAQUE STRUCTURES DECLARED IN LTDL.H --- */ + + +/* Extract the diagnostic strings from the error table macro in the same +   order as the enumerated indices in ltdl.h. */ + +static const char *lt_dlerror_strings[] = +  { +#define LT_ERROR(name, diagnostic)	(diagnostic), +    lt_dlerror_table +#undef LT_ERROR + +    0 +  }; + +/* This structure is used for the list of registered loaders. */ +struct lt_dlloader { +  struct lt_dlloader   *next; +  const char	       *loader_name;	/* identifying name for each loader */ +  const char	       *sym_prefix;	/* prefix for symbols */ +  lt_module_open       *module_open; +  lt_module_close      *module_close; +  lt_find_sym	       *find_sym; +  lt_dlloader_exit     *dlloader_exit; +  lt_user_data		dlloader_data; +}; + +struct lt_dlhandle_struct { +  struct lt_dlhandle_struct   *next; +  lt_dlloader	       *loader;		/* dlopening interface */ +  lt_dlinfo		info; +  int			depcount;	/* number of dependencies */ +  lt_dlhandle	       *deplibs;	/* dependencies */ +  lt_module		module;		/* system module handle */ +  lt_ptr		system;		/* system specific data */ +  lt_caller_data       *caller_data;	/* per caller associated data */ +  int			flags;		/* various boolean stats */ +}; + +/* Various boolean flags can be stored in the flags field of an +   lt_dlhandle_struct... */ +#define LT_DLGET_FLAG(handle, flag) (((handle)->flags & (flag)) == (flag)) +#define LT_DLSET_FLAG(handle, flag) ((handle)->flags |= (flag)) + +#define LT_DLRESIDENT_FLAG	    (0x01 << 0) +/* ...add more flags here... */ + +#define LT_DLIS_RESIDENT(handle)    LT_DLGET_FLAG(handle, LT_DLRESIDENT_FLAG) + + +#define LT_DLSTRERROR(name)	lt_dlerror_strings[LT_CONC(LT_ERROR_,name)] + +static	const char	objdir[]		= LTDL_OBJDIR; +static	const char	archive_ext[]		= LTDL_ARCHIVE_EXT; +#ifdef	LTDL_SHLIB_EXT +static	const char	shlib_ext[]		= LTDL_SHLIB_EXT; +#endif +#ifdef	LTDL_SYSSEARCHPATH +static	const char	sys_search_path[]	= LTDL_SYSSEARCHPATH; +#endif + + + + +/* --- MUTEX LOCKING --- */ + + +/* Macros to make it easier to run the lock functions only if they have +   been registered.  The reason for the complicated lock macro is to +   ensure that the stored error message from the last error is not +   accidentally erased if the current function doesn't generate an +   error of its own.  */ +#define LT_DLMUTEX_LOCK()			LT_STMT_START {	\ +	if (lt_dlmutex_lock_func) (*lt_dlmutex_lock_func)();	\ +						} LT_STMT_END +#define LT_DLMUTEX_UNLOCK()			LT_STMT_START { \ +	if (lt_dlmutex_unlock_func) (*lt_dlmutex_unlock_func)();\ +						} LT_STMT_END +#define LT_DLMUTEX_SETERROR(errormsg)		LT_STMT_START {	\ +	if (lt_dlmutex_seterror_func)				\ +		(*lt_dlmutex_seterror_func) (errormsg);		\ +	else 	lt_dllast_error = (errormsg);	} LT_STMT_END +#define LT_DLMUTEX_GETERROR(errormsg)		LT_STMT_START {	\ +	if (lt_dlmutex_seterror_func)				\ +		(errormsg) = (*lt_dlmutex_geterror_func) ();	\ +	else	(errormsg) = lt_dllast_error;	} LT_STMT_END + +/* The mutex functions stored here are global, and are necessarily the +   same for all threads that wish to share access to libltdl.  */ +static	lt_dlmutex_lock	    *lt_dlmutex_lock_func     = 0; +static	lt_dlmutex_unlock   *lt_dlmutex_unlock_func   = 0; +static	lt_dlmutex_seterror *lt_dlmutex_seterror_func = 0; +static	lt_dlmutex_geterror *lt_dlmutex_geterror_func = 0; +static	const char	    *lt_dllast_error	      = 0; + + +/* Either set or reset the mutex functions.  Either all the arguments must +   be valid functions, or else all can be NULL to turn off locking entirely. +   The registered functions should be manipulating a static global lock +   from the lock() and unlock() callbacks, which needs to be reentrant.  */ +int +lt_dlmutex_register (lock, unlock, seterror, geterror) +     lt_dlmutex_lock *lock; +     lt_dlmutex_unlock *unlock; +     lt_dlmutex_seterror *seterror; +     lt_dlmutex_geterror *geterror; +{ +  lt_dlmutex_unlock *old_unlock = unlock; +  int		     errors	= 0; + +  /* Lock using the old lock() callback, if any.  */ +  LT_DLMUTEX_LOCK (); + +  if ((lock && unlock && seterror && geterror) +      || !(lock || unlock || seterror || geterror)) +    { +      lt_dlmutex_lock_func     = lock; +      lt_dlmutex_unlock_func   = unlock; +      lt_dlmutex_geterror_func = geterror; +    } +  else +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (INVALID_MUTEX_ARGS)); +      ++errors; +    } + +  /* Use the old unlock() callback we saved earlier, if any.  Otherwise +     record any errors using internal storage.  */ +  if (old_unlock) +    (*old_unlock) (); + +  /* Return the number of errors encountered during the execution of +     this function.  */ +  return errors; +} + + + + +/* --- ERROR HANDLING --- */ + + +static	const char    **user_error_strings	= 0; +static	int		errorcount		= LT_ERROR_MAX; + +int +lt_dladderror (diagnostic) +     const char *diagnostic; +{ +  int		errindex = 0; +  int		result	 = -1; +  const char  **temp     = (const char **) 0; + +  assert (diagnostic); + +  LT_DLMUTEX_LOCK (); + +  errindex = errorcount - LT_ERROR_MAX; +  temp = LT_EREALLOC (const char *, user_error_strings, 1 + errindex); +  if (temp) +    { +      user_error_strings		= temp; +      user_error_strings[errindex]	= diagnostic; +      result				= errorcount++; +    } + +  LT_DLMUTEX_UNLOCK (); + +  return result; +} + +int +lt_dlseterror (errindex) +     int errindex; +{ +  int		errors	 = 0; + +  LT_DLMUTEX_LOCK (); + +  if (errindex >= errorcount || errindex < 0) +    { +      /* Ack!  Error setting the error message! */ +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (INVALID_ERRORCODE)); +      ++errors; +    } +  else if (errindex < LT_ERROR_MAX) +    { +      /* No error setting the error message! */ +      LT_DLMUTEX_SETERROR (lt_dlerror_strings[errindex]); +    } +  else +    { +      /* No error setting the error message! */ +      LT_DLMUTEX_SETERROR (user_error_strings[errindex - LT_ERROR_MAX]); +    } + +  LT_DLMUTEX_UNLOCK (); + +  return errors; +} + +static lt_ptr +lt_emalloc (size) +     size_t size; +{ +  lt_ptr mem = lt_dlmalloc (size); +  if (size && !mem) +    LT_DLMUTEX_SETERROR (LT_DLSTRERROR (NO_MEMORY)); +  return mem; +} + +static lt_ptr +lt_erealloc (addr, size) +     lt_ptr addr; +     size_t size; +{ +  lt_ptr mem = lt_dlrealloc (addr, size); +  if (size && !mem) +    LT_DLMUTEX_SETERROR (LT_DLSTRERROR (NO_MEMORY)); +  return mem; +} + +static char * +lt_estrdup (str) +     const char *str; +{ +  char *copy = strdup (str); +  if (LT_STRLEN (str) && !copy) +    LT_DLMUTEX_SETERROR (LT_DLSTRERROR (NO_MEMORY)); +  return copy; +} + + + + +/* --- DLOPEN() INTERFACE LOADER --- */ + + +#if HAVE_LIBDL + +/* dynamic linking with dlopen/dlsym */ + +#if HAVE_DLFCN_H +#  include <dlfcn.h> +#endif + +#if HAVE_SYS_DL_H +#  include <sys/dl.h> +#endif + +#ifdef RTLD_GLOBAL +#  define LT_GLOBAL		RTLD_GLOBAL +#else +#  ifdef DL_GLOBAL +#    define LT_GLOBAL		DL_GLOBAL +#  endif +#endif /* !RTLD_GLOBAL */ +#ifndef LT_GLOBAL +#  define LT_GLOBAL		0 +#endif /* !LT_GLOBAL */ + +/* We may have to define LT_LAZY_OR_NOW in the command line if we +   find out it does not work in some platform. */ +#ifndef LT_LAZY_OR_NOW +#  ifdef RTLD_LAZY +#    define LT_LAZY_OR_NOW	RTLD_LAZY +#  else +#    ifdef DL_LAZY +#      define LT_LAZY_OR_NOW	DL_LAZY +#    endif +#  endif /* !RTLD_LAZY */ +#endif +#ifndef LT_LAZY_OR_NOW +#  ifdef RTLD_NOW +#    define LT_LAZY_OR_NOW	RTLD_NOW +#  else +#    ifdef DL_NOW +#      define LT_LAZY_OR_NOW	DL_NOW +#    endif +#  endif /* !RTLD_NOW */ +#endif +#ifndef LT_LAZY_OR_NOW +#  define LT_LAZY_OR_NOW	0 +#endif /* !LT_LAZY_OR_NOW */ + +#if HAVE_DLERROR +#  define DLERROR(arg)	dlerror () +#else +#  define DLERROR(arg)	LT_DLSTRERROR (arg) +#endif + +static lt_module +sys_dl_open (loader_data, filename) +     lt_user_data loader_data; +     const char *filename; +{ +  lt_module   module   = dlopen (filename, LT_GLOBAL | LT_LAZY_OR_NOW); + +  if (!module) +    { +      LT_DLMUTEX_SETERROR (DLERROR (CANNOT_OPEN)); +    } + +  return module; +} + +static int +sys_dl_close (loader_data, module) +     lt_user_data loader_data; +     lt_module module; +{ +  int errors = 0; + +  if (dlclose (module) != 0) +    { +      LT_DLMUTEX_SETERROR (DLERROR (CANNOT_CLOSE)); +      ++errors; +    } + +  return errors; +} + +static lt_ptr +sys_dl_sym (loader_data, module, symbol) +     lt_user_data loader_data; +     lt_module module; +     const char *symbol; +{ +  lt_ptr address = dlsym (module, symbol); + +  if (!address) +    { +      LT_DLMUTEX_SETERROR (DLERROR (SYMBOL_NOT_FOUND)); +    } + +  return address; +} + +static struct lt_user_dlloader sys_dl = +  { +#  ifdef NEED_USCORE +    "_", +#  else +    0, +#  endif +    sys_dl_open, sys_dl_close, sys_dl_sym, 0, 0 }; + + +#endif /* HAVE_LIBDL */ + + + +/* --- SHL_LOAD() INTERFACE LOADER --- */ + +#if HAVE_SHL_LOAD + +/* dynamic linking with shl_load (HP-UX) (comments from gmodule) */ + +#ifdef HAVE_DL_H +#  include <dl.h> +#endif + +/* some flags are missing on some systems, so we provide + * harmless defaults. + * + * Mandatory: + * BIND_IMMEDIATE  - Resolve symbol references when the library is loaded. + * BIND_DEFERRED   - Delay code symbol resolution until actual reference. + * + * Optionally: + * BIND_FIRST	   - Place the library at the head of the symbol search + * 		     order. + * BIND_NONFATAL   - The default BIND_IMMEDIATE behavior is to treat all + * 		     unsatisfied symbols as fatal.  This flag allows + * 		     binding of unsatisfied code symbols to be deferred + * 		     until use. + *		     [Perl: For certain libraries, like DCE, deferred + *		     binding often causes run time problems. Adding + *		     BIND_NONFATAL to BIND_IMMEDIATE still allows + *		     unresolved references in situations like this.] + * BIND_NOSTART	   - Do not call the initializer for the shared library + *		     when the library is loaded, nor on a future call to + *		     shl_unload(). + * BIND_VERBOSE	   - Print verbose messages concerning possible + *		     unsatisfied symbols. + * + * hp9000s700/hp9000s800: + * BIND_RESTRICTED - Restrict symbols visible by the library to those + *		     present at library load time. + * DYNAMIC_PATH	   - Allow the loader to dynamically search for the + *		     library specified by the path argument. + */ + +#ifndef	DYNAMIC_PATH +#  define DYNAMIC_PATH		0 +#endif +#ifndef	BIND_RESTRICTED +#  define BIND_RESTRICTED	0 +#endif + +#define	LT_BIND_FLAGS	(BIND_IMMEDIATE | BIND_NONFATAL | DYNAMIC_PATH) + +static lt_module +sys_shl_open (loader_data, filename) +     lt_user_data loader_data; +     const char *filename; +{ +  static shl_t self = (shl_t) 0; +  lt_module module = shl_load (filename, LT_BIND_FLAGS, 0L); + +  /* Since searching for a symbol against a NULL module handle will also +     look in everything else that was already loaded and exported with +     the -E compiler flag, we always cache a handle saved before any +     modules are loaded.  */ +  if (!self) +    { +      lt_ptr address; +      shl_findsym (&self, "main", TYPE_UNDEFINED, &address); +    } + +  if (!filename) +    { +      module = self; +    } +  else +    { +      module = shl_load (filename, LT_BIND_FLAGS, 0L); + +      if (!module) +	{ +	  LT_DLMUTEX_SETERROR (LT_DLSTRERROR (CANNOT_OPEN)); +	} +    } + +  return module; +} + +static int +sys_shl_close (loader_data, module) +     lt_user_data loader_data; +     lt_module module; +{ +  int errors = 0; + +  if (module && (shl_unload ((shl_t) (module)) != 0)) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (CANNOT_CLOSE)); +      ++errors; +    } + +  return errors; +} + +static lt_ptr +sys_shl_sym (loader_data, module, symbol) +     lt_user_data loader_data; +     lt_module module; +     const char *symbol; +{ +  lt_ptr address = 0; + +  /* sys_shl_open should never return a NULL module handle */ +  if (module == (lt_module) 0) +  { +    LT_DLMUTEX_SETERROR (LT_DLSTRERROR (INVALID_HANDLE)); +  } +  else if (!shl_findsym((shl_t*) &module, symbol, TYPE_UNDEFINED, &address)) +    { +      if (!address) +	{ +	  LT_DLMUTEX_SETERROR (LT_DLSTRERROR (SYMBOL_NOT_FOUND)); +	} +    } + +  return address; +} + +static struct lt_user_dlloader sys_shl = { +  0, sys_shl_open, sys_shl_close, sys_shl_sym, 0, 0 +}; + +#endif /* HAVE_SHL_LOAD */ + + + + +/* --- LOADLIBRARY() INTERFACE LOADER --- */ + +#ifdef __WINDOWS__ + +/* dynamic linking for Win32 */ + +#include <windows.h> + +/* Forward declaration; required to implement handle search below. */ +static lt_dlhandle handles; + +static lt_module +sys_wll_open (loader_data, filename) +     lt_user_data loader_data; +     const char *filename; +{ +  lt_dlhandle	cur; +  lt_module	module	   = 0; +  const char   *errormsg   = 0; +  char	       *searchname = 0; +  char	       *ext; +  char		self_name_buf[MAX_PATH]; + +  if (!filename) +    { +      /* Get the name of main module */ +      *self_name_buf = 0; +      GetModuleFileName (NULL, self_name_buf, sizeof (self_name_buf)); +      filename = ext = self_name_buf; +    } +  else +    { +      ext = strrchr (filename, '.'); +    } + +  if (ext) +    { +      /* FILENAME already has an extension. */ +      searchname = lt_estrdup (filename); +    } +  else +    { +      /* Append a `.' to stop Windows from adding an +	 implicit `.dll' extension. */ +      searchname = LT_EMALLOC (char, 2+ LT_STRLEN (filename)); +      if (searchname) +	sprintf (searchname, "%s.", filename); +    } +  if (!searchname) +    return 0; + +  { +    /* Silence dialog from LoadLibrary on some failures. +       No way to get the error mode, but to set it, +       so set it twice to preserve any previous flags. */ +    UINT errormode = SetErrorMode(SEM_FAILCRITICALERRORS); +    SetErrorMode(errormode | SEM_FAILCRITICALERRORS); + +#if defined(__CYGWIN__) +    { +      char wpath[MAX_PATH]; +      cygwin_conv_to_full_win32_path (searchname, wpath); +      module = LoadLibrary (wpath); +    } +#else +    module = LoadLibrary (searchname); +#endif + +    /* Restore the error mode. */ +    SetErrorMode(errormode); +  } + +  LT_DLFREE (searchname); + +  /* libltdl expects this function to fail if it is unable +     to physically load the library.  Sadly, LoadLibrary +     will search the loaded libraries for a match and return +     one of them if the path search load fails. + +     We check whether LoadLibrary is returning a handle to +     an already loaded module, and simulate failure if we +     find one. */ +  LT_DLMUTEX_LOCK (); +  cur = handles; +  while (cur) +    { +      if (!cur->module) +	{ +	  cur = 0; +	  break; +	} + +      if (cur->module == module) +	{ +	  break; +	} + +      cur = cur->next; +  } +  LT_DLMUTEX_UNLOCK (); + +  if (cur || !module) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (CANNOT_OPEN)); +      module = 0; +    } + +  return module; +} + +static int +sys_wll_close (loader_data, module) +     lt_user_data loader_data; +     lt_module module; +{ +  int	      errors   = 0; + +  if (FreeLibrary(module) == 0) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (CANNOT_CLOSE)); +      ++errors; +    } + +  return errors; +} + +static lt_ptr +sys_wll_sym (loader_data, module, symbol) +     lt_user_data loader_data; +     lt_module module; +     const char *symbol; +{ +  lt_ptr      address  = GetProcAddress (module, symbol); + +  if (!address) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (SYMBOL_NOT_FOUND)); +    } + +  return address; +} + +static struct lt_user_dlloader sys_wll = { +  0, sys_wll_open, sys_wll_close, sys_wll_sym, 0, 0 +}; + +#endif /* __WINDOWS__ */ + + + + +/* --- LOAD_ADD_ON() INTERFACE LOADER --- */ + + +#ifdef __BEOS__ + +/* dynamic linking for BeOS */ + +#include <kernel/image.h> + +static lt_module +sys_bedl_open (loader_data, filename) +     lt_user_data loader_data; +     const char *filename; +{ +  image_id image = 0; + +  if (filename) +    { +      image = load_add_on (filename); +    } +  else +    { +      image_info info; +      int32 cookie = 0; +      if (get_next_image_info (0, &cookie, &info) == B_OK) +	image = load_add_on (info.name); +    } + +  if (image <= 0) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (CANNOT_OPEN)); +      image = 0; +    } + +  return (lt_module) image; +} + +static int +sys_bedl_close (loader_data, module) +     lt_user_data loader_data; +     lt_module module; +{ +  int errors = 0; + +  if (unload_add_on ((image_id) module) != B_OK) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (CANNOT_CLOSE)); +      ++errors; +    } + +  return errors; +} + +static lt_ptr +sys_bedl_sym (loader_data, module, symbol) +     lt_user_data loader_data; +     lt_module module; +     const char *symbol; +{ +  lt_ptr address = 0; +  image_id image = (image_id) module; + +  if (get_image_symbol (image, symbol, B_SYMBOL_TYPE_ANY, address) != B_OK) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (SYMBOL_NOT_FOUND)); +      address = 0; +    } + +  return address; +} + +static struct lt_user_dlloader sys_bedl = { +  0, sys_bedl_open, sys_bedl_close, sys_bedl_sym, 0, 0 +}; + +#endif /* __BEOS__ */ + + + + +/* --- DLD_LINK() INTERFACE LOADER --- */ + + +#if HAVE_DLD + +/* dynamic linking with dld */ + +#if HAVE_DLD_H +#include <dld.h> +#endif + +static lt_module +sys_dld_open (loader_data, filename) +     lt_user_data loader_data; +     const char *filename; +{ +  lt_module module = strdup (filename); + +  if (dld_link (filename) != 0) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (CANNOT_OPEN)); +      LT_DLFREE (module); +      module = 0; +    } + +  return module; +} + +static int +sys_dld_close (loader_data, module) +     lt_user_data loader_data; +     lt_module module; +{ +  int errors = 0; + +  if (dld_unlink_by_file ((char*)(module), 1) != 0) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (CANNOT_CLOSE)); +      ++errors; +    } +  else +    { +      LT_DLFREE (module); +    } + +  return errors; +} + +static lt_ptr +sys_dld_sym (loader_data, module, symbol) +     lt_user_data loader_data; +     lt_module module; +     const char *symbol; +{ +  lt_ptr address = dld_get_func (symbol); + +  if (!address) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (SYMBOL_NOT_FOUND)); +    } + +  return address; +} + +static struct lt_user_dlloader sys_dld = { +  0, sys_dld_open, sys_dld_close, sys_dld_sym, 0, 0 +}; + +#endif /* HAVE_DLD */ + +/* --- DYLD() MACOSX/DARWIN INTERFACE LOADER --- */ +#if HAVE_DYLD + + +#if HAVE_MACH_O_DYLD_H +#if !defined(__APPLE_CC__) && !defined(__MWERKS__) && !defined(__private_extern__) +/* Is this correct? Does it still function properly? */ +#define __private_extern__ extern +#endif +# include <mach-o/dyld.h> +#endif +#include <mach-o/getsect.h> + +/* We have to put some stuff here that isn't in older dyld.h files */ +#ifndef ENUM_DYLD_BOOL +# define ENUM_DYLD_BOOL +# undef FALSE +# undef TRUE + enum DYLD_BOOL { +    FALSE, +    TRUE + }; +#endif +#ifndef LC_REQ_DYLD +# define LC_REQ_DYLD 0x80000000 +#endif +#ifndef LC_LOAD_WEAK_DYLIB +# define LC_LOAD_WEAK_DYLIB (0x18 | LC_REQ_DYLD) +#endif +static const struct mach_header * (*ltdl_NSAddImage)(const char *image_name, unsigned long options) = 0; +static NSSymbol (*ltdl_NSLookupSymbolInImage)(const struct mach_header *image,const char *symbolName, unsigned long options) = 0; +static enum DYLD_BOOL (*ltdl_NSIsSymbolNameDefinedInImage)(const struct mach_header *image, const char *symbolName) = 0; +static enum DYLD_BOOL (*ltdl_NSMakePrivateModulePublic)(NSModule module) = 0; + +#ifndef NSADDIMAGE_OPTION_NONE +#define NSADDIMAGE_OPTION_NONE                          0x0 +#endif +#ifndef NSADDIMAGE_OPTION_RETURN_ON_ERROR +#define NSADDIMAGE_OPTION_RETURN_ON_ERROR               0x1 +#endif +#ifndef NSADDIMAGE_OPTION_WITH_SEARCHING +#define NSADDIMAGE_OPTION_WITH_SEARCHING                0x2 +#endif +#ifndef NSADDIMAGE_OPTION_RETURN_ONLY_IF_LOADED +#define NSADDIMAGE_OPTION_RETURN_ONLY_IF_LOADED         0x4 +#endif +#ifndef NSADDIMAGE_OPTION_MATCH_FILENAME_BY_INSTALLNAME +#define NSADDIMAGE_OPTION_MATCH_FILENAME_BY_INSTALLNAME 0x8 +#endif +#ifndef NSLOOKUPSYMBOLINIMAGE_OPTION_BIND +#define NSLOOKUPSYMBOLINIMAGE_OPTION_BIND            0x0 +#endif +#ifndef NSLOOKUPSYMBOLINIMAGE_OPTION_BIND_NOW +#define NSLOOKUPSYMBOLINIMAGE_OPTION_BIND_NOW        0x1 +#endif +#ifndef NSLOOKUPSYMBOLINIMAGE_OPTION_BIND_FULLY +#define NSLOOKUPSYMBOLINIMAGE_OPTION_BIND_FULLY      0x2 +#endif +#ifndef NSLOOKUPSYMBOLINIMAGE_OPTION_RETURN_ON_ERROR +#define NSLOOKUPSYMBOLINIMAGE_OPTION_RETURN_ON_ERROR 0x4 +#endif + + +static const char * +lt_int_dyld_error(othererror) +	char* othererror; +{ +/* return the dyld error string, or the passed in error string if none */ +	NSLinkEditErrors ler; +	int lerno; +	const char *errstr; +	const char *file; +	NSLinkEditError(&ler,&lerno,&file,&errstr); +	if (!errstr || !strlen(errstr)) errstr = othererror; +	return errstr; +} + +static const struct mach_header * +lt_int_dyld_get_mach_header_from_nsmodule(module) +	NSModule module; +{ +/* There should probably be an apple dyld api for this */ +	int i=_dyld_image_count(); +	int j; +	const char *modname=NSNameOfModule(module); +	const struct mach_header *mh=NULL; +	if (!modname) return NULL; +	for (j = 0; j < i; j++) +	{ +		if (!strcmp(_dyld_get_image_name(j),modname)) +		{ +			mh=_dyld_get_image_header(j); +			break; +		} +	} +	return mh; +} + +static const char* lt_int_dyld_lib_install_name(mh) +	const struct mach_header *mh; +{ +/* NSAddImage is also used to get the loaded image, but it only works if the lib +   is installed, for uninstalled libs we need to check the install_names against +   each other. Note that this is still broken if DYLD_IMAGE_SUFFIX is set and a +   different lib was loaded as a result +*/ +	int j; +	struct load_command *lc; +	unsigned long offset = sizeof(struct mach_header); +	const char* retStr=NULL; +	for (j = 0; j < mh->ncmds; j++) +	{ +		lc = (struct load_command*)(((unsigned long)mh) + offset); +		if (LC_ID_DYLIB == lc->cmd) +		{ +			retStr=(char*)(((struct dylib_command*)lc)->dylib.name.offset + +									(unsigned long)lc); +		} +		offset += lc->cmdsize; +	} +	return retStr; +} + +static const struct mach_header * +lt_int_dyld_match_loaded_lib_by_install_name(const char *name) +{ +	int i=_dyld_image_count(); +	int j; +	const struct mach_header *mh=NULL; +	const char *id=NULL; +	for (j = 0; j < i; j++) +	{ +		id=lt_int_dyld_lib_install_name(_dyld_get_image_header(j)); +		if ((id) && (!strcmp(id,name))) +		{ +			mh=_dyld_get_image_header(j); +			break; +		} +	} +	return mh; +} + +static NSSymbol +lt_int_dyld_NSlookupSymbolInLinkedLibs(symbol,mh) +	const char *symbol; +	const struct mach_header *mh; +{ +	/* Safe to assume our mh is good */ +	int j; +	struct load_command *lc; +	unsigned long offset = sizeof(struct mach_header); +	NSSymbol retSym = 0; +	const struct mach_header *mh1; +	if ((ltdl_NSLookupSymbolInImage) && NSIsSymbolNameDefined(symbol) ) +	{ +		for (j = 0; j < mh->ncmds; j++) +		{ +			lc = (struct load_command*)(((unsigned long)mh) + offset); +			if ((LC_LOAD_DYLIB == lc->cmd) || (LC_LOAD_WEAK_DYLIB == lc->cmd)) +			{ +				mh1=lt_int_dyld_match_loaded_lib_by_install_name((char*)(((struct dylib_command*)lc)->dylib.name.offset + +										(unsigned long)lc)); +				if (!mh1) +				{ +					/* Maybe NSAddImage can find it */ +					mh1=ltdl_NSAddImage((char*)(((struct dylib_command*)lc)->dylib.name.offset + +										(unsigned long)lc), +										NSADDIMAGE_OPTION_RETURN_ONLY_IF_LOADED + +										NSADDIMAGE_OPTION_WITH_SEARCHING + +										NSADDIMAGE_OPTION_RETURN_ON_ERROR ); +				} +				if (mh1) +				{ +					retSym = ltdl_NSLookupSymbolInImage(mh1, +											symbol, +											NSLOOKUPSYMBOLINIMAGE_OPTION_BIND_NOW +											| NSLOOKUPSYMBOLINIMAGE_OPTION_RETURN_ON_ERROR +											); +					if (retSym) break; +				} +			} +			offset += lc->cmdsize; +		} +	} +	return retSym; +} + +static int +sys_dyld_init() +{ +	int retCode = 0; +	int err = 0; +	if (!_dyld_present()) { +		retCode=1; +	} +	else { +      err = _dyld_func_lookup("__dyld_NSAddImage",(unsigned long*)<dl_NSAddImage); +      err = _dyld_func_lookup("__dyld_NSLookupSymbolInImage",(unsigned long*)<dl_NSLookupSymbolInImage); +      err = _dyld_func_lookup("__dyld_NSIsSymbolNameDefinedInImage",(unsigned long*)<dl_NSIsSymbolNameDefinedInImage); +      err = _dyld_func_lookup("__dyld_NSMakePrivateModulePublic",(unsigned long*)<dl_NSMakePrivateModulePublic); +    } + return retCode; +} + +static lt_module +sys_dyld_open (loader_data, filename) +     lt_user_data loader_data; +     const char *filename; +{ +	lt_module   module   = 0; +	NSObjectFileImage ofi = 0; +	NSObjectFileImageReturnCode ofirc; + +  	if (!filename) +  		return (lt_module)-1; +	ofirc = NSCreateObjectFileImageFromFile(filename, &ofi); +	switch (ofirc) +	{ +		case NSObjectFileImageSuccess: +			module = NSLinkModule(ofi, filename, +						NSLINKMODULE_OPTION_RETURN_ON_ERROR +						 | NSLINKMODULE_OPTION_PRIVATE +						 | NSLINKMODULE_OPTION_BINDNOW); +			NSDestroyObjectFileImage(ofi); +			if (module) +				ltdl_NSMakePrivateModulePublic(module); +			break; +		case NSObjectFileImageInappropriateFile: +		    if (ltdl_NSIsSymbolNameDefinedInImage && ltdl_NSLookupSymbolInImage) +		    { +				module = (lt_module)ltdl_NSAddImage(filename, NSADDIMAGE_OPTION_RETURN_ON_ERROR); +				break; +			} +		default: +			LT_DLMUTEX_SETERROR (lt_int_dyld_error(LT_DLSTRERROR(CANNOT_OPEN))); +			return 0; +	} +	if (!module) LT_DLMUTEX_SETERROR (lt_int_dyld_error(LT_DLSTRERROR(CANNOT_OPEN))); +  return module; +} + +static int +sys_dyld_close (loader_data, module) +     lt_user_data loader_data; +     lt_module module; +{ +	int retCode = 0; +	int flags = 0; +	if (module == (lt_module)-1) return 0; +#ifdef __BIG_ENDIAN__ +  	if (((struct mach_header *)module)->magic == MH_MAGIC) +#else +    if (((struct mach_header *)module)->magic == MH_CIGAM) +#endif +	{ +	  LT_DLMUTEX_SETERROR("Can not close a dylib"); +	  retCode = 1; +	} +	else +	{ +#if 1 +/* Currently, if a module contains c++ static destructors and it is unloaded, we +   get a segfault in atexit(), due to compiler and dynamic loader differences of +   opinion, this works around that. +*/ +		if ((const struct section *)NULL != +		   getsectbynamefromheader(lt_int_dyld_get_mach_header_from_nsmodule(module), +		   "__DATA","__mod_term_func")) +		{ +			flags += NSUNLINKMODULE_OPTION_KEEP_MEMORY_MAPPED; +		} +#endif +#ifdef __ppc__ +			flags += NSUNLINKMODULE_OPTION_RESET_LAZY_REFERENCES; +#endif +		if (!NSUnLinkModule(module,flags)) +		{ +			retCode=1; +			LT_DLMUTEX_SETERROR (lt_int_dyld_error(LT_DLSTRERROR(CANNOT_CLOSE))); +		} +	} + + return retCode; +} + +static lt_ptr +sys_dyld_sym (loader_data, module, symbol) +     lt_user_data loader_data; +     lt_module module; +     const char *symbol; +{ +	lt_ptr address = 0; +  	NSSymbol *nssym = 0; +  	void *unused; +  	const struct mach_header *mh=NULL; +  	char saveError[256] = "Symbol not found"; +  	if (module == (lt_module)-1) +  	{ +  		_dyld_lookup_and_bind(symbol,(unsigned long*)&address,&unused); +  		return address; +  	} +#ifdef __BIG_ENDIAN__ +  	if (((struct mach_header *)module)->magic == MH_MAGIC) +#else +    if (((struct mach_header *)module)->magic == MH_CIGAM) +#endif +  	{ +  	    if (ltdl_NSIsSymbolNameDefinedInImage && ltdl_NSLookupSymbolInImage) +  	    { +  	    	mh=module; +			if (ltdl_NSIsSymbolNameDefinedInImage((struct mach_header*)module,symbol)) +			{ +				nssym = ltdl_NSLookupSymbolInImage((struct mach_header*)module, +											symbol, +											NSLOOKUPSYMBOLINIMAGE_OPTION_BIND_NOW +											| NSLOOKUPSYMBOLINIMAGE_OPTION_RETURN_ON_ERROR +											); +			} +	    } + +  	} +  else { +	nssym = NSLookupSymbolInModule(module, symbol); +	} +	if (!nssym) +	{ +		strncpy(saveError, lt_int_dyld_error(LT_DLSTRERROR(SYMBOL_NOT_FOUND)), 255); +		saveError[255] = 0; +		if (!mh) mh=lt_int_dyld_get_mach_header_from_nsmodule(module); +		nssym = lt_int_dyld_NSlookupSymbolInLinkedLibs(symbol,mh); +	} +	if (!nssym) +	{ +		LT_DLMUTEX_SETERROR (saveError); +		return NULL; +	} +	return NSAddressOfSymbol(nssym); +} + +static struct lt_user_dlloader sys_dyld = +  { "_", sys_dyld_open, sys_dyld_close, sys_dyld_sym, 0, 0 }; + + +#endif /* HAVE_DYLD */ + + +/* --- DLPREOPEN() INTERFACE LOADER --- */ + + +/* emulate dynamic linking using preloaded_symbols */ + +typedef struct lt_dlsymlists_t +{ +  struct lt_dlsymlists_t       *next; +  const lt_dlsymlist	       *syms; +} lt_dlsymlists_t; + +static	const lt_dlsymlist     *default_preloaded_symbols	= 0; +static	lt_dlsymlists_t	       *preloaded_symbols		= 0; + +static int +presym_init (loader_data) +     lt_user_data loader_data; +{ +  int errors = 0; + +  LT_DLMUTEX_LOCK (); + +  preloaded_symbols = 0; +  if (default_preloaded_symbols) +    { +      errors = lt_dlpreload (default_preloaded_symbols); +    } + +  LT_DLMUTEX_UNLOCK (); + +  return errors; +} + +static int +presym_free_symlists () +{ +  lt_dlsymlists_t *lists; + +  LT_DLMUTEX_LOCK (); + +  lists = preloaded_symbols; +  while (lists) +    { +      lt_dlsymlists_t	*tmp = lists; + +      lists = lists->next; +      LT_DLFREE (tmp); +    } +  preloaded_symbols = 0; + +  LT_DLMUTEX_UNLOCK (); + +  return 0; +} + +static int +presym_exit (loader_data) +     lt_user_data loader_data; +{ +  presym_free_symlists (); +  return 0; +} + +static int +presym_add_symlist (preloaded) +     const lt_dlsymlist *preloaded; +{ +  lt_dlsymlists_t *tmp; +  lt_dlsymlists_t *lists; +  int		   errors   = 0; + +  LT_DLMUTEX_LOCK (); + +  lists = preloaded_symbols; +  while (lists) +    { +      if (lists->syms == preloaded) +	{ +	  goto done; +	} +      lists = lists->next; +    } + +  tmp = LT_EMALLOC (lt_dlsymlists_t, 1); +  if (tmp) +    { +      memset (tmp, 0, sizeof(lt_dlsymlists_t)); +      tmp->syms = preloaded; +      tmp->next = preloaded_symbols; +      preloaded_symbols = tmp; +    } +  else +    { +      ++errors; +    } + + done: +  LT_DLMUTEX_UNLOCK (); +  return errors; +} + +static lt_module +presym_open (loader_data, filename) +     lt_user_data loader_data; +     const char *filename; +{ +  lt_dlsymlists_t *lists; +  lt_module	   module = (lt_module) 0; + +  LT_DLMUTEX_LOCK (); +  lists = preloaded_symbols; + +  if (!lists) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (NO_SYMBOLS)); +      goto done; +    } + +  /* Can't use NULL as the reflective symbol header, as NULL is +     used to mark the end of the entire symbol list.  Self-dlpreopened +     symbols follow this magic number, chosen to be an unlikely +     clash with a real module name.  */ +  if (!filename) +    { +      filename = "@PROGRAM@"; +    } + +  while (lists) +    { +      const lt_dlsymlist *syms = lists->syms; + +      while (syms->name) +	{ +	  if (!syms->address && strcmp(syms->name, filename) == 0) +	    { +	      module = (lt_module) syms; +	      goto done; +	    } +	  ++syms; +	} + +      lists = lists->next; +    } + +  LT_DLMUTEX_SETERROR (LT_DLSTRERROR (FILE_NOT_FOUND)); + + done: +  LT_DLMUTEX_UNLOCK (); +  return module; +} + +static int +presym_close (loader_data, module) +     lt_user_data loader_data; +     lt_module module; +{ +  /* Just to silence gcc -Wall */ +  module = 0; +  return 0; +} + +static lt_ptr +presym_sym (loader_data, module, symbol) +     lt_user_data loader_data; +     lt_module module; +     const char *symbol; +{ +  lt_dlsymlist *syms = (lt_dlsymlist*) module; + +  ++syms; +  while (syms->address) +    { +      if (strcmp(syms->name, symbol) == 0) +	{ +	  return syms->address; +	} + +    ++syms; +  } + +  LT_DLMUTEX_SETERROR (LT_DLSTRERROR (SYMBOL_NOT_FOUND)); + +  return 0; +} + +static struct lt_user_dlloader presym = { +  0, presym_open, presym_close, presym_sym, presym_exit, 0 +}; + + + + + +/* --- DYNAMIC MODULE LOADING --- */ + + +/* The type of a function used at each iteration of  foreach_dirinpath().  */ +typedef int	foreach_callback_func LT_PARAMS((char *filename, lt_ptr data1, +						 lt_ptr data2)); + +static	int	foreach_dirinpath     LT_PARAMS((const char *search_path, +						 const char *base_name, +						 foreach_callback_func *func, +						 lt_ptr data1, lt_ptr data2)); + +static	int	find_file_callback    LT_PARAMS((char *filename, lt_ptr data, +						 lt_ptr ignored)); +static	int	find_handle_callback  LT_PARAMS((char *filename, lt_ptr data, +						 lt_ptr ignored)); +static	int	foreachfile_callback  LT_PARAMS((char *filename, lt_ptr data1, +						 lt_ptr data2)); + + +static	int     canonicalize_path     LT_PARAMS((const char *path, +						 char **pcanonical)); +static	int	argzize_path 	      LT_PARAMS((const char *path, +						 char **pargz, +						 size_t *pargz_len)); +static	FILE   *find_file	      LT_PARAMS((const char *search_path, +						 const char *base_name, +						 char **pdir)); +static	lt_dlhandle *find_handle      LT_PARAMS((const char *search_path, +						 const char *base_name, +						 lt_dlhandle *handle)); +static	int	find_module	      LT_PARAMS((lt_dlhandle *handle, +						 const char *dir, +						 const char *libdir, +						 const char *dlname, +						 const char *old_name, +						 int installed)); +static	int	free_vars	      LT_PARAMS((char *dlname, char *oldname, +						 char *libdir, char *deplibs)); +static	int	load_deplibs	      LT_PARAMS((lt_dlhandle handle, +						 char *deplibs)); +static	int	trim		      LT_PARAMS((char **dest, +						 const char *str)); +static	int	try_dlopen	      LT_PARAMS((lt_dlhandle *handle, +						 const char *filename)); +static	int	tryall_dlopen	      LT_PARAMS((lt_dlhandle *handle, +						 const char *filename)); +static	int	unload_deplibs	      LT_PARAMS((lt_dlhandle handle)); +static	int	lt_argz_insert	      LT_PARAMS((char **pargz, +						 size_t *pargz_len, +						 char *before, +						 const char *entry)); +static	int	lt_argz_insertinorder LT_PARAMS((char **pargz, +						 size_t *pargz_len, +						 const char *entry)); +static	int	lt_argz_insertdir     LT_PARAMS((char **pargz, +						 size_t *pargz_len, +						 const char *dirnam, +						 struct dirent *dp)); +static	int	lt_dlpath_insertdir   LT_PARAMS((char **ppath, +						 char *before, +						 const char *dir)); +static	int	list_files_by_dir     LT_PARAMS((const char *dirnam, +						 char **pargz, +						 size_t *pargz_len)); +static	int	file_not_found	      LT_PARAMS((void)); + +static	char	       *user_search_path= 0; +static	lt_dlloader    *loaders		= 0; +static	lt_dlhandle	handles 	= 0; +static	int		initialized 	= 0; + +/* Initialize libltdl. */ +int +lt_dlinit () +{ +  int	      errors   = 0; + +  LT_DLMUTEX_LOCK (); + +  /* Initialize only at first call. */ +  if (++initialized == 1) +    { +      handles = 0; +      user_search_path = 0; /* empty search path */ + +#if HAVE_LIBDL +      errors += lt_dlloader_add (lt_dlloader_next (0), &sys_dl, "dlopen"); +#endif +#if HAVE_SHL_LOAD +      errors += lt_dlloader_add (lt_dlloader_next (0), &sys_shl, "dlopen"); +#endif +#ifdef __WINDOWS__ +      errors += lt_dlloader_add (lt_dlloader_next (0), &sys_wll, "dlopen"); +#endif +#ifdef __BEOS__ +      errors += lt_dlloader_add (lt_dlloader_next (0), &sys_bedl, "dlopen"); +#endif +#if HAVE_DLD +      errors += lt_dlloader_add (lt_dlloader_next (0), &sys_dld, "dld"); +#endif +#if HAVE_DYLD +       errors += lt_dlloader_add (lt_dlloader_next (0), &sys_dyld, "dyld"); +       errors += sys_dyld_init(); +#endif +      errors += lt_dlloader_add (lt_dlloader_next (0), &presym, "dlpreload"); + +      if (presym_init (presym.dlloader_data)) +	{ +	  LT_DLMUTEX_SETERROR (LT_DLSTRERROR (INIT_LOADER)); +	  ++errors; +	} +      else if (errors != 0) +	{ +	  LT_DLMUTEX_SETERROR (LT_DLSTRERROR (DLOPEN_NOT_SUPPORTED)); +	  ++errors; +	} +    } + +  LT_DLMUTEX_UNLOCK (); + +  return errors; +} + +int +lt_dlpreload (preloaded) +     const lt_dlsymlist *preloaded; +{ +  int errors = 0; + +  if (preloaded) +    { +      errors = presym_add_symlist (preloaded); +    } +  else +    { +      presym_free_symlists(); + +      LT_DLMUTEX_LOCK (); +      if (default_preloaded_symbols) +	{ +	  errors = lt_dlpreload (default_preloaded_symbols); +	} +      LT_DLMUTEX_UNLOCK (); +    } + +  return errors; +} + +int +lt_dlpreload_default (preloaded) +     const lt_dlsymlist *preloaded; +{ +  LT_DLMUTEX_LOCK (); +  default_preloaded_symbols = preloaded; +  LT_DLMUTEX_UNLOCK (); +  return 0; +} + +int +lt_dlexit () +{ +  /* shut down libltdl */ +  lt_dlloader *loader; +  int	       errors   = 0; + +  LT_DLMUTEX_LOCK (); +  loader = loaders; + +  if (!initialized) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (SHUTDOWN)); +      ++errors; +      goto done; +    } + +  /* shut down only at last call. */ +  if (--initialized == 0) +    { +      int	level; + +      while (handles && LT_DLIS_RESIDENT (handles)) +	{ +	  handles = handles->next; +	} + +      /* close all modules */ +      for (level = 1; handles; ++level) +	{ +	  lt_dlhandle cur = handles; +	  int saw_nonresident = 0; + +	  while (cur) +	    { +	      lt_dlhandle tmp = cur; +	      cur = cur->next; +	      if (!LT_DLIS_RESIDENT (tmp)) +		saw_nonresident = 1; +	      if (!LT_DLIS_RESIDENT (tmp) && tmp->info.ref_count <= level) +		{ +		  if (lt_dlclose (tmp)) +		    { +		      ++errors; +		    } +		} +	    } +	  /* done if only resident modules are left */ +	  if (!saw_nonresident) +	    break; +	} + +      /* close all loaders */ +      while (loader) +	{ +	  lt_dlloader *next = loader->next; +	  lt_user_data data = loader->dlloader_data; +	  if (loader->dlloader_exit && loader->dlloader_exit (data)) +	    { +	      ++errors; +	    } + +	  LT_DLMEM_REASSIGN (loader, next); +	} +      loaders = 0; +    } + + done: +  LT_DLMUTEX_UNLOCK (); +  return errors; +} + +static int +tryall_dlopen (handle, filename) +     lt_dlhandle *handle; +     const char *filename; +{ +  lt_dlhandle	 cur; +  lt_dlloader   *loader; +  const char	*saved_error; +  int		 errors		= 0; + +  LT_DLMUTEX_GETERROR (saved_error); +  LT_DLMUTEX_LOCK (); + +  cur	 = handles; +  loader = loaders; + +  /* check whether the module was already opened */ +  while (cur) +    { +      /* try to dlopen the program itself? */ +      if (!cur->info.filename && !filename) +	{ +	  break; +	} + +      if (cur->info.filename && filename +	  && strcmp (cur->info.filename, filename) == 0) +	{ +	  break; +	} + +      cur = cur->next; +    } + +  if (cur) +    { +      ++cur->info.ref_count; +      *handle = cur; +      goto done; +    } + +  cur = *handle; +  if (filename) +    { +      /* Comment out the check of file permissions using access. +	 This call seems to always return -1 with error EACCES. +      */ +      /* We need to catch missing file errors early so that +	 file_not_found() can detect what happened. +      if (access (filename, R_OK) != 0) +	{ +	  LT_DLMUTEX_SETERROR (LT_DLSTRERROR (FILE_NOT_FOUND)); +	  ++errors; +	  goto done; +	} */ + +      cur->info.filename = lt_estrdup (filename); +      if (!cur->info.filename) +	{ +	  ++errors; +	  goto done; +	} +    } +  else +    { +      cur->info.filename = 0; +    } + +  while (loader) +    { +      lt_user_data data = loader->dlloader_data; + +      cur->module = loader->module_open (data, filename); + +      if (cur->module != 0) +	{ +	  break; +	} +      loader = loader->next; +    } + +  if (!loader) +    { +      LT_DLFREE (cur->info.filename); +      ++errors; +      goto done; +    } + +  cur->loader	= loader; +  LT_DLMUTEX_SETERROR (saved_error); + + done: +  LT_DLMUTEX_UNLOCK (); + +  return errors; +} + +static int +tryall_dlopen_module (handle, prefix, dirname, dlname) +     lt_dlhandle *handle; +     const char *prefix; +     const char *dirname; +     const char *dlname; +{ +  int      error	= 0; +  char     *filename	= 0; +  size_t   filename_len	= 0; +  size_t   dirname_len	= LT_STRLEN (dirname); + +  assert (handle); +  assert (dirname); +  assert (dlname); +#ifdef LT_DIRSEP_CHAR +  /* Only canonicalized names (i.e. with DIRSEP chars already converted) +     should make it into this function:  */ +  assert (strchr (dirname, LT_DIRSEP_CHAR) == 0); +#endif + +  if (dirname_len > 0) +    if (dirname[dirname_len -1] == '/') +      --dirname_len; +  filename_len = dirname_len + 1 + LT_STRLEN (dlname); + +  /* Allocate memory, and combine DIRNAME and MODULENAME into it. +     The PREFIX (if any) is handled below.  */ +  filename  = LT_EMALLOC (char, dirname_len + 1 + filename_len + 1); +  if (!filename) +    return 1; + +  sprintf (filename, "%.*s/%s", (int) dirname_len, dirname, dlname); + +  /* Now that we have combined DIRNAME and MODULENAME, if there is +     also a PREFIX to contend with, simply recurse with the arguments +     shuffled.  Otherwise, attempt to open FILENAME as a module.  */ +  if (prefix) +    { +      error += tryall_dlopen_module (handle, +				     (const char *) 0, prefix, filename); +    } +  else if (tryall_dlopen (handle, filename) != 0) +    { +      ++error; +    } + +  LT_DLFREE (filename); +  return error; +} + +static int +find_module (handle, dir, libdir, dlname, old_name, installed) +     lt_dlhandle *handle; +     const char *dir; +     const char *libdir; +     const char *dlname; +     const char *old_name; +     int installed; +{ +  /* Try to open the old library first; if it was dlpreopened, +     we want the preopened version of it, even if a dlopenable +     module is available.  */ +  if (old_name && tryall_dlopen (handle, old_name) == 0) +    { +      return 0; +    } + +  /* Try to open the dynamic library.  */ +  if (dlname) +    { +      /* try to open the installed module */ +      if (installed && libdir) +	{ +	  if (tryall_dlopen_module (handle, +				    (const char *) 0, libdir, dlname) == 0) +	    return 0; +	} + +      /* try to open the not-installed module */ +      if (!installed) +	{ +	  if (tryall_dlopen_module (handle, dir, objdir, dlname) == 0) +	    return 0; +	} + +      /* maybe it was moved to another directory */ +      { +	  if (dir && (tryall_dlopen_module (handle, +				    (const char *) 0, dir, dlname) == 0)) +	    return 0; +      } +    } + +  return 1; +} + + +static int +canonicalize_path (path, pcanonical) +     const char *path; +     char **pcanonical; +{ +  char *canonical = 0; + +  assert (path && *path); +  assert (pcanonical); + +  canonical = LT_EMALLOC (char, 1+ LT_STRLEN (path)); +  if (!canonical) +    return 1; + +  { +    size_t dest = 0; +    size_t src; +    for (src = 0; path[src] != LT_EOS_CHAR; ++src) +      { +	/* Path separators are not copied to the beginning or end of +	   the destination, or if another separator would follow +	   immediately.  */ +	if (path[src] == LT_PATHSEP_CHAR) +	  { +	    if ((dest == 0) +		|| (path[1+ src] == LT_PATHSEP_CHAR) +		|| (path[1+ src] == LT_EOS_CHAR)) +	      continue; +	  } + +	/* Anything other than a directory separator is copied verbatim.  */ +	if ((path[src] != '/') +#ifdef LT_DIRSEP_CHAR +	    && (path[src] != LT_DIRSEP_CHAR) +#endif +	    ) +	  { +	    canonical[dest++] = path[src]; +	  } +	/* Directory separators are converted and copied only if they are +	   not at the end of a path -- i.e. before a path separator or +	   NULL terminator.  */ +	else if ((path[1+ src] != LT_PATHSEP_CHAR) +		 && (path[1+ src] != LT_EOS_CHAR) +#ifdef LT_DIRSEP_CHAR +		 && (path[1+ src] != LT_DIRSEP_CHAR) +#endif +		 && (path[1+ src] != '/')) +	  { +	    canonical[dest++] = '/'; +	  } +      } + +    /* Add an end-of-string marker at the end.  */ +    canonical[dest] = LT_EOS_CHAR; +  } + +  /* Assign new value.  */ +  *pcanonical = canonical; + +  return 0; +} + +static int +argzize_path (path, pargz, pargz_len) +     const char *path; +     char **pargz; +     size_t *pargz_len; +{ +  error_t error; + +  assert (path); +  assert (pargz); +  assert (pargz_len); + +  if ((error = argz_create_sep (path, LT_PATHSEP_CHAR, pargz, pargz_len))) +    { +      switch (error) +	{ +	case ENOMEM: +	  LT_DLMUTEX_SETERROR (LT_DLSTRERROR (NO_MEMORY)); +	  break; +	default: +	  LT_DLMUTEX_SETERROR (LT_DLSTRERROR (UNKNOWN)); +	  break; +	} + +      return 1; +    } + +  return 0; +} + +/* Repeatedly call FUNC with each LT_PATHSEP_CHAR delimited element +   of SEARCH_PATH and references to DATA1 and DATA2, until FUNC returns +   non-zero or all elements are exhausted.  If BASE_NAME is non-NULL, +   it is appended to each SEARCH_PATH element before FUNC is called.  */ +static int +foreach_dirinpath (search_path, base_name, func, data1, data2) +     const char *search_path; +     const char *base_name; +     foreach_callback_func *func; +     lt_ptr data1; +     lt_ptr data2; +{ +  int	 result		= 0; +  int	 filenamesize	= 0; +  size_t lenbase	= LT_STRLEN (base_name); +  size_t argz_len	= 0; +  char *argz		= 0; +  char *filename	= 0; +  char *canonical	= 0; + +  LT_DLMUTEX_LOCK (); + +  if (!search_path || !*search_path) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (FILE_NOT_FOUND)); +      goto cleanup; +    } + +  if (canonicalize_path (search_path, &canonical) != 0) +    goto cleanup; + +  if (argzize_path (canonical, &argz, &argz_len) != 0) +    goto cleanup; + +  { +    char *dir_name = 0; +    while ((dir_name = argz_next (argz, argz_len, dir_name))) +      { +	size_t lendir = LT_STRLEN (dir_name); + +	if (lendir +1 +lenbase >= (size_t)filenamesize) +	{ +	  LT_DLFREE (filename); +	  filenamesize	= lendir +1 +lenbase +1; /* "/d" + '/' + "f" + '\0' */ +	  filename	= LT_EMALLOC (char, filenamesize); +	  if (!filename) +	    goto cleanup; +	} + +	assert ((size_t)filenamesize > lendir); +	strcpy (filename, dir_name); + +	if (base_name && *base_name) +	  { +	    if (filename[lendir -1] != '/') +	      filename[lendir++] = '/'; +	    strcpy (filename +lendir, base_name); +	  } + +	if ((result = (*func) (filename, data1, data2))) +	  { +	    break; +	  } +      } +  } + + cleanup: +  LT_DLFREE (argz); +  LT_DLFREE (canonical); +  LT_DLFREE (filename); + +  LT_DLMUTEX_UNLOCK (); + +  return result; +} + +/* If FILEPATH can be opened, store the name of the directory component +   in DATA1, and the opened FILE* structure address in DATA2.  Otherwise +   DATA1 is unchanged, but DATA2 is set to a pointer to NULL.  */ +static int +find_file_callback (filename, data1, data2) +     char *filename; +     lt_ptr data1; +     lt_ptr data2; +{ +  char	     **pdir	= (char **) data1; +  FILE	     **pfile	= (FILE **) data2; +  int	     is_done	= 0; + +  assert (filename && *filename); +  assert (pdir); +  assert (pfile); + +  if ((*pfile = fopen (filename, LT_READTEXT_MODE))) +    { +      char *dirend = strrchr (filename, '/'); + +      if (dirend > filename) +	*dirend   = LT_EOS_CHAR; + +      LT_DLFREE (*pdir); +      *pdir   = lt_estrdup (filename); +      is_done = (*pdir == 0) ? -1 : 1; +    } + +  return is_done; +} + +static FILE * +find_file (search_path, base_name, pdir) +     const char *search_path; +     const char *base_name; +     char **pdir; +{ +  FILE *file = 0; + +  foreach_dirinpath (search_path, base_name, find_file_callback, pdir, &file); + +  return file; +} + +static int +find_handle_callback (filename, data, ignored) +     char *filename; +     lt_ptr data; +     lt_ptr ignored; +{ +  lt_dlhandle  *handle		= (lt_dlhandle *) data; +  int		notfound	= access (filename, R_OK); + +  /* Bail out if file cannot be read...  */ +  if (notfound) +    return 0; + +  /* Try to dlopen the file, but do not continue searching in any +     case.  */ +  if (tryall_dlopen (handle, filename) != 0) +    *handle = 0; + +  return 1; +} + +/* If HANDLE was found return it, otherwise return 0.  If HANDLE was +   found but could not be opened, *HANDLE will be set to 0.  */ +static lt_dlhandle * +find_handle (search_path, base_name, handle) +     const char *search_path; +     const char *base_name; +     lt_dlhandle *handle; +{ +  if (!search_path) +    return 0; + +  if (!foreach_dirinpath (search_path, base_name, find_handle_callback, +			  handle, 0)) +    return 0; + +  return handle; +} + +static int +load_deplibs (handle, deplibs) +     lt_dlhandle handle; +     char *deplibs; +{ +#if LTDL_DLOPEN_DEPLIBS +  char	*p, *save_search_path = 0; +  int   depcount = 0; +  int	i; +  char	**names = 0; +#endif +  int	errors = 0; + +  handle->depcount = 0; + +#if LTDL_DLOPEN_DEPLIBS +  if (!deplibs) +    { +      return errors; +    } +  ++errors; + +  LT_DLMUTEX_LOCK (); +  if (user_search_path) +    { +      save_search_path = lt_estrdup (user_search_path); +      if (!save_search_path) +	goto cleanup; +    } + +  /* extract search paths and count deplibs */ +  p = deplibs; +  while (*p) +    { +      if (!isspace ((int) *p)) +	{ +	  char *end = p+1; +	  while (*end && !isspace((int) *end)) +	    { +	      ++end; +	    } + +	  if (strncmp(p, "-L", 2) == 0 || strncmp(p, "-R", 2) == 0) +	    { +	      char save = *end; +	      *end = 0; /* set a temporary string terminator */ +	      if (lt_dladdsearchdir(p+2)) +		{ +		  goto cleanup; +		} +	      *end = save; +	    } +	  else +	    { +	      ++depcount; +	    } + +	  p = end; +	} +      else +	{ +	  ++p; +	} +    } + +  if (!depcount) +    { +      errors = 0; +      goto cleanup; +    } + +  names = LT_EMALLOC (char *, depcount * sizeof (char*)); +  if (!names) +    goto cleanup; + +  /* now only extract the actual deplibs */ +  depcount = 0; +  p = deplibs; +  while (*p) +    { +      if (isspace ((int) *p)) +	{ +	  ++p; +	} +      else +	{ +	  char *end = p+1; +	  while (*end && !isspace ((int) *end)) +	    { +	      ++end; +	    } + +	  if (strncmp(p, "-L", 2) != 0 && strncmp(p, "-R", 2) != 0) +	    { +	      char *name; +	      char save = *end; +	      *end = 0; /* set a temporary string terminator */ +	      if (strncmp(p, "-l", 2) == 0) +		{ +		  size_t name_len = 3+ /* "lib" */ LT_STRLEN (p + 2); +		  name = LT_EMALLOC (char, 1+ name_len); +		  if (name) +		    sprintf (name, "lib%s", p+2); +		} +	      else +		name = lt_estrdup(p); + +	      if (!name) +		goto cleanup_names; + +	      names[depcount++] = name; +	      *end = save; +	    } +	  p = end; +	} +    } + +  /* load the deplibs (in reverse order) +     At this stage, don't worry if the deplibs do not load correctly, +     they may already be statically linked into the loading application +     for instance.  There will be a more enlightening error message +     later on if the loaded module cannot resolve all of its symbols.  */ +  if (depcount) +    { +      int	j = 0; + +      handle->deplibs = (lt_dlhandle*) LT_EMALLOC (lt_dlhandle *, depcount); +      if (!handle->deplibs) +	goto cleanup; + +      for (i = 0; i < depcount; ++i) +	{ +	  handle->deplibs[j] = lt_dlopenext(names[depcount-1-i]); +	  if (handle->deplibs[j]) +	    { +	      ++j; +	    } +	} + +      handle->depcount	= j;	/* Number of successfully loaded deplibs */ +      errors		= 0; +    } + + cleanup_names: +  for (i = 0; i < depcount; ++i) +    { +      LT_DLFREE (names[i]); +    } + + cleanup: +  LT_DLFREE (names); +  /* restore the old search path */ +  if (user_search_path) { +    LT_DLFREE (user_search_path); +    user_search_path = save_search_path; +  } +  LT_DLMUTEX_UNLOCK (); + +#endif + +  return errors; +} + +static int +unload_deplibs (handle) +     lt_dlhandle handle; +{ +  int i; +  int errors = 0; + +  if (handle->depcount) +    { +      for (i = 0; i < handle->depcount; ++i) +	{ +	  if (!LT_DLIS_RESIDENT (handle->deplibs[i])) +	    { +	      errors += lt_dlclose (handle->deplibs[i]); +	    } +	} +    } + +  return errors; +} + +static int +trim (dest, str) +     char **dest; +     const char *str; +{ +  /* remove the leading and trailing "'" from str +     and store the result in dest */ +  const char *end   = strrchr (str, '\''); +  size_t len	    = LT_STRLEN (str); +  char *tmp; + +  LT_DLFREE (*dest); + +  if (!end) +    return 1; + +  if (len > 3 && str[0] == '\'') +    { +      tmp = LT_EMALLOC (char, end - str); +      if (!tmp) +	return 1; + +      strncpy(tmp, &str[1], (end - str) - 1); +      tmp[len-3] = LT_EOS_CHAR; +      *dest = tmp; +    } +  else +    { +      *dest = 0; +    } + +  return 0; +} + +static int +free_vars (dlname, oldname, libdir, deplibs) +     char *dlname; +     char *oldname; +     char *libdir; +     char *deplibs; +{ +  LT_DLFREE (dlname); +  LT_DLFREE (oldname); +  LT_DLFREE (libdir); +  LT_DLFREE (deplibs); + +  return 0; +} + +static int +try_dlopen (phandle, filename) +     lt_dlhandle *phandle; +     const char *filename; +{ +  const char *	ext		= 0; +  const char *	saved_error	= 0; +  char *	canonical	= 0; +  char *	base_name	= 0; +  char *	dir		= 0; +  char *	name		= 0; +  int		errors		= 0; +  lt_dlhandle	newhandle; + +  assert (phandle); +  assert (*phandle == 0); + +  LT_DLMUTEX_GETERROR (saved_error); + +  /* dlopen self? */ +  if (!filename) +    { +      *phandle = (lt_dlhandle) LT_EMALLOC (struct lt_dlhandle_struct, 1); +      if (*phandle == 0) +	return 1; + +      memset (*phandle, 0, sizeof(struct lt_dlhandle_struct)); +      newhandle	= *phandle; + +      /* lt_dlclose()ing yourself is very bad!  Disallow it.  */ +      LT_DLSET_FLAG (*phandle, LT_DLRESIDENT_FLAG); + +      if (tryall_dlopen (&newhandle, 0) != 0) +	{ +	  LT_DLFREE (*phandle); +	  return 1; +	} + +      goto register_handle; +    } + +  assert (filename && *filename); + +  /* Doing this immediately allows internal functions to safely +     assume only canonicalized paths are passed.  */ +  if (canonicalize_path (filename, &canonical) != 0) +    { +      ++errors; +      goto cleanup; +    } + +  /* If the canonical module name is a path (relative or absolute) +     then split it into a directory part and a name part.  */ +  base_name = strrchr (canonical, '/'); +  if (base_name) +    { +      size_t dirlen = (1+ base_name) - canonical; + +      dir = LT_EMALLOC (char, 1+ dirlen); +      if (!dir) +	{ +	  ++errors; +	  goto cleanup; +	} + +      strncpy (dir, canonical, dirlen); +      dir[dirlen] = LT_EOS_CHAR; + +      ++base_name; +    } +  else +    base_name = canonical; + +  assert (base_name && *base_name); + +  /* Check whether we are opening a libtool module (.la extension).  */ +  ext = strrchr (base_name, '.'); +  if (ext && strcmp (ext, archive_ext) == 0) +    { +      /* this seems to be a libtool module */ +      FILE *	file	 = 0; +      char *	dlname	 = 0; +      char *	old_name = 0; +      char *	libdir	 = 0; +      char *	deplibs	 = 0; +      char *    line	 = 0; +      size_t	line_len; + +      /* if we can't find the installed flag, it is probably an +	 installed libtool archive, produced with an old version +	 of libtool */ +      int	installed = 1; + +      /* extract the module name from the file name */ +      name = LT_EMALLOC (char, ext - base_name + 1); +      if (!name) +	{ +	  ++errors; +	  goto cleanup; +	} + +      /* canonicalize the module name */ +      { +        size_t i; +        for (i = 0; i < (size_t)(ext - base_name); ++i) +	  { +	    if (isalnum ((int)(base_name[i]))) +	      { +	        name[i] = base_name[i]; +	      } +	    else +	      { +	        name[i] = '_'; +	      } +	  } +        name[ext - base_name] = LT_EOS_CHAR; +      } + +      /* Now try to open the .la file.  If there is no directory name +         component, try to find it first in user_search_path and then other +         prescribed paths.  Otherwise (or in any case if the module was not +         yet found) try opening just the module name as passed.  */ +      if (!dir) +	{ +	  const char *search_path; + +	  LT_DLMUTEX_LOCK (); +	  search_path = user_search_path; +	  if (search_path) +	    file = find_file (user_search_path, base_name, &dir); +	  LT_DLMUTEX_UNLOCK (); + +	  if (!file) +	    { +	      search_path = getenv (LTDL_SEARCHPATH_VAR); +	      if (search_path) +		file = find_file (search_path, base_name, &dir); +	    } + +#ifdef LTDL_SHLIBPATH_VAR +	  if (!file) +	    { +	      search_path = getenv (LTDL_SHLIBPATH_VAR); +	      if (search_path) +		file = find_file (search_path, base_name, &dir); +	    } +#endif +#ifdef LTDL_SYSSEARCHPATH +	  if (!file && sys_search_path) +	    { +	      file = find_file (sys_search_path, base_name, &dir); +	    } +#endif +	} +      if (!file) +	{ +	  file = fopen (filename, LT_READTEXT_MODE); +	} + +      /* If we didn't find the file by now, it really isn't there.  Set +	 the status flag, and bail out.  */ +      if (!file) +	{ +	  LT_DLMUTEX_SETERROR (LT_DLSTRERROR (FILE_NOT_FOUND)); +	  ++errors; +	  goto cleanup; +	} + +      line_len = LT_FILENAME_MAX; +      line = LT_EMALLOC (char, line_len); +      if (!line) +	{ +	  fclose (file); +	  ++errors; +	  goto cleanup; +	} + +      /* read the .la file */ +      while (!feof (file)) +	{ +	  if (!fgets (line, (int) line_len, file)) +	    { +	      break; +	    } + +	  /* Handle the case where we occasionally need to read a line +	     that is longer than the initial buffer size.  */ +	  while ((line[LT_STRLEN(line) -1] != '\n') && (!feof (file))) +	    { +	      line = LT_DLREALLOC (char, line, line_len *2); +	      if (!fgets (&line[line_len -1], (int) line_len +1, file)) +		{ +		  break; +		} +	      line_len *= 2; +	    } + +	  if (line[0] == '\n' || line[0] == '#') +	    { +	      continue; +	    } + +#undef  STR_DLNAME +#define STR_DLNAME	"dlname=" +	  if (strncmp (line, STR_DLNAME, sizeof (STR_DLNAME) - 1) == 0) +	    { +	      errors += trim (&dlname, &line[sizeof (STR_DLNAME) - 1]); +	    } + +#undef  STR_OLD_LIBRARY +#define STR_OLD_LIBRARY	"old_library=" +	  else if (strncmp (line, STR_OLD_LIBRARY, +			    sizeof (STR_OLD_LIBRARY) - 1) == 0) +	    { +	      errors += trim (&old_name, &line[sizeof (STR_OLD_LIBRARY) - 1]); +	    } +#undef  STR_LIBDIR +#define STR_LIBDIR	"libdir=" +	  else if (strncmp (line, STR_LIBDIR, sizeof (STR_LIBDIR) - 1) == 0) +	    { +	      errors += trim (&libdir, &line[sizeof(STR_LIBDIR) - 1]); +	    } + +#undef  STR_DL_DEPLIBS +#define STR_DL_DEPLIBS	"dependency_libs=" +	  else if (strncmp (line, STR_DL_DEPLIBS, +			    sizeof (STR_DL_DEPLIBS) - 1) == 0) +	    { +	      errors += trim (&deplibs, &line[sizeof (STR_DL_DEPLIBS) - 1]); +	    } +	  else if (strcmp (line, "installed=yes\n") == 0) +	    { +	      installed = 1; +	    } +	  else if (strcmp (line, "installed=no\n") == 0) +	    { +	      installed = 0; +	    } + +#undef  STR_LIBRARY_NAMES +#define STR_LIBRARY_NAMES "library_names=" +	  else if (! dlname && strncmp (line, STR_LIBRARY_NAMES, +					sizeof (STR_LIBRARY_NAMES) - 1) == 0) +	    { +	      char *last_libname; +	      errors += trim (&dlname, &line[sizeof (STR_LIBRARY_NAMES) - 1]); +	      if (!errors +		  && dlname +		  && (last_libname = strrchr (dlname, ' ')) != 0) +		{ +		  last_libname = lt_estrdup (last_libname + 1); +		  if (!last_libname) +		    { +		      ++errors; +		      goto cleanup; +		    } +		  LT_DLMEM_REASSIGN (dlname, last_libname); +		} +	    } + +	  if (errors) +	    break; +	} + +      fclose (file); +      LT_DLFREE (line); + +      /* allocate the handle */ +      *phandle = (lt_dlhandle) LT_EMALLOC (struct lt_dlhandle_struct, 1); +      if (*phandle == 0) +	++errors; + +      if (errors) +	{ +	  free_vars (dlname, old_name, libdir, deplibs); +	  LT_DLFREE (*phandle); +	  goto cleanup; +	} + +      assert (*phandle); + +      memset (*phandle, 0, sizeof(struct lt_dlhandle_struct)); +      if (load_deplibs (*phandle, deplibs) == 0) +	{ +	  newhandle = *phandle; +	  /* find_module may replace newhandle */ +	  if (find_module (&newhandle, dir, libdir, dlname, old_name, installed)) +	    { +	      unload_deplibs (*phandle); +	      ++errors; +	    } +	} +      else +	{ +	  ++errors; +	} + +      free_vars (dlname, old_name, libdir, deplibs); +      if (errors) +	{ +	  LT_DLFREE (*phandle); +	  goto cleanup; +	} + +      if (*phandle != newhandle) +	{ +	  unload_deplibs (*phandle); +	} +    } +  else +    { +      /* not a libtool module */ +      *phandle = (lt_dlhandle) LT_EMALLOC (struct lt_dlhandle_struct, 1); +      if (*phandle == 0) +	{ +	  ++errors; +	  goto cleanup; +	} + +      memset (*phandle, 0, sizeof (struct lt_dlhandle_struct)); +      newhandle = *phandle; + +      /* If the module has no directory name component, try to find it +	 first in user_search_path and then other prescribed paths. +	 Otherwise (or in any case if the module was not yet found) try +	 opening just the module name as passed.  */ +      if ((dir || (!find_handle (user_search_path, base_name, &newhandle) +		   && !find_handle (getenv (LTDL_SEARCHPATH_VAR), base_name, +				    &newhandle) +#ifdef LTDL_SHLIBPATH_VAR +		   && !find_handle (getenv (LTDL_SHLIBPATH_VAR), base_name, +				    &newhandle) +#endif +#ifdef LTDL_SYSSEARCHPATH +		   && !find_handle (sys_search_path, base_name, &newhandle) +#endif +		   ))) +	{ +          if (tryall_dlopen (&newhandle, filename) != 0) +            { +              newhandle = NULL; +            } +	} + +      if (!newhandle) +	{ +	  LT_DLFREE (*phandle); +	  ++errors; +	  goto cleanup; +	} +    } + + register_handle: +  LT_DLMEM_REASSIGN (*phandle, newhandle); + +  if ((*phandle)->info.ref_count == 0) +    { +      (*phandle)->info.ref_count	= 1; +      LT_DLMEM_REASSIGN ((*phandle)->info.name, name); + +      LT_DLMUTEX_LOCK (); +      (*phandle)->next		= handles; +      handles			= *phandle; +      LT_DLMUTEX_UNLOCK (); +    } + +  LT_DLMUTEX_SETERROR (saved_error); + + cleanup: +  LT_DLFREE (dir); +  LT_DLFREE (name); +  LT_DLFREE (canonical); + +  return errors; +} + +lt_dlhandle +lt_dlopen (filename) +     const char *filename; +{ +  lt_dlhandle handle = 0; + +  /* Just incase we missed a code path in try_dlopen() that reports +     an error, but forgets to reset handle... */ +  if (try_dlopen (&handle, filename) != 0) +    return 0; + +  return handle; +} + +/* If the last error messge store was `FILE_NOT_FOUND', then return +   non-zero.  */ +static int +file_not_found () +{ +  const char *error = 0; + +  LT_DLMUTEX_GETERROR (error); +  if (error == LT_DLSTRERROR (FILE_NOT_FOUND)) +    return 1; + +  return 0; +} + +/* If FILENAME has an ARCHIVE_EXT or SHLIB_EXT extension, try to +   open the FILENAME as passed.  Otherwise try appending ARCHIVE_EXT, +   and if a file is still not found try again with SHLIB_EXT appended +   instead.  */ +lt_dlhandle +lt_dlopenext (filename) +     const char *filename; +{ +  lt_dlhandle	handle		= 0; +  char *	tmp		= 0; +  char *	ext		= 0; +  size_t	len; +  int		errors		= 0; + +  if (!filename) +    { +      return lt_dlopen (filename); +    } + +  assert (filename); + +  len = LT_STRLEN (filename); +  ext = strrchr (filename, '.'); + +  /* If FILENAME already bears a suitable extension, there is no need +     to try appending additional extensions.  */ +  if (ext && ((strcmp (ext, archive_ext) == 0) +#ifdef LTDL_SHLIB_EXT +	      || (strcmp (ext, shlib_ext) == 0) +#endif +      )) +    { +      return lt_dlopen (filename); +    } + +  /* First try appending ARCHIVE_EXT.  */ +  tmp = LT_EMALLOC (char, len + LT_STRLEN (archive_ext) + 1); +  if (!tmp) +    return 0; + +  strcpy (tmp, filename); +  strcat (tmp, archive_ext); +  errors = try_dlopen (&handle, tmp); + +  /* If we found FILENAME, stop searching -- whether we were able to +     load the file as a module or not.  If the file exists but loading +     failed, it is better to return an error message here than to +     report FILE_NOT_FOUND when the alternatives (foo.so etc) are not +     in the module search path.  */ +  if (handle || ((errors > 0) && !file_not_found ())) +    { +      LT_DLFREE (tmp); +      return handle; +    } + +#ifdef LTDL_SHLIB_EXT +  /* Try appending SHLIB_EXT.   */ +  if (LT_STRLEN (shlib_ext) > LT_STRLEN (archive_ext)) +    { +      LT_DLFREE (tmp); +      tmp = LT_EMALLOC (char, len + LT_STRLEN (shlib_ext) + 1); +      if (!tmp) +	return 0; + +      strcpy (tmp, filename); +    } +  else +    { +      tmp[len] = LT_EOS_CHAR; +    } + +  strcat(tmp, shlib_ext); +  errors = try_dlopen (&handle, tmp); + +  /* As before, if the file was found but loading failed, return now +     with the current error message.  */ +  if (handle || ((errors > 0) && !file_not_found ())) +    { +      LT_DLFREE (tmp); +      return handle; +    } +#endif + +  /* Still here?  Then we really did fail to locate any of the file +     names we tried.  */ +  LT_DLMUTEX_SETERROR (LT_DLSTRERROR (FILE_NOT_FOUND)); +  LT_DLFREE (tmp); +  return 0; +} + + +static int +lt_argz_insert (pargz, pargz_len, before, entry) +     char **pargz; +     size_t *pargz_len; +     char *before; +     const char *entry; +{ +  error_t error; + +  /* Prior to Sep 8, 2005, newlib had a bug where argz_insert(pargz, +     pargz_len, NULL, entry) failed with EINVAL.  */ +  if (before) +    error = argz_insert (pargz, pargz_len, before, entry); +  else +    error = argz_append (pargz, pargz_len, entry, 1 + LT_STRLEN (entry)); + +  if (error) +    { +      switch (error) +	{ +	case ENOMEM: +	  LT_DLMUTEX_SETERROR (LT_DLSTRERROR (NO_MEMORY)); +	  break; +	default: +	  LT_DLMUTEX_SETERROR (LT_DLSTRERROR (UNKNOWN)); +	  break; +	} +      return 1; +    } + +  return 0; +} + +static int +lt_argz_insertinorder (pargz, pargz_len, entry) +     char **pargz; +     size_t *pargz_len; +     const char *entry; +{ +  char *before = 0; + +  assert (pargz); +  assert (pargz_len); +  assert (entry && *entry); + +  if (*pargz) +    while ((before = argz_next (*pargz, *pargz_len, before))) +      { +	int cmp = strcmp (entry, before); + +	if (cmp < 0)  break; +	if (cmp == 0) return 0;	/* No duplicates! */ +      } + +  return lt_argz_insert (pargz, pargz_len, before, entry); +} + +static int +lt_argz_insertdir (pargz, pargz_len, dirnam, dp) +     char **pargz; +     size_t *pargz_len; +     const char *dirnam; +     struct dirent *dp; +{ +  char   *buf	    = 0; +  size_t buf_len    = 0; +  char   *end	    = 0; +  size_t end_offset = 0; +  size_t dir_len    = 0; +  int    errors	    = 0; + +  assert (pargz); +  assert (pargz_len); +  assert (dp); + +  dir_len = LT_STRLEN (dirnam); +  end     = dp->d_name + LT_D_NAMLEN(dp); + +  /* Ignore version numbers.  */ +  { +    char *p; +    for (p = end; p -1 > dp->d_name; --p) +      if (strchr (".0123456789", p[-1]) == 0) +	break; + +    if (*p == '.') +      end = p; +  } + +  /* Ignore filename extension.  */ +  { +    char *p; +    for (p = end -1; p > dp->d_name; --p) +      if (*p == '.') +	{ +	  end = p; +	  break; +	} +  } + +  /* Prepend the directory name.  */ +  end_offset	= end - dp->d_name; +  buf_len	= dir_len + 1+ end_offset; +  buf		= LT_EMALLOC (char, 1+ buf_len); +  if (!buf) +    return ++errors; + +  assert (buf); + +  strcpy  (buf, dirnam); +  strcat  (buf, "/"); +  strncat (buf, dp->d_name, end_offset); +  buf[buf_len] = LT_EOS_CHAR; + +  /* Try to insert (in order) into ARGZ/ARGZ_LEN.  */ +  if (lt_argz_insertinorder (pargz, pargz_len, buf) != 0) +    ++errors; + +  LT_DLFREE (buf); + +  return errors; +} + +static int +list_files_by_dir (dirnam, pargz, pargz_len) +     const char *dirnam; +     char **pargz; +     size_t *pargz_len; +{ +  DIR	*dirp	  = 0; +  int    errors	  = 0; + +  assert (dirnam && *dirnam); +  assert (pargz); +  assert (pargz_len); +  assert (dirnam[LT_STRLEN(dirnam) -1] != '/'); + +  dirp = opendir (dirnam); +  if (dirp) +    { +      struct dirent *dp	= 0; + +      while ((dp = readdir (dirp))) +	if (dp->d_name[0] != '.') +	  if (lt_argz_insertdir (pargz, pargz_len, dirnam, dp)) +	    { +	      ++errors; +	      break; +	    } + +      closedir (dirp); +    } +  else +    ++errors; + +  return errors; +} + + +/* If there are any files in DIRNAME, call the function passed in +   DATA1 (with the name of each file and DATA2 as arguments).  */ +static int +foreachfile_callback (dirname, data1, data2) +     char *dirname; +     lt_ptr data1; +     lt_ptr data2; +{ +  int (*func) LT_PARAMS((const char *filename, lt_ptr data)) +	= (int (*) LT_PARAMS((const char *filename, lt_ptr data))) data1; + +  int	  is_done  = 0; +  char   *argz     = 0; +  size_t  argz_len = 0; + +  if (list_files_by_dir (dirname, &argz, &argz_len) != 0) +    goto cleanup; +  if (!argz) +    goto cleanup; + +  { +    char *filename = 0; +    while ((filename = argz_next (argz, argz_len, filename))) +      if ((is_done = (*func) (filename, data2))) +	break; +  } + + cleanup: +  LT_DLFREE (argz); + +  return is_done; +} + + +/* Call FUNC for each unique extensionless file in SEARCH_PATH, along +   with DATA.  The filenames passed to FUNC would be suitable for +   passing to lt_dlopenext.  The extensions are stripped so that +   individual modules do not generate several entries (e.g. libfoo.la, +   libfoo.so, libfoo.so.1, libfoo.so.1.0.0).  If SEARCH_PATH is NULL, +   then the same directories that lt_dlopen would search are examined.  */ +int +lt_dlforeachfile (search_path, func, data) +     const char *search_path; +     int (*func) LT_PARAMS ((const char *filename, lt_ptr data)); +     lt_ptr data; +{ +  int is_done = 0; + +  if (search_path) +    { +      /* If a specific path was passed, search only the directories +	 listed in it.  */ +      is_done = foreach_dirinpath (search_path, 0, +				   foreachfile_callback, func, data); +    } +  else +    { +      /* Otherwise search the default paths.  */ +      is_done = foreach_dirinpath (user_search_path, 0, +				   foreachfile_callback, func, data); +      if (!is_done) +	{ +	  is_done = foreach_dirinpath (getenv("LTDL_LIBRARY_PATH"), 0, +				       foreachfile_callback, func, data); +	} + +#ifdef LTDL_SHLIBPATH_VAR +      if (!is_done) +	{ +	  is_done = foreach_dirinpath (getenv(LTDL_SHLIBPATH_VAR), 0, +				       foreachfile_callback, func, data); +	} +#endif +#ifdef LTDL_SYSSEARCHPATH +      if (!is_done) +	{ +	  is_done = foreach_dirinpath (getenv(LTDL_SYSSEARCHPATH), 0, +				       foreachfile_callback, func, data); +	} +#endif +    } + +  return is_done; +} + +int +lt_dlclose (handle) +     lt_dlhandle handle; +{ +  lt_dlhandle cur, last; +  int errors = 0; + +  LT_DLMUTEX_LOCK (); + +  /* check whether the handle is valid */ +  last = cur = handles; +  while (cur && handle != cur) +    { +      last = cur; +      cur = cur->next; +    } + +  if (!cur) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (INVALID_HANDLE)); +      ++errors; +      goto done; +    } + +  handle->info.ref_count--; + +  /* Note that even with resident modules, we must track the ref_count +     correctly incase the user decides to reset the residency flag +     later (even though the API makes no provision for that at the +     moment).  */ +  if (handle->info.ref_count <= 0 && !LT_DLIS_RESIDENT (handle)) +    { +      lt_user_data data = handle->loader->dlloader_data; + +      if (handle != handles) +	{ +	  last->next = handle->next; +	} +      else +	{ +	  handles = handle->next; +	} + +      errors += handle->loader->module_close (data, handle->module); +      errors += unload_deplibs(handle); + +      /* It is up to the callers to free the data itself.  */ +      LT_DLFREE (handle->caller_data); + +      LT_DLFREE (handle->info.filename); +      LT_DLFREE (handle->info.name); +      LT_DLFREE (handle); + +      goto done; +    } + +  if (LT_DLIS_RESIDENT (handle)) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (CLOSE_RESIDENT_MODULE)); +      ++errors; +    } + + done: +  LT_DLMUTEX_UNLOCK (); + +  return errors; +} + +lt_ptr +lt_dlsym (handle, symbol) +     lt_dlhandle handle; +     const char *symbol; +{ +  size_t lensym; +  char	lsym[LT_SYMBOL_LENGTH]; +  char	*sym; +  lt_ptr address; +  lt_user_data data; + +  if (!handle) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (INVALID_HANDLE)); +      return 0; +    } + +  if (!symbol) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (SYMBOL_NOT_FOUND)); +      return 0; +    } + +  lensym = LT_STRLEN (symbol) + LT_STRLEN (handle->loader->sym_prefix) +					+ LT_STRLEN (handle->info.name); + +  if (lensym + LT_SYMBOL_OVERHEAD < LT_SYMBOL_LENGTH) +    { +      sym = lsym; +    } +  else +    { +      sym = LT_EMALLOC (char, lensym + LT_SYMBOL_OVERHEAD + 1); +      if (!sym) +	{ +	  LT_DLMUTEX_SETERROR (LT_DLSTRERROR (BUFFER_OVERFLOW)); +	  return 0; +	} +    } + +  data = handle->loader->dlloader_data; +  if (handle->info.name) +    { +      const char *saved_error; + +      LT_DLMUTEX_GETERROR (saved_error); + +      /* this is a libtool module */ +      if (handle->loader->sym_prefix) +	{ +	  strcpy(sym, handle->loader->sym_prefix); +	  strcat(sym, handle->info.name); +	} +      else +	{ +	  strcpy(sym, handle->info.name); +	} + +      strcat(sym, "_LTX_"); +      strcat(sym, symbol); + +      /* try "modulename_LTX_symbol" */ +      address = handle->loader->find_sym (data, handle->module, sym); +      if (address) +	{ +	  if (sym != lsym) +	    { +	      LT_DLFREE (sym); +	    } +	  return address; +	} +      LT_DLMUTEX_SETERROR (saved_error); +    } + +  /* otherwise try "symbol" */ +  if (handle->loader->sym_prefix) +    { +      strcpy(sym, handle->loader->sym_prefix); +      strcat(sym, symbol); +    } +  else +    { +      strcpy(sym, symbol); +    } + +  address = handle->loader->find_sym (data, handle->module, sym); +  if (sym != lsym) +    { +      LT_DLFREE (sym); +    } + +  return address; +} + +const char * +lt_dlerror () +{ +  const char *error; + +  LT_DLMUTEX_GETERROR (error); +  LT_DLMUTEX_SETERROR (0); + +  return error ? error : NULL; +} + +static int +lt_dlpath_insertdir (ppath, before, dir) +     char **ppath; +     char *before; +     const char *dir; +{ +  int    errors		= 0; +  char  *canonical	= 0; +  char  *argz		= 0; +  size_t argz_len	= 0; + +  assert (ppath); +  assert (dir && *dir); + +  if (canonicalize_path (dir, &canonical) != 0) +    { +      ++errors; +      goto cleanup; +    } + +  assert (canonical && *canonical); + +  /* If *PPATH is empty, set it to DIR.  */ +  if (*ppath == 0) +    { +      assert (!before);		/* BEFORE cannot be set without PPATH.  */ +      assert (dir);		/* Without DIR, don't call this function!  */ + +      *ppath = lt_estrdup (dir); +      if (*ppath == 0) +	++errors; + +      return errors; +    } + +  assert (ppath && *ppath); + +  if (argzize_path (*ppath, &argz, &argz_len) != 0) +    { +      ++errors; +      goto cleanup; +    } + +  /* Convert BEFORE into an equivalent offset into ARGZ.  This only works +     if *PPATH is already canonicalized, and hence does not change length +     with respect to ARGZ.  We canonicalize each entry as it is added to +     the search path, and don't call this function with (uncanonicalized) +     user paths, so this is a fair assumption.  */ +  if (before) +    { +      assert (*ppath <= before); +      assert ((size_t)(before - *ppath) <= strlen (*ppath)); + +      before = before - *ppath + argz; +    } + +  if (lt_argz_insert (&argz, &argz_len, before, dir) != 0) +    { +      ++errors; +      goto cleanup; +    } + +  argz_stringify (argz, argz_len, LT_PATHSEP_CHAR); +  LT_DLMEM_REASSIGN (*ppath,  argz); + + cleanup: +  LT_DLFREE (canonical); +  LT_DLFREE (argz); + +  return errors; +} + +int +lt_dladdsearchdir (search_dir) +     const char *search_dir; +{ +  int errors = 0; + +  if (search_dir && *search_dir) +    { +      LT_DLMUTEX_LOCK (); +      if (lt_dlpath_insertdir (&user_search_path, 0, search_dir) != 0) +	++errors; +      LT_DLMUTEX_UNLOCK (); +    } + +  return errors; +} + +int +lt_dlinsertsearchdir (before, search_dir) +     const char *before; +     const char *search_dir; +{ +  int errors = 0; + +  if (before) +    { +      LT_DLMUTEX_LOCK (); +      if ((before < user_search_path) +	  || (before >= user_search_path + LT_STRLEN (user_search_path))) +	{ +	  LT_DLMUTEX_UNLOCK (); +	  LT_DLMUTEX_SETERROR (LT_DLSTRERROR (INVALID_POSITION)); +	  return 1; +	} +      LT_DLMUTEX_UNLOCK (); +    } + +  if (search_dir && *search_dir) +    { +      LT_DLMUTEX_LOCK (); +      if (lt_dlpath_insertdir (&user_search_path, +			       (char *) before, search_dir) != 0) +	{ +	  ++errors; +	} +      LT_DLMUTEX_UNLOCK (); +    } + +  return errors; +} + +int +lt_dlsetsearchpath (search_path) +     const char *search_path; +{ +  int   errors	    = 0; + +  LT_DLMUTEX_LOCK (); +  LT_DLFREE (user_search_path); +  LT_DLMUTEX_UNLOCK (); + +  if (!search_path || !LT_STRLEN (search_path)) +    { +      return errors; +    } + +  LT_DLMUTEX_LOCK (); +  if (canonicalize_path (search_path, &user_search_path) != 0) +    ++errors; +  LT_DLMUTEX_UNLOCK (); + +  return errors; +} + +const char * +lt_dlgetsearchpath () +{ +  const char *saved_path; + +  LT_DLMUTEX_LOCK (); +  saved_path = user_search_path; +  LT_DLMUTEX_UNLOCK (); + +  return saved_path; +} + +int +lt_dlmakeresident (handle) +     lt_dlhandle handle; +{ +  int errors = 0; + +  if (!handle) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (INVALID_HANDLE)); +      ++errors; +    } +  else +    { +      LT_DLSET_FLAG (handle, LT_DLRESIDENT_FLAG); +    } + +  return errors; +} + +int +lt_dlisresident	(handle) +     lt_dlhandle handle; +{ +  if (!handle) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (INVALID_HANDLE)); +      return -1; +    } + +  return LT_DLIS_RESIDENT (handle); +} + + + + +/* --- MODULE INFORMATION --- */ + +const lt_dlinfo * +lt_dlgetinfo (handle) +     lt_dlhandle handle; +{ +  if (!handle) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (INVALID_HANDLE)); +      return 0; +    } + +  return &(handle->info); +} + +lt_dlhandle +lt_dlhandle_next (place) +     lt_dlhandle place; +{ +  return place ? place->next : handles; +} + +int +lt_dlforeach (func, data) +     int (*func) LT_PARAMS((lt_dlhandle handle, lt_ptr data)); +     lt_ptr data; +{ +  int errors = 0; +  lt_dlhandle cur; + +  LT_DLMUTEX_LOCK (); + +  cur = handles; +  while (cur) +    { +      lt_dlhandle tmp = cur; + +      cur = cur->next; +      if ((*func) (tmp, data)) +	{ +	  ++errors; +	  break; +	} +    } + +  LT_DLMUTEX_UNLOCK (); + +  return errors; +} + +lt_dlcaller_id +lt_dlcaller_register () +{ +  static lt_dlcaller_id last_caller_id = 0; +  int result; + +  LT_DLMUTEX_LOCK (); +  result = ++last_caller_id; +  LT_DLMUTEX_UNLOCK (); + +  return result; +} + +lt_ptr +lt_dlcaller_set_data (key, handle, data) +     lt_dlcaller_id key; +     lt_dlhandle handle; +     lt_ptr data; +{ +  int n_elements = 0; +  lt_ptr stale = (lt_ptr) 0; +  int i; + +  /* This needs to be locked so that the caller data can be updated +     simultaneously by different threads.  */ +  LT_DLMUTEX_LOCK (); + +  if (handle->caller_data) +    while (handle->caller_data[n_elements].key) +      ++n_elements; + +  for (i = 0; i < n_elements; ++i) +    { +      if (handle->caller_data[i].key == key) +	{ +	  stale = handle->caller_data[i].data; +	  break; +	} +    } + +  /* Ensure that there is enough room in this handle's caller_data +     array to accept a new element (and an empty end marker).  */ +  if (i == n_elements) +    { +      lt_caller_data *temp +	= LT_DLREALLOC (lt_caller_data, handle->caller_data, 2+ n_elements); + +      if (!temp) +	{ +	  stale = 0; +	  goto done; +	} + +      handle->caller_data = temp; + +      /* We only need this if we needed to allocate a new caller_data.  */ +      handle->caller_data[i].key  = key; +      handle->caller_data[1+ i].key = 0; +    } + +  handle->caller_data[i].data = data; + + done: +  LT_DLMUTEX_UNLOCK (); + +  return stale; +} + +lt_ptr +lt_dlcaller_get_data  (key, handle) +     lt_dlcaller_id key; +     lt_dlhandle handle; +{ +  lt_ptr result = (lt_ptr) 0; + +  /* This needs to be locked so that the caller data isn't updated by +     another thread part way through this function.  */ +  LT_DLMUTEX_LOCK (); + +  /* Locate the index of the element with a matching KEY.  */ +  { +    int i; +    for (i = 0; handle->caller_data[i].key; ++i) +      { +	if (handle->caller_data[i].key == key) +	  { +	    result = handle->caller_data[i].data; +	    break; +	  } +      } +  } + +  LT_DLMUTEX_UNLOCK (); + +  return result; +} + + + +/* --- USER MODULE LOADER API --- */ + + +int +lt_dlloader_add (place, dlloader, loader_name) +     lt_dlloader *place; +     const struct lt_user_dlloader *dlloader; +     const char *loader_name; +{ +  int errors = 0; +  lt_dlloader *node = 0, *ptr = 0; + +  if ((dlloader == 0)	/* diagnose null parameters */ +      || (dlloader->module_open == 0) +      || (dlloader->module_close == 0) +      || (dlloader->find_sym == 0)) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (INVALID_LOADER)); +      return 1; +    } + +  /* Create a new dlloader node with copies of the user callbacks.  */ +  node = LT_EMALLOC (lt_dlloader, 1); +  if (!node) +    return 1; + +  node->next		= 0; +  node->loader_name	= loader_name; +  node->sym_prefix	= dlloader->sym_prefix; +  node->dlloader_exit	= dlloader->dlloader_exit; +  node->module_open	= dlloader->module_open; +  node->module_close	= dlloader->module_close; +  node->find_sym	= dlloader->find_sym; +  node->dlloader_data	= dlloader->dlloader_data; + +  LT_DLMUTEX_LOCK (); +  if (!loaders) +    { +      /* If there are no loaders, NODE becomes the list! */ +      loaders = node; +    } +  else if (!place) +    { +      /* If PLACE is not set, add NODE to the end of the +	 LOADERS list. */ +      for (ptr = loaders; ptr->next; ptr = ptr->next) +	{ +	  /*NOWORK*/; +	} + +      ptr->next = node; +    } +  else if (loaders == place) +    { +      /* If PLACE is the first loader, NODE goes first. */ +      node->next = place; +      loaders = node; +    } +  else +    { +      /* Find the node immediately preceding PLACE. */ +      for (ptr = loaders; ptr->next != place; ptr = ptr->next) +	{ +	  /*NOWORK*/; +	} + +      if (ptr->next != place) +	{ +	  LT_DLMUTEX_SETERROR (LT_DLSTRERROR (INVALID_LOADER)); +	  ++errors; +	} +      else +	{ +	  /* Insert NODE between PTR and PLACE. */ +	  node->next = place; +	  ptr->next  = node; +	} +    } + +  LT_DLMUTEX_UNLOCK (); + +  return errors; +} + +int +lt_dlloader_remove (loader_name) +     const char *loader_name; +{ +  lt_dlloader *place = lt_dlloader_find (loader_name); +  lt_dlhandle handle; +  int errors = 0; + +  if (!place) +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (INVALID_LOADER)); +      return 1; +    } + +  LT_DLMUTEX_LOCK (); + +  /* Fail if there are any open modules which use this loader. */ +  for  (handle = handles; handle; handle = handle->next) +    { +      if (handle->loader == place) +	{ +	  LT_DLMUTEX_SETERROR (LT_DLSTRERROR (REMOVE_LOADER)); +	  ++errors; +	  goto done; +	} +    } + +  if (place == loaders) +    { +      /* PLACE is the first loader in the list. */ +      loaders = loaders->next; +    } +  else +    { +      /* Find the loader before the one being removed. */ +      lt_dlloader *prev; +      for (prev = loaders; prev->next; prev = prev->next) +	{ +	  if (!strcmp (prev->next->loader_name, loader_name)) +	    { +	      break; +	    } +	} + +      place = prev->next; +      prev->next = prev->next->next; +    } + +  if (place->dlloader_exit) +    { +      errors = place->dlloader_exit (place->dlloader_data); +    } + +  LT_DLFREE (place); + + done: +  LT_DLMUTEX_UNLOCK (); + +  return errors; +} + +lt_dlloader * +lt_dlloader_next (place) +     lt_dlloader *place; +{ +  lt_dlloader *next; + +  LT_DLMUTEX_LOCK (); +  next = place ? place->next : loaders; +  LT_DLMUTEX_UNLOCK (); + +  return next; +} + +const char * +lt_dlloader_name (place) +     lt_dlloader *place; +{ +  const char *name = 0; + +  if (place) +    { +      LT_DLMUTEX_LOCK (); +      name = place ? place->loader_name : 0; +      LT_DLMUTEX_UNLOCK (); +    } +  else +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (INVALID_LOADER)); +    } + +  return name; +} + +lt_user_data * +lt_dlloader_data (place) +     lt_dlloader *place; +{ +  lt_user_data *data = 0; + +  if (place) +    { +      LT_DLMUTEX_LOCK (); +      data = place ? &(place->dlloader_data) : 0; +      LT_DLMUTEX_UNLOCK (); +    } +  else +    { +      LT_DLMUTEX_SETERROR (LT_DLSTRERROR (INVALID_LOADER)); +    } + +  return data; +} + +lt_dlloader * +lt_dlloader_find (loader_name) +     const char *loader_name; +{ +  lt_dlloader *place = 0; + +  LT_DLMUTEX_LOCK (); +  for (place = loaders; place; place = place->next) +    { +      if (strcmp (place->loader_name, loader_name) == 0) +	{ +	  break; +	} +    } +  LT_DLMUTEX_UNLOCK (); + +  return place; +} diff --git a/lib/System/ltdl.h b/lib/System/ltdl.h new file mode 100644 index 0000000..8aaf342 --- /dev/null +++ b/lib/System/ltdl.h @@ -0,0 +1,366 @@ +/* ltdl.h -- generic dlopen functions +   Copyright (C) 1998-2000 Free Software Foundation, Inc. +   Originally by Thomas Tanner <tanner@ffii.org> +   This file is part of GNU Libtool. + +This library is free software; you can redistribute it and/or +modify it under the terms of the GNU Lesser General Public +License as published by the Free Software Foundation; either +version 2 of the License, or (at your option) any later version. + +As a special exception to the GNU Lesser General Public License, +if you distribute this file as part of a program or library that +is built using GNU libtool, you may include it under the same +distribution terms that you use for the rest of that program. + +This library is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU +Lesser General Public License for more details. + +You should have received a copy of the GNU Lesser General Public +License along with this library; if not, write to the Free +Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA +02110-1301  USA +*/ + +/* Only include this header file once. */ +#ifndef LTDL_H +#define LTDL_H 1 + +#include <sys/types.h>		/* for size_t declaration */ + + +/* --- MACROS FOR PORTABILITY --- */ + + +/* Saves on those hard to debug '\0' typos....  */ +#define LT_EOS_CHAR	'\0' + +/* LTDL_BEGIN_C_DECLS should be used at the beginning of your declarations, +   so that C++ compilers don't mangle their names.  Use LTDL_END_C_DECLS at +   the end of C declarations. */ +#ifdef __cplusplus +# define LT_BEGIN_C_DECLS	extern "C" { +# define LT_END_C_DECLS		} +#else +# define LT_BEGIN_C_DECLS	/* empty */ +# define LT_END_C_DECLS		/* empty */ +#endif + +LT_BEGIN_C_DECLS + + +/* LT_PARAMS is a macro used to wrap function prototypes, so that compilers +   that don't understand ANSI C prototypes still work, and ANSI C +   compilers can issue warnings about type mismatches.  */ +#if defined (__STDC__) || defined (_AIX) || (defined (__mips) && defined (_SYSTYPE_SVR4)) || defined(WIN32) || defined(__cplusplus) +# define LT_PARAMS(protos)	protos +# define lt_ptr		void* +#else +# define LT_PARAMS(protos)	() +# define lt_ptr		char* +#endif + +/* LT_STMT_START/END are used to create macros which expand to a +   a single compound statement in a portable way.  */ +#if defined (__GNUC__) && !defined (__STRICT_ANSI__) && !defined (__cplusplus) +#  define LT_STMT_START        (void)( +#  define LT_STMT_END          ) +#else +#  if (defined (sun) || defined (__sun__)) +#    define LT_STMT_START      if (1) +#    define LT_STMT_END        else (void)0 +#  else +#    define LT_STMT_START      do +#    define LT_STMT_END        while (0) +#  endif +#endif + +/* LT_CONC creates a new concatenated symbol for the compiler +   in a portable way.  */ +#if defined(__STDC__) || defined(__cplusplus) || defined(_MSC_VER) +#  define LT_CONC(s,t)	s##t +#else +#  define LT_CONC(s,t)	s/**/t +#endif + +/* LT_STRLEN can be used safely on NULL pointers.  */ +#define LT_STRLEN(s)	(((s) && (s)[0]) ? strlen (s) : 0) + + + +/* --- WINDOWS SUPPORT --- */ + + +/* Canonicalise Windows and Cygwin recognition macros.  */ +#ifdef __CYGWIN32__ +#  ifndef __CYGWIN__ +#    define __CYGWIN__ __CYGWIN32__ +#  endif +#endif +#if defined(_WIN32) || defined(WIN32) +#  ifndef __WINDOWS__ +#    ifdef _WIN32 +#      define __WINDOWS__ _WIN32 +#    else +#      ifdef WIN32 +#        define __WINDOWS__ WIN32 +#      endif +#    endif +#  endif +#endif + + +#ifdef __WINDOWS__ +#  ifndef __CYGWIN__ +/* LT_DIRSEP_CHAR is accepted *in addition* to '/' as a directory +   separator when it is set. */ +#    define LT_DIRSEP_CHAR	'\\' +#    define LT_PATHSEP_CHAR	';' +#  endif +#endif +#ifndef LT_PATHSEP_CHAR +#  define LT_PATHSEP_CHAR	':' +#endif + +/* DLL building support on win32 hosts;  mostly to workaround their +   ridiculous implementation of data symbol exporting. */ +#ifndef LT_SCOPE +#  ifdef __WINDOWS__ +#    ifdef DLL_EXPORT		/* defined by libtool (if required) */ +#      define LT_SCOPE	__declspec(dllexport) +#    endif +#    ifdef LIBLTDL_DLL_IMPORT	/* define if linking with this dll */ +#      define LT_SCOPE	extern __declspec(dllimport) +#    endif +#  endif +#  ifndef LT_SCOPE		/* static linking or !__WINDOWS__ */ +#    define LT_SCOPE	extern +#  endif +#endif + + +#if defined(_MSC_VER) /* Visual Studio */ +#  define R_OK 4 +#endif + + + +/* --- DYNAMIC MODULE LOADING API --- */ + + +typedef	struct lt_dlhandle_struct *lt_dlhandle;	/* A loaded module.  */ + +/* Initialisation and finalisation functions for libltdl. */ +LT_SCOPE	int	    lt_dlinit		LT_PARAMS((void)); +LT_SCOPE	int	    lt_dlexit		LT_PARAMS((void)); + +/* Module search path manipulation.  */ +LT_SCOPE	int	    lt_dladdsearchdir	 LT_PARAMS((const char *search_dir)); +LT_SCOPE	int	    lt_dlinsertsearchdir LT_PARAMS((const char *before, +						    const char *search_dir)); +LT_SCOPE	int 	    lt_dlsetsearchpath	 LT_PARAMS((const char *search_path)); +LT_SCOPE	const char *lt_dlgetsearchpath	 LT_PARAMS((void)); +LT_SCOPE	int	    lt_dlforeachfile	 LT_PARAMS(( +			const char *search_path, +			int (*func) (const char *filename, lt_ptr data), +			lt_ptr data)); + +/* Portable libltdl versions of the system dlopen() API. */ +LT_SCOPE	lt_dlhandle lt_dlopen		LT_PARAMS((const char *filename)); +LT_SCOPE	lt_dlhandle lt_dlopenext	LT_PARAMS((const char *filename)); +LT_SCOPE	lt_ptr	    lt_dlsym		LT_PARAMS((lt_dlhandle handle, +						     const char *name)); +LT_SCOPE	const char *lt_dlerror		LT_PARAMS((void)); +LT_SCOPE	int	    lt_dlclose		LT_PARAMS((lt_dlhandle handle)); + +/* Module residency management. */ +LT_SCOPE	int	    lt_dlmakeresident	LT_PARAMS((lt_dlhandle handle)); +LT_SCOPE	int	    lt_dlisresident	LT_PARAMS((lt_dlhandle handle)); + + + + +/* --- MUTEX LOCKING --- */ + + +typedef void	lt_dlmutex_lock		LT_PARAMS((void)); +typedef void	lt_dlmutex_unlock	LT_PARAMS((void)); +typedef void	lt_dlmutex_seterror	LT_PARAMS((const char *errmsg)); +typedef const char *lt_dlmutex_geterror	LT_PARAMS((void)); + +LT_SCOPE	int	lt_dlmutex_register	LT_PARAMS((lt_dlmutex_lock *lock, +					    lt_dlmutex_unlock *unlock, +					    lt_dlmutex_seterror *seterror, +					    lt_dlmutex_geterror *geterror)); + + + + +/* --- MEMORY HANDLING --- */ + + +/* By default, the realloc function pointer is set to our internal +   realloc implementation which iself uses lt_dlmalloc and lt_dlfree. +   libltdl relies on a featureful realloc, but if you are sure yours +   has the right semantics then you can assign it directly.  Generally, +   it is safe to assign just a malloc() and a free() function.  */ +LT_SCOPE  lt_ptr   (*lt_dlmalloc)	LT_PARAMS((size_t size)); +LT_SCOPE  lt_ptr   (*lt_dlrealloc)	LT_PARAMS((lt_ptr ptr, size_t size)); +LT_SCOPE  void	   (*lt_dlfree)		LT_PARAMS((lt_ptr ptr)); + + + + +/* --- PRELOADED MODULE SUPPORT --- */ + + +/* A preopened symbol. Arrays of this type comprise the exported +   symbols for a dlpreopened module. */ +typedef struct { +  const char *name; +  lt_ptr      address; +} lt_dlsymlist; + +LT_SCOPE	int	lt_dlpreload	LT_PARAMS((const lt_dlsymlist *preloaded)); +LT_SCOPE	int	lt_dlpreload_default +				LT_PARAMS((const lt_dlsymlist *preloaded)); + +#define LTDL_SET_PRELOADED_SYMBOLS() 		LT_STMT_START{	\ +	extern const lt_dlsymlist lt_preloaded_symbols[];		\ +	lt_dlpreload_default(lt_preloaded_symbols);			\ +						}LT_STMT_END + + + + +/* --- MODULE INFORMATION --- */ + + +/* Read only information pertaining to a loaded module. */ +typedef	struct { +  char	*filename;		/* file name */ +  char	*name;			/* module name */ +  int	ref_count;		/* number of times lt_dlopened minus +				   number of times lt_dlclosed. */ +} lt_dlinfo; + +LT_SCOPE	const lt_dlinfo	*lt_dlgetinfo	    LT_PARAMS((lt_dlhandle handle)); +LT_SCOPE	lt_dlhandle	lt_dlhandle_next    LT_PARAMS((lt_dlhandle place)); +LT_SCOPE	int		lt_dlforeach	    LT_PARAMS(( +				int (*func) (lt_dlhandle handle, lt_ptr data), +				lt_ptr data)); + +/* Associating user data with loaded modules. */ +typedef unsigned lt_dlcaller_id; + +LT_SCOPE	lt_dlcaller_id	lt_dlcaller_register  LT_PARAMS((void)); +LT_SCOPE	lt_ptr		lt_dlcaller_set_data  LT_PARAMS((lt_dlcaller_id key, +						lt_dlhandle handle, +						lt_ptr data)); +LT_SCOPE	lt_ptr		lt_dlcaller_get_data  LT_PARAMS((lt_dlcaller_id key, +						lt_dlhandle handle)); + + + +/* --- USER MODULE LOADER API --- */ + + +typedef	struct lt_dlloader	lt_dlloader; +typedef lt_ptr			lt_user_data; +typedef lt_ptr			lt_module; + +/* Function pointer types for creating user defined module loaders. */ +typedef lt_module   lt_module_open	LT_PARAMS((lt_user_data loader_data, +					    const char *filename)); +typedef int	    lt_module_close	LT_PARAMS((lt_user_data loader_data, +					    lt_module handle)); +typedef lt_ptr	    lt_find_sym		LT_PARAMS((lt_user_data loader_data, +					    lt_module handle, +					    const char *symbol)); +typedef int	    lt_dlloader_exit	LT_PARAMS((lt_user_data loader_data)); + +struct lt_user_dlloader { +  const char	       *sym_prefix; +  lt_module_open       *module_open; +  lt_module_close      *module_close; +  lt_find_sym	       *find_sym; +  lt_dlloader_exit     *dlloader_exit; +  lt_user_data		dlloader_data; +}; + +LT_SCOPE	lt_dlloader    *lt_dlloader_next    LT_PARAMS((lt_dlloader *place)); +LT_SCOPE	lt_dlloader    *lt_dlloader_find    LT_PARAMS(( +						const char *loader_name)); +LT_SCOPE	const char     *lt_dlloader_name    LT_PARAMS((lt_dlloader *place)); +LT_SCOPE	lt_user_data   *lt_dlloader_data    LT_PARAMS((lt_dlloader *place)); +LT_SCOPE	int		lt_dlloader_add     LT_PARAMS((lt_dlloader *place, +				const struct lt_user_dlloader *dlloader, +				const char *loader_name)); +LT_SCOPE	int		lt_dlloader_remove  LT_PARAMS(( +						const char *loader_name)); + + + +/* --- ERROR MESSAGE HANDLING --- */ + + +/* Defining error strings alongside their symbolic names in a macro in +   this way allows us to expand the macro in different contexts with +   confidence that the enumeration of symbolic names will map correctly +   onto the table of error strings.  */ +#define lt_dlerror_table						\ +    LT_ERROR(UNKNOWN,		    "unknown error")			\ +    LT_ERROR(DLOPEN_NOT_SUPPORTED,  "dlopen support not available")	\ +    LT_ERROR(INVALID_LOADER,	    "invalid loader")			\ +    LT_ERROR(INIT_LOADER,	    "loader initialization failed")	\ +    LT_ERROR(REMOVE_LOADER,	    "loader removal failed")		\ +    LT_ERROR(FILE_NOT_FOUND,	    "file not found")			\ +    LT_ERROR(DEPLIB_NOT_FOUND,      "dependency library not found")	\ +    LT_ERROR(NO_SYMBOLS,	    "no symbols defined")		\ +    LT_ERROR(CANNOT_OPEN,	    "can't open the module")		\ +    LT_ERROR(CANNOT_CLOSE,	    "can't close the module")		\ +    LT_ERROR(SYMBOL_NOT_FOUND,      "symbol not found")			\ +    LT_ERROR(NO_MEMORY,		    "not enough memory")		\ +    LT_ERROR(INVALID_HANDLE,	    "invalid module handle")		\ +    LT_ERROR(BUFFER_OVERFLOW,	    "internal buffer overflow")		\ +    LT_ERROR(INVALID_ERRORCODE,     "invalid errorcode")		\ +    LT_ERROR(SHUTDOWN,		    "library already shutdown")		\ +    LT_ERROR(CLOSE_RESIDENT_MODULE, "can't close resident module")	\ +    LT_ERROR(INVALID_MUTEX_ARGS,    "invalid mutex handler registration") \ +    LT_ERROR(INVALID_POSITION,	    "invalid search path insert position") + +/* Enumerate the symbolic error names. */ +enum { +#define LT_ERROR(name, diagnostic)	LT_CONC(LT_ERROR_, name), +	lt_dlerror_table +#undef LT_ERROR + +	LT_ERROR_MAX +}; + +/* These functions are only useful from inside custom module loaders. */ +LT_SCOPE	int	lt_dladderror	LT_PARAMS((const char *diagnostic)); +LT_SCOPE	int	lt_dlseterror	LT_PARAMS((int errorcode)); + + + + +/* --- SOURCE COMPATIBILITY WITH OLD LIBLTDL --- */ + + +#ifdef LT_NON_POSIX_NAMESPACE +#  define lt_ptr_t		lt_ptr +#  define lt_module_t		lt_module +#  define lt_module_open_t	lt_module_open +#  define lt_module_close_t	lt_module_close +#  define lt_find_sym_t		lt_find_sym +#  define lt_dlloader_exit_t	lt_dlloader_exit +#  define lt_dlloader_t		lt_dlloader +#  define lt_dlloader_data_t	lt_user_data +#endif + +LT_END_C_DECLS + +#endif /* !LTDL_H */ | 
