Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

selftests/powerpc: Add a test of the switch_endian() syscall

This adds a test of the switch_endian() syscall we added in the previous
commit.

We test it by calling the endian switch syscall, and then executing some
code in the other endian to check everything went as expected. That code
checks registers we expect to be maintained are. If the endian switch
failed to happen that code sequence will be illegal and cause the test
to abort.

We then switch back to the original endian, do the same checks and
finally write a success message and exit(0).

Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>

+214 -1
+1 -1
tools/testing/selftests/powerpc/Makefile
··· 13 13 14 14 export CC CFLAGS 15 15 16 - SUB_DIRS = pmu copyloops mm tm primitives stringloops vphn 16 + SUB_DIRS = pmu copyloops mm tm primitives stringloops vphn switch_endian 17 17 18 18 endif 19 19
+2
tools/testing/selftests/powerpc/switch_endian/.gitignore
··· 1 + switch_endian_test 2 + check-reversed.S
+24
tools/testing/selftests/powerpc/switch_endian/Makefile
··· 1 + CC := $(CROSS_COMPILE)gcc 2 + PROGS := switch_endian_test 3 + 4 + ASFLAGS += -O2 -Wall -g -nostdlib -m64 5 + 6 + all: $(PROGS) 7 + 8 + switch_endian_test: check-reversed.S 9 + 10 + check-reversed.o: check.o 11 + $(CROSS_COMPILE)objcopy -j .text --reverse-bytes=4 -O binary $< $@ 12 + 13 + check-reversed.S: check-reversed.o 14 + hexdump -v -e '/1 ".byte 0x%02X\n"' $< > $@ 15 + 16 + run_tests: all 17 + @-for PROG in $(PROGS); do \ 18 + ./$$PROG; \ 19 + done; 20 + 21 + clean: 22 + rm -f $(PROGS) *.o check-reversed.S 23 + 24 + .PHONY: all run_tests clean
+100
tools/testing/selftests/powerpc/switch_endian/check.S
··· 1 + #include "common.h" 2 + 3 + /* 4 + * Checks that registers contain what we expect, ie. they were not clobbered by 5 + * the syscall. 6 + * 7 + * r15: pattern to check registers against. 8 + * 9 + * At the end r3 == 0 if everything's OK. 10 + */ 11 + nop # guaranteed to be illegal in reverse-endian 12 + mr r9,r15 13 + cmpd r9,r3 # check r3 14 + bne 1f 15 + addi r9,r15,4 # check r4 16 + cmpd r9,r4 17 + bne 1f 18 + lis r9,0x00FF # check CR 19 + ori r9,r9,0xF000 20 + mfcr r10 21 + and r10,r10,r9 22 + cmpw r9,r10 23 + addi r9,r15,34 24 + bne 1f 25 + addi r9,r15,32 # check LR 26 + mflr r10 27 + cmpd r9,r10 28 + bne 1f 29 + addi r9,r15,5 # check r5 30 + cmpd r9,r5 31 + bne 1f 32 + addi r9,r15,6 # check r6 33 + cmpd r9,r6 34 + bne 1f 35 + addi r9,r15,7 # check r7 36 + cmpd r9,r7 37 + bne 1f 38 + addi r9,r15,8 # check r8 39 + cmpd r9,r8 40 + bne 1f 41 + addi r9,r15,13 # check r13 42 + cmpd r9,r13 43 + bne 1f 44 + addi r9,r15,14 # check r14 45 + cmpd r9,r14 46 + bne 1f 47 + addi r9,r15,16 # check r16 48 + cmpd r9,r16 49 + bne 1f 50 + addi r9,r15,17 # check r17 51 + cmpd r9,r17 52 + bne 1f 53 + addi r9,r15,18 # check r18 54 + cmpd r9,r18 55 + bne 1f 56 + addi r9,r15,19 # check r19 57 + cmpd r9,r19 58 + bne 1f 59 + addi r9,r15,20 # check r20 60 + cmpd r9,r20 61 + bne 1f 62 + addi r9,r15,21 # check r21 63 + cmpd r9,r21 64 + bne 1f 65 + addi r9,r15,22 # check r22 66 + cmpd r9,r22 67 + bne 1f 68 + addi r9,r15,23 # check r23 69 + cmpd r9,r23 70 + bne 1f 71 + addi r9,r15,24 # check r24 72 + cmpd r9,r24 73 + bne 1f 74 + addi r9,r15,25 # check r25 75 + cmpd r9,r25 76 + bne 1f 77 + addi r9,r15,26 # check r26 78 + cmpd r9,r26 79 + bne 1f 80 + addi r9,r15,27 # check r27 81 + cmpd r9,r27 82 + bne 1f 83 + addi r9,r15,28 # check r28 84 + cmpd r9,r28 85 + bne 1f 86 + addi r9,r15,29 # check r29 87 + cmpd r9,r29 88 + bne 1f 89 + addi r9,r15,30 # check r30 90 + cmpd r9,r30 91 + bne 1f 92 + addi r9,r15,31 # check r31 93 + cmpd r9,r31 94 + bne 1f 95 + b 2f 96 + 1: mr r3, r9 97 + li r0, __NR_exit 98 + sc 99 + 2: li r0, __NR_switch_endian 100 + nop
+6
tools/testing/selftests/powerpc/switch_endian/common.h
··· 1 + #include <ppc-asm.h> 2 + #include <asm/unistd.h> 3 + 4 + #ifndef __NR_switch_endian 5 + #define __NR_switch_endian 363 6 + #endif
+81
tools/testing/selftests/powerpc/switch_endian/switch_endian_test.S
··· 1 + #include "common.h" 2 + 3 + .data 4 + .balign 8 5 + message: 6 + .ascii "success: switch_endian_test\n\0" 7 + 8 + .section ".toc" 9 + .balign 8 10 + pattern: 11 + .llong 0x5555AAAA5555AAAA 12 + 13 + .text 14 + FUNC_START(_start) 15 + /* Load the pattern */ 16 + ld r15, pattern@TOC(%r2) 17 + 18 + /* Setup CR, only CR2-CR4 are maintained */ 19 + lis r3, 0x00FF 20 + ori r3, r3, 0xF000 21 + mtcr r3 22 + 23 + /* Load the pattern slightly modified into the registers */ 24 + mr r3, r15 25 + addi r4, r15, 4 26 + 27 + addi r5, r15, 32 28 + mtlr r5 29 + 30 + addi r5, r15, 5 31 + addi r6, r15, 6 32 + addi r7, r15, 7 33 + addi r8, r15, 8 34 + 35 + /* r9 - r12 are clobbered */ 36 + 37 + addi r13, r15, 13 38 + addi r14, r15, 14 39 + 40 + /* Skip r15 we're using it */ 41 + 42 + addi r16, r15, 16 43 + addi r17, r15, 17 44 + addi r18, r15, 18 45 + addi r19, r15, 19 46 + addi r20, r15, 20 47 + addi r21, r15, 21 48 + addi r22, r15, 22 49 + addi r23, r15, 23 50 + addi r24, r15, 24 51 + addi r25, r15, 25 52 + addi r26, r15, 26 53 + addi r27, r15, 27 54 + addi r28, r15, 28 55 + addi r29, r15, 29 56 + addi r30, r15, 30 57 + addi r31, r15, 31 58 + 59 + /* 60 + * Call the syscall to switch endian. 61 + * It clobbers r9-r12, XER, CTR and CR0-1,5-7. 62 + */ 63 + li r0, __NR_switch_endian 64 + sc 65 + 66 + #include "check-reversed.S" 67 + 68 + /* Flip back, r0 already has the switch syscall number */ 69 + .long 0x02000044 /* sc */ 70 + 71 + #include "check.S" 72 + 73 + li r0, __NR_write 74 + li r3, 1 /* stdout */ 75 + ld r4, message@got(%r2) 76 + li r5, 28 /* strlen(message3) */ 77 + sc 78 + li r0, __NR_exit 79 + li r3, 0 80 + sc 81 + b .