2011-08-17 04:38:08 +02:00
|
|
|
/** @file
|
2021-06-08 14:12:50 +02:00
|
|
|
This file is for Challenge-Handshake Authentication Protocol (CHAP)
|
|
|
|
Configuration.
|
2011-08-17 04:38:08 +02:00
|
|
|
|
2018-06-27 15:12:32 +02:00
|
|
|
Copyright (c) 2004 - 2018, Intel Corporation. All rights reserved.<BR>
|
2019-04-04 01:06:13 +02:00
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
2011-08-17 04:38:08 +02:00
|
|
|
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include "IScsiImpl.h"
|
|
|
|
|
|
|
|
/**
|
2020-02-07 02:07:54 +01:00
|
|
|
Initiator calculates its own expected hash value.
|
2018-06-27 15:12:32 +02:00
|
|
|
|
2011-08-17 04:38:08 +02:00
|
|
|
@param[in] ChapIdentifier iSCSI CHAP identifier sent by authenticator.
|
|
|
|
@param[in] ChapSecret iSCSI CHAP secret of the authenticator.
|
|
|
|
@param[in] SecretLength The length of iSCSI CHAP secret.
|
|
|
|
@param[in] ChapChallenge The challenge message sent by authenticator.
|
|
|
|
@param[in] ChallengeLength The length of iSCSI CHAP challenge message.
|
|
|
|
@param[out] ChapResponse The calculation of the expected hash value.
|
2018-06-27 15:12:32 +02:00
|
|
|
|
2021-06-08 14:12:50 +02:00
|
|
|
@retval EFI_SUCCESS The expected hash value was calculatedly
|
|
|
|
successfully.
|
|
|
|
@retval EFI_PROTOCOL_ERROR The length of the secret should be at least
|
|
|
|
the length of the hash value for the hashing
|
|
|
|
algorithm chosen.
|
2011-08-17 04:38:08 +02:00
|
|
|
@retval EFI_PROTOCOL_ERROR MD5 hash operation fail.
|
|
|
|
@retval EFI_OUT_OF_RESOURCES Fail to allocate resource to complete MD5.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
IScsiCHAPCalculateResponse (
|
|
|
|
IN UINT32 ChapIdentifier,
|
|
|
|
IN CHAR8 *ChapSecret,
|
|
|
|
IN UINT32 SecretLength,
|
|
|
|
IN UINT8 *ChapChallenge,
|
|
|
|
IN UINT32 ChallengeLength,
|
|
|
|
OUT UINT8 *ChapResponse
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINTN Md5ContextSize;
|
|
|
|
VOID *Md5Ctx;
|
|
|
|
CHAR8 IdByte[1];
|
|
|
|
EFI_STATUS Status;
|
|
|
|
|
|
|
|
if (SecretLength < ISCSI_CHAP_SECRET_MIN_LEN) {
|
|
|
|
return EFI_PROTOCOL_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
Md5ContextSize = Md5GetContextSize ();
|
|
|
|
Md5Ctx = AllocatePool (Md5ContextSize);
|
|
|
|
if (Md5Ctx == NULL) {
|
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = EFI_PROTOCOL_ERROR;
|
|
|
|
|
|
|
|
if (!Md5Init (Md5Ctx)) {
|
|
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Hash Identifier - Only calculate 1 byte data (RFC1994)
|
|
|
|
//
|
|
|
|
IdByte[0] = (CHAR8) ChapIdentifier;
|
|
|
|
if (!Md5Update (Md5Ctx, IdByte, 1)) {
|
|
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Hash Secret
|
|
|
|
//
|
|
|
|
if (!Md5Update (Md5Ctx, ChapSecret, SecretLength)) {
|
|
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Hash Challenge received from Target
|
|
|
|
//
|
|
|
|
if (!Md5Update (Md5Ctx, ChapChallenge, ChallengeLength)) {
|
|
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Md5Final (Md5Ctx, ChapResponse)) {
|
|
|
|
Status = EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
Exit:
|
|
|
|
FreePool (Md5Ctx);
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-02-07 02:07:54 +01:00
|
|
|
The initiator checks the CHAP response replied by target against its own
|
2018-06-27 15:12:32 +02:00
|
|
|
calculation of the expected hash value.
|
|
|
|
|
|
|
|
@param[in] AuthData iSCSI CHAP authentication data.
|
|
|
|
@param[in] TargetResponse The response from target.
|
2011-08-17 04:38:08 +02:00
|
|
|
|
2021-06-08 14:12:50 +02:00
|
|
|
@retval EFI_SUCCESS The response from target passed
|
|
|
|
authentication.
|
|
|
|
@retval EFI_SECURITY_VIOLATION The response from target was not expected
|
|
|
|
value.
|
2011-08-17 04:38:08 +02:00
|
|
|
@retval Others Other errors as indicated.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
IScsiCHAPAuthTarget (
|
|
|
|
IN ISCSI_CHAP_AUTH_DATA *AuthData,
|
|
|
|
IN UINT8 *TargetResponse
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINT32 SecretSize;
|
|
|
|
UINT8 VerifyRsp[ISCSI_CHAP_RSP_LEN];
|
|
|
|
|
|
|
|
Status = EFI_SUCCESS;
|
|
|
|
|
|
|
|
SecretSize = (UINT32) AsciiStrLen (AuthData->AuthConfig->ReverseCHAPSecret);
|
|
|
|
Status = IScsiCHAPCalculateResponse (
|
|
|
|
AuthData->OutIdentifier,
|
|
|
|
AuthData->AuthConfig->ReverseCHAPSecret,
|
|
|
|
SecretSize,
|
|
|
|
AuthData->OutChallenge,
|
NetworkPkg/IScsiDxe: clean up "ISCSI_CHAP_AUTH_DATA.OutChallengeLength"
The "ISCSI_CHAP_AUTH_DATA.OutChallenge" field is declared as a UINT8 array
with ISCSI_CHAP_AUTH_MAX_LEN (1024) elements. However, when the challenge
is generated and formatted, only ISCSI_CHAP_RSP_LEN (16) octets are used
in the array.
Change the array size to ISCSI_CHAP_RSP_LEN, and remove the (now unused)
ISCSI_CHAP_AUTH_MAX_LEN macro.
Remove the "ISCSI_CHAP_AUTH_DATA.OutChallengeLength" field, which is
superfluous too.
Most importantly, explain in a new comment *why* tying the challenge size
to the digest size (ISCSI_CHAP_RSP_LEN) has always made sense. (See also
Linux kernel commit 19f5f88ed779, "scsi: target: iscsi: tie the challenge
length to the hash digest size", 2019-11-06.) For sure, the motivation
that the new comment now explains has always been there, and has always
been the same, for IScsiDxe; it's just that now we spell it out too.
No change in peer-visible behavior.
Cc: Jiaxin Wu <jiaxin.wu@intel.com>
Cc: Maciej Rabeda <maciej.rabeda@linux.intel.com>
Cc: Philippe Mathieu-Daudé <philmd@redhat.com>
Cc: Siyuan Fu <siyuan.fu@intel.com>
Ref: https://bugzilla.tianocore.org/show_bug.cgi?id=3356
Signed-off-by: Laszlo Ersek <lersek@redhat.com>
Reviewed-by: Philippe Mathieu-Daudé <philmd@redhat.com>
Reviewed-by: Maciej Rabeda <maciej.rabeda@linux.intel.com>
Message-Id: <20210608121259.32451-4-lersek@redhat.com>
2021-06-08 14:12:52 +02:00
|
|
|
ISCSI_CHAP_RSP_LEN, // ChallengeLength
|
2011-08-17 04:38:08 +02:00
|
|
|
VerifyRsp
|
|
|
|
);
|
|
|
|
|
|
|
|
if (CompareMem (VerifyRsp, TargetResponse, ISCSI_CHAP_RSP_LEN) != 0) {
|
|
|
|
Status = EFI_SECURITY_VIOLATION;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
This function checks the received iSCSI Login Response during the security
|
|
|
|
negotiation stage.
|
|
|
|
|
|
|
|
@param[in] Conn The iSCSI connection.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS The Login Response passed the CHAP validation.
|
|
|
|
@retval EFI_OUT_OF_RESOURCES Failed to allocate memory.
|
|
|
|
@retval EFI_PROTOCOL_ERROR Some kind of protocol error occurred.
|
|
|
|
@retval Others Other errors as indicated.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
IScsiCHAPOnRspReceived (
|
|
|
|
IN ISCSI_CONNECTION *Conn
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
ISCSI_SESSION *Session;
|
|
|
|
ISCSI_CHAP_AUTH_DATA *AuthData;
|
|
|
|
CHAR8 *Value;
|
|
|
|
UINT8 *Data;
|
|
|
|
UINT32 Len;
|
|
|
|
LIST_ENTRY *KeyValueList;
|
|
|
|
UINTN Algorithm;
|
|
|
|
CHAR8 *Identifier;
|
|
|
|
CHAR8 *Challenge;
|
|
|
|
CHAR8 *Name;
|
|
|
|
CHAR8 *Response;
|
|
|
|
UINT8 TargetRsp[ISCSI_CHAP_RSP_LEN];
|
|
|
|
UINT32 RspLen;
|
|
|
|
UINTN Result;
|
|
|
|
|
|
|
|
ASSERT (Conn->CurrentStage == ISCSI_SECURITY_NEGOTIATION);
|
|
|
|
ASSERT (Conn->RspQue.BufNum != 0);
|
|
|
|
|
|
|
|
Session = Conn->Session;
|
|
|
|
AuthData = &Session->AuthData.CHAP;
|
|
|
|
Len = Conn->RspQue.BufSize;
|
|
|
|
Data = AllocateZeroPool (Len);
|
|
|
|
if (Data == NULL) {
|
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
//
|
|
|
|
// Copy the data in case the data spans over multiple PDUs.
|
|
|
|
//
|
|
|
|
NetbufQueCopy (&Conn->RspQue, 0, Len, Data);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Build the key-value list from the data segment of the Login Response.
|
|
|
|
//
|
|
|
|
KeyValueList = IScsiBuildKeyValueList ((CHAR8 *) Data, Len);
|
|
|
|
if (KeyValueList == NULL) {
|
|
|
|
Status = EFI_OUT_OF_RESOURCES;
|
|
|
|
goto ON_EXIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = EFI_PROTOCOL_ERROR;
|
|
|
|
|
|
|
|
switch (Conn->AuthStep) {
|
|
|
|
case ISCSI_AUTH_INITIAL:
|
|
|
|
//
|
|
|
|
// The first Login Response.
|
|
|
|
//
|
2021-06-08 14:12:50 +02:00
|
|
|
Value = IScsiGetValueByKeyFromList (
|
|
|
|
KeyValueList,
|
|
|
|
ISCSI_KEY_TARGET_PORTAL_GROUP_TAG
|
|
|
|
);
|
2011-08-17 04:38:08 +02:00
|
|
|
if (Value == NULL) {
|
|
|
|
goto ON_EXIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
Result = IScsiNetNtoi (Value);
|
|
|
|
if (Result > 0xFFFF) {
|
|
|
|
goto ON_EXIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
Session->TargetPortalGroupTag = (UINT16) Result;
|
|
|
|
|
2021-06-08 14:12:50 +02:00
|
|
|
Value = IScsiGetValueByKeyFromList (
|
|
|
|
KeyValueList,
|
|
|
|
ISCSI_KEY_AUTH_METHOD
|
|
|
|
);
|
2011-08-17 04:38:08 +02:00
|
|
|
if (Value == NULL) {
|
|
|
|
goto ON_EXIT;
|
|
|
|
}
|
|
|
|
//
|
2021-06-08 14:12:50 +02:00
|
|
|
// Initiator mandates CHAP authentication but target replies without
|
|
|
|
// "CHAP", or initiator suggets "None" but target replies with some kind of
|
|
|
|
// auth method.
|
2011-08-17 04:38:08 +02:00
|
|
|
//
|
|
|
|
if (Session->AuthType == ISCSI_AUTH_TYPE_NONE) {
|
|
|
|
if (AsciiStrCmp (Value, ISCSI_KEY_VALUE_NONE) != 0) {
|
|
|
|
goto ON_EXIT;
|
|
|
|
}
|
|
|
|
} else if (Session->AuthType == ISCSI_AUTH_TYPE_CHAP) {
|
|
|
|
if (AsciiStrCmp (Value, ISCSI_AUTH_METHOD_CHAP) != 0) {
|
|
|
|
goto ON_EXIT;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
goto ON_EXIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Transit to CHAP step one.
|
|
|
|
//
|
|
|
|
Conn->AuthStep = ISCSI_CHAP_STEP_ONE;
|
|
|
|
Status = EFI_SUCCESS;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ISCSI_CHAP_STEP_TWO:
|
|
|
|
//
|
|
|
|
// The Target replies with CHAP_A=<A> CHAP_I=<I> CHAP_C=<C>
|
|
|
|
//
|
2021-06-08 14:12:50 +02:00
|
|
|
Value = IScsiGetValueByKeyFromList (
|
|
|
|
KeyValueList,
|
|
|
|
ISCSI_KEY_CHAP_ALGORITHM
|
|
|
|
);
|
2011-08-17 04:38:08 +02:00
|
|
|
if (Value == NULL) {
|
|
|
|
goto ON_EXIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
Algorithm = IScsiNetNtoi (Value);
|
|
|
|
if (Algorithm != ISCSI_CHAP_ALGORITHM_MD5) {
|
|
|
|
//
|
|
|
|
// Unsupported algorithm is chosen by target.
|
|
|
|
//
|
|
|
|
goto ON_EXIT;
|
|
|
|
}
|
|
|
|
|
2021-06-08 14:12:50 +02:00
|
|
|
Identifier = IScsiGetValueByKeyFromList (
|
|
|
|
KeyValueList,
|
|
|
|
ISCSI_KEY_CHAP_IDENTIFIER
|
|
|
|
);
|
2011-08-17 04:38:08 +02:00
|
|
|
if (Identifier == NULL) {
|
|
|
|
goto ON_EXIT;
|
|
|
|
}
|
|
|
|
|
2021-06-08 14:12:50 +02:00
|
|
|
Challenge = IScsiGetValueByKeyFromList (
|
|
|
|
KeyValueList,
|
|
|
|
ISCSI_KEY_CHAP_CHALLENGE
|
|
|
|
);
|
2011-08-17 04:38:08 +02:00
|
|
|
if (Challenge == NULL) {
|
|
|
|
goto ON_EXIT;
|
|
|
|
}
|
|
|
|
//
|
|
|
|
// Process the CHAP identifier and CHAP Challenge from Target.
|
|
|
|
// Calculate Response value.
|
2018-06-27 15:12:32 +02:00
|
|
|
//
|
2011-08-17 04:38:08 +02:00
|
|
|
Result = IScsiNetNtoi (Identifier);
|
|
|
|
if (Result > 0xFF) {
|
|
|
|
goto ON_EXIT;
|
2018-06-27 15:12:32 +02:00
|
|
|
}
|
|
|
|
|
2011-08-17 04:38:08 +02:00
|
|
|
AuthData->InIdentifier = (UINT32) Result;
|
2021-06-08 14:12:51 +02:00
|
|
|
AuthData->InChallengeLength = (UINT32) sizeof (AuthData->InChallenge);
|
NetworkPkg/IScsiDxe: check IScsiHexToBin() return values
IScsiDxe (that is, the initiator) receives two hex-encoded strings from
the iSCSI target:
- CHAP_C, where the target challenges the initiator,
- CHAP_R, where the target answers the challenge from the initiator (in
case the initiator wants mutual authentication).
Accordingly, we have two IScsiHexToBin() call sites:
- At the CHAP_C decoding site, check whether the decoding succeeds. The
decoded buffer ("AuthData->InChallenge") can accommodate 1024 bytes,
which is a permissible restriction on the target, per
<https://tools.ietf.org/html/rfc7143#section-12.1.3>. Shorter challenges
from the target are acceptable.
- At the CHAP_R decoding site, enforce that the decoding both succeed, and
provide exactly ISCSI_CHAP_RSP_LEN bytes. CHAP_R contains the digest
calculated by the target, therefore it must be of fixed size. We may
only call IScsiCHAPAuthTarget() if "TargetRsp" has been fully populated.
Cc: Jiaxin Wu <jiaxin.wu@intel.com>
Cc: Maciej Rabeda <maciej.rabeda@linux.intel.com>
Cc: Philippe Mathieu-Daudé <philmd@redhat.com>
Cc: Siyuan Fu <siyuan.fu@intel.com>
Ref: https://bugzilla.tianocore.org/show_bug.cgi?id=3356
Signed-off-by: Laszlo Ersek <lersek@redhat.com>
Reviewed-by: Philippe Mathieu-Daudé <philmd@redhat.com>
Reviewed-by: Maciej Rabeda <maciej.rabeda@linux.intel.com>
Message-Id: <20210608121259.32451-11-lersek@redhat.com>
2021-06-08 14:12:59 +02:00
|
|
|
Status = IScsiHexToBin (
|
|
|
|
(UINT8 *) AuthData->InChallenge,
|
|
|
|
&AuthData->InChallengeLength,
|
|
|
|
Challenge
|
|
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
Status = EFI_PROTOCOL_ERROR;
|
|
|
|
goto ON_EXIT;
|
|
|
|
}
|
2011-08-17 04:38:08 +02:00
|
|
|
Status = IScsiCHAPCalculateResponse (
|
|
|
|
AuthData->InIdentifier,
|
|
|
|
AuthData->AuthConfig->CHAPSecret,
|
|
|
|
(UINT32) AsciiStrLen (AuthData->AuthConfig->CHAPSecret),
|
|
|
|
AuthData->InChallenge,
|
|
|
|
AuthData->InChallengeLength,
|
|
|
|
AuthData->CHAPResponse
|
|
|
|
);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Transit to next step.
|
|
|
|
//
|
|
|
|
Conn->AuthStep = ISCSI_CHAP_STEP_THREE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ISCSI_CHAP_STEP_THREE:
|
|
|
|
//
|
|
|
|
// One way CHAP authentication and the target would like to
|
|
|
|
// authenticate us.
|
|
|
|
//
|
|
|
|
Status = EFI_SUCCESS;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ISCSI_CHAP_STEP_FOUR:
|
|
|
|
ASSERT (AuthData->AuthConfig->CHAPType == ISCSI_CHAP_MUTUAL);
|
|
|
|
//
|
|
|
|
// The forth step, CHAP_N=<N> CHAP_R=<R> is received from Target.
|
|
|
|
//
|
|
|
|
Name = IScsiGetValueByKeyFromList (KeyValueList, ISCSI_KEY_CHAP_NAME);
|
|
|
|
if (Name == NULL) {
|
|
|
|
goto ON_EXIT;
|
|
|
|
}
|
|
|
|
|
2021-06-08 14:12:50 +02:00
|
|
|
Response = IScsiGetValueByKeyFromList (
|
|
|
|
KeyValueList,
|
|
|
|
ISCSI_KEY_CHAP_RESPONSE
|
|
|
|
);
|
2011-08-17 04:38:08 +02:00
|
|
|
if (Response == NULL) {
|
|
|
|
goto ON_EXIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
RspLen = ISCSI_CHAP_RSP_LEN;
|
NetworkPkg/IScsiDxe: check IScsiHexToBin() return values
IScsiDxe (that is, the initiator) receives two hex-encoded strings from
the iSCSI target:
- CHAP_C, where the target challenges the initiator,
- CHAP_R, where the target answers the challenge from the initiator (in
case the initiator wants mutual authentication).
Accordingly, we have two IScsiHexToBin() call sites:
- At the CHAP_C decoding site, check whether the decoding succeeds. The
decoded buffer ("AuthData->InChallenge") can accommodate 1024 bytes,
which is a permissible restriction on the target, per
<https://tools.ietf.org/html/rfc7143#section-12.1.3>. Shorter challenges
from the target are acceptable.
- At the CHAP_R decoding site, enforce that the decoding both succeed, and
provide exactly ISCSI_CHAP_RSP_LEN bytes. CHAP_R contains the digest
calculated by the target, therefore it must be of fixed size. We may
only call IScsiCHAPAuthTarget() if "TargetRsp" has been fully populated.
Cc: Jiaxin Wu <jiaxin.wu@intel.com>
Cc: Maciej Rabeda <maciej.rabeda@linux.intel.com>
Cc: Philippe Mathieu-Daudé <philmd@redhat.com>
Cc: Siyuan Fu <siyuan.fu@intel.com>
Ref: https://bugzilla.tianocore.org/show_bug.cgi?id=3356
Signed-off-by: Laszlo Ersek <lersek@redhat.com>
Reviewed-by: Philippe Mathieu-Daudé <philmd@redhat.com>
Reviewed-by: Maciej Rabeda <maciej.rabeda@linux.intel.com>
Message-Id: <20210608121259.32451-11-lersek@redhat.com>
2021-06-08 14:12:59 +02:00
|
|
|
Status = IScsiHexToBin (TargetRsp, &RspLen, Response);
|
|
|
|
if (EFI_ERROR (Status) || RspLen != ISCSI_CHAP_RSP_LEN) {
|
|
|
|
Status = EFI_PROTOCOL_ERROR;
|
|
|
|
goto ON_EXIT;
|
|
|
|
}
|
2011-08-17 04:38:08 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// Check the CHAP Name and Response replied by Target.
|
|
|
|
//
|
|
|
|
Status = IScsiCHAPAuthTarget (AuthData, TargetRsp);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
ON_EXIT:
|
|
|
|
|
|
|
|
if (KeyValueList != NULL) {
|
|
|
|
IScsiFreeKeyValueList (KeyValueList);
|
2018-06-27 15:12:32 +02:00
|
|
|
}
|
2011-08-17 04:38:08 +02:00
|
|
|
|
|
|
|
FreePool (Data);
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
This function fills the CHAP authentication information into the login PDU
|
|
|
|
during the security negotiation stage in the iSCSI connection login.
|
|
|
|
|
|
|
|
@param[in] Conn The iSCSI connection.
|
|
|
|
@param[in, out] Pdu The PDU to send out.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS All check passed and the phase-related CHAP
|
2021-06-08 14:12:50 +02:00
|
|
|
authentication info is filled into the iSCSI
|
|
|
|
PDU.
|
2011-08-17 04:38:08 +02:00
|
|
|
@retval EFI_OUT_OF_RESOURCES Failed to allocate memory.
|
|
|
|
@retval EFI_PROTOCOL_ERROR Some kind of protocol error occurred.
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
IScsiCHAPToSendReq (
|
|
|
|
IN ISCSI_CONNECTION *Conn,
|
|
|
|
IN OUT NET_BUF *Pdu
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
ISCSI_SESSION *Session;
|
|
|
|
ISCSI_LOGIN_REQUEST *LoginReq;
|
|
|
|
ISCSI_CHAP_AUTH_DATA *AuthData;
|
|
|
|
CHAR8 *Value;
|
|
|
|
CHAR8 ValueStr[256];
|
|
|
|
CHAR8 *Response;
|
|
|
|
UINT32 RspLen;
|
|
|
|
CHAR8 *Challenge;
|
|
|
|
UINT32 ChallengeLen;
|
2021-06-08 14:12:55 +02:00
|
|
|
EFI_STATUS BinToHexStatus;
|
2011-08-17 04:38:08 +02:00
|
|
|
|
|
|
|
ASSERT (Conn->CurrentStage == ISCSI_SECURITY_NEGOTIATION);
|
|
|
|
|
|
|
|
Session = Conn->Session;
|
|
|
|
AuthData = &Session->AuthData.CHAP;
|
|
|
|
LoginReq = (ISCSI_LOGIN_REQUEST *) NetbufGetByte (Pdu, 0, 0);
|
2012-04-10 04:27:05 +02:00
|
|
|
if (LoginReq == NULL) {
|
|
|
|
return EFI_PROTOCOL_ERROR;
|
|
|
|
}
|
2011-08-17 04:38:08 +02:00
|
|
|
Status = EFI_SUCCESS;
|
|
|
|
|
|
|
|
RspLen = 2 * ISCSI_CHAP_RSP_LEN + 3;
|
|
|
|
Response = AllocateZeroPool (RspLen);
|
|
|
|
if (Response == NULL) {
|
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
ChallengeLen = 2 * ISCSI_CHAP_RSP_LEN + 3;
|
|
|
|
Challenge = AllocateZeroPool (ChallengeLen);
|
|
|
|
if (Challenge == NULL) {
|
|
|
|
FreePool (Response);
|
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (Conn->AuthStep) {
|
|
|
|
case ISCSI_AUTH_INITIAL:
|
|
|
|
//
|
|
|
|
// It's the initial Login Request. Fill in the key=value pairs mandatory
|
|
|
|
// for the initial Login Request.
|
|
|
|
//
|
2021-06-08 14:12:50 +02:00
|
|
|
IScsiAddKeyValuePair (
|
|
|
|
Pdu,
|
|
|
|
ISCSI_KEY_INITIATOR_NAME,
|
|
|
|
mPrivate->InitiatorName
|
|
|
|
);
|
2011-08-17 04:38:08 +02:00
|
|
|
IScsiAddKeyValuePair (Pdu, ISCSI_KEY_SESSION_TYPE, "Normal");
|
|
|
|
IScsiAddKeyValuePair (
|
|
|
|
Pdu,
|
|
|
|
ISCSI_KEY_TARGET_NAME,
|
|
|
|
Session->ConfigData->SessionConfigData.TargetName
|
|
|
|
);
|
|
|
|
|
|
|
|
if (Session->AuthType == ISCSI_AUTH_TYPE_NONE) {
|
|
|
|
Value = ISCSI_KEY_VALUE_NONE;
|
|
|
|
ISCSI_SET_FLAG (LoginReq, ISCSI_LOGIN_REQ_PDU_FLAG_TRANSIT);
|
|
|
|
} else {
|
|
|
|
Value = ISCSI_AUTH_METHOD_CHAP;
|
|
|
|
}
|
|
|
|
|
|
|
|
IScsiAddKeyValuePair (Pdu, ISCSI_KEY_AUTH_METHOD, Value);
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ISCSI_CHAP_STEP_ONE:
|
|
|
|
//
|
2021-06-08 14:12:50 +02:00
|
|
|
// First step, send the Login Request with CHAP_A=<A1,A2...> key-value
|
|
|
|
// pair.
|
2011-08-17 04:38:08 +02:00
|
|
|
//
|
|
|
|
AsciiSPrint (ValueStr, sizeof (ValueStr), "%d", ISCSI_CHAP_ALGORITHM_MD5);
|
|
|
|
IScsiAddKeyValuePair (Pdu, ISCSI_KEY_CHAP_ALGORITHM, ValueStr);
|
|
|
|
|
|
|
|
Conn->AuthStep = ISCSI_CHAP_STEP_TWO;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ISCSI_CHAP_STEP_THREE:
|
|
|
|
//
|
|
|
|
// Third step, send the Login Request with CHAP_N=<N> CHAP_R=<R> or
|
|
|
|
// CHAP_N=<N> CHAP_R=<R> CHAP_I=<I> CHAP_C=<C> if target authentication is
|
|
|
|
// required too.
|
|
|
|
//
|
|
|
|
// CHAP_N=<N>
|
|
|
|
//
|
2021-06-08 14:12:50 +02:00
|
|
|
IScsiAddKeyValuePair (
|
|
|
|
Pdu,
|
|
|
|
ISCSI_KEY_CHAP_NAME,
|
|
|
|
(CHAR8 *) &AuthData->AuthConfig->CHAPName
|
|
|
|
);
|
2011-08-17 04:38:08 +02:00
|
|
|
//
|
|
|
|
// CHAP_R=<R>
|
|
|
|
//
|
2021-06-08 14:12:55 +02:00
|
|
|
BinToHexStatus = IScsiBinToHex (
|
|
|
|
(UINT8 *) AuthData->CHAPResponse,
|
|
|
|
ISCSI_CHAP_RSP_LEN,
|
|
|
|
Response,
|
|
|
|
&RspLen
|
|
|
|
);
|
|
|
|
ASSERT_EFI_ERROR (BinToHexStatus);
|
2011-08-17 04:38:08 +02:00
|
|
|
IScsiAddKeyValuePair (Pdu, ISCSI_KEY_CHAP_RESPONSE, Response);
|
|
|
|
|
|
|
|
if (AuthData->AuthConfig->CHAPType == ISCSI_CHAP_MUTUAL) {
|
|
|
|
//
|
|
|
|
// CHAP_I=<I>
|
|
|
|
//
|
|
|
|
IScsiGenRandom ((UINT8 *) &AuthData->OutIdentifier, 1);
|
|
|
|
AsciiSPrint (ValueStr, sizeof (ValueStr), "%d", AuthData->OutIdentifier);
|
|
|
|
IScsiAddKeyValuePair (Pdu, ISCSI_KEY_CHAP_IDENTIFIER, ValueStr);
|
|
|
|
//
|
|
|
|
// CHAP_C=<C>
|
|
|
|
//
|
|
|
|
IScsiGenRandom ((UINT8 *) AuthData->OutChallenge, ISCSI_CHAP_RSP_LEN);
|
2021-06-08 14:12:55 +02:00
|
|
|
BinToHexStatus = IScsiBinToHex (
|
|
|
|
(UINT8 *) AuthData->OutChallenge,
|
|
|
|
ISCSI_CHAP_RSP_LEN,
|
|
|
|
Challenge,
|
|
|
|
&ChallengeLen
|
|
|
|
);
|
|
|
|
ASSERT_EFI_ERROR (BinToHexStatus);
|
2011-08-17 04:38:08 +02:00
|
|
|
IScsiAddKeyValuePair (Pdu, ISCSI_KEY_CHAP_CHALLENGE, Challenge);
|
|
|
|
|
|
|
|
Conn->AuthStep = ISCSI_CHAP_STEP_FOUR;
|
|
|
|
}
|
|
|
|
//
|
|
|
|
// Set the stage transition flag.
|
|
|
|
//
|
|
|
|
ISCSI_SET_FLAG (LoginReq, ISCSI_LOGIN_REQ_PDU_FLAG_TRANSIT);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
Status = EFI_PROTOCOL_ERROR;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
FreePool (Response);
|
|
|
|
FreePool (Challenge);
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|