added EAP-SIM authentication

client side only
  uses an external SIM reader library specified with SIM_READER_LIB
  untested
This commit is contained in:
Martin Willi 2007-03-13 15:01:02 +00:00
parent d2940fd78c
commit b0f24449dd
8 changed files with 6000 additions and 2 deletions

View File

@ -119,6 +119,13 @@ AC_ARG_ENABLE(
)
AM_CONDITIONAL(USE_LEAK_DETECTIVE, test x$leak_detective = xtrue)
AC_ARG_ENABLE(
[eap-sim],
AS_HELP_STRING([--enable-eap-sim],[build SIM authenication module for EAP (default is NO).]),
eap_sim=true
)
AM_CONDITIONAL(BUILD_EAP_SIM, test x$eap_sim = xtrue)
dnl =========================
dnl check required programs
dnl =========================

View File

@ -1,8 +1,17 @@
# SUBDIRS = . testing
eap_LTLIBRARIES =
eap_LTLIBRARIES = libeapidentity.la
# always build EAP Identity module
libeapidentity_la_SOURCES = sa/authenticators/eap/eap_identity.h sa/authenticators/eap/eap_identity.c
libeapidentity_la_LDFLAGS = -module
# build optional EAP modules
if BUILD_EAP_SIM
eap_LTLIBRARIES += libeapsim.la
libeapsim_la_SOURCES = sa/authenticators/eap/eap_sim.h sa/authenticators/eap/eap_sim.c
libeapsim_la_LDFLAGS = -module
endif
ipsec_PROGRAMS = charon

File diff suppressed because it is too large Load Diff

View File

