To use the NDK before version r19b standalone toolchains had to be generated. Version r19b added prebuilt standalone toolchains. Use the prebuilt for gomobile build and gomobile bind and stop generating toolchains during gomobile init. gomobile init is now only necessary for building OpenAL for gomobile build programs. This change is not compatible with NDK versions < r19b, but the user is instructed how to upgrade when running gomobile build or gomobile bind. Change-Id: I96953298ecce42402459a9dd15169c09fe6b6f8b Reviewed-on: https://go-review.googlesource.com/c/163378 Reviewed-by: Hyang-Ah Hana Kim <hyangah@gmail.com>
321 lines
7.3 KiB
Go
321 lines
7.3 KiB
Go
package main
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"runtime"
|
|
"strings"
|
|
)
|
|
|
|
// General mobile build environment. Initialized by envInit.
|
|
var (
|
|
cwd string
|
|
gomobilepath string // $GOPATH/pkg/gomobile
|
|
|
|
androidEnv map[string][]string // android arch -> []string
|
|
|
|
darwinEnv map[string][]string
|
|
|
|
androidArmNM string
|
|
darwinArmNM string
|
|
|
|
allArchs = []string{"arm", "arm64", "386", "amd64"}
|
|
)
|
|
|
|
func buildEnvInit() (cleanup func(), err error) {
|
|
// Find gomobilepath.
|
|
gopath := goEnv("GOPATH")
|
|
for _, p := range filepath.SplitList(gopath) {
|
|
gomobilepath = filepath.Join(p, "pkg", "gomobile")
|
|
if _, err := os.Stat(gomobilepath); buildN || err == nil {
|
|
break
|
|
}
|
|
}
|
|
|
|
if buildX {
|
|
fmt.Fprintln(xout, "GOMOBILE="+gomobilepath)
|
|
}
|
|
|
|
// Check the toolchain is in a good state.
|
|
// Pick a temporary directory for assembling an apk/app.
|
|
if gomobilepath == "" {
|
|
return nil, errors.New("toolchain not installed, run `gomobile init`")
|
|
}
|
|
|
|
if err := envInit(); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
cleanupFn := func() {
|
|
if buildWork {
|
|
fmt.Printf("WORK=%s\n", tmpdir)
|
|
return
|
|
}
|
|
removeAll(tmpdir)
|
|
}
|
|
if buildN {
|
|
tmpdir = "$WORK"
|
|
cleanupFn = func() {}
|
|
} else {
|
|
tmpdir, err = ioutil.TempDir("", "gomobile-work-")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
if buildX {
|
|
fmt.Fprintln(xout, "WORK="+tmpdir)
|
|
}
|
|
|
|
return cleanupFn, nil
|
|
}
|
|
|
|
func envInit() (err error) {
|
|
// TODO(crawshaw): cwd only used by ctx.Import, which can take "."
|
|
cwd, err = os.Getwd()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Setup the cross-compiler environments.
|
|
if ndkRoot, err := ndkRoot(); err == nil {
|
|
androidEnv = make(map[string][]string)
|
|
for arch, toolchain := range ndk {
|
|
androidEnv[arch] = []string{
|
|
"GOOS=android",
|
|
"GOARCH=" + arch,
|
|
"CC=" + toolchain.Path(ndkRoot, "clang"),
|
|
"CXX=" + toolchain.Path(ndkRoot, "clang++"),
|
|
"CGO_ENABLED=1",
|
|
}
|
|
if arch == "arm" {
|
|
androidEnv[arch] = append(androidEnv[arch], "GOARM=7")
|
|
}
|
|
}
|
|
}
|
|
|
|
if !xcodeAvailable() {
|
|
return nil
|
|
}
|
|
|
|
darwinArmNM = "nm"
|
|
darwinEnv = make(map[string][]string)
|
|
for _, arch := range allArchs {
|
|
var env []string
|
|
var err error
|
|
var clang, cflags string
|
|
switch arch {
|
|
case "arm":
|
|
env = append(env, "GOARM=7")
|
|
fallthrough
|
|
case "arm64":
|
|
clang, cflags, err = envClang("iphoneos")
|
|
cflags += " -miphoneos-version-min=" + buildIOSVersion
|
|
case "386", "amd64":
|
|
clang, cflags, err = envClang("iphonesimulator")
|
|
cflags += " -mios-simulator-version-min=" + buildIOSVersion
|
|
default:
|
|
panic(fmt.Errorf("unknown GOARCH: %q", arch))
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
env = append(env,
|
|
"GOOS=darwin",
|
|
"GOARCH="+arch,
|
|
"CC="+clang,
|
|
"CXX="+clang+"++",
|
|
"CGO_CFLAGS="+cflags+" -arch "+archClang(arch),
|
|
"CGO_CXXFLAGS="+cflags+" -arch "+archClang(arch),
|
|
"CGO_LDFLAGS="+cflags+" -arch "+archClang(arch),
|
|
"CGO_ENABLED=1",
|
|
)
|
|
darwinEnv[arch] = env
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func ndkRoot() (string, error) {
|
|
if buildN {
|
|
return "$NDK_PATH", nil
|
|
}
|
|
androidHome := os.Getenv("ANDROID_HOME")
|
|
if androidHome == "" {
|
|
return "", errors.New("The Android SDK was not found. Please set ANDROID_HOME to the root of the Android SDK.")
|
|
}
|
|
ndkRoot := filepath.Join(androidHome, "ndk-bundle")
|
|
_, err := os.Stat(ndkRoot)
|
|
if err != nil {
|
|
return "", fmt.Errorf("The NDK was not found in $ANDROID_HOME/ndk-bundle (%q). Install the NDK with `sdkmanager 'ndk-bundle'`", ndkRoot)
|
|
}
|
|
prebuiltPath := filepath.Join(androidHome, "ndk-bundle", "toolchains", "llvm", "prebuilt")
|
|
_, err = os.Stat(prebuiltPath)
|
|
if err != nil {
|
|
return "", fmt.Errorf("No prebuilt toolchains found in $ANDROID_HOME/ndk-bundle/toolchains/llvm/prebuilt (%q). Make sure your NDK version is >= r19b. Use `sdkmanager --update` to update it.", prebuiltPath)
|
|
}
|
|
return ndkRoot, nil
|
|
}
|
|
|
|
func envClang(sdkName string) (clang, cflags string, err error) {
|
|
if buildN {
|
|
return "clang-" + sdkName, "-isysroot=" + sdkName, nil
|
|
}
|
|
cmd := exec.Command("xcrun", "--sdk", sdkName, "--find", "clang")
|
|
out, err := cmd.CombinedOutput()
|
|
if err != nil {
|
|
return "", "", fmt.Errorf("xcrun --find: %v\n%s", err, out)
|
|
}
|
|
clang = strings.TrimSpace(string(out))
|
|
|
|
cmd = exec.Command("xcrun", "--sdk", sdkName, "--show-sdk-path")
|
|
out, err = cmd.CombinedOutput()
|
|
if err != nil {
|
|
return "", "", fmt.Errorf("xcrun --show-sdk-path: %v\n%s", err, out)
|
|
}
|
|
sdk := strings.TrimSpace(string(out))
|
|
return clang, "-isysroot " + sdk, nil
|
|
}
|
|
|
|
func archClang(goarch string) string {
|
|
switch goarch {
|
|
case "arm":
|
|
return "armv7"
|
|
case "arm64":
|
|
return "arm64"
|
|
case "386":
|
|
return "i386"
|
|
case "amd64":
|
|
return "x86_64"
|
|
default:
|
|
panic(fmt.Sprintf("unknown GOARCH: %q", goarch))
|
|
}
|
|
}
|
|
|
|
// environ merges os.Environ and the given "key=value" pairs.
|
|
// If a key is in both os.Environ and kv, kv takes precedence.
|
|
func environ(kv []string) []string {
|
|
cur := os.Environ()
|
|
new := make([]string, 0, len(cur)+len(kv))
|
|
|
|
envs := make(map[string]string, len(cur))
|
|
for _, ev := range cur {
|
|
elem := strings.SplitN(ev, "=", 2)
|
|
if len(elem) != 2 || elem[0] == "" {
|
|
// pass the env var of unusual form untouched.
|
|
// e.g. Windows may have env var names starting with "=".
|
|
new = append(new, ev)
|
|
continue
|
|
}
|
|
if goos == "windows" {
|
|
elem[0] = strings.ToUpper(elem[0])
|
|
}
|
|
envs[elem[0]] = elem[1]
|
|
}
|
|
for _, ev := range kv {
|
|
elem := strings.SplitN(ev, "=", 2)
|
|
if len(elem) != 2 || elem[0] == "" {
|
|
panic(fmt.Sprintf("malformed env var %q from input", ev))
|
|
}
|
|
if goos == "windows" {
|
|
elem[0] = strings.ToUpper(elem[0])
|
|
}
|
|
envs[elem[0]] = elem[1]
|
|
}
|
|
for k, v := range envs {
|
|
new = append(new, k+"="+v)
|
|
}
|
|
return new
|
|
}
|
|
|
|
func getenv(env []string, key string) string {
|
|
prefix := key + "="
|
|
for _, kv := range env {
|
|
if strings.HasPrefix(kv, prefix) {
|
|
return kv[len(prefix):]
|
|
}
|
|
}
|
|
return ""
|
|
}
|
|
|
|
func archNDK() string {
|
|
if runtime.GOOS == "windows" && runtime.GOARCH == "386" {
|
|
return "windows"
|
|
} else {
|
|
var arch string
|
|
switch runtime.GOARCH {
|
|
case "386":
|
|
arch = "x86"
|
|
case "amd64":
|
|
arch = "x86_64"
|
|
default:
|
|
panic("unsupported GOARCH: " + runtime.GOARCH)
|
|
}
|
|
return runtime.GOOS + "-" + arch
|
|
}
|
|
}
|
|
|
|
type ndkToolchain struct {
|
|
arch string
|
|
abi string
|
|
toolPrefix string
|
|
clangPrefix string
|
|
}
|
|
|
|
func (tc *ndkToolchain) Path(ndkRoot, toolName string) string {
|
|
var pref string
|
|
switch toolName {
|
|
case "clang", "clang++":
|
|
pref = tc.clangPrefix
|
|
default:
|
|
pref = tc.toolPrefix
|
|
}
|
|
return filepath.Join(ndkRoot, "toolchains", "llvm", "prebuilt", archNDK(), "bin", pref+"-"+toolName)
|
|
}
|
|
|
|
type ndkConfig map[string]ndkToolchain // map: GOOS->androidConfig.
|
|
|
|
func (nc ndkConfig) Toolchain(arch string) ndkToolchain {
|
|
tc, ok := nc[arch]
|
|
if !ok {
|
|
panic(`unsupported architecture: ` + arch)
|
|
}
|
|
return tc
|
|
}
|
|
|
|
var ndk = ndkConfig{
|
|
"arm": {
|
|
arch: "arm",
|
|
abi: "armeabi-v7a",
|
|
toolPrefix: "arm-linux-androideabi",
|
|
clangPrefix: "armv7a-linux-androideabi16",
|
|
},
|
|
"arm64": {
|
|
arch: "arm64",
|
|
abi: "arm64-v8a",
|
|
toolPrefix: "aarch64-linux-android",
|
|
clangPrefix: "aarch64-linux-android21",
|
|
},
|
|
|
|
"386": {
|
|
arch: "x86",
|
|
abi: "x86",
|
|
toolPrefix: "i686-linux-android",
|
|
clangPrefix: "i686-linux-android16",
|
|
},
|
|
"amd64": {
|
|
arch: "x86_64",
|
|
abi: "x86_64",
|
|
toolPrefix: "x86_64-linux-android",
|
|
clangPrefix: "x86_64-linux-android21",
|
|
},
|
|
}
|
|
|
|
func xcodeAvailable() bool {
|
|
err := exec.Command("xcrun", "xcodebuild", "-version").Run()
|
|
return err == nil
|
|
}
|