-
Notifications
You must be signed in to change notification settings - Fork 17.9k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
x/mobile: Issues converting Data to []byte in Swift #33745
Comments
CC @hyangah @eliasnaur @steeve My guess (but it's just a guess) would be that the In other words: I suspect a classic use-after-free bug, obscured by the use of two garbage collectors. The usual fix for such a bug is to extend the aliased object for the lifetime of the alias. (In a reference-counted language, that would imply pinning the Or, you can copy passed-in slices rather than retaining references. |
Indeed, a NSData is copied before being passed to Go, and then freed, according to https://github.com/golang/mobile/blob/e8b3e6111d02c5a0a25ea00257319a0c01c12806/bind/objc/seq_darwin.m.support#L191 |
Strangely enough, the following code prints the correct results in all 5 cases: var data = Data(repeating: 0, count: 8)
let mutableData = NSMutableData(bytes: &data, length: 8)
print("Data: \(Array(UnsafeBufferPointer(start: mutableData.bytes.assumingMemoryBound(to: UInt8.self), count: 8)))")
let bytes = GocodeNewBytes(mutableData as Data)!
print("Elements: \([UInt8](bytes.getElements()!))")
print("Data: \([UInt8](data))") This is interesting since now |
After some more investigation I am now pretty sure what the actual problematic behavior is. These are the steps happening in my small example from the issue description:
It is the same like doing the following in Go: p := C.malloc(4)
arr := (*[1 << 30]byte)(p)[:4:4]
// Put some values into 'arr'.
b := NewBytes(arr)
// Values of 'b.elements' are okay.
C.free(p)
// Now 'b.elements' could be anything. In the cgo wiki it is explicitly stated that ...
So from my understanding, this nondeterminism is exactly the root cause of this issue. The following solutions come into my mind:
|
Good one ! |
From what I'm reading online:
I'm not able to confirm this yet, however.... It makes me wonder, though, wether it's gomobile's job to ensure one does not mutable a |
On Sun Sep 8, 2019 at 6:37 AM Steeve Morin wrote:
From what I'm reading online:
```
let d = Data() // should be NSData
var d2 = Data() // should be NSMutableData
```
I'm not able to confirm this yet, however....
It makes me wonder, though, wether it's gomobile's job to ensure one does not mutable a `[]byte` passed from ObjC/Swift code. Since the underlying data structure is never backed by Go, either it is copied into Go's memory, or not copied/freed at all. @eliasnaur, is there a reason not to do that ?
The intention behind not copying []byte is to enable io.Reader-like interfaces:
func ReadData(d []byte) error {
// Fill in d.
d[...] = ...
}
Would then be callable with a native byte array. In Java:
byte[] d = new byte[100];
go.Pkg.readData(d);
// Use d
...
Additionally, avoiding the copy leads to performance improvements for very
large byte arrays.
See also #12113.
Now, non-copying byte slices that are only valid during the cross language call
are problematic for several reasons:
- The static lifetime leads to tricky bugs when a byte slice is retained for longer
than the underlying data lives. It seems to me that's what this issue is about.
- Confusingly, returning a byte slice from a function will result in a copy, because
gomobile doesn't have anywhere to tell the other side to release the byte slice data.
In retrospect, the current design is probably a mistake.
Perhaps all byte slices should be copied, and a special type, say
golang.org/x/mobile/bind.ByteSlice, added that allows for zero-copy transfer of
data across the language barrier. ByteSlice can use the usual gomobile
reference counting and will not release the underlying byte array or slice
memory until its finalizer runs.
In other words, ByteSlice will provide dynamic and correct lifetimes of zero-copy
memory, replacing the Cgo-like static lifetimes.
@steeve, did that answer your question?
|
…leData This process is necessary because of an issue (golang/go#33745) in gomobile. Passing bare Data objects to Go functions leads to nondeterministic behavior.
@eliasnaur it does! Actually I was wondering why gobind would copy at all. |
…leData This process is necessary because of an issue (golang/go#33745) in gomobile. Passing bare Data objects to Go functions leads to nondeterministic behavior.
…leData This process is necessary because of an issue (golang/go#33745) in gomobile. Passing bare Data objects to Go functions leads to nondeterministic behavior.
…leData This process is necessary because of an issue (golang/go#33745) in gomobile. Passing bare Data objects to Go functions leads to nondeterministic behavior.
Is it possible to read and write to a |
This is exactly the thing I'm failing completely to do with
... and then call it from Swift:
... and the data change in Go is never reflected back in Swift. It doesn't matter if I explicitly create an What am I missing? This works fine with Java. |
…leData This process is necessary because of an issue (golang/go#33745) in gomobile. Passing bare Data objects to Go functions leads to nondeterministic behavior.
What version of Go are you using (
go version
)?Does this issue reproduce with the latest release?
Yes.
What operating system and processor architecture are you using (
go env
)?go env
OutputWhat did you do?
The Go source code
is compiled into an iOS framework using
$GOPATH/bin/gomobile bind -target ios -o Gocode.framework gocode
.In a minimal iOS application there is the Swift code
which can be compiled and run just fine. However, the printed values are not what I would expect.
What did you expect to see?
What did you see instead?
The data is correctly passed to the Go functions and back to Swift, but internally constructing the
Bytes
object it somehow changes. If I rewrite the constructor in Go asit works as anticipated. I'm not sure whether this is a bug or expected but unpleasant behavior due to Go's and Swift's handling of values and references. If the latter is true I really have no idea how to solve this issue on the Swift side in order to utilize Go libraries using
[]byte
at any interface function.The text was updated successfully, but these errors were encountered: