s390/fpu: decrease stack usage for some cases

The kernel_fpu structure has a quite large size of 520 bytes. In order to
reduce stack footprint introduce several kernel fpu structures with
different and also smaller sizes. This way every kernel fpu user must use
the correct variant. A compile time check verifies that the correct variant
is used.

There are several users which use only 16 instead of all 32 vector
registers. For those users the new kernel_fpu_16 structure with a size of
only 266 bytes can be used.

Signed-off-by: Heiko Carstens <hca@linux.ibm.com>
This commit is contained in:
Heiko Carstens 2024-02-03 11:45:17 +01:00
parent cad8c3abaa
commit 066c40918b
8 changed files with 97 additions and 43 deletions

View File

@ -22,7 +22,7 @@ static void chacha20_crypt_s390(u32 *state, u8 *dst, const u8 *src,
unsigned int nbytes, const u32 *key, unsigned int nbytes, const u32 *key,
u32 *counter) u32 *counter)
{ {
DECLARE_KERNEL_FPU_ONSTACK(vxstate); DECLARE_KERNEL_FPU_ONSTACK32(vxstate);
kernel_fpu_begin(&vxstate, KERNEL_VXR); kernel_fpu_begin(&vxstate, KERNEL_VXR);
chacha20_vx(dst, src, nbytes, key, counter); chacha20_vx(dst, src, nbytes, key, counter);

View File

@ -50,7 +50,7 @@ u32 crc32c_le_vgfm_16(u32 crc, unsigned char const *buf, size_t size);
unsigned char const *data, size_t datalen) \ unsigned char const *data, size_t datalen) \
{ \ { \
unsigned long prealign, aligned, remaining; \ unsigned long prealign, aligned, remaining; \
DECLARE_KERNEL_FPU_ONSTACK(vxstate); \ DECLARE_KERNEL_FPU_ONSTACK16(vxstate); \
\ \
if (datalen < VX_MIN_LEN + VX_ALIGN_MASK) \ if (datalen < VX_MIN_LEN + VX_ALIGN_MASK) \
return ___crc32_sw(crc, data, datalen); \ return ___crc32_sw(crc, data, datalen); \

View File

@ -16,14 +16,32 @@ struct fpu {
__vector128 vxrs[__NUM_VXRS] __aligned(8); __vector128 vxrs[__NUM_VXRS] __aligned(8);
}; };
/* In-kernel FPU state structure */ struct kernel_fpu_hdr {
struct kernel_fpu { int mask;
int mask; u32 fpc;
u32 fpc;
__vector128 vxrs[__NUM_VXRS] __aligned(8);
}; };
#define DECLARE_KERNEL_FPU_ONSTACK(name) \ struct kernel_fpu {
struct kernel_fpu name __uninitialized struct kernel_fpu_hdr hdr;
__vector128 vxrs[] __aligned(8);
};
#define KERNEL_FPU_STRUCT(vxr_size) \
struct kernel_fpu_##vxr_size { \
struct kernel_fpu_hdr hdr; \
__vector128 vxrs[vxr_size] __aligned(8); \
}
KERNEL_FPU_STRUCT(16);
KERNEL_FPU_STRUCT(32);
#define DECLARE_KERNEL_FPU_ONSTACK(vxr_size, name) \
struct kernel_fpu_##vxr_size name __uninitialized
#define DECLARE_KERNEL_FPU_ONSTACK16(name) \
DECLARE_KERNEL_FPU_ONSTACK(16, name)
#define DECLARE_KERNEL_FPU_ONSTACK32(name) \
DECLARE_KERNEL_FPU_ONSTACK(32, name)
#endif /* _ASM_S390_FPU_TYPES_H */ #endif /* _ASM_S390_FPU_TYPES_H */

View File

