2015-02-11 15:23:09 -05:00
// Copyright 2015 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
package main
import (
cmd/gomobile: replace stripped NDK with external NDK
Gomobile has up until now used stripped NDKs hosted by Google. This
arrangement adds maintenance overhead and blocks the use of custom
NDKs or custom API levels. Also, as noted in issue 16211, the stripped
NDK is no longer tiny because Gomobile supports more platforms.
This CL removed the code for generating and packaging stripped NDKs and
adds support for using external NDKs to the gomobile tool.
gomobile init will now use the NDK installed by the Android SDK manager,
if present, or a user specified NDK if the -ndk flag is given. If no
NDK was found or specified, Android initialization is skipped. gomobile
will instruct the user to run init with a valid NDK if bind or build is
invoked without Android initialization.
gomobile init will also attempt to build OpenAL for Android if the -openal
flag specifies a source directory. It needs cmake and, on Windows, nmake
installed. If gomobile build is run on an app that requires
golang.org/x/mobile/exp/audio/al and OpenAL wasn't built by init, the user
is instructed to do so.
Tested on Linux, macOS, Windows.
Fixes golang/go#16211
Fixes golang/go#18522
Change-Id: Ia38f6e43e671a207dad562678c65225b426e7e3e
Reviewed-on: https://go-review.googlesource.com/35173
Reviewed-by: David Crawshaw <crawshaw@golang.org>
2017-01-13 00:59:09 +01:00
"errors"
2015-02-11 15:23:09 -05:00
"fmt"
"go/build"
"io"
"io/ioutil"
"os"
2015-11-20 11:26:00 -05:00
"path"
2015-02-11 15:23:09 -05:00
"path/filepath"
)
2015-07-10 16:47:46 -06:00
// ctx, pkg, tmpdir in build.go
2015-02-11 15:23:09 -05:00
var cmdBind = & command {
run : runBind ,
Name : "bind" ,
2016-09-30 12:39:25 +02:00
Usage : "[-target android|ios] [-bootclasspath <path>] [-classpath <path>] [-o output] [build flags] [package]" ,
2015-08-29 16:27:39 -07:00
Short : "build a library for Android and iOS" ,
2015-02-11 15:23:09 -05:00
Long : `
2015-06-19 12:27:15 -04:00
Bind generates language bindings for the package named by the import
path , and compiles a library for the named target system .
The - target flag takes a target system name , either android ( the
default ) or ios .
For - target android , the bind command produces an AAR ( Android ARchive )
file that archives the precompiled Java API stub classes , the compiled
shared libraries , and all asset files in the / assets subdirectory under
the package directory . The output is named ' < package_name > . aar ' by
default . This AAR file is commonly used for binary distribution of an
Android library project and most Android IDEs support AAR import . For
example , in Android Studio ( 1.2 + ) , an AAR file can be imported using
the module import wizard ( File > New > New Module > Import . JAR or
. AAR package ) , and setting it as a new dependency
( File > Project Structure > Dependencies ) . This requires ' javac '
2016-07-07 11:36:51 -04:00
( version 1.7 + ) and Android SDK ( API level 15 or newer ) to build the
2015-06-19 12:27:15 -04:00
library for Android . The environment variable ANDROID_HOME must be set
2018-03-08 00:15:06 +01:00
to the path to Android SDK . Use the - javapkg flag to specify the Java
package prefix for the generated classes .
2015-06-19 12:27:15 -04:00
2016-06-14 10:43:25 -04:00
By default , - target = android builds shared libraries for all supported
instruction sets ( arm , arm64 , 386 , amd64 ) . A subset of instruction sets
can be selected by specifying target type with the architecture name . E . g . ,
- target = android / arm , android / 386.
2015-06-19 12:27:15 -04:00
For - target ios , gomobile must be run on an OS X machine with Xcode
2018-03-08 00:15:06 +01:00
installed . The generated Objective - C types can be prefixed with the - prefix
flag .
2015-02-11 15:23:09 -05:00
2016-09-30 12:39:25 +02:00
For - target android , the - bootclasspath and - classpath flags are used to
control the bootstrap classpath and the classpath for Go wrappers to Java
classes .
2015-02-11 15:23:09 -05:00
The - v flag provides verbose output , including the list of packages built .
2015-11-30 14:01:47 -05:00
The build flags - a , - n , - x , - gcflags , - ldflags , - tags , and - work
2015-07-27 16:33:08 -04:00
are shared with the build command . For documentation , see ' go help build ' .
2015-02-11 15:23:09 -05:00
` ,
}
func runBind ( cmd * command ) error {
2015-07-16 13:32:51 -04:00
cleanup , err := buildEnvInit ( )
2015-02-11 15:23:09 -05:00
if err != nil {
2015-07-10 16:47:46 -06:00
return err
2015-02-11 15:23:09 -05:00
}
2015-07-10 16:47:46 -06:00
defer cleanup ( )
2015-02-11 15:23:09 -05:00
args := cmd . flag . Args ( )
2015-12-11 18:36:28 -05:00
targetOS , targetArchs , err := parseBuildTarget ( buildTarget )
if err != nil {
return fmt . Errorf ( ` invalid -target=%q: %v ` , buildTarget , err )
2015-07-24 07:04:40 -04:00
}
2015-12-11 18:36:28 -05:00
ctx . GOARCH = "arm"
ctx . GOOS = targetOS
2017-01-08 18:10:11 +01:00
if ctx . GOOS == "darwin" {
ctx . BuildTags = append ( ctx . BuildTags , "ios" )
}
2015-08-28 13:39:30 -04:00
if bindJavaPkg != "" && ctx . GOOS != "android" {
return fmt . Errorf ( "-javapkg is supported only for android target" )
}
2016-12-23 11:18:18 +01:00
if bindPrefix != "" && ctx . GOOS != "darwin" {
2015-08-28 13:39:30 -04:00
return fmt . Errorf ( "-prefix is supported only for ios target" )
}
2018-03-15 17:30:30 +01:00
if ctx . GOOS == "android" && ! hasNDK ( ) {
cmd/gomobile: replace stripped NDK with external NDK
Gomobile has up until now used stripped NDKs hosted by Google. This
arrangement adds maintenance overhead and blocks the use of custom
NDKs or custom API levels. Also, as noted in issue 16211, the stripped
NDK is no longer tiny because Gomobile supports more platforms.
This CL removed the code for generating and packaging stripped NDKs and
adds support for using external NDKs to the gomobile tool.
gomobile init will now use the NDK installed by the Android SDK manager,
if present, or a user specified NDK if the -ndk flag is given. If no
NDK was found or specified, Android initialization is skipped. gomobile
will instruct the user to run init with a valid NDK if bind or build is
invoked without Android initialization.
gomobile init will also attempt to build OpenAL for Android if the -openal
flag specifies a source directory. It needs cmake and, on Windows, nmake
installed. If gomobile build is run on an app that requires
golang.org/x/mobile/exp/audio/al and OpenAL wasn't built by init, the user
is instructed to do so.
Tested on Linux, macOS, Windows.
Fixes golang/go#16211
Fixes golang/go#18522
Change-Id: Ia38f6e43e671a207dad562678c65225b426e7e3e
Reviewed-on: https://go-review.googlesource.com/35173
Reviewed-by: David Crawshaw <crawshaw@golang.org>
2017-01-13 00:59:09 +01:00
return errors . New ( "no Android NDK path is set. Please run gomobile init with the ndk-bundle installed through the Android SDK manager or with the -ndk flag set." )
}
2015-11-19 10:51:29 +05:30
var pkgs [ ] * build . Package
2015-02-11 15:23:09 -05:00
switch len ( args ) {
case 0 :
2015-11-19 10:51:29 +05:30
pkgs = make ( [ ] * build . Package , 1 )
pkgs [ 0 ] , err = ctx . ImportDir ( cwd , build . ImportComment )
2015-02-11 15:23:09 -05:00
default :
2015-11-19 10:51:29 +05:30
pkgs , err = importPackages ( args )
2015-02-11 15:23:09 -05:00
}
if err != nil {
return err
}
2016-05-05 09:22:14 -07:00
// check if any of the package is main
for _ , pkg := range pkgs {
if pkg . Name == "main" {
return fmt . Errorf ( "binding 'main' package (%s) is not supported" , pkg . ImportComment )
}
}
2015-12-18 12:04:02 -05:00
switch targetOS {
2015-06-19 12:27:15 -04:00
case "android" :
2015-12-11 18:36:28 -05:00
return goAndroidBind ( pkgs , targetArchs )
2015-12-18 12:04:02 -05:00
case "darwin" :
2015-12-11 18:36:28 -05:00
// TODO: use targetArchs?
2015-11-19 10:51:29 +05:30
return goIOSBind ( pkgs )
2015-07-24 16:02:39 -04:00
default :
2015-12-11 18:36:28 -05:00
return fmt . Errorf ( ` invalid -target=%q ` , buildTarget )
2015-06-19 12:27:15 -04:00
}
2015-02-11 15:23:09 -05:00
}
2015-11-19 10:51:29 +05:30
func importPackages ( args [ ] string ) ( [ ] * build . Package , error ) {
pkgs := make ( [ ] * build . Package , len ( args ) )
2017-02-21 18:48:50 +01:00
for i , a := range args {
a = path . Clean ( a )
2015-11-19 10:51:29 +05:30
var err error
2017-02-21 18:48:50 +01:00
if pkgs [ i ] , err = ctx . Import ( a , cwd , build . ImportComment ) ; err != nil {
return nil , fmt . Errorf ( "package %q: %v" , a , err )
2015-11-19 10:51:29 +05:30
}
}
return pkgs , nil
}
2015-08-28 13:39:30 -04:00
var (
2016-09-30 12:39:25 +02:00
bindPrefix string // -prefix
bindJavaPkg string // -javapkg
bindClasspath string // -classpath
bindBootClasspath string // -bootclasspath
2015-08-28 13:39:30 -04:00
)
func init ( ) {
// bind command specific commands.
cmdBind . flag . StringVar ( & bindJavaPkg , "javapkg" , "" ,
2017-01-01 22:43:46 +01:00
"specifies custom Java package path prefix. Valid only with -target=android." )
2016-12-23 11:18:18 +01:00
cmdBind . flag . StringVar ( & bindPrefix , "prefix" , "" ,
2017-01-01 22:43:46 +01:00
"custom Objective-C name prefix. Valid only with -target=ios." )
2016-09-30 12:39:25 +02:00
cmdBind . flag . StringVar ( & bindClasspath , "classpath" , "" , "The classpath for imported Java classes. Valid only with -target=android." )
cmdBind . flag . StringVar ( & bindBootClasspath , "bootclasspath" , "" , "The bootstrap classpath for imported Java classes. Valid only with -target=android." )
2015-08-28 13:39:30 -04:00
}
2016-09-30 12:39:25 +02:00
func bootClasspath ( ) ( string , error ) {
if bindBootClasspath != "" {
return bindBootClasspath , nil
}
2016-09-07 18:27:36 +02:00
apiPath , err := androidAPIPath ( )
if err != nil {
2016-09-30 12:39:25 +02:00
return "" , err
2016-09-07 18:27:36 +02:00
}
2016-09-30 12:39:25 +02:00
return filepath . Join ( apiPath , "android.jar" ) , nil
}
2015-07-14 18:09:25 -04:00
func copyFile ( dst , src string ) error {
if buildX {
printcmd ( "cp %s %s" , src , dst )
}
return writeFile ( dst , func ( w io . Writer ) error {
2015-07-19 21:29:35 -04:00
if buildN {
return nil
}
f , err := os . Open ( src )
if err != nil {
return err
}
defer f . Close ( )
if _ , err := io . Copy ( w , f ) ; err != nil {
return fmt . Errorf ( "cp %s %s failed: %v" , src , dst , err )
}
return nil
2015-07-14 18:09:25 -04:00
} )
}
2015-02-11 15:23:09 -05:00
func writeFile ( filename string , generate func ( io . Writer ) error ) error {
if buildV {
fmt . Fprintf ( os . Stderr , "write %s\n" , filename )
}
err := mkdir ( filepath . Dir ( filename ) )
if err != nil {
return err
}
if buildN {
return generate ( ioutil . Discard )
}
f , err := os . Create ( filename )
if err != nil {
return err
}
defer func ( ) {
if cerr := f . Close ( ) ; err == nil {
err = cerr
}
} ( )
return generate ( f )
}