mirror of
https://github.com/upx/upx
synced 2025-09-28 19:06:07 +08:00
721 lines
15 KiB
C++
721 lines
15 KiB
C++
/* util.cpp --
|
|
|
|
This file is part of the UPX executable compressor.
|
|
|
|
Copyright (C) 1996-2004 Markus Franz Xaver Johannes Oberhumer
|
|
Copyright (C) 1996-2004 Laszlo Molnar
|
|
All Rights Reserved.
|
|
|
|
UPX and the UCL library are free software; you can redistribute them
|
|
and/or modify them under the terms of the GNU General Public License as
|
|
published by the Free Software Foundation; either version 2 of
|
|
the License, or (at your option) any later version.
|
|
|
|
This program 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 General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; see the file COPYING.
|
|
If not, write to the Free Software Foundation, Inc.,
|
|
59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
Markus F.X.J. Oberhumer Laszlo Molnar
|
|
<mfx@users.sourceforge.net> <ml1050@users.sourceforge.net>
|
|
*/
|
|
|
|
|
|
#include "conf.h"
|
|
#include "util.h"
|
|
|
|
#if (ACC_CC_MSC && (_MSC_VER >= 1000 && _MSC_VER < 1200))
|
|
/* avoid -W4 warnings in <conio.h> */
|
|
# pragma warning(disable: 4032)
|
|
/* avoid -W4 warnings in <windows.h> */
|
|
# pragma warning(disable: 4201 4214 4514)
|
|
#endif
|
|
#include "acc/acc_inci.h"
|
|
#include "acc/acclib/misc.ch"
|
|
#include "acc/acclib/hsread.ch"
|
|
#include "acc/acclib/wildargv.ch"
|
|
|
|
|
|
/*************************************************************************
|
|
// qsort() util
|
|
**************************************************************************/
|
|
|
|
int __acc_cdecl_qsort be16_compare(const void *e1, const void *e2)
|
|
{
|
|
const unsigned d1 = get_be16(e1);
|
|
const unsigned d2 = get_be16(e2);
|
|
return (d1 < d2) ? -1 : ((d1 > d2) ? 1 : 0);
|
|
}
|
|
|
|
int __acc_cdecl_qsort be32_compare(const void *e1, const void *e2)
|
|
{
|
|
const unsigned d1 = get_be32(e1);
|
|
const unsigned d2 = get_be32(e2);
|
|
return (d1 < d2) ? -1 : ((d1 > d2) ? 1 : 0);
|
|
}
|
|
|
|
int __acc_cdecl_qsort le16_compare(const void *e1, const void *e2)
|
|
{
|
|
const unsigned d1 = get_le16(e1);
|
|
const unsigned d2 = get_le16(e2);
|
|
return (d1 < d2) ? -1 : ((d1 > d2) ? 1 : 0);
|
|
}
|
|
|
|
int __acc_cdecl_qsort le32_compare(const void *e1, const void *e2)
|
|
{
|
|
const unsigned d1 = get_le32(e1);
|
|
const unsigned d2 = get_le32(e2);
|
|
return (d1 < d2) ? -1 : ((d1 > d2) ? 1 : 0);
|
|
}
|
|
|
|
|
|
int __acc_cdecl_qsort be16_compare_signed(const void *e1, const void *e2)
|
|
{
|
|
const int d1 = get_be16_signed(e1);
|
|
const int d2 = get_be16_signed(e2);
|
|
return (d1 < d2) ? -1 : ((d1 > d2) ? 1 : 0);
|
|
}
|
|
|
|
int __acc_cdecl_qsort be32_compare_signed(const void *e1, const void *e2)
|
|
{
|
|
const int d1 = get_be32_signed(e1);
|
|
const int d2 = get_be32_signed(e2);
|
|
return (d1 < d2) ? -1 : ((d1 > d2) ? 1 : 0);
|
|
}
|
|
|
|
int __acc_cdecl_qsort le16_compare_signed(const void *e1, const void *e2)
|
|
{
|
|
const int d1 = get_le16_signed(e1);
|
|
const int d2 = get_le16_signed(e2);
|
|
return (d1 < d2) ? -1 : ((d1 > d2) ? 1 : 0);
|
|
}
|
|
|
|
int __acc_cdecl_qsort le32_compare_signed(const void *e1, const void *e2)
|
|
{
|
|
const int d1 = get_le32_signed(e1);
|
|
const int d2 = get_le32_signed(e2);
|
|
return (d1 < d2) ? -1 : ((d1 > d2) ? 1 : 0);
|
|
}
|
|
|
|
|
|
/*************************************************************************
|
|
// find util
|
|
**************************************************************************/
|
|
|
|
int find(const void *b, int blen, const void *what, int wlen)
|
|
{
|
|
if (b == NULL || blen <= 0 || what == NULL || wlen <= 0)
|
|
return -1;
|
|
|
|
int i;
|
|
const unsigned char *base = (const unsigned char *) b;
|
|
unsigned char firstc = * (const unsigned char *) what;
|
|
|
|
blen -= wlen;
|
|
for (i = 0; i <= blen; i++, base++)
|
|
if (*base == firstc && memcmp(base, what, wlen) == 0)
|
|
return i;
|
|
|
|
return -1;
|
|
}
|
|
|
|
|
|
int find_be16(const void *b, int blen, unsigned what)
|
|
{
|
|
unsigned char w[2];
|
|
set_be16(w, what);
|
|
return find(b, blen, w, 2);
|
|
}
|
|
|
|
|
|
int find_be32(const void *b, int blen, unsigned what)
|
|
{
|
|
unsigned char w[4];
|
|
set_be32(w, what);
|
|
return find(b, blen, w, 4);
|
|
}
|
|
|
|
|
|
int find_le16(const void *b, int blen, unsigned what)
|
|
{
|
|
unsigned char w[2];
|
|
set_le16(w, what);
|
|
return find(b, blen, w, 2);
|
|
}
|
|
|
|
|
|
int find_le32(const void *b, int blen, unsigned what)
|
|
{
|
|
unsigned char w[4];
|
|
set_le32(w, what);
|
|
return find(b, blen, w, 4);
|
|
}
|
|
|
|
|
|
/*************************************************************************
|
|
// find util
|
|
**************************************************************************/
|
|
|
|
#if (UPX_VERSION_HEX < 0x019000)
|
|
|
|
upx_bytep pfind(const void *b, int blen, const void *what, int wlen)
|
|
{
|
|
if (b == NULL || blen <= 0 || what == NULL || wlen <= 0)
|
|
return NULL;
|
|
|
|
int i;
|
|
const upx_bytep base = (const upx_bytep) b;
|
|
unsigned char firstc = * (const upx_bytep) what;
|
|
|
|
blen -= wlen;
|
|
for (i = 0; i <= blen; i++, base++)
|
|
if (*base == firstc && memcmp(base, what, wlen) == 0)
|
|
return const_cast<upx_bytep>(base);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
upx_bytep pfind_be16(const void *b, int blen, unsigned what)
|
|
{
|
|
unsigned char w[2];
|
|
set_be16(w, what);
|
|
return pfind(b, blen, w, 2);
|
|
}
|
|
|
|
|
|
upx_bytep pfind_be32(const void *b, int blen, unsigned what)
|
|
{
|
|
unsigned char w[4];
|
|
set_be32(w, what);
|
|
return pfind(b, blen, w, 4);
|
|
}
|
|
|
|
|
|
upx_bytep pfind_le16(const void *b, int blen, unsigned what)
|
|
{
|
|
unsigned char w[2];
|
|
set_le16(w, what);
|
|
return pfind(b, blen, w, 2);
|
|
}
|
|
|
|
|
|
upx_bytep pfind_le32(const void *b, int blen, unsigned what)
|
|
{
|
|
unsigned char w[4];
|
|
set_le32(w, what);
|
|
return pfind(b, blen, w, 4);
|
|
}
|
|
|
|
#endif /* UPX_VERSION_HEX */
|
|
|
|
|
|
/*************************************************************************
|
|
// ctype util
|
|
**************************************************************************/
|
|
|
|
#if 0
|
|
bool upx_isdigit(int c)
|
|
{
|
|
return c >= '0' && c <= '9';
|
|
}
|
|
|
|
bool upx_islower(int c)
|
|
{
|
|
return c >= 'a' && c <= 'z';
|
|
}
|
|
|
|
bool upx_isspace(int c)
|
|
{
|
|
// according to "C" and "POSIX" locales
|
|
return strchr(" \f\n\r\t\v", c) != NULL;
|
|
}
|
|
|
|
int upx_tolower(int c)
|
|
{
|
|
if (c >= 'A' && c <= 'Z')
|
|
c += 'a' - 'A';
|
|
return c;
|
|
}
|
|
#endif
|
|
|
|
|
|
/*************************************************************************
|
|
// filename util
|
|
**************************************************************************/
|
|
|
|
#if (ACC_OS_CYGWIN || ACC_OS_DOS16 || ACC_OS_DOS32 || ACC_OS_EMX || ACC_OS_OS2 || ACC_OS_OS16 || ACC_OS_TOS || ACC_OS_WIN16 || ACC_OS_WIN32 || ACC_OS_WIN64)
|
|
|
|
static const char dir_sep[] = "/\\";
|
|
#define fn_is_drive(s) (s[0] && s[1] == ':')
|
|
#define fn_is_sep(c) (strchr(dir_sep,c) != NULL)
|
|
#define fn_skip_drive(s) (fn_is_drive(s) ? (s) + 2 : (s))
|
|
#define fn_tolower(c) (tolower(((unsigned char)(c))))
|
|
|
|
#else
|
|
|
|
static const char dir_sep[] = "/";
|
|
#define fn_is_drive(s) (0)
|
|
#define fn_is_sep(c) ((c) == '/')
|
|
#define fn_skip_drive(s) (s)
|
|
#define fn_tolower(c) (c)
|
|
|
|
#endif
|
|
|
|
|
|
char *fn_basename(const char *name)
|
|
{
|
|
const char *n, *nn;
|
|
|
|
name = fn_skip_drive(name);
|
|
for (nn = n = name; *nn; nn++)
|
|
if (fn_is_sep(*nn))
|
|
n = nn + 1;
|
|
return const_cast<char *>(n);
|
|
}
|
|
|
|
|
|
bool fn_has_ext(const char *name, const char *ext, bool ignore_case)
|
|
{
|
|
const char *n, *e;
|
|
|
|
name = fn_basename(name);
|
|
for (n = e = name; *n; n++)
|
|
if (*n == '.')
|
|
e = n;
|
|
if (ignore_case)
|
|
return (strcasecmp(ext,e+1) == 0);
|
|
else
|
|
return (fn_strcmp(ext,e+1) == 0);
|
|
}
|
|
|
|
|
|
void fn_addslash(char *name, bool slash)
|
|
{
|
|
char *p;
|
|
|
|
name = fn_skip_drive(name);
|
|
p = name + strlen(name);
|
|
while (p > name && fn_is_sep(p[-1]))
|
|
*p-- = 0;
|
|
if (p > name)
|
|
{
|
|
if (slash)
|
|
*p++ = dir_sep[0];
|
|
*p = 0;
|
|
}
|
|
}
|
|
|
|
|
|
char *fn_strlwr(char *n)
|
|
{
|
|
char *p;
|
|
for (p = n; *p; p++)
|
|
*p = (char) fn_tolower(*p);
|
|
return n;
|
|
}
|
|
|
|
|
|
int fn_strcmp(const char *n1, const char *n2)
|
|
{
|
|
for (;;)
|
|
{
|
|
if (*n1 != *n2)
|
|
{
|
|
int c = fn_tolower(*n1) - fn_tolower(*n2);
|
|
if (c)
|
|
return c;
|
|
}
|
|
if (*n1 == 0)
|
|
return 0;
|
|
n1++; n2++;
|
|
}
|
|
}
|
|
|
|
|
|
bool fn_is_same_file(const char *n1, const char *n2)
|
|
{
|
|
/* very simple... */
|
|
if (fn_strcmp(n1, n2) == 0)
|
|
return 1;
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*************************************************************************
|
|
// time util
|
|
**************************************************************************/
|
|
|
|
#if 0 // not used
|
|
|
|
#if defined(HAVE_LOCALTIME)
|
|
void tm2str(char *s, size_t size, const struct tm *tmp)
|
|
{
|
|
upx_snprintf(s, size, "%04d-%02d-%02d %02d:%02d:%02d",
|
|
(int) tmp->tm_year + 1900, (int) tmp->tm_mon + 1,
|
|
(int) tmp->tm_mday,
|
|
(int) tmp->tm_hour, (int) tmp->tm_min, (int) tmp->tm_sec);
|
|
}
|
|
#endif
|
|
|
|
|
|
void time2str(char *s, size_t size, const time_t *t)
|
|
{
|
|
assert(size >= 18);
|
|
#if defined(HAVE_LOCALTIME)
|
|
tm2str(s, size, localtime(t));
|
|
#elif defined(HAVE_CTIME)
|
|
const char *p = ctime(t);
|
|
memset(s, ' ', 16);
|
|
memcpy(s + 2, p + 4, 6);
|
|
memcpy(s + 11, p + 11, 5);
|
|
s[16] = 0;
|
|
#else
|
|
s[0] = 0;
|
|
#endif
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
/*************************************************************************
|
|
// misc.
|
|
**************************************************************************/
|
|
|
|
bool set_method_name(char *buf, size_t size, int method, int level)
|
|
{
|
|
bool r = true;
|
|
const char *alg;
|
|
if (M_IS_NRV2B(method))
|
|
alg = "NRV2B";
|
|
else if (M_IS_NRV2D(method))
|
|
alg = "NRV2D";
|
|
else if (M_IS_NRV2E(method))
|
|
alg = "NRV2E";
|
|
else
|
|
{
|
|
alg = "???";
|
|
r = false;
|
|
}
|
|
if (level > 0)
|
|
upx_snprintf(buf, size, "%s/%d", alg, level);
|
|
else
|
|
upx_snprintf(buf, size, "%s", alg);
|
|
return r;
|
|
}
|
|
|
|
|
|
void center_string(char *buf, size_t size, const char *s)
|
|
{
|
|
size_t l1 = size - 1;
|
|
size_t l2 = strlen(s);
|
|
assert(size > 0);
|
|
assert(l2 < size);
|
|
memset(buf, ' ', l1);
|
|
memcpy(buf+(l1-l2)/2, s, l2);
|
|
buf[l1] = 0;
|
|
}
|
|
|
|
|
|
bool file_exists(const char *name)
|
|
{
|
|
int fd, r;
|
|
struct stat st;
|
|
|
|
/* return true if we can open it */
|
|
fd = open(name, O_RDONLY, 0);
|
|
if (fd >= 0)
|
|
{
|
|
(void) close(fd);
|
|
return true;
|
|
}
|
|
|
|
/* return true if we can stat it */
|
|
//memset(&st, 0, sizeof(st));
|
|
r = stat(name, &st);
|
|
if (r != -1)
|
|
return true;
|
|
|
|
/* return true if we can lstat it */
|
|
#if defined(HAVE_LSTAT)
|
|
//memset(&st, 0, sizeof(st));
|
|
r = lstat(name, &st);
|
|
if (r != -1)
|
|
return true;
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
bool maketempname(char *ofilename, size_t size,
|
|
const char *ifilename, const char *ext, bool force)
|
|
{
|
|
char *ofext = NULL, *ofname;
|
|
int ofile;
|
|
|
|
if (size <= 0)
|
|
return false;
|
|
|
|
strcpy(ofilename, ifilename);
|
|
for (ofname = fn_basename(ofilename); *ofname; ofname++)
|
|
{
|
|
if (*ofname == '.')
|
|
ofext = ofname;
|
|
}
|
|
if (ofext == NULL)
|
|
ofext = ofilename + strlen(ofilename);
|
|
strcpy(ofext, ext);
|
|
|
|
for (ofile = 0; ofile < 1000; ofile++)
|
|
{
|
|
assert(strlen(ofilename) < size);
|
|
if (!file_exists(ofilename))
|
|
return true;
|
|
if (!force)
|
|
break;
|
|
upx_snprintf(ofext, 5, ".%03d", ofile);
|
|
}
|
|
|
|
ofilename[0] = 0;
|
|
return false;
|
|
}
|
|
|
|
|
|
bool makebakname(char *ofilename, size_t size,
|
|
const char *ifilename, bool force)
|
|
{
|
|
char *ofext = NULL, *ofname;
|
|
int ofile;
|
|
|
|
if (size <= 0)
|
|
return false;
|
|
|
|
strcpy(ofilename, ifilename);
|
|
for (ofname = fn_basename(ofilename); *ofname; ofname++)
|
|
{
|
|
if (*ofname == '.')
|
|
ofext = ofname;
|
|
}
|
|
if (ofext == NULL)
|
|
{
|
|
ofext = ofilename + strlen(ofilename);
|
|
strcpy(ofext, ".~");
|
|
}
|
|
else if (strlen(ofext) < 1 + 3)
|
|
strcat(ofilename, "~");
|
|
else
|
|
ofext[strlen(ofext)-1] = '~';
|
|
|
|
for (ofile = 0; ofile < 1000; ofile++)
|
|
{
|
|
assert(strlen(ofilename) < size);
|
|
if (!file_exists(ofilename))
|
|
return true;
|
|
if (!force)
|
|
break;
|
|
upx_snprintf(ofext, 5, ".%03d", ofile);
|
|
}
|
|
|
|
ofilename[0] = 0;
|
|
return false;
|
|
}
|
|
|
|
|
|
/*************************************************************************
|
|
// return compression ratio, where 100% == 1000*1000 == 1e6
|
|
**************************************************************************/
|
|
|
|
unsigned get_ratio(unsigned u_len, unsigned c_len)
|
|
{
|
|
const unsigned n = 1000000;
|
|
if (u_len <= 0)
|
|
return c_len <= 0 ? 0 : n;
|
|
#if defined(acc_uint64l_t)
|
|
return (unsigned) ((c_len * (acc_uint64l_t)n) / u_len);
|
|
#else
|
|
# if 0
|
|
return (unsigned) acc_umuldiv32(c_len, n, u_len);
|
|
# else
|
|
return (unsigned) ((c_len * (double)n) / u_len);
|
|
# endif
|
|
#endif
|
|
}
|
|
|
|
|
|
/*************************************************************************
|
|
// memory debugging
|
|
**************************************************************************/
|
|
|
|
#if defined(WITH_GC)
|
|
extern "C" {
|
|
|
|
#undef malloc
|
|
#undef realloc
|
|
#undef free
|
|
#ifndef __malloc_ptr_t
|
|
# define __malloc_ptr_t __ptr_t
|
|
#endif
|
|
|
|
__malloc_ptr_t malloc(size_t size)
|
|
{
|
|
return GC_MALLOC(size);
|
|
}
|
|
|
|
__malloc_ptr_t realloc(__malloc_ptr_t ptr, size_t size)
|
|
{
|
|
return GC_REALLOC(ptr, size);
|
|
}
|
|
|
|
void free(__malloc_ptr_t ptr)
|
|
{
|
|
GC_FREE(ptr);
|
|
}
|
|
|
|
}; // extern "C"
|
|
#endif
|
|
|
|
|
|
/*************************************************************************
|
|
// Don't link these functions from libc ==> save xxx bytes
|
|
**************************************************************************/
|
|
|
|
extern "C" {
|
|
|
|
#if defined(__DJGPP__)
|
|
int _is_executable(const char *, int, const char *)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
// FIXME: something wants to link in ctime.o
|
|
time_t mktime(struct tm *)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
time_t time(time_t *t)
|
|
{
|
|
if (t) *t = 0;
|
|
return 0;
|
|
}
|
|
#endif /* __DJGPP__ */
|
|
|
|
|
|
// These space savings are only useful when building a statically
|
|
// linked version, so this is disabled for now.
|
|
#if 0 && defined(__linux__) && defined(__GLIBC__)
|
|
|
|
#if 1
|
|
// We don't need floating point support in printf().
|
|
// See glibc/stdio-common/*
|
|
int __printf_fp(void)
|
|
{
|
|
assert(0);
|
|
return -1;
|
|
}
|
|
int __printf_fphex(void)
|
|
{
|
|
assert(0);
|
|
return -1;
|
|
}
|
|
#endif
|
|
|
|
|
|
#if 1
|
|
// We don't need multibyte character support.
|
|
// See <wchar.h> and glibc/xxx/vfprintf.c
|
|
int mblen(const char *, size_t)
|
|
{
|
|
assert(0);
|
|
return -1;
|
|
}
|
|
int mbtowc(...)
|
|
{
|
|
assert(0);
|
|
return -1;
|
|
}
|
|
int mbrtowc(...)
|
|
{
|
|
assert(0);
|
|
return -1;
|
|
}
|
|
// glibc internals
|
|
size_t __ctype_get_mb_cur_max(void)
|
|
{
|
|
assert(0);
|
|
return 1;
|
|
}
|
|
int __mbrlen(...)
|
|
{
|
|
assert(0);
|
|
return -1;
|
|
}
|
|
int __mbrtowc(...)
|
|
{
|
|
assert(0);
|
|
return -1;
|
|
}
|
|
int __wcrtomb(...)
|
|
{
|
|
assert(0);
|
|
return -1;
|
|
}
|
|
int __wcsrtombs(...)
|
|
{
|
|
assert(0);
|
|
return -1;
|
|
}
|
|
// see strtol.c
|
|
int __btowc(...)
|
|
{
|
|
assert(0);
|
|
return -1;
|
|
}
|
|
#endif
|
|
|
|
|
|
#if 1
|
|
// We don't need *scanf.
|
|
// See iovsscanf.c
|
|
// (libc6 2.1.1-10: this function pulls in ~80kB code !)
|
|
int _IO_vsscanf(void)
|
|
{
|
|
assert(0);
|
|
return -1;
|
|
}
|
|
#endif
|
|
|
|
|
|
#if 0
|
|
// FIXME - also should get rid of this intl stuff
|
|
// see glibc/intl/*
|
|
static const char _nl_default_default_domain[] = "messages";
|
|
static const char *_nl_current_default_domain = _nl_default_default_domain;
|
|
static const char _nl_default_dirname[] = "";
|
|
char *bindtextdomain(const char *, const char *)
|
|
{
|
|
return NULL;
|
|
}
|
|
char *textdomain(const char *)
|
|
{
|
|
return (char *) _nl_current_default_domain;
|
|
}
|
|
#endif
|
|
|
|
#endif /* __linux__ && __GLIBC__ */
|
|
|
|
|
|
} // extern "C"
|
|
|
|
|
|
/*
|
|
vi:ts=4:et
|
|
*/
|
|
|