mirror of
				https://github.com/jart/cosmopolitan.git
				synced 2025-10-25 10:40:57 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			200 lines
		
	
	
	
		
			6.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			200 lines
		
	
	
	
		
			6.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #ifndef COSMOPOLITAN_LIBC_RUNTIME_MEMTRACK_H_
 | |
| #define COSMOPOLITAN_LIBC_RUNTIME_MEMTRACK_H_
 | |
| #include "libc/assert.h"
 | |
| #include "libc/bits/midpoint.h"
 | |
| #include "libc/dce.h"
 | |
| #include "libc/intrin/nopl.h"
 | |
| #include "libc/macros.internal.h"
 | |
| #include "libc/nexgen32e/threaded.h"
 | |
| #include "libc/nt/version.h"
 | |
| #include "libc/runtime/stack.h"
 | |
| #include "libc/sysv/consts/ss.h"
 | |
| #if !(__ASSEMBLER__ + __LINKER__ + 0)
 | |
| COSMOPOLITAN_C_START_
 | |
| 
 | |
| #define kAutomapStart _kMem(0x100080000000, 0x000010000000)
 | |
| #define kAutomapSize                                             \
 | |
|   _kMem(0x200000000000 - 0x100080000000 - _kMmi(0x800000000000), \
 | |
|         0x000040000000 - 0x000010000000 - _kMmi(0x000080000000))
 | |
| #define kMemtrackStart                                      \
 | |
|   (ROUNDDOWN(_kMem(0x200000000000 - _kMmi(0x800000000000),  \
 | |
|                    0x000040000000 - _kMmi(0x000080000000)), \
 | |
|              FRAMESIZE * 8) -                               \
 | |
|    0x8000 * 8 /* so frame aligned after adding 0x7fff8000 */)
 | |
| #define kMemtrackSize  _kMem(_kMmi(0x800000000000), _kMmi(0x000080000000))
 | |
| #define kMemtrackGran  (!IsAsan() ? FRAMESIZE : FRAMESIZE * 8)
 | |
| #define kFixedmapStart _kMem(0x300000000000, 0x000040000000)
 | |
| #define kFixedmapSize \
 | |
|   _kMem(0x400000000000 - 0x300000000000, 0x000070000000 - 0x000040000000)
 | |
| #define kMemtrackFdsStart \
 | |
|   (ROUNDDOWN(_kMem(0x6fe000000000, 0x68000000), FRAMESIZE * 8) - 0x8000 * 8)
 | |
| #define kMemtrackFdsSize _kMem(0x001000000000, 0x04000000)
 | |
| #define kMemtrackZiposStart \
 | |
|   (ROUNDDOWN(_kMem(0x6fd000000000, 0x6c000000), FRAMESIZE * 8) - 0x8000 * 8)
 | |
| #define kMemtrackZiposSize _kMem(0x001000000000, 0x10000000)
 | |
| #define _kMmi(VSPACE)                                                   \
 | |
|   ROUNDUP(VSPACE / FRAMESIZE * (intptr_t)sizeof(struct MemoryInterval), \
 | |
|           FRAMESIZE)
 | |
| #define _kMem(NORMAL, WIN7) \
 | |
|   (!IsWindows() || IsAtLeastWindows10() ? NORMAL : WIN7)
 | |
| 
 | |
| struct MemoryInterval {
 | |
|   int x;
 | |
|   int y;
 | |
|   long h;
 | |
|   long size;
 | |
|   int prot;
 | |
|   int flags;
 | |
|   long offset;
 | |
|   bool iscow;
 | |
|   bool readonlyfile;
 | |
| };
 | |
| 
 | |
| struct MemoryIntervals {
 | |
|   size_t i, n;
 | |
|   struct MemoryInterval *p;
 | |
|   struct MemoryInterval s[OPEN_MAX];
 | |
| };
 | |
| 
 | |
| extern hidden struct MemoryIntervals _mmi;
 | |
| 
 | |
| void __mmi_lock(void) hidden;
 | |
| void __mmi_unlock(void) hidden;
 | |
| bool IsMemtracked(int, int) hidden;
 | |
| void PrintSystemMappings(int) hidden;
 | |
| bool AreMemoryIntervalsOk(const struct MemoryIntervals *) nosideeffect hidden;
 | |
| void PrintMemoryIntervals(int, const struct MemoryIntervals *) hidden;
 | |
| int TrackMemoryInterval(struct MemoryIntervals *, int, int, long, int, int,
 | |
|                         bool, bool, long, long) hidden;
 | |
| int ReleaseMemoryIntervals(struct MemoryIntervals *, int, int,
 | |
|                            void (*)(struct MemoryIntervals *, int, int)) hidden;
 | |
| void ReleaseMemoryNt(struct MemoryIntervals *, int, int) hidden;
 | |
| int UntrackMemoryIntervals(void *, size_t) hidden;
 | |
| size_t GetMemtrackSize(struct MemoryIntervals *);
 | |
| 
 | |
| #ifdef _NOPL0
 | |
| #define __mmi_lock()   _NOPL0("__threadcalls", __mmi_lock)
 | |
| #define __mmi_unlock() _NOPL0("__threadcalls", __mmi_unlock)
 | |
| #else
 | |
| #define __mmi_lock()   (__threaded ? __mmi_lock() : 0)
 | |
| #define __mmi_unlock() (__threaded ? __mmi_unlock() : 0)
 | |
| #endif
 | |
| 
 | |
| #define IsLegalPointer(p) \
 | |
|   (-0x800000000000 <= (intptr_t)(p) && (intptr_t)(p) <= 0x7fffffffffff)
 | |
| 
 | |
