xref: /xnu-8792.81.2/libsyscall/wrappers/mach_approximate_time.s (revision 19c3b8c28c31cb8130e034cfb5df6bf9ba342d90)
1*19c3b8c2SApple OSS Distributions/*
2*19c3b8c2SApple OSS Distributions * Copyright (c) 2003-2007 Apple Inc. All rights reserved.
3*19c3b8c2SApple OSS Distributions *
4*19c3b8c2SApple OSS Distributions * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
5*19c3b8c2SApple OSS Distributions *
6*19c3b8c2SApple OSS Distributions * This file contains Original Code and/or Modifications of Original Code
7*19c3b8c2SApple OSS Distributions * as defined in and that are subject to the Apple Public Source License
8*19c3b8c2SApple OSS Distributions * Version 2.0 (the 'License'). You may not use this file except in
9*19c3b8c2SApple OSS Distributions * compliance with the License. The rights granted to you under the License
10*19c3b8c2SApple OSS Distributions * may not be used to create, or enable the creation or redistribution of,
11*19c3b8c2SApple OSS Distributions * unlawful or unlicensed copies of an Apple operating system, or to
12*19c3b8c2SApple OSS Distributions * circumvent, violate, or enable the circumvention or violation of, any
13*19c3b8c2SApple OSS Distributions * terms of an Apple operating system software license agreement.
14*19c3b8c2SApple OSS Distributions *
15*19c3b8c2SApple OSS Distributions * Please obtain a copy of the License at
16*19c3b8c2SApple OSS Distributions * http://www.opensource.apple.com/apsl/ and read it before using this file.
17*19c3b8c2SApple OSS Distributions *
18*19c3b8c2SApple OSS Distributions * The Original Code and all software distributed under the License are
19*19c3b8c2SApple OSS Distributions * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20*19c3b8c2SApple OSS Distributions * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21*19c3b8c2SApple OSS Distributions * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22*19c3b8c2SApple OSS Distributions * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23*19c3b8c2SApple OSS Distributions * Please see the License for the specific language governing rights and
24*19c3b8c2SApple OSS Distributions * limitations under the License.
25*19c3b8c2SApple OSS Distributions *
26*19c3b8c2SApple OSS Distributions * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
27*19c3b8c2SApple OSS Distributions */
28*19c3b8c2SApple OSS Distributions
29*19c3b8c2SApple OSS Distributions#include <sys/appleapiopts.h>
30*19c3b8c2SApple OSS Distributions#include <machine/cpu_capabilities.h>
31*19c3b8c2SApple OSS Distributions
32*19c3b8c2SApple OSS Distributions#if defined(__arm__)
33*19c3b8c2SApple OSS Distributions
34*19c3b8c2SApple OSS Distributions	.text
35*19c3b8c2SApple OSS Distributions	.align	2
36*19c3b8c2SApple OSS Distributions	.globl	_mach_approximate_time
37*19c3b8c2SApple OSS Distributions_mach_approximate_time:
38*19c3b8c2SApple OSS Distributions
39*19c3b8c2SApple OSS Distributions	movw	r0, #((_COMM_PAGE_APPROX_TIME_SUPPORTED>>0)&0x0FFFF)
40*19c3b8c2SApple OSS Distributions	movt	r0, #((_COMM_PAGE_APPROX_TIME_SUPPORTED>>16)&0x0FFFF)
41*19c3b8c2SApple OSS Distributions	ldrb	r0, [r0]		// load COMM_PAGE_APPROX_TIME_SUPPORTED
42*19c3b8c2SApple OSS Distributions	cmp	r0, #1			// check if approx time is supported
43*19c3b8c2SApple OSS Distributions
44*19c3b8c2SApple OSS Distributions        bne      _mach_absolute_time	// if not supported, fall through to
45*19c3b8c2SApple OSS Distributions					//   absolute_time
46*19c3b8c2SApple OSS Distributions
47*19c3b8c2SApple OSS Distributions	movw	r2, #((_COMM_PAGE_APPROX_TIME>>0)&0x0FFFF)
48*19c3b8c2SApple OSS Distributions	movt	r2, #((_COMM_PAGE_APPROX_TIME>>16)&0x0FFFF)
49*19c3b8c2SApple OSS Distributions
50*19c3b8c2SApple OSS Distributions        // at this point, r2->COMM_PAGE_APPROX_TIME, which is a 64-bit value.
51*19c3b8c2SApple OSS Distributions        // Since this is a 32-bit architecture, and the commpage is readonly,
52*19c3b8c2SApple OSS Distributions        // there is no "guaranteed" atomic way to read all 64-bits with
53*19c3b8c2SApple OSS Distributions        // hardware protection. Even the ldrd instruction is not guaranteed to
54*19c3b8c2SApple OSS Distributions        // complete atomically. The solution is to use a 32-bit high/low/high
55*19c3b8c2SApple OSS Distributions        // read with a consistency check on the high bits. To further
56*19c3b8c2SApple OSS Distributions        // complicate things, reading the same location in memory back to back
57*19c3b8c2SApple OSS Distributions        // could trigger a predictive read, which would defeat the purpose of
58*19c3b8c2SApple OSS Distributions        // doing the consistency check so we insert a data memory barrier to
59*19c3b8c2SApple OSS Distributions        // prevent this.
60*19c3b8c2SApple OSS Distributions_consistency_check:
61*19c3b8c2SApple OSS Distributions        ldr     r1, [r2,#4]		// load high
62*19c3b8c2SApple OSS Distributions        ldr     r0, [r2]		// load low
63*19c3b8c2SApple OSS Distributions        dsb				// avoid predictive reads that could
64*19c3b8c2SApple OSS Distributions					//   be invalid if interrupted
65*19c3b8c2SApple OSS Distributions        ldr     r3, [r2,#4]		// load high again
66*19c3b8c2SApple OSS Distributions        cmp     r1, r3			// check that high1 == high2
67*19c3b8c2SApple OSS Distributions	bne	_consistency_check	// try again if not
68*19c3b8c2SApple OSS Distributions        bx      lr
69*19c3b8c2SApple OSS Distributions
70*19c3b8c2SApple OSS Distributions#elif defined(__i386__)
71*19c3b8c2SApple OSS Distributions
72*19c3b8c2SApple OSS Distributions	.text
73*19c3b8c2SApple OSS Distributions	.align	4, 0x90
74*19c3b8c2SApple OSS Distributions	.globl	_mach_approximate_time
75*19c3b8c2SApple OSS Distributions_mach_approximate_time:
76*19c3b8c2SApple OSS Distributions	pushl	%ebp				// push a frame
77*19c3b8c2SApple OSS Distributions	movl	%esp, %ebp
78*19c3b8c2SApple OSS Distributions	cmpb	$0, _COMM_PAGE_APPROX_TIME_SUPPORTED // check commpage
79*19c3b8c2SApple OSS Distributions						//   "is supported" flag.
80*19c3b8c2SApple OSS Distributions	je	_approx_time_not_supported	// if zero, fall through to
81*19c3b8c2SApple OSS Distributions						//   absolute_time
82*19c3b8c2SApple OSS Distributions        // note the comment above for 32-bit ARM applies equally to 32-bit i386.
83*19c3b8c2SApple OSS Distributions	.align	4, 0x90
84*19c3b8c2SApple OSS Distributions_approx_time_consistency_check:
85*19c3b8c2SApple OSS Distributions	movl	_COMM_PAGE_APPROX_TIME+4, %edx	// load high
86*19c3b8c2SApple OSS Distributions	movl	_COMM_PAGE_APPROX_TIME, %eax	// load low
87*19c3b8c2SApple OSS Distributions        lfence					// avoid predictive reads that
88*19c3b8c2SApple OSS Distributions						//   could be invalid if
89*19c3b8c2SApple OSS Distributions						//   interrupted
90*19c3b8c2SApple OSS Distributions	cmpl	_COMM_PAGE_APPROX_TIME+4, %edx  // load high and check if equal
91*19c3b8c2SApple OSS Distributions						// to the first read
92*19c3b8c2SApple OSS Distributions	jne	_approx_time_consistency_check	// if not, try again.
93*19c3b8c2SApple OSS Distributions	popl	%ebp
94*19c3b8c2SApple OSS Distributions	ret
95*19c3b8c2SApple OSS Distributions
96*19c3b8c2SApple OSS Distributions_approx_time_not_supported:
97*19c3b8c2SApple OSS Distributions	popl	%ebp
98*19c3b8c2SApple OSS Distributions	jmp	_mach_absolute_time
99*19c3b8c2SApple OSS Distributions
100*19c3b8c2SApple OSS Distributions#endif
101