forked from dotnet/runtime
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathTarget.cs
241 lines (215 loc) · 9.64 KB
/
Target.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
using System;
using System.Collections.Generic;
using System.Diagnostics.CodeAnalysis;
using System.Numerics;
namespace Microsoft.Diagnostics.DataContractReader;
/// <summary>
/// Representation of the target under inspection
/// </summary>
/// <remarks>
/// This class provides APIs used by contracts for reading from the target and getting type and globals
/// information. Like the contracts themselves in cdacreader, these are throwing APIs. Any callers at the boundaries
/// (for example, unmanaged entry points, COM) should handle any exceptions.
/// </remarks>
public abstract class Target
{
/// <summary>
/// CorDebugPlatform represents the platform of the target.
/// </summary>
public enum CorDebugPlatform : int
{
CORDB_PLATFORM_WINDOWS_X86 = 0,
CORDB_PLATFORM_WINDOWS_AMD64 = 1,
CORDB_PLATFORM_WINDOWS_IA64 = 2,
CORDB_PLATFORM_MAC_PPC = 3,
CORDB_PLATFORM_MAC_X86 = 4,
CORDB_PLATFORM_WINDOWS_ARM = 5,
CORDB_PLATFORM_MAC_AMD64 = 6,
CORDB_PLATFORM_WINDOWS_ARM64 = 7,
CORDB_PLATFORM_POSIX_AMD64 = 8,
CORDB_PLATFORM_POSIX_X86 = 9,
CORDB_PLATFORM_POSIX_ARM = 10,
CORDB_PLATFORM_POSIX_ARM64 = 11,
CORDB_PLATFORM_POSIX_LOONGARCH64 = 12,
CORDB_PLATFORM_POSIX_RISCV64 = 13,
}
/// <summary>
/// Pointer size of the target
/// </summary>
public abstract int PointerSize { get; }
/// <summary>
/// Endianness of the target
/// </summary>
public abstract bool IsLittleEndian { get; }
/// <summary>
/// Platform of the target
/// </summary>
public abstract CorDebugPlatform Platform { get; }
/// <summary>
/// Fills a buffer with the context of the given thread
/// </summary>
/// <param name="threadId">The identifier of the thread whose context is to be retrieved. The identifier is defined by the operating system.</param>
/// <param name="contextFlags">A bitwise combination of platform-dependent flags that indicate which portions of the context should be read.</param>
/// <param name="buffer">Buffer to be filled with thread context.</param>
/// <returns>true if successful, false otherwise</returns>
public abstract bool TryGetThreadContext(ulong threadId, uint contextFlags, Span<byte> buffer);
/// <summary>
/// Reads a well-known global pointer value from the target process
/// </summary>
/// <param name="global">The name of the global</param>
/// <returns>The value of the global</returns>
public abstract TargetPointer ReadGlobalPointer(string global);
/// <summary>
/// Read a pointer from the target in target endianness
/// </summary>
/// <param name="address">Address to start reading from</param>
/// <returns>Pointer read from the target</returns>}
public abstract TargetPointer ReadPointer(ulong address);
/// <summary>
/// Read a code pointer from the target in target endianness
/// </summary>
/// <param name="address">Address to start reading from</param>
/// <returns>Pointer read from the target</returns>}
public abstract TargetCodePointer ReadCodePointer(ulong address);
/// <summary>
/// Read some bytes from the target
/// </summary>
/// <param name="address">The address where to start reading</param>
/// <param name="buffer">Destination to copy the bytes, the number of bytes to read is the span length</param>
public abstract void ReadBuffer(ulong address, Span<byte> buffer);
/// <summary>
/// Read a null-terminated UTF-8 string from the target
/// </summary>
/// <param name="address">Address to start reading from</param>
/// <returns>String read from the target</returns>}
public abstract string ReadUtf8String(ulong address);
/// <summary>
/// Read a null-terminated UTF-16 string from the target in target endianness
/// </summary>
/// <param name="address">Address to start reading from</param>
/// <returns>String read from the target</returns>}
public abstract string ReadUtf16String(ulong address);
/// <summary>
/// Read a native unsigned integer from the target in target endianness
/// </summary>
/// <param name="address">Address to start reading from</param>
/// <returns>Value read from the target</returns>
public abstract TargetNUInt ReadNUInt(ulong address);
/// <summary>
/// Read a well known global from the target process as a number in the target endianness
/// </summary>
/// <typeparam name="T">The numeric type to be read</typeparam>
/// <param name="name">The name of the global</param>
/// <returns>A numeric value</returns>
public abstract T ReadGlobal<T>(string name) where T : struct, INumber<T>;
/// <summary>
/// Read a well known global from the target process as a number in the target endianness
/// </summary>
/// <typeparam name="T">The numeric type to be read</typeparam>
/// <param name="name">The name of the global</param>
/// <param name="value">The numeric value read.</param>
/// <returns>True if a global is found, false otherwise.</returns>
public abstract bool TryReadGlobal<T>(string name, [NotNullWhen(true)] out T? value) where T : struct, INumber<T>;
/// <summary>
/// Read a value from the target in target endianness
/// </summary>
/// <typeparam name="T">Type of value to read</typeparam>
/// <param name="address">Address to start reading from</param>
/// <returns>Value read from the target</returns>
public abstract T Read<T>(ulong address) where T : unmanaged, IBinaryInteger<T>, IMinMaxValue<T>;
/// <summary>
/// Read a target pointer from a span of bytes
/// </summary>
/// <param name="bytes">The span of bytes to read from</param>
/// <returns>The target pointer read from the span</returns>
public abstract TargetPointer ReadPointerFromSpan(ReadOnlySpan<byte> bytes);
/// <summary>
/// Returns true if the given pointer is aligned to the pointer size of the target
/// </summary>
/// <param name="pointer">A target pointer value</param>
/// <returns></returns>
public abstract bool IsAlignedToPointerSize(TargetPointer pointer);
/// <summary>
/// Returns the information about the given well-known data type in the target process
/// </summary>
/// <param name="type">The name of the well known type</param>
/// <returns>The information about the given type in the target process</returns>
public abstract TypeInfo GetTypeInfo(DataType type);
/// <summary>
/// Get the data cache for the target
/// </summary>
public abstract IDataCache ProcessedData { get; }
/// <summary>
/// Holds a snapshot of the target's structured data
/// </summary>
public interface IDataCache
{
/// <summary>
/// Read a value from the target and cache it, or return the currently cached value
/// </summary>
/// <typeparam name="T">The type of data to be read</typeparam>
/// <param name="address">The address in the target where the data resides</param>
/// <returns>The value that has been read from the target</returns>
T GetOrAdd<T>(TargetPointer address) where T : Data.IData<T>;
/// <summary>
/// Return the cached value for the given address, or null if no value is cached
/// </summary>
/// <typeparam name="T">The type of the data to be read</typeparam>
/// <param name="address">The address in the target where the data resides</param>
/// <param name="data">On return, set to the cached data value, or null if the data hasn't been cached yet.</param>
/// <returns>True if a copy of the data is cached, or false otherwise</returns>
bool TryGet<T>(ulong address, [NotNullWhen(true)] out T? data);
/// <summary>
/// Clear all cached data
/// </summary>
void Clear();
}
/// <summary>
/// Information about a well-known data type in the target process
/// </summary>
public readonly record struct TypeInfo
{
/// <summary>
/// The stride of the type in the target process.
/// This is a value that can be used to calculate the byte offset of an element in an array of this type.
/// </summary>
public uint? Size { get; init; }
/// <summary>
/// Information about the well-known fields of the type in the target process
/// </summary>
public readonly IReadOnlyDictionary<string, FieldInfo> Fields
{
get;
init;
}
public TypeInfo()
{
Fields = new Dictionary<string, FieldInfo>();
}
}
/// <summary>
/// Information about a well-known field of a well-known data type in the target process
/// </summary>
public readonly record struct FieldInfo
{
/// <summary>
/// The byte offset of the field in an instance of the type in the target process
/// </summary>
public int Offset {get; init; }
/// <summary>
/// The well known data type of the field in the target process
/// </summary>
public readonly DataType Type {get; init;}
/// <summary>
/// The name of the well known data type of the field in the target process, or null
/// if the target data descriptor did not record a name
/// </summary>
public readonly string? TypeName {get; init; }
}
/// <summary>
/// A cache of the contracts for the target process
/// </summary>
public abstract ContractRegistry Contracts { get; }
}