2013-06-14 00:51:33 +00:00
|
|
|
/* radare - LGPL - Copyright 2009-2013 - pancake */
|
2010-03-12 17:46:11 +00:00
|
|
|
|
2011-03-10 10:36:16 +00:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <dirent.h>
|
2009-04-01 01:40:04 +00:00
|
|
|
#include <r_types.h>
|
2009-12-30 10:03:18 +00:00
|
|
|
#include <r_util.h>
|
2011-09-25 04:57:13 +00:00
|
|
|
#if __linux__ && __GNU_LIBRARY__
|
|
|
|
# include <execinfo.h>
|
2010-11-08 18:30:25 +00:00
|
|
|
#endif
|
2013-08-18 22:36:17 +00:00
|
|
|
#if __APPLE__
|
|
|
|
#include <errno.h>
|
|
|
|
#include <libproc.h>
|
|
|
|
#endif
|
2011-09-25 04:57:13 +00:00
|
|
|
#if __UNIX__
|
|
|
|
# include <sys/wait.h>
|
|
|
|
# include <sys/stat.h>
|
|
|
|
# include <errno.h>
|
|
|
|
# include <signal.h>
|
2013-06-05 21:58:31 +00:00
|
|
|
#ifdef __HAIKU__
|
|
|
|
# define Sleep sleep
|
|
|
|
#endif
|
2010-02-19 18:59:22 +00:00
|
|
|
#elif __WINDOWS__
|
2011-09-25 04:57:13 +00:00
|
|
|
# include <io.h>
|
2013-01-23 13:50:50 +00:00
|
|
|
# include <winbase.h>
|
2010-01-15 00:32:28 +00:00
|
|
|
#endif
|
2009-04-01 01:30:36 +00:00
|
|
|
|
2013-06-14 00:51:33 +00:00
|
|
|
R_LIB_VERSION(r_util);
|
|
|
|
|
2010-10-17 21:03:54 +00:00
|
|
|
/* TODO: import stuff fron bininfo/p/bininfo_addr2line */
|
2010-03-12 18:11:43 +00:00
|
|
|
/* TODO: check endianness issues here */
|
2010-06-30 00:30:07 +00:00
|
|
|
R_API ut64 r_sys_now(void) {
|
2010-03-12 17:46:11 +00:00
|
|
|
ut64 ret;
|
|
|
|
struct timeval now;
|
|
|
|
gettimeofday (&now, NULL);
|
|
|
|
ret = now.tv_sec;
|
|
|
|
ret <<= 32;
|
2012-10-19 22:31:18 +00:00
|
|
|
ret += now.tv_usec;
|
2010-03-12 17:46:11 +00:00
|
|
|
//(sizeof (now.tv_sec) == 4
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-08-10 09:35:38 +00:00
|
|
|
R_API int r_sys_truncate(const char *file, int sz) {
|
|
|
|
#if __WINDOWS__
|
2012-12-20 11:11:09 +00:00
|
|
|
int fd = r_sandbox_open (file, O_RDWR, 0644);
|
2012-08-10 09:35:38 +00:00
|
|
|
if (!fd) return R_FALSE;
|
|
|
|
ftruncate (fd, sz);
|
|
|
|
close (fd);
|
|
|
|
return R_TRUE;
|
|
|
|
#else
|
|
|
|
return truncate (file, sz)? R_FALSE: R_TRUE;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2011-02-13 00:37:02 +00:00
|
|
|
R_API RList *r_sys_dir(const char *path) {
|
|
|
|
struct dirent *entry;
|
2013-04-18 07:39:37 +00:00
|
|
|
DIR *dir = r_sandbox_opendir (path);
|
2011-02-13 00:37:02 +00:00
|
|
|
if (dir) {
|
|
|
|
RList *list = r_list_new ();
|
|
|
|
if (list) {
|
|
|
|
list->free = free;
|
|
|
|
while ((entry = readdir (dir))) {
|
|
|
|
r_list_append (list, strdup (entry->d_name));
|
|
|
|
}
|
|
|
|
closedir (dir);
|
|
|
|
return list;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2010-04-13 19:19:54 +00:00
|
|
|
R_API char *r_sys_cmd_strf(const char *fmt, ...) {
|
2010-09-09 22:08:53 +00:00
|
|
|
char *ret, cmd[4096];
|
2010-04-13 19:19:54 +00:00
|
|
|
va_list ap;
|
2011-11-15 23:44:18 +00:00
|
|
|
va_start (ap, fmt);
|
2010-04-13 19:19:54 +00:00
|
|
|
vsnprintf (cmd, sizeof (cmd), fmt, ap);
|
|
|
|
ret = r_sys_cmd_str (cmd, NULL, NULL);
|
2011-11-15 23:44:18 +00:00
|
|
|
va_end (ap);
|
2010-04-13 19:19:54 +00:00
|
|
|
return ret;
|
2009-04-01 01:30:36 +00:00
|
|
|
}
|
2009-04-03 11:11:17 +00:00
|
|
|
|
2013-01-02 01:49:18 +00:00
|
|
|
#ifdef __MAC_10_7
|
|
|
|
#define APPLE_WITH_BACKTRACE 1
|
|
|
|
#endif
|
|
|
|
#ifdef __IPHONE_4_0
|
|
|
|
#define APPLE_WITH_BACKTRACE 1
|
|
|
|
#endif
|
|
|
|
|
2010-06-30 00:30:07 +00:00
|
|
|
R_API void r_sys_backtrace(void) {
|
2013-01-02 01:49:18 +00:00
|
|
|
#if (__linux__ && __GNU_LIBRARY__) || (__APPLE__ && APPLE_WITH_BACKTRACE)
|
2010-05-24 10:07:54 +00:00
|
|
|
void *array[10];
|
2012-10-19 22:31:18 +00:00
|
|
|
size_t i, size = backtrace (array, 10);
|
2013-07-15 00:51:55 +00:00
|
|
|
char **strings = (char **)(size_t)backtrace_symbols (array, size);
|
2010-05-24 10:07:54 +00:00
|
|
|
printf ("Backtrace %zd stack frames.\n", size);
|
|
|
|
for (i = 0; i < size; i++)
|
|
|
|
printf ("%s\n", strings[i]);
|
|
|
|
free (strings);
|
2011-03-13 20:00:46 +00:00
|
|
|
#elif __APPLE__
|
|
|
|
void **fp = (void **) __builtin_frame_address (0);
|
|
|
|
void *saved_pc = __builtin_return_address (0);
|
|
|
|
void *saved_fp = __builtin_frame_address (1);
|
|
|
|
int depth = 0;
|
|
|
|
|
|
|
|
printf ("[%d] pc == %p fp == %p\n", depth++, saved_pc, saved_fp);
|
|
|
|
fp = saved_fp;
|
|
|
|
while (fp != NULL) {
|
|
|
|
saved_fp = *fp;
|
|
|
|
fp = saved_fp;
|
|
|
|
if (*fp == NULL)
|
|
|
|
break;
|
|
|
|
saved_pc = *(fp + 2);
|
|
|
|
printf ("[%d] pc == %p fp == %p\n", depth++, saved_pc, saved_fp);
|
|
|
|
}
|
2010-05-24 10:07:54 +00:00
|
|
|
#else
|
|
|
|
#warning TODO: r_sys_bt : unimplemented
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2010-03-04 00:46:25 +00:00
|
|
|
R_API int r_sys_sleep(int secs) {
|
2009-04-03 11:11:17 +00:00
|
|
|
#if __UNIX__
|
|
|
|
return sleep(secs);
|
|
|
|
#else
|
2010-03-25 16:19:58 +00:00
|
|
|
Sleep (secs * 1000); // W32
|
2009-04-03 11:11:17 +00:00
|
|
|
return 0;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2010-03-04 00:46:25 +00:00
|
|
|
R_API int r_sys_usleep(int usecs) {
|
2009-04-03 11:11:17 +00:00
|
|
|
#if __UNIX__
|
2010-03-25 16:19:58 +00:00
|
|
|
return usleep (usecs);
|
2009-04-03 11:11:17 +00:00
|
|
|
#else
|
2010-03-25 16:19:58 +00:00
|
|
|
Sleep (usecs); // W32
|
|
|
|
return 0;
|
2009-04-03 11:11:17 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2010-04-12 00:22:52 +00:00
|
|
|
R_API int r_sys_setenv(const char *key, const char *value) {
|
2009-04-03 11:11:17 +00:00
|
|
|
#if __UNIX__
|
2013-01-02 23:47:58 +00:00
|
|
|
if (!key) return 0;
|
2010-05-23 21:04:46 +00:00
|
|
|
if (value == NULL) {
|
|
|
|
unsetenv (key);
|
|
|
|
return 0;
|
|
|
|
}
|
2010-04-12 00:22:52 +00:00
|
|
|
return setenv (key, value, 1);
|
|
|
|
#elif __WINDOWS__
|
|
|
|
SetEnvironmentVariable (key, (LPSTR)value);
|
|
|
|
return 0; // TODO. get ret
|
2009-04-03 11:11:17 +00:00
|
|
|
#else
|
2010-04-12 00:22:52 +00:00
|
|
|
#warning r_sys_setenv : unimplemented for this platform
|
|
|
|
return 0;
|
2009-04-03 11:11:17 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2011-05-15 02:20:59 +00:00
|
|
|
static char *crash_handler_cmd = NULL;
|
|
|
|
|
|
|
|
static void signal_handler(int signum) {
|
|
|
|
int len;
|
|
|
|
char *cmd;
|
|
|
|
if (!crash_handler_cmd)
|
|
|
|
return;
|
|
|
|
len = strlen (crash_handler_cmd)+32;
|
|
|
|
cmd = malloc (len);
|
|
|
|
snprintf (cmd, len, crash_handler_cmd, getpid ());
|
|
|
|
r_sys_backtrace ();
|
2012-10-19 22:31:18 +00:00
|
|
|
exit (r_sys_cmd (cmd));
|
2011-05-15 02:20:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int checkcmd(const char *c) {
|
|
|
|
char oc = 0;
|
|
|
|
for (;*c;c++) {
|
|
|
|
if (oc == '%')
|
|
|
|
if (*c!='d' && *c!='%')
|
|
|
|
return 0;
|
|
|
|
oc = *c;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API int r_sys_crash_handler(const char *cmd) {
|
|
|
|
#if __UNIX__
|
|
|
|
struct sigaction sigact;
|
|
|
|
if (!checkcmd (cmd))
|
|
|
|
return R_FALSE;
|
|
|
|
free (crash_handler_cmd);
|
|
|
|
crash_handler_cmd = strdup (cmd);
|
|
|
|
sigact.sa_handler = signal_handler;
|
|
|
|
sigemptyset (&sigact.sa_mask);
|
|
|
|
sigact.sa_flags = 0;
|
|
|
|
sigaction (SIGINT, &sigact, (struct sigaction *)NULL);
|
|
|
|
|
|
|
|
sigaddset (&sigact.sa_mask, SIGSEGV);
|
|
|
|
sigaction (SIGSEGV, &sigact, (struct sigaction *)NULL);
|
|
|
|
|
|
|
|
sigaddset (&sigact.sa_mask, SIGBUS);
|
|
|
|
sigaction (SIGBUS, &sigact, (struct sigaction *)NULL);
|
|
|
|
|
|
|
|
sigaddset (&sigact.sa_mask, SIGQUIT);
|
|
|
|
sigaction (SIGQUIT, &sigact, (struct sigaction *)NULL);
|
|
|
|
|
|
|
|
sigaddset (&sigact.sa_mask, SIGHUP);
|
|
|
|
sigaction (SIGHUP, &sigact, (struct sigaction *)NULL);
|
|
|
|
|
|
|
|
sigaddset (&sigact.sa_mask, SIGKILL);
|
|
|
|
sigaction (SIGKILL, &sigact, (struct sigaction *)NULL);
|
|
|
|
return R_TRUE;
|
|
|
|
#else
|
|
|
|
return R_FALSE;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2011-08-27 18:25:37 +00:00
|
|
|
R_API char *r_sys_getenv(const char *key) {
|
2011-09-04 00:34:54 +00:00
|
|
|
#if __WINDOWS__
|
2010-10-14 17:01:14 +00:00
|
|
|
static char envbuf[1024];
|
2010-10-11 23:18:17 +00:00
|
|
|
envbuf[0] = 0;
|
2010-04-12 00:22:52 +00:00
|
|
|
GetEnvironmentVariable (key, (LPSTR)&envbuf, sizeof (envbuf));
|
2011-08-27 18:25:37 +00:00
|
|
|
// TODO: handle return value of GEV
|
2011-09-04 00:34:54 +00:00
|
|
|
return *envbuf? strdup (envbuf): NULL;
|
2010-04-13 19:19:54 +00:00
|
|
|
#else
|
2011-08-27 18:25:37 +00:00
|
|
|
char *b = getenv (key);
|
2011-09-04 00:34:54 +00:00
|
|
|
return b? strdup (b): NULL;
|
2010-04-13 19:19:54 +00:00
|
|
|
#endif
|
2011-09-04 00:34:54 +00:00
|
|
|
}
|
2009-04-13 22:47:02 +00:00
|
|
|
|
2011-09-04 00:34:54 +00:00
|
|
|
R_API char *r_sys_getdir(void) {
|
|
|
|
#if __WINDOWS__
|
|
|
|
char *cwd = _getcwd (NULL, 0);
|
2010-02-19 18:59:22 +00:00
|
|
|
#else
|
2011-09-04 00:34:54 +00:00
|
|
|
char *cwd = getcwd (NULL, 0);
|
2010-02-19 18:59:22 +00:00
|
|
|
#endif
|
2011-09-04 00:34:54 +00:00
|
|
|
return cwd? strdup (cwd): NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API int r_sys_chdir(const char *s) {
|
2012-10-19 22:31:18 +00:00
|
|
|
return r_sandbox_chdir (s)==0;
|
2010-02-19 18:59:22 +00:00
|
|
|
}
|
|
|
|
|
2009-04-13 22:47:02 +00:00
|
|
|
#if __UNIX__
|
2012-02-05 01:39:04 +00:00
|
|
|
R_API int r_sys_cmd_str_full(const char *cmd, const char *input, char **output, int *len, char **sterr) {
|
|
|
|
char buffer[1024], *outputptr = NULL;
|
2009-12-30 10:03:18 +00:00
|
|
|
char *inputptr = (char *)input;
|
2011-06-30 15:41:48 +00:00
|
|
|
int pid, bytes = 0, status;
|
2010-05-03 19:24:58 +00:00
|
|
|
int sh_in[2], sh_out[2], sh_err[2];
|
|
|
|
|
2010-04-20 17:32:04 +00:00
|
|
|
if (len) *len = 0;
|
2010-10-27 14:31:51 +00:00
|
|
|
if (pipe (sh_in))
|
2012-02-05 01:39:04 +00:00
|
|
|
return R_FALSE;
|
|
|
|
if (output) {
|
|
|
|
if (pipe (sh_out)) {
|
|
|
|
close (sh_in[0]);
|
|
|
|
close (sh_in[1]);
|
|
|
|
return R_FALSE;
|
|
|
|
}
|
2010-10-27 14:31:51 +00:00
|
|
|
}
|
|
|
|
if (pipe (sh_err)) {
|
|
|
|
close (sh_in[0]);
|
|
|
|
close (sh_in[1]);
|
|
|
|
close (sh_out[0]);
|
|
|
|
close (sh_out[1]);
|
2012-02-05 01:39:04 +00:00
|
|
|
return R_FALSE;
|
2010-10-27 14:31:51 +00:00
|
|
|
}
|
2009-12-30 10:03:18 +00:00
|
|
|
|
2010-06-30 00:30:07 +00:00
|
|
|
switch ((pid=fork ())) {
|
2010-05-03 19:24:58 +00:00
|
|
|
case -1:
|
2012-02-05 01:39:04 +00:00
|
|
|
return R_FALSE;
|
2010-05-03 19:24:58 +00:00
|
|
|
case 0:
|
2010-04-13 19:19:54 +00:00
|
|
|
dup2 (sh_in[0], 0); close (sh_in[0]); close (sh_in[1]);
|
2012-02-05 01:39:04 +00:00
|
|
|
if (output) { dup2 (sh_out[1], 1); close (sh_out[0]); close (sh_out[1]); }
|
2010-05-03 19:24:58 +00:00
|
|
|
if (sterr) dup2 (sh_err[1], 2); else close (2);
|
2010-04-13 19:19:54 +00:00
|
|
|
close (sh_err[0]); close (sh_err[1]);
|
2012-10-19 22:31:18 +00:00
|
|
|
exit (r_sandbox_system (cmd, 0));
|
2010-05-03 19:24:58 +00:00
|
|
|
default:
|
2012-02-05 01:39:04 +00:00
|
|
|
outputptr = strdup ("");
|
|
|
|
if (!outputptr)
|
|
|
|
return R_FALSE;
|
2010-05-06 18:24:16 +00:00
|
|
|
if (sterr) {
|
2011-11-11 16:14:09 +00:00
|
|
|
*sterr = strdup ("");
|
2010-05-06 18:24:16 +00:00
|
|
|
if (!*sterr) {
|
2012-02-05 01:39:04 +00:00
|
|
|
free (outputptr);
|
|
|
|
return R_FALSE;
|
2010-05-06 18:24:16 +00:00
|
|
|
}
|
2010-05-03 19:24:58 +00:00
|
|
|
}
|
2012-02-05 01:39:04 +00:00
|
|
|
if (output) close (sh_out[1]);
|
2010-04-13 19:19:54 +00:00
|
|
|
close (sh_err[1]);
|
|
|
|
close (sh_in[0]);
|
2010-02-15 21:59:26 +00:00
|
|
|
if (!inputptr || !*inputptr)
|
2010-04-13 19:19:54 +00:00
|
|
|
close (sh_in[1]);
|
2010-02-15 21:59:26 +00:00
|
|
|
|
2010-04-13 19:19:54 +00:00
|
|
|
for (;;) {
|
2009-12-30 10:03:18 +00:00
|
|
|
fd_set rfds, wfds;
|
|
|
|
int nfd;
|
|
|
|
|
2010-04-13 19:19:54 +00:00
|
|
|
FD_ZERO (&rfds);
|
|
|
|
FD_ZERO (&wfds);
|
2012-02-05 01:39:04 +00:00
|
|
|
if (output)
|
|
|
|
FD_SET (sh_out[0], &rfds);
|
2010-02-15 21:59:26 +00:00
|
|
|
if (sterr)
|
2010-04-13 19:19:54 +00:00
|
|
|
FD_SET (sh_err[0], &rfds);
|
2009-12-30 10:03:18 +00:00
|
|
|
if (inputptr && *inputptr)
|
2010-04-13 19:19:54 +00:00
|
|
|
FD_SET (sh_in[1], &wfds);
|
2010-05-03 19:24:58 +00:00
|
|
|
memset (buffer, 0, sizeof (buffer));
|
2010-04-13 19:19:54 +00:00
|
|
|
nfd = select (sh_err[0] + 1, &rfds, &wfds, NULL, NULL);
|
|
|
|
if (nfd < 0)
|
2010-02-15 21:59:26 +00:00
|
|
|
break;
|
2012-02-05 01:39:04 +00:00
|
|
|
if (output && FD_ISSET (sh_out[0], &rfds)) {
|
2010-04-13 19:19:54 +00:00
|
|
|
if ((bytes = read (sh_out[0], buffer, sizeof (buffer)-1)) == 0) break;
|
2010-04-20 17:32:04 +00:00
|
|
|
if (len) *len += bytes;
|
2012-02-05 01:39:04 +00:00
|
|
|
outputptr = r_str_concat (outputptr, buffer);
|
2010-04-13 19:19:54 +00:00
|
|
|
} else if (FD_ISSET (sh_err[0], &rfds) && sterr) {
|
|
|
|
if (read (sh_err[0], buffer, sizeof (buffer)-1) == 0) break;
|
|
|
|
*sterr = r_str_concat (*sterr, buffer);
|
|
|
|
} else if (FD_ISSET (sh_in[1], &wfds) && inputptr && *inputptr) {
|
2010-05-03 19:24:58 +00:00
|
|
|
bytes = write (sh_in[1], inputptr, strlen (inputptr));
|
2010-04-13 19:19:54 +00:00
|
|
|
inputptr += bytes;
|
2012-02-05 01:39:04 +00:00
|
|
|
if (!*inputptr) {
|
|
|
|
close (sh_in[1]);
|
|
|
|
/* If neither stdout nor stderr should be captured,
|
|
|
|
* abort now - nothing more to do for select(). */
|
|
|
|
if (!output && !sterr) break;
|
|
|
|
}
|
2009-12-30 10:03:18 +00:00
|
|
|
}
|
|
|
|
}
|
2012-02-05 01:39:04 +00:00
|
|
|
if (output)
|
|
|
|
close (sh_out[0]);
|
2010-05-03 19:24:58 +00:00
|
|
|
close (sh_err[0]);
|
|
|
|
close (sh_in[1]);
|
2011-09-04 00:34:54 +00:00
|
|
|
waitpid (pid, &status, 0);
|
2011-06-30 15:41:48 +00:00
|
|
|
if (status != 0) {
|
2011-10-06 23:16:45 +00:00
|
|
|
eprintf ("%s: command '%s' returned !0\n", __func__, cmd);
|
2012-02-05 01:39:04 +00:00
|
|
|
return R_FALSE;
|
2011-06-30 15:41:48 +00:00
|
|
|
}
|
2010-02-15 21:59:26 +00:00
|
|
|
|
2011-11-11 16:14:09 +00:00
|
|
|
if (output) {
|
2012-02-05 01:39:04 +00:00
|
|
|
*output = outputptr;
|
|
|
|
} else if (outputptr) {
|
|
|
|
free(outputptr);
|
2011-11-11 16:14:09 +00:00
|
|
|
}
|
2012-02-05 01:39:04 +00:00
|
|
|
return R_TRUE;
|
2009-12-30 10:03:18 +00:00
|
|
|
}
|
2013-01-25 01:03:57 +00:00
|
|
|
free(outputptr);
|
2012-02-05 01:39:04 +00:00
|
|
|
return R_FALSE;
|
2010-04-12 09:46:15 +00:00
|
|
|
}
|
2010-04-12 00:22:52 +00:00
|
|
|
#elif __WINDOWS__
|
2012-09-06 01:12:54 +00:00
|
|
|
// TODO: fully implement the rest
|
2012-02-05 01:39:04 +00:00
|
|
|
R_API int r_sys_cmd_str_full(const char *cmd, const char *input, char **output, int *len, char **sterr) {
|
2012-09-06 01:12:54 +00:00
|
|
|
char *result = r_sys_cmd_str_w32 (cmd);
|
2010-04-12 00:22:52 +00:00
|
|
|
if (len) *len = 0;
|
2012-09-06 01:12:54 +00:00
|
|
|
if (output) *output = result;
|
|
|
|
if (result) return R_TRUE;
|
2012-02-05 01:39:04 +00:00
|
|
|
return R_FALSE;
|
2010-04-12 00:22:52 +00:00
|
|
|
}
|
2009-04-13 22:47:02 +00:00
|
|
|
#else
|
2012-02-05 01:39:04 +00:00
|
|
|
R_API int r_sys_cmd_str_full(const char *cmd, const char *input, char **output, int *len, char **sterr) {
|
2010-03-25 16:19:58 +00:00
|
|
|
eprintf ("r_sys_cmd_str: not yet implemented for this platform\n");
|
2012-02-05 01:39:04 +00:00
|
|
|
return R_FALSE;
|
2009-04-13 22:47:02 +00:00
|
|
|
}
|
2010-04-12 00:22:52 +00:00
|
|
|
#endif
|
2010-02-02 10:09:52 +00:00
|
|
|
|
2010-09-09 22:08:53 +00:00
|
|
|
R_API int r_sys_cmdf (const char *fmt, ...) {
|
2010-10-11 23:18:17 +00:00
|
|
|
int ret;
|
|
|
|
char cmd[4096];
|
2010-09-09 22:08:53 +00:00
|
|
|
va_list ap;
|
|
|
|
va_start(ap, fmt);
|
|
|
|
vsnprintf (cmd, sizeof (cmd), fmt, ap);
|
|
|
|
ret = r_sys_cmd (cmd);
|
2010-10-11 23:18:17 +00:00
|
|
|
va_end (ap);
|
2010-09-09 22:08:53 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2013-05-10 23:58:05 +00:00
|
|
|
R_API int r_sys_cmdbg (const char *str) {
|
|
|
|
#if __UNIX__
|
|
|
|
int ret, pid = fork ();
|
|
|
|
if (pid == -1) return -1;
|
|
|
|
if (pid) return pid;
|
|
|
|
ret = r_sandbox_system (str, 0);
|
|
|
|
eprintf ("{exit: %d, pid: %d, cmd: \"%s\"}", ret, pid, str);
|
|
|
|
exit (0);
|
|
|
|
#else
|
|
|
|
#warning r_sys_cmdbg is not implemented for this platform
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2010-03-01 09:49:04 +00:00
|
|
|
R_API int r_sys_cmd (const char *str) {
|
|
|
|
#if __FreeBSD__
|
|
|
|
/* freebsd system() is broken */
|
2012-10-19 22:31:18 +00:00
|
|
|
int st, pid, fds[2];
|
|
|
|
if (pipe (fds))
|
|
|
|
return -1;
|
2010-03-04 00:46:25 +00:00
|
|
|
pid = vfork ();
|
2012-10-19 22:31:18 +00:00
|
|
|
if (pid == -1)
|
|
|
|
return -1;
|
2010-03-01 09:49:04 +00:00
|
|
|
if (pid == 0) {
|
2010-03-04 00:46:25 +00:00
|
|
|
dup2 (1, fds[1]);
|
2012-10-19 22:31:18 +00:00
|
|
|
// char *argv[] = { "/bin/sh", "-c", str, NULL};
|
|
|
|
// execv (argv[0], argv);
|
|
|
|
r_sandbox_system (str, 0);
|
2010-03-04 00:46:25 +00:00
|
|
|
_exit (127); /* error */
|
2010-03-01 09:49:04 +00:00
|
|
|
} else {
|
2010-03-04 00:46:25 +00:00
|
|
|
dup2 (1, fds[0]);
|
|
|
|
waitpid (pid, &st, 0);
|
2010-03-01 09:49:04 +00:00
|
|
|
}
|
2010-03-04 00:46:25 +00:00
|
|
|
return WEXITSTATUS (st);
|
2010-03-01 09:49:04 +00:00
|
|
|
#else
|
2012-10-19 22:31:18 +00:00
|
|
|
return r_sandbox_system (str, 1);
|
2010-03-01 09:49:04 +00:00
|
|
|
#endif
|
2010-02-02 10:09:52 +00:00
|
|
|
}
|
2010-02-15 21:59:26 +00:00
|
|
|
|
|
|
|
R_API char *r_sys_cmd_str(const char *cmd, const char *input, int *len) {
|
2012-02-05 01:39:04 +00:00
|
|
|
char *output;
|
|
|
|
if (r_sys_cmd_str_full (cmd, input, &output, len, NULL))
|
|
|
|
return output;
|
|
|
|
return NULL;
|
2010-02-15 21:59:26 +00:00
|
|
|
}
|
2010-03-30 22:03:59 +00:00
|
|
|
|
2010-10-01 17:05:07 +00:00
|
|
|
R_API int r_sys_rmkdir(const char *dir) {
|
2010-10-11 23:18:17 +00:00
|
|
|
int ret = R_TRUE;
|
2010-10-04 11:57:48 +00:00
|
|
|
char *path = strdup (dir), *ptr = path;
|
2010-10-17 21:03:54 +00:00
|
|
|
// XXX: Wrong for w32 (/).. and no errno ?
|
2012-09-07 08:07:41 +00:00
|
|
|
if (*ptr=='/') ptr++;
|
2010-10-04 11:57:48 +00:00
|
|
|
while ((ptr = strchr (ptr, '/'))) {
|
2010-10-01 17:05:07 +00:00
|
|
|
*ptr = 0;
|
2010-10-23 12:27:13 +00:00
|
|
|
if (!r_sys_mkdir (path) && r_sys_mkdir_failed ()) {
|
|
|
|
eprintf ("r_sys_rmkdir: fail %s\n", dir);
|
|
|
|
free (path);
|
|
|
|
return R_FALSE;
|
2010-10-11 23:18:17 +00:00
|
|
|
}
|
2010-10-04 11:57:48 +00:00
|
|
|
*ptr = '/';
|
2010-10-17 21:03:54 +00:00
|
|
|
ptr++;
|
2010-10-01 17:05:07 +00:00
|
|
|
}
|
2010-10-23 12:27:13 +00:00
|
|
|
if (!r_sys_mkdir (path) && r_sys_mkdir_failed ())
|
|
|
|
ret = R_FALSE;
|
2010-10-01 17:05:07 +00:00
|
|
|
free (path);
|
2010-10-11 23:18:17 +00:00
|
|
|
return ret;
|
2010-03-30 22:03:59 +00:00
|
|
|
}
|
2010-04-12 00:22:52 +00:00
|
|
|
|
2010-04-14 21:56:27 +00:00
|
|
|
R_API void r_sys_perror(const char *fun) {
|
2010-04-12 00:22:52 +00:00
|
|
|
#if __UNIX__
|
|
|
|
perror (fun);
|
|
|
|
#elif __WINDOWS__
|
|
|
|
char *lpMsgBuf;
|
|
|
|
LPVOID lpDisplayBuf;
|
2012-10-19 22:31:18 +00:00
|
|
|
DWORD dw = GetLastError ();
|
2010-04-12 00:22:52 +00:00
|
|
|
|
|
|
|
FormatMessage ( FORMAT_MESSAGE_ALLOCATE_BUFFER |
|
|
|
|
FORMAT_MESSAGE_FROM_SYSTEM |
|
|
|
|
FORMAT_MESSAGE_IGNORE_INSERTS,
|
|
|
|
NULL,
|
|
|
|
dw,
|
2012-10-19 22:31:18 +00:00
|
|
|
MAKELANGID (LANG_NEUTRAL, SUBLANG_DEFAULT),
|
2010-04-12 00:22:52 +00:00
|
|
|
(LPTSTR) &lpMsgBuf,
|
|
|
|
0, NULL );
|
|
|
|
|
|
|
|
lpDisplayBuf = (LPVOID)LocalAlloc (LMEM_ZEROINIT,
|
2012-10-19 22:31:18 +00:00
|
|
|
(lstrlen ((LPCTSTR)lpMsgBuf)+
|
|
|
|
lstrlen ((LPCTSTR)fun)+40)*sizeof (TCHAR));
|
2010-04-12 00:22:52 +00:00
|
|
|
eprintf ("%s: %s\n", fun, lpMsgBuf);
|
|
|
|
|
|
|
|
LocalFree (lpMsgBuf);
|
|
|
|
LocalFree (lpDisplayBuf);
|
|
|
|
#endif
|
|
|
|
}
|
2011-05-05 22:59:10 +00:00
|
|
|
|
2012-11-30 00:06:30 +00:00
|
|
|
// TODO: use array :P
|
2011-05-05 22:59:10 +00:00
|
|
|
R_API int r_sys_arch_id(const char *arch) {
|
|
|
|
if (!strcmp (arch, "x86")) return R_SYS_ARCH_X86;
|
|
|
|
if (!strcmp (arch, "arm")) return R_SYS_ARCH_ARM;
|
|
|
|
if (!strcmp (arch, "ppc")) return R_SYS_ARCH_PPC;
|
|
|
|
if (!strcmp (arch, "m68k")) return R_SYS_ARCH_M68K;
|
|
|
|
if (!strcmp (arch, "java")) return R_SYS_ARCH_JAVA;
|
|
|
|
if (!strcmp (arch, "mips")) return R_SYS_ARCH_MIPS;
|
|
|
|
if (!strcmp (arch, "sparc")) return R_SYS_ARCH_SPARC;
|
|
|
|
if (!strcmp (arch, "csr")) return R_SYS_ARCH_CSR;
|
2013-04-23 10:53:21 +00:00
|
|
|
if (!strcmp (arch, "c55+")) return R_SYS_ARCH_C55PLUS;
|
2011-05-05 22:59:10 +00:00
|
|
|
if (!strcmp (arch, "msil")) return R_SYS_ARCH_MSIL;
|
|
|
|
if (!strcmp (arch, "objd")) return R_SYS_ARCH_OBJD;
|
|
|
|
if (!strcmp (arch, "bf")) return R_SYS_ARCH_BF;
|
|
|
|
if (!strcmp (arch, "sh")) return R_SYS_ARCH_SH;
|
|
|
|
if (!strcmp (arch, "avr")) return R_SYS_ARCH_AVR;
|
2012-11-30 00:06:30 +00:00
|
|
|
if (!strcmp (arch, "dalvik")) return R_SYS_ARCH_DALVIK;
|
|
|
|
if (!strcmp (arch, "z80")) return R_SYS_ARCH_Z80;
|
|
|
|
if (!strcmp (arch, "arc")) return R_SYS_ARCH_ARC;
|
|
|
|
if (!strcmp (arch, "i8080")) return R_SYS_ARCH_I8080;
|
|
|
|
if (!strcmp (arch, "rar")) return R_SYS_ARCH_RAR;
|
2011-05-05 22:59:10 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
R_API const char *r_sys_arch_str(int arch) {
|
|
|
|
if (arch & R_SYS_ARCH_X86) return "x86";
|
|
|
|
if (arch & R_SYS_ARCH_ARM) return "arm";
|
|
|
|
if (arch & R_SYS_ARCH_PPC) return "ppc";
|
|
|
|
if (arch & R_SYS_ARCH_M68K) return "m68k";
|
|
|
|
if (arch & R_SYS_ARCH_JAVA) return "java";
|
|
|
|
if (arch & R_SYS_ARCH_MIPS) return "mips";
|
|
|
|
if (arch & R_SYS_ARCH_SPARC) return "sparc";
|
|
|
|
if (arch & R_SYS_ARCH_CSR) return "csr";
|
2013-04-23 10:53:21 +00:00
|
|
|
if (arch & R_SYS_ARCH_C55PLUS) return "c55+";
|
2011-05-05 22:59:10 +00:00
|
|
|
if (arch & R_SYS_ARCH_MSIL) return "msil";
|
|
|
|
if (arch & R_SYS_ARCH_OBJD) return "objd";
|
|
|
|
if (arch & R_SYS_ARCH_BF) return "bf";
|
|
|
|
if (arch & R_SYS_ARCH_SH) return "sh";
|
|
|
|
if (arch & R_SYS_ARCH_AVR) return "avr";
|
2012-11-30 00:06:30 +00:00
|
|
|
if (arch & R_SYS_ARCH_DALVIK) return "dalvik";
|
|
|
|
if (arch & R_SYS_ARCH_Z80) return "z80";
|
|
|
|
if (arch & R_SYS_ARCH_ARC) return "arc";
|
|
|
|
if (arch & R_SYS_ARCH_I8080) return "i8080";
|
|
|
|
if (arch & R_SYS_ARCH_RAR) return "rar";
|
2011-05-05 22:59:10 +00:00
|
|
|
return "none";
|
|
|
|
}
|
2011-11-15 23:44:18 +00:00
|
|
|
|
|
|
|
R_API int r_sys_run(const ut8 *buf, int len) {
|
2012-10-19 22:31:18 +00:00
|
|
|
const int sz = 4096;
|
2013-04-21 22:01:41 +00:00
|
|
|
int pdelta, ret, (*cb)();
|
|
|
|
// TODO: define R_SYS_ALIGN_FORWARD in r_util.h
|
2012-10-19 22:31:18 +00:00
|
|
|
ut8 *ptr, *p = malloc ((sz+len)<<1);
|
2013-04-21 22:01:41 +00:00
|
|
|
ptr = p;
|
|
|
|
pdelta = ((size_t)(p)) & (4096-1);
|
|
|
|
if (pdelta)
|
|
|
|
ptr += (4096-pdelta);
|
2012-09-26 08:01:43 +00:00
|
|
|
if (!ptr) {
|
|
|
|
free (p);
|
|
|
|
return R_FALSE;
|
|
|
|
}
|
2012-10-19 22:31:18 +00:00
|
|
|
memcpy (ptr, buf, sz);
|
2013-04-21 23:34:25 +00:00
|
|
|
r_mem_protect (ptr, sz, "rx");
|
2013-04-21 22:01:41 +00:00
|
|
|
//r_mem_protect (ptr, sz, "rwx"); // try, ignore if fail
|
2011-11-15 23:44:18 +00:00
|
|
|
cb = (void*)ptr;
|
|
|
|
ret = cb ();
|
|
|
|
free (p);
|
|
|
|
return ret;
|
|
|
|
}
|
2013-04-11 23:15:00 +00:00
|
|
|
|
|
|
|
R_API int r_is_heap (void *p) {
|
2013-04-12 00:39:46 +00:00
|
|
|
void *q = malloc (8);
|
2013-04-11 23:15:00 +00:00
|
|
|
ut64 mask = UT64_MAX;
|
|
|
|
ut64 addr = (ut64)(size_t)q;
|
|
|
|
addr>>=16;
|
|
|
|
addr<<=16;
|
|
|
|
mask>>=16;
|
|
|
|
mask<<=16;
|
2013-04-12 00:39:46 +00:00
|
|
|
free (q);
|
2013-04-11 23:15:00 +00:00
|
|
|
return (((ut64)(size_t)p) == mask);
|
|
|
|
}
|
2013-08-18 22:36:17 +00:00
|
|
|
|
|
|
|
R_API char *r_sys_pid_to_path(int pid) {
|
2013-08-18 22:41:44 +00:00
|
|
|
#if __WINDOWS__
|
|
|
|
// TODO: implement r_sys_pid_to_path on W32
|
|
|
|
return NULL;
|
|
|
|
#elif __APPLE__
|
2013-08-18 22:36:17 +00:00
|
|
|
char pathbuf[PROC_PIDPATHINFO_MAXSIZE];
|
2013-08-21 22:15:05 +00:00
|
|
|
int ret = proc_pidpath (pid, pathbuf, sizeof(pathbuf));
|
2013-08-18 22:36:17 +00:00
|
|
|
if (ret <= 0)
|
|
|
|
return NULL;
|
2013-08-21 22:15:05 +00:00
|
|
|
return strdup (pathbuf);
|
2013-08-18 22:36:17 +00:00
|
|
|
#else
|
|
|
|
char buf[128], pathbuf[1024];
|
|
|
|
snprintf (buf, "/proc/%d/exe", pid);
|
|
|
|
if (readlink (buf, pathbuf, sizeof (pathbuf))<1)
|
|
|
|
return NULL;
|
|
|
|
return strdup (pathbuf);
|
2013-08-21 22:15:05 +00:00
|
|
|
#endif
|
2013-08-18 22:36:17 +00:00
|
|
|
}
|