2015-06-16 20:30:55 +03:00
/*
* Public Key Encryption
*
* Copyright ( c ) 2015 , Intel Corporation
* Authors : Tadeusz Struk < tadeusz . struk @ intel . com >
*
* 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/errno.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/seq_file.h>
# include <linux/slab.h>
# include <linux/string.h>
# include <linux/crypto.h>
2016-12-31 18:56:23 +03:00
# include <linux/compiler.h>
2015-06-16 20:30:55 +03:00
# include <crypto/algapi.h>
# include <linux/cryptouser.h>
# include <net/netlink.h>
# include <crypto/akcipher.h>
2015-12-05 19:09:33 +03:00
# include <crypto/internal/akcipher.h>
2015-06-16 20:30:55 +03:00
# include "internal.h"
# ifdef CONFIG_NET
static int crypto_akcipher_report ( struct sk_buff * skb , struct crypto_alg * alg )
{
struct crypto_report_akcipher rakcipher ;
2018-11-04 00:56:03 +03:00
memset ( & rakcipher , 0 , sizeof ( rakcipher ) ) ;
2015-06-16 20:30:55 +03:00
2018-11-04 00:56:03 +03:00
strscpy ( rakcipher . type , " akcipher " , sizeof ( rakcipher . type ) ) ;
2015-06-16 20:30:55 +03:00
2018-11-04 00:56:03 +03:00
return nla_put ( skb , CRYPTOCFGA_REPORT_AKCIPHER ,
sizeof ( rakcipher ) , & rakcipher ) ;
2015-06-16 20:30:55 +03:00
}
# else
static int crypto_akcipher_report ( struct sk_buff * skb , struct crypto_alg * alg )
{
return - ENOSYS ;
}
# endif
static void crypto_akcipher_show ( struct seq_file * m , struct crypto_alg * alg )
2016-12-31 18:56:23 +03:00
__maybe_unused ;
2015-06-16 20:30:55 +03:00
static void crypto_akcipher_show ( struct seq_file * m , struct crypto_alg * alg )
{
seq_puts ( m , " type : akcipher \n " ) ;
}
static void crypto_akcipher_exit_tfm ( struct crypto_tfm * tfm )
{
struct crypto_akcipher * akcipher = __crypto_akcipher_tfm ( tfm ) ;
struct akcipher_alg * alg = crypto_akcipher_alg ( akcipher ) ;
alg - > exit ( akcipher ) ;
}
static int crypto_akcipher_init_tfm ( struct crypto_tfm * tfm )
{
struct crypto_akcipher * akcipher = __crypto_akcipher_tfm ( tfm ) ;
struct akcipher_alg * alg = crypto_akcipher_alg ( akcipher ) ;
if ( alg - > exit )
akcipher - > base . exit = crypto_akcipher_exit_tfm ;
if ( alg - > init )
return alg - > init ( akcipher ) ;
return 0 ;
}
2015-12-05 19:09:33 +03:00
static void crypto_akcipher_free_instance ( struct crypto_instance * inst )
{
struct akcipher_instance * akcipher = akcipher_instance ( inst ) ;
akcipher - > free ( akcipher ) ;
}
2015-06-16 20:30:55 +03:00
static const struct crypto_type crypto_akcipher_type = {
. extsize = crypto_alg_extsize ,
. init_tfm = crypto_akcipher_init_tfm ,
2015-12-05 19:09:33 +03:00
. free = crypto_akcipher_free_instance ,
2015-06-16 20:30:55 +03:00
# ifdef CONFIG_PROC_FS
. show = crypto_akcipher_show ,
# endif
. report = crypto_akcipher_report ,
. maskclear = ~ CRYPTO_ALG_TYPE_MASK ,
. maskset = CRYPTO_ALG_TYPE_MASK ,
. type = CRYPTO_ALG_TYPE_AKCIPHER ,
. tfmsize = offsetof ( struct crypto_akcipher , base ) ,
} ;
2015-12-05 19:09:33 +03:00
int crypto_grab_akcipher ( struct crypto_akcipher_spawn * spawn , const char * name ,
u32 type , u32 mask )
{
spawn - > base . frontend = & crypto_akcipher_type ;
return crypto_grab_spawn ( & spawn - > base , name , type , mask ) ;
}
EXPORT_SYMBOL_GPL ( crypto_grab_akcipher ) ;
2015-06-16 20:30:55 +03:00
struct crypto_akcipher * crypto_alloc_akcipher ( const char * alg_name , u32 type ,
u32 mask )
{
return crypto_alloc_tfm ( alg_name , & crypto_akcipher_type , type , mask ) ;
}
EXPORT_SYMBOL_GPL ( crypto_alloc_akcipher ) ;
2015-12-05 19:09:33 +03:00
static void akcipher_prepare_alg ( struct akcipher_alg * alg )
2015-06-16 20:30:55 +03:00
{
struct crypto_alg * base = & alg - > base ;
base - > cra_type = & crypto_akcipher_type ;
base - > cra_flags & = ~ CRYPTO_ALG_TYPE_MASK ;
base - > cra_flags | = CRYPTO_ALG_TYPE_AKCIPHER ;
2015-12-05 19:09:33 +03:00
}
2019-04-11 18:51:13 +03:00
static int akcipher_default_op ( struct akcipher_request * req )
{
return - ENOSYS ;
}
2015-12-05 19:09:33 +03:00
int crypto_register_akcipher ( struct akcipher_alg * alg )
{
struct crypto_alg * base = & alg - > base ;
2019-04-11 18:51:13 +03:00
if ( ! alg - > sign )
alg - > sign = akcipher_default_op ;
if ( ! alg - > verify )
alg - > verify = akcipher_default_op ;
if ( ! alg - > encrypt )
alg - > encrypt = akcipher_default_op ;
if ( ! alg - > decrypt )
alg - > decrypt = akcipher_default_op ;
2015-12-05 19:09:33 +03:00
akcipher_prepare_alg ( alg ) ;
2015-06-16 20:30:55 +03:00
return crypto_register_alg ( base ) ;
}
EXPORT_SYMBOL_GPL ( crypto_register_akcipher ) ;
void crypto_unregister_akcipher ( struct akcipher_alg * alg )
{
crypto_unregister_alg ( & alg - > base ) ;
}
EXPORT_SYMBOL_GPL ( crypto_unregister_akcipher ) ;
2015-12-05 19:09:33 +03:00
int akcipher_register_instance ( struct crypto_template * tmpl ,
struct akcipher_instance * inst )
{
akcipher_prepare_alg ( & inst - > alg ) ;
return crypto_register_instance ( tmpl , akcipher_crypto_instance ( inst ) ) ;
}
EXPORT_SYMBOL_GPL ( akcipher_register_instance ) ;
2015-06-16 20:30:55 +03:00
MODULE_LICENSE ( " GPL " ) ;
2015-06-23 20:18:53 +03:00
MODULE_DESCRIPTION ( " Generic public key cipher type " ) ;