@ -41,7 +41,9 @@ ENUM_BEGIN(eap_type_names, EAP_IDENTITY, EAP_TOKEN_CARD,
"EAP_MD5",
"EAP_ONE_TIME_PASSWORD",
"EAP_TOKEN_CARD");
ENUM_NEXT(eap_type_names, EAP_AKA, EAP_AKA, EAP_TOKEN_CARD,
ENUM_NEXT(eap_type_names, EAP_SIM, EAP_SIM, EAP_TOKEN_CARD,
"EAP_SIM");
ENUM_NEXT(eap_type_names, EAP_AKA, EAP_AKA, EAP_SIM,
"EAP_AKA");
ENUM_END(eap_type_names, EAP_AKA);

View File

@ -60,6 +60,7 @@ enum eap_type_t {
EAP_MD5 = 4,
EAP_ONE_TIME_PASSWORD = 5,
EAP_TOKEN_CARD = 6,
EAP_SIM = 18,
EAP_AKA = 23,
};

View File

@ -0,0 +1,677 @@
/**
* @file eap_sim.c
*
* @brief Implementation of eap_sim_t.
*
*/
/*
* Copyright (C) 2007 Martin Willi
* Hochschule fuer Technik Rapperswil
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License as published by the
* Free Software Foundation; either version 2 of the License, or (at your
* option) any later version. See <http://www.fsf.org/copyleft/gpl.txt>.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* for more details.
*/
#include "eap_sim.h"
#include <dlfcn.h>
#include <daemon.h>
#include <library.h>
ENUM(sim_subtype_names, SIM_START, SIM_CLIENT_ERROR,
"SIM_START",
"SIM_CHALLENGE",
"SIM_NOTIFICATION",
"SIM_13",
"SIM_CLIENT_ERROR",
);
ENUM_BEGIN(sim_attribute_names, AT_END, AT_CLIENT_ERROR_CODE,
"AT_END",
"AT_0",
"AT_RAND",
"AT_AUTN",
"AT_RES",
"AT_AUTS",
"AT_5",
"AT_PADDING",
"AT_NONCE_MT",
"AT_8",
"AT_9",
"AT_PERMANENT_ID_REQ",
"AT_MAC",
"AT_NOTIFICATION",
"AT_ANY_ID_REQ",
"AT_IDENTITY",
"AT_VERSION_LIST",
"AT_SELECTED_VERSION",
"AT_FULLAUTH_ID_REQ",
"AT_18",
"AT_COUNTER",
"AT_COUNTER_TOO_SMALL",
"AT_NONCE_S",
"AT_CLIENT_ERROR_CODE");
ENUM_NEXT(sim_attribute_names, AT_IV, AT_RESULT_IND, AT_CLIENT_ERROR_CODE,
"AT_IV",
"AT_ENCR_DATA",
"AT_131",
"AT_NEXT_PSEUDONYM",
"AT_NEXT_REAUTH_ID",
"AT_CHECKCODE",
"AT_RESULT_IND");
ENUM_END(sim_attribute_names, AT_RESULT_IND);
typedef struct private_eap_sim_t private_eap_sim_t;
/**
* Private data of an eap_sim_t object.
*/
struct private_eap_sim_t {
/**
* Public authenticator_t interface.
*/
eap_sim_t public;
/**
* ID of ourself
*/
identification_t *peer;
/**
* SIM cardreader function loaded from library
*/
sim_algo_t alg;
/**
* handle of the loaded library
*/
void *handle;
/**
* version this implementation uses
*/
chunk_t version;
/**
* version list received from server
*/
chunk_t version_list;
/**
* Nonce value used in AT_NONCE_MT
*/
chunk_t nonce;
/**
* k_encr key derived from MK
*/
chunk_t k_encr;
/**
* k_auth key derived from MK, used for AT_MAC verification
*/
chunk_t k_auth;
/**
* MSK, used for EAP-SIM based IKEv2 authentication
*/
chunk_t msk;
/**
* EMSK, extendes MSK for further uses
*/
chunk_t emsk;
};
/** length of the AT_NONCE_MT nonce value */
#define NONCE_LEN 16
/** length of the AT_MAC value */
#define MAC_LEN 16
/** length of the AT_RAND value */
#define RAND_LEN 16
/** length of the k_encr key */
#define KENCR_LEN 16
/** length of the k_Auth value */
#define KAUTH_LEN 16
/** length of the MSK */
#define MSK_LEN 64
/** length of the EMSK */
#define EMSK_LEN 64
/* client error codes used in AT_CLIENT_ERROR_CODE */
char client_error_general_buf[] = {0x00, 0x01};
char client_error_unsupported_buf[] = {0x00, 0x02};
char client_error_insufficient_buf[] = {0x00, 0x03};
char client_error_notfresh_buf[] = {0x00, 0x04};
chunk_t client_error_general = chunk_from_buf(client_error_general_buf);
chunk_t client_error_unsupported = chunk_from_buf(client_error_unsupported_buf);
chunk_t client_error_insufficient = chunk_from_buf(client_error_insufficient_buf);
chunk_t client_error_notfresh = chunk_from_buf(client_error_notfresh_buf);
/**
* Read EAP and EAP-SIM header, return SIM type
*/
static sim_subtype_t read_header(chunk_t *message)
{
sim_subtype_t type;
if (message->len < 8)
{
*message = chunk_empty;
return 0;
}
type = *(message->ptr + 5);
*message = chunk_skip(*message, 8);
return type;
}
/**
* read the next attribute from the chunk data
*/
static sim_attribute_t read_attribute(chunk_t *message, chunk_t *data)
{
sim_attribute_t attribute;
size_t length;
DBG3(DBG_IKE, "reading attribute from %B", message);
if (message->len < 2)
{
return AT_END;
}
attribute = *message->ptr++;
length = *message->ptr++ * 4 - 2;
message->len -= 2;
DBG3(DBG_IKE, "found attribute %N with length %d",
sim_attribute_names, attribute, length);
if (length > message->len)
{
return AT_END;
}
data->len = length;
data->ptr = message->ptr;
*message = chunk_skip(*message, length);
return attribute;
}
/**
* Build an EAP-SIM payload using a variable length attribute list.
* The variable argument takes a sim_attribute_t followed by its data in a chunk.
*/
static eap_payload_t *build_payload(private_eap_sim_t *this, u_int8_t identifier,
sim_subtype_t type, ...)
{
chunk_t message = chunk_alloca(512);
chunk_t pos = message;
eap_payload_t *payload;
va_list args;
sim_attribute_t attr;
u_int8_t *mac_pos = NULL;
chunk_t mac_data = chunk_empty;
/* write EAP header, skip length bytes */
*pos.ptr++ = EAP_RESPONSE;
*pos.ptr++ = identifier;
pos.ptr += 2;
pos.len -= 4;
/* write SIM header with type and subtype, zero reserved bytes */
*pos.ptr++ = EAP_SIM;
*pos.ptr++ = type;
*pos.ptr++ = 0;
*pos.ptr++ = 0;
pos.len -= 4;
va_start(args, type);
while ((attr = va_arg(args, sim_attribute_t)) != AT_END)
{
chunk_t data = va_arg(args, chunk_t);
DBG3(DBG_IKE, "building %N %B", sim_attribute_names, attr, &data);
/* write attribute header */
*pos.ptr++ = attr;
pos.len--;
switch (attr)
{
case AT_CLIENT_ERROR_CODE:
case AT_SELECTED_VERSION:
{
*pos.ptr = data.len/4 + 1;
pos = chunk_skip(pos, 1);
memcpy(pos.ptr, data.ptr, data.len);
pos = chunk_skip(pos, data.len);
break;
}
case AT_IDENTITY:
{
*pos.ptr = data.len/4 + 1;
pos = chunk_skip(pos, 1);
/* actual length in bytes */
*(u_int16_t*)pos.ptr = htons(data.len);
pos = chunk_skip(pos, sizeof(u_int16_t));
memcpy(pos.ptr, data.ptr, data.len);
pos = chunk_skip(pos, data.len);
break;
}
case AT_NONCE_MT:
{
*pos.ptr = data.len/4 + 1;
pos = chunk_skip(pos, 1);
memset(pos.ptr, 0, 2);
pos = chunk_skip(pos, 2);
memcpy(pos.ptr, data.ptr, data.len);
pos = chunk_skip(pos, data.len);
break;
}
case AT_MAC:
{
*pos.ptr++ = 5; pos.len--;
*pos.ptr++ = 0; pos.len--;
*pos.ptr++ = 0; pos.len--;
mac_pos = pos.ptr;
memset(mac_pos, 0, MAC_LEN);
pos = chunk_skip(pos, MAC_LEN);
mac_data = data;
break;
}
case AT_RAND:
{
*pos.ptr++ = data.len/4 + 1; pos.len--;
*pos.ptr++ = 0; pos.len--;
*pos.ptr++ = 0; pos.len--;
memcpy(pos.ptr, data.ptr, data.len);
pos = chunk_skip(pos, data.len);
break;
}
default:
DBG1(DBG_IKE, "no rule to build EAP_SIM attribute %N, skipped",
sim_attribute_names, attr);
break;
}
}
va_end(args);
/* calculate message length, write into header */
message.len = pos.ptr - message.ptr;
*(u_int16_t*)(message.ptr + 2) = htons(message.len);
/* create MAC if AT_MAC attribte was included. Append supplied va_arg
* chunk mac_data to "to-sign" chunk */
if (mac_pos)
{
signer_t *signer = signer_create(AUTH_HMAC_SHA1_128);
signer->set_key(signer, this->k_auth);
mac_data = chunk_cata("cc", message, mac_data);
DBG3(DBG_IKE, "AT_MAC signature of %B", &mac_data);
DBG3(DBG_IKE, "using k_auth %B", &this->k_auth);
signer->get_signature(signer, mac_data, mac_pos);
DBG3(DBG_IKE, "is %b", mac_pos, MAC_LEN);
signer->destroy(signer);
}
payload = eap_payload_create_data(message);
DBG3(DBG_IKE, "created EAP message %B", &message);
return payload;
}
/**
* process an EAP-SIM/Request/Start message
*/
static status_t process_start(private_eap_sim_t *this, eap_payload_t *in,
eap_payload_t **out)
{
chunk_t message, data;
sim_attribute_t attribute, include_id = AT_END;
u_int8_t identifier;
identifier = in->get_identifier(in);
message = in->get_data(in);
read_header(&message);
while ((attribute = read_attribute(&message, &data)) != AT_END)
{
switch (attribute)
{
case AT_VERSION_LIST:
{
/* check if server supports our implementation */
bool found = FALSE;
if (data.len > 2)
{
/* read actual length first */
data.len = ntohs(*(u_int16_t*)data.ptr);
data = chunk_skip(data, 2);
this->version_list = chunk_clone(data);
while (data.len >= this->version.len)
{
if (memeq(data.ptr, this->version.ptr, this->version.len))
{
found = TRUE;
break;
}
data = chunk_skip(data, this->version.len);
}
}
if (!found)
{
DBG1(DBG_IKE, "server does not support EAP_SIM "
"version number %#B", &this->version);
*out = build_payload(this, identifier, SIM_CLIENT_ERROR,
AT_CLIENT_ERROR_CODE, client_error_unsupported,
AT_END);
return FAILED;
}
break;
}
case AT_PERMANENT_ID_REQ:
case AT_FULLAUTH_ID_REQ:
case AT_ANY_ID_REQ:
/* only include AT_IDENTITY if requested */
include_id = AT_IDENTITY;
break;
default:
DBG1(DBG_IKE, "ignoring EAP_SIM attribute %N",
sim_attribute_names, attribute);
break;
}
}
/* build payload. If "include_id" is AT_END, AT_IDENTITY is ommited */
*out = build_payload(this, identifier, SIM_START,
AT_SELECTED_VERSION, this->version,
AT_NONCE_MT, this->nonce,
include_id, this->peer->get_encoding(this->peer),
AT_END);
return NEED_MORE;
}
/**
* process an EAP-SIM/Request/Challenge message
*/
static status_t process_challenge(private_eap_sim_t *this, eap_payload_t *in,
eap_payload_t **out)
{
chunk_t message, data, tmp, kcs, kc, sreses, sres, mk;
sim_attribute_t attribute;
u_int8_t identifier;
chunk_t mac = chunk_empty, rands = chunk_empty;
signer_t *signer;
hasher_t *hasher;
prf_t *prf;
identifier = in->get_identifier(in);
message = in->get_data(in);
read_header(&message);
while ((attribute = read_attribute(&message, &data)) != AT_END)
{
switch (attribute)
{
case AT_RAND:
{
rands = chunk_skip(data, 2);
break;
}
case AT_MAC:
{
/* backup MAC, zero it inline for later verification */
mac = chunk_clonea(chunk_skip(data, 2));
memset(data.ptr, 0, data.len);
break;
}
default:
DBG1(DBG_IKE, "ignoring EAP_SIM attribute %N",
sim_attribute_names, attribute);
break;
}
}
/* excepting two or three RAND, each 16 bytes. We require two valid
* and different (!) RANDs */
if ((rands.len != 2 * RAND_LEN && rands.len != 3 * RAND_LEN) ||
memeq(rands.ptr, rands.ptr + RAND_LEN, RAND_LEN))
{
DBG1(DBG_IKE, "no valid AT_RAND received");
*out = build_payload(this, identifier, SIM_CLIENT_ERROR,
AT_CLIENT_ERROR_CODE, client_error_insufficient,
AT_END);
return FAILED;
}
if (mac.len != MAC_LEN)
{
DBG1(DBG_IKE, "no valid AT_MAC received");
*out = build_payload(this, identifier, SIM_CLIENT_ERROR,
AT_CLIENT_ERROR_CODE, client_error_general,
AT_END);
return FAILED;
}
/* get two or three KCs/SRESes from SIM using RANDs */
kcs = kc = chunk_alloca(rands.len / 2);
sreses = sres = chunk_alloca(kcs.len / 2);
while (rands.len > 0)
{
if (this->alg(rands.ptr, RAND_LEN, kc.ptr, &kc.len, sres.ptr, &sres.len))
{
DBG1(DBG_IKE, "unable to get triplets from SIM");
*out = build_payload(this, identifier, SIM_CLIENT_ERROR,
AT_CLIENT_ERROR_CODE, client_error_general,
AT_END);
return FAILED;
}
kc = chunk_skip(kc, kc.len);
sres = chunk_skip(sres, sres.len);
rands = chunk_skip(rands, RAND_LEN);
}
/* build MK */
tmp = chunk_cata("ccccc", this->peer->get_encoding(this->peer), kcs,
this->nonce, this->version_list, this->version);
hasher = hasher_create(HASH_SHA1);
mk = chunk_alloca(hasher->get_hash_size(hasher));
hasher->get_hash(hasher, tmp, mk.ptr);
hasher->destroy(hasher);
/* K_encr | K_auth | MSK | EMSK = prf() | prf() | prf() | prf()
* FIPS PRF has 320 bit block size, we need 160 byte for keys
* => run prf four times */
prf = prf_create(PRF_FIPS_SHA1_160);
prf->set_key(prf, mk);
tmp = chunk_alloca(prf->get_block_size(prf) * 4);
prf->get_bytes(prf, chunk_empty, tmp.ptr);
prf->get_bytes(prf, chunk_empty, tmp.ptr + tmp.len / 4 * 1);
prf->get_bytes(prf, chunk_empty, tmp.ptr + tmp.len / 4 * 2);
prf->get_bytes(prf, chunk_empty, tmp.ptr + tmp.len / 4 * 3);
prf->destroy(prf);
chunk_split(tmp, "aaaa", KENCR_LEN, &this->k_encr, KAUTH_LEN, &this->k_auth,
MSK_LEN, &this->msk, EMSK_LEN, &this->emsk);
DBG3(DBG_IKE, "MK %B", &mk);
DBG3(DBG_IKE, "PRF res %B", &tmp);
DBG3(DBG_IKE, "K_encr %B", &this->k_encr);
DBG3(DBG_IKE, "K_auth %B", &this->k_auth);
DBG3(DBG_IKE, "MSK %B", &this->msk);
DBG3(DBG_IKE, "EMSK %B", &this->emsk);
/* verify AT_MAC attribute, signature is over "EAP packet | NONCE_MT" */
signer = signer_create(AUTH_HMAC_SHA1_128);
signer->set_key(signer, this->k_auth);
tmp = chunk_cata("cc", message, this->nonce);
DBG3(DBG_IKE, "verifying AT_MAC signature of %B", &tmp);
DBG3(DBG_IKE, "using k_auth %B", &this->k_auth);
if (!signer->verify_signature(signer, tmp, mac))
{
DBG1(DBG_IKE, "MAC in AT_MAC attribute verification failed");
signer->destroy(signer);
chunk_free(&this->msk);
*out = build_payload(this, identifier, SIM_CLIENT_ERROR,
AT_CLIENT_ERROR_CODE, client_error_general,
AT_END);
return FAILED;
}
signer->destroy(signer);
/* build response, AT_MAC is built over "EAP packet | n*SRES" */
*out = build_payload(this, identifier, SIM_CHALLENGE,
AT_MAC, sreses,
AT_END);
return NEED_MORE;
}
/**
* Implementation of eap_method_t.process for the peer
*/
static status_t process(private_eap_sim_t *this,
eap_payload_t *in, eap_payload_t **out)
{
sim_subtype_t type;
chunk_t message;
message = in->get_data(in);
type = read_header(&message);
switch (type)
{
case SIM_START:
return process_start(this, in, out);
case SIM_CHALLENGE:
return process_challenge(this, in, out);
default:
DBG1(DBG_IKE, "unable to process EAP_SIM subtype %N",
sim_subtype_names, type);
*out = build_payload(this, in->get_identifier(in), SIM_CLIENT_ERROR,
AT_CLIENT_ERROR_CODE, client_error_general, AT_END);
return NEED_MORE;
}
}
/**
* Implementation of eap_method_t.initiate for the peer
*/
static status_t initiate(private_eap_sim_t *this, eap_payload_t **out)
{
/* peer never initiates */
return FAILED;
}
/**
* Implementation of eap_method_t.get_type.
*/
static eap_type_t get_type(private_eap_sim_t *this)
{
return EAP_SIM;
}
/**
* Implementation of eap_method_t.get_msk.
*/
static status_t get_msk(private_eap_sim_t *this, chunk_t *msk)
{
if (this->msk.ptr)
{
*msk = this->msk;
return SUCCESS;
}
return FAILED;
}
/**
* Implementation of eap_method_t.is_mutual.
*/
static bool is_mutual(private_eap_sim_t *this)
{
return TRUE;
}
/**
* Implementation of eap_method_t.destroy.
*/
static void destroy(private_eap_sim_t *this)
{
dlclose(this->handle);
chunk_free(&this->nonce);
chunk_free(&this->version_list);
chunk_free(&this->k_auth);
chunk_free(&this->k_encr);
chunk_free(&this->msk);
chunk_free(&this->emsk);
free(this);
}
/*
* Described in header.
*/
eap_sim_t *eap_create(eap_role_t role,
identification_t *server, identification_t *peer)
{
private_eap_sim_t *this;
randomizer_t *randomizer;
static char version[] = {0x00,0x01};
if (role != EAP_PEER)
{
return NULL;
}
this = malloc_thing(private_eap_sim_t);
this->handle = dlopen(SIM_READER_LIB, RTLD_LAZY);
if (this->handle == NULL)
{
DBG1(DBG_IKE, "unable to open SIM reader '%s'", SIM_READER_LIB);
free(this);
return NULL;
}
this->alg = dlsym(this->handle, SIM_READER_ALG);
if (this->alg == NULL)
{
DBG1(DBG_IKE, "unable to open SIM reader function '%s' in '%s'",
SIM_READER_ALG, SIM_READER_LIB);
dlclose(this->handle);
free(this);
return NULL;
}
randomizer = randomizer_create();
if (randomizer->allocate_pseudo_random_bytes(randomizer, NONCE_LEN,
&this->nonce))
{
DBG1(DBG_IKE, "unable to generate NONCE for EAP_SIM");
randomizer->destroy(randomizer);
free(this);
return NULL;
}
randomizer->destroy(randomizer);
/* public functions */
this->public.eap_method_interface.initiate = (status_t(*)(eap_method_t*,eap_payload_t**))initiate;
this->public.eap_method_interface.process = (status_t(*)(eap_method_t*,eap_payload_t*,eap_payload_t**))process;
this->public.eap_method_interface.get_type = (eap_type_t(*)(eap_method_t*))get_type;
this->public.eap_method_interface.is_mutual = (bool(*)(eap_method_t*))is_mutual;
this->public.eap_method_interface.get_msk = (status_t(*)(eap_method_t*,chunk_t*))get_msk;
this->public.eap_method_interface.destroy = (void(*)(eap_method_t*))destroy;
/* private data */
this->peer = peer;
this->version.ptr = version;
this->version.len = sizeof(version);
this->version_list = chunk_empty;
this->k_auth = chunk_empty;
this->k_encr = chunk_empty;
this->msk = chunk_empty;
this->emsk = chunk_empty;
return &this->public;
}

View File

@ -0,0 +1,142 @@
/**
* @file eap_sim.h
*
* @brief Interface of eap_sim_t.
*
*/
/*
* Copyright (C) 2007 Martin Willi
* Hochschule fuer Technik Rapperswil
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License as published by the
* Free Software Foundation; either version 2 of the License, or (at your
* option) any later version. See <http://www.fsf.org/copyleft/gpl.txt>.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* for more details.
*/
#ifndef EAP_SIM_H_
#define EAP_SIM_H_
typedef struct eap_sim_t eap_sim_t;
typedef enum sim_subtype_t sim_subtype_t;
typedef enum sim_attribute_t sim_attribute_t;
#include <sa/authenticators/eap/eap_method.h>
/**
* Subtypes of SIM messages
*/
enum sim_subtype_t {
SIM_START = 10,
SIM_CHALLENGE = 11,
SIM_NOTIFICATION = 12,
SIM_CLIENT_ERROR = 14,
};
/**
* enum names for sim_subtype_t
*/
extern enum_name_t *sim_subtype_names;
enum sim_attribute_t {
/** defines the end of attribute list */
AT_END = -1,
AT_RAND = 1,
AT_AUTN = 2,
AT_RES = 3,
AT_AUTS = 4,
AT_PADDING = 6,
AT_NONCE_MT = 7,
AT_PERMANENT_ID_REQ = 10,
AT_MAC = 11,
AT_NOTIFICATION = 12,
AT_ANY_ID_REQ = 13,
AT_IDENTITY = 14,
AT_VERSION_LIST = 15,
AT_SELECTED_VERSION = 16,
AT_FULLAUTH_ID_REQ = 17,
AT_COUNTER = 19,
AT_COUNTER_TOO_SMALL = 20,
AT_NONCE_S = 21,
AT_CLIENT_ERROR_CODE = 22,
AT_IV = 129,
AT_ENCR_DATA = 130,
AT_NEXT_PSEUDONYM = 132,
AT_NEXT_REAUTH_ID = 133,
AT_CHECKCODE = 134,
AT_RESULT_IND = 135,
};
/**
* enum names for sim_subtype_t
*/
extern enum_name_t *sim_attribute_names;
/**
* @brief Cardreaders SIM function.
*
* @param rand RAND to run algo with
* @param rand_length length of value in rand
* @param sres buffer to get SRES
* @param sres_length size of buffer in sres, returns bytes written to SRES
* @param kc buffer to get Kc
* @param kc_length size of buffer in Kc, returns bytes written to Kc
* @return zero on success
*/
typedef int (*sim_algo_t)(const unsigned char *rand, int rand_length,
unsigned char *sres, int *sres_length,
unsigned char *kc, int *kc_length);
#ifndef SIM_READER_LIB
/** the library containing the cardreader with the SIM function */
#define SIM_READER_LIB "/home/martin/strongswan/trunk/src/charon/sa/authenticators/eap/sim_reader/sim_api.so"
#endif /* SIM_READER_LIB */
#ifndef SIM_READER_ALG
/** the SIM_READER_LIB's algorithm, uses sim_algo_t signature */
#define SIM_READER_ALG "sim_run_alg"
#endif /* SIM_READER_ALG */
/**
* @brief Implementation of the eap_method_t interface using EAP-SIM.
*
* This EAP-SIM client implementation uses another pluggable library to
* access the SIM card. This module is specified using the SIM_READER_LIB
* definition. The function to run the algorithm has the sim_algo_t type and
* is named as SIM_READER_ALG is defined.
*
* @b Constructors:
* - eap_sim_create()
* - eap_client_create() using eap_method EAP_SIM
*
* @ingroup eap
*/
struct eap_sim_t {
/**
* Implemented eap_method_t interface.
*/
eap_method_t eap_method_interface;
};
/**
* @brief Creates the EAP method EAP-SIM.
*
* @param server ID of the EAP server
* @param peer ID of the EAP client
* @return eap_sim_t object
*
* @ingroup eap
*/
eap_sim_t *eap_create(eap_role_t role,
identification_t *server, identification_t *peer);
#endif /* EAP_SIM_H_ */

View File

@ -466,6 +466,11 @@ load_conn(starter_conn_t *conn, kw_list_t *kw, starter_config_t *cfg)
/* TODO: a gperf function for all EAP types */
if (streq(kw->value, "aka"))
conn->eap = 23;
else if (streq(kw->value, "sim"))
{
conn->eap = 18;
}
else
{
conn->eap = atoi(kw->value);