857 lines
30 KiB
C++
857 lines
30 KiB
C++
|
#include "pch.h"
|
||
|
#pragma hdrstop
|
||
|
|
||
|
#include <initguid.h>
|
||
|
#include "uuid.h"
|
||
|
#include <ccstock.h>
|
||
|
|
||
|
#include "util.h"
|
||
|
#include "config.h"
|
||
|
#include "security.h"
|
||
|
|
||
|
const TCHAR g_szKeyPolicy[] = TEXT("Software\\Policies\\Microsoft\\Windows\\NetCache");
|
||
|
const TCHAR g_szKeyPrefs[] = TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\NetCache");
|
||
|
const TCHAR g_szKeyCustomActions[] = TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\NetCache\\CustomGoOfflineActions");
|
||
|
const TCHAR g_szValEnabled[] = TEXT("Enabled");
|
||
|
const TCHAR g_szValEncrypted[] = TEXT("Encrypted");
|
||
|
const TCHAR g_szValGoOfflineAction[] = TEXT("GoOfflineAction");
|
||
|
const TCHAR g_szValDefCacheSize[] = TEXT("DefCacheSize");
|
||
|
const TCHAR g_szValNoConfigCache[] = TEXT("NoConfigCache");
|
||
|
const TCHAR g_szValNoCacheViewer[] = TEXT("NoCacheViewer");
|
||
|
const TCHAR g_szValNoMakeAvailableOffline[] = TEXT("NoMakeAvailableOffline");
|
||
|
const TCHAR g_szValNoReminders[] = TEXT("NoReminders");
|
||
|
const TCHAR g_szValNoConfigReminders[] = TEXT("NoConfigReminders");
|
||
|
const TCHAR g_szValSyncAtLogoff[] = TEXT("SyncAtLogoff");
|
||
|
const TCHAR g_szValReminderFreqMinutes[] = TEXT("ReminderFreqMinutes");
|
||
|
const TCHAR g_szValInitialBalloonTimeout[] = TEXT("InitialBalloonTimeoutSeconds");
|
||
|
const TCHAR g_szValReminderBalloonTimeout[] = TEXT("ReminderBalloonTimeoutSeconds");
|
||
|
const TCHAR g_szValFirstPinWizardShown[] = TEXT("FirstPinWizardShown");
|
||
|
const TCHAR g_szValExpandStatusDlg[] = TEXT("ExpandStatusDlg");
|
||
|
const TCHAR g_szValFormatCscDb[] = TEXT("FormatDatabase");
|
||
|
const TCHAR g_szValEventLoggingLevel[] = TEXT("EventLoggingLevel");
|
||
|
const TCHAR g_szValPurgeAtLogoff[] = TEXT("PurgeAtLogoff");
|
||
|
const TCHAR g_szValExtExclusionList[] = TEXT("ExtExclusionList");
|
||
|
const TCHAR g_szValAlwaysPinSubFolders[] = TEXT("AlwaysPinSubFolders");
|
||
|
const TCHAR g_szNA[] = TEXT("<n/a>");
|
||
|
|
||
|
const DWORD POL_CU = 0x0001; // Current user
|
||
|
const DWORD POL_LM = 0x0002; // Local machine
|
||
|
const DWORD PREF_CU = 0x0004;
|
||
|
const DWORD PREF_LM = 0x0008;
|
||
|
const DWORD PREF_BOTH = PREF_CU | PREF_LM;
|
||
|
const DWORD POL_BOTH = POL_CU | POL_LM;
|
||
|
const DWORD REG_ALL = PREF_BOTH | POL_BOTH;
|
||
|
|
||
|
const DWORD PREF = 1;
|
||
|
const DWORD POLICY = 2;
|
||
|
|
||
|
|
||
|
LPTSTR GoOfflineActionText(
|
||
|
int iValue,
|
||
|
LPTSTR pszDest
|
||
|
)
|
||
|
{
|
||
|
LPCTSTR rgpsz[] = { TEXT("(Work offline)"),
|
||
|
TEXT("(No offline)") };
|
||
|
|
||
|
if (iValue >= 0 && iValue < ARRAYSIZE(rgpsz))
|
||
|
lstrcpy(pszDest, rgpsz[iValue]);
|
||
|
else
|
||
|
lstrcpy(pszDest, TEXT("<unknown>"));
|
||
|
|
||
|
return pszDest;
|
||
|
}
|
||
|
|
||
|
|
||
|
LPTSTR SyncAtLogoffText(
|
||
|
int iValue,
|
||
|
LPTSTR pszDest
|
||
|
)
|
||
|
{
|
||
|
LPCTSTR rgpsz[] = { TEXT("(Part sync)"),
|
||
|
TEXT("(Full sync)") };
|
||
|
|
||
|
if (iValue >= 0 && iValue < ARRAYSIZE(rgpsz))
|
||
|
lstrcpy(pszDest, rgpsz[iValue]);
|
||
|
else
|
||
|
lstrcpy(pszDest, TEXT("<unknown>"));
|
||
|
|
||
|
return pszDest;
|
||
|
}
|
||
|
|
||
|
|
||
|
LPTSTR DefCacheSizeText(
|
||
|
int iValue,
|
||
|
LPTSTR pszDest
|
||
|
)
|
||
|
{
|
||
|
wsprintf(pszDest, TEXT("(%2d.%02d %%)"), iValue / 100, iValue % 100);
|
||
|
return pszDest;
|
||
|
}
|
||
|
|
||
|
|
||
|
LPCTSTR BoolText(bool b)
|
||
|
{
|
||
|
static const LPCTSTR rgYN[] = { TEXT("No"), TEXT("Yes") };
|
||
|
return rgYN[int(b)];
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
LPCTSTR
|
||
|
RegValStr(
|
||
|
HKEY hkeyRoot,
|
||
|
LPCTSTR pszSubkey,
|
||
|
LPCTSTR pszValue,
|
||
|
LPTSTR pszDest
|
||
|
)
|
||
|
{
|
||
|
DWORD dwValue = 0;
|
||
|
DWORD cbData = sizeof(dwValue);
|
||
|
DWORD dwType;
|
||
|
|
||
|
DWORD dwError = SHGetValue(hkeyRoot,
|
||
|
pszSubkey,
|
||
|
pszValue,
|
||
|
&dwType,
|
||
|
(LPVOID)&dwValue,
|
||
|
&cbData);
|
||
|
|
||
|
if (ERROR_SUCCESS == dwError)
|
||
|
{
|
||
|
wsprintf(pszDest, TEXT("%d"), dwValue);
|
||
|
}
|
||
|
else if (ERROR_FILE_NOT_FOUND == dwError)
|
||
|
{
|
||
|
lstrcpy(pszDest, TEXT("<none>"));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
wsprintf(pszDest, TEXT("Err %d"), dwError);
|
||
|
}
|
||
|
return pszDest;
|
||
|
}
|
||
|
|
||
|
|
||
|
LPTSTR GetOsVersionInfoText(
|
||
|
LPTSTR pszDest
|
||
|
)
|
||
|
{
|
||
|
OSVERSIONINFO osvi;
|
||
|
ZeroMemory(&osvi, sizeof(osvi));
|
||
|
osvi.dwOSVersionInfoSize = sizeof(osvi);
|
||
|
|
||
|
*pszDest = TEXT('\0');
|
||
|
if (GetVersionEx(&osvi))
|
||
|
{
|
||
|
static const TCHAR szUnknown[] = TEXT("Unknown OS");
|
||
|
static const struct
|
||
|
{
|
||
|
DWORD dwPlatform;
|
||
|
DWORD dwMinorVersion;
|
||
|
LPCTSTR pszPlatform;
|
||
|
|
||
|
} rgPlatforms[] = {{ VER_PLATFORM_WIN32s, (DWORD)-1, TEXT("Win32s") },
|
||
|
{ VER_PLATFORM_WIN32_WINDOWS, 0, TEXT("Windows 95") },
|
||
|
{ VER_PLATFORM_WIN32_WINDOWS, 10, TEXT("Windows 98") },
|
||
|
{ VER_PLATFORM_WIN32_NT, (DWORD)-1, TEXT("Windows NT") }};
|
||
|
|
||
|
static const struct
|
||
|
{
|
||
|
DWORD dwOS;
|
||
|
LPCTSTR pszOS;
|
||
|
|
||
|
} rgOS[] = {{ OS_WIN2000TERMINAL, TEXT("Windows 2000 Terminal") },
|
||
|
{ OS_WIN2000PRO, TEXT("Windows 2000 Professional") },
|
||
|
{ OS_WIN2000ADVSERVER, TEXT("Windows 2000 Advanced Server") },
|
||
|
{ OS_WIN2000DATACENTER, TEXT("Windows 2000 Data Center") },
|
||
|
{ OS_WIN2000SERVER, TEXT("Windows 2000 Server") },
|
||
|
{ OS_WIN2000, TEXT("Windows 2000") },
|
||
|
{ OS_NT5, TEXT("Windows NT") },
|
||
|
{ OS_NT4, TEXT("Windows NT") },
|
||
|
{ OS_NT, TEXT("Windows NT") },
|
||
|
{ OS_MEMPHIS_GOLD, TEXT("Windows 98") },
|
||
|
{ OS_MEMPHIS, TEXT("Windows 98 (beta)") },
|
||
|
{ OS_WIN95, TEXT("Windows 95") },
|
||
|
{ OS_WINDOWS, TEXT("Windows") }};
|
||
|
|
||
|
|
||
|
LPCTSTR pszPlatform = szUnknown;
|
||
|
int i = 0;
|
||
|
|
||
|
//
|
||
|
// IsOS() is the shlwapi API for figuring out the OS type. Currently, it
|
||
|
// provides better granularity than GetVersionEx.
|
||
|
//
|
||
|
for (i = 0; i < ARRAYSIZE(rgOS); i++)
|
||
|
{
|
||
|
if (IsOS(rgOS[i].dwOS))
|
||
|
{
|
||
|
pszPlatform = rgOS[i].pszOS;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
//
|
||
|
// If IsOS() wasn't able to determine the platform, take the platform from
|
||
|
// the GetVersionEx results.
|
||
|
//
|
||
|
if (szUnknown == pszPlatform)
|
||
|
{
|
||
|
for (i = 0; i < ARRAYSIZE(rgPlatforms); i++)
|
||
|
{
|
||
|
if (rgPlatforms[i].dwPlatform == osvi.dwPlatformId)
|
||
|
{
|
||
|
if ((DWORD)-1 == rgPlatforms[i].dwMinorVersion ||
|
||
|
osvi.dwMinorVersion == rgPlatforms[i].dwMinorVersion)
|
||
|
{
|
||
|
pszPlatform = rgPlatforms[i].pszPlatform;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
wsprintf(pszDest, TEXT("%s version %d.%d %s build %d\n\n"), pszPlatform,
|
||
|
osvi.dwMajorVersion,
|
||
|
osvi.dwMinorVersion,
|
||
|
osvi.szCSDVersion,
|
||
|
osvi.dwBuildNumber);
|
||
|
}
|
||
|
return pszDest;
|
||
|
}
|
||
|
|
||
|
|
||
|
void DumpRegStats(void)
|
||
|
{
|
||
|
TCHAR szScratch[MAX_PATH];
|
||
|
CConfig &config = CConfig::GetSingleton();
|
||
|
|
||
|
typedef bool (CConfig::*PBMF)(bool *bSetByPolicy) const;
|
||
|
typedef int (CConfig::*PIMF)(bool *bSetByPolicy) const;
|
||
|
typedef LPTSTR (*PDF)(int iValue, LPTSTR pszDest);
|
||
|
|
||
|
struct
|
||
|
{
|
||
|
LPCTSTR pszTitle;
|
||
|
DWORD dwReg;
|
||
|
HKEY hkeyRoot;
|
||
|
LPCTSTR pszSubkey;
|
||
|
|
||
|
} rgRegKeys[] = {{ TEXT("Policy (LM)"), POL_LM, HKEY_LOCAL_MACHINE, g_szKeyPolicy },
|
||
|
{ TEXT("Policy (CU)"), POL_CU, HKEY_CURRENT_USER, g_szKeyPolicy },
|
||
|
{ TEXT("Pref (LM)"), PREF_LM, HKEY_LOCAL_MACHINE, g_szKeyPrefs },
|
||
|
{ TEXT("Pref (CU)"), PREF_CU, HKEY_CURRENT_USER, g_szKeyPrefs }};
|
||
|
|
||
|
struct
|
||
|
{
|
||
|
LPCTSTR pszValue;
|
||
|
DWORD dwReg;
|
||
|
PBMF pfnBool;
|
||
|
PIMF pfnInt;
|
||
|
PDF pfnDesc;
|
||
|
|
||
|
} rgRegValues[] = {
|
||
|
{ g_szValDefCacheSize, POL_LM, NULL, (PIMF)&CConfig::DefaultCacheSize, &DefCacheSizeText },
|
||
|
{ g_szValEnabled, POL_LM, &CConfig::CscEnabled, NULL, NULL },
|
||
|
{ g_szValNoConfigCache, POL_BOTH, &CConfig::NoConfigCache, NULL, NULL },
|
||
|
{ g_szValNoCacheViewer, POL_BOTH, &CConfig::NoCacheViewer, NULL, NULL },
|
||
|
{ g_szValNoMakeAvailableOffline, POL_BOTH, &CConfig::NoMakeAvailableOffline, NULL, NULL },
|
||
|
{ g_szValGoOfflineAction, PREF_CU | POL_BOTH, NULL, &CConfig::GoOfflineAction, &GoOfflineActionText },
|
||
|
{ g_szValEventLoggingLevel, REG_ALL, NULL, &CConfig::EventLoggingLevel, NULL },
|
||
|
{ g_szValFirstPinWizardShown, PREF_CU, (PBMF)&CConfig::FirstPinWizardShown, NULL, NULL },
|
||
|
{ g_szValNoReminders, POL_BOTH | PREF_CU, &CConfig::NoReminders, NULL, NULL },
|
||
|
{ g_szValPurgeAtLogoff, POL_LM, &CConfig::PurgeAtLogoff, NULL, NULL },
|
||
|
{ g_szValSyncAtLogoff, POL_BOTH | PREF_CU, NULL, &CConfig::SyncAtLogoff, &SyncAtLogoffText },
|
||
|
{ g_szValInitialBalloonTimeout, POL_BOTH, NULL, &CConfig::InitialBalloonTimeoutSeconds, NULL },
|
||
|
{ g_szValReminderBalloonTimeout, POL_BOTH, NULL, &CConfig::ReminderBalloonTimeoutSeconds, NULL },
|
||
|
{ g_szValReminderFreqMinutes, POL_BOTH | PREF_CU, NULL, &CConfig::ReminderFreqMinutes, NULL },
|
||
|
{ g_szValAlwaysPinSubFolders, POL_LM, &CConfig::AlwaysPinSubFolders, NULL, NULL }
|
||
|
};
|
||
|
|
||
|
|
||
|
_tprintf(TEXT("Registry Information:\n\n"));
|
||
|
_tprintf(TEXT("%-30s%15s%15s%15s%15s%15s\n"),
|
||
|
TEXT("Value"),
|
||
|
rgRegKeys[0].pszTitle,
|
||
|
rgRegKeys[1].pszTitle,
|
||
|
rgRegKeys[2].pszTitle,
|
||
|
rgRegKeys[3].pszTitle,
|
||
|
TEXT("Result"));
|
||
|
|
||
|
for (int iVal = 0; iVal < ARRAYSIZE(rgRegValues); iVal++)
|
||
|
{
|
||
|
PBMF pfnBool = rgRegValues[iVal].pfnBool;
|
||
|
PIMF pfnInt = rgRegValues[iVal].pfnInt;
|
||
|
|
||
|
_tprintf(TEXT("%-30s"), rgRegValues[iVal].pszValue);
|
||
|
|
||
|
for (int iKey = 0; iKey < ARRAYSIZE(rgRegKeys); iKey++)
|
||
|
{
|
||
|
if (rgRegValues[iVal].dwReg & rgRegKeys[iKey].dwReg)
|
||
|
{
|
||
|
_tprintf(TEXT("%15s"), RegValStr(rgRegKeys[iKey].hkeyRoot,
|
||
|
rgRegKeys[iKey].pszSubkey,
|
||
|
rgRegValues[iVal].pszValue,
|
||
|
szScratch));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
_tprintf(TEXT("%15s"), g_szNA);
|
||
|
}
|
||
|
}
|
||
|
int iValue = 0;
|
||
|
if (NULL != pfnBool)
|
||
|
iValue = (config.*pfnBool)(NULL);
|
||
|
else
|
||
|
iValue = (config.*pfnInt)(NULL);
|
||
|
|
||
|
_tprintf(TEXT("%15d"), iValue);
|
||
|
if (NULL != rgRegValues[iVal].pfnDesc)
|
||
|
_tprintf(TEXT(" %s"), (*rgRegValues[iVal].pfnDesc)(iValue, szScratch));
|
||
|
_tprintf(TEXT("\n"));
|
||
|
}
|
||
|
_tprintf(TEXT("\n"));
|
||
|
|
||
|
CConfig::OfflineActionIter iter = config.CreateOfflineActionIter();
|
||
|
CConfig::OfflineActionInfo oai;
|
||
|
_tprintf(TEXT("Offline action exceptions. Default is %s:\n\n"), GoOfflineActionText(config.GoOfflineAction(), szScratch));
|
||
|
_tprintf(TEXT("%-30s%s\n"), TEXT("Server"), TEXT("Action"));
|
||
|
|
||
|
while(S_OK == iter.Next(&oai))
|
||
|
{
|
||
|
_tprintf(TEXT("%-30s%s\n"), oai.szServer, GoOfflineActionText(oai.iAction, szScratch));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
LPTSTR GetMachineName(
|
||
|
LPTSTR pszDest,
|
||
|
UINT cchDest
|
||
|
)
|
||
|
{
|
||
|
ULONG cchComputer = cchDest;
|
||
|
GetComputerName(pszDest, &cchComputer);
|
||
|
return pszDest;
|
||
|
}
|
||
|
|
||
|
LPTSTR FormatDateTime(
|
||
|
const SYSTEMTIME& time,
|
||
|
LPTSTR pszDest,
|
||
|
UINT cchDest
|
||
|
)
|
||
|
{
|
||
|
LPTSTR pszWrite = pszDest;
|
||
|
|
||
|
GetDateFormat(LOCALE_USER_DEFAULT,
|
||
|
DATE_SHORTDATE,
|
||
|
&time,
|
||
|
NULL,
|
||
|
pszWrite,
|
||
|
cchDest);
|
||
|
|
||
|
lstrcat(pszWrite, TEXT(" "));
|
||
|
int len = lstrlen(pszWrite);
|
||
|
pszWrite += len;
|
||
|
cchDest -= len;
|
||
|
GetTimeFormat(LOCALE_USER_DEFAULT,
|
||
|
LOCALE_NOUSEROVERRIDE,
|
||
|
&time,
|
||
|
NULL,
|
||
|
pszWrite,
|
||
|
cchDest);
|
||
|
|
||
|
return pszDest;
|
||
|
}
|
||
|
|
||
|
|
||
|
LPTSTR GetCurrentDateTime(
|
||
|
LPTSTR pszDest,
|
||
|
UINT cchDest
|
||
|
)
|
||
|
{
|
||
|
SYSTEMTIME now;
|
||
|
GetLocalTime(&now);
|
||
|
return FormatDateTime(now, pszDest, cchDest);
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
LPTSTR
|
||
|
Int64ToCommaSepString(
|
||
|
LONGLONG n,
|
||
|
LPTSTR pszOut,
|
||
|
int cchOut
|
||
|
)
|
||
|
{
|
||
|
ULONG ulTemp;
|
||
|
UNICODE_STRING s;
|
||
|
NUMBERFMTW nfmtW;
|
||
|
LPWSTR pszFmtOutW;
|
||
|
int cchFmtOut;
|
||
|
WCHAR szTextW[30];
|
||
|
WCHAR szSep[5];
|
||
|
//
|
||
|
// Convert the 64-bit int to a text string.
|
||
|
//
|
||
|
s.Length = 0;
|
||
|
s.MaximumLength = ARRAYSIZE(szTextW);
|
||
|
s.Buffer = szTextW;
|
||
|
RtlInt64ToUnicodeString(n, 10, &s);
|
||
|
//
|
||
|
// Format the number with commas according to locale conventions.
|
||
|
//
|
||
|
nfmtW.NumDigits = 0;
|
||
|
nfmtW.LeadingZero = 0;
|
||
|
GetLocaleInfoW(LOCALE_USER_DEFAULT, LOCALE_SGROUPING, szSep, ARRAYSIZE(szSep));
|
||
|
RtlInitUnicodeString(&s, szSep);
|
||
|
RtlUnicodeStringToInteger(&s, 10, &ulTemp);
|
||
|
nfmtW.Grouping = UINT(ulTemp);
|
||
|
GetLocaleInfoW(LOCALE_USER_DEFAULT, LOCALE_STHOUSAND, szSep, ARRAYSIZE(szSep));
|
||
|
nfmtW.lpDecimalSep = nfmtW.lpThousandSep = szSep;
|
||
|
nfmtW.NegativeOrder = 0;
|
||
|
|
||
|
#ifndef UNICODE
|
||
|
//
|
||
|
// If ansi build we need a wide-char buffer as format destination.
|
||
|
//
|
||
|
WCHAR szNumW[30];
|
||
|
pszFmtOutW = szNumW;
|
||
|
cchFmtOut = ARRAYSIZE(szNumW);
|
||
|
#else
|
||
|
//
|
||
|
// If unicode build we can format directly to the destination buffer.
|
||
|
//
|
||
|
pszFmtOutW = pszOut;
|
||
|
cchFmtOut = cchOut;
|
||
|
#endif
|
||
|
|
||
|
GetNumberFormatW(LOCALE_USER_DEFAULT,
|
||
|
0,
|
||
|
szTextW,
|
||
|
&nfmtW,
|
||
|
pszFmtOutW,
|
||
|
cchFmtOut);
|
||
|
#ifndef UNICODE
|
||
|
//
|
||
|
// If ansi build, need extra step to convert formatted number string
|
||
|
// (wide char) back to ansi.
|
||
|
//
|
||
|
WideCharToMultiByte(CP_ACP,
|
||
|
0,
|
||
|
pszFmtOutW,
|
||
|
-1,
|
||
|
pszOut,
|
||
|
cchOut,
|
||
|
NULL,
|
||
|
NULL);
|
||
|
#endif
|
||
|
return pszOut;
|
||
|
}
|
||
|
|
||
|
|
||
|
void PrintFileStatusFlags(void)
|
||
|
{
|
||
|
struct
|
||
|
{
|
||
|
DWORD dwFlags;
|
||
|
LPCTSTR pszText;
|
||
|
|
||
|
} rgMap[] = {{ FLAG_CSC_COPY_STATUS_DATA_LOCALLY_MODIFIED, TEXT("Data locally modified") },
|
||
|
{ FLAG_CSC_COPY_STATUS_ATTRIB_LOCALLY_MODIFIED, TEXT("Attrib locally modified") },
|
||
|
{ FLAG_CSC_COPY_STATUS_TIME_LOCALLY_MODIFIED, TEXT("Time locally modified") },
|
||
|
{ FLAG_CSC_COPY_STATUS_STALE, TEXT("Stale") },
|
||
|
{ FLAG_CSC_COPY_STATUS_LOCALLY_DELETED, TEXT("Locally deleted") },
|
||
|
{ FLAG_CSC_COPY_STATUS_SPARSE, TEXT("Sparse") },
|
||
|
{ FLAG_CSC_COPY_STATUS_ORPHAN, TEXT("Orphan") },
|
||
|
{ FLAG_CSC_COPY_STATUS_SUSPECT, TEXT("Suspect") },
|
||
|
{ FLAG_CSC_COPY_STATUS_LOCALLY_CREATED, TEXT("Locally created") },
|
||
|
{ 0x00010000, TEXT("User has READ access") },
|
||
|
{ 0x00020000, TEXT("User has WRITE access") },
|
||
|
{ 0x00040000, TEXT("Guest has READ access") },
|
||
|
{ 0x00080000, TEXT("Guest has WRITE access") },
|
||
|
{ 0x00100000, TEXT("Other has READ access") },
|
||
|
{ 0x00200000, TEXT("Other has WRITE access") },
|
||
|
{ FLAG_CSC_COPY_STATUS_IS_FILE, TEXT("Entry is a file") },
|
||
|
{ FLAG_CSC_COPY_STATUS_FILE_IN_USE, TEXT("File in use") },
|
||
|
};
|
||
|
|
||
|
_tprintf(TEXT("\nFile status flags ------------------------\n\n"));
|
||
|
for (int i = 0; i < ARRAYSIZE(rgMap); i++)
|
||
|
{
|
||
|
_tprintf(TEXT("0x%08X %s\n"), rgMap[i].dwFlags, rgMap[i].pszText);
|
||
|
}
|
||
|
_tprintf(TEXT("\n\n"));
|
||
|
}
|
||
|
|
||
|
|
||
|
void PrintShareStatusFlags(void)
|
||
|
{
|
||
|
struct
|
||
|
{
|
||
|
DWORD dwFlags;
|
||
|
LPCTSTR pszText;
|
||
|
|
||
|
} rgMap[] = {{ FLAG_CSC_SHARE_STATUS_MANUAL_REINT, TEXT("Manual Caching") }, // 0x00000000
|
||
|
{ FLAG_CSC_SHARE_STATUS_MODIFIED_OFFLINE, TEXT("Modified offline") }, // 0x00000001
|
||
|
{ FLAG_CSC_SHARE_STATUS_AUTO_REINT, TEXT("Auto Caching") }, // 0x00000040
|
||
|
{ FLAG_CSC_SHARE_STATUS_VDO, TEXT("Virtually Disconnected Ops") }, // 0x00000080
|
||
|
{ FLAG_CSC_SHARE_STATUS_NO_CACHING, TEXT("No Caching") }, // 0x000000C0
|
||
|
{ FLAG_CSC_SHARE_STATUS_FINDS_IN_PROGRESS, TEXT("Finds in progress") }, // 0x00000200
|
||
|
{ FLAG_CSC_SHARE_STATUS_FILES_OPEN, TEXT("Open files") }, // 0x00000400
|
||
|
{ FLAG_CSC_SHARE_STATUS_CONNECTED, TEXT("Connected") }, // 0x00000800
|
||
|
{ FLAG_CSC_SHARE_MERGING, TEXT("Merging") }, // 0x40000000
|
||
|
{ FLAG_CSC_SHARE_STATUS_DISCONNECTED_OP, TEXT("Disconnected Op") }, // 0x80000000
|
||
|
};
|
||
|
|
||
|
_tprintf(TEXT("\nShare status flags -------------------\n\n"));
|
||
|
for (int i = 0; i < ARRAYSIZE(rgMap); i++)
|
||
|
{
|
||
|
_tprintf(TEXT("%30s0x%08X %s\n"), TEXT(""), rgMap[i].dwFlags, rgMap[i].pszText);
|
||
|
}
|
||
|
_tprintf(TEXT("\n\n"));
|
||
|
}
|
||
|
|
||
|
|
||
|
void PrintHintFlags(void)
|
||
|
{
|
||
|
struct
|
||
|
{
|
||
|
DWORD dwFlags;
|
||
|
LPCTSTR pszText;
|
||
|
|
||
|
} rgMap[] = {{ FLAG_CSC_HINT_PIN_USER, TEXT("Pin User") }, // 0x00000001
|
||
|
{ FLAG_CSC_HINT_PIN_INHERIT_USER, TEXT("Pin Inherit User") }, // 0x00000002
|
||
|
{ FLAG_CSC_HINT_PIN_INHERIT_SYSTEM, TEXT("Pin Inherit System") }, // 0x00000004
|
||
|
{ FLAG_CSC_HINT_CONSERVE_BANDWIDTH, TEXT("Conserve Bandwidth") }, // 0x00000008
|
||
|
{ FLAG_CSC_HINT_PIN_SYSTEM, TEXT("Pin System") }, // 0x00000010
|
||
|
};
|
||
|
|
||
|
_tprintf(TEXT("\nFile hint flags -------------------\n\n"));
|
||
|
for (int i = 0; i < ARRAYSIZE(rgMap); i++)
|
||
|
{
|
||
|
_tprintf(TEXT("0x%08X %s\n"), rgMap[i].dwFlags, rgMap[i].pszText);
|
||
|
}
|
||
|
_tprintf(TEXT("\n\n"));
|
||
|
}
|
||
|
|
||
|
|
||
|
void DumpCacheStats(void)
|
||
|
{
|
||
|
DWORD dwStatus;
|
||
|
WIN32_FIND_DATA fd;
|
||
|
TCHAR szScratch[MAX_PATH];
|
||
|
BOOL bEnabled = CSCIsCSCEnabled();
|
||
|
CSCSPACEUSAGEINFO sui;
|
||
|
|
||
|
ZeroMemory(&sui, sizeof(sui));
|
||
|
GetCscSpaceUsageInfo(&sui);
|
||
|
|
||
|
_tprintf(TEXT("Cache information:\n\n"));
|
||
|
|
||
|
_tprintf(TEXT("CSC enabled.........: %s\n"), BoolText(boolify(CSCIsCSCEnabled())));
|
||
|
_tprintf(TEXT("Volume..............: %s\n"), sui.szVolume);
|
||
|
_tprintf(TEXT("Bytes on volume.....: %s\n"), Int64ToCommaSepString(sui.llBytesOnVolume, szScratch, ARRAYSIZE(szScratch)));
|
||
|
_tprintf(TEXT("Bytes in cache......: %s\n"), Int64ToCommaSepString(sui.llBytesTotalInCache, szScratch, ARRAYSIZE(szScratch)));
|
||
|
_tprintf(TEXT("Bytes used in cache.: %s\n"), Int64ToCommaSepString(sui.llBytesUsedInCache, szScratch, ARRAYSIZE(szScratch)));
|
||
|
_tprintf(TEXT("Files in cache......: %d\n"), sui.dwNumFilesInCache);
|
||
|
_tprintf(TEXT("Directories in cache: %d\n\n"), sui.dwNumDirsInCache);
|
||
|
|
||
|
CCscFindHandle hFind(CacheFindFirst(NULL, &fd, &dwStatus, NULL, NULL, NULL));
|
||
|
if (hFind.IsValid())
|
||
|
{
|
||
|
BOOL bResult = TRUE;
|
||
|
CSCSHARESTATS ss;
|
||
|
CSCCACHESTATS cs;
|
||
|
CSCGETSTATSINFO si = { SSEF_NONE, SSUF_NONE, true, false };
|
||
|
|
||
|
ZeroMemory(&cs, sizeof(cs));
|
||
|
|
||
|
_tprintf(TEXT("%-30s%-12s%10s%10s%10s%10s%10s%10s%10s%10s%10s%12s\n"),
|
||
|
TEXT("Share"),
|
||
|
TEXT("Status"),
|
||
|
TEXT("Files"),
|
||
|
TEXT("Dirs"),
|
||
|
TEXT("Pinned"),
|
||
|
TEXT("Modified"),
|
||
|
TEXT("Sparse"),
|
||
|
TEXT("USER"),
|
||
|
TEXT("GUEST"),
|
||
|
TEXT("OTHER"),
|
||
|
TEXT("Offline?"),
|
||
|
TEXT("OpenFiles?"));
|
||
|
do
|
||
|
{
|
||
|
cs.cShares++;
|
||
|
if (bResult = _GetShareStatistics(fd.cFileName,
|
||
|
&si,
|
||
|
&ss))
|
||
|
{
|
||
|
_tprintf(TEXT("%-30s0x%08X %10d%10d%10d%10d%10d%10d%10d%10d%10s%12s\n"),
|
||
|
fd.cFileName,
|
||
|
dwStatus,
|
||
|
ss.cTotal - ss.cDirs,
|
||
|
ss.cDirs,
|
||
|
ss.cPinned,
|
||
|
ss.cModified,
|
||
|
ss.cSparse,
|
||
|
ss.cAccessUser,
|
||
|
ss.cAccessGuest,
|
||
|
ss.cAccessOther,
|
||
|
BoolText(ss.bOffline),
|
||
|
BoolText(ss.bOpenFiles));
|
||
|
|
||
|
cs.cTotal += ss.cTotal;
|
||
|
cs.cPinned += ss.cPinned;
|
||
|
cs.cModified += ss.cModified;
|
||
|
cs.cSparse += ss.cSparse;
|
||
|
cs.cDirs += ss.cDirs;
|
||
|
cs.cAccessUser += ss.cAccessUser;
|
||
|
cs.cAccessGuest += ss.cAccessGuest;
|
||
|
cs.cAccessOther += ss.cAccessOther;
|
||
|
cs.cSharesOffline += int(ss.bOffline);
|
||
|
cs.cSharesWithOpenFiles += int(ss.bOpenFiles);
|
||
|
}
|
||
|
}
|
||
|
while(bResult && CacheFindNext(hFind, &fd, &dwStatus, NULL, NULL, NULL));
|
||
|
|
||
|
_tprintf(TEXT("%-30s%-12s%10d%10d%10d%10d%10d%10d%10d%10d%10d%12d\n\n"),
|
||
|
TEXT("SUMMARY"),
|
||
|
TEXT(""),
|
||
|
cs.cTotal - cs.cDirs,
|
||
|
cs.cDirs,
|
||
|
cs.cPinned,
|
||
|
cs.cModified,
|
||
|
cs.cSparse,
|
||
|
cs.cAccessUser,
|
||
|
cs.cAccessGuest,
|
||
|
cs.cAccessOther,
|
||
|
cs.cSharesOffline,
|
||
|
cs.cSharesWithOpenFiles);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void DumpFileInformation(LPCTSTR pszFile)
|
||
|
{
|
||
|
TCHAR szExpanded[MAX_PATH*2];
|
||
|
ExpandEnvironmentStrings(pszFile, szExpanded, ARRAYSIZE(szExpanded));
|
||
|
|
||
|
HANDLE hFile = CreateFile(szExpanded,
|
||
|
GENERIC_READ,
|
||
|
FILE_SHARE_READ,
|
||
|
NULL,
|
||
|
OPEN_EXISTING,
|
||
|
FILE_ATTRIBUTE_NORMAL,
|
||
|
NULL);
|
||
|
|
||
|
if (INVALID_HANDLE_VALUE != hFile)
|
||
|
{
|
||
|
BY_HANDLE_FILE_INFORMATION fi;
|
||
|
if (GetFileInformationByHandle(hFile, &fi))
|
||
|
{
|
||
|
TCHAR szScratch[80];
|
||
|
ULARGE_INTEGER llSize = { fi.nFileSizeLow, fi.nFileSizeHigh };
|
||
|
FILETIME localfiletime;
|
||
|
SYSTEMTIME systime;
|
||
|
FileTimeToLocalFileTime(&fi.ftLastWriteTime, &localfiletime);
|
||
|
FileTimeToSystemTime(&localfiletime, &systime);
|
||
|
|
||
|
_tprintf(TEXT("File.......: %s\n"), szExpanded);
|
||
|
_tprintf(TEXT("Size.......: %s bytes\n"), Int64ToCommaSepString(llSize.QuadPart, szScratch, ARRAYSIZE(szScratch)));
|
||
|
_tprintf(TEXT("Created....: %s\n\n"), FormatDateTime(systime, szScratch, ARRAYSIZE(szScratch)));
|
||
|
}
|
||
|
|
||
|
CloseHandle(hFile);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
_tprintf(TEXT("ERROR %d opening %s\n\n"), GetLastError(), szExpanded);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
void PrintOneEnumEntry(LPCTSTR pszPath, DWORD dwStatus, DWORD dwPinCount, DWORD dwHintFlags)
|
||
|
{
|
||
|
_tprintf(TEXT("0x%08X %d 0x%08X %s\n"), dwStatus, dwPinCount, dwHintFlags, pszPath);
|
||
|
}
|
||
|
|
||
|
|
||
|
void EnumTree(LPTSTR pszPath, LPTSTR pszPathToPrint)
|
||
|
{
|
||
|
WIN32_FIND_DATA fd;
|
||
|
DWORD dwStatus, dwPinCount, dwHintFlags;
|
||
|
|
||
|
CCscFindHandle hFind(CacheFindFirst(pszPath, &fd, &dwStatus, &dwPinCount, &dwHintFlags, NULL));
|
||
|
if (hFind.IsValid())
|
||
|
{
|
||
|
do
|
||
|
{
|
||
|
if (!PathIsDotOrDotDot(pszPath))
|
||
|
{
|
||
|
const bool bIsDir = 0 != (FILE_ATTRIBUTE_DIRECTORY & fd.dwFileAttributes);
|
||
|
PathCombine(pszPathToPrint, pszPath, fd.cFileName);
|
||
|
if (bIsDir)
|
||
|
{
|
||
|
lstrcat(pszPathToPrint, TEXT(" [DIR]"));
|
||
|
}
|
||
|
PrintOneEnumEntry(pszPathToPrint, dwStatus, dwPinCount, dwHintFlags);
|
||
|
if (bIsDir)
|
||
|
{
|
||
|
PathAppend(pszPath, fd.cFileName);
|
||
|
EnumTree(pszPath, pszPathToPrint);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
while(CacheFindNext(hFind, &fd, &dwStatus, &dwPinCount, &dwHintFlags, NULL));
|
||
|
}
|
||
|
PathRemoveFileSpec(pszPath);
|
||
|
}
|
||
|
|
||
|
|
||
|
void EnumFiles(void)
|
||
|
{
|
||
|
WIN32_FIND_DATA fd;
|
||
|
DWORD dwStatus, dwPinCount, dwHintFlags;
|
||
|
|
||
|
_tprintf(TEXT("Status PinCnt Hints Name\n"));
|
||
|
CCscFindHandle hFind(CacheFindFirst(NULL, &fd, &dwStatus, &dwPinCount, &dwHintFlags, NULL));
|
||
|
if (hFind.IsValid())
|
||
|
{
|
||
|
do
|
||
|
{
|
||
|
//
|
||
|
// We create only one path and print-path buffer that we pass
|
||
|
// into the recursive EnumTree() function. This way we don't have
|
||
|
// a path buffer on each call stack as we recurse.
|
||
|
//
|
||
|
TCHAR szPath[MAX_PATH * 2]; // Working path buffer.
|
||
|
TCHAR szPathToPrint[MAX_PATH * 2]; // For printing only.
|
||
|
|
||
|
wsprintf(szPathToPrint, TEXT("%s [SHARE]"), fd.cFileName);
|
||
|
PrintOneEnumEntry(szPathToPrint, dwStatus, dwPinCount, dwHintFlags);
|
||
|
|
||
|
lstrcpyn(szPath, fd.cFileName, ARRAYSIZE(szPath));
|
||
|
EnumTree(szPath, szPathToPrint);
|
||
|
}
|
||
|
while(CacheFindNext(hFind, &fd, &dwStatus, &dwPinCount, &dwHintFlags, NULL));
|
||
|
}
|
||
|
_tprintf(TEXT("\n\n"));
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
void ShowUsage(void)
|
||
|
{
|
||
|
_tprintf(TEXT("\aUsage: cscsniff [-f] [-c] [-r] [-e] [-a]\n\n"));
|
||
|
_tprintf(TEXT("\t-f = Show file information.\n"));
|
||
|
_tprintf(TEXT("\t-c = Show cache information.\n"));
|
||
|
_tprintf(TEXT("\t-r = Show registry information.\n"));
|
||
|
_tprintf(TEXT("\t-e = Enumerate all files.\n"));
|
||
|
_tprintf(TEXT("\t-a = Show ALL output.\n\n"));
|
||
|
_tprintf(TEXT("Default is -f -c -r\n"));
|
||
|
}
|
||
|
|
||
|
|
||
|
void __cdecl main(int argc, char **argv)
|
||
|
{
|
||
|
|
||
|
// if (!IsCurrentUserAnAdminMember())
|
||
|
// {
|
||
|
// _tprintf(TEXT("\aYou must be an administrator on this computer to run cscsniff.\n"));
|
||
|
// return;
|
||
|
// }
|
||
|
|
||
|
const char chDash = '-';
|
||
|
const char chSlash = '/';
|
||
|
|
||
|
const DWORD SHOW_CACHEINFO = 0x00000001;
|
||
|
const DWORD SHOW_REGINFO = 0x00000002;
|
||
|
const DWORD SHOW_FILEINFO = 0x00000004;
|
||
|
const DWORD SHOW_ENUMFILES = 0x00000008;
|
||
|
const DWORD SHOW_DEFAULT = SHOW_CACHEINFO | SHOW_REGINFO | SHOW_FILEINFO;
|
||
|
const DWORD SHOW_ALL = SHOW_DEFAULT | SHOW_ENUMFILES;
|
||
|
|
||
|
DWORD dwShow = 0;
|
||
|
|
||
|
for (int i = 1; i < argc; i++)
|
||
|
{
|
||
|
if (chDash == argv[i][0] || chSlash == argv[i][0])
|
||
|
{
|
||
|
switch(argv[i][1])
|
||
|
{
|
||
|
case 'C':
|
||
|
case 'c':
|
||
|
dwShow |= SHOW_CACHEINFO;
|
||
|
break;
|
||
|
|
||
|
case 'R':
|
||
|
case 'r':
|
||
|
dwShow |= SHOW_REGINFO;
|
||
|
break;
|
||
|
|
||
|
case 'F':
|
||
|
case 'f':
|
||
|
dwShow |= SHOW_FILEINFO;
|
||
|
break;
|
||
|
|
||
|
case 'E':
|
||
|
case 'e':
|
||
|
dwShow |= SHOW_ENUMFILES;
|
||
|
break;
|
||
|
|
||
|
case 'A':
|
||
|
case 'a':
|
||
|
dwShow |= SHOW_ALL;
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
ShowUsage();
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (0 == dwShow)
|
||
|
{
|
||
|
dwShow = SHOW_DEFAULT;
|
||
|
}
|
||
|
|
||
|
|
||
|
TCHAR szScratch[MAX_PATH];
|
||
|
TCHAR szComputer[MAX_COMPUTERNAME_LENGTH + 1];
|
||
|
TCHAR szDateTime[80];
|
||
|
_tprintf(TEXT("Status of CSC for %s (%s)\n\n"),
|
||
|
GetMachineName(szComputer, ARRAYSIZE(szComputer)),
|
||
|
GetCurrentDateTime(szDateTime, ARRAYSIZE(szDateTime)));
|
||
|
|
||
|
_tprintf(TEXT("Operating system: %s\n\n"), GetOsVersionInfoText(szScratch));
|
||
|
|
||
|
if (dwShow & SHOW_FILEINFO)
|
||
|
{
|
||
|
_tprintf(TEXT("Binary file information:\n\n"));
|
||
|
DumpFileInformation(TEXT("%systemroot%\\system32\\CSCDLL.DLL"));
|
||
|
DumpFileInformation(TEXT("%systemroot%\\system32\\CSCUI.DLL"));
|
||
|
}
|
||
|
|
||
|
if (dwShow & SHOW_CACHEINFO)
|
||
|
{
|
||
|
DumpCacheStats();
|
||
|
PrintShareStatusFlags();
|
||
|
}
|
||
|
if (dwShow & SHOW_ENUMFILES)
|
||
|
{
|
||
|
EnumFiles();
|
||
|
PrintFileStatusFlags();
|
||
|
PrintHintFlags();
|
||
|
}
|
||
|
|
||
|
if (dwShow & SHOW_REGINFO)
|
||
|
DumpRegStats();
|
||
|
}
|
||
|
|
||
|
|
||
|
|