Line data Source code
1 : #ifndef HEADER_fd_src_funk_fd_funk_val_h
2 : #define HEADER_fd_src_funk_fd_funk_val_h
3 :
4 : /* This provides APIs for managing funk record values. It is generally
5 : not meant to be included directly. Use fd_funk.h instead. */
6 :
7 : #include "fd_funk_rec.h" /* Includes fd_funk_txn.h, fd_funk_base.h */
8 :
9 : /* FD_FUNK_REC_VAL_MAX is the maximum size of a record value. */
10 :
11 516712734 : #define FD_FUNK_REC_VAL_MAX UINT_MAX
12 198608604 : #define FD_FUNK_VAL_ALIGN 8UL
13 :
14 : FD_PROTOTYPES_BEGIN
15 :
16 : /* Accessors */
17 :
18 : /* fd_funk_val_{sz,max} returns the current size of the value associated
19 : with a record and the amount of wksp allocated currently for a value.
20 : Assumes funk is a current local join. These value might change on
21 : subsequent calls if the record is resized.
22 : 0<=sz<=max<=FD_FUNK_REC_VAL_MAX. */
23 :
24 : FD_FN_PURE static inline ulong /* Current size of the record's value in bytes */
25 456285561 : fd_funk_val_sz( fd_funk_rec_t const * rec ) { /* Assumes pointer in caller's address space to a live funk record */
26 456285561 : return (ulong)rec->val_sz; /* Covers the marked ERASE case too */
27 456285561 : }
28 :
29 : FD_FN_PURE static inline ulong /* Current size of the record's value allocation in bytes */
30 472764129 : fd_funk_val_max( fd_funk_rec_t const * rec ) { /* Assumes pointer in caller's address space to a live funk record */
31 472764129 : return (ulong)rec->val_max; /* Covers the marked ERASE case too */
32 472764129 : }
33 :
34 : /* fd_funk_val returns a pointer in the caller's address space to the
35 : current value associated with a record. fd_funk_rec_val_const is a
36 : const-correct version. There are sz bytes at the returned pointer.
37 : IMPORTANT SAFETY TIP! There are _no_ alignment guarantees on the
38 : returned value. Returns NULL if the record has a zero sz (which also
39 : covers the case where rec has been marked ERASE). max 0 implies val
40 : NULL and vice versa. Assumes no concurrent operations on rec. */
41 :
42 : FD_FN_PURE static inline void * /* Lifetime is the lesser of rec or the value size is modified */
43 : fd_funk_val( fd_funk_rec_t const * rec, /* Assumes pointer in caller's address space to a live funk record */
44 116052192 : fd_wksp_t const * wksp ) { /* ==fd_funk_wksp( funk ) where funk is a current local join */
45 116052192 : ulong val_gaddr = rec->val_gaddr;
46 116052192 : if( !val_gaddr ) return NULL; /* Covers the marked ERASE case too */ /* TODO: consider branchless */
47 83544180 : return fd_wksp_laddr_fast( wksp, val_gaddr );
48 116052192 : }
49 :
50 : FD_FN_PURE static inline void const * /* Lifetime is the lesser of rec or the value size is modified */
51 : fd_funk_val_const( fd_funk_rec_t const * rec, /* Assumes pointer in caller's address space to a live funk record */
52 526183140 : fd_wksp_t const * wksp ) { /* ==fd_funk_wksp( funk ) where funk is a current local join */
53 526183140 : ulong val_gaddr = rec->val_gaddr;
54 526183140 : if( !val_gaddr ) return NULL; /* Covers the marked ERASE case too */ /* TODO: consider branchless */
55 376776480 : return fd_wksp_laddr_fast( wksp, val_gaddr );
56 526183140 : }
57 :
58 : /* fd_funk_val_safe copies out the record value into a buffer
59 : * allocated by the valloc. The result should eventually be freed by
60 : * the same valloc. This API is safe in the presence of concurrent writes. */
61 :
62 : void *
63 : fd_funk_val_safe( fd_funk_rec_t const * rec, /* Assumes pointer in caller's address space to a live funk record */
64 : fd_wksp_t const * wksp,
65 : fd_valloc_t valloc,
66 : ulong * result_len );
67 :
68 :
69 : /* fd_funk_rec_read reads bytes [off,off+sz) and returns a pointer to
70 : the requested data on success and NULL on failure. Reasons for
71 : failure include NULL rec, 0 sz, [off,off+sz) does not overlap
72 : completely val, NULL wksp, marked ERASE. Assumes no concurrent
73 : operations on rec.
74 :
75 : The returned pointer is in the caller's address space and, if
76 : non-NULL, the value at the pointer is stable for its lifetime or
77 : until it is modified.
78 :
79 : IMPORTANT SAFETY TIP! There are _no_ alignment guarantees on the
80 : returned value (even if off itself is aligned).
81 :
82 : Note that if reading two overlapping regions of a record, the return
83 : pointers can overlap (along these will also overlap the regions
84 : returned by the above accessors). Further, if the region is changed
85 : by a below write, the value at the returned pointers will immediately
86 : reflect those writes. (That is, the returned pointers are zero copy
87 : into the actual value data of the record.) */
88 :
89 : FD_FN_PURE static inline void const * /* Lifetime is lesser of current local join, the record or val is resized */
90 : fd_funk_val_read( fd_funk_rec_t const * rec, /* Assumes pointer in caller's address space to a live funk record
91 : (NULL returns NULL) */
92 : ulong off, /* Should be in [0,sz] */
93 : ulong sz, /* Should be in [1,val_sz-off] */
94 3610168071 : fd_wksp_t const * wksp ) { /* ==fd_funk_wksp( funk ) where funk is current local join */
95 :
96 3610168071 : ulong end = off + sz;
97 :
98 3610168071 : if( FD_UNLIKELY( (!rec) | (end<=off) | (!wksp) ) || /* NULL rec, sz==0 or off+sz wrapped, NULL wksp */
99 3610168071 : FD_UNLIKELY( (end>(ulong)rec->val_sz) ) ) return NULL; /* Read past end (covers marked ERASE case too) */
100 :
101 1560036294 : return fd_wksp_laddr_fast( wksp, rec->val_gaddr + off );
102 3610168071 : }
103 :
104 : /* Operations */
105 :
106 : /* fd_funk_rec_write writes record bytes [off,off+sz) and returns rec on
107 : success and NULL on failure.
108 :
109 : data points to the bytes to sz write. [data,data+sz) should not
110 : overlap with the current value bytes [off,off+sz) but otherwise can
111 : point anywhere valid in the caller's address space.
112 :
113 : The write retains no interest in data on return. sz 0 is considered
114 : a no-op regardless of anything else and immediately returns rec.
115 :
116 : Reasons for failure include NULL rec, NULL data with non-zero sz,
117 : [data,data+sz) wraps, [off,off+sz) does not overlap with the record
118 : completely, [data,data+sz) overlaps with [off,off+sz). Assumes no
119 : concurrent operations on rec or data. */
120 :
121 : FD_FN_UNUSED static fd_funk_rec_t * /* Returns rec on success, NULL on failure */ /* Workaround -Winline */
122 : fd_funk_val_write( fd_funk_rec_t * rec, /* Assumed in caller's address space to live funk record (NULL returns NULL) */
123 : ulong off, /* First byte of record to write, in [0,val_sz], NULL if too large */
124 : ulong sz, /* Number of bytes to write, 0 is a no-op, in [0,val_sz-off], NULL if too large */
125 : void const * data, /* Assumed in caller's address space, NULL okay if sz 0 */
126 2603469507 : fd_wksp_t const * wksp ) { /* ==fd_funk_wksp( funk ) where funk is current local join */
127 :
128 2603469507 : if( FD_UNLIKELY( !sz ) ) return rec; /* Empty write request */
129 :
130 2285831058 : ulong end = off + sz;
131 :
132 2285831058 : ulong d0 = (ulong)data;
133 2285831058 : ulong d1 = d0 + sz;
134 :
135 2285831058 : if( FD_UNLIKELY( (!rec) | (end<off) | (!data) | (d1<d0) | (!wksp) ) || /* NULL rec, off+sz wrapped, NULL data w sz!=0, data wrapped, NULL wksp */
136 2285831058 : FD_UNLIKELY( end > (ulong)rec->val_max ) ) return NULL; /* too large (covers marked ERASE case too) */
137 :
138 36835422 : ulong v0 = (ulong)fd_wksp_laddr_fast( wksp, rec->val_gaddr + off );
139 36835422 : ulong v1 = v0 + sz;
140 :
141 36835422 : if( FD_UNLIKELY( !((d1<=v0) | (d0>=v1)) ) ) return NULL; /* data overlaps with val */
142 :
143 36835422 : fd_memcpy( (void *)v0, data, sz );
144 :
145 36835422 : if ( FD_UNLIKELY( end > (ulong)rec->val_sz ) )
146 0 : rec->val_sz = (uint)end;
147 :
148 36835422 : return rec;
149 36835422 : }
150 :
151 : /* fd_funk_val_copy copies sz bytes starting at data into the record
152 : value, replacing the existing record value. rec's value will be able
153 : to accommodate at least sz_est in the future without resizing on
154 : return. If sz_est is 0 on entry, it will be set to sz as a
155 : reasonable default before any argument checking.
156 :
157 : data points to the bytes to write. [data,data+sz) should not overlap
158 : with the existing record but can otherwise can point anywhere valid
159 : in the caller's address space.
160 :
161 : This generally will resize the record value to something at least
162 : sz_est so this function should be assumed to kill any existing
163 : pointers into this record's value storage. Note that sz_est==0 will
164 : set the record to the NULL val. And sz==0 with sz_est!=0 can be used
165 : to preallocate record sizes with a NULL initial value.
166 :
167 : Returns rec on success and NULL on failure. If opt_err is non-NULL,
168 : on return, *opt_err will hold FD_FUNK_SUCCESS if successful or a
169 : FD_FUNK_ERR_* code on failure. Reasons for failure include
170 : FD_FUNK_ERR_INVAL (NULL rec, NULL data with non-zero sz, NULL alloc,
171 : NULL wksp, data region wraps, sz>sz_est, sz_est too large, rec is
172 : marked as ERASE, data region overlaps the existing val allocation)
173 : and FD_FUNK_ERR_MEM (allocation failure, need a larger wksp). On
174 : failure, the current value is unchanged.
175 :
176 : Assumes no concurrent operations on rec or data. The copy retains no
177 : interest in data on return. */
178 :
179 : fd_funk_rec_t * /* Returns rec on success, NULL on failure */
180 : fd_funk_val_copy( fd_funk_rec_t * rec, /* Assumed in caller's address space to live funk record (NULL returns NULL) */
181 : void const * data, /* Points to first byte to copy in caller's address space, NULL okay if sz 0 */
182 : ulong sz, /* Number of bytes to copy, in [0,sz_est], NULL if too large */
183 : ulong sz_est, /* Est final size, 0 means use sz, in [sz,FD_FUNK_REC_VAL_MAX], NULL if too large */
184 : fd_alloc_t * alloc, /* ==fd_funk_alloc( funk, wksp ) */
185 : fd_wksp_t * wksp, /* ==fd_funk_wksp( funk ) where funk is current local join */
186 : int * opt_err ); /* If non-NULL, *opt_err returns operation error code */
187 :
188 : /* fd_funk_val_append appends sz bytes starting at data to the end of
189 : the record rec. [data,data+sz) should not overlap with the existing
190 : value allocation but can otherwise can point anywhere valid in the
191 : caller's address space.
192 :
193 : This might need to resize the record value so this function should be
194 : assumed to kill any existing pointers into this record's value
195 : storage. Unlike copy above and truncate below, this function does
196 : try to minimize the amount of value allocations it might do.
197 :
198 : Returns rec on success and NULL on failure. If opt_err is non-NULL,
199 : on return, *opt_err will hold FD_FUNK_SUCCESS if successful or a
200 : FD_FUNK_ERR_* code on failure. Reasons for failure include
201 : FD_FUNK_ERR_INVAL (NULL rec, NULL data with non-zero sz,
202 : [data,data+sz) wraps, NULL alloc, NULL wksp, rec marked ERASE, sz too
203 : large, data region overlaps with existing record value allocation)
204 : and FD_FUNK_ERR_MEM (allocation failure, need a larger wksp). On
205 : failure, the current value is unchanged.
206 :
207 : Assumes no concurrent operations on rec or data. The append retains
208 : no interest in data on return. sz 0 is considered a no-op regardless
209 : of anything else and immediately returns rec / SUCCESS. */
210 :
211 : fd_funk_rec_t * /* Returns rec on success, NULL on failure */
212 : fd_funk_val_append( fd_funk_rec_t * rec, /* Assumed in caller's address space to a live funk record (NULL returns NULL) */
213 : void const * data, /* Points to first byte to append in caller's address space, NULL okay if sz 0 */
214 : ulong sz, /* Number of bytes to append, 0 is a no-op, NULL if too large */
215 : fd_alloc_t * alloc, /* ==fd_funk_alloc( funk, wksp ) */
216 : fd_wksp_t * wksp, /* ==fd_funk_wksp( funk ) where funk is current local join */
217 : int * opt_err ); /* If non-NULL, *opt_err returns operation error code */
218 :
219 : /* fd_funk_val_truncate resizes a record to be new_val_sz bytes in size.
220 :
221 : This function is optimized for the user knowing the actual long term
222 : record size when they call this. So avoid using this to
223 : incrementally increase a size of a value by a constant amount over
224 : time. See append above for that.
225 :
226 : Likewise, regardless of the current and new value sizes, this will
227 : always attempt to resize the record in order to minimize the amount
228 : of excess allocation used by the record. So this function should be
229 : assumed to kill any existing pointers into this record's value
230 : storage.
231 :
232 : Returns rec on success and NULL on failure. If opt_err is non-NULL,
233 : on return, *opt_err will hold FD_FUNK_SUCCESS if successful or a
234 : FD_FUNK_ERR_* code on failure. Reasons for failure include
235 : FD_FUNK_ERR_INVAL (NULL rec, too large new_val_sz, rec is marked
236 : ERASE) and FD_FUNK_ERR_MEM (allocation failure, need a larger wksp).
237 : On failure, the current value is unchanged.
238 :
239 : Assumes no concurrent operations on rec. */
240 :
241 : fd_funk_rec_t * /* Returns rec on success, NULL on failure */
242 : fd_funk_val_truncate( fd_funk_rec_t * rec, /* Assumed in caller's address space to a live funk record (NULL returns NULL) */
243 : ulong new_val_sz, /* Should be in [0,FD_FUNK_REC_VAL_MAX] (returns NULL otherwise) */
244 : fd_alloc_t * alloc, /* ==fd_funk_alloc( funk, wksp ) */
245 : fd_wksp_t * wksp, /* ==fd_funk_wksp( funk ) where funk is current local join */
246 : int * opt_err ); /* If non-NULL, *opt_err returns operation error code */
247 :
248 : /* Misc */
249 :
250 : /* fd_funk_val_init sets a record with uninitialized value metadata to
251 : the NULL value. Meant for internal use. */
252 :
253 : static inline fd_funk_rec_t * /* Returns rec */
254 22497615 : fd_funk_val_init( fd_funk_rec_t * rec ) { /* Assumed record in caller's address space with uninitialized value metadata */
255 22497615 : rec->val_sz = 0U;
256 22497615 : rec->val_max = 0U;
257 22497615 : rec->val_gaddr = 0UL;
258 22497615 : return rec;
259 22497615 : }
260 :
261 : /* fd_funk_val_flush sets a record to the NULL value, discarding the
262 : current value if any. Meant for internal use. */
263 :
264 : static inline fd_funk_rec_t * /* Returns rec */
265 : fd_funk_val_flush( fd_funk_rec_t * rec, /* Assumed live funk record in caller's address space */
266 : fd_alloc_t * alloc, /* ==fd_funk_alloc( funk, wksp ) */
267 14641236 : fd_wksp_t * wksp ) { /* ==fd_funk_wksp( funk ) where funk is a current local join */
268 14641236 : ulong val_gaddr = rec->val_gaddr;
269 14641236 : fd_funk_val_init( rec );
270 14641236 : if( val_gaddr ) fd_alloc_free( alloc, fd_wksp_laddr_fast( wksp, val_gaddr ) );
271 14641236 : return rec;
272 14641236 : }
273 :
274 : /* fd_funk_val_verify verifies the record values. Returns
275 : FD_FUNK_SUCCESS if the values appear intact and FD_FUNK_ERR_INVAL if
276 : not (logs details). Meant to be called as part of fd_funk_verify.
277 : As such, it assumes funk is non-NULL, fd_funk_{wksp,rec_map,wksp_tag}
278 : have been verified to work and the rec_map has been verified. */
279 :
280 : int
281 : fd_funk_val_verify( fd_funk_t * funk );
282 :
283 : FD_PROTOTYPES_END
284 :
285 : /* TODO: Retune fd_alloc and fd_wksp for Solana record size optimized
286 : size classes and transition point to fd_wksp backing. */
287 :
288 : #endif /* HEADER_fd_src_funk_fd_funk_val_h */
|