Viewing: lustre_crypto.h

/* SPDX-License-Identifier: GPL-2.0 */

/*
 * Copyright (c) 2019, 2020, Whamcloud.
 */

/*
 * This file is part of Lustre, http://www.lustre.org/
 */

#ifndef _LUSTRE_CRYPTO_H_
#define _LUSTRE_CRYPTO_H_

#if defined(HAVE_LUSTRE_CRYPTO) && !defined(CONFIG_LL_ENCRYPTION)
#define __FS_HAS_ENCRYPTION 1
#include <linux/fscrypt.h>

/* LLCRYPT_DIGESTED_* is provided by llcrypt.h but that is not present
 * for native fscrypt builds
 */
#define LLCRYPT_DIGESTED_CHAR		'+'
#define LLCRYPT_DIGESTED_CHAR_OLD	'_'
#define LL_CRYPTO_BLOCK_SIZE		FS_CRYPTO_BLOCK_SIZE
#define llcrypt_name			fscrypt_name
#define llcrypt_str			fscrypt_str
#define LLTR_INIT			FSTR_INIT
#define llcrypt_operations		fscrypt_operations
#define llcrypt_symlink_data		fscrypt_symlink_data
#define llcrypt_init()			0
#define llcrypt_exit()			{}
#ifndef HAVE_FSCRYPT_DUMMY_CONTEXT_ENABLED
#ifdef HAVE_FSCRYPT_DUMMY_POLICY
#define llcrypt_policy			fscrypt_policy
#define llcrypt_dummy_policy		fscrypt_dummy_policy
#else
#define llcrypt_policy			fscrypt_context
#define llcrypt_dummy_policy		fscrypt_dummy_context
#endif
#endif
#ifdef HAVE_FSCRYPT_D_REVALIDATE
#define llcrypt_d_revalidate(dentry, flags)				\
	fscrypt_d_revalidate(dentry, flags)
#else
	int llcrypt_d_revalidate(struct dentry *dentry, unsigned int flags);
#endif
#define llcrypt_has_encryption_key(inode)				\
	fscrypt_has_encryption_key(inode)
#define llcrypt_encrypt_pagecache_blocks(page, len, offs, gfp_flags)	\
	fscrypt_encrypt_pagecache_blocks(page, len, offs, gfp_flags)
#define llcrypt_decrypt_pagecache_blocks(page, len, offs)	\
	fscrypt_decrypt_pagecache_blocks(page, len, offs)
#define llcrypt_decrypt_block_inplace(inode, page, len, offs, lblk_num)	\
	fscrypt_decrypt_block_inplace(inode, page, len, offs, lblk_num)
#define llcrypt_inherit_context(parent, child, fs_data, preload)	\
	fscrypt_inherit_context(parent, child, fs_data, preload)
#ifdef HAVE_FSCRYPT_PREPARE_READDIR
#define llcrypt_prepare_readdir(inode) fscrypt_prepare_readdir(inode)
#else
#define llcrypt_prepare_readdir(inode) fscrypt_get_encryption_info(inode)
#endif
#define llcrypt_put_encryption_info(inode) fscrypt_put_encryption_info(inode)
#define llcrypt_free_inode(inode)	   fscrypt_free_inode(inode)
#define llcrypt_finalize_bounce_page(pagep)  fscrypt_finalize_bounce_page(pagep)
#define llcrypt_file_open(inode, filp)	fscrypt_file_open(inode, filp)
#define llcrypt_ioctl_set_policy(filp, arg)  fscrypt_ioctl_set_policy(filp, arg)
#define llcrypt_ioctl_get_policy_ex(filp, arg)	\
	fscrypt_ioctl_get_policy_ex(filp, arg)
#define llcrypt_policy_has_filename_enc(inode) true
#define llcrypt_ioctl_add_key(filp, arg)	fscrypt_ioctl_add_key(filp, arg)
#define llcrypt_ioctl_remove_key(filp, arg)  fscrypt_ioctl_remove_key(filp, arg)
#define llcrypt_ioctl_remove_key_all_users(filp, arg)	\
	fscrypt_ioctl_remove_key_all_users(filp, arg)
#define llcrypt_ioctl_get_key_status(filp, arg)	\
	fscrypt_ioctl_get_key_status(filp, arg)
#define llcrypt_drop_inode(inode)	fscrypt_drop_inode(inode)
#define llcrypt_prepare_rename(olddir, olddentry, newdir, newdentry, flags) \
	fscrypt_prepare_rename(olddir, olddentry, newdir, newdentry, flags)
