384 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			384 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
| /* pseudo-reloc.cc
 | |
| 
 | |
|    Contributed by Egor Duda  <deo@logos-m.ru>
 | |
|    Modified by addition of runtime_pseudo_reloc version 2
 | |
|    by Kai Tietz  <kai.tietz@onevision.com>
 | |
| 
 | |
|    THIS SOFTWARE IS NOT COPYRIGHTED
 | |
| 
 | |
|    This source code is offered for use in the public domain. You may
 | |
|    use, modify or distribute it freely.
 | |
| 
 | |
|    This code is distributed in the hope that it will be useful but
 | |
|    WITHOUT ANY WARRANTY. ALL WARRENTIES, EXPRESS OR IMPLIED ARE HEREBY
 | |
|    DISCLAMED. This includes but is not limited to warrenties of
 | |
|    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
 | |
| */
 | |
| 
 | |
| #ifndef __CYGWIN__
 | |
| # include "windows.h"
 | |
| # define NO_COPY
 | |
| #else
 | |
| # include "winsup.h"
 | |
| # include <sys/cygwin.h>
 | |
| /* custom status code: */
 | |
| # define STATUS_ILLEGAL_DLL_PSEUDO_RELOCATION ((NTSTATUS) 0xe0000269)
 | |
| #endif
 | |
| 
 | |
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <stdarg.h>
 | |
| #include <memory.h>
 | |
| 
 | |
| #ifdef __GNUC__
 | |
| #define ATTRIBUTE_NORETURN __attribute__ ((noreturn))
 | |
| #else
 | |
| #define ATTRIBUTE_NORETURN
 | |
| #endif
 | |
| 
 | |
| #ifndef __MINGW_LSYMBOL
 | |
| #define __MINGW_LSYMBOL(sym) sym
 | |
| #endif
 | |
| 
 | |
| extern char __RUNTIME_PSEUDO_RELOC_LIST__;
 | |
| extern char __RUNTIME_PSEUDO_RELOC_LIST_END__;
 | |
| extern char __MINGW_LSYMBOL(_image_base__);
 | |
| 
 | |
| /* v1 relocation is basically:
 | |
|  *   *(base + .target) += .addend
 | |
|  * where (base + .target) is always assumed to point
 | |
|  * to a DWORD (4 bytes).
 | |
|  */
 | |
| typedef struct {
 | |
|   DWORD addend;
 | |
|   DWORD target;
 | |
| } runtime_pseudo_reloc_item_v1;
 | |
| 
 | |
| /* v2 relocation is more complex. In effect, it is
 | |
|  *    *(base + .target) += *(base + .sym) - (base + .sym)
 | |
|  * with care taken in both reading, sign extension, and writing
 | |
|  * because .flags may indicate that (base + .target) may point
 | |
|  * to a BYTE, WORD, DWORD, or QWORD (w64).
 | |
|  */
 | |
| typedef struct {
 | |
|   DWORD sym;
 | |
|   DWORD target;
 | |
|   DWORD flags;
 | |
| } runtime_pseudo_reloc_item_v2;
 | |
| 
 | |
| typedef struct {
 | |
|   DWORD magic1;
 | |
|   DWORD magic2;
 | |
|   DWORD version;
 | |
| } runtime_pseudo_reloc_v2;
 | |
| 
 | |
| static void ATTRIBUTE_NORETURN
 | |
| __report_error (const char *msg, ...)
 | |
