Get develop builds working on macOS as well

This commit is contained in:
Kovid Goyal 2023-07-28 21:14:31 +05:30
parent e9cc1cc9f4
commit f8fd6031c9
No known key found for this signature in database
GPG key ID: 06BC317B515ACE7C
3 changed files with 126 additions and 11 deletions

View file

@ -3,6 +3,7 @@
package main
import (
"bufio"
"bytes"
"errors"
"fmt"
@ -17,12 +18,18 @@ import (
"strings"
)
const folder = "dependencies"
const (
folder = "dependencies"
macos_prefix = "/Users/Shared/kitty-build/sw/sw"
macos_python = "python/Python.framework/Versions/Current/bin/python3"
macos_python_framework = "python/Python.framework/Versions/Current/Python"
macos_python_framework_exe = "python/Python.framework/Versions/Current/Resources/Python.app/Contents/MacOS/Python"
)
func root_dir() string {
f, e := filepath.Abs(filepath.Join(folder, runtime.GOOS+"-"+runtime.GOARCH))
if e != nil {
panic(e)
exit(e)
}
return f
}
@ -51,6 +58,94 @@ func exit(x any) {
}
// download deps {{{
type dependency struct {
path string
basename string
is_id bool
}
func lines(exe string, cmd ...string) []string {
c := exec.Command(exe, cmd...)
c.Stderr = os.Stderr
out, err := c.Output()
if err != nil {
exit(fmt.Errorf("Failed to run '%s' with error: %w", strings.Join(append([]string{exe}, cmd...), " "), err))
}
ans := []string{}
for s := bufio.NewScanner(bytes.NewReader(out)); s.Scan(); {
ans = append(ans, s.Text())
}
return ans
}
func get_dependencies(path string) (ans []dependency) {
a := lines("otool", "-D", path)
install_name := strings.TrimSpace(a[len(a)-1])
for _, line := range lines("otool", "-L", path) {
line = strings.TrimSpace(line)
if strings.Contains(line, "compatibility") && !strings.HasSuffix(line, ":") {
idx := strings.IndexByte(line, '(')
dep := strings.TrimSpace(line[:idx])
ans = append(ans, dependency{path: dep, is_id: dep == install_name})
}
}
return
}
func get_local_dependencies(path string) (ans []dependency) {
for _, dep := range get_dependencies(path) {
for _, y := range []string{filepath.Join(macos_prefix, "lib") + "/", filepath.Join(macos_prefix, "python", "Python.framework") + "/", "@rpath/"} {
if strings.HasPrefix(dep.path, y) {
if y == "@rpath/" {
dep.basename = "lib/" + dep.path[len(y):]
} else {
y = macos_prefix + "/"
dep.basename = dep.path[len(y):]
}
ans = append(ans, dep)
break
}
}
}
return
}
func change_dep(path string, dep dependency) {
cmd := []string{}
fid := filepath.Join(root_dir(), dep.basename)
if dep.is_id {
cmd = append(cmd, "-id", fid)
} else {
cmd = append(cmd, "-change", dep.path, fid)
}
cmd = append(cmd, path)
c := exec.Command("install_name_tool", cmd...)
c.Stdout = os.Stdout
c.Stderr = os.Stderr
if err := c.Run(); err != nil {
exit(fmt.Errorf("Failed to run command '%s' with error: %w", strings.Join(c.Args, " "), err))
}
}
func fix_dependencies_in_lib(path string) {
path, err := filepath.EvalSymlinks(path)
if err != nil {
exit(err)
}
if s, err := os.Stat(path); err != nil {
exit(err)
} else if err := os.Chmod(path, s.Mode().Perm()|0o200); err != nil {
exit(err)
}
for _, dep := range get_local_dependencies(path) {
change_dep(path, dep)
}
if ldeps := get_local_dependencies(path); len(ldeps) > 0 {
exit(fmt.Errorf("Failed to fix local dependencies in: %s", path))
}
}
func cached_download(url string) string {
fname := filepath.Base(url)
fmt.Println("Downloading", fname)
@ -126,7 +221,7 @@ func dependencies(args []string) {
var which string
switch runtime.GOOS {
case "darwin":
prefix = "/Users/Shared/kitty-build/sw/sw"
prefix = macos_prefix
which = "macos"
case "linux":
which = "linux"
@ -156,6 +251,11 @@ func dependencies(args []string) {
if err = cmd.Run(); err != nil {
exit(err)
}
if runtime.GOOS == "darwin" {
fix_dependencies_in_lib(filepath.Join(root, macos_python))
fix_dependencies_in_lib(filepath.Join(root, macos_python_framework))
fix_dependencies_in_lib(filepath.Join(root, macos_python_framework_exe))
}
if err = filepath.WalkDir(root, func(path string, d fs.DirEntry, err error) error {
if err != nil {
return err
@ -171,6 +271,11 @@ func dependencies(args []string) {
if strings.HasPrefix(name, "libfontconfig.so") {
os.Remove(path)
}
if runtime.GOOS == "darwin" {
if ext == ".so" || ext == ".dylib" {
fix_dependencies_in_lib(path)
}
}
}
return err
}); err != nil {
@ -209,6 +314,9 @@ func build(args []string) {
prepend("LD_LIBRARY_PATH", filepath.Join(root, "lib"))
os.Setenv("PYTHONHOME", root)
python = filepath.Join(root, "bin", "python")
case "darwin":
python = filepath.Join(root, macos_python)
os.Setenv("PKGCONFIG_EXE", filepath.Join(root, "bin", "pkg-config"))
default:
exit("Building is only supported on Linux and macOS")
}

View file

@ -182,6 +182,9 @@ run_embedded(RunData *run_data) {
PyPreConfig_InitPythonConfig(&preconfig);
preconfig.utf8_mode = 1;
preconfig.coerce_c_locale = 1;
#ifdef SET_PYTHON_HOME
preconfig.isolated = 1;
#endif
status = Py_PreInitialize(&preconfig);
if (PyStatus_Exception(status)) goto fail;
PyConfig config;
@ -195,10 +198,13 @@ run_embedded(RunData *run_data) {
status = PyConfig_SetBytesString(&config, &config.run_filename, run_data->lib_dir);
if (PyStatus_Exception(status)) goto fail;
#ifdef SET_PYTHON_HOME
#ifndef __APPLE__
char pyhome[256];
snprintf(pyhome, sizeof(pyhome), "%s/%s", run_data->lib_dir, SET_PYTHON_HOME);
status = PyConfig_SetBytesString(&config, &config.home, pyhome);
if (PyStatus_Exception(status)) goto fail;
#endif
config.isolated = 1;
#endif
status = Py_InitializeFromConfig(&config);
if (PyStatus_Exception(status)) goto fail;

View file

@ -430,6 +430,10 @@ def init_env(
for path in extra_library_dirs:
ldpaths.append(f'-L{path}')
if os.environ.get("DEVELOP_ROOT"):
cflags.insert(0, f'-I{os.environ["DEVELOP_ROOT"]}/include')
ldpaths.insert(0, f'-L{os.environ["DEVELOP_ROOT"]}/lib')
rs_cflag = detect_librsync(cc, cflags, ldflags + ldpaths)
if rs_cflag:
cflags.append(rs_cflag)
@ -438,10 +442,6 @@ def init_env(
set_arches(cflags)
set_arches(ldflags)
if os.environ.get("DEVELOP_ROOT"):
cflags.insert(0, f'-I{os.environ["DEVELOP_ROOT"]}/include')
ldpaths.insert(0, f'-L{os.environ["DEVELOP_ROOT"]}/lib')
return Env(cc, cppflags, cflags, ldflags, library_paths, ccver=ccver, ldpaths=ldpaths, vcs_rev=vcs_rev)
@ -709,10 +709,11 @@ class CompilationDatabase:
compilation_database = [
{'file': c.key.src, 'arguments': c.cmd, 'directory': src_base, 'output': c.key.dest} for c in self.compile_commands if c.key is not None
]
with open(self.dbpath, 'w') as f:
json.dump(compilation_database, f, indent=2, sort_keys=True)
with open(self.linkdbpath, 'w') as f:
json.dump([{'output': c.key, 'arguments': c.cmd, 'directory': src_base} for c in self.link_commands], f, indent=2, sort_keys=True)
with suppress(FileNotFoundError):
with open(self.dbpath, 'w') as f:
json.dump(compilation_database, f, indent=2, sort_keys=True)
with open(self.linkdbpath, 'w') as f:
json.dump([{'output': c.key, 'arguments': c.cmd, 'directory': src_base} for c in self.link_commands], f, indent=2, sort_keys=True)
def compile_c_extension(