[K/N] Add a stress test on dispose-on-main ^KT-63423

This commit is contained in:
Alexander Shabalin
2023-11-20 15:32:53 +01:00
committed by Space Team
parent 0741250b12
commit eb0bd0112a
5 changed files with 143 additions and 0 deletions
@@ -1239,6 +1239,14 @@ if (PlatformInfo.isAppleTarget(project)) {
it.headers "$projectDir/interop/objc/overridabilityCondition/lib.h"
it.extraOpts "-Xcompile-source", "$projectDir/interop/objc/overridabilityCondition/lib.m"
}
createInterop("kt63423_dispose_on_main_stress") {
it.defFile 'interop/objc/kt63423_dispose_on_main_stress/objclib.def'
it.headers "$projectDir/interop/objc/kt63423_dispose_on_main_stress/objclib.h"
it.extraOpts "-Xcompile-source", "$projectDir/interop/objc/kt63423_dispose_on_main_stress/objclib.m"
it.extraOpts "-Xsource-compiler-option", "-DNS_FORMAT_ARGUMENT(A)="
it.extraOpts "-Xsource-compiler-option", "-fobjc-arc"
it.extraOpts "-Xsource-compiler-option", "-ObjC++"
}
}
createInterop("withSpaces") {
@@ -1848,6 +1856,18 @@ if (PlatformInfo.isAppleTarget(project)) {
source = 'interop/kt59167/main.kt'
interop = 'kt59167'
}
interopTest("interop_objc_kt63423_dispose_on_main_stress") {
// Test depends on macOS-specific AppKit
enabled = (project.testTarget == 'macos_x64' || project.testTarget == 'macos_arm64' || project.testTarget == null)
&& !isNoopGC // requires some GC
source = 'interop/objc/kt63423_dispose_on_main_stress/main.kt'
interop = "kt63423_dispose_on_main_stress"
flags = [
'-opt-in=kotlin.native.internal.InternalForKotlinNative', // MemoryUsageInfo is internal
'-Xbinary=gcSchedulerType=manual', // This test requires very careful timing when GC can happen
]
}
}
tasks.register("KT-50983", KonanDriverTest) {
@@ -0,0 +1,45 @@
/*
* Copyright 2010-2023 JetBrains s.r.o. and Kotlin Programming Language contributors.
* Use of this source code is governed by the Apache 2.0 license that can be found in the license/LICENSE.txt file.
*/
@file:OptIn(kotlin.experimental.ExperimentalNativeApi::class)
import objclib.*
import kotlin.native.internal.MemoryUsageInfo
import kotlin.test.*
import kotlinx.cinterop.*
class PeakRSSChecker(private val rssDiffLimitBytes: Long) {
// On Linux, the child process might immediately commit the same amount of memory as the parent.
// So, measure difference between peak RSS measurements.
private val initialBytes = MemoryUsageInfo.peakResidentSetSizeBytes.also {
check(it != 0L) { "Error trying to obtain peak RSS. Check if current platform is supported" }
}
fun check(): Long {
val diffBytes = MemoryUsageInfo.peakResidentSetSizeBytes - initialBytes
check(diffBytes <= rssDiffLimitBytes) { "Increased peak RSS by $diffBytes bytes which is more than $rssDiffLimitBytes" }
return diffBytes
}
}
fun alloc(): Unit = autoreleasepool {
OnDestroyHook()
Unit
}
fun waitDestruction() {
assertTrue(isMainThread())
kotlin.native.internal.GC.collect()
spin()
}
fun main() = startApp {
repeat(500000) {
alloc()
}
val peakRSSChecker = PeakRSSChecker(10_000_000L) // ~10MiB allowed difference for running finalizers
waitDestruction()
peakRSSChecker.check()
}
@@ -0,0 +1,3 @@
language = Objective-C
headerFilter = **/objclib.h
linkerOpts = -framework AppKit
@@ -0,0 +1,17 @@
#include <objc/NSObject.h>
@interface OnDestroyHook : NSObject
- (instancetype)init;
@end
#ifdef __cplusplus
extern "C" {
#endif
void startApp(void (^task)());
BOOL isMainThread();
void spin();
#ifdef __cplusplus
}
#endif
@@ -0,0 +1,58 @@
#include "objclib.h"
#include <cinttypes>
#include <dispatch/dispatch.h>
#include <map>
#import <AppKit/NSApplication.h>
#import <Foundation/NSRunLoop.h>
#import <Foundation/NSThread.h>
std::map<uintptr_t, bool> dictionary;
@implementation OnDestroyHook
- (instancetype)init {
if (self = [super init]) {
dictionary[(uintptr_t)self] = true;
}
return self;
}
- (void)dealloc {
dictionary[(uintptr_t)self] = false;
}
@end
extern "C" void startApp(void (^task)()) {
dispatch_async(dispatch_get_main_queue(), ^{
// At this point all other scheduled main queue tasks were already executed.
// Executing via performBlock to allow a recursive run loop in `spin()`.
[[NSRunLoop currentRunLoop] performBlock:^{
task();
[NSApp terminate:NULL];
}];
});
[[NSApplication sharedApplication] run];
}
extern "C" BOOL isMainThread() {
return [NSThread isMainThread];
}
extern "C" void spin() {
if ([NSRunLoop currentRunLoop] != [NSRunLoop mainRunLoop]) {
fprintf(stderr, "Must spin main run loop\n");
exit(1);
}
while (true) {
[[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate dateWithTimeIntervalSinceNow:0.1]];
bool done = true;
for (auto kvp : dictionary) {
if (kvp.second) {
done = false;
break;
}
}
if (done) return;
}
}