windows-nt/Source/XPSP1/NT/shell/ext/cscui/sniff/main.cpp

857 lines
30 KiB
C++
Raw Permalink Normal View History

2020-09-26 03:20:57 -05:00
#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();
}