2020-09-30 16:53:55 +02:00

685 lines
16 KiB
C++

/*++
Copyright (c) 1992-2002 Microsoft Corporation
Module Name:
callswin.cpp
Abstract:
This module contains the main line code for display of calls window.
Environment:
Win32, User Mode
--*/
#include "precomp.hxx"
#pragma hdrstop
#define MIN_FRAMES 10
#define MORE_LESS 10
#define CALLS_CONTEXT_ID_BASE 0x100
#define TBB_MORE 9
#define TBB_LESS 10
#define TBB_COPY_ALL 13
// The IDs of the buttons are the bit shift of the
// corresponding flag.
TBBUTTON g_CallsTbButtons[] =
{
TEXT_TB_BTN(0, "Args", BTNS_CHECK),
TEXT_TB_BTN(1, "Func info", BTNS_CHECK),
TEXT_TB_BTN(2, "Source", BTNS_CHECK),
TEXT_TB_BTN(3, "Addrs", BTNS_CHECK),
TEXT_TB_BTN(4, "Headings", BTNS_CHECK),
TEXT_TB_BTN(5, "Nonvolatile regs", BTNS_CHECK),
TEXT_TB_BTN(6, "Frame nums", BTNS_CHECK),
TEXT_TB_BTN(7, "Arg types", BTNS_CHECK),
SEP_TB_BTN(),
TEXT_TB_BTN(TBB_MORE, "More", 0),
TEXT_TB_BTN(TBB_LESS, "Less", 0),
SEP_TB_BTN(),
TEXT_TB_BTN(ID_SHOW_TOOLBAR, "Toolbar", 0),
SEP_TB_BTN(),
TEXT_TB_BTN(TBB_COPY_ALL, "Copy stack to clipboard", 0),
};
#define NUM_CALLS_MENU_BUTTONS \
(sizeof(g_CallsTbButtons) / sizeof(g_CallsTbButtons[0]))
#define NUM_CALLS_TB_BUTTONS \
(NUM_CALLS_MENU_BUTTONS - 4)
HMENU CALLSWIN_DATA::s_ContextMenu;
//
//
//
CALLSWIN_DATA::CALLSWIN_DATA()
: SINGLE_CHILDWIN_DATA(1024)
{
m_enumType = CALLS_WINDOW;
m_Flags = 0;
m_Frames = 20;
m_FramesFound = 0;
m_TextOffset = 0;
m_WarningLine = 0xffffffff;
}
void
CALLSWIN_DATA::Validate()
{
SINGLE_CHILDWIN_DATA::Validate();
Assert(CALLS_WINDOW == m_enumType);
}
HRESULT
CALLSWIN_DATA::ReadState(void)
{
HRESULT Status;
ULONG FramesFound;
ULONG TextOffset;
Empty();
//
// Record the raw frame data first.
//
// Preallocate space to record the raw frames.
if (AddData(sizeof(DEBUG_STACK_FRAME) * m_Frames) == NULL)
{
return E_OUTOFMEMORY;
}
// Allocate a separate buffer to hold the frames while
// calling OutputStackTrace on them. We can't just pass
// in the state buffer pointer as resizing of the state
// buffer may cause the data pointer to change.
PDEBUG_STACK_FRAME RawFrames = (PDEBUG_STACK_FRAME)malloc(m_DataUsed);
if (RawFrames == NULL)
{
return E_OUTOFMEMORY;
}
Status = g_pDbgControl->GetStackTrace(0, 0, 0, RawFrames, m_Frames,
&FramesFound);
if (Status != S_OK)
{
free(RawFrames);
m_FramesFound = 0;
m_TextOffset = 0;
return Status;
}
TextOffset = m_DataUsed;
g_OutStateBuf.SetBuffer(this);
if ((Status = g_OutStateBuf.Start(FALSE)) != S_OK)
{
free(RawFrames);
m_FramesFound = 0;
m_TextOffset = 0;
return Status;
}
// If nonvolatile registers were requested we can't use just
// our saved frames as they require full context information.
Status = g_pOutCapControl->
OutputStackTrace(DEBUG_OUTCTL_THIS_CLIENT |
DEBUG_OUTCTL_OVERRIDE_MASK |
DEBUG_OUTCTL_NOT_LOGGED,
(m_Flags & DEBUG_STACK_NONVOLATILE_REGISTERS) ?
NULL : RawFrames, FramesFound, m_Flags);
if (Status == S_OK)
{
Status = g_OutStateBuf.End(FALSE);
}
else
{
g_OutStateBuf.End(FALSE);
}
// Now that the state buffer is stable put the raw frame
// data in.
memcpy(m_Data, RawFrames, TextOffset);
m_FramesFound = FramesFound;
m_TextOffset = TextOffset;
free(RawFrames);
// Check and see if there's a stack trace warning and remember
// it so we can ignore it. We still want it in the text
// as a reminder so we can't just remove it.
ULONG Line = 0;
PSTR Scan = (PSTR)m_Data + m_TextOffset;
m_WarningLine = 0xffffffff;
while (Scan < (PSTR)m_Data + m_DataUsed)
{
if (!memcmp(Scan, "WARNING:", 8))
{
m_WarningLine = Line;
break;
}
Scan = strchr(Scan, '\n');
if (!Scan)
{
break;
}
else
{
Scan++;
}
Line++;
}
return Status;
}
void
CALLSWIN_DATA::Copy()
{
LRESULT Line = SendMessage(m_hwndChild, LB_GETCURSEL, 0, 0);
Assert(Line != LB_ERR);
LRESULT Len = SendMessage(m_hwndChild, LB_GETTEXTLEN, Line, 0);
if (Len <= 0)
{
return;
}
Len++;
PSTR Text = (PSTR)malloc(Len);
if (!Text)
{
return;
}
SendMessage(m_hwndChild, LB_GETTEXT, Line, (LPARAM)Text);
Text[Len - 1] = 0;
CopyToClipboard(Text, FALSE);
free (Text);
return;
}
BOOL
CALLSWIN_DATA::CanWriteTextToFile()
{
return TRUE;
}
HRESULT
CALLSWIN_DATA::WriteTextToFile(HANDLE File)
{
HRESULT Status;
BOOL Write;
ULONG Done;
if ((Status = UiLockForRead()) != S_OK)
{
return Status;
}
Write = WriteFile(File, (PSTR)m_Data + m_TextOffset,
m_DataUsed - m_TextOffset, &Done, NULL);
UnlockStateBuffer(this);
if (!Write)
{
return WIN32_LAST_STATUS();
}
if (Done < m_DataUsed - m_TextOffset)
{
return HRESULT_FROM_WIN32(ERROR_WRITE_FAULT);
}
return S_OK;
}
HMENU
CALLSWIN_DATA::GetContextMenu(void)
{
ULONG i;
//
// We only keep one menu around for all call windows
// so apply the menu check state for this particular
// window.
// In reality there's only one calls window anyway,
// but this is a good example of how to handle
// multi-instance windows.
//
for (i = 0; i < NUM_CALLS_TB_BUTTONS; i++)
{
CheckMenuItem(s_ContextMenu, i + CALLS_CONTEXT_ID_BASE,
MF_BYCOMMAND | ((m_Flags & (1 << i)) ? MF_CHECKED : 0));
}
CheckMenuItem(s_ContextMenu, ID_SHOW_TOOLBAR + CALLS_CONTEXT_ID_BASE,
MF_BYCOMMAND | (m_ShowToolbar ? MF_CHECKED : 0));
return s_ContextMenu;
}
void
CALLSWIN_DATA::OnContextMenuSelection(UINT Item)
{
Item -= CALLS_CONTEXT_ID_BASE;
switch(Item)
{
case TBB_MORE:
m_Frames += MORE_LESS;
break;
case TBB_LESS:
if (m_Frames >= MIN_FRAMES + MORE_LESS)
{
m_Frames -= MORE_LESS;
}
break;
case TBB_COPY_ALL:
if (UiLockForRead() == S_OK)
{
CopyToClipboard((PSTR)m_Data + m_TextOffset, TRUE);
UnlockStateBuffer(this);
}
break;
case ID_SHOW_TOOLBAR:
SetShowToolbar(!m_ShowToolbar);
break;
default:
m_Flags ^= 1 << Item;
SyncUiWithFlags(1 << Item);
break;
}
if (g_Workspace != NULL)
{
g_Workspace->AddDirty(WSPF_DIRTY_WINDOWS);
}
UiRequestRead();
}
HRESULT
CALLSWIN_DATA::CodeExprAtCaret(PSTR Expr, PULONG64 Offset)
{
HRESULT Status;
ULONG Line = SelectionToFrame();
if (Line >= m_FramesFound)
{
return E_INVALIDARG;
}
if ((Status = UiLockForRead()) != S_OK)
{
// Don't want to return any success codes here.
return FAILED(Status) ? Status : E_FAIL;
}
PDEBUG_STACK_FRAME RawFrames = (PDEBUG_STACK_FRAME)m_Data;
if (Expr != NULL)
{
sprintf(Expr, "0x%I64x", RawFrames[Line].InstructionOffset);
}
if (Offset != NULL)
{
*Offset = RawFrames[Line].InstructionOffset;
}
UnlockStateBuffer(this);
return S_OK;
}
HRESULT
CALLSWIN_DATA::StackFrameAtCaret(PDEBUG_STACK_FRAME Frame)
{
HRESULT Status;
ULONG Line = SelectionToFrame();
if (Line >= m_FramesFound)
{
return E_INVALIDARG;
}
if ((Status = UiLockForRead()) != S_OK)
{
// Don't want to return any success codes here.
return FAILED(Status) ? Status : E_FAIL;
}
PDEBUG_STACK_FRAME RawFrames = (PDEBUG_STACK_FRAME)m_Data;
*Frame = RawFrames[Line];
UnlockStateBuffer(this);
return S_OK;
}
BOOL
CALLSWIN_DATA::OnCreate(void)
{
if (s_ContextMenu == NULL)
{
s_ContextMenu = CreateContextMenuFromToolbarButtons
(NUM_CALLS_MENU_BUTTONS, g_CallsTbButtons, CALLS_CONTEXT_ID_BASE);
if (s_ContextMenu == NULL)
{
return FALSE;
}
}
m_Toolbar = CreateWindowEx(0, TOOLBARCLASSNAME, NULL,
WS_CHILD | WS_VISIBLE |
TBSTYLE_WRAPABLE | TBSTYLE_LIST | CCS_TOP,
0, 0, m_Size.cx, 0, m_Win, (HMENU)ID_TOOLBAR,
g_hInst, NULL);
if (m_Toolbar == NULL)
{
return FALSE;
}
SendMessage(m_Toolbar, TB_BUTTONSTRUCTSIZE, sizeof(TBBUTTON), 0);
SendMessage(m_Toolbar, TB_ADDBUTTONS, NUM_CALLS_TB_BUTTONS,
(LPARAM)&g_CallsTbButtons);
SendMessage(m_Toolbar, TB_AUTOSIZE, 0, 0);
RECT Rect;
GetClientRect(m_Toolbar, &Rect);
m_ToolbarHeight = Rect.bottom - Rect.top;
m_ShowToolbar = TRUE;
m_hwndChild = CreateWindowEx(
WS_EX_CLIENTEDGE, // Extended style
_T("LISTBOX"), // class name
NULL, // title
WS_CHILD | WS_VISIBLE
| WS_MAXIMIZE
| WS_HSCROLL | WS_VSCROLL
| LBS_NOTIFY | LBS_WANTKEYBOARDINPUT
| LBS_NOINTEGRALHEIGHT, // style
0, // x
m_ToolbarHeight, // y
m_Size.cx, // width
m_Size.cy - m_ToolbarHeight, // height
m_Win, // parent
0, // control id
g_hInst, // hInstance
NULL // user defined data
);
if (m_hwndChild != NULL)
{
SetFont( FONT_FIXED );
return TRUE;
}
else
{
return FALSE;
}
}
LRESULT
CALLSWIN_DATA::OnCommand(
WPARAM wParam,
LPARAM lParam
)
{
if (HIWORD(wParam) == LBN_DBLCLK)
{
ULONG64 Offset;
if (CodeExprAtCaret(NULL, &Offset) == S_OK)
{
UIC_DISPLAY_CODE_DATA* DispCode =
StartStructCommand(UIC_DISPLAY_CODE);
if (DispCode != NULL)
{
DispCode->Offset = Offset;
FinishCommand();
}
}
DEBUG_STACK_FRAME StkFrame;
if (StackFrameAtCaret(&StkFrame) == S_OK)
{
UIC_SET_SCOPE_DATA* SetScope =
StartStructCommand(UIC_SET_SCOPE);
if (SetScope != NULL)
{
SetScope->StackFrame = StkFrame;
FinishCommand();
}
}
return 0;
}
if ((HWND)lParam == m_Toolbar)
{
OnContextMenuSelection(LOWORD(wParam) + CALLS_CONTEXT_ID_BASE);
return 0;
}
return 0;
}
LRESULT
CALLSWIN_DATA::OnVKeyToItem(
WPARAM wParam,
LPARAM lParam
)
{
if (LOWORD(wParam) == VK_RETURN)
{
ULONG64 Offset;
if (CodeExprAtCaret(NULL, &Offset) == S_OK)
{
UIC_DISPLAY_CODE_DATA* DispCode =
StartStructCommand(UIC_DISPLAY_CODE);
if (DispCode != NULL)
{
DispCode->Offset = Offset;
FinishCommand();
}
}
DEBUG_STACK_FRAME StkFrame;
if (StackFrameAtCaret(&StkFrame) == S_OK)
{
UIC_SET_SCOPE_DATA* SetScope =
StartStructCommand(UIC_SET_SCOPE);
if (SetScope != NULL)
{
SetScope->StackFrame = StkFrame;
FinishCommand();
}
}
}
else if (_T('G') == LOWORD(wParam))
{
ULONG64 Offset;
if (CodeExprAtCaret(NULL, &Offset) == S_OK)
{
PrintStringCommand(UIC_EXECUTE, "g 0x%I64x", Offset);
}
}
else if (_T('R') == LOWORD(wParam))
{
OnUpdate(UPDATE_BUFFER);
}
else
{
// Default behavior.
return -1;
}
// Keystroke processed.
return -2;
}
void
CALLSWIN_DATA::OnUpdate(
UpdateType Type
)
{
if (Type != UPDATE_BUFFER)
{
return;
}
LRESULT lbItem;
int nFrameCount;
HRESULT Status;
lbItem = SendMessage( m_hwndChild, LB_GETCURSEL, 0, 0 );
SendMessage( m_hwndChild, WM_SETREDRAW, FALSE, 0L );
SendMessage( m_hwndChild, LB_RESETCONTENT, 0, 0 );
Status = UiLockForRead();
if (Status == S_OK)
{
PSTR Buf = (PSTR)m_Data + m_TextOffset;
// Ignore final terminator.
PSTR End = (PSTR)m_Data + m_DataUsed - 1;
ULONG Width = 0;
nFrameCount = 0;
while (Buf < End)
{
PSTR Sep = strchr(Buf, '\n');
if (!Sep)
{
// Shouldn't happen, but just in case.
break;
}
ULONG Len = (ULONG)(Sep - Buf);
ULONG StrWidth = Len * m_Font->Metrics.tmAveCharWidth;
*Sep = 0;
SendMessage(m_hwndChild, LB_ADDSTRING, 0, (LPARAM)Buf);
Buf = Sep;
*Buf++ = '\n';
if (StrWidth > Width)
{
Width = StrWidth;
}
nFrameCount++;
}
SendMessage(m_hwndChild, LB_SETHORIZONTALEXTENT, Width, 0);
UnlockStateBuffer(this);
}
else
{
SendLockStatusMessage(m_hwndChild, LB_ADDSTRING, Status);
nFrameCount = 1;
}
SendMessage( m_hwndChild, LB_SETCURSEL,
(lbItem > nFrameCount) ? 0 : lbItem, 0 );
SendMessage( m_hwndChild, WM_SETREDRAW, TRUE, 0L );
}
ULONG
CALLSWIN_DATA::GetWorkspaceSize(void)
{
return SINGLE_CHILDWIN_DATA::GetWorkspaceSize() + 2 * sizeof(ULONG);
}
PUCHAR
CALLSWIN_DATA::SetWorkspace(PUCHAR Data)
{
Data = SINGLE_CHILDWIN_DATA::SetWorkspace(Data);
*(PULONG)Data = m_Flags;
Data += sizeof(ULONG);
*(PULONG)Data = m_Frames;
Data += sizeof(ULONG);
return Data;
}
PUCHAR
CALLSWIN_DATA::ApplyWorkspace1(PUCHAR Data, PUCHAR End)
{
Data = SINGLE_CHILDWIN_DATA::ApplyWorkspace1(Data, End);
if (End - Data >= 2 * sizeof(ULONG))
{
m_Flags = *(PULONG)Data;
Data += sizeof(ULONG);
m_Frames = *(PULONG)Data;
Data += sizeof(ULONG);
SyncUiWithFlags(0xffffffff);
UiRequestRead();
}
return Data;
}
void
CALLSWIN_DATA::SyncUiWithFlags(ULONG Changed)
{
ULONG i;
//
// Set toolbar button state from flags.
//
for (i = 0; i < NUM_CALLS_TB_BUTTONS; i++)
{
if (Changed & (1 << i))
{
SendMessage(m_Toolbar, TB_SETSTATE, g_CallsTbButtons[i].idCommand,
TBSTATE_ENABLED |
((m_Flags & (1 << i)) ? TBSTATE_CHECKED : 0));
}
}
}
ULONG
CALLSWIN_DATA::SelectionToFrame(void)
{
LRESULT CurSel = SendMessage(m_hwndChild, LB_GETCURSEL, 0, 0);
// Check for no-selection.
if (CurSel < 0)
{
return 0xffffffff;
}
ULONG Line = (ULONG)CurSel;
// If there's a WARNING line ignore it.
if (Line == m_WarningLine)
{
return 0xffffffff;
}
else if (Line > m_WarningLine)
{
Line--;
}
// If the column headers are on ignore the header line.
if (m_Flags & DEBUG_STACK_COLUMN_NAMES)
{
if (Line == 0)
{
return 0xffffffff;
}
else
{
Line--;
}
}
return Line;
}