blob: 908e8fc4f592fc71778507e05477c96e35aade96 [file] [log] [blame]
// Copyright (c) 2022, 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')
import 'dart:async';
import 'dart:ffi';
import 'dart:io';
import 'package:test/test.dart';
import 'package:ffi/ffi.dart';
import '../test_utils.dart';
import 'block_bindings.dart';
import 'util.dart';
void main() {
late BlockTestObjCLibrary lib;
late void Function(Pointer<Char>, Pointer<Void>) executeInternalCommand;
group('Blocks', () {
setUpAll(() {
logWarnings();
final dylib = File('test/native_objc_test/block_test.dylib');
verifySetupFile(dylib);
lib = BlockTestObjCLibrary(DynamicLibrary.open(dylib.absolute.path));
executeInternalCommand = DynamicLibrary.process().lookupFunction<
Void Function(Pointer<Char>, Pointer<Void>),
void Function(
Pointer<Char>, Pointer<Void>)>('Dart_ExecuteInternalCommand');
generateBindingsForCoverage('block');
});
doGC() {
final gcNow = "gc-now".toNativeUtf8();
executeInternalCommand(gcNow.cast(), nullptr);
calloc.free(gcNow);
}
test('BlockTester is working', () {
// This doesn't test any Block functionality, just that the BlockTester
// itself is working correctly.
final blockTester = BlockTester.makeFromMultiplier_(lib, 10);
expect(blockTester.call_(123), 1230);
final intBlock = blockTester.getBlock();
final blockTester2 = BlockTester.makeFromBlock_(lib, intBlock);
blockTester2.pokeBlock();
expect(blockTester2.call_(456), 4560);
});
test('Block from function pointer', () {
final block = ObjCBlock1.fromFunctionPointer(
lib, Pointer.fromFunction(_add100, 999));
final blockTester = BlockTester.makeFromBlock_(lib, block);
blockTester.pokeBlock();
expect(blockTester.call_(123), 223);
expect(block(123), 223);
});
int Function(int) makeAdder(int addTo) {
return (int x) => addTo + x;
}
test('Block from function', () {
final block = ObjCBlock1.fromFunction(lib, makeAdder(4000));
final blockTester = BlockTester.makeFromBlock_(lib, block);
blockTester.pokeBlock();
expect(blockTester.call_(123), 4123);
expect(block(123), 4123);
});
test('Listener block same thread', () async {
final hasRun = Completer();
int value = 0;
final block = ObjCBlock.listener(lib, () {
value = 123;
hasRun.complete();
});
BlockTester.callOnSameThread_(lib, block);
await hasRun.future;
expect(value, 123);
});
test('Listener block new thread', () async {
final hasRun = Completer();
int value = 0;
final block = ObjCBlock.listener(lib, () {
value = 123;
hasRun.complete();
});
final thread = BlockTester.callOnNewThread_(lib, block);
thread.start();
await hasRun.future;
expect(value, 123);
});
test('Float block', () {
final block = ObjCBlock2.fromFunction(lib, (double x) {
return x + 4.56;
});
expect(block(1.23), closeTo(5.79, 1e-6));
expect(BlockTester.callFloatBlock_(lib, block), closeTo(5.79, 1e-6));
});
test('Double block', () {
final block = ObjCBlock3.fromFunction(lib, (double x) {
return x + 4.56;
});
expect(block(1.23), closeTo(5.79, 1e-6));
expect(BlockTester.callDoubleBlock_(lib, block), closeTo(5.79, 1e-6));
});
test('Struct block', () {
final inputPtr = calloc<Vec4>();
final input = inputPtr.ref;
input.x = 1.2;
input.y = 3.4;
input.z = 5.6;
input.w = 7.8;
final tempPtr = calloc<Vec4>();
final temp = tempPtr.ref;
final block = ObjCBlock4.fromFunction(lib, (Vec4 v) {
// Twiddle the Vec4 components.
temp.x = v.y;
temp.y = v.z;
temp.z = v.w;
temp.w = v.x;
return temp;
});
final result1 = block(input);
expect(result1.x, 3.4);
expect(result1.y, 5.6);
expect(result1.z, 7.8);
expect(result1.w, 1.2);
final result2Ptr = calloc<Vec4>();
final result2 = result2Ptr.ref;
BlockTester.callVec4Block_(lib, result2Ptr, block);
expect(result2.x, 3.4);
expect(result2.y, 5.6);
expect(result2.z, 7.8);
expect(result2.w, 1.2);
calloc.free(inputPtr);
calloc.free(tempPtr);
calloc.free(result2Ptr);
});
Pointer<Void> funcPointerBlockRefCountTest() {
final block = ObjCBlock1.fromFunctionPointer(
lib, Pointer.fromFunction(_add100, 999));
expect(BlockTester.getBlockRetainCount_(lib, block.pointer.cast()), 1);
return block.pointer.cast();
}
test('Function pointer block ref counting', () {
final rawBlock = funcPointerBlockRefCountTest();
doGC();
expect(BlockTester.getBlockRetainCount_(lib, rawBlock.cast()), 0);
});
Pointer<Void> funcBlockRefCountTest() {
final block = ObjCBlock1.fromFunction(lib, makeAdder(4000));
expect(BlockTester.getBlockRetainCount_(lib, block.pointer.cast()), 1);
return block.pointer.cast();
}
test('Function pointer block ref counting', () {
final rawBlock = funcBlockRefCountTest();
doGC();
expect(BlockTester.getBlockRetainCount_(lib, rawBlock.cast()), 0);
});
test('Block fields have sensible values', () {
final block = ObjCBlock1.fromFunction(lib, makeAdder(4000));
final blockPtr = block.pointer;
expect(blockPtr.ref.isa, isNot(0));
expect(blockPtr.ref.flags, isNot(0)); // Set by Block_copy.
expect(blockPtr.ref.reserved, 0);
expect(blockPtr.ref.invoke, isNot(0));
expect(blockPtr.ref.target, isNot(0));
final descPtr = blockPtr.ref.descriptor;
expect(descPtr.ref.reserved, 0);
expect(descPtr.ref.size, isNot(0));
expect(descPtr.ref.copy_helper, nullptr);
expect(descPtr.ref.dispose_helper, nullptr);
expect(descPtr.ref.signature, nullptr);
});
});
}
int _add100(int x) {
return x + 100;
}