2006-09-21 05:31:44 +04:00
/*
* Create default crypto algorithm instances .
*
* Copyright ( c ) 2006 Herbert Xu < herbert @ gondor . apana . org . au >
*
* 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 .
*
*/
# include <linux/crypto.h>
# include <linux/ctype.h>
# include <linux/err.h>
# include <linux/init.h>
2007-03-29 11:32:59 +04:00
# include <linux/kthread.h>
2006-09-21 05:31:44 +04:00
# include <linux/module.h>
# include <linux/notifier.h>
# include <linux/rtnetlink.h>
2006-09-21 05:39:29 +04:00
# include <linux/sched.h>
2006-09-21 05:31:44 +04:00
# include <linux/string.h>
# include "internal.h"
struct cryptomgr_param {
2007-01-01 10:37:02 +03:00
struct rtattr * tb [ CRYPTOA_MAX ] ;
struct {
struct rtattr attr ;
struct crypto_attr_type data ;
} type ;
2006-09-21 05:31:44 +04:00
struct {
struct rtattr attr ;
struct crypto_attr_alg data ;
} alg ;
struct {
char name [ CRYPTO_MAX_ALG_NAME ] ;
} larval ;
char template [ CRYPTO_MAX_ALG_NAME ] ;
} ;
2007-03-29 11:32:59 +04:00
static int cryptomgr_probe ( void * data )
2006-09-21 05:31:44 +04:00
{
2007-03-29 11:32:59 +04:00
struct cryptomgr_param * param = data ;
2006-09-21 05:31:44 +04:00
struct crypto_template * tmpl ;
struct crypto_instance * inst ;
2006-09-21 05:39:29 +04:00
int err ;
2006-09-21 05:31:44 +04:00
tmpl = crypto_lookup_template ( param - > template ) ;
if ( ! tmpl )
goto err ;
2006-09-21 05:39:29 +04:00
do {
2007-01-01 10:37:02 +03:00
inst = tmpl - > alloc ( param - > tb ) ;
2006-09-21 05:39:29 +04:00
if ( IS_ERR ( inst ) )
err = PTR_ERR ( inst ) ;
else if ( ( err = crypto_register_instance ( tmpl , inst ) ) )
tmpl - > free ( inst ) ;
} while ( err = = - EAGAIN & & ! signal_pending ( current ) ) ;
2006-09-21 05:31:44 +04:00
crypto_tmpl_put ( tmpl ) ;
2006-09-21 05:39:29 +04:00
if ( err )
goto err ;
2006-09-21 05:31:44 +04:00
out :
kfree ( param ) ;
2007-03-29 11:32:59 +04:00
module_put_and_exit ( 0 ) ;
2006-09-21 05:31:44 +04:00
err :
2007-01-01 10:37:02 +03:00
crypto_larval_error ( param - > larval . name , param - > type . data . type ,
param - > type . data . mask ) ;
2006-09-21 05:31:44 +04:00
goto out ;
}
static int cryptomgr_schedule_probe ( struct crypto_larval * larval )
{
2007-05-09 07:04:39 +04:00
struct task_struct * thread ;
2006-09-21 05:31:44 +04:00
struct cryptomgr_param * param ;
const char * name = larval - > alg . cra_name ;
const char * p ;
unsigned int len ;
2007-03-29 11:32:59 +04:00
if ( ! try_module_get ( THIS_MODULE ) )
goto err ;
2007-01-01 10:37:02 +03:00
param = kzalloc ( sizeof ( * param ) , GFP_KERNEL ) ;
2006-09-21 05:31:44 +04:00
if ( ! param )
2007-03-29 11:32:59 +04:00
goto err_put_module ;
2006-09-21 05:31:44 +04:00
for ( p = name ; isalnum ( * p ) | | * p = = ' - ' | | * p = = ' _ ' ; p + + )
;
len = p - name ;
if ( ! len | | * p ! = ' ( ' )
goto err_free_param ;
memcpy ( param - > template , name , len ) ;
name = p + 1 ;
2007-03-29 11:32:59 +04:00
len = 0 ;
for ( p = name ; * p ; p + + ) {
for ( ; isalnum ( * p ) | | * p = = ' - ' | | * p = = ' _ ' | | * p = = ' ( ' ; p + + )
;
2006-09-21 05:31:44 +04:00
2007-03-29 11:32:59 +04:00
if ( * p ! = ' ) ' )
goto err_free_param ;
len = p - name ;
}
if ( ! len | | name [ len + 1 ] )
2006-09-21 05:31:44 +04:00
goto err_free_param ;
2007-01-01 10:37:02 +03:00
param - > type . attr . rta_len = sizeof ( param - > type ) ;
param - > type . attr . rta_type = CRYPTOA_TYPE ;
param - > type . data . type = larval - > alg . cra_flags ;
param - > type . data . mask = larval - > mask ;
param - > tb [ CRYPTOA_TYPE - 1 ] = & param - > type . attr ;
2006-09-21 05:31:44 +04:00
param - > alg . attr . rta_len = sizeof ( param - > alg ) ;
param - > alg . attr . rta_type = CRYPTOA_ALG ;
memcpy ( param - > alg . data . name , name , len ) ;
2007-01-01 10:37:02 +03:00
param - > tb [ CRYPTOA_ALG - 1 ] = & param - > alg . attr ;
2006-09-21 05:31:44 +04:00
memcpy ( param - > larval . name , larval - > alg . cra_name , CRYPTO_MAX_ALG_NAME ) ;
2007-05-09 07:04:39 +04:00
thread = kthread_run ( cryptomgr_probe , param , " cryptomgr " ) ;
if ( IS_ERR ( thread ) )
2007-03-29 11:32:59 +04:00
goto err_free_param ;
2006-09-21 05:31:44 +04:00
return NOTIFY_STOP ;
err_free_param :
kfree ( param ) ;
2007-03-29 11:32:59 +04:00
err_put_module :
module_put ( THIS_MODULE ) ;
2006-09-21 05:31:44 +04:00
err :
return NOTIFY_OK ;
}
static int cryptomgr_notify ( struct notifier_block * this , unsigned long msg ,
void * data )
{
switch ( msg ) {
case CRYPTO_MSG_ALG_REQUEST :
return cryptomgr_schedule_probe ( data ) ;
}
return NOTIFY_DONE ;
}
static struct notifier_block cryptomgr_notifier = {
. notifier_call = cryptomgr_notify ,
} ;
static int __init cryptomgr_init ( void )
{
return crypto_register_notifier ( & cryptomgr_notifier ) ;
}
static void __exit cryptomgr_exit ( void )
{
int err = crypto_unregister_notifier ( & cryptomgr_notifier ) ;
BUG_ON ( err ) ;
}
module_init ( cryptomgr_init ) ;
module_exit ( cryptomgr_exit ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_DESCRIPTION ( " Crypto Algorithm Manager " ) ;