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
| /* SPDX-License-Identifier: GPL-2.0 */
| /*
| * arch/alpha/lib/strncpy.S
| * Contributed by Richard Henderson (rth@tamu.edu)
| *
| * Copy no more than COUNT bytes of the null-terminated string from
| * SRC to DST. If SRC does not cover all of COUNT, the balance is
| * zeroed.
| *
| * Or, rather, if the kernel cared about that weird ANSI quirk. This
| * version has cropped that bit o' nastiness as well as assuming that
| * __stxncpy is in range of a branch.
| */
| #include <asm/export.h>
| .set noat
| .set noreorder
|
| .text
|
| .align 4
| .globl strncpy
| .ent strncpy
| strncpy:
| .frame $30, 0, $26
| .prologue 0
|
| mov $16, $0 # set return value now
| beq $18, $zerolen
| unop
| bsr $23, __stxncpy # do the work of the copy
|
| unop
| bne $18, $multiword # do we have full words left?
| subq $24, 1, $3 # nope
| subq $27, 1, $4
|
| or $3, $24, $3 # clear the bits between the last
| or $4, $27, $4 # written byte and the last byte in COUNT
| andnot $3, $4, $4
| zap $1, $4, $1
|
| stq_u $1, 0($16)
| ret
|
| .align 4
| $multiword:
| subq $27, 1, $2 # clear the final bits in the prev word
| or $2, $27, $2
| zapnot $1, $2, $1
| subq $18, 1, $18
|
| stq_u $1, 0($16)
| addq $16, 8, $16
| unop
| beq $18, 1f
|
| nop
| unop
| nop
| blbc $18, 0f
|
| stq_u $31, 0($16) # zero one word
| subq $18, 1, $18
| addq $16, 8, $16
| beq $18, 1f
|
| 0: stq_u $31, 0($16) # zero two words
| subq $18, 2, $18
| stq_u $31, 8($16)
| addq $16, 16, $16
| bne $18, 0b
|
| 1: ldq_u $1, 0($16) # clear the leading bits in the final word
| subq $24, 1, $2
| or $2, $24, $2
|
| zap $1, $2, $1
| stq_u $1, 0($16)
| $zerolen:
| ret
|
| .end strncpy
| EXPORT_SYMBOL(strncpy)
|
|