| // Copyright (c) 2023, the Dart project authors. Please see the AUTHORS file |
| // for details. All rights reserved. Use of this source code is governed by a |
| // BSD-style license that can be found in the LICENSE file. |
| |
| // Objective C support is only available on mac. |
| @TestOn('mac-os') |
| // Keep in sync with static_func_test.dart. These are the same tests, but |
| // without using @Native. |
| import 'dart:ffi'; |
| import 'dart:io'; |
| |
| import 'package:ffi/ffi.dart'; |
| import 'package:objective_c/objective_c.dart'; |
| import 'package:path/path.dart' as path; |
| import 'package:test/test.dart'; |
| |
| import '../test_utils.dart'; |
| import 'static_func_bindings.dart'; |
| import 'util.dart'; |
| |
| typedef IntBlock = ObjCBlock_Int32_Int32; |
| |
| void main() { |
| late StaticFuncTestObjCLibrary lib; |
| |
| group('static functions', () { |
| setUpAll(() { |
| final dylib = File( |
| path.join( |
| packagePathForTests, |
| 'test', |
| 'native_objc_test', |
| 'objc_test.dylib', |
| ), |
| ); |
| verifySetupFile(dylib); |
| lib = StaticFuncTestObjCLibrary(DynamicLibrary.open(dylib.absolute.path)); |
| |
| generateBindingsForCoverage('static_func'); |
| }); |
| |
| Pointer<Int32> staticFuncOfObjectRefCountTest(Allocator alloc) { |
| final counter = alloc<Int32>(); |
| counter.value = 0; |
| |
| final obj = StaticFuncTestObj.newWithCounter(counter); |
| expect(counter.value, 1); |
| |
| final pool = lib.objc_autoreleasePoolPush(); |
| final outputObj = lib.staticFuncOfObject(obj); |
| lib.objc_autoreleasePoolPop(pool); |
| expect(obj, outputObj); |
| expect(counter.value, 1); |
| |
| return counter; |
| } |
| |
| test( |
| 'Objects passed through static functions have correct ref counts', |
| () { |
| using((Arena arena) { |
| final (counter) = staticFuncOfObjectRefCountTest(arena); |
| doGC(); |
| expect(counter.value, 0); |
| }); |
| }, |
| skip: !canDoGC, |
| ); |
| |
| Pointer<Int32> staticFuncOfNullableObjectRefCountTest(Allocator alloc) { |
| final counter = alloc<Int32>(); |
| counter.value = 0; |
| |
| final obj = StaticFuncTestObj.newWithCounter(counter); |
| expect(counter.value, 1); |
| |
| final pool = lib.objc_autoreleasePoolPush(); |
| final outputObj = lib.staticFuncOfNullableObject(obj); |
| lib.objc_autoreleasePoolPop(pool); |
| expect(obj, outputObj); |
| expect(counter.value, 1); |
| |
| return counter; |
| } |
| |
| test( |
| 'Nullables passed through static functions have correct ref counts', |
| () { |
| using((Arena arena) { |
| final (counter) = staticFuncOfNullableObjectRefCountTest(arena); |
| doGC(); |
| expect(counter.value, 0); |
| |
| expect(lib.staticFuncOfNullableObject(null), isNull); |
| }); |
| }, |
| skip: !canDoGC, |
| ); |
| |
| Pointer<ObjCBlockImpl> staticFuncOfBlockRefCountTest() { |
| final block = IntBlock.fromFunction((int x) => 2 * x); |
| expect(blockRetainCount(block.ref.pointer.cast()), 1); |
| |
| final pool = lib.objc_autoreleasePoolPush(); |
| final outputBlock = lib.staticFuncOfBlock(block); |
| lib.objc_autoreleasePoolPop(pool); |
| expect(block, outputBlock); |
| expect(blockRetainCount(block.ref.pointer.cast()), 2); |
| |
| return block.ref.pointer; |
| } |
| |
| test( |
| 'Blocks passed through static functions have correct ref counts', |
| () { |
| final (rawBlock) = staticFuncOfBlockRefCountTest(); |
| doGC(); |
| expect(blockRetainCount(rawBlock), 0); |
| }, |
| skip: !canDoGC, |
| ); |
| |
| Pointer<Int32> staticFuncReturnsRetainedRefCountTest(Allocator alloc) { |
| final counter = alloc<Int32>(); |
| counter.value = 0; |
| |
| final outputObj = lib.staticFuncReturnsRetained(counter); |
| expect(counter.value, 1); |
| |
| return counter; |
| } |
| |
| test('Objects returned from static functions with NS_RETURNS_RETAINED ' |
| 'have correct ref counts', () { |
| using((Arena arena) { |
| final (counter) = staticFuncReturnsRetainedRefCountTest(arena); |
| doGC(); |
| expect(counter.value, 0); |
| }); |
| }, skip: !canDoGC); |
| |
| Pointer<Int32> staticFuncOfObjectReturnsRetainedRefCountTest( |
| Allocator alloc, |
| ) { |
| final counter = alloc<Int32>(); |
| counter.value = 0; |
| |
| final obj = StaticFuncTestObj.newWithCounter(counter); |
| expect(counter.value, 1); |
| |
| final outputObj = lib.staticFuncReturnsRetainedArg(obj); |
| expect(obj, outputObj); |
| expect(counter.value, 1); |
| |
| return counter; |
| } |
| |
| test('Objects passed through static functions with NS_RETURNS_RETAINED ' |
| 'have correct ref counts', () { |
| using((Arena arena) { |
| final (counter) = staticFuncOfObjectReturnsRetainedRefCountTest(arena); |
| doGC(); |
| expect(counter.value, 0); |
| }); |
| }, skip: !canDoGC); |
| |
| test('Objects passed to static functions that consume them ' |
| 'have correct ref counts', () { |
| final counter = calloc<Int32>(); |
| StaticFuncTestObj? obj1 = StaticFuncTestObj.newWithCounter(counter); |
| final obj1raw = obj1.ref.pointer; |
| |
| expect(objectRetainCount(obj1raw), 1); |
| expect(counter.value, 1); |
| |
| lib.staticFuncConsumesArg(obj1); |
| |
| expect(objectRetainCount(obj1raw), 1); |
| expect(counter.value, 1); |
| |
| obj1 = null; |
| doGC(); |
| expect(objectRetainCount(obj1raw), 0); |
| expect(counter.value, 0); |
| calloc.free(counter); |
| }, skip: !canDoGC); |
| |
| test('Internal variable conflict resolution', () { |
| // Regression test for https://github.com/dart-lang/native/issues/2760 |
| expect(lib.foo(123), 1230); |
| expect(lib.fooPtr(123), 12300); |
| }); |
| }); |
| } |