| /* |
| * arch/xtensa/lib/usercopy.S |
| * |
| * Copy to/from user space (derived from arch/xtensa/lib/hal/memcopy.S) |
| * |
| * DO NOT COMBINE this function with <arch/xtensa/lib/hal/memcopy.S>. |
| * It needs to remain separate and distinct. The hal files are part |
| * of the Xtensa link-time HAL, and those files may differ per |
| * processor configuration. Patching the kernel for another |
| * processor configuration includes replacing the hal files, and we |
| * could lose the special functionality for accessing user-space |
| * memory during such a patch. We sacrifice a little code space here |
| * in favor to simplify code maintenance. |
| * |
| * This file is subject to the terms and conditions of the GNU General |
| * Public License. See the file "COPYING" in the main directory of |
| * this archive for more details. |
| * |
| * Copyright (C) 2002 Tensilica Inc. |
| */ |
| |
| |
| /* |
| * size_t __xtensa_copy_user (void *dst, const void *src, size_t len); |
| * |
| * The returned value is the number of bytes not copied. Implies zero |
| * is success. |
| * |
| * The general case algorithm is as follows: |
| * If the destination and source are both aligned, |
| * do 16B chunks with a loop, and then finish up with |
| * 8B, 4B, 2B, and 1B copies conditional on the length. |
| * If destination is aligned and source unaligned, |
| * do the same, but use SRC to align the source data. |
| * If destination is unaligned, align it by conditionally |
| * copying 1B and 2B and then retest. |
| * This code tries to use fall-through braches for the common |
| * case of aligned destinations (except for the branches to |
| * the alignment label). |
| * |
| * Register use: |
| * a0/ return address |
| * a1/ stack pointer |
| * a2/ return value |
| * a3/ src |
| * a4/ length |
| * a5/ dst |
| * a6/ tmp |
| * a7/ tmp |
| * a8/ tmp |
| * a9/ tmp |
| * a10/ tmp |
| * a11/ original length |
| */ |
| |
| #include <variant/core.h> |
| |
| #ifdef __XTENSA_EB__ |
| #define ALIGN(R, W0, W1) src R, W0, W1 |
| #define SSA8(R) ssa8b R |
| #else |
| #define ALIGN(R, W0, W1) src R, W1, W0 |
| #define SSA8(R) ssa8l R |
| #endif |
| |
| /* Load or store instructions that may cause exceptions use the EX macro. */ |
| |
| #define EX(insn,reg1,reg2,offset,handler) \ |
| 9: insn reg1, reg2, offset; \ |
| .section __ex_table, "a"; \ |
| .word 9b, handler; \ |
| .previous |
| |
| |
| .text |
| .align 4 |
| .global __xtensa_copy_user |
| .type __xtensa_copy_user,@function |
| __xtensa_copy_user: |
| entry sp, 16 # minimal stack frame |
| # a2/ dst, a3/ src, a4/ len |
| mov a5, a2 # copy dst so that a2 is return value |
| mov a11, a4 # preserve original len for error case |
| .Lcommon: |
| bbsi.l a2, 0, .Ldst1mod2 # if dst is 1 mod 2 |
| bbsi.l a2, 1, .Ldst2mod4 # if dst is 2 mod 4 |
| .Ldstaligned: # return here from .Ldstunaligned when dst is aligned |
| srli a7, a4, 4 # number of loop iterations with 16B |
| # per iteration |
| movi a8, 3 # if source is also aligned, |
| bnone a3, a8, .Laligned # then use word copy |
| SSA8( a3) # set shift amount from byte offset |
| bnez a4, .Lsrcunaligned |
| movi a2, 0 # return success for len==0 |
| retw |
| |
| /* |
| * Destination is unaligned |
| */ |
| |
| .Ldst1mod2: # dst is only byte aligned |
| bltui a4, 7, .Lbytecopy # do short copies byte by byte |
| |
| # copy 1 byte |
| EX(l8ui, a6, a3, 0, l_fixup) |
| addi a3, a3, 1 |
| EX(s8i, a6, a5, 0, s_fixup) |
| addi a5, a5, 1 |
| addi a4, a4, -1 |
| bbci.l a5, 1, .Ldstaligned # if dst is now aligned, then |
| # return to main algorithm |
| .Ldst2mod4: # dst 16-bit aligned |
| # copy 2 bytes |
| bltui a4, 6, .Lbytecopy # do short copies byte by byte |
| EX(l8ui, a6, a3, 0, l_fixup) |
| EX(l8ui, a7, a3, 1, l_fixup) |
| addi a3, a3, 2 |
| EX(s8i, a6, a5, 0, s_fixup) |
| EX(s8i, a7, a5, 1, s_fixup) |
| addi a5, a5, 2 |
| addi a4, a4, -2 |
| j .Ldstaligned # dst is now aligned, return to main algorithm |
| |
| /* |
| * Byte by byte copy |
| */ |
| .align 4 |
| .byte 0 # 1 mod 4 alignment for LOOPNEZ |
| # (0 mod 4 alignment for LBEG) |
| .Lbytecopy: |
| #if XCHAL_HAVE_LOOPS |
| loopnez a4, .Lbytecopydone |
| #else /* !XCHAL_HAVE_LOOPS */ |
| beqz a4, .Lbytecopydone |
| add a7, a3, a4 # a7 = end address for source |
| #endif /* !XCHAL_HAVE_LOOPS */ |
| .Lnextbyte: |
| EX(l8ui, a6, a3, 0, l_fixup) |
| addi a3, a3, 1 |
| EX(s8i, a6, a5, 0, s_fixup) |
| addi a5, a5, 1 |
| #if !XCHAL_HAVE_LOOPS |
| blt a3, a7, .Lnextbyte |
| #endif /* !XCHAL_HAVE_LOOPS */ |
| .Lbytecopydone: |
| movi a2, 0 # return success for len bytes copied |
| retw |
| |
| /* |
| * Destination and source are word-aligned. |
| */ |
| # copy 16 bytes per iteration for word-aligned dst and word-aligned src |
| .align 4 # 1 mod 4 alignment for LOOPNEZ |
| .byte 0 # (0 mod 4 alignment for LBEG) |
| .Laligned: |
| #if XCHAL_HAVE_LOOPS |
| loopnez a7, .Loop1done |
| #else /* !XCHAL_HAVE_LOOPS */ |
| beqz a7, .Loop1done |
| slli a8, a7, 4 |
| add a8, a8, a3 # a8 = end of last 16B source chunk |
| #endif /* !XCHAL_HAVE_LOOPS */ |
| .Loop1: |
| EX(l32i, a6, a3, 0, l_fixup) |
| EX(l32i, a7, a3, 4, l_fixup) |
| EX(s32i, a6, a5, 0, s_fixup) |
| EX(l32i, a6, a3, 8, l_fixup) |
| EX(s32i, a7, a5, 4, s_fixup) |
| EX(l32i, a7, a3, 12, l_fixup) |
| EX(s32i, a6, a5, 8, s_fixup) |
| addi a3, a3, 16 |
| EX(s32i, a7, a5, 12, s_fixup) |
| addi a5, a5, 16 |
| #if !XCHAL_HAVE_LOOPS |
| blt a3, a8, .Loop1 |
| #endif /* !XCHAL_HAVE_LOOPS */ |
| .Loop1done: |
| bbci.l a4, 3, .L2 |
| # copy 8 bytes |
| EX(l32i, a6, a3, 0, l_fixup) |
| EX(l32i, a7, a3, 4, l_fixup) |
| addi a3, a3, 8 |
| EX(s32i, a6, a5, 0, s_fixup) |
| EX(s32i, a7, a5, 4, s_fixup) |
| addi a5, a5, 8 |
| .L2: |
| bbci.l a4, 2, .L3 |
| # copy 4 bytes |
| EX(l32i, a6, a3, 0, l_fixup) |
| addi a3, a3, 4 |
| EX(s32i, a6, a5, 0, s_fixup) |
| addi a5, a5, 4 |
| .L3: |
| bbci.l a4, 1, .L4 |
| # copy 2 bytes |
| EX(l16ui, a6, a3, 0, l_fixup) |
| addi a3, a3, 2 |
| EX(s16i, a6, a5, 0, s_fixup) |
| addi a5, a5, 2 |
| .L4: |
| bbci.l a4, 0, .L5 |
| # copy 1 byte |
| EX(l8ui, a6, a3, 0, l_fixup) |
| EX(s8i, a6, a5, 0, s_fixup) |
| .L5: |
| movi a2, 0 # return success for len bytes copied |
| retw |
| |
| /* |
| * Destination is aligned, Source is unaligned |
| */ |
| |
| .align 4 |
| .byte 0 # 1 mod 4 alignement for LOOPNEZ |
| # (0 mod 4 alignment for LBEG) |
| .Lsrcunaligned: |
| # copy 16 bytes per iteration for word-aligned dst and unaligned src |
| and a10, a3, a8 # save unalignment offset for below |
| sub a3, a3, a10 # align a3 (to avoid sim warnings only; not needed for hardware) |
| EX(l32i, a6, a3, 0, l_fixup) # load first word |
| #if XCHAL_HAVE_LOOPS |
| loopnez a7, .Loop2done |
| #else /* !XCHAL_HAVE_LOOPS */ |
| beqz a7, .Loop2done |
| slli a12, a7, 4 |
| add a12, a12, a3 # a12 = end of last 16B source chunk |
| #endif /* !XCHAL_HAVE_LOOPS */ |
| .Loop2: |
| EX(l32i, a7, a3, 4, l_fixup) |
| EX(l32i, a8, a3, 8, l_fixup) |
| ALIGN( a6, a6, a7) |
| EX(s32i, a6, a5, 0, s_fixup) |
| EX(l32i, a9, a3, 12, l_fixup) |
| ALIGN( a7, a7, a8) |
| EX(s32i, a7, a5, 4, s_fixup) |
| EX(l32i, a6, a3, 16, l_fixup) |
| ALIGN( a8, a8, a9) |
| EX(s32i, a8, a5, 8, s_fixup) |
| addi a3, a3, 16 |
| ALIGN( a9, a9, a6) |
| EX(s32i, a9, a5, 12, s_fixup) |
| addi a5, a5, 16 |
| #if !XCHAL_HAVE_LOOPS |
| blt a3, a12, .Loop2 |
| #endif /* !XCHAL_HAVE_LOOPS */ |
| .Loop2done: |
| bbci.l a4, 3, .L12 |
| # copy 8 bytes |
| EX(l32i, a7, a3, 4, l_fixup) |
| EX(l32i, a8, a3, 8, l_fixup) |
| ALIGN( a6, a6, a7) |
| EX(s32i, a6, a5, 0, s_fixup) |
| addi a3, a3, 8 |
| ALIGN( a7, a7, a8) |
| EX(s32i, a7, a5, 4, s_fixup) |
| addi a5, a5, 8 |
| mov a6, a8 |
| .L12: |
| bbci.l a4, 2, .L13 |
| # copy 4 bytes |
| EX(l32i, a7, a3, 4, l_fixup) |
| addi a3, a3, 4 |
| ALIGN( a6, a6, a7) |
| EX(s32i, a6, a5, 0, s_fixup) |
| addi a5, a5, 4 |
| mov a6, a7 |
| .L13: |
| add a3, a3, a10 # readjust a3 with correct misalignment |
| bbci.l a4, 1, .L14 |
| # copy 2 bytes |
| EX(l8ui, a6, a3, 0, l_fixup) |
| EX(l8ui, a7, a3, 1, l_fixup) |
| addi a3, a3, 2 |
| EX(s8i, a6, a5, 0, s_fixup) |
| EX(s8i, a7, a5, 1, s_fixup) |
| addi a5, a5, 2 |
| .L14: |
| bbci.l a4, 0, .L15 |
| # copy 1 byte |
| EX(l8ui, a6, a3, 0, l_fixup) |
| EX(s8i, a6, a5, 0, s_fixup) |
| .L15: |
| movi a2, 0 # return success for len bytes copied |
| retw |
| |
| |
| .section .fixup, "ax" |
| .align 4 |
| |
| /* a2 = original dst; a5 = current dst; a11= original len |
| * bytes_copied = a5 - a2 |
| * retval = bytes_not_copied = original len - bytes_copied |
| * retval = a11 - (a5 - a2) |
| * |
| * Clearing the remaining pieces of kernel memory plugs security |
| * holes. This functionality is the equivalent of the *_zeroing |
| * functions that some architectures provide. |
| */ |
| |
| .Lmemset: |
| .word memset |
| |
| s_fixup: |
| sub a2, a5, a2 /* a2 <-- bytes copied */ |
| sub a2, a11, a2 /* a2 <-- bytes not copied */ |
| retw |
| |
| l_fixup: |
| sub a2, a5, a2 /* a2 <-- bytes copied */ |
| sub a2, a11, a2 /* a2 <-- bytes not copied == return value */ |
| |
| /* void *memset(void *s, int c, size_t n); */ |
| mov a6, a5 /* s */ |
| movi a7, 0 /* c */ |
| mov a8, a2 /* n */ |
| l32r a4, .Lmemset |
| callx4 a4 |
| /* Ignore memset return value in a6. */ |
| /* a2 still contains bytes not copied. */ |
| retw |