@ -162,28 +162,64 @@ static __always_inline void load_fp_regs_vx(__vector128 *vxrs)
__load_fp_regs(fprs, sizeof(__vector128) / sizeof(freg_t)); __load_fp_regs(fprs, sizeof(__vector128) / sizeof(freg_t));
} }
static inline void kernel_fpu_begin(struct kernel_fpu *state, int flags) static inline void _kernel_fpu_begin(struct kernel_fpu *state, int flags)
{ {
state->mask = READ_ONCE(current->thread.kfpu_flags); state->hdr.mask = READ_ONCE(current->thread.kfpu_flags);
if (!test_thread_flag(TIF_FPU)) { if (!test_thread_flag(TIF_FPU)) {
/* Save user space FPU state and register contents */ /* Save user space FPU state and register contents */
save_user_fpu_regs(); save_user_fpu_regs();
} else if (state->mask & flags) { } else if (state->hdr.mask & flags) {
/* Save FPU/vector register in-use by the kernel */ /* Save FPU/vector register in-use by the kernel */
__kernel_fpu_begin(state, flags); __kernel_fpu_begin(state, flags);
} }
__atomic_or(flags, &current->thread.kfpu_flags); __atomic_or(flags, &current->thread.kfpu_flags);
} }
static inline void kernel_fpu_end(struct kernel_fpu *state, int flags) static inline void _kernel_fpu_end(struct kernel_fpu *state, int flags)
{ {
WRITE_ONCE(current->thread.kfpu_flags, state->mask); WRITE_ONCE(current->thread.kfpu_flags, state->hdr.mask);
if (state->mask & flags) { if (state->hdr.mask & flags) {
/* Restore FPU/vector register in-use by the kernel */ /* Restore FPU/vector register in-use by the kernel */
__kernel_fpu_end(state, flags); __kernel_fpu_end(state, flags);
} }
} }
void __kernel_fpu_invalid_size(void);
static __always_inline void kernel_fpu_check_size(int flags, unsigned int size)
{
unsigned int cnt = 0;
if (flags & KERNEL_VXR_V0V7)
cnt += 8;
if (flags & KERNEL_VXR_V8V15)
cnt += 8;
if (flags & KERNEL_VXR_V16V23)
cnt += 8;
if (flags & KERNEL_VXR_V24V31)
cnt += 8;
if (cnt != size)
__kernel_fpu_invalid_size();
}
#define kernel_fpu_begin(state, flags) \
{ \
typeof(state) s = (state); \
int _flags = (flags); \
\
kernel_fpu_check_size(_flags, ARRAY_SIZE(s->vxrs)); \
_kernel_fpu_begin((struct kernel_fpu *)s, _flags); \
}
#define kernel_fpu_end(state, flags) \
{ \
typeof(state) s = (state); \
int _flags = (flags); \
\
kernel_fpu_check_size(_flags, ARRAY_SIZE(s->vxrs)); \
_kernel_fpu_end((struct kernel_fpu *)s, _flags); \
}
static inline void save_kernel_fpu_regs(struct thread_struct *thread) static inline void save_kernel_fpu_regs(struct thread_struct *thread)
{ {
struct fpu *state = &thread->kfpu; struct fpu *state = &thread->kfpu;

View File

@ -19,41 +19,41 @@ void __kernel_fpu_begin(struct kernel_fpu *state, int flags)
* Limit the save to the FPU/vector registers already * Limit the save to the FPU/vector registers already
* in use by the previous context. * in use by the previous context.
*/ */
flags &= state->mask; flags &= state->hdr.mask;
if (flags & KERNEL_FPC) if (flags & KERNEL_FPC)
fpu_stfpc(&state->fpc); fpu_stfpc(&state->hdr.fpc);
if (!cpu_has_vx()) { if (!cpu_has_vx()) {
if (flags & KERNEL_VXR_LOW) if (flags & KERNEL_VXR_LOW)
save_fp_regs_vx(state->vxrs); save_fp_regs_vx(vxrs);
return; return;
} }
mask = flags & KERNEL_VXR; mask = flags & KERNEL_VXR;
if (mask == KERNEL_VXR) { if (mask == KERNEL_VXR) {
fpu_vstm(0, 15, &vxrs[0]); vxrs += fpu_vstm(0, 15, vxrs);
fpu_vstm(16, 31, &vxrs[16]); vxrs += fpu_vstm(16, 31, vxrs);
return; return;
} }
if (mask == KERNEL_VXR_MID) { if (mask == KERNEL_VXR_MID) {
fpu_vstm(8, 23, &vxrs[8]); vxrs += fpu_vstm(8, 23, vxrs);
return; return;
} }
mask = flags & KERNEL_VXR_LOW; mask = flags & KERNEL_VXR_LOW;
if (mask) { if (mask) {
if (mask == KERNEL_VXR_LOW) if (mask == KERNEL_VXR_LOW)
fpu_vstm(0, 15, &vxrs[0]); vxrs += fpu_vstm(0, 15, vxrs);
else if (mask == KERNEL_VXR_V0V7) else if (mask == KERNEL_VXR_V0V7)
fpu_vstm(0, 7, &vxrs[0]); vxrs += fpu_vstm(0, 7, vxrs);
else else
fpu_vstm(8, 15, &vxrs[8]); vxrs += fpu_vstm(8, 15, vxrs);
} }
mask = flags & KERNEL_VXR_HIGH; mask = flags & KERNEL_VXR_HIGH;
if (mask) { if (mask) {
if (mask == KERNEL_VXR_HIGH) if (mask == KERNEL_VXR_HIGH)
fpu_vstm(16, 31, &vxrs[16]); vxrs += fpu_vstm(16, 31, vxrs);
else if (mask == KERNEL_VXR_V16V23) else if (mask == KERNEL_VXR_V16V23)
fpu_vstm(16, 23, &vxrs[16]); vxrs += fpu_vstm(16, 23, vxrs);
else else
fpu_vstm(24, 31, &vxrs[24]); vxrs += fpu_vstm(24, 31, vxrs);
} }
} }
EXPORT_SYMBOL(__kernel_fpu_begin); EXPORT_SYMBOL(__kernel_fpu_begin);
@ -68,41 +68,41 @@ void __kernel_fpu_end(struct kernel_fpu *state, int flags)
* previous context that have been overwritten by the * previous context that have been overwritten by the
* current context. * current context.
*/ */
flags &= state->mask; flags &= state->hdr.mask;
if (flags & KERNEL_FPC) if (flags & KERNEL_FPC)
fpu_lfpc(&state->fpc); fpu_lfpc(&state->hdr.fpc);
if (!cpu_has_vx()) { if (!cpu_has_vx()) {
if (flags & KERNEL_VXR_LOW) if (flags & KERNEL_VXR_LOW)
load_fp_regs_vx(state->vxrs); load_fp_regs_vx(vxrs);
return; return;
} }
mask = flags & KERNEL_VXR; mask = flags & KERNEL_VXR;
if (mask == KERNEL_VXR) { if (mask == KERNEL_VXR) {
fpu_vlm(0, 15, &vxrs[0]); vxrs += fpu_vlm(0, 15, vxrs);
fpu_vlm(16, 31, &vxrs[16]); vxrs += fpu_vlm(16, 31, vxrs);
return; return;
} }
if (mask == KERNEL_VXR_MID) { if (mask == KERNEL_VXR_MID) {
fpu_vlm(8, 23, &vxrs[8]); vxrs += fpu_vlm(8, 23, vxrs);
return; return;
} }
mask = flags & KERNEL_VXR_LOW; mask = flags & KERNEL_VXR_LOW;
if (mask) { if (mask) {
if (mask == KERNEL_VXR_LOW) if (mask == KERNEL_VXR_LOW)
fpu_vlm(0, 15, &vxrs[0]); vxrs += fpu_vlm(0, 15, vxrs);
else if (mask == KERNEL_VXR_V0V7) else if (mask == KERNEL_VXR_V0V7)
fpu_vlm(0, 7, &vxrs[0]); vxrs += fpu_vlm(0, 7, vxrs);
else else
fpu_vlm(8, 15, &vxrs[8]); vxrs += fpu_vlm(8, 15, vxrs);
} }
mask = flags & KERNEL_VXR_HIGH; mask = flags & KERNEL_VXR_HIGH;
if (mask) { if (mask) {
if (mask == KERNEL_VXR_HIGH) if (mask == KERNEL_VXR_HIGH)
fpu_vlm(16, 31, &vxrs[16]); vxrs += fpu_vlm(16, 31, vxrs);
else if (mask == KERNEL_VXR_V16V23) else if (mask == KERNEL_VXR_V16V23)
fpu_vlm(16, 23, &vxrs[16]); vxrs += fpu_vlm(16, 23, vxrs);
else else
fpu_vlm(24, 31, &vxrs[24]); vxrs += fpu_vlm(24, 31, vxrs);
} }
} }
EXPORT_SYMBOL(__kernel_fpu_end); EXPORT_SYMBOL(__kernel_fpu_end);

