mirror of
https://github.com/jimeh/build-emacs-for-macos.git
synced 2026-02-19 13:06:38 +00:00
Compare commits
14 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
4a7c507858
|
|||
|
884f1607f6
|
|||
|
c7daa1350b
|
|||
|
9223ff8e28
|
|||
|
70bf6b05d5
|
|||
|
8936f4762a
|
|||
|
be8e04e7a5
|
|||
| aadf32cbbe | |||
|
d8bbcb72b3
|
|||
|
d6f11b5459
|
|||
|
09e67381cb
|
|||
|
d21ccad3e4
|
|||
| 036d2a82d9 | |||
|
111cb64993
|
2
Brewfile
2
Brewfile
@@ -3,11 +3,13 @@
|
|||||||
brew 'coreutils'
|
brew 'coreutils'
|
||||||
brew 'curl'
|
brew 'curl'
|
||||||
brew 'expat'
|
brew 'expat'
|
||||||
|
brew 'gcc'
|
||||||
brew 'gmp'
|
brew 'gmp'
|
||||||
brew 'gnu-sed'
|
brew 'gnu-sed'
|
||||||
brew 'gnutls'
|
brew 'gnutls'
|
||||||
brew 'jansson'
|
brew 'jansson'
|
||||||
brew 'libffi'
|
brew 'libffi'
|
||||||
|
brew 'libgccjit'
|
||||||
brew 'libiconv'
|
brew 'libiconv'
|
||||||
brew 'librsvg'
|
brew 'librsvg'
|
||||||
brew 'libtasn1'
|
brew 'libtasn1'
|
||||||
|
|||||||
37
CHANGELOG.md
37
CHANGELOG.md
@@ -2,6 +2,43 @@
|
|||||||
|
|
||||||
All notable changes to this project will be documented in this file. See [standard-version](https://github.com/conventional-changelog/standard-version) for commit guidelines.
|
All notable changes to this project will be documented in this file. See [standard-version](https://github.com/conventional-changelog/standard-version) for commit guidelines.
|
||||||
|
|
||||||
|
### [0.4.2](https://github.com/jimeh/build-emacs-for-macos/compare/0.4.1...0.4.2) (2020-12-09)
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* **cli:** avoid error if --git-sha is used without a branch/tag/sha argument ([884f160](https://github.com/jimeh/build-emacs-for-macos/commit/884f1607f6707ca187b1abfb0ce562757d872230)), closes [#21](https://github.com/jimeh/build-emacs-for-macos/issues/21)
|
||||||
|
* **native_comp:** update env setup patch for recent changes to comp.el ([c7daa13](https://github.com/jimeh/build-emacs-for-macos/commit/c7daa1350bd69df172ce6484c54189d2cee8d97e))
|
||||||
|
|
||||||
|
### [0.4.1](https://github.com/jimeh/build-emacs-for-macos/compare/0.4.0...0.4.1) (2020-10-29)
|
||||||
|
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* **native_comp:** remove patch based on feature/native-comp-macos-fixes branch ([70bf6b0](https://github.com/jimeh/build-emacs-for-macos/commit/70bf6b05d584976632b2fd2947c0bf692f5b6421))
|
||||||
|
|
||||||
|
## [0.4.0](https://github.com/jimeh/build-emacs-for-macos/compare/0.3.0...0.4.0) (2020-10-04)
|
||||||
|
|
||||||
|
|
||||||
|
### ⚠ BREAKING CHANGES
|
||||||
|
|
||||||
|
* **native_comp:** Standard Homewbrew `gcc` and `libgccjit` formula are now required for native-comp, instead of the custom patched gcc formula.
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* **native_comp:** use new libgccjit Homebrew formula ([d8bbcb7](https://github.com/jimeh/build-emacs-for-macos/commit/d8bbcb72b33f6bde8678c9d37548217ffdf3d641))
|
||||||
|
|
||||||
|
## [0.3.0](https://github.com/jimeh/build-emacs-for-macos/compare/0.2.0...0.3.0) (2020-09-22)
|
||||||
|
|
||||||
|
|
||||||
|
### ⚠ BREAKING CHANGES
|
||||||
|
|
||||||
|
* **native_comp:** `--[no-]launcher` option is deprecated, as launcher script is no longer used.
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
* **native_comp:** use elisp patch instead of launcher script to set LIBRARY_PATH ([111cb64](https://github.com/jimeh/build-emacs-for-macos/commit/111cb6499368d14853a5927d38a43fc5e2f759f4)), closes [#14](https://github.com/jimeh/build-emacs-for-macos/issues/14)
|
||||||
|
|
||||||
## [0.2.0](https://github.com/jimeh/build-emacs-for-macos/compare/0.1.1...0.2.0) (2020-09-20)
|
## [0.2.0](https://github.com/jimeh/build-emacs-for-macos/compare/0.1.1...0.2.0) (2020-09-20)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,21 +0,0 @@
|
|||||||
diff --git a/Formula/gcc.rb b/Formula/gcc.rb
|
|
||||||
index 1bd636d496..03ad124218 100644
|
|
||||||
--- a/Formula/gcc.rb
|
|
||||||
+++ b/Formula/gcc.rb
|
|
||||||
@@ -53,7 +53,7 @@ class Gcc < Formula
|
|
||||||
# - Ada, which requires a pre-existing GCC Ada compiler to bootstrap
|
|
||||||
# - Go, currently not supported on macOS
|
|
||||||
# - BRIG
|
|
||||||
- languages = %w[c c++ objc obj-c++ fortran]
|
|
||||||
+ languages = %w[c c++ objc obj-c++ fortran jit]
|
|
||||||
|
|
||||||
osmajor = `uname -r`.split(".").first
|
|
||||||
pkgversion = "Homebrew GCC #{pkg_version} #{build.used_options*" "}".strip
|
|
||||||
@@ -73,6 +73,7 @@ class Gcc < Formula
|
|
||||||
--with-system-zlib
|
|
||||||
--with-pkgversion=#{pkgversion}
|
|
||||||
--with-bugurl=https://github.com/Homebrew/homebrew-core/issues
|
|
||||||
+ --enable-host-shared
|
|
||||||
]
|
|
||||||
|
|
||||||
# Xcode 10 dropped 32-bit support
|
|
||||||
81
README.md
81
README.md
@@ -18,7 +18,7 @@ Use this script at your own risk.
|
|||||||
|
|
||||||
## Status
|
## Status
|
||||||
|
|
||||||
As of writing (2020-08-19) it works for me on my machine. Your luck may vary.
|
As of writing (2020-10-04) it works for me on my machine. Your luck may vary.
|
||||||
|
|
||||||
I have successfully built:
|
I have successfully built:
|
||||||
|
|
||||||
@@ -30,7 +30,7 @@ For reference, my machine is:
|
|||||||
|
|
||||||
- 13-inch MacBook Pro (2020), 10th-gen 2.3 GHz Quad-Core Intel Core i7 (4c/8t)
|
- 13-inch MacBook Pro (2020), 10th-gen 2.3 GHz Quad-Core Intel Core i7 (4c/8t)
|
||||||
- macOS 10.15.6 (19G2021)
|
- macOS 10.15.6 (19G2021)
|
||||||
- Xcode 11.7
|
- Xcode 12.0
|
||||||
|
|
||||||
## Limitations
|
## Limitations
|
||||||
|
|
||||||
@@ -76,11 +76,11 @@ Options:
|
|||||||
--[no-]native-full-aot Enable/disable NATIVE_FULL_AOT / Ahead of Time compilation (default: disabled)
|
--[no-]native-full-aot Enable/disable NATIVE_FULL_AOT / Ahead of Time compilation (default: disabled)
|
||||||
--[no-]native-comp-macos-fixes
|
--[no-]native-comp-macos-fixes
|
||||||
Enable/disable fix based on feature/native-comp-macos-fixes branch (default: enabled if native-comp supported)
|
Enable/disable fix based on feature/native-comp-macos-fixes branch (default: enabled if native-comp supported)
|
||||||
--[no-]launcher Enable/disable embedded launcher script (default: enabled if native-comp is enabled)
|
|
||||||
--rsvg Enable SVG image support via librsvg, can yield a unstable build (default: disabled)
|
--rsvg Enable SVG image support via librsvg, can yield a unstable build (default: disabled)
|
||||||
--no-titlebar Apply no-titlebar patch (default: disabled)
|
--no-titlebar Apply no-titlebar patch (default: disabled)
|
||||||
--no-frame-refocus Apply no-frame-refocus patch (default: disabled)
|
--no-frame-refocus Apply no-frame-refocus patch (default: disabled)
|
||||||
--[no-]native-fast-boot DEPRECATED: use --[no-]native-full-aot instead
|
--[no-]native-fast-boot DEPRECATED: use --[no-]native-full-aot instead
|
||||||
|
--[no-]launcher DEPRECATED: Launcher script is no longer used.
|
||||||
```
|
```
|
||||||
|
|
||||||
Resulting applications are saved to the `builds` directory in a bzip2 compressed
|
Resulting applications are saved to the `builds` directory in a bzip2 compressed
|
||||||
@@ -89,6 +89,9 @@ tarball.
|
|||||||
If you don't want the build process to eat all your CPU cores, pass in a `-j`
|
If you don't want the build process to eat all your CPU cores, pass in a `-j`
|
||||||
value of how many CPU cores you want it to use.
|
value of how many CPU cores you want it to use.
|
||||||
|
|
||||||
|
Re-building the same Git SHA again can yield weird results unless you first
|
||||||
|
trash the corresponding directory from the `sources` directory.
|
||||||
|
|
||||||
### Examples
|
### Examples
|
||||||
|
|
||||||
To download a tarball of the `master` branch (Emacs 28.x as of writing) and
|
To download a tarball of the `master` branch (Emacs 28.x as of writing) and
|
||||||
@@ -109,31 +112,42 @@ All sources as downloaded as tarballs from the
|
|||||||
to get a list of tags/branches available to install, simply check said
|
to get a list of tags/branches available to install, simply check said
|
||||||
repository.
|
repository.
|
||||||
|
|
||||||
|
## Use Self-Contained Emacs.app as `emacs` CLI Tool
|
||||||
|
|
||||||
|
As the application bundle is self-contained, the main executable needs to be run
|
||||||
|
from within the application bundle. This means a simple symlink to
|
||||||
|
`Emacs.app/Contents/MacOS/Emacs` will not work. Instead the best approach is to
|
||||||
|
create a shell alias called `emacs` pointing to the right place.
|
||||||
|
|
||||||
|
Personally I use something similar to this:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
if [ -f "/Applications/Emacs.app/Contents/MacOS/Emacs" ]; then
|
||||||
|
export EMACS="/Applications/Emacs.app/Contents/MacOS/Emacs"
|
||||||
|
alias emacs="$EMACS -nw"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -f "/Applications/Emacs.app/Contents/MacOS/bin/emacsclient" ]; then
|
||||||
|
alias emacsclient="/Applications/Emacs.app/Contents/MacOS/bin/emacsclient"
|
||||||
|
fi
|
||||||
|
```
|
||||||
|
|
||||||
|
Setting the `EMACS` variable to the binary path seems to be a good idea, as some
|
||||||
|
tools seems to use it to figure out the path to Emacs' executable, including
|
||||||
|
[doom-emacs](https://github.com/hlissner/doom-emacs)' `doom` CLI tool.
|
||||||
|
|
||||||
## Native-Comp
|
## Native-Comp
|
||||||
|
|
||||||
To build a Emacs.app with native-comp support
|
Building a Emacs.app with native-comp support
|
||||||
([gccemacs](https://akrl.sdf.org/gccemacs.html)) from the `feature/native-comp`
|
([gccemacs](https://akrl.sdf.org/gccemacs.html)) from the `feature/native-comp`
|
||||||
branch, you will need to install a patched version of Homebrew's `gcc` formula
|
branch is now supported without much hassle thanks to the newly released
|
||||||
that includes libgccjit.
|
`libgccjit` Homebrew formula.
|
||||||
|
|
||||||
The patch itself is in `./Formula/gcc.rb.patch`, and comes from
|
Changes from the `feature/native-comp-macos-fixes` branch are also applied
|
||||||
[this](https://gist.github.com/mikroskeem/0a5c909c1880408adf732ceba6d3f9ab#1-gcc-with-libgccjit-enabled)
|
through a custom patch process which should be more future-proof compared to a
|
||||||
gist.
|
regular git diff patch.
|
||||||
|
|
||||||
You can install the patched formula by running the helper script:
|
To build a Emacs.app with native compilation enabled, simply run:
|
||||||
|
|
||||||
```
|
|
||||||
./install-patched-gcc
|
|
||||||
```
|
|
||||||
|
|
||||||
The helper script will copy your local `gcc.rb` Forumla from Homebrew to
|
|
||||||
`./Formula`, and apply the `./Formula/gcc.rb.patch` to it. After which it then
|
|
||||||
proceed to install the patched gcc formula which includes libgccjit.
|
|
||||||
|
|
||||||
As it requires installing and compiling GCC from source, it can take anywhere
|
|
||||||
between 30-60 minutes or more depending on your machine.
|
|
||||||
|
|
||||||
And finally to build a Emacs.app with native compilation enabled, run:
|
|
||||||
|
|
||||||
```
|
```
|
||||||
./build-emacs-for-macos feature/native-comp
|
./build-emacs-for-macos feature/native-comp
|
||||||
@@ -142,7 +156,7 @@ And finally to build a Emacs.app with native compilation enabled, run:
|
|||||||
By default `NATIVE_FULL_AOT` is disabled which ensures a fast build by native
|
By default `NATIVE_FULL_AOT` is disabled which ensures a fast build by native
|
||||||
compiling as few lisp source files as possible to build the app. Any remaining
|
compiling as few lisp source files as possible to build the app. Any remaining
|
||||||
lisp files will be dynamically compiled in the background the first time you use
|
lisp files will be dynamically compiled in the background the first time you use
|
||||||
them.
|
them. To enable native full AoT, pass in the `--native-full-aot` option.
|
||||||
|
|
||||||
On my machine it takes around 10 minutes to build Emacs.app with
|
On my machine it takes around 10 minutes to build Emacs.app with
|
||||||
`NATIVE_FULL_AOT` disabled. With it enabled it takes around 20-25 minutes.
|
`NATIVE_FULL_AOT` disabled. With it enabled it takes around 20-25 minutes.
|
||||||
@@ -160,26 +174,13 @@ By default natively compiled `*.eln` files will be cached in
|
|||||||
the first element of the `comp-eln-load-path` variable. The path string must end
|
the first element of the `comp-eln-load-path` variable. The path string must end
|
||||||
with a `/`.
|
with a `/`.
|
||||||
|
|
||||||
Also it seems somewhat common that some `*.eln` files are left behind with a
|
|
||||||
zero-byte file size if Emacs is quit while async native compilation is in
|
|
||||||
progress. Such empty files causes errors on startup, and needs to be deleted.
|
|
||||||
|
|
||||||
Below is an example which stores all compiled `*.eln` files in `cache/eln-cache`
|
Below is an example which stores all compiled `*.eln` files in `cache/eln-cache`
|
||||||
within your Emacs configuration directory, and also deletes any `*.eln` files in
|
within your Emacs configuration directory:
|
||||||
said directory which have a file size of zero bytes:
|
|
||||||
|
|
||||||
```elisp
|
```elisp
|
||||||
(when (boundp 'comp-eln-load-path)
|
(when (boundp 'comp-eln-load-path)
|
||||||
(let ((eln-cache-dir (expand-file-name "cache/eln-cache/"
|
(setcar comp-eln-load-path
|
||||||
user-emacs-directory))
|
(expand-file-name "cache/eln-cache/" user-emacs-directory)))
|
||||||
(find-exec (executable-find "find")))
|
|
||||||
(setcar comp-eln-load-path eln-cache-dir)
|
|
||||||
;; Quitting emacs while native compilation in progress can leave zero byte
|
|
||||||
;; sized *.eln files behind. Hence delete such files during startup.
|
|
||||||
(when find-exec
|
|
||||||
(call-process find-exec nil nil nil eln-cache-dir
|
|
||||||
"-name" "*.eln" "-size" "0" "-delete" "-or"
|
|
||||||
"-name" "*.eln.tmp" "-size" "0" "-delete"))))
|
|
||||||
```
|
```
|
||||||
|
|
||||||
### Issues
|
### Issues
|
||||||
|
|||||||
@@ -72,14 +72,13 @@ class Build
|
|||||||
DOWNLOAD_URL = 'https://github.com/emacs-mirror/emacs/tarball/%s'
|
DOWNLOAD_URL = 'https://github.com/emacs-mirror/emacs/tarball/%s'
|
||||||
LATEST_URL = 'https://api.github.com/repos/emacs-mirror/emacs/commits/%s'
|
LATEST_URL = 'https://api.github.com/repos/emacs-mirror/emacs/commits/%s'
|
||||||
NATIVE_COMP_REF_REGEXP = %r{^feature/native-comp}.freeze
|
NATIVE_COMP_REF_REGEXP = %r{^feature/native-comp}.freeze
|
||||||
LAUNCHER_TEMPLATE = './launcher.bash.erb'
|
|
||||||
|
|
||||||
attr_reader :root_dir
|
attr_reader :root_dir
|
||||||
attr_reader :source_dir
|
attr_reader :source_dir
|
||||||
attr_reader :ref
|
attr_reader :ref
|
||||||
attr_reader :options
|
attr_reader :options
|
||||||
|
|
||||||
def initialize(root_dir, ref = 'master', options = {})
|
def initialize(root_dir, ref = nil, options = {})
|
||||||
@root_dir = root_dir
|
@root_dir = root_dir
|
||||||
@ref = ref
|
@ref = ref
|
||||||
@options = options
|
@options = options
|
||||||
@@ -96,16 +95,11 @@ class Build
|
|||||||
autogen
|
autogen
|
||||||
detect_native_comp if options[:native_comp].nil?
|
detect_native_comp if options[:native_comp].nil?
|
||||||
|
|
||||||
if options[:native_comp] && options[:launcher].nil?
|
|
||||||
options[:launcher] = true
|
|
||||||
end
|
|
||||||
|
|
||||||
app = compile_source(@source_dir)
|
app = compile_source(@source_dir)
|
||||||
symlink_internals(app)
|
symlink_internals(app)
|
||||||
|
|
||||||
LibEmbedder.new(app, brew_dir, extra_libs).embed
|
LibEmbedder.new(app, brew_dir, extra_libs).embed
|
||||||
LibGccJitEmbedder.new(app, gcc_dir).embed if options[:native_comp]
|
GccLibEmbedder.new(app, gcc_dir).embed if options[:native_comp]
|
||||||
LauncherEmbedder.new(app, LAUNCHER_TEMPLATE).embed if options[:launcher]
|
|
||||||
|
|
||||||
archive_app(app)
|
archive_app(app)
|
||||||
end
|
end
|
||||||
@@ -132,6 +126,10 @@ class Build
|
|||||||
@gcc_dir ||= `brew --prefix gcc`.chomp
|
@gcc_dir ||= `brew --prefix gcc`.chomp
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def libgccjit_dir
|
||||||
|
@libgccjit_dir ||= `brew --prefix libgccjit`.chomp
|
||||||
|
end
|
||||||
|
|
||||||
def extra_libs
|
def extra_libs
|
||||||
@extra_libs ||= [
|
@extra_libs ||= [
|
||||||
"#{brew_dir}/opt/expat/lib/libexpat.1.dylib",
|
"#{brew_dir}/opt/expat/lib/libexpat.1.dylib",
|
||||||
@@ -145,7 +143,7 @@ class Build
|
|||||||
|
|
||||||
url = (DOWNLOAD_URL % sha)
|
url = (DOWNLOAD_URL % sha)
|
||||||
filename = "emacs-mirror-emacs-#{sha[0..6]}.tgz"
|
filename = "emacs-mirror-emacs-#{sha[0..6]}.tgz"
|
||||||
target = File.join(tarballs_dir, filename)
|
target = File.join(tarballs_dir, filename)
|
||||||
|
|
||||||
if File.exist?(target)
|
if File.exist?(target)
|
||||||
info "#{filename} already exists locally, attempting to use."
|
info "#{filename} already exists locally, attempting to use."
|
||||||
@@ -210,17 +208,25 @@ class Build
|
|||||||
|
|
||||||
def verify_libgccjit
|
def verify_libgccjit
|
||||||
err 'gcc not installed' unless Dir.exist?(gcc_dir)
|
err 'gcc not installed' unless Dir.exist?(gcc_dir)
|
||||||
|
err 'libgccjit not installed' unless Dir.exist?(libgccjit_dir)
|
||||||
|
|
||||||
return if Dir["#{gcc_dir}/lib/**/libgccjit.so*"].any?
|
if Dir["#{libgccjit_dir}/lib/**/libgccjit.so*"].empty?
|
||||||
|
err "Detected libgccjit (#{libgccjit_dir}) does not have any " \
|
||||||
|
'libgccjit.so* files. Please try reinstalling libgccjit: ' \
|
||||||
|
'brew reinstall libgccjit'
|
||||||
|
end
|
||||||
|
|
||||||
err "Detected GCC (#{gcc_dir}) does not have libgccjit. Ensure patched " \
|
dirs = Dir["{#{gcc_dir},#{libgccjit_dir}}/lib/gcc/*"]
|
||||||
'gcc brew formula has been installed via ./install-patched-gcc'
|
return if dirs.map { |dir| File.basename(dir) }.uniq == %w[10]
|
||||||
|
|
||||||
|
err 'Detected gcc and libgccjit library paths do not belong to the ' \
|
||||||
|
"same major version. Detected paths:\n - #{dirs.join(' - ')}"
|
||||||
end
|
end
|
||||||
|
|
||||||
def gcc_library_paths
|
def gcc_library_paths
|
||||||
@gcc_library_paths ||= Dir[
|
@gcc_library_paths ||= Dir[
|
||||||
"#{gcc_dir}/lib/gcc/*",
|
"{#{gcc_dir},#{libgccjit_dir}}/lib/gcc/*",
|
||||||
"#{gcc_dir}/lib/gcc/*/gcc/*apple-darwin*/*"
|
"{#{gcc_dir},#{libgccjit_dir}}/lib/gcc/*/gcc/*apple-darwin*/*"
|
||||||
].sort_by { |p| [p.size, p] }
|
].sort_by { |p| [p.size, p] }
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -252,9 +258,7 @@ class Build
|
|||||||
verify_native_comp
|
verify_native_comp
|
||||||
verify_libgccjit
|
verify_libgccjit
|
||||||
|
|
||||||
if options[:macos_fixes] && ref != 'feature/native-comp-macos-fixes'
|
apply_native_comp_env_setup_patch(source)
|
||||||
apply_native_comp_macos_fixes
|
|
||||||
end
|
|
||||||
|
|
||||||
ENV['CFLAGS'] = [
|
ENV['CFLAGS'] = [
|
||||||
"-I#{gcc_dir}/include",
|
"-I#{gcc_dir}/include",
|
||||||
@@ -264,7 +268,8 @@ class Build
|
|||||||
|
|
||||||
ENV['LDFLAGS'] = [
|
ENV['LDFLAGS'] = [
|
||||||
gcc_library_paths.map { |path| "-L#{path}" },
|
gcc_library_paths.map { |path| "-L#{path}" },
|
||||||
"-I#{gcc_dir}/include"
|
"-I#{gcc_dir}/include",
|
||||||
|
"-I#{libgccjit_dir}/include"
|
||||||
].flatten.compact.join(' ')
|
].flatten.compact.join(' ')
|
||||||
|
|
||||||
ENV['LIBRARY_PATH'] = [
|
ENV['LIBRARY_PATH'] = [
|
||||||
@@ -361,18 +366,18 @@ class Build
|
|||||||
FileUtils.mkdir_p(builds_dir)
|
FileUtils.mkdir_p(builds_dir)
|
||||||
|
|
||||||
metadata = [
|
metadata = [
|
||||||
ref.gsub(/\W/, '-'),
|
meta[:ref]&.gsub(/\W/, '-'),
|
||||||
meta[:date],
|
meta[:date],
|
||||||
meta[:sha][0..6],
|
meta[:sha][0..6],
|
||||||
"macOS-#{OS.version}",
|
"macOS-#{OS.version}",
|
||||||
OS.arch
|
OS.arch
|
||||||
]
|
].compact
|
||||||
|
|
||||||
filename = "Emacs.app-[#{metadata.join('][')}].tbz"
|
filename = "Emacs.app-[#{metadata.join('][')}].tbz"
|
||||||
target = "#{builds_dir}/#{filename}"
|
target = "#{builds_dir}/#{filename}"
|
||||||
|
|
||||||
app_base = File.basename(app)
|
app_base = File.basename(app)
|
||||||
app_dir = File.dirname(app)
|
app_dir = File.dirname(app)
|
||||||
|
|
||||||
if !File.exist?(target)
|
if !File.exist?(target)
|
||||||
info "Creating #{filename} archive in \"#{builds_dir}\"..."
|
info "Creating #{filename} archive in \"#{builds_dir}\"..."
|
||||||
@@ -407,11 +412,13 @@ class Build
|
|||||||
err "Failed to get commit info about: #{ref_sha}" if commit_json.nil?
|
err "Failed to get commit info about: #{ref_sha}" if commit_json.nil?
|
||||||
|
|
||||||
commit = JSON.parse(commit_json)
|
commit = JSON.parse(commit_json)
|
||||||
|
meta = {
|
||||||
@meta = {
|
|
||||||
sha: commit['sha'],
|
sha: commit['sha'],
|
||||||
date: Date.parse(commit['commit']['committer']['date'])
|
date: Date.parse(commit['commit']['committer']['date'])
|
||||||
}
|
}
|
||||||
|
meta[:ref] = ref if ref && ref[0..6] != meta[:sha][0..6]
|
||||||
|
|
||||||
|
@meta = meta
|
||||||
end
|
end
|
||||||
|
|
||||||
def http_get(url)
|
def http_get(url)
|
||||||
@@ -426,35 +433,15 @@ class Build
|
|||||||
system(*args) || err("Exit code: #{$CHILD_STATUS.exitstatus}")
|
system(*args) || err("Exit code: #{$CHILD_STATUS.exitstatus}")
|
||||||
end
|
end
|
||||||
|
|
||||||
def apply_native_comp_macos_fixes
|
def apply_native_comp_env_setup_patch(source)
|
||||||
filename = 'Makefile.in'
|
term = 'native-compile-setup-environment-variables'
|
||||||
pattern = /^src: Makefile\n(.*BIN_DESTDIR.*)\nblessmail: Makefile src\n/m
|
file = 'lisp/emacs-lisp/comp.el'
|
||||||
content = File.read(filename).gsub(pattern) do
|
return if `grep '#{term}' '#{file}'`.strip.size.positive?
|
||||||
old_src_body = Regexp.last_match(1).strip
|
|
||||||
|
|
||||||
# check if already patched
|
apply_patch(
|
||||||
if old_src_body.include?('BIN_DESTDIR=\'${ns_appbindir}/\'')
|
{ file: "#{__dir__}/patches/native-comp-env-setup.patch" },
|
||||||
return old_src_body
|
source
|
||||||
end
|
)
|
||||||
|
|
||||||
self_contained = old_src_body.gsub(
|
|
||||||
'BIN_DESTDIR=\'$(DESTDIR)${bindir}/\'',
|
|
||||||
'BIN_DESTDIR=\'${ns_appbindir}/\''
|
|
||||||
)
|
|
||||||
|
|
||||||
<<~REPLACEMENT
|
|
||||||
src: Makefile
|
|
||||||
ifeq (${ns_self_contained},no)
|
|
||||||
\t#{old_src_body}
|
|
||||||
else
|
|
||||||
\t#{self_contained}
|
|
||||||
endif
|
|
||||||
|
|
||||||
blessmail: Makefile src
|
|
||||||
REPLACEMENT
|
|
||||||
end
|
|
||||||
|
|
||||||
File.open(filename, 'w') { |f| f.write(content) }
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def effective_version
|
def effective_version
|
||||||
@@ -521,7 +508,10 @@ class Build
|
|||||||
def apply_patch(patch, target)
|
def apply_patch(patch, target)
|
||||||
err "\"#{target}\" does not exist." unless File.exist?(target)
|
err "\"#{target}\" does not exist." unless File.exist?(target)
|
||||||
|
|
||||||
if patch[:url]
|
if patch[:file]
|
||||||
|
info 'Applying patch...'
|
||||||
|
FileUtils.cd(target) { run_cmd('patch', '-f', '-p1', '-i', patch[:file]) }
|
||||||
|
elsif patch[:url]
|
||||||
patch_dir = "#{target}/macos_patches"
|
patch_dir = "#{target}/macos_patches"
|
||||||
run_cmd('mkdir', '-p', patch_dir)
|
run_cmd('mkdir', '-p', patch_dir)
|
||||||
|
|
||||||
@@ -535,8 +525,7 @@ class Build
|
|||||||
info "Downloading patch: #{patch[:url]}"
|
info "Downloading patch: #{patch[:url]}"
|
||||||
run_cmd('curl', '-L#', patch[:url], '-o', patch_file)
|
run_cmd('curl', '-L#', patch[:url], '-o', patch_file)
|
||||||
|
|
||||||
info 'Applying patch...'
|
apply_patch({ file: patch_file }, target)
|
||||||
FileUtils.cd(target) { run_cmd('patch', '-f', '-p1', '-i', patch_file) }
|
|
||||||
elsif patch[:replace]
|
elsif patch[:replace]
|
||||||
err 'Patch replace input error' unless patch[:replace].size == 3
|
err 'Patch replace input error' unless patch[:replace].size == 3
|
||||||
|
|
||||||
@@ -578,7 +567,7 @@ class AbstractEmbedder
|
|||||||
end
|
end
|
||||||
|
|
||||||
def lib_dir_name
|
def lib_dir_name
|
||||||
"lib-#{OS.arch}-#{OS.version.to_s.tr('.', '_')}"
|
'lib'
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -691,7 +680,7 @@ class LibEmbedder < AbstractEmbedder
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
class LibGccJitEmbedder < AbstractEmbedder
|
class GccLibEmbedder < AbstractEmbedder
|
||||||
attr_reader :gcc_dir
|
attr_reader :gcc_dir
|
||||||
|
|
||||||
def initialize(app, gcc_dir)
|
def initialize(app, gcc_dir)
|
||||||
@@ -712,6 +701,7 @@ class LibGccJitEmbedder < AbstractEmbedder
|
|||||||
|
|
||||||
FileUtils.mkdir_p(File.dirname(target_dir))
|
FileUtils.mkdir_p(File.dirname(target_dir))
|
||||||
FileUtils.cp_r(source_dir, target_dir)
|
FileUtils.cp_r(source_dir, target_dir)
|
||||||
|
FileUtils.rm(Dir[File.join(target_dir, '**', '.DS_Store')], force: true)
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
@@ -725,13 +715,10 @@ class LibGccJitEmbedder < AbstractEmbedder
|
|||||||
end
|
end
|
||||||
|
|
||||||
def gcc_version
|
def gcc_version
|
||||||
@gcc_version ||= Dir[File.join(gcc_dir, 'lib', 'gcc', '*', 'libgccjit.so*')]
|
@gcc_version ||= Dir[File.join(gcc_dir, 'lib', 'gcc', '*', 'libgcc*')]
|
||||||
.map { |path| File.dirname(path) }
|
.map { |path| File.basename(File.dirname(path)) }
|
||||||
.select { |path| path.match(%r{/\d+$}) }
|
.select { |path| path.match(/^\d+$/) }
|
||||||
.uniq
|
.uniq.map(&:to_i).max.to_s
|
||||||
.map { |dir| File.basename(dir).to_i }
|
|
||||||
.max
|
|
||||||
.to_s
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def source_dir
|
def source_dir
|
||||||
@@ -739,67 +726,6 @@ class LibGccJitEmbedder < AbstractEmbedder
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
class LauncherEmbedder < AbstractEmbedder
|
|
||||||
attr_reader :template
|
|
||||||
|
|
||||||
def initialize(app, template)
|
|
||||||
super(app)
|
|
||||||
|
|
||||||
@template = template
|
|
||||||
end
|
|
||||||
|
|
||||||
def embed
|
|
||||||
if embedded?
|
|
||||||
info 'Launcher script already embedded in Emacs.app'
|
|
||||||
return
|
|
||||||
end
|
|
||||||
|
|
||||||
info 'Embedding launcher script into Emacs.app'
|
|
||||||
|
|
||||||
unless File.exist?("#{bin}#{bin_suffix}")
|
|
||||||
FileUtils.mv(bin, "#{bin}#{bin_suffix}")
|
|
||||||
end
|
|
||||||
|
|
||||||
unless File.exist?("#{bin}#{bin_suffix}#{dump_ext}")
|
|
||||||
FileUtils.mv("#{bin}#{dump_ext}", "#{bin}#{bin_suffix}#{dump_ext}")
|
|
||||||
end
|
|
||||||
|
|
||||||
unless File.exist?(bin)
|
|
||||||
File.write(bin, launcher)
|
|
||||||
File.chmod(0o775, bin)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
private
|
|
||||||
|
|
||||||
def bin_suffix
|
|
||||||
'-bin'
|
|
||||||
end
|
|
||||||
|
|
||||||
def dump_ext
|
|
||||||
'.pdmp'
|
|
||||||
end
|
|
||||||
|
|
||||||
def embedded?
|
|
||||||
File.exist?(bin) &&
|
|
||||||
File.exist?("#{bin}#{bin_suffix}") &&
|
|
||||||
File.exist?("#{bin}#{bin_suffix}#{dump_ext}")
|
|
||||||
end
|
|
||||||
|
|
||||||
def launcher
|
|
||||||
@launcher ||= ERB.new(File.read(template)).result(binding)
|
|
||||||
end
|
|
||||||
|
|
||||||
def library_paths
|
|
||||||
@library_paths ||= Dir[
|
|
||||||
"#{lib_dir}/gcc/*",
|
|
||||||
"#{lib_dir}/gcc/*/gcc/*apple-darwin*/*"
|
|
||||||
].map do |p|
|
|
||||||
p.gsub(/^#{Regexp.escape(lib_dir + '/')}/, '')
|
|
||||||
end.sort_by { |p| [p.size, p] }
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
if __FILE__ == $PROGRAM_NAME
|
if __FILE__ == $PROGRAM_NAME
|
||||||
cli_options = {
|
cli_options = {
|
||||||
macos_fixes: true,
|
macos_fixes: true,
|
||||||
@@ -855,12 +781,6 @@ if __FILE__ == $PROGRAM_NAME
|
|||||||
cli_options[:macos_fixes] = v
|
cli_options[:macos_fixes] = v
|
||||||
end
|
end
|
||||||
|
|
||||||
opts.on('--[no-]launcher',
|
|
||||||
'Enable/disable embedded launcher script ' \
|
|
||||||
'(default: enabled if native-comp is enabled)') do |v|
|
|
||||||
cli_options[:launcher] = v
|
|
||||||
end
|
|
||||||
|
|
||||||
opts.on('--rsvg', 'Enable SVG image support via librsvg, ' \
|
opts.on('--rsvg', 'Enable SVG image support via librsvg, ' \
|
||||||
'can yield a unstable build (default: disabled)') do
|
'can yield a unstable build (default: disabled)') do
|
||||||
cli_options[:rsvg] = true
|
cli_options[:rsvg] = true
|
||||||
@@ -885,6 +805,12 @@ if __FILE__ == $PROGRAM_NAME
|
|||||||
'use --native-full-aot instead'
|
'use --native-full-aot instead'
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
opts.on('--[no-]launcher',
|
||||||
|
'DEPRECATED: Launcher script is no longer used.') do |_|
|
||||||
|
raise Error, '--[no-]launcher option is deprecated, launcher ' \
|
||||||
|
'script is no longer used.'
|
||||||
|
end
|
||||||
end.parse!
|
end.parse!
|
||||||
|
|
||||||
Build.new(File.expand_path(__dir__), ARGV.shift, cli_options).build
|
Build.new(File.expand_path(__dir__), ARGV.shift, cli_options).build
|
||||||
|
|||||||
@@ -1,27 +0,0 @@
|
|||||||
#! /usr/bin/env bash
|
|
||||||
set -e
|
|
||||||
|
|
||||||
brewdir="$(brew --prefix)"
|
|
||||||
formula="${brewdir}/Homebrew/Library/Taps/homebrew/homebrew-core/Formula/gcc.rb"
|
|
||||||
|
|
||||||
if [ ! -f "$formula" ]; then
|
|
||||||
echo "ERROR: ${formula} does not exist." 1>&2
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
gnubins=(
|
|
||||||
"${brewdir}/opt/coreutils/libexec/gnubin"
|
|
||||||
"${brewdir}/opt/make/libexec/gnubin"
|
|
||||||
"${brewdir}/opt/gnu-sed/libexec/gnubin"
|
|
||||||
)
|
|
||||||
|
|
||||||
for gnubin in "${gnubins[@]}"; do
|
|
||||||
if [ -d "$gnubin" ]; then
|
|
||||||
export PATH="${gnubin}:$PATH"
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
|
|
||||||
cp "$formula" ./Formula/
|
|
||||||
|
|
||||||
patch -f -p1 -i ./Formula/gcc.rb.patch
|
|
||||||
brew install ./Formula/gcc.rb --build-from-source --force
|
|
||||||
@@ -1,65 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
# This launcher script is not part of Emacs proper. It is from the
|
|
||||||
# build-emacs-for-macos project (https://github.com/jimeh/build-emacs-for-macos)
|
|
||||||
# and helps facilitate proper startup of Emacs with environment varibales set as
|
|
||||||
# needed.
|
|
||||||
#
|
|
||||||
# Licensed under CC0 1.0 Universal:
|
|
||||||
# https://creativecommons.org/publicdomain/zero/1.0/
|
|
||||||
#
|
|
||||||
set -e
|
|
||||||
|
|
||||||
resolve_link() {
|
|
||||||
local file="$1"
|
|
||||||
|
|
||||||
while [ -L "$file" ]; do
|
|
||||||
file="$(readlink "$file")"
|
|
||||||
done
|
|
||||||
|
|
||||||
echo "$file"
|
|
||||||
}
|
|
||||||
|
|
||||||
realname() {
|
|
||||||
local path="$1"
|
|
||||||
local resolved
|
|
||||||
local cwd
|
|
||||||
|
|
||||||
cwd="$(pwd)"
|
|
||||||
resolved="$(resolve_link "$path")"
|
|
||||||
cd "$(dirname "$resolved")"
|
|
||||||
echo "$(pwd)/$(basename "$resolved")"
|
|
||||||
cd "$cwd"
|
|
||||||
}
|
|
||||||
|
|
||||||
join() {
|
|
||||||
local IFS="$1"
|
|
||||||
local parts=()
|
|
||||||
shift
|
|
||||||
|
|
||||||
for arg in "$@"; do
|
|
||||||
if [ "$arg" != "" ]; then
|
|
||||||
parts+=("$arg")
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
|
|
||||||
echo "${parts[*]}"
|
|
||||||
}
|
|
||||||
|
|
||||||
DIR="$(dirname "$(realname "$0")")"
|
|
||||||
BIN="${DIR}/Emacs<%= bin_suffix %>"
|
|
||||||
|
|
||||||
export PATH="${DIR}/bin:${DIR}/libexec:${PATH}"
|
|
||||||
<% if library_paths.any? %>
|
|
||||||
LIB_PATHS=(
|
|
||||||
'<%= library_paths.map { |p| p.gsub('\'', "\"'\"") }.join("'\n '") %>'
|
|
||||||
)
|
|
||||||
for lib in "${LIB_PATHS[@]}"; do
|
|
||||||
if [ -d "${DIR}/<%= lib_dir_name %>/${lib}" ]; then
|
|
||||||
libs="$(join : "$libs" "${DIR}/<%= lib_dir_name %>/${lib}")"
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
|
|
||||||
LIBRARY_PATH="$(join : "$libs" "$LIBRARY_PATH")"
|
|
||||||
export LIBRARY_PATH
|
|
||||||
<% end %>
|
|
||||||
exec "$BIN" "$@"
|
|
||||||
58
patches/native-comp-env-setup.patch
Normal file
58
patches/native-comp-env-setup.patch
Normal file
@@ -0,0 +1,58 @@
|
|||||||
|
diff --git a/lisp/emacs-lisp/comp.el b/lisp/emacs-lisp/comp.el
|
||||||
|
index 25e2de9..bcedd31 100644
|
||||||
|
--- a/lisp/emacs-lisp/comp.el
|
||||||
|
+++ b/lisp/emacs-lisp/comp.el
|
||||||
|
@@ -2801,6 +2801,53 @@ queued with LOAD %"
|
||||||
|
(comp-run-async-workers)
|
||||||
|
(message "Compilation started."))))
|
||||||
|
|
||||||
|
+;;;###autoload
|
||||||
|
+(defun native-compile-setup-environment-variables (&rest _args)
|
||||||
|
+ "Ensure LIBRARY_PATH is set correctly when libgccjit is bundled."
|
||||||
|
+ (when (and (eq system-type 'darwin)
|
||||||
|
+ (string-match-p "\.app\/Contents\/MacOS\/?$"
|
||||||
|
+ invocation-directory))
|
||||||
|
+ (let* ((library-path-env (getenv "LIBRARY_PATH"))
|
||||||
|
+ (gcc-base-dir (concat invocation-directory "lib/gcc"))
|
||||||
|
+ (gcc-version (car (seq-filter
|
||||||
|
+ (lambda (dir) (string-match-p "^[0-9]+$" dir))
|
||||||
|
+ (directory-files gcc-base-dir))))
|
||||||
|
+ (gcc-dir (concat gcc-base-dir "/" gcc-version))
|
||||||
|
+ (darwin-base-dir (car (file-expand-wildcards
|
||||||
|
+ (concat gcc-dir "/gcc/*apple-darwin*"))))
|
||||||
|
+ (darwin-version (car (seq-filter
|
||||||
|
+ (lambda (dir)
|
||||||
|
+ (string-match-p
|
||||||
|
+ "^[0-9]+\\(\.[0-9]+\\(\.[0-9]+\\)?\\)?$" dir))
|
||||||
|
+ (directory-files darwin-base-dir))))
|
||||||
|
+ (darwin-dir (concat darwin-base-dir "/" darwin-version))
|
||||||
|
+ (lib-paths (append
|
||||||
|
+ (list gcc-dir darwin-dir)
|
||||||
|
+ (if library-path-env (list library-path-env) (list)))))
|
||||||
|
+
|
||||||
|
+ (when (and gcc-dir darwin-dir)
|
||||||
|
+ (setenv "LIBRARY_PATH" (mapconcat 'identity lib-paths ":")))))
|
||||||
|
+
|
||||||
|
+ ;; Remove advice, as it only needs to run once.
|
||||||
|
+ (advice-remove 'native-compile
|
||||||
|
+ 'native-compile-setup-environment-variables)
|
||||||
|
+ (advice-remove 'comp--native-compile
|
||||||
|
+ 'native-compile-setup-environment-variables)
|
||||||
|
+ (advice-remove 'native-compile-async
|
||||||
|
+ 'native-compile-setup-environment-variables)
|
||||||
|
+ (advice-remove 'native--compile-async
|
||||||
|
+ 'native-compile-setup-environment-variables))
|
||||||
|
+
|
||||||
|
+;; Ensure environment setup runs before any native compilation.
|
||||||
|
+(advice-add 'native-compile :before
|
||||||
|
+ 'native-compile-setup-environment-variables)
|
||||||
|
+(advice-add 'comp--native-compile :before
|
||||||
|
+ 'native-compile-setup-environment-variables)
|
||||||
|
+(advice-add 'native-compile-async :before
|
||||||
|
+ 'native-compile-setup-environment-variables)
|
||||||
|
+(advice-add 'native--compile-async :before
|
||||||
|
+ 'native-compile-setup-environment-variables)
|
||||||
|
+
|
||||||
|
(provide 'comp)
|
||||||
|
|
||||||
|
;;; comp.el ends here
|
||||||
Reference in New Issue
Block a user