1*0f4c859eSApple OSS Distributions /*
2*0f4c859eSApple OSS Distributions * Copyright (c) 2016 Apple Inc. All rights reserved.
3*0f4c859eSApple OSS Distributions *
4*0f4c859eSApple OSS Distributions * @APPLE_APACHE_LICENSE_HEADER_START@
5*0f4c859eSApple OSS Distributions *
6*0f4c859eSApple OSS Distributions * Licensed under the Apache License, Version 2.0 (the "License");
7*0f4c859eSApple OSS Distributions * you may not use this file except in compliance with the License.
8*0f4c859eSApple OSS Distributions * You may obtain a copy of the License at
9*0f4c859eSApple OSS Distributions *
10*0f4c859eSApple OSS Distributions * http://www.apache.org/licenses/LICENSE-2.0
11*0f4c859eSApple OSS Distributions *
12*0f4c859eSApple OSS Distributions * Unless required by applicable law or agreed to in writing, software
13*0f4c859eSApple OSS Distributions * distributed under the License is distributed on an "AS IS" BASIS,
14*0f4c859eSApple OSS Distributions * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15*0f4c859eSApple OSS Distributions * See the License for the specific language governing permissions and
16*0f4c859eSApple OSS Distributions * limitations under the License.
17*0f4c859eSApple OSS Distributions *
18*0f4c859eSApple OSS Distributions * @APPLE_APACHE_LICENSE_HEADER_END@
19*0f4c859eSApple OSS Distributions */
20*0f4c859eSApple OSS Distributions
21*0f4c859eSApple OSS Distributions #ifndef __FIREHOSE_CHUNK_PRIVATE__
22*0f4c859eSApple OSS Distributions #define __FIREHOSE_CHUNK_PRIVATE__
23*0f4c859eSApple OSS Distributions
24*0f4c859eSApple OSS Distributions #include <sys/param.h>
25*0f4c859eSApple OSS Distributions #include "firehose_types_private.h"
26*0f4c859eSApple OSS Distributions #include "tracepoint_private.h"
27*0f4c859eSApple OSS Distributions
28*0f4c859eSApple OSS Distributions __BEGIN_DECLS
29*0f4c859eSApple OSS Distributions
30*0f4c859eSApple OSS Distributions #define FIREHOSE_CHUNK_SIZE 4096ul
31*0f4c859eSApple OSS Distributions
32*0f4c859eSApple OSS Distributions #define FIREHOSE_CHUNK_POS_ENTRY_OFFS_INC (1ULL << 0)
33*0f4c859eSApple OSS Distributions #define FIREHOSE_CHUNK_POS_PRIVATE_OFFS_INC (1ULL << 16)
34*0f4c859eSApple OSS Distributions #define FIREHOSE_CHUNK_POS_REFCNT_INC (1ULL << 32)
35*0f4c859eSApple OSS Distributions #define FIREHOSE_CHUNK_POS_FULL_BIT (1ULL << 56)
36*0f4c859eSApple OSS Distributions #define FIREHOSE_CHUNK_POS_USABLE_FOR_STREAM(pos, stream) \
37*0f4c859eSApple OSS Distributions ((((pos).fcp_pos >> 48) & 0x1ff) == (uint16_t)stream)
38*0f4c859eSApple OSS Distributions
39*0f4c859eSApple OSS Distributions typedef union {
40*0f4c859eSApple OSS Distributions os_atomic(uint64_t) fcp_atomic_pos;
41*0f4c859eSApple OSS Distributions uint64_t fcp_pos;
42*0f4c859eSApple OSS Distributions struct {
43*0f4c859eSApple OSS Distributions uint16_t fcp_next_entry_offs;
44*0f4c859eSApple OSS Distributions uint16_t fcp_private_offs;
45*0f4c859eSApple OSS Distributions uint8_t fcp_refcnt;
46*0f4c859eSApple OSS Distributions uint8_t fcp_qos;
47*0f4c859eSApple OSS Distributions uint8_t fcp_stream;
48*0f4c859eSApple OSS Distributions uint8_t fcp_flag_full : 1;
49*0f4c859eSApple OSS Distributions uint8_t fcp_flag_io : 1;
50*0f4c859eSApple OSS Distributions uint8_t fcp_quarantined : 1;
51*0f4c859eSApple OSS Distributions uint8_t _fcp_flag_unused : 5;
52*0f4c859eSApple OSS Distributions };
53*0f4c859eSApple OSS Distributions } firehose_chunk_pos_u;
54*0f4c859eSApple OSS Distributions
55*0f4c859eSApple OSS Distributions typedef struct firehose_chunk_s {
56*0f4c859eSApple OSS Distributions union {
57*0f4c859eSApple OSS Distributions uint8_t fc_start[FIREHOSE_CHUNK_SIZE];
58*0f4c859eSApple OSS Distributions struct {
59*0f4c859eSApple OSS Distributions firehose_chunk_pos_u fc_pos;
60*0f4c859eSApple OSS Distributions uint64_t fc_timestamp;
61*0f4c859eSApple OSS Distributions uint8_t fc_data[FIREHOSE_CHUNK_SIZE - 8 - 8];
62*0f4c859eSApple OSS Distributions };
63*0f4c859eSApple OSS Distributions };
64*0f4c859eSApple OSS Distributions } *firehose_chunk_t;
65*0f4c859eSApple OSS Distributions
66*0f4c859eSApple OSS Distributions typedef struct firehose_chunk_range_s {
67*0f4c859eSApple OSS Distributions uint16_t fcr_offset; // offset from the start of the chunk
68*0f4c859eSApple OSS Distributions uint16_t fcr_length;
69*0f4c859eSApple OSS Distributions } *firehose_chunk_range_t;
70*0f4c859eSApple OSS Distributions
71*0f4c859eSApple OSS Distributions #if __has_include(<os/atomic_private.h>)
72*0f4c859eSApple OSS Distributions #if defined(KERNEL) || defined(OS_FIREHOSE_SPI)
73*0f4c859eSApple OSS Distributions
74*0f4c859eSApple OSS Distributions OS_ALWAYS_INLINE
75*0f4c859eSApple OSS Distributions static inline bool
firehose_chunk_pos_fits(firehose_chunk_pos_u * pos,uint16_t size)76*0f4c859eSApple OSS Distributions firehose_chunk_pos_fits(firehose_chunk_pos_u *pos, uint16_t size)
77*0f4c859eSApple OSS Distributions {
78*0f4c859eSApple OSS Distributions return pos->fcp_next_entry_offs + size <= pos->fcp_private_offs;
79*0f4c859eSApple OSS Distributions }
80*0f4c859eSApple OSS Distributions
81*0f4c859eSApple OSS Distributions #define FIREHOSE_CHUNK_TRY_RESERVE_FAIL_ENQUEUE (-1)
82*0f4c859eSApple OSS Distributions #define FIREHOSE_CHUNK_TRY_RESERVE_FAIL ( 0)
83*0f4c859eSApple OSS Distributions
84*0f4c859eSApple OSS Distributions OS_ALWAYS_INLINE
85*0f4c859eSApple OSS Distributions static inline long
firehose_chunk_tracepoint_try_reserve(firehose_chunk_t fc,uint64_t stamp,firehose_stream_t stream,uint8_t qos,uint16_t pubsize,uint16_t privsize,uint8_t ** privptr)86*0f4c859eSApple OSS Distributions firehose_chunk_tracepoint_try_reserve(firehose_chunk_t fc, uint64_t stamp,
87*0f4c859eSApple OSS Distributions firehose_stream_t stream, uint8_t qos, uint16_t pubsize,
88*0f4c859eSApple OSS Distributions uint16_t privsize, uint8_t **privptr)
89*0f4c859eSApple OSS Distributions {
90*0f4c859eSApple OSS Distributions const uint16_t ft_size = offsetof(struct firehose_tracepoint_s, ft_data);
91*0f4c859eSApple OSS Distributions firehose_chunk_pos_u orig, pos;
92*0f4c859eSApple OSS Distributions bool reservation_failed, stamp_delta_fits;
93*0f4c859eSApple OSS Distributions
94*0f4c859eSApple OSS Distributions stamp_delta_fits = ((stamp - fc->fc_timestamp) >> 48) == 0;
95*0f4c859eSApple OSS Distributions
96*0f4c859eSApple OSS Distributions // no acquire barrier because the returned space is written to only
97*0f4c859eSApple OSS Distributions os_atomic_rmw_loop(&fc->fc_pos.fcp_atomic_pos,
98*0f4c859eSApple OSS Distributions orig.fcp_pos, pos.fcp_pos, relaxed, {
99*0f4c859eSApple OSS Distributions if (orig.fcp_pos == 0) {
100*0f4c859eSApple OSS Distributions // we acquired a really really old reference, and we probably
101*0f4c859eSApple OSS Distributions // just faulted in a new page
102*0f4c859eSApple OSS Distributions os_atomic_rmw_loop_give_up(return FIREHOSE_CHUNK_TRY_RESERVE_FAIL);
103*0f4c859eSApple OSS Distributions }
104*0f4c859eSApple OSS Distributions if (!FIREHOSE_CHUNK_POS_USABLE_FOR_STREAM(orig, stream)) {
105*0f4c859eSApple OSS Distributions // nothing to do if the chunk is full, or the stream doesn't match,
106*0f4c859eSApple OSS Distributions // in which case the thread probably:
107*0f4c859eSApple OSS Distributions // - loaded the chunk ref
108*0f4c859eSApple OSS Distributions // - been suspended a long while
109*0f4c859eSApple OSS Distributions // - read the chunk to find a very old thing
110*0f4c859eSApple OSS Distributions os_atomic_rmw_loop_give_up(return FIREHOSE_CHUNK_TRY_RESERVE_FAIL);
111*0f4c859eSApple OSS Distributions }
112*0f4c859eSApple OSS Distributions pos.fcp_pos = orig.fcp_pos;
113*0f4c859eSApple OSS Distributions if (!firehose_chunk_pos_fits(&orig,
114*0f4c859eSApple OSS Distributions ft_size + pubsize + privsize) || !stamp_delta_fits) {
115*0f4c859eSApple OSS Distributions pos.fcp_flag_full = true;
116*0f4c859eSApple OSS Distributions reservation_failed = true;
117*0f4c859eSApple OSS Distributions } else {
118*0f4c859eSApple OSS Distributions if (qos > pos.fcp_qos) {
119*0f4c859eSApple OSS Distributions pos.fcp_qos = qos;
120*0f4c859eSApple OSS Distributions }
121*0f4c859eSApple OSS Distributions // using these *_INC macros is so that the compiler generates better
122*0f4c859eSApple OSS Distributions // assembly: using the struct individual fields forces the compiler
123*0f4c859eSApple OSS Distributions // to handle carry propagations, and we know it won't happen
124*0f4c859eSApple OSS Distributions pos.fcp_pos += roundup(ft_size + pubsize, 8) *
125*0f4c859eSApple OSS Distributions FIREHOSE_CHUNK_POS_ENTRY_OFFS_INC;
126*0f4c859eSApple OSS Distributions pos.fcp_pos -= privsize * FIREHOSE_CHUNK_POS_PRIVATE_OFFS_INC;
127*0f4c859eSApple OSS Distributions pos.fcp_pos += FIREHOSE_CHUNK_POS_REFCNT_INC;
128*0f4c859eSApple OSS Distributions const uint16_t minimum_payload_size = 16;
129*0f4c859eSApple OSS Distributions if (!firehose_chunk_pos_fits(&pos,
130*0f4c859eSApple OSS Distributions roundup(ft_size + minimum_payload_size, 8))) {
131*0f4c859eSApple OSS Distributions // if we can't even have minimum_payload_size bytes of payload
132*0f4c859eSApple OSS Distributions // for the next tracepoint, just flush right away
133*0f4c859eSApple OSS Distributions pos.fcp_flag_full = true;
134*0f4c859eSApple OSS Distributions }
135*0f4c859eSApple OSS Distributions reservation_failed = false;
136*0f4c859eSApple OSS Distributions }
137*0f4c859eSApple OSS Distributions });
138*0f4c859eSApple OSS Distributions
139*0f4c859eSApple OSS Distributions if (reservation_failed) {
140*0f4c859eSApple OSS Distributions if (pos.fcp_refcnt) {
141*0f4c859eSApple OSS Distributions // nothing to do, there is a thread writing that will pick up
142*0f4c859eSApple OSS Distributions // the "FULL" flag on flush and push as a consequence
143*0f4c859eSApple OSS Distributions return FIREHOSE_CHUNK_TRY_RESERVE_FAIL;
144*0f4c859eSApple OSS Distributions }
145*0f4c859eSApple OSS Distributions // caller must enqueue chunk
146*0f4c859eSApple OSS Distributions return FIREHOSE_CHUNK_TRY_RESERVE_FAIL_ENQUEUE;
147*0f4c859eSApple OSS Distributions }
148*0f4c859eSApple OSS Distributions if (privptr) {
149*0f4c859eSApple OSS Distributions *privptr = fc->fc_start + pos.fcp_private_offs;
150*0f4c859eSApple OSS Distributions }
151*0f4c859eSApple OSS Distributions return orig.fcp_next_entry_offs;
152*0f4c859eSApple OSS Distributions }
153*0f4c859eSApple OSS Distributions
154*0f4c859eSApple OSS Distributions OS_ALWAYS_INLINE
155*0f4c859eSApple OSS Distributions static inline firehose_tracepoint_t
firehose_chunk_tracepoint_begin(firehose_chunk_t fc,uint64_t stamp,uint16_t pubsize,uint64_t thread_id,long offset)156*0f4c859eSApple OSS Distributions firehose_chunk_tracepoint_begin(firehose_chunk_t fc, uint64_t stamp,
157*0f4c859eSApple OSS Distributions uint16_t pubsize, uint64_t thread_id, long offset)
158*0f4c859eSApple OSS Distributions {
159*0f4c859eSApple OSS Distributions firehose_tracepoint_t ft = (firehose_tracepoint_t)
160*0f4c859eSApple OSS Distributions __builtin_assume_aligned(fc->fc_start + offset, 8);
161*0f4c859eSApple OSS Distributions stamp -= fc->fc_timestamp;
162*0f4c859eSApple OSS Distributions stamp |= (uint64_t)pubsize << 48;
163*0f4c859eSApple OSS Distributions // The compiler barrier is needed for userland process death handling, see
164*0f4c859eSApple OSS Distributions // (tracepoint-begin) in libdispatch's firehose_buffer_stream_chunk_install.
165*0f4c859eSApple OSS Distributions os_atomic_std(atomic_store_explicit)(&ft->ft_atomic_stamp_and_length, stamp,
166*0f4c859eSApple OSS Distributions os_atomic_std(memory_order_relaxed));
167*0f4c859eSApple OSS Distributions __asm__ __volatile__ ("" ::: "memory");
168*0f4c859eSApple OSS Distributions ft->ft_thread = thread_id;
169*0f4c859eSApple OSS Distributions return ft;
170*0f4c859eSApple OSS Distributions }
171*0f4c859eSApple OSS Distributions
172*0f4c859eSApple OSS Distributions OS_ALWAYS_INLINE
173*0f4c859eSApple OSS Distributions static inline bool
firehose_chunk_tracepoint_end(firehose_chunk_t fc,firehose_tracepoint_t ft,firehose_tracepoint_id_u ftid)174*0f4c859eSApple OSS Distributions firehose_chunk_tracepoint_end(firehose_chunk_t fc,
175*0f4c859eSApple OSS Distributions firehose_tracepoint_t ft, firehose_tracepoint_id_u ftid)
176*0f4c859eSApple OSS Distributions {
177*0f4c859eSApple OSS Distributions firehose_chunk_pos_u pos;
178*0f4c859eSApple OSS Distributions
179*0f4c859eSApple OSS Distributions os_atomic_std(atomic_store_explicit)(&ft->ft_id.ftid_atomic_value,
180*0f4c859eSApple OSS Distributions ftid.ftid_value, os_atomic_std(memory_order_release));
181*0f4c859eSApple OSS Distributions pos.fcp_pos = os_atomic_std(atomic_fetch_sub_explicit)(&fc->fc_pos.fcp_atomic_pos,
182*0f4c859eSApple OSS Distributions FIREHOSE_CHUNK_POS_REFCNT_INC, os_atomic_std(memory_order_relaxed));
183*0f4c859eSApple OSS Distributions return pos.fcp_refcnt == 1 && pos.fcp_flag_full;
184*0f4c859eSApple OSS Distributions }
185*0f4c859eSApple OSS Distributions
186*0f4c859eSApple OSS Distributions #endif // defined(KERNEL) || defined(OS_FIREHOSE_SPI)
187*0f4c859eSApple OSS Distributions #endif // __has_include(<os/atomic_private.h>)
188*0f4c859eSApple OSS Distributions
189*0f4c859eSApple OSS Distributions __END_DECLS
190*0f4c859eSApple OSS Distributions
191*0f4c859eSApple OSS Distributions #endif // __FIREHOSE_CHUNK_PRIVATE__
192