]> git.ipfire.org Git - thirdparty/strongswan.git/commitdiff
eap-sim-file plugin supports volatile in-memory storage of fast reauthentication...
authorMartin Willi <martin@strongswan.org>
Wed, 28 Oct 2009 13:18:33 +0000 (14:18 +0100)
committerMartin Willi <martin@strongswan.org>
Thu, 12 Nov 2009 09:34:01 +0000 (10:34 +0100)
src/charon/plugins/eap_sim_file/eap_sim_file_card.c
src/charon/plugins/eap_sim_file/eap_sim_file_provider.c

index 1413d97d33e35bae02d76da1743c59fb354efa89..077163a82aa7ff636b38703b2c03658e13bdcdbe 100644 (file)
@@ -36,16 +36,32 @@ struct private_eap_sim_file_card_t {
        eap_sim_file_triplets_t *triplets;
 
        /**
-        * Permanent -> pseudonym mappongs
+        * Permanent -> pseudonym mappings
         */
        hashtable_t *pseudonym;
 
        /**
-        * Pseudonym -> permanent mappings
+        * Permanent -> reauth_data_t mappings
+        */
+       hashtable_t *reauth;
+
+       /**
+        * Reverse pseudonym -> permanent mappings
         */
        hashtable_t *permanent;
 };
 
+typedef struct {
+       /** currently used reauthentication identity */
+       identification_t *id;
+       /** associated permanent identity */
+       identification_t *permanent;
+       /** counter value */
+       u_int16_t counter;
+       /** master key */
+       char mk[HASH_SIZE_SHA1];
+} reauth_data_t;
+
 /**
  * hashtable hash function
  */
@@ -130,6 +146,55 @@ static void set_pseudonym(private_eap_sim_file_card_t *this,
        DESTROY_IF(pseudonym);
 }
 
+/**
+ * Implementation of sim_card_t.get_reauth
+ */
+static identification_t *get_reauth(private_eap_sim_file_card_t *this,
+                                                               identification_t *id, char mk[HASH_SIZE_SHA1],
+                                                               u_int16_t *counter)
+{
+       reauth_data_t *data;
+       identification_t *reauth;
+
+       /* look up reauthentication data */
+       data = this->reauth->remove(this->reauth, id);
+       if (!data)
+       {
+               return NULL;
+       }
+       *counter = ++data->counter;
+       memcpy(mk, data->mk, HASH_SIZE_SHA1);
+       reauth = data->id;
+       data->permanent->destroy(data->permanent);
+       free(data);
+       return reauth;
+}
+
+/**
+ * Implementation of sim_card_t.set_reauth
+ */
+static void set_reauth(private_eap_sim_file_card_t *this,
+                                          identification_t *id, identification_t* next,
+                                          char mk[HASH_SIZE_SHA1], u_int16_t counter)
+{
+       reauth_data_t *data;
+
+       data = this->reauth->get(this->reauth, id);
+       if (data)
+       {
+               data->id->destroy(data->id);
+       }
+       else
+       {
+               data = malloc_thing(reauth_data_t);
+               data->permanent = id->clone(id);
+               this->reauth->put(this->reauth, data->permanent, data);
+       }
+       data->counter = counter;
+       data->id = next->clone(next);
+       memcpy(data->mk, mk, HASH_SIZE_SHA1);
+}
+
 /**
  * Implementation of sim_card_t.get_quintuplet
  */
@@ -144,24 +209,36 @@ static bool get_quintuplet()
 static void destroy(private_eap_sim_file_card_t *this)
 {
        enumerator_t *enumerator;
-       identification_t *key, *value;
+       identification_t *id;
+       reauth_data_t *data;
+       void *key;
 
        enumerator = this->pseudonym->create_enumerator(this->pseudonym);
-       while (enumerator->enumerate(enumerator, &key, &value))
+       while (enumerator->enumerate(enumerator, &key, &id))
        {
-               value->destroy(value);
+               id->destroy(id);
        }
        enumerator->destroy(enumerator);
 
        enumerator = this->permanent->create_enumerator(this->permanent);
-       while (enumerator->enumerate(enumerator, &key, &value))
+       while (enumerator->enumerate(enumerator, &key, &id))
+       {
+               id->destroy(id);
+       }
+       enumerator->destroy(enumerator);
+
+       enumerator = this->reauth->create_enumerator(this->reauth);
+       while (enumerator->enumerate(enumerator, &key, &data))
        {
-               value->destroy(value);
+               data->id->destroy(data->id);
+               data->permanent->destroy(data->permanent);
+               free(data);
        }
        enumerator->destroy(enumerator);
 
        this->pseudonym->destroy(this->pseudonym);
        this->permanent->destroy(this->permanent);
+       this->reauth->destroy(this->reauth);
        free(this);
 }
 