#define llcrypt_prepare_link(old_dentry, dir, dentry)	\
	fscrypt_prepare_link(old_dentry, dir, dentry)
#define llcrypt_prepare_setattr(dentry, attr)		\
	fscrypt_prepare_setattr(dentry, attr)
#define __llcrypt_prepare_lookup(inode, dentry, fname)	\
	__fscrypt_prepare_lookup(inode, dentry, fname)
#define llcrypt_set_ops(sb, cop)	fscrypt_set_ops(sb, cop)
#define llcrypt_sb_free(sb)		{}
#ifdef HAVE_FSCRYPT_FNAME_ALLOC_BUFFER_NO_INODE
#define llcrypt_fname_alloc_buffer(inode, max_encrypted_len, crypto_str) \
	fscrypt_fname_alloc_buffer(max_encrypted_len, crypto_str)
#else
#define llcrypt_fname_alloc_buffer(inode, max_encrypted_len, crypto_str) \
	fscrypt_fname_alloc_buffer(inode, max_encrypted_len, crypto_str)
#endif
#define llcrypt_fname_disk_to_usr(inode, hash, minor_hash, iname, oname) \
	fscrypt_fname_disk_to_usr(inode, hash, minor_hash, iname, oname)
#define llcrypt_fname_free_buffer(crypto_str) \
	fscrypt_fname_free_buffer(crypto_str)
#define llcrypt_setup_filename(dir, iname, lookup, fname) \
	fscrypt_setup_filename(dir, iname, lookup, fname)
#define llcrypt_free_filename(fname) \
	fscrypt_free_filename(fname)
#define llcrypt_match_name(fname, de_name, name_len)		\
	fscrypt_match_name(fname, de_name, name_len)
#define llcrypt_prepare_lookup(dir, dentry, fname) \
	fscrypt_prepare_lookup(dir, dentry, fname)
#define llcrypt_encrypt_symlink(inode, target, len, disk_link) \
	fscrypt_encrypt_symlink(inode, target, len, disk_link)
#define __llcrypt_encrypt_symlink(inode, target, len, disk_link) \
	__fscrypt_encrypt_symlink(inode, target, len, disk_link)
#define llcrypt_prepare_symlink(dir, target, len, max_len, disk_link)	\
	fscrypt_prepare_symlink(dir, target, len, max_len, disk_link)
#define llcrypt_get_symlink(inode, caddr, max_size, done) \
	fscrypt_get_symlink(inode, caddr, max_size, done)

#define LL_IOC_SET_ENCRYPTION_POLICY FS_IOC_SET_ENCRYPTION_POLICY
#define LL_IOC_GET_ENCRYPTION_POLICY_EX FS_IOC_GET_ENCRYPTION_POLICY_EX
#define LL_IOC_ADD_ENCRYPTION_KEY FS_IOC_ADD_ENCRYPTION_KEY
#define LL_IOC_REMOVE_ENCRYPTION_KEY FS_IOC_REMOVE_ENCRYPTION_KEY
#define LL_IOC_REMOVE_ENCRYPTION_KEY_ALL_USERS \
	FS_IOC_REMOVE_ENCRYPTION_KEY_ALL_USERS
#define LL_IOC_GET_ENCRYPTION_KEY_STATUS FS_IOC_GET_ENCRYPTION_KEY_STATUS

#else /* HAVE_LUSTRE_CRYPTO && !CONFIG_LL_ENCRYPTION */
#include <lustre_compat/linux/llcrypt.h>

#define llcrypt_prepare_readdir(inode) llcrypt_get_encryption_info(inode)

int llcrypt_d_revalidate(struct dentry *dentry, unsigned int flags);

#endif /* !HAVE_LUSTRE_CRYPTO || CONFIG_LL_ENCRYPTION */

#if !defined(HAVE_FSCRYPT_IS_NOKEY_NAME) || defined(CONFIG_LL_ENCRYPTION)

#ifndef DCACHE_NOKEY_NAME
#define DCACHE_NOKEY_NAME               0x02000000 /* Enc name without key */
#endif

static inline bool llcrypt_is_nokey_name(const struct dentry *dentry)
{
	return dentry->d_flags & DCACHE_NOKEY_NAME;
}
#else
#define llcrypt_is_nokey_name(dentry)		\
	fscrypt_is_nokey_name(dentry)
#endif

#if defined(HAVE_LUSTRE_CRYPTO) && !defined(HAVE_FSCRYPT_DUMMY_CONTEXT_ENABLED)
#define llcrypt_show_test_dummy_encryption(seq, sep, sb)	\
	fscrypt_show_test_dummy_encryption(seq, sep, sb)
