Suggestion - Runtime warnings in debug builds if @MainActor methods/vars are called from background threads

That works as expected (you can return arbitrary objects from the MainActor annotated function, those objects could be called on arbitrary threads).

This doesn't work either but it is closer to what you need:

    // no @MainActor annotation here
    var binding: Binding<Bool> {
        Binding<Bool> { @MainActor in
            dispatchPrecondition(condition: .onQueue(.main))
            return true
        } set: { @MainActor _, _ in
            dispatchPrecondition(condition: .onQueue(.main))
        }
    }

I'd also recommend to send code snippets as text embedded in ``` brackets.

you might expect that @MainActor functions can be called on arbitrary threads.
(I know it to be true)

But that's explicitly what the compiler is trying to stop, and is also what many users think is guaranteed not to happen. (and indeed what Apple videos promise)

Given the almost complete lack of documentation around @MainActor - it's not very surprising that people don't know what it does.

Still - the fact that the compiler will object if the variable is not marked @MainActor even though it doesn't enforce the @MainActor-ness is at the very least misleading

struct ContentView: View {
    var body: some View {
        //Button just updates the binding when clicked
        //But on a background thread
        BoundButton(val:binding)
    }
    
    @MainActor
    var binding:Binding<Bool> {
        //The Binding is created on @MainActor
        return Binding<Bool> {
            return true
        } set: { _, _ in
            //But the setter could be called from any thread
            //So the compiler shouldn't allow this call to
            //@MainActor annotated shouldBeOnMain
            
            //Error generated here if variable is -not- marked @MainActor
            //Call to main actor-isolated instance method 'shouldBeOnMain()' in a synchronous nonisolated context
            shouldBeOnMain()
        }
    }


    
    @MainActor
    func shouldBeOnMain() {
        if !Thread.isMainThread {
            fatalError("Not on Main!")
        }
    }
}

struct BoundButton: View {
    @Binding var val:Bool
    
    var body:some View {
        Button {
            Task.detached {
                val = false
            }
        } label: {
            Text("TryMe")
        }
    }
}
1 Like

Try adding @MainActor annotation on the @Binding variable:

@MainActor @Binding var val: Bool

Note that the checks in the compiler are quite shallow:

DispatchQueue.main.async {
    val = true // ok
}

let queue = DispatchQueue.main
queue.async {
    val = true // not ok
}

I agree in that it's under documented and underperforming.

I think you're missing my point.

I'm not looking for coding help. I'm pointing out yet another way that @MainActor doesn't work in the way that is expected.

1 Like

I was merely pointing out that while it's far from ideal, at least this situation is handled correctly:

struct BoundButton: View {
    @MainActor @Binding var val: Bool
    
    var body:some View {
        Button {
            Task.detached {
                val = false // šŸ›‘ Main actor-isolated property 'val' can not be mutated from a Sendable closure
                DispatchQueue.main.async {
                    val = false // ok
                }
            }
            Task.detached { @MainActor in
                val = false // ok
            }
        } label: {
            Text("TryMe")
        }
    }
}

I would have expected a compilation error in this code when compiling under strict concurrency checking (which will be enabled by default in Swift 6). However, it appears that even under strict concurrency checking, this isn't being diagnosed.

I would expect it because the parameters passed to Binding.init(get:set:) are not main-actor isolated. The set closure declared here seems to be inferring @MainActor isolation from its declaring context, but it is being passed off to Binding.init(), which doesn't promise to fulfill that contract.

Is it expected that a @MainActor closure can be passed to a non-actor-isolated argument?

1 Like

I think this is ā€˜okayā€™ because the closure is formed on the main actor and is not marked @Sendable so it Should :tm: be impossible for the formed function reference to be handed off to a different isolation context and executed there.

That thatā€™s happening seems like a bug internal to SwiftUI to me but I donā€™t think itā€™s a problem in the client code, unless Iā€™m missing something.

the promise of modern concurrency is that the compiler guarantees correctness (at least at some level)

this shouldn't depend on whether the framework you're passing to is buggy...

Having said that - there is no SwiftUI bug here. SwiftUI makes no promises - so we can't blame it for not keeping them!

public init(get: @escaping () -> Value, set: @escaping (Value) -> Void)

the compiler doesn't know anything about where this closure is going to be run.
(The Binding API gives no guarantees) so, it makes zero sense that the compiler is assessing correctness on the basis of the enclosing function's isolation.

1 Like

Function values default to non-Sendable, so this signature implicitly promises that get and set won't cross an isolation domain. If Binding allows that to happen internally, then either the interface is misrepresented (and these values should be @Sendable) or there's an implementation bug that violates the concurrency rules.

This can be reproduced locally fairly easily. On its own, the following is fine:

func f(_ h: @escaping () -> Void) {}

actor A {
    var x: Int = 0
    func g() {
        f {
            self.x += 1
        }
    }
}

but if the body of f is changed to:

    Task.detached {
        h()
    }

then we get a warning that a non-Sendable closure is being captured by a Sendable closure. If we mark h as @Sendable, then we do get an error at the call site of f:

test.swift:12:18: error: actor-isolated property 'x' can not be mutated from a Sendable closure
            self.x += 1
                 ^
test.swift:9:9: note: mutation of this property is only permitted within the actor
    var x: Int = 0
        ^
1 Like

This is going beyond my understanding.

Are you saying that if I write a library that has a callback (just a regular escaping function) - then somehow the library code needs to know what isolation domain the callback was created in - and ensure to only call it in that domain?

That seems kinda weird if so. How would my library even know what isolation domain the callback was created in? I just saved it to a variable, then called it at a later point...

Well, yes and no. It doesn't need to know which specific isolation domain the callback was created in, but if it's a non-sendable function then the callee does know that the function was created in the current isolation domain (because non-sendable functions can't cross isolation domains). So what the callee is responsible for ensuring is that the function doesn't leave the current isolation domain, which should be covered by sendability checking. Moving a non-sendable function (or any non-sendable value) to another isolation domain is invalid.

The problem here is not with the Binding ā€“ which isnā€™t Sendable and should not be ā€“ but with Task.detached { val = false }, which improperly captures the Binding in a non-@Sendable closure, and this is correctly diagnosed with full strict concurrency checking. I personally think this should be an error rather than a warning, but :man_shrugging:t2:

In the end I traced this crash to something related to the Secure Enclave ECC encryption that Iā€™m working with, but then it was inadvertently resolved by another fix that I implemented, and that other fix may well have had an effect on the interweaving of background and @MainActor methods. I donā€™t know to what extent Secure Enclave encryption may involve the Objective-C-related things that expose one to these @MainActor foot-guns, but I just wanted to post this here in case the information Iā€™ve provided sparks any helpful feedback in response, because inadvertently fixing a completely cryptic crash that reproduces 100% of the time on my bossā€™ TestFlight instance and 0% of the time on my devices is unsettlingā€¦