xref: /openbmc/linux/arch/arm64/include/asm/word-at-a-time.h (revision 1ac731c529cd4d6adbce134754b51ff7d822b145)
1caab277bSThomas Gleixner /* SPDX-License-Identifier: GPL-2.0-only */
212a0ef7bSWill Deacon /*
312a0ef7bSWill Deacon  * Copyright (C) 2013 ARM Ltd.
412a0ef7bSWill Deacon  */
512a0ef7bSWill Deacon #ifndef __ASM_WORD_AT_A_TIME_H
612a0ef7bSWill Deacon #define __ASM_WORD_AT_A_TIME_H
712a0ef7bSWill Deacon 
87c0f6ba6SLinus Torvalds #include <linux/uaccess.h>
96c94f27aSArd Biesheuvel 
1012a0ef7bSWill Deacon #ifndef __AARCH64EB__
1112a0ef7bSWill Deacon 
1212a0ef7bSWill Deacon #include <linux/kernel.h>
1312a0ef7bSWill Deacon 
1412a0ef7bSWill Deacon struct word_at_a_time {
1512a0ef7bSWill Deacon 	const unsigned long one_bits, high_bits;
1612a0ef7bSWill Deacon };
1712a0ef7bSWill Deacon 
1812a0ef7bSWill Deacon #define WORD_AT_A_TIME_CONSTANTS { REPEAT_BYTE(0x01), REPEAT_BYTE(0x80) }
1912a0ef7bSWill Deacon 
has_zero(unsigned long a,unsigned long * bits,const struct word_at_a_time * c)2012a0ef7bSWill Deacon static inline unsigned long has_zero(unsigned long a, unsigned long *bits,
2112a0ef7bSWill Deacon 				     const struct word_at_a_time *c)
2212a0ef7bSWill Deacon {
2312a0ef7bSWill Deacon 	unsigned long mask = ((a - c->one_bits) & ~a) & c->high_bits;
2412a0ef7bSWill Deacon 	*bits = mask;
2512a0ef7bSWill Deacon 	return mask;
2612a0ef7bSWill Deacon }
2712a0ef7bSWill Deacon 
2812a0ef7bSWill Deacon #define prep_zero_mask(a, bits, c) (bits)
2912a0ef7bSWill Deacon 
create_zero_mask(unsigned long bits)3012a0ef7bSWill Deacon static inline unsigned long create_zero_mask(unsigned long bits)
3112a0ef7bSWill Deacon {
3212a0ef7bSWill Deacon 	bits = (bits - 1) & ~bits;
3312a0ef7bSWill Deacon 	return bits >> 7;
3412a0ef7bSWill Deacon }
3512a0ef7bSWill Deacon 
find_zero(unsigned long mask)3612a0ef7bSWill Deacon static inline unsigned long find_zero(unsigned long mask)
3712a0ef7bSWill Deacon {
3812a0ef7bSWill Deacon 	return fls64(mask) >> 3;
3912a0ef7bSWill Deacon }
4012a0ef7bSWill Deacon 
417bc13fd3SWill Deacon #define zero_bytemask(mask) (mask)
427bc13fd3SWill Deacon 
4312a0ef7bSWill Deacon #else	/* __AARCH64EB__ */
4412a0ef7bSWill Deacon #include <asm-generic/word-at-a-time.h>
4512a0ef7bSWill Deacon #endif
4612a0ef7bSWill Deacon 
477bc13fd3SWill Deacon /*
487bc13fd3SWill Deacon  * Load an unaligned word from kernel space.
497bc13fd3SWill Deacon  *
507bc13fd3SWill Deacon  * In the (very unlikely) case of the word being a page-crosser
517bc13fd3SWill Deacon  * and the next page not being mapped, take the exception and
527bc13fd3SWill Deacon  * return zeroes in the non-existing part.
537bc13fd3SWill Deacon  */
load_unaligned_zeropad(const void * addr)547bc13fd3SWill Deacon static inline unsigned long load_unaligned_zeropad(const void *addr)
557bc13fd3SWill Deacon {
56753b3236SMark Rutland 	unsigned long ret;
577bc13fd3SWill Deacon 
58*2cc029a0SVincenzo Frascino 	__mte_enable_tco_async();
59e60beb95SVincenzo Frascino 
607bc13fd3SWill Deacon 	/* Load word from unaligned pointer addr */
617bc13fd3SWill Deacon 	asm(
62753b3236SMark Rutland 	"1:	ldr	%0, %2\n"
637bc13fd3SWill Deacon 	"2:\n"
64753b3236SMark Rutland 	_ASM_EXTABLE_LOAD_UNALIGNED_ZEROPAD(1b, 2b, %0, %1)
65753b3236SMark Rutland 	: "=&r" (ret)
667bc13fd3SWill Deacon 	: "r" (addr), "Q" (*(unsigned long *)addr));
677bc13fd3SWill Deacon 
68*2cc029a0SVincenzo Frascino 	__mte_disable_tco_async();
69e60beb95SVincenzo Frascino 
707bc13fd3SWill Deacon 	return ret;
717bc13fd3SWill Deacon }
727bc13fd3SWill Deacon 
7312a0ef7bSWill Deacon #endif /* __ASM_WORD_AT_A_TIME_H */
74