#define llcrypt_set_test_dummy_encryption(sb, arg, ctx)		\
	fscrypt_set_test_dummy_encryption(sb, arg, ctx)
#ifdef HAVE_FSCRYPT_DUMMY_POLICY
#define llcrypt_free_dummy_policy(policy)			\
	fscrypt_free_dummy_policy(policy)
#else
#define llcrypt_free_dummy_policy(policy)			\
	fscrypt_free_dummy_context(policy)
#endif
#else
#define llcrypt_show_test_dummy_encryption(seq, sep, sb)	{}
#define llcrypt_free_dummy_policy(policy)			{}
#endif

/* Macro to extract digest from Lustre specific structures */
#if defined(HAVE_FSCRYPT_DIGESTED_NAME) && !defined(CONFIG_LL_ENCRYPTION)
#define LLCRYPT_EXTRACT_DIGEST		FSCRYPT_FNAME_DIGEST
#else
#define LLCRYPT_EXTRACT_DIGEST(name, len)			\
	((name) + round_down((len) - LL_CRYPTO_BLOCK_SIZE - 1,	\
			     LL_CRYPTO_BLOCK_SIZE))
#endif

struct ll_sb_info;
int ll_set_encflags(struct inode *inode, void *encctx, __u32 encctxlen,
		    bool preload);
bool ll_sb_has_test_dummy_encryption(struct super_block *sb);
bool ll_sbi_has_encrypt(struct ll_sb_info *sbi);
void ll_sbi_set_encrypt(struct ll_sb_info *sbi, bool set);
bool ll_sbi_has_name_encrypt(struct ll_sb_info *sbi);
void ll_sbi_set_name_encrypt(struct ll_sb_info *sbi, bool set);
/* sizeof(struct fscrypt_context_v2) = 40 */
#define LLCRYPT_ENC_CTX_SIZE 40

/* Encoding/decoding routines inspired from yEnc principles.
 * We just take care of a few critical characters:
 * NULL, LF, CR, /, DEL and =.
 * If such a char is found, it is replaced with '=' followed by
 * the char value + 64.
 * All other chars are left untouched.
 * Efficiency of this encoding depends on the occurences of the
 * critical chars, but statistically on binary data it can be much higher
 * than base64 for instance.
 */
static inline int critical_encode(const u8 *src, int len, char *dst)
{
	u8 *p = (u8 *)src, *q = dst;

	while (p - src < len) {
		/* escape NULL, LF, CR, /, DEL and = */
		if (unlikely(*p == 0x0 || *p == 0xA || *p == 0xD ||
			     *p == '/' || *p == 0x7F || *p == '=')) {
			*(q++) = '=';
			*(q++) = *(p++) + 64;
		} else {
			*(q++) = *(p++);
		}
	}

	return (char *)q - dst;
}

/* returns the number of chars encoding would produce */
static inline int critical_chars(const u8 *src, int len)
{
	u8 *p = (u8 *)src;
	int newlen = len;

	while (p - src < len) {
		/* NULL, LF, CR, /, DEL and = cost an additional '=' */
		if (unlikely(*p == 0x0 || *p == 0xA || *p == 0xD ||
			     *p == '/' || *p == 0x7F || *p == '='))
			newlen++;
		p++;
	}

	return newlen;
}

/* decoding routine - returns the number of chars in output */
static inline int critical_decode(const u8 *src, int len, char *dst)
{
	u8 *p = (u8 *)src, *q = dst;

	while (p - src < len) {
		if (unlikely(*p == '=')) {
			*(q++) = *(++p) - 64;
			p++;
		} else {
			*(q++) = *(p++);
		}
	}

	return (char *)q - dst;
}

/* Used for support of PCC-RO for encrypted files.
 * In case an encrypted file is put in PCC-RO, we want it to be in ciphertext.
 * The way to achieve this is to set the S_PCCCOPY flag on the inode, so that
 * we directly return -ENOKEY in this case. This makes the lustre inode be
 * treated as if we do not have the key, hence fetching ciphertext content
 * instead of decrypting it.
 * S_PCCCOPY is actually S_DIRSYNC, as it is safe so far to use it on regular
 * files in this scenario.
 */
#define S_PCCCOPY S_DIRSYNC
#define IS_PCCCOPY(inode)	((inode)->i_flags & S_PCCCOPY)
#define ll_has_encryption_key(inode)	\
	(IS_PCCCOPY(inode) ? false : llcrypt_has_encryption_key(inode))

#endif /* _LUSTRE_CRYPTO_H_ */