bind_test.go 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404
  1. // Copyright 2015 The Go Authors. All rights reserved.
  2. // Use of this source code is governed by a BSD-style
  3. // license that can be found in the LICENSE file.
  4. package main
  5. import (
  6. "bytes"
  7. "os"
  8. "os/exec"
  9. "path/filepath"
  10. "runtime"
  11. "strings"
  12. "testing"
  13. "text/template"
  14. "golang.org/x/mobile/internal/sdkpath"
  15. )
  16. func TestBindAndroid(t *testing.T) {
  17. platform, err := sdkpath.AndroidAPIPath(minAndroidAPI)
  18. if err != nil {
  19. t.Skip("No compatible Android API platform found, skipping bind")
  20. }
  21. // platform is a path like "/path/to/Android/sdk/platforms/android-32"
  22. components := strings.Split(platform, string(filepath.Separator))
  23. if len(components) < 2 {
  24. t.Fatalf("API path is too short: %s", platform)
  25. }
  26. components = components[len(components)-2:]
  27. platformRel := filepath.Join("$ANDROID_HOME", components[0], components[1])
  28. defer func() {
  29. xout = os.Stderr
  30. buildN = false
  31. buildX = false
  32. buildO = ""
  33. buildTarget = ""
  34. bindJavaPkg = ""
  35. }()
  36. buildN = true
  37. buildX = true
  38. buildO = "asset.aar"
  39. buildTarget = "android/arm"
  40. tests := []struct {
  41. javaPkg string
  42. }{
  43. {
  44. // Empty javaPkg
  45. },
  46. {
  47. javaPkg: "com.example.foo",
  48. },
  49. }
  50. for _, tc := range tests {
  51. bindJavaPkg = tc.javaPkg
  52. buf := new(bytes.Buffer)
  53. xout = buf
  54. gopath = filepath.SplitList(goEnv("GOPATH"))[0]
  55. if goos == "windows" {
  56. os.Setenv("HOMEDRIVE", "C:")
  57. }
  58. cmdBind.flag.Parse([]string{"golang.org/x/mobile/asset"})
  59. err := runBind(cmdBind)
  60. if err != nil {
  61. t.Log(buf.String())
  62. t.Fatal(err)
  63. }
  64. got := filepath.ToSlash(buf.String())
  65. output, err := defaultOutputData("")
  66. if err != nil {
  67. t.Fatal(err)
  68. }
  69. data := struct {
  70. outputData
  71. AndroidPlatform string
  72. JavaPkg string
  73. }{
  74. outputData: output,
  75. AndroidPlatform: platformRel,
  76. JavaPkg: tc.javaPkg,
  77. }
  78. wantBuf := new(bytes.Buffer)
  79. if err := bindAndroidTmpl.Execute(wantBuf, data); err != nil {
  80. t.Errorf("%+v: computing diff failed: %v", tc, err)
  81. continue
  82. }
  83. diff, err := diff(got, wantBuf.String())
  84. if err != nil {
  85. t.Errorf("%+v: computing diff failed: %v", tc, err)
  86. continue
  87. }
  88. if diff != "" {
  89. t.Errorf("%+v: unexpected output:\n%s", tc, diff)
  90. }
  91. }
  92. }
  93. func TestBindApple(t *testing.T) {
  94. if !xcodeAvailable() {
  95. t.Skip("Xcode is missing")
  96. }
  97. defer func() {
  98. xout = os.Stderr
  99. buildN = false
  100. buildX = false
  101. buildO = ""
  102. buildTarget = ""
  103. bindPrefix = ""
  104. }()
  105. buildN = true
  106. buildX = true
  107. buildO = "Asset.xcframework"
  108. buildTarget = "ios/arm64"
  109. tests := []struct {
  110. prefix string
  111. out string
  112. }{
  113. {
  114. // empty prefix
  115. },
  116. {
  117. prefix: "Foo",
  118. },
  119. {
  120. out: "Abcde.xcframework",
  121. },
  122. }
  123. for _, tc := range tests {
  124. bindPrefix = tc.prefix
  125. if tc.out != "" {
  126. buildO = tc.out
  127. }
  128. buf := new(bytes.Buffer)
  129. xout = buf
  130. gopath = filepath.SplitList(goEnv("GOPATH"))[0]
  131. if goos == "windows" {
  132. os.Setenv("HOMEDRIVE", "C:")
  133. }
  134. cmdBind.flag.Parse([]string{"golang.org/x/mobile/asset"})
  135. if err := runBind(cmdBind); err != nil {
  136. t.Log(buf.String())
  137. t.Fatal(err)
  138. }
  139. got := filepath.ToSlash(buf.String())
  140. output, err := defaultOutputData("")
  141. if err != nil {
  142. t.Fatal(err)
  143. }
  144. data := struct {
  145. outputData
  146. Output string
  147. Prefix string
  148. }{
  149. outputData: output,
  150. Output: buildO[:len(buildO)-len(".xcframework")],
  151. Prefix: tc.prefix,
  152. }
  153. wantBuf := new(bytes.Buffer)
  154. if err := bindAppleTmpl.Execute(wantBuf, data); err != nil {
  155. t.Errorf("%+v: computing diff failed: %v", tc, err)
  156. continue
  157. }
  158. diff, err := diff(got, wantBuf.String())
  159. if err != nil {
  160. t.Errorf("%+v: computing diff failed: %v", tc, err)
  161. continue
  162. }
  163. if diff != "" {
  164. t.Errorf("%+v: unexpected output:\n%s", tc, diff)
  165. }
  166. }
  167. }
  168. var bindAndroidTmpl = template.Must(template.New("output").Parse(`GOMOBILE={{.GOPATH}}/pkg/gomobile
  169. WORK=$WORK
  170. GOOS=android CGO_ENABLED=1 gobind -lang=go,java -outdir=$WORK{{if .JavaPkg}} -javapkg={{.JavaPkg}}{{end}} golang.org/x/mobile/asset
  171. mkdir -p $WORK/src-android-arm
  172. PWD=$WORK/src-android-arm GOMODCACHE=$GOPATH/pkg/mod GOOS=android GOARCH=arm CC=$NDK_PATH/toolchains/llvm/prebuilt/{{.NDKARCH}}/bin/armv7a-linux-androideabi16-clang CXX=$NDK_PATH/toolchains/llvm/prebuilt/{{.NDKARCH}}/bin/armv7a-linux-androideabi16-clang++ CGO_ENABLED=1 GOARM=7 GOPATH=$WORK:$GOPATH go mod tidy
  173. PWD=$WORK/src-android-arm GOMODCACHE=$GOPATH/pkg/mod GOOS=android GOARCH=arm CC=$NDK_PATH/toolchains/llvm/prebuilt/{{.NDKARCH}}/bin/armv7a-linux-androideabi16-clang CXX=$NDK_PATH/toolchains/llvm/prebuilt/{{.NDKARCH}}/bin/armv7a-linux-androideabi16-clang++ CGO_ENABLED=1 GOARM=7 GOPATH=$WORK:$GOPATH go build -x -buildmode=c-shared -o=$WORK/android/src/main/jniLibs/armeabi-v7a/libgojni.so ./gobind
  174. PWD=$WORK/java javac -d $WORK/javac-output -source 1.8 -target 1.8 -bootclasspath {{.AndroidPlatform}}/android.jar *.java
  175. jar c -C $WORK/javac-output .
  176. `))
  177. var bindAppleTmpl = template.Must(template.New("output").Parse(`GOMOBILE={{.GOPATH}}/pkg/gomobile
  178. WORK=$WORK
  179. rm -r -f "{{.Output}}.xcframework"
  180. GOOS=ios CGO_ENABLED=1 gobind -lang=go,objc -outdir=$WORK/ios -tags=ios{{if .Prefix}} -prefix={{.Prefix}}{{end}} golang.org/x/mobile/asset
  181. mkdir -p $WORK/ios/src-arm64
  182. PWD=$WORK/ios/src-arm64 GOMODCACHE=$GOPATH/pkg/mod GOOS=ios GOARCH=arm64 GOFLAGS=-tags=ios CC=iphoneos-clang CXX=iphoneos-clang++ CGO_CFLAGS=-isysroot iphoneos -miphoneos-version-min=13.0 -fembed-bitcode -arch arm64 CGO_CXXFLAGS=-isysroot iphoneos -miphoneos-version-min=13.0 -fembed-bitcode -arch arm64 CGO_LDFLAGS=-isysroot iphoneos -miphoneos-version-min=13.0 -fembed-bitcode -arch arm64 CGO_ENABLED=1 DARWIN_SDK=iphoneos GOPATH=$WORK/ios:$GOPATH go mod tidy
  183. PWD=$WORK/ios/src-arm64 GOMODCACHE=$GOPATH/pkg/mod GOOS=ios GOARCH=arm64 GOFLAGS=-tags=ios CC=iphoneos-clang CXX=iphoneos-clang++ CGO_CFLAGS=-isysroot iphoneos -miphoneos-version-min=13.0 -fembed-bitcode -arch arm64 CGO_CXXFLAGS=-isysroot iphoneos -miphoneos-version-min=13.0 -fembed-bitcode -arch arm64 CGO_LDFLAGS=-isysroot iphoneos -miphoneos-version-min=13.0 -fembed-bitcode -arch arm64 CGO_ENABLED=1 DARWIN_SDK=iphoneos GOPATH=$WORK/ios:$GOPATH go build -x -buildmode=c-archive -o $WORK/{{.Output}}-ios-arm64.a ./gobind
  184. mkdir -p $WORK/ios/iphoneos/{{.Output}}.framework/Versions/A/Headers
  185. ln -s A $WORK/ios/iphoneos/{{.Output}}.framework/Versions/Current
  186. ln -s Versions/Current/Headers $WORK/ios/iphoneos/{{.Output}}.framework/Headers
  187. ln -s Versions/Current/{{.Output}} $WORK/ios/iphoneos/{{.Output}}.framework/{{.Output}}
  188. xcrun lipo $WORK/{{.Output}}-ios-arm64.a -create -o $WORK/ios/iphoneos/{{.Output}}.framework/Versions/A/{{.Output}}
  189. cp $WORK/ios/src/gobind/{{.Prefix}}Asset.objc.h $WORK/ios/iphoneos/{{.Output}}.framework/Versions/A/Headers/{{.Prefix}}Asset.objc.h
  190. mkdir -p $WORK/ios/iphoneos/{{.Output}}.framework/Versions/A/Headers
  191. cp $WORK/ios/src/gobind/Universe.objc.h $WORK/ios/iphoneos/{{.Output}}.framework/Versions/A/Headers/Universe.objc.h
  192. mkdir -p $WORK/ios/iphoneos/{{.Output}}.framework/Versions/A/Headers
  193. cp $WORK/ios/src/gobind/ref.h $WORK/ios/iphoneos/{{.Output}}.framework/Versions/A/Headers/ref.h
  194. mkdir -p $WORK/ios/iphoneos/{{.Output}}.framework/Versions/A/Headers
  195. mkdir -p $WORK/ios/iphoneos/{{.Output}}.framework/Versions/A/Headers
  196. mkdir -p $WORK/ios/iphoneos/{{.Output}}.framework/Versions/A/Resources
  197. ln -s Versions/Current/Resources $WORK/ios/iphoneos/{{.Output}}.framework/Resources
  198. mkdir -p $WORK/ios/iphoneos/{{.Output}}.framework/Resources
  199. mkdir -p $WORK/ios/iphoneos/{{.Output}}.framework/Versions/A/Modules
  200. ln -s Versions/Current/Modules $WORK/ios/iphoneos/{{.Output}}.framework/Modules
  201. xcodebuild -create-xcframework -framework $WORK/ios/iphoneos/{{.Output}}.framework -output {{.Output}}.xcframework
  202. `))
  203. func TestBindAppleAll(t *testing.T) {
  204. if !xcodeAvailable() {
  205. t.Skip("Xcode is missing")
  206. }
  207. defer func() {
  208. xout = os.Stderr
  209. buildN = false
  210. buildX = false
  211. buildO = ""
  212. buildTarget = ""
  213. bindPrefix = ""
  214. }()
  215. buildN = true
  216. buildX = true
  217. buildO = "Asset.xcframework"
  218. buildTarget = "ios"
  219. buf := new(bytes.Buffer)
  220. xout = buf
  221. gopath = filepath.SplitList(goEnv("GOPATH"))[0]
  222. if goos == "windows" {
  223. os.Setenv("HOMEDRIVE", "C:")
  224. }
  225. cmdBind.flag.Parse([]string{"golang.org/x/mobile/asset"})
  226. if err := runBind(cmdBind); err != nil {
  227. t.Log(buf.String())
  228. t.Fatal(err)
  229. }
  230. }
  231. const ambiguousPathsGoMod = `module ambiguouspaths
  232. go 1.18
  233. require golang.org/x/mobile v0.0.0-20230905140555-fbe1c053b6a9
  234. require (
  235. golang.org/x/exp/shiny v0.0.0-20230817173708-d852ddb80c63 // indirect
  236. golang.org/x/image v0.11.0 // indirect
  237. golang.org/x/sys v0.11.0 // indirect
  238. )
  239. `
  240. const ambiguousPathsGoSum = `github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY=
  241. golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
  242. golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc=
  243. golang.org/x/exp/shiny v0.0.0-20230817173708-d852ddb80c63 h1:3AGKexOYqL+ztdWdkB1bDwXgPBuTS/S8A4WzuTvJ8Cg=
  244. golang.org/x/exp/shiny v0.0.0-20230817173708-d852ddb80c63/go.mod h1:UH99kUObWAZkDnWqppdQe5ZhPYESUw8I0zVV1uWBR+0=
  245. golang.org/x/image v0.11.0 h1:ds2RoQvBvYTiJkwpSFDwCcDFNX7DqjL2WsUgTNk0Ooo=
  246. golang.org/x/image v0.11.0/go.mod h1:bglhjqbqVuEb9e9+eNR45Jfu7D+T4Qan+NhQk8Ck2P8=
  247. golang.org/x/mobile v0.0.0-20230905140555-fbe1c053b6a9 h1:LaLfQUz4L1tfuOlrtEouZLZ0qHDwKn87E1NKoiudP/o=
  248. golang.org/x/mobile v0.0.0-20230905140555-fbe1c053b6a9/go.mod h1:2jxcxt/JNJik+N+QcB8q308+SyrE3bu43+sGZDmJ02M=
  249. golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4=
  250. golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs=
  251. golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
  252. golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg=
  253. golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c=
  254. golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs=
  255. golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
  256. golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
  257. golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
  258. golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
  259. golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
  260. golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
  261. golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
  262. golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
  263. golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
  264. golang.org/x/sys v0.11.0 h1:eG7RXZHdqOJ1i+0lgLgCpSXAp6M3LYlAo6osgSi0xOM=
  265. golang.org/x/sys v0.11.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
  266. golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
  267. golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8=
  268. golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k=
  269. golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
  270. golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
  271. golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ=
  272. golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8=
  273. golang.org/x/text v0.12.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE=
  274. golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
  275. golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
  276. golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc=
  277. golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU=
  278. golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
  279. `
  280. const ambiguousPathsGo = `package ambiguouspaths
  281. import (
  282. _ "golang.org/x/mobile/app"
  283. )
  284. func Dummy() {}
  285. `
  286. func TestBindWithGoModules(t *testing.T) {
  287. if runtime.GOOS == "android" || runtime.GOOS == "ios" {
  288. t.Skipf("gomobile and gobind are not available on %s", runtime.GOOS)
  289. }
  290. dir := t.TempDir()
  291. if out, err := exec.Command("go", "build", "-o="+dir, "golang.org/x/mobile/cmd/gobind").CombinedOutput(); err != nil {
  292. t.Fatalf("%v: %s", err, string(out))
  293. }
  294. if out, err := exec.Command("go", "build", "-o="+dir, "golang.org/x/mobile/cmd/gomobile").CombinedOutput(); err != nil {
  295. t.Fatalf("%v: %s", err, string(out))
  296. }
  297. path := dir
  298. if p := os.Getenv("PATH"); p != "" {
  299. path += string(filepath.ListSeparator) + p
  300. }
  301. // Create a source package dynamically to avoid go.mod files in this repository. See golang/go#34352 for more details.
  302. if err := os.Mkdir(filepath.Join(dir, "ambiguouspaths"), 0755); err != nil {
  303. t.Fatal(err)
  304. }
  305. if err := os.WriteFile(filepath.Join(dir, "ambiguouspaths", "go.mod"), []byte(ambiguousPathsGoMod), 0644); err != nil {
  306. t.Fatal(err)
  307. }
  308. if err := os.WriteFile(filepath.Join(dir, "ambiguouspaths", "go.sum"), []byte(ambiguousPathsGoSum), 0644); err != nil {
  309. t.Fatal(err)
  310. }
  311. if err := os.WriteFile(filepath.Join(dir, "ambiguouspaths", "ambiguouspaths.go"), []byte(ambiguousPathsGo), 0644); err != nil {
  312. t.Fatal(err)
  313. }
  314. for _, target := range []string{"android", "ios"} {
  315. target := target
  316. t.Run(target, func(t *testing.T) {
  317. switch target {
  318. case "android":
  319. if _, err := sdkpath.AndroidAPIPath(minAndroidAPI); err != nil {
  320. t.Skip("No compatible Android API platform found, skipping bind")
  321. }
  322. case "ios":
  323. if !xcodeAvailable() {
  324. t.Skip("Xcode is missing")
  325. }
  326. }
  327. var out string
  328. switch target {
  329. case "android":
  330. out = filepath.Join(dir, "cgopkg.aar")
  331. case "ios":
  332. out = filepath.Join(dir, "Cgopkg.xcframework")
  333. }
  334. tests := []struct {
  335. Name string
  336. Path string
  337. Dir string
  338. }{
  339. {
  340. Name: "Absolute Path",
  341. Path: "golang.org/x/mobile/bind/testdata/cgopkg",
  342. },
  343. {
  344. Name: "Relative Path",
  345. Path: "./bind/testdata/cgopkg",
  346. Dir: filepath.Join("..", ".."),
  347. },
  348. {
  349. Name: "Ambiguous Paths",
  350. Path: ".",
  351. Dir: filepath.Join(dir, "ambiguouspaths"),
  352. },
  353. }
  354. for _, tc := range tests {
  355. tc := tc
  356. t.Run(tc.Name, func(t *testing.T) {
  357. cmd := exec.Command(filepath.Join(dir, "gomobile"), "bind", "-target="+target, "-o="+out, tc.Path)
  358. cmd.Env = append(os.Environ(), "PATH="+path, "GO111MODULE=on")
  359. cmd.Dir = tc.Dir
  360. if out, err := cmd.CombinedOutput(); err != nil {
  361. t.Errorf("gomobile bind failed: %v\n%s", err, string(out))
  362. }
  363. })
  364. }
  365. })
  366. }
  367. }