/* R : A Computer Language for Statistical Data Analysis Copyright (C) 1995-1996 Robert Gentleman and Ross Ihaka Copyright (C) 1997-2015 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/ */ /* See ../unix/system.txt for a description of some of these functions */ #ifdef HAVE_CONFIG_H #include #endif #include "Defn.h" #include "Fileio.h" /* for R_fopen */ #include "Startup.h" #ifdef HAVE_UNISTD_H #include #endif /* These are used in ../gnuwin32/system.c, ../unix/sys-std.c */ SA_TYPE SaveAction = SA_SAVEASK; SA_TYPE RestoreAction = SA_RESTORE; static Rboolean LoadSiteFile = TRUE; attribute_hidden Rboolean LoadInitFile = TRUE; /* Used in R_OpenInitFile */ static Rboolean DebugInitFile = FALSE; /* * INITIALIZATION AND TERMINATION ACTIONS */ void attribute_hidden R_InitialData(void) { R_RestoreGlobalEnv(); } attribute_hidden FILE *R_OpenLibraryFile(const char *file) { char buf[PATH_MAX]; FILE *fp; snprintf(buf, PATH_MAX, "%s/library/base/R/%s", R_Home, file); fp = R_fopen(buf, "r"); return fp; } attribute_hidden char *R_LibraryFileName(const char *file, char *buf, size_t bsize) { if (snprintf(buf, bsize, "%s/library/base/R/%s", R_Home, file) < 0) error(_("R_LibraryFileName: buffer too small")); return buf; } attribute_hidden FILE *R_OpenSysInitFile(void) { char buf[PATH_MAX]; FILE *fp; snprintf(buf, PATH_MAX, "%s/library/base/R/Rprofile", R_Home); fp = R_fopen(buf, "r"); return fp; } attribute_hidden FILE *R_OpenSiteFile(void) { char buf[PATH_MAX]; FILE *fp; fp = NULL; if (LoadSiteFile) { char *p = getenv("R_PROFILE"); if (p) { if (*p) return R_fopen(R_ExpandFileName(p), "r"); else return NULL; } #ifdef R_ARCH snprintf(buf, PATH_MAX, "%s/etc/%s/Rprofile.site", R_Home, R_ARCH); if ((fp = R_fopen(buf, "r"))) return fp; #endif snprintf(buf, PATH_MAX, "%s/etc/Rprofile.site", R_Home); if ((fp = R_fopen(buf, "r"))) return fp; } return fp; } /* Saving and Restoring the Global Environment */ #ifndef Win32 static char workspace_name[1000] = ".RData"; /* set_workspace_name is in src/gnuwin32/system.c and used to implement drag-and-drop on Windows. */ #else static char workspace_name[PATH_MAX] = ".RData"; attribute_hidden void set_workspace_name(const char *fn) { strncpy(workspace_name, fn, PATH_MAX); workspace_name[PATH_MAX - 1] = '\0'; } #endif attribute_hidden const char* get_workspace_name() { return workspace_name; } void R_RestoreGlobalEnv(void) { if(RestoreAction == SA_RESTORE) { R_RestoreGlobalEnvFromFile(workspace_name, R_Quiet); } } void R_SaveGlobalEnv(void) { R_SaveGlobalEnvToFile(".RData"); } /* * INITIALIZATION HELPER CODE */ void R_DefParams(Rstart Rp) { Rp->R_Quiet = FALSE; Rp->R_Slave = FALSE; Rp->R_Interactive = TRUE; Rp->R_Verbose = FALSE; Rp->RestoreAction = SA_RESTORE; Rp->SaveAction = SA_SAVEASK; Rp->LoadSiteFile = TRUE; Rp->LoadInitFile = TRUE; Rp->DebugInitFile = FALSE; Rp->vsize = R_VSIZE; Rp->nsize = R_NSIZE; Rp->max_vsize = R_SIZE_T_MAX; Rp->max_nsize = R_SIZE_T_MAX; Rp->ppsize = R_PPSSIZE; Rp->NoRenviron = FALSE; R_SizeFromEnv(Rp); } #define Max_Nsize 50000000 /* about 1.4Gb 32-bit, 2.8Gb 64-bit */ #define Max_Vsize R_SIZE_T_MAX /* unlimited */ // small values ok for R_DEFAULT_PACKAGES=NULL (= 'base' only) #define Min_Nsize 50000 #define Min_Vsize 262144 // = (Mega/4) void R_SizeFromEnv(Rstart Rp) { int ierr; R_size_t value; char *p, msg[256]; if ((p = getenv("R_MAX_VSIZE"))) { value = R_Decode2Long(p, &ierr); if(ierr != 0 || value > Max_Vsize) R_ShowMessage("WARNING: invalid R_VSIZE ignored\n"); else if(value < Min_Vsize) { snprintf(msg, 256, "WARNING: R_MAX_VSIZE smaller than Min_Vsize = %lu is ignored\n", (unsigned long) Min_Vsize); R_ShowMessage(msg); } else Rp->max_vsize = value; } #if defined(__APPLE__) && defined(_SC_PHYS_PAGES) && defined(_SC_PAGE_SIZE) /* For now only on macOS place a default limit on the vector heap size to avoid having R killed due to memory overcommit. Setting the limit at the maximum of 16Gb and available physical memory seems reasonable, but there may be better options. LT */ else { R_size_t pages = sysconf(_SC_PHYS_PAGES); R_size_t page_size = sysconf(_SC_PAGE_SIZE); R_size_t sysmem = pages * page_size; R_size_t MinMaxVSize = 17179869184; /* 16 Gb */ Rp->max_vsize = sysmem > MinMaxVSize ? sysmem : MinMaxVSize; } #endif if((p = getenv("R_VSIZE"))) { value = R_Decode2Long(p, &ierr); if(ierr != 0 || value > Max_Vsize) R_ShowMessage("WARNING: invalid R_VSIZE ignored\n"); else if(value < Min_Vsize) { snprintf(msg, 256, "WARNING: R_VSIZE smaller than Min_Vsize = %lu is ignored\n", (unsigned long) Min_Vsize); R_ShowMessage(msg); } else Rp->vsize = value; } if((p = getenv("R_NSIZE"))) { value = R_Decode2Long(p, &ierr); if(ierr != 0 || value > Max_Nsize) R_ShowMessage("WARNING: invalid R_NSIZE ignored\n"); else if(value < Min_Nsize) { snprintf(msg, 256, "WARNING: R_NSIZE smaller than Min_Nsize = %lu is ignored\n", (unsigned long) Min_Nsize); R_ShowMessage(msg); } else Rp->nsize = value; } } static void SetSize(R_size_t vsize, R_size_t nsize) { char msg[1024]; Rboolean sml; /* vsize > 0 to catch long->int overflow */ if (vsize < 1000 && vsize > 0) { R_ShowMessage("WARNING: vsize ridiculously low, Megabytes assumed\n"); vsize *= (R_size_t) Mega; } if((sml = vsize < Min_Vsize) || vsize > Max_Vsize) { snprintf(msg, 1024, "WARNING: %s v(ector heap)size '%lu' ignored," " using default = %gM\n", sml ? "too small" : "too large", (unsigned long) vsize, R_VSIZE / Mega); R_ShowMessage(msg); R_VSize = R_VSIZE; } else R_VSize = vsize; if((sml = nsize < Min_Nsize) || nsize > Max_Nsize) { snprintf(msg, 1024, "WARNING: %s language heap (n)size '%lu' ignored," " using default = %ld\n", sml ? "too small" : "too large", (unsigned long) nsize, R_NSIZE); R_ShowMessage(msg); R_NSize = R_NSIZE; } else R_NSize = nsize; } void R_SetParams(Rstart Rp) { R_Quiet = Rp->R_Quiet; R_Slave = Rp->R_Slave; R_Interactive = Rp->R_Interactive; R_Verbose = Rp->R_Verbose; RestoreAction = Rp->RestoreAction; SaveAction = Rp->SaveAction; LoadSiteFile = Rp->LoadSiteFile; LoadInitFile = Rp->LoadInitFile; DebugInitFile = Rp->DebugInitFile; SetSize(Rp->vsize, Rp->nsize); R_SetMaxNSize(Rp->max_nsize); R_SetMaxVSize(Rp->max_vsize); R_SetPPSize(Rp->ppsize); #ifdef Win32 R_SetWin32(Rp); #endif }