123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227 |
- /*
- * Copyright (C) 2008 Martin Willi
- * HSR 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 <stdint.h>
- #include "credential_factory.h"
- #include <utils/debug.h>
- #include <collections/linked_list.h>
- #include <threading/thread_value.h>
- #include <threading/rwlock.h>
- #include <credentials/certificates/x509.h>
- #include <credentials/containers/container.h>
- ENUM(credential_type_names, CRED_PRIVATE_KEY, CRED_CONTAINER,
- "CRED_PRIVATE_KEY",
- "CRED_PUBLIC_KEY",
- "CRED_CERTIFICATE",
- "CRED_CONTAINER",
- );
- typedef struct private_credential_factory_t private_credential_factory_t;
- /**
- * private data of credential_factory
- */
- struct private_credential_factory_t {
- /**
- * public functions
- */
- credential_factory_t public;
- /**
- * list with entry_t
- */
- linked_list_t *constructors;
- /**
- * Thread specific recursiveness counter
- */
- thread_value_t *recursive;
- /**
- * lock access to builders
- */
- rwlock_t *lock;
- };
- typedef struct entry_t entry_t;
- struct entry_t {
- /** kind of credential builder */
- credential_type_t type;
- /** subtype of credential, e.g. certificate_type_t */
- int subtype;
- /** registered with final flag? */
- bool final;
- /** builder function */
- builder_function_t constructor;
- };
- METHOD(credential_factory_t, add_builder, void,
- private_credential_factory_t *this, credential_type_t type, int subtype,
- bool final, builder_function_t constructor)
- {
- entry_t *entry = malloc_thing(entry_t);
- entry->type = type;
- entry->subtype = subtype;
- entry->final = final;
- entry->constructor = constructor;
- this->lock->write_lock(this->lock);
- this->constructors->insert_last(this->constructors, entry);
- this->lock->unlock(this->lock);
- }
- METHOD(credential_factory_t, remove_builder, void,
- private_credential_factory_t *this, builder_function_t constructor)
- {
- enumerator_t *enumerator;
- entry_t *entry;
- this->lock->write_lock(this->lock);
- enumerator = this->constructors->create_enumerator(this->constructors);
- while (enumerator->enumerate(enumerator, &entry))
- {
- if (entry->constructor == constructor)
- {
- this->constructors->remove_at(this->constructors, enumerator);
- free(entry);
- }
- }
- enumerator->destroy(enumerator);
- this->lock->unlock(this->lock);
- }
- METHOD(credential_factory_t, create, void*,
- private_credential_factory_t *this, credential_type_t type, int subtype, ...)
- {
- enumerator_t *enumerator;
- entry_t *entry;
- va_list args;
- void *construct = NULL;
- int failures = 0;
- uintptr_t level;
- level = (uintptr_t)this->recursive->get(this->recursive);
- this->recursive->set(this->recursive, (void*)level + 1);
- this->lock->read_lock(this->lock);
- enumerator = this->constructors->create_enumerator(this->constructors);
- while (enumerator->enumerate(enumerator, &entry))
- {
- if (entry->type == type && entry->subtype == subtype)
- {
- va_start(args, subtype);
- construct = entry->constructor(subtype, args);
- va_end(args);
- if (construct)
- {
- break;
- }
- failures++;
- }
- }
- enumerator->destroy(enumerator);
- this->lock->unlock(this->lock);
- if (!construct && !level)
- {
- enum_name_t *names;
- switch (type)
- {
- case CRED_CERTIFICATE:
- names = certificate_type_names;
- break;
- case CRED_CONTAINER:
- names = container_type_names;
- break;
- case CRED_PRIVATE_KEY:
- case CRED_PUBLIC_KEY:
- default:
- names = key_type_names;
- break;
- }
- DBG1(DBG_LIB, "building %N - %N failed, tried %d builders",
- credential_type_names, type, names, subtype, failures);
- }
- this->recursive->set(this->recursive, (void*)level);
- return construct;
- }
- CALLBACK(builder_filter, bool,
- void *null, enumerator_t *orig, va_list args)
- {
- entry_t *entry;
- credential_type_t *type;
- int *subtype;
- VA_ARGS_VGET(args, type, subtype);
- while (orig->enumerate(orig, &entry))
- {
- if (entry->final)
- {
- *type = entry->type;
- *subtype = entry->subtype;
- return TRUE;
- }
- }
- return FALSE;
- }
- METHOD(credential_factory_t, create_builder_enumerator, enumerator_t*,
- private_credential_factory_t *this)
- {
- this->lock->read_lock(this->lock);
- return enumerator_create_filter(
- this->constructors->create_enumerator(this->constructors),
- builder_filter, this->lock, (void*)this->lock->unlock);
- }
- METHOD(credential_factory_t, destroy, void,
- private_credential_factory_t *this)
- {
- this->constructors->destroy_function(this->constructors, free);
- this->recursive->destroy(this->recursive);
- this->lock->destroy(this->lock);
- free(this);
- }
- /*
- * see header file
- */
- credential_factory_t *credential_factory_create()
- {
- private_credential_factory_t *this;
- INIT(this,
- .public = {
- .create = _create,
- .create_builder_enumerator = _create_builder_enumerator,
- .add_builder = _add_builder,
- .remove_builder = _remove_builder,
- .destroy = _destroy,
- },
- .constructors = linked_list_create(),
- .recursive = thread_value_create(NULL),
- .lock = rwlock_create(RWLOCK_TYPE_DEFAULT),
- );
- return &this->public;
- }
|