@@ -54,6 +54,7 @@ internal class AD7DebugSession : DebugAdapterBase, IDebugPortNotify2, IDebugEven
54
54
private bool m_isAttach ;
55
55
private bool m_isCoreDump ;
56
56
private bool m_isStopped = false ;
57
+ private bool m_isStepping = false ;
57
58
58
59
private readonly TaskCompletionSource < object > m_configurationDoneTCS = new TaskCompletionSource < object > ( ) ;
59
60
@@ -237,6 +238,31 @@ private ProtocolException VerifyProcessId(string processId, string telemetryEven
237
238
return null ;
238
239
}
239
240
241
+ private IList < Tracepoint > GetTracepoints ( IDebugBreakpointEvent2 debugEvent )
242
+ {
243
+ IList < Tracepoint > tracepoints = new List < Tracepoint > ( ) ;
244
+
245
+ if ( debugEvent != null )
246
+ {
247
+ debugEvent . EnumBreakpoints ( out IEnumDebugBoundBreakpoints2 pBoundBreakpoints ) ;
248
+ IDebugBoundBreakpoint2 [ ] boundBp = new IDebugBoundBreakpoint2 [ 1 ] ;
249
+
250
+ uint numReturned = 0 ;
251
+ while ( pBoundBreakpoints . Next ( 1 , boundBp , ref numReturned ) == HRConstants . S_OK && numReturned == 1 )
252
+ {
253
+ if ( boundBp [ 0 ] . GetPendingBreakpoint ( out IDebugPendingBreakpoint2 ppPendingBreakpoint ) == HRConstants . S_OK &&
254
+ ppPendingBreakpoint . GetBreakpointRequest ( out IDebugBreakpointRequest2 ppBPRequest ) == HRConstants . S_OK &&
255
+ ppBPRequest is AD7BreakPointRequest ad7BreakpointRequest &&
256
+ ad7BreakpointRequest . HasTracepoint )
257
+ {
258
+ tracepoints . Add ( ad7BreakpointRequest . Tracepoint ) ;
259
+ }
260
+ }
261
+ }
262
+
263
+ return tracepoints ;
264
+ }
265
+
240
266
#endregion
241
267
242
268
#region AD7EventHandlers helper methods
@@ -245,6 +271,7 @@ public void BeforeContinue()
245
271
{
246
272
if ( ! m_isCoreDump )
247
273
{
274
+ m_isStepping = false ;
248
275
m_isStopped = false ;
249
276
m_variableManager . Reset ( ) ;
250
277
m_frameHandles . Reset ( ) ;
@@ -499,7 +526,6 @@ private void StepInternal(int threadId, enum_STEPKIND stepKind, enum_STEPUNIT st
499
526
// If we are already running ignore additional step requests
500
527
if ( m_isStopped )
501
528
{
502
-
503
529
IDebugThread2 thread = null ;
504
530
lock ( m_threads )
505
531
{
@@ -511,6 +537,7 @@ private void StepInternal(int threadId, enum_STEPKIND stepKind, enum_STEPUNIT st
511
537
512
538
BeforeContinue ( ) ;
513
539
ErrorBuilder builder = new ErrorBuilder ( ( ) => errorMessage ) ;
540
+ m_isStepping = true ;
514
541
try
515
542
{
516
543
builder . CheckHR ( m_program . Step ( thread , stepKind , stepUnit ) ) ;
@@ -590,7 +617,8 @@ protected override void HandleInitializeRequestAsync(IRequestResponder<Initializ
590
617
SupportsFunctionBreakpoints = m_engineConfiguration . FunctionBP ,
591
618
SupportsConditionalBreakpoints = m_engineConfiguration . ConditionalBP ,
592
619
ExceptionBreakpointFilters = m_engineConfiguration . ExceptionSettings . ExceptionBreakpointFilters . Select ( item => new ExceptionBreakpointsFilter ( ) { Default = item . @default , Filter = item . filter , Label = item . label } ) . ToList ( ) ,
593
- SupportsClipboardContext = m_engineConfiguration . ClipboardContext
620
+ SupportsClipboardContext = m_engineConfiguration . ClipboardContext ,
621
+ SupportsLogPoints = true
594
622
} ;
595
623
596
624
responder . SetResponse ( initializeResponse ) ;
@@ -1607,6 +1635,14 @@ protected override void HandleSetBreakpointsRequestAsync(IRequestResponder<SetBr
1607
1635
toRemove . Delete ( ) ;
1608
1636
dict . Remove ( bp . Line ) ;
1609
1637
}
1638
+ // Check to see if tracepoint changed
1639
+ else if ( ! StringComparer . Ordinal . Equals ( ad7BPRequest . LogMessage , bp . LogMessage ) )
1640
+ {
1641
+ ad7BPRequest . ClearTracepoint ( ) ;
1642
+ var toRemove = dict [ bp . Line ] ;
1643
+ toRemove . Delete ( ) ;
1644
+ dict . Remove ( bp . Line ) ;
1645
+ }
1610
1646
else
1611
1647
{
1612
1648
if ( ad7BPRequest . BindResult != null )
@@ -1637,16 +1673,37 @@ protected override void HandleSetBreakpointsRequestAsync(IRequestResponder<SetBr
1637
1673
1638
1674
try
1639
1675
{
1640
- eb . CheckHR ( m_engine . CreatePendingBreakpoint ( pBPRequest , out pendingBp ) ) ;
1641
- eb . CheckHR ( pendingBp . Bind ( ) ) ;
1676
+ bool verified = true ;
1677
+ if ( ! string . IsNullOrEmpty ( bp . LogMessage ) )
1678
+ {
1679
+ // Make sure tracepoint is valid.
1680
+ verified = pBPRequest . SetLogMessage ( bp . LogMessage ) ;
1681
+ }
1642
1682
1643
- dict [ bp . Line ] = pendingBp ;
1644
- resBreakpoints . Add ( new Breakpoint ( )
1683
+ if ( verified )
1645
1684
{
1646
- Id = ( int ) pBPRequest . Id ,
1647
- Verified = true ,
1648
- Line = bp . Line
1649
- } ) ;
1685
+ eb . CheckHR ( m_engine . CreatePendingBreakpoint ( pBPRequest , out pendingBp ) ) ;
1686
+ eb . CheckHR ( pendingBp . Bind ( ) ) ;
1687
+
1688
+ dict [ bp . Line ] = pendingBp ;
1689
+
1690
+ resBreakpoints . Add ( new Breakpoint ( )
1691
+ {
1692
+ Id = ( int ) pBPRequest . Id ,
1693
+ Verified = verified ,
1694
+ Line = bp . Line
1695
+ } ) ;
1696
+ }
1697
+ else
1698
+ {
1699
+ resBreakpoints . Add ( new Breakpoint ( )
1700
+ {
1701
+ Id = ( int ) pBPRequest . Id ,
1702
+ Verified = verified ,
1703
+ Line = bp . Line ,
1704
+ Message = string . Format ( CultureInfo . CurrentCulture , AD7Resources . Error_UnableToParseLogMessage )
1705
+ } ) ;
1706
+ }
1650
1707
}
1651
1708
catch ( Exception e )
1652
1709
{
@@ -1893,12 +1950,10 @@ protected override void HandleEvaluateRequestAsync(IRequestResponder<EvaluateArg
1893
1950
hr = frame . GetExpressionContext ( out expressionContext ) ;
1894
1951
eb . CheckHR ( hr ) ;
1895
1952
1896
- const uint InputRadix = 10 ;
1897
- DAPEvalFlags dapEvalFlags = DAPEvalFlags . NONE ;
1898
1953
IDebugExpression2 expressionObject ;
1899
1954
string error ;
1900
1955
uint errorIndex ;
1901
- hr = expressionContext . ParseText ( expression , enum_PARSEFLAGS . PARSE_EXPRESSION , InputRadix , out expressionObject , out error , out errorIndex ) ;
1956
+ hr = expressionContext . ParseText ( expression , enum_PARSEFLAGS . PARSE_EXPRESSION , Constants . ParseRadix , out expressionObject , out error , out errorIndex ) ;
1902
1957
if ( ! string . IsNullOrEmpty ( error ) )
1903
1958
{
1904
1959
// TODO: Is this how errors should be returned?
@@ -1919,14 +1974,14 @@ protected override void HandleEvaluateRequestAsync(IRequestResponder<EvaluateArg
1919
1974
flags |= enum_EVALFLAGS . EVAL_NOSIDEEFFECTS ;
1920
1975
}
1921
1976
1922
- if ( context == EvaluateArguments . ContextValue . Clipboard )
1923
- {
1924
- dapEvalFlags |= DAPEvalFlags . CLIPBOARD_CONTEXT ;
1925
- }
1926
-
1927
1977
IDebugProperty2 property ;
1928
1978
if ( expressionObject is IDebugExpressionDAP expressionDapObject )
1929
1979
{
1980
+ DAPEvalFlags dapEvalFlags = DAPEvalFlags . NONE ;
1981
+ if ( context == EvaluateArguments . ContextValue . Clipboard )
1982
+ {
1983
+ dapEvalFlags |= DAPEvalFlags . CLIPBOARD_CONTEXT ;
1984
+ }
1930
1985
hr = expressionDapObject . EvaluateSync ( flags , dapEvalFlags , Constants . EvaluationTimeout , null , out property ) ;
1931
1986
}
1932
1987
else
@@ -2094,7 +2149,45 @@ public void HandleIDebugEntryPointEvent2(IDebugEngine2 pEngine, IDebugProcess2 p
2094
2149
2095
2150
public void HandleIDebugBreakpointEvent2 ( IDebugEngine2 pEngine , IDebugProcess2 pProcess , IDebugProgram2 pProgram , IDebugThread2 pThread , IDebugEvent2 pEvent )
2096
2151
{
2097
- FireStoppedEvent ( pThread , StoppedEvent . ReasonValue . Breakpoint ) ;
2152
+ IList < Tracepoint > tracepoints = GetTracepoints ( pEvent as IDebugBreakpointEvent2 ) ;
2153
+ if ( tracepoints . Any ( ) )
2154
+ {
2155
+ ThreadPool . QueueUserWorkItem ( ( o ) =>
2156
+ {
2157
+ foreach ( var tp in tracepoints )
2158
+ {
2159
+ int hr = tp . GetLogMessage ( pThread , Constants . ParseRadix , m_processName , out string logMessage ) ;
2160
+ if ( hr != HRConstants . S_OK )
2161
+ {
2162
+ DebuggerTelemetry . ReportError ( DebuggerTelemetry . TelemetryTracepointEventName , logMessage ) ;
2163
+ m_logger . WriteLine ( LoggingCategory . DebuggerError , logMessage ) ;
2164
+ }
2165
+ else
2166
+ {
2167
+ m_logger . WriteLine ( LoggingCategory . DebuggerStatus , logMessage ) ;
2168
+ }
2169
+ }
2170
+
2171
+ // Need to check to see if the previous continuation of the debuggee was a step.
2172
+ // If so, we need to send a stopping event to the UI to signal the step completed successfully.
2173
+ if ( ! m_isStepping )
2174
+ {
2175
+ ThreadPool . QueueUserWorkItem ( ( obj ) =>
2176
+ {
2177
+ BeforeContinue ( ) ;
2178
+ m_program . Continue ( pThread ) ;
2179
+ } ) ;
2180
+ }
2181
+ else
2182
+ {
2183
+ FireStoppedEvent ( pThread , StoppedEvent . ReasonValue . Breakpoint ) ;
2184
+ }
2185
+ } ) ;
2186
+ }
2187
+ else
2188
+ {
2189
+ FireStoppedEvent ( pThread , StoppedEvent . ReasonValue . Breakpoint ) ;
2190
+ }
2098
2191
}
2099
2192
2100
2193
public void HandleIDebugBreakEvent2 ( IDebugEngine2 pEngine , IDebugProcess2 pProcess , IDebugProgram2 pProgram , IDebugThread2 pThread , IDebugEvent2 pEvent )
0 commit comments