2008-02-27 20:34:43 -05:00
|
|
|
/*
|
|
|
|
* hostapd / EAP-SIM (RFC 4186)
|
2012-09-01 11:51:09 -04:00
|
|
|
* Copyright (c) 2005-2012, Jouni Malinen <j@w1.fi>
|
2008-02-27 20:34:43 -05:00
|
|
|
*
|
2012-02-11 09:46:35 -05:00
|
|
|
* This software may be distributed under the terms of the BSD license.
|
|
|
|
* See README for more details.
|
2008-02-27 20:34:43 -05:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "includes.h"
|
|
|
|
|
|
|
|
#include "common.h"
|
2010-11-23 18:05:20 -05:00
|
|
|
#include "crypto/random.h"
|
2008-02-27 20:34:43 -05:00
|
|
|
#include "eap_server/eap_i.h"
|
|
|
|
#include "eap_common/eap_sim_common.h"
|
|
|
|
#include "eap_server/eap_sim_db.h"
|
|
|
|
|
|
|
|
|
|
|
|
struct eap_sim_data {
|
|
|
|
u8 mk[EAP_SIM_MK_LEN];
|
|
|
|
u8 nonce_mt[EAP_SIM_NONCE_MT_LEN];
|
|
|
|
u8 nonce_s[EAP_SIM_NONCE_S_LEN];
|
|
|
|
u8 k_aut[EAP_SIM_K_AUT_LEN];
|
|
|
|
u8 k_encr[EAP_SIM_K_ENCR_LEN];
|
|
|
|
u8 msk[EAP_SIM_KEYING_DATA_LEN];
|
|
|
|
u8 emsk[EAP_EMSK_LEN];
|
|
|
|
u8 kc[EAP_SIM_MAX_CHAL][EAP_SIM_KC_LEN];
|
|
|
|
u8 sres[EAP_SIM_MAX_CHAL][EAP_SIM_SRES_LEN];
|
|
|
|
u8 rand[EAP_SIM_MAX_CHAL][GSM_RAND_LEN];
|
|
|
|
int num_chal;
|
|
|
|
enum {
|
|
|
|
START, CHALLENGE, REAUTH, NOTIFICATION, SUCCESS, FAILURE
|
|
|
|
} state;
|
|
|
|
char *next_pseudonym;
|
|
|
|
char *next_reauth_id;
|
|
|
|
u16 counter;
|
|
|
|
struct eap_sim_reauth *reauth;
|
|
|
|
u16 notification;
|
|
|
|
int use_result_ind;
|
2012-09-01 11:51:09 -04:00
|
|
|
int start_round;
|
2012-09-01 12:10:33 -04:00
|
|
|
char permanent[20]; /* Permanent username */
|
2008-02-27 20:34:43 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
static const char * eap_sim_state_txt(int state)
|
|
|
|
{
|
|
|
|
switch (state) {
|
|
|
|
case START:
|
|
|
|
return "START";
|
|
|
|
case CHALLENGE:
|
|
|
|
return "CHALLENGE";
|
|
|
|
case REAUTH:
|
|
|
|
return "REAUTH";
|
|
|
|
case SUCCESS:
|
|
|
|
return "SUCCESS";
|
|
|
|
case FAILURE:
|
|
|
|
return "FAILURE";
|
|
|
|
case NOTIFICATION:
|
|
|
|
return "NOTIFICATION";
|
|
|
|
default:
|
|
|
|
return "Unknown?!";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void eap_sim_state(struct eap_sim_data *data, int state)
|
|
|
|
{
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: %s -> %s",
|
|
|
|
eap_sim_state_txt(data->state),
|
|
|
|
eap_sim_state_txt(state));
|
|
|
|
data->state = state;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void * eap_sim_init(struct eap_sm *sm)
|
|
|
|
{
|
|
|
|
struct eap_sim_data *data;
|
|
|
|
|
|
|
|
if (sm->eap_sim_db_priv == NULL) {
|
|
|
|
wpa_printf(MSG_WARNING, "EAP-SIM: eap_sim_db not configured");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
data = os_zalloc(sizeof(*data));
|
|
|
|
if (data == NULL)
|
|
|
|
return NULL;
|
|
|
|
data->state = START;
|
|
|
|
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void eap_sim_reset(struct eap_sm *sm, void *priv)
|
|
|
|
{
|
|
|
|
struct eap_sim_data *data = priv;
|
|
|
|
os_free(data->next_pseudonym);
|
|
|
|
os_free(data->next_reauth_id);
|
2014-06-29 18:48:41 -04:00
|
|
|
bin_clear_free(data, sizeof(*data));
|
2008-02-27 20:34:43 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static struct wpabuf * eap_sim_build_start(struct eap_sm *sm,
|
|
|
|
struct eap_sim_data *data, u8 id)
|
|
|
|
{
|
|
|
|
struct eap_sim_msg *msg;
|
|
|
|
u8 ver[2];
|
|
|
|
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Generating Start");
|
|
|
|
msg = eap_sim_msg_init(EAP_CODE_REQUEST, id, EAP_TYPE_SIM,
|
|
|
|
EAP_SIM_SUBTYPE_START);
|
2012-09-01 11:51:09 -04:00
|
|
|
data->start_round++;
|
|
|
|
if (data->start_round == 1) {
|
2008-02-27 20:34:43 -05:00
|
|
|
/*
|
|
|
|
* RFC 4186, Chap. 4.2.4 recommends that identity from EAP is
|
|
|
|
* ignored and the SIM/Start is used to request the identity.
|
|
|
|
*/
|
|
|
|
wpa_printf(MSG_DEBUG, " AT_ANY_ID_REQ");
|
|
|
|
eap_sim_msg_add(msg, EAP_SIM_AT_ANY_ID_REQ, 0, NULL, 0);
|
2012-09-01 11:51:09 -04:00
|
|
|
} else if (data->start_round > 3) {
|
|
|
|
/* Cannot use more than three rounds of Start messages */
|
2012-11-11 06:15:49 -05:00
|
|
|
eap_sim_msg_free(msg);
|
2012-09-01 11:51:09 -04:00
|
|
|
return NULL;
|
2012-09-01 17:38:39 -04:00
|
|
|
} else if (data->start_round == 0) {
|
|
|
|
/*
|
|
|
|
* This is a special case that is used to recover from
|
|
|
|
* AT_COUNTER_TOO_SMALL during re-authentication. Since we
|
|
|
|
* already know the identity of the peer, there is no need to
|
|
|
|
* request any identity in this case.
|
|
|
|
*/
|
2012-09-01 11:51:09 -04:00
|
|
|
} else if (sm->identity && sm->identity_len > 0 &&
|
|
|
|
sm->identity[0] == EAP_SIM_REAUTH_ID_PREFIX) {
|
|
|
|
/* Reauth id may have expired - try fullauth */
|
|
|
|
wpa_printf(MSG_DEBUG, " AT_FULLAUTH_ID_REQ");
|
|
|
|
eap_sim_msg_add(msg, EAP_SIM_AT_FULLAUTH_ID_REQ, 0, NULL, 0);
|
|
|
|
} else {
|
|
|
|
wpa_printf(MSG_DEBUG, " AT_PERMANENT_ID_REQ");
|
|
|
|
eap_sim_msg_add(msg, EAP_SIM_AT_PERMANENT_ID_REQ, 0, NULL, 0);
|
2008-02-27 20:34:43 -05:00
|
|
|
}
|
|
|
|
wpa_printf(MSG_DEBUG, " AT_VERSION_LIST");
|
|
|
|
ver[0] = 0;
|
|
|
|
ver[1] = EAP_SIM_VERSION;
|
|
|
|
eap_sim_msg_add(msg, EAP_SIM_AT_VERSION_LIST, sizeof(ver),
|
|
|
|
ver, sizeof(ver));
|
2014-06-29 11:46:29 -04:00
|
|
|
return eap_sim_msg_finish(msg, EAP_TYPE_SIM, NULL, NULL, 0);
|
2008-02-27 20:34:43 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int eap_sim_build_encr(struct eap_sm *sm, struct eap_sim_data *data,
|
|
|
|
struct eap_sim_msg *msg, u16 counter,
|
|
|
|
const u8 *nonce_s)
|
|
|
|
{
|
|
|
|
os_free(data->next_pseudonym);
|
2012-02-16 16:27:01 -05:00
|
|
|
if (nonce_s == NULL) {
|
|
|
|
data->next_pseudonym =
|
2012-05-02 13:45:01 -04:00
|
|
|
eap_sim_db_get_next_pseudonym(sm->eap_sim_db_priv,
|
|
|
|
EAP_SIM_DB_SIM);
|
2012-02-16 16:27:01 -05:00
|
|
|
} else {
|
|
|
|
/* Do not update pseudonym during re-authentication */
|
|
|
|
data->next_pseudonym = NULL;
|
|
|
|
}
|
2008-02-27 20:34:43 -05:00
|
|
|
os_free(data->next_reauth_id);
|
|
|
|
if (data->counter <= EAP_SIM_MAX_FAST_REAUTHS) {
|
|
|
|
data->next_reauth_id =
|
2012-05-02 13:45:01 -04:00
|
|
|
eap_sim_db_get_next_reauth_id(sm->eap_sim_db_priv,
|
|
|
|
EAP_SIM_DB_SIM);
|
2008-02-27 20:34:43 -05:00
|
|
|
} else {
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Max fast re-authentication "
|
|
|
|
"count exceeded - force full authentication");
|
|
|
|
data->next_reauth_id = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data->next_pseudonym == NULL && data->next_reauth_id == NULL &&
|
|
|
|
counter == 0 && nonce_s == NULL)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
wpa_printf(MSG_DEBUG, " AT_IV");
|
|
|
|
wpa_printf(MSG_DEBUG, " AT_ENCR_DATA");
|
|
|
|
eap_sim_msg_add_encr_start(msg, EAP_SIM_AT_IV, EAP_SIM_AT_ENCR_DATA);
|
|
|
|
|
|
|
|
if (counter > 0) {
|
|
|
|
wpa_printf(MSG_DEBUG, " *AT_COUNTER (%u)", counter);
|
|
|
|
eap_sim_msg_add(msg, EAP_SIM_AT_COUNTER, counter, NULL, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nonce_s) {
|
|
|
|
wpa_printf(MSG_DEBUG, " *AT_NONCE_S");
|
|
|
|
eap_sim_msg_add(msg, EAP_SIM_AT_NONCE_S, 0, nonce_s,
|
|
|
|
EAP_SIM_NONCE_S_LEN);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data->next_pseudonym) {
|
|
|
|
wpa_printf(MSG_DEBUG, " *AT_NEXT_PSEUDONYM (%s)",
|
|
|
|
data->next_pseudonym);
|
|
|
|
eap_sim_msg_add(msg, EAP_SIM_AT_NEXT_PSEUDONYM,
|
|
|
|
os_strlen(data->next_pseudonym),
|
|
|
|
(u8 *) data->next_pseudonym,
|
|
|
|
os_strlen(data->next_pseudonym));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data->next_reauth_id) {
|
|
|
|
wpa_printf(MSG_DEBUG, " *AT_NEXT_REAUTH_ID (%s)",
|
|
|
|
data->next_reauth_id);
|
|
|
|
eap_sim_msg_add(msg, EAP_SIM_AT_NEXT_REAUTH_ID,
|
|
|
|
os_strlen(data->next_reauth_id),
|
|
|
|
(u8 *) data->next_reauth_id,
|
|
|
|
os_strlen(data->next_reauth_id));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (eap_sim_msg_add_encr_end(msg, data->k_encr, EAP_SIM_AT_PADDING)) {
|
|
|
|
wpa_printf(MSG_WARNING, "EAP-SIM: Failed to encrypt "
|
|
|
|
"AT_ENCR_DATA");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static struct wpabuf * eap_sim_build_challenge(struct eap_sm *sm,
|
|
|
|
struct eap_sim_data *data,
|
|
|
|
u8 id)
|
|
|
|
{
|
|
|
|
struct eap_sim_msg *msg;
|
|
|
|
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Generating Challenge");
|
|
|
|
msg = eap_sim_msg_init(EAP_CODE_REQUEST, id, EAP_TYPE_SIM,
|
|
|
|
EAP_SIM_SUBTYPE_CHALLENGE);
|
|
|
|
wpa_printf(MSG_DEBUG, " AT_RAND");
|
|
|
|
eap_sim_msg_add(msg, EAP_SIM_AT_RAND, 0, (u8 *) data->rand,
|
|
|
|
data->num_chal * GSM_RAND_LEN);
|
|
|
|
|
|
|
|
if (eap_sim_build_encr(sm, data, msg, 0, NULL)) {
|
|
|
|
eap_sim_msg_free(msg);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sm->eap_sim_aka_result_ind) {
|
|
|
|
wpa_printf(MSG_DEBUG, " AT_RESULT_IND");
|
|
|
|
eap_sim_msg_add(msg, EAP_SIM_AT_RESULT_IND, 0, NULL, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
wpa_printf(MSG_DEBUG, " AT_MAC");
|
|
|
|
eap_sim_msg_add_mac(msg, EAP_SIM_AT_MAC);
|
2014-06-29 11:46:29 -04:00
|
|
|
return eap_sim_msg_finish(msg, EAP_TYPE_SIM, data->k_aut,
|
|
|
|
data->nonce_mt, EAP_SIM_NONCE_MT_LEN);
|
2008-02-27 20:34:43 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static struct wpabuf * eap_sim_build_reauth(struct eap_sm *sm,
|
|
|
|
struct eap_sim_data *data, u8 id)
|
|
|
|
{
|
|
|
|
struct eap_sim_msg *msg;
|
|
|
|
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Generating Re-authentication");
|
|
|
|
|
2010-11-23 18:05:20 -05:00
|
|
|
if (random_get_bytes(data->nonce_s, EAP_SIM_NONCE_S_LEN))
|
2008-02-27 20:34:43 -05:00
|
|
|
return NULL;
|
|
|
|
wpa_hexdump_key(MSG_MSGDUMP, "EAP-SIM: NONCE_S",
|
|
|
|
data->nonce_s, EAP_SIM_NONCE_S_LEN);
|
|
|
|
|
|
|
|
eap_sim_derive_keys(data->mk, data->k_encr, data->k_aut, data->msk,
|
|
|
|
data->emsk);
|
|
|
|
eap_sim_derive_keys_reauth(data->counter, sm->identity,
|
|
|
|
sm->identity_len, data->nonce_s, data->mk,
|
|
|
|
data->msk, data->emsk);
|
|
|
|
|
|
|
|
msg = eap_sim_msg_init(EAP_CODE_REQUEST, id, EAP_TYPE_SIM,
|
|
|
|
EAP_SIM_SUBTYPE_REAUTHENTICATION);
|
|
|
|
|
|
|
|
if (eap_sim_build_encr(sm, data, msg, data->counter, data->nonce_s)) {
|
|
|
|
eap_sim_msg_free(msg);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sm->eap_sim_aka_result_ind) {
|
|
|
|
wpa_printf(MSG_DEBUG, " AT_RESULT_IND");
|
|
|
|
eap_sim_msg_add(msg, EAP_SIM_AT_RESULT_IND, 0, NULL, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
wpa_printf(MSG_DEBUG, " AT_MAC");
|
|
|
|
eap_sim_msg_add_mac(msg, EAP_SIM_AT_MAC);
|
2014-06-29 11:46:29 -04:00
|
|
|
return eap_sim_msg_finish(msg, EAP_TYPE_SIM, data->k_aut, NULL, 0);
|
2008-02-27 20:34:43 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static struct wpabuf * eap_sim_build_notification(struct eap_sm *sm,
|
|
|
|
struct eap_sim_data *data,
|
|
|
|
u8 id)
|
|
|
|
{
|
|
|
|
struct eap_sim_msg *msg;
|
|
|
|
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Generating Notification");
|
|
|
|
msg = eap_sim_msg_init(EAP_CODE_REQUEST, id, EAP_TYPE_SIM,
|
|
|
|
EAP_SIM_SUBTYPE_NOTIFICATION);
|
|
|
|
wpa_printf(MSG_DEBUG, " AT_NOTIFICATION (%d)", data->notification);
|
|
|
|
eap_sim_msg_add(msg, EAP_SIM_AT_NOTIFICATION, data->notification,
|
|
|
|
NULL, 0);
|
|
|
|
if (data->use_result_ind) {
|
|
|
|
if (data->reauth) {
|
|
|
|
wpa_printf(MSG_DEBUG, " AT_IV");
|
|
|
|
wpa_printf(MSG_DEBUG, " AT_ENCR_DATA");
|
|
|
|
eap_sim_msg_add_encr_start(msg, EAP_SIM_AT_IV,
|
|
|
|
EAP_SIM_AT_ENCR_DATA);
|
|
|
|
wpa_printf(MSG_DEBUG, " *AT_COUNTER (%u)",
|
|
|
|
data->counter);
|
|
|
|
eap_sim_msg_add(msg, EAP_SIM_AT_COUNTER, data->counter,
|
|
|
|
NULL, 0);
|
|
|
|
|
|
|
|
if (eap_sim_msg_add_encr_end(msg, data->k_encr,
|
|
|
|
EAP_SIM_AT_PADDING)) {
|
|
|
|
wpa_printf(MSG_WARNING, "EAP-SIM: Failed to "
|
|
|
|
"encrypt AT_ENCR_DATA");
|
|
|
|
eap_sim_msg_free(msg);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
wpa_printf(MSG_DEBUG, " AT_MAC");
|
|
|
|
eap_sim_msg_add_mac(msg, EAP_SIM_AT_MAC);
|
|
|
|
}
|
2014-06-29 11:46:29 -04:00
|
|
|
return eap_sim_msg_finish(msg, EAP_TYPE_SIM, data->k_aut, NULL, 0);
|
2008-02-27 20:34:43 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static struct wpabuf * eap_sim_buildReq(struct eap_sm *sm, void *priv, u8 id)
|
|
|
|
{
|
|
|
|
struct eap_sim_data *data = priv;
|
|
|
|
|
|
|
|
switch (data->state) {
|
|
|
|
case START:
|
|
|
|
return eap_sim_build_start(sm, data, id);
|
|
|
|
case CHALLENGE:
|
|
|
|
return eap_sim_build_challenge(sm, data, id);
|
|
|
|
case REAUTH:
|
|
|
|
return eap_sim_build_reauth(sm, data, id);
|
|
|
|
case NOTIFICATION:
|
|
|
|
return eap_sim_build_notification(sm, data, id);
|
|
|
|
default:
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Unknown state %d in "
|
|
|
|
"buildReq", data->state);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static Boolean eap_sim_check(struct eap_sm *sm, void *priv,
|
|
|
|
struct wpabuf *respData)
|
|
|
|
{
|
|
|
|
const u8 *pos;
|
|
|
|
size_t len;
|
|
|
|
|
|
|
|
pos = eap_hdr_validate(EAP_VENDOR_IETF, EAP_TYPE_SIM, respData, &len);
|
|
|
|
if (pos == NULL || len < 3) {
|
|
|
|
wpa_printf(MSG_INFO, "EAP-SIM: Invalid frame");
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2012-09-01 18:26:05 -04:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static Boolean eap_sim_unexpected_subtype(struct eap_sim_data *data,
|
|
|
|
u8 subtype)
|
|
|
|
{
|
2008-02-27 20:34:43 -05:00
|
|
|
if (subtype == EAP_SIM_SUBTYPE_CLIENT_ERROR)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
switch (data->state) {
|
|
|
|
case START:
|
|
|
|
if (subtype != EAP_SIM_SUBTYPE_START) {
|
|
|
|
wpa_printf(MSG_INFO, "EAP-SIM: Unexpected response "
|
|
|
|
"subtype %d", subtype);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case CHALLENGE:
|
|
|
|
if (subtype != EAP_SIM_SUBTYPE_CHALLENGE) {
|
|
|
|
wpa_printf(MSG_INFO, "EAP-SIM: Unexpected response "
|
|
|
|
"subtype %d", subtype);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case REAUTH:
|
|
|
|
if (subtype != EAP_SIM_SUBTYPE_REAUTHENTICATION) {
|
|
|
|
wpa_printf(MSG_INFO, "EAP-SIM: Unexpected response "
|
|
|
|
"subtype %d", subtype);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case NOTIFICATION:
|
|
|
|
if (subtype != EAP_SIM_SUBTYPE_NOTIFICATION) {
|
|
|
|
wpa_printf(MSG_INFO, "EAP-SIM: Unexpected response "
|
|
|
|
"subtype %d", subtype);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
wpa_printf(MSG_INFO, "EAP-SIM: Unexpected state (%d) for "
|
|
|
|
"processing a response", data->state);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int eap_sim_supported_ver(struct eap_sim_data *data, int version)
|
|
|
|
{
|
|
|
|
return version == EAP_SIM_VERSION;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void eap_sim_process_start(struct eap_sm *sm,
|
|
|
|
struct eap_sim_data *data,
|
|
|
|
struct wpabuf *respData,
|
|
|
|
struct eap_sim_attrs *attr)
|
|
|
|
{
|
|
|
|
size_t identity_len;
|
|
|
|
u8 ver_list[2];
|
2012-09-01 11:56:35 -04:00
|
|
|
u8 *new_identity;
|
2012-09-01 12:10:33 -04:00
|
|
|
char *username;
|
2008-02-27 20:34:43 -05:00
|
|
|
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Receive start response");
|
|
|
|
|
2012-09-01 17:38:39 -04:00
|
|
|
if (data->start_round == 0) {
|
|
|
|
/*
|
|
|
|
* Special case for AT_COUNTER_TOO_SMALL recovery - no identity
|
|
|
|
* was requested since we already know it.
|
|
|
|
*/
|
|
|
|
goto skip_id_update;
|
|
|
|
}
|
|
|
|
|
2012-09-01 11:56:35 -04:00
|
|
|
/*
|
|
|
|
* We always request identity in SIM/Start, so the peer is required to
|
|
|
|
* have replied with one.
|
|
|
|
*/
|
|
|
|
if (!attr->identity || attr->identity_len == 0) {
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Peer did not provide any "
|
|
|
|
"identity");
|
2012-09-01 18:20:29 -04:00
|
|
|
goto failed;
|
2008-02-27 20:34:43 -05:00
|
|
|
}
|
|
|
|
|
2012-09-01 11:56:35 -04:00
|
|
|
new_identity = os_malloc(attr->identity_len);
|
2012-09-01 18:20:29 -04:00
|
|
|
if (new_identity == NULL)
|
|
|
|
goto failed;
|
2012-09-01 11:56:35 -04:00
|
|
|
os_free(sm->identity);
|
|
|
|
sm->identity = new_identity;
|
|
|
|
os_memcpy(sm->identity, attr->identity, attr->identity_len);
|
|
|
|
sm->identity_len = attr->identity_len;
|
|
|
|
|
|
|
|
wpa_hexdump_ascii(MSG_DEBUG, "EAP-SIM: Identity",
|
|
|
|
sm->identity, sm->identity_len);
|
2012-09-01 12:10:33 -04:00
|
|
|
username = sim_get_username(sm->identity, sm->identity_len);
|
2012-09-01 18:20:29 -04:00
|
|
|
if (username == NULL)
|
|
|
|
goto failed;
|
2012-09-01 11:56:35 -04:00
|
|
|
|
2012-09-01 12:10:33 -04:00
|
|
|
if (username[0] == EAP_SIM_REAUTH_ID_PREFIX) {
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Reauth username '%s'",
|
|
|
|
username);
|
|
|
|
data->reauth = eap_sim_db_get_reauth_entry(
|
2012-09-01 14:10:19 -04:00
|
|
|
sm->eap_sim_db_priv, username);
|
2012-09-01 12:10:33 -04:00
|
|
|
os_free(username);
|
|
|
|
if (data->reauth == NULL) {
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Unknown reauth "
|
|
|
|
"identity - request full auth identity");
|
|
|
|
/* Remain in START state for another round */
|
|
|
|
return;
|
2008-02-27 20:34:43 -05:00
|
|
|
}
|
2012-09-01 12:10:33 -04:00
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Using fast re-authentication");
|
2012-09-01 14:10:19 -04:00
|
|
|
os_strlcpy(data->permanent, data->reauth->permanent,
|
|
|
|
sizeof(data->permanent));
|
2012-09-01 12:10:33 -04:00
|
|
|
data->counter = data->reauth->counter;
|
|
|
|
os_memcpy(data->mk, data->reauth->mk, EAP_SIM_MK_LEN);
|
|
|
|
eap_sim_state(data, REAUTH);
|
|
|
|
return;
|
2008-02-27 20:34:43 -05:00
|
|
|
}
|
|
|
|
|
2012-09-01 12:10:33 -04:00
|
|
|
if (username[0] == EAP_SIM_PSEUDONYM_PREFIX) {
|
2012-09-01 14:10:19 -04:00
|
|
|
const char *permanent;
|
2012-09-01 12:10:33 -04:00
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Pseudonym username '%s'",
|
|
|
|
username);
|
|
|
|
permanent = eap_sim_db_get_permanent(
|
2012-09-01 14:10:19 -04:00
|
|
|
sm->eap_sim_db_priv, username);
|
2012-09-01 12:10:33 -04:00
|
|
|
os_free(username);
|
|
|
|
if (permanent == NULL) {
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Unknown pseudonym "
|
|
|
|
"identity - request permanent identity");
|
|
|
|
/* Remain in START state for another round */
|
|
|
|
return;
|
|
|
|
}
|
2012-09-01 14:10:19 -04:00
|
|
|
os_strlcpy(data->permanent, permanent,
|
|
|
|
sizeof(data->permanent));
|
2012-09-01 12:10:33 -04:00
|
|
|
} else if (username[0] == EAP_SIM_PERMANENT_PREFIX) {
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Permanent username '%s'",
|
|
|
|
username);
|
|
|
|
os_strlcpy(data->permanent, username, sizeof(data->permanent));
|
|
|
|
os_free(username);
|
|
|
|
} else {
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Unrecognized username '%s'",
|
|
|
|
username);
|
|
|
|
os_free(username);
|
2012-09-01 18:20:29 -04:00
|
|
|
goto failed;
|
2008-02-27 20:34:43 -05:00
|
|
|
}
|
|
|
|
|
2012-09-01 17:38:39 -04:00
|
|
|
skip_id_update:
|
2012-09-01 12:10:33 -04:00
|
|
|
/* Full authentication */
|
2008-02-27 20:34:43 -05:00
|
|
|
|
|
|
|
if (attr->nonce_mt == NULL || attr->selected_version < 0) {
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Start/Response missing "
|
|
|
|
"required attributes");
|
2012-09-01 18:20:29 -04:00
|
|
|
goto failed;
|
2008-02-27 20:34:43 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!eap_sim_supported_ver(data, attr->selected_version)) {
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Peer selected unsupported "
|
|
|
|
"version %d", attr->selected_version);
|
2012-09-01 18:20:29 -04:00
|
|
|
goto failed;
|
2008-02-27 20:34:43 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
data->counter = 0; /* reset re-auth counter since this is full auth */
|
|
|
|
data->reauth = NULL;
|
|
|
|
|
|
|
|
data->num_chal = eap_sim_db_get_gsm_triplets(
|
2012-09-01 14:10:19 -04:00
|
|
|
sm->eap_sim_db_priv, data->permanent, EAP_SIM_MAX_CHAL,
|
2008-02-27 20:34:43 -05:00
|
|
|
(u8 *) data->rand, (u8 *) data->kc, (u8 *) data->sres, sm);
|
|
|
|
if (data->num_chal == EAP_SIM_DB_PENDING) {
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: GSM authentication triplets "
|
|
|
|
"not yet available - pending request");
|
|
|
|
sm->method_pending = METHOD_PENDING_WAIT;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (data->num_chal < 2) {
|
|
|
|
wpa_printf(MSG_INFO, "EAP-SIM: Failed to get GSM "
|
|
|
|
"authentication triplets for the peer");
|
2012-09-01 18:20:29 -04:00
|
|
|
goto failed;
|
2008-02-27 20:34:43 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
identity_len = sm->identity_len;
|
|
|
|
while (identity_len > 0 && sm->identity[identity_len - 1] == '\0') {
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Workaround - drop last null "
|
|
|
|
"character from identity");
|
|
|
|
identity_len--;
|
|
|
|
}
|
|
|
|
wpa_hexdump_ascii(MSG_DEBUG, "EAP-SIM: Identity for MK derivation",
|
|
|
|
sm->identity, identity_len);
|
|
|
|
|
|
|
|
os_memcpy(data->nonce_mt, attr->nonce_mt, EAP_SIM_NONCE_MT_LEN);
|
|
|
|
WPA_PUT_BE16(ver_list, EAP_SIM_VERSION);
|
|
|
|
eap_sim_derive_mk(sm->identity, identity_len, attr->nonce_mt,
|
|
|
|
attr->selected_version, ver_list, sizeof(ver_list),
|
|
|
|
data->num_chal, (const u8 *) data->kc, data->mk);
|
|
|
|
eap_sim_derive_keys(data->mk, data->k_encr, data->k_aut, data->msk,
|
|
|
|
data->emsk);
|
|
|
|
|
|
|
|
eap_sim_state(data, CHALLENGE);
|
2012-09-01 18:20:29 -04:00
|
|
|
return;
|
|
|
|
|
|
|
|
failed:
|
|
|
|
data->notification = EAP_SIM_GENERAL_FAILURE_BEFORE_AUTH;
|
|
|
|
eap_sim_state(data, NOTIFICATION);
|
2008-02-27 20:34:43 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void eap_sim_process_challenge(struct eap_sm *sm,
|
|
|
|
struct eap_sim_data *data,
|
|
|
|
struct wpabuf *respData,
|
|
|
|
struct eap_sim_attrs *attr)
|
|
|
|
{
|
|
|
|
if (attr->mac == NULL ||
|
|
|
|
eap_sim_verify_mac(data->k_aut, respData, attr->mac,
|
|
|
|
(u8 *) data->sres,
|
|
|
|
data->num_chal * EAP_SIM_SRES_LEN)) {
|
|
|
|
wpa_printf(MSG_WARNING, "EAP-SIM: Challenge message "
|
|
|
|
"did not include valid AT_MAC");
|
2012-09-01 18:20:29 -04:00
|
|
|
data->notification = EAP_SIM_GENERAL_FAILURE_BEFORE_AUTH;
|
|
|
|
eap_sim_state(data, NOTIFICATION);
|
2008-02-27 20:34:43 -05:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Challenge response includes the "
|
|
|
|
"correct AT_MAC");
|
|
|
|
if (sm->eap_sim_aka_result_ind && attr->result_ind) {
|
|
|
|
data->use_result_ind = 1;
|
|
|
|
data->notification = EAP_SIM_SUCCESS;
|
|
|
|
eap_sim_state(data, NOTIFICATION);
|
|
|
|
} else
|
|
|
|
eap_sim_state(data, SUCCESS);
|
|
|
|
|
|
|
|
if (data->next_pseudonym) {
|
2012-09-01 14:10:19 -04:00
|
|
|
eap_sim_db_add_pseudonym(sm->eap_sim_db_priv, data->permanent,
|
2008-02-27 20:34:43 -05:00
|
|
|
data->next_pseudonym);
|
|
|
|
data->next_pseudonym = NULL;
|
|
|
|
}
|
|
|
|
if (data->next_reauth_id) {
|
2012-09-01 14:10:19 -04:00
|
|
|
eap_sim_db_add_reauth(sm->eap_sim_db_priv, data->permanent,
|
2008-02-27 20:34:43 -05:00
|
|
|
data->next_reauth_id, data->counter + 1,
|
|
|
|
data->mk);
|
|
|
|
data->next_reauth_id = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void eap_sim_process_reauth(struct eap_sm *sm,
|
|
|
|
struct eap_sim_data *data,
|
|
|
|
struct wpabuf *respData,
|
|
|
|
struct eap_sim_attrs *attr)
|
|
|
|
{
|
|
|
|
struct eap_sim_attrs eattr;
|
|
|
|
u8 *decrypted = NULL;
|
|
|
|
|
|
|
|
if (attr->mac == NULL ||
|
|
|
|
eap_sim_verify_mac(data->k_aut, respData, attr->mac, data->nonce_s,
|
|
|
|
EAP_SIM_NONCE_S_LEN)) {
|
|
|
|
wpa_printf(MSG_WARNING, "EAP-SIM: Re-authentication message "
|
|
|
|
"did not include valid AT_MAC");
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (attr->encr_data == NULL || attr->iv == NULL) {
|
|
|
|
wpa_printf(MSG_WARNING, "EAP-SIM: Reauthentication "
|
|
|
|
"message did not include encrypted data");
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
decrypted = eap_sim_parse_encr(data->k_encr, attr->encr_data,
|
|
|
|
attr->encr_data_len, attr->iv, &eattr,
|
|
|
|
0);
|
|
|
|
if (decrypted == NULL) {
|
|
|
|
wpa_printf(MSG_WARNING, "EAP-SIM: Failed to parse encrypted "
|
|
|
|
"data from reauthentication message");
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (eattr.counter != data->counter) {
|
|
|
|
wpa_printf(MSG_WARNING, "EAP-SIM: Re-authentication message "
|
|
|
|
"used incorrect counter %u, expected %u",
|
|
|
|
eattr.counter, data->counter);
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
os_free(decrypted);
|
|
|
|
decrypted = NULL;
|
|
|
|
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Re-authentication response includes "
|
|
|
|
"the correct AT_MAC");
|
2012-09-01 17:38:39 -04:00
|
|
|
|
|
|
|
if (eattr.counter_too_small) {
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-AKA: Re-authentication response "
|
|
|
|
"included AT_COUNTER_TOO_SMALL - starting full "
|
|
|
|
"authentication");
|
|
|
|
data->start_round = -1;
|
|
|
|
eap_sim_state(data, START);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2008-02-27 20:34:43 -05:00
|
|
|
if (sm->eap_sim_aka_result_ind && attr->result_ind) {
|
|
|
|
data->use_result_ind = 1;
|
|
|
|
data->notification = EAP_SIM_SUCCESS;
|
|
|
|
eap_sim_state(data, NOTIFICATION);
|
|
|
|
} else
|
|
|
|
eap_sim_state(data, SUCCESS);
|
|
|
|
|
|
|
|
if (data->next_reauth_id) {
|
2012-09-01 14:10:19 -04:00
|
|
|
eap_sim_db_add_reauth(sm->eap_sim_db_priv, data->permanent,
|
2012-09-01 12:10:33 -04:00
|
|
|
data->next_reauth_id,
|
2008-02-27 20:34:43 -05:00
|
|
|
data->counter + 1, data->mk);
|
|
|
|
data->next_reauth_id = NULL;
|
|
|
|
} else {
|
|
|
|
eap_sim_db_remove_reauth(sm->eap_sim_db_priv, data->reauth);
|
|
|
|
data->reauth = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
fail:
|
2012-09-01 18:20:29 -04:00
|
|
|
data->notification = EAP_SIM_GENERAL_FAILURE_BEFORE_AUTH;
|
|
|
|
eap_sim_state(data, NOTIFICATION);
|
2008-02-27 20:34:43 -05:00
|
|
|
eap_sim_db_remove_reauth(sm->eap_sim_db_priv, data->reauth);
|
|
|
|
data->reauth = NULL;
|
|
|
|
os_free(decrypted);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void eap_sim_process_client_error(struct eap_sm *sm,
|
|
|
|
struct eap_sim_data *data,
|
|
|
|
struct wpabuf *respData,
|
|
|
|
struct eap_sim_attrs *attr)
|
|
|
|
{
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Client reported error %d",
|
|
|
|
attr->client_error_code);
|
|
|
|
if (data->notification == EAP_SIM_SUCCESS && data->use_result_ind)
|
|
|
|
eap_sim_state(data, SUCCESS);
|
|
|
|
else
|
|
|
|
eap_sim_state(data, FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void eap_sim_process_notification(struct eap_sm *sm,
|
|
|
|
struct eap_sim_data *data,
|
|
|
|
struct wpabuf *respData,
|
|
|
|
struct eap_sim_attrs *attr)
|
|
|
|
{
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Client replied to notification");
|
|
|
|
if (data->notification == EAP_SIM_SUCCESS && data->use_result_ind)
|
|
|
|
eap_sim_state(data, SUCCESS);
|
|
|
|
else
|
|
|
|
eap_sim_state(data, FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void eap_sim_process(struct eap_sm *sm, void *priv,
|
|
|
|
struct wpabuf *respData)
|
|
|
|
{
|
|
|
|
struct eap_sim_data *data = priv;
|
|
|
|
const u8 *pos, *end;
|
|
|
|
u8 subtype;
|
|
|
|
size_t len;
|
|
|
|
struct eap_sim_attrs attr;
|
|
|
|
|
|
|
|
pos = eap_hdr_validate(EAP_VENDOR_IETF, EAP_TYPE_SIM, respData, &len);
|
|
|
|
if (pos == NULL || len < 3)
|
|
|
|
return;
|
|
|
|
|
|
|
|
end = pos + len;
|
|
|
|
subtype = *pos;
|
|
|
|
pos += 3;
|
|
|
|
|
2012-09-01 18:26:05 -04:00
|
|
|
if (eap_sim_unexpected_subtype(data, subtype)) {
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Unrecognized or unexpected "
|
|
|
|
"EAP-SIM Subtype in EAP Response");
|
|
|
|
data->notification = EAP_SIM_GENERAL_FAILURE_BEFORE_AUTH;
|
|
|
|
eap_sim_state(data, NOTIFICATION);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2008-02-27 20:34:43 -05:00
|
|
|
if (eap_sim_parse_attr(pos, end, &attr, 0, 0)) {
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Failed to parse attributes");
|
2012-09-01 18:20:29 -04:00
|
|
|
if (subtype != EAP_SIM_SUBTYPE_CLIENT_ERROR &&
|
|
|
|
(data->state == START || data->state == CHALLENGE ||
|
|
|
|
data->state == REAUTH)) {
|
|
|
|
data->notification =
|
|
|
|
EAP_SIM_GENERAL_FAILURE_BEFORE_AUTH;
|
|
|
|
eap_sim_state(data, NOTIFICATION);
|
|
|
|
return;
|
|
|
|
}
|
2008-02-27 20:34:43 -05:00
|
|
|
eap_sim_state(data, FAILURE);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (subtype == EAP_SIM_SUBTYPE_CLIENT_ERROR) {
|
|
|
|
eap_sim_process_client_error(sm, data, respData, &attr);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (data->state) {
|
|
|
|
case START:
|
|
|
|
eap_sim_process_start(sm, data, respData, &attr);
|
|
|
|
break;
|
|
|
|
case CHALLENGE:
|
|
|
|
eap_sim_process_challenge(sm, data, respData, &attr);
|
|
|
|
break;
|
|
|
|
case REAUTH:
|
|
|
|
eap_sim_process_reauth(sm, data, respData, &attr);
|
|
|
|
break;
|
|
|
|
case NOTIFICATION:
|
|
|
|
eap_sim_process_notification(sm, data, respData, &attr);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
wpa_printf(MSG_DEBUG, "EAP-SIM: Unknown state %d in "
|
|
|
|
"process", data->state);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static Boolean eap_sim_isDone(struct eap_sm *sm, void *priv)
|
|
|
|
{
|
|
|
|
struct eap_sim_data *data = priv;
|
|
|
|
return data->state == SUCCESS || data->state == FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static u8 * eap_sim_getKey(struct eap_sm *sm, void *priv, size_t *len)
|
|
|
|
{
|
|
|
|
struct eap_sim_data *data = priv;
|
|
|
|
u8 *key;
|
|
|
|
|
|
|
|
if (data->state != SUCCESS)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
key = os_malloc(EAP_SIM_KEYING_DATA_LEN);
|
|
|
|
if (key == NULL)
|
|
|
|
return NULL;
|
|
|
|
os_memcpy(key, data->msk, EAP_SIM_KEYING_DATA_LEN);
|
|
|
|
*len = EAP_SIM_KEYING_DATA_LEN;
|
|
|
|
return key;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static u8 * eap_sim_get_emsk(struct eap_sm *sm, void *priv, size_t *len)
|
|
|
|
{
|
|
|
|
struct eap_sim_data *data = priv;
|
|
|
|
u8 *key;
|
|
|
|
|
|
|
|
if (data->state != SUCCESS)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
key = os_malloc(EAP_EMSK_LEN);
|
|
|
|
if (key == NULL)
|
|
|
|
return NULL;
|
|
|
|
os_memcpy(key, data->emsk, EAP_EMSK_LEN);
|
|
|
|
*len = EAP_EMSK_LEN;
|
|
|
|
return key;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static Boolean eap_sim_isSuccess(struct eap_sm *sm, void *priv)
|
|
|
|
{
|
|
|
|
struct eap_sim_data *data = priv;
|
|
|
|
return data->state == SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int eap_server_sim_register(void)
|
|
|
|
{
|
|
|
|
struct eap_method *eap;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
eap = eap_server_method_alloc(EAP_SERVER_METHOD_INTERFACE_VERSION,
|
|
|
|
EAP_VENDOR_IETF, EAP_TYPE_SIM, "SIM");
|
|
|
|
if (eap == NULL)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
eap->init = eap_sim_init;
|
|
|
|
eap->reset = eap_sim_reset;
|
|
|
|
eap->buildReq = eap_sim_buildReq;
|
|
|
|
eap->check = eap_sim_check;
|
|
|
|
eap->process = eap_sim_process;
|
|
|
|
eap->isDone = eap_sim_isDone;
|
|
|
|
eap->getKey = eap_sim_getKey;
|
|
|
|
eap->isSuccess = eap_sim_isSuccess;
|
|
|
|
eap->get_emsk = eap_sim_get_emsk;
|
|
|
|
|
|
|
|
ret = eap_server_method_register(eap);
|
|
|
|
if (ret)
|
|
|
|
eap_server_method_free(eap);
|
|
|
|
return ret;
|
|
|
|
}
|