diff options
Diffstat (limited to 'dom/webauthn/WinWebAuthnService.cpp')
-rw-r--r-- | dom/webauthn/WinWebAuthnService.cpp | 1047 |
1 files changed, 1047 insertions, 0 deletions
diff --git a/dom/webauthn/WinWebAuthnService.cpp b/dom/webauthn/WinWebAuthnService.cpp new file mode 100644 index 0000000000..8667cf5615 --- /dev/null +++ b/dom/webauthn/WinWebAuthnService.cpp @@ -0,0 +1,1047 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ +/* vim: set ts=8 sts=2 et sw=2 tw=80: */ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this file, + * You can obtain one at http://mozilla.org/MPL/2.0/. */ + +#include "mozilla/dom/PWebAuthnTransactionParent.h" +#include "mozilla/ipc/BackgroundParent.h" +#include "mozilla/Assertions.h" +#include "mozilla/MozPromise.h" +#include "mozilla/Preferences.h" +#include "mozilla/ScopeExit.h" +#include "mozilla/StaticMutex.h" +#include "mozilla/Unused.h" + +#include "nsTextFormatter.h" +#include "nsWindowsHelpers.h" +#include "WebAuthnAutoFillEntry.h" +#include "WebAuthnEnumStrings.h" +#include "WebAuthnResult.h" +#include "WebAuthnTransportIdentifiers.h" +#include "winwebauthn/webauthn.h" +#include "WinWebAuthnService.h" + +namespace mozilla::dom { + +namespace { +StaticRWLock gWinWebAuthnModuleLock; + +static bool gWinWebAuthnModuleUnusable = false; +static HMODULE gWinWebAuthnModule = 0; + +static decltype(WebAuthNIsUserVerifyingPlatformAuthenticatorAvailable)* + gWinWebauthnIsUVPAA = nullptr; +static decltype(WebAuthNAuthenticatorMakeCredential)* + gWinWebauthnMakeCredential = nullptr; +static decltype(WebAuthNFreeCredentialAttestation)* + gWinWebauthnFreeCredentialAttestation = nullptr; +static decltype(WebAuthNAuthenticatorGetAssertion)* gWinWebauthnGetAssertion = + nullptr; +static decltype(WebAuthNFreeAssertion)* gWinWebauthnFreeAssertion = nullptr; +static decltype(WebAuthNGetCancellationId)* gWinWebauthnGetCancellationId = + nullptr; +static decltype(WebAuthNCancelCurrentOperation)* + gWinWebauthnCancelCurrentOperation = nullptr; +static decltype(WebAuthNGetErrorName)* gWinWebauthnGetErrorName = nullptr; +static decltype(WebAuthNGetApiVersionNumber)* gWinWebauthnGetApiVersionNumber = + nullptr; +static decltype(WebAuthNGetPlatformCredentialList)* + gWinWebauthnGetPlatformCredentialList = nullptr; +static decltype(WebAuthNFreePlatformCredentialList)* + gWinWebauthnFreePlatformCredentialList = nullptr; + +} // namespace + +/*********************************************************************** + * WinWebAuthnService Implementation + **********************************************************************/ + +constexpr uint32_t kMinWinWebAuthNApiVersion = WEBAUTHN_API_VERSION_1; + +NS_IMPL_ISUPPORTS(WinWebAuthnService, nsIWebAuthnService) + +/* static */ +nsresult WinWebAuthnService::EnsureWinWebAuthnModuleLoaded() { + { + StaticAutoReadLock moduleLock(gWinWebAuthnModuleLock); + if (gWinWebAuthnModule) { + // The module is already loaded. + return NS_OK; + } + if (gWinWebAuthnModuleUnusable) { + // A previous attempt to load the module failed. + return NS_ERROR_NOT_AVAILABLE; + } + } + + StaticAutoWriteLock lock(gWinWebAuthnModuleLock); + if (gWinWebAuthnModule) { + // Another thread successfully loaded the module while we were waiting. + return NS_OK; + } + if (gWinWebAuthnModuleUnusable) { + // Another thread failed to load the module while we were waiting. + return NS_ERROR_NOT_AVAILABLE; + } + + gWinWebAuthnModule = LoadLibrarySystem32(L"webauthn.dll"); + auto markModuleUnusable = MakeScopeExit([]() { + if (gWinWebAuthnModule) { + FreeLibrary(gWinWebAuthnModule); + gWinWebAuthnModule = 0; + } + gWinWebAuthnModuleUnusable = true; + }); + + if (!gWinWebAuthnModule) { + return NS_ERROR_NOT_AVAILABLE; + } + + gWinWebauthnIsUVPAA = reinterpret_cast< + decltype(WebAuthNIsUserVerifyingPlatformAuthenticatorAvailable)*>( + GetProcAddress(gWinWebAuthnModule, + "WebAuthNIsUserVerifyingPlatformAuthenticatorAvailable")); + gWinWebauthnMakeCredential = + reinterpret_cast<decltype(WebAuthNAuthenticatorMakeCredential)*>( + GetProcAddress(gWinWebAuthnModule, + "WebAuthNAuthenticatorMakeCredential")); + gWinWebauthnFreeCredentialAttestation = + reinterpret_cast<decltype(WebAuthNFreeCredentialAttestation)*>( + GetProcAddress(gWinWebAuthnModule, + "WebAuthNFreeCredentialAttestation")); + gWinWebauthnGetAssertion = + reinterpret_cast<decltype(WebAuthNAuthenticatorGetAssertion)*>( + GetProcAddress(gWinWebAuthnModule, + "WebAuthNAuthenticatorGetAssertion")); + gWinWebauthnFreeAssertion = + reinterpret_cast<decltype(WebAuthNFreeAssertion)*>( + GetProcAddress(gWinWebAuthnModule, "WebAuthNFreeAssertion")); + gWinWebauthnGetCancellationId = + reinterpret_cast<decltype(WebAuthNGetCancellationId)*>( + GetProcAddress(gWinWebAuthnModule, "WebAuthNGetCancellationId")); + gWinWebauthnCancelCurrentOperation = + reinterpret_cast<decltype(WebAuthNCancelCurrentOperation)*>( + GetProcAddress(gWinWebAuthnModule, "WebAuthNCancelCurrentOperation")); + gWinWebauthnGetErrorName = reinterpret_cast<decltype(WebAuthNGetErrorName)*>( + GetProcAddress(gWinWebAuthnModule, "WebAuthNGetErrorName")); + gWinWebauthnGetApiVersionNumber = + reinterpret_cast<decltype(WebAuthNGetApiVersionNumber)*>( + GetProcAddress(gWinWebAuthnModule, "WebAuthNGetApiVersionNumber")); + + if (!(gWinWebauthnIsUVPAA && gWinWebauthnMakeCredential && + gWinWebauthnFreeCredentialAttestation && gWinWebauthnGetAssertion && + gWinWebauthnFreeAssertion && gWinWebauthnGetCancellationId && + gWinWebauthnCancelCurrentOperation && gWinWebauthnGetErrorName && + gWinWebauthnGetApiVersionNumber)) { + return NS_ERROR_NOT_AVAILABLE; + } + + DWORD version = gWinWebauthnGetApiVersionNumber(); + + if (version >= WEBAUTHN_API_VERSION_4) { + gWinWebauthnGetPlatformCredentialList = + reinterpret_cast<decltype(WebAuthNGetPlatformCredentialList)*>( + GetProcAddress(gWinWebAuthnModule, + "WebAuthNGetPlatformCredentialList")); + gWinWebauthnFreePlatformCredentialList = + reinterpret_cast<decltype(WebAuthNFreePlatformCredentialList)*>( + GetProcAddress(gWinWebAuthnModule, + "WebAuthNFreePlatformCredentialList")); + if (!(gWinWebauthnGetPlatformCredentialList && + gWinWebauthnFreePlatformCredentialList)) { + return NS_ERROR_NOT_AVAILABLE; + } + } + + // Bug 1869584: In some of our tests, a content process can end up here due to + // a call to WinWebAuthnService::AreWebAuthNApisAvailable. This causes us to + // fail an assertion in Preferences::SetBool, which is parent-process only. + if (XRE_IsParentProcess()) { + NS_DispatchToMainThread(NS_NewRunnableFunction(__func__, [version]() { + Preferences::SetBool("security.webauthn.show_ms_settings_link", + version >= WEBAUTHN_API_VERSION_7); + })); + } + + markModuleUnusable.release(); + return NS_OK; +} + +WinWebAuthnService::~WinWebAuthnService() { + StaticAutoWriteLock lock(gWinWebAuthnModuleLock); + if (gWinWebAuthnModule) { + FreeLibrary(gWinWebAuthnModule); + } + gWinWebAuthnModule = 0; +} + +// static +bool WinWebAuthnService::AreWebAuthNApisAvailable() { + nsresult rv = EnsureWinWebAuthnModuleLoaded(); + NS_ENSURE_SUCCESS(rv, false); + + StaticAutoReadLock moduleLock(gWinWebAuthnModuleLock); + return gWinWebAuthnModule && + gWinWebauthnGetApiVersionNumber() >= kMinWinWebAuthNApiVersion; +} + +NS_IMETHODIMP +WinWebAuthnService::GetIsUVPAA(bool* aAvailable) { + nsresult rv = EnsureWinWebAuthnModuleLoaded(); + NS_ENSURE_SUCCESS(rv, rv); + + if (WinWebAuthnService::AreWebAuthNApisAvailable()) { + BOOL isUVPAA = FALSE; + StaticAutoReadLock moduleLock(gWinWebAuthnModuleLock); + *aAvailable = gWinWebAuthnModule && gWinWebauthnIsUVPAA(&isUVPAA) == S_OK && + isUVPAA == TRUE; + } else { + *aAvailable = false; + } + return NS_OK; +} + +NS_IMETHODIMP +WinWebAuthnService::Cancel(uint64_t aTransactionId) { + return NS_ERROR_NOT_IMPLEMENTED; +} + +NS_IMETHODIMP +WinWebAuthnService::Reset() { + // Reset will never be the first function to use gWinWebAuthnModule, so + // we shouldn't try to initialize it here. + auto guard = mTransactionState.Lock(); + if (guard->isSome()) { + StaticAutoReadLock moduleLock(gWinWebAuthnModuleLock); + if (gWinWebAuthnModule) { + const GUID cancellationId = guard->ref().cancellationId; + gWinWebauthnCancelCurrentOperation(&cancellationId); + } + if (guard->ref().pendingSignPromise.isSome()) { + // This request was never dispatched to the platform API, so + // we need to reject the promise ourselves. + guard->ref().pendingSignPromise.ref()->Reject( + NS_ERROR_DOM_NOT_ALLOWED_ERR); + } + guard->reset(); + } + + return NS_OK; +} + +NS_IMETHODIMP +WinWebAuthnService::MakeCredential(uint64_t aTransactionId, + uint64_t aBrowsingContextId, + nsIWebAuthnRegisterArgs* aArgs, + nsIWebAuthnRegisterPromise* aPromise) { + nsresult rv = EnsureWinWebAuthnModuleLoaded(); + NS_ENSURE_SUCCESS(rv, rv); + + Reset(); + auto guard = mTransactionState.Lock(); + StaticAutoReadLock moduleLock(gWinWebAuthnModuleLock); + GUID cancellationId; + if (gWinWebauthnGetCancellationId(&cancellationId) != S_OK) { + // caller will reject promise + return NS_ERROR_DOM_UNKNOWN_ERR; + } + *guard = Some(TransactionState{ + aTransactionId, + aBrowsingContextId, + Nothing(), + Nothing(), + cancellationId, + }); + + nsCOMPtr<nsIRunnable> runnable(NS_NewRunnableFunction( + "WinWebAuthnService::MakeCredential", + [self = RefPtr{this}, aArgs = RefPtr{aArgs}, aPromise = RefPtr{aPromise}, + cancellationId]() mutable { + // Take a read lock on gWinWebAuthnModuleLock to prevent the module from + // being unloaded while the operation is in progress. This does not + // prevent the operation from being cancelled, so it does not block a + // clean shutdown. + StaticAutoReadLock moduleLock(gWinWebAuthnModuleLock); + if (!gWinWebAuthnModule) { + aPromise->Reject(NS_ERROR_DOM_UNKNOWN_ERR); + return; + } + + BOOL HmacCreateSecret = FALSE; + BOOL MinPinLength = FALSE; + + // RP Information + nsString rpId; + Unused << aArgs->GetRpId(rpId); + WEBAUTHN_RP_ENTITY_INFORMATION rpInfo = { + WEBAUTHN_RP_ENTITY_INFORMATION_CURRENT_VERSION, rpId.get(), nullptr, + nullptr}; + + // User Information + WEBAUTHN_USER_ENTITY_INFORMATION userInfo = { + WEBAUTHN_USER_ENTITY_INFORMATION_CURRENT_VERSION, + 0, + nullptr, + nullptr, + nullptr, + nullptr}; + + // Client Data + nsCString clientDataJSON; + Unused << aArgs->GetClientDataJSON(clientDataJSON); + WEBAUTHN_CLIENT_DATA WebAuthNClientData = { + WEBAUTHN_CLIENT_DATA_CURRENT_VERSION, + (DWORD)clientDataJSON.Length(), (BYTE*)(clientDataJSON.get()), + WEBAUTHN_HASH_ALGORITHM_SHA_256}; + + // User Verification Requirement + DWORD winUserVerificationReq = + WEBAUTHN_USER_VERIFICATION_REQUIREMENT_ANY; + + // Resident Key + BOOL winRequireResidentKey = FALSE; + BOOL winPreferResidentKey = FALSE; + + // AttestationConveyance + DWORD winAttestation = WEBAUTHN_ATTESTATION_CONVEYANCE_PREFERENCE_ANY; + + nsString rpName; + Unused << aArgs->GetRpName(rpName); + rpInfo.pwszName = rpName.get(); + rpInfo.pwszIcon = nullptr; + + nsTArray<uint8_t> userId; + Unused << aArgs->GetUserId(userId); + userInfo.cbId = static_cast<DWORD>(userId.Length()); + userInfo.pbId = const_cast<unsigned char*>(userId.Elements()); + + nsString userName; + Unused << aArgs->GetUserName(userName); + userInfo.pwszName = userName.get(); + + userInfo.pwszIcon = nullptr; + + nsString userDisplayName; + Unused << aArgs->GetUserDisplayName(userDisplayName); + userInfo.pwszDisplayName = userDisplayName.get(); + + // Algorithms + nsTArray<WEBAUTHN_COSE_CREDENTIAL_PARAMETER> coseParams; + nsTArray<int32_t> coseAlgs; + Unused << aArgs->GetCoseAlgs(coseAlgs); + for (const int32_t& coseAlg : coseAlgs) { + WEBAUTHN_COSE_CREDENTIAL_PARAMETER coseAlgorithm = { + WEBAUTHN_COSE_CREDENTIAL_PARAMETER_CURRENT_VERSION, + WEBAUTHN_CREDENTIAL_TYPE_PUBLIC_KEY, coseAlg}; + coseParams.AppendElement(coseAlgorithm); + } + + nsString userVerificationReq; + Unused << aArgs->GetUserVerification(userVerificationReq); + // This mapping needs to be reviewed if values are added to the + // UserVerificationRequirement enum. + static_assert(MOZ_WEBAUTHN_ENUM_STRINGS_VERSION == 3); + if (userVerificationReq.EqualsLiteral( + MOZ_WEBAUTHN_USER_VERIFICATION_REQUIREMENT_REQUIRED)) { + winUserVerificationReq = + WEBAUTHN_USER_VERIFICATION_REQUIREMENT_REQUIRED; + } else if (userVerificationReq.EqualsLiteral( + MOZ_WEBAUTHN_USER_VERIFICATION_REQUIREMENT_PREFERRED)) { + winUserVerificationReq = + WEBAUTHN_USER_VERIFICATION_REQUIREMENT_PREFERRED; + } else if (userVerificationReq.EqualsLiteral( + MOZ_WEBAUTHN_RESIDENT_KEY_REQUIREMENT_DISCOURAGED)) { + winUserVerificationReq = + WEBAUTHN_USER_VERIFICATION_REQUIREMENT_DISCOURAGED; + } else { + winUserVerificationReq = WEBAUTHN_USER_VERIFICATION_REQUIREMENT_ANY; + } + + // Attachment + DWORD winAttachment = WEBAUTHN_AUTHENTICATOR_ATTACHMENT_ANY; + nsString authenticatorAttachment; + nsresult rv = + aArgs->GetAuthenticatorAttachment(authenticatorAttachment); + if (rv != NS_ERROR_NOT_AVAILABLE) { + if (NS_FAILED(rv)) { + aPromise->Reject(rv); + return; + } + // This mapping needs to be reviewed if values are added to the + // AuthenticatorAttachement enum. + static_assert(MOZ_WEBAUTHN_ENUM_STRINGS_VERSION == 3); + if (authenticatorAttachment.EqualsLiteral( + MOZ_WEBAUTHN_AUTHENTICATOR_ATTACHMENT_PLATFORM)) { + winAttachment = WEBAUTHN_AUTHENTICATOR_ATTACHMENT_PLATFORM; + } else if ( + authenticatorAttachment.EqualsLiteral( + MOZ_WEBAUTHN_AUTHENTICATOR_ATTACHMENT_CROSS_PLATFORM)) { + winAttachment = WEBAUTHN_AUTHENTICATOR_ATTACHMENT_CROSS_PLATFORM; + } else { + winAttachment = WEBAUTHN_AUTHENTICATOR_ATTACHMENT_ANY; + } + } + + nsString residentKey; + Unused << aArgs->GetResidentKey(residentKey); + // This mapping needs to be reviewed if values are added to the + // ResidentKeyRequirement enum. + static_assert(MOZ_WEBAUTHN_ENUM_STRINGS_VERSION == 3); + if (residentKey.EqualsLiteral( + MOZ_WEBAUTHN_RESIDENT_KEY_REQUIREMENT_REQUIRED)) { + winRequireResidentKey = TRUE; + winPreferResidentKey = TRUE; + } else if (residentKey.EqualsLiteral( + MOZ_WEBAUTHN_RESIDENT_KEY_REQUIREMENT_PREFERRED)) { + winRequireResidentKey = FALSE; + winPreferResidentKey = TRUE; + } else if (residentKey.EqualsLiteral( + MOZ_WEBAUTHN_RESIDENT_KEY_REQUIREMENT_DISCOURAGED)) { + winRequireResidentKey = FALSE; + winPreferResidentKey = FALSE; + } else { + // WebAuthnManager::MakeCredential is supposed to assign one of the + // above values, so this shouldn't happen. + MOZ_ASSERT_UNREACHABLE(); + aPromise->Reject(NS_ERROR_DOM_UNKNOWN_ERR); + return; + } + + // AttestationConveyance + nsString attestation; + Unused << aArgs->GetAttestationConveyancePreference(attestation); + bool anonymize = false; + // This mapping needs to be reviewed if values are added to the + // AttestationConveyancePreference enum. + static_assert(MOZ_WEBAUTHN_ENUM_STRINGS_VERSION == 3); + if (attestation.EqualsLiteral( + MOZ_WEBAUTHN_ATTESTATION_CONVEYANCE_PREFERENCE_NONE)) { + winAttestation = WEBAUTHN_ATTESTATION_CONVEYANCE_PREFERENCE_NONE; + anonymize = true; + } else if ( + attestation.EqualsLiteral( + MOZ_WEBAUTHN_ATTESTATION_CONVEYANCE_PREFERENCE_INDIRECT)) { + winAttestation = WEBAUTHN_ATTESTATION_CONVEYANCE_PREFERENCE_INDIRECT; + } else if (attestation.EqualsLiteral( + MOZ_WEBAUTHN_ATTESTATION_CONVEYANCE_PREFERENCE_DIRECT)) { + winAttestation = WEBAUTHN_ATTESTATION_CONVEYANCE_PREFERENCE_DIRECT; + } else { + winAttestation = WEBAUTHN_ATTESTATION_CONVEYANCE_PREFERENCE_ANY; + } + + bool requestedCredProps; + Unused << aArgs->GetCredProps(&requestedCredProps); + + bool requestedMinPinLength; + Unused << aArgs->GetMinPinLength(&requestedMinPinLength); + + bool requestedHmacCreateSecret; + Unused << aArgs->GetHmacCreateSecret(&requestedHmacCreateSecret); + + // Extensions that might require an entry: hmac-secret, minPinLength. + WEBAUTHN_EXTENSION rgExtension[2] = {}; + DWORD cExtensions = 0; + if (requestedHmacCreateSecret) { + HmacCreateSecret = TRUE; + rgExtension[cExtensions].pwszExtensionIdentifier = + WEBAUTHN_EXTENSIONS_IDENTIFIER_HMAC_SECRET; + rgExtension[cExtensions].cbExtension = sizeof(BOOL); + rgExtension[cExtensions].pvExtension = &HmacCreateSecret; + cExtensions++; + } + if (requestedMinPinLength) { + MinPinLength = TRUE; + rgExtension[cExtensions].pwszExtensionIdentifier = + WEBAUTHN_EXTENSIONS_IDENTIFIER_MIN_PIN_LENGTH; + rgExtension[cExtensions].cbExtension = sizeof(BOOL); + rgExtension[cExtensions].pvExtension = &MinPinLength; + cExtensions++; + } + + WEBAUTHN_COSE_CREDENTIAL_PARAMETERS WebAuthNCredentialParameters = { + static_cast<DWORD>(coseParams.Length()), coseParams.Elements()}; + + // Exclude Credentials + nsTArray<nsTArray<uint8_t>> excludeList; + Unused << aArgs->GetExcludeList(excludeList); + + nsTArray<uint8_t> excludeListTransports; + Unused << aArgs->GetExcludeListTransports(excludeListTransports); + + if (excludeList.Length() != excludeListTransports.Length()) { + aPromise->Reject(NS_ERROR_DOM_UNKNOWN_ERR); + return; + } + + nsTArray<WEBAUTHN_CREDENTIAL_EX> excludeCredentials; + WEBAUTHN_CREDENTIAL_EX* pExcludeCredentials = nullptr; + nsTArray<WEBAUTHN_CREDENTIAL_EX*> excludeCredentialsPtrs; + WEBAUTHN_CREDENTIAL_LIST excludeCredentialList = {0}; + WEBAUTHN_CREDENTIAL_LIST* pExcludeCredentialList = nullptr; + + for (size_t i = 0; i < excludeList.Length(); i++) { + nsTArray<uint8_t>& cred = excludeList[i]; + uint8_t& transports = excludeListTransports[i]; + DWORD winTransports = 0; + if (transports & MOZ_WEBAUTHN_AUTHENTICATOR_TRANSPORT_ID_USB) { + winTransports |= WEBAUTHN_CTAP_TRANSPORT_USB; + } + if (transports & MOZ_WEBAUTHN_AUTHENTICATOR_TRANSPORT_ID_NFC) { + winTransports |= WEBAUTHN_CTAP_TRANSPORT_NFC; + } + if (transports & MOZ_WEBAUTHN_AUTHENTICATOR_TRANSPORT_ID_BLE) { + winTransports |= WEBAUTHN_CTAP_TRANSPORT_BLE; + } + if (transports & MOZ_WEBAUTHN_AUTHENTICATOR_TRANSPORT_ID_INTERNAL) { + winTransports |= WEBAUTHN_CTAP_TRANSPORT_INTERNAL; + } + if (transports & MOZ_WEBAUTHN_AUTHENTICATOR_TRANSPORT_ID_HYBRID) { + winTransports |= WEBAUTHN_CTAP_TRANSPORT_HYBRID; + } + + WEBAUTHN_CREDENTIAL_EX credential = { + WEBAUTHN_CREDENTIAL_EX_CURRENT_VERSION, + static_cast<DWORD>(cred.Length()), (PBYTE)(cred.Elements()), + WEBAUTHN_CREDENTIAL_TYPE_PUBLIC_KEY, winTransports}; + excludeCredentials.AppendElement(credential); + } + + if (!excludeCredentials.IsEmpty()) { + pExcludeCredentials = excludeCredentials.Elements(); + for (DWORD i = 0; i < excludeCredentials.Length(); i++) { + excludeCredentialsPtrs.AppendElement(&pExcludeCredentials[i]); + } + excludeCredentialList.cCredentials = excludeCredentials.Length(); + excludeCredentialList.ppCredentials = + excludeCredentialsPtrs.Elements(); + pExcludeCredentialList = &excludeCredentialList; + } + + uint32_t timeout_u32; + Unused << aArgs->GetTimeoutMS(&timeout_u32); + DWORD timeout = timeout_u32; + + // MakeCredentialOptions + WEBAUTHN_AUTHENTICATOR_MAKE_CREDENTIAL_OPTIONS + WebAuthNCredentialOptions = { + WEBAUTHN_AUTHENTICATOR_MAKE_CREDENTIAL_OPTIONS_VERSION_7, + timeout, + {0, NULL}, + {0, NULL}, + winAttachment, + winRequireResidentKey, + winUserVerificationReq, + winAttestation, + 0, // Flags + &cancellationId, // CancellationId + pExcludeCredentialList, + WEBAUTHN_ENTERPRISE_ATTESTATION_NONE, + WEBAUTHN_LARGE_BLOB_SUPPORT_NONE, + winPreferResidentKey, // PreferResidentKey + FALSE, // BrowserInPrivateMode + FALSE, // EnablePrf + NULL, // LinkedDevice + 0, // size of JsonExt + NULL, // JsonExt + }; + + if (cExtensions != 0) { + WebAuthNCredentialOptions.Extensions.cExtensions = cExtensions; + WebAuthNCredentialOptions.Extensions.pExtensions = rgExtension; + } + + PWEBAUTHN_CREDENTIAL_ATTESTATION pWebAuthNCredentialAttestation = + nullptr; + + // Bug 1518876: Get Window Handle from Content process for Windows + // WebAuthN APIs + HWND hWnd = GetForegroundWindow(); + + HRESULT hr = gWinWebauthnMakeCredential( + hWnd, &rpInfo, &userInfo, &WebAuthNCredentialParameters, + &WebAuthNClientData, &WebAuthNCredentialOptions, + &pWebAuthNCredentialAttestation); + + if (hr == S_OK) { + RefPtr<WebAuthnRegisterResult> result = new WebAuthnRegisterResult( + clientDataJSON, pWebAuthNCredentialAttestation); + + // WEBAUTHN_CREDENTIAL_ATTESTATION structs of version >= 4 always + // include a flag to indicate whether a resident key was created. We + // copy that flag to the credProps extension output only if the RP + // requested the credProps extension. + if (requestedCredProps && + pWebAuthNCredentialAttestation->dwVersion >= + WEBAUTHN_CREDENTIAL_ATTESTATION_VERSION_4) { + BOOL rk = pWebAuthNCredentialAttestation->bResidentKey; + Unused << result->SetCredPropsRk(rk == TRUE); + } + gWinWebauthnFreeCredentialAttestation(pWebAuthNCredentialAttestation); + + if (anonymize) { + nsresult rv = result->Anonymize(); + if (NS_FAILED(rv)) { + aPromise->Reject(NS_ERROR_DOM_NOT_ALLOWED_ERR); + return; + } + } + aPromise->Resolve(result); + } else { + PCWSTR errorName = gWinWebauthnGetErrorName(hr); + nsresult aError = NS_ERROR_DOM_ABORT_ERR; + + if (_wcsicmp(errorName, L"InvalidStateError") == 0) { + aError = NS_ERROR_DOM_INVALID_STATE_ERR; + } else if (_wcsicmp(errorName, L"ConstraintError") == 0 || + _wcsicmp(errorName, L"UnknownError") == 0) { + aError = NS_ERROR_DOM_UNKNOWN_ERR; + } else if (_wcsicmp(errorName, L"NotSupportedError") == 0) { + aError = NS_ERROR_DOM_INVALID_STATE_ERR; + } else if (_wcsicmp(errorName, L"NotAllowedError") == 0) { + aError = NS_ERROR_DOM_NOT_ALLOWED_ERR; + } + + aPromise->Reject(aError); + } + })); + + NS_DispatchBackgroundTask(runnable, NS_DISPATCH_EVENT_MAY_BLOCK); + return NS_OK; +} + +NS_IMETHODIMP +WinWebAuthnService::GetAssertion(uint64_t aTransactionId, + uint64_t aBrowsingContextId, + nsIWebAuthnSignArgs* aArgs, + nsIWebAuthnSignPromise* aPromise) { + nsresult rv = EnsureWinWebAuthnModuleLoaded(); + NS_ENSURE_SUCCESS(rv, rv); + + Reset(); + + auto guard = mTransactionState.Lock(); + + GUID cancellationId; + { + StaticAutoReadLock moduleLock(gWinWebAuthnModuleLock); + if (gWinWebauthnGetCancellationId(&cancellationId) != S_OK) { + // caller will reject promise + return NS_ERROR_DOM_UNKNOWN_ERR; + } + } + + *guard = Some(TransactionState{ + aTransactionId, + aBrowsingContextId, + Some(RefPtr{aArgs}), + Some(RefPtr{aPromise}), + cancellationId, + }); + + bool conditionallyMediated; + Unused << aArgs->GetConditionallyMediated(&conditionallyMediated); + if (!conditionallyMediated) { + DoGetAssertion(Nothing(), guard); + } + return NS_OK; +} + +void WinWebAuthnService::DoGetAssertion( + Maybe<nsTArray<uint8_t>>&& aSelectedCredentialId, + const TransactionStateMutex::AutoLock& aGuard) { + if (aGuard->isNothing() || aGuard->ref().pendingSignArgs.isNothing() || + aGuard->ref().pendingSignPromise.isNothing()) { + return; + } + + // Take the pending Args and Promise to prevent repeated calls to + // DoGetAssertion for this transaction. + RefPtr<nsIWebAuthnSignArgs> aArgs = aGuard->ref().pendingSignArgs.extract(); + RefPtr<nsIWebAuthnSignPromise> aPromise = + aGuard->ref().pendingSignPromise.extract(); + + nsCOMPtr<nsIRunnable> runnable(NS_NewRunnableFunction( + "WinWebAuthnService::MakeCredential", + [self = RefPtr{this}, aArgs, aPromise, + aSelectedCredentialId = std::move(aSelectedCredentialId), + aCancellationId = aGuard->ref().cancellationId]() mutable { + // Take a read lock on gWinWebAuthnModuleLock to prevent the module from + // being unloaded while the operation is in progress. This does not + // prevent the operation from being cancelled, so it does not block a + // clean shutdown. + StaticAutoReadLock moduleLock(gWinWebAuthnModuleLock); + if (!gWinWebAuthnModule) { + aPromise->Reject(NS_ERROR_DOM_UNKNOWN_ERR); + return; + } + + // Attachment + DWORD winAttachment = WEBAUTHN_AUTHENTICATOR_ATTACHMENT_ANY; + + // AppId + BOOL bAppIdUsed = FALSE; + BOOL* pbAppIdUsed = nullptr; + PCWSTR winAppIdentifier = nullptr; + + // Client Data + nsCString clientDataJSON; + Unused << aArgs->GetClientDataJSON(clientDataJSON); + WEBAUTHN_CLIENT_DATA WebAuthNClientData = { + WEBAUTHN_CLIENT_DATA_CURRENT_VERSION, + (DWORD)clientDataJSON.Length(), (BYTE*)(clientDataJSON.get()), + WEBAUTHN_HASH_ALGORITHM_SHA_256}; + + nsString appId; + nsresult rv = aArgs->GetAppId(appId); + if (rv != NS_ERROR_NOT_AVAILABLE) { + if (NS_FAILED(rv)) { + aPromise->Reject(rv); + return; + } + winAppIdentifier = appId.get(); + pbAppIdUsed = &bAppIdUsed; + } + + // RPID + nsString rpId; + Unused << aArgs->GetRpId(rpId); + + // User Verification Requirement + nsString userVerificationReq; + Unused << aArgs->GetUserVerification(userVerificationReq); + DWORD winUserVerificationReq; + // This mapping needs to be reviewed if values are added to the + // UserVerificationRequirement enum. + static_assert(MOZ_WEBAUTHN_ENUM_STRINGS_VERSION == 3); + if (userVerificationReq.EqualsLiteral( + MOZ_WEBAUTHN_USER_VERIFICATION_REQUIREMENT_REQUIRED)) { + winUserVerificationReq = + WEBAUTHN_USER_VERIFICATION_REQUIREMENT_REQUIRED; + } else if (userVerificationReq.EqualsLiteral( + MOZ_WEBAUTHN_USER_VERIFICATION_REQUIREMENT_PREFERRED)) { + winUserVerificationReq = + WEBAUTHN_USER_VERIFICATION_REQUIREMENT_PREFERRED; + } else if (userVerificationReq.EqualsLiteral( + MOZ_WEBAUTHN_RESIDENT_KEY_REQUIREMENT_DISCOURAGED)) { + winUserVerificationReq = + WEBAUTHN_USER_VERIFICATION_REQUIREMENT_DISCOURAGED; + } else { + winUserVerificationReq = WEBAUTHN_USER_VERIFICATION_REQUIREMENT_ANY; + } + + // allow Credentials + nsTArray<nsTArray<uint8_t>> allowList; + nsTArray<uint8_t> allowListTransports; + if (aSelectedCredentialId.isSome()) { + allowList.AppendElement(aSelectedCredentialId.extract()); + allowListTransports.AppendElement( + MOZ_WEBAUTHN_AUTHENTICATOR_TRANSPORT_ID_INTERNAL); + } else { + Unused << aArgs->GetAllowList(allowList); + Unused << aArgs->GetAllowListTransports(allowListTransports); + } + + if (allowList.Length() != allowListTransports.Length()) { + aPromise->Reject(NS_ERROR_DOM_UNKNOWN_ERR); + return; + } + + nsTArray<WEBAUTHN_CREDENTIAL_EX> allowCredentials; + WEBAUTHN_CREDENTIAL_EX* pAllowCredentials = nullptr; + nsTArray<WEBAUTHN_CREDENTIAL_EX*> allowCredentialsPtrs; + WEBAUTHN_CREDENTIAL_LIST allowCredentialList = {0}; + WEBAUTHN_CREDENTIAL_LIST* pAllowCredentialList = nullptr; + + for (size_t i = 0; i < allowList.Length(); i++) { + nsTArray<uint8_t>& cred = allowList[i]; + uint8_t& transports = allowListTransports[i]; + DWORD winTransports = 0; + if (transports & MOZ_WEBAUTHN_AUTHENTICATOR_TRANSPORT_ID_USB) { + winTransports |= WEBAUTHN_CTAP_TRANSPORT_USB; + } + if (transports & MOZ_WEBAUTHN_AUTHENTICATOR_TRANSPORT_ID_NFC) { + winTransports |= WEBAUTHN_CTAP_TRANSPORT_NFC; + } + if (transports & MOZ_WEBAUTHN_AUTHENTICATOR_TRANSPORT_ID_BLE) { + winTransports |= WEBAUTHN_CTAP_TRANSPORT_BLE; + } + if (transports & MOZ_WEBAUTHN_AUTHENTICATOR_TRANSPORT_ID_INTERNAL) { + winTransports |= WEBAUTHN_CTAP_TRANSPORT_INTERNAL; + } + if (transports & MOZ_WEBAUTHN_AUTHENTICATOR_TRANSPORT_ID_HYBRID) { + winTransports |= WEBAUTHN_CTAP_TRANSPORT_HYBRID; + } + + WEBAUTHN_CREDENTIAL_EX credential = { + WEBAUTHN_CREDENTIAL_EX_CURRENT_VERSION, + static_cast<DWORD>(cred.Length()), (PBYTE)(cred.Elements()), + WEBAUTHN_CREDENTIAL_TYPE_PUBLIC_KEY, winTransports}; + allowCredentials.AppendElement(credential); + } + + if (allowCredentials.Length()) { + pAllowCredentials = allowCredentials.Elements(); + for (DWORD i = 0; i < allowCredentials.Length(); i++) { + allowCredentialsPtrs.AppendElement(&pAllowCredentials[i]); + } + allowCredentialList.cCredentials = allowCredentials.Length(); + allowCredentialList.ppCredentials = allowCredentialsPtrs.Elements(); + pAllowCredentialList = &allowCredentialList; + } + + uint32_t timeout_u32; + Unused << aArgs->GetTimeoutMS(&timeout_u32); + DWORD timeout = timeout_u32; + + WEBAUTHN_AUTHENTICATOR_GET_ASSERTION_OPTIONS WebAuthNAssertionOptions = + { + WEBAUTHN_AUTHENTICATOR_GET_ASSERTION_OPTIONS_VERSION_7, + timeout, + {0, NULL}, + {0, NULL}, + winAttachment, + winUserVerificationReq, + 0, // dwFlags + winAppIdentifier, + pbAppIdUsed, + &aCancellationId, // CancellationId + pAllowCredentialList, + WEBAUTHN_CRED_LARGE_BLOB_OPERATION_NONE, + 0, // Size of CredLargeBlob + NULL, // CredLargeBlob + NULL, // HmacSecretSaltValues + FALSE, // BrowserInPrivateMode + NULL, // LinkedDevice + FALSE, // AutoFill + 0, // Size of JsonExt + NULL, // JsonExt + }; + + PWEBAUTHN_ASSERTION pWebAuthNAssertion = nullptr; + + // Bug 1518876: Get Window Handle from Content process for Windows + // WebAuthN APIs + HWND hWnd = GetForegroundWindow(); + + HRESULT hr = gWinWebauthnGetAssertion( + hWnd, rpId.get(), &WebAuthNClientData, &WebAuthNAssertionOptions, + &pWebAuthNAssertion); + + if (hr == S_OK) { + RefPtr<WebAuthnSignResult> result = + new WebAuthnSignResult(clientDataJSON, pWebAuthNAssertion); + gWinWebauthnFreeAssertion(pWebAuthNAssertion); + if (winAppIdentifier != nullptr) { + // The gWinWebauthnGetAssertion call modified bAppIdUsed through + // a pointer provided in WebAuthNAssertionOptions. + Unused << result->SetUsedAppId(bAppIdUsed == TRUE); + } + aPromise->Resolve(result); + } else { + PCWSTR errorName = gWinWebauthnGetErrorName(hr); + nsresult aError = NS_ERROR_DOM_ABORT_ERR; + + if (_wcsicmp(errorName, L"InvalidStateError") == 0) { + aError = NS_ERROR_DOM_INVALID_STATE_ERR; + } else if (_wcsicmp(errorName, L"ConstraintError") == 0 || + _wcsicmp(errorName, L"UnknownError") == 0) { + aError = NS_ERROR_DOM_UNKNOWN_ERR; + } else if (_wcsicmp(errorName, L"NotSupportedError") == 0) { + aError = NS_ERROR_DOM_INVALID_STATE_ERR; + } else if (_wcsicmp(errorName, L"NotAllowedError") == 0) { + aError = NS_ERROR_DOM_NOT_ALLOWED_ERR; + } + + aPromise->Reject(aError); + } + })); + + NS_DispatchBackgroundTask(runnable, NS_DISPATCH_EVENT_MAY_BLOCK); +} + +NS_IMETHODIMP +WinWebAuthnService::HasPendingConditionalGet(uint64_t aBrowsingContextId, + const nsAString& aOrigin, + uint64_t* aRv) { + auto guard = mTransactionState.Lock(); + if (guard->isNothing() || + guard->ref().browsingContextId != aBrowsingContextId || + guard->ref().pendingSignArgs.isNothing()) { + *aRv = 0; + return NS_OK; + } + + nsString origin; + Unused << guard->ref().pendingSignArgs.ref()->GetOrigin(origin); + if (origin != aOrigin) { + *aRv = 0; + return NS_OK; + } + + *aRv = guard->ref().transactionId; + return NS_OK; +} + +NS_IMETHODIMP +WinWebAuthnService::GetAutoFillEntries( + uint64_t aTransactionId, nsTArray<RefPtr<nsIWebAuthnAutoFillEntry>>& aRv) { + auto guard = mTransactionState.Lock(); + if (guard->isNothing() || guard->ref().transactionId != aTransactionId || + guard->ref().pendingSignArgs.isNothing()) { + return NS_ERROR_NOT_AVAILABLE; + } + + StaticAutoReadLock moduleLock(gWinWebAuthnModuleLock); + if (!gWinWebAuthnModule) { + return NS_ERROR_NOT_AVAILABLE; + } + + aRv.Clear(); + + if (gWinWebauthnGetApiVersionNumber() < WEBAUTHN_API_VERSION_4) { + // GetPlatformCredentialList was added in version 4. Earlier versions + // can still present a generic "Use a Passkey" autofill entry, so + // this isn't an error. + return NS_OK; + } + + nsString rpId; + Unused << guard->ref().pendingSignArgs.ref()->GetRpId(rpId); + + WEBAUTHN_GET_CREDENTIALS_OPTIONS getCredentialsOptions{ + WEBAUTHN_GET_CREDENTIALS_OPTIONS_VERSION_1, + rpId.get(), // pwszRpId + FALSE, // bBrowserInPrivateMode + }; + PWEBAUTHN_CREDENTIAL_DETAILS_LIST pCredentialList = nullptr; + HRESULT hr = gWinWebauthnGetPlatformCredentialList(&getCredentialsOptions, + &pCredentialList); + // WebAuthNGetPlatformCredentialList has an _Outptr_result_maybenull_ + // annotation and a comment "Returns NTE_NOT_FOUND when credentials are + // not found." + if (pCredentialList == nullptr) { + if (hr != NTE_NOT_FOUND) { + return NS_ERROR_FAILURE; + } + return NS_OK; + } + MOZ_ASSERT(hr == S_OK); + for (size_t i = 0; i < pCredentialList->cCredentialDetails; i++) { + RefPtr<nsIWebAuthnAutoFillEntry> entry( + new WebAuthnAutoFillEntry(pCredentialList->ppCredentialDetails[i])); + aRv.AppendElement(entry); + } + gWinWebauthnFreePlatformCredentialList(pCredentialList); + return NS_OK; +} + +NS_IMETHODIMP +WinWebAuthnService::SelectAutoFillEntry( + uint64_t aTransactionId, const nsTArray<uint8_t>& aCredentialId) { + auto guard = mTransactionState.Lock(); + if (guard->isNothing() || guard->ref().transactionId != aTransactionId || + guard->ref().pendingSignArgs.isNothing()) { + return NS_ERROR_NOT_AVAILABLE; + } + + nsTArray<nsTArray<uint8_t>> allowList; + Unused << guard->ref().pendingSignArgs.ref()->GetAllowList(allowList); + if (!allowList.IsEmpty() && !allowList.Contains(aCredentialId)) { + return NS_ERROR_FAILURE; + } + + Maybe<nsTArray<uint8_t>> id; + id.emplace(); + id.ref().Assign(aCredentialId); + DoGetAssertion(std::move(id), guard); + + return NS_OK; +} + +NS_IMETHODIMP +WinWebAuthnService::ResumeConditionalGet(uint64_t aTransactionId) { + auto guard = mTransactionState.Lock(); + if (guard->isNothing() || guard->ref().transactionId != aTransactionId || + guard->ref().pendingSignArgs.isNothing()) { + return NS_ERROR_NOT_AVAILABLE; + } + DoGetAssertion(Nothing(), guard); + return NS_OK; +} + +NS_IMETHODIMP +WinWebAuthnService::PinCallback(uint64_t aTransactionId, + const nsACString& aPin) { + return NS_ERROR_NOT_IMPLEMENTED; +} + +NS_IMETHODIMP +WinWebAuthnService::ResumeMakeCredential(uint64_t aTransactionId, + bool aForceNoneAttestation) { + return NS_ERROR_NOT_IMPLEMENTED; +} + +NS_IMETHODIMP +WinWebAuthnService::SelectionCallback(uint64_t aTransactionId, + uint64_t aIndex) { + return NS_ERROR_NOT_IMPLEMENTED; +} + +NS_IMETHODIMP +WinWebAuthnService::AddVirtualAuthenticator( + const nsACString& protocol, const nsACString& transport, + bool hasResidentKey, bool hasUserVerification, bool isUserConsenting, + bool isUserVerified, uint64_t* _retval) { + return NS_ERROR_NOT_IMPLEMENTED; +} + +NS_IMETHODIMP +WinWebAuthnService::RemoveVirtualAuthenticator(uint64_t authenticatorId) { + return NS_ERROR_NOT_IMPLEMENTED; +} + +NS_IMETHODIMP +WinWebAuthnService::AddCredential(uint64_t authenticatorId, + const nsACString& credentialId, + bool isResidentCredential, + const nsACString& rpId, + const nsACString& privateKey, + const nsACString& userHandle, + uint32_t signCount) { + return NS_ERROR_NOT_IMPLEMENTED; +} + +NS_IMETHODIMP +WinWebAuthnService::GetCredentials( + uint64_t authenticatorId, + nsTArray<RefPtr<nsICredentialParameters>>& _retval) { + return NS_ERROR_NOT_IMPLEMENTED; +} + +NS_IMETHODIMP +WinWebAuthnService::RemoveCredential(uint64_t authenticatorId, + const nsACString& credentialId) { + return NS_ERROR_NOT_IMPLEMENTED; +} + +NS_IMETHODIMP +WinWebAuthnService::RemoveAllCredentials(uint64_t authenticatorId) { + return NS_ERROR_NOT_IMPLEMENTED; +} + +NS_IMETHODIMP +WinWebAuthnService::SetUserVerified(uint64_t authenticatorId, + bool isUserVerified) { + return NS_ERROR_NOT_IMPLEMENTED; +} + +NS_IMETHODIMP +WinWebAuthnService::Listen() { return NS_ERROR_NOT_IMPLEMENTED; } + +NS_IMETHODIMP +WinWebAuthnService::RunCommand(const nsACString& cmd) { + return NS_ERROR_NOT_IMPLEMENTED; +} + +} // namespace mozilla::dom |