Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

1734 lines
44 KiB

/*++
Copyright (c) 1999-2002 Microsoft Corporation
Module Name:
wrkspace.cpp
Abstract:
This module contains the workspace implementation.
--*/
#include "precomp.hxx"
#pragma hdrstop
// #define DBG_WSP
#define WSP_ALIGN(Size) (((Size) + 7) & ~7)
#define WSP_GROW_BY 1024
#if DBG
#define SCORCH_ENTRY(Entry) \
memset((Entry) + 1, 0xdb, (Entry)->FullSize - sizeof(*(Entry)))
#else
#define SCORCH_ENTRY(Entry)
#endif
ULONG g_WspSwitchKey;
TCHAR g_WspSwitchValue[MAX_PATH];
BOOL g_WspSwitchBufferAvailable = TRUE;
Workspace* g_Workspace;
BOOL g_ExplicitWorkspace;
char* g_WorkspaceKeyNames[] =
{
"",
"Kernel",
"User",
"Dump",
"Remote",
"Explicit",
"File",
};
char* g_WorkspaceDefaultName = "Default";
char* g_WorkspaceKeyDescriptions[] =
{
"Base workspace",
"Kernel mode workspaces",
"User mode workspaces",
"Dump file workspaces",
"Remote client workspaces",
"User-saved workspaces",
"Workspaces in files",
};
Workspace::Workspace(void)
{
m_Flags = 0;
m_Data = NULL;
m_DataLen = 0;
m_DataUsed = 0;
m_Key = WSP_NAME_BASE;
m_Value = NULL;
}
Workspace::~Workspace(void)
{
free(m_Data);
free(m_Value);
}
WSP_ENTRY*
Workspace::Get(WSP_TAG Tag)
{
WSP_ENTRY* Entry = NULL;
while ((Entry = NextEntry(Entry)) != NULL)
{
if (Entry->Tag == Tag)
{
return Entry;
}
}
return NULL;
}
WSP_ENTRY*
Workspace::GetNext(WSP_ENTRY* Entry, WSP_TAG Tag, WSP_TAG TagMask)
{
while ((Entry = NextEntry(Entry)) != NULL)
{
if ((Entry->Tag & TagMask) == Tag)
{
return Entry;
}
}
return NULL;
}
WSP_ENTRY*
Workspace::GetString(WSP_TAG Tag, PSTR Str, ULONG MaxSize)
{
WSP_ENTRY* Entry = Get(Tag);
if (Entry != NULL)
{
if (Entry->DataSize > MaxSize)
{
return NULL;
}
strcpy(Str, WSP_ENTRY_DATA(PSTR, Entry));
}
return Entry;
}
WSP_ENTRY*
Workspace::GetAllocString(WSP_TAG Tag, PSTR* Str)
{
WSP_ENTRY* Entry = Get(Tag);
if (Entry != NULL)
{
*Str = (PSTR)malloc(Entry->DataSize);
if (*Str == NULL)
{
return NULL;
}
strcpy(*Str, WSP_ENTRY_DATA(PSTR, Entry));
}
return Entry;
}
WSP_ENTRY*
Workspace::GetBuffer(WSP_TAG Tag, PVOID Buf, ULONG Size)
{
WSP_ENTRY* Entry = Get(Tag);
if (Entry != NULL)
{
if (Entry->DataSize != Size)
{
return NULL;
}
memcpy(Buf, WSP_ENTRY_DATA(PUCHAR, Entry), Size);
}
return Entry;
}
WSP_ENTRY*
Workspace::Set(WSP_TAG Tag, ULONG Size)
{
WSP_ENTRY* Entry;
ULONG FullSize;
// Compute full rounded size.
FullSize = sizeof(WSP_ENTRY) + WSP_ALIGN(Size);
// Check and see if there's already an entry.
Entry = Get(Tag);
if (Entry != NULL)
{
// If it's already large enough use it and
// pack in remaining data.
if (Entry->FullSize >= FullSize)
{
ULONG Pack = Entry->FullSize - FullSize;
if (Pack > 0)
{
PackData((PUCHAR)Entry + FullSize, Pack);
Entry->FullSize = (USHORT)FullSize;
}
Entry->DataSize = (USHORT)Size;
SCORCH_ENTRY(Entry);
m_Flags |= WSPF_DIRTY_WRITE;
return Entry;
}
// Entry is too small so remove it.
PackData((PUCHAR)Entry, Entry->FullSize);
}
return Add(Tag, Size);
}
WSP_ENTRY*
Workspace::SetString(WSP_TAG Tag, PCSTR Str)
{
ULONG Size = strlen(Str) + 1;
WSP_ENTRY* Entry = Set(Tag, Size);
if (Entry != NULL)
{
memcpy(WSP_ENTRY_DATA(PSTR, Entry), Str, Size);
}
return Entry;
}
WSP_ENTRY*
Workspace::SetStrings(WSP_TAG Tag, ULONG Count, PCSTR* Strs)
{
ULONG i;
ULONG Size = 0;
for (i = 0; i < Count; i++)
{
Size += strlen(Strs[i]) + 1;
}
// Put a double terminator at the very end.
Size++;
WSP_ENTRY* Entry = Set(Tag, Size);
if (Entry != NULL)
{
PSTR Data = WSP_ENTRY_DATA(PSTR, Entry);
for (i = 0; i < Count; i++)
{
Size = strlen(Strs[i]) + 1;
memcpy(Data, Strs[i], Size);
Data += Size;
}
*Data = 0;
}
return Entry;
}
WSP_ENTRY*
Workspace::SetBuffer(WSP_TAG Tag, PVOID Buf, ULONG Size)
{
WSP_ENTRY* Entry = Set(Tag, Size);
if (Entry != NULL)
{
memcpy(WSP_ENTRY_DATA(PUCHAR, Entry), Buf, Size);
}
return Entry;
}
WSP_ENTRY*
Workspace::Add(WSP_TAG Tag, ULONG Size)
{
// Compute full rounded size.
ULONG FullSize = sizeof(WSP_ENTRY) + WSP_ALIGN(Size);
WSP_ENTRY* Entry = AllocateEntry(FullSize);
if (Entry != NULL)
{
Entry->Tag = Tag;
Entry->FullSize = (USHORT)FullSize;
Entry->DataSize = (USHORT)Size;
SCORCH_ENTRY(Entry);
m_Flags |= WSPF_DIRTY_WRITE;
}
return Entry;
}
void
Workspace::DeleteEntry(WSP_ENTRY* Entry)
{
if (ValidEntry(Entry))
{
PackData((PUCHAR)Entry, Entry->FullSize);
}
}
ULONG
Workspace::Delete(WSP_TAG Tag, WSP_TAG TagMask)
{
ULONG Deleted = 0;
WSP_ENTRY* Entry = NextEntry(NULL);
while (Entry != NULL)
{
if ((Entry->Tag & TagMask) == Tag)
{
DeleteEntry(Entry);
Deleted++;
m_Flags |= WSPF_DIRTY_WRITE;
// Check and see if we packed away the last entry.
if (!ValidEntry(Entry))
{
break;
}
}
else
{
Entry = NextEntry(Entry);
}
}
return Deleted;
}
void
Workspace::Empty(void)
{
// Reset used to just the header.
m_DataUsed = sizeof(WSP_HEADER);
// Nothing is dirty now except the write of emptiness.
m_Flags = (m_Flags & ~WSPF_DIRTY_ALL) | WSPF_DIRTY_WRITE;
}
PTSTR
Workspace::GetName(BOOL Verbose)
{
static TCHAR s_Buffer[MAX_PATH];
s_Buffer[0] = 0;
if (Verbose)
{
PSTR Type;
switch(m_Key)
{
case WSP_NAME_FILE:
Type = "File: ";
break;
case WSP_NAME_EXPLICIT:
Type = "Explicit: ";
break;
default:
Type = "Implicit: ";
break;
}
CatString(s_Buffer, Type, DIMA(s_Buffer));
}
if (m_Key != WSP_NAME_FILE && !strcmp(m_Value, g_WorkspaceDefaultName))
{
if (m_Key == WSP_NAME_BASE)
{
CatString(s_Buffer, "base", DIMA(s_Buffer));
}
else
{
CatString(s_Buffer, g_WorkspaceKeyNames[m_Key], DIMA(s_Buffer));
CatString(s_Buffer, " default", DIMA(s_Buffer));
}
}
else
{
CatString(s_Buffer, m_Value, DIMA(s_Buffer));
}
return s_Buffer;
}
HRESULT
Workspace::Create(ULONG Key, PTSTR Value,
Workspace** NewWsp)
{
Workspace* Wsp = new Workspace;
if (Wsp == NULL)
{
return E_OUTOFMEMORY;
}
Wsp->m_Key = Key;
if (Value != NULL)
{
Wsp->m_Value = _tcsdup(Value);
if (Wsp->m_Value == NULL)
{
delete Wsp;
return E_OUTOFMEMORY;
}
}
WSP_ENTRY* Entry;
WSP_HEADER* Header;
// Allocate intial space for the header and eight
// small entries. The workspace grows by large amounts
// so this will immediately allocate a reasonable chunk.
Entry = Wsp->AllocateEntry(sizeof(WSP_HEADER) +
8 * (sizeof(WSP_ENTRY) + 2 * sizeof(ULONG64)));
if (Entry == NULL)
{
delete Wsp;
return E_OUTOFMEMORY;
}
Header = (WSP_HEADER*)Entry;
Header->Signature = WSP_SIGNATURE;
Header->Version = WSP_VERSION;
// Reset used to just the header.
Wsp->m_DataUsed = sizeof(*Header);
// Start out dirty so the workspace will be written
// out and therefore can be opened later.
Wsp->m_Flags |= WSPF_DIRTY_WRITE;
*NewWsp = Wsp;
return S_OK;
}
HRESULT
Workspace::ReadFromReg(void)
{
HRESULT Status;
HKEY RegKey;
LONG RegStatus;
BOOL InPrimary;
//
// First check and see if the value exists under the
// primary key. If not, check the secondary key.
//
RegKey = OpenKey(TRUE, m_Key, FALSE);
if (RegKey)
{
RegStatus = RegQueryValueEx(RegKey, m_Value, NULL, NULL, NULL, NULL);
if (RegStatus != ERROR_SUCCESS && RegStatus != ERROR_MORE_DATA)
{
RegCloseKey(RegKey);
RegKey = NULL;
}
}
if (RegKey == NULL)
{
RegKey = OpenKey(FALSE, m_Key, FALSE);
if (RegKey == NULL)
{
return HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND);
}
InPrimary = FALSE;
}
else
{
InPrimary = TRUE;
}
DWORD Type;
DWORD Size;
Size = 0;
RegStatus = RegQueryValueEx(RegKey, m_Value, NULL, &Type, NULL, &Size);
if (RegStatus != ERROR_SUCCESS && RegStatus != ERROR_MORE_DATA)
{
Status = HRESULT_FROM_WIN32(RegStatus);
goto EH_Key;
}
if (Type != REG_BINARY ||
WSP_ALIGN(Size) != Size)
{
Status = E_INVALIDARG;
goto EH_Key;
}
WSP_ENTRY* Entry;
WSP_HEADER* Header;
Entry = AllocateEntry(Size);
if (Entry == NULL)
{
Status = E_OUTOFMEMORY;
goto EH_Key;
}
Header = (WSP_HEADER*)Entry;
if (RegQueryValueEx(RegKey, m_Value, NULL, &Type, (LPBYTE)Header, &Size) !=
ERROR_SUCCESS ||
Header->Signature != WSP_SIGNATURE ||
Header->Version != WSP_VERSION)
{
Status = E_INVALIDARG;
goto EH_Key;
}
RegCloseKey(RegKey);
//
// If the workspace was read from the secondary key
// migrate it to the primary and remove the secondary
// entry.
//
if (!InPrimary)
{
if (WriteToReg() == S_OK)
{
DeleteReg(FALSE);
}
}
return S_OK;
EH_Key:
RegCloseKey(RegKey);
return Status;
}
HRESULT
Workspace::ReadFromFile(void)
{
HRESULT Status;
HANDLE File;
File = CreateFile(m_Value, GENERIC_READ, FILE_SHARE_READ,
NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
if (!File || File == INVALID_HANDLE_VALUE)
{
return WIN32_LAST_STATUS();
}
DWORD Size = GetFileSize(File, NULL);
if (Size == INVALID_FILE_SIZE ||
WSP_ALIGN(Size) != Size)
{
Status = E_INVALIDARG;
goto EH_File;
}
WSP_ENTRY* Entry;
WSP_HEADER* Header;
Entry = AllocateEntry(Size);
if (Entry == NULL)
{
Status = E_OUTOFMEMORY;
goto EH_File;
}
Header = (WSP_HEADER*)Entry;
DWORD Done;
if (!ReadFile(File, Header, Size, &Done, NULL) ||
Done != Size ||
Header->Signature != WSP_SIGNATURE ||
Header->Version != WSP_VERSION)
{
Status = E_INVALIDARG;
goto EH_File;
}
CloseHandle(File);
return S_OK;
EH_File:
CloseHandle(File);
return Status;
}
HRESULT
Workspace::Read(ULONG Key, PTSTR Value,
Workspace** NewWsp)
{
// Make sure basic structures preserve alignment.
C_ASSERT(sizeof(WSP_HEADER) == WSP_ALIGN(sizeof(WSP_HEADER)));
C_ASSERT(sizeof(WSP_ENTRY) == WSP_ALIGN(sizeof(WSP_ENTRY)));
C_ASSERT(sizeof(WSP_COMMONWIN_HEADER) ==
WSP_ALIGN(sizeof(WSP_COMMONWIN_HEADER)));
HRESULT Status;
Workspace* Wsp = new Workspace;
if (Wsp == NULL)
{
return E_OUTOFMEMORY;
}
Wsp->m_Key = Key;
if (Value != NULL)
{
Wsp->m_Value = _tcsdup(Value);
if (Wsp->m_Value == NULL)
{
delete Wsp;
return E_OUTOFMEMORY;
}
}
if (Key == WSP_NAME_FILE)
{
Status = Wsp->ReadFromFile();
}
else
{
Status = Wsp->ReadFromReg();
}
if (Status != S_OK)
{
delete Wsp;
}
else
{
*NewWsp = Wsp;
}
return Status;
}
HRESULT
Workspace::ChangeName(ULONG Key, PTSTR Value, BOOL Force)
{
if (!Force)
{
if (Key == WSP_NAME_FILE)
{
if (GetFileAttributes(Value) != -1)
{
return HRESULT_FROM_WIN32(ERROR_ALREADY_EXISTS);
}
}
else
{
HKEY RegKey;
//
// Check and see if a workspace entry already
// exists under the given name. We only need
// to check the primary key as we're only concerned
// with overwriting and writing always occurs
// to the primary key.
//
RegKey = OpenKey(TRUE, Key, FALSE);
if (RegKey != NULL)
{
LONG RegStatus;
RegStatus = RegQueryValueEx(RegKey, Value, NULL, NULL,
NULL, NULL);
RegCloseKey(RegKey);
if (RegStatus == ERROR_SUCCESS || RegStatus == ERROR_MORE_DATA)
{
return HRESULT_FROM_WIN32(ERROR_ALREADY_EXISTS);
}
}
}
}
//
// Swap the workspace name.
//
PTSTR NewValue;
if (Value != NULL)
{
NewValue = _tcsdup(Value);
if (NewValue == NULL)
{
return E_OUTOFMEMORY;
}
}
else
{
NewValue = NULL;
}
delete m_Value;
m_Key = Key;
m_Value = NewValue;
// Need to write data out to the new location.
m_Flags |= WSPF_DIRTY_WRITE;
return S_OK;
}
void
Workspace::UpdateBreakpointInformation(void)
{
HRESULT Status;
Status = g_BpCmdsBuffer->UiLockForRead();
if (Status == S_OK)
{
// Clear old information.
Delete(WSP_GLOBAL_BREAKPOINTS, WSP_TAG_MASK);
// Only save an entry if there are breakpoints.
// Minimum output is a newline and terminator so
// don't count those.
if (g_BpCmdsBuffer->GetDataLen() > 2)
{
PSTR Cmds = (PSTR)g_BpCmdsBuffer->GetDataBuffer();
SetString(WSP_GLOBAL_BREAKPOINTS, Cmds);
}
UnlockStateBuffer(g_BpCmdsBuffer);
}
}
void
Workspace::UpdateWindowInformation(void)
{
// Clear old information.
Delete(DEF_WSP_TAG(WSP_GROUP_WINDOW, 0), WSP_GROUP_MASK);
//
// Record the frame window state.
//
WINDOWPLACEMENT Place;
Place.length = sizeof(Place);
GetWindowPlacement(g_hwndFrame, &Place);
SetBuffer(WSP_WINDOW_FRAME_PLACEMENT, &Place, sizeof(Place));
//
// Persist windows from the bottom of the Z order up
// so that when they're recreated in the same order
// the Z order is also recreated.
//
HWND Win = MDIGetActive(g_hwndMDIClient, NULL);
if (Win == NULL ||
(Win = GetWindow(Win, GW_HWNDLAST)) == NULL)
{
// No windows.
return;
}
while (Win != NULL)
{
PCOMMONWIN_DATA WinData = GetCommonWinData(Win);
if (WinData != NULL)
{
WSP_ENTRY* Entry;
ULONG Size;
Size = WinData->GetWorkspaceSize();
Entry = Add(WSP_WINDOW_COMMONWIN_1,
Size + sizeof(WSP_COMMONWIN_HEADER));
if (Entry != NULL)
{
WSP_COMMONWIN_HEADER* Hdr =
WSP_ENTRY_DATA(WSP_COMMONWIN_HEADER*, Entry);
Hdr->Type = WinData->m_enumType;
Hdr->Reserved = 0;
if (Size > 0)
{
WinData->SetWorkspace((PUCHAR)(Hdr + 1));
}
}
}
Win = GetWindow(Win, GW_HWNDPREV);
}
}
void
Workspace::UpdateLogFileInformation(void)
{
HRESULT Status;
char LogFile[MAX_PATH];
BOOL Append;
ULONG FileLen;
Status = g_pUiControl->GetLogFile(LogFile, sizeof(LogFile), NULL,
&Append);
if (Status != S_OK && Status != E_NOINTERFACE)
{
return;
}
// Clear old information.
Delete(WSP_GLOBAL_LOG_FILE, WSP_TAG_MASK);
if (Status == E_NOINTERFACE)
{
// No log is open.
return;
}
FileLen = strlen(LogFile) + 1;
WSP_ENTRY* Entry = Set(WSP_GLOBAL_LOG_FILE, sizeof(BOOL) + FileLen);
if (Entry != NULL)
{
PSTR Data = WSP_ENTRY_DATA(PSTR, Entry);
*(PBOOL)Data = Append;
strcpy(Data + sizeof(Append), LogFile);
}
}
void
Workspace::UpdatePathInformation(void)
{
HRESULT Status;
char Path[MAX_ENGINE_PATH];
Status = g_pUiSymbols->GetSymbolPath(Path, sizeof(Path), NULL);
if (Status == S_OK)
{
SetString(WSP_GLOBAL_SYMBOL_PATH, Path);
}
Status = g_pUiSymbols->GetImagePath(Path, sizeof(Path), NULL);
if (Status == S_OK)
{
SetString(WSP_GLOBAL_IMAGE_PATH, Path);
}
Status = g_pUiSymbols->GetSourcePath(Path, sizeof(Path), NULL);
if (Status == S_OK)
{
SetString(WSP_GLOBAL_SOURCE_PATH, Path);
}
// Local source path is only set explicitly.
}
void
Workspace::UpdateFilterInformation(void)
{
HRESULT Status;
Status = g_FilterBuffer->UiLockForRead();
if (Status == S_OK)
{
// Clear old information.
Delete(WSP_GLOBAL_FILTERS, WSP_TAG_MASK);
// Only save an entry if there are changes.
// Minimum output is a newline and terminator so
// don't count those.
if (g_FilterWspCmdsOffset < g_FilterBuffer->GetDataLen() - 2)
{
PSTR Cmds = (PSTR)g_FilterBuffer->GetDataBuffer() +
g_FilterWspCmdsOffset;
SetString(WSP_GLOBAL_FILTERS, Cmds);
}
UnlockStateBuffer(g_FilterBuffer);
}
}
void
Workspace::UpdateMruListInformation(void)
{
ULONG Size;
WSP_ENTRY* Entry;
// Clear old information.
Delete(WSP_GLOBAL_MRU_LIST, WSP_TAG_MASK);
Size = GetMruSize();
Entry = Set(WSP_GLOBAL_MRU_LIST, Size);
if (Entry != NULL)
{
WriteMru(WSP_ENTRY_DATA(PUCHAR, Entry));
}
}
void
Workspace::UpdateAliasInformation(void)
{
HRESULT Status;
Status = g_AliasBuffer->UiLockForRead();
if (Status == S_OK)
{
// Clear old information.
Delete(WSP_GLOBAL_ALIASES, WSP_TAG_MASK);
// Only save an entry if there are changes.
// Minimum output is a newline and terminator so
// don't count those.
if (g_AliasBuffer->GetDataLen() > 2)
{
SetString(WSP_GLOBAL_ALIASES,
(PSTR)g_AliasBuffer->GetDataBuffer());
}
UnlockStateBuffer(g_AliasBuffer);
}
}
HRESULT
Workspace::WriteToReg(void)
{
// Writing always occurs to the primary key.
HKEY RegKey = OpenKey(TRUE, m_Key, TRUE);
if (RegKey == NULL)
{
return E_FAIL;
}
LONG Status = RegSetValueEx(RegKey, m_Value, 0, REG_BINARY,
m_Data, m_DataUsed);
RegCloseKey(RegKey);
if (Status != ERROR_SUCCESS)
{
return HRESULT_FROM_WIN32(Status);
}
else
{
m_Flags &= ~WSPF_DIRTY_ALL;
return S_OK;
}
}
HRESULT
Workspace::WriteToFile(void)
{
HRESULT Status;
HANDLE File;
File = CreateFile(m_Value, GENERIC_WRITE, 0,
NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
if (!File || File == INVALID_HANDLE_VALUE)
{
return WIN32_LAST_STATUS();
}
DWORD Done;
if (!WriteFile(File, m_Data, m_DataUsed, &Done, NULL) ||
Done != m_DataUsed)
{
Status = WIN32_LAST_STATUS();
}
else
{
m_Flags &= ~WSPF_DIRTY_ALL;
Status = S_OK;
}
CloseHandle(File);
return Status;
}
void
Workspace::DeleteReg(BOOL Primary)
{
DeleteRegKey(Primary, m_Key, m_Value);
// We don't want to leave any dirty bits
// on because the workspace would just be written
// out again at the next flush.
m_Flags &= ~WSPF_DIRTY_ALL;
}
void
Workspace::DeleteRegKey(BOOL Primary, ULONG Key, PTSTR Value)
{
HKEY RegKey = OpenKey(Primary, Key, FALSE);
if (RegKey != NULL)
{
RegDeleteValue(RegKey, Value);
RegCloseKey(RegKey);
}
}
HRESULT
Workspace::Flush(BOOL ForceSave, BOOL Cancellable)
{
if (getenv("WINDBG_NO_WORKSPACE_WINDOWS") != NULL)
{
// Window layout saving is suppressed so don't
// consider them dirty.
m_Flags &= ~WSPF_DIRTY_WINDOWS;
}
if ((m_Flags & WSPF_DIRTY_ALL) == 0 ||
(g_QuietMode == QMODE_ALWAYS_NO && !ForceSave))
{
return S_OK;
}
#ifdef DBG_WSP
DebugPrint("Workspace dirty flags %X\n", m_Flags & WSPF_DIRTY_ALL);
#endif
int Answer;
if (g_QuietMode == QMODE_ALWAYS_YES || ForceSave)
{
Answer = IDOK;
}
else
{
Answer = QuestionBox(STR_Save_Workspace,
Cancellable ? MB_YESNOCANCEL : MB_YESNO,
GetName(FALSE));
}
if (Answer == IDNO)
{
return S_OK;
}
else if (Answer == IDCANCEL)
{
Assert(Cancellable);
return S_FALSE;
}
if (m_Flags & WSPF_DIRTY_BREAKPOINTS)
{
UpdateBreakpointInformation();
}
if (m_Flags & WSPF_DIRTY_WINDOWS)
{
UpdateWindowInformation();
}
if (m_Flags & WSPF_DIRTY_LOG_FILE)
{
UpdateLogFileInformation();
}
if (m_Flags & WSPF_DIRTY_PATHS)
{
UpdatePathInformation();
}
if (m_Flags & WSPF_DIRTY_FILTERS)
{
UpdateFilterInformation();
}
if (m_Flags & WSPF_DIRTY_MRU_LIST)
{
UpdateMruListInformation();
}
if (m_Flags & WSPF_DIRTY_ALIASES)
{
UpdateAliasInformation();
}
if (m_Key == WSP_NAME_FILE)
{
return WriteToFile();
}
else
{
return WriteToReg();
}
}
WSP_ENTRY*
Workspace::AllocateEntry(ULONG FullSize)
{
// Sizes must fit in USHORTs. This shouldn't be
// a big problem since workspaces shouldn't have
// huge data items in them.
if (FullSize > 0xffff)
{
return NULL;
}
if (m_DataUsed + FullSize > m_DataLen)
{
ULONG NewLen = m_DataLen;
do
{
NewLen += WSP_GROW_BY;
}
while (m_DataUsed + FullSize > NewLen);
PUCHAR NewData = (PUCHAR)realloc(m_Data, NewLen);
if (NewData == NULL)
{
return NULL;
}
m_Data = NewData;
m_DataLen = NewLen;
}
WSP_ENTRY* Entry = (WSP_ENTRY*)(m_Data + m_DataUsed);
m_DataUsed += FullSize;
return Entry;
}
void
Workspace::GetKeyName(ULONG Key, PSTR KeyName)
{
_tcscpy(KeyName, WSP_REG_KEY);
if (Key > WSP_NAME_BASE)
{
_tcscat(KeyName, "\\");
_tcscat(KeyName, g_WorkspaceKeyNames[Key]);
}
}
HKEY
Workspace::OpenKey(BOOL Primary, ULONG Key, BOOL Create)
{
TCHAR KeyName[MAX_PATH];
HKEY RegKey;
HKEY Base = Primary ? HKEY_CURRENT_USER : HKEY_LOCAL_MACHINE;
GetKeyName(Key, KeyName);
if (Create)
{
if (RegCreateKeyEx(Base, KeyName, 0, NULL, 0,
KEY_ALL_ACCESS, NULL, &RegKey,
NULL) == ERROR_SUCCESS)
{
return RegKey;
}
}
else if (RegOpenKeyEx(Base, KeyName, 0, KEY_ALL_ACCESS,
&RegKey) == ERROR_SUCCESS)
{
return RegKey;
}
return NULL;
}
int
Workspace::Apply(ULONG Flags)
{
WSP_ENTRY* Entry;
PUCHAR Data;
BOOL UpdateColors = FALSE;
int SessionStarts;
ULONG MemWins = 0;
RegisterNamesStateBuffer* NameBuf;
#ifdef DBG_WSP
DebugPrint("Applying workspace %s%s%s with:\n",
m_Key == NULL ? "" : m_Key,
m_Key == NULL ? "" : "\\",
m_Value);
#endif
//
// Scan for explicit session start entries first.
// If any are present and a session is active
// fail the apply before anything actually happens.
//
if ((Flags & (WSP_APPLY_AGAIN |
WSP_APPLY_EXPLICIT)) == WSP_APPLY_EXPLICIT &&
g_EngineThreadId)
{
Entry = NULL;
while ((Entry = NextEntry(Entry)) != NULL)
{
switch(Entry->Tag)
{
case WSP_GLOBAL_EXE_COMMAND_LINE:
case WSP_GLOBAL_DUMP_FILE_NAME:
case WSP_GLOBAL_ATTACH_KERNEL_FLAGS:
return -1;
}
}
}
SessionStarts = 0;
Entry = NULL;
while ((Entry = NextEntry(Entry)) != NULL)
{
#ifdef DBG_WSP
DebugPrint(" %04X: Tag: %08X Size %X:%X\n",
(PUCHAR)Entry - m_Data, Entry->Tag,
Entry->DataSize, Entry->FullSize);
#endif
// If this is a reapply only a subset of the
// workspace is applied to prevent duplication
// and problems.
if ((Flags & WSP_APPLY_AGAIN) &&
Entry->Tag != WSP_GLOBAL_BREAKPOINTS &&
Entry->Tag != WSP_GLOBAL_REGISTER_MAP)
{
continue;
}
if (WSP_TAG_GROUP(Entry->Tag) == WSP_GROUP_COLORS)
{
if (SetColor(WSP_TAG_ITEM(Entry->Tag),
*WSP_ENTRY_DATA(COLORREF*, Entry)))
{
UpdateColors = TRUE;
}
continue;
}
switch(Entry->Tag)
{
case WSP_GLOBAL_SYMBOL_PATH:
g_pUiSymbols->SetSymbolPath(WSP_ENTRY_DATA(PSTR, Entry));
break;
case WSP_GLOBAL_IMAGE_PATH:
g_pUiSymbols->SetImagePath(WSP_ENTRY_DATA(PSTR, Entry));
break;
case WSP_GLOBAL_SOURCE_PATH:
g_pUiSymbols->SetSourcePath(WSP_ENTRY_DATA(PSTR, Entry));
break;
case WSP_GLOBAL_WINDOW_OPTIONS:
g_WinOptions = *WSP_ENTRY_DATA(PULONG, Entry);
if (g_WinOptions & WOPT_AUTO_ARRANGE)
{
Arrange();
}
break;
case WSP_GLOBAL_REGISTER_MAP:
NameBuf = GetRegisterNames(g_ActualProcType);
if (NameBuf)
{
NameBuf->SetRegisterMap(Entry->DataSize / sizeof(USHORT),
WSP_ENTRY_DATA(PUSHORT, Entry));
}
break;
case WSP_GLOBAL_PROC_REGISTER_MAP:
NameBuf = GetRegisterNames(*WSP_ENTRY_DATA(PULONG, Entry));
if (NameBuf)
{
NameBuf->SetRegisterMap((Entry->DataSize - sizeof(ULONG)) /
sizeof(USHORT),
(PUSHORT)(WSP_ENTRY_DATA(PULONG,
Entry) + 1));
}
break;
case WSP_GLOBAL_PROC_FLAGS_REGISTER_MAP:
NameBuf = GetRegisterNames(*WSP_ENTRY_DATA(PULONG, Entry));
if (NameBuf)
{
NameBuf->m_Flags = *(WSP_ENTRY_DATA(PULONG, Entry) + 1);
NameBuf->SetRegisterMap((Entry->DataSize - sizeof(ULONG)) /
sizeof(USHORT),
(PUSHORT)(WSP_ENTRY_DATA(PULONG,
Entry) + 2));
}
break;
case WSP_GLOBAL_BREAKPOINTS:
Assert(Entry->DataSize > 1);
AddStringMultiCommand(UIC_INVISIBLE_EXECUTE,
WSP_ENTRY_DATA(PSTR, Entry), FALSE);
break;
case WSP_GLOBAL_LOG_FILE:
Data = WSP_ENTRY_DATA(PUCHAR, Entry);
g_pUiControl->OpenLogFile((PSTR)Data + sizeof(BOOL), *(PBOOL)Data);
break;
case WSP_GLOBAL_LOCAL_SOURCE_PATH:
if (g_RemoteClient)
{
g_pUiLocSymbols->SetSourcePath(WSP_ENTRY_DATA(PSTR, Entry));
}
break;
case WSP_GLOBAL_FILTERS:
Assert(Entry->DataSize > 1);
AddStringMultiCommand(UIC_INVISIBLE_EXECUTE,
WSP_ENTRY_DATA(PSTR, Entry), FALSE);
break;
case WSP_GLOBAL_FIXED_LOGFONT:
g_Fonts[FONT_FIXED].LogFont = *WSP_ENTRY_DATA(LPLOGFONT, Entry);
CreateIndexedFont(FONT_FIXED, TRUE);
break;
case WSP_GLOBAL_TAB_WIDTH:
SetTabWidth(*WSP_ENTRY_DATA(PULONG, Entry));
break;
case WSP_GLOBAL_MRU_LIST:
Data = WSP_ENTRY_DATA(PUCHAR, Entry);
ReadMru(Data, Data + Entry->DataSize);
break;
case WSP_GLOBAL_REPEAT_COMMANDS:
if (*WSP_ENTRY_DATA(PULONG, Entry))
{
g_pUiControl->
RemoveEngineOptions(DEBUG_ENGOPT_NO_EXECUTE_REPEAT);
}
else
{
g_pUiControl->
AddEngineOptions(DEBUG_ENGOPT_NO_EXECUTE_REPEAT);
}
break;
case WSP_GLOBAL_COM_SETTINGS:
if (Entry->DataSize <= sizeof(g_ComSettings))
{
memcpy(g_ComSettings, WSP_ENTRY_DATA(PSTR, Entry),
Entry->DataSize);
PrintAllocString(&g_KernelConnectOptions, 256,
"com:port=%s,baud=%s", g_ComSettings,
g_ComSettings + strlen(g_ComSettings) + 1);
}
break;
case WSP_GLOBAL_1394_SETTINGS:
if (Entry->DataSize <= sizeof(g_1394Settings))
{
memcpy(g_1394Settings, WSP_ENTRY_DATA(PSTR, Entry),
Entry->DataSize);
PrintAllocString(&g_KernelConnectOptions, 256,
"1394:channel=%s", g_1394Settings);
}
break;
case WSP_GLOBAL_DISASM_ACTIVATE_SOURCE:
g_DisasmActivateSource = *WSP_ENTRY_DATA(PULONG, Entry);
break;
case WSP_GLOBAL_VIEW_TOOL_BAR:
CheckMenuItem(g_hmenuMain, IDM_VIEW_TOOLBAR,
*WSP_ENTRY_DATA(PULONG, Entry) ?
MF_CHECKED : MF_UNCHECKED);
Show_Toolbar(*WSP_ENTRY_DATA(PULONG, Entry));
break;
case WSP_GLOBAL_VIEW_STATUS_BAR:
CheckMenuItem(g_hmenuMain, IDM_VIEW_STATUS,
*WSP_ENTRY_DATA(PULONG, Entry) ?
MF_CHECKED : MF_UNCHECKED);
Show_StatusBar(*WSP_ENTRY_DATA(PULONG, Entry));
break;
case WSP_GLOBAL_AUTO_CMD_SCROLL:
g_AutoCmdScroll = *WSP_ENTRY_DATA(PULONG, Entry);
break;
case WSP_GLOBAL_SRC_FILE_PATH:
CopyString(g_SrcFilePath, WSP_ENTRY_DATA(PSTR, Entry),
DIMA(g_SrcFilePath));
break;
case WSP_GLOBAL_EXE_COMMAND_LINE:
if ((Flags & WSP_APPLY_EXPLICIT) &&
DupAllocString(&g_DebugCommandLine,
WSP_ENTRY_DATA(PSTR, Entry)))
{
SessionStarts++;
}
break;
case WSP_GLOBAL_EXE_CREATE_FLAGS:
g_DebugCreateFlags = *WSP_ENTRY_DATA(PULONG, Entry);
break;
case WSP_GLOBAL_DUMP_FILE_NAME:
if ((Flags & WSP_APPLY_EXPLICIT) &&
DupAllocString(&g_DumpFiles[0],
WSP_ENTRY_DATA(PSTR, Entry)))
{
g_NumDumpFiles = 1;
SessionStarts++;
}
break;
case WSP_GLOBAL_ATTACH_KERNEL_FLAGS:
if ((Flags & WSP_APPLY_EXPLICIT))
{
g_AttachKernelFlags = *WSP_ENTRY_DATA(PULONG, Entry);
SessionStarts++;
}
break;
case WSP_GLOBAL_TYPE_OPTIONS:
{
g_TypeOptions = *WSP_ENTRY_DATA(PULONG, Entry);
if (g_pUiSymbols2 != NULL)
{
g_pUiSymbols2->SetTypeOptions(g_TypeOptions);
}
}
break;
case WSP_GLOBAL_DUMP_FILE_PATH:
CopyString(g_DumpFilePath, WSP_ENTRY_DATA(PSTR, Entry),
DIMA(g_DumpFilePath));
break;
case WSP_GLOBAL_EXE_FILE_PATH:
CopyString(g_ExeFilePath, WSP_ENTRY_DATA(PSTR, Entry),
DIMA(g_ExeFilePath));
break;
case WSP_GLOBAL_ASSEMBLY_OPTIONS:
if (g_pUiControl3)
{
g_pUiControl3->
SetAssemblyOptions(*WSP_ENTRY_DATA(PULONG, Entry));
}
break;
case WSP_GLOBAL_EXPRESSION_SYNTAX:
if (g_pUiControl3)
{
g_pUiControl3->
SetExpressionSyntax(*WSP_ENTRY_DATA(PULONG, Entry));
}
break;
case WSP_GLOBAL_ALIASES:
Assert(Entry->DataSize > 1);
AddStringMultiCommand(UIC_INVISIBLE_EXECUTE,
WSP_ENTRY_DATA(PSTR, Entry), TRUE);
break;
case WSP_GLOBAL_PROCESS_START_DIR:
if (Flags & WSP_APPLY_EXPLICIT)
{
DupAllocString(&g_ProcessStartDir,
WSP_ENTRY_DATA(PSTR, Entry));
}
break;
case WSP_WINDOW_COMMONWIN_1:
WSP_COMMONWIN_HEADER* Hdr;
HWND Win;
PCOMMONWIN_DATA WinData;
Hdr = WSP_ENTRY_DATA(WSP_COMMONWIN_HEADER*, Entry);
Win = New_OpenDebugWindow(Hdr->Type, TRUE, MemWins);
if (Win != NULL &&
(WinData = GetCommonWinData(Win)) != NULL &&
Entry->DataSize > sizeof(WSP_COMMONWIN_HEADER))
{
Data = (PUCHAR)(Hdr + 1);
WinData->m_InAutoOp++;
WinData->ApplyWorkspace1(Data, Data +
(Entry->DataSize -
sizeof(WSP_COMMONWIN_HEADER)));
WinData->m_InAutoOp--;
}
// A user can have as many open memory windows as
// they like, which makes things a little tricky
// for workspaces as applying stacked workspaces
// could result in memory windows multiplying out
// of control if the same set of memory windows
// is saved in each workspace level. To avoid
// this and to function more like the other windows
// we reuse memory windows from any that are
// already in existence.
if (Hdr->Type == MEM_WINDOW)
{
MemWins++;
}
break;
case WSP_WINDOW_FRAME_PLACEMENT:
LPWINDOWPLACEMENT Place;
Place = WSP_ENTRY_DATA(LPWINDOWPLACEMENT, Entry);
SetWindowPlacement(g_hwndFrame, Place);
break;
case WSP_WINDOW_FRAME_TITLE:
SetTitleExplicitText(WSP_ENTRY_DATA(PSTR, Entry));
break;
}
}
if (UpdateColors)
{
UpdateAllColors();
}
return SessionStarts;
}
HRESULT
UiSwitchWorkspace(ULONG Key, PTSTR Value, WSP_CREATE_OPTION Create,
ULONG Flags, int* SessionStarts)
{
if (getenv("WINDBG_NO_WORKSPACE") != NULL)
{
return E_NOTIMPL;
}
HRESULT Status;
Workspace* OldWsp;
Workspace* NewWsp;
int Starts = 0;
Status = Workspace::Read(Key, Value, &NewWsp);
if (Status != S_OK)
{
if (Status == HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) &&
Create != WSP_OPEN_EXISTING)
{
if (Create == WSP_CREATE_QUERY)
{
if (QuestionBox(STR_Create_New_Workspace, MB_OKCANCEL) != IDOK)
{
return Status;
}
}
// Workspace does not exist so create a new one.
Status = Workspace::Create(Key, Value, &NewWsp);
}
if (Status != S_OK)
{
return Status;
}
}
// We have a new workspace ready to go so flush the old one.
OldWsp = g_Workspace;
if (OldWsp != NULL)
{
OldWsp->Flush(FALSE, FALSE);
}
// Apply the new workspace with no global workspace to
// avoid writing changes into the workspace as we apply it.
g_Workspace = NULL;
if (NewWsp != NULL)
{
Starts = NewWsp->Apply(Flags);
// Clear any window messages queued during the workspace
// application so that they're processed with no
// active workspace.
ProcessPendingMessages();
}
if (SessionStarts != NULL)
{
*SessionStarts = Starts;
}
if (Starts < 0)
{
// Apply failed so put the old workspace back.
g_Workspace = OldWsp;
return E_FAIL;
}
else
{
// Apply succeeded to replace the old workspace.
g_Workspace = NewWsp;
delete OldWsp;
return S_OK;
}
}
void
UiSwitchToExplicitWorkspace(ULONG Key, PTSTR Value)
{
HRESULT Status;
int Starts = 0;
if ((Status = UiSwitchWorkspace(Key, Value, WSP_OPEN_EXISTING,
WSP_APPLY_EXPLICIT, &Starts)) != S_OK)
{
if (Starts < 0)
{
InformationBox(ERR_Workspace_Session_Conflict);
}
else
{
InformationBox(ERR_Cant_Open_Workspace,
FormatStatusCode(Status),
FormatStatus(Status));
}
}
else
{
g_ExplicitWorkspace = TRUE;
if (Starts == 1)
{
StartDebugging();
}
}
}
void
UiSaveWorkspaceAs(ULONG Key, PTSTR Value)
{
HRESULT Status;
Status = g_Workspace->ChangeName(Key, Value, FALSE);
if (Status == HRESULT_FROM_WIN32(ERROR_ALREADY_EXISTS))
{
if (QuestionBox(ERR_Workspace_Already_Exists, MB_YESNO, Value) == IDNO)
{
return;
}
Status = g_Workspace->ChangeName(Key, Value, TRUE);
}
if (Status != S_OK)
{
InformationBox(ERR_Cant_Save_Workspace,
FormatStatusCode(Status), FormatStatus(Status));
return;
}
g_Workspace->Flush(TRUE, FALSE);
g_ExplicitWorkspace = TRUE;
}
HRESULT
UiDelayedSwitchWorkspace(void)
{
Assert(!g_WspSwitchBufferAvailable);
HRESULT Status = UiSwitchWorkspace(g_WspSwitchKey, g_WspSwitchValue,
WSP_CREATE_ALWAYS, WSP_APPLY_DEFAULT,
NULL);
// Mark the delayed switch buffer as available and
// wait for acknowledgement.
g_WspSwitchBufferAvailable = TRUE;
while (g_WspSwitchValue[0])
{
Sleep(10);
}
return Status;
}
void
EngSwitchWorkspace(ULONG Key, PTSTR Value)
{
// If the user explicitly selected a workspace
// don't override it due to engine activity.
if (g_ExplicitWorkspace ||
g_Exit)
{
return;
}
// We can't switch workspaces on the engine thread
// because of the UI work involved. Send the
// switch over to the UI thread and wait for
// it to be processed.
Assert(g_WspSwitchBufferAvailable);
g_WspSwitchBufferAvailable = FALSE;
g_WspSwitchKey = Key;
CopyString(g_WspSwitchValue, Value, DIMA(g_WspSwitchValue));
PostMessage(g_hwndFrame, WU_SWITCH_WORKSPACE, 0, 0);
if (g_pDbgClient != NULL)
{
// Temporarily disable event callbacks to keep
// activity at a minimum while we're in this halfway state.
g_pDbgClient->SetEventCallbacks(NULL);
while (!g_WspSwitchBufferAvailable)
{
if (FAILED(g_pDbgClient->DispatchCallbacks(10)))
{
Sleep(10);
}
}
g_pDbgClient->SetEventCallbacks(&g_EventCb);
}
else
{
while (!g_WspSwitchBufferAvailable)
{
Sleep(10);
}
}
// We know that at this point the new workspace cannot be dirty
// so just clear the dirty flags.
if (g_Workspace)
{
g_Workspace->ClearDirty();
}
// Let the UI thread continue.
g_WspSwitchKey = WSP_NAME_BASE;
g_WspSwitchValue[0] = 0;
Sleep(50);
//
// Warn the user is the workspace was not be created properly.
//
if (!g_Workspace)
{
InformationBox(ERR_NULL_Workspace, NULL);
return;
}
}
PSTR g_WspGlobalNames[] =
{
"Symbol path", "Image path", "Source path", "Window menu checks",
"Register customization", "Breakpoints", "Log file settings",
"Local source path", "Event filter settings", "Fixed-width font",
"Tab width", "MRU list", "Repeat commands setting", "COM port settings",
"1394 settings", "Activate source windows in disassembly mode",
"Show tool bar", "Show status bar", "Automatically scroll command window",
"Source open dialog path", "Executable command line",
"Executable create flags", "Dump file name", "Kernel attach flags",
"Type options", "Dump open dialog path", "Executable open dialog path",
"Per-processor register customization", "Assembly/disassembly options",
"Expression evaluator syntax", "Command window text aliases",
"Executable start directory", "Per-processor register customization",
};
PSTR g_WspWindowNames[] =
{
"Child window settings", "WinDBG window settings", "WinDBG window title",
};
PSTR
GetWspTagName(WSP_TAG Tag)
{
ULONG Item = WSP_TAG_ITEM(Tag);
static char Buffer[128];
switch(WSP_TAG_GROUP(Tag))
{
case WSP_GROUP_GLOBAL:
if (Item < WSP_GLOBAL_COUNT)
{
C_ASSERT(DIMA(g_WspGlobalNames) == WSP_GLOBAL_COUNT);
return g_WspGlobalNames[Item];
}
break;
case WSP_GROUP_WINDOW:
if (Item < WSP_WINDOW_COUNT)
{
C_ASSERT(DIMA(g_WspWindowNames) == WSP_WINDOW_COUNT);
return g_WspWindowNames[Item];
}
break;
case WSP_GROUP_COLORS:
INDEXED_COLOR* IdxCol = GetIndexedColor(Item);
if (IdxCol != NULL)
{
sprintf(Buffer, "%s color", IdxCol->Name);
return Buffer;
}
break;
}
return "Unknown tag";
}