Commit 5f6cb2e6 authored by Ard Biesheuvel's avatar Ard Biesheuvel Committed by Herbert Xu
Browse files

crypto: arm64/sha512-ce - simplify NEON yield



Instead of calling into kernel_neon_end() and kernel_neon_begin() (and
potentially into schedule()) from the assembler code when running in
task mode and a reschedule is pending, perform only the preempt count
check in assembler, but simply return early in this case, and let the C
code deal with the consequences.

This reverts commit 6caf7adc.

Signed-off-by: default avatarArd Biesheuvel <ardb@kernel.org>
Signed-off-by: default avatarHerbert Xu <herbert@gondor.apana.org.au>
parent 9ecc9f31
Loading
Loading
Loading
Loading
+8 −21
Original line number Diff line number Diff line
@@ -107,23 +107,17 @@
	 */
	.text
SYM_FUNC_START(sha512_ce_transform)
	frame_push	3

	mov		x19, x0
	mov		x20, x1
	mov		x21, x2

	/* load state */
0:	ld1		{v8.2d-v11.2d}, [x19]
	ld1		{v8.2d-v11.2d}, [x0]

	/* load first 4 round constants */
	adr_l		x3, .Lsha512_rcon
	ld1		{v20.2d-v23.2d}, [x3], #64

	/* load input */
1:	ld1		{v12.2d-v15.2d}, [x20], #64
	ld1		{v16.2d-v19.2d}, [x20], #64
	sub		w21, w21, #1
0:	ld1		{v12.2d-v15.2d}, [x1], #64
	ld1		{v16.2d-v19.2d}, [x1], #64
	sub		w2, w2, #1

CPU_LE(	rev64		v12.16b, v12.16b	)
CPU_LE(	rev64		v13.16b, v13.16b	)
@@ -201,19 +195,12 @@ CPU_LE( rev64 v19.16b, v19.16b )
	add		v10.2d, v10.2d, v2.2d
	add		v11.2d, v11.2d, v3.2d

	cond_yield	3f, x4
	/* handled all input blocks? */
	cbz		w21, 3f

	if_will_cond_yield_neon
	st1		{v8.2d-v11.2d}, [x19]
	do_cond_yield_neon
	b		0b
	endif_yield_neon

	b		1b
	cbnz		w2, 0b

	/* store new state */
3:	st1		{v8.2d-v11.2d}, [x19]
	frame_pop
3:	st1		{v8.2d-v11.2d}, [x0]
	mov		w0, w2
	ret
SYM_FUNC_END(sha512_ce_transform)
+26 −27
Original line number Diff line number Diff line
@@ -26,11 +26,25 @@ MODULE_LICENSE("GPL v2");
MODULE_ALIAS_CRYPTO("sha384");
MODULE_ALIAS_CRYPTO("sha512");

asmlinkage void sha512_ce_transform(struct sha512_state *sst, u8 const *src,
asmlinkage int sha512_ce_transform(struct sha512_state *sst, u8 const *src,
				   int blocks);

asmlinkage void sha512_block_data_order(u64 *digest, u8 const *src, int blocks);

static void __sha512_ce_transform(struct sha512_state *sst, u8 const *src,
				  int blocks)
{
	while (blocks) {
		int rem;

		kernel_neon_begin();
		rem = sha512_ce_transform(sst, src, blocks);
		kernel_neon_end();
		src += (blocks - rem) * SHA512_BLOCK_SIZE;
		blocks = rem;
	}
}

static void __sha512_block_data_order(struct sha512_state *sst, u8 const *src,
				      int blocks)
{
@@ -40,45 +54,30 @@ static void __sha512_block_data_order(struct sha512_state *sst, u8 const *src,
static int sha512_ce_update(struct shash_desc *desc, const u8 *data,
			    unsigned int len)
{
	if (!crypto_simd_usable())
		return sha512_base_do_update(desc, data, len,
					     __sha512_block_data_order);

	kernel_neon_begin();
	sha512_base_do_update(desc, data, len, sha512_ce_transform);
	kernel_neon_end();
	sha512_block_fn *fn = crypto_simd_usable() ? __sha512_ce_transform
						   : __sha512_block_data_order;

	sha512_base_do_update(desc, data, len, fn);
	return 0;
}

static int sha512_ce_finup(struct shash_desc *desc, const u8 *data,
			   unsigned int len, u8 *out)
{
	if (!crypto_simd_usable()) {
		if (len)
			sha512_base_do_update(desc, data, len,
					      __sha512_block_data_order);
		sha512_base_do_finalize(desc, __sha512_block_data_order);
		return sha512_base_finish(desc, out);
	}
	sha512_block_fn *fn = crypto_simd_usable() ? __sha512_ce_transform
						   : __sha512_block_data_order;

	kernel_neon_begin();
	sha512_base_do_update(desc, data, len, sha512_ce_transform);
	sha512_base_do_finalize(desc, sha512_ce_transform);
	kernel_neon_end();
	sha512_base_do_update(desc, data, len, fn);
	sha512_base_do_finalize(desc, fn);
	return sha512_base_finish(desc, out);
}

static int sha512_ce_final(struct shash_desc *desc, u8 *out)
{
	if (!crypto_simd_usable()) {
		sha512_base_do_finalize(desc, __sha512_block_data_order);
		return sha512_base_finish(desc, out);
	}
	sha512_block_fn *fn = crypto_simd_usable() ? __sha512_ce_transform
						   : __sha512_block_data_order;

	kernel_neon_begin();
	sha512_base_do_finalize(desc, sha512_ce_transform);
	kernel_neon_end();
	sha512_base_do_finalize(desc, fn);
	return sha512_base_finish(desc, out);
}