// This file is part of BOINC. // http://boinc.berkeley.edu // Copyright (C) 2008 University of California // // BOINC is free software; you can redistribute it and/or modify it // under the terms of the GNU Lesser General Public License // as published by the Free Software Foundation, // either version 3 of the License, or (at your option) any later version. // // BOINC 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 Lesser General Public License for more details. // // You should have received a copy of the GNU Lesser General Public License // along with BOINC. If not, see . // There is a reason that having a file called "cpp.h" that includes config.h // and some of the C++ header files is bad. That reason is because there are // #defines that alter the behiour of the standard C and C++ headers. In // this case we need to use the "small files" environment on some unix // systems. That can't be done if we include "cpp.h" // copied directly from cpp.h #if defined(_WIN32) && !defined(__CYGWIN32__) #if defined(_WIN64) && defined(_M_X64) #define HOSTTYPE "windows_x86_64" #define HOSTTYPEALT "windows_intelx86" #else #define HOSTTYPE "windows_intelx86" #endif #include "version.h" // version numbers from autoconf #endif #include "config.h" #if !defined(_WIN32) || defined(__CYGWIN32__) // Access to binary files in /proc filesystem doesn't work in the 64bit // files environment on some systems. None of the functions here need // 64bit file functions, so we'll undefine _FILE_OFFSET_BITS and _LARGE_FILES. #undef _FILE_OFFSET_BITS #undef _LARGE_FILES #undef _LARGEFILE_SOURCE #undef _LARGEFILE64_SOURCE #include #include #include #include #endif #include #include #include #include #include #if HAVE_SYS_TYPES_H #include #endif #ifdef HAVE_SYS_MOUNT_H #include #endif #ifdef HAVE_SYS_VFS_H #include #endif #ifdef HAVE_SYS_VMMETER_H #include #endif #include #if HAVE_SYS_SWAP_H #include #endif #if HAVE_SYS_SYSCTL_H #include #endif #if HAVE_SYS_SYSTEMINFO_H #include #endif #if HAVE_SYS_UTSNAME_H #include #endif #if HAVE_UNISTD_H #include #endif #if HAVE_UTMP_H #include #endif #if HAVE_NETINET_IN_H #include #endif #ifdef __EMX__ #define INCL_DOSMISC #include #include "win/opt_x86.h" #endif #include "client_types.h" #include "filesys.h" #include "error_numbers.h" #include "str_util.h" #include "client_state.h" #include "hostinfo_network.h" #include "client_msgs.h" using std::string; #ifdef __APPLE__ #include #include #ifdef __cplusplus extern "C" { #endif #include #include #ifdef __cplusplus } // extern "C" #endif NXEventHandle gEventHandle = NULL; #endif // __APPLE__ #ifdef _HPUX_SOURCE #include #endif // Tru64 UNIX. // 2005-12-26 SMS. #ifdef __osf__ #include #include #include #endif // The following is intended to be true both on Linux // and Debian GNU/kFreeBSD (see trac #521) // #define LINUX_LIKE_SYSTEM defined(__linux__) || defined(__GNU__) || defined(__GLIBC__) // functions to get name/addr of local host // Converts a int ip address to a string representation (i.e. "66.218.71.198") // #if 0 char* ip_addr_string(int ip_addr) { in_addr ia; ia.s_addr = ip_addr; return inet_ntoa(ia); } #endif // Returns the offset between LOCAL STANDARD TIME and UTC. // LOCAL_STANDARD_TIME = UTC_TIME + get_timezone(). // int get_timezone() { tzset(); // TODO: take daylight savings time into account #ifdef HAVE_STRUCT_TM_TM_ZONE time_t cur_time; struct tm *time_data; cur_time = time(NULL); time_data = localtime( &cur_time ); // tm_gmtoff is already adjusted for daylight savings time return time_data->tm_gmtoff; #elif LINUX_LIKE_SYSTEM return -1*(__timezone); #elif defined(__CYGWIN32__) return -1*(_timezone); #elif defined(unix) return -1*timezone; #elif defined(HAVE_TZNAME) return -1*timezone; #else #error timezone #endif return 0; } // Returns true if the host is currently running off battery power // If you can't figure out, return false // bool HOST_INFO::host_is_running_on_batteries() { #if defined(__APPLE__) CFDictionaryRef pSource = NULL; CFStringRef psState; int i; bool retval = false; CFTypeRef blob = IOPSCopyPowerSourcesInfo(); CFArrayRef list = IOPSCopyPowerSourcesList(blob); for (i=0; i ==buf necessary */ if ((strstr(buf, "model\t\t: ") == buf) && model<0) { model = atoi(buf+strlen("model\t\t: ")); } /* ia64 */ if (strstr(buf, "model : ") && model<0) { model = atoi(buf+strlen("model : ")); } #endif if (strstr(buf, "stepping\t: ") && stepping<0) { stepping = atoi(buf+strlen("stepping\t: ")); } #ifdef __hppa__ if (!icache_found && strstr(buf, "I-cache\t\t: ")) { icache_found = true; sscanf(buf, "I-cache\t\t: %d", &n); host.m_cache += n*1024; } if (!dcache_found && strstr(buf, "D-cache\t\t: ")) { dcache_found = true; sscanf(buf, "D-cache\t\t: %d", &n); host.m_cache += n*1024; } #elif __powerpc__ if (!cache_found && strstr(buf, "L2 cache\t: ")) { cache_found = true; sscanf(buf, "L2 cache\t: %d", &n); host.m_cache = n*1024; } #else if (!cache_found && (strstr(buf, "cache size\t: ") == buf)) { cache_found = true; sscanf(buf, "cache size\t: %d", &n); host.m_cache = n*1024; } #endif if (!features_found) { // Some versions of the linux kernel call them flags, // others call them features, so look for both. // if ((strstr(buf, "flags\t\t: ") == buf)) { strlcpy(features, strchr(buf, ':') + 2, sizeof(features)); } else if ((strstr(buf, "features\t\t: ") == buf)) { strlcpy(features, strchr(buf, ':') + 2, sizeof(features)); } else if ((strstr(buf, "features : ") == buf)) { /* ia64 */ strlcpy(features, strchr(buf, ':') + 2, sizeof(features)); } if (strlen(features)) { features_found = true; } } } strcpy(model_buf, host.p_model); if (family>=0 || model>=0 || stepping>0) { strcat(model_buf, " ["); if (family>=0) { sprintf(buf, "Family %d ", family); strcat(model_buf, buf); } if (model>=0) { sprintf(buf, "Model %d ", model); strcat(model_buf, buf); } if (stepping>=0) { sprintf(buf, "Stepping %d", stepping); strcat(model_buf, buf); } strcat(model_buf, "]"); } if (strlen(features)) { strlcpy(host.p_features, features, sizeof(host.p_features)); } strlcpy(host.p_model, model_buf, sizeof(host.p_model)); fclose(f); } #endif // LINUX_LIKE_SYSTEM #ifdef __FreeBSD__ #if defined(__i386__) || defined(__amd64__) #include #include #include void use_cpuid(HOST_INFO& host) { u_int p[4]; int hasMMX, hasSSE, hasSSE2, hasSSE3, has3DNow, has3DNowExt = 0; char capabilities[256]; do_cpuid(0x0, p); if (p[0] >= 0x1) { do_cpuid(0x1, p); hasMMX = (p[3] & (1 << 23 )) >> 23; // 0x0800000 hasSSE = (p[3] & (1 << 25 )) >> 25; // 0x2000000 hasSSE2 = (p[3] & (1 << 26 )) >> 26; // 0x4000000 hasSSE3 = (p[2] & (1 << 0 )) >> 0; } do_cpuid(0x80000000, p); if (p[0]>=0x80000001) { do_cpuid(0x80000001, p); hasMMX |= (p[3] & (1 << 23 )) >> 23; // 0x0800000 has3DNow = (p[3] & (1 << 31 )) >> 31; //0x80000000 has3DNowExt = (p[3] & (1 << 30 )) >> 30; } capabilities[0] = '\0'; if (hasSSE) strncat(capabilities, "sse ", 4); if (hasSSE2) strncat(capabilities, "sse2 ", 5); if (hasSSE3) strncat(capabilities, "sse3 ", 5); if (has3DNow) strncat(capabilities, "3dnow ", 6); if (has3DNowExt) strncat(capabilities, "3dnowext ", 9); if (hasMMX) strncat(capabilities, "mmx ", 4); strip_whitespace(capabilities); snprintf(host.p_model, sizeof(host.p_model), "%s [] [%s]", host.p_model, capabilities); } #endif #endif #ifdef __APPLE__ static void get_cpu_info_maxosx(HOST_INFO& host) { int p_model_size = sizeof(host.p_model); size_t len; #if defined(__i386__) || defined(__x86_64__) char brand_string[256]; int family, stepping, model; len = sizeof(host.p_vendor); sysctlbyname("machdep.cpu.vendor", host.p_vendor, &len, NULL, 0); len = sizeof(brand_string); sysctlbyname("machdep.cpu.brand_string", brand_string, &len, NULL, 0); len = sizeof(family); sysctlbyname("machdep.cpu.family", &family, &len, NULL, 0); len = sizeof(model); sysctlbyname("machdep.cpu.model", &model, &len, NULL, 0); len = sizeof(stepping); sysctlbyname("machdep.cpu.stepping", &stepping, &len, NULL, 0); len = sizeof(host.p_features); sysctlbyname("machdep.cpu.features", host.p_features, &len, NULL, 0); snprintf( host.p_model, sizeof(host.p_model), "%s [x86 Family %d Model %d Stepping %d]", brand_string, family, model, stepping ); #else // PowerPC char capabilities[256], model[256]; int response = 0; int retval; len = sizeof(response); safe_strcpy(host.p_vendor, "Power Macintosh"); retval = sysctlbyname("hw.optional.altivec", &response, &len, NULL, 0); if (response && (!retval)) { safe_strcpy(capabilities, "AltiVec"); } len = sizeof(model); sysctlbyname("hw.model", model, &len, NULL, 0); snprintf(host.p_model, p_model_size, "%s [%s Model %s] [%s]", host.p_vendor, host.p_vendor, model, capabilities); #endif host.p_model[p_model_size-1] = 0; char *in = host.p_model + 1; char *out = in; // Strip out runs of multiple spaces do { if ((!isspace(*(in-1))) || (!isspace(*in))) { *out++ = *in; } } while (*in++); } #endif // Rules: // - Keep code in the right place // - only one level of #if // int HOST_INFO::get_host_info() { get_filesystem_info(d_total, d_free); ///////////// p_vendor, p_model, p_features ///////////////// #if LINUX_LIKE_SYSTEM parse_cpuinfo_linux(*this); #elif defined( __APPLE__) int mib[2]; size_t len; get_cpu_info_maxosx(*this); #elif defined(__EMX__) CPU_INFO_t cpuInfo; strlcpy( p_vendor, cpuInfo.vendor.company, sizeof(p_vendor)); strlcpy( p_model, cpuInfo.name.fromID, sizeof(p_model)); #elif defined(HAVE_SYS_SYSCTL_H) int mib[2]; size_t len; // Get machine mib[0] = CTL_HW; mib[1] = HW_MACHINE; len = sizeof(p_vendor); sysctl(mib, 2, &p_vendor, &len, NULL, 0); // Get model mib[0] = CTL_HW; mib[1] = HW_MODEL; len = sizeof(p_model); sysctl(mib, 2, &p_model, &len, NULL, 0); #elif defined(__osf__) // Tru64 UNIX. // 2005-12-26 SMS. long cpu_type; char *cpu_type_name; strcpy(p_vendor, "HP (DEC)"); getsysinfo( GSI_PROC_TYPE, (caddr_t) &cpu_type, sizeof( cpu_type)); CPU_TYPE_TO_TEXT( (cpu_type& 0xffffffff), cpu_type_name); strncpy( p_model, "Alpha ", sizeof( p_model)); strncat( p_model, cpu_type_name, (sizeof( p_model)- strlen( p_model)- 1)); p_model[sizeof(p_model)-1]=0; #elif defined(HAVE_SYS_SYSTEMINFO_H) sysinfo(SI_PLATFORM, p_vendor, sizeof(p_vendor)); sysinfo(SI_ISALIST, p_model, sizeof(p_model)); for (unsigned int i=0; i 2GB. The Gestalt selector // gestaltPhysicalRAMSizeInMegabytes is available starting with OS 10.3.0. SInt32 mem_size; if (Gestalt(gestaltPhysicalRAMSizeInMegabytes, &mem_size)) { msg_printf(NULL, MSG_INTERNAL_ERROR, "Couldn't determine physical RAM size" ); } m_nbytes = (1024. * 1024.) * (double)mem_size; #elif defined(_HPUX_SOURCE) struct pst_static pst; pstat_getstatic(&pst, sizeof(pst), (size_t)1, 0); m_nbytes = (long double)pst.physical_memory * (long double)pst.page_size; #elif defined(__osf__) // Tru64 UNIX. // 2005-12-26 SMS. int mem_size; getsysinfo( GSI_PHYSMEM, (caddr_t) &mem_size, sizeof( mem_size)); m_nbytes = 1024.* (double)mem_size; #elif defined(HW_PHYSMEM) // for OpenBSD mib[0] = CTL_HW; int mem_size; mib[1] = HW_PHYSMEM; len = sizeof(mem_size); sysctl(mib, 2, &mem_size, &len, NULL, 0); m_nbytes = mem_size; #elif defined(__FreeBSD__) unsigned int mem_size; mib[0] = CTL_HW; mib[1] = HW_PHYSMEM; len = sizeof(mem_size); sysctl(mib, 2, &mem_size, &len, NULL, 0); m_nbytes = mem_size; #else #error Need to specify a method to get memory size #endif #if defined(HAVE_SYS_SWAP_H) && defined(SC_GETNSWP) // Solaris, ... char buf[256]; swaptbl_t* s; int i, n; n = swapctl(SC_GETNSWP, 0); s = (swaptbl_t*)malloc(n*sizeof(swapent_t) + sizeof(struct swaptable)); for (i=0; iswt_ent[i].ste_path = buf; } s->swt_n = n; n = swapctl(SC_LIST, s); m_swap = 0.0; for (i=0; iswt_ent[i].ste_length; } #elif defined(HAVE_SYS_SWAP_H) && defined(SWAP_NSWAP) // NetBSD (the above line should probably be more comprehensive struct swapent * s; int i, n; n = swapctl(SWAP_NSWAP, NULL, 0); s = (struct swapent*)malloc(n * sizeof(struct swapent)); swapctl(SWAP_STATS, s, n); m_swap = 0.0; for (i = 0; i < n; i ++) { if (s[i].se_flags & SWF_ENABLE) { m_swap += 512. * (double)s[i].se_nblks; } } #elif defined(__APPLE__) // The sysctl(vm.vmmeter) function doesn't work on OS X. However, swap // space is limited only by free disk space, so we get that info instead. // This is larger than free disk space reported by get_filesystem_info() // because it includes space available only to the kernel / super-user. // // http://developer.apple.com/documentation/Performance/Conceptual/ManagingMemory/Articles/AboutMemory.html says: // Unlike most UNIX-based operating systems, Mac OS X does not use a // preallocated swap partition for virtual memory. Instead, it uses all // of the available space on the machineÕs boot partition. struct statfs fs_info; statfs(".", &fs_info); m_swap = (double)fs_info.f_bsize * (double)fs_info.f_bfree; #elif defined(HAVE_VMMETER_H) && defined(HAVE_SYS_SYSCTL_H) && defined(CTL_VM) && defined(VM_METER) // MacOSX, I think... // // The sysctl(vm.vmmeter) function doesn't work on OS X, so the following // code fails to get the total swap space. See note above for APPLE case. // I've left this code here in case it is used by a different platform, // though I believe the first argument should be CTL_VM instead of CTL_USER. struct vmtotal vm_info; mib[0] = CTL_USER; // Should this be CTL_VM ? mib[1] = VM_METER; len = sizeof(vm_info); if (!sysctl(mib, 2, &vm_info, &len, NULL, 0)) { m_swap = 1024. * getpagesize() * (double) vm_info.t_vm; } #elif defined(_HPUX_SOURCE) struct pst_vminfo vminfo; pstat_getvminfo(&vminfo, sizeof(vminfo), (size_t)1, 0); m_swap = (vminfo.psv_swapspc_max * pst.page_size); #else //#error Need to specify a method to obtain swap space #endif get_local_network_info(); timezone = get_timezone(); ///////////// os_name, os_version ///////////////// #ifdef HAVE_SYS_UTSNAME_H struct utsname u; uname(&u); safe_strcpy(os_name, u.sysname); #ifdef __EMX__ // OS2: version is in u.version safe_strcpy(os_version, u.version); #else safe_strcpy(os_version, u.release); #endif #ifdef _HPUX_SOURCE safe_strcpy(p_model, u.machine); safe_strcpy(p_vendor, "Hewlett-Packard"); #endif #elif defined(HAVE_SYS_SYSCTL_H) && defined(CTL_KERN) && defined(KERN_OSTYPE) && defined(KERN_OSRELEASE) mib[0] = CTL_KERN; mib[1] = KERN_OSTYPE; len = sizeof(os_name); sysctl(mib, 2, &os_name, &len, NULL, 0); mib[0] = CTL_KERN; mib[1] = KERN_OSRELEASE; len = sizeof(os_version); sysctl(mib, 2, &os_version, &len, NULL, 0); #elif HAVE_SYS_SYSTEMINFO_H sysinfo(SI_SYSNAME, os_name, sizeof(os_name)); sysinfo(SI_RELEASE, os_version, sizeof(os_version)); #else #error Need to specify a method to obtain OS name/version #endif if (!strlen(host_cpid)) { generate_host_cpid(); } return 0; } // returns true iff device was last accessed before t // or if an error occurred looking at the device. // inline bool device_idle(time_t t, const char *device) { struct stat sbuf; return stat(device, &sbuf) || (sbuf.st_atime < t); } static const struct dir_dev { char *dir; char *dev; } tty_patterns[] = { #ifdef unix { "/dev","tty" }, { "/dev","pty" }, { "/dev/pts","" }, #endif // add other ifdefs here as necessary. { NULL, NULL }, }; std::vector get_tty_list() { // Create a list of all terminal devices on the system. char devname[1024]; char fullname[1024]; int done,i=0; std::vector tty_list; do { DIRREF dev=dir_open(tty_patterns[i].dir); if (dev) { do { // get next file done=dir_scan(devname,dev,1024); // does it match our tty pattern? If so, add it to the tty list. if (!done && (strstr(devname,tty_patterns[i].dev) == devname)) { // don't add anything starting with . if (devname[0] != '.') { sprintf(fullname,"%s/%s",tty_patterns[i].dir,devname); tty_list.push_back(fullname); } } } while (!done); } i++; } while (tty_patterns[i].dir != NULL); return tty_list; } inline bool all_tty_idle(time_t t) { static std::vector tty_list; struct stat sbuf; unsigned int i; if (tty_list.size()==0) tty_list=get_tty_list(); for (i=0; i= t) { return false; } } } return true; } #ifdef HAVE_UTMP_H inline bool user_idle(time_t t, struct utmp* u) { char tty[5 + sizeof u->ut_line + 1] = "/dev/"; unsigned int i; for (i=0; i < sizeof(u->ut_line); i++) { // clean up tty if garbled if (isalnum((int) u->ut_line[i]) || (u->ut_line[i]=='/')) { tty[i+5] = u->ut_line[i]; } else { tty[i+5] = '\0'; } } return device_idle(t, tty); } #if !defined(HAVE_SETUTENT) || !defined(HAVE_GETUTENT) static FILE *ufp = NULL; static struct utmp ut; // get next user login record // (this is defined on everything except BSD) // struct utmp *getutent() { if (ufp == NULL) { #if defined(UTMP_LOCATION) if ((ufp = fopen(UTMP_LOCATION, "r")) == NULL) { #elif defined(UTMP_FILE) if ((ufp = fopen(UTMP_FILE, "r")) == NULL) { #elif defined(_PATH_UTMP) if ((ufp = fopen(_PATH_UTMP, "r")) == NULL) { #else if ((ufp = fopen("/etc/utmp", "r")) == NULL) { #endif return((struct utmp *)NULL); } } do { if (fread((char *)&ut, sizeof(ut), 1, ufp) != 1) { return((struct utmp *)NULL); } } while (ut.ut_name[0] == 0); return(&ut); } void setutent() { if (ufp != NULL) rewind(ufp); } #endif // scan list of logged-in users, and see if they're all idle // inline bool all_logins_idle(time_t t) { struct utmp* u; setutent(); while ((u = getutent()) != NULL) { if (!user_idle(t, u)) { return false; } } return true; } #endif // HAVE_UTMP_H #ifdef __APPLE__ bool HOST_INFO::users_idle( bool check_all_logins, double idle_time_to_run, double *actual_idle_time ) { double idleTime = 0; if (gEventHandle) { idleTime = NXIdleTime(gEventHandle); } else { // Initialize Mac OS X idle time measurement / idle detection // Do this here because NXOpenEventStatus() may not be available // immediately on system startup when running as a deaemon. gEventHandle = NXOpenEventStatus(); } if (actual_idle_time) { *actual_idle_time = idleTime; } return (idleTime > (60 * idle_time_to_run)); } #else // ! __APPLE__ #if LINUX_LIKE_SYSTEM bool interrupts_idle(time_t t) { static FILE *ifp = NULL; static long irq_count[256]; static time_t last_irq = time(NULL); char line[256]; int i = 0; long ccount = 0; if (ifp == NULL) { if ((ifp = fopen("/proc/interrupts", "r")) == NULL) { return true; } } rewind(ifp); while (fgets(line, sizeof(line), ifp)) { // Check for mouse, keyboard and PS/2 devices. if (strcasestr(line, "mouse") != NULL || strcasestr(line, "keyboard") != NULL || strcasestr(line, "i8042") != NULL) { // If any IRQ count changed, update last_irq. if (sscanf(line, "%d: %ld", &i, &ccount) == 2 && irq_count[i] != ccount) { last_irq = time(NULL); irq_count[i] = ccount; } } } return last_irq < t; } #endif bool HOST_INFO::users_idle(bool check_all_logins, double idle_time_to_run) { time_t idle_time = time(0) - (long) (60 * idle_time_to_run); #ifdef HAVE_UTMP_H if (check_all_logins) { if (!all_logins_idle(idle_time)) return false; } #endif if (!all_tty_idle(idle_time)) return false; #if LINUX_LIKE_SYSTEM // Check /proc/interrupts to detect keyboard or mouse activity. if (!interrupts_idle(idle_time)) return false; #else // We should find out which of the following are actually relevant // on which systems (if any) // if (!device_idle(idle_time, "/dev/mouse")) return false; // solaris, linux if (!device_idle(idle_time, "/dev/input/mice")) return false; if (!device_idle(idle_time, "/dev/kbd")) return false; // solaris #endif return true; } #endif // ! __APPLE__ const char *BOINC_RCSID_2cf92d205b = "$Id$";