2019-05-27 08:55:01 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2016-10-21 13:19:48 +01:00
/*
* Synchronous Compression operations
*
* Copyright 2015 LG Electronics Inc .
* Copyright ( c ) 2016 , Intel Corporation
* Author : Giovanni Cabiddu < giovanni . cabiddu @ intel . com >
*/
2023-02-16 18:35:17 +08:00
# include <crypto/internal/acompress.h>
# include <crypto/internal/scompress.h>
# include <crypto/scatterwalk.h>
# include <linux/cryptouser.h>
# include <linux/err.h>
2016-10-21 13:19:48 +01:00
# include <linux/kernel.h>
# include <linux/module.h>
2023-02-16 18:35:17 +08:00
# include <linux/scatterlist.h>
2016-10-21 13:19:48 +01:00
# include <linux/seq_file.h>
# include <linux/slab.h>
# include <linux/string.h>
# include <linux/vmalloc.h>
# include <net/netlink.h>
2023-02-16 18:35:17 +08:00
# include "compress.h"
2016-10-21 13:19:48 +01:00
2019-03-29 14:09:56 +01:00
struct scomp_scratch {
spinlock_t lock ;
void * src ;
void * dst ;
} ;
static DEFINE_PER_CPU ( struct scomp_scratch , scomp_scratch ) = {
. lock = __SPIN_LOCK_UNLOCKED ( scomp_scratch . lock ) ,
} ;
2016-10-21 13:19:48 +01:00
static const struct crypto_type crypto_scomp_type ;
static int scomp_scratch_users ;
static DEFINE_MUTEX ( scomp_lock ) ;
2023-02-16 18:35:28 +08:00
static int __maybe_unused crypto_scomp_report (
struct sk_buff * skb , struct crypto_alg * alg )
2016-10-21 13:19:48 +01:00
{
struct crypto_report_comp rscomp ;
2018-11-03 14:56:03 -07:00
memset ( & rscomp , 0 , sizeof ( rscomp ) ) ;
2016-10-21 13:19:48 +01:00
2018-11-03 14:56:03 -07:00
strscpy ( rscomp . type , " scomp " , sizeof ( rscomp . type ) ) ;
2016-10-21 13:19:48 +01:00
2018-11-03 14:56:03 -07:00
return nla_put ( skb , CRYPTOCFGA_REPORT_COMPRESS ,
sizeof ( rscomp ) , & rscomp ) ;
2016-10-21 13:19:48 +01:00
}
static void crypto_scomp_show ( struct seq_file * m , struct crypto_alg * alg )
2016-12-31 21:26:23 +05:30
__maybe_unused ;
2016-10-21 13:19:48 +01:00
static void crypto_scomp_show ( struct seq_file * m , struct crypto_alg * alg )
{
seq_puts ( m , " type : scomp \n " ) ;
}
2019-03-29 14:09:56 +01:00
static void crypto_scomp_free_scratches ( void )
2016-10-21 13:19:48 +01:00
{
2019-03-29 14:09:56 +01:00
struct scomp_scratch * scratch ;
2016-10-21 13:19:48 +01:00
int i ;
2019-03-29 14:09:56 +01:00
for_each_possible_cpu ( i ) {
2019-04-12 17:14:15 +02:00
scratch = per_cpu_ptr ( & scomp_scratch , i ) ;
2016-10-21 13:19:48 +01:00
2019-03-29 14:09:56 +01:00
vfree ( scratch - > src ) ;
vfree ( scratch - > dst ) ;
scratch - > src = NULL ;
scratch - > dst = NULL ;
}
2016-10-21 13:19:48 +01:00
}
2019-03-29 14:09:56 +01:00
static int crypto_scomp_alloc_scratches ( void )
2016-10-21 13:19:48 +01:00
{
2019-03-29 14:09:56 +01:00
struct scomp_scratch * scratch ;
2016-10-21 13:19:48 +01:00
int i ;
for_each_possible_cpu ( i ) {
2019-03-29 14:09:56 +01:00
void * mem ;
2016-10-21 13:19:48 +01:00
2019-04-12 17:14:15 +02:00
scratch = per_cpu_ptr ( & scomp_scratch , i ) ;
2016-10-21 13:19:48 +01:00
2019-03-29 14:09:56 +01:00
mem = vmalloc_node ( SCOMP_SCRATCH_SIZE , cpu_to_node ( i ) ) ;
if ( ! mem )
goto error ;
scratch - > src = mem ;
mem = vmalloc_node ( SCOMP_SCRATCH_SIZE , cpu_to_node ( i ) ) ;
if ( ! mem )
goto error ;
scratch - > dst = mem ;
2016-10-21 13:19:48 +01:00
}
return 0 ;
2019-03-29 14:09:56 +01:00
error :
crypto_scomp_free_scratches ( ) ;
return - ENOMEM ;
2016-10-21 13:19:48 +01:00
}
2017-07-21 16:42:38 +01:00
static int crypto_scomp_init_tfm ( struct crypto_tfm * tfm )
{
2019-03-29 14:09:56 +01:00
int ret = 0 ;
2017-07-21 16:42:38 +01:00
mutex_lock ( & scomp_lock ) ;
2019-03-29 14:09:56 +01:00
if ( ! scomp_scratch_users + + )
ret = crypto_scomp_alloc_scratches ( ) ;
2017-07-21 16:42:38 +01:00
mutex_unlock ( & scomp_lock ) ;
return ret ;
}
2016-10-21 13:19:48 +01:00
static int scomp_acomp_comp_decomp ( struct acomp_req * req , int dir )
{
struct crypto_acomp * tfm = crypto_acomp_reqtfm ( req ) ;
void * * tfm_ctx = acomp_tfm_ctx ( tfm ) ;
struct crypto_scomp * scomp = * tfm_ctx ;
void * * ctx = acomp_request_ctx ( req ) ;
2019-03-29 14:09:56 +01:00
struct scomp_scratch * scratch ;
2024-03-02 08:27:45 +13:00
void * src , * dst ;
2023-12-27 09:35:23 +00:00
unsigned int dlen ;
2016-10-21 13:19:48 +01:00
int ret ;
2019-03-29 14:09:56 +01:00
if ( ! req - > src | | ! req - > slen | | req - > slen > SCOMP_SCRATCH_SIZE )
return - EINVAL ;
2016-10-21 13:19:48 +01:00
2019-03-29 14:09:56 +01:00
if ( req - > dst & & ! req - > dlen )
return - EINVAL ;
2016-10-21 13:19:48 +01:00
if ( ! req - > dlen | | req - > dlen > SCOMP_SCRATCH_SIZE )
req - > dlen = SCOMP_SCRATCH_SIZE ;
2023-12-27 09:35:23 +00:00
dlen = req - > dlen ;
2019-03-29 14:09:56 +01:00
scratch = raw_cpu_ptr ( & scomp_scratch ) ;
spin_lock ( & scratch - > lock ) ;
2024-03-02 08:27:45 +13:00
if ( sg_nents ( req - > src ) = = 1 & & ! PageHighMem ( sg_page ( req - > src ) ) ) {
src = page_to_virt ( sg_page ( req - > src ) ) + req - > src - > offset ;
} else {
scatterwalk_map_and_copy ( scratch - > src , req - > src , 0 ,
req - > slen , 0 ) ;
src = scratch - > src ;
}
if ( req - > dst & & sg_nents ( req - > dst ) = = 1 & & ! PageHighMem ( sg_page ( req - > dst ) ) )
dst = page_to_virt ( sg_page ( req - > dst ) ) + req - > dst - > offset ;
else
dst = scratch - > dst ;
2016-10-21 13:19:48 +01:00
if ( dir )
2024-03-02 08:27:45 +13:00
ret = crypto_scomp_compress ( scomp , src , req - > slen ,
dst , & req - > dlen , * ctx ) ;
2016-10-21 13:19:48 +01:00
else
2024-03-02 08:27:45 +13:00
ret = crypto_scomp_decompress ( scomp , src , req - > slen ,
dst , & req - > dlen , * ctx ) ;
2016-10-21 13:19:48 +01:00
if ( ! ret ) {
if ( ! req - > dst ) {
2018-01-05 08:26:47 -08:00
req - > dst = sgl_alloc ( req - > dlen , GFP_ATOMIC , NULL ) ;
2019-03-29 14:09:55 +01:00
if ( ! req - > dst ) {
ret = - ENOMEM ;
2016-10-21 13:19:48 +01:00
goto out ;
2019-03-29 14:09:55 +01:00
}
2023-12-27 09:35:23 +00:00
} else if ( req - > dlen > dlen ) {
ret = - ENOSPC ;
goto out ;
2016-10-21 13:19:48 +01:00
}
2024-03-02 08:27:45 +13:00
if ( dst = = scratch - > dst ) {
scatterwalk_map_and_copy ( scratch - > dst , req - > dst , 0 ,
req - > dlen , 1 ) ;
} else {
int nr_pages = DIV_ROUND_UP ( req - > dst - > offset + req - > dlen , PAGE_SIZE ) ;
int i ;
struct page * dst_page = sg_page ( req - > dst ) ;
for ( i = 0 ; i < nr_pages ; i + + )
flush_dcache_page ( dst_page + i ) ;
}
2016-10-21 13:19:48 +01:00
}
out :
2019-03-29 14:09:56 +01:00
spin_unlock ( & scratch - > lock ) ;
2016-10-21 13:19:48 +01:00
return ret ;
}
static int scomp_acomp_compress ( struct acomp_req * req )
{
return scomp_acomp_comp_decomp ( req , 1 ) ;
}
static int scomp_acomp_decompress ( struct acomp_req * req )
{
return scomp_acomp_comp_decomp ( req , 0 ) ;
}
static void crypto_exit_scomp_ops_async ( struct crypto_tfm * tfm )
{
struct crypto_scomp * * ctx = crypto_tfm_ctx ( tfm ) ;
crypto_free_scomp ( * ctx ) ;
2017-07-21 16:42:38 +01:00
mutex_lock ( & scomp_lock ) ;
2019-03-29 14:09:56 +01:00
if ( ! - - scomp_scratch_users )
crypto_scomp_free_scratches ( ) ;
2017-07-21 16:42:38 +01:00
mutex_unlock ( & scomp_lock ) ;
2016-10-21 13:19:48 +01:00
}
int crypto_init_scomp_ops_async ( struct crypto_tfm * tfm )
{
struct crypto_alg * calg = tfm - > __crt_alg ;
struct crypto_acomp * crt = __crypto_acomp_tfm ( tfm ) ;
struct crypto_scomp * * ctx = crypto_tfm_ctx ( tfm ) ;
struct crypto_scomp * scomp ;
if ( ! crypto_mod_get ( calg ) )
return - EAGAIN ;
scomp = crypto_create_tfm ( calg , & crypto_scomp_type ) ;
if ( IS_ERR ( scomp ) ) {
crypto_mod_put ( calg ) ;
return PTR_ERR ( scomp ) ;
}
* ctx = scomp ;
tfm - > exit = crypto_exit_scomp_ops_async ;
crt - > compress = scomp_acomp_compress ;
crt - > decompress = scomp_acomp_decompress ;
2018-01-05 08:26:47 -08:00
crt - > dst_free = sgl_free ;
2016-10-21 13:19:48 +01:00
crt - > reqsize = sizeof ( void * ) ;
return 0 ;
}
struct acomp_req * crypto_acomp_scomp_alloc_ctx ( struct acomp_req * req )
{
struct crypto_acomp * acomp = crypto_acomp_reqtfm ( req ) ;
struct crypto_tfm * tfm = crypto_acomp_tfm ( acomp ) ;
struct crypto_scomp * * tfm_ctx = crypto_tfm_ctx ( tfm ) ;
struct crypto_scomp * scomp = * tfm_ctx ;
void * ctx ;
ctx = crypto_scomp_alloc_ctx ( scomp ) ;
if ( IS_ERR ( ctx ) ) {
kfree ( req ) ;
return NULL ;
}
* req - > __ctx = ctx ;
return req ;
}
void crypto_acomp_scomp_free_ctx ( struct acomp_req * req )
{
struct crypto_acomp * acomp = crypto_acomp_reqtfm ( req ) ;
struct crypto_tfm * tfm = crypto_acomp_tfm ( acomp ) ;
struct crypto_scomp * * tfm_ctx = crypto_tfm_ctx ( tfm ) ;
struct crypto_scomp * scomp = * tfm_ctx ;
void * ctx = * req - > __ctx ;
if ( ctx )
crypto_scomp_free_ctx ( scomp , ctx ) ;
}
static const struct crypto_type crypto_scomp_type = {
. extsize = crypto_alg_extsize ,
. init_tfm = crypto_scomp_init_tfm ,
# ifdef CONFIG_PROC_FS
. show = crypto_scomp_show ,
# endif
2023-05-02 10:02:33 +02:00
# if IS_ENABLED(CONFIG_CRYPTO_USER)
2016-10-21 13:19:48 +01:00
. report = crypto_scomp_report ,
2024-03-13 09:49:37 +08:00
# endif
# ifdef CONFIG_CRYPTO_STATS
. report_stat = crypto_acomp_report_stat ,
2023-02-16 18:35:17 +08:00
# endif
2016-10-21 13:19:48 +01:00
. maskclear = ~ CRYPTO_ALG_TYPE_MASK ,
. maskset = CRYPTO_ALG_TYPE_MASK ,
. type = CRYPTO_ALG_TYPE_SCOMPRESS ,
. tfmsize = offsetof ( struct crypto_scomp , base ) ,
} ;
int crypto_register_scomp ( struct scomp_alg * alg )
{
2024-03-13 09:49:37 +08:00
struct crypto_alg * base = & alg - > calg . base ;
comp_prepare_alg ( & alg - > calg ) ;
2016-10-21 13:19:48 +01:00
base - > cra_type = & crypto_scomp_type ;
base - > cra_flags | = CRYPTO_ALG_TYPE_SCOMPRESS ;
2017-07-21 16:42:38 +01:00
return crypto_register_alg ( base ) ;
2016-10-21 13:19:48 +01:00
}
EXPORT_SYMBOL_GPL ( crypto_register_scomp ) ;
2019-12-15 15:51:19 -08:00
void crypto_unregister_scomp ( struct scomp_alg * alg )
2016-10-21 13:19:48 +01:00
{
2019-12-15 15:51:19 -08:00
crypto_unregister_alg ( & alg - > base ) ;
2016-10-21 13:19:48 +01:00
}
EXPORT_SYMBOL_GPL ( crypto_unregister_scomp ) ;
2017-04-21 21:54:29 +01:00
int crypto_register_scomps ( struct scomp_alg * algs , int count )
{
int i , ret ;
for ( i = 0 ; i < count ; i + + ) {
ret = crypto_register_scomp ( & algs [ i ] ) ;
if ( ret )
goto err ;
}
return 0 ;
err :
for ( - - i ; i > = 0 ; - - i )
crypto_unregister_scomp ( & algs [ i ] ) ;
return ret ;
}
EXPORT_SYMBOL_GPL ( crypto_register_scomps ) ;
void crypto_unregister_scomps ( struct scomp_alg * algs , int count )
{
int i ;
for ( i = count - 1 ; i > = 0 ; - - i )
crypto_unregister_scomp ( & algs [ i ] ) ;
}
EXPORT_SYMBOL_GPL ( crypto_unregister_scomps ) ;
2016-10-21 13:19:48 +01:00
MODULE_LICENSE ( " GPL " ) ;
MODULE_DESCRIPTION ( " Synchronous compression type " ) ;