Since this buffer is used as an array of 12 32bits integers, and C++ string expect a NULL-terminated C-string, we need an extra char to ensure last one is always NULL. See D8906. Thanks to @brecht for noting this one too.
		
			
				
	
	
		
			197 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			197 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 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(
 | 
						|
    /* 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
 | 
						|
  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);
 | 
						|
}
 |