198 lines
4.7 KiB
C
198 lines
4.7 KiB
C
/*
|
|
* 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, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
/** \file
|
|
* \ingroup bli
|
|
*/
|
|
|
|
#include <limits.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
|
|
#include "BLI_math_base.h"
|
|
#include "BLI_string.h"
|
|
#include "BLI_system.h"
|
|
#include "BLI_utildefines.h"
|
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
/* for backtrace and gethostname/GetComputerName */
|
|
#if defined(WIN32)
|
|
# include <intrin.h>
|
|
|
|
# include "BLI_winstuff.h"
|
|
#else
|
|
# include <execinfo.h>
|
|
# include <unistd.h>
|
|
#endif
|
|
|
|
int BLI_cpu_support_sse2(void)
|
|
{
|
|
#if defined(__x86_64__) || defined(_M_X64)
|
|
/* x86_64 always has SSE2 instructions */
|
|
return 1;
|
|
#elif defined(__GNUC__) && defined(i386)
|
|
/* for GCC x86 we check cpuid */
|
|
unsigned int d;
|
|
__asm__(
|
|
"pushl %%ebx\n\t"
|
|
"cpuid\n\t"
|
|
"popl %%ebx\n\t"
|
|
: "=d"(d)
|
|
: "a"(1));
|
|
return (d & 0x04000000) != 0;
|
|
#elif (defined(_MSC_VER) && defined(_M_IX86))
|
|
/* also check cpuid for MSVC x86 */
|
|
unsigned int d;
|
|
__asm {
|
|
xor eax, eax
|
|
inc eax
|
|
push ebx
|
|
cpuid
|
|
pop ebx
|
|
mov d, edx
|
|
}
|
|
return (d & 0x04000000) != 0;
|
|
#else
|
|
return 0;
|
|
#endif
|
|
}
|
|
|
|
/* Windows stack-walk lives in system_win32.c */
|
|
#if !defined(_MSC_VER)
|
|
/**
|
|
* Write a backtrace into a file for systems which support it.
|
|
*/
|
|
void BLI_system_backtrace(FILE *fp)
|
|
{
|
|
/* ------------- */
|
|
/* Linux / Apple */
|
|
# if defined(__linux__) || defined(__APPLE__)
|
|
|
|
# define SIZE 100
|
|
void *buffer[SIZE];
|
|
int nptrs;
|
|
char **strings;
|
|
int i;
|
|
|
|
/* include a backtrace for good measure */
|
|
nptrs = backtrace(buffer, SIZE);
|
|
strings = backtrace_symbols(buffer, nptrs);
|
|
for (i = 0; i < nptrs; i++) {
|
|
fputs(strings[i], fp);
|
|
fputc('\n', fp);
|
|
}
|
|
|
|
free(strings);
|
|
# undef SIZE
|
|
|
|
# else
|
|
/* ------------------ */
|
|
/* non msvc/osx/linux */
|
|
(void)fp;
|
|
# endif
|
|
}
|
|
#endif
|
|
/* end BLI_system_backtrace */
|
|
|
|
/* NOTE: The code for CPU brand string is adopted from Cycles. */
|
|
|
|
#if !defined(_WIN32) || defined(FREE_WINDOWS)
|
|
static void __cpuid(
|
|
/* Cannot be const, because it is modified below.
|
|
* NOLINTNEXTLINE: readability-non-const-parameter. */
|
|
int data[4],
|
|
int selector)
|
|
{
|
|
# if defined(__x86_64__)
|
|
asm("cpuid" : "=a"(data[0]), "=b"(data[1]), "=c"(data[2]), "=d"(data[3]) : "a"(selector));
|
|
# elif defined(__i386__)
|
|
asm("pushl %%ebx \n\t"
|
|
"cpuid \n\t"
|
|
"movl %%ebx, %1 \n\t"
|
|
"popl %%ebx \n\t"
|
|
: "=a"(data[0]), "=r"(data[1]), "=c"(data[2]), "=d"(data[3])
|
|
: "a"(selector)
|
|
: "ebx");
|
|
# else
|
|
(void)selector;
|
|
data[0] = data[1] = data[2] = data[3] = 0;
|
|
# endif
|
|
}
|
|
#endif
|
|
|
|
char *BLI_cpu_brand_string(void)
|
|
{
|
|
char buf[49] = {0};
|
|
int result[4] = {0};
|
|
__cpuid(result, 0x80000000);
|
|
if (result[0] >= (int)0x80000004) {
|
|
__cpuid((int *)(buf + 0), 0x80000002);
|
|
__cpuid((int *)(buf + 16), 0x80000003);
|
|
__cpuid((int *)(buf + 32), 0x80000004);
|
|
char *brand = BLI_strdup(buf);
|
|
/* TODO(sergey): Make it a bit more presentable by removing trademark. */
|
|
return brand;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
int BLI_cpu_support_sse41(void)
|
|
{
|
|
int result[4], num;
|
|
__cpuid(result, 0);
|
|
num = result[0];
|
|
|
|
if (num >= 1) {
|
|
__cpuid(result, 0x00000001);
|
|
return (result[2] & ((int)1 << 19)) != 0;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void BLI_hostname_get(char *buffer, size_t bufsize)
|
|
{
|
|
#ifndef WIN32
|
|
if (gethostname(buffer, bufsize - 1) < 0) {
|
|
BLI_strncpy(buffer, "-unknown-", bufsize);
|
|
}
|
|
/* When gethostname() truncates, it doesn't guarantee the trailing \0. */
|
|
buffer[bufsize - 1] = '\0';
|
|
#else
|
|
DWORD bufsize_inout = bufsize;
|
|
if (!GetComputerName(buffer, &bufsize_inout)) {
|
|
strncpy(buffer, "-unknown-", bufsize);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
size_t BLI_system_memory_max_in_megabytes(void)
|
|
{
|
|
/* Maximum addressable bytes on this platform.
|
|
*
|
|
* NOTE: Due to the shift arithmetic this is a half of the memory. */
|
|
const size_t limit_bytes_half = (((size_t)1) << ((sizeof(size_t[8])) - 1));
|
|
/* Convert it to megabytes and return. */
|
|
return (limit_bytes_half >> 20) * 2;
|
|
}
|
|
|
|
int BLI_system_memory_max_in_megabytes_int(void)
|
|
{
|
|
const size_t limit_megabytes = BLI_system_memory_max_in_megabytes();
|
|
/* NOTE: The result will fit into integer. */
|
|
return (int)min_zz(limit_megabytes, (size_t)INT_MAX);
|
|
}
|