Loading...
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 | /* * arch/alpha/lib/copy_user.S * * Copy to/from user space, handling exceptions as we go.. This * isn't exactly pretty. * * This is essentially the same as "memcpy()", but with a few twists. * Notably, we have to make sure that $0 is always up-to-date and * contains the right "bytes left to copy" value (and that it is updated * only _after_ a successful copy). There is also some rather minor * exception setup stuff.. * * NOTE! This is not directly C-callable, because the calling semantics are * different: * * Inputs: * length in $0 * destination address in $6 * source address in $7 * return address in $28 * * Outputs: * bytes left to copy in $0 * * Clobbers: * $1,$2,$3,$4,$5,$6,$7 */ /* Allow an exception for an insn; exit if we get one. */ #define EX(x,y...) \ 99: x,##y; \ .section __ex_table,"a"; \ .gprel32 99b; \ lda $31, $exit-99b($31); \ .previous .set noat .align 3 .globl __copy_user .ent __copy_user __copy_user: and $6,7,$3 beq $0,$35 beq $3,$36 subq $3,8,$3 .align 5 $37: EX( ldq_u $1,0($7) ) EX( ldq_u $2,0($6) ) extbl $1,$7,$1 mskbl $2,$6,$2 insbl $1,$6,$1 addq $3,1,$3 bis $1,$2,$1 EX( stq_u $1,0($6) ) subq $0,1,$0 addq $6,1,$6 addq $7,1,$7 beq $0,$41 bne $3,$37 $36: and $7,7,$1 bic $0,7,$4 beq $1,$43 beq $4,$48 EX( ldq_u $3,0($7) ) .align 5 $50: EX( ldq_u $2,8($7) ) subq $4,8,$4 extql $3,$7,$3 extqh $2,$7,$1 bis $3,$1,$1 stq $1,0($6) addq $7,8,$7 subq $0,8,$0 addq $6,8,$6 bis $2,$2,$3 bne $4,$50 $48: beq $0,$41 .align 5 $57: EX( ldq_u $1,0($7) ) EX( ldq_u $2,0($6) ) extbl $1,$7,$1 mskbl $2,$6,$2 insbl $1,$6,$1 bis $1,$2,$1 EX( stq_u $1,0($6) ) subq $0,1,$0 addq $6,1,$6 addq $7,1,$7 bne $0,$57 br $31,$41 .align 4 $43: beq $4,$65 .align 5 $66: EX( ldq $1,0($7) ) subq $4,8,$4 stq $1,0($6) addq $7,8,$7 subq $0,8,$0 addq $6,8,$6 bne $4,$66 $65: beq $0,$41 EX( ldq $2,0($7) ) EX( ldq $1,0($6) ) mskql $2,$0,$2 mskqh $1,$0,$1 bis $2,$1,$2 EX( stq $2,0($6) ) bis $31,$31,$0 $41: $35: $exit: ret $31,($28),1 .end __copy_user |