
This diff add supports for crash logs on windows for release builds. This can be toggled on/off with the `WITH_WINDOWS_PDB` cmake option. by default it is on. Things to take into consideration: Release builds are hightly optimized and the resulting backtraces can be wrong/misleading, take the backtrace as a general area where the problem resides rather than an exact location. By default we ship a minimized symbol file that can only resolve the function names. This was chosen to strike a balance between growth in size of the download vs functionality gained. If more detailed information is required such as source file + line number information a full pdb can be shipped by setting `WITH_WINDOWS_STRIPPED_PDB` to off. The Release in the title of this diff refers to the release build type, not the official blender releases. Initially this will only be enabled for nightly build bot versions of blender, official releases as of now will not ship with symbols. Differential Revision: https://developer.blender.org/D7520 Reviewed by: brecht
192 lines
4.6 KiB
C
192 lines
4.6 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 stackwalk 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(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
|
|
data[0] = data[1] = data[2] = data[3] = 0;
|
|
# endif
|
|
}
|
|
#endif
|
|
|
|
char *BLI_cpu_brand_string(void)
|
|
{
|
|
char buf[48] = {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);
|
|
}
|