blob: 082c1ce4becc632aa43ecd427cf5cb6e1411aefd [file] [log] [blame]
// Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
#include "platform/globals.h"
#if defined(DART_HOST_OS_MACOS)
#include "platform/utils.h"
#include "platform/utils_macos.h"
#include <errno.h> // NOLINT
#include <sys/utsname.h> // NOLINT
namespace dart {
char* Utils::StrNDup(const char* s, intptr_t n) {
// strndup has only been added to Mac OS X in 10.7. We are supplying
// our own copy here if needed.
#if !defined(__ENVIRONMENT_MAC_OS_X_VERSION_MIN_REQUIRED__) || \
__ENVIRONMENT_MAC_OS_X_VERSION_MIN_REQUIRED__ <= 1060
intptr_t len = strlen(s);
if ((n < 0) || (len < 0)) {
return nullptr;
}
if (n < len) {
len = n;
}
char* result = reinterpret_cast<char*>(malloc(len + 1));
result[len] = '\0';
return reinterpret_cast<char*>(memmove(result, s, len));
#else // !defined(__ENVIRONMENT_MAC_OS_X_VERSION_MIN_REQUIRED__) || ...
return strndup(s, n);
#endif // !defined(__ENVIRONMENT_MAC_OS_X_VERSION_MIN_REQUIRED__) || ...
}
char* Utils::StrDup(const char* s) {
return strdup(s);
}
intptr_t Utils::StrNLen(const char* s, intptr_t n) {
// strnlen has only been added to Mac OS X in 10.7. We are supplying
// our own copy here if needed.
#if !defined(__ENVIRONMENT_MAC_OS_X_VERSION_MIN_REQUIRED__) || \
__ENVIRONMENT_MAC_OS_X_VERSION_MIN_REQUIRED__ <= 1060
intptr_t len = 0;
while ((len <= n) && (*s != '\0')) {
s++;
len++;
}
return len;
#else // !defined(__ENVIRONMENT_MAC_OS_X_VERSION_MIN_REQUIRED__) || ...
return strnlen(s, n);
#endif // !defined(__ENVIRONMENT_MAC_OS_X_VERSION_MIN_REQUIRED__) || ...
}
int Utils::SNPrint(char* str, size_t size, const char* format, ...) {
va_list args;
va_start(args, format);
int retval = VSNPrint(str, size, format, args);
va_end(args);
return retval;
}
int Utils::VSNPrint(char* str, size_t size, const char* format, va_list args) {
int retval = vsnprintf(str, size, format, args);
if (retval < 0) {
FATAL("Fatal error in Utils::VSNPrint with format '%s'", format);
}
return retval;
}
int Utils::Close(int fildes) {
return close(fildes);
}
size_t Utils::Read(int filedes, void* buf, size_t nbyte) {
return read(filedes, buf, nbyte);
}
int Utils::Unlink(const char* path) {
return unlink(path);
}
namespace internal {
// Returns the running system's Darwin major version. Don't call this, it's
// an implementation detail and its result is meant to be cached by
// MacOSXMinorVersion.
int32_t DarwinMajorVersionInternal() {
// uname is implemented as a simple series of sysctl system calls to
// obtain the relevant data from the kernel. The data is
// compiled right into the kernel, so no threads or blocking or other
// funny business is necessary.
struct utsname uname_info;
if (uname(&uname_info) != 0) {
FATAL("Fatal error in DarwinMajorVersionInternal : invalid return uname");
return 0;
}
if (strcmp(uname_info.sysname, "Darwin") != 0) {
FATAL(
"Fatal error in DarwinMajorVersionInternal : unexpected uname"
" sysname '%s'",
uname_info.sysname);
return 0;
}
int32_t darwin_major_version = 0;
char* dot = strchr(uname_info.release, '.');
if (dot) {
errno = 0;
char* end_ptr = nullptr;
darwin_major_version = strtol(uname_info.release, &end_ptr, 10);
if (errno != 0 || (end_ptr == uname_info.release)) {
dot = nullptr;
}
}
if (!dot) {
FATAL(
"Fatal error in DarwinMajorVersionInternal :"
" could not parse uname release '%s'",
uname_info.release);
return 0;
}
return darwin_major_version;
}
// Returns the running system's Mac OS X version which matches the encoding
// of MAC_OS_X_VERSION_* defines in AvailabilityMacros.h
int32_t MacOSXVersionInternal() {
const int32_t darwin_major_version = DarwinMajorVersionInternal();
int32_t major_version;
int32_t minor_version;
if (darwin_major_version < 20) {
// For Mac OS X 10.* minor version is off by 4 from Darwin's major
// version, e.g. 5.* is v10.1.*, 6.* is v10.2.* and so on.
// Pretend that anything below Darwin v5 is just Mac OS X Cheetah (v10.0).
major_version = 10;
minor_version = Utils::Maximum(0, darwin_major_version - 4);
} else {
// Starting from Darwin v20 major version increment in lock-step:
// Darwin v20 - Mac OS X v11, Darwin v21 - Mac OS X v12, etc
major_version = (darwin_major_version - 9);
minor_version = 0;
}
// Caveat: MAC_OS_X_VERSION_* is encoded using decimal encoding.
// Starting at MAC_OS_X_VERSION_10_10 versions use 2 decimal digits for
// minor version and patch number.
const int32_t field_multiplier = (darwin_major_version < 14) ? 10 : 100;
const int32_t major_multiplier = field_multiplier * field_multiplier;
const int32_t minor_multiplier = field_multiplier;
return major_version * major_multiplier + minor_version * minor_multiplier;
}
int32_t MacOSXVersion() {
static int mac_os_x_version = MacOSXVersionInternal();
return mac_os_x_version;
}
} // namespace internal
namespace {
int32_t MacOSMinorVersion(int32_t version) {
// Caveat: MAC_OS_X_VERSION_* is encoded using decimal encoding.
// Starting at MAC_OS_X_VERSION_10_10 versions use 2 decimal digits for
// minor version and patch number.
const int32_t field_multiplier =
(version < MAC_OS_X_VERSION_10_10) ? 10 : 100;
return (version / field_multiplier) % field_multiplier;
}
int32_t MacOSMajorVersion(int32_t version) {
// Caveat: MAC_OS_X_VERSION_* is encoded using decimal encoding.
// Starting at MAC_OS_X_VERSION_10_10 versions use 2 decimal digits for
// minor version and patch number.
const int32_t field_multiplier =
(version < MAC_OS_X_VERSION_10_10) ? 10 : 100;
return version / (field_multiplier * field_multiplier);
}
} // namespace
char* CheckIsAtLeastMinRequiredMacOSVersion() {
const int32_t current_version = internal::MacOSXVersion();
if (current_version >= MAC_OS_X_VERSION_MIN_REQUIRED) {
return nullptr;
}
return Utils::SCreate(
"Current Mac OS X version %d.%d is lower than minimum supported version "
"%d.%d",
MacOSMajorVersion(current_version), MacOSMinorVersion(current_version),
MacOSMajorVersion(MAC_OS_X_VERSION_MIN_REQUIRED),
MacOSMinorVersion(MAC_OS_X_VERSION_MIN_REQUIRED));
}
} // namespace dart
#endif // defined(DART_HOST_OS_MACOS)