2010-03-12 17:46:11 +00:00
|
|
|
/* radare - LGPL - Copyright 2009-2010 pancake<nopcode.org> */
|
|
|
|
|
2009-04-01 01:40:04 +00:00
|
|
|
#include <r_types.h>
|
2009-12-30 10:03:18 +00:00
|
|
|
#include <r_util.h>
|
2010-01-15 00:32:28 +00:00
|
|
|
#if __UNIX__
|
|
|
|
#include <sys/wait.h>
|
|
|
|
#include <signal.h>
|
2010-02-19 18:59:22 +00:00
|
|
|
#elif __WINDOWS__
|
|
|
|
#include <io.h>
|
2010-01-15 00:32:28 +00:00
|
|
|
#endif
|
|
|
|
#include <sys/types.h>
|
2009-04-01 01:30:36 +00:00
|
|
|
/* TODO: import stuff fron bininfo/p/bininfo_addr2line */
|
|
|
|
|
2010-03-12 18:11:43 +00:00
|
|
|
/* TODO: check endianness issues here */
|
2010-03-12 17:46:11 +00:00
|
|
|
R_API ut64 r_sys_now() {
|
|
|
|
ut64 ret;
|
|
|
|
struct timeval now;
|
|
|
|
gettimeofday (&now, NULL);
|
|
|
|
ret = now.tv_sec;
|
|
|
|
ret <<= 32;
|
|
|
|
ret |= now.tv_usec;
|
|
|
|
//(sizeof (now.tv_sec) == 4
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-03-01 09:49:04 +00:00
|
|
|
R_API char *r_sys_cmd_strf(const char *cmd, ...) {
|
2010-02-15 21:59:26 +00:00
|
|
|
// FIXME Implement r_sys_cmd_strf
|
2009-04-01 01:30:36 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
2009-04-03 11:11:17 +00:00
|
|
|
|
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__
|
2010-04-12 00:22:52 +00:00
|
|
|
if (value == NULL)
|
|
|
|
return unsetenv (key);
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2010-03-25 16:19:58 +00:00
|
|
|
static char envbuf[1024];
|
2010-03-01 09:49:04 +00:00
|
|
|
R_API const char *r_sys_getenv(const char *key) {
|
2009-04-03 11:11:17 +00:00
|
|
|
#if __UNIX__
|
2010-03-25 16:19:58 +00:00
|
|
|
return getenv (key);
|
2009-04-03 11:11:17 +00:00
|
|
|
#else
|
2010-04-12 00:22:52 +00:00
|
|
|
GetEnvironmentVariable (key, (LPSTR)&envbuf, sizeof (envbuf));
|
|
|
|
return envbuf;
|
2009-04-03 11:11:17 +00:00
|
|
|
#endif
|
|
|
|
}
|
2009-04-13 22:47:02 +00:00
|
|
|
|
2010-03-01 09:49:04 +00:00
|
|
|
R_API char *r_sys_getcwd() {
|
2010-02-19 18:59:22 +00:00
|
|
|
#if __UNIX__
|
2010-03-25 16:19:58 +00:00
|
|
|
return getcwd (NULL, 0);
|
2010-02-19 18:59:22 +00:00
|
|
|
#elif __WINDOWS__
|
2010-03-25 16:19:58 +00:00
|
|
|
return _getcwd (NULL, 0);
|
2010-02-19 18:59:22 +00:00
|
|
|
#else
|
|
|
|
#warning TODO: r_sys_getcwd
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2009-04-13 22:47:02 +00:00
|
|
|
#if __UNIX__
|
2010-04-12 00:22:52 +00:00
|
|
|
R_API char *r_sys_cmd_str_full(const char *cmd, const char *input, int *len, char **sterr) {
|
2009-12-30 10:03:18 +00:00
|
|
|
char *inputptr = (char *)input;
|
|
|
|
int bytes = 0;
|
|
|
|
int sh_in[2];
|
|
|
|
int sh_out[2];
|
|
|
|
int sh_err[2];
|
|
|
|
pipe(sh_in);
|
|
|
|
pipe(sh_out);
|
|
|
|
pipe(sh_err);
|
2009-04-13 22:47:02 +00:00
|
|
|
*len = 0;
|
2009-12-30 10:03:18 +00:00
|
|
|
|
|
|
|
int pid = fork();
|
|
|
|
if (!pid) {
|
2010-02-15 21:59:26 +00:00
|
|
|
dup2(sh_in[0], 0); close(sh_in[0]); close(sh_in[1]);
|
|
|
|
dup2(sh_out[1], 1); close(sh_out[0]); close(sh_out[1]);
|
2009-12-30 10:03:18 +00:00
|
|
|
if (sterr) dup2(sh_err[1], 2);
|
|
|
|
else close(2);
|
2010-02-15 21:59:26 +00:00
|
|
|
close(sh_err[0]); close(sh_err[1]);
|
2009-12-30 10:03:18 +00:00
|
|
|
execl("/bin/sh", "sh", "-c", cmd, NULL);
|
|
|
|
} else {
|
|
|
|
char buffer[1024];
|
|
|
|
char *output = calloc(1, 1024);
|
|
|
|
if (sterr)
|
|
|
|
*sterr = calloc(1, 1024);
|
|
|
|
|
2010-02-15 21:59:26 +00:00
|
|
|
close(sh_out[1]);
|
|
|
|
close(sh_err[1]);
|
|
|
|
close(sh_in[0]);
|
|
|
|
if (!inputptr || !*inputptr)
|
|
|
|
close(sh_in[1]);
|
|
|
|
|
2009-12-30 10:03:18 +00:00
|
|
|
while (1) {
|
|
|
|
fd_set rfds, wfds;
|
|
|
|
int nfd;
|
|
|
|
|
|
|
|
FD_ZERO(&rfds);
|
|
|
|
FD_ZERO(&wfds);
|
|
|
|
FD_SET(sh_out[0], &rfds);
|
2010-02-15 21:59:26 +00:00
|
|
|
if (sterr)
|
|
|
|
FD_SET(sh_err[0], &rfds);
|
2009-12-30 10:03:18 +00:00
|
|
|
if (inputptr && *inputptr)
|
|
|
|
FD_SET(sh_in[1], &wfds);
|
|
|
|
|
|
|
|
memset(buffer, 0, sizeof(buffer));
|
2010-02-15 21:59:26 +00:00
|
|
|
nfd = select(sh_err[0] + 1, &rfds, &wfds, NULL, NULL);
|
|
|
|
if (nfd < 0) {
|
|
|
|
break;
|
2009-12-30 10:03:18 +00:00
|
|
|
} else {
|
|
|
|
if (FD_ISSET(sh_out[0], &rfds)) {
|
2010-02-15 21:59:26 +00:00
|
|
|
if ((bytes = read(sh_out[0], buffer, sizeof(buffer)-1)) == 0) break;
|
|
|
|
*len += bytes;
|
2009-12-30 10:03:18 +00:00
|
|
|
output = r_str_concat(output, buffer);
|
|
|
|
} else if (FD_ISSET(sh_err[0], &rfds) && sterr) {
|
2010-02-15 21:59:26 +00:00
|
|
|
if (read(sh_err[0], buffer, sizeof(buffer)-1) == 0) break;
|
2009-12-30 10:03:18 +00:00
|
|
|
*sterr = r_str_concat(*sterr, buffer);
|
|
|
|
} else if (FD_ISSET(sh_in[1], &wfds) && inputptr && *inputptr) {
|
|
|
|
bytes = write(sh_in[1], inputptr, strlen(inputptr));
|
|
|
|
inputptr += bytes;
|
2010-02-15 21:59:26 +00:00
|
|
|
if (!*inputptr) close(sh_in[1]);
|
2009-12-30 10:03:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-02-15 21:59:26 +00:00
|
|
|
close(sh_out[0]);
|
|
|
|
close(sh_err[0]);
|
|
|
|
close(sh_in[1]);
|
|
|
|
|
2009-12-30 10:03:18 +00:00
|
|
|
if (strlen(output))
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
return NULL;
|
2010-04-12 00:22:52 +00:00
|
|
|
#elif __WINDOWS__
|
|
|
|
R_API char *r_sys_cmd_str_full(const char *cmd, const char *input, int *len, char **sterr) {
|
|
|
|
// TODO: fully implement the rest
|
|
|
|
if (len) *len = 0;
|
|
|
|
return r_sys_cmd_str_w32 (cmd);
|
|
|
|
}
|
2009-04-13 22:47:02 +00:00
|
|
|
#else
|
2010-04-12 00:22:52 +00:00
|
|
|
R_API char *r_sys_cmd_str_full(const char *cmd, const char *input, int *len, char **sterr) {
|
2010-03-25 16:19:58 +00:00
|
|
|
eprintf ("r_sys_cmd_str: not yet implemented for this platform\n");
|
2009-04-13 22:47:02 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
2010-04-12 00:22:52 +00:00
|
|
|
#endif
|
2010-02-02 10:09:52 +00:00
|
|
|
|
2010-03-01 09:49:04 +00:00
|
|
|
R_API int r_sys_cmd (const char *str) {
|
2010-02-02 10:09:52 +00:00
|
|
|
/* TODO: implement for other systems */
|
2010-03-01 09:49:04 +00:00
|
|
|
#if __FreeBSD__
|
|
|
|
/* freebsd system() is broken */
|
|
|
|
int fds[2];
|
|
|
|
int st,pid;
|
|
|
|
char *argv[] ={ "/bin/sh", "-c", input, NULL};
|
2010-03-04 00:46:25 +00:00
|
|
|
pipe (fds);
|
2010-03-01 09:49:04 +00:00
|
|
|
/* not working ?? */
|
|
|
|
//pid = rfork(RFPROC|RFCFDG);
|
2010-03-04 00:46:25 +00:00
|
|
|
pid = vfork ();
|
2010-03-01 09:49:04 +00:00
|
|
|
if (pid == 0) {
|
2010-03-04 00:46:25 +00:00
|
|
|
dup2 (1, fds[1]);
|
|
|
|
execv (argv[0], argv);
|
|
|
|
_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
|
2010-02-02 10:09:52 +00:00
|
|
|
return system (str);
|
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) {
|
|
|
|
return r_sys_cmd_str_full (cmd, input, len, NULL);
|
|
|
|
}
|
2010-03-30 22:03:59 +00:00
|
|
|
|
|
|
|
R_API int r_sys_mkdir(const char *dir) {
|
|
|
|
int ret;
|
|
|
|
#if __WINDOWS__
|
|
|
|
ret = mkdir (dir);
|
|
|
|
#else
|
|
|
|
ret = mkdir (dir, 0755);
|
|
|
|
#endif
|
|
|
|
return (ret != -1);
|
|
|
|
}
|
2010-04-12 00:22:52 +00:00
|
|
|
|
|
|
|
void r_sys_perror(const char *fun) {
|
|
|
|
#if __UNIX__
|
|
|
|
perror (fun);
|
|
|
|
#elif __WINDOWS__
|
|
|
|
char *lpMsgBuf;
|
|
|
|
LPVOID lpDisplayBuf;
|
|
|
|
DWORD dw = GetLastError();
|
|
|
|
|
|
|
|
FormatMessage ( FORMAT_MESSAGE_ALLOCATE_BUFFER |
|
|
|
|
FORMAT_MESSAGE_FROM_SYSTEM |
|
|
|
|
FORMAT_MESSAGE_IGNORE_INSERTS,
|
|
|
|
NULL,
|
|
|
|
dw,
|
|
|
|
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
|
|
|
|
(LPTSTR) &lpMsgBuf,
|
|
|
|
0, NULL );
|
|
|
|
|
|
|
|
lpDisplayBuf = (LPVOID)LocalAlloc (LMEM_ZEROINIT,
|
|
|
|
(lstrlen((LPCTSTR)lpMsgBuf)+
|
|
|
|
lstrlen((LPCTSTR)fun)+40)*sizeof (TCHAR));
|
|
|
|
eprintf ("%s: %s\n", fun, lpMsgBuf);
|
|
|
|
|
|
|
|
LocalFree (lpMsgBuf);
|
|
|
|
LocalFree (lpDisplayBuf);
|
|
|
|
#endif
|
|
|
|
}
|