mirror of
https://github.com/dart-lang/sdk
synced 2024-11-02 12:24:24 +00:00
f407419d0a
This relands https://dart-review.googlesource.com/c/sdk/+/205633 but without renaming TARGET_OS_IPHONE to DART_TARGET_OS_IPHONE. It also changes uses of TARGET_OS_IOS to DART_TARGET_OS_MACOS_IOS to be consistent with the rest of the VM. TargetConditionals.h for XCode 13 defines several TARGET_OS_* preprocessor symbols that confuse the Dart build. There is probably a more targeted fix for this, but renaming the symbols that Dart uses will also prevent this problem if more symbols are added to the platform headers in the future. See: https://github.com/dart-lang/sdk/issues/46499 TEST=It builds. Change-Id: Ie775c19dd23cfdf5f65e5ebc6ee4ec3a561676fa Reviewed-on: https://dart-review.googlesource.com/c/sdk/+/205860 Commit-Queue: Zach Anderson <zra@google.com> Reviewed-by: Alexander Aprelev <aam@google.com>
125 lines
4.3 KiB
C++
125 lines
4.3 KiB
C++
// Copyright (c) 2020, 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 "bin/exe_utils.h"
|
|
|
|
#include "bin/directory.h"
|
|
#include "bin/file.h"
|
|
#include "bin/platform.h"
|
|
#include "platform/utils.h"
|
|
|
|
namespace dart {
|
|
namespace bin {
|
|
|
|
static bool StartsWithPathSeparator(const char* path,
|
|
const char* sep,
|
|
intptr_t sep_length) {
|
|
return (strncmp(path, sep, sep_length) == 0
|
|
#if defined(DART_HOST_OS_WINDOWS)
|
|
// TODO(aam): GetExecutableName doesn't work reliably on Windows,
|
|
|| *path == '/'
|
|
#endif
|
|
); // NOLINT
|
|
}
|
|
|
|
// Returns the directory portion of a given path.
|
|
//
|
|
// If dir is NULL, the result must be freed by the caller. Otherwise, the
|
|
// result is copied into dir.
|
|
static char* GetDirectoryFromPath(const char* path, char* dir) {
|
|
const char* sep = File::PathSeparator();
|
|
const intptr_t sep_length = strlen(sep);
|
|
intptr_t path_len = strlen(path);
|
|
|
|
for (intptr_t i = path_len - 1; i >= 0; --i) {
|
|
const char* str = path + i;
|
|
if (StartsWithPathSeparator(str, sep, sep_length)) {
|
|
if (dir != nullptr) {
|
|
strncpy(dir, path, i);
|
|
dir[i] = '\0';
|
|
return dir;
|
|
} else {
|
|
return Utils::StrNDup(path, i + 1);
|
|
}
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
// Returns the file portion of a given path. Returned string is either
|
|
// `path` if no path separators are found or `path + separator_loc + sep_length`
|
|
// if a separator is found.
|
|
static const char* GetFileNameFromPath(const char* path) {
|
|
const char* sep = File::PathSeparator();
|
|
const intptr_t sep_length = strlen(sep);
|
|
intptr_t path_len = strlen(path);
|
|
|
|
for (intptr_t i = path_len - 1; i >= 0; --i) {
|
|
const char* str = path + i;
|
|
if (StartsWithPathSeparator(str, sep, sep_length)) {
|
|
return str + sep_length;
|
|
}
|
|
}
|
|
// No path separators, assume that path is a file name.
|
|
return path;
|
|
}
|
|
|
|
Utils::CStringUniquePtr EXEUtils::GetDirectoryPrefixFromExeName() {
|
|
const char* name = nullptr;
|
|
const int kTargetSize = PATH_MAX;
|
|
char target[kTargetSize];
|
|
intptr_t target_size =
|
|
Platform::ResolveExecutablePathInto(target, kTargetSize);
|
|
if (target_size > 0 && target_size < kTargetSize - 1) {
|
|
target[target_size] = 0;
|
|
name = target;
|
|
}
|
|
if (name == nullptr) {
|
|
name = Platform::GetExecutableName();
|
|
target_size = strlen(name);
|
|
ASSERT(target_size < kTargetSize);
|
|
}
|
|
Namespace* namespc = Namespace::Create(Namespace::Default());
|
|
char* result;
|
|
if (File::GetType(namespc, name, false) == File::kIsLink) {
|
|
char dir_path[kTargetSize];
|
|
// cwd is currently wherever we launched from, so set the cwd to the
|
|
// directory of the symlink while we try and resolve it. If we don't
|
|
// do this, we won't be able to properly resolve relative paths.
|
|
auto initial_dir_path =
|
|
Utils::CreateCStringUniquePtr(Directory::CurrentNoScope());
|
|
// We might run into symlinks of symlinks, so make sure we follow the
|
|
// links all the way. See https://github.com/dart-lang/sdk/issues/41057 for
|
|
// an example where this happens with brew on MacOS.
|
|
do {
|
|
Directory::SetCurrent(namespc, GetDirectoryFromPath(name, dir_path));
|
|
// Resolve the link without creating Dart scope String.
|
|
name = File::LinkTarget(namespc, GetFileNameFromPath(name), target,
|
|
kTargetSize);
|
|
if (name == nullptr) {
|
|
return Utils::CreateCStringUniquePtr(Utils::StrDup(""));
|
|
}
|
|
} while (File::GetType(namespc, name, false) == File::kIsLink);
|
|
target_size = strlen(name);
|
|
|
|
char absolute_path[kTargetSize];
|
|
|
|
// Get the absolute path after we've resolved all the symlinks and before
|
|
// we reset the cwd, otherwise path resolution will fail.
|
|
File::GetCanonicalPath(namespc, name, absolute_path, kTargetSize);
|
|
|
|
// Reset cwd to the original value.
|
|
Directory::SetCurrent(namespc, initial_dir_path.get());
|
|
|
|
result = GetDirectoryFromPath(absolute_path, nullptr);
|
|
} else {
|
|
result = GetDirectoryFromPath(target, nullptr);
|
|
}
|
|
namespc->Release();
|
|
return Utils::CreateCStringUniquePtr(result == nullptr ? Utils::StrDup("")
|
|
: result);
|
|
}
|
|
|
|
} // namespace bin
|
|
} // namespace dart
|