Look for the compiler directly, not just the existence of the directory containing. Fixes golang/go#30637 Change-Id: Iae4429406da8622e7085bf3701c0f84d5c95d2fd Reviewed-on: https://go-review.googlesource.com/c/mobile/+/165937 TryBot-Result: Gobot Gobot <gobot@golang.org> Reviewed-by: Hyang-Ah Hana Kim <hyangah@gmail.com>
352 lines
8.3 KiB
Go
352 lines
8.3 KiB
Go
package main
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"log"
|
|
"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`")
|
|
}
|
|
|
|
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)
|
|
}
|
|
|
|
if err := envInit(); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
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 {
|
|
clang := toolchain.Path(ndkRoot, "clang")
|
|
if !buildN {
|
|
_, err = os.Stat(clang)
|
|
if err != nil {
|
|
return fmt.Errorf("No compiler for %s was found in the NDK (tried %q). Make sure your NDK version is >= r19b. Use `sdkmanager --update` to update it.", arch, clang)
|
|
}
|
|
}
|
|
androidEnv[arch] = []string{
|
|
"GOOS=android",
|
|
"GOARCH=" + arch,
|
|
"CC=" + 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)
|
|
}
|
|
return ndkRoot, nil
|
|
}
|
|
|
|
func envClang(sdkName string) (clang, cflags string, err error) {
|
|
if buildN {
|
|
return sdkName + "-clang", "-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++":
|
|
if runtime.GOOS == "windows" {
|
|
return tc.createNDKr19bWorkaroundTool(ndkRoot, toolName)
|
|
}
|
|
pref = tc.clangPrefix
|
|
default:
|
|
pref = tc.toolPrefix
|
|
}
|
|
return filepath.Join(ndkRoot, "toolchains", "llvm", "prebuilt", archNDK(), "bin", pref+"-"+toolName)
|
|
}
|
|
|
|
// createNDKr19bWorkaroundTool creates a Windows wrapper script for clang or clang++.
|
|
// The scripts included in r19b are broken on Windows: https://github.com/android-ndk/ndk/issues/920.
|
|
// TODO: Remove this when r19c is out; the code inside is hacky and panicky.
|
|
func (tc *ndkToolchain) createNDKr19bWorkaroundTool(ndkRoot, toolName string) string {
|
|
toolCmd := filepath.Join(tmpdir, fmt.Sprintf("%s-%s.cmd", tc.arch, toolName))
|
|
tool, err := os.Create(toolCmd)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
defer func() {
|
|
if err := tool.Close(); err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
}()
|
|
tcBin := filepath.Join(ndkRoot, "toolchains", "llvm", "prebuilt", archNDK(), "bin")
|
|
// Adapted from the NDK cmd wrappers.
|
|
toolCmdContent := fmt.Sprintf(`@echo off
|
|
set _BIN_DIR=%s\
|
|
%%_BIN_DIR%%%s.exe --target=%s -fno-addrsig %%*"`, tcBin, toolName, tc.clangPrefix)
|
|
if _, err = tool.Write([]byte(toolCmdContent)); err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
return toolCmd
|
|
}
|
|
|
|
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
|
|
}
|