mirror of
https://github.com/jart/cosmopolitan.git
synced 2025-01-31 11:37:35 +00:00
ba180e754d
- Decouple zlib from libc - Add some underscores to mostly internal names
178 lines
6.5 KiB
C
178 lines
6.5 KiB
C
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
|
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
|
|
╞══════════════════════════════════════════════════════════════════════════════╡
|
|
│ Copyright 2020 Justine Alexandra Roberts Tunney │
|
|
│ │
|
|
│ Permission to use, copy, modify, and/or distribute this software for │
|
|
│ any purpose with or without fee is hereby granted, provided that the │
|
|
│ above copyright notice and this permission notice appear in all copies. │
|
|
│ │
|
|
│ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │
|
|
│ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │
|
|
│ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │
|
|
│ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │
|
|
│ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │
|
|
│ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │
|
|
│ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │
|
|
│ PERFORMANCE OF THIS SOFTWARE. │
|
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
|
#include "libc/assert.h"
|
|
#include "libc/intrin/bits.h"
|
|
#include "libc/intrin/promises.internal.h"
|
|
#include "libc/intrin/strace.internal.h"
|
|
#include "libc/intrin/weaken.h"
|
|
#include "libc/macros.internal.h"
|
|
#include "libc/runtime/internal.h"
|
|
#include "libc/runtime/runtime.h"
|
|
#include "libc/runtime/symbols.internal.h"
|
|
#include "libc/str/str.h"
|
|
#include "libc/thread/thread.h"
|
|
#include "libc/x/x.h"
|
|
#include "libc/zip.h"
|
|
#include "libc/zipos/zipos.internal.h"
|
|
#include "third_party/puff/puff.h"
|
|
|
|
static pthread_spinlock_t g_lock;
|
|
_Hide struct SymbolTable *__symtab; // for kprintf
|
|
|
|
/**
|
|
* Looks for `.symtab` in zip central directory.
|
|
*/
|
|
static ssize_t FindSymtabInZip(struct Zipos *zipos) {
|
|
size_t i, n, c;
|
|
c = GetZipCdirOffset(zipos->cdir);
|
|
n = GetZipCdirRecords(zipos->cdir);
|
|
for (i = 0; i < n; ++i, c += ZIP_CFILE_HDRSIZE(zipos->map + c)) {
|
|
if (ZIP_CFILE_NAMESIZE(zipos->map + c) == 7 &&
|
|
READ32LE(ZIP_CFILE_NAME(zipos->map + c + 0)) == READ32LE(".sym") &&
|
|
READ16LE(ZIP_CFILE_NAME(zipos->map + c + 4)) == READ16LE("ta") &&
|
|
*ZIP_CFILE_NAME(zipos->map + c + 6) == 'b') {
|
|
return c;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
/**
|
|
* Reads symbol table from zip directory.
|
|
* @note This code can't depend on dlmalloc()
|
|
*/
|
|
static struct SymbolTable *GetSymbolTableFromZip(struct Zipos *zipos) {
|
|
size_t size, size2;
|
|
ssize_t rc, cf, lf;
|
|
struct SymbolTable *res = 0;
|
|
if ((cf = FindSymtabInZip(zipos)) != -1) {
|
|
lf = GetZipCfileOffset(zipos->map + cf);
|
|
size = GetZipLfileUncompressedSize(zipos->map + lf);
|
|
size2 = ROUNDUP(size, FRAMESIZE);
|
|
if ((res = _mapanon(size2))) {
|
|
switch (ZIP_LFILE_COMPRESSIONMETHOD(zipos->map + lf)) {
|
|
case kZipCompressionNone:
|
|
memcpy(res, (void *)ZIP_LFILE_CONTENT(zipos->map + lf), size);
|
|
break;
|
|
case kZipCompressionDeflate:
|
|
if (__inflate((void *)res, size,
|
|
(void *)ZIP_LFILE_CONTENT(zipos->map + lf),
|
|
GetZipLfileCompressedSize(zipos->map + lf))) {
|
|
munmap(res, size2);
|
|
res = 0;
|
|
}
|
|
break;
|
|
default:
|
|
munmap(res, size2);
|
|
res = 0;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
STRACE("GetSymbolTableFromZip() → %p", res);
|
|
return res;
|
|
}
|
|
|
|
/**
|
|
* Reads symbol table from .com.dbg file.
|
|
* @note This code can't depend on dlmalloc()
|
|
*/
|
|
static struct SymbolTable *GetSymbolTableFromElf(void) {
|
|
int e;
|
|
const char *s;
|
|
if (PLEDGED(RPATH) && (s = FindDebugBinary())) {
|
|
return OpenSymbolTable(s);
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Returns symbol table singleton.
|
|
*
|
|
* This uses multiple strategies to find the symbol table. The first
|
|
* strategy, depends on whether or not the following is linked:
|
|
*
|
|
* STATIC_YOINK("__zipos_get");
|
|
*
|
|
* In that case, the symbol table may be read from `/zip/.symtab` which
|
|
* is generated by `o//tool/build/symtab.com`. The second strategy is to
|
|
* look for the concomitant `.com.dbg` executable, which may very well
|
|
* be the one currently executing, or it could be placed in the same
|
|
* folder as your `.com` binary, or lastly, it could be explicitly
|
|
* specified via the `COMDBG` environment variable.
|
|
*
|
|
* Function tracing is disabled throughout the duration of this call.
|
|
* Backtraces and other core runtime functionality depend on this.
|
|
*
|
|
* @return symbol table, or NULL if not found
|
|
*/
|
|
struct SymbolTable *GetSymbolTable(void) {
|
|
struct Zipos *z;
|
|
if (pthread_spin_trylock(&g_lock)) return 0;
|
|
if (!__symtab && !__isworker) {
|
|
if (_weaken(__zipos_get) && (z = _weaken(__zipos_get)())) {
|
|
if ((__symtab = GetSymbolTableFromZip(z))) {
|
|
__symtab->names =
|
|
(uint32_t *)((char *)__symtab + __symtab->names_offset);
|
|
__symtab->name_base =
|
|
(char *)((char *)__symtab + __symtab->name_base_offset);
|
|
}
|
|
}
|
|
if (!__symtab) {
|
|
__symtab = GetSymbolTableFromElf();
|
|
}
|
|
}
|
|
pthread_spin_unlock(&g_lock);
|
|
return __symtab;
|
|
}
|
|
|
|
/**
|
|
* Returns low index into symbol table for address.
|
|
*
|
|
* @param t if null will be auto-populated only if already open
|
|
* @return index or -1 if nothing found
|
|
*/
|
|
noinstrument privileged int __get_symbol(struct SymbolTable *t, intptr_t a) {
|
|
// we need privileged because:
|
|
// kprintf is privileged and it depends on this
|
|
// we don't want function tracing because:
|
|
// function tracing depends on this function via kprintf
|
|
unsigned l, m, r, n, k;
|
|
if (!t && __symtab) {
|
|
t = __symtab;
|
|
}
|
|
if (t) {
|
|
l = 0;
|
|
r = n = t->count;
|
|
k = a - t->addr_base;
|
|
while (l < r) {
|
|
m = (l + r) >> 1;
|
|
if (t->symbols[m].y < k) {
|
|
l = m + 1;
|
|
} else {
|
|
r = m;
|
|
}
|
|
}
|
|
if (l < n && t->symbols[l].x <= k && k <= t->symbols[l].y) {
|
|
return l;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|