From cf19f8287547c523f65879b9e9477950c81a9eed Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?P=C5=99emysl=20Eric=20Janouch?= Date: Wed, 10 Aug 2022 18:11:05 +0200 Subject: Automate Windows builds, add icons to executables Scripts have been ported from sdtui, and adjusted for Meson. The port is broken through and through on WINE, but sort-of works natively. --- README.adoc | 13 ++++++ fiv.manifest | 11 +++++ fiv.rc | 3 ++ meson.build | 47 +++++++++++++++++--- msys2-cross-configure.sh | 111 +++++++++++++++++++++++++++++++++++++++++++++++ msys2-cross-install.sh | 65 +++++++++++++++++++++++++++ 6 files changed, 245 insertions(+), 5 deletions(-) create mode 100644 fiv.manifest create mode 100644 fiv.rc create mode 100755 msys2-cross-configure.sh create mode 100755 msys2-cross-install.sh diff --git a/README.adoc b/README.adoc index 13d92ba..031b110 100644 --- a/README.adoc +++ b/README.adoc @@ -56,6 +56,19 @@ direct installations via `ninja install`. To test the program: The lossless JPEG cropper is intended to be invoked from a context menu. +Windows +~~~~~~~ +'fiv' can be cross-compiled for Windows, provided that you install a bunch of +dependencies listed at the beginning of 'msys2-cross-configure.sh', +plus rsvg-convert from librsvg2, and icotool from icoutils. +Beware that the build will take up close to a gigabyte of disk space. + + $ sh msys2-cross-configure.sh builddir + $ meson install -C builddir + +If everything succeeds, you will find a portable build of the application +in the 'builddir/package' subdirectory. Keep your expectations low. + Documentation ------------- For information concerning usage, refer to link:docs/fiv.html[the user guide], diff --git a/fiv.manifest b/fiv.manifest new file mode 100644 index 0000000..6009f3c --- /dev/null +++ b/fiv.manifest @@ -0,0 +1,11 @@ + + + + + + + + + diff --git a/fiv.rc b/fiv.rc new file mode 100644 index 0000000..7e029f5 --- /dev/null +++ b/fiv.rc @@ -0,0 +1,3 @@ +#include +LD_ICON ICON "fiv.ico" +CREATEPROCESS_MANIFEST_RESOURCE_ID RT_MANIFEST "fiv.manifest" diff --git a/meson.build b/meson.build index 9c40889..d646a2a 100644 --- a/meson.build +++ b/meson.build @@ -91,8 +91,32 @@ configure_file( configuration : conf, ) +rc = [] +if host_machine.system() == 'windows' + windows = import('windows') + rsvg_convert = find_program('rsvg-convert') + icotool = find_program('icotool') + + # Meson is brain-dead and retarded, so these PNGs cannot be installed, + # only because they must all have the same name when installed. + # The largest size is mainly for an appropriately sized Windows icon. + icon_png_list = [] + foreach size : ['16', '32', '48', '256'] + icon_dimensions = size + 'x' + size + icon_png_list += custom_target(icon_dimensions + ' icon', + input : 'fiv.svg', + output : 'fiv.' + icon_dimensions + '.png', + command : [rsvg_convert, '--output', '@OUTPUT@', + '--width', size, '--height', size, '@INPUT@']) + endforeach + + icon_ico = custom_target(input : icon_png_list, output : 'fiv.ico', + command : [icotool, '-c', '-o', '@OUTPUT@', '@INPUT@']) + rc += windows.compile_resources('fiv.rc', depends : icon_ico) +endif + gnome = import('gnome') -resources = gnome.compile_resources('resources', +gresources = gnome.compile_resources('resources', 'resources/resources.gresource.xml', source_dir : 'resources', c_name : 'resources', @@ -108,9 +132,11 @@ tiff_tables = custom_target('tiff-tables.h', desktops = ['fiv.desktop', 'fiv-browse.desktop'] exe = executable('fiv', 'fiv.c', 'fiv-view.c', 'fiv-io.c', 'fiv-context-menu.c', 'fiv-browser.c', 'fiv-sidebar.c', 'fiv-thumbnail.c', 'fiv-collection.c', - 'xdg.c', resources, + 'xdg.c', gresources, rc, install : true, - dependencies : dependencies) + dependencies : dependencies, + win_subsystem : 'windows', +) if gdkpixbuf.found() executable('io-benchmark', 'fiv-io-benchmark.c', 'fiv-io.c', 'xdg.c', build_by_default : false, @@ -118,12 +144,14 @@ if gdkpixbuf.found() endif desktops += 'fiv-jpegcrop.desktop' -jpegcrop = executable('fiv-jpegcrop', 'fiv-jpegcrop.c', +jpegcrop = executable('fiv-jpegcrop', 'fiv-jpegcrop.c', rc, install : true, dependencies : [ dependency('gtk+-3.0'), dependency('libturbojpeg'), - ]) + ], + win_subsystem : 'windows', +) if get_option('tools').enabled() # libjq 1.6 lacks a pkg-config file, and there is no release in sight. @@ -170,6 +198,15 @@ if host_machine.system() != 'windows' if not meson.is_cross_build() meson.add_install_script(updater, skip_if_destdir : dynamic_desktops) endif +elif meson.is_cross_build() + msys2_root = meson.get_external_property('msys2_root') + meson.add_install_script('msys2-cross-install.sh', msys2_root) + + # This is the minimum to run targets from msys2-cross-configure.sh builds. + meson.add_devenv({ + 'WINEPATH' : msys2_root / 'bin', + 'XDG_DATA_DIRS' : msys2_root / 'share', + }) endif install_data('fiv.svg', diff --git a/msys2-cross-configure.sh b/msys2-cross-configure.sh new file mode 100755 index 0000000..602548e --- /dev/null +++ b/msys2-cross-configure.sh @@ -0,0 +1,111 @@ +#!/bin/sh -e +# msys2-cross-configure.sh: set up an MSYS2-based cross-compiled Meson build. +# Dependencies: AWK, sed, sha256sum, cURL, bsdtar, +# wine64, Meson, mingw-w64-binutils, mingw-w64-gcc, pkg-config +repository=https://repo.msys2.org/mingw/mingw64/ + +status() { + echo "$(tput bold)-- $*$(tput sgr0)" +} + +dbsync() { + status Fetching repository DB + [ -f db.tsv ] || curl -# "$repository/mingw64.db" | bsdtar -xOf- | awk ' + function flush() { print f["%NAME%"] f["%FILENAME%"] f["%DEPENDS%"] } + NR > 1 && $0 == "%FILENAME%" { flush(); for (i in f) delete f[i] } + !/^[^%]/ { field = $0; next } { f[field] = f[field] $0 "\t" } + field == "%SHA256SUM%" { path = "*packages/" f["%FILENAME%"] + sub(/\t$/, "", path); print $0, path > "db.sums" } END { flush() } + ' > db.tsv +} + +fetch() { + status Resolving "$@" + mkdir -p packages + awk -F'\t' 'function get(name, i, a) { + if (visited[name]++ || !(name in filenames)) return + print filenames[name]; split(deps[name], a); for (i in a) get(a[i]) + } BEGIN { while ((getline < "db.tsv") > 0) { + filenames[$1] = $2; deps[$1] = ""; for (i = 3; i <= NF; i++) { + gsub(/[<=>].*/, "", $i); deps[$1] = deps[$1] $i FS } + } for (i = 0; i < ARGC; i++) get(ARGV[i]) }' "$@" | while IFS= read -r name + do + status Fetching "$name" + [ -f "packages/$name" ] || curl -#o "packages/$name" "$repository/$name" + done +} + +verify() { + status Verifying checksums + sha256sum --ignore-missing --quiet -c db.sums +} + +extract() { + status Extracting packages + for subdir in * + do [ -d "$subdir" -a "$subdir" != packages ] && rm -rf -- "$subdir" + done + for i in packages/* + do bsdtar -xf "$i" --strip-components 1 mingw64 + done +} + +configure() { + # Don't let GLib programs inherit wrong paths from the environment. + export XDG_DATA_DIRS=$msys2_root/share + + status Configuring packages + wine64 bin/update-mime-database.exe share/mime + wine64 bin/glib-compile-schemas.exe share/glib-2.0/schemas + wine64 bin/gdk-pixbuf-query-loaders.exe \ + > lib/gdk-pixbuf-2.0/2.10.0/loaders.cache +} + +setup() { + status Setting up Meson + cat >"$toolchain" <<-EOF + [binaries] + c = 'x86_64-w64-mingw32-gcc' + cpp = 'x86_64-w64-mingw32-g++' + ar = 'x86_64-w64-mingw32-gcc-ar' + ranlib = 'x86_64-w64-mingw32-gcc-ranlib' + strip = 'x86_64-w64-mingw32-strip' + windres = 'x86_64-w64-mingw32-windres' + pkgconfig = 'pkg-config' + + [properties] + sys_root = '$builddir' + msys2_root = '$msys2_root' + pkg_config_libdir = '$msys2_root/share/pkgconfig:$msys2_root/lib/pkgconfig' + needs_exe_wrapper = true + + [host_machine] + system = 'windows' + cpu_family = 'x86_64' + cpu = 'x86_64' + endian = 'little' + EOF + + meson --buildtype=debugoptimized --prefix="$packagedir" \ + --bindir . --libdir . --cross-file="$toolchain" "$builddir" "$sourcedir" +} + +sourcedir=$(realpath "${2:-$(dirname "$0")}") +builddir=$(realpath "${1:-builddir}") +packagedir=$builddir/package +toolchain=$builddir/msys2-cross-toolchain.meson + +# This directory name matches the prefix in .pc files, so we don't need to +# modify them (pkgconf has --prefix-variable, but Meson can't pass that option). +msys2_root=$builddir/mingw64 + +mkdir -p "$msys2_root" +cd "$msys2_root" +dbsync +fetch mingw-w64-x86_64-gtk3 mingw-w64-x86_64-lcms2 \ + mingw-w64-x86_64-libraw mingw-w64-x86_64-libheif \ + mingw-w64-x86_64-libwinpthread-git # Because we don't do "provides"? +verify +extract +configure +setup diff --git a/msys2-cross-install.sh b/msys2-cross-install.sh new file mode 100755 index 0000000..f349e86 --- /dev/null +++ b/msys2-cross-install.sh @@ -0,0 +1,65 @@ +#!/bin/sh -e +export LC_ALL=C +cd "$MESON_INSTALL_DESTDIR_PREFIX" +msys2_root=$1 + +# Copy libraries we depend on, and a few files required by GTK+. +cp -p "$msys2_root"/bin/*.dll . +cp -pR "$msys2_root"/etc/ . + +mkdir -p lib +cp -pR "$msys2_root"/lib/gdk-pixbuf-2.0/ lib +mkdir -p share/glib-2.0 +cp -pR "$msys2_root"/share/glib-2.0/schemas/ share/glib-2.0 +mkdir -p share +cp -pR "$msys2_root"/share/mime/ share +mkdir -p share/icons +cp -pR "$msys2_root"/share/icons/Adwaita/ share/icons +mkdir -p share/icons/hicolor +cp -p "$msys2_root"/share/icons/hicolor/index.theme share/icons/hicolor + +# Remove unreferenced libraries. +find lib -name '*.a' -exec rm -- {} + +awk 'function whitelist(binary) { + if (seen[binary]++) + return + + delete orphans[binary] + while (("strings -a \"" binary "\" 2>/dev/null" | getline string) > 0) + if (match(string, /[-.+_a-zA-Z0-9]+[.][Dd][Ll][Ll]$/)) + whitelist("./" substr(string, RSTART, RLENGTH)) +} BEGIN { + while (("find . -type f -path \"./*.[Dd][Ll][Ll]\"" | getline) > 0) + orphans[$0]++ + while (("find . -type f -path \"./*.[Ee][Xx][Ee]\"" | getline) > 0) + whitelist($0) + while (("find ./lib/gdk-pixbuf-2.0 -type f " \ + "-path \"./*.[Dd][Ll][Ll]\"" | getline) > 0) + whitelist($0) + for (library in orphans) + print library +}' | xargs rm -- + +# Removes unused icons from the Adwaita theme. It could be even more aggressive, +# since it keeps around lots of sizes and all the GTK+ stock icons. +find share/icons/Adwaita -type f | awk 'BEGIN { + while (("grep -aho \"[a-z][a-z-]*\" *.dll *.exe" | getline) > 0) + good[$0] = 1 +} /[.](png|svg|cur|ani)$/ { + # Cut out the basename without extensions. + match($0, /[^\/]+$/) + base = substr($0, RSTART) + sub(/[.].+$/, "", base) + + # Try matching while cutting off suffixes. + # Disregarding the not-much-used GTK_ICON_LOOKUP_GENERIC_FALLBACK. + while (!(keep = good[base]) && + sub(/-(ltr|rtl|symbolic)$/, "", base)) {} + if (!keep) + print +}' | xargs rm -- + +wine64 "$msys2_root"/bin/glib-compile-schemas.exe share/glib-2.0/schemas + +# This may speed up program start-up a little bit. +wine64 "$msys2_root"/bin/gtk-update-icon-cache-3.0.exe share/icons/Adwaita -- cgit v1.2.3