@@ -177,13 +254,14 @@ eap_sim_file_card_t *eap_sim_file_card_create(eap_sim_file_triplets_t *triplets)
        this->public.card.resync = (bool(*)(sim_card_t*, identification_t *id, char rand[AKA_RAND_LEN], char auts[AKA_AUTS_LEN]))return_false;
        this->public.card.get_pseudonym = (identification_t*(*)(sim_card_t*, identification_t *perm))get_pseudonym;
        this->public.card.set_pseudonym = (void(*)(sim_card_t*, identification_t *id, identification_t *pseudonym))set_pseudonym;
-       this->public.card.get_reauth = (identification_t*(*)(sim_card_t*, identification_t *id, char mk[HASH_SIZE_SHA1], u_int16_t *counter))return_null;
-       this->public.card.set_reauth = (void(*)(sim_card_t*, identification_t *id, identification_t* next, char mk[HASH_SIZE_SHA1], u_int16_t counter))nop;
+       this->public.card.get_reauth = (identification_t*(*)(sim_card_t*, identification_t *id, char mk[HASH_SIZE_SHA1], u_int16_t *counter))get_reauth;
+       this->public.card.set_reauth = (void(*)(sim_card_t*, identification_t *id, identification_t* next, char mk[HASH_SIZE_SHA1], u_int16_t counter))set_reauth;
        this->public.destroy = (void(*)(eap_sim_file_card_t*))destroy;
 
        this->triplets = triplets;
        this->pseudonym = hashtable_create((void*)hash, (void*)equals, 0);
        this->permanent = hashtable_create((void*)hash, (void*)equals, 0);
+       this->reauth = hashtable_create((void*)hash, (void*)equals, 0);
 
        return &this->public;
 }
index 22c4f26e3a56c2d4a0bf39e16c4ca4853a71d1dd..aa2eabc432b9e3efd0443a298175a1fcd23dd336 100644 (file)
@@ -36,21 +36,35 @@ struct private_eap_sim_file_provider_t {
        eap_sim_file_triplets_t *triplets;
 
        /**
-        * Permanent -> pseudonym mappongs
+        * Permanent -> pseudonym mappings
         */
        hashtable_t *pseudonym;
 
        /**
-        * Pseudonym -> permanent mappings
+        * Permanent -> reauth_data_t mappings
+        */
+       hashtable_t *reauth;
+
+       /**
+        * Reverse pseudonym/reauth -> permanent mappings
         */
        hashtable_t *permanent;
 
        /**
-        * RNG for pseudonyms
+        * RNG for pseudonyms/reauth identities
         */
        rng_t *rng;
 };
 
+typedef struct {
+       /** currently used reauthentication identity */
+       identification_t *id;
+       /** counter value */
+       u_int16_t counter;
+       /** master key */
+       char mk[HASH_SIZE_SHA1];
+} reauth_data_t;
+
 /**
  * hashtable hash function
  */
