private void ProcessFrame(IXCLRDataStackWalk stackWalk) { object tmp; if (HR.Failed(stackWalk.GetFrame(out tmp))) return; IXCLRDataFrame frame = (IXCLRDataFrame)tmp; StringBuilder methodName = new StringBuilder(MaxNameSize); uint methodNameLen; if (HR.Failed(frame.GetCodeName(0 /*default flags*/, (uint)methodName.Capacity, out methodNameLen, methodName))) return; uint numArgs, numLocals; if (HR.Failed(frame.GetNumArguments(out numArgs))) numArgs = 0; if (HR.Failed(frame.GetNumLocalVariables(out numLocals))) numLocals = 0; FrameArgumentsAndLocals frameArgsLocals = new FrameArgumentsAndLocals() { MethodName = methodName.ToString() }; for (uint argIdx = 0; argIdx < numArgs; ++argIdx) { StringBuilder argName = new StringBuilder(MaxNameSize); uint argNameLen; if (FailedUnlessOnCLR2DAC(frame.GetArgumentByIndex(argIdx, out tmp, (uint)argName.Capacity, out argNameLen, argName)) || tmp == null) continue; var arg = new ArgumentOrLocal() { Name = argName.ToString() }; FillValue(arg, (IXCLRDataValue)tmp); frameArgsLocals.Arguments.Add(arg); } for (uint lclIdx = 0; lclIdx < numLocals; ++lclIdx) { // The mscordacwks!ClrDataFrame::GetLocalVariableByIndex implementation never returns // names for local variables. Need to go through metadata to get them. StringBuilder dummy = new StringBuilder(2); uint lclNameLen; if (FailedUnlessOnCLR2DAC(frame.GetLocalVariableByIndex(lclIdx, out tmp, (uint)dummy.Capacity, out lclNameLen, dummy)) || tmp == null) continue; string lclName = dummy.ToString(); var matchingFrame = _stackTrace.SingleOrDefault(f => f.DisplayString == frameArgsLocals.MethodName); if (matchingFrame != null) { lclName = GetLocalVariableName(matchingFrame.InstructionPointer, lclIdx); } var lcl = new ArgumentOrLocal() { Name = lclName }; FillValue(lcl, (IXCLRDataValue)tmp); frameArgsLocals.LocalVariables.Add(lcl); } _results.Add(frameArgsLocals); }
private void ProcessFrame(IXCLRDataStackWalk stackWalk) { object tmp; if (HR.Failed(stackWalk.GetFrame(out tmp))) { return; } IXCLRDataFrame frame = (IXCLRDataFrame)tmp; StringBuilder methodName = new StringBuilder(MaxNameSize); uint methodNameLen; if (HR.Failed(frame.GetCodeName(0 /*default flags*/, (uint)methodName.Capacity, out methodNameLen, methodName))) { return; } uint numArgs, numLocals; if (HR.Failed(frame.GetNumArguments(out numArgs))) { numArgs = 0; } if (HR.Failed(frame.GetNumLocalVariables(out numLocals))) { numLocals = 0; } FrameArgumentsAndLocals frameArgsLocals = new FrameArgumentsAndLocals() { MethodName = methodName.ToString() }; for (uint argIdx = 0; argIdx < numArgs; ++argIdx) { StringBuilder argName = new StringBuilder(MaxNameSize); uint argNameLen; if (FailedUnlessOnCLR2DAC(frame.GetArgumentByIndex(argIdx, out tmp, (uint)argName.Capacity, out argNameLen, argName)) || tmp == null) { continue; } var arg = new ArgumentOrLocal() { Name = argName.ToString() }; FillValue(arg, (IXCLRDataValue)tmp); frameArgsLocals.Arguments.Add(arg); } for (uint lclIdx = 0; lclIdx < numLocals; ++lclIdx) { // The mscordacwks!ClrDataFrame::GetLocalVariableByIndex implementation never returns // names for local variables. Need to go through metadata to get them. StringBuilder dummy = new StringBuilder(2); uint lclNameLen; if (FailedUnlessOnCLR2DAC(frame.GetLocalVariableByIndex(lclIdx, out tmp, (uint)dummy.Capacity, out lclNameLen, dummy)) || tmp == null) { continue; } string lclName = dummy.ToString(); var matchingFrame = _stackTrace.SingleOrDefault(f => f.DisplayString == frameArgsLocals.MethodName); if (matchingFrame != null) { lclName = GetLocalVariableName(matchingFrame.InstructionPointer, lclIdx); } var lcl = new ArgumentOrLocal() { Name = lclName }; FillValue(lcl, (IXCLRDataValue)tmp); frameArgsLocals.LocalVariables.Add(lcl); } _results.Add(frameArgsLocals); }