View File

@ -426,7 +426,7 @@ subsys_initcall(create_proc_service_level);
*/ */
void s390_adjust_jiffies(void) void s390_adjust_jiffies(void)
{ {
DECLARE_KERNEL_FPU_ONSTACK(fpu); DECLARE_KERNEL_FPU_ONSTACK16(fpu);
struct sysinfo_1_2_2 *info; struct sysinfo_1_2_2 *info;
unsigned long capability; unsigned long capability;

View File

@ -5026,7 +5026,7 @@ static void store_regs(struct kvm_vcpu *vcpu)
int kvm_arch_vcpu_ioctl_run(struct kvm_vcpu *vcpu) int kvm_arch_vcpu_ioctl_run(struct kvm_vcpu *vcpu)
{ {
struct kvm_run *kvm_run = vcpu->run; struct kvm_run *kvm_run = vcpu->run;
DECLARE_KERNEL_FPU_ONSTACK(fpu); DECLARE_KERNEL_FPU_ONSTACK32(fpu);
int rc; int rc;
/* /*

View File

@ -80,7 +80,7 @@ static inline void COPY_VEC(int x, int y)
static void raid6_s390vx$#_gen_syndrome(int disks, size_t bytes, void **ptrs) static void raid6_s390vx$#_gen_syndrome(int disks, size_t bytes, void **ptrs)
{ {
DECLARE_KERNEL_FPU_ONSTACK(vxstate); DECLARE_KERNEL_FPU_ONSTACK32(vxstate);
u8 **dptr, *p, *q; u8 **dptr, *p, *q;
int d, z, z0; int d, z, z0;
@ -113,7 +113,7 @@ static void raid6_s390vx$#_gen_syndrome(int disks, size_t bytes, void **ptrs)
static void raid6_s390vx$#_xor_syndrome(int disks, int start, int stop, static void raid6_s390vx$#_xor_syndrome(int disks, int start, int stop,
size_t bytes, void **ptrs) size_t bytes, void **ptrs)
{ {
DECLARE_KERNEL_FPU_ONSTACK(vxstate); DECLARE_KERNEL_FPU_ONSTACK32(vxstate);
u8 **dptr, *p, *q; u8 **dptr, *p, *q;
int d, z, z0; int d, z, z0;