@@ -110,13 +124,25 @@ static identification_t* is_pseudonym(private_eap_sim_file_provider_t *this,
 }
 
 /**
- * Implementation of sim_provider_t.get_triplet
+ * Generate a random identity
+ */
+static identification_t *gen_identity(private_eap_sim_file_provider_t *this)
+{
+       char buf[8], hex[sizeof(buf) * 2 + 1];
+
+       this->rng->get_bytes(this->rng, sizeof(buf), buf);
+       chunk_to_hex(chunk_create(buf, sizeof(buf)), hex, FALSE);
+
+       return identification_create_from_string(hex);
+}
+
+/**
+ * Implementation of sim_provider_t.get_pseudonym
  */
 static identification_t* gen_pseudonym(private_eap_sim_file_provider_t *this,
                                                                           identification_t *id)
 {
        identification_t *pseudonym, *permanent;
-       char buf[8], hex[sizeof(buf) * 2 + 1];
 
        /* remove old entry */
        pseudonym = this->pseudonym->remove(this->pseudonym, id);
@@ -130,10 +156,7 @@ static identification_t* gen_pseudonym(private_eap_sim_file_provider_t *this,
                pseudonym->destroy(pseudonym);
        }
 
-       /* generate new pseudonym */
-       this->rng->get_bytes(this->rng, sizeof(buf), buf);
-       chunk_to_hex(chunk_create(buf, sizeof(buf)), hex, FALSE);
-       pseudonym = identification_create_from_string(hex);
+       pseudonym = gen_identity(this);
 
        /* create new entries */
        id = id->clone(id);
@@ -143,30 +166,102 @@ static identification_t* gen_pseudonym(private_eap_sim_file_provider_t *this,
        return pseudonym->clone(pseudonym);
 }
 
+/**
+ * Implementation of sim_provider_t.is_reauth
+ */
+static identification_t *is_reauth(private_eap_sim_file_provider_t *this,
+                                                               identification_t *id, char mk[HASH_SIZE_SHA1],
+                                                               u_int16_t *counter)
+{
+       identification_t *permanent;
+       reauth_data_t *data;
+
+       /* look up permanent identity */
+       permanent = this->permanent->get(this->permanent, id);
+       if (!permanent)
+       {
+               return NULL;
+       }
+       /* look up reauthentication data */
+       data = this->reauth->get(this->reauth, permanent);
+       if (!data)
+       {
+               return NULL;
+       }
+       *counter = ++data->counter;
+       memcpy(mk, data->mk, HASH_SIZE_SHA1);
+       return permanent->clone(permanent);
+}
+
+/**
+ * Implementation of sim_provider_t.gen_reauth
+ */
+static identification_t *gen_reauth(private_eap_sim_file_provider_t *this,
+                                                               identification_t *id, char mk[HASH_SIZE_SHA1])
+{
+       reauth_data_t *data;
+       identification_t *permanent;
+
+       data = this->reauth->get(this->reauth, id);
+       id = id->clone(id);
+       if (data)
+       {       /* update existing entry */
+               permanent = this->permanent->remove(this->permanent, data->id);
+               if (permanent)
+               {
+                       permanent->destroy(permanent);
+               }
+               data->id->destroy(data->id);
+       }
+       else
+       {       /* generate new entry */
+               data = malloc_thing(reauth_data_t);
+               data->counter = 0;
+               this->reauth->put(this->reauth, id, data);
+       }
+       memcpy(data->mk, mk, HASH_SIZE_SHA1);
+       data->id = gen_identity(this);
+
+       this->permanent->put(this->permanent, data->id, id);
+
+       return data->id->clone(data->id);
+}
+
 /**
  * Implementation of eap_sim_file_provider_t.destroy.
  */
 static void destroy(private_eap_sim_file_provider_t *this)
 {
        enumerator_t *enumerator;
-       identification_t *key, *value;
+       identification_t *id;
+       reauth_data_t *data;
+       void *key;
 
        enumerator = this->pseudonym->create_enumerator(this->pseudonym);
-       while (enumerator->enumerate(enumerator, &key, &value))
+       while (enumerator->enumerate(enumerator, &key, &id))
        {
-               value->destroy(value);
+               id->destroy(id);
        }
        enumerator->destroy(enumerator);
 
        enumerator = this->permanent->create_enumerator(this->permanent);
-       while (enumerator->enumerate(enumerator, &key, &value))
+       while (enumerator->enumerate(enumerator, &key, &id))
+       {
+               id->destroy(id);
+       }
+       enumerator->destroy(enumerator);
+
+       enumerator = this->reauth->create_enumerator(this->reauth);
+       while (enumerator->enumerate(enumerator, &key, &data))
        {
-               value->destroy(value);
+               data->id->destroy(data->id);
+               free(data);
        }
        enumerator->destroy(enumerator);
 
        this->pseudonym->destroy(this->pseudonym);
        this->permanent->destroy(this->permanent);
+       this->reauth->destroy(this->reauth);
        this->rng->destroy(this->rng);
        free(this);
 }
@@ -184,8 +279,8 @@ eap_sim_file_provider_t *eap_sim_file_provider_create(
        this->public.provider.resync = (bool(*)(sim_provider_t*, identification_t *id, char rand[AKA_RAND_LEN], char auts[AKA_AUTS_LEN]))return_false;
        this->public.provider.is_pseudonym = (identification_t*(*)(sim_provider_t*, identification_t *id))is_pseudonym;
        this->public.provider.gen_pseudonym = (identification_t*(*)(sim_provider_t*, identification_t *id))gen_pseudonym;
-       this->public.provider.is_reauth = (identification_t*(*)(sim_provider_t*, identification_t *id, char [HASH_SIZE_SHA1], u_int16_t *counter))return_null;
-       this->public.provider.gen_reauth = (identification_t*(*)(sim_provider_t*, identification_t *id, char mk[HASH_SIZE_SHA1]))return_null;
+       this->public.provider.is_reauth = (identification_t*(*)(sim_provider_t*, identification_t *id, char [HASH_SIZE_SHA1], u_int16_t *counter))is_reauth;
+       this->public.provider.gen_reauth = (identification_t*(*)(sim_provider_t*, identification_t *id, char mk[HASH_SIZE_SHA1]))gen_reauth;
        this->public.destroy = (void(*)(eap_sim_file_provider_t*))destroy;
 
        this->triplets = triplets;
@@ -197,6 +292,7 @@ eap_sim_file_provider_t *eap_sim_file_provider_create(
        }
        this->pseudonym = hashtable_create((void*)hash, (void*)equals, 0);
        this->permanent = hashtable_create((void*)hash, (void*)equals, 0);
+       this->reauth = hashtable_create((void*)hash, (void*)equals, 0);
 
        return &this->public;
 }