randomenv.cpp
1 // Copyright (c) 2009-2010 Satoshi Nakamoto 2 // Copyright (c) 2009-present The Bitcoin Core developers 3 // Distributed under the MIT software license, see the accompanying 4 // file COPYING or http://www.opensource.org/licenses/mit-license.php. 5 6 #include <bitcoin-build-config.h> // IWYU pragma: keep 7 8 #include <randomenv.h> 9 10 #include <clientversion.h> 11 #include <compat/compat.h> 12 #include <compat/cpuid.h> 13 #include <crypto/sha512.h> 14 #include <span.h> 15 #include <support/cleanse.h> 16 #include <util/byte_units.h> 17 #include <util/time.h> 18 19 #include <algorithm> 20 #include <atomic> 21 #include <cstdint> 22 #include <cstring> 23 #include <chrono> 24 #include <climits> 25 #include <thread> 26 #include <vector> 27 28 #include <sys/types.h> // must go before a number of other headers 29 30 #ifdef WIN32 31 #include <windows.h> 32 #else 33 #include <fcntl.h> 34 #include <netinet/in.h> 35 #include <sys/resource.h> 36 #include <sys/socket.h> 37 #include <sys/stat.h> 38 #include <sys/time.h> 39 #include <sys/utsname.h> 40 #include <unistd.h> 41 #endif 42 #ifdef HAVE_IFADDRS 43 #include <ifaddrs.h> 44 #endif 45 #ifdef HAVE_SYSCTL 46 #include <sys/sysctl.h> 47 #if __has_include(<vm/vm_param.h>) 48 #include <vm/vm_param.h> 49 #endif 50 #if __has_include(<sys/resources.h>) 51 #include <sys/resources.h> 52 #endif 53 #if __has_include(<sys/vmmeter.h>) 54 #include <sys/vmmeter.h> 55 #endif 56 #endif 57 #if defined(HAVE_STRONG_GETAUXVAL) 58 #include <sys/auxv.h> 59 #endif 60 61 #if defined(__APPLE__) || \ 62 defined(__FreeBSD__) || \ 63 defined(__NetBSD__) || \ 64 defined(__OpenBSD__) || \ 65 defined(__illumos__) 66 extern char** environ; // Necessary on the above platforms 67 #endif 68 69 namespace { 70 71 /** Helper to easily feed data into a CSHA512. 72 * 73 * Note that this does not serialize the passed object (like stream.h's << operators do). 74 * Its raw memory representation is used directly. 75 */ 76 template<typename T> 77 CSHA512& operator<<(CSHA512& hasher, const T& data) { 78 static_assert(!std::is_same_v<std::decay_t<T>, char*>, "Calling operator<<(CSHA512, char*) is probably not what you want"); 79 static_assert(!std::is_same_v<std::decay_t<T>, unsigned char*>, "Calling operator<<(CSHA512, unsigned char*) is probably not what you want"); 80 static_assert(!std::is_same_v<std::decay_t<T>, const char*>, "Calling operator<<(CSHA512, const char*) is probably not what you want"); 81 static_assert(!std::is_same_v<std::decay_t<T>, const unsigned char*>, "Calling operator<<(CSHA512, const unsigned char*) is probably not what you want"); 82 hasher.Write((const unsigned char*)&data, sizeof(data)); 83 return hasher; 84 } 85 86 #ifndef WIN32 87 void AddSockaddr(CSHA512& hasher, const struct sockaddr *addr) 88 { 89 if (addr == nullptr) return; 90 switch (addr->sa_family) { 91 case AF_INET: 92 hasher.Write((const unsigned char*)addr, sizeof(sockaddr_in)); 93 break; 94 case AF_INET6: 95 hasher.Write((const unsigned char*)addr, sizeof(sockaddr_in6)); 96 break; 97 default: 98 hasher.Write((const unsigned char*)&addr->sa_family, sizeof(addr->sa_family)); 99 } 100 } 101 102 void AddFile(CSHA512& hasher, const char *path) 103 { 104 struct stat sb = {}; 105 int f = open(path, O_RDONLY); 106 size_t total = 0; 107 if (f != -1) { 108 unsigned char fbuf[4096]; 109 int n; 110 hasher.Write((const unsigned char*)&f, sizeof(f)); 111 if (fstat(f, &sb) == 0) hasher << sb; 112 do { 113 n = read(f, fbuf, sizeof(fbuf)); 114 if (n > 0) hasher.Write(fbuf, n); 115 total += n; 116 /* not bothering with EINTR handling. */ 117 } while (n == sizeof(fbuf) && total < 1_MiB); // Read only the first 1 Mbyte 118 close(f); 119 } 120 } 121 122 void AddPath(CSHA512& hasher, const char *path) 123 { 124 struct stat sb = {}; 125 if (stat(path, &sb) == 0) { 126 hasher.Write((const unsigned char*)path, strlen(path) + 1); 127 hasher << sb; 128 } 129 } 130 #endif 131 132 #ifdef HAVE_SYSCTL 133 template<int... S> 134 void AddSysctl(CSHA512& hasher) 135 { 136 int CTL[sizeof...(S)] = {S...}; 137 unsigned char buffer[65536]; 138 size_t siz = 65536; 139 int ret = sysctl(CTL, sizeof...(S), buffer, &siz, nullptr, 0); 140 if (ret == 0 || (ret == -1 && errno == ENOMEM)) { 141 hasher << sizeof(CTL); 142 hasher.Write((const unsigned char*)CTL, sizeof(CTL)); 143 if (siz > sizeof(buffer)) siz = sizeof(buffer); 144 hasher << siz; 145 hasher.Write(buffer, siz); 146 } 147 } 148 #endif 149 150 #ifdef HAVE_GETCPUID 151 void inline AddCPUID(CSHA512& hasher, uint32_t leaf, uint32_t subleaf, uint32_t& ax, uint32_t& bx, uint32_t& cx, uint32_t& dx) 152 { 153 GetCPUID(leaf, subleaf, ax, bx, cx, dx); 154 hasher << leaf << subleaf << ax << bx << cx << dx; 155 } 156 157 void AddAllCPUID(CSHA512& hasher) 158 { 159 uint32_t ax, bx, cx, dx; 160 // Iterate over all standard leaves 161 AddCPUID(hasher, 0, 0, ax, bx, cx, dx); // Returns max leaf in ax 162 uint32_t max = ax; 163 for (uint32_t leaf = 1; leaf <= max && leaf <= 0xFF; ++leaf) { 164 uint32_t maxsub = 0; 165 for (uint32_t subleaf = 0; subleaf <= 0xFF; ++subleaf) { 166 AddCPUID(hasher, leaf, subleaf, ax, bx, cx, dx); 167 // Iterate subleafs for leaf values 4, 7, 11, 13 168 if (leaf == 4) { 169 if ((ax & 0x1f) == 0) break; 170 } else if (leaf == 7) { 171 if (subleaf == 0) maxsub = ax; 172 if (subleaf == maxsub) break; 173 } else if (leaf == 11) { 174 if ((cx & 0xff00) == 0) break; 175 } else if (leaf == 13) { 176 if (ax == 0 && bx == 0 && cx == 0 && dx == 0) break; 177 } else { 178 // For any other leaf, stop after subleaf 0. 179 break; 180 } 181 } 182 } 183 // Iterate over all extended leaves 184 AddCPUID(hasher, 0x80000000, 0, ax, bx, cx, dx); // Returns max extended leaf in ax 185 uint32_t ext_max = ax; 186 for (uint32_t leaf = 0x80000001; leaf <= ext_max && leaf <= 0x800000FF; ++leaf) { 187 AddCPUID(hasher, leaf, 0, ax, bx, cx, dx); 188 } 189 } 190 #endif 191 } // namespace 192 193 void RandAddDynamicEnv(CSHA512& hasher) 194 { 195 // Various clocks 196 #ifdef WIN32 197 FILETIME ftime; 198 GetSystemTimeAsFileTime(&ftime); 199 hasher << ftime; 200 #else 201 struct timespec ts = {}; 202 # ifdef CLOCK_MONOTONIC 203 clock_gettime(CLOCK_MONOTONIC, &ts); 204 hasher << ts; 205 # endif 206 # ifdef CLOCK_REALTIME 207 clock_gettime(CLOCK_REALTIME, &ts); 208 hasher << ts; 209 # endif 210 # ifdef CLOCK_BOOTTIME 211 clock_gettime(CLOCK_BOOTTIME, &ts); 212 hasher << ts; 213 # endif 214 // gettimeofday is available on all UNIX systems, but only has microsecond precision. 215 struct timeval tv = {}; 216 gettimeofday(&tv, nullptr); 217 hasher << tv; 218 #endif 219 // Probably redundant, but also use all the standard library clocks: 220 hasher << std::chrono::system_clock::now().time_since_epoch().count(); 221 hasher << std::chrono::steady_clock::now().time_since_epoch().count(); 222 hasher << std::chrono::high_resolution_clock::now().time_since_epoch().count(); 223 224 #ifndef WIN32 225 // Current resource usage. 226 struct rusage usage = {}; 227 if (getrusage(RUSAGE_SELF, &usage) == 0) hasher << usage; 228 #endif 229 230 #ifdef __linux__ 231 AddFile(hasher, "/proc/diskstats"); 232 AddFile(hasher, "/proc/vmstat"); 233 AddFile(hasher, "/proc/schedstat"); 234 AddFile(hasher, "/proc/zoneinfo"); 235 AddFile(hasher, "/proc/meminfo"); 236 AddFile(hasher, "/proc/softirqs"); 237 AddFile(hasher, "/proc/stat"); 238 AddFile(hasher, "/proc/self/schedstat"); 239 AddFile(hasher, "/proc/self/status"); 240 #endif 241 242 #ifdef HAVE_SYSCTL 243 # ifdef CTL_KERN 244 # if defined(KERN_PROC) && defined(KERN_PROC_ALL) 245 AddSysctl<CTL_KERN, KERN_PROC, KERN_PROC_ALL>(hasher); 246 # endif 247 # endif 248 # ifdef CTL_HW 249 # ifdef HW_DISKSTATS 250 AddSysctl<CTL_HW, HW_DISKSTATS>(hasher); 251 # endif 252 # endif 253 # ifdef CTL_VM 254 # ifdef VM_LOADAVG 255 AddSysctl<CTL_VM, VM_LOADAVG>(hasher); 256 # endif 257 # ifdef VM_TOTAL 258 AddSysctl<CTL_VM, VM_TOTAL>(hasher); 259 # endif 260 # ifdef VM_METER 261 AddSysctl<CTL_VM, VM_METER>(hasher); 262 # endif 263 # endif 264 #endif 265 266 // Stack and heap location 267 void* addr = malloc(4097); 268 hasher << &addr << addr; 269 free(addr); 270 } 271 272 void RandAddStaticEnv(CSHA512& hasher) 273 { 274 // Some compile-time static properties 275 hasher << (CHAR_MIN < 0) << sizeof(void*) << sizeof(long) << sizeof(int); 276 #if defined(__GNUC__) && defined(__GNUC_MINOR__) && defined(__GNUC_PATCHLEVEL__) 277 hasher << __GNUC__ << __GNUC_MINOR__ << __GNUC_PATCHLEVEL__; 278 #endif 279 #ifdef _MSC_VER 280 hasher << _MSC_VER; 281 #endif 282 hasher << __cplusplus; 283 #ifdef _XOPEN_VERSION 284 hasher << _XOPEN_VERSION; 285 #endif 286 #ifdef __VERSION__ 287 const char* COMPILER_VERSION = __VERSION__; 288 hasher.Write((const unsigned char*)COMPILER_VERSION, strlen(COMPILER_VERSION) + 1); 289 #endif 290 291 // Bitcoin client version 292 hasher << CLIENT_VERSION; 293 294 #if defined(HAVE_STRONG_GETAUXVAL) 295 // Information available through getauxval() 296 # ifdef AT_HWCAP 297 hasher << getauxval(AT_HWCAP); 298 # endif 299 # ifdef AT_HWCAP2 300 hasher << getauxval(AT_HWCAP2); 301 # endif 302 # ifdef AT_RANDOM 303 const unsigned char* random_aux = (const unsigned char*)getauxval(AT_RANDOM); 304 if (random_aux) hasher.Write(random_aux, 16); 305 # endif 306 # ifdef AT_PLATFORM 307 const char* platform_str = (const char*)getauxval(AT_PLATFORM); 308 if (platform_str) hasher.Write((const unsigned char*)platform_str, strlen(platform_str) + 1); 309 # endif 310 # ifdef AT_EXECFN 311 const char* exec_str = (const char*)getauxval(AT_EXECFN); 312 if (exec_str) hasher.Write((const unsigned char*)exec_str, strlen(exec_str) + 1); 313 # endif 314 #endif // HAVE_STRONG_GETAUXVAL 315 316 #ifdef HAVE_GETCPUID 317 AddAllCPUID(hasher); 318 #endif 319 320 // Memory locations 321 hasher << &hasher << &RandAddStaticEnv << &malloc << &errno << &environ; 322 323 // Hostname 324 #ifdef WIN32 325 constexpr DWORD max_size = MAX_COMPUTERNAME_LENGTH + 1; 326 char hname[max_size]; 327 DWORD size = max_size; 328 if (GetComputerNameA(hname, &size) != 0) { 329 hasher.Write(UCharCast(hname), size); 330 } 331 #else 332 char hname[256]; 333 if (gethostname(hname, 256) == 0) { 334 hasher.Write((const unsigned char*)hname, strnlen(hname, 256)); 335 } 336 #endif 337 338 #ifdef HAVE_IFADDRS 339 // Network interfaces 340 struct ifaddrs *ifad = nullptr; 341 getifaddrs(&ifad); 342 struct ifaddrs *ifit = ifad; 343 while (ifit != nullptr) { 344 hasher.Write((const unsigned char*)&ifit, sizeof(ifit)); 345 hasher.Write((const unsigned char*)ifit->ifa_name, strlen(ifit->ifa_name) + 1); 346 hasher.Write((const unsigned char*)&ifit->ifa_flags, sizeof(ifit->ifa_flags)); 347 AddSockaddr(hasher, ifit->ifa_addr); 348 AddSockaddr(hasher, ifit->ifa_netmask); 349 AddSockaddr(hasher, ifit->ifa_dstaddr); 350 ifit = ifit->ifa_next; 351 } 352 freeifaddrs(ifad); 353 #endif 354 355 #ifndef WIN32 356 // UNIX kernel information 357 struct utsname name; 358 if (uname(&name) != -1) { 359 hasher.Write((const unsigned char*)&name.sysname, strlen(name.sysname) + 1); 360 hasher.Write((const unsigned char*)&name.nodename, strlen(name.nodename) + 1); 361 hasher.Write((const unsigned char*)&name.release, strlen(name.release) + 1); 362 hasher.Write((const unsigned char*)&name.version, strlen(name.version) + 1); 363 hasher.Write((const unsigned char*)&name.machine, strlen(name.machine) + 1); 364 } 365 366 /* Path and filesystem provided data */ 367 AddPath(hasher, "/"); 368 AddPath(hasher, "."); 369 AddPath(hasher, "/tmp"); 370 AddPath(hasher, "/home"); 371 AddPath(hasher, "/proc"); 372 #ifdef __linux__ 373 AddFile(hasher, "/proc/cmdline"); 374 AddFile(hasher, "/proc/cpuinfo"); 375 AddFile(hasher, "/proc/version"); 376 #endif 377 AddFile(hasher, "/etc/passwd"); 378 AddFile(hasher, "/etc/group"); 379 AddFile(hasher, "/etc/hosts"); 380 AddFile(hasher, "/etc/resolv.conf"); 381 AddFile(hasher, "/etc/timezone"); 382 AddFile(hasher, "/etc/localtime"); 383 #endif 384 385 // For MacOS/BSDs, gather data through sysctl instead of /proc. Not all of these 386 // will exist on every system. 387 #ifdef HAVE_SYSCTL 388 # ifdef CTL_HW 389 # ifdef HW_MACHINE 390 AddSysctl<CTL_HW, HW_MACHINE>(hasher); 391 # endif 392 # ifdef HW_MODEL 393 AddSysctl<CTL_HW, HW_MODEL>(hasher); 394 # endif 395 # ifdef HW_NCPU 396 AddSysctl<CTL_HW, HW_NCPU>(hasher); 397 # endif 398 # ifdef HW_PHYSMEM 399 AddSysctl<CTL_HW, HW_PHYSMEM>(hasher); 400 # endif 401 # ifdef HW_USERMEM 402 AddSysctl<CTL_HW, HW_USERMEM>(hasher); 403 # endif 404 # ifdef HW_MACHINE_ARCH 405 AddSysctl<CTL_HW, HW_MACHINE_ARCH>(hasher); 406 # endif 407 # ifdef HW_REALMEM 408 AddSysctl<CTL_HW, HW_REALMEM>(hasher); 409 # endif 410 # ifdef HW_CPU_FREQ 411 AddSysctl<CTL_HW, HW_CPU_FREQ>(hasher); 412 # endif 413 # ifdef HW_BUS_FREQ 414 AddSysctl<CTL_HW, HW_BUS_FREQ>(hasher); 415 # endif 416 # ifdef HW_CACHELINE 417 AddSysctl<CTL_HW, HW_CACHELINE>(hasher); 418 # endif 419 # endif 420 # ifdef CTL_KERN 421 # ifdef KERN_BOOTFILE 422 AddSysctl<CTL_KERN, KERN_BOOTFILE>(hasher); 423 # endif 424 # ifdef KERN_BOOTTIME 425 AddSysctl<CTL_KERN, KERN_BOOTTIME>(hasher); 426 # endif 427 # ifdef KERN_CLOCKRATE 428 AddSysctl<CTL_KERN, KERN_CLOCKRATE>(hasher); 429 # endif 430 # ifdef KERN_HOSTID 431 AddSysctl<CTL_KERN, KERN_HOSTID>(hasher); 432 # endif 433 # ifdef KERN_HOSTUUID 434 AddSysctl<CTL_KERN, KERN_HOSTUUID>(hasher); 435 # endif 436 # ifdef KERN_HOSTNAME 437 AddSysctl<CTL_KERN, KERN_HOSTNAME>(hasher); 438 # endif 439 # ifdef KERN_OSRELDATE 440 AddSysctl<CTL_KERN, KERN_OSRELDATE>(hasher); 441 # endif 442 # ifdef KERN_OSRELEASE 443 AddSysctl<CTL_KERN, KERN_OSRELEASE>(hasher); 444 # endif 445 # ifdef KERN_OSREV 446 AddSysctl<CTL_KERN, KERN_OSREV>(hasher); 447 # endif 448 # ifdef KERN_OSTYPE 449 AddSysctl<CTL_KERN, KERN_OSTYPE>(hasher); 450 # endif 451 # ifdef KERN_POSIX1 452 AddSysctl<CTL_KERN, KERN_OSREV>(hasher); 453 # endif 454 # ifdef KERN_VERSION 455 AddSysctl<CTL_KERN, KERN_VERSION>(hasher); 456 # endif 457 # endif 458 #endif 459 460 // Env variables 461 if (environ) { 462 for (size_t i = 0; environ[i]; ++i) { 463 hasher.Write((const unsigned char*)environ[i], strlen(environ[i])); 464 } 465 } 466 467 // Process, thread, user, session, group, ... ids. 468 #ifdef WIN32 469 hasher << GetCurrentProcessId() << GetCurrentThreadId(); 470 #else 471 hasher << getpid() << getppid() << getsid(0) << getpgid(0) << getuid() << geteuid() << getgid() << getegid(); 472 #endif 473 hasher << std::this_thread::get_id(); 474 }