7.7 KiB
Most HarfBuzz developers do so on Linux or macOS. However, HarfBuzz is a cross-platform library and it is important to ensure that it works on Windows as well. In particular, we use this workflow to develop and test the HarfBuzz Uniscribe shaper and DirectWrite shaper and font backend, all from Linux or macOS.
This document provides instructions for cross-compiling HarfBuzz on Linux or macOS, for Windows, using the MinGW toolchain, and running tests and utilties under Wine.
We then discuss using native Windows Uniscribe or DirectWrite DLLs, which allows you to test HarfBuzz's shaping against the Microsoft shaping engines instead of those provided by Wine.
This document assumes that you are familiar with building HarfBuzz on Linux or macOS.
You can build for 32bit or 64bit Windows. If your intention is to use a native Uniscribe usp10.dll from Windows 7 or before, you would need to build for 32bit. If you want to use a native DirectWrite DLL from Windows 10 or later, you would need to build for 64bit.
- Install Wine.
- Fedora:
dnf install wine
. - Mac:
brew install wine-stable
.
- Install the
mingw-w64
cross-compiler.
- Fedora, 32bit:
dnf install mingw32-gcc-c++
- Fedora, 64bit:
dnf install mingw64-gcc-c++
- Debian:
apt install g++-mingw-w64
- Mac:
brew install mingw-w64
- Install dependencies.
- Fedora, 32bit:
dnf install mingw32-glib2 mingw32-cairo mingw32-freetype
- Fedora, 64bit:
dnf install mingw64-glib2 mingw64-cairo mingw64-freetype
Or you could simply pull them all in one go with:
- Fedora, 32bit:
dnf install mingw32-harfbuzz
- Fedora, 64bit:
dnf install mingw64-harfbuzz
If you cannot find these packages for your distribution, or you are on macOS, you can skip to the next step, as meson will automatically download and build the dependencies for you.
- If you are familiar with
meson
, you can use the cross-compile files we provide to find your way around. Read until the end of this section before deciding which one to use.
- 32bit:
meson --cross-file=.ci/win32-cross-file.txt build-win -Dglib-enabled -Dcairo=enabled -Dgdi=enabled -Ddirectwrite=enabled
- 64bit:
meson --cross-file=.ci/win64-cross-file.txt build-win -Dglib-enabled -Dcairo=enabled -Dgdi=enabled -Ddirectwrite=enabled
In which case, you will proceed to run ninja
as usual to build:
ninja -C build-win
Or you can simply invoke the scripts we provide for our Continuous Integration system, to configure and build HarfBuzz for you. This is the easiest way to build HarfBuzz for Windows and how we build our Windows binaries:
- 32bit:
./.ci/build-win.sh 32 && ln -s build-win32 build-win
- 64bit:
./.ci/build-win.sh 64 && ln -s build-win64 build-win
This might take a while, since, if you do not have the dependencies installed, meson will download and build them for you.
-
If everything succeeds, you should have the
hb-shape.exe
,hb-view.exe
,hb-subset.exe
, andhb-info.exe
executables inbuild-win/util
. -
Configure your wine to find system mingw libraries. While there, set it also to find the built HarfBuzz DLLs:
- Fedora, 32bit:
export WINEPATH="$HOME/harfbuzz/build-win/src;/usr/i686-w64-mingw32/sys-root/mingw/bin"
- Fedora, 64bit:
export WINEPATH="$HOME/harfbuzz/build-win/src;/usr/x86_64-w64-mingw32/sys-root/mingw/bin"
- Other systems:
export WINEPATH="$HOME/harfbuzz/build-win/src"
Adjust for the path where you have built HarfBuzz. You might want to add this
to your .bashrc
or .zshrc
file.
- Run the
hb-shape
executable under Wine:
wine build-win/util/hb-shape.exe perf/fonts/Roboto-Regular.ttf Test
You probably will get lots of Wine warnings, but if all works fine, you should see:
[gid57=0+1123|gid74=1+1086|gid88=2+1057|gid89=3+670]
You can make Wine less verbose, without hiding all errors, by setting:
export WINEDEBUG=fixme-all,warn-all,err-plugplay,err-seh,err-rpc,err-ntoskrnl,err-winediag,err-systray,err-hid
Add this to your .bashrc
or .zshrc
file as well.
Next, let's try some non-Latin text. Unfortunately, the command-line parsing of our cross-compiled glib is not quite Unicode-aware, at least when run under Wine. So you will need to find some other way to feed Unicode text to the shaper. There are three different ways you can try:
echo حرف | wine build-win/util/hb-shape.exe perf/fonts/Amiri-Regular.ttf
wine build-win/util/hb-shape.exe perf/fonts/Amiri-Regular.ttf -u 062D,0631,0641
wine build-win/util/hb-shape.exe perf/fonts/Amiri-Regular.ttf --text-file harf.txt
To get the Unicode codepoints for a string, you can use the hb-unicode-decode
utility:
$ test/shape/hb-unicode-decode حرف
U+062D,U+0631,U+0641
- Next, let's try the
hb-view
utility. By default,hb-view
outputs ANSI text, which Wine will not display correctly. You can use the-o
option to redirect the output to a file, or just redirect the output using the shell, which will produce a PNG file.
wine build-win/util/hb-view.exe perf/fonts/Roboto-Regular.ttf Test > test.png
- If you are on Linux and your Wine has configured itself as the processor for
the EXE files (called
binfmt
in Linux), you can run the executables directly:
build-win/util/hb-shape.exe perf/fonts/Roboto-Regular.ttf Test
If that does not work, you can use the wine
command as shown above.
- You can try running the test suite. If on Linux with
binfmt
enabled, you can run the tests directly:
ninja -C build-win test
If not, you might need to reconfigure the build directory to tell meson to use
Wine to run the tests. This part needs improvement, but for now: edit the
cross-file you used to build HarfBuzz and uncomment the exe_wrapper
line:
- 32bit:
.ci/win32-cross-file.txt
- 64bit:
.ci/win64-cross-file.txt
Then go back to step 4 above and rebuild HarfBuzz with the exe_wrapper
set.
If all goes well, tests should run with the above ninja
command. If all is
well, you should probably see over 300 tests pass, some skipped, and just the
directwrite
test fail.
The reason the directwrite
test fails is that we are running against the
Wine-provided DirectWrite DLL, which is an incomplete reimplementation of the
DirectWrite API by Wine, and not the real thing. If you want to test the
Uniscribe or DirectWrite shapers against the real Uniscribe / DirectWrite, you
can follow the instructions below.
- Uniscribe: Assuming a 32bit build for now. Bring a 32bit version of
usp10.dll
for yourself fromC:\Windows\SysWOW64\usp10.dll
of your Windows installation (assuming you have a 64-bit installation, otherwiseC:\Windows\System32\usp10.dll
). You want one from Windows 7 or earlier. One that is not just a proxy fortextshaping.dll
. Rule of thumb, yourusp10.dll
should have a size more than 500kb. Put the file inbuild-win/src
.
You can now tell wine to use the native usp10.dll
:
export WINEDLLOVERRIDES="usp10=n"
wine build-win/util/hb-shape.exe perf/fonts/Roboto-Regular.ttf Test --shaper=uniscribe
- DirectWrite: You can use the same method to test the DirectWrite shaper
against the native DirectWrite DLL. Try with a 64bit build this time. Bring
textshaping.dll
,dwrite.dll
, andusp10.dll
from your Windows installation (C:\Windows\System32
) tobuild-win/src
. You can now tell wine to use the native DirectWrite:
export WINEDLLOVERRIDES="textshaping,dwrite,usp10=n"
wine build-win/util/hb-shape.exe perf/fonts/Roboto-Regular.ttf Test --shaper=directwrite
If all works well, you should be able to rerun the tests and see all pass this time.
-
For some old instructions on how to test HarfBuzz's native Indic shaper against Uniscribe, see: https://github.com/harfbuzz/harfbuzz/issues/3671
-
That's it! If you made it this far, you are now able to develop and test HarfBuzz on Windows, from Linux or macOS. Enjoy!