| forceinline pureconst bool IsLegalSize(size_t n) {
 | |
|   return n <= 0x7fffffffffff;
 | |
| }
 | |
| 
 | |
| forceinline pureconst bool IsAutoFrame(int x) {
 | |
|   return (kAutomapStart >> 16) <= x &&
 | |
|          x <= ((kAutomapStart + (kAutomapSize - 1)) >> 16);
 | |
| }
 | |
| 
 | |
| forceinline pureconst bool IsMemtrackFrame(int x) {
 | |
|   return (kAutomapStart >> 16) <= x &&
 | |
|          x <= ((kAutomapStart + (kAutomapSize - 1)) >> 16);
 | |
| }
 | |
| 
 | |
| forceinline pureconst bool IsArenaFrame(int x) {
 | |
|   return 0x5000 <= x && x < 0x7ffe;
 | |
| }
 | |
| 
 | |
| forceinline pureconst bool IsShadowFrame(int x) {
 | |
|   return 0x7fff <= x && x < 0x10008000;
 | |
| }
 | |
| 
 | |
| forceinline pureconst bool IsKernelFrame(int x) {
 | |
|   intptr_t stack = (intptr_t)GetStaticStackAddr(0);
 | |
|   return (int)(stack >> 16) <= x &&
 | |
|          x <= (int)((stack + (GetStackSize() - FRAMESIZE)) >> 16);
 | |
| }
 | |
| 
 | |
| forceinline pureconst bool IsStaticStackFrame(int x) {
 | |
|   intptr_t stack = (intptr_t)GetStaticStackAddr(0);
 | |
|   return (int)(stack >> 16) <= x &&
 | |
|          x <= (int)((stack + (GetStackSize() - FRAMESIZE)) >> 16);
 | |
| }
 | |
| 
 | |
| forceinline pureconst bool IsStackFrame(int x) {
 | |
|   intptr_t stack = (intptr_t)GetStackAddr(0);
 | |
|   return (int)(stack >> 16) <= x &&
 | |
|          x <= (int)((stack + (GetStackSize() - FRAMESIZE)) >> 16);
 | |
| }
 | |
| 
 | |
| forceinline pureconst bool IsSigAltStackFrame(int x) {
 | |
|   intptr_t stack = (intptr_t)GetStackAddr(0);
 | |
|   return (int)(stack >> 16) <= x &&
 | |
|          x <= (int)((stack + (SIGSTKSZ - FRAMESIZE)) >> 16);
 | |
| }
 | |
| 
 | |
| forceinline pureconst bool IsOldStackFrame(int x) {
 | |
|   intptr_t old = ROUNDDOWN(__oldstack, STACKSIZE);
 | |
|   return (old >> 16) <= x && x <= ((old + (STACKSIZE - FRAMESIZE)) >> 16);
 | |
| }
 | |
| 
 | |
| forceinline pureconst bool IsFixedFrame(int x) {
 | |
|   return (kFixedmapStart >> 16) <= x &&
 | |
|          x <= ((kFixedmapStart + (kFixedmapSize - 1)) >> 16);
 | |
| }
 | |
| 
 | |
| forceinline pureconst bool OverlapsImageSpace(const void *p, size_t n) {
 | |
|   const unsigned char *start, *ender;
 | |
|   if (n) {
 | |
|     start = p;
 | |
|     ender = start + (n - 1);
 | |
|     return ((_base <= start && start < _end) ||
 | |
|             (_base <= ender && ender < _end) ||
 | |
|             (start < _base && _end <= ender));
 | |
|   } else {
 | |
|     return 0;
 | |
|   }
 | |
| }
 | |
| 
 | |
| forceinline pureconst bool OverlapsArenaSpace(const void *p, size_t n) {
 | |
|   intptr_t x, y;
 | |
|   if (n) {
 | |
|     x = (intptr_t)p;
 | |
|     y = x + (n - 1);
 | |
|     return ((0x50000000 <= x && x <= 0x7ffdffff) ||
 | |
|             (0x50000000 <= y && y <= 0x7ffdffff) ||
 | |
|             (x < 0x50000000 && 0x7ffdffff < y));
 | |
|   } else {
 | |
|     return 0;
 | |
|   }
 | |
| }
 | |
| 
 | |
| forceinline pureconst bool OverlapsShadowSpace(const void *p, size_t n) {
 | |
|   intptr_t x, y;
 | |
|   if (n) {
 | |
|     x = (intptr_t)p;
 | |
|     y = x + (n - 1);
 | |
|     return ((0x7fff0000 <= x && x <= 0x10007fffffff) ||
 | |
|             (0x7fff0000 <= y && y <= 0x10007fffffff) ||
 | |
|             (x < 0x7fff0000 && 0x10007fffffff < y));
 | |
|   } else {
 | |
|     return 0;
 | |
|   }
 | |
| }
 | |
| 
 | |
| forceinline unsigned FindMemoryInterval(const struct MemoryIntervals *mm,
 | |
|                                         int x) {
 | |
|   unsigned l, m, r;
 | |
|   l = 0;
 | |
|   r = mm->i;
 | |
|   while (l < r) {
 | |
|     m = _midpoint(l, r);
 | |
|     if (mm->p[m].y < x) {
 | |
|       l = m + 1;
 | |
|     } else {
 | |
|       r = m;
 | |
|     }
 | |
|   }
 | |
|   assert(l == mm->i || x <= mm->p[l].y);
 | |
|   return l;
 | |
| }
 | |
| 
 | |
| COSMOPOLITAN_C_END_
 | |
| #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */
 | |
| #endif /* COSMOPOLITAN_LIBC_RUNTIME_MEMTRACK_H_ */
 |