]> git.proxmox.com Git - mirror_ubuntu-hirsute-kernel.git/blob - arch/arm64/kernel/vdso/gettimeofday.S
Merge remote-tracking branches 'regmap/fix/irq', 'regmap/fix/rbtree' and 'regmap...
[mirror_ubuntu-hirsute-kernel.git] / arch / arm64 / kernel / vdso / gettimeofday.S
1 /*
2 * Userspace implementations of gettimeofday() and friends.
3 *
4 * Copyright (C) 2012 ARM Limited
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License version 2 as
8 * published by the Free Software Foundation.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program. If not, see <http://www.gnu.org/licenses/>.
17 *
18 * Author: Will Deacon <will.deacon@arm.com>
19 */
20
21 #include <linux/linkage.h>
22 #include <asm/asm-offsets.h>
23 #include <asm/unistd.h>
24
25 #define NSEC_PER_SEC_LO16 0xca00
26 #define NSEC_PER_SEC_HI16 0x3b9a
27
28 vdso_data .req x6
29 use_syscall .req w7
30 seqcnt .req w8
31
32 .macro seqcnt_acquire
33 9999: ldr seqcnt, [vdso_data, #VDSO_TB_SEQ_COUNT]
34 tbnz seqcnt, #0, 9999b
35 dmb ishld
36 ldr use_syscall, [vdso_data, #VDSO_USE_SYSCALL]
37 .endm
38
39 .macro seqcnt_read, cnt
40 dmb ishld
41 ldr \cnt, [vdso_data, #VDSO_TB_SEQ_COUNT]
42 .endm
43
44 .macro seqcnt_check, cnt, fail
45 cmp \cnt, seqcnt
46 b.ne \fail
47 .endm
48
49 .text
50
51 /* int __kernel_gettimeofday(struct timeval *tv, struct timezone *tz); */
52 ENTRY(__kernel_gettimeofday)
53 .cfi_startproc
54 mov x2, x30
55 .cfi_register x30, x2
56
57 /* Acquire the sequence counter and get the timespec. */
58 adr vdso_data, _vdso_data
59 1: seqcnt_acquire
60 cbnz use_syscall, 4f
61
62 /* If tv is NULL, skip to the timezone code. */
63 cbz x0, 2f
64 bl __do_get_tspec
65 seqcnt_check w9, 1b
66
67 /* Convert ns to us. */
68 mov x13, #1000
69 lsl x13, x13, x12
70 udiv x11, x11, x13
71 stp x10, x11, [x0, #TVAL_TV_SEC]
72 2:
73 /* If tz is NULL, return 0. */
74 cbz x1, 3f
75 ldp w4, w5, [vdso_data, #VDSO_TZ_MINWEST]
76 stp w4, w5, [x1, #TZ_MINWEST]
77 3:
78 mov x0, xzr
79 ret x2
80 4:
81 /* Syscall fallback. */
82 mov x8, #__NR_gettimeofday
83 svc #0
84 ret x2
85 .cfi_endproc
86 ENDPROC(__kernel_gettimeofday)
87
88 /* int __kernel_clock_gettime(clockid_t clock_id, struct timespec *tp); */
89 ENTRY(__kernel_clock_gettime)
90 .cfi_startproc
91 cmp w0, #CLOCK_REALTIME
92 ccmp w0, #CLOCK_MONOTONIC, #0x4, ne
93 b.ne 2f
94
95 mov x2, x30
96 .cfi_register x30, x2
97
98 /* Get kernel timespec. */
99 adr vdso_data, _vdso_data
100 1: seqcnt_acquire
101 cbnz use_syscall, 7f
102
103 bl __do_get_tspec
104 seqcnt_check w9, 1b
105
106 mov x30, x2
107
108 cmp w0, #CLOCK_MONOTONIC
109 b.ne 6f
110
111 /* Get wtm timespec. */
112 ldp x13, x14, [vdso_data, #VDSO_WTM_CLK_SEC]
113
114 /* Check the sequence counter. */
115 seqcnt_read w9
116 seqcnt_check w9, 1b
117 b 4f
118 2:
119 cmp w0, #CLOCK_REALTIME_COARSE
120 ccmp w0, #CLOCK_MONOTONIC_COARSE, #0x4, ne
121 b.ne 8f
122
123 /* xtime_coarse_nsec is already right-shifted */
124 mov x12, #0
125
126 /* Get coarse timespec. */
127 adr vdso_data, _vdso_data
128 3: seqcnt_acquire
129 ldp x10, x11, [vdso_data, #VDSO_XTIME_CRS_SEC]
130
131 /* Get wtm timespec. */
132 ldp x13, x14, [vdso_data, #VDSO_WTM_CLK_SEC]
133
134 /* Check the sequence counter. */
135 seqcnt_read w9
136 seqcnt_check w9, 3b
137
138 cmp w0, #CLOCK_MONOTONIC_COARSE
139 b.ne 6f
140 4:
141 /* Add on wtm timespec. */
142 add x10, x10, x13
143 lsl x14, x14, x12
144 add x11, x11, x14
145
146 /* Normalise the new timespec. */
147 mov x15, #NSEC_PER_SEC_LO16
148 movk x15, #NSEC_PER_SEC_HI16, lsl #16
149 lsl x15, x15, x12
150 cmp x11, x15
151 b.lt 5f
152 sub x11, x11, x15
153 add x10, x10, #1
154 5:
155 cmp x11, #0
156 b.ge 6f
157 add x11, x11, x15
158 sub x10, x10, #1
159
160 6: /* Store to the user timespec. */
161 lsr x11, x11, x12
162 stp x10, x11, [x1, #TSPEC_TV_SEC]
163 mov x0, xzr
164 ret
165 7:
166 mov x30, x2
167 8: /* Syscall fallback. */
168 mov x8, #__NR_clock_gettime
169 svc #0
170 ret
171 .cfi_endproc
172 ENDPROC(__kernel_clock_gettime)
173
174 /* int __kernel_clock_getres(clockid_t clock_id, struct timespec *res); */
175 ENTRY(__kernel_clock_getres)
176 .cfi_startproc
177 cmp w0, #CLOCK_REALTIME
178 ccmp w0, #CLOCK_MONOTONIC, #0x4, ne
179 b.ne 1f
180
181 ldr x2, 5f
182 b 2f
183 1:
184 cmp w0, #CLOCK_REALTIME_COARSE
185 ccmp w0, #CLOCK_MONOTONIC_COARSE, #0x4, ne
186 b.ne 4f
187 ldr x2, 6f
188 2:
189 cbz w1, 3f
190 stp xzr, x2, [x1]
191
192 3: /* res == NULL. */
193 mov w0, wzr
194 ret
195
196 4: /* Syscall fallback. */
197 mov x8, #__NR_clock_getres
198 svc #0
199 ret
200 5:
201 .quad CLOCK_REALTIME_RES
202 6:
203 .quad CLOCK_COARSE_RES
204 .cfi_endproc
205 ENDPROC(__kernel_clock_getres)
206
207 /*
208 * Read the current time from the architected counter.
209 * Expects vdso_data to be initialised.
210 * Clobbers the temporary registers (x9 - x15).
211 * Returns:
212 * - w9 = vDSO sequence counter
213 * - (x10, x11) = (ts->tv_sec, shifted ts->tv_nsec)
214 * - w12 = cs_shift
215 */
216 ENTRY(__do_get_tspec)
217 .cfi_startproc
218
219 /* Read from the vDSO data page. */
220 ldr x10, [vdso_data, #VDSO_CS_CYCLE_LAST]
221 ldp x13, x14, [vdso_data, #VDSO_XTIME_CLK_SEC]
222 ldp w11, w12, [vdso_data, #VDSO_CS_MULT]
223 seqcnt_read w9
224
225 /* Read the virtual counter. */
226 isb
227 mrs x15, cntvct_el0
228
229 /* Calculate cycle delta and convert to ns. */
230 sub x10, x15, x10
231 /* We can only guarantee 56 bits of precision. */
232 movn x15, #0xff00, lsl #48
233 and x10, x15, x10
234 mul x10, x10, x11
235
236 /* Use the kernel time to calculate the new timespec. */
237 mov x11, #NSEC_PER_SEC_LO16
238 movk x11, #NSEC_PER_SEC_HI16, lsl #16
239 lsl x11, x11, x12
240 add x15, x10, x14
241 udiv x14, x15, x11
242 add x10, x13, x14
243 mul x13, x14, x11
244 sub x11, x15, x13
245
246 ret
247 .cfi_endproc
248 ENDPROC(__do_get_tspec)