1
0
mirror of https://github.com/veracrypt/VeraCrypt.git synced 2026-06-10 06:46:59 -05:00

Windows: allow cancelling long mount operations

Add a root-driver abort IOCTL that bypasses the mount control mutex and sets cooperative KDF abort flags for the active mount.

Restrict abort requests to privileged callers or to the user that initiated the pending mount, and retry early wait-dialog cancel requests until the driver has registered the cancellable mount context.

Wire the wait dialog Cancel button to send the abort request through a fresh driver handle, and propagate ERR_USER_ABORT through header/cache processing.

Add a /cancelmount command-line switch that sends the same abort request without displaying UI, so users can cancel hidden-wait-dialog mount operations from another process.
This commit is contained in:
Mounir IDRASSI
2026-06-07 17:14:37 +09:00
parent 105425ebb0
commit 1871765a76
13 changed files with 437 additions and 23 deletions
@@ -110,6 +110,10 @@ if it is followed by <strong>n </strong>or<strong> no</strong>: don't try to mou
If it is followed by <strong>n</strong> or <strong>no</strong>: force the display waiting dialog is displayed while performing operations.</td> If it is followed by <strong>n</strong> or <strong>no</strong>: force the display waiting dialog is displayed while performing operations.</td>
</tr> </tr>
<tr> <tr>
<td><em>/cancelmount</em></td>
<td>Cancels any currently running mount operation and exits without displaying a status message. The process returns exit code 0 when the cancel request is accepted by the driver, or 1 otherwise.</td>
</tr>
<tr>
<td><em>/secureDesktop</em></td> <td><em>/secureDesktop</em></td>
<td>If it is followed by <strong>y</strong> or <strong>yes</strong> or if no parameter is specified: display password dialog and token PIN dialog in a dedicated secure desktop to protect against certain types of attacks.<br> <td>If it is followed by <strong>y</strong> or <strong>yes</strong> or if no parameter is specified: display password dialog and token PIN dialog in a dedicated secure desktop to protect against certain types of attacks.<br>
If it is followed by <strong>n</strong> or <strong>no</strong>: the password dialog and token PIN dialog are displayed in the normal desktop.</td> If it is followed by <strong>n</strong> or <strong>no</strong>: the password dialog and token PIN dialog are displayed in the normal desktop.</td>
+8
View File
@@ -129,6 +129,8 @@
#define VC_IOCTL_ENCRYPTION_QUEUE_PARAMS TC_IOCTL (43) #define VC_IOCTL_ENCRYPTION_QUEUE_PARAMS TC_IOCTL (43)
#define TC_IOCTL_ABORT_MOUNT_VOLUME TC_IOCTL (44)
// Undocumented IOCTL sent by Windows 10 when handling EFS data on volumes // Undocumented IOCTL sent by Windows 10 when handling EFS data on volumes
#define IOCTL_UNKNOWN_WINDOWS10_EFS_ACCESS 0x455610D8 #define IOCTL_UNKNOWN_WINDOWS10_EFS_ACCESS 0x455610D8
@@ -180,6 +182,12 @@ typedef struct
BOOL VolumeMasterKeyVulnerable; BOOL VolumeMasterKeyVulnerable;
} MOUNT_STRUCT; } MOUNT_STRUCT;
typedef struct
{
int nDosDriveNo; /* Drive number whose pending mount should be aborted; -1 aborts any pending mount */
int nReturnCode; /* Return code back from driver */
} MOUNT_ABORT_STRUCT;
typedef struct typedef struct
{ {
int nDosDriveNo; /* Drive letter to unmount */ int nDosDriveNo; /* Drive letter to unmount */
+34 -3
View File
@@ -24,6 +24,22 @@ int CachedPim[CACHE_SIZE];
int cacheEmpty = 1; int cacheEmpty = 1;
static int nPasswordIdx = 0; static int nPasswordIdx = 0;
static BOOL IsUserAbortRequested (long volatile *pUserAbort)
{
return pUserAbort && *pUserAbort;
}
static BOOL ResetAbortKeyDerivation (long volatile *pAbortKeyDerivation, long volatile *pUserAbort)
{
if (IsUserAbortRequested (pUserAbort))
return FALSE;
if (pAbortKeyDerivation)
*pAbortKeyDerivation = 0;
return TRUE;
}
uint64 VcGetPasswordEncryptionID (Password* pPassword) uint64 VcGetPasswordEncryptionID (Password* pPassword)
{ {
return ((uint64) pPassword->Text) + ((uint64) pPassword); return ((uint64) pPassword->Text) + ((uint64) pPassword);
@@ -39,7 +55,7 @@ void VcUnprotectPassword (Password* pPassword, uint64 encID)
VcProtectPassword (pPassword, encID); VcProtectPassword (pPassword, encID);
} }
int ReadVolumeHeaderWCache (BOOL bBoot, BOOL bCache, BOOL bCachePim, unsigned char *header, Password *password, int pkcs5_prf, int pim, PCRYPTO_INFO *retInfo) int ReadVolumeHeaderWCacheWithAbort (BOOL bBoot, BOOL bCache, BOOL bCachePim, unsigned char *header, Password *password, int pkcs5_prf, int pim, PCRYPTO_INFO *retInfo, long volatile *pAbortKeyDerivation, long volatile *pUserAbort)
{ {
int nReturnCode = ERR_PASSWORD_WRONG; int nReturnCode = ERR_PASSWORD_WRONG;
int i, effectivePim; int i, effectivePim;
@@ -47,7 +63,10 @@ int ReadVolumeHeaderWCache (BOOL bBoot, BOOL bCache, BOOL bCachePim, unsigned ch
/* Attempt to recognize volume using mount password */ /* Attempt to recognize volume using mount password */
if (password->Length > 0) if (password->Length > 0)
{ {
nReturnCode = ReadVolumeHeader (bBoot, header, password, pkcs5_prf, pim, retInfo, NULL); if (!ResetAbortKeyDerivation (pAbortKeyDerivation, pUserAbort))
return ERR_USER_ABORT;
nReturnCode = ReadVolumeHeaderWithAbort (bBoot, header, password, pkcs5_prf, pim, retInfo, NULL, pAbortKeyDerivation, pUserAbort);
/* Save mount passwords back into cache if asked to do so */ /* Save mount passwords back into cache if asked to do so */
if (bCache && (nReturnCode == 0 || nReturnCode == ERR_CIPHER_INIT_WEAK_KEY)) if (bCache && (nReturnCode == 0 || nReturnCode == ERR_CIPHER_INIT_WEAK_KEY))
@@ -113,7 +132,14 @@ int ReadVolumeHeaderWCache (BOOL bBoot, BOOL bCache, BOOL bCachePim, unsigned ch
effectivePim = CachedPim[i]; effectivePim = CachedPim[i];
else else
effectivePim = pim; effectivePim = pim;
nReturnCode = ReadVolumeHeader (bBoot, header, pCurrentPassword, pkcs5_prf, effectivePim, retInfo, NULL);
if (!ResetAbortKeyDerivation (pAbortKeyDerivation, pUserAbort))
{
nReturnCode = ERR_USER_ABORT;
break;
}
nReturnCode = ReadVolumeHeaderWithAbort (bBoot, header, pCurrentPassword, pkcs5_prf, effectivePim, retInfo, NULL, pAbortKeyDerivation, pUserAbort);
if (nReturnCode != ERR_PASSWORD_WRONG) if (nReturnCode != ERR_PASSWORD_WRONG)
break; break;
@@ -128,6 +154,11 @@ int ReadVolumeHeaderWCache (BOOL bBoot, BOOL bCache, BOOL bCachePim, unsigned ch
return nReturnCode; return nReturnCode;
} }
int ReadVolumeHeaderWCache (BOOL bBoot, BOOL bCache, BOOL bCachePim, unsigned char *header, Password *password, int pkcs5_prf, int pim, PCRYPTO_INFO *retInfo)
{
return ReadVolumeHeaderWCacheWithAbort (bBoot, bCache, bCachePim, header, password, pkcs5_prf, pim, retInfo, NULL, NULL);
}
void AddPasswordToCache (Password *password, int pim, BOOL bCachePim) void AddPasswordToCache (Password *password, int pim, BOOL bCachePim)
{ {
+1
View File
@@ -23,4 +23,5 @@ extern int cacheEmpty;
void AddPasswordToCache (Password *password, int pim, BOOL bCachePim); void AddPasswordToCache (Password *password, int pim, BOOL bCachePim);
void AddLegacyPasswordToCache (__unaligned PasswordLegacy *password, int pim); void AddLegacyPasswordToCache (__unaligned PasswordLegacy *password, int pim);
int ReadVolumeHeaderWCache (BOOL bBoot, BOOL bCache, BOOL bCachePim, unsigned char *header, Password *password, int pkcs5_prf, int pim, PCRYPTO_INFO *retInfo); int ReadVolumeHeaderWCache (BOOL bBoot, BOOL bCache, BOOL bCachePim, unsigned char *header, Password *password, int pkcs5_prf, int pim, PCRYPTO_INFO *retInfo);
int ReadVolumeHeaderWCacheWithAbort (BOOL bBoot, BOOL bCache, BOOL bCachePim, unsigned char *header, Password *password, int pkcs5_prf, int pim, PCRYPTO_INFO *retInfo, long volatile *pAbortKeyDerivation, long volatile *pUserAbort);
void WipeCache (void); void WipeCache (void);
+3 -2
View File
@@ -338,8 +338,9 @@ STYLE DS_SETFONT | DS_MODALFRAME | DS_SETFOREGROUND | DS_FIXEDSYS | DS_CENTER |
CAPTION "VeraCrypt" CAPTION "VeraCrypt"
FONT 8, "MS Shell Dlg", 0, 0, 0x0 FONT 8, "MS Shell Dlg", 0, 0, 0x0
BEGIN BEGIN
CTEXT "Please wait...\nThis process may take a long time and VeraCrypt may seem unresponsive.",IDT_STATIC_MODAL_WAIT_DLG_INFO,9,11,274,33 CTEXT "Please wait...\nThis process may take a long time and VeraCrypt may seem unresponsive.",IDT_STATIC_MODAL_WAIT_DLG_INFO,9,7,274,30
CONTROL "",IDC_WAIT_PROGRESS_BAR,"msctls_progress32",WS_BORDER,7,49,278,14 CONTROL "",IDC_WAIT_PROGRESS_BAR,"msctls_progress32",WS_BORDER,7,41,278,12
PUSHBUTTON "Cancel",IDCANCEL,121,56,50,14
END END
IDD_TEXT_EDIT_DLG DIALOGEX 0, 0, 372, 220 IDD_TEXT_EDIT_DLG DIALOGEX 0, 0, 372, 220
+89 -3
View File
@@ -8880,12 +8880,18 @@ BOOL GetPhysicalDriveStorageInformation(UINT nDriveNumber, STORAGE_ACCESS_ALIGNM
// implementation of the generic wait dialog mechanism // implementation of the generic wait dialog mechanism
static UINT g_wmWaitDlg = ::RegisterWindowMessage(L"VeraCryptWaitDlgMessage"); static UINT g_wmWaitDlg = ::RegisterWindowMessage(L"VeraCryptWaitDlgMessage");
#define WAIT_DLG_CANCEL_RETRY_TIMER_ID 1
#define WAIT_DLG_CANCEL_RETRY_INTERVAL 250
typedef BOOL (CALLBACK* WaitCancelProc)(void* pArg, HWND hWaitDlg);
typedef struct typedef struct
{ {
HWND hwnd; HWND hwnd;
void* pArg; void* pArg;
WaitThreadProc callback; WaitThreadProc callback;
WaitCancelProc cancelCallback;
BOOL cancelRequested;
} WaitThreadParam; } WaitThreadParam;
static void _cdecl WaitThread (void* pParam) static void _cdecl WaitThread (void* pParam)
@@ -8899,6 +8905,22 @@ static void _cdecl WaitThread (void* pParam)
PostMessage (pThreadParam->hwnd, g_wmWaitDlg, 0, 0); PostMessage (pThreadParam->hwnd, g_wmWaitDlg, 0, 0);
} }
static BOOL WaitDlgTryCancel (HWND hwndDlg, WaitThreadParam* thParam)
{
if (thParam && thParam->cancelCallback)
{
if (thParam->cancelCallback (thParam->pArg, hwndDlg))
{
KillTimer (hwndDlg, WAIT_DLG_CANCEL_RETRY_TIMER_ID);
return TRUE;
}
SetTimer (hwndDlg, WAIT_DLG_CANCEL_RETRY_TIMER_ID, WAIT_DLG_CANCEL_RETRY_INTERVAL, NULL);
}
return FALSE;
}
BOOL CALLBACK WaitDlgProc (HWND hwndDlg, UINT msg, WPARAM wParam, LPARAM lParam) BOOL CALLBACK WaitDlgProc (HWND hwndDlg, UINT msg, WPARAM wParam, LPARAM lParam)
{ {
WORD lw = LOWORD (wParam); WORD lw = LOWORD (wParam);
@@ -8908,6 +8930,7 @@ BOOL CALLBACK WaitDlgProc (HWND hwndDlg, UINT msg, WPARAM wParam, LPARAM lParam)
case WM_INITDIALOG: case WM_INITDIALOG:
{ {
WaitThreadParam* thParam = (WaitThreadParam*) lParam; WaitThreadParam* thParam = (WaitThreadParam*) lParam;
SetWindowLongPtr (hwndDlg, DWLP_USER, (LONG_PTR) thParam);
// set the progress bar type to MARQUEE (indefinite progress) // set the progress bar type to MARQUEE (indefinite progress)
HWND hProgress = GetDlgItem (hwndDlg, IDC_WAIT_PROGRESS_BAR); HWND hProgress = GetDlgItem (hwndDlg, IDC_WAIT_PROGRESS_BAR);
@@ -8935,24 +8958,50 @@ BOOL CALLBACK WaitDlgProc (HWND hwndDlg, UINT msg, WPARAM wParam, LPARAM lParam)
} }
LocalizeDialog (hwndDlg, NULL); LocalizeDialog (hwndDlg, NULL);
if (!thParam->cancelCallback)
ShowWindow (GetDlgItem (hwndDlg, IDCANCEL), SW_HIDE);
_beginthread(WaitThread, 0, thParam); _beginthread(WaitThread, 0, thParam);
return 0; return 0;
} }
case WM_COMMAND: case WM_COMMAND:
if (lw == IDOK || lw == IDCANCEL) if (lw == IDCANCEL)
{
WaitThreadParam* thParam = (WaitThreadParam*) GetWindowLongPtr (hwndDlg, DWLP_USER);
if (thParam && thParam->cancelCallback && !thParam->cancelRequested)
{
thParam->cancelRequested = TRUE;
EnableWindow (GetDlgItem (hwndDlg, IDCANCEL), FALSE);
WaitDlgTryCancel (hwndDlg, thParam);
}
return 1;
}
if (lw == IDOK)
return 1; return 1;
else else
return 0; return 0;
case WM_TIMER:
if (wParam == WAIT_DLG_CANCEL_RETRY_TIMER_ID)
{
WaitThreadParam* thParam = (WaitThreadParam*) GetWindowLongPtr (hwndDlg, DWLP_USER);
if (thParam && thParam->cancelRequested)
WaitDlgTryCancel (hwndDlg, thParam);
return 1;
}
return 0;
case WM_DESTROY: case WM_DESTROY:
KillTimer (hwndDlg, WAIT_DLG_CANCEL_RETRY_TIMER_ID);
DetachProtectionFromCurrentThread(); DetachProtectionFromCurrentThread();
return 0; return 0;
default: default:
if (msg == g_wmWaitDlg) if (msg == g_wmWaitDlg)
{ {
KillTimer (hwndDlg, WAIT_DLG_CANCEL_RETRY_TIMER_ID);
EndDialog (hwndDlg, IDOK); EndDialog (hwndDlg, IDOK);
return 1; return 1;
} }
@@ -9015,11 +9064,13 @@ static LRESULT CALLBACK ShowWaitDialogParentWndProc (HWND hWnd, UINT message, WP
} }
void ShowWaitDialog(HWND hwnd, BOOL bUseHwndAsParent, WaitThreadProc callback, void* pArg) static void ShowWaitDialogEx(HWND hwnd, BOOL bUseHwndAsParent, WaitThreadProc callback, WaitCancelProc cancelCallback, void* pArg)
{ {
BOOL bEffectiveHideWaitingDialog = bCmdHideWaitingDialogValid? bCmdHideWaitingDialog : bHideWaitingDialog; BOOL bEffectiveHideWaitingDialog = bCmdHideWaitingDialogValid? bCmdHideWaitingDialog : bHideWaitingDialog;
WaitThreadParam threadParam; WaitThreadParam threadParam;
threadParam.callback = callback; threadParam.callback = callback;
threadParam.cancelCallback = cancelCallback;
threadParam.cancelRequested = FALSE;
threadParam.pArg = pArg; threadParam.pArg = pArg;
if (WaitDialogDisplaying || bEffectiveHideWaitingDialog) if (WaitDialogDisplaying || bEffectiveHideWaitingDialog)
@@ -9082,6 +9133,11 @@ void ShowWaitDialog(HWND hwnd, BOOL bUseHwndAsParent, WaitThreadProc callback, v
} }
} }
void ShowWaitDialog(HWND hwnd, BOOL bUseHwndAsParent, WaitThreadProc callback, void* pArg)
{
ShowWaitDialogEx (hwnd, bUseHwndAsParent, callback, NULL, pArg);
}
#ifndef SETUP #ifndef SETUP
/************************************************************************/ /************************************************************************/
@@ -9108,6 +9164,26 @@ static BOOL PerformMountIoctl (MOUNT_STRUCT* pmount, LPDWORD pdwResult, BOOL use
sizeof (MOUNT_STRUCT), pmount, sizeof (MOUNT_STRUCT), pdwResult, NULL); sizeof (MOUNT_STRUCT), pmount, sizeof (MOUNT_STRUCT), pdwResult, NULL);
} }
BOOL AbortMountOperation (int nDosDriveNo)
{
BOOL bResult;
DWORD dwResult;
HANDLE hAbortDriver = CreateFile (WIN32_ROOT_PREFIX, 0, FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL);
MOUNT_ABORT_STRUCT abortMount;
if (hAbortDriver == INVALID_HANDLE_VALUE)
return FALSE;
memset (&abortMount, 0, sizeof (abortMount));
abortMount.nDosDriveNo = nDosDriveNo;
bResult = DeviceIoControl (hAbortDriver, TC_IOCTL_ABORT_MOUNT_VOLUME, &abortMount,
sizeof (abortMount), &abortMount, sizeof (abortMount), &dwResult, NULL);
CloseHandle (hAbortDriver);
return bResult && abortMount.nReturnCode == ERR_USER_ABORT;
}
// specific definitions and implementation for support of mount operation // specific definitions and implementation for support of mount operation
// in wait dialog mechanism // in wait dialog mechanism
@@ -9130,6 +9206,13 @@ void CALLBACK MountWaitThreadProc(void* pArg, HWND )
pThreadParam->dwLastError = GetLastError (); pThreadParam->dwLastError = GetLastError ();
} }
BOOL CALLBACK MountWaitCancelProc(void* pArg, HWND )
{
MountThreadParam* pThreadParam = (MountThreadParam*) pArg;
return AbortMountOperation (pThreadParam->pmount->nDosDriveNo);
}
/************************************************************************/ /************************************************************************/
// Use only cached passwords if password = NULL // Use only cached passwords if password = NULL
@@ -9366,7 +9449,7 @@ retry:
mountThreadParam.pdwResult = &dwResult; mountThreadParam.pdwResult = &dwResult;
mountThreadParam.dwLastError = ERROR_SUCCESS; mountThreadParam.dwLastError = ERROR_SUCCESS;
ShowWaitDialog (hwndDlg, FALSE, MountWaitThreadProc, &mountThreadParam); ShowWaitDialogEx (hwndDlg, FALSE, MountWaitThreadProc, MountWaitCancelProc, &mountThreadParam);
dwLastError = mountThreadParam.dwLastError; dwLastError = mountThreadParam.dwLastError;
} }
@@ -9428,6 +9511,9 @@ retry:
if (mount.nReturnCode != 0) if (mount.nReturnCode != 0)
{ {
if (mount.nReturnCode == ERR_USER_ABORT)
return -1;
if (mount.nReturnCode == ERR_PASSWORD_WRONG) if (mount.nReturnCode == ERR_PASSWORD_WRONG)
{ {
// Do not report wrong password, if not instructed to // Do not report wrong password, if not instructed to
+1
View File
@@ -412,6 +412,7 @@ BOOL IsDriveAvailable (int driveNo);
BOOL IsDeviceMounted (wchar_t *deviceName); BOOL IsDeviceMounted (wchar_t *deviceName);
int DriverUnmountVolume (HWND hwndDlg, int nDosDriveNo, BOOL forced); int DriverUnmountVolume (HWND hwndDlg, int nDosDriveNo, BOOL forced);
void BroadcastDeviceChange (WPARAM message, int nDosDriveNo, DWORD driveMap); void BroadcastDeviceChange (WPARAM message, int nDosDriveNo, DWORD driveMap);
BOOL AbortMountOperation (int nDosDriveNo);
int MountVolume (HWND hwndDlg, int driveNo, wchar_t *volumePath, Password *password, int pkcs5, int pim, BOOL cachePassword, BOOL cachePim, BOOL sharedAccess, const MountOptions* const mountOptions, BOOL quiet, BOOL bReportWrongPassword); int MountVolume (HWND hwndDlg, int driveNo, wchar_t *volumePath, Password *password, int pkcs5, int pim, BOOL cachePassword, BOOL cachePim, BOOL sharedAccess, const MountOptions* const mountOptions, BOOL quiet, BOOL bReportWrongPassword);
BOOL UnmountVolume (HWND hwndDlg , int nDosDriveNo, BOOL forceUnmount); BOOL UnmountVolume (HWND hwndDlg , int nDosDriveNo, BOOL forceUnmount);
BOOL UnmountVolumeAfterFormatExCall (HWND hwndDlg, int nDosDriveNo); BOOL UnmountVolumeAfterFormatExCall (HWND hwndDlg, int nDosDriveNo);
+37 -10
View File
@@ -186,7 +186,7 @@ static int MapArgon2ResultToVcError (int result)
BOOL ReadVolumeHeaderRecoveryMode = FALSE; BOOL ReadVolumeHeaderRecoveryMode = FALSE;
int ReadVolumeHeader (BOOL bBoot, unsigned char *encryptedHeader, Password *password, int selected_pkcs5_prf, int pim, PCRYPTO_INFO *retInfo, CRYPTO_INFO *retHeaderCryptoInfo) int ReadVolumeHeaderWithAbort (BOOL bBoot, unsigned char *encryptedHeader, Password *password, int selected_pkcs5_prf, int pim, PCRYPTO_INFO *retInfo, CRYPTO_INFO *retHeaderCryptoInfo, long volatile *pAbortKeyDerivation, long volatile *pUserAbort)
{ {
unsigned char header[TC_VOLUME_HEADER_EFFECTIVE_SIZE]; unsigned char header[TC_VOLUME_HEADER_EFFECTIVE_SIZE];
unsigned char* keyInfoBuffer = NULL; unsigned char* keyInfoBuffer = NULL;
@@ -203,6 +203,7 @@ int ReadVolumeHeader (BOOL bBoot, unsigned char *encryptedHeader, Password *pass
int iterationsCount = 0; int iterationsCount = 0;
int memoryCost = 0; int memoryCost = 0;
LONG volatile abortKeyDerivation = 0; LONG volatile abortKeyDerivation = 0;
LONG volatile *effectiveAbortKeyDerivation = pAbortKeyDerivation ? (LONG volatile *) pAbortKeyDerivation : &abortKeyDerivation;
#ifndef VC_DCS_DISABLE_ARGON2 #ifndef VC_DCS_DISABLE_ARGON2
int lastArgon2DerivationResult = 0; int lastArgon2DerivationResult = 0;
#endif #endif
@@ -326,6 +327,12 @@ int ReadVolumeHeader (BOOL bBoot, unsigned char *encryptedHeader, Password *pass
// Test all available PKCS5 PRFs // Test all available PKCS5 PRFs
for (enqPkcs5Prf = FIRST_PRF_ID; enqPkcs5Prf <= LAST_PRF_ID || queuedWorkItems > 0; ++enqPkcs5Prf) for (enqPkcs5Prf = FIRST_PRF_ID; enqPkcs5Prf <= LAST_PRF_ID || queuedWorkItems > 0; ++enqPkcs5Prf)
{ {
if (pUserAbort && *pUserAbort)
{
status = ERR_USER_ABORT;
goto err;
}
// if a PRF is specified, we skip all other PRFs // if a PRF is specified, we skip all other PRFs
if (selected_pkcs5_prf != 0 && enqPkcs5Prf != selected_pkcs5_prf) if (selected_pkcs5_prf != 0 && enqPkcs5Prf != selected_pkcs5_prf)
continue; continue;
@@ -355,7 +362,7 @@ int ReadVolumeHeader (BOOL bBoot, unsigned char *encryptedHeader, Password *pass
iterationsCount = get_pkcs5_iteration_count (enqPkcs5Prf, pim, bBoot, &memoryCost); iterationsCount = get_pkcs5_iteration_count (enqPkcs5Prf, pim, bBoot, &memoryCost);
EncryptionThreadPoolBeginKeyDerivation (keyDerivationCompletedEvent, noOutstandingWorkItemEvent, EncryptionThreadPoolBeginKeyDerivation (keyDerivationCompletedEvent, noOutstandingWorkItemEvent,
&item->KeyReady, outstandingWorkItemCount, enqPkcs5Prf, keyInfo->userKey, &item->KeyReady, outstandingWorkItemCount, enqPkcs5Prf, keyInfo->userKey,
keyInfo->keyLength, keyInfo->salt, iterationsCount, memoryCost, item->DerivedKey, &item->DerivationResult, &abortKeyDerivation); keyInfo->keyLength, keyInfo->salt, iterationsCount, memoryCost, item->DerivedKey, &item->DerivationResult, effectiveAbortKeyDerivation);
++queuedWorkItems; ++queuedWorkItems;
break; break;
@@ -376,6 +383,12 @@ int ReadVolumeHeader (BOOL bBoot, unsigned char *encryptedHeader, Password *pass
item = &keyDerivationWorkItems[i]; item = &keyDerivationWorkItems[i];
if (!item->Free && InterlockedExchangeAdd (&item->KeyReady, 0) == TRUE) if (!item->Free && InterlockedExchangeAdd (&item->KeyReady, 0) == TRUE)
{ {
if (pUserAbort && *pUserAbort)
{
status = ERR_USER_ABORT;
goto err;
}
LONG derivationResult = InterlockedExchangeAdd (&item->DerivationResult, 0); LONG derivationResult = InterlockedExchangeAdd (&item->DerivationResult, 0);
if (derivationResult != 0) if (derivationResult != 0)
{ {
@@ -418,29 +431,29 @@ KeyReady: ;
{ {
case SHA512: case SHA512:
derive_key_sha512 (keyInfo->userKey, keyInfo->keyLength, keyInfo->salt, derive_key_sha512 (keyInfo->userKey, keyInfo->keyLength, keyInfo->salt,
PKCS5_SALT_SIZE, keyInfo->noIterations, dk, GetMaxPkcs5OutSize(), &abortKeyDerivation); PKCS5_SALT_SIZE, keyInfo->noIterations, dk, GetMaxPkcs5OutSize(), effectiveAbortKeyDerivation);
break; break;
case SHA256: case SHA256:
derive_key_sha256 (keyInfo->userKey, keyInfo->keyLength, keyInfo->salt, derive_key_sha256 (keyInfo->userKey, keyInfo->keyLength, keyInfo->salt,
PKCS5_SALT_SIZE, keyInfo->noIterations, dk, GetMaxPkcs5OutSize(), &abortKeyDerivation); PKCS5_SALT_SIZE, keyInfo->noIterations, dk, GetMaxPkcs5OutSize(), effectiveAbortKeyDerivation);
break; break;
#ifndef WOLFCRYPT_BACKEND #ifndef WOLFCRYPT_BACKEND
case BLAKE2S: case BLAKE2S:
derive_key_blake2s (keyInfo->userKey, keyInfo->keyLength, keyInfo->salt, derive_key_blake2s (keyInfo->userKey, keyInfo->keyLength, keyInfo->salt,
PKCS5_SALT_SIZE, keyInfo->noIterations, dk, GetMaxPkcs5OutSize(), &abortKeyDerivation); PKCS5_SALT_SIZE, keyInfo->noIterations, dk, GetMaxPkcs5OutSize(), effectiveAbortKeyDerivation);
break; break;
case WHIRLPOOL: case WHIRLPOOL:
derive_key_whirlpool (keyInfo->userKey, keyInfo->keyLength, keyInfo->salt, derive_key_whirlpool (keyInfo->userKey, keyInfo->keyLength, keyInfo->salt,
PKCS5_SALT_SIZE, keyInfo->noIterations, dk, GetMaxPkcs5OutSize(), &abortKeyDerivation); PKCS5_SALT_SIZE, keyInfo->noIterations, dk, GetMaxPkcs5OutSize(), effectiveAbortKeyDerivation);
break; break;
case STREEBOG: case STREEBOG:
derive_key_streebog(keyInfo->userKey, keyInfo->keyLength, keyInfo->salt, derive_key_streebog(keyInfo->userKey, keyInfo->keyLength, keyInfo->salt,
PKCS5_SALT_SIZE, keyInfo->noIterations, dk, GetMaxPkcs5OutSize(), &abortKeyDerivation); PKCS5_SALT_SIZE, keyInfo->noIterations, dk, GetMaxPkcs5OutSize(), effectiveAbortKeyDerivation);
break; break;
@@ -448,7 +461,7 @@ KeyReady: ;
case ARGON2: case ARGON2:
{ {
int derivationResult = derive_key_argon2(keyInfo->userKey, keyInfo->keyLength, keyInfo->salt, int derivationResult = derive_key_argon2(keyInfo->userKey, keyInfo->keyLength, keyInfo->salt,
PKCS5_SALT_SIZE, keyInfo->noIterations, keyInfo->memoryCost, dk, ARGON2_HEADER_KEYDATA_SIZE, &abortKeyDerivation); PKCS5_SALT_SIZE, keyInfo->noIterations, keyInfo->memoryCost, dk, ARGON2_HEADER_KEYDATA_SIZE, effectiveAbortKeyDerivation);
if (derivationResult != 0) if (derivationResult != 0)
{ {
if (selected_pkcs5_prf == 0) if (selected_pkcs5_prf == 0)
@@ -470,6 +483,12 @@ KeyReady: ;
} }
} }
if (pUserAbort && *pUserAbort)
{
status = ERR_USER_ABORT;
goto err;
}
// Test all available modes of operation // Test all available modes of operation
for (cryptoInfo->mode = FIRST_MODE_OF_OPERATION_ID; for (cryptoInfo->mode = FIRST_MODE_OF_OPERATION_ID;
cryptoInfo->mode <= LAST_MODE_OF_OPERATION; cryptoInfo->mode <= LAST_MODE_OF_OPERATION;
@@ -677,7 +696,7 @@ KeyReady: ;
if ((selected_pkcs5_prf == 0) && (encryptionThreadCount > 1)) if ((selected_pkcs5_prf == 0) && (encryptionThreadCount > 1))
{ {
// Signal other threads to stop // Signal other threads to stop
InterlockedExchange(&abortKeyDerivation, 1); InterlockedExchange(effectiveAbortKeyDerivation, 1);
} }
#endif #endif
goto ret; goto ret;
@@ -689,12 +708,15 @@ KeyReady: ;
status = MapArgon2ResultToVcError (lastArgon2DerivationResult); status = MapArgon2ResultToVcError (lastArgon2DerivationResult);
else else
#endif #endif
if (pUserAbort && *pUserAbort)
status = ERR_USER_ABORT;
else
status = ERR_PASSWORD_WRONG; status = ERR_PASSWORD_WRONG;
err: err:
#if !defined(_UEFI) #if !defined(_UEFI)
// Signal threads to stop // Signal threads to stop
InterlockedExchange(&abortKeyDerivation, 1); InterlockedExchange(effectiveAbortKeyDerivation, 1);
#endif #endif
if (cryptoInfo != retHeaderCryptoInfo) if (cryptoInfo != retHeaderCryptoInfo)
{ {
@@ -744,6 +766,11 @@ ret:
return status; return status;
} }
int ReadVolumeHeader (BOOL bBoot, unsigned char *encryptedHeader, Password *password, int selected_pkcs5_prf, int pim, PCRYPTO_INFO *retInfo, CRYPTO_INFO *retHeaderCryptoInfo)
{
return ReadVolumeHeaderWithAbort (bBoot, encryptedHeader, password, selected_pkcs5_prf, pim, retInfo, retHeaderCryptoInfo, NULL, NULL);
}
#if defined(_WIN32) && !defined(_UEFI) #if defined(_WIN32) && !defined(_UEFI)
void ComputeBootloaderFingerprint (uint8 *bootLoaderBuf, unsigned int bootLoaderSize, uint8* fingerprint) void ComputeBootloaderFingerprint (uint8 *bootLoaderBuf, unsigned int bootLoaderSize, uint8* fingerprint)
{ {
+1
View File
@@ -152,6 +152,7 @@ int CreateVolumeHeaderInMemory(BOOL bBoot, unsigned char *encryptedHeader, int e
BOOL RandgetBytes(unsigned char *buf, int len, BOOL forceSlowPoll); BOOL RandgetBytes(unsigned char *buf, int len, BOOL forceSlowPoll);
#else #else
int ReadVolumeHeader (BOOL bBoot, unsigned char *encryptedHeader, Password *password, int pkcs5_prf, int pim, PCRYPTO_INFO *retInfo, CRYPTO_INFO *retHeaderCryptoInfo); int ReadVolumeHeader (BOOL bBoot, unsigned char *encryptedHeader, Password *password, int pkcs5_prf, int pim, PCRYPTO_INFO *retInfo, CRYPTO_INFO *retHeaderCryptoInfo);
int ReadVolumeHeaderWithAbort (BOOL bBoot, unsigned char *encryptedHeader, Password *password, int pkcs5_prf, int pim, PCRYPTO_INFO *retInfo, CRYPTO_INFO *retHeaderCryptoInfo, long volatile *pAbortKeyDerivation, long volatile *pUserAbort);
#if defined(_WIN32) && !defined(_UEFI) #if defined(_WIN32) && !defined(_UEFI)
void ComputeBootloaderFingerprint (uint8 *bootLoaderBuf, unsigned int bootLoaderSize, uint8* fingerprint); void ComputeBootloaderFingerprint (uint8 *bootLoaderBuf, unsigned int bootLoaderSize, uint8* fingerprint);
#endif #endif
+201
View File
@@ -122,6 +122,8 @@
PDRIVER_OBJECT TCDriverObject; PDRIVER_OBJECT TCDriverObject;
PDEVICE_OBJECT RootDeviceObject = NULL; PDEVICE_OBJECT RootDeviceObject = NULL;
static KMUTEX RootDeviceControlMutex; static KMUTEX RootDeviceControlMutex;
static KMUTEX MountCancelContextMutex;
static MOUNT_CANCEL_CONTEXT ActiveMountCancelContext;
BOOL DriverShuttingDown = FALSE; BOOL DriverShuttingDown = FALSE;
BOOL SelfTestsPassed; BOOL SelfTestsPassed;
int LastUniqueVolumeId; int LastUniqueVolumeId;
@@ -570,6 +572,9 @@ NTSTATUS TCDispatchQueueIRP (PDEVICE_OBJECT DeviceObject, PIRP Irp)
{ {
if (irpSp->MajorFunction == IRP_MJ_DEVICE_CONTROL) if (irpSp->MajorFunction == IRP_MJ_DEVICE_CONTROL)
{ {
if (irpSp->Parameters.DeviceIoControl.IoControlCode == TC_IOCTL_ABORT_MOUNT_VOLUME)
return ProcessMainDeviceControlIrp (DeviceObject, Extension, Irp);
NTSTATUS status = KeWaitForMutexObject (&RootDeviceControlMutex, Executive, KernelMode, FALSE, NULL); NTSTATUS status = KeWaitForMutexObject (&RootDeviceControlMutex, Executive, KernelMode, FALSE, NULL);
if (!NT_SUCCESS (status)) if (!NT_SUCCESS (status))
return status; return status;
@@ -694,6 +699,7 @@ NTSTATUS TCCreateRootDeviceObject (PDRIVER_OBJECT DriverObject)
*bRootExtension = TRUE; *bRootExtension = TRUE;
KeInitializeMutex (&RootDeviceControlMutex, 0); KeInitializeMutex (&RootDeviceControlMutex, 0);
KeInitializeMutex (&MountCancelContextMutex, 0);
ntStatus = IoCreateSymbolicLink (&Win32NameString, &ntUnicodeString); ntStatus = IoCreateSymbolicLink (&Win32NameString, &ntUnicodeString);
@@ -800,6 +806,177 @@ void RootDeviceControlMutexRelease ()
KeReleaseMutex (&RootDeviceControlMutex, FALSE); KeReleaseMutex (&RootDeviceControlMutex, FALSE);
} }
static void RegisterMountCancelContext (PEXTENSION Extension, int nDosDriveNo)
{
if (!NT_SUCCESS (KeWaitForMutexObject (&MountCancelContextMutex, Executive, KernelMode, FALSE, NULL)))
return;
ActiveMountCancelContext.nDosDriveNo = nDosDriveNo;
ActiveMountCancelContext.UserSidLength = 0;
InterlockedExchange (&ActiveMountCancelContext.UserSidValid, 0);
InterlockedExchange (&ActiveMountCancelContext.UserAbortRequested, 0);
InterlockedExchange (&ActiveMountCancelContext.KeyDerivationAbort, 0);
InterlockedIncrement (&ActiveMountCancelContext.SequenceNumber);
Extension->MountCancelContext = &ActiveMountCancelContext;
InterlockedExchange (&ActiveMountCancelContext.Active, 1);
KeReleaseMutex (&MountCancelContextMutex, FALSE);
}
static void SetMountCancelContextUserSid (PEXTENSION Extension, PSID userSid)
{
ULONG sidLength;
if (!userSid || Extension->MountCancelContext != &ActiveMountCancelContext)
return;
sidLength = RtlLengthSid (userSid);
if (sidLength > sizeof (ActiveMountCancelContext.UserSid))
return;
if (!NT_SUCCESS (KeWaitForMutexObject (&MountCancelContextMutex, Executive, KernelMode, FALSE, NULL)))
return;
if (Extension->MountCancelContext == &ActiveMountCancelContext
&& InterlockedExchangeAdd (&ActiveMountCancelContext.Active, 0) != 0
&& NT_SUCCESS (RtlCopySid (sizeof (ActiveMountCancelContext.UserSid), ActiveMountCancelContext.UserSid, userSid)))
{
ActiveMountCancelContext.UserSidLength = sidLength;
InterlockedExchange (&ActiveMountCancelContext.UserSidValid, 1);
}
KeReleaseMutex (&MountCancelContextMutex, FALSE);
}
static void UnregisterMountCancelContext (PEXTENSION Extension)
{
if (!NT_SUCCESS (KeWaitForMutexObject (&MountCancelContextMutex, Executive, KernelMode, FALSE, NULL)))
return;
if (Extension->MountCancelContext == &ActiveMountCancelContext)
{
InterlockedExchange (&ActiveMountCancelContext.Active, 0);
InterlockedExchange (&ActiveMountCancelContext.KeyDerivationAbort, 1);
InterlockedExchange (&ActiveMountCancelContext.UserSidValid, 0);
ActiveMountCancelContext.UserSidLength = 0;
Extension->MountCancelContext = NULL;
}
KeReleaseMutex (&MountCancelContextMutex, FALSE);
}
static BOOL CurrentUserMatchesSid (PSID userSid)
{
SECURITY_SUBJECT_CONTEXT subContext;
PACCESS_TOKEN accessToken;
PTOKEN_USER tokenUser;
BOOL result = FALSE;
if (!userSid)
return FALSE;
SeCaptureSubjectContext (&subContext);
SeLockSubjectContext(&subContext);
if (subContext.ClientToken && subContext.ImpersonationLevel >= SecurityImpersonation)
accessToken = subContext.ClientToken;
else
accessToken = subContext.PrimaryToken;
if (!accessToken)
goto ret;
if (SeTokenIsAdmin (accessToken))
{
result = TRUE;
goto ret;
}
if (!NT_SUCCESS (SeQueryInformationToken (accessToken, TokenUser, &tokenUser)))
goto ret;
result = RtlEqualSid (userSid, tokenUser->User.Sid);
ExFreePool (tokenUser); // Documented in newer versions of WDK
ret:
SeUnlockSubjectContext(&subContext);
SeReleaseSubjectContext (&subContext);
return result;
}
static BOOL CurrentUserCanAbortPendingMount (PSID userSid, BOOL userSidValid)
{
if (IoIsSystemThread (PsGetCurrentThread()))
return TRUE;
if (UserCanAccessDriveDevice())
return TRUE;
return userSidValid && CurrentUserMatchesSid (userSid);
}
static NTSTATUS AbortPendingMount (MOUNT_ABORT_STRUCT *abortRequest)
{
UCHAR userSid[SECURITY_MAX_SID_SIZE];
ULONG userSidLength = 0;
LONG sequenceNumber = 0;
BOOL userSidValid = FALSE;
BOOL activeMountMatches = FALSE;
if (!NT_SUCCESS (KeWaitForMutexObject (&MountCancelContextMutex, Executive, KernelMode, FALSE, NULL)))
return STATUS_UNSUCCESSFUL;
if (InterlockedExchangeAdd (&ActiveMountCancelContext.Active, 0) != 0
&& (abortRequest->nDosDriveNo < 0 || abortRequest->nDosDriveNo == ActiveMountCancelContext.nDosDriveNo))
{
activeMountMatches = TRUE;
sequenceNumber = InterlockedExchangeAdd (&ActiveMountCancelContext.SequenceNumber, 0);
userSidValid = InterlockedExchangeAdd (&ActiveMountCancelContext.UserSidValid, 0) != 0;
userSidLength = ActiveMountCancelContext.UserSidLength;
if (userSidValid && userSidLength <= sizeof (userSid))
memcpy (userSid, ActiveMountCancelContext.UserSid, userSidLength);
else
userSidValid = FALSE;
}
KeReleaseMutex (&MountCancelContextMutex, FALSE);
if (!activeMountMatches)
{
abortRequest->nReturnCode = ERR_DRIVE_NOT_FOUND;
return STATUS_SUCCESS;
}
if (!CurrentUserCanAbortPendingMount (userSidValid ? (PSID) userSid : NULL, userSidValid))
{
abortRequest->nReturnCode = ERR_ACCESS_DENIED;
return STATUS_ACCESS_DENIED;
}
if (!NT_SUCCESS (KeWaitForMutexObject (&MountCancelContextMutex, Executive, KernelMode, FALSE, NULL)))
return STATUS_UNSUCCESSFUL;
if (InterlockedExchangeAdd (&ActiveMountCancelContext.Active, 0) != 0
&& sequenceNumber == InterlockedExchangeAdd (&ActiveMountCancelContext.SequenceNumber, 0)
&& (abortRequest->nDosDriveNo < 0 || abortRequest->nDosDriveNo == ActiveMountCancelContext.nDosDriveNo))
{
InterlockedExchange (&ActiveMountCancelContext.UserAbortRequested, 1);
InterlockedExchange (&ActiveMountCancelContext.KeyDerivationAbort, 1);
abortRequest->nReturnCode = ERR_USER_ABORT;
KeReleaseMutex (&MountCancelContextMutex, FALSE);
return STATUS_SUCCESS;
}
abortRequest->nReturnCode = ERR_DRIVE_NOT_FOUND;
KeReleaseMutex (&MountCancelContextMutex, FALSE);
return STATUS_SUCCESS;
}
/* /*
IOCTL_STORAGE_GET_DEVICE_NUMBER 0x002D1080 IOCTL_STORAGE_GET_DEVICE_NUMBER 0x002D1080
IOCTL_STORAGE_GET_HOTPLUG_INFO 0x002D0C14 IOCTL_STORAGE_GET_HOTPLUG_INFO 0x002D0C14
@@ -2605,6 +2782,23 @@ NTSTATUS ProcessMainDeviceControlIrp (PDEVICE_OBJECT DeviceObject, PEXTENSION Ex
} }
break; break;
case TC_IOCTL_ABORT_MOUNT_VOLUME:
if (ValidateIOBufferSize (Irp, sizeof (MOUNT_ABORT_STRUCT), ValidateInputOutput))
{
MOUNT_ABORT_STRUCT *abortRequest = (MOUNT_ABORT_STRUCT *) Irp->AssociatedIrp.SystemBuffer;
if (irpSp->Parameters.DeviceIoControl.InputBufferLength != sizeof (MOUNT_ABORT_STRUCT))
{
Irp->IoStatus.Status = STATUS_INVALID_PARAMETER;
Irp->IoStatus.Information = 0;
break;
}
Irp->IoStatus.Status = AbortPendingMount (abortRequest);
Irp->IoStatus.Information = sizeof (MOUNT_ABORT_STRUCT);
}
break;
case TC_IOCTL_MOUNT_VOLUME: case TC_IOCTL_MOUNT_VOLUME:
if (ValidateIOBufferSize (Irp, sizeof (MOUNT_STRUCT), ValidateInputOutput)) if (ValidateIOBufferSize (Irp, sizeof (MOUNT_STRUCT), ValidateInputOutput))
{ {
@@ -3216,6 +3410,7 @@ LPWSTR TCTranslateCode (ULONG ulCode)
{ {
#define TC_CASE_RET_NAME(CODE) case CODE : return L###CODE #define TC_CASE_RET_NAME(CODE) case CODE : return L###CODE
TC_CASE_RET_NAME (TC_IOCTL_ABORT_MOUNT_VOLUME);
TC_CASE_RET_NAME (TC_IOCTL_ABORT_BOOT_ENCRYPTION_SETUP); TC_CASE_RET_NAME (TC_IOCTL_ABORT_BOOT_ENCRYPTION_SETUP);
TC_CASE_RET_NAME (TC_IOCTL_ABORT_DECOY_SYSTEM_WIPE); TC_CASE_RET_NAME (TC_IOCTL_ABORT_DECOY_SYSTEM_WIPE);
TC_CASE_RET_NAME (TC_IOCTL_BOOT_ENCRYPTION_SETUP); TC_CASE_RET_NAME (TC_IOCTL_BOOT_ENCRYPTION_SETUP);
@@ -4013,6 +4208,8 @@ NTSTATUS MountDevice (PDEVICE_OBJECT DeviceObject, MOUNT_STRUCT *mount)
SECURITY_SUBJECT_CONTEXT subContext; SECURITY_SUBJECT_CONTEXT subContext;
PACCESS_TOKEN accessToken; PACCESS_TOKEN accessToken;
RegisterMountCancelContext (NewExtension, mount->nDosDriveNo);
SeCaptureSubjectContext (&subContext); SeCaptureSubjectContext (&subContext);
SeLockSubjectContext(&subContext); SeLockSubjectContext(&subContext);
if (subContext.ClientToken && subContext.ImpersonationLevel >= SecurityImpersonation) if (subContext.ClientToken && subContext.ImpersonationLevel >= SecurityImpersonation)
@@ -4033,6 +4230,8 @@ NTSTATUS MountDevice (PDEVICE_OBJECT DeviceObject, MOUNT_STRUCT *mount)
{ {
ULONG sidLength = RtlLengthSid (tokenUser->User.Sid); ULONG sidLength = RtlLengthSid (tokenUser->User.Sid);
SetMountCancelContextUserSid (NewExtension, tokenUser->User.Sid);
NewExtension->UserSid = TCalloc (sidLength); NewExtension->UserSid = TCalloc (sidLength);
if (!NewExtension->UserSid) if (!NewExtension->UserSid)
ntStatus = STATUS_INSUFFICIENT_RESOURCES; ntStatus = STATUS_INSUFFICIENT_RESOURCES;
@@ -4049,6 +4248,8 @@ NTSTATUS MountDevice (PDEVICE_OBJECT DeviceObject, MOUNT_STRUCT *mount)
if (NT_SUCCESS (ntStatus)) if (NT_SUCCESS (ntStatus))
ntStatus = TCStartVolumeThread (NewDeviceObject, NewExtension, mount); ntStatus = TCStartVolumeThread (NewDeviceObject, NewExtension, mount);
UnregisterMountCancelContext (NewExtension);
if (!NT_SUCCESS (ntStatus)) if (!NT_SUCCESS (ntStatus))
{ {
Dump ("Mount FAILURE NT ERROR, ntStatus = 0x%08x\n", ntStatus); Dump ("Mount FAILURE NT ERROR, ntStatus = 0x%08x\n", ntStatus);
+17
View File
@@ -26,6 +26,22 @@ typedef struct _THREAD_BLOCK_
MOUNT_STRUCT *mount; MOUNT_STRUCT *mount;
} THREAD_BLOCK, *PTHREAD_BLOCK; } THREAD_BLOCK, *PTHREAD_BLOCK;
#ifndef SECURITY_MAX_SID_SIZE
#define SECURITY_MAX_SID_SIZE 68
#endif
typedef struct _MOUNT_CANCEL_CONTEXT_
{
LONG Active;
LONG UserAbortRequested;
LONG KeyDerivationAbort;
LONG UserSidValid;
LONG SequenceNumber;
int nDosDriveNo;
ULONG UserSidLength;
UCHAR UserSid[SECURITY_MAX_SID_SIZE];
} MOUNT_CANCEL_CONTEXT, *PMOUNT_CANCEL_CONTEXT;
/* This structure is allocated for non-root devices! WARNING: bRootDevice /* This structure is allocated for non-root devices! WARNING: bRootDevice
must be the first member of the structure! */ must be the first member of the structure! */
@@ -43,6 +59,7 @@ typedef struct EXTENSION
BOOL bThreadShouldQuit; /* Instruct per device worker thread to quit */ BOOL bThreadShouldQuit; /* Instruct per device worker thread to quit */
PETHREAD peThread; /* Thread handle */ PETHREAD peThread; /* Thread handle */
KEVENT keCreateEvent; /* Device creation event */ KEVENT keCreateEvent; /* Device creation event */
PMOUNT_CANCEL_CONTEXT MountCancelContext;
KSPIN_LOCK ListSpinLock; /* IRP spinlock */ KSPIN_LOCK ListSpinLock; /* IRP spinlock */
LIST_ENTRY ListEntry; /* IRP listentry */ LIST_ENTRY ListEntry; /* IRP listentry */
KSEMAPHORE RequestSemaphore; /* IRP list request Semaphore */ KSEMAPHORE RequestSemaphore; /* IRP list request Semaphore */
+30 -4
View File
@@ -36,6 +36,19 @@
volatile BOOL ProbingHostDeviceForWrite = FALSE; volatile BOOL ProbingHostDeviceForWrite = FALSE;
static BOOL MountCancelRequested (PEXTENSION Extension)
{
PMOUNT_CANCEL_CONTEXT context = Extension->MountCancelContext;
return context && InterlockedExchangeAdd (&context->UserAbortRequested, 0) != 0;
}
static void ResetMountKeyDerivationAbort (PEXTENSION Extension)
{
PMOUNT_CANCEL_CONTEXT context = Extension->MountCancelContext;
if (context && !MountCancelRequested (Extension))
InterlockedExchange (&context->KeyDerivationAbort, 0);
}
NTSTATUS TCOpenVolume (PDEVICE_OBJECT DeviceObject, NTSTATUS TCOpenVolume (PDEVICE_OBJECT DeviceObject,
PEXTENSION Extension, PEXTENSION Extension,
@@ -593,11 +606,20 @@ NTSTATUS TCOpenVolume (PDEVICE_OBJECT DeviceObject,
/* Attempt to recognize the volume (decrypt the header) */ /* Attempt to recognize the volume (decrypt the header) */
if (MountCancelRequested (Extension))
{
mount->nReturnCode = ERR_USER_ABORT;
ntStatus = STATUS_SUCCESS;
goto error;
}
ResetMountKeyDerivationAbort (Extension);
ReadVolumeHeaderRecoveryMode = mount->RecoveryMode; ReadVolumeHeaderRecoveryMode = mount->RecoveryMode;
if ((volumeType == TC_VOLUME_TYPE_HIDDEN) && mount->bProtectHiddenVolume) if ((volumeType == TC_VOLUME_TYPE_HIDDEN) && mount->bProtectHiddenVolume)
{ {
mount->nReturnCode = ReadVolumeHeaderWCache ( mount->nReturnCode = ReadVolumeHeaderWCacheWithAbort (
FALSE, FALSE,
bAutoCachePassword, bAutoCachePassword,
mount->bCachePim, mount->bCachePim,
@@ -605,11 +627,13 @@ NTSTATUS TCOpenVolume (PDEVICE_OBJECT DeviceObject,
&mount->ProtectedHidVolPassword, &mount->ProtectedHidVolPassword,
mount->ProtectedHidVolPkcs5Prf, mount->ProtectedHidVolPkcs5Prf,
mount->ProtectedHidVolPim, mount->ProtectedHidVolPim,
&tmpCryptoInfo); &tmpCryptoInfo,
Extension->MountCancelContext ? &Extension->MountCancelContext->KeyDerivationAbort : NULL,
Extension->MountCancelContext ? &Extension->MountCancelContext->UserAbortRequested : NULL);
} }
else else
{ {
mount->nReturnCode = ReadVolumeHeaderWCache ( mount->nReturnCode = ReadVolumeHeaderWCacheWithAbort (
mount->bPartitionInInactiveSysEncScope && volumeType == TC_VOLUME_TYPE_NORMAL, mount->bPartitionInInactiveSysEncScope && volumeType == TC_VOLUME_TYPE_NORMAL,
bAutoCachePassword, bAutoCachePassword,
mount->bCachePim, mount->bCachePim,
@@ -617,7 +641,9 @@ NTSTATUS TCOpenVolume (PDEVICE_OBJECT DeviceObject,
&mount->VolumePassword, &mount->VolumePassword,
mount->pkcs5_prf, mount->pkcs5_prf,
mount->VolumePim, mount->VolumePim,
&Extension->cryptoInfo); &Extension->cryptoInfo,
Extension->MountCancelContext ? &Extension->MountCancelContext->KeyDerivationAbort : NULL,
Extension->MountCancelContext ? &Extension->MountCancelContext->UserAbortRequested : NULL);
} }
ReadVolumeHeaderRecoveryMode = FALSE; ReadVolumeHeaderRecoveryMode = FALSE;
+11 -1
View File
@@ -103,6 +103,7 @@ enum hidden_os_read_only_notif_mode
#define TIMER_INTERVAL_KEYB_LAYOUT_GUARD 10 #define TIMER_INTERVAL_KEYB_LAYOUT_GUARD 10
#define TIMER_INTERVAL_UPDATE_DEVICE_LIST 1000 #define TIMER_INTERVAL_UPDATE_DEVICE_LIST 1000
#define TIMER_INTERVAL_CHECK_FOREGROUND 500 #define TIMER_INTERVAL_CHECK_FOREGROUND 500
#define TC_COMMAND_CANCEL_MOUNT L"/cancelmount"
BootEncryption *BootEncObj = NULL; BootEncryption *BootEncObj = NULL;
BootEncryptionStatus BootEncStatus; BootEncryptionStatus BootEncStatus;
@@ -5502,7 +5503,7 @@ static BOOL Mount (HWND hwndDlg, int nDosDriveNo, wchar_t *szVolFileName, int pi
NormalCursor (); NormalCursor ();
if (mounted) if (mounted > 0)
{ {
// Check for problematic file extensions (exe, dll, sys) // Check for problematic file extensions (exe, dll, sys)
@@ -9558,6 +9559,7 @@ void ExtractCommandLine (HWND hwndDlg, wchar_t *lpszCommandLine)
OptionEnableMemoryProtection, OptionEnableMemoryProtection,
OptionEnableScreenProtection, OptionEnableScreenProtection,
OptionSignalExit, OptionSignalExit,
CommandCancelMount,
CommandUnmount, CommandUnmount,
}; };
@@ -9591,6 +9593,8 @@ void ExtractCommandLine (HWND hwndDlg, wchar_t *lpszCommandLine)
{ OptionEnableMemoryProtection, L"/protectMemory", NULL, FALSE }, { OptionEnableMemoryProtection, L"/protectMemory", NULL, FALSE },
{ OptionEnableScreenProtection, L"/protectScreen", NULL, FALSE }, { OptionEnableScreenProtection, L"/protectScreen", NULL, FALSE },
{ OptionSignalExit, L"/signalExit", NULL, FALSE }, { OptionSignalExit, L"/signalExit", NULL, FALSE },
// Add /cancelmount to the command table so it appears in /help.
{ CommandCancelMount, TC_COMMAND_CANCEL_MOUNT, NULL, FALSE },
{ CommandUnmount, L"/unmount", L"/u", FALSE }, { CommandUnmount, L"/unmount", L"/u", FALSE },
}; };
@@ -10588,6 +10592,12 @@ int WINAPI wWinMain (HINSTANCE hInstance, HINSTANCE hPrevInstance, wchar_t *lpsz
for (int i = 0; argv && i < argc; i++) for (int i = 0; argv && i < argc; i++)
{ {
if (_wcsicmp (argv[i], TC_COMMAND_CANCEL_MOUNT) == 0)
{
BOOL abortSent = AbortMountOperation (-1);
LocalFree (argv); // free memory allocated by CommandLineToArgvW
return abortSent ? 0 : 1;
}
if (_wcsicmp (argv[i], L"/protectScreen") == 0) if (_wcsicmp (argv[i], L"/protectScreen") == 0)
{ {
if ((i < argc - 1) && _wcsicmp (argv[i + 1], L"no") == 0) if ((i < argc - 1) && _wcsicmp (argv[i + 1], L"no") == 0)