Skip to content

Commit

Permalink
frontend/qt: update to Qt6 on macOS
Browse files Browse the repository at this point in the history
brew's qt@6 would work excpt that it is missing all of the
qt@6/libexec folder, which contains the rcc tool we need to bundle the
frontend assets. Instead we use `aqtinstall` which installs Qt
binaries from official mirrors and includes all we need.

We get rid of osx-brew.sh which is only referenced in the BUILD.md and
instead inline the instructions, as it is strange to install brew and
other unprompted. Devs can follow along the BUILD.md instructions.

This commit also introduces universal builds for macOS. They are fat
binaries/libraries that include code for both the amd64 and arm64
architectures, so the app can run natively. As a result, the resulting
app size is roughly double to before.

Alternative considered but dismissed:
- Make and distribute two separate BitBox.app outputs per
  architecture. This reduces the app size, but is a bit more error
  prone for users and incurs more effort to distribute correctly. Also
  the AppStore requires universal builds, if we ever want to distbute
  the app through the AppStore.
  • Loading branch information
benma committed Oct 30, 2024
1 parent 9e69e82 commit adf3704
Show file tree
Hide file tree
Showing 11 changed files with 58 additions and 53 deletions.
14 changes: 14 additions & 0 deletions .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -104,6 +104,20 @@ jobs:
uses: actions/setup-node@v4
with:
node-version: '20.x'
- name: Set up Python
uses: actions/setup-python@v4
with:
python-version: '3.x'
- name: Install Qt
# Qt modules: Not sure why we need qtpositioning - app compilation fails without. Maybe the webengine depends on it.
# qtpositioning depends on qtserialport.
# qtwebchannel is for the JS/backend bridge.
# qtwebengine is for rendering the frontend.
run: |
pip install aqtinstall
aqt install-qt mac desktop 6.2.4 --modules qtpositioning qtserialport qtwebchannel qtwebengine --outputdir ~/Qt
echo "$HOME/Qt/6.2.4/macos/bin" >> $GITHUB_PATH
echo "$HOME/Qt/6.2.4/macos/libexec" >> $GITHUB_PATH
- name: Build macOS app
run: >
./scripts/github-ci.sh qt-osx;
Expand Down
7 changes: 0 additions & 7 deletions Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -30,13 +30,6 @@ gomobileinit:
# TODO: replace with go install golang.org/x/mobile/cmd/gomobile@latest once https://github.com/golang/mobile/pull/105 is merged.
git clone https://github.com/BitBoxSwiss/mobile.git /tmp/mobile && cd /tmp/mobile/cmd/gomobile && go install .
gomobile init
# Initializiation on MacOS
# - run make from $GOPATH/src/github.com/BitBoxSwiss/bitbox-wallet-app
# - additional dependencies: Qt 5.15 & Xcode command line tools
# - add to $PATH: /usr/local/opt/[email protected]/bin
osx-init:
./scripts/osx-brew.sh
$(MAKE) envinit
servewallet:
go run -mod=vendor ./cmd/servewallet
servewallet-mainnet:
Expand Down
8 changes: 0 additions & 8 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -53,14 +53,6 @@ To build the app or run the development workflow, the following dependencies nee
- [Qt5](https://www.qt.io) version 5.15.2
- install Qt for your platform, including the WebEngine component

#### Environment variables

Make sure the environment variables for qt, go and node are set, see also [docs/BUILD.md](./docs/BUILD.md) for platform specific instructions

To initialize the build environment and install the required go utilities (linters, ...), call
* `make envinit`, or
* on MacOS `make osx-init`

## Build the BitBoxApp

Clone this repository using `git clone --recursive`.
Expand Down
21 changes: 16 additions & 5 deletions docs/BUILD.md
Original file line number Diff line number Diff line change
Expand Up @@ -30,17 +30,28 @@ Build artifacts:

## MacOS

Make sure you have `qt@5/bin`, `[email protected]/bin` and `go/bin` in your PATH, i.e. add to your `.zshrc`
Install Go, Qt and create-dmg. Note that qt@6 from homebrew does **not** work as it is missing the
`rcc` tool.

```
# Install Go. Can also use the official installer
brew install [email protected]
brew install create-dmg
# InstalL Qt. Can also use the official installer.
pip install aqtinstall
aqt list-qt mac desktop --arch 6.2.4
aqt install-qt mac desktop 6.2.4 --modules qtpositioning qtserialport qtwebchannel qtwebengine --outputdir ~/Qt
```

Make sure you have `qt@6/bin`, `qt@6/libexec`, `[email protected]/bin` and `go/bin` in your PATH, i.e. add to your `.zshrc`:

```bash
export PATH="$PATH:/usr/local/opt/qt@5/bin"
export PATH="$PATH:~/Qt/6.2.4/macos/bin"
export PATH="$PATH:~/Qt/6.2.4/macos/libexec"
export PATH="$PATH:/usr/local/opt/[email protected]/bin"
export PATH="$PATH:$HOME/go/bin"
```

Prepare the MacOS system to have the build environment:
`make osx-init`

Build the QT frontend for MacOS:
`make qt-osx`

Expand Down
4 changes: 4 additions & 0 deletions frontends/qt/.gitignore
Original file line number Diff line number Diff line change
@@ -1,5 +1,9 @@
/build/
/config/
/server/libserver.h
/server/libserver_arm64.h
/server/libserver_amd64.h
/server/libserver.so
/server/libserver_arm64.so
/server/libserver_amd64.so
.qmake.stash
2 changes: 2 additions & 0 deletions frontends/qt/BitBox.pro
Original file line number Diff line number Diff line change
Expand Up @@ -64,6 +64,8 @@ SOURCES += \
HEADERS += libserver.h webclass.h filedialog.h

unix:macx {
CONFIG += sdk_no_version_check
QMAKE_APPLE_DEVICE_ARCHS = x86_64 arm64
# Those frameworks are needed for Go's http/net packages.
# Waiting for https://github.com/golang/go/issues/11258 to be able to automatically capture link flags.
LIBS += -framework CoreFoundation -framework Security
Expand Down
14 changes: 8 additions & 6 deletions frontends/qt/Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -14,19 +14,20 @@

include ../../env.mk.inc

PATH := $(PATH)

base:
mkdir build
./genassets.sh
qmake -set MACOS_MIN_VERSION $(MACOS_MIN_VERSION)
qmake -o build/Makefile
cd build && $(MAKE)
clean:
-rm -rf build
cd server && $(MAKE) clean
linux:
$(MAKE) clean
cd server && $(MAKE) linux
$(MAKE) base
qmake -o build/Makefile
cd build && $(MAKE)
mkdir build/linux-tmp build/linux
mv build/BitBox build/linux-tmp
cp build/assets.rcc build/linux-tmp/
Expand All @@ -47,13 +48,14 @@ osx:
$(MAKE) clean
cd server && $(MAKE) macosx
$(MAKE) base
# See https://stackoverflow.com/questions/77136519/qmake-failing-with-sdk-14-ventura-13-5-2
qmake -early QMAKE_DEFAULT_LIBDIRS=$(shell xcrun -show-sdk-path)/usr/lib -o build/Makefile
cd build && $(MAKE)
mkdir build/osx
mv build/BitBox.app build/osx/
cp resources/MacOS/Info.plist build/osx/BitBox.app/Contents/
cp resources/MacOS/icon.icns build/osx/BitBox.app/Contents/Resources/
# The `-executable` flag is needed to make the macOS build work when Qt is installed from homebrew.
# It works without it when Qt is installed using the official installer.
cd build/osx/ && macdeployqt BitBox.app -executable=BitBox.app/Contents/Frameworks/QtWebEngineCore.framework/Versions/Current/Helpers/QtWebEngineProcess.app/Contents/MacOS/QtWebEngineProcess
cd build/osx/ && macdeployqt BitBox.app
cp server/libserver.so build/osx/BitBox.app/Contents/Frameworks
cp build/assets.rcc build/osx/BitBox.app/Contents/MacOS/
install_name_tool -change libserver.so @executable_path/../Frameworks/libserver.so build/osx/BitBox.app/Contents/MacOS/BitBox
Expand Down
15 changes: 9 additions & 6 deletions frontends/qt/server/Makefile.macosx
Original file line number Diff line number Diff line change
Expand Up @@ -2,18 +2,21 @@ include ../../../hardening.mk.inc
include ../../../env.mk.inc
CGO=1
BUILDMODE=c-shared
GOARCH=amd64
GOOS=darwin
GOTOOLCHAIN=local

darwin:
CGO_CFLAGS="-g ${GODARWINSECFLAGS} ${CFLAGS}" \
CGO_LDFLAGS="${GODARWINLDFLAGS} ${LFLAGS}" \
MACOSX_DEPLOYMENT_TARGET=${MACOS_MIN_VERSION} \
GOARCH=${GOARCH} CGO_ENABLED=${CGO} GOOS=${GOOS} GOTOOLCHAIN=${GOTOOLCHAIN} \
go build -ldflags='-s' -x -mod=vendor \
-buildmode="c-shared" -o ${LIBNAME}.so
GOARCH=arm64 CGO_ENABLED=${CGO} GOOS=${GOOS} GOTOOLCHAIN=${GOTOOLCHAIN} go build -ldflags='-s' -x -mod=vendor -buildmode="c-shared" -o ${LIBNAME}_arm64.so
GOARCH=amd64 CGO_ENABLED=${CGO} GOOS=${GOOS} GOTOOLCHAIN=${GOTOOLCHAIN} go build -ldflags='-s' -x -mod=vendor -buildmode="c-shared" -o ${LIBNAME}_amd64.so
lipo -create -output ${LIBNAME}.so ${LIBNAME}_arm64.so ${LIBNAME}_amd64.so
# Without this, the internal names of the two libs are libserver_arm64.so and libserver_amd64.so, and
# will be linked with these names in the final executable (even though the library is fat and its name is libserver.so).
# With this, we fix the internal name which will be used when linking.
install_name_tool -id ${LIBNAME}.so ${LIBNAME}.so

clean:
-rm -f ${LIBNAME}.so
-rm -f ${LIBNAME}.h
-rm -f ${LIBNAME}.so ${LIBNAME}_arm64.so ${LIBNAME}_amd64.so
-rm -f ${LIBNAME}_arm64.h {LIBNAME}_amd64.h
2 changes: 1 addition & 1 deletion scripts/docker_install.sh
Original file line number Diff line number Diff line change
Expand Up @@ -73,7 +73,7 @@ apt-get -y install --no-install-recommends python3-pip python3-dev
pip install -U pip && pip install aqtinstall
# Not sure why we need qtpositioning - app compilation fails without. Maybe the webengine depends on it.
# qtpositioning depends on qtserialport.
# qtwebechannel is for the JS/backend bridge.
# qtwebchannel is for the JS/backend bridge.
# qtwebengine is for rendering the frontend.
aqt install-qt linux desktop 6.2.4 -m qtpositioning qtserialport qtwebchannel qtwebengine --outputdir /opt/qt6

Expand Down
10 changes: 4 additions & 6 deletions scripts/github-ci.sh
Original file line number Diff line number Diff line change
Expand Up @@ -29,15 +29,13 @@ fi

# The following is executed only on macOS machines.
if [ "$OS_NAME" == "osx" ]; then
# GitHub CI installs Go directly in the macos action, before executing
# GitHub CI installs Go and Qt directly in the macos action, before executing
# this script.
go version
brew install qt@5
export PATH="/usr/local/opt/qt@5/bin:$PATH"
export LDFLAGS="-L/usr/local/opt/qt@5/lib"
export CPPFLAGS="-I/usr/local/opt/qt@5/include"
export LDFLAGS="-L~/Qt/6.2.4/macos/lib"
export CPPFLAGS="-I~/Qt/6.2.4/macos/include"
export GOPATH=~/go
export PATH=$PATH:~/go/bin
export PATH="$PATH:~/go/bin"
mkdir -p $GOPATH/$(dirname $GO_SRC_DIR)
# GitHub checkout action (git clone) seem to require current work dir
# to be the root of the repo during its clean up phase. So, we push it
Expand Down
14 changes: 0 additions & 14 deletions scripts/osx-brew.sh

This file was deleted.

0 comments on commit adf3704

Please sign in to comment.