1*e3723e1fSApple OSS Distributions // Copyright (c) 2021-2022 Apple Inc. All rights reserved.
2*e3723e1fSApple OSS Distributions
3*e3723e1fSApple OSS Distributions #include <darwintest.h>
4*e3723e1fSApple OSS Distributions #include "test_utils.h"
5*e3723e1fSApple OSS Distributions #include <fcntl.h>
6*e3723e1fSApple OSS Distributions #include <inttypes.h>
7*e3723e1fSApple OSS Distributions #ifndef PRIVATE
8*e3723e1fSApple OSS Distributions /*
9*e3723e1fSApple OSS Distributions * Need new CPU families.
10*e3723e1fSApple OSS Distributions */
11*e3723e1fSApple OSS Distributions #define PRIVATE
12*e3723e1fSApple OSS Distributions #include <mach/machine.h>
13*e3723e1fSApple OSS Distributions #undef PRIVATE
14*e3723e1fSApple OSS Distributions #else /* !defined(PRIVATE) */
15*e3723e1fSApple OSS Distributions #include <mach/machine.h>
16*e3723e1fSApple OSS Distributions #endif /* defined(PRIVATE) */
17*e3723e1fSApple OSS Distributions #include <stdint.h>
18*e3723e1fSApple OSS Distributions #include <System/sys/guarded.h>
19*e3723e1fSApple OSS Distributions #include <System/sys/monotonic.h>
20*e3723e1fSApple OSS Distributions #include <sys/ioctl.h>
21*e3723e1fSApple OSS Distributions #include <sys/sysctl.h>
22*e3723e1fSApple OSS Distributions #include <unistd.h>
23*e3723e1fSApple OSS Distributions
24*e3723e1fSApple OSS Distributions T_GLOBAL_META(
25*e3723e1fSApple OSS Distributions T_META_NAMESPACE("xnu.monotonic"),
26*e3723e1fSApple OSS Distributions T_META_CHECK_LEAKS(false),
27*e3723e1fSApple OSS Distributions XNU_T_META_REQUIRES_DEVELOPMENT_KERNEL
28*e3723e1fSApple OSS Distributions );
29*e3723e1fSApple OSS Distributions
30*e3723e1fSApple OSS Distributions static bool
device_supports_uncore(void)31*e3723e1fSApple OSS Distributions device_supports_uncore(void)
32*e3723e1fSApple OSS Distributions {
33*e3723e1fSApple OSS Distributions int r;
34*e3723e1fSApple OSS Distributions int type, subtype;
35*e3723e1fSApple OSS Distributions unsigned int family;
36*e3723e1fSApple OSS Distributions size_t size = sizeof(type);
37*e3723e1fSApple OSS Distributions
38*e3723e1fSApple OSS Distributions /*
39*e3723e1fSApple OSS Distributions * Only arm64 Monsoon devices support uncore counters.
40*e3723e1fSApple OSS Distributions */
41*e3723e1fSApple OSS Distributions
42*e3723e1fSApple OSS Distributions r = sysctlbyname("hw.cputype", &type, &size, NULL, 0);
43*e3723e1fSApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(r, "sysctlbyname(\"hw.cputype\")");
44*e3723e1fSApple OSS Distributions r = sysctlbyname("hw.cpusubtype", &subtype, &size, NULL, 0);
45*e3723e1fSApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(r, "sysctlbyname(\"hw.cpusubtype\")");
46*e3723e1fSApple OSS Distributions r = sysctlbyname("hw.cpufamily", &family, &size, NULL, 0);
47*e3723e1fSApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(r, "sysctlbyname(\"hw.cpufamily\")");
48*e3723e1fSApple OSS Distributions
49*e3723e1fSApple OSS Distributions if (type == CPU_TYPE_ARM64 &&
50*e3723e1fSApple OSS Distributions subtype == CPU_SUBTYPE_ARM64_V8 &&
51*e3723e1fSApple OSS Distributions (family == CPUFAMILY_ARM_MONSOON_MISTRAL ||
52*e3723e1fSApple OSS Distributions family == CPUFAMILY_ARM_VORTEX_TEMPEST)) {
53*e3723e1fSApple OSS Distributions return true;
54*e3723e1fSApple OSS Distributions }
55*e3723e1fSApple OSS Distributions
56*e3723e1fSApple OSS Distributions return false;
57*e3723e1fSApple OSS Distributions }
58*e3723e1fSApple OSS Distributions
59*e3723e1fSApple OSS Distributions #define UNCORE_DEV_PATH "/dev/monotonic/uncore"
60*e3723e1fSApple OSS Distributions
61*e3723e1fSApple OSS Distributions static int
open_uncore_error(int * error)62*e3723e1fSApple OSS Distributions open_uncore_error(int *error)
63*e3723e1fSApple OSS Distributions {
64*e3723e1fSApple OSS Distributions guardid_t guard;
65*e3723e1fSApple OSS Distributions int fd;
66*e3723e1fSApple OSS Distributions
67*e3723e1fSApple OSS Distributions guard = 0xa5adcafe;
68*e3723e1fSApple OSS Distributions
69*e3723e1fSApple OSS Distributions T_SETUPBEGIN;
70*e3723e1fSApple OSS Distributions
71*e3723e1fSApple OSS Distributions fd = guarded_open_np(UNCORE_DEV_PATH, &guard,
72*e3723e1fSApple OSS Distributions GUARD_CLOSE | GUARD_DUP | GUARD_WRITE, O_CLOEXEC | O_EXCL);
73*e3723e1fSApple OSS Distributions if (fd < 0 && errno == ENOENT) {
74*e3723e1fSApple OSS Distributions T_ASSERT_FALSE(device_supports_uncore(),
75*e3723e1fSApple OSS Distributions "lack of dev node implies no uncore support");
76*e3723e1fSApple OSS Distributions T_SKIP("uncore counters are unsupported");
77*e3723e1fSApple OSS Distributions __builtin_unreachable();
78*e3723e1fSApple OSS Distributions }
79*e3723e1fSApple OSS Distributions
80*e3723e1fSApple OSS Distributions if (error == NULL) {
81*e3723e1fSApple OSS Distributions T_ASSERT_POSIX_SUCCESS(fd, "open '%s'", UNCORE_DEV_PATH);
82*e3723e1fSApple OSS Distributions } else {
83*e3723e1fSApple OSS Distributions *error = errno;
84*e3723e1fSApple OSS Distributions }
85*e3723e1fSApple OSS Distributions
86*e3723e1fSApple OSS Distributions T_SETUPEND;
87*e3723e1fSApple OSS Distributions
88*e3723e1fSApple OSS Distributions return fd;
89*e3723e1fSApple OSS Distributions }
90*e3723e1fSApple OSS Distributions
91*e3723e1fSApple OSS Distributions static void
uncore_counts(int fd,uint64_t ctr_mask,uint64_t * counts)92*e3723e1fSApple OSS Distributions uncore_counts(int fd, uint64_t ctr_mask, uint64_t *counts)
93*e3723e1fSApple OSS Distributions {
94*e3723e1fSApple OSS Distributions int r;
95*e3723e1fSApple OSS Distributions union monotonic_ctl_counts *cts_ctl;
96*e3723e1fSApple OSS Distributions
97*e3723e1fSApple OSS Distributions cts_ctl = (union monotonic_ctl_counts *)counts;
98*e3723e1fSApple OSS Distributions cts_ctl->in.ctr_mask = ctr_mask;
99*e3723e1fSApple OSS Distributions
100*e3723e1fSApple OSS Distributions r = ioctl(fd, MT_IOC_COUNTS, cts_ctl);
101*e3723e1fSApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(r, "MT_IOC_COUNTS got counter values");
102*e3723e1fSApple OSS Distributions }
103*e3723e1fSApple OSS Distributions
104*e3723e1fSApple OSS Distributions #define REF_TIMEBASE_EVENT 0x3
105*e3723e1fSApple OSS Distributions #define CTRS_MAX 128
106*e3723e1fSApple OSS Distributions
107*e3723e1fSApple OSS Distributions T_DECL(uncore_max_counters,
108*e3723e1fSApple OSS Distributions "ensure that the maximum number of uncore countes is sane",
109*e3723e1fSApple OSS Distributions XNU_T_META_SOC_SPECIFIC,
110*e3723e1fSApple OSS Distributions T_META_ASROOT(true), T_META_TAG_VM_NOT_ELIGIBLE)
111*e3723e1fSApple OSS Distributions {
112*e3723e1fSApple OSS Distributions int nctrs = 0;
113*e3723e1fSApple OSS Distributions int fd;
114*e3723e1fSApple OSS Distributions
115*e3723e1fSApple OSS Distributions fd = open_uncore_error(NULL);
116*e3723e1fSApple OSS Distributions
117*e3723e1fSApple OSS Distributions do {
118*e3723e1fSApple OSS Distributions union monotonic_ctl_add add_ctl;
119*e3723e1fSApple OSS Distributions int r;
120*e3723e1fSApple OSS Distributions
121*e3723e1fSApple OSS Distributions add_ctl.in.config.event = REF_TIMEBASE_EVENT;
122*e3723e1fSApple OSS Distributions add_ctl.in.config.allowed_ctr_mask = UINT64_MAX;
123*e3723e1fSApple OSS Distributions
124*e3723e1fSApple OSS Distributions r = ioctl(fd, MT_IOC_ADD, &add_ctl);
125*e3723e1fSApple OSS Distributions if (r < 0 && errno == E2BIG) {
126*e3723e1fSApple OSS Distributions break;
127*e3723e1fSApple OSS Distributions }
128*e3723e1fSApple OSS Distributions
129*e3723e1fSApple OSS Distributions T_QUIET;
130*e3723e1fSApple OSS Distributions T_ASSERT_POSIX_SUCCESS(r, "added reference timebase event to counters");
131*e3723e1fSApple OSS Distributions nctrs++;
132*e3723e1fSApple OSS Distributions } while (nctrs < CTRS_MAX);
133*e3723e1fSApple OSS Distributions
134*e3723e1fSApple OSS Distributions T_EXPECT_LT(nctrs, CTRS_MAX, "able to allocate %d uncore PMCs", nctrs);
135*e3723e1fSApple OSS Distributions }
136*e3723e1fSApple OSS Distributions
137*e3723e1fSApple OSS Distributions static uint32_t
uncore_add(int fd,uint64_t event,uint64_t allowed_ctrs,int error)138*e3723e1fSApple OSS Distributions uncore_add(int fd, uint64_t event, uint64_t allowed_ctrs, int error)
139*e3723e1fSApple OSS Distributions {
140*e3723e1fSApple OSS Distributions int save_errno;
141*e3723e1fSApple OSS Distributions int r;
142*e3723e1fSApple OSS Distributions uint32_t ctr;
143*e3723e1fSApple OSS Distributions union monotonic_ctl_add add_ctl;
144*e3723e1fSApple OSS Distributions
145*e3723e1fSApple OSS Distributions add_ctl.in.config.event = event;
146*e3723e1fSApple OSS Distributions add_ctl.in.config.allowed_ctr_mask = allowed_ctrs;
147*e3723e1fSApple OSS Distributions r = ioctl(fd, MT_IOC_ADD, &add_ctl);
148*e3723e1fSApple OSS Distributions if (error) {
149*e3723e1fSApple OSS Distributions save_errno = errno;
150*e3723e1fSApple OSS Distributions T_EXPECT_LT(r, 0, "adding event to counter should fail");
151*e3723e1fSApple OSS Distributions T_EXPECT_EQ(save_errno, error,
152*e3723e1fSApple OSS Distributions "adding event to counter should fail with %d: %s",
153*e3723e1fSApple OSS Distributions error, strerror(error));
154*e3723e1fSApple OSS Distributions return UINT32_MAX;
155*e3723e1fSApple OSS Distributions } else {
156*e3723e1fSApple OSS Distributions T_QUIET;
157*e3723e1fSApple OSS Distributions T_ASSERT_POSIX_SUCCESS(r,
158*e3723e1fSApple OSS Distributions "added event %#" PRIx64 " to counters", event);
159*e3723e1fSApple OSS Distributions }
160*e3723e1fSApple OSS Distributions
161*e3723e1fSApple OSS Distributions ctr = add_ctl.out.ctr;
162*e3723e1fSApple OSS Distributions T_QUIET; T_ASSERT_LT(ctr, (uint32_t)CTRS_MAX, "counter returned should be sane");
163*e3723e1fSApple OSS Distributions return ctr;
164*e3723e1fSApple OSS Distributions }
165*e3723e1fSApple OSS Distributions
166*e3723e1fSApple OSS Distributions T_DECL(uncore_collision,
167*e3723e1fSApple OSS Distributions "ensure that trying to add an event on the same counter fails",
168*e3723e1fSApple OSS Distributions T_META_ASROOT(true), T_META_TAG_VM_NOT_ELIGIBLE)
169*e3723e1fSApple OSS Distributions {
170*e3723e1fSApple OSS Distributions int fd;
171*e3723e1fSApple OSS Distributions uint32_t ctr;
172*e3723e1fSApple OSS Distributions
173*e3723e1fSApple OSS Distributions fd = open_uncore_error(NULL);
174*e3723e1fSApple OSS Distributions
175*e3723e1fSApple OSS Distributions ctr = uncore_add(fd, REF_TIMEBASE_EVENT, UINT64_MAX, 0);
176*e3723e1fSApple OSS Distributions T_LOG("added event to uncore counter %d\n", ctr);
177*e3723e1fSApple OSS Distributions
178*e3723e1fSApple OSS Distributions (void)uncore_add(fd, REF_TIMEBASE_EVENT, UINT64_C(1) << ctr, ENOSPC);
179*e3723e1fSApple OSS Distributions }
180*e3723e1fSApple OSS Distributions
181*e3723e1fSApple OSS Distributions static void
uncore_enable(int fd)182*e3723e1fSApple OSS Distributions uncore_enable(int fd)
183*e3723e1fSApple OSS Distributions {
184*e3723e1fSApple OSS Distributions union monotonic_ctl_enable en_ctl = {
185*e3723e1fSApple OSS Distributions .in = { .enable = true }
186*e3723e1fSApple OSS Distributions };
187*e3723e1fSApple OSS Distributions
188*e3723e1fSApple OSS Distributions T_ASSERT_POSIX_SUCCESS(ioctl(fd, MT_IOC_ENABLE, &en_ctl),
189*e3723e1fSApple OSS Distributions "enabling counters");
190*e3723e1fSApple OSS Distributions }
191*e3723e1fSApple OSS Distributions
192*e3723e1fSApple OSS Distributions T_DECL(uncore_enabled_busy,
193*e3723e1fSApple OSS Distributions "ensure that trying to add an event while enabled fails",
194*e3723e1fSApple OSS Distributions T_META_ASROOT(true), T_META_TAG_VM_NOT_ELIGIBLE)
195*e3723e1fSApple OSS Distributions {
196*e3723e1fSApple OSS Distributions int fd;
197*e3723e1fSApple OSS Distributions
198*e3723e1fSApple OSS Distributions fd = open_uncore_error(NULL);
199*e3723e1fSApple OSS Distributions
200*e3723e1fSApple OSS Distributions (void)uncore_add(fd, REF_TIMEBASE_EVENT, UINT64_MAX, 0);
201*e3723e1fSApple OSS Distributions
202*e3723e1fSApple OSS Distributions uncore_enable(fd);
203*e3723e1fSApple OSS Distributions (void)uncore_add(fd, REF_TIMEBASE_EVENT, UINT64_MAX, EBUSY);
204*e3723e1fSApple OSS Distributions }
205*e3723e1fSApple OSS Distributions
206*e3723e1fSApple OSS Distributions T_DECL(uncore_reset,
207*e3723e1fSApple OSS Distributions "ensure that resetting the counters works", T_META_TAG_VM_NOT_ELIGIBLE)
208*e3723e1fSApple OSS Distributions {
209*e3723e1fSApple OSS Distributions int fd;
210*e3723e1fSApple OSS Distributions int r;
211*e3723e1fSApple OSS Distributions
212*e3723e1fSApple OSS Distributions fd = open_uncore_error(NULL);
213*e3723e1fSApple OSS Distributions
214*e3723e1fSApple OSS Distributions (void)uncore_add(fd, REF_TIMEBASE_EVENT, UINT64_C(1), 0);
215*e3723e1fSApple OSS Distributions (void)uncore_add(fd, REF_TIMEBASE_EVENT, UINT64_C(1), ENOSPC);
216*e3723e1fSApple OSS Distributions
217*e3723e1fSApple OSS Distributions r = ioctl(fd, MT_IOC_RESET);
218*e3723e1fSApple OSS Distributions T_ASSERT_POSIX_SUCCESS(r, "resetting succeeds");
219*e3723e1fSApple OSS Distributions
220*e3723e1fSApple OSS Distributions T_LOG("adding event to same counter after reset");
221*e3723e1fSApple OSS Distributions (void)uncore_add(fd, REF_TIMEBASE_EVENT, UINT64_C(1), 0);
222*e3723e1fSApple OSS Distributions }
223*e3723e1fSApple OSS Distributions
224*e3723e1fSApple OSS Distributions #define SLEEP_USECS (500 * 1000)
225*e3723e1fSApple OSS Distributions
226*e3723e1fSApple OSS Distributions static int
uncore_add_all(int fd,uint64_t event,int * nmonitors)227*e3723e1fSApple OSS Distributions uncore_add_all(int fd, uint64_t event, int *nmonitors)
228*e3723e1fSApple OSS Distributions {
229*e3723e1fSApple OSS Distributions int nctrs = 0;
230*e3723e1fSApple OSS Distributions int r;
231*e3723e1fSApple OSS Distributions
232*e3723e1fSApple OSS Distributions do {
233*e3723e1fSApple OSS Distributions union monotonic_ctl_add add_ctl;
234*e3723e1fSApple OSS Distributions
235*e3723e1fSApple OSS Distributions add_ctl.in.config.event = event;
236*e3723e1fSApple OSS Distributions add_ctl.in.config.allowed_ctr_mask = UINT64_MAX;
237*e3723e1fSApple OSS Distributions
238*e3723e1fSApple OSS Distributions r = ioctl(fd, MT_IOC_ADD, &add_ctl);
239*e3723e1fSApple OSS Distributions if (r < 0 && errno == E2BIG) {
240*e3723e1fSApple OSS Distributions break;
241*e3723e1fSApple OSS Distributions }
242*e3723e1fSApple OSS Distributions
243*e3723e1fSApple OSS Distributions T_QUIET;
244*e3723e1fSApple OSS Distributions T_ASSERT_POSIX_SUCCESS(r, "added event %#" PRIx64 " to counters",
245*e3723e1fSApple OSS Distributions event);
246*e3723e1fSApple OSS Distributions nctrs++;
247*e3723e1fSApple OSS Distributions } while (nctrs < CTRS_MAX);
248*e3723e1fSApple OSS Distributions
249*e3723e1fSApple OSS Distributions if (nmonitors) {
250*e3723e1fSApple OSS Distributions union monotonic_ctl_info info_ctl;
251*e3723e1fSApple OSS Distributions r = ioctl(fd, MT_IOC_GET_INFO, &info_ctl);
252*e3723e1fSApple OSS Distributions T_QUIET; T_ASSERT_POSIX_SUCCESS(r, "got info about uncore counters");
253*e3723e1fSApple OSS Distributions
254*e3723e1fSApple OSS Distributions *nmonitors = (int)info_ctl.out.nmonitors;
255*e3723e1fSApple OSS Distributions }
256*e3723e1fSApple OSS Distributions
257*e3723e1fSApple OSS Distributions return nctrs;
258*e3723e1fSApple OSS Distributions }
259*e3723e1fSApple OSS Distributions
260*e3723e1fSApple OSS Distributions T_DECL(uncore_accuracy,
261*e3723e1fSApple OSS Distributions "ensure that the uncore counters count accurately",
262*e3723e1fSApple OSS Distributions T_META_ASROOT(true),
263*e3723e1fSApple OSS Distributions XNU_T_META_SOC_SPECIFIC,
264*e3723e1fSApple OSS Distributions T_META_MAYFAIL("rdar://88973518, threads need to be forced onto clusters"), T_META_TAG_VM_NOT_ELIGIBLE)
265*e3723e1fSApple OSS Distributions {
266*e3723e1fSApple OSS Distributions int fd;
267*e3723e1fSApple OSS Distributions int nctrs = 0;
268*e3723e1fSApple OSS Distributions int nmonitors = 0;
269*e3723e1fSApple OSS Distributions uint64_t ctr_mask;
270*e3723e1fSApple OSS Distributions uint64_t counts[2][CTRS_MAX];
271*e3723e1fSApple OSS Distributions uint64_t times[2];
272*e3723e1fSApple OSS Distributions
273*e3723e1fSApple OSS Distributions fd = open_uncore_error(NULL);
274*e3723e1fSApple OSS Distributions
275*e3723e1fSApple OSS Distributions /*
276*e3723e1fSApple OSS Distributions * The reference timebase event counts the same as mach_continuous_time
277*e3723e1fSApple OSS Distributions * (on hardware supporting uncore counters). Make sure that the counter
278*e3723e1fSApple OSS Distributions * is close to the values returned from the trap.
279*e3723e1fSApple OSS Distributions *
280*e3723e1fSApple OSS Distributions * Fill all the counters with this event.
281*e3723e1fSApple OSS Distributions */
282*e3723e1fSApple OSS Distributions nctrs = uncore_add_all(fd, REF_TIMEBASE_EVENT, &nmonitors);
283*e3723e1fSApple OSS Distributions ctr_mask = (UINT64_C(1) << nctrs) - 1;
284*e3723e1fSApple OSS Distributions
285*e3723e1fSApple OSS Distributions T_LOG("added %d counters to check", nctrs);
286*e3723e1fSApple OSS Distributions
287*e3723e1fSApple OSS Distributions uncore_enable(fd);
288*e3723e1fSApple OSS Distributions
289*e3723e1fSApple OSS Distributions /*
290*e3723e1fSApple OSS Distributions * First, make sure there's an upper bound on the counter -- take the
291*e3723e1fSApple OSS Distributions * time around getting the counter values.
292*e3723e1fSApple OSS Distributions */
293*e3723e1fSApple OSS Distributions
294*e3723e1fSApple OSS Distributions times[0] = mach_absolute_time();
295*e3723e1fSApple OSS Distributions uncore_counts(fd, ctr_mask, counts[0]);
296*e3723e1fSApple OSS Distributions
297*e3723e1fSApple OSS Distributions usleep(SLEEP_USECS);
298*e3723e1fSApple OSS Distributions
299*e3723e1fSApple OSS Distributions uncore_counts(fd, ctr_mask, counts[1]);
300*e3723e1fSApple OSS Distributions times[1] = mach_absolute_time();
301*e3723e1fSApple OSS Distributions
302*e3723e1fSApple OSS Distributions T_QUIET; T_EXPECT_GT(times[1], times[0],
303*e3723e1fSApple OSS Distributions "mach_continuous_time is monotonically increasing");
304*e3723e1fSApple OSS Distributions for (int i = 0; i < nctrs; i++) {
305*e3723e1fSApple OSS Distributions T_EXPECT_GT(counts[1][i], counts[0][i],
306*e3723e1fSApple OSS Distributions "uncore counter %d value is monotonically increasing", i);
307*e3723e1fSApple OSS Distributions T_EXPECT_LT(counts[1][i] - counts[0][i], times[1] - times[0],
308*e3723e1fSApple OSS Distributions "reference timebase on uncore counter %d satisfies upper bound "
309*e3723e1fSApple OSS Distributions "from mach_absolute_time", i);
310*e3723e1fSApple OSS Distributions }
311*e3723e1fSApple OSS Distributions
312*e3723e1fSApple OSS Distributions /*
313*e3723e1fSApple OSS Distributions * Next, the lower bound -- put mach_absolute_time inside getting the
314*e3723e1fSApple OSS Distributions * counter values.
315*e3723e1fSApple OSS Distributions */
316*e3723e1fSApple OSS Distributions
317*e3723e1fSApple OSS Distributions uncore_counts(fd, ctr_mask, counts[0]);
318*e3723e1fSApple OSS Distributions times[0] = mach_absolute_time();
319*e3723e1fSApple OSS Distributions
320*e3723e1fSApple OSS Distributions volatile int iterations = 100000;
321*e3723e1fSApple OSS Distributions while (iterations--) {
322*e3723e1fSApple OSS Distributions ;
323*e3723e1fSApple OSS Distributions }
324*e3723e1fSApple OSS Distributions
325*e3723e1fSApple OSS Distributions times[1] = mach_absolute_time();
326*e3723e1fSApple OSS Distributions uncore_counts(fd, ctr_mask, counts[1]);
327*e3723e1fSApple OSS Distributions
328*e3723e1fSApple OSS Distributions for (int mon = 0; mon < nmonitors; mon++) {
329*e3723e1fSApple OSS Distributions for (int i = 0; i < nctrs; i++) {
330*e3723e1fSApple OSS Distributions uint64_t after = counts[1][i * mon];
331*e3723e1fSApple OSS Distributions uint64_t before = counts[0][i * mon];
332*e3723e1fSApple OSS Distributions uint64_t delta = after - before;
333*e3723e1fSApple OSS Distributions T_LOG("%d,%2d: %12" PRIu64 "%12" PRIu64 " = %10" PRIu64, mon, i,
334*e3723e1fSApple OSS Distributions before, after, delta);
335*e3723e1fSApple OSS Distributions T_QUIET;
336*e3723e1fSApple OSS Distributions T_EXPECT_GT(after, before,
337*e3723e1fSApple OSS Distributions "uncore %d counter %d value is monotonically increasing",
338*e3723e1fSApple OSS Distributions mon, i);
339*e3723e1fSApple OSS Distributions T_QUIET;
340*e3723e1fSApple OSS Distributions T_EXPECT_GT(counts[1][i * mon] - counts[0][i * mon],
341*e3723e1fSApple OSS Distributions times[1] - times[0],
342*e3723e1fSApple OSS Distributions "reference timebase on uncore %d counter %d satisfies "
343*e3723e1fSApple OSS Distributions "lower bound from mach_absolute_time", mon, i);
344*e3723e1fSApple OSS Distributions }
345*e3723e1fSApple OSS Distributions }
346*e3723e1fSApple OSS Distributions }
347*e3723e1fSApple OSS Distributions
348*e3723e1fSApple OSS Distributions T_DECL(uncore_ownership,
349*e3723e1fSApple OSS Distributions "ensure the dev node cannot be open in two places",
350*e3723e1fSApple OSS Distributions T_META_ASROOT(true), T_META_TAG_VM_NOT_ELIGIBLE)
351*e3723e1fSApple OSS Distributions {
352*e3723e1fSApple OSS Distributions int fd;
353*e3723e1fSApple OSS Distributions int other_fd;
354*e3723e1fSApple OSS Distributions int error;
355*e3723e1fSApple OSS Distributions
356*e3723e1fSApple OSS Distributions fd = open_uncore_error(NULL);
357*e3723e1fSApple OSS Distributions
358*e3723e1fSApple OSS Distributions other_fd = open_uncore_error(&error);
359*e3723e1fSApple OSS Distributions T_ASSERT_LT(other_fd, 0, "opening a second uncore fd should fail");
360*e3723e1fSApple OSS Distributions }
361*e3723e1fSApple OSS Distributions
362*e3723e1fSApple OSS Distributions T_DECL(uncore_root_required,
363*e3723e1fSApple OSS Distributions "ensure the dev node cannot be opened by non-root users",
364*e3723e1fSApple OSS Distributions T_META_ASROOT(false), T_META_TAG_VM_NOT_ELIGIBLE)
365*e3723e1fSApple OSS Distributions {
366*e3723e1fSApple OSS Distributions int fd;
367*e3723e1fSApple OSS Distributions int error = 0;
368*e3723e1fSApple OSS Distributions
369*e3723e1fSApple OSS Distributions T_SKIP("libdarwintest doesn't drop privileges properly");
370*e3723e1fSApple OSS Distributions
371*e3723e1fSApple OSS Distributions fd = open_uncore_error(&error);
372*e3723e1fSApple OSS Distributions T_ASSERT_LT(fd, 0, "opening dev node should not return an fd");
373*e3723e1fSApple OSS Distributions T_ASSERT_EQ(error, EPERM,
374*e3723e1fSApple OSS Distributions "opening dev node as non-root user should fail with EPERM");
375*e3723e1fSApple OSS Distributions }
376*e3723e1fSApple OSS Distributions
377*e3723e1fSApple OSS Distributions T_DECL(perf_uncore,
378*e3723e1fSApple OSS Distributions "measure the latency of accessing the counters",
379*e3723e1fSApple OSS Distributions XNU_T_META_SOC_SPECIFIC,
380*e3723e1fSApple OSS Distributions T_META_TAG_PERF, T_META_TAG_VM_NOT_ELIGIBLE)
381*e3723e1fSApple OSS Distributions {
382*e3723e1fSApple OSS Distributions int fd;
383*e3723e1fSApple OSS Distributions int nctrs;
384*e3723e1fSApple OSS Distributions int nmonitors;
385*e3723e1fSApple OSS Distributions int r;
386*e3723e1fSApple OSS Distributions uint64_t ctr_mask;
387*e3723e1fSApple OSS Distributions dt_stat_thread_instructions_t counts_instrs;
388*e3723e1fSApple OSS Distributions dt_stat_t counter_deltas;
389*e3723e1fSApple OSS Distributions
390*e3723e1fSApple OSS Distributions counts_instrs = dt_stat_thread_instructions_create("ioctl_counts");
391*e3723e1fSApple OSS Distributions counter_deltas = dt_stat_create("abs_time", "between_each_counter");
392*e3723e1fSApple OSS Distributions
393*e3723e1fSApple OSS Distributions fd = open_uncore_error(NULL);
394*e3723e1fSApple OSS Distributions
395*e3723e1fSApple OSS Distributions nctrs = uncore_add_all(fd, REF_TIMEBASE_EVENT, &nmonitors);
396*e3723e1fSApple OSS Distributions ctr_mask = (UINT64_C(1) << nctrs) - 1;
397*e3723e1fSApple OSS Distributions
398*e3723e1fSApple OSS Distributions uncore_enable(fd);
399*e3723e1fSApple OSS Distributions
400*e3723e1fSApple OSS Distributions do {
401*e3723e1fSApple OSS Distributions dt_stat_token token;
402*e3723e1fSApple OSS Distributions uint64_t counts[nctrs * nmonitors];
403*e3723e1fSApple OSS Distributions union monotonic_ctl_counts *cts_ctl;
404*e3723e1fSApple OSS Distributions
405*e3723e1fSApple OSS Distributions cts_ctl = (union monotonic_ctl_counts *)counts;
406*e3723e1fSApple OSS Distributions cts_ctl->in.ctr_mask = ctr_mask;
407*e3723e1fSApple OSS Distributions
408*e3723e1fSApple OSS Distributions token = dt_stat_thread_instructions_begin(counts_instrs);
409*e3723e1fSApple OSS Distributions r = ioctl(fd, MT_IOC_COUNTS, cts_ctl);
410*e3723e1fSApple OSS Distributions dt_stat_thread_instructions_end(counts_instrs, token);
411*e3723e1fSApple OSS Distributions T_QUIET;
412*e3723e1fSApple OSS Distributions T_ASSERT_POSIX_SUCCESS(r,
413*e3723e1fSApple OSS Distributions "getting uncore counter values %#" PRIx64, ctr_mask);
414*e3723e1fSApple OSS Distributions
415*e3723e1fSApple OSS Distributions for (int i = 0; i < (nctrs - 1); i++) {
416*e3723e1fSApple OSS Distributions dt_stat_add(counter_deltas, (double)(counts[i + 1] - counts[i]));
417*e3723e1fSApple OSS Distributions }
418*e3723e1fSApple OSS Distributions } while (!dt_stat_stable(counts_instrs) || !dt_stat_stable(counter_deltas));
419*e3723e1fSApple OSS Distributions
420*e3723e1fSApple OSS Distributions dt_stat_finalize(counts_instrs);
421*e3723e1fSApple OSS Distributions dt_stat_finalize(counter_deltas);
422*e3723e1fSApple OSS Distributions }
423