mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
Enhance the Remote / RemoteAST libraries with an error mechanism.
Nothing is producing meaningful errors yet, however.
This commit is contained in:
337
include/swift/Remote/Failure.h
Normal file
337
include/swift/Remote/Failure.h
Normal file
@@ -0,0 +1,337 @@
|
|||||||
|
//===--- Failure.h - Failure to access remote memory ------------*- C++ -*-===//
|
||||||
|
//
|
||||||
|
// This source file is part of the Swift.org open source project
|
||||||
|
//
|
||||||
|
// Copyright (c) 2014 - 2016 Apple Inc. and the Swift project authors
|
||||||
|
// Licensed under Apache License v2.0 with Runtime Library Exception
|
||||||
|
//
|
||||||
|
// See http://swift.org/LICENSE.txt for license information
|
||||||
|
// See http://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
|
||||||
|
//
|
||||||
|
//===----------------------------------------------------------------------===//
|
||||||
|
//
|
||||||
|
// This file provides a simple diagnostics library for the various
|
||||||
|
// operations for working with a remote process.
|
||||||
|
//
|
||||||
|
//===----------------------------------------------------------------------===//
|
||||||
|
|
||||||
|
#ifndef SWIFT_REMOTE_FAILURE_H
|
||||||
|
#define SWIFT_REMOTE_FAILURE_H
|
||||||
|
|
||||||
|
#include "swift/Remote/RemoteAddress.h"
|
||||||
|
|
||||||
|
#include "llvm/Support/Compiler.h"
|
||||||
|
#include <string>
|
||||||
|
#include <cstring>
|
||||||
|
#include <type_traits>
|
||||||
|
|
||||||
|
namespace swift {
|
||||||
|
namespace remote {
|
||||||
|
|
||||||
|
class Failure {
|
||||||
|
public:
|
||||||
|
/// An enum which unifies all of the failure kinds into a single namespace.
|
||||||
|
/// This is how kinds are stored internally.
|
||||||
|
enum class Kind {
|
||||||
|
#define FAILURE(KIND, TEXT, ARGTYS) KIND,
|
||||||
|
#include "swift/Remote/FailureKinds.def"
|
||||||
|
};
|
||||||
|
|
||||||
|
/// A bunch of enums for all the different kinds of failure.
|
||||||
|
/// Users should construct a Failure by passing Failure::KIND instead
|
||||||
|
/// of Failure::Kind::KIND.
|
||||||
|
#define FAILURE(KIND, TEXT, ARGTYS) \
|
||||||
|
enum KIND##_t { KIND = unsigned(Kind::KIND) };
|
||||||
|
#include "swift/Remote/FailureKinds.def"
|
||||||
|
|
||||||
|
private:
|
||||||
|
// A whole lot of template machinery to validate that the right argument
|
||||||
|
// types were passed:
|
||||||
|
|
||||||
|
// A sequence of types.
|
||||||
|
template <class... Tys> struct TypeBundle {};
|
||||||
|
|
||||||
|
// A way of turning (X, Y, Z) into a TypeBundle:
|
||||||
|
// typename TypeBundleForFunctionTypeParameters<void ARGS>::types
|
||||||
|
// Partially specialized at the end of this file.
|
||||||
|
template <class FnTy> struct TypeBundleForFunctionTypeParameters;
|
||||||
|
|
||||||
|
// A way of getting the expected TypeBundle of (nonce) argument types
|
||||||
|
// for a particular failure kind:
|
||||||
|
// typename ArgTypesForFailureKind<KindTy>::types
|
||||||
|
// Explicitly specialized for each kind type at the end of this file.
|
||||||
|
template <class KindTy> struct ArgTypesForFailureKind;
|
||||||
|
|
||||||
|
// Nonce expected argument types.
|
||||||
|
enum ArgType_String {};
|
||||||
|
enum ArgType_Address {};
|
||||||
|
|
||||||
|
// A predicate that decides whether the given argument type satisfies
|
||||||
|
// the given nonce expected argument type.
|
||||||
|
template <class Expected, class Actual> struct IsAcceptableArgType {
|
||||||
|
static constexpr bool value = false;
|
||||||
|
};
|
||||||
|
|
||||||
|
template <class... Ts>
|
||||||
|
static void validateFailureArgsRecursive(TypeBundle<Ts...> expected,
|
||||||
|
TypeBundle<> actual) {
|
||||||
|
static_assert(std::is_same<decltype(expected), decltype(actual)>::value,
|
||||||
|
"too few arguments to diagnostic");
|
||||||
|
}
|
||||||
|
|
||||||
|
template <class Expected, class... RestOfExpected,
|
||||||
|
class Actual, class... RestOfActual>
|
||||||
|
static void validateFailureArgsRecursive(
|
||||||
|
TypeBundle<Expected, RestOfExpected...> expected,
|
||||||
|
TypeBundle<Actual, RestOfActual...> actual) {
|
||||||
|
using SimplifiedActual = typename std::decay<Actual>::type;
|
||||||
|
static_assert(IsAcceptableArgType<Expected, SimplifiedActual>::value,
|
||||||
|
"argument does no match");
|
||||||
|
validateFailureArgsRecursive(TypeBundle<RestOfExpected...>(),
|
||||||
|
TypeBundle<RestOfActual...>());
|
||||||
|
}
|
||||||
|
|
||||||
|
public:
|
||||||
|
/// Validate that it's okay to construct a Failure with the given arguments.
|
||||||
|
template <class KindTy, class... ArgTys>
|
||||||
|
static void validateFailureArgs(KindTy ty, ArgTys &&... argTys) {
|
||||||
|
using ExpectedArgTypes = typename ArgTypesForFailureKind<KindTy>::types;
|
||||||
|
validateFailureArgsRecursive(ExpectedArgTypes(), TypeBundle<ArgTys...>());
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
static const char *getTextForKind(Kind kind) {
|
||||||
|
switch (kind) {
|
||||||
|
#define FAILURE(KIND, TEXT, ARGTYS) \
|
||||||
|
case Kind::KIND: return TEXT;
|
||||||
|
#include "swift/Remote/FailureKinds.def"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
union ArgStorage {
|
||||||
|
std::string String;
|
||||||
|
RemoteAddress Address;
|
||||||
|
|
||||||
|
ArgStorage() {}
|
||||||
|
ArgStorage(const ArgStorage &other) = delete;
|
||||||
|
ArgStorage &operator=(const ArgStorage &other) = delete;
|
||||||
|
~ArgStorage() {}
|
||||||
|
};
|
||||||
|
enum class ArgStorageKind : char {
|
||||||
|
None,
|
||||||
|
String, // std::string
|
||||||
|
Address, // RemoteAddress
|
||||||
|
};
|
||||||
|
|
||||||
|
enum {
|
||||||
|
MaxArgs = 4
|
||||||
|
};
|
||||||
|
|
||||||
|
Kind TheKind;
|
||||||
|
ArgStorageKind ArgKinds[MaxArgs];
|
||||||
|
ArgStorage Args[MaxArgs];
|
||||||
|
|
||||||
|
template <unsigned Index>
|
||||||
|
void initArgs() {
|
||||||
|
static_assert(Index <= MaxArgs, "index out of bounds");
|
||||||
|
for (unsigned i = Index; i < MaxArgs; ++i) {
|
||||||
|
ArgKinds[i] = ArgStorageKind::None;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
template <unsigned Index, class T, class... Ts>
|
||||||
|
void initArgs(T &&nextArg, Ts &&... restOfArgs) {
|
||||||
|
static_assert(Index < MaxArgs, "index out of bounds");
|
||||||
|
initArg(Index, std::forward<T>(nextArg));
|
||||||
|
initArgs<Index + 1>(std::forward<T>(restOfArgs)...);
|
||||||
|
}
|
||||||
|
|
||||||
|
void initArg(unsigned index, std::string &&string) {
|
||||||
|
ArgKinds[index] = ArgStorageKind::String;
|
||||||
|
new (&Args[index]) std::string(std::move(string));
|
||||||
|
}
|
||||||
|
|
||||||
|
void initArg(unsigned index, const std::string &string) {
|
||||||
|
ArgKinds[index] = ArgStorageKind::String;
|
||||||
|
new (&Args[index]) std::string(string);
|
||||||
|
}
|
||||||
|
|
||||||
|
void initArg(unsigned index, RemoteAddress address) {
|
||||||
|
ArgKinds[index] = ArgStorageKind::Address;
|
||||||
|
new (&Args[index]) RemoteAddress(address);
|
||||||
|
}
|
||||||
|
|
||||||
|
public:
|
||||||
|
template <class KindTy, class... Ts>
|
||||||
|
explicit Failure(KindTy kind, Ts &&...args) : TheKind(Kind(kind)) {
|
||||||
|
validateFailureArgs(kind, std::forward<Ts>(args)...);
|
||||||
|
initArgs<0>(std::forward<Ts>(args)...);
|
||||||
|
}
|
||||||
|
|
||||||
|
Failure(const Failure &other) : TheKind(other.TheKind) {
|
||||||
|
for (unsigned i = 0; i != MaxArgs; ++i) {
|
||||||
|
ArgKinds[i] = other.ArgKinds[i];
|
||||||
|
switch (ArgKinds[i]) {
|
||||||
|
case ArgStorageKind::None:
|
||||||
|
break;
|
||||||
|
case ArgStorageKind::String:
|
||||||
|
::new (&Args[i].String) std::string(other.Args[i].String);
|
||||||
|
break;
|
||||||
|
case ArgStorageKind::Address:
|
||||||
|
::new (&Args[i].Address) RemoteAddress(other.Args[i].Address);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Failure(Failure &&other) : TheKind(other.TheKind) {
|
||||||
|
for (unsigned i = 0; i != MaxArgs; ++i) {
|
||||||
|
ArgKinds[i] = other.ArgKinds[i];
|
||||||
|
switch (ArgKinds[i]) {
|
||||||
|
case ArgStorageKind::None:
|
||||||
|
break;
|
||||||
|
case ArgStorageKind::String:
|
||||||
|
::new (&Args[i].String) std::string(std::move(other.Args[i].String));
|
||||||
|
break;
|
||||||
|
case ArgStorageKind::Address:
|
||||||
|
::new (&Args[i].Address) RemoteAddress(std::move(other.Args[i].Address));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Failure &operator=(const Failure &other) {
|
||||||
|
this->~Failure();
|
||||||
|
::new (this) Failure(other);
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
Failure &operator=(Failure &&other) {
|
||||||
|
this->~Failure();
|
||||||
|
::new (this) Failure(std::move(other));
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
~Failure() {
|
||||||
|
for (unsigned i = 0; i != MaxArgs; ++i) {
|
||||||
|
switch (ArgKinds[i]) {
|
||||||
|
case ArgStorageKind::None:
|
||||||
|
break;
|
||||||
|
case ArgStorageKind::String:
|
||||||
|
Args[i].String.~basic_string();
|
||||||
|
break;
|
||||||
|
case ArgStorageKind::Address:
|
||||||
|
Args[i].Address.~RemoteAddress();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Return the kind of failure.
|
||||||
|
Kind getKind() const { return TheKind; }
|
||||||
|
|
||||||
|
/// Return the number of arguments to the failure.
|
||||||
|
unsigned getNumArgs() const {
|
||||||
|
unsigned i = 0;
|
||||||
|
for (; i != MaxArgs; ++i) {
|
||||||
|
// Stop at the first missing argument.
|
||||||
|
if (ArgKinds[i] == ArgStorageKind::None)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Render the failure as an error message.
|
||||||
|
std::string render() const {
|
||||||
|
std::string result;
|
||||||
|
|
||||||
|
const char *text = getTextForKind(TheKind);
|
||||||
|
while (const char *next = std::strchr(text, '%')) {
|
||||||
|
// Append everything we just skipped over.
|
||||||
|
result.append(text, next - text);
|
||||||
|
|
||||||
|
// Skip the '%'.
|
||||||
|
next++;
|
||||||
|
|
||||||
|
// Do something based on the character after '%'.
|
||||||
|
char c = *next++;
|
||||||
|
if (c == '%') {
|
||||||
|
result += c;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
assert('0' <= c && c <= '9');
|
||||||
|
unsigned argIndex = c - '0';
|
||||||
|
assert(argIndex < MaxArgs);
|
||||||
|
|
||||||
|
switch (ArgKinds[argIndex]) {
|
||||||
|
case ArgStorageKind::None:
|
||||||
|
LLVM_BUILTIN_UNREACHABLE;
|
||||||
|
|
||||||
|
// Stringize a string argument by just appending it.
|
||||||
|
case ArgStorageKind::String:
|
||||||
|
result += Args[argIndex].String;
|
||||||
|
continue;
|
||||||
|
|
||||||
|
// Stringize an address argument as a hex number. We assume that an
|
||||||
|
// address less than 2^32 is for a 32-bit target and accordingly
|
||||||
|
// only print 8 digits for it.
|
||||||
|
//
|
||||||
|
// It is really silly to provide our own hex-stringization here,
|
||||||
|
// but swift::remote is supposed to be a minimal, header-only library.
|
||||||
|
case ArgStorageKind::Address: {
|
||||||
|
result += '0';
|
||||||
|
result += 'x';
|
||||||
|
uint64_t address = Args[argIndex].Address.getAddressData();
|
||||||
|
unsigned max = ((address >> 32) != 0 ? 16 : 8);
|
||||||
|
for (unsigned i = 0; i != max; ++i) {
|
||||||
|
result += "0123456789abcdef"[(address >> (max - 1 - i) * 4) & 0xF];
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
LLVM_BUILTIN_UNREACHABLE;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Append the rest of the string.
|
||||||
|
result.append(text);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
template <class... Tys>
|
||||||
|
struct Failure::TypeBundleForFunctionTypeParameters<void(Tys...)> {
|
||||||
|
using types = TypeBundle<Tys...>;
|
||||||
|
};
|
||||||
|
|
||||||
|
template <class... Tys> class TypeBundle {};
|
||||||
|
|
||||||
|
#define FAILURE(KIND, TEXT, ARGTYS) \
|
||||||
|
template <> struct Failure::ArgTypesForFailureKind<Failure::KIND##_t> { \
|
||||||
|
using String = ArgType_String; \
|
||||||
|
using Address = ArgType_Address; \
|
||||||
|
using types = TypeBundleForFunctionTypeParameters<void ARGTYS>::types; \
|
||||||
|
};
|
||||||
|
#include "swift/Remote/FailureKinds.def"
|
||||||
|
|
||||||
|
template <>
|
||||||
|
struct Failure::IsAcceptableArgType<Failure::ArgType_String, const char *> {
|
||||||
|
static constexpr bool value = true;
|
||||||
|
};
|
||||||
|
|
||||||
|
template <>
|
||||||
|
struct Failure::IsAcceptableArgType<Failure::ArgType_String, std::string> {
|
||||||
|
static constexpr bool value = true;
|
||||||
|
};
|
||||||
|
|
||||||
|
template <>
|
||||||
|
struct Failure::IsAcceptableArgType<Failure::ArgType_Address, RemoteAddress> {
|
||||||
|
static constexpr bool value = true;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // end namespace remote
|
||||||
|
} // end namespace swift
|
||||||
|
|
||||||
|
#endif // SWIFT_REMOTE_FAILURE_H
|
||||||
|
|
||||||
19
include/swift/Remote/FailureKinds.def
Normal file
19
include/swift/Remote/FailureKinds.def
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
//===--- FailureKinds.def - Remote failure definitions ----------*- C++ -*-===//
|
||||||
|
//
|
||||||
|
// This source file is part of the Swift.org open source project
|
||||||
|
//
|
||||||
|
// Copyright (c) 2014 - 2016 Apple Inc. and the Swift project authors
|
||||||
|
// Licensed under Apache License v2.0 with Runtime Library Exception
|
||||||
|
//
|
||||||
|
// See http://swift.org/LICENSE.txt for license information
|
||||||
|
// See http://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
|
||||||
|
//
|
||||||
|
//===----------------------------------------------------------------------===//
|
||||||
|
|
||||||
|
// FAILURE(KIND, TEXT, ARGTYS)
|
||||||
|
|
||||||
|
FAILURE(Unknown, "an unknown failure occurred", ())
|
||||||
|
FAILURE(Memory, "an unknown failure occurred while reading %0 at address %1",
|
||||||
|
(String, Address))
|
||||||
|
|
||||||
|
#undef FAILURE
|
||||||
@@ -18,35 +18,13 @@
|
|||||||
#ifndef SWIFT_REMOTE_MEMORYREADER_H
|
#ifndef SWIFT_REMOTE_MEMORYREADER_H
|
||||||
#define SWIFT_REMOTE_MEMORYREADER_H
|
#define SWIFT_REMOTE_MEMORYREADER_H
|
||||||
|
|
||||||
#include <cstdint>
|
#include "swift/Remote/RemoteAddress.h"
|
||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
namespace swift {
|
namespace swift {
|
||||||
namespace remote {
|
namespace remote {
|
||||||
|
|
||||||
/// An abstract address in the remote process's address space.
|
|
||||||
class RemoteAddress {
|
|
||||||
uint64_t Data;
|
|
||||||
public:
|
|
||||||
explicit RemoteAddress(const void *localPtr)
|
|
||||||
: Data(reinterpret_cast<uintptr_t>(localPtr)) {}
|
|
||||||
|
|
||||||
explicit RemoteAddress(uint64_t addressData) : Data(addressData) {}
|
|
||||||
|
|
||||||
explicit operator bool() const {
|
|
||||||
return Data != 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <class T>
|
|
||||||
const T *getLocalPointer() const {
|
|
||||||
return reinterpret_cast<const T*>(static_cast<uintptr_t>(Data));
|
|
||||||
}
|
|
||||||
|
|
||||||
uint64_t getAddressData() const {
|
|
||||||
return Data;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
/// An abstract interface for reading memory.
|
/// An abstract interface for reading memory.
|
||||||
///
|
///
|
||||||
/// This abstraction presents memory as if it were a read-only
|
/// This abstraction presents memory as if it were a read-only
|
||||||
|
|||||||
53
include/swift/Remote/RemoteAddress.h
Normal file
53
include/swift/Remote/RemoteAddress.h
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
//===--- RemoteAddress.h - Address of remote memory -------------*- C++ -*-===//
|
||||||
|
//
|
||||||
|
// This source file is part of the Swift.org open source project
|
||||||
|
//
|
||||||
|
// Copyright (c) 2014 - 2016 Apple Inc. and the Swift project authors
|
||||||
|
// Licensed under Apache License v2.0 with Runtime Library Exception
|
||||||
|
//
|
||||||
|
// See http://swift.org/LICENSE.txt for license information
|
||||||
|
// See http://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
|
||||||
|
//
|
||||||
|
//===----------------------------------------------------------------------===//
|
||||||
|
//
|
||||||
|
// This file defines the RemoteAddress type, which abstracts over an
|
||||||
|
// address in a remote process.
|
||||||
|
//
|
||||||
|
//===----------------------------------------------------------------------===//
|
||||||
|
|
||||||
|
#ifndef SWIFT_REMOTE_REMOTEADDRESS_H
|
||||||
|
#define SWIFT_REMOTE_REMOTEADDRESS_H
|
||||||
|
|
||||||
|
#include <cstdint>
|
||||||
|
|
||||||
|
namespace swift {
|
||||||
|
namespace remote {
|
||||||
|
|
||||||
|
/// An abstract address in the remote process's address space.
|
||||||
|
class RemoteAddress {
|
||||||
|
uint64_t Data;
|
||||||
|
public:
|
||||||
|
explicit RemoteAddress(const void *localPtr)
|
||||||
|
: Data(reinterpret_cast<uintptr_t>(localPtr)) {}
|
||||||
|
|
||||||
|
explicit RemoteAddress(uint64_t addressData) : Data(addressData) {}
|
||||||
|
|
||||||
|
explicit operator bool() const {
|
||||||
|
return Data != 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
template <class T>
|
||||||
|
const T *getLocalPointer() const {
|
||||||
|
return reinterpret_cast<const T*>(static_cast<uintptr_t>(Data));
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t getAddressData() const {
|
||||||
|
return Data;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
} // end namespace remote
|
||||||
|
} // end namespace swift
|
||||||
|
|
||||||
|
#endif // SWIFT_REMOTE_REMOTEADDRESS_H
|
||||||
|
|
||||||
@@ -19,6 +19,7 @@
|
|||||||
#ifndef SWIFT_REMOTEAST_REMOTEAST_H
|
#ifndef SWIFT_REMOTEAST_REMOTEAST_H
|
||||||
#define SWIFT_REMOTEAST_REMOTEAST_H
|
#define SWIFT_REMOTEAST_REMOTEAST_H
|
||||||
|
|
||||||
|
#include "swift/Remote/Failure.h"
|
||||||
#include "swift/Remote/MemoryReader.h"
|
#include "swift/Remote/MemoryReader.h"
|
||||||
#include "swift/Basic/LLVM.h"
|
#include "swift/Basic/LLVM.h"
|
||||||
#include "llvm/ADT/Optional.h"
|
#include "llvm/ADT/Optional.h"
|
||||||
@@ -33,6 +34,107 @@ class Type;
|
|||||||
|
|
||||||
namespace remoteAST {
|
namespace remoteAST {
|
||||||
|
|
||||||
|
template <class T>
|
||||||
|
class Result {
|
||||||
|
union Storage {
|
||||||
|
remote::Failure Failure;
|
||||||
|
T Success;
|
||||||
|
|
||||||
|
Storage() {}
|
||||||
|
Storage(const Storage &) = delete;
|
||||||
|
Storage &operator=(const Storage &) = delete;
|
||||||
|
~Storage() {}
|
||||||
|
};
|
||||||
|
|
||||||
|
Storage S;
|
||||||
|
bool IsSuccess;
|
||||||
|
|
||||||
|
Result(bool isSuccess) {}
|
||||||
|
|
||||||
|
public:
|
||||||
|
/*implicit*/ Result(const T &value) : IsSuccess(true) {
|
||||||
|
new (&S.Success) T(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*implicit*/ Result(T &&value) : IsSuccess(true) {
|
||||||
|
new (&S.Success) T(std::move(value));
|
||||||
|
}
|
||||||
|
|
||||||
|
/*implicit*/ Result(remote::Failure &&failure) : IsSuccess(false) {
|
||||||
|
new (&S.Failure) remote::Failure(std::move(failure));
|
||||||
|
}
|
||||||
|
|
||||||
|
Result(const Result &other) : IsSuccess(other.IsSuccess) {
|
||||||
|
if (IsSuccess) {
|
||||||
|
::new (&S.Success) T(other.S.Success);
|
||||||
|
} else {
|
||||||
|
::new (&S.Failure) remote::Failure(other.S.Failure);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Result(Result &&other) : IsSuccess(other.IsSuccess) {
|
||||||
|
if (IsSuccess) {
|
||||||
|
::new (&S.Success) T(std::move(other.S.Success));
|
||||||
|
} else {
|
||||||
|
::new (&S.Failure) remote::Failure(std::move(other.S.Failure));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Result &operator=(const Result &other) {
|
||||||
|
this->~Result();
|
||||||
|
::new (this) Result(other);
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
Result &operator=(Result &&other) {
|
||||||
|
this->~Result();
|
||||||
|
::new (this) Result(std::move(other));
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
~Result() {
|
||||||
|
if (IsSuccess) {
|
||||||
|
S.Success.~T();
|
||||||
|
} else {
|
||||||
|
S.Failure.~Failure();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
template <class... ArgTys>
|
||||||
|
static Result emplaceSuccess(ArgTys &&...args) {
|
||||||
|
Result result(true);
|
||||||
|
::new (&result.S.Success) T(std::forward<ArgTys>(args)...);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
template <class KindTy, class... ArgTys>
|
||||||
|
static Result emplaceFailure(KindTy kind, ArgTys &&...args) {
|
||||||
|
Result result(false);
|
||||||
|
::new (&result.S.Failure)
|
||||||
|
remote::Failure(kind, std::forward<ArgTys>(args)...);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
explicit operator bool() const { return isSuccess(); }
|
||||||
|
bool isSuccess() const { return IsSuccess; }
|
||||||
|
bool isFailure() const { return !IsSuccess; }
|
||||||
|
|
||||||
|
const remote::Failure &getFailure() const {
|
||||||
|
assert(isFailure());
|
||||||
|
return S.Failure;
|
||||||
|
}
|
||||||
|
|
||||||
|
const T &getValue() const & {
|
||||||
|
assert(isSuccess());
|
||||||
|
return S.Success;
|
||||||
|
}
|
||||||
|
|
||||||
|
T &&getValue() && {
|
||||||
|
assert(isSuccess());
|
||||||
|
return std::move(S.Success);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
/// A context for performing an operation relating the remote process with
|
/// A context for performing an operation relating the remote process with
|
||||||
/// the AST. This may be discarded and recreated at any time without danger,
|
/// the AST. This may be discarded and recreated at any time without danger,
|
||||||
/// but reusing a context across multiple calls may allow some redundant work
|
/// but reusing a context across multiple calls may allow some redundant work
|
||||||
@@ -65,16 +167,12 @@ public:
|
|||||||
|
|
||||||
/// Given an address which is supposedly of type metadata, try to
|
/// Given an address which is supposedly of type metadata, try to
|
||||||
/// resolve it to a specific type in the local AST.
|
/// resolve it to a specific type in the local AST.
|
||||||
///
|
Result<Type> getTypeForRemoteTypeMetadata(remote::RemoteAddress address);
|
||||||
/// This may fail by returning a null type.
|
|
||||||
Type getTypeForRemoteTypeMetadata(remote::RemoteAddress address);
|
|
||||||
|
|
||||||
/// Given an address which is supposedly of a nominal type descriptor,
|
/// Given an address which is supposedly of a nominal type descriptor,
|
||||||
/// try to resolve it to a specific nominal type declaration in the
|
/// try to resolve it to a specific nominal type declaration in the
|
||||||
/// local AST.
|
/// local AST.
|
||||||
///
|
Result<NominalTypeDecl *>
|
||||||
/// This may fail by returning null.
|
|
||||||
NominalTypeDecl *
|
|
||||||
getDeclForRemoteNominalTypeDescriptor(remote::RemoteAddress address);
|
getDeclForRemoteNominalTypeDescriptor(remote::RemoteAddress address);
|
||||||
|
|
||||||
/// Given a type in the local AST, try to resolve the offset of its
|
/// Given a type in the local AST, try to resolve the offset of its
|
||||||
@@ -83,7 +181,7 @@ public:
|
|||||||
/// This may fail by returning an empty optional. Failure may indicate
|
/// This may fail by returning an empty optional. Failure may indicate
|
||||||
/// that an offset for the property could not be resolved, or it may
|
/// that an offset for the property could not be resolved, or it may
|
||||||
/// simply indicate that the property has a non-zero offset.
|
/// simply indicate that the property has a non-zero offset.
|
||||||
Optional<uint64_t> getOffsetForProperty(Type type, StringRef propertyName);
|
Result<uint64_t> getOffsetForProperty(Type type, StringRef propertyName);
|
||||||
};
|
};
|
||||||
|
|
||||||
} // end namespace remoteAST
|
} // end namespace remoteAST
|
||||||
|
|||||||
@@ -506,26 +506,33 @@ public:
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Type getTypeForRemoteTypeMetadata(RemoteAddress address) {
|
Result<Type> getTypeForRemoteTypeMetadata(RemoteAddress address) {
|
||||||
|
Type result;
|
||||||
if (Is32) {
|
if (Is32) {
|
||||||
return Reader32->readTypeFromMetadata(address.getAddressData());
|
result = Reader32->readTypeFromMetadata(address.getAddressData());
|
||||||
} else {
|
} else {
|
||||||
return Reader64->readTypeFromMetadata(address.getAddressData());
|
result = Reader64->readTypeFromMetadata(address.getAddressData());
|
||||||
}
|
}
|
||||||
|
if (result) return result;
|
||||||
|
return Result<Type>::emplaceFailure(Failure::Unknown);
|
||||||
}
|
}
|
||||||
|
|
||||||
NominalTypeDecl *getDeclForRemoteNominalTypeDescriptor(RemoteAddress address){
|
Result<NominalTypeDecl *>
|
||||||
|
getDeclForRemoteNominalTypeDescriptor(RemoteAddress address) {
|
||||||
|
NominalTypeDecl *result;
|
||||||
if (Is32) {
|
if (Is32) {
|
||||||
return Reader32->readNominalTypeFromDescriptor(address.getAddressData());
|
result =Reader32->readNominalTypeFromDescriptor(address.getAddressData());
|
||||||
} else {
|
} else {
|
||||||
return Reader64->readNominalTypeFromDescriptor(address.getAddressData());
|
result =Reader64->readNominalTypeFromDescriptor(address.getAddressData());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (result) return result;
|
||||||
|
return Result<NominalTypeDecl *>::emplaceFailure(Failure::Unknown);
|
||||||
}
|
}
|
||||||
|
|
||||||
Optional<uint64_t> getOffsetForProperty(Type type, StringRef propertyName) {
|
Result<uint64_t> getOffsetForProperty(Type type, StringRef propertyName) {
|
||||||
// TODO
|
// TODO
|
||||||
return None;
|
return Result<uint64_t>::emplaceFailure(Failure::Unknown);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -544,16 +551,17 @@ RemoteASTContext::~RemoteASTContext() {
|
|||||||
delete asImpl(Impl);
|
delete asImpl(Impl);
|
||||||
}
|
}
|
||||||
|
|
||||||
Type RemoteASTContext::getTypeForRemoteTypeMetadata(RemoteAddress address) {
|
Result<Type>
|
||||||
|
RemoteASTContext::getTypeForRemoteTypeMetadata(RemoteAddress address) {
|
||||||
return asImpl(Impl)->getTypeForRemoteTypeMetadata(address);
|
return asImpl(Impl)->getTypeForRemoteTypeMetadata(address);
|
||||||
}
|
}
|
||||||
|
|
||||||
NominalTypeDecl *
|
Result<NominalTypeDecl *>
|
||||||
RemoteASTContext::getDeclForRemoteNominalTypeDescriptor(RemoteAddress address) {
|
RemoteASTContext::getDeclForRemoteNominalTypeDescriptor(RemoteAddress address) {
|
||||||
return asImpl(Impl)->getDeclForRemoteNominalTypeDescriptor(address);
|
return asImpl(Impl)->getDeclForRemoteNominalTypeDescriptor(address);
|
||||||
}
|
}
|
||||||
|
|
||||||
Optional<uint64_t>
|
Result<uint64_t>
|
||||||
RemoteASTContext::getOffsetForProperty(Type type, StringRef propertyName) {
|
RemoteASTContext::getOffsetForProperty(Type type, StringRef propertyName) {
|
||||||
return asImpl(Impl)->getOffsetForProperty(type, propertyName);
|
return asImpl(Impl)->getOffsetForProperty(type, propertyName);
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user