| {
 | |
| #ifdef __CYGWIN__
 | |
|   /* This function is used to print short error messages
 | |
|    * to stderr, which may occur during DLL initialization
 | |
|    * while fixing up 'pseudo' relocations. This early, we
 | |
|    * may not be able to use cygwin stdio functions, so we
 | |
|    * use the win32 WriteFile api. This should work with both
 | |
|    * normal win32 console IO handles, redirected ones, and
 | |
|    * cygwin ptys.
 | |
|    */
 | |
|   char buf[128];
 | |
|   char *posix_module = NULL;
 | |
|   static const char UNKNOWN_MODULE[] = "<unknown module>: ";
 | |
|   static const char CYGWIN_FAILURE_MSG[] = "Cygwin runtime failure: ";
 | |
|   HANDLE errh = GetStdHandle (STD_ERROR_HANDLE);
 | |
|   va_list args;
 | |
| 
 | |
|   /* FIXME: cleanup further to avoid old use of cygwin_internal */
 | |
|   if (errh == INVALID_HANDLE_VALUE)
 | |
|     cygwin_internal (CW_EXIT_PROCESS, STATUS_ILLEGAL_DLL_PSEUDO_RELOCATION, 1);
 | |
| 
 | |
|   posix_module = (char *) cygwin_create_path (CCP_WIN_W_TO_POSIX,
 | |
| 					      global_progname);
 | |
| 
 | |
|   va_start (args, msg);
 | |
|   vsnprintf (buf, sizeof (buf), msg, args);
 | |
|   va_end (args);
 | |
|   buf[sizeof (buf) - 1] = '\0'; /* paranoia */
 | |
| 
 | |
|   small_printf ("%s%s: %s\n", CYGWIN_FAILURE_MSG, posix_module ?: UNKNOWN_MODULE, buf);
 | |
|   if (posix_module)
 | |
|     free (posix_module);
 | |
| 
 | |
|   cygwin_internal (CW_EXIT_PROCESS, STATUS_ILLEGAL_DLL_PSEUDO_RELOCATION, 1);
 | |
|   /* not reached, but silences noreturn warning */
 | |
|   abort ();
 | |
| #else
 | |
|   va_list argp;
 | |
|   va_start (argp, msg);
 | |
| # ifdef __MINGW64_VERSION_MAJOR
 | |
|   fprintf (stderr, "Mingw-w64 runtime failure:\n");
 | |
| # else
 | |
|   fprintf (stderr, "Mingw runtime failure:\n");
 | |
| # endif
 | |
|   vfprintf (stderr, msg, argp);
 | |
|   va_end (argp);
 | |
|   abort ();
 | |
| #endif
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * This function automatically sets addr as PAGE_EXECUTE_READWRITE
 | |
|  * by deciding whether VirtualQuery for the addr is actually needed.
 | |
|  * And it assumes that it is called in LdrpCallInitRoutine.
 | |
|  * Hence not thread safe.
 | |
|  */
 | |
| static void
 | |
| auto_protect_for (void* addr)
 | |
| {
 | |
|   static MEMORY_BASIC_INFORMATION mbi;
 | |
|   static bool state = false;
 | |
|   static DWORD oldprot;
 | |
| 
 | |
|   if (!addr)
 | |
|     {
 | |
|       /* Restore original protection. */
 | |
|       if (!(mbi.Protect & (PAGE_EXECUTE_READWRITE | PAGE_READWRITE)))
 | |
|         VirtualProtect (mbi.BaseAddress, mbi.RegionSize, oldprot, &oldprot);
 | |
|       state = false;
 | |
|       return;
 | |
|     }
 | |
|   if (state)
 | |
|     {
 | |
|       /* We have valid region information.  Are we still within this region?
 | |
|          If so, just leave. */
 | |
|       void *ptr = ((void*) ((ptrdiff_t) mbi.BaseAddress + mbi.RegionSize));
 | |
|       if (addr >= mbi.BaseAddress && addr < ptr)
 | |
| 	return;
 | |
|       /* Otherwise, restore original protection and fall through to querying
 | |
|          and potentially changing next region. */
 | |
|       if (!(mbi.Protect & (PAGE_EXECUTE_READWRITE | PAGE_READWRITE)))
 | |
| 	VirtualProtect (mbi.BaseAddress, mbi.RegionSize, oldprot, &oldprot);
 | |
|     }
 | |
|   else
 | |
|     state = true;
 | |
|   /* Query region and temporarily allow write access to read-only protected
 | |
|      memory.  */
 | |
|   VirtualQuery (addr, &mbi, sizeof mbi);
 | |
|   if (!(mbi.Protect & (PAGE_EXECUTE_READWRITE | PAGE_READWRITE)))
 | |
|     VirtualProtect (mbi.BaseAddress, mbi.RegionSize,
 | |
| 	PAGE_EXECUTE_READWRITE, &oldprot);
 | |
| }
 | |
| 
 | |
| /* This function temporarily marks the page containing addr
 | |
|  * writable, before copying len bytes from *src to *addr, and
 | |
|  * then restores the original protection settings to the page.
 | |
|  *
 | |
|  * Using this function eliminates the requirement with older
 | |
|  * pseudo-reloc implementations, that sections containing
 | |
|  * pseudo-relocs (such as .text and .rdata) be permanently
 | |
|  * marked writable. This older behavior sabotaged any memory
 | |
|  * savings achieved by shared libraries on win32 -- and was
 | |
|  * slower, too.  However, on cygwin as of binutils 2.20 the
 | |
|  * .text section is still marked writable, and the .rdata section
 | |
|  * is folded into the (writable) .data when --enable-auto-import.
 | |
|  */
 | |
| static void
 | |
| __write_memory (void *addr, const void *src, size_t len)
 | |
| {
 | |
|   if (!len)
 | |
|     return;
 | |
|   /* Fix page protection for writing. */
 | |
|   auto_protect_for (addr);
 | |
|   /* write the data. */
 | |
|   memcpy (addr, src, len);
 | |
| }
 | |
| 
 | |
| #define RP_VERSION_V1 0
 | |
| #define RP_VERSION_V2 1
 | |
| 
 | |
| static void
 | |
| do_pseudo_reloc (void * start, void * end, void * base)
 | |
| {
 | |
|   ptrdiff_t addr_imp, reldata;
 | |
|   ptrdiff_t reloc_target = (ptrdiff_t) ((char *)end - (char*)start);
 | |
|   runtime_pseudo_reloc_v2 *v2_hdr = (runtime_pseudo_reloc_v2 *) start;
 | |
|   runtime_pseudo_reloc_item_v2 *r;
 | |
| 
 | |
|   /* A valid relocation list will contain at least one entry, and
 | |
|    * one v1 data structure (the smallest one) requires two DWORDs.
 | |
|    * So, if the relocation list is smaller than 8 bytes, bail.
 | |
|    */
 | |
|   if (reloc_target < 8)
 | |
|     return;
 | |
| 
 | |
|   /* Check if this is the old pseudo relocation version.  */
 | |
|   /* There are two kinds of v1 relocation lists:
 | |
|    *   1) With a (v2-style) version header. In this case, the
 | |
|    *      first entry in the list is a 3-DWORD structure, with
 | |
|    *      value:
 | |
|    *	  { 0, 0, RP_VERSION_V1 }
 | |
|    *      In this case, we skip to the next entry in the list,
 | |
|    *      knowing that all elements after the head item can
 | |
|    *      be cast to runtime_pseudo_reloc_item_v1.
 | |
|    *   2) Without a (v2-style) version header. In this case, the
 | |
|    *      first element in the list IS an actual v1 relocation
 | |
|    *      record, which is two DWORDs.  Because there will never
 | |
|    *      be a case where a v1 relocation record has both
 | |
|    *      addend == 0 and target == 0, this case will not be
 | |
|    *      confused with the prior one.
 | |
|    * All current binutils, when generating a v1 relocation list,
 | |
|    * use the second (e.g. original) form -- that is, without the
 | |
|    * v2-style version header.
 | |
|    */
 | |
|   if (reloc_target >= 12
 | |
|       && v2_hdr->magic1 == 0 && v2_hdr->magic2 == 0
 | |
|       && v2_hdr->version == RP_VERSION_V1)
 | |
|     {
 | |
|       /* We have a list header item indicating that the rest
 | |
|        * of the list contains v1 entries.  Move the pointer to
 | |
|        * the first true v1 relocation record.  By definition,
 | |
|        * that v1 element will not have both addend == 0 and
 | |
|        * target == 0 (and thus, when interpreted as a
 | |
|        * runtime_pseudo_reloc_v2, it will not have both
 | |
|        * magic1 == 0 and magic2 == 0).
 | |
|        */
 | |
|       v2_hdr++;
 | |
|     }
 | |
| 
 | |
|   if (v2_hdr->magic1 != 0 || v2_hdr->magic2 != 0)
 | |
|     {
 | |
|       /*************************
 | |
|        * Handle v1 relocations *
 | |
|        *************************/
 | |
|       runtime_pseudo_reloc_item_v1 * o;
 | |
|       for (o = (runtime_pseudo_reloc_item_v1 *) v2_hdr;
 | |
| 	   o < (runtime_pseudo_reloc_item_v1 *)end;
 | |
| 	   o++)
 | |
| 	{
 | |
| 	  DWORD newval;
 | |
| 	  reloc_target = (ptrdiff_t) base + o->target;
 | |
| 	  newval = (*((DWORD*) reloc_target)) + o->addend;
 | |
| 	  __write_memory ((void *) reloc_target, &newval, sizeof (DWORD));
 | |
| 	}
 | |
|       /* Restore original protection. */
 | |
|       auto_protect_for (NULL);
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|   /* If we got this far, then we have relocations of version 2 or newer */
 | |
| 
 | |
|   /* Check if this is a known version.  */
 | |
|   if (v2_hdr->version != RP_VERSION_V2)
 | |
|     {
 | |
|       __report_error ("  Unknown pseudo relocation protocol version %d.\n",
 | |
| 		      (int) v2_hdr->version);
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|   /*************************
 | |
|    * Handle v2 relocations *
 | |
|    *************************/
 | |
| 
 | |
|   /* Walk over header. */
 | |
|   r = (runtime_pseudo_reloc_item_v2 *) &v2_hdr[1];
 | |
| 
 | |
|   for (; r < (runtime_pseudo_reloc_item_v2 *) end; r++)
 | |
|     {
 | |
|       /* location where new address will be written */
 | |
|       reloc_target = (ptrdiff_t) base + r->target;
 | |
| 
 | |
|       /* get sym pointer. It points either to the iat entry
 | |
|        * of the referenced element, or to the stub function.
 | |
|        */
 | |
|       addr_imp = (ptrdiff_t) base + r->sym;
 | |
|       addr_imp = *((ptrdiff_t *) addr_imp);
 | |
| 
 | |
|       /* read existing relocation value from image, casting to the
 | |
|        * bitsize indicated by the 8 LSBs of flags. If the value is
 | |
|        * negative, manually sign-extend to ptrdiff_t width. Raise an
 | |
|        * error if the bitsize indicated by the 8 LSBs of flags is not
 | |
|        * supported.
 | |
|        */
 | |
|       switch ((r->flags & 0xff))
 | |
| 	{
 | |
| 	case 8:
 | |
| 	  reldata = (ptrdiff_t) (*((unsigned char *)reloc_target));
 | |
| 	  if ((reldata & 0x80) != 0)
 | |
| 	    reldata |= ~((ptrdiff_t) 0xff);
 | |
| 	  break;
 | |
| 	case 16:
 | |
| 	  reldata = (ptrdiff_t) (*((unsigned short *)reloc_target));
 | |
| 	  if ((reldata & 0x8000) != 0)
 | |
| 	    reldata |= ~((ptrdiff_t) 0xffff);
 | |
| 	  break;
 | |
| 	case 32:
 | |
| 	  reldata = (ptrdiff_t) (*((unsigned int *)reloc_target));
 | |
| #if defined (__x86_64__) || defined (_WIN64)
 | |
| 	  if ((reldata & 0x80000000) != 0)
 | |
| 	    reldata |= ~((ptrdiff_t) 0xffffffff);
 | |
| #endif
 | |
| 	  break;
 | |
| #if defined (__x86_64__) || defined (_WIN64)
 | |
| 	case 64:
 | |
| 	  reldata = (ptrdiff_t) (*((unsigned long long *)reloc_target));
 | |
| 	  break;
 | |
| #endif
 | |
| 	default:
 | |
| 	  reldata=0;
 | |
| 	  __report_error ("  Unknown pseudo relocation bit size %d.\n",
 | |
| 		  (int) (r->flags & 0xff));
 | |
| 	  break;
 | |
| 	}
 | |
| 
 | |
|       /* Adjust the relocation value */
 | |
|       reldata -= ((ptrdiff_t) base + r->sym);
 | |
|       reldata += addr_imp;
 | |
| 
 | |
|       /* Write the new relocation value back to *reloc_target */
 | |
|       switch ((r->flags & 0xff))
 | |
| 	{
 | |
| 	case 8:
 | |
| 	  __write_memory ((void *) reloc_target, &reldata, 1);
 | |
| 	  break;
 | |
| 	case 16:
 | |
| 	  __write_memory ((void *) reloc_target, &reldata, 2);
 | |
| 	  break;
 | |
| 	case 32:
 | |
| #if defined (__CYGWIN__) && defined (__x86_64__)
 | |
| 	  if (reldata > (ptrdiff_t) __INT32_MAX__
 | |
| 	      || reldata < -((ptrdiff_t) __INT32_MAX__) - 1)
 | |
| 	    __report_error ("Invalid relocation.  Offset %p at address %p "
 | |
| 			    "doesn't fit into 32 bits", reldata, reloc_target);
 | |
| #endif
 | |
| 	  __write_memory ((void *) reloc_target, &reldata, 4);
 | |
| 	  break;
 | |
| #if defined (__x86_64__) || defined (_WIN64)
 | |
| 	case 64:
 | |
| 	  __write_memory ((void *) reloc_target, &reldata, 8);
 | |
| 	  break;
 | |
| #endif
 | |
| 	}
 | |
|     }
 | |
|   /* Restore original protection. */
 | |
|   auto_protect_for (NULL);
 | |
| }
 | |
| 
 | |
| #ifdef __CYGWIN__
 | |
| extern "C" void
 | |
| _pei386_runtime_relocator (per_process *u)
 | |
| {
 | |
|   if (u && CYGWIN_VERSION_USE_PSEUDO_RELOC_IN_DLL (u))
 | |
|     do_pseudo_reloc (u->pseudo_reloc_start, u->pseudo_reloc_end, u->image_base);
 | |
| }
 | |
| #else
 | |
| extern "C" void
 | |
| _pei386_runtime_relocator (void)
 | |
| {
 | |
|   static NO_COPY int was_init = 0;
 | |
|   if (was_init)
 | |
|     return;
 | |
|   ++was_init;
 | |
|   do_pseudo_reloc (&__RUNTIME_PSEUDO_RELOC_LIST__,
 | |
| 		   &__RUNTIME_PSEUDO_RELOC_LIST_END__,
 | |
| 		   &__MINGW_LSYMBOL(_image_base__));
 | |
| }
 | |
| #endif
 |