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