/* * R : A Computer Language for Statistical Data Analysis * Copyright (C) 1995, 1996 Robert Gentleman and Ross Ihaka * Copyright (C) 1997--2017 The R Core Team * * This program is free software; you can redistribute it and/or modify * it 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; if not, a copy is available at * https://www.R-project.org/Licenses/ */ /* char here is mainly handled as a whole string. Does handle file names. Chopping final \n is OK in UTF-8. */ /* See system.txt for a description of functions */ #ifdef HAVE_CONFIG_H # include #endif #define R_USE_SIGNALS 1 #include #include #include #include /* for fround */ #include "Runix.h" #ifdef HAVE_UNISTD_H # include #endif #ifndef HAVE_GETRUSAGE # ifdef HAVE_SYS_TIME_H # include # endif #endif #ifdef HAVE_FCNTL_H # include #endif #if defined(HAVE_SYS_RESOURCE_H) && defined(HAVE_GETRUSAGE) /* on macOS it seems sys/resource.h needs sys/time.h first */ # include # include #endif #include /* * 4) INITIALIZATION AND TERMINATION ACTIONS */ attribute_hidden FILE *R_OpenInitFile(void) { char buf[PATH_MAX], *home, *p = getenv("R_PROFILE_USER"); FILE *fp; fp = NULL; if (LoadInitFile) { if(p) { if(!*p) return NULL; /* set to "" */ return R_fopen(R_ExpandFileName(p), "r"); } if((fp = R_fopen(".Rprofile", "r"))) return fp; if((home = getenv("HOME")) == NULL) return NULL; snprintf(buf, PATH_MAX, "%s/.Rprofile", home); if((fp = R_fopen(buf, "r"))) return fp; } return fp; } /* * R_CleanUp is interface-specific */ /* * 5) FILESYSTEM INTERACTION */ /* * R_ShowFiles is interface-specific */ /* * R_ChooseFile is interface-specific */ char *R_ExpandFileName_readline(const char *s, char *buff); /* sys-std.c */ static char newFileName[PATH_MAX]; static int HaveHOME=-1; static char UserHOME[PATH_MAX]; /* Only interpret inputs of the form ~ and ~/... */ static const char *R_ExpandFileName_unix(const char *s, char *buff) { char *p; if(s[0] != '~') return s; if(strlen(s) > 1 && s[1] != '/') return s; if(HaveHOME < 0) { p = getenv("HOME"); if(p && *p && (strlen(p) < PATH_MAX)) { strcpy(UserHOME, p); HaveHOME = 1; } else HaveHOME = 0; } if(HaveHOME > 0 && (strlen(UserHOME) + strlen(s+1) < PATH_MAX)) { strcpy(buff, UserHOME); strcat(buff, s+1); return buff; } else return s; } /* tilde_expand (in libreadline) mallocs storage for its return value. The R entry point does not require that storage to be freed, so we copy the value to a static buffer, to void a memory leak in R<=1.6.0. This is not thread-safe, but as R_ExpandFileName is a public entry point (in R-exts.texi) it will need to deprecated and replaced by a version which takes a buffer as an argument. BDR 10/2002 */ extern Rboolean UsingReadline; const char *R_ExpandFileName(const char *s) { #ifdef HAVE_LIBREADLINE if(UsingReadline) { const char * c = R_ExpandFileName_readline(s, newFileName); /* we can return the result only if tilde_expand is not broken */ if (!c || c[0]!='~' || (c[1]!='\0' && c[1]!='/')) return c; } #endif return R_ExpandFileName_unix(s, newFileName); } /* * 7) PLATFORM DEPENDENT FUNCTIONS */ SEXP attribute_hidden do_machine(SEXP call, SEXP op, SEXP args, SEXP env) { checkArity(op, args); return mkString("Unix"); } # ifdef HAVE_SYS_TIMES_H # include /* times */ # endif static double clk_tck, StartTime; void R_setStartTime(void) { #ifdef HAVE_SYSCONF clk_tck = (double) sysconf(_SC_CLK_TCK); #else # ifndef CLK_TCK /* this is in ticks/second, generally 60 on BSD style Unix, 100? on SysV */ # ifdef HZ # define CLK_TCK HZ # else # define CLK_TCK 60 # endif # endif /* not CLK_TCK */ clk_tck = (double) CLK_TCK; #endif /* printf("CLK_TCK = %d\n", CLK_TCK); */ StartTime = currentTime(); } /* NOTE This used to use times() for elapsed times, which is measured in clock ticks (which can overflow). It is possible this version uses time() and so is in seconds. But even Cygwin has gettimeofday. */ attribute_hidden void R_getProcTime(double *data) { /* docs say this is rounded to the nearest ms */ double et = currentTime() - StartTime; data[2] = 1e-3 * rint(1000*et); #ifdef HAVE_GETRUSAGE /* all known current OSes */ struct rusage self, children; getrusage(RUSAGE_SELF, &self); getrusage(RUSAGE_CHILDREN, &children); data[0] = (double) self.ru_utime.tv_sec + 1e-3 * (self.ru_utime.tv_usec/1000); data[1] = (double) self.ru_stime.tv_sec + 1e-3 * (self.ru_stime.tv_usec/1000); data[3] = (double) children.ru_utime.tv_sec + 1e-3 * (children.ru_utime.tv_usec/1000); data[4] = (double) children.ru_stime.tv_sec + 1e-3 * (children.ru_stime.tv_usec/1000); #else /* Not known to be currently used */ struct tms timeinfo; times(&timeinfo); data[0] = fround(timeinfo.tms_utime / clk_tck, 3); data[1] = fround(timeinfo.tms_stime / clk_tck, 3); data[3] = fround(timeinfo.tms_cutime / clk_tck, 3); data[4] = fround(timeinfo.tms_cstime / clk_tck, 3); #endif } /* used in memory.c */ /* FIXME: maybe should try to find the increment for getrusage */ attribute_hidden double R_getClockIncrement(void) { return 1.0 / clk_tck; } #ifdef HAVE_SYS_WAIT_H # include #endif /* The timeout support is inspired by timeout utility from coreutils. However, here the child process creates a new process group rather than the parent (R) process, because changing the group leader for the whole of R might have undesirable consequences. According to comments in coreutils, this could lead to issues with propagating signals between foreground and background process groups. Like with coreutils, the timeout is not always enforced: an external application can run longer than the timeout when it creates a new process group or when it spawns a child process and exits without waiting for it to finish (becomes a daemon). This implementation only works for processes that do not read from the standard input - the new process group is always created, and hence the executed process can no longer access the terminal. To prevent interference with job control, the new process is thus started with standard input redirected from /dev/null. Note that while the timeout utility allows to run processes also without creating the new group (option `foreground`), that approach would interfere with job control in "/bin/sh" that is documented to be used by the R system call. There does not seem to be a simple way to address this issue, and hence interactive applications cannot be executed with timeout (and note the same issues arise when timeout utility is used with /bin/sh). Currently we only have a single global structure and hence only one call to R_popen_timeout/R_system_timeout may be active at the same time. A more general implementation could use a linked list and identify entries by file pointer and child pid. Background jobs (ending with &) are not supported. */ #define KILL_SIGNAL1 SIGINT #define KILL_SIGNAL2 SIGTERM #define KILL_SIGNAL3 SIGKILL #define EMERGENCY_TIMEOUT 20 /* The child processes are sent KILL_SIGNAL1 after the specified timeout. As a backup, KILL_SIGNAL2 would be sent after additional EMERGENCY_TIMEOUT seconds. As a backup of the backup, KILL_SIGNAL3 would be sent after yet additional EMERGENCY_TIMEOUT seconds. SIGINT is used first because it seems to be handled better by applications: applications happen to wait for child processes to terminate, and hence their execution is included into getrusage/RUSAGE_CHILDREN (proc.time). As follows from empirical observations, SIGTERM can sometimes terminate applications that cannot be terminated by SIGINT. */ int kill_signals[] = { KILL_SIGNAL1, KILL_SIGNAL2, KILL_SIGNAL3 }; static struct { pid_t child_pid; int timedout; /* set when the child has been timed out */ int kill_attempts; /* 1 after sending KILL_SIGNAL1, etc */ sigset_t oldset; struct sigaction oldalrm, oldint, oldquit, oldhup, oldterm, oldttin, oldttou, oldchld; RCNTXT cntxt; /* for popen/pclose */ FILE *fp; /* for popen/pclose, sanity check */ } tost; static void timeout_handler(int sig); static void timeout_init() { tost.child_pid = 0; tost.timedout = 0; tost.kill_attempts = 0; sigprocmask(0, NULL, &tost.oldset); sigaction(SIGALRM, NULL, &tost.oldalrm); sigaction(SIGINT, NULL, &tost.oldint); sigaction(SIGQUIT, NULL, &tost.oldquit); sigaction(SIGHUP, NULL, &tost.oldhup); sigaction(SIGTERM, NULL, &tost.oldterm); sigaction(SIGTTIN, NULL, &tost.oldttin); sigaction(SIGTTOU, NULL, &tost.oldttou); sigaction(SIGCHLD, NULL, &tost.oldchld); tost.fp = NULL; /* install handler */ struct sigaction sa; sigemptyset(&sa.sa_mask); sa.sa_handler = &timeout_handler; sa.sa_flags = SA_RESTART; sigaction(SIGALRM, &sa, NULL); sigaction(SIGINT, &sa, NULL); sigaction(SIGQUIT, &sa, NULL); sigaction(SIGHUP, &sa, NULL); sigaction(SIGTERM, &sa, NULL); sigaction(SIGCHLD, &sa, NULL); } static void timeout_cleanup_set(sigset_t *ss) { sigemptyset(ss); sigaddset(ss, SIGALRM); sigaddset(ss, SIGINT); sigaddset(ss, SIGQUIT); sigaddset(ss, SIGHUP); sigaddset(ss, SIGTERM); sigaddset(ss, SIGTTIN); sigaddset(ss, SIGTTOU); sigaddset(ss, SIGCHLD); } static void timeout_cleanup() { sigset_t ss; timeout_cleanup_set(&ss); sigprocmask(SIG_BLOCK, &ss, NULL); alarm(0); /* clear alarm */ sigaction(SIGALRM, &tost.oldalrm, NULL); sigaction(SIGINT, &tost.oldalrm, NULL); sigaction(SIGQUIT, &tost.oldquit, NULL); sigaction(SIGHUP, &tost.oldhup, NULL); sigaction(SIGTERM, &tost.oldterm, NULL); sigaction(SIGTTIN, &tost.oldttin, NULL); sigaction(SIGTTOU, &tost.oldttou, NULL); sigaction(SIGCHLD, &tost.oldchld, NULL); sigprocmask(SIG_SETMASK, &tost.oldset, NULL); } static void timeout_handler(int sig) { if (sig == SIGCHLD) return; /* needed for sigsuspend() to be interrupted */ if (tost.child_pid > 0 && sig == SIGALRM) { tost.timedout = 1; if (tost.kill_attempts < 3) { sig = kill_signals[tost.kill_attempts]; if (tost.kill_attempts < 2) { int saveerrno = errno; alarm(EMERGENCY_TIMEOUT); errno = saveerrno; } tost.kill_attempts++; } else sig = KILL_SIGNAL1; /* should not happen */ } if (tost.child_pid > 0) { /* parent, received a signal */ kill(tost.child_pid, sig); /* NOTE: don't signal the group and don't send SIGCONT for interactive jobs */ int saveerrno = errno; /* on macOS, killpg fails with EPERM for groups with zombies */ killpg(tost.child_pid, sig); errno = saveerrno; if (sig != SIGKILL && sig != SIGCONT) { kill(tost.child_pid, SIGCONT); saveerrno = errno; /* on macOS, killpg fails with EPERM for groups with zombies */ killpg(tost.child_pid, SIGCONT); errno = saveerrno; } } else if (tost.child_pid == 0) { /* child */ _exit(128 + sig); /* arbitrary status, such as in timeout utility */ } /* tost.child_pid is -1 when child process no longer exists */ } static pid_t timeout_wait(int *wstatus) { pid_t wres; /* make sure we do not accidentally send signals to a new process with re-used pid from the child */ sigset_t ss; timeout_cleanup_set(&ss); sigset_t unblocked_ss; sigprocmask(SIG_BLOCK, &ss, &unblocked_ss); int saveerrno = errno; while((wres = waitpid(tost.child_pid, wstatus, WNOHANG)) == 0) sigsuspend(&unblocked_ss); if (errno == EINTR) /* EINTR is not really an error but expected situation here, however, R's "system" call would report any non-zero errno as an error. */ errno = saveerrno; if (wres == tost.child_pid) tost.child_pid = -1; /* the process no longer exists */ timeout_cleanup(); return wres; } static void timeout_cend(void *data) { if (tost.child_pid > 0) { timeout_handler(SIGALRM); timeout_wait(NULL); } timeout_cleanup(); } /* Fork with blocked SIGCHLD to make sure that tost.child_pid is set in the parent before the signal is received. Also makes sure SIGCHLD is unblocked in the parent after the call. */ static void timeout_fork() { sigset_t css; sigemptyset(&css); sigaddset(&css, SIGCHLD); sigprocmask(SIG_BLOCK, &css, NULL); tost.child_pid = fork(); sigprocmask(SIG_UNBLOCK, &css, NULL); } /* R_popen_timeout, R_pclose_timeout - a partial implementation of popen/close with support for timeout. The POSIX/Unix popen/pclose cannot be re-used, because the PID of the child process is not accessible via POSIX API. This simple implementation only supports a single pipe to be open at a time and R_system_timeout cannot be used at the same time. It does not support close-on-exec ("e" flag). A pipe opened with R_popen_timeout cannot be closed by pclose. A pipe opened with popen cannot be closed by R_pclose_timeout. Timeout is in seconds. After timing out, the child process is interrupted. */ static FILE *R_popen_timeout(const char *cmd, const char *type, int timeout) { /* close-on-exec is not supported */ if (!type || type[1] || (type[0] != 'r' && type[0] != 'w')) { errno = EINVAL; return NULL; } int doread = (type[0] == 'r'); int pipefd[2]; int parent_end, child_end; if (pipe(pipefd) < 0) return NULL; if (doread) { parent_end = pipefd[0]; child_end = pipefd[1]; } else { parent_end = pipefd[1]; child_end = pipefd[0]; } /* Earlier version of R would block SIGPROF here on old Apple systems following Luke's recommendation on how to fix PR#1140 (see R_open, R_system). */ timeout_init(); /* set up a context to recover from R error between popen and pclose */ begincontext(&tost.cntxt, CTXT_CCODE, R_NilValue, R_BaseEnv, R_BaseEnv, R_NilValue, R_NilValue); tost.cntxt.cenddata = NULL; tost.cntxt.cend = &timeout_cend; signal(SIGTTIN, SIG_IGN); signal(SIGTTOU, SIG_IGN); timeout_fork(); if (tost.child_pid == 0) { /* child */ setpgid(0, 0); /* NOTE: don't create new group in interactive jobs */ signal(SIGTTIN, SIG_DFL); signal(SIGTTOU, SIG_DFL); dup2(child_end, doread ? 1 : 0); close(child_end); close(parent_end); close(doread ? 0 : 1); /* ensure there is no read from terminal to avoid SIGTTIN */ if (open("/dev/null", O_RDONLY) < 0) { perror("Cannot open /dev/null for reading:"); _exit(127); } execl("/bin/sh", "sh", "-c", cmd, (char *)NULL); _exit(127); /* execl failed */ } else if (tost.child_pid > 0) { /* parent */ close(child_end); tost.fp = fdopen(parent_end, type); if (!tost.fp) { close(parent_end); return NULL; } sigset_t ss; sigemptyset(&ss); sigaddset(&ss, SIGALRM); sigprocmask(SIG_UNBLOCK, &ss, NULL); alarm(timeout); /* will get SIGALRM on timeout */ return tost.fp; } else { close(parent_end); return NULL; } } int R_pclose_timeout(FILE *fp) { if (fp != tost.fp) /* should not happen */ error("Invalid file pointer in pclose"); /* Do not use fclose, because on Solaris it sets errno to "Invalid seek" when the pipe is already closed (e.g. because of timeout). fclose would not return an error, but it would set errno and the non-zero errno would then be reported by R's "system" function. */ int fd = fileno(fp); if (fd >= 0) close(fd); pid_t wres; int wstatus; wres = timeout_wait(&wstatus); endcontext(&tost.cntxt); if (wres < 0) return -1; return wstatus; } /* Similar to system, but supports timeout in seconds. Calls to R_system_timeout cannot be used when a pipe is open using R_popen_timeout. */ static int R_system_timeout(const char *cmd, int timeout) { if (!cmd) return R_system(cmd); /* Earlier version of R would block SIGPROF here on old Apple systems following Luke's recommendation on how to fix PR#1140 (see R_open, R_system). */ timeout_init(); signal(SIGTTIN, SIG_IGN); signal(SIGTTOU, SIG_IGN); timeout_fork(); if (tost.child_pid == 0) { /* child */ close(0); /* ensure there is no read from terminal to avoid SIGTTIN */ if (open("/dev/null", O_RDONLY) < 0) { perror("Cannot open /dev/null for reading:"); _exit(127); } setpgid(0, 0); signal(SIGTTIN, SIG_DFL); signal(SIGTTOU, SIG_DFL); execl("/bin/sh", "sh", "-c", cmd, (char *)NULL); _exit(127); /* execl failed */ } else if (tost.child_pid > 0) { /* parent */ sigset_t ss; sigemptyset(&ss); sigaddset(&ss, SIGALRM); sigprocmask(SIG_UNBLOCK, &ss, NULL); alarm(timeout); /* will get SIGALRM on timeout */ int wstatus; timeout_wait(&wstatus); if (tost.child_pid != -1) return -1; #ifdef HAVE_SYS_WAIT_H if (WIFEXITED(wstatus)) wstatus = WEXITSTATUS(wstatus); #else /* assume that this is shifted if a multiple of 256 */ if ((wstatus % 256) == 0) wstatus = wstatus/256; #endif if (wstatus == -1) { /* this means that system() failed badly - it didn't even get to try to run the shell */ warning(_("system call failed: %s"), strerror(errno)); /* R system() is documented to return 127 on failure, and a lot of code relies on that - it will misinterpret -1 as success */ wstatus = 127; } return wstatus; } else return -1; } static void warn_status(const char *cmd, int res) { if (!res) return; if (errno) /* FIXME: TK: non-zero errno is a sign of an error only when a function that modified it also signals an error by its return value, usually -1 or EOF. We should not be reporting an error here (CERT ERR30-C).*/ /* on Solaris, if the command ends with non-zero status and timeout is 0, "Illegal seek" error is reported; the timeout version works this around by using close(fileno) */ warning(_("running command '%s' had status %d and error message '%s'"), cmd, res, strerror(errno)); else warning(_("running command '%s' had status %d"), cmd, res); } #define INTERN_BUFSIZE 8096 SEXP attribute_hidden do_system(SEXP call, SEXP op, SEXP args, SEXP rho) { SEXP tlist = R_NilValue; int intern = 0; int timeout = 0; checkArity(op, args); if (!isValidStringF(CAR(args))) error(_("non-empty character argument expected")); intern = asLogical(CADR(args)); if (intern == NA_INTEGER) error(_("'intern' must be logical and not NA")); timeout = asInteger(CADDR(args)); if (timeout == NA_INTEGER || timeout < 0) error(_("invalid '%s' argument"), "timeout"); const char *cmd = translateChar(STRING_ELT(CAR(args), 0)); if (timeout > 0) { /* command ending with & is not supported by timeout */ const void *vmax = vmaxget(); const char *c = translateCharUTF8(STRING_ELT(CAR(args), 0)); int last_is_amp = 0; int len = 0; for(;*c; c += len) { len = utf8clen(*c); if (len == 1) { if (*c == '&') last_is_amp = 1; else if (*c != ' ' && *c != '\t' && *c != '\r' && *c != '\n') last_is_amp = 0; } else last_is_amp = 0; } if (last_is_amp) error("Timeout with background running processes is not supported."); vmaxset(vmax); } if (intern) { /* intern = TRUE */ FILE *fp; char *x = "r", #ifdef HAVE_GETLINE *buf = NULL; size_t buf_len = 0; #else buf[INTERN_BUFSIZE]; #endif int i, j, res; SEXP tchar, rval; PROTECT(tlist); errno = 0; /* precaution */ if (timeout == 0) fp = R_popen(cmd, x); else fp = R_popen_timeout(cmd, x, timeout); if(!fp) error(_("cannot popen '%s', probable reason '%s'"), cmd, strerror(errno)); #ifdef HAVE_GETLINE size_t read; for(i = 0; (read = getline(&buf, &buf_len, fp)) != (size_t)-1; i++) { if (buf[read - 1] == '\n') #else for (i = 0; fgets(buf, INTERN_BUFSIZE, fp); i++) { size_t read = strlen(buf); if(read >= INTERN_BUFSIZE - 1) warning(_("line %d may be truncated in call to system(, intern = TRUE)"), i + 1); if (read > 0 && buf[read-1] == '\n') #endif buf[read - 1] = '\0'; /* chop final CR */ tchar = mkChar(buf); UNPROTECT(1); PROTECT(tlist = CONS(tchar, tlist)); } #ifdef HAVE_GETLINE if (buf != NULL) free(buf); #endif if (timeout == 0) res = pclose(fp); else res = R_pclose_timeout(fp); /* On Solaris, pclose sometimes returns -1 and sets errno to ESPIPE (Illegal seek). In that case, do_system reports 0 exit status and displays a warning via warn_status. ESPIPE is not mentioned by POSIX as possible outcome of pclose. */ #ifdef HAVE_SYS_WAIT_H if (WIFEXITED(res)) res = WEXITSTATUS(res); else res = 0; #else /* assume that this is shifted if a multiple of 256 */ if ((res % 256) == 0) res = res/256; #endif if ((res & 0xff) == 127) {/* 127, aka -1 */ if (errno) error(_("error in running command: '%s'"), strerror(errno)); else error(_("error in running command")); } if (timeout && tost.timedout) { res = 124; warning(_("command '%s' timed out after %ds"), cmd, timeout); } else warn_status(cmd, res); rval = PROTECT(allocVector(STRSXP, i)); for (j = (i - 1); j >= 0; j--) { SET_STRING_ELT(rval, j, CAR(tlist)); tlist = CDR(tlist); } if(res) { SEXP lsym = install("status"); setAttrib(rval, lsym, ScalarInteger(res)); if(errno) { lsym = install("errmsg"); setAttrib(rval, lsym, mkString(strerror(errno))); } } UNPROTECT(2); return rval; } else { /* intern = FALSE */ #ifdef HAVE_AQUA R_Busy(1); #endif tlist = PROTECT(allocVector(INTSXP, 1)); fflush(stdout); int res; if (timeout == 0) res = R_system(cmd); else res = R_system_timeout(cmd, timeout); if (res == 127) warning(_("error in running command")); if (timeout && tost.timedout) { res = 124; warning(_("command '%s' timed out after %ds"), cmd, timeout); } INTEGER(tlist)[0] = res; #ifdef HAVE_AQUA R_Busy(0); #endif UNPROTECT(1); R_Visible = 0; return tlist; } } #ifdef HAVE_SYS_UTSNAME_H # include # ifdef HAVE_UNISTD_H # include # endif # ifdef HAVE_PWD_H # include # endif SEXP attribute_hidden do_sysinfo(SEXP call, SEXP op, SEXP args, SEXP rho) { SEXP ans, ansnames; struct utsname name; char *login; checkArity(op, args); PROTECT(ans = allocVector(STRSXP, 8)); if(uname(&name) == -1) { UNPROTECT(1); return R_NilValue; } SET_STRING_ELT(ans, 0, mkChar(name.sysname)); SET_STRING_ELT(ans, 1, mkChar(name.release)); SET_STRING_ELT(ans, 2, mkChar(name.version)); SET_STRING_ELT(ans, 3, mkChar(name.nodename)); SET_STRING_ELT(ans, 4, mkChar(name.machine)); login = getlogin(); SET_STRING_ELT(ans, 5, login ? mkChar(login) : mkChar("unknown")); #if defined(HAVE_PWD_H) && defined(HAVE_GETPWUID) && defined(HAVE_GETUID) { struct passwd *stpwd; stpwd = getpwuid(getuid()); SET_STRING_ELT(ans, 6, stpwd ? mkChar(stpwd->pw_name) : mkChar("unknown")); } #else SET_STRING_ELT(ans, 6, mkChar("unknown")); #endif #if defined(HAVE_PWD_H) && defined(HAVE_GETPWUID) && defined(HAVE_GETEUID) { struct passwd *stpwd; stpwd = getpwuid(geteuid()); SET_STRING_ELT(ans, 7, stpwd ? mkChar(stpwd->pw_name) : mkChar("unknown")); } #else SET_STRING_ELT(ans, 7, mkChar("unknown")); #endif PROTECT(ansnames = allocVector(STRSXP, 8)); SET_STRING_ELT(ansnames, 0, mkChar("sysname")); SET_STRING_ELT(ansnames, 1, mkChar("release")); SET_STRING_ELT(ansnames, 2, mkChar("version")); SET_STRING_ELT(ansnames, 3, mkChar("nodename")); SET_STRING_ELT(ansnames, 4, mkChar("machine")); SET_STRING_ELT(ansnames, 5, mkChar("login")); SET_STRING_ELT(ansnames, 6, mkChar("user")); SET_STRING_ELT(ansnames, 7, mkChar("effective_user")); setAttrib(ans, R_NamesSymbol, ansnames); UNPROTECT(2); return ans; } #else /* not HAVE_SYS_UTSNAME_H */ SEXP do_sysinfo(SEXP call, SEXP op, SEXP args, SEXP rho) { warning(_("Sys.info() is not implemented on this system")); return R_NilValue; /* -Wall */ } #endif /* not HAVE_SYS_UTSNAME_H */ /* The pointer here is used in the Mac GUI */ #include /* for R_PolledEvents */ #include DL_FUNC ptr_R_ProcessEvents; void R_ProcessEvents(void) { #ifdef HAVE_AQUA /* disable ProcessEvents in child, since we can't call CoreFoundation there. */ if (ptr_R_ProcessEvents && !R_isForkedChild) ptr_R_ProcessEvents(); #else /* We might in due course want to always inhibit in a child */ if (ptr_R_ProcessEvents) ptr_R_ProcessEvents(); #endif R_PolledEvents(); if (cpuLimit > 0.0 || elapsedLimit > 0.0) { double cpu, data[5]; R_getProcTime(data); cpu = data[0] + data[1] + data[3] + data[4]; if (elapsedLimit > 0.0 && data[2] > elapsedLimit) { cpuLimit = elapsedLimit = -1; if (elapsedLimit2 > 0.0 && data[2] > elapsedLimit2) { elapsedLimit2 = -1.0; error(_("reached session elapsed time limit")); } else error(_("reached elapsed time limit")); } if (cpuLimit > 0.0 && cpu > cpuLimit) { cpuLimit = elapsedLimit = -1; if (cpuLimit2 > 0.0 && cpu > cpuLimit2) { cpuLimit2 = -1.0; error(_("reached session CPU time limit")); } else error(_("reached CPU time limit")); } } } /* * helpers for start-up code */ #ifdef __FreeBSD__ # ifdef HAVE_FLOATINGPOINT_H # include # endif #endif /* patch from Ei-ji Nakama for Intel compilers on ix86. From http://www.nakama.ne.jp/memo/ia32_linux/R-2.1.1.iccftzdaz.patch.txt. Since updated to include x86_64. */ #if (defined(__i386) || defined(__x86_64)) && defined(__INTEL_COMPILER) && __INTEL_COMPILER > 800 #include #include #endif /* exported for Rembedded.h */ void fpu_setup(Rboolean start) { if (start) { #ifdef __FreeBSD__ fpsetmask(0); #endif #if (defined(__i386) || defined(__x86_64)) && defined(__INTEL_COMPILER) && __INTEL_COMPILER > 800 _MM_SET_FLUSH_ZERO_MODE(_MM_FLUSH_ZERO_OFF); _MM_SET_DENORMALS_ZERO_MODE(_MM_DENORMALS_ZERO_OFF); #endif } else { #ifdef __FreeBSD__ fpsetmask(~0); #endif } }