123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233 |
- /*++
- Copyright (c) 2015 Minoca Corp. All Rights Reserved
- Module Name:
- crypto.c
- Abstract:
- This module implements cryptographic functionality for the 802.11 core
- wireless networking library.
- Author:
- Chris Stevens 5-Nov-2015
- Environment:
- Kernel
- --*/
- //
- // ------------------------------------------------------------------- Includes
- //
- #include "net80211.h"
- #include "eapol.h"
- #include <minoca/lib/crypto.h>
- //
- // ---------------------------------------------------------------- Definitions
- //
- //
- // Define the default key ID to use for transmitting data.
- //
- #define NET80211_DEFAULT_ENCRYPTION_KEY 0
- //
- // ------------------------------------------------------ Data Type Definitions
- //
- //
- // ----------------------------------------------- Internal Function Prototypes
- //
- KSTATUS
- Net80211pCcmEncrypt (
- PNET80211_KEY Key,
- PUCHAR Message,
- ULONG MessageLength,
- PUCHAR Aad,
- ULONG AadLength,
- PUCHAR Nonce,
- ULONG NonceLength,
- PUCHAR AuthenticationField,
- ULONG AuthenticationFieldSize,
- ULONG LengthFieldSize
- );
- KSTATUS
- Net80211pCcmDecrypt (
- PNET80211_KEY Key,
- PUCHAR Message,
- ULONG MessageLength,
- PUCHAR Aad,
- ULONG AadLength,
- PUCHAR Nonce,
- ULONG NonceLength,
- PUCHAR AuthenticationField,
- ULONG AuthenticationFieldSize,
- ULONG LengthFieldSize
- );
- VOID
- Net80211pCcmComputeAuthenticationField (
- PNET80211_KEY Key,
- PUCHAR Message,
- ULONG MessageLength,
- PUCHAR Aad,
- ULONG AadLength,
- PUCHAR Nonce,
- ULONG NonceLength,
- PUCHAR AuthenticationField,
- ULONG AuthenticationFieldSize,
- ULONG LengthFieldSize
- );
- VOID
- Net80211pEapolCompletionRoutine (
- PVOID Context,
- KSTATUS Status
- );
- //
- // -------------------------------------------------------------------- Globals
- //
- //
- // ------------------------------------------------------------------ Functions
- //
- NET80211_API
- KSTATUS
- Net80211SetKey (
- PNET80211_LINK Link,
- PUCHAR KeyValue,
- ULONG KeyLength,
- ULONG KeyFlags,
- ULONG KeyId
- )
- /*++
- Routine Description:
- This routine sets the given key into the given network link. The 802.11
- networking library makes a local copy of all parameters.
- Arguments:
- Link - Supplies a pointer to the networking link to which the keys should
- be added.
- KeyValue - Supplies a pointer to the key value.
- KeyLength - Supplies the length of the key value, in bytes.
- KeyFlags - Supplies a bitmask of flags to describe the key. See
- NET80211_KEY_FLAG_* for definitions.
- KeyId - Supplies the ID of the key negotiated between this station and its
- peers and/or access point.
- Return Value:
- Status code.
- --*/
- {
- ULONG AllocationSize;
- PNET80211_ENCRYPTION Encryption;
- PNET80211_KEY Key;
- PNET80211_KEY OldKey;
- KSTATUS Status;
- OldKey = NULL;
- //
- // Make sure the key ID is valid and supported. The CCMP header only has
- // two bits for the key ID.
- //
- if (KeyId >= NET80211_MAX_KEY_COUNT) {
- return STATUS_INVALID_PARAMETER;
- }
- AllocationSize = sizeof(NET80211_KEY) + KeyLength - ANYSIZE_ARRAY;
- Key = MmAllocatePagedPool(AllocationSize, NET80211_ALLOCATION_TAG);
- if (Key == NULL) {
- Status = STATUS_INSUFFICIENT_RESOURCES;
- goto SetKeyEnd;
- }
- RtlZeroMemory(Key, AllocationSize);
- Key->Flags = KeyFlags;
- Key->Id = KeyId;
- Key->Length = KeyLength;
- RtlCopyMemory(Key->Value, KeyValue, KeyLength);
- //
- // Update the pointer in the array of keys for the active BSS.
- //
- KeAcquireQueuedLock(Link->Lock);
- if ((Link->ActiveBss == NULL) ||
- (Link->State != Net80211StateAssociated)) {
- Status = STATUS_NOT_READY;
- } else {
- Encryption = &(Link->ActiveBss->Encryption);
- OldKey = Encryption->Keys[KeyId];
- Encryption->Keys[KeyId] = Key;
- //
- // Update the key indices if this is a group key.
- //
- if ((KeyFlags & NET80211_KEY_FLAG_GLOBAL) != 0) {
- Encryption->GroupKeyIndex = KeyId;
- if ((Encryption->Flags &
- NET80211_ENCRYPTION_FLAG_USE_GROUP_CIPHER) != 0) {
- Encryption->PairwiseKeyIndex = KeyId;
- }
- }
- Status = STATUS_SUCCESS;
- }
- KeReleaseQueuedLock(Link->Lock);
- SetKeyEnd:
- if (!KSUCCESS(Status)) {
- if (Key != NULL) {
- Net80211pDestroyKey(Key);
- }
- }
- if (OldKey != NULL) {
- Net80211pDestroyKey(OldKey);
- }
- return Status;
- }
- VOID
- Net80211pDestroyKey (
- PNET80211_KEY Key
- )
- /*++
- Routine Description:
- This routine destroys the given 802.11 encryption key.
- Arguments:
- Key - Supplies a pointer to the key to be destroyed.
- Return Value:
- None.
- --*/
- {
- RtlZeroMemory(Key->Value, Key->Length);
- MmFreePagedPool(Key);
- return;
- }
- KSTATUS
- Net80211pInitializeEncryption (
- PNET80211_LINK Link,
- PNET80211_BSS_ENTRY Bss
- )
- /*++
- Routine Description:
- This routine initializes the 802.11 core to handle the completion of an
- advanced encryption handshake.
- Arguments:
- Link - Supplies a pointer to the 802.11 link establishing an ecrypted
- connection.
- Bss - Supplies a pointer to the BSS on which the encryption handshake will
- take place.
- Return Value:
- Status code.
- --*/
- {
- ULONG ApRsnSize;
- NETWORK_ADDRESS AuthenticatorAddress;
- EAPOL_CREATION_PARAMETERS Parameters;
- ULONG StationRsnSize;
- KSTATUS Status;
- ASSERT(Bss != NULL);
- //
- // The BSS is good to go if there is alreayd an EAPOL instance associated
- // with it.
- //
- if (Bss->EapolHandle != INVALID_HANDLE) {
- return STATUS_SUCCESS;
- }
- //
- // If there is no encryption required by the BSS or it is using the basic
- // authentication built into 802.11, there is no work to be done.
- //
- if ((Bss->Encryption.Pairwise == NetworkEncryptionNone) ||
- (Bss->Encryption.Pairwise == NetworkEncryptionWep)) {
- return STATUS_SUCCESS;
- }
- //
- // Set both the pairwise and group key indices to the default.
- //
- Bss->Encryption.PairwiseKeyIndex = NET80211_DEFAULT_ENCRYPTION_KEY;
- Bss->Encryption.GroupKeyIndex = NET80211_DEFAULT_ENCRYPTION_KEY;
- //
- // Otherwise, EAPOL must be invoked in order to derive the PTK.
- //
- ASSERT(Bss->Encryption.Pairwise == NetworkEncryptionWpa2Psk);
- RtlZeroMemory(&AuthenticatorAddress, sizeof(NETWORK_ADDRESS));
- AuthenticatorAddress.Domain = NetDomain80211;
- RtlCopyMemory(AuthenticatorAddress.Address,
- Bss->State.Bssid,
- NET80211_ADDRESS_SIZE);
- RtlZeroMemory(&Parameters, sizeof(EAPOL_CREATION_PARAMETERS));
- Parameters.Mode = EapolModeSupplicant;
- Parameters.NetworkLink = Link->NetworkLink;
- Parameters.Net80211Link = Link;
- Parameters.SupplicantAddress = &(Link->Properties.PhysicalAddress);
- Parameters.AuthenticatorAddress = &AuthenticatorAddress;
- Parameters.Ssid = NET80211_GET_ELEMENT_DATA(Bss->Ssid);
- Parameters.SsidLength = NET80211_GET_ELEMENT_LENGTH(Bss->Ssid);
- Parameters.Passphrase = Bss->Passphrase;
- Parameters.PassphraseLength = Bss->PassphraseLength;
- Parameters.SupplicantRsn = Bss->Encryption.StationRsn;
- StationRsnSize = NET80211_GET_ELEMENT_LENGTH(Bss->Encryption.StationRsn) +
- NET80211_ELEMENT_HEADER_SIZE;
- Parameters.SupplicantRsnSize = StationRsnSize;
- Parameters.AuthenticatorRsn = Bss->Encryption.ApRsn;
- ApRsnSize = NET80211_GET_ELEMENT_LENGTH(Bss->Encryption.ApRsn) +
- NET80211_ELEMENT_HEADER_SIZE;
- Parameters.AuthenticatorRsnSize = ApRsnSize;
- Parameters.CompletionRoutine = Net80211pEapolCompletionRoutine;
- Parameters.CompletionContext = Link;
- Status = Net80211pEapolCreateInstance(&Parameters, &(Bss->EapolHandle));
- if (!KSUCCESS(Status)) {
- return Status;
- }
- return STATUS_SUCCESS;
- }
- VOID
- Net80211pDestroyEncryption (
- PNET80211_BSS_ENTRY Bss
- )
- /*++
- Routine Description:
- This routine destroys the context used to handle encryption initialization.
- It is not necessary to keep this context once the encrypted state is
- reached.
- Arguments:
- Bss - Supplies a pointer to the BSS on which encryption initialization took
- place.
- Return Value:
- None.
- --*/
- {
- ASSERT(Bss != NULL);
- if (Bss->EapolHandle == INVALID_HANDLE) {
- return;
- }
- Net80211pEapolDestroyInstance(Bss->EapolHandle);
- Bss->EapolHandle = INVALID_HANDLE;
- return;
- }
- KSTATUS
- Net80211pEncryptPacket (
- PNET80211_LINK Link,
- PNET80211_BSS_ENTRY Bss,
- PNET_PACKET_BUFFER Packet
- )
- /*++
- Routine Description:
- This routine encrypts the given network packet's plaintext data. The
- supplied packet buffer is modified directly and should already include the
- full MPDU (i.e. the 802.11 headers should be present).
- Arguments:
- Link - Supplies a pointer to the 802.11 network link that owns the packet.
- Bss - Supplies a pointer to the BSS over which this packet should be sent.
- Packet - Supplies a pointer to the packet to encrypt.
- Return Value:
- Status code.
- --*/
- {
- NET80211_AAD Aad;
- PUCHAR AuthField;
- NET80211_CCM_NONCE CcmNonce;
- PNET80211_CCMP_HEADER CcmpHeader;
- PNET80211_DATA_FRAME_HEADER DataHeader;
- ULONG Index;
- PNET80211_KEY Key;
- ULONG KeyId;
- PUCHAR Message;
- ULONG MessageLength;
- PNET80211_DATA_FRAME_HEADER MovedDataHeader;
- ULONGLONG PacketNumber;
- PUCHAR PacketNumberArray;
- KSTATUS Status;
- //
- // Use the pairwise key by default.
- //
- KeyId = Bss->Encryption.PairwiseKeyIndex;
- Key = Bss->Encryption.Keys[KeyId];
- if ((Key == NULL) || ((Key->Flags & NET80211_KEY_FLAG_TRANSMIT) == 0)) {
- RtlDebugPrint("802.11: Failed to find valid key for transmit.\n");
- Status = STATUS_INVALID_CONFIGURATION;
- goto EncryptPacketEnd;
- }
- //
- // The start of the packet's valid data should point to the 802.11 header.
- //
- DataHeader = Packet->Buffer + Packet->DataOffset;
- AuthField = Packet->Buffer + Packet->FooterOffset;
- Message = (PUCHAR)(DataHeader + 1);
- MessageLength = AuthField - Message;
- Packet->FooterOffset += NET80211_CCMP_MIC_SIZE;
- //
- // Get a new packet number for the temporal key. The first 48-bits cannot
- // wrap. It's time to get a new temporal key if they do.
- //
- PacketNumber = RtlAtomicAdd64(&(Key->PacketNumber), 1);
- ASSERT(PacketNumber < 0x1000000000000);
- //
- // Construct the AAD based on the 802.11 header.
- //
- Aad.FrameControl = DataHeader->FrameControl &
- NET80211_AAD_FRAME_CONTROL_DEFAULT_MASK;
- RtlCopyMemory(Aad.Address1,
- DataHeader->ReceiverAddress,
- NET80211_ADDRESS_SIZE * 3);
- Aad.SequenceControl = DataHeader->SequenceControl &
- NET80211_AAD_SEQUENCE_CONTROL_MASK;
- //
- // Construct the CCM nonce. This is based on the packet number, which must
- // be stored in big-endian byte order.
- //
- CcmNonce.Flags = 0;
- RtlCopyMemory(CcmNonce.Address2, Aad.Address2, NET80211_ADDRESS_SIZE);
- PacketNumberArray = (PUCHAR)&PacketNumber;
- for (Index = 0; Index < NET80211_CCMP_PACKET_NUMBER_SIZE; Index += 1) {
- CcmNonce.PacketNumber[Index] =
- PacketNumberArray[NET80211_CCMP_PACKET_NUMBER_SIZE - 1 - Index];
- }
- //
- // Perform the CCM originator processing to produce the cipher text.
- //
- Status = Net80211pCcmEncrypt(Key,
- Message,
- MessageLength,
- (PUCHAR)&Aad,
- sizeof(NET80211_AAD),
- (PUCHAR)&CcmNonce,
- sizeof(NET80211_CCM_NONCE),
- AuthField,
- NET80211_CCMP_MIC_SIZE,
- NET80211_CCMP_LENGTH_FIELD_SIZE);
- if (!KSUCCESS(Status)) {
- goto EncryptPacketEnd;
- }
- //
- // Build the finalized encrypted packet. First move the 802.11 header
- // forward to make space for the CCMP header.
- //
- Packet->DataOffset -= sizeof(NET80211_CCMP_HEADER);
- MovedDataHeader = Packet->Buffer + Packet->DataOffset;
- RtlCopyMemory(MovedDataHeader,
- DataHeader,
- sizeof(NET80211_DATA_FRAME_HEADER));
- //
- // Construct the CCMP header using key ID 0.
- //
- CcmpHeader = (PNET80211_CCMP_HEADER)(MovedDataHeader + 1);
- CcmpHeader->Reserved = 0;
- CcmpHeader->Flags = NET80211_CCMP_FLAG_EXT_IV |
- ((KeyId << NET80211_CCMP_FLAG_KEY_ID_SHIFT) &
- NET80211_CCMP_FLAG_KEY_ID_MASK);
- NET80211_SET_CCMP_HEADER_PACKET_NUMBER(CcmpHeader, PacketNumber);
- //
- // The plaintext was encrypted in place and is right where it should be and
- // the MIC was placed where it should be in the footer. This packet is good
- // to go!
- //
- EncryptPacketEnd:
- return Status;
- }
- KSTATUS
- Net80211pDecryptPacket (
- PNET80211_LINK Link,
- PNET80211_BSS_ENTRY Bss,
- PNET_PACKET_BUFFER Packet
- )
- /*++
- Routine Description:
- This routine decrypts the given network packet's ciphertext. The supplied
- packet buffer is modified directly and should contain the full encrypted
- MPDU, including the 802.11 headers.
- Arguments:
- Link - Supplies a pointer to the 802.11 network link that owns the packet.
- Bss - Supplies a pointer to the BSS over which this packet was received.
- Packet - Supplies a pointer to the packet to decrypt.
- Return Value:
- Status code.
- --*/
- {
- NET80211_AAD Aad;
- PUCHAR AuthField;
- NET80211_CCM_NONCE CcmNonce;
- PNET80211_CCMP_HEADER CcmpHeader;
- PNET80211_DATA_FRAME_HEADER DataHeader;
- ULONG Index;
- PNET80211_KEY Key;
- ULONG KeyId;
- PUCHAR Message;
- ULONG MessageLength;
- ULONGLONG PacketNumber;
- PUCHAR PacketNumberArray;
- KSTATUS Status;
- //
- // The start of the packet's valid data should point to the 802.11 header.
- //
- DataHeader = Packet->Buffer + Packet->DataOffset;
- CcmpHeader = (PNET80211_CCMP_HEADER)(DataHeader + 1);
- Message = (PUCHAR)(CcmpHeader + 1);
- AuthField = Packet->Buffer + Packet->FooterOffset - NET80211_CCMP_MIC_SIZE;
- MessageLength = AuthField - Message;
- //
- // Get the correct key to use for the decryption.
- //
- KeyId = (CcmpHeader->Flags & NET80211_CCMP_FLAG_KEY_ID_MASK) >>
- NET80211_CCMP_FLAG_KEY_ID_SHIFT;
- Key = Bss->Encryption.Keys[KeyId];
- if (Key == NULL) {
- Status = STATUS_UNSUCCESSFUL;
- goto DecryptPacketEnd;
- }
- //
- // Construct the AAD based on the 802.11 header.
- //
- Aad.FrameControl = DataHeader->FrameControl &
- NET80211_AAD_FRAME_CONTROL_DEFAULT_MASK;
- RtlCopyMemory(Aad.Address1,
- DataHeader->ReceiverAddress,
- NET80211_ADDRESS_SIZE * 3);
- Aad.SequenceControl = DataHeader->SequenceControl &
- NET80211_AAD_SEQUENCE_CONTROL_MASK;
- //
- // Construct the CCM nonce. This is based on the packet number retrieved
- // from the CCMP header.
- //
- CcmNonce.Flags = 0;
- RtlCopyMemory(CcmNonce.Address2, Aad.Address2, NET80211_ADDRESS_SIZE);
- NET80211_GET_CCMP_HEADER_PACKET_NUMBER(CcmpHeader, PacketNumber);
- PacketNumberArray = (PUCHAR)&PacketNumber;
- for (Index = 0; Index < NET80211_CCMP_PACKET_NUMBER_SIZE; Index += 1) {
- CcmNonce.PacketNumber[Index] =
- PacketNumberArray[NET80211_CCMP_PACKET_NUMBER_SIZE - 1 - Index];
- }
- //
- // Perform the CCM originator processing to produce the cipher text.
- //
- Status = Net80211pCcmDecrypt(Key,
- Message,
- MessageLength,
- (PUCHAR)&Aad,
- sizeof(NET80211_AAD),
- (PUCHAR)&CcmNonce,
- sizeof(NET80211_CCM_NONCE),
- AuthField,
- NET80211_CCMP_MIC_SIZE,
- NET80211_CCMP_LENGTH_FIELD_SIZE);
- if (!KSUCCESS(Status)) {
- RtlDebugPrint("802.11: Failed to decrypt packet 0x%08x for link "
- "0x%08x.\n",
- Packet,
- Link);
- goto DecryptPacketEnd;
- }
- //
- // Compare the packet number to the replay counter and toss the packet if
- // if it's number is too low.
- //
- if (PacketNumber <= Key->ReplayCounter) {
- Status = STATUS_TOO_LATE;
- goto DecryptPacketEnd;
- }
- Key->ReplayCounter = PacketNumber;
- //
- // Move past both the encryption header and the data header. Fully
- // recreating a decrypted packet is not useful unless tooling requires it.
- //
- Packet->DataOffset += sizeof(NET80211_DATA_FRAME_HEADER) +
- sizeof(NET80211_CCMP_HEADER);
- Packet->FooterOffset -= NET80211_CCMP_MIC_SIZE;
- DecryptPacketEnd:
- return Status;
- }
- //
- // --------------------------------------------------------- Internal Functions
- //
- KSTATUS
- Net80211pCcmEncrypt (
- PNET80211_KEY Key,
- PUCHAR Message,
- ULONG MessageLength,
- PUCHAR Aad,
- ULONG AadLength,
- PUCHAR Nonce,
- ULONG NonceLength,
- PUCHAR AuthenticationField,
- ULONG AuthenticationFieldSize,
- ULONG LengthFieldSize
- )
- /*++
- Routine Description:
- This routine performs CCM originator processing on the given plaintext
- message, updating it in place with the corresponding encrypted text.
- Arguments:
- Key - Supplies a pointer to the temporal key to use for CCM encryption.
- Message - Supplies a pointer to the plaintext message that is to be
- encrypted in place.
- MessageLength - Supplies the length of the plaintext message, in bytes.
- Aad - Supplies a pointer to the additional authentication data (AAD).
- AadLength - Supplies the length of the AAD, in bytes.
- Nonce - Supplies a pointer to the nonce value to use for CCM encryption.
- NonceLength - Supplies the length of the nonce value, in bytes.
- AuthenticationField - Supplies a pointer to a buffer that is to receive
- the authentication bytes for the CCM encryption (i.e. the MIC).
- AuthenticationFieldSize - Supplies the length of the authentication field
- in bytes.
- LengthFieldSize - Supplies the size, in bytes, of the CCM length field.
- Return Value:
- Status code.
- --*/
- {
- AES_CONTEXT AesContext;
- UCHAR AesInitializationVector[AES_INITIALIZATION_VECTOR_SIZE];
- ULONG BlockCount;
- UCHAR BlockIn[AES_BLOCK_SIZE];
- ULONG BytesRemaining;
- ULONG Index;
- ASSERT(NonceLength >= (AES_BLOCK_SIZE - 1 - LengthFieldSize));
- //
- // Compute the authentication field and store it in the input block, as it
- // used as the first input for the encryption process.
- //
- Net80211pCcmComputeAuthenticationField(Key,
- Message,
- MessageLength,
- Aad,
- AadLength,
- Nonce,
- NonceLength,
- BlockIn,
- AuthenticationFieldSize,
- LengthFieldSize);
- //
- // Initialize the AES context in counter mode.
- //
- AesInitializationVector[0] = LengthFieldSize - 1;
- RtlCopyMemory(AesInitializationVector + 1,
- Nonce,
- (AES_BLOCK_SIZE - 1 - LengthFieldSize));
- for (Index = AES_INITIALIZATION_VECTOR_SIZE - LengthFieldSize;
- Index < AES_INITIALIZATION_VECTOR_SIZE;
- Index += 1) {
- AesInitializationVector[Index] = 0;
- }
- CyAesInitialize(&AesContext,
- AesModeCtr128,
- Key->Value,
- AesInitializationVector);
- //
- // The first block encrypts the authentication field to get the
- // authentication value.
- //
- CyAesCtrEncrypt(&AesContext, BlockIn, BlockIn, AES_BLOCK_SIZE);
- RtlCopyMemory(AuthenticationField, BlockIn, AuthenticationFieldSize);
- //
- // The message is now encrypted with the rest of the counter sequence.
- // Because of how the counter algorithm works, this can actually be done
- // in-place.
- //
- BlockCount = MessageLength / AES_BLOCK_SIZE;
- BytesRemaining = MessageLength % AES_BLOCK_SIZE;
- CyAesCtrEncrypt(&AesContext, Message, Message, BlockCount * AES_BLOCK_SIZE);
- //
- // If there are leftover bytes, then copy them to a local block, perform
- // the encryption and then copy back the ciphertext.
- //
- if (BytesRemaining != 0) {
- RtlCopyMemory(BlockIn,
- Message + (BlockCount * AES_BLOCK_SIZE),
- BytesRemaining);
- CyAesCtrEncrypt(&AesContext, BlockIn, BlockIn, AES_BLOCK_SIZE);
- RtlCopyMemory(Message + (BlockCount * AES_BLOCK_SIZE),
- BlockIn,
- BytesRemaining);
- }
- return STATUS_SUCCESS;
- }
- KSTATUS
- Net80211pCcmDecrypt (
- PNET80211_KEY Key,
- PUCHAR Message,
- ULONG MessageLength,
- PUCHAR Aad,
- ULONG AadLength,
- PUCHAR Nonce,
- ULONG NonceLength,
- PUCHAR AuthenticationField,
- ULONG AuthenticationFieldSize,
- ULONG LengthFieldSize
- )
- /*++
- Routine Description:
- This routine performs CCM recipient processing on the given ciphertext
- message, updating it in place with the corresponding decrypted text.
- Arguments:
- Key - Supplies a pointer to the temporal key to use for CCM decryption.
- Message - Supplies a pointer to the encrypted message that is to be
- decrypted in place.
- MessageLength - Supplies the length of the ciphertext message, in bytes.
- Aad - Supplies a pointer to the additional authentication data (AAD).
- AadLength - Supplies the length of the AAD, in bytes.
- Nonce - Supplies a pointer to the nonce value to use for CCM encryption.
- NonceLength - Supplies the length of the nonce value, in bytes.
- AuthenticationField - Supplies a pointer to the authentication bytes for
- the CCM decryption (i.e. the MIC).
- AuthenticationFieldSize - Supplies the length of the authentication field
- in bytes.
- LengthFieldSize - Supplies the size, in bytes, of the CCM length field.
- Return Value:
- Status code.
- --*/
- {
- AES_CONTEXT AesContext;
- UCHAR AesInitializationVector[AES_INITIALIZATION_VECTOR_SIZE];
- ULONG BlockCount;
- UCHAR BlockIn[AES_BLOCK_SIZE];
- ULONG BytesRemaining;
- ULONG Index;
- UCHAR LocalAuthenticationField[NET80211_CCM_MAX_AUTHENTICATION_FIELD_SIZE];
- BOOL Match;
- ASSERT(NonceLength >= (AES_BLOCK_SIZE - 1 - LengthFieldSize));
- ASSERT(MessageLength < MAX_USHORT);
- //
- // Initialize the AES context in CTR mode.
- //
- AesInitializationVector[0] = LengthFieldSize - 1;
- RtlCopyMemory(AesInitializationVector + 1,
- Nonce,
- (AES_BLOCK_SIZE - 1 - LengthFieldSize));
- for (Index = AES_INITIALIZATION_VECTOR_SIZE - LengthFieldSize;
- Index < AES_INITIALIZATION_VECTOR_SIZE;
- Index += 1) {
- AesInitializationVector[Index] = 0;
- }
- CyAesInitialize(&AesContext,
- AesModeCtr128,
- Key->Value,
- AesInitializationVector);
- //
- // The authentication value passes through the counter decryption first
- // to recompute the authentification field. The uninitialized portions of
- // the supplied block do not need to be zeroed.
- //
- RtlCopyMemory(BlockIn, AuthenticationField, AuthenticationFieldSize);
- CyAesCtrDecrypt(&AesContext, BlockIn, BlockIn, AES_BLOCK_SIZE);
- RtlCopyMemory(AuthenticationField, BlockIn, AuthenticationFieldSize);
- //
- // The message is now decrypted with the rest of the counter sequence.
- // Because of how the counter algorithm works, this can actually be done
- // in-place.
- //
- BlockCount = MessageLength / AES_BLOCK_SIZE;
- BytesRemaining = MessageLength % AES_BLOCK_SIZE;
- CyAesCtrDecrypt(&AesContext, Message, Message, BlockCount * AES_BLOCK_SIZE);
- //
- // If there are leftover bytes, then copy them to a local block, perform
- // the decryption and then copy back the ciphertext.
- //
- if (BytesRemaining != 0) {
- RtlCopyMemory(BlockIn,
- Message + (BlockCount * AES_BLOCK_SIZE),
- BytesRemaining);
- CyAesCtrDecrypt(&AesContext, BlockIn, BlockIn, AES_BLOCK_SIZE);
- RtlCopyMemory(Message + (BlockCount * AES_BLOCK_SIZE),
- BlockIn,
- BytesRemaining);
- }
- //
- // Compute the authentication field for the now decrypted message and
- // compare it to the recomputed authentication field.
- //
- Net80211pCcmComputeAuthenticationField(Key,
- Message,
- MessageLength,
- Aad,
- AadLength,
- Nonce,
- NonceLength,
- LocalAuthenticationField,
- AuthenticationFieldSize,
- LengthFieldSize);
- Match = RtlCompareMemory(AuthenticationField,
- LocalAuthenticationField,
- AuthenticationFieldSize);
- if (Match != FALSE) {
- return STATUS_SUCCESS;
- }
- RtlDebugPrint("802.11: CCM decryption found a bad authentication value!\n");
- return STATUS_UNSUCCESSFUL;
- }
- VOID
- Net80211pCcmComputeAuthenticationField (
- PNET80211_KEY Key,
- PUCHAR Message,
- ULONG MessageLength,
- PUCHAR Aad,
- ULONG AadLength,
- PUCHAR Nonce,
- ULONG NonceLength,
- PUCHAR AuthenticationField,
- ULONG AuthenticationFieldSize,
- ULONG LengthFieldSize
- )
- /*++
- Routine Description:
- This routine computes the authentication field for the given plaintext
- message, additional information, and key. This is used for both encryption
- and decryption as a MIC at the end of the packet.
- Arguments:
- Key - Supplies a pointer to the temporal key to use for CBC encryption.
- Message - Supplies a pointer to the plaintext message that is to be
- used as input to compute the authentication field.
- MessageLength - Supplies the length of the plaintext message, in bytes.
- Aad - Supplies a pointer to the additional authentication data (AAD).
- AadLength - Supplies the length of the AAD, in bytes.
- Nonce - Supplies a pointer to the nonce value to use.
- NonceLength - Supplies the length of the nonce value, in bytes.
- AuthenticationField - Supplies a pointer to a buffer that is to receive
- the authentication field for the CCM encryption (i.e. the MIC).
- AuthenticationFieldSize - Supplies the length of the authentication field
- in bytes.
- LengthFieldSize - Supplies the size, in bytes, of the CCM length field.
- Return Value:
- None.
- --*/
- {
- ULONG AadIndex;
- AES_CONTEXT AesContext;
- UCHAR BlockIn[AES_BLOCK_SIZE];
- ULONG BlockIndex;
- UCHAR BlockOut[AES_BLOCK_SIZE];
- ULONG BytesRemaining;
- ULONG BytesThisRound;
- UCHAR Flags;
- ULONG Index;
- ULONG MessageIndex;
- ASSERT(AuthenticationFieldSize <=
- NET80211_CCM_MAX_AUTHENTICATION_FIELD_SIZE);
- ASSERT((LengthFieldSize >= NET80211_CCM_MIN_LENGTH_FIELD_SIZE) &&
- (LengthFieldSize <= NET80211_CCM_MAX_LENGTH_FIELD_SIZE));
- //
- // Initialize the AES context for CBC mode.
- //
- CyAesInitialize(&AesContext, AesModeCbc128, Key->Value, NULL);
- //
- // Initialize the first block based on the length field size,
- // authentication field size, nonce, and message length.
- //
- Flags = ((AuthenticationFieldSize - 2) / 2) <<
- NET80211_CCM_FLAG_AUTHENTICATION_FIELD_SHIFT;
- Flags |= (LengthFieldSize - 1) << NET80211_CCM_FLAG_LENGTH_SHIFT;
- if (AadLength != 0) {
- Flags |= NET80211_CCM_FLAG_AAD;
- }
- BlockIn[0] = Flags;
- RtlCopyMemory(BlockIn + 1, Nonce, (AES_BLOCK_SIZE - 1 - LengthFieldSize));
- for (Index = LengthFieldSize; Index > 0; Index -= 1) {
- BlockIn[AES_BLOCK_SIZE - Index] = ((PUCHAR)&MessageLength)[Index - 1];
- }
- //
- // Encrypt the first output block. Because this is a CBC algorithm that can
- // be called multiple times, the AES library internally remembers the last
- // out block and will XOR that with the next supplied in block before
- // encrypting.
- //
- CyAesCbcEncrypt(&AesContext, BlockIn, BlockOut, AES_BLOCK_SIZE);
- //
- // If an AAD was supplied, then that makes up the second block.
- //
- if (AadLength != 0) {
- if (AadLength <= NET80211_CCM_AAD_MAX_SHORT_LENGTH) {
- *((PUSHORT)&(BlockIn[0])) = CPU_TO_NETWORK16(AadLength);
- BlockIndex = 2;
- } else {
- *((PUSHORT)&(BlockIn[0])) = NET80211_CCM_AAD_LONG_ENCODING;
- *((PULONG)&(BlockIn[2])) = CPU_TO_NETWORK32(AadLength);
- BlockIndex = 6;
- }
- AadIndex = 0;
- BytesRemaining = AadLength;
- while (BytesRemaining != 0) {
- BytesThisRound = BytesRemaining;
- if (BytesThisRound > (AES_BLOCK_SIZE - BlockIndex)) {
- BytesThisRound = AES_BLOCK_SIZE - BlockIndex;
- }
- RtlCopyMemory(BlockIn + BlockIndex,
- Aad + AadIndex,
- BytesThisRound);
- BlockIndex += BytesThisRound;
- AadIndex += BytesThisRound;
- BytesRemaining -= BytesThisRound;
- //
- // Pad the block with zeros if necessary.
- //
- if (BlockIndex != AES_BLOCK_SIZE) {
- ASSERT(BytesRemaining == 0);
- RtlZeroMemory(BlockIn + BlockIndex,
- AES_BLOCK_SIZE - BlockIndex);
- }
- //
- // Encrypt this block. It will get XOR'd with the previous block,
- // which is stored internally to the AES context.
- //
- CyAesCbcEncrypt(&AesContext, BlockIn, BlockOut, AES_BLOCK_SIZE);
- BlockIndex = 0;
- }
- }
- //
- // Fold the message into the computation. This must not modify the contents
- // of the message buffer. As a result, it is done block-by-block.
- //
- MessageIndex = 0;
- BytesRemaining = MessageLength;
- while (BytesRemaining != 0) {
- BytesThisRound = BytesRemaining;
- if (BytesThisRound > (AES_BLOCK_SIZE - BlockIndex)) {
- BytesThisRound = AES_BLOCK_SIZE - BlockIndex;
- }
- RtlCopyMemory(BlockIn, Message + MessageIndex, BytesThisRound);
- MessageIndex += BytesThisRound;
- BytesRemaining -= BytesThisRound;
- //
- // Pad the block with zeros if necessary.
- //
- if (BytesThisRound != AES_BLOCK_SIZE) {
- ASSERT(BytesRemaining == 0);
- RtlZeroMemory(BlockIn + BytesThisRound,
- AES_BLOCK_SIZE - BytesThisRound);
- }
- //
- // Encrypt this block. It will get XOR'd with the previous block, which
- // is stored internally to the AES context.
- //
- CyAesCbcEncrypt(&AesContext, BlockIn, BlockOut, AES_BLOCK_SIZE);
- }
- //
- // The out block now stores the authentication field.
- //
- RtlCopyMemory(AuthenticationField, BlockOut, AuthenticationFieldSize);
- return;
- }
- VOID
- Net80211pEapolCompletionRoutine (
- PVOID Context,
- KSTATUS Status
- )
- /*++
- Routine Description:
- This routine is called when an EAPOL exchange completes. It is supplied by
- the creator of the EAPOL instance.
- Arguments:
- Context - Supplies a pointer to the context supplied by the creator of the
- EAPOL instance.
- Status - Supplies the completion status of the EAPOL exchange.
- Return Value:
- None.
- --*/
- {
- PNET80211_LINK Link;
- NET80211_STATE State;
- Link = (PNET80211_LINK)Context;
- State = Net80211StateEncrypted;
- if (!KSUCCESS(Status)) {
- RtlDebugPrint("802.11: EAPOL failed with status 0x%08x\n", Status);
- State = Net80211StateInitialized;
- }
- Net80211pSetState(Link, State);
- return;
- }
|