//--------------------------------------------------------------------------- // // Copyright (c) Microsoft Corporation 1993-1994 // // File: cbs.c // // This files contains code for the cached briefcase structs // // History: // 09-02-93 ScottH Created // 01-31-94 ScottH Moved from cache.c // //--------------------------------------------------------------------------- ///////////////////////////////////////////////////// INCLUDES #include "brfprv.h" // common headers #include "res.h" CACHE g_cacheCBS = {0, 0, 0}; // Briefcase structure cache #define CBS_EnterCS() EnterCriticalSection(&g_cacheCBS.cs) #define CBS_LeaveCS() LeaveCriticalSection(&g_cacheCBS.cs) SETbl const c_rgseOpenBriefcase[] = { { E_TR_OUT_OF_MEMORY, IDS_OOM_OPENBRIEFCASE, MB_ERROR }, { E_OUTOFMEMORY, IDS_OOM_OPENBRIEFCASE, MB_ERROR }, { E_TR_BRIEFCASE_LOCKED, IDS_ERR_BRIEFCASE_LOCKED, MB_WARNING }, { E_TR_BRIEFCASE_OPEN_FAILED, IDS_ERR_OPEN_ACCESS_DENIED, MB_WARNING }, { E_TR_NEWER_BRIEFCASE, IDS_ERR_NEWER_BRIEFCASE, MB_INFO }, { E_TR_SUBTREE_CYCLE_FOUND, IDS_ERR_OPEN_SUBTREECYCLE, MB_WARNING }, }; #ifdef DEBUG void PRIVATE CBS_DumpEntry( CBS * pcbs) { ASSERT(pcbs); TRACE_MSG(TF_ALWAYS, TEXT("CBS: Atom %d: %s"), pcbs->atomBrf, Atom_GetName(pcbs->atomBrf)); TRACE_MSG(TF_ALWAYS, TEXT(" Ref [%u] Hbrf = %lx "), Cache_GetRefCount(&g_cacheCBS, pcbs->atomBrf), pcbs->hbrf); } void PUBLIC CBS_DumpAll() { CBS * pcbs; int atom; BOOL bDump; ENTEREXCLUSIVE(); { bDump = IsFlagSet(g_uDumpFlags, DF_CBS); } LEAVEEXCLUSIVE(); if (!bDump) return ; atom = Cache_FindFirstKey(&g_cacheCBS); while (atom != ATOM_ERR) { pcbs = Cache_GetPtr(&g_cacheCBS, atom); ASSERT(pcbs); if (pcbs) { CBS_DumpEntry(pcbs); CBS_Delete(atom, NULL); // Decrement count } atom = Cache_FindNextKey(&g_cacheCBS, atom); } } #endif /*---------------------------------------------------------- Purpose: Save and close the briefcase. Returns: -- Cond: This function is serialized by the caller (Cache_Term or Cache_DeleteItem). */ void CALLBACK CBS_Free( LPVOID lpv, HWND hwndOwner) { HBRFCASE hbrf; CBS * pcbs = (CBS *)lpv; CRL * pcrl; int atomPath = pcbs->atomBrf; int atom; TWINRESULT tr1; TWINRESULT tr2; DECLAREHOURGLASS; hbrf = pcbs->hbrf; // Save the briefcase with the same name it was opened // DEBUG_CODE( TRACE_MSG(TF_GENERAL, TEXT("Saving and closing Briefcase %s (0x%lx)"), Atom_GetName(atomPath), hbrf); ) // Search thru the CRL cache for entries // sharing the same partial path as this briefcase // and nuke them. // atom = Cache_FindFirstKey(&g_cacheCRL); while (atom != ATOM_ERR) { pcrl = Cache_GetPtr(&g_cacheCRL, atom); ASSERT(pcrl); if (pcrl) { if (hbrf == pcrl->hbrf) { // This atomKey belongs to this briefcase. Nuke it. // DEBUG_CODE( TRACE_MSG(TF_CACHE, TEXT("CACHE Nuking CRL %d"), atom); ) CRL_Nuke(atom); } #ifdef DEBUG else DEBUG_CODE( TRACE_MSG(TF_CACHE, TEXT("CACHE NOT Nuking CRL %d"), atom); ) #endif Cache_DeleteItem(&g_cacheCRL, atom, FALSE, hwndOwner, CRL_Free); // Decrement count } atom = Cache_FindNextKey(&g_cacheCRL, atom); } // Save the briefcase. We normally (re)specify the database // pathname to handle the rename case. However, if the // move bit has been set, then we use the NULL parameter // (save under current name) because we will depend on the // shell to move the database. // ASSERT(Sync_IsEngineLoaded()); // First check if the disk is available. If it isn't, Windows will // blue-screen because we cannot close the database file. So before // that happens, bring up a friendlier retry messagebox. RETRY_BEGIN(FALSE) { // Is disk unavailable? if ( !PathExists(Atom_GetName(atomPath)) ) { // Yes; ask user to retry/cancel int id = MsgBox(hwndOwner, MAKEINTRESOURCE(IDS_ERR_CLOSE_UNAVAIL_VOL), MAKEINTRESOURCE(IDS_CAP_SAVE), NULL, MB_RETRYCANCEL | MB_ICONWARNING); if (IDRETRY == id) RETRY_SET(); } } RETRY_END() SetHourglass(); tr1 = Sync_SaveBriefcase(pcbs->hbrf); tr2 = Sync_CloseBriefcase(pcbs->hbrf); if (TR_SUCCESS != tr1 || TR_SUCCESS != tr2) { DWORD dwError = GetLastError(); switch (dwError) { case ERROR_ACCESS_DENIED: MsgBox(hwndOwner, MAKEINTRESOURCE(IDS_ERR_SAVE_UNAVAIL_VOL), MAKEINTRESOURCE(IDS_CAP_SAVE), NULL, MB_ERROR); break; default: if (TR_BRIEFCASE_WRITE_FAILED == tr1 || TR_BRIEFCASE_WRITE_FAILED == tr2) { LPTSTR psz; static UINT rgids[2] = { IDS_ERR_1_FullDiskSave, IDS_ERR_2_FullDiskSave }; if (FmtString(&psz, IDS_ERR_F_FullDiskSave, rgids, ARRAYSIZE(rgids))) { MsgBox(hwndOwner, psz, MAKEINTRESOURCE(IDS_CAP_SAVE), NULL, MB_ERROR); GFree(psz); } } break; } } ResetHourglass(); AbortEvt_Free(pcbs->pabortevt); SharedFree(&pcbs); } /*---------------------------------------------------------- Purpose: Actually opens the briefcase and adds the briefcase handle to the given CBS struct. Returns: standard hresult Cond: -- */ HRESULT PRIVATE OpenTheBriefcase( LPCTSTR pszDatPath, int atomPath, CBS * pcbs, HWND hwndOwner) { HRESULT hres; TWINRESULT tr; BOOL bRet = FALSE; DWORD dwFlags = OB_FL_OPEN_DATABASE | OB_FL_TRANSLATE_DB_FOLDER | OB_FL_ALLOW_UI; int nDrive; int nDriveType; // Determine if we want to record the existence of this briefcase. // We don't care about briefcases on remote or floppy drives. nDrive = PathGetDriveNumber(pszDatPath); // Record this briefcase? nDriveType = DriveType(nDrive); if (DRIVE_CDROM != nDriveType && DRIVE_REMOVABLE != nDriveType && DRIVE_RAMDRIVE != nDriveType && !PathIsUNC(pszDatPath) && !IsNetDrive(nDrive)) { // Yes SetFlag(dwFlags, OB_FL_LIST_DATABASE); TRACE_MSG(TF_GENERAL, TEXT("Remembering briefcase %s"), pszDatPath); } RETRY_BEGIN(FALSE) { tr = Sync_OpenBriefcase(pszDatPath, dwFlags, GetDesktopWindow(), &pcbs->hbrf); hres = HRESULT_FROM_TR(tr); // Unavailable disk? if (FAILED(hres)) { DWORD dwError = GetLastError(); if (ERROR_INVALID_DATA == dwError || ERROR_ACCESS_DENIED == dwError) { // Yes; ask user to retry/cancel int id = MsgBox(hwndOwner, MAKEINTRESOURCE(IDS_ERR_OPEN_UNAVAIL_VOL), MAKEINTRESOURCE(IDS_CAP_OPEN), NULL, MB_RETRYCANCEL | MB_ICONWARNING); // Set specific error value hres = E_TR_UNAVAILABLE_VOLUME; if (IDRETRY == id) { RETRY_SET(); // Try again } } } } RETRY_END() if (SUCCEEDED(hres)) { if (!Cache_AddItem(&g_cacheCBS, atomPath, (LPVOID)pcbs)) { Sync_CloseBriefcase(pcbs->hbrf); hres = ResultFromScode(E_OUTOFMEMORY); } } return hres; } /*---------------------------------------------------------- Purpose: This function handles the case when the engine fails to open the database because the database file is corrupt. Returns: standard hresult Cond: -- */ HRESULT PRIVATE HandleCorruptDatabase( CBS * pcbs, int atomPath, LPCTSTR pszDatPath, // Path of database file HWND hwndOwner) { TCHAR szTemplate[MAXPATHLEN]; TCHAR szNewFile[MAXPATHLEN]; LPTSTR pszNewPath = szTemplate; LPCTSTR pszPath = Atom_GetName(atomPath); LPTSTR psz; DWORD dwAttr; HRESULT hr = E_FAIL; static UINT rgids[2] = { IDS_ERR_1_CorruptDB, IDS_ERR_2_CorruptDB }; ASSERT(pszPath); // Create the new database name // SzFromIDS(IDS_BOGUSDBTEMPLATE, szTemplate, ARRAYSIZE(szTemplate)); if (PathMakeUniqueName(szNewFile, ARRAYSIZE(szNewFile), TEXT("badbc.dat"), szTemplate, pszPath)) { lstrcpyn(szTemplate, pszPath, ARRAYSIZE(szTemplate)); if (PathAppend(pszNewPath, szNewFile)) { // Move the database // MoveFile(pszDatPath, pszNewPath); // Unhide the corrupt database // dwAttr = GetFileAttributes(pszNewPath); if (dwAttr != 0xFFFFFFFF) { ClearFlag(dwAttr, FILE_ATTRIBUTE_HIDDEN); SetFileAttributes(pszNewPath, dwAttr); } if (FmtString(&psz, IDS_ERR_F_CorruptDB, rgids, ARRAYSIZE(rgids))) { MsgBox(hwndOwner, psz, MAKEINTRESOURCE(IDS_CAP_OPEN), NULL, MB_ERROR); GFree(psz); } DEBUG_CODE( TRACE_MSG(TF_GENERAL, TEXT("Renaming corrupt database to %s"), pszNewPath); ) // Retry opening... // hr = OpenTheBriefcase(pszDatPath, atomPath, pcbs, hwndOwner); } } return hr; } /*---------------------------------------------------------- Purpose: Add the atomPath to the cache. We open the briefcase database if it needs opening. If atomPath is already in the cache, simply return the pointer to the entry. Returns: standard hresult Cond: Must call CBS_Delete for every call to this function */ HRESULT PUBLIC CBS_Add( PCBS * ppcbs, int atomPath, HWND hwndOwner) { HRESULT hres = NOERROR; TCHAR szDatPath[MAXPATHLEN]; CBS * pcbs; CBS_EnterCS(); { pcbs = Cache_GetPtr(&g_cacheCBS, atomPath); if (NULL == pcbs) { // Allocate using commctrl's Alloc, so the structure will be in // shared heap space across processes. pcbs = SharedAllocType(CBS); if (NULL == pcbs) { hres = ResultFromScode(E_OUTOFMEMORY); } else { LPCTSTR pszPath = Atom_GetName(atomPath); LPCTSTR pszDBName; ASSERT(pszPath); pcbs->atomBrf = atomPath; pcbs->uFlags = 0; // Create an abort event object simply so we can programmatically // cancel a createreclist call in the worker thread. This // would happen if the user closed the briefcase during // CreateRecList. // (it is ok if this fails) AbortEvt_Create(&pcbs->pabortevt, AEF_SHARED); DEBUG_CODE( TRACE_MSG(TF_GENERAL, TEXT("Opening Briefcase %s..."), pszPath); ) if (IsLFNDrive(pszPath)) { pszDBName = g_szDBName; SetFlag(pcbs->uFlags, CBSF_LFNDRIVE); } else pszDBName = g_szDBNameShort; if (PathsTooLong(pszPath, pszDBName)) { MsgBox(hwndOwner, MAKEINTRESOURCE(IDS_ERR_OPEN_TOOLONG), MAKEINTRESOURCE(IDS_CAP_OPEN), NULL, MB_ERROR); hres = E_FAIL; } else { PathCombine(szDatPath, pszPath, pszDBName); hres = OpenTheBriefcase(szDatPath, atomPath, pcbs, hwndOwner); if (FAILED(hres)) { DEBUG_CODE( TRACE_MSG(TF_ERROR, TEXT("Open failed. Error is %s"), SzFromTR(GET_TR(hres))); ) SEMsgBox(hwndOwner, IDS_CAP_OPEN, hres, c_rgseOpenBriefcase, ARRAYSIZE(c_rgseOpenBriefcase)); // Is this a corrupt briefcase? if (E_TR_CORRUPT_BRIEFCASE == hres) { // Yes; try to create a new database hres = HandleCorruptDatabase(pcbs, atomPath, szDatPath, hwndOwner); } } } } } // Did something fail above? if (FAILED(hres)) { // Yes; cleanup if (pcbs) { if (pcbs->hbrf) Sync_CloseBriefcase(pcbs->hbrf); SharedFree(&pcbs); } } *ppcbs = pcbs; } CBS_LeaveCS(); return hres; }