2018-09-19 10:10:54 +00:00
// SPDX-License-Identifier: GPL-2.0
/*
* Crypto user configuration API .
*
* Copyright ( C ) 2017 - 2018 Corentin Labbe < clabbe @ baylibre . com >
*
*/
2023-02-16 18:35:23 +08:00
# include <crypto/algapi.h>
# include <crypto/internal/cryptouser.h>
# include <linux/errno.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/string.h>
2018-09-19 10:10:54 +00:00
# include <net/netlink.h>
2019-07-09 13:11:24 +02:00
# include <net/sock.h>
2018-09-19 10:10:54 +00:00
# define null_terminated(x) (strnlen(x, sizeof(x)) < sizeof(x))
struct crypto_dump_info {
struct sk_buff * in_skb ;
struct sk_buff * out_skb ;
u32 nlmsg_seq ;
u16 nlmsg_flags ;
} ;
static int crypto_report_cipher ( struct sk_buff * skb , struct crypto_alg * alg )
{
2018-11-29 14:42:19 +00:00
struct crypto_stat_cipher rcipher ;
2018-09-19 10:10:54 +00:00
2018-11-03 14:56:01 -07:00
memset ( & rcipher , 0 , sizeof ( rcipher ) ) ;
2018-11-03 14:56:03 -07:00
strscpy ( rcipher . type , " cipher " , sizeof ( rcipher . type ) ) ;
2018-09-19 10:10:54 +00:00
2018-11-03 14:56:03 -07:00
return nla_put ( skb , CRYPTOCFGA_STAT_CIPHER , sizeof ( rcipher ) , & rcipher ) ;
2018-09-19 10:10:54 +00:00
}
static int crypto_report_comp ( struct sk_buff * skb , struct crypto_alg * alg )
{
2018-11-29 14:42:19 +00:00
struct crypto_stat_compress rcomp ;
2018-09-19 10:10:54 +00:00
2018-11-03 14:56:01 -07:00
memset ( & rcomp , 0 , sizeof ( rcomp ) ) ;
2018-11-03 14:56:03 -07:00
strscpy ( rcomp . type , " compression " , sizeof ( rcomp . type ) ) ;
2018-09-19 10:10:54 +00:00
2018-11-03 14:56:03 -07:00
return nla_put ( skb , CRYPTOCFGA_STAT_COMPRESS , sizeof ( rcomp ) , & rcomp ) ;
2018-09-19 10:10:54 +00:00
}
static int crypto_reportstat_one ( struct crypto_alg * alg ,
struct crypto_user_alg * ualg ,
struct sk_buff * skb )
{
2018-11-03 14:56:01 -07:00
memset ( ualg , 0 , sizeof ( * ualg ) ) ;
2018-11-03 14:56:03 -07:00
strscpy ( ualg - > cru_name , alg - > cra_name , sizeof ( ualg - > cru_name ) ) ;
strscpy ( ualg - > cru_driver_name , alg - > cra_driver_name ,
2018-09-19 10:10:54 +00:00
sizeof ( ualg - > cru_driver_name ) ) ;
2018-11-03 14:56:03 -07:00
strscpy ( ualg - > cru_module_name , module_name ( alg - > cra_module ) ,
2018-09-19 10:10:54 +00:00
sizeof ( ualg - > cru_module_name ) ) ;
ualg - > cru_type = 0 ;
ualg - > cru_mask = 0 ;
ualg - > cru_flags = alg - > cra_flags ;
ualg - > cru_refcnt = refcount_read ( & alg - > cra_refcnt ) ;
if ( nla_put_u32 ( skb , CRYPTOCFGA_PRIORITY_VAL , alg - > cra_priority ) )
goto nla_put_failure ;
if ( alg - > cra_flags & CRYPTO_ALG_LARVAL ) {
2018-11-29 14:42:19 +00:00
struct crypto_stat_larval rl ;
2018-09-19 10:10:54 +00:00
2018-11-03 14:56:01 -07:00
memset ( & rl , 0 , sizeof ( rl ) ) ;
2018-11-03 14:56:03 -07:00
strscpy ( rl . type , " larval " , sizeof ( rl . type ) ) ;
if ( nla_put ( skb , CRYPTOCFGA_STAT_LARVAL , sizeof ( rl ) , & rl ) )
2018-09-19 10:10:54 +00:00
goto nla_put_failure ;
goto out ;
}
2023-02-16 18:35:09 +08:00
if ( alg - > cra_type & & alg - > cra_type - > report_stat ) {
if ( alg - > cra_type - > report_stat ( skb , alg ) )
goto nla_put_failure ;
goto out ;
}
2018-09-19 10:10:54 +00:00
switch ( alg - > cra_flags & ( CRYPTO_ALG_TYPE_MASK | CRYPTO_ALG_LARVAL ) ) {
case CRYPTO_ALG_TYPE_CIPHER :
if ( crypto_report_cipher ( skb , alg ) )
goto nla_put_failure ;
break ;
case CRYPTO_ALG_TYPE_COMPRESS :
if ( crypto_report_comp ( skb , alg ) )
goto nla_put_failure ;
break ;
default :
pr_err ( " ERROR: Unhandled alg %d in %s \n " ,
alg - > cra_flags & ( CRYPTO_ALG_TYPE_MASK | CRYPTO_ALG_LARVAL ) ,
__func__ ) ;
}
out :
return 0 ;
nla_put_failure :
return - EMSGSIZE ;
}
static int crypto_reportstat_alg ( struct crypto_alg * alg ,
struct crypto_dump_info * info )
{
struct sk_buff * in_skb = info - > in_skb ;
struct sk_buff * skb = info - > out_skb ;
struct nlmsghdr * nlh ;
struct crypto_user_alg * ualg ;
int err = 0 ;
nlh = nlmsg_put ( skb , NETLINK_CB ( in_skb ) . portid , info - > nlmsg_seq ,
CRYPTO_MSG_GETSTAT , sizeof ( * ualg ) , info - > nlmsg_flags ) ;
if ( ! nlh ) {
err = - EMSGSIZE ;
goto out ;
}
ualg = nlmsg_data ( nlh ) ;
err = crypto_reportstat_one ( alg , ualg , skb ) ;
if ( err ) {
nlmsg_cancel ( skb , nlh ) ;
goto out ;
}
nlmsg_end ( skb , nlh ) ;
out :
return err ;
}
int crypto_reportstat ( struct sk_buff * in_skb , struct nlmsghdr * in_nlh ,
struct nlattr * * attrs )
{
2019-07-09 13:11:24 +02:00
struct net * net = sock_net ( in_skb - > sk ) ;
2018-09-19 10:10:54 +00:00
struct crypto_user_alg * p = nlmsg_data ( in_nlh ) ;
struct crypto_alg * alg ;
struct sk_buff * skb ;
struct crypto_dump_info info ;
int err ;
if ( ! null_terminated ( p - > cru_name ) | | ! null_terminated ( p - > cru_driver_name ) )
return - EINVAL ;
alg = crypto_alg_match ( p , 0 ) ;
if ( ! alg )
return - ENOENT ;
err = - ENOMEM ;
skb = nlmsg_new ( NLMSG_DEFAULT_SIZE , GFP_ATOMIC ) ;
if ( ! skb )
goto drop_alg ;
info . in_skb = in_skb ;
info . out_skb = skb ;
info . nlmsg_seq = in_nlh - > nlmsg_seq ;
info . nlmsg_flags = 0 ;
err = crypto_reportstat_alg ( alg , & info ) ;
drop_alg :
crypto_mod_put ( alg ) ;
2019-10-04 14:34:54 -05:00
if ( err ) {
kfree_skb ( skb ) ;
2018-09-19 10:10:54 +00:00
return err ;
2019-10-04 14:34:54 -05:00
}
2018-09-19 10:10:54 +00:00
2019-07-09 13:11:24 +02:00
return nlmsg_unicast ( net - > crypto_nlsk , skb , NETLINK_CB ( in_skb ) . portid ) ;
2018-09-19 10:10:54 +00:00
}
MODULE_LICENSE ( " GPL " ) ;