@@ -15,6 +15,7 @@ use cubeb_backend::{
15
15
use std:: ffi:: CStr ;
16
16
use std:: os:: raw:: c_void;
17
17
use std:: ptr;
18
+ use std:: sync:: OnceLock ;
18
19
19
20
pub const OPS : Ops = capi_new ! ( TestContext , TestStream ) ;
20
21
@@ -144,12 +145,13 @@ fn test_ops_context_init() {
144
145
unsafe { OPS . init. unwrap( ) ( & mut c, ptr:: null( ) ) } ,
145
146
ffi:: CUBEB_OK
146
147
) ;
148
+ assert ! ( !c. is_null( ) ) ;
147
149
unsafe { OPS . destroy . unwrap ( ) ( c) }
148
150
}
149
151
150
152
#[ test]
151
153
fn test_ops_context_max_channel_count ( ) {
152
- let c: * mut ffi:: cubeb = ptr :: null_mut ( ) ;
154
+ let c: * mut ffi:: cubeb = get_ctx ( ) ;
153
155
let mut max_channel_count = u32:: max_value ( ) ;
154
156
assert_eq ! (
155
157
unsafe { OPS . get_max_channel_count. unwrap( ) ( c, & mut max_channel_count) } ,
@@ -160,7 +162,7 @@ fn test_ops_context_max_channel_count() {
160
162
161
163
#[ test]
162
164
fn test_ops_context_min_latency ( ) {
163
- let c: * mut ffi:: cubeb = ptr :: null_mut ( ) ;
165
+ let c: * mut ffi:: cubeb = get_ctx ( ) ;
164
166
let params: ffi:: cubeb_stream_params = unsafe { :: std:: mem:: zeroed ( ) } ;
165
167
let mut latency = u32:: max_value ( ) ;
166
168
assert_eq ! (
@@ -172,7 +174,7 @@ fn test_ops_context_min_latency() {
172
174
173
175
#[ test]
174
176
fn test_ops_context_preferred_sample_rate ( ) {
175
- let c: * mut ffi:: cubeb = ptr :: null_mut ( ) ;
177
+ let c: * mut ffi:: cubeb = get_ctx ( ) ;
176
178
let mut rate = u32:: max_value ( ) ;
177
179
assert_eq ! (
178
180
unsafe { OPS . get_preferred_sample_rate. unwrap( ) ( c, & mut rate) } ,
@@ -183,7 +185,7 @@ fn test_ops_context_preferred_sample_rate() {
183
185
184
186
#[ test]
185
187
fn test_ops_context_supported_input_processing_params ( ) {
186
- let c: * mut ffi:: cubeb = ptr :: null_mut ( ) ;
188
+ let c: * mut ffi:: cubeb = get_ctx ( ) ;
187
189
let mut params: ffi:: cubeb_input_processing_params = InputProcessingParams :: all ( ) . bits ( ) ;
188
190
assert_eq ! (
189
191
unsafe { OPS . get_supported_input_processing_params. unwrap( ) ( c, & mut params) } ,
@@ -194,7 +196,7 @@ fn test_ops_context_supported_input_processing_params() {
194
196
195
197
#[ test]
196
198
fn test_ops_context_enumerate_devices ( ) {
197
- let c: * mut ffi:: cubeb = ptr :: null_mut ( ) ;
199
+ let c: * mut ffi:: cubeb = get_ctx ( ) ;
198
200
let mut coll = ffi:: cubeb_device_collection {
199
201
device : ptr:: null_mut ( ) ,
200
202
count : 0 ,
@@ -209,7 +211,7 @@ fn test_ops_context_enumerate_devices() {
209
211
210
212
#[ test]
211
213
fn test_ops_context_device_collection_destroy ( ) {
212
- let c: * mut ffi:: cubeb = ptr :: null_mut ( ) ;
214
+ let c: * mut ffi:: cubeb = get_ctx ( ) ;
213
215
let mut coll = ffi:: cubeb_device_collection {
214
216
device : 0xDEAD_BEEF as * mut _ ,
215
217
count : usize:: max_value ( ) ,
@@ -296,3 +298,35 @@ fn test_ops_stream_device_destroy() {
296
298
OPS . stream_device_destroy . unwrap ( ) ( s, 0xDEAD_BEEF as * mut _ ) ;
297
299
}
298
300
}
301
+
302
+ fn get_ctx ( ) -> * mut ffi:: cubeb {
303
+ CONTEXT . get_or_init ( TestContextPtr :: new) . ptr
304
+ }
305
+
306
+ static CONTEXT : OnceLock < TestContextPtr > = OnceLock :: new ( ) ;
307
+
308
+ struct TestContextPtr {
309
+ ptr : * mut ffi:: cubeb ,
310
+ }
311
+
312
+ // Safety: ffi::cubeb implementations are expected to be thread-safe.
313
+ unsafe impl Send for TestContextPtr { }
314
+ unsafe impl Sync for TestContextPtr { }
315
+
316
+ impl TestContextPtr {
317
+ fn new ( ) -> Self {
318
+ let mut c: * mut ffi:: cubeb = ptr:: null_mut ( ) ;
319
+ assert_eq ! (
320
+ unsafe { OPS . init. unwrap( ) ( & mut c, ptr:: null( ) ) } ,
321
+ ffi:: CUBEB_OK
322
+ ) ;
323
+ assert ! ( !c. is_null( ) ) ;
324
+ TestContextPtr { ptr : c }
325
+ }
326
+ }
327
+
328
+ impl Drop for TestContextPtr {
329
+ fn drop ( & mut self ) {
330
+ unsafe { OPS . destroy . unwrap ( ) ( self . ptr ) }
331
+ }
332
+ }
0 commit comments