-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
ARM: 7449/1: use generic strnlen_user and strncpy_from_user functions
This patch implements the word-at-a-time interface for ARM using the same algorithm as x86. We use the fls macro from ARMv5 onwards, where we have a clz instruction available which saves us a mov instruction when targetting Thumb-2. For older CPUs, we use the magic 0x0ff0001 constant. Big-endian configurations make use of the implementation from asm-generic. With this implemented, we can replace our byte-at-a-time strnlen_user and strncpy_from_user functions with the optimised generic versions. Reviewed-by: Nicolas Pitre <nico@linaro.org> Signed-off-by: Will Deacon <will.deacon@arm.com> Signed-off-by: Russell King <rmk+kernel@arm.linux.org.uk>
- Loading branch information
Will Deacon
authored and
Russell King
committed
Jul 9, 2012
1 parent
4295b89
commit 8c56cc8
Showing
7 changed files
with
63 additions
and
109 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,55 @@ | ||
#ifndef __ASM_ARM_WORD_AT_A_TIME_H | ||
#define __ASM_ARM_WORD_AT_A_TIME_H | ||
|
||
#ifndef __ARMEB__ | ||
|
||
/* | ||
* Little-endian word-at-a-time zero byte handling. | ||
* Heavily based on the x86 algorithm. | ||
*/ | ||
#include <linux/kernel.h> | ||
|
||
struct word_at_a_time { | ||
const unsigned long one_bits, high_bits; | ||
}; | ||
|
||
#define WORD_AT_A_TIME_CONSTANTS { REPEAT_BYTE(0x01), REPEAT_BYTE(0x80) } | ||
|
||
static inline unsigned long has_zero(unsigned long a, unsigned long *bits, | ||
const struct word_at_a_time *c) | ||
{ | ||
unsigned long mask = ((a - c->one_bits) & ~a) & c->high_bits; | ||
*bits = mask; | ||
return mask; | ||
} | ||
|
||
#define prep_zero_mask(a, bits, c) (bits) | ||
|
||
static inline unsigned long create_zero_mask(unsigned long bits) | ||
{ | ||
bits = (bits - 1) & ~bits; | ||
return bits >> 7; | ||
} | ||
|
||
static inline unsigned long find_zero(unsigned long mask) | ||
{ | ||
unsigned long ret; | ||
|
||
#if __LINUX_ARM_ARCH__ >= 5 | ||
/* We have clz available. */ | ||
ret = fls(mask) >> 3; | ||
#else | ||
/* (000000 0000ff 00ffff ffffff) -> ( 1 1 2 3 ) */ | ||
ret = (0x0ff0001 + mask) >> 23; | ||
/* Fix the 1 for 00 case */ | ||
ret &= mask; | ||
#endif | ||
|
||
return ret; | ||
} | ||
|
||
#else /* __ARMEB__ */ | ||
#include <asm-generic/word-at-a-time.h> | ||
#endif | ||
|
||
#endif /* __ASM_ARM_WORD_AT_A_TIME_H */ |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file was deleted.
Oops, something went wrong.
This file was deleted.
Oops, something went wrong.