// Microsoft Windows // Copyright (C) Microsoft Corporation, 1997 - 1999 // File: update.cpp // Authors; // Jeff Saathoff (jeffreys) // Notes; // SyncMgr integration #include "pch.h" #include "msgbox.h" // CscWin32Message #include "folder.h" #include // OpenOfflineFile #include "cscst.h" // PostToSystray #include "uihooks.h" // Self-host notifications #include "fopendlg.h" // OpenFilesWarningDialog #include "statdlg.h" // ReconnectServers #include "security.h" #define RAS_CONNECT_DELAY (10 * 1000) // Maximum length of username #define MAX_USERNAME_CHARS 64 // SYNCTHREADDATA.dwSyncStatus flags #define SDS_SYNC_OUT 0x00000001 // CSCMergeShare #define SDS_SYNC_IN_QUICK 0x00000002 // CSCFillSparseFiles(FALSE) #define SDS_SYNC_IN_FULL 0x00000004 // CSCFillSparseFiles(TRUE) #define SDS_SYNC_FORCE_INWARD 0x00000008 #define SDS_SYNC_RAS_CONNECTED 0x00000010 #define SDS_SYNC_RESTART_MERGE 0x00000020 #define SDS_SYNC_DELETE_DELETE 0x00000040 #define SDS_SYNC_DELETE_RESTORE 0x00000080 #define SDS_SYNC_AUTOCACHE 0x00000100 #define SDS_SYNC_CONFLICT_KEEPLOCAL 0x00000200 #define SDS_SYNC_CONFLICT_KEEPNET 0x00000400 #define SDS_SYNC_CONFLICT_KEEPBOTH 0x00000800 #define SDS_SYNC_STARTED 0x00010000 #define SDS_SYNC_ERROR 0x00020000 #define SDS_SYNC_CANCELLED 0x00040000 #define SDS_SYNC_FILE_SKIPPED 0x00080000 #define SDS_SYNC_DELETE_CONFLICT_MASK (SDS_SYNC_DELETE_DELETE | SDS_SYNC_DELETE_RESTORE) #define SDS_SYNC_FILE_CONFLICT_MASK (SDS_SYNC_CONFLICT_KEEPLOCAL | SDS_SYNC_CONFLICT_KEEPNET | SDS_SYNC_CONFLICT_KEEPBOTH) // Sync Flags used internally by CCscUpdate #define CSC_SYNC_OUT 0x00000001L #define CSC_SYNC_IN_QUICK 0x00000002L #define CSC_SYNC_IN_FULL 0x00000004L #define CSC_SYNC_SETTINGS 0x00000008L #define CSC_SYNC_MAYBOTHERUSER 0x00000010L #define CSC_SYNC_NOTIFY_SYSTRAY 0x00000020L #define CSC_SYNC_LOGOFF 0x00000040L #define CSC_SYNC_LOGON 0x00000080L #define CSC_SYNC_IDLE 0x00000100L #define CSC_SYNC_NONET 0x00000200L #define CSC_SYNC_PINFILES 0x00000400L #define CSC_SYNC_PIN_RECURSE 0x00000800L #define CSC_SYNC_OFWARNINGDONE 0x00001000L #define CSC_SYNC_CANCELLED 0x00002000L #define CSC_SYNC_SHOWUI_ALWAYS 0x00004000L #define CSC_SYNC_IGNORE_ACCESS 0x00008000L #define CSC_SYNC_SKIP_EFS 0x00010000L #define CSC_SYNC_EFS_WARNING_SHOWN 0x00020000L #define CSC_SYNC_RECONNECT 0x00040000L #define CSC_LOCALLY_MODIFIED (FLAG_CSC_COPY_STATUS_DATA_LOCALLY_MODIFIED \ | FLAG_CSC_COPY_STATUS_LOCALLY_DELETED \ | FLAG_CSC_COPY_STATUS_LOCALLY_CREATED) HICON g_hCscIcon = NULL; // Used for marshalling data into the SyncMgr process typedef struct _CSC_UPDATE_DATA { DWORD dwUpdateFlags; DWORD dwFileBufferOffset; } CSC_UPDATE_DATA, *PCSC_UPDATE_DATA; LPTSTR GetErrorText(DWORD dwErr) { UINT idString = (UINT)-1; LPTSTR pszError = NULL; switch (dwErr) { case ERROR_INVALID_NAME: // "Files of this type cannot be made available offline." idString = IDS_CACHING_DISALLOWED; break; } if ((UINT)-1 != idString) { LoadStringAlloc(&pszError, g_hInstance, idString); } else if (NOERROR != dwErr) { FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM, NULL, dwErr, 0, (LPTSTR)&pszError, 1, NULL); } return pszError; } // CscRegisterHandler // Purpose: Register/unregister CSC Update handler with SyncMgr // Parameters: bRegister - TRUE to register, FALSE to unregister // punkSyncMgr - (optional) instance of SyncMgr to use // Return: HRESULT HRESULT CscRegisterHandler(BOOL bRegister, LPUNKNOWN punkSyncMgr) { HRESULT hr; HRESULT hrComInit = E_FAIL; ISyncMgrRegister *pSyncRegister = NULL; const DWORD dwRegFlags = SYNCMGRREGISTERFLAG_CONNECT | SYNCMGRREGISTERFLAG_PENDINGDISCONNECT; static BOOL s_bSyncHandlerRegistered = FALSE; TraceEnter(TRACE_UPDATE, "CscRegisterHandler"); if (bRegister && s_bSyncHandlerRegistered) TraceLeaveResult(S_OK); // already registered // Note: if bRegister and s_bSyncHandlerRegistered are both FALSE, unregister anyway. if (punkSyncMgr) { hr = punkSyncMgr->QueryInterface(IID_ISyncMgrRegister, (LPVOID*)&pSyncRegister); } else { hrComInit = CoInitialize(NULL); hr = CoCreateInstance(CLSID_SyncMgr, NULL, CLSCTX_SERVER, IID_ISyncMgrRegister, (LPVOID*)&pSyncRegister); } FailGracefully(hr, "Unable to get ISyncMgrRegister interface"); if (bRegister) hr = pSyncRegister->RegisterSyncMgrHandler(CLSID_CscUpdateHandler, NULL, dwRegFlags); else hr = pSyncRegister->UnregisterSyncMgrHandler(CLSID_CscUpdateHandler, dwRegFlags); if (SUCCEEDED(hr)) s_bSyncHandlerRegistered = bRegister; exit_gracefully: DoRelease(pSyncRegister); if (SUCCEEDED(hrComInit)) CoUninitialize(); TraceLeaveResult(hr); } // CscUpdateCache // Purpose: Invoke SyncMgr to update the CSC cache // Parameters: pNamelist - list of files passed to the CSC SyncMgr handler // Return: HRESULT HRESULT CscUpdateCache(DWORD dwUpdateFlags, CscFilenameList *pfnl) { HRESULT hr; HRESULT hrComInit = E_FAIL; ISyncMgrSynchronizeInvoke *pSyncInvoke = NULL; DWORD dwSyncMgrFlags = 0; ULONG cbDataLength = sizeof(CSC_UPDATE_DATA); PCSC_UPDATE_DATA pUpdateData = NULL; PCSC_NAMELIST_HDR pNamelist = NULL; TraceEnter(TRACE_UPDATE, "CscUpdateCache"); hrComInit = CoInitialize(NULL); hr = CoCreateInstance(CLSID_SyncMgr, NULL, CLSCTX_SERVER, IID_ISyncMgrSynchronizeInvoke, (LPVOID*)&pSyncInvoke); FailGracefully(hr, "Unable to create SyncMgr object"); if (dwUpdateFlags & CSC_UPDATE_SELECTION) { if (NULL == pfnl || (0 == (CSC_UPDATE_SHOWUI_ALWAYS & dwUpdateFlags) && 0 == pfnl->GetShareCount())) ExitGracefully(hr, E_INVALIDARG, "CSC_UPDATE_SELECTION with no selection"); pNamelist = pfnl->CreateListBuffer(); if (!pNamelist) ExitGracefully(hr, E_OUTOFMEMORY, "Unable to create namelist buffer"); cbDataLength += pNamelist->cbSize; } // Alloc a buffer for the cookie data pUpdateData = (PCSC_UPDATE_DATA)LocalAlloc(LPTR, cbDataLength); if (!pUpdateData) ExitGracefully(hr, E_OUTOFMEMORY, "LocalAlloc failed"); pUpdateData->dwUpdateFlags = dwUpdateFlags; if (pNamelist) { pUpdateData->dwFileBufferOffset = sizeof(CSC_UPDATE_DATA); CopyMemory(ByteOffset(pUpdateData, pUpdateData->dwFileBufferOffset), pNamelist, pNamelist->cbSize); } if (dwUpdateFlags & CSC_UPDATE_STARTNOW) dwSyncMgrFlags |= SYNCMGRINVOKE_STARTSYNC; // Start SyncMgr hr = pSyncInvoke->UpdateItems(dwSyncMgrFlags, CLSID_CscUpdateHandler, cbDataLength, (LPBYTE)pUpdateData); exit_gracefully: if (pNamelist) CscFilenameList::FreeListBuffer(pNamelist); if (pUpdateData) LocalFree(pUpdateData); DoRelease(pSyncInvoke); if (SUCCEEDED(hrComInit)) CoUninitialize(); TraceLeaveResult(hr); } // GetNewVersionName // Purpose: Create unique names for copies of a file // Parameters: LPTSTR pszUNCPath - fully qualified UNC name of file // LPTSTR pszShare - \\server\share that file lives on // LPTSTR pszDrive - drive mapping to use for net operations // LPTSTR *ppszNewName - filename for new version returned here (must free) // Return: Win32 error code DWORD GetNewVersionName(LPCTSTR pszUNCPath, LPCTSTR pszShare, LPCTSTR pszDrive, LPTSTR *ppszNewName) { DWORD dwErr = NOERROR; LPTSTR pszDriveLetterPath = NULL; LPTSTR pszPath = NULL; LPTSTR pszFile = NULL; LPTSTR pszExt = NULL; LPTSTR pszWildCardName = NULL; TCHAR szUserName[MAX_USERNAME_CHARS]; ULONG nLength; ULONG nMaxVersion = 0; ULONG cOlderVersions = 0; HANDLE hFind = INVALID_HANDLE_VALUE; WIN32_FIND_DATA fd; LPTSTR pszT; TraceEnter(TRACE_UPDATE, "GetNewVersionName"); TraceAssert(pszUNCPath != NULL); TraceAssert(ppszNewName != NULL); *ppszNewName = NULL; // 1. Split the path into components. // 2. Build wildcard name "X:\dir\foo (johndoe v*).txt" // 3. Do a findfirst/findnext loop to get the min & max version # // and count the number of old versions. // 4. Increment the max version # and build the new filename as: // "foo (johndoe v).txt" // Assume that the UNC name contains more than the share TraceAssert(!StrCmpNI(pszUNCPath, pszShare, lstrlen(pszShare))); TraceAssert(lstrlen(pszUNCPath) > lstrlen(pszShare)); // Copy the path (without \\server\share) if (!LocalAllocString(&pszPath, pszUNCPath + lstrlen(pszShare))) ExitGracefully(dwErr, ERROR_OUTOFMEMORY, "LocalAllocString failed"); // Find the file part of the name pszT = PathFindFileName(pszPath); if (!pszT) ExitGracefully(dwErr, ERROR_INVALID_PARAMETER, "Incomplete path"); // Copy the filename if (!LocalAllocString(&pszFile, pszT)) ExitGracefully(dwErr, ERROR_OUTOFMEMORY, "LocalAllocString failed"); // Look for the file extension pszT = PathFindExtension(pszFile); if (pszT) { // Copy the extension and truncate the file root at this point LocalAllocString(&pszExt, pszT); *pszT = TEXT('\0'); } // Truncate the path PathRemoveFileSpec(pszPath); // Get the user name nLength = ARRAYSIZE(szUserName); if (!GetUserName(szUserName, &nLength)) LoadString(g_hInstance, IDS_UNKNOWN_USER, szUserName, ARRAYSIZE(szUserName)); // Build the wildcard path "X:\dir\foo (johndoe v*).txt" nLength = FormatStringID(&pszWildCardName, g_hInstance, IDS_VERSION_FORMAT, pszFile, szUserName, c_szStar, pszExt); if (!nLength) ExitGracefully(dwErr, GetLastError(), "Unable to format string"); nLength += lstrlen(pszUNCPath) + lstrlen(szUserName); pszDriveLetterPath = (LPTSTR)LocalAlloc(LPTR, MAX(nLength, ULONG(MAX_PATH)) * sizeof(TCHAR)); if (!pszDriveLetterPath) ExitGracefully(dwErr, ERROR_OUTOFMEMORY, "LocalAlloc failed"); PathCombine(pszDriveLetterPath, pszDrive, pszPath); PathAppend(pszDriveLetterPath, pszWildCardName); nLength = (ULONG)(StrStr(pszWildCardName, c_szStar) - pszWildCardName); // remember where the '*' is // Search for existing versions of the file with this username hFind = FindFirstFile(pszDriveLetterPath, &fd); if (hFind != INVALID_HANDLE_VALUE) { ULONG nVersion; do { nVersion = StrToLong(&fd.cFileName[nLength]); if (nVersion > nMaxVersion) { nMaxVersion = nVersion; } cOlderVersions++; } while (FindNextFile(hFind, &fd)); FindClose(hFind); } // Build the new file name to return to the caller. // This one is version nMaxVersion+1. ULongToString(nMaxVersion+1, pszDriveLetterPath, lstrlen(pszDriveLetterPath)); nLength = FormatStringID(ppszNewName, g_hInstance, IDS_VERSION_FORMAT, pszFile, szUserName, pszDriveLetterPath, pszExt); if (!nLength) ExitGracefully(dwErr, GetLastError(), "Unable to format string"); exit_gracefully: LocalFreeString(&pszDriveLetterPath); LocalFreeString(&pszPath); LocalFreeString(&pszFile); LocalFreeString(&pszExt); LocalFreeString(&pszWildCardName); if (NOERROR != dwErr) { LocalFreeString(ppszNewName); } TraceLeaveValue(dwErr); } // ConflictDlgCallback // Purpose: Display local or remote file from conflict dialog // Parameters: hWnd - conflict dialog handle (used as parent for UI) // uMsg - one of RFCCM_* // wParam - depends on uMsg (unused) // lParam - pointer to context data (RFCDLGPARAM) // Return: TRUE on success, FALSE otherwise typedef struct _CONFLICT_DATA { LPCTSTR pszShare; LPCTSTR pszDrive; } CONFLICT_DATA; BOOL ConflictDlgCallback(HWND hWnd, UINT uMsg, WPARAM /*wParam*/, LPARAM lParam) { RFCDLGPARAM *pdlgParam = (RFCDLGPARAM*)lParam; CONFLICT_DATA cd = {0}; LPTSTR pszTmpName = NULL; ULONG cchShare = 0; LPTSTR szFile; DWORD dwErr = NOERROR; TraceEnter(TRACE_UPDATE, "ConflictDlgCallback"); if (NULL == pdlgParam) { TraceAssert(FALSE); TraceLeaveValue(FALSE); } szFile = (LPTSTR)LocalAlloc(LMEM_FIXED, MAX(StringByteSize(pdlgParam->pszLocation) + StringByteSize(pdlgParam->pszFilename), MAX_PATH_BYTES)); if (!szFile) TraceLeaveValue(FALSE); if (pdlgParam->lCallerData) cd = *(CONFLICT_DATA*)pdlgParam->lCallerData; if (cd.pszShare) cchShare = lstrlen(cd.pszShare); switch (uMsg) { case RFCCM_VIEWLOCAL: // Build UNC path and view what's in the cache PathCombine(szFile, pdlgParam->pszLocation, pdlgParam->pszFilename); dwErr = OpenOfflineFile(szFile); break; case RFCCM_VIEWNETWORK: // Build drive letter (non-UNC) path and ShellExecute it PathCombine(szFile, cd.pszDrive, pdlgParam->pszLocation + cchShare); PathAppend(szFile, pdlgParam->pszFilename); { SHELLEXECUTEINFO si = {0}; si.cbSize = sizeof(si); si.fMask = SEE_MASK_FLAG_NO_UI; si.hwnd = hWnd; si.lpFile = szFile; si.nShow = SW_NORMAL; Trace((TEXT("ShellExecuting \"%s\""), szFile)); if (!ShellExecuteEx(&si)) dwErr = GetLastError(); } break; } if (NOERROR != dwErr) CscWin32Message(hWnd, dwErr, CSCUI::SEV_ERROR); LocalFree(szFile); TraceLeaveValue(TRUE); } // ShowConflictDialog // Purpose: Invoke the conflict resolution dialog // Parameters: hWndParent - dialog parent window // pszUNCPath - full UNC of file that conflicts // pszNewName - filespec to use for new copy of file (e.g. "foo (johndoe v1).txt" // pszShare - "\\server\share" // pszDrive - "X:" drive mapping of remote connection // pfdLocal - Information about local file // pfdRemote - Information about remote file // Return: HRESULT typedef int (WINAPI *PFNSYNCMGRRESOLVECONFLICT)(HWND hWndParent, RFCDLGPARAM *pdlgParam); TCHAR const c_szSyncMgrDll[] = TEXT("mobsync.dll"); #ifdef UNICODE CHAR const c_szResolveConflict[] = "SyncMgrResolveConflictW"; #else CHAR const c_szResolveConflict[] = "SyncMgrResolveConflictA"; #endif BOOL FileHasAssociation(LPCTSTR pszFile) { HRESULT hr = HRESULT_FROM_WIN32(ERROR_NO_ASSOCIATION); if (pszFile) { pszFile = PathFindExtension(pszFile); if (pszFile && *pszFile) { IQueryAssociations *pAssoc = NULL; hr = AssocCreate(CLSID_QueryAssociations, IID_IQueryAssociations, (LPVOID*)&pAssoc); if (SUCCEEDED(hr)) { hr = pAssoc->Init(ASSOCF_IGNOREBASECLASS, pszFile, NULL, NULL); pAssoc->Release(); } } } return SUCCEEDED(hr); } int ShowConflictDialog(HWND hWndParent, LPCTSTR pszUNCPath, LPCTSTR pszNewName, LPCTSTR pszShare, LPCTSTR pszDrive, LPWIN32_FIND_DATA pfdLocal, LPWIN32_FIND_DATA pfdRemote) { int nResult = 0; TCHAR szUser[MAX_USERNAME_CHARS]; LPTSTR pszPath = NULL; LPTSTR pszFile = NULL; TCHAR szRemoteDate[MAX_PATH]; TCHAR szLocalDate[MAX_PATH]; ULONG nLength; SYSTEMTIME st; RFCDLGPARAM dp = {0}; CONFLICT_DATA cd; BOOL bLocalIsDir = FALSE; BOOL bRemoteIsDir = FALSE; static PFNSYNCMGRRESOLVECONFLICT pfnResolveConflict = NULL; TraceEnter(TRACE_UPDATE, "ShowConflictDialog"); TraceAssert(pszUNCPath); if (NULL == pfnResolveConflict) { // The CSC Update handler is loaded by SyncMgr, so assume the SyncMgr // dll is already loaded. We don't want to link to the LIB to keep // SyncMgr from loading every time our context menu or icon overlay // handler is loaded (for example). HMODULE hSyncMgrDll = GetModuleHandle(c_szSyncMgrDll); if (NULL != hSyncMgrDll) pfnResolveConflict = (PFNSYNCMGRRESOLVECONFLICT)GetProcAddress(hSyncMgrDll, c_szResolveConflict); if (NULL == pfnResolveConflict) return 0; } TraceAssert(NULL != pfnResolveConflict); szUser[0] = TEXT('\0'); nLength = ARRAYSIZE(szUser); GetUserName(szUser, &nLength); szRemoteDate[0] = TEXT('\0'); if (NULL != pfdRemote) { DWORD dwFlags = FDTF_DEFAULT; SHFormatDateTime(&pfdRemote->ftLastWriteTime, &dwFlags, szRemoteDate, ARRAYSIZE(szRemoteDate)); if (pfdRemote->dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) bRemoteIsDir = TRUE; } szLocalDate[0] = TEXT('\0'); if (NULL != pfdLocal) { DWORD dwFlags = FDTF_DEFAULT; SHFormatDateTime(&pfdLocal->ftLastWriteTime, &dwFlags, szLocalDate, ARRAYSIZE(szLocalDate)); if (pfdLocal->dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) bLocalIsDir = TRUE; } if (!LocalAllocString(&pszPath, pszUNCPath)) ExitGracefully(nResult, 0, "LocalAllocString failed"); pszFile = PathFindFileName(pszUNCPath); PathRemoveFileSpec(pszPath); dp.dwFlags = RFCF_APPLY_ALL; dp.pszFilename = pszFile; dp.pszLocation = pszPath; dp.pszNewName = pszNewName; dp.pszNetworkModifiedBy = NULL; dp.pszLocalModifiedBy = szUser; dp.pszNetworkModifiedOn = szRemoteDate; dp.pszLocalModifiedOn = szLocalDate; dp.pfnCallBack = NULL; dp.lCallerData = 0; // Only turn on the View buttons (set a callback) if we're // dealing with files that have associations. if (!(bLocalIsDir || bRemoteIsDir) && FileHasAssociation(pszFile)) { // Save both the share name and drive letter for building paths to view files cd.pszShare = pszShare; cd.pszDrive = pszDrive; dp.pfnCallBack = ConflictDlgCallback; dp.lCallerData = (LPARAM)&cd; } nResult = (*pfnResolveConflict)(hWndParent, &dp); exit_gracefully: LocalFreeString(&pszPath); // No need to free pszFile TraceLeaveValue(nResult); } // SyncMgr integration implementation // CCscUpdate::CCscUpdate() : m_cRef(1), m_ShareLog(HKEY_CURRENT_USER, c_szCSCShareKey), m_pSyncMgrCB(NULL), m_hSyncThreads(NULL), m_pFileList(NULL), m_hSyncItems(NULL), m_hwndDlgParent(NULL), m_hSyncInProgMutex(NULL), m_pConflictPinList(NULL), m_pSilentFolderList(NULL), m_pSpecialFolderList(NULL) { DllAddRef(); InitializeCriticalSection(&m_csThreadList); if (!g_hCscIcon) g_hCscIcon = LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_CSCUI_ICON)); m_hSyncMutex = CreateMutex(NULL, FALSE, c_szSyncMutex); } CCscUpdate::~CCscUpdate() { TraceEnter(TRACE_UPDATE, "CCscUpdate::~CCscUpdate"); SyncCompleted(); TraceAssert(NULL == m_hSyncInProgMutex); // We should never get here while a sync thread is still running TraceAssert(NULL == m_hSyncThreads || 0 == DPA_GetPtrCount(m_hSyncThreads)); if (NULL != m_hSyncThreads) DPA_Destroy(m_hSyncThreads); DeleteCriticalSection(&m_csThreadList); if (NULL != m_hSyncItems) DSA_Destroy(m_hSyncItems); DoRelease(m_pSyncMgrCB); delete m_pFileList; delete m_pConflictPinList; delete m_pSilentFolderList; delete m_pSpecialFolderList; if (NULL != m_hSyncMutex) CloseHandle(m_hSyncMutex); DllRelease(); TraceLeaveVoid(); } HRESULT WINAPI CCscUpdate::CreateInstance(REFIID riid, LPVOID *ppv) { HRESULT hr; CCscUpdate *pThis; TraceEnter(TRACE_UPDATE, "CCscUpdate::CreateInstance"); TraceAssert(IsCSCEnabled()); pThis = new CCscUpdate; if (pThis) { hr = pThis->QueryInterface(riid, ppv); pThis->Release(); } else hr = E_OUTOFMEMORY; TraceLeaveResult(hr); } // SyncMgr integration implementation (IUnknown) // STDMETHODIMP CCscUpdate::QueryInterface(REFIID riid, void **ppv) { static const QITAB qit[] = { QITABENT(CCscUpdate, ISyncMgrSynchronize), { 0 }, }; return QISearch(this, qit, riid, ppv); } STDMETHODIMP_(ULONG) CCscUpdate::AddRef() { return InterlockedIncrement(&m_cRef); } STDMETHODIMP_(ULONG) CCscUpdate::Release() { if (InterlockedDecrement(&m_cRef)) return m_cRef; delete this; return 0; } // Sync Manager integration implementation (ISyncMgrSynchronize) // STDMETHODIMP CCscUpdate::Initialize(DWORD /*dwReserved*/, DWORD dwSyncFlags, DWORD cbCookie, const BYTE *pCookie) { HRESULT hr = S_OK; HKEY hkCSC; BOOL bNoNet = TRUE; TraceEnter(TRACE_UPDATE, "CCscUpdate::Initialize"); TraceAssert(IsCSCEnabled()); if (!(SYNCMGRFLAG_SETTINGS & dwSyncFlags) && ::IsSyncInProgress()) { // We need to guard against running multiple syncs at the // same time. User notification in the UI is handled where // the UI code calls CscUpdate(). This is so that the UI // message contains the proper context with respect to what // the user is doing. TraceLeaveResult(E_FAIL); } m_dwSyncFlags = 0; delete m_pFileList; m_pFileList = NULL; delete m_pConflictPinList; m_pConflictPinList = NULL; // We used to get the tray status to check for NoNet, but // there's a timing problem at logon (the tray window may not // be created yet). So ask RDR instead. If this call fails, // then RDR must be dead, so bNoNet defaults to TRUE. CSCIsServerOffline(NULL, &bNoNet); switch (dwSyncFlags & SYNCMGRFLAG_EVENTMASK) { case SYNCMGRFLAG_CONNECT: // Logon if (bNoNet) ExitGracefully(hr, E_FAIL, "No Logon sync when no net"); m_dwSyncFlags = CSC_SYNC_OUT | CSC_SYNC_LOGON | CSC_SYNC_NOTIFY_SYSTRAY; // | CSC_SYNC_RECONNECT; break; case SYNCMGRFLAG_PENDINGDISCONNECT: // Logoff if (bNoNet) ExitGracefully(hr, E_FAIL, "No Logoff sync when no net"); m_dwSyncFlags = CSC_SYNC_LOGOFF; if (CConfig::eSyncFull == CConfig::GetSingleton().SyncAtLogoff()) m_dwSyncFlags |= CSC_SYNC_OUT | CSC_SYNC_IN_FULL; else m_dwSyncFlags |= CSC_SYNC_IN_QUICK; break; case SYNCMGRFLAG_INVOKE: // CscUpdateCache if (pCookie != NULL && cbCookie > 0) { PCSC_UPDATE_DATA pUpdateData = (PCSC_UPDATE_DATA)pCookie; TraceAssert(cbCookie >= sizeof(CSC_UPDATE_DATA)); DWORD dwUpdateFlags = pUpdateData->dwUpdateFlags; if (dwUpdateFlags & CSC_UPDATE_SELECTION) { TraceAssert(cbCookie > sizeof(CSC_UPDATE_DATA)); // Create the filelist from the selection provided m_pFileList = new CscFilenameList((PCSC_NAMELIST_HDR)ByteOffset(pUpdateData, pUpdateData->dwFileBufferOffset), true); if (!m_pFileList) ExitGracefully(hr, E_OUTOFMEMORY, "Unable to create CscFilenameList object"); if (!m_pFileList->IsValid()) ExitGracefully(hr, E_FAIL, "Unable to initialize CscFilenameList object"); if (CSC_UPDATE_SHOWUI_ALWAYS & dwUpdateFlags) { m_dwSyncFlags |= CSC_SYNC_SHOWUI_ALWAYS; } else if (0 == m_pFileList->GetShareCount()) ExitGracefully(hr, E_UNEXPECTED, "CSC_UPDATE_SELECTION with no selection"); } if (dwUpdateFlags & CSC_UPDATE_RECONNECT) { m_dwSyncFlags |= CSC_SYNC_RECONNECT; } if (dwUpdateFlags & CSC_UPDATE_NOTIFY_DONE) { // Caller of CscUpdateCache want's systray notification // when sync is complete. m_dwSyncFlags |= CSC_SYNC_NOTIFY_SYSTRAY; } if (dwUpdateFlags & CSC_UPDATE_FILL_ALL) m_dwSyncFlags |= CSC_SYNC_IN_FULL; else if (dwUpdateFlags & CSC_UPDATE_FILL_QUICK) m_dwSyncFlags |= CSC_SYNC_IN_QUICK; if (dwUpdateFlags & CSC_UPDATE_REINT) m_dwSyncFlags |= CSC_SYNC_OUT; if (dwUpdateFlags & CSC_UPDATE_PIN_RECURSE) m_dwSyncFlags |= CSC_SYNC_PINFILES | CSC_SYNC_PIN_RECURSE | CSC_SYNC_IN_QUICK; else if (dwUpdateFlags & CSC_UPDATE_PINFILES) m_dwSyncFlags |= CSC_SYNC_PINFILES | CSC_SYNC_IN_QUICK; if (dwUpdateFlags & CSC_UPDATE_IGNORE_ACCESS) m_dwSyncFlags |= CSC_SYNC_IGNORE_ACCESS; } break; case SYNCMGRFLAG_IDLE: // Auto-sync at idle time if (bNoNet) ExitGracefully(hr, E_FAIL, "No idle sync when no net"); m_dwSyncFlags = CSC_SYNC_OUT | CSC_SYNC_IN_QUICK | CSC_SYNC_IDLE | CSC_SYNC_NOTIFY_SYSTRAY; break; case SYNCMGRFLAG_MANUAL: // Run "mobsync.exe" m_dwSyncFlags = CSC_SYNC_OUT | CSC_SYNC_IN_FULL | CSC_SYNC_NOTIFY_SYSTRAY | CSC_SYNC_RECONNECT; break; case SYNCMGRFLAG_SCHEDULED: // User scheduled sync m_dwSyncFlags = CSC_SYNC_OUT | CSC_SYNC_IN_FULL | CSC_SYNC_NOTIFY_SYSTRAY; break; } if (!(m_dwSyncFlags & CSC_SYNC_PINFILES)) m_dwSyncFlags |= CSC_SYNC_SKIP_EFS; // skip EFS if not pinning if (dwSyncFlags & SYNCMGRFLAG_SETTINGS) m_dwSyncFlags |= CSC_SYNC_SETTINGS; if (!m_dwSyncFlags) ExitGracefully(hr, E_UNEXPECTED, "Nothing to do"); if (dwSyncFlags & SYNCMGRFLAG_MAYBOTHERUSER) m_dwSyncFlags |= CSC_SYNC_MAYBOTHERUSER; if (bNoNet) m_dwSyncFlags |= CSC_SYNC_NONET; GetSilentFolderList(); exit_gracefully: TraceLeaveResult(hr); } STDMETHODIMP CCscUpdate::GetHandlerInfo(LPSYNCMGRHANDLERINFO *ppSyncMgrHandlerInfo) { HRESULT hr = S_OK; LPSYNCMGRHANDLERINFO pHandlerInfo; TraceEnter(TRACE_UPDATE, "CCscUpdate::GetHandlerInfo"); if (NULL == ppSyncMgrHandlerInfo) TraceLeaveResult(E_INVALIDARG); *ppSyncMgrHandlerInfo = NULL; pHandlerInfo = (LPSYNCMGRHANDLERINFO)CoTaskMemAlloc(sizeof(SYNCMGRHANDLERINFO)); if (NULL == pHandlerInfo) ExitGracefully(hr, E_OUTOFMEMORY, "LocalAlloc failed"); pHandlerInfo->cbSize = sizeof(SYNCMGRHANDLERINFO); pHandlerInfo->hIcon = LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_CSCUI_ICON)); pHandlerInfo->SyncMgrHandlerFlags = SYNCMGRHANDLER_HASPROPERTIES | SYNCMGRHANDLER_MAYESTABLISHCONNECTION; LoadStringW(g_hInstance, IDS_APPLICATION, pHandlerInfo->wszHandlerName, ARRAYSIZE(pHandlerInfo->wszHandlerName)); *ppSyncMgrHandlerInfo = pHandlerInfo; exit_gracefully: TraceLeaveResult(hr); } STDMETHODIMP CCscUpdate::EnumSyncMgrItems(LPSYNCMGRENUMITEMS *ppenum) { HRESULT hr; PUPDATEENUM pNewEnum; TraceEnter(TRACE_UPDATE, "CCscUpdate::EnumSyncMgrItems"); *ppenum = NULL; pNewEnum = new CUpdateEnumerator(this); if (pNewEnum) { hr = pNewEnum->QueryInterface(IID_ISyncMgrEnumItems, (LPVOID*)ppenum); pNewEnum->Release(); } else hr = E_OUTOFMEMORY; TraceLeaveResult(hr); } STDMETHODIMP CCscUpdate::GetItemObject(REFSYNCMGRITEMID /*rItemID*/, REFIID /*riid*/, LPVOID * /*ppv*/) { return E_NOTIMPL; } STDMETHODIMP CCscUpdate::ShowProperties(HWND hWndParent, REFSYNCMGRITEMID rItemID) { CSCEntry *pShareEntry; LPCTSTR pszShareName = TEXT(""); pShareEntry = m_ShareLog.Get(rItemID); // We don't enumerate shares to SyncMgr unless a share entry // exists in the registry, so m_ShareLog.Get should never fail here. if (pShareEntry) pszShareName = pShareEntry->Name(); COfflineFilesFolder::Open(); // Notify SyncMgr that the ShowProperties is done. if (NULL != m_pSyncMgrCB) m_pSyncMgrCB->ShowPropertiesCompleted(S_OK); return S_OK; } STDMETHODIMP CCscUpdate::SetProgressCallback(LPSYNCMGRSYNCHRONIZECALLBACK pCallback) { TraceEnter(TRACE_UPDATE, "CCscUpdate::SetProgressCallback"); DoRelease(m_pSyncMgrCB); m_pSyncMgrCB = pCallback; if (m_pSyncMgrCB) m_pSyncMgrCB->AddRef(); TraceLeaveResult(S_OK); } STDMETHODIMP CCscUpdate::PrepareForSync(ULONG cNumItems, SYNCMGRITEMID *pItemID, HWND /*hWndParent*/, DWORD /*dwReserved*/) { HRESULT hr = S_OK; TraceEnter(TRACE_UPDATE, "CCscUpdate::PrepareForSync"); TraceAssert(0 != cNumItems); TraceAssert(NULL != pItemID); // Copy the list of item ID's if (NULL == m_hSyncItems) { m_hSyncItems = DSA_Create(sizeof(SYNCMGRITEMID), 4); if (NULL == m_hSyncItems) ExitGracefully(hr, E_OUTOFMEMORY, "Unable to create DSA for SYNCMGRITEMID list"); } else DSA_DeleteAllItems(m_hSyncItems); while (cNumItems--) DSA_AppendItem(m_hSyncItems, pItemID++); exit_gracefully: // ISyncMgrSynchronize::PrepareForSync is now an asynchronous call // so we could create another thread to do the work and return from // this call immediately. However, since all we do is copy the list // of Item IDs, let's do it here and call // m_pSyncMgrCB->PrepareForSyncCompleted before returning. if (NULL != m_pSyncMgrCB) m_pSyncMgrCB->PrepareForSyncCompleted(hr); TraceLeaveResult(hr); } STDMETHODIMP CCscUpdate::Synchronize(HWND hWndParent) { HRESULT hr = E_FAIL; ULONG cItems = 0; BOOL bConnectionEstablished = FALSE; TraceEnter(TRACE_UPDATE, "CCscUpdate::Synchronize"); if (NULL != m_hSyncItems) cItems = DSA_GetItemCount(m_hSyncItems); // Don't want systray UI updates while syncing. // Whenever the systray UI is updated, the code checks first // for this global mutex object. If it's non-signaled, the // systray knows there's a sync in progress and the UI isn't // updated. TraceAssert(NULL == m_hSyncInProgMutex); m_hSyncInProgMutex = CreateMutex(NULL, TRUE, c_szSyncInProgMutex); if (0 == cItems) { ExitGracefully(hr, E_UNEXPECTED, "Nothing to synchronize"); } else if (1 == cItems) { SYNCMGRITEMID *pItemID = (SYNCMGRITEMID*)DSA_GetItemPtr(m_hSyncItems, 0); if (NULL != pItemID && IsEqualGUID(GUID_CscNullSyncItem, *pItemID)) { // A single item in the DSA and it's our "null sync" GUID. // This means we really have nothing to sync but the invoker // of the sync wants to see some SyncMgr progress UI. In // this scenario the update item enumerator already enumerated // the "null sync" item. Here we set this single item's progress // UI info to 100% complete and skip any sync activity. SYNCMGRPROGRESSITEM spi = {0}; spi.mask = SYNCMGRPROGRESSITEM_STATUSTYPE | SYNCMGRPROGRESSITEM_STATUSTEXT | SYNCMGRPROGRESSITEM_PROGVALUE | SYNCMGRPROGRESSITEM_MAXVALUE; spi.cbSize = sizeof(spi); spi.dwStatusType = SYNCMGRSTATUS_SUCCEEDED; spi.lpcStatusText = L" "; spi.iProgValue = 1; spi.iMaxValue = 1; m_pSyncMgrCB->Progress(GUID_CscNullSyncItem, &spi); m_pSyncMgrCB->SynchronizeCompleted(S_OK); ExitGracefully(hr, NOERROR, "Nothing to sync. Progress UI displayed"); } } m_hwndDlgParent = hWndParent; // We can pin autocached files without a net (no sync required); // otherwise we need to establish a RAS connection to do anything. if ((m_dwSyncFlags & CSC_SYNC_NONET) && !(m_dwSyncFlags & CSC_SYNC_PINFILES)) { hr = m_pSyncMgrCB->EstablishConnection(NULL, 0); FailGracefully(hr, "Unable to establish RAS connection"); bConnectionEstablished = TRUE; } // For each share, kick off a thread to do the work while (cItems > 0) { SYNCMGRITEMID *pItemID; CSCEntry *pShareEntry; --cItems; pItemID = (SYNCMGRITEMID*)DSA_GetItemPtr(m_hSyncItems, cItems); pShareEntry = m_ShareLog.Get(*pItemID); // We don't enumerate shares to SyncMgr unless a share entry // exists in the registry, so m_ShareLog.Get should never fail here. if (NULL == pShareEntry) ExitGracefully(hr, E_UNEXPECTED, "No share entry"); hr = SynchronizeShare(pItemID, pShareEntry->Name(), bConnectionEstablished); DSA_DeleteItem(m_hSyncItems, cItems); FailGracefully(hr, "Unable to create sync thread"); } TraceAssert(0 == DSA_GetItemCount(m_hSyncItems)); exit_gracefully: if (FAILED(hr)) SetItemStatus(GUID_NULL, SYNCMGRSTATUS_STOPPED); TraceLeaveResult(hr); } STDMETHODIMP CCscUpdate::SetItemStatus(REFSYNCMGRITEMID rItemID, DWORD dwSyncMgrStatus) { HRESULT hr = E_FAIL; ULONG cItems; BOOL bAllItems; TraceEnter(TRACE_UPDATE, "CCscUpdate::SetItemStatus"); if (SYNCMGRSTATUS_SKIPPED != dwSyncMgrStatus && SYNCMGRSTATUS_STOPPED != dwSyncMgrStatus) TraceLeaveResult(E_NOTIMPL); bAllItems = FALSE; if (SYNCMGRSTATUS_STOPPED == dwSyncMgrStatus) { bAllItems = TRUE; m_dwSyncFlags |= CSC_SYNC_CANCELLED; } // SetItemStatus can be called between PrepareForSync and Synchronize, in // in which case the correct thing to do is remove the item from m_hSyncItems. if (NULL != m_hSyncItems) { cItems = DSA_GetItemCount(m_hSyncItems); while (cItems > 0) { SYNCMGRITEMID *pItemID; --cItems; pItemID = (SYNCMGRITEMID*)DSA_GetItemPtr(m_hSyncItems, cItems); if (bAllItems || IsEqualGUID(rItemID, *pItemID)) { // Remove the item from the list of items to sync DSA_DeleteItem(m_hSyncItems, cItems); if (!bAllItems) ExitGracefully(hr, S_OK, "Skipping item"); } } } // Lookup the thread for the item ID and set its status // to cause it to terminate. hr = SetSyncThreadStatus(SyncStop, bAllItems ? GUID_NULL : rItemID); exit_gracefully: TraceLeaveResult(hr); } STDMETHODIMP CCscUpdate::ShowError(HWND /*hWndParent*/ , REFSYNCMGRERRORID /*ErrorID*/) { return E_NOTIMPL; } HRESULT CCscUpdate::SynchronizeShare(SYNCMGRITEMID *pItemID, LPCTSTR pszShareName, BOOL bRasConnected) { HRESULT hr = S_OK; DWORD dwThreadID; PSYNCTHREADDATA pThreadData; ULONG cbShareName = 0; TraceEnter(TRACE_UPDATE, "CCscUpdate::SynchronizeShare"); TraceAssert(NULL != pItemID); TraceAssert(NULL != pszShareName); TraceAssert(*pszShareName); EnterCriticalSection(&m_csThreadList); if (NULL == m_hSyncThreads) m_hSyncThreads = DPA_Create(4); LeaveCriticalSection(&m_csThreadList); if (NULL == m_hSyncThreads) ExitGracefully(hr, E_OUTOFMEMORY, "Unable to create DPA for threads"); cbShareName = StringByteSize(pszShareName); pThreadData = (PSYNCTHREADDATA)LocalAlloc(LPTR, sizeof(SYNCTHREADDATA) + cbShareName); if (!pThreadData) ExitGracefully(hr, E_OUTOFMEMORY, "LocalAlloc failed"); pThreadData->pThis = this; pThreadData->ItemID = *pItemID; pThreadData->pszShareName = (LPTSTR)(pThreadData + 1); CopyMemory(pThreadData->pszShareName, pszShareName, cbShareName); // If we established a RAS connection, then it will go away // right after the sync completes, so there's no point trying // to reconnect. That is, only check CSC_SYNC_RECONNECT and // add the share to the reconnect list if we aren't doing RAS. if (bRasConnected) { pThreadData->dwSyncStatus |= SDS_SYNC_RAS_CONNECTED; } else if (m_dwSyncFlags & CSC_SYNC_RECONNECT) { CscFilenameList::HSHARE hShare; m_ReconnectList.AddShare(pszShareName, &hShare); } pThreadData->hThread = CreateThread(NULL, 0, _SyncThread, pThreadData, CREATE_SUSPENDED, &dwThreadID); if (NULL != pThreadData->hThread) { EnterCriticalSection(&m_csThreadList); DPA_AppendPtr(m_hSyncThreads, pThreadData); LeaveCriticalSection(&m_csThreadList); ResumeThread(pThreadData->hThread); } else { DWORD dwErr = GetLastError(); LocalFree(pThreadData); LPTSTR pszErr = GetErrorText(GetLastError()); LogError(*pItemID, SYNCMGRLOGLEVEL_ERROR, IDS_FILL_SPARSE_FILES_ERROR, pszShareName, pszErr); LocalFreeString(&pszErr); hr = HRESULT_FROM_WIN32(dwErr); } exit_gracefully: TraceLeaveResult(hr); } void CCscUpdate::SetLastSyncTime(LPCTSTR pszShareName) { HKEY hKey = NULL; hKey = m_ShareLog.OpenKey(pszShareName, KEY_SET_VALUE); if (hKey) { FILETIME ft = {0}; GetSystemTimeAsFileTime(&ft); RegSetValueEx(hKey, c_szLastSync, 0, REG_BINARY, (LPBYTE)&ft, sizeof(ft)); RegCloseKey(hKey); } } DWORD CCscUpdate::GetLastSyncTime(LPCTSTR pszShareName, LPFILETIME pft) { DWORD dwResult = ERROR_PATH_NOT_FOUND; HKEY hKey = NULL; hKey = m_ShareLog.OpenKey(pszShareName, KEY_QUERY_VALUE); if (hKey) { DWORD dwSize = sizeof(*pft); dwResult = RegQueryValueEx(hKey, c_szLastSync, NULL, NULL, (LPBYTE)pft, &dwSize); RegCloseKey(hKey); } return dwResult; } void CCscUpdate::SyncThreadCompleted(PSYNCTHREADDATA pSyncData) { int iThread; TraceEnter(TRACE_UPDATE, "CCscUpdate::SyncThreadCompleted"); TraceAssert(NULL != pSyncData); TraceAssert(NULL != m_hSyncThreads); EnterCriticalSection(&m_csThreadList); iThread = DPA_GetPtrIndex(m_hSyncThreads, pSyncData); TraceAssert(-1 != iThread); DPA_DeletePtr(m_hSyncThreads, iThread); CloseHandle(pSyncData->hThread); pSyncData->hThread = NULL; iThread = DPA_GetPtrCount(m_hSyncThreads); LeaveCriticalSection(&m_csThreadList); if (0 == iThread) { SyncCompleted(); } TraceLeaveVoid(); } void CCscUpdate::SyncCompleted(void) { if ((m_dwSyncFlags & CSC_SYNC_RECONNECT) && !(m_dwSyncFlags & CSC_SYNC_CANCELLED)) { m_dwSyncFlags &= ~CSC_SYNC_RECONNECT; ReconnectServers(&m_ReconnectList, FALSE, FALSE); } if (NULL != m_hSyncInProgMutex) { // We're not syncing so reset the global event ReleaseMutex(m_hSyncInProgMutex); CloseHandle(m_hSyncInProgMutex); m_hSyncInProgMutex = NULL; } if (m_dwSyncFlags & CSC_SYNC_NOTIFY_SYSTRAY) { // Notify systray that we're done PostToSystray(CSCWM_DONESYNCING, 0, 0); m_dwSyncFlags &= ~CSC_SYNC_NOTIFY_SYSTRAY; } // Notify SyncMgr that the sync is done if (NULL != m_pSyncMgrCB) { m_pSyncMgrCB->SynchronizeCompleted(S_OK); } } UINT GetErrorFormat(DWORD dwErr, BOOL bMerging = FALSE) { UINT idString = 0; // BUGBUG these are all just initial guesses. Not sure // which error codes we'll get from CSC. switch (dwErr) { case ERROR_DISK_FULL: // "The server disk is full." // "The local disk is full." idString = bMerging ? IDS_SERVER_FULL_ERROR : IDS_LOCAL_DISK_FULL_ERROR; break; case ERROR_LOCK_VIOLATION: case ERROR_SHARING_VIOLATION: case ERROR_OPEN_FILES: case ERROR_ACTIVE_CONNECTIONS: case ERROR_DEVICE_IN_USE: // "'%1' is in use on %2" idString = IDS_FILE_OPEN_ERROR; break; case ERROR_BAD_NETPATH: case ERROR_DEV_NOT_EXIST: case ERROR_NETNAME_DELETED: case ERROR_BAD_NET_NAME: case ERROR_SHARING_PAUSED: case ERROR_REQ_NOT_ACCEP: case ERROR_REDIR_PAUSED: case ERROR_BAD_DEVICE: case ERROR_CONNECTION_UNAVAIL: case ERROR_NO_NET_OR_BAD_PATH: case ERROR_NO_NETWORK: case ERROR_CONNECTION_REFUSED: case ERROR_GRACEFUL_DISCONNECT: case ERROR_NETWORK_UNREACHABLE: case ERROR_HOST_UNREACHABLE: case ERROR_PROTOCOL_UNREACHABLE: case ERROR_PORT_UNREACHABLE: case ERROR_LOGON_FAILURE: // "Unable to connect to '%1.' %2" idString = IDS_SHARE_CONNECT_ERROR; break; case ERROR_OPEN_FAILED: case ERROR_UNEXP_NET_ERR: case ERROR_NETWORK_BUSY: case ERROR_BAD_NET_RESP: // "Unable to access '%1' on %2. %3" idString = IDS_NET_ERROR; break; case ERROR_ACCESS_DENIED: case ERROR_NETWORK_ACCESS_DENIED: // "Access to '%1' is denied on %2" idString = IDS_ACCESS_ERROR; break; case ERROR_BAD_FORMAT: // "The Offline Files cache is corrupt. Restart the computer to correct the cache." idString = IDS_CACHE_CORRUPT; break; default: // "Error accessing '%1' on %2. %3" idString = IDS_UNKNOWN_SYNC_ERROR; break; } return idString; } HRESULT CCscUpdate::LogError(REFSYNCMGRITEMID rItemID, LPCTSTR pszText, DWORD dwLogLevel, REFSYNCMGRERRORID ErrorID) { HRESULT hr; SYNCMGRLOGERRORINFO slei; USES_CONVERSION; TraceEnter(TRACE_UPDATE, "CCscUpdate::LogError"); if (NULL == m_pSyncMgrCB) TraceLeaveResult(E_UNEXPECTED); slei.cbSize = sizeof(slei); slei.mask = SYNCMGRLOGERROR_ITEMID | SYNCMGRLOGERROR_ERRORID; slei.ItemID = rItemID; slei.ErrorID = ErrorID; // if we have a jumptext associated with this item then // set the enable jumptext flag if (ErrorID != GUID_NULL) { slei.mask |= SYNCMGRLOGERROR_ERRORFLAGS; slei.dwSyncMgrErrorFlags = SYNCMGRERRORFLAG_ENABLEJUMPTEXT; } Trace((pszText)); hr = m_pSyncMgrCB->LogError(dwLogLevel, T2CW(pszText), &slei); TraceLeaveResult(hr); } DWORD CCscUpdate::LogError(REFSYNCMGRITEMID rItemID, DWORD dwLogLevel, UINT nFormatID, ...) { LPTSTR pszError = NULL; va_list args; va_start(args, nFormatID); if (vFormatStringID(&pszError, g_hInstance, nFormatID, &args)) { LogError(rItemID, pszError, dwLogLevel); LocalFree(pszError); } va_end(args); return 0; } DWORD CCscUpdate::LogError(REFSYNCMGRITEMID rItemID, UINT nFormatID, LPCTSTR pszName, DWORD dwErr, DWORD dwLogLevel) { // Break the filename into "file" and "path" components TCHAR szPath[MAX_PATH] = TEXT("\\"); LPCTSTR pszFile = NULL; if (pszName) { pszFile = PathFindFileName(pszName); lstrcpyn(szPath, pszName, min(ARRAYSIZE(szPath),(int)(pszFile-pszName))); } // Get the system error text and format the error LPTSTR pszErr = GetErrorText(dwErr); LogError(rItemID, dwLogLevel, nFormatID, pszFile, szPath, pszErr); LocalFreeString(&pszErr); return 0; } BOOL MakeDriveLetterPath(LPCTSTR pszUNC, LPCTSTR pszShare, LPCTSTR pszDrive, LPTSTR *ppszResult) { BOOL bResult = FALSE; ULONG cchShare; if (!pszUNC || !pszShare || !ppszResult) return FALSE; *ppszResult = NULL; cchShare = lstrlen(pszShare); // If the path is on the share, use the drive letter instead if (pszDrive && *pszDrive && CSTR_EQUAL == CompareString(LOCALE_USER_DEFAULT, NORM_IGNORECASE, pszUNC, cchShare, pszShare, cchShare)) { *ppszResult = (LPTSTR)LocalAlloc(LPTR, MAX(StringByteSize(pszUNC), MAX_PATH_BYTES)); if (*ppszResult) { PathCombine(*ppszResult, pszDrive, pszUNC + cchShare); bResult = TRUE; } } return bResult; } DWORD CCscUpdate::CopyLocalFileWithDriveMapping(LPCTSTR pszSrc, LPCTSTR pszDst, LPCTSTR pszShare, LPCTSTR pszDrive, BOOL bDirectory) { DWORD dwErr = NOERROR; LPTSTR szDst = NULL; if (!pszSrc || !pszDst || !pszShare) return ERROR_INVALID_PARAMETER; // If the destination is on the share, use the drive letter instead if (MakeDriveLetterPath(pszDst, pszShare, pszDrive, &szDst)) pszDst = szDst; if (bDirectory) { // We don't need to copy the directory contents here, just create // the tree structure on the server. if (!CreateDirectory(pszDst, NULL)) { dwErr = GetLastError(); if (ERROR_ALREADY_EXISTS == dwErr) dwErr = NOERROR; } } else { LPTSTR pszTmpName = NULL; if (!CSCCopyReplica(pszSrc, &pszTmpName) || !MoveFileEx(pszTmpName, pszDst, MOVEFILE_COPY_ALLOWED | MOVEFILE_REPLACE_EXISTING | MOVEFILE_WRITE_THROUGH)) { dwErr = GetLastError(); } if (NULL != pszTmpName) { DeleteFile(pszTmpName); LocalFree(pszTmpName); } } if (ERROR_PATH_NOT_FOUND == dwErr) { // The parent directory doesn't exist, create it now. TCHAR szParent[MAX_PATH]; lstrcpyn(szParent, pszDst, ARRAYSIZE(szParent)); PathRemoveFileSpec(szParent); dwErr = CopyLocalFileWithDriveMapping(pszSrc, szParent, pszShare, NULL, TRUE); // If that worked, retry the original operation. if (NOERROR == dwErr) dwErr = CopyLocalFileWithDriveMapping(pszSrc, pszDst, pszShare, NULL, bDirectory); } LocalFreeString(&szDst); return dwErr; } BOOL HandleConflictLocally(PSYNCTHREADDATA pSyncData, LPCTSTR pszPath, DWORD dwCscStatus, DWORD dwLocalAttr, DWORD dwRemoteAttr = 0) { BOOL bResult = FALSE; LPTSTR szParent = NULL; // If it's super-hidden or not modified locally, we can always // handle the conflict locally. if (!(dwCscStatus & CSC_LOCALLY_MODIFIED) || IsHiddenSystem(dwLocalAttr) || IsHiddenSystem(dwRemoteAttr)) return TRUE; // If we're dealing with 2 folders, the worst that happens is that the // underlying files/folders get merged. if ((FILE_ATTRIBUTE_DIRECTORY & dwLocalAttr) && (FILE_ATTRIBUTE_DIRECTORY & dwRemoteAttr)) return TRUE; // Next, check whether the parent path is super-hidden. // For example, recycle bin makes super-hidden folders and puts // metadata files in them. // Do this on the server, since CSC has exclusive access to the database // while merging, causing GetFileAttributes to fail with Access Denied. if (MakeDriveLetterPath(pszPath, pSyncData->pszShareName, pSyncData->szDrive, &szParent)) { do { PathRemoveFileSpec(szParent); dwRemoteAttr = GetFileAttributes(szParent); if ((DWORD)-1 == dwRemoteAttr) { // Path doesn't exist, access denied, etc. break; } if (IsHiddenSystem(dwRemoteAttr)) { bResult = TRUE; break; } } while (!PathIsRoot(szParent)); } LocalFreeString(&szParent); return bResult; } DWORD CCscUpdate::HandleFileConflict(PSYNCTHREADDATA pSyncData, LPCTSTR pszName, DWORD dwStatus, DWORD dwHintFlags, LPWIN32_FIND_DATA pFind32) { DWORD dwResult = CSCPROC_RETURN_CONTINUE; DWORD dwErr = NOERROR; int nErrorResolution = RFC_KEEPBOTH; LPTSTR pszNewName = NULL; LPTSTR szFullPath = NULL; BOOL bApplyToAll = FALSE; TraceEnter(TRACE_UPDATE, "CCscUpdate::HandleFileConflict"); Trace((TEXT("File conflict: %s"), pszName)); TraceAssert(pSyncData->dwSyncStatus & SDS_SYNC_OUT); szFullPath = (LPTSTR)LocalAlloc(LPTR, StringByteSize(pszName) + MAX_PATH*sizeof(TCHAR)); if (!szFullPath) { dwErr = ERROR_OUTOFMEMORY; ExitGracefully(dwResult, CSCPROC_RETURN_SKIP, "LocalAlloc failed"); } HANDLE hFind; WIN32_FIND_DATA fdRemote; PathCombine(szFullPath, pSyncData->szDrive, pszName + lstrlen(pSyncData->pszShareName)); hFind = FindFirstFile(szFullPath, &fdRemote); // Does the net version still exist? if (hFind == INVALID_HANDLE_VALUE) ExitGracefully(dwResult, HandleDeleteConflict(pSyncData, pszName, dwStatus, dwHintFlags, pFind32), "Net file deleted"); // Still exists, continue FindClose(hFind); // If only the attributes or file times were modified locally, // or if the file is hidden+system, keep the server copy and // don't bother the user. (e.g. desktop.ini) if (HandleConflictLocally(pSyncData, pszName, dwStatus, pFind32->dwFileAttributes, fdRemote.dwFileAttributes)) { ExitGracefully(dwResult, CSCPROC_RETURN_FORCE_INWARD, "Ignoring conflict"); } else if (IsSilentFolder(pszName)) { // It's in a per-user shell special folder. Last writer wins. if (CompareFileTime(&pFind32->ftLastWriteTime, &fdRemote.ftLastWriteTime) < 0) { ExitGracefully(dwResult, CSCPROC_RETURN_FORCE_INWARD, "Handling special folder conflict - server copy wins"); } else { ExitGracefully(dwResult, CSCPROC_RETURN_FORCE_OUTWARD, "Handling special folder conflict - local copy wins"); } } dwErr = GetNewVersionName(pszName, pSyncData->pszShareName, pSyncData->szDrive, &pszNewName); if (NOERROR != dwErr) { ExitGracefully(dwResult, CSCPROC_RETURN_SKIP, "GetNewVersionName failed"); } switch (SDS_SYNC_FILE_CONFLICT_MASK & pSyncData->dwSyncStatus) { case 0: if (CSC_SYNC_MAYBOTHERUSER & m_dwSyncFlags) { nErrorResolution = ShowConflictDialog(m_hwndDlgParent, pszName, pszNewName, pSyncData->pszShareName, pSyncData->szDrive, pFind32, &fdRemote); if (RFC_APPLY_TO_ALL & nErrorResolution) { bApplyToAll = TRUE; nErrorResolution &= ~RFC_APPLY_TO_ALL; } } break; case SDS_SYNC_CONFLICT_KEEPLOCAL: nErrorResolution = RFC_KEEPLOCAL; break; case SDS_SYNC_CONFLICT_KEEPNET: nErrorResolution = RFC_KEEPNETWORK; break; case SDS_SYNC_CONFLICT_KEEPBOTH: nErrorResolution = RFC_KEEPBOTH; break; } // Self-host notification callback CSCUI_NOTIFYHOOK((CSCH_UpdateConflict, TEXT("Update conflict: %1, resolution %2!d!"), pszName, nErrorResolution)); switch (nErrorResolution) { default: case RFC_KEEPBOTH: if (bApplyToAll) pSyncData->dwSyncStatus |= SDS_SYNC_CONFLICT_KEEPBOTH; lstrcpy(szFullPath, pszName); PathRemoveFileSpec(szFullPath); if (FILE_ATTRIBUTE_DIRECTORY & pFind32->dwFileAttributes) { // Rename the local version in the cache and merge again. lstrcpyn(pFind32->cFileName, pszNewName, ARRAYSIZE(pFind32->cFileName)); if (!CSCDoLocalRenameEx(pszName, szFullPath, pFind32, TRUE, TRUE)) { dwErr = GetLastError(); ExitGracefully(dwResult, CSCPROC_RETURN_SKIP, "CSCDoLocalRenameEx failed"); } // Because CSCDoLocalRenameEx and CSCMergeShare are separate operations, // we have to abort the current merge operation and start over. // Otherwise, the current merge operation fails due to the "left // hand not knowing what the right hande is doing". Trace((TEXT("Restarting merge on: %s"), pSyncData->pszShareName)); pSyncData->dwSyncStatus |= SDS_SYNC_RESTART_MERGE; dwResult = CSCPROC_RETURN_ABORT; } else { // Note that CSCDoLocalRenameEx would work for files also, but we // prefer to avoid restarting CSCMergeShare so do these ourselves. PathAppend(szFullPath, pszNewName); dwErr = CopyLocalFileWithDriveMapping(pszName, szFullPath, pSyncData->pszShareName, pSyncData->szDrive); if (NOERROR != dwErr) ExitGracefully(dwResult, CSCPROC_RETURN_SKIP, "CopyLocalFileWithDriveMapping failed"); // If the original file was pinned, we want to pin the copy also. // Unfortunately, we can't reliably pin during a merge, so we have // to remember these in a list and pin them later. if (dwHintFlags & FLAG_CSC_HINT_PIN_USER) { if (!m_pConflictPinList) m_pConflictPinList = new CscFilenameList; if (m_pConflictPinList) { m_pConflictPinList->AddFile(szFullPath, !!(pFind32->dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)); } } // Tell CSCMergeShare to copy the server copy to the cache // (with the old name). This clears the dirty cache. dwResult = CSCPROC_RETURN_FORCE_INWARD; } break; case RFC_KEEPNETWORK: // Tell CSCMergeShare to copy the server copy to the cache dwResult = CSCPROC_RETURN_FORCE_INWARD; if (bApplyToAll) pSyncData->dwSyncStatus |= SDS_SYNC_CONFLICT_KEEPNET; break; case RFC_KEEPLOCAL: // Tell CSCMergeShare to push the local copy to the server dwResult = CSCPROC_RETURN_FORCE_OUTWARD; if (bApplyToAll) pSyncData->dwSyncStatus |= SDS_SYNC_CONFLICT_KEEPLOCAL; break; case RFC_CANCEL: TraceMsg("HandleFileConflict: Cancelling sync - user bailed"); SetItemStatus(GUID_NULL, SYNCMGRSTATUS_STOPPED); dwResult = CSCPROC_RETURN_ABORT; break; } exit_gracefully: if (CSCPROC_RETURN_FORCE_INWARD == dwResult) { // CSCMergeShare truncates (makes sparse) the // file if we return this. We'd like to fill // it during this sync. pSyncData->cFilesToSync++; pSyncData->dwSyncStatus |= SDS_SYNC_FORCE_INWARD; } if (NOERROR != dwErr) { pszName += lstrlen(pSyncData->pszShareName); if (*pszName == TEXT('\\')) pszName++; LogError(pSyncData->ItemID, IDS_NAME_CONFLICT_ERROR, pszName, dwErr); pSyncData->dwSyncStatus |= SDS_SYNC_ERROR; } LocalFreeString(&szFullPath); LocalFreeString(&pszNewName); TraceLeaveResult(dwResult); } // Returns values for the Resolve Delete Conflict dialog #define RDC_CANCEL 0x00 #define RDC_DELETE 0x01 #define RDC_RESTORE 0x02 #define RDC_APPLY_ALL 0x04 #define RDC_DELETE_ALL (RDC_APPLY_ALL | RDC_DELETE) #define RDC_RESTORE_ALL (RDC_APPLY_ALL | RDC_RESTORE) TCHAR const c_szDeleteSelection[] = TEXT("DeleteConflictSelection"); INT_PTR CALLBACK DeleteConflictProc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) { int nResult; switch (uMsg) { case WM_INITDIALOG: { TCHAR szShare[MAX_PATH]; LPCTSTR pszPath = (LPCTSTR)lParam; LPTSTR pszT = NULL; szShare[0] = TEXT('\0'); lstrcpyn(szShare, pszPath, ARRAYSIZE(szShare)); PathStripToRoot(szShare); // Format the file name PathSetDlgItemPath(hDlg, IDC_FILENAME, pszPath); // Build the "Do this for all on " string FormatStringID(&pszT, g_hInstance, IDS_FMT_DELETE_APPLY_ALL, szShare); if (pszT) { SetDlgItemText(hDlg, IDC_APPLY_TO_ALL, pszT); LocalFreeString(&pszT); } // else default text is OK (no share name) // Select whatever the user chose last time, default is "restore" DWORD dwPrevSelection = RDC_RESTORE; DWORD dwType; DWORD cbData = sizeof(dwPrevSelection); SHGetValue(HKEY_CURRENT_USER, c_szCSCKey, c_szDeleteSelection, &dwType, &dwPrevSelection, &cbData); dwPrevSelection = (RDC_DELETE == dwPrevSelection ? IDC_DELETE_LOCAL : IDC_KEEP_LOCAL); CheckRadioButton(hDlg, IDC_KEEP_LOCAL, IDC_DELETE_LOCAL, dwPrevSelection); // Get the file-type icon pszT = PathFindExtension(pszPath); if (pszT) { SHFILEINFO sfi = {0}; SHGetFileInfo(pszT, 0, &sfi, sizeof(sfi), SHGFI_ICON); if (sfi.hIcon) { SendDlgItemMessage(hDlg, IDC_DLGTYPEICON, STM_SETICON, (WPARAM)sfi.hIcon, 0L); } } } return TRUE; case WM_COMMAND: nResult = -1; switch (LOWORD(wParam)) { case IDCANCEL: nResult = RDC_CANCEL; break; case IDOK: if (BST_CHECKED == IsDlgButtonChecked(hDlg, IDC_DELETE_LOCAL)) nResult = RDC_DELETE; else nResult = RDC_RESTORE; // Remember the selection for next time SHSetValue(HKEY_CURRENT_USER, c_szCSCKey, c_szDeleteSelection, REG_DWORD, &nResult, sizeof(nResult)); if (BST_CHECKED == IsDlgButtonChecked(hDlg, IDC_APPLY_TO_ALL)) nResult |= RDC_APPLY_ALL; break; } if (-1 != nResult) { EndDialog(hDlg, nResult); return TRUE; } break; } return FALSE; } BOOL CALLBACK ConflictPurgeCallback(LPCWSTR /*pszFile*/, LPARAM lParam) { PSYNCTHREADDATA pSyncData = (PSYNCTHREADDATA)lParam; return !(SDS_SYNC_CANCELLED & pSyncData->dwSyncStatus); } DWORD CCscUpdate::HandleDeleteConflict(PSYNCTHREADDATA pSyncData, LPCTSTR pszName, DWORD dwStatus, DWORD dwHintFlags, LPWIN32_FIND_DATA pFind32) { DWORD dwResult = CSCPROC_RETURN_CONTINUE; int nErrorResolution = RDC_DELETE; // default action BOOL bDirectory = (FILE_ATTRIBUTE_DIRECTORY & pFind32->dwFileAttributes); TraceEnter(TRACE_UPDATE, "CCscUpdate::HandleDeleteConflict"); Trace((TEXT("Net file deleted: %s"), pszName)); // We already know that the net file was deleted, or HandleDeleteConflict // wouldn't be called. If the local copy was also deleted, then there // isn't really a conflict and we can continue without prompting. // If the file isn't pinned, handle the conflict silently if // only attributes changed or it's super-hidden. // Finally, if the file lives in certain special folder locations, // such as AppData, handle the conflict silently. // If we get past all that, ask the user what to do, but only bother // the user as a last resort. if ( !(dwStatus & FLAG_CSC_COPY_STATUS_LOCALLY_DELETED) && ((dwHintFlags & (FLAG_CSC_HINT_PIN_USER | FLAG_CSC_HINT_PIN_ADMIN)) || !HandleConflictLocally(pSyncData, pszName, dwStatus, pFind32->dwFileAttributes)) && !IsSilentFolder(pszName) ) { // The file is either pinned or modified locally, so // default action is now "restore". nErrorResolution = RDC_RESTORE; switch (SDS_SYNC_DELETE_CONFLICT_MASK & pSyncData->dwSyncStatus) { case 0: if (CSC_SYNC_MAYBOTHERUSER & m_dwSyncFlags) { int idDialog = (bDirectory ? IDD_FOLDER_CONFLICT_DELETE : IDD_FILE_CONFLICT_DELETE); nErrorResolution = (int)DialogBoxParam(g_hInstance, MAKEINTRESOURCE(idDialog), m_hwndDlgParent, DeleteConflictProc, (LPARAM)pszName); if (RDC_DELETE_ALL == nErrorResolution) { pSyncData->dwSyncStatus |= SDS_SYNC_DELETE_DELETE; nErrorResolution = RDC_DELETE; } else if (RDC_RESTORE_ALL == nErrorResolution) { pSyncData->dwSyncStatus |= SDS_SYNC_DELETE_RESTORE; nErrorResolution = RDC_RESTORE; } } break; case SDS_SYNC_DELETE_DELETE: nErrorResolution = RDC_DELETE; break; case SDS_SYNC_DELETE_RESTORE: nErrorResolution = RDC_RESTORE; break; } // Self-host notification callback CSCUI_NOTIFYHOOK((CSCH_DeleteConflict, TEXT("Delete conflict: %1, resolution %2!d!"), pszName, nErrorResolution)); } switch (nErrorResolution) { default: case RDC_RESTORE: Trace((TEXT("HandleDeleteConflict: restoring %s"), pszName)); // Tell CSCMergeShare to push the local copy to the server dwResult = CSCPROC_RETURN_FORCE_OUTWARD; break; case RDC_DELETE: Trace((TEXT("HandleDeleteConflict: deleting %s"), pszName)); if (bDirectory) { // Deep delete CSCUIRemoveFolderFromCache(pszName, 0, ConflictPurgeCallback, (LPARAM)pSyncData); } else { CscDelete(pszName); } dwResult = CSCPROC_RETURN_SKIP; break; case RDC_CANCEL: TraceMsg("HandleDeleteConflict: Cancelling sync - user bailed"); SetItemStatus(GUID_NULL, SYNCMGRSTATUS_STOPPED); dwResult = CSCPROC_RETURN_ABORT; break; } TraceLeaveResult(dwResult); } DWORD CCscUpdate::CscCallback(PSYNCTHREADDATA pSyncData, LPCTSTR pszName, DWORD dwStatus, DWORD dwHintFlags, DWORD dwPinCount, LPWIN32_FIND_DATA pFind32, DWORD dwReason, DWORD dwParam1, DWORD dwParam2) { DWORD dwResult = CSCPROC_RETURN_CONTINUE; SYNCMGRPROGRESSITEM spi = { sizeof(spi), 0 }; TraceEnter(TRACE_UPDATE, "CCscUpdate::CscCallback"); TraceAssert(pSyncData != NULL); TraceAssert(pSyncData->pThis == this); // Check for Cancel if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED) { TraceMsg("Cancelling sync operation"); TraceLeaveValue(CSCPROC_RETURN_ABORT); } switch (dwReason) { case CSCPROC_REASON_BEGIN: // First thing to do is determine if this is for the entire share // or an individual file in the share. if (!(pSyncData->dwSyncStatus & SDS_SYNC_STARTED)) { // SHARE BEGIN pSyncData->dwSyncStatus |= SDS_SYNC_STARTED; TraceAssert(!lstrcmpi(pszName, pSyncData->pszShareName)); Trace((TEXT("Share begin: %s"), pszName)); if (pSyncData->dwSyncStatus & SDS_SYNC_OUT) { // Save the drive letter to use for net operations Trace((TEXT("Drive %s"), pFind32->cFileName)); lstrcpyn(pSyncData->szDrive, pFind32->cFileName, ARRAYSIZE(pSyncData->szDrive)); } else { pSyncData->szDrive[0] = TEXT('\0'); } // Remember whether it's an autocache share or not switch (dwStatus & FLAG_CSC_SHARE_STATUS_CACHING_MASK) { case FLAG_CSC_SHARE_STATUS_AUTO_REINT: case FLAG_CSC_SHARE_STATUS_VDO: pSyncData->dwSyncStatus |= SDS_SYNC_AUTOCACHE; break; } } else { // FILE BEGIN BOOL bSkipFile = FALSE; TraceAssert(lstrlen(pszName) > lstrlen(pSyncData->pszShareName)); if (!(pFind32->dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)) { // If we're updating a file selection and this file // isn't part of the selection, skip it. if (m_pFileList && !m_pFileList->FileExists(pszName, false)) { bSkipFile = TRUE; } else if (!(pSyncData->dwSyncStatus & (SDS_SYNC_AUTOCACHE | SDS_SYNC_OUT)) && !(dwHintFlags & (FLAG_CSC_HINT_PIN_USER | FLAG_CSC_HINT_PIN_ADMIN)) && !IsSpecialFolder(pszName)) { // Skip autocached files when filling on a // non-autocache share. Raid #341786 bSkipFile = TRUE; } else if (!(pSyncData->dwSyncStatus & CSC_SYNC_IGNORE_ACCESS)) { // dwReserved0 is the current user's access mask // dwReserved1 is the Guest access mask DWORD dwCurrentAccess = pFind32->dwReserved0 | pFind32->dwReserved1; if (pSyncData->dwSyncStatus & SDS_SYNC_OUT) { // If the current user doesn't have sufficient access // to merge offline changes, then don't bother trying. // (It must be some other user's file.) // Have the attributes changed offline? if (FLAG_CSC_COPY_STATUS_ATTRIB_LOCALLY_MODIFIED & dwStatus) { // Yes. Continue if the current user has // write-attribute access. bSkipFile = !(dwCurrentAccess & FILE_WRITE_ATTRIBUTES); } // Have the contents changed offline? if (!bSkipFile && ((FLAG_CSC_COPY_STATUS_DATA_LOCALLY_MODIFIED | FLAG_CSC_COPY_STATUS_LOCALLY_CREATED | FLAG_CSC_COPY_STATUS_LOCALLY_DELETED) & dwStatus)) { // Yes. Continue if the current user has // write-data access. bSkipFile = !(dwCurrentAccess & FILE_WRITE_DATA); } } else { // We're filling. Continue if the current user has // read-data access, otherwise skip. bSkipFile = !(dwCurrentAccess & FILE_READ_DATA); } } } else if (!(pSyncData->dwSyncStatus & SDS_SYNC_OUT)) { // It's a directory and we're in CSCFillSparseFiles. // Note that we never skip directories when merging (we may be // interested in a file further down the tree) although we // can skip directories when filling. // If it's not in the file selection, skip it. if (m_pFileList && !m_pFileList->FileExists(pszName, false)) { bSkipFile = TRUE; } } if (bSkipFile) { Trace((TEXT("Skipping: %s"), pszName)); dwResult = CSCPROC_RETURN_SKIP; pSyncData->dwSyncStatus |= SDS_SYNC_FILE_SKIPPED; break; } Trace((TEXT("File begin: %s"), pszName)); // Since we sometimes don't skip directories, even when it turns // out they have nothing that the current user is interested in, // don't display directory names in SyncMgr. // If we sync a file farther down the tree, we will display the // filename and the intervening directory names will be visible // at that time. if (!(pFind32->dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)) { USES_CONVERSION; // Tell SyncMgr what we're doing spi.mask = SYNCMGRPROGRESSITEM_STATUSTEXT; spi.lpcStatusText = T2CW(pszName + lstrlen(pSyncData->pszShareName) + 1); NotifySyncMgr(pSyncData, &spi); } // dwParam1 is non-zero when there is a conflict, i.e. both the // local and remote versions of the file have been modified. if (dwParam1) { if (dwParam2) // indicates server file deleted { Trace((TEXT("Delete conflict: %d"), dwParam2)); dwResult = HandleDeleteConflict(pSyncData, pszName, dwStatus, dwHintFlags, pFind32); } else { Trace((TEXT("Update conflict: %d"), dwParam1)); dwResult = HandleFileConflict(pSyncData, pszName, dwStatus, dwHintFlags, pFind32); } } } break; case CSCPROC_REASON_END: // dwParam2 == error code (winerror.h) if (3000 <= dwParam2 && dwParam2 <= 3200) { // Private error codes used in cscdll Trace((TEXT("CSC error: %d"), dwParam2)); dwParam2 = NOERROR; } else if (ERROR_OPERATION_ABORTED == dwParam2) { // We returned CSCPROC_RETURN_ABORT for some reason. // Whatever it was, we already reported it. dwParam2 = NOERROR; dwResult = CSCPROC_RETURN_ABORT; } if (lstrlen(pszName) == lstrlen(pSyncData->pszShareName)) { // SHARE END TraceAssert(!lstrcmpi(pszName, pSyncData->pszShareName)); Trace((TEXT("Share end: %s"), pszName)); pSyncData->dwSyncStatus &= ~SDS_SYNC_STARTED; } else { BOOL bUpdateProgress = FALSE; // FILE END if (!(pSyncData->dwSyncStatus & SDS_SYNC_FILE_SKIPPED)) { Trace((TEXT("File end: %s"), pszName)); bUpdateProgress = TRUE; // BUGBUG special case errors switch (dwParam2) { case ERROR_ACCESS_DENIED: if (FILE_ATTRIBUTE_DIRECTORY & pFind32->dwFileAttributes) { // 317751 directories are not per-user, so if a // different user syncs, we can hit this. Don't want // to show an error message unless we are ignoring // access (when the user explicitly selected something // to pin/sync). // 394362 BrianV hit this running as an admin, so don't // show this error for admins either. if (!(pSyncData->dwSyncStatus & CSC_SYNC_IGNORE_ACCESS)) { TraceMsg("Suppressing ERROR_ACCESS_DENIED on folder"); dwParam2 = NOERROR; } } break; case ERROR_GEN_FAILURE: TraceMsg("Received ERROR_GEN_FAILURE from cscdll"); if (dwStatus & FLAG_CSC_COPY_STATUS_FILE_IN_USE) dwParam2 = ERROR_OPEN_FILES; break; case ERROR_FILE_NOT_FOUND: case ERROR_PATH_NOT_FOUND: // We either handle the error here or the user is // prompted, so no need for another error message. dwParam2 = NOERROR; // If this is an autocache file and has not been modified // offline, nuke it now. Otherwise, prompt for action. if (CSCPROC_RETURN_FORCE_OUTWARD == HandleDeleteConflict(pSyncData, pszName, dwStatus, dwHintFlags, pFind32)) { dwParam2 = CopyLocalFileWithDriveMapping(pszName, pszName, pSyncData->pszShareName, pSyncData->szDrive, (FILE_ATTRIBUTE_DIRECTORY & pFind32->dwFileAttributes)); } break; case ERROR_DISK_FULL: // There's no point continuing dwResult = CSCPROC_RETURN_ABORT; break; default: // nothing break; } } else { pSyncData->dwSyncStatus &= ~SDS_SYNC_FILE_SKIPPED; dwParam2 = NOERROR; // If doing full sync, then we count progress for skipped // files as well. Not true for quick fill or merge. if (pSyncData->dwSyncStatus & SDS_SYNC_IN_FULL) bUpdateProgress = TRUE; } // Update progress in SyncMgr if (bUpdateProgress) { pSyncData->cFilesDone++; spi.mask = SYNCMGRPROGRESSITEM_PROGVALUE; spi.iProgValue = min(pSyncData->cFilesDone, pSyncData->cFilesToSync - 1); Trace((TEXT("%d of %d files done"), spi.iProgValue, pSyncData->cFilesToSync)); NotifySyncMgr(pSyncData, &spi); } } if (dwParam2 != NOERROR) { UINT idsError = GetErrorFormat(dwParam2, boolify(pSyncData->dwSyncStatus & SDS_SYNC_OUT)); if (IDS_SHARE_CONNECT_ERROR == idsError) { LPTSTR pszErr = GetErrorText(dwParam2); // Special-case the "can't connect to share" error. // Display only the share name in the error message // and abort the synchronization of this share. LogError(pSyncData->ItemID, SYNCMGRLOGLEVEL_ERROR, idsError, pSyncData->pszShareName, pszErr ? pszErr : TEXT("")); LocalFreeString(&pszErr); dwResult = CSCPROC_RETURN_ABORT; } else { LogError(pSyncData->ItemID, idsError, pszName, dwParam2); } pSyncData->dwSyncStatus |= SDS_SYNC_ERROR; } break; } // Check for Cancel if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED) { TraceMsg("Cancelling sync operation"); dwResult = CSCPROC_RETURN_ABORT; } TraceLeaveValue(dwResult); } void CCscUpdate::NotifySyncMgr(PSYNCTHREADDATA pSyncData, LPSYNCMGRPROGRESSITEM pspi) { LPSYNCMGRSYNCHRONIZECALLBACK pSyncMgr = pSyncData->pThis->m_pSyncMgrCB; if (pSyncMgr) { HRESULT hr = pSyncMgr->Progress(pSyncData->ItemID, pspi); if (hr == S_SYNCMGR_CANCELITEM || hr == S_SYNCMGR_CANCELALL) pSyncData->dwSyncStatus |= SDS_SYNC_CANCELLED; } } DWORD WINAPI CCscUpdate::_CscCallback(LPCTSTR pszName, DWORD dwStatus, DWORD dwHintFlags, DWORD dwPinCount, LPWIN32_FIND_DATA pFind32, DWORD dwReason, DWORD dwParam1, DWORD dwParam2, DWORD_PTR dwContext) { DWORD dwResult = CSCPROC_RETURN_ABORT; PSYNCTHREADDATA pSyncData = (PSYNCTHREADDATA)dwContext; if (pSyncData != NULL && pSyncData->pThis != NULL) dwResult = pSyncData->pThis->CscCallback(pSyncData, pszName, dwStatus, dwHintFlags, dwPinCount, pFind32, dwReason, dwParam1, dwParam2); return dwResult; } BOOL CCscUpdate::PinLinkTarget(LPCTSTR pszName, PSYNCTHREADDATA pSyncData) { BOOL bResult = FALSE; LPTSTR pszTarget = NULL; DWORD dwAttr = 0; TraceEnter(TRACE_SHELLEX, "PinLinkTarget"); GetLinkTarget(pszName, NULL, &pszTarget, &dwAttr); if (pszTarget) { TraceAssert(!(dwAttr & FILE_ATTRIBUTE_DIRECTORY)); // Check for EFS if ((FILE_ATTRIBUTE_ENCRYPTED & dwAttr) && SkipEFSPin(pSyncData, pszTarget)) ExitGracefully(bResult, FALSE, "Skipping EFS link target"); if (!(pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED) && CSCPinFile(pszTarget, FLAG_CSC_HINT_PIN_USER | FLAG_CSC_HINT_PIN_INHERIT_USER, NULL, NULL, NULL)) { WIN32_FIND_DATA fd = {0}; LPCTSTR pszT = PathFindFileName(pszTarget); fd.dwFileAttributes = dwAttr; lstrcpyn(fd.cFileName, pszT ? pszT : pszTarget, ARRAYSIZE(fd.cFileName)); ShellChangeNotify(pszTarget, &fd, FALSE); bResult = TRUE; if (FILE_ATTRIBUTE_ENCRYPTED & dwAttr) { LogError(pSyncData->ItemID, IDS_PIN_ENCRYPT_WARNING, pszTarget, NOERROR, SYNCMGRLOGLEVEL_WARNING); } } } exit_gracefully: LocalFreeString(&pszTarget); TraceLeaveValue(bResult); } DWORD WINAPI CCscUpdate::_PinNewFilesW32Callback(LPCTSTR pszName, ENUM_REASON eReason, LPWIN32_FIND_DATA pFind32, LPARAM lpContext) { PSYNCTHREADDATA pSyncData = (PSYNCTHREADDATA)lpContext; DWORD dwHintFlags = 0; DWORD dwErr = NOERROR; // This callback is used when enumerating a pinned folder looking // for new files on the server. Since the parent folder is pinned, // any files in it that aren't pinned get pinned here. TraceEnter(TRACE_UPDATE, "CCscUpdate::_PinNewFilesW32Callback"); TraceAssert(pSyncData != NULL); // Check for Cancel if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED) { TraceMsg("Cancelling sync operation"); TraceLeaveValue(CSCPROC_RETURN_ABORT); } // Always ignore folder_end and ignore folder_begin if we // aren't doing a recursive pin operation. if (eReason == ENUM_REASON_FOLDER_END || (eReason == ENUM_REASON_FOLDER_BEGIN && !(pSyncData->pThis->m_dwSyncFlags & CSC_SYNC_PIN_RECURSE))) { TraceLeaveValue(CSCPROC_RETURN_SKIP); } // At this point, we either have 1) a file or 2) folder_begin + recurse, // so pin anything that isn't pinned. // Is this file already pinned? if (!CSCQueryFileStatus(pszName, NULL, NULL, &dwHintFlags)) dwErr = GetLastError(); if (ERROR_FILE_NOT_FOUND == dwErr || (NOERROR == dwErr && !(dwHintFlags & (FLAG_CSC_HINT_PIN_USER | FLAG_CSC_HINT_PIN_ADMIN)))) { // Check for EFS BOOL bIsEFSFile = (FILE_ATTRIBUTE_ENCRYPTED & pFind32->dwFileAttributes) && !(FILE_ATTRIBUTE_DIRECTORY & pFind32->dwFileAttributes); if (bIsEFSFile && pSyncData->pThis->SkipEFSPin(pSyncData, pszName)) TraceLeaveResult(CSCPROC_RETURN_SKIP); if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED) TraceLeaveValue(CSCPROC_RETURN_ABORT); // Pin it now. dwHintFlags |= FLAG_CSC_HINT_PIN_USER | FLAG_CSC_HINT_PIN_INHERIT_USER; if (CSCPinFile(pszName, dwHintFlags, NULL, NULL, NULL)) { pSyncData->cFilesToSync++; ShellChangeNotify(pszName, pFind32, FALSE); if (bIsEFSFile) { pSyncData->pThis->LogError(pSyncData->ItemID, IDS_PIN_ENCRYPT_WARNING, pszName, NOERROR, SYNCMGRLOGLEVEL_WARNING); } // If this is a link file, pin the target (if appropriate) LPTSTR pszExtn = PathFindExtension(pszName); if (pszExtn && !lstrcmpi(pszExtn, c_szLNK)) { if (pSyncData->pThis->PinLinkTarget(pszName, pSyncData)) pSyncData->cFilesToSync++; } } else { DWORD dwError = GetLastError(); UINT idsError = GetErrorFormat(dwError); if (IDS_SHARE_CONNECT_ERROR == idsError) { LPTSTR pszErr = GetErrorText(dwError); // Special-case the "can't connect to share" error. // Display only the share name in the error message // and abort the pinning of this share. pSyncData->pThis->LogError(pSyncData->ItemID, SYNCMGRLOGLEVEL_ERROR, idsError, pSyncData->pszShareName, pszErr ? pszErr : TEXT("")); LocalFreeString(&pszErr); pSyncData->dwSyncStatus |= SDS_SYNC_CANCELLED; } else { pSyncData->pThis->LogError(pSyncData->ItemID, IDS_PIN_FILE_ERROR, pszName, dwError); } pSyncData->dwSyncStatus |= SDS_SYNC_ERROR; } LPTSTR pszScanMsg = NULL; SYNCMGRPROGRESSITEM spi; spi.cbSize = sizeof(spi); spi.mask = SYNCMGRPROGRESSITEM_STATUSTEXT; spi.lpcStatusText = L" "; // Skip the share name TraceAssert(PathIsPrefix(pSyncData->pszShareName, pszName)); pszName += lstrlen(pSyncData->pszShareName); if (*pszName == TEXT('\\')) pszName++; LPCTSTR pszFile = PathFindFileName(pszName); TCHAR szPath[MAX_PATH] = TEXT("\\"); if (*pszName) lstrcpyn(szPath, pszName, min(ARRAYSIZE(szPath),(int)(pszFile-pszName))); // If we still have a name, build a string like // "scanning: dir\foo.txt" to display in SyncMgr if (FormatStringID(&pszScanMsg, g_hInstance, IDS_NEW_SCAN, pszFile, szPath)) { USES_CONVERSION; spi.lpcStatusText = T2CW(pszScanMsg); } NotifySyncMgr(pSyncData, &spi); LocalFreeString(&pszScanMsg); } else if ((dwHintFlags & FLAG_CSC_HINT_PIN_USER) && (pSyncData->pThis->m_dwSyncFlags & CSC_SYNC_PINFILES)) { // FLAG_CSC_HINT_PIN_USER being set implies that CSCQueryFileStatus // succeeded above. // The item was already pinned. Save it in the undo exclusion list. if (!pSyncData->pUndoExclusionList) pSyncData->pUndoExclusionList = new CscFilenameList; if (pSyncData->pUndoExclusionList) pSyncData->pUndoExclusionList->AddFile(pszName); } TraceLeaveValue(CSCPROC_RETURN_CONTINUE); } DWORD WINAPI CCscUpdate::_PinNewFilesCSCCallback(LPCTSTR pszName, ENUM_REASON eReason, DWORD /*dwStatus*/, DWORD dwHintFlags, DWORD /*dwPinCount*/, LPWIN32_FIND_DATA /*pFind32*/, LPARAM lpContext) { PSYNCTHREADDATA pSyncData = (PSYNCTHREADDATA)lpContext; PCSCUPDATE pThis; // This callback is used when enumerating the CSC database looking // for pinned folders, with the intention of pinning new files // in those folders on the server. TraceEnter(TRACE_UPDATE, "CCscUpdate::_PinNewFilesCSCCallback"); TraceAssert(pSyncData != NULL); TraceAssert(pSyncData->pThis != NULL); pThis = pSyncData->pThis; // Check for Cancel if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED) { TraceMsg("Cancelling sync operation"); TraceLeaveValue(CSCPROC_RETURN_ABORT); } // If this isn't a directory with the user hint flag, keep looking. if (eReason != ENUM_REASON_FOLDER_BEGIN || !(dwHintFlags & FLAG_CSC_HINT_PIN_USER)) { TraceLeaveValue(CSCPROC_RETURN_CONTINUE); } // If we have a file list and this directory isn't in the list, // continue without doing anything here. if (pSyncData->pThis->m_pFileList && !pSyncData->pThis->m_pFileList->FileExists(pszName, false)) { TraceLeaveValue(CSCPROC_RETURN_CONTINUE); } // Ok, we've found a directory with the user hint flag set. Walk // this directory on the server, pinning any files that aren't pinned. _Win32EnumFolder(pszName, FALSE, _PinNewFilesW32Callback, (LPARAM)pSyncData); TraceLeaveValue(CSCPROC_RETURN_CONTINUE); } DWORD WINAPI CCscUpdate::_SyncThread(LPVOID pThreadData) { // Make sure the DLL stays loaded while this thread is running HINSTANCE hInstThisDll = LoadLibrary(c_szDllName); PSYNCTHREADDATA pSyncData = (PSYNCTHREADDATA)pThreadData; PCSCUPDATE pThis; HRESULT hrComInit = E_FAIL; SYNCMGRPROGRESSITEM spi = {0}; DWORD dwErr = NOERROR; CSCSHARESTATS shareStats; CSCGETSTATSINFO si = { SSEF_NONE, SSUF_NONE, false, // No access info reqd (faster). false }; ULONG cDirtyFiles = 0; ULONG cStaleFiles = 0; DWORD dwShareStatus = 0; BOOL bShareOnline = FALSE; DWORD dwConnectionSpeed = 0; if (!pSyncData) return ERROR_INVALID_PARAMETER; pThis = pSyncData->pThis; if (!pThis || !pSyncData->pszShareName || !*pSyncData->pszShareName) { LocalFree(pSyncData); return ERROR_INVALID_PARAMETER; } // Make sure the object stays alive as long as we need it pThis->AddRef(); spi.cbSize = sizeof(spi); TraceEnter(TRACE_UPDATE, "CCscUpdate::_SyncThread"); hrComInit = CoInitialize(NULL); if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED) ExitGracefully(dwErr, NOERROR, "Cancelling sync operation"); // Figure out how many files need updating pSyncData->cFilesDone = 0; pSyncData->cFilesToSync = 0; _GetShareStatisticsForUser(pSyncData->pszShareName, &si, &shareStats); // Get share status CSCQueryFileStatus(pSyncData->pszShareName, &dwShareStatus, NULL, NULL); // The root of a special folder is pinned with a pin count, but // not the user-hint flag, so _GetShareStats doesn't count it. // We need to count this for some of the checks below. // (If the share is manual-cache, these look exactly like the // Siemens scenario in 341786, but we want to sync them.) if (shareStats.cTotal && pThis->IsSpecialFolderShare(pSyncData->pszShareName)) { shareStats.cPinned++; } if (pThis->m_dwSyncFlags & CSC_SYNC_OUT) { cDirtyFiles = shareStats.cModified; // Force the merge code if there are open files, so we are // sure to do the open file warning. The danger here is that we // don't warn because the share with open files has nothing dirty, // but we merge changes on another share and then transition online. // We don't want to transition online without warning about open files. if (0 == cDirtyFiles) { if ((FLAG_CSC_SHARE_STATUS_DISCONNECTED_OP & dwShareStatus) && (FLAG_CSC_SHARE_STATUS_FILES_OPEN & dwShareStatus)) { cDirtyFiles++; } } } if (pThis->m_dwSyncFlags & CSC_SYNC_IN_FULL) { // For full inward sync, always set cStaleFiles to at least 1 to force // a call to CSCFillSparseFiles. // Also, we get callbacks for each file and folder, even if they // are not sparse or stale, so go with cTotal here to make // the progress bar look right. cStaleFiles = max(shareStats.cTotal, 1); } else if (pThis->m_dwSyncFlags & CSC_SYNC_IN_QUICK) { cStaleFiles = shareStats.cSparse; // If we're pinning, then it's possible that nothing is sparse yet, // but we'll need to call CSCFillSparseFiles. if (pThis->m_dwSyncFlags & CSC_SYNC_PINFILES) pSyncData->dwSyncStatus |= SDS_SYNC_FORCE_INWARD; } // Self-host notification callback CSCUI_NOTIFYHOOK((CSCH_SyncShare, TEXT("Sync: %1, Dirty: %2!d!, Stale: %3!d!"), pSyncData->pszShareName, cDirtyFiles, cStaleFiles)); if (dwShareStatus & FLAG_CSC_SHARE_STATUS_DISCONNECTED_OP) { // Can't call CSCFillSparseFiles when disconnected (it just fails) cStaleFiles = 0; } else if ((dwShareStatus & FLAG_CSC_SHARE_STATUS_CACHING_MASK) == FLAG_CSC_SHARE_STATUS_MANUAL_REINT && 0 == shareStats.cPinned && !(pThis->m_dwSyncFlags & CSC_SYNC_PINFILES)) { // On a manual share, if nothing is pinned (and we aren't pinning) // then we prefer not to call CSCFillSparseFiles on the share. // Raid #341786 Trace((TEXT("Manual cache share '%s' has only autocached files"), pSyncData->pszShareName)); cStaleFiles = 0; } pSyncData->cFilesToSync = cDirtyFiles + cStaleFiles; // At this point, if pSyncData->cFilesToSync is nonzero, then we are doing // a sync, and will be calling CSCBeginSynchronization to connect to the // share (with prompt for credentials if necessary). // If SDS_SYNC_FORCE_INWARD is on, then we are pinning files. We will only // call CSCFillSparseFiles is the server is in connected mode, and we will // only call CSCBeginSynchronization if pSyncData->cFilesToSync is nonzero // (to pin something, you must already have a connection to the share). if (0 == pSyncData->cFilesToSync && !(pSyncData->dwSyncStatus & SDS_SYNC_FORCE_INWARD)) ExitGracefully(dwErr, NOERROR, "Nothing to synchronize"); // Tell SyncMgr how many files we're updating spi.mask = SYNCMGRPROGRESSITEM_STATUSTYPE | SYNCMGRPROGRESSITEM_PROGVALUE | SYNCMGRPROGRESSITEM_MAXVALUE; spi.dwStatusType = SYNCMGRSTATUS_UPDATING; spi.iProgValue = 0; spi.iMaxValue = pSyncData->cFilesToSync; Trace((TEXT("%d files to sync on %s"), spi.iMaxValue, pSyncData->pszShareName)); NotifySyncMgr(pSyncData, &spi); #if 1 // BUGBUG RAS isn't notifying RDR before returning success, so // do this bogus delay. Remove this when RAS is fixed. if (pSyncData->dwSyncStatus & SDS_SYNC_RAS_CONNECTED) { // We've just established a RAS connection, but there's a timing // problem with RDR. Wait until we can connect to the share, but // don't wait longer than RAS_CONNECT_DELAY. Trace((TEXT("RAS delay for %s"), pSyncData->pszShareName)); DWORD dwDelayTime = GetTickCount() + RAS_CONNECT_DELAY; #ifdef DEBUG int cAttempts = 0; #endif while (!bShareOnline && GetTickCount() < dwDelayTime && !(pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED)) { WaitForSingleObject(pThis->m_hSyncMutex, INFINITE); #ifdef DEBUG cAttempts++; #endif bShareOnline = CSCBeginSynchronization(pSyncData->pszShareName, &dwConnectionSpeed, &pSyncData->dwCscContext); ReleaseMutex(pThis->m_hSyncMutex); } Trace((TEXT("%s %s after %d attempts (%dms)"), pSyncData->pszShareName, (bShareOnline ? TEXT("connected") : TEXT("not reachable")), cAttempts, GetTickCount() - (dwDelayTime - RAS_CONNECT_DELAY))); } else #endif // RAS delay if (pSyncData->cFilesToSync) { // CSCBeginSynchronization makes a net connection to the share // using the "interactive" flag. This causes a credential popup // if the current user doesn't have access to the share. // Use the sync mutex to avoid multiple concurrent popups. WaitForSingleObject(pThis->m_hSyncMutex, INFINITE); bShareOnline = CSCBeginSynchronization(pSyncData->pszShareName, &dwConnectionSpeed, &pSyncData->dwCscContext); ReleaseMutex(pThis->m_hSyncMutex); } if (pSyncData->cFilesToSync && !bShareOnline) { // The share isn't reachable, so there's no point in continuing. dwErr = GetLastError(); if (ERROR_CANCELLED == dwErr) { // The user cancelled the credential popup pSyncData->dwSyncStatus |= SDS_SYNC_CANCELLED; ExitGracefully(dwErr, NOERROR, "User cancelled sync"); } LPTSTR pszErr = GetErrorText(dwErr); pThis->LogError(pSyncData->ItemID, SYNCMGRLOGLEVEL_ERROR, IDS_SHARE_CONNECT_ERROR, pSyncData->pszShareName, pszErr); LocalFreeString(&pszErr); ExitGracefully(dwErr, dwErr, "Share not reachable"); } if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED) ExitGracefully(dwErr, NOERROR, "Cancelling sync operation"); // Note the time of this sync pThis->SetLastSyncTime(pSyncData->pszShareName); // Merge if (0 != cDirtyFiles) { dwErr = pThis->MergeShare(pSyncData); if (NOERROR != dwErr) { LPTSTR pszErr = GetErrorText(dwErr); pThis->LogError(pSyncData->ItemID, SYNCMGRLOGLEVEL_ERROR, IDS_MERGE_SHARE_ERROR, pSyncData->pszShareName, pszErr); LocalFreeString(&pszErr); ExitGracefully(dwErr, dwErr, "Aborting due to merge error"); } } if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED) ExitGracefully(dwErr, NOERROR, "Cancelling sync operation"); // Fill if (0 != cStaleFiles || (pSyncData->dwSyncStatus & SDS_SYNC_FORCE_INWARD)) { dwErr = pThis->FillShare(pSyncData, shareStats.cPinned, dwConnectionSpeed); } exit_gracefully: // If we called CSCBeginSynchronization and it succeeded, // we need to call CSCEndSynchronization. if (bShareOnline) CSCEndSynchronization(pSyncData->pszShareName, pSyncData->dwCscContext); // Tell SyncMgr that we're done (succeeded, failed, or stopped) spi.mask = SYNCMGRPROGRESSITEM_STATUSTYPE | SYNCMGRPROGRESSITEM_STATUSTEXT | SYNCMGRPROGRESSITEM_PROGVALUE | SYNCMGRPROGRESSITEM_MAXVALUE; spi.dwStatusType = SYNCMGRSTATUS_SUCCEEDED; // Assume success if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED) spi.dwStatusType = SYNCMGRSTATUS_STOPPED; if (NOERROR != dwErr || (pSyncData->dwSyncStatus & SDS_SYNC_ERROR)) spi.dwStatusType = SYNCMGRSTATUS_FAILED; spi.lpcStatusText = L" "; spi.iProgValue = spi.iMaxValue = pSyncData->cFilesToSync; // This tells syncmgr that the item is done NotifySyncMgr(pSyncData, &spi); if ((pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED) && (pThis->m_dwSyncFlags & CSC_SYNC_PINFILES)) { // We cancelled a pin operation, roll back to the previous state CscUnpinFileList(pThis->m_pFileList, (pThis->m_dwSyncFlags & CSC_SYNC_PIN_RECURSE), pSyncData->pszShareName, _UndoProgress, (LPARAM)pSyncData); } // Tell the Update Handler that this thread is exiting // This may use OLE to notify SyncMgr that the sync is done, // so do this before CoUninitialize. Trace((TEXT("%s finished"), pSyncData->pszShareName)); pThis->SyncThreadCompleted(pSyncData); if (SUCCEEDED(hrComInit)) CoUninitialize(); delete pSyncData->pUndoExclusionList; LocalFree(pSyncData); // Release our ref on the object // (also release our ref on the DLL) pThis->Release(); TraceLeave(); FreeLibraryAndExitThread(hInstThisDll, dwErr); return 0; } DWORD CCscUpdate::MergeShare(PSYNCTHREADDATA pSyncData) { DWORD dwErr = NOERROR; BOOL bMergeResult = TRUE; TraceEnter(TRACE_UPDATE, "CCscUpdate::MergeShare"); // CSCMergeShare fails if another thread (or process) is // currently merging. This is because CSCMergeShare uses // a drive letter connection to the share to bypass CSC, // and we don't want to use up all of the drive letters. // So let's protect the call to CSCMergeShare with a mutex // (rather than dealing with failure and retrying, etc.) WaitForSingleObject(m_hSyncMutex, INFINITE); if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED) ExitGracefully(dwErr, NOERROR, "Merge cancelled"); // It would be nice to skip the open file warning if we knew // that the open files were on a "silent folder". The best // we can do, though, is detect that the open files are on // the same share as a silent folder. There's no guarantee // that the open files are not from a different folder on // the same share, so we have to show the warning. //if (!IsSilentShare(pSyncData->pszShareName)) { DWORD dwShareStatus = 0; CSCQueryFileStatus(pSyncData->pszShareName, &dwShareStatus, NULL, NULL); if (FLAG_CSC_SHARE_STATUS_FILES_OPEN & dwShareStatus) { if (CSC_SYNC_MAYBOTHERUSER & m_dwSyncFlags) { // Only show this warning once per sync (not per thread) if (!(CSC_SYNC_OFWARNINGDONE & m_dwSyncFlags)) { m_dwSyncFlags |= CSC_SYNC_OFWARNINGDONE; if (IDOK != OpenFilesWarningDialog()) { TraceMsg("Cancelling sync - user bailed at open file warning"); SetItemStatus(GUID_NULL, SYNCMGRSTATUS_STOPPED); } } // else we already put up the warning on another thread. If the // user cancelled, SDS_SYNC_CANCELLED will be set. } else { // Don't merge, but continue otherwise. LogError(pSyncData->ItemID, SYNCMGRLOGLEVEL_WARNING, IDS_OPENFILE_MERGE_WARNING, pSyncData->pszShareName); ExitGracefully(dwErr, NOERROR, "Skipping merge due to open files"); } } } pSyncData->dwSyncStatus = SDS_SYNC_OUT; // Conflict resolution may require stopping and restarting CSCMergeShare, // so do this in a loop while (!(pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED)) { Trace((TEXT("Calling CSCMergeShare(%s)"), pSyncData->pszShareName)); bMergeResult = CSCMergeShare(pSyncData->pszShareName, CCscUpdate::_CscCallback, (DWORD_PTR)pSyncData); Trace((TEXT("CSCMergeShare(%s) returned"), pSyncData->pszShareName)); // Do we need to merge again? if (!(SDS_SYNC_RESTART_MERGE & pSyncData->dwSyncStatus)) break; pSyncData->dwSyncStatus &= ~SDS_SYNC_RESTART_MERGE; } if (!(pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED) && !bMergeResult) { dwErr = GetLastError(); if (ERROR_OPERATION_ABORTED == dwErr) dwErr = NOERROR; } exit_gracefully: ReleaseMutex(m_hSyncMutex); TraceLeaveValue(dwErr); } DWORD CCscUpdate::FillShare(PSYNCTHREADDATA pSyncData, int cPinned, DWORD dwConnectionSpeed) { DWORD dwErr = NOERROR; DWORD dwShareStatus = 0; DWORD dwShareHints = 0; TraceEnter(TRACE_UPDATE, "CCscUpdate::FillShare"); CSCQueryFileStatus(pSyncData->pszShareName, &dwShareStatus, NULL, &dwShareHints); if (m_dwSyncFlags & CSC_SYNC_IN_FULL) { pSyncData->dwSyncStatus = SDS_SYNC_IN_FULL; Trace((TEXT("Full sync at %d00 bps"), dwConnectionSpeed)); // Check the server for new files that should be pinned. // We can't do this when disconnected. Also, this is // time consuming, so don't do it on a slow connection. if (!(FLAG_CSC_SHARE_STATUS_DISCONNECTED_OP & dwShareStatus) && cPinned && !_PathIsSlow(dwConnectionSpeed)) { // Look for pinned folders on this share by enumerating // in the CSC database. Go out to the server only if/when // we find a pinned folder. TraceMsg("Running FrankAr code"); if (CConfig::GetSingleton().AlwaysPinSubFolders()) { // If the "AlwaysPinSubFolders" policy is set, we // do a recursive pin. This will cause any content // (including folders) of a pinned folder to become pinned. pSyncData->pThis->m_dwSyncFlags |= CSC_SYNC_PIN_RECURSE; } // First check the root folder if (_PinNewFilesCSCCallback(pSyncData->pszShareName, ENUM_REASON_FOLDER_BEGIN, 0, dwShareHints, 0, NULL, (LPARAM)pSyncData) == CSCPROC_RETURN_CONTINUE) { _CSCEnumDatabase(pSyncData->pszShareName, TRUE, _PinNewFilesCSCCallback, (LPARAM)pSyncData); } TraceMsg("FrankAr code complete"); } } else { pSyncData->dwSyncStatus = SDS_SYNC_IN_QUICK; if (m_dwSyncFlags & CSC_SYNC_PINFILES) { // Enumerate the file list and pin everything, checking with // SyncMgr periodically. PinFiles(pSyncData); } } if (m_pConflictPinList) { // Make sure that any files we created because of merge // conflicts are pinned. PinFiles(pSyncData, TRUE); } // Can't fill when disconnected if (!(FLAG_CSC_SHARE_STATUS_DISCONNECTED_OP & dwShareStatus)) { // Clear the status text and update the max count in case we // pinned somthing above SYNCMGRPROGRESSITEM spi; spi.cbSize = sizeof(spi); spi.mask = SYNCMGRPROGRESSITEM_STATUSTEXT | SYNCMGRPROGRESSITEM_MAXVALUE; spi.lpcStatusText = L" "; spi.iMaxValue = pSyncData->cFilesToSync; Trace((TEXT("%d files to sync on %s"), spi.iMaxValue, pSyncData->pszShareName)); NotifySyncMgr(pSyncData, &spi); if (!(pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED)) { Trace((TEXT("Calling CSCFillSparseFiles(%s, %s)"), pSyncData->pszShareName, (pSyncData->dwSyncStatus & SDS_SYNC_IN_FULL) ? TEXT("full") : TEXT("quick"))); if (!CSCFillSparseFiles(pSyncData->pszShareName, !!(pSyncData->dwSyncStatus & SDS_SYNC_IN_FULL), CCscUpdate::_CscCallback, (DWORD_PTR)pSyncData)) { dwErr = GetLastError(); if (ERROR_OPERATION_ABORTED == dwErr) dwErr = NOERROR; } Trace((TEXT("CSCFillSparseFiles(%s) complete"), pSyncData->pszShareName)); } } else { Trace((TEXT("Skipping CSCFillSparseFiles(%s) - server is offline"), pSyncData->pszShareName)); } TraceLeaveValue(dwErr); } void CCscUpdate::PinFiles(PSYNCTHREADDATA pSyncData, BOOL bConflictPinList) { CscFilenameList *pfnl; CscFilenameList::HSHARE hShare; LPCTSTR pszFile; BOOL bRecurse; TraceEnter(TRACE_UPDATE, "CCscUpdate::PinFiles"); TraceAssert((m_dwSyncFlags & CSC_SYNC_PINFILES) || bConflictPinList); pfnl = m_pFileList; bRecurse = m_dwSyncFlags & CSC_SYNC_PIN_RECURSE; if (bConflictPinList) { pfnl = m_pConflictPinList; bRecurse = FALSE; } if (!pfnl || !pfnl->GetShareHandle(pSyncData->pszShareName, &hShare)) { TraceLeaveVoid(); } CscFilenameList::FileIter fi = pfnl->CreateFileIterator(hShare); // Iterate over the filenames associated with the share. while (pszFile = fi.Next()) { TCHAR szFullPath[MAX_PATH]; TCHAR szRelativePath[MAX_PATH]; WIN32_FIND_DATA fd; ULONG cchFile = lstrlen(pszFile) + 1; // include NULL // Check for Cancel if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED) break; ZeroMemory(&fd, sizeof(fd)); // Directories have a trailing "\*" if (StrChr(pszFile, TEXT('*'))) { // It's a directory. Trim off the "\*" cchFile -= 2; // When pinning at the share level, pszFile points to "*" // and cchFile is now zero. } szRelativePath[0] = TEXT('\0'); lstrcpyn(szRelativePath, pszFile, min(cchFile, ARRAYSIZE(szRelativePath))); // Build the full path PathCombine(szFullPath, pSyncData->pszShareName, szRelativePath); // Get attributes and test for existence fd.dwFileAttributes = GetFileAttributes(szFullPath); if ((DWORD)-1 == fd.dwFileAttributes) continue; pszFile = PathFindFileName(szFullPath); lstrcpyn(fd.cFileName, pszFile ? pszFile : szFullPath, ARRAYSIZE(fd.cFileName)); // Check for EFS BOOL bIsEFSFile; bIsEFSFile = (FILE_ATTRIBUTE_ENCRYPTED & fd.dwFileAttributes) && !(FILE_ATTRIBUTE_DIRECTORY & fd.dwFileAttributes); if (bIsEFSFile && SkipEFSPin(pSyncData, szFullPath)) continue; if (pSyncData->dwSyncStatus & SDS_SYNC_CANCELLED) break; // Pin it if (CSCPinFile(szFullPath, FLAG_CSC_HINT_PIN_USER | FLAG_CSC_HINT_PIN_INHERIT_USER, NULL, NULL, NULL)) { if (bConflictPinList && m_pFileList) m_pFileList->AddFile(szFullPath, !!(fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)); pSyncData->cFilesToSync++; ShellChangeNotify(szFullPath, &fd, FALSE); if (bIsEFSFile) { LogError(pSyncData->ItemID, IDS_PIN_ENCRYPT_WARNING, szFullPath, NOERROR, SYNCMGRLOGLEVEL_WARNING); } } else { DWORD dwError = GetLastError(); UINT idsError = GetErrorFormat(dwError); if (IDS_SHARE_CONNECT_ERROR == idsError) { LPTSTR pszErr = GetErrorText(dwError); // Special-case the "can't connect to share" error. // Display only the share name in the error message // and abort the pinning of this share. LogError(pSyncData->ItemID, SYNCMGRLOGLEVEL_ERROR, idsError, pSyncData->pszShareName, pszErr ? pszErr : TEXT("")); LocalFreeString(&pszErr); pSyncData->dwSyncStatus |= SDS_SYNC_CANCELLED; } else { LogError(pSyncData->ItemID, IDS_PIN_FILE_ERROR, szFullPath, dwError); } pSyncData->dwSyncStatus |= SDS_SYNC_ERROR; } // If it's a directory, pin its contents if (fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) { _Win32EnumFolder(szFullPath, bRecurse, CCscUpdate::_PinNewFilesW32Callback, (LPARAM)pSyncData); } } // Flush the shell notify queue ShellChangeNotify(NULL, TRUE); TraceLeaveVoid(); } void CCscUpdate::NotifyUndo(PSYNCTHREADDATA pSyncData, LPCTSTR pszName) { LPTSTR pszMsg; SYNCMGRPROGRESSITEM spi; spi.cbSize = sizeof(spi); spi.mask = SYNCMGRPROGRESSITEM_STATUSTEXT; spi.lpcStatusText = L" "; // Skip the share name if (PathIsPrefix(pSyncData->pszShareName, pszName)) { pszName += lstrlen(pSyncData->pszShareName); if (*pszName == TEXT('\\')) pszName++; } LPCTSTR pszFile = PathFindFileName(pszName); TCHAR szPath[MAX_PATH] = TEXT("\\"); if (*pszName) lstrcpyn(szPath, pszName, min(ARRAYSIZE(szPath),(int)(pszFile-pszName))); // If we still have a name, build a string like // "undo: dir\foo.txt" to display in SyncMgr if (FormatStringID(&pszMsg, g_hInstance, IDS_UNDO_SCAN, pszFile, szPath)) { USES_CONVERSION; spi.lpcStatusText = T2CW(pszMsg); } NotifySyncMgr(pSyncData, &spi); LocalFreeString(&pszMsg); } DWORD WINAPI CCscUpdate::_UndoProgress(LPCTSTR pszItem, LPARAM lpContext) { PSYNCTHREADDATA pSyncData = reinterpret_cast(lpContext); if (pSyncData->pUndoExclusionList && pSyncData->pUndoExclusionList->FileExists(pszItem)) { return CSCPROC_RETURN_SKIP; } // Update SyncMgr pSyncData->pThis->NotifyUndo(pSyncData, pszItem); return CSCPROC_RETURN_CONTINUE; } #define PINEFS_SKIP 0 #define PINEFS_PIN 1 #define PINEFS_CANCEL 2 INT_PTR CALLBACK ConfirmEFSPinProc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) { INT_PTR nResult = 0; switch (uMsg) { case WM_INITDIALOG: { LPTSTR pszMsg = NULL; LPCTSTR pszFile = PathFindFileName((LPCTSTR)lParam); FormatStringID(&pszMsg, g_hInstance, IDS_FMT_PIN_EFS_MSG, pszFile); if (pszMsg) { SetDlgItemText(hDlg, IDC_EFS_MSG, pszMsg); LocalFree(pszMsg); } else EndDialog(hDlg, PINEFS_SKIP); CheckDlgButton(hDlg, IDC_SKIP_EFS, BST_CHECKED); } nResult = TRUE; break; case WM_COMMAND: switch (LOWORD(wParam)) { case IDOK: EndDialog(hDlg, BST_CHECKED == IsDlgButtonChecked(hDlg, IDC_SKIP_EFS) ? PINEFS_SKIP : PINEFS_PIN); nResult = TRUE; break; case IDCANCEL: EndDialog(hDlg, PINEFS_CANCEL); nResult = TRUE; break; } break; } return nResult; } BOOL CCscUpdate::SkipEFSPin(PSYNCTHREADDATA pSyncData, LPCTSTR pszItem) { BOOL bSkip = FALSE; int iResult = PINEFS_PIN; if ((CSC_SYNC_SKIP_EFS & m_dwSyncFlags) || !(CSC_SYNC_MAYBOTHERUSER & m_dwSyncFlags)) { iResult = PINEFS_SKIP; } else if (!(CSC_SYNC_EFS_WARNING_SHOWN & m_dwSyncFlags)) { EnterCriticalSection(&m_csThreadList); // Another thread may have come through here while we were // waiting for the critical section, so recheck any flags // that are modified here. // We could take the critical section at the top of the function and // only check flags once, but this way we avoid extra blocking. if (CSC_SYNC_SKIP_EFS & m_dwSyncFlags) { iResult = PINEFS_SKIP; } else if (!(CSC_SYNC_EFS_WARNING_SHOWN & m_dwSyncFlags)) { m_dwSyncFlags |= CSC_SYNC_EFS_WARNING_SHOWN; // Suspend the other sync threads SetSyncThreadStatus(SyncPause, pSyncData->ItemID); iResult = (int)DialogBoxParam(g_hInstance, MAKEINTRESOURCE(IDD_CONFIRM_PIN_EFS), m_hwndDlgParent, ConfirmEFSPinProc, (LPARAM)pszItem); if (PINEFS_SKIP == iResult) m_dwSyncFlags |= CSC_SYNC_SKIP_EFS; // Resume syncing SetSyncThreadStatus(SyncResume, pSyncData->ItemID); } LeaveCriticalSection(&m_csThreadList); } switch (iResult) { default: case PINEFS_SKIP: bSkip = TRUE; break; case PINEFS_PIN: // continue break; case PINEFS_CANCEL: // stop all threads SetItemStatus(GUID_NULL, SYNCMGRSTATUS_STOPPED); break; } return bSkip; } HRESULT CCscUpdate::SetSyncThreadStatus(eSetSyncStatus status, REFGUID rItemID) { // Assume success here. If we don't find the thread, // it's probably already finished. HRESULT hr = S_OK; BOOL bOneItem; TraceEnter(TRACE_UPDATE, "CCscUpdate::SetSyncThreadStatus"); bOneItem = (SyncStop == status && !IsEqualGUID(rItemID, GUID_NULL)); EnterCriticalSection(&m_csThreadList); if (NULL != m_hSyncThreads) { int cItems = DPA_GetPtrCount(m_hSyncThreads); SYNCMGRPROGRESSITEM spi = {0}; DWORD (WINAPI *pfnStartStop)(HANDLE); pfnStartStop = ResumeThread; spi.cbSize = sizeof(spi); spi.mask = SYNCMGRPROGRESSITEM_STATUSTYPE | SYNCMGRPROGRESSITEM_STATUSTEXT; spi.lpcStatusText = L" "; spi.dwStatusType = SYNCMGRSTATUS_UPDATING; if (SyncPause == status) { spi.dwStatusType = SYNCMGRSTATUS_PAUSED; pfnStartStop = SuspendThread; } while (cItems > 0) { PSYNCTHREADDATA pSyncData; --cItems; pSyncData = (PSYNCTHREADDATA)DPA_FastGetPtr(m_hSyncThreads, cItems); TraceAssert(NULL != pSyncData); if (SyncStop == status) { // Tell the thread to abort if (!bOneItem || IsEqualGUID(rItemID, pSyncData->ItemID)) { pSyncData->dwSyncStatus |= SDS_SYNC_CANCELLED; if (bOneItem) break; } } else { // Suspend or resume the thread if it's not the current thread if (!IsEqualGUID(rItemID, pSyncData->ItemID)) (*pfnStartStop)(pSyncData->hThread); m_pSyncMgrCB->Progress(pSyncData->ItemID, &spi); } } } LeaveCriticalSection(&m_csThreadList); TraceLeaveResult(hr); } void CCscUpdate::GetSilentFolderList(void) { delete m_pSilentFolderList; m_pSilentFolderList = new CscFilenameList; delete m_pSpecialFolderList; m_pSpecialFolderList = new CscFilenameList; BuildSilentFolderList(m_pSilentFolderList, m_pSpecialFolderList); if (0 == m_pSilentFolderList->GetShareCount()) { delete m_pSilentFolderList; m_pSilentFolderList = NULL; } if (0 == m_pSpecialFolderList->GetShareCount()) { delete m_pSpecialFolderList; m_pSpecialFolderList = NULL; } } void BuildSilentFolderList(CscFilenameList *pfnlSilentFolders, CscFilenameList *pfnlSpecialFolders) { // We will silently handle sync conflicts in any of the folders // below that have a '1' after them. // If we get complaints about conflicts in folders that we // think we can handle silently and safely, add them. // Note that CSIDL_PERSONAL (MyDocs) and CSIDL_MYPICTURES // should probably never be silent, since the user // interacts with them directly. // This list corresponds to the list of shell folders that may // be redirected. See also // HKCU\Software\Microsoft\Windows\CurrentVersion\Explorer\Shell Folders static const int s_csidlFolders[][2] = { { CSIDL_PROGRAMS, 0 }, { CSIDL_PERSONAL, 0 }, { CSIDL_FAVORITES, 0 }, { CSIDL_STARTUP, 0 }, { CSIDL_RECENT, 1 }, { CSIDL_SENDTO, 0 }, { CSIDL_STARTMENU, 1 }, { CSIDL_DESKTOPDIRECTORY, 0 }, { CSIDL_NETHOOD, 0 }, { CSIDL_TEMPLATES, 0 }, { CSIDL_APPDATA, 1 }, { CSIDL_PRINTHOOD, 0 }, { CSIDL_MYPICTURES, 0 }, { CSIDL_PROFILE, 1 }, { CSIDL_ADMINTOOLS, 0 }, }; TCHAR szPath[MAX_PATH]; for (int i = 0; i < ARRAYSIZE(s_csidlFolders); i++) { if (SHGetSpecialFolderPath(NULL, szPath, s_csidlFolders[i][0] | CSIDL_FLAG_DONT_VERIFY, FALSE)) { // We only want UNC net paths LPTSTR pszUNC = NULL; GetRemotePath(szPath, &pszUNC); if (!pszUNC) continue; if (s_csidlFolders[i][1]) { if (pfnlSilentFolders) pfnlSilentFolders->AddFile(pszUNC, true); } else { if (pfnlSpecialFolders) pfnlSpecialFolders->AddFile(pszUNC, true); } LocalFreeString(&pszUNC); } } } // SyncMgr integration (ISyncMgrEnumItems) // CUpdateEnumerator::CUpdateEnumerator(PCSCUPDATE pUpdate) : m_cRef(1), m_pUpdate(pUpdate), m_hFind(INVALID_HANDLE_VALUE), m_bEnumFileSelection(FALSE), m_cCheckedItemsEnumerated(0) { DllAddRef(); if (m_pUpdate) { m_pUpdate->AddRef(); if (m_pUpdate->m_pFileList) { m_bEnumFileSelection = TRUE; m_SelectionIterator = m_pUpdate->m_pFileList->CreateShareIterator(); } } } CUpdateEnumerator::~CUpdateEnumerator() { if (m_hFind != INVALID_HANDLE_VALUE) CSCFindClose(m_hFind); DoRelease(m_pUpdate); DllRelease(); } STDMETHODIMP CUpdateEnumerator::QueryInterface(REFIID riid, void **ppv) { static const QITAB qit[] = { QITABENT(CUpdateEnumerator, ISyncMgrEnumItems), { 0 }, }; return QISearch(this, qit, riid, ppv); } STDMETHODIMP_(ULONG) CUpdateEnumerator::AddRef() { return InterlockedIncrement(&m_cRef); } STDMETHODIMP_(ULONG) CUpdateEnumerator::Release() { if (InterlockedDecrement(&m_cRef)) return m_cRef; delete this; return 0; } STDMETHODIMP CUpdateEnumerator::Next(ULONG celt, LPSYNCMGRITEM rgelt, PULONG pceltFetched) { HRESULT hr = S_OK; ULONG cFetched = 0; LPSYNCMGRITEM pItem = rgelt; WIN32_FIND_DATA fd = {0}; DWORD dwShareStatus = 0; DWORD dwSyncFlags; CscFilenameList::HSHARE hShare; LPCTSTR pszShareName = NULL; TraceEnter(TRACE_UPDATE, "CUpdateEnumerator::Next"); TraceAssert(m_pUpdate != NULL); if (NULL == rgelt) TraceLeaveResult(E_INVALIDARG); dwSyncFlags = m_pUpdate->m_dwSyncFlags; while (cFetched < celt) { CSCEntry *pShareEntry; CSCSHARESTATS shareStats; CSCGETSTATSINFO si = { SSEF_NONE, SSUF_TOTAL | SSUF_PINNED | SSUF_MODIFIED | SSUF_SPARSE | SSUF_DIRS, false, false }; if (m_bEnumFileSelection) { if (!m_SelectionIterator.Next(&hShare)) break; pszShareName = m_pUpdate->m_pFileList->GetShareName(hShare); CSCQueryFileStatus(pszShareName, &dwShareStatus, NULL, NULL); fd.dwFileAttributes = FILE_ATTRIBUTE_DIRECTORY; lstrcpyn(fd.cFileName, pszShareName, ARRAYSIZE(fd.cFileName)); } else { if (m_hFind == INVALID_HANDLE_VALUE) { m_hFind = CacheFindFirst(NULL, &fd, &dwShareStatus, NULL, NULL, NULL); if (m_hFind == INVALID_HANDLE_VALUE) { // The database is empty, so there's nothing to enumerate.. break; } pszShareName = fd.cFileName; } else if (CacheFindNext(m_hFind, &fd, &dwShareStatus, NULL, NULL, NULL)) { pszShareName = fd.cFileName; } else break; } TraceAssert(pszShareName); // BUGBUG This was proposed as a fix for part of 383011. However, // that bug only applies in a multi-user scenario, and if there // are more than 3 users using the machine, this would cause a // user whose SID had been expelled from the CSC database to not // be able to sync a share where they do indeed have access. // // If the current user has no access to the share, don't enumerate it. // if (!(CscAccessUser(dwShareStatus) || CscAccessGuest(dwShareStatus))) // continue; // Count the # of pinned files, sparse files, etc. _GetShareStatisticsForUser(pszShareName, &si, &shareStats); // The root of a special folder is pinned with a pin count, but // not the user-hint flag, so _GetShareStats doesn't count it. // We need to count this for some of the checks below. // (If the share is manual-cache, these look exactly like the // Siemens scenario in 341786, but we want to sync them.) if (shareStats.cTotal && m_pUpdate->IsSpecialFolderShare(pszShareName)) { shareStats.cPinned++; } // If we're pinning, then even if nothing is sparse now, // there will be sparse files after we pin them. if (dwSyncFlags & CSC_SYNC_PINFILES) { shareStats.cSparse++; shareStats.cTotal++; } // If there's nothing cached on this share, then don't even // enumerate it to SyncMgr. This avoids listing extra junk // in SyncMgr. if ((0 == shareStats.cTotal) || (shareStats.cTotal == shareStats.cDirs && 0 == shareStats.cPinned)) { // Either there is nothing cached for this share, or the only // things found were unpinned dirs (no files, no pinned dirs). // The second case can happen if you delete files from the viewer, // in which case you think you deleted everything but the viewer // doesn't show directories, so they weren't deleted. Trace((TEXT("Nothing cached on %s, not enumerating"), pszShareName)); continue; } if ((dwShareStatus & FLAG_CSC_SHARE_STATUS_CACHING_MASK) == FLAG_CSC_SHARE_STATUS_NO_CACHING) { // Don't enumerate "no-cache" shares if there's nothing to merge. // These can exist in the cache if the share was previously // cacheable, but has since been changed to "no caching". // If there's something to merge, we should still sync it to // get everything squared away. if (!((dwSyncFlags & CSC_SYNC_OUT) && (shareStats.cModified))) { Trace((TEXT("Not enumerating no-cache share %s"), pszShareName)); continue; } Trace((TEXT("Enumerating no-cache share %s with offline changes."), pszShareName)); } pItem->dwFlags = SYNCMGRITEM_HASPROPERTIES; if ((dwShareStatus & FLAG_CSC_SHARE_STATUS_CACHING_MASK) == FLAG_CSC_SHARE_STATUS_MANUAL_REINT && 0 == shareStats.cPinned && !(dwSyncFlags & CSC_SYNC_PINFILES)) { // On a manual share, if nothing is pinned (and we aren't pinning) // then we don't want to call CSCFillSparseFiles on the share. // Raid #341786 Trace((TEXT("Manual cache share '%s' has only autocached files"), pszShareName)); // However, if there is something to merge, then we need to sync. if (!((dwSyncFlags & CSC_SYNC_OUT) && shareStats.cModified)) { Trace((TEXT("Not enumerating manual-cache share %s"), pszShareName)); continue; } // There is something to merge, so enumerate the share but // tell SyncMgr that it's temporary so it doesn't save state // for this share. pItem->dwFlags |= SYNCMGRITEM_TEMPORARY; } // In some circumstances, we may want to merge even if there // are no modified files, in order to show the open files warning. // See comments in CCscUpdate::_SyncThread if (0 == shareStats.cModified) { if ((FLAG_CSC_SHARE_STATUS_DISCONNECTED_OP & dwShareStatus) && (FLAG_CSC_SHARE_STATUS_FILES_OPEN & dwShareStatus)) { shareStats.cModified++; } } // Enumerate this share cFetched++; // Get existing share entry or create a new one pShareEntry = m_pUpdate->m_ShareLog.Add(pszShareName); if (!pShareEntry) TraceLeaveResult(E_OUTOFMEMORY); pItem->cbSize = sizeof(SYNCMGRITEM); pItem->ItemID = pShareEntry->Guid(); pItem->hIcon = g_hCscIcon; // BUGBUG SYNCMGRITEM_TEMPORARY causes items to not show up in // SyncMgr's logon/logoff settings page. Raid #237288 //if (0 == shareStats.cPinned) // pItem->dwFlags |= SYNCMGRITEM_TEMPORARY; if (ERROR_SUCCESS == m_pUpdate->GetLastSyncTime(pszShareName, &pItem->ftLastUpdate)) pItem->dwFlags |= SYNCMGRITEM_LASTUPDATETIME; // Determine whether this share needs syncing. // At settings time, assume everything needs syncing (check everything) // If outbound, shares with modified files are checked // If inbound (full), shares with sparse or pinned files are checked // If inbound (quick), shares with sparse files are checked // Anything else doesn't need to be sync'ed at this time (unchecked) pItem->dwItemState = SYNCMGRITEMSTATE_CHECKED; if (!(dwSyncFlags & CSC_SYNC_SETTINGS) && !((dwSyncFlags & CSC_SYNC_OUT) && shareStats.cModified) && !((dwSyncFlags & CSC_SYNC_IN_FULL) && shareStats.cTotal ) && !((dwSyncFlags & CSC_SYNC_IN_QUICK) && shareStats.cSparse )) { pItem->dwItemState = SYNCMGRITEMSTATE_UNCHECKED; } // Get friendly share name here LPITEMIDLIST pidl = NULL; SHFILEINFO sfi = {0}; if (SUCCEEDED(SHSimpleIDListFromFindData(pszShareName, &fd, &pidl))) { SHGetFileInfo((LPCTSTR)pidl, 0, &sfi, sizeof(sfi), SHGFI_PIDL | SHGFI_DISPLAYNAME); SHFree(pidl); } if (TEXT('\0') != sfi.szDisplayName[0]) pszShareName = sfi.szDisplayName; SHTCharToUnicode((LPTSTR)pszShareName, pItem->wszItemName, ARRAYSIZE(pItem->wszItemName)); if (SYNCMGRITEMSTATE_CHECKED == pItem->dwItemState) { m_cCheckedItemsEnumerated++; Trace((TEXT("Enumerating %s, checked"), pszShareName)); } else { Trace((TEXT("Enumerating %s, unchecked"), pszShareName)); } pItem++; } if (pceltFetched) *pceltFetched = cFetched; if (cFetched != celt) hr = S_FALSE; if ((S_FALSE == hr) && 0 == m_cCheckedItemsEnumerated && (CSC_SYNC_SHOWUI_ALWAYS & dwSyncFlags)) { // Special-case where we're synching nothing but still // want to display SyncMgr progress UI. We enumerate a // special string rather than a share name for display in // the status UI. Force hr == S_OK so the caller will accept // this "dummy" item. Next() will be called once more but // m_cCheckedItemsEnumerated will be 1 so this block won't be // entered and we'll return S_FALSE indicating the end of the // enumeration. pItem->cbSize = sizeof(SYNCMGRITEM); pItem->hIcon = g_hCscIcon; pItem->dwFlags = SYNCMGRITEM_HASPROPERTIES; pItem->dwItemState = SYNCMGRITEMSTATE_CHECKED; pItem->ItemID = GUID_CscNullSyncItem; UINT idString = IDS_NULLSYNC_ITEMNAME; if ((CSC_SYNC_OUT & dwSyncFlags) && !((CSC_SYNC_IN_QUICK | CSC_SYNC_IN_FULL) & dwSyncFlags)) { // Use different text if we are only merging idString = IDS_NULLMERGE_ITEMNAME; } LoadStringW(g_hInstance, idString, pItem->wszItemName, ARRAYSIZE(pItem->wszItemName)); m_cCheckedItemsEnumerated = 1; TraceMsg("Enumerating NULL item"); hr = S_OK; } TraceLeaveResult(hr); } STDMETHODIMP CUpdateEnumerator::Skip(ULONG celt) { return Next(celt, NULL, NULL); } STDMETHODIMP CUpdateEnumerator::Reset() { m_cCheckedItemsEnumerated = 0; if (m_bEnumFileSelection) { m_SelectionIterator.Reset(); } else if (m_hFind != INVALID_HANDLE_VALUE) { CSCFindClose(m_hFind); m_hFind = INVALID_HANDLE_VALUE; } return S_OK; } STDMETHODIMP CUpdateEnumerator::Clone(LPSYNCMGRENUMITEMS *ppenum) { return E_NOTIMPL; }