Skip to content

Commit be75e66

Browse files
author
Christian Melchior
committed
Merge branch 'releases'
# Conflicts: # CHANGELOG.md
2 parents e19b1d1 + 0dcdbfb commit be75e66

File tree

10 files changed

+112
-36
lines changed

10 files changed

+112
-36
lines changed

CHANGELOG.md

+29
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,35 @@
2727
* None.
2828

2929

30+
## 1.12.1-SNAPSHOT (YYYY-MM-DD)
31+
32+
### Breaking Changes
33+
* None.
34+
35+
### Enhancements
36+
* None.
37+
38+
### Fixed
39+
* Fix craches caused by posting to a released scheduler. (Issue [#1543](https://github.com/realm/realm-kotlin/issues/1543))
40+
41+
### Compatibility
42+
* File format: Generates Realms with file format v23.
43+
* Realm Studio 13.0.0 or above is required to open Realms created by this version.
44+
* This release is compatible with the following Kotlin releases:
45+
* Kotlin 1.8.0 and above. The K2 compiler is not supported yet.
46+
* Ktor 2.1.2 and above.
47+
* Coroutines 1.7.0 and above.
48+
* AtomicFu 0.18.3 and above.
49+
* The new memory model only. See https://github.com/realm/realm-kotlin#kotlin-memory-model-and-coroutine-compatibility
50+
* Minimum Kbson 0.3.0.
51+
* Minimum Gradle version: 6.8.3.
52+
* Minimum Android Gradle Plugin version: 4.1.3.
53+
* Minimum Android SDK: 16.
54+
55+
### Internal
56+
* Updated to Realm Core 13.23.3, commit 7556b535aa7b27d49c13444894f7e9db778b3203.
57+
58+
3059
## 1.12.0 (2023-11-02)
3160

3261
This release upgrades the Sync metadata in a way that is not compatible with older versions. To downgrade a Sync app from this version, you'll need to manually delete the metadata folder located at `$[SYNC-ROOT-DIRECTORY]/mongodb-realm/[APP-ID]/server-utility/metadata/`. This will log out all users.

dependencies.list

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,11 @@
11
# Version of MongoDB Realm used by integration tests
22
# See https://github.com/realm/ci/packages/147854 for available versions
3-
MONGODB_REALM_SERVER=2023-10-10
3+
MONGODB_REALM_SERVER=2023-11-07
44

55
# `BAAS` and `BAAS-UI` projects commit hashes matching MONGODB_REALM_SERVER image version
66
# note that the MONGODB_REALM_SERVER image is a nightly build, find the matching commits
77
# for that date within the following repositories:
88
# https://github.com/10gen/baas/
99
# https://github.com/10gen/baas-ui/
10-
REALM_BAAS_GIT_HASH=8246fc548763eb908b8090df864e9924e3330a0d
11-
REALM_BAAS_UI_GIT_HASH=8a1843be2bf24f2faa705c5470a5bdd8d954f7ea
10+
REALM_BAAS_GIT_HASH=41fa6cdbca47826c20a64f756e21b2c184393e90
11+
REALM_BAAS_UI_GIT_HASH=b97a27ac858e0e8126aeb63f6ff9734d11029a91

packages/cinterop/src/jvm/kotlin/io/realm/kotlin/internal/interop/RealmInterop.kt

+13-1
Original file line numberDiff line numberDiff line change
@@ -2138,10 +2138,22 @@ fun ObjectId.asRealmObjectIdT(): realm_object_id_t {
21382138

21392139
private class JVMScheduler(dispatcher: CoroutineDispatcher) {
21402140
val scope: CoroutineScope = CoroutineScope(dispatcher)
2141+
val lock = SynchronizableObject()
2142+
var cancelled = false
21412143

21422144
fun notifyCore(schedulerPointer: Long) {
21432145
scope.launch {
2144-
realmc.invoke_core_notify_callback(schedulerPointer)
2146+
lock.withLock {
2147+
if (!cancelled) {
2148+
realmc.invoke_core_notify_callback(schedulerPointer)
2149+
}
2150+
}
2151+
}
2152+
}
2153+
2154+
fun cancel() {
2155+
lock.withLock {
2156+
cancelled = true
21452157
}
21462158
}
21472159
}

packages/cinterop/src/nativeDarwin/kotlin/io/realm/kotlin/internal/interop/RealmInterop.kt

+21-6
Original file line numberDiff line numberDiff line change
@@ -124,6 +124,7 @@ import realm_wrapper.realm_user_t
124124
import realm_wrapper.realm_value_t
125125
import realm_wrapper.realm_value_type
126126
import realm_wrapper.realm_version_id_t
127+
import realm_wrapper.realm_work_queue_t
127128
import kotlin.collections.set
128129
import kotlin.native.internal.createCleaner
129130

@@ -560,17 +561,19 @@ actual object RealmInterop {
560561
// free: realm_wrapper.realm_free_userdata_func_t? /* = kotlinx.cinterop.CPointer<kotlinx.cinterop.CFunction<(kotlinx.cinterop.COpaquePointer? /* = kotlinx.cinterop.CPointer<out kotlinx.cinterop.CPointed>? */) -> kotlin.Unit>>? */,
561562
staticCFunction<COpaquePointer?, Unit> { userdata ->
562563
printlntid("free")
563-
userdata?.asStableRef<SingleThreadDispatcherScheduler>()?.dispose()
564+
val stableSchedulerRef: StableRef<SingleThreadDispatcherScheduler>? = userdata?.asStableRef<SingleThreadDispatcherScheduler>()
565+
stableSchedulerRef?.get()?.cancel()
566+
stableSchedulerRef?.dispose()
564567
},
565568

566569
// notify: realm_wrapper.realm_scheduler_notify_func_t? /* = kotlinx.cinterop.CPointer<kotlinx.cinterop.CFunction<(kotlinx.cinterop.COpaquePointer? /* = kotlinx.cinterop.CPointer<out kotlinx.cinterop.CPointed>? */) -> kotlin.Unit>>? */,
567-
staticCFunction<COpaquePointer?, Unit> { userdata ->
570+
staticCFunction<COpaquePointer?, CPointer<realm_work_queue_t>?, Unit> { userdata, work_queue ->
568571
// Must be thread safe
569572
val scheduler =
570573
userdata!!.asStableRef<SingleThreadDispatcherScheduler>().get()
571574
printlntid("$scheduler notify")
572575
try {
573-
scheduler.notify()
576+
scheduler.notify(work_queue)
574577
} catch (e: Exception) {
575578
// Should never happen, but is included for development to get some indicators
576579
// on errors instead of silent crashes.
@@ -3392,7 +3395,7 @@ actual object RealmInterop {
33923395
}
33933396

33943397
interface Scheduler {
3395-
fun notify()
3398+
fun notify(work_queue: CPointer<realm_work_queue_t>?)
33963399
}
33973400

33983401
class SingleThreadDispatcherScheduler(
@@ -3402,23 +3405,35 @@ actual object RealmInterop {
34023405
private val scope: CoroutineScope = CoroutineScope(dispatcher)
34033406
val ref: CPointer<out CPointed> = StableRef.create(this).asCPointer()
34043407
private lateinit var scheduler: CPointer<realm_scheduler_t>
3408+
private val lock = SynchronizableObject()
3409+
private var cancelled = false
34053410

34063411
fun setScheduler(scheduler: CPointer<realm_scheduler_t>) {
34073412
this.scheduler = scheduler
34083413
}
34093414

3410-
override fun notify() {
3415+
override fun notify(work_queue: CPointer<realm_work_queue_t>?) {
34113416
scope.launch {
34123417
try {
34133418
printlntid("on dispatcher")
3414-
realm_wrapper.realm_scheduler_perform_work(scheduler)
3419+
lock.withLock {
3420+
if (!cancelled) {
3421+
realm_wrapper.realm_scheduler_perform_work(work_queue)
3422+
}
3423+
}
34153424
} catch (e: Exception) {
34163425
// Should never happen, but is included for development to get some indicators
34173426
// on errors instead of silent crashes.
34183427
e.printStackTrace()
34193428
}
34203429
}
34213430
}
3431+
3432+
fun cancel() {
3433+
lock.withLock {
3434+
cancelled = true
3435+
}
3436+
}
34223437
}
34233438
}
34243439

packages/external/core

Submodule core updated 74 files

packages/jni-swig-stub/src/main/jni/realm_api_helpers.cpp

+18-12
Original file line numberDiff line numberDiff line change
@@ -264,25 +264,22 @@ class CustomJVMScheduler {
264264
JNIEnv *jenv = get_env();
265265
jclass jvm_scheduler_class = jenv->FindClass("io/realm/kotlin/internal/interop/JVMScheduler");
266266
m_notify_method = jenv->GetMethodID(jvm_scheduler_class, "notifyCore", "(J)V");
267+
m_cancel_method = jenv->GetMethodID(jvm_scheduler_class, "cancel", "()V");
267268
m_jvm_dispatch_scheduler = jenv->NewGlobalRef(dispatchScheduler);
268269
}
269270

270271
~CustomJVMScheduler() {
271272
get_env(true)->DeleteGlobalRef(m_jvm_dispatch_scheduler);
272273
}
273274

274-
void set_scheduler(realm_scheduler_t* scheduler) {
275-
m_scheduler = scheduler;
276-
}
277-
278-
void notify() {
275+
void notify(realm_work_queue_t* work_queue) {
279276
// There is currently no signaling of creation/tear down of the core notifier thread, so we
280277
// just attach it as a daemon thread here on first notification to allow the JVM to
281278
// shutdown propertly. See https://github.com/realm/realm-core/issues/6429
282279
auto jenv = get_env(true, true, "core-notifier");
283280
jni_check_exception(jenv);
284281
jenv->CallVoidMethod(m_jvm_dispatch_scheduler, m_notify_method,
285-
reinterpret_cast<jlong>(m_scheduler));
282+
reinterpret_cast<jlong>(work_queue));
286283
}
287284

288285
bool is_on_thread() const noexcept {
@@ -293,12 +290,18 @@ class CustomJVMScheduler {
293290
return true;
294291
}
295292

293+
void cancel() {
294+
auto jenv = get_env(true, true, "core-notifier");
295+
jenv->CallVoidMethod(m_jvm_dispatch_scheduler, m_cancel_method);
296+
jni_check_exception(jenv);
297+
}
298+
296299

297300
private:
298301
std::thread::id m_id;
299302
jmethodID m_notify_method;
303+
jmethodID m_cancel_method;
300304
jobject m_jvm_dispatch_scheduler;
301-
realm_scheduler_t *m_scheduler;
302305
};
303306

304307
// Note: using jlong here will create a linker issue
@@ -309,8 +312,8 @@ class CustomJVMScheduler {
309312
//
310313
// I suspect this could be related to the fact that jni.h defines jlong differently between Android (typedef int64_t)
311314
// and JVM which is a (typedef long long) resulting in a different signature of the method that could be found by the linker.
312-
void invoke_core_notify_callback(int64_t scheduler) {
313-
realm_scheduler_perform_work(reinterpret_cast<realm_scheduler_t *>(scheduler));
315+
void invoke_core_notify_callback(int64_t work_queue) {
316+
realm_scheduler_perform_work(reinterpret_cast<realm_work_queue_t *>(work_queue));
314317
}
315318

316319
realm_scheduler_t*
@@ -319,13 +322,16 @@ realm_create_scheduler(jobject dispatchScheduler) {
319322
auto jvmScheduler = new CustomJVMScheduler(dispatchScheduler);
320323
auto scheduler = realm_scheduler_new(
321324
jvmScheduler,
322-
[](void *userdata) { delete(static_cast<CustomJVMScheduler *>(userdata)); },
323-
[](void *userdata) { static_cast<CustomJVMScheduler *>(userdata)->notify(); },
325+
[](void *userdata) {
326+
auto jvmScheduler = static_cast<CustomJVMScheduler *>(userdata);
327+
jvmScheduler->cancel();
328+
delete(jvmScheduler);
329+
},
330+
[](void *userdata, realm_work_queue_t* work_queue) { static_cast<CustomJVMScheduler *>(userdata)->notify(work_queue); },
324331
[](void *userdata) { return static_cast<CustomJVMScheduler *>(userdata)->is_on_thread(); },
325332
[](const void *userdata, const void *userdata_other) { return userdata == userdata_other; },
326333
[](void *userdata) { return static_cast<CustomJVMScheduler *>(userdata)->can_invoke(); }
327334
);
328-
jvmScheduler->set_scheduler(scheduler);
329335
return scheduler;
330336
}
331337
throw std::runtime_error("Null dispatchScheduler");

packages/test-base/src/commonMain/kotlin/io/realm/kotlin/test/util/Utils.kt

+10-4
Original file line numberDiff line numberDiff line change
@@ -21,8 +21,10 @@ import io.realm.kotlin.Realm
2121
import io.realm.kotlin.test.platform.PlatformUtils
2222
import io.realm.kotlin.types.RealmInstant
2323
import io.realm.kotlin.types.RealmObject
24+
import kotlinx.coroutines.TimeoutCancellationException
2425
import kotlinx.coroutines.channels.Channel
25-
import kotlinx.coroutines.withTimeout
26+
import kotlinx.coroutines.selects.onTimeout
27+
import kotlinx.coroutines.selects.select
2628
import kotlinx.datetime.Instant
2729
import kotlin.time.Duration
2830
import kotlin.time.Duration.Companion.minutes
@@ -92,8 +94,12 @@ fun Instant.toRealmInstant(): RealmInstant {
9294
}
9395

9496
// Variant of `Channel.receiveOrFail()` that will will throw if a timeout is hit.
95-
suspend fun <T : Any?> Channel<T>.receiveOrFail(timeout: Duration = 1.minutes): T {
96-
return withTimeout(timeout) {
97-
receive()
97+
suspend fun <T : Any?> Channel<T>.receiveOrFail(timeout: Duration = 1.minutes, message: String? = null): T {
98+
return select {
99+
this@receiveOrFail.onReceive { it }
100+
onTimeout(timeout) {
101+
@Suppress("invisible_member")
102+
throw TimeoutCancellationException("Timeout: $message")
103+
}
98104
}
99105
}

packages/test-base/src/commonTest/kotlin/io/realm/kotlin/test/common/VersionTrackingTests.kt

+3-1
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,7 @@ import kotlin.test.Test
4545
import kotlin.test.assertEquals
4646
import kotlin.test.assertNotNull
4747
import kotlin.test.assertNull
48+
import kotlin.test.assertTrue
4849

4950
class VersionTrackingTests {
5051
private lateinit var initialLogLevel: LogLevel
@@ -150,7 +151,8 @@ class VersionTrackingTests {
150151
realm.write<Unit> { copyToRealm(Sample()) }
151152
realm.write<Unit> { copyToRealm(Sample()) }
152153
realm.activeVersions().run {
153-
assertEquals(1, allTracked.size, toString())
154+
// Initially tracked version from user facing realm might have been released by now
155+
assertTrue(allTracked.size <= 1, toString())
154156
assertNotNull(notifier, toString())
155157
assertEquals(0, notifier?.active?.size, toString())
156158
assertNotNull(writer, toString())

packages/test-base/src/nativeDarwinTest/kotlin/io/realm/kotlin/test/darwin/MemoryTests.kt

+9-5
Original file line numberDiff line numberDiff line change
@@ -61,6 +61,15 @@ class MemoryTests {
6161
@OptIn(ExperimentalStdlibApi::class)
6262
println("NEW_MEMORY_MODEL: " + isExperimentalMM())
6363

64+
// Referencing things like
65+
// NSProcessInfo.Companion.processInfo().operatingSystemVersionString
66+
// platform.Foundation.NSFileManager.defaultManager
67+
// as done in Darwin SystemUtils.kt and initialized lazily, so do a full realm-lifecycle
68+
// to only measure increases over the actual test
69+
// - Ensure that we clean up any released memory to get a nice baseline
70+
platform.posix.sleep(1 * 5) // give chance to the Collector Thread to process out of scope references
71+
triggerGC()
72+
// - Record the baseline
6473
val initialAllocation = parseSizeString(runSystemCommand(amountOfMemoryMappedInProcessCMD))
6574

6675
val referenceHolder = mutableListOf<Sample>();
@@ -91,11 +100,6 @@ class MemoryTests {
91100
triggerGC()
92101
platform.posix.sleep(1 * 5) // give chance to the Collector Thread to process out of scope references
93102

94-
// Referencing things like
95-
// NSProcessInfo.Companion.processInfo().operatingSystemVersionString
96-
// platform.Foundation.NSFileManager.defaultManager
97-
// as done in Darwin SystemUtils.kt cause allocations so we just assert the increase over
98-
// the test
99103
val allocation = parseSizeString(runSystemCommand(amountOfMemoryMappedInProcessCMD))
100104
assertEquals(initialAllocation, allocation, "mmap allocation exceeds expectations: initial=$initialAllocation current=$allocation")
101105
}

packages/test-sync/src/commonTest/kotlin/io/realm/kotlin/test/mongodb/common/SyncedRealmTests.kt

+5-3
Original file line numberDiff line numberDiff line change
@@ -739,7 +739,7 @@ class SyncedRealmTests {
739739
realm.writeBlocking { copyToRealm(masterObject) }
740740
realm.syncSession.uploadAllLocalChanges()
741741
}
742-
assertEquals(42, counterValue.receiveOrFail(), "Failed to receive 42")
742+
assertEquals(42, counterValue.receiveOrFail(message = "Failed to receive 42"))
743743

744744
// Increment counter asynchronously after download initial data (1)
745745
val increment1 = async {
@@ -753,9 +753,10 @@ class SyncedRealmTests {
753753
.mutableRealmIntField
754754
.increment(1)
755755
}
756+
realm.syncSession.uploadAllLocalChanges(10.seconds)
756757
}
757758
}
758-
assertEquals(43, counterValue.receiveOrFail(), "Failed to receive 43")
759+
assertEquals(43, counterValue.receiveOrFail(message = "Failed to receive 43"))
759760

760761
// Increment counter asynchronously after download initial data (2)
761762
val increment2 = async {
@@ -769,9 +770,10 @@ class SyncedRealmTests {
769770
.mutableRealmIntField
770771
.increment(1)
771772
}
773+
realm.syncSession.uploadAllLocalChanges(10.seconds)
772774
}
773775
}
774-
assertEquals(44, counterValue.receiveOrFail(), "Failed to receive 44")
776+
assertEquals(44, counterValue.receiveOrFail(message = "Failed to receive 44"))
775777

776778
increment1.cancel()
777779
increment2.cancel()

0 commit comments

Comments
 (0)