Skip to content

ci/gha/meson: use various library types for Windows builds #44

ci/gha/meson: use various library types for Windows builds

ci/gha/meson: use various library types for Windows builds #44

Workflow file for this run

name: GitHub CI
on:
push:
branches: [master, ci, coverity_scan]
pull_request:
jobs:
build:
runs-on: ${{ matrix.os }}
name: build(${{ matrix.msystem || matrix.docker_image || matrix.os }},
${{ matrix.cc }}${{ matrix.api && ', ' }}${{ matrix.api }}${{ matrix.jobname_suffix && ', ' }}${{ matrix.jobname_suffix }})
strategy:
fail-fast: false
matrix:
include:
# Enable distcheck for one build
- os: ubuntu-latest
cc: gcc
do_distc: yes
# Run Coverity on a clang build; Coverity's gcc causes issues
- os: ubuntu-latest
cc: clang
do_coverity: yes
# Add a tcc build
- os: ubuntu-latest
cc: tcc
ld: tcc
# Add docker-build on Alpine
- os: ubuntu-latest
cc: gcc
docker_image: alpine:latest
shell: '/tmp/docker_shell {0}'
art_reg_skip: 'font_nonunicode'
# Add docker-build with minimum version of dependencies
- os: ubuntu-latest
cc: gcc
docker_image: oldlibs
docker_pullprefix: 'ghcr.io/theoneric/libass-containers/'
shell: '/tmp/docker_shell {0}'
# Crash Tests detect (false?) leaks in Fontconfig, and
# various regression test fail, assumed due to older deps
skip_tests: yes
# MacOS Intel
- os: macos-13
cc: clang
# MacOS arm64
- os: macos-14
cc: clang
omit_asan: no
- os: macos-latest
cc: clang
omit_asan: yes
art_binwrap: 'leaks --atExit --'
jobname_suffix: leaks
# Add a Windows build (MinGW-gcc via MSYS2) with no extras
- os: windows-2019
msystem: MINGW32
cc: gcc
api: desktop
shell: 'msys2 {0}'
# Add a best-effort build for UWP apps for Microsoft Store
- os: windows-2019
msystem: UCRT64
cc: gcc
api: app
extra_cflags: -DWINAPI_FAMILY=WINAPI_FAMILY_APP -specs=/tmp/windowsapp.specs
shell: 'msys2 {0}'
defaults:
run:
shell: ${{ matrix.shell || 'bash' }}
env:
ART_SAMPLES: ext_art-samples
ART_BINWRAP: ${{ matrix.art_binwrap }}
steps:
# Available core count changes on runner type and over time, but GHA
# doesn’t easily expose this, so we’ll have to query this ourselves. Based on:
# https://github.com/orgs/community/discussions/25057#discussioncomment-5957241
- name: determine logical cpu core count
id: cpus
shell: python
run: |
import os
import multiprocessing
try:
num_cpus = len(os.sched_getaffinity(0))
except AttributeError:
num_cpus = multiprocessing.cpu_count()
print(f"num_cpus={num_cpus}")
output_file = os.environ["GITHUB_OUTPUT"]
with open(output_file, "a", encoding="utf-8") as output_stream:
output_stream.write(f"count={num_cpus}\n")
- name: checkout code
uses: actions/checkout@v4
- name: download test samples
uses: actions/checkout@v4
with:
repository: libass/libass-tests
path: ${{ env.ART_SAMPLES }}
- name: Start Docker
if: matrix.docker_image
shell: bash
run: |
# Note: Many containers default to the root user
docker pull "${{ matrix.docker_pullprefix }}${{ matrix.docker_image }}"
docker create --name dockerciimage \
-v "/home/runner/work:/home/runner/work" --workdir "$PWD" \
--entrypoint "tail" \
"${{ matrix.docker_pullprefix }}${{ matrix.docker_image }}" \
"-f" "/dev/null"
docker start dockerciimage
# Create a proxy-shell for Docker containers
# Scripts for each step and the output file are inside the mounted
# directories, but some environment variable must be forwarded.
echo '#!/bin/sh
set -eu
if [ "$#" -ne 1 ] ; then
echo "Usage: $0 <script file>"
exit 1
fi
exec /usr/bin/docker exec \
--env GITHUB_OUTPUT --env GITHUB_ENV --env GITHUB_PATH --env GITHUB_STATE \
dockerciimage sh -e "$1"
' > /tmp/docker_shell
chmod a+x /tmp/docker_shell
- name: Setup MSys2
uses: msys2/setup-msys2@v2
if: matrix.msystem
with:
msystem: ${{ matrix.msystem }}
update: false
- name: install deps
run: |
case "${{ matrix.docker_image || matrix.os }}" in
macos-*)
#brew update
brew install autoconf automake libtool nasm pkg-config \
harfbuzz freetype fribidi fontconfig
;;
windows-*)
pre="$MINGW_PACKAGE_PREFIX"
pacman --noconfirm -S \
automake autoconf libtool nasm make \
$pre-pkg-config $pre-gcc \
$pre-fribidi $pre-freetype $pre-harfbuzz $pre-fontconfig \
$pre-libpng
;;
alpine:*)
apk add nasm ${{ matrix.cc }} musl-dev \
make automake autoconf libtool pkgconf \
fontconfig-dev freetype-dev fribidi-dev harfbuzz-dev \
libpng-dev
;;
oldlibs)
: # Everything is preinstalled
;;
*)
sudo apt-get update #&& sudo apt-get upgrade
ubver="$(apt-cache search libubsan | awk '/^libubsan[0-9]* / {print substr($1, 9)}' | sort -rn | head -n1)"
asver="$(apt-cache search libasan | awk '/^libasan[0-9]* / {print substr($1, 8)}' | sort -rn | head -n1)"
sudo apt-get install -y --no-install-recommends \
autoconf automake make libtool \
libfontconfig1-dev libfreetype6-dev libfribidi-dev \
libharfbuzz-dev nasm ${{ matrix.cc }} \
libpng-dev libubsan"$ubver" libasan"$asver"
;;
esac
- name: Determine Sanitizer Flags
id: sanitizer
run: |
aflags="-fsanitize=address"
uflags="-fsanitize=undefined -fsanitize=float-cast-overflow"
if [ "${{ startsWith(matrix.cc, 'clang') }}" = "true" ] ; then
# Clang's UBSAN exits with code zero even if issues were found
# This options will instead force an SIGILL, but remove a report being printed
# see https://reviews.llvm.org/D35085
uflags="$uflags -fsanitize-undefined-trap-on-error"
fi
# ASAN is incompatible with other tools so some jobs want to opt-out
if [ "${{ matrix.omit_asan }}" = "yes" ] ; then
aflags=""
fi
# UBSAN: Alpine and MSys2 doesn't ship the UBSAN library,
# but when SIGILL'ing the lib is not needed
# ASAN: Not supported with musl and in Windows
case "${{ matrix.docker_image || matrix.os }}" in
alpine*|windows-*)
flags="$uflags -fsanitize-undefined-trap-on-error" ;;
*)
flags="$aflags $uflags" ;;
esac
if [ -n "$flags" ] ; then
flags="$flags -fno-sanitize-recover=all"
fi
if [ "${{ matrix.cc }}" = "tcc" ] || [ "${{ matrix.skip_tests }}" = "yes" ] ; then
flags=""
fi
echo "SANFLAGS=$flags"
echo "flags=${flags}" >> $GITHUB_OUTPUT
- name: Customize compiler
if: matrix.api == 'app' && matrix.cc == 'gcc'
run: >
gcc -dumpspecs
| sed 's/-lmsvcrt/-lucrtapp/g; s/-lkernel32/-lwindowsapp/g;
s/-ladvapi32//g; s/-lshell32//g; s/-luser32//g'
> /tmp/windowsapp.specs
- name: configure
run: |
export CC="${{ matrix.cc }}\
-DDEBUG_LEVEL=3\
${{ matrix.extra_cflags && ' ' }}${{ matrix.extra_cflags }}\
${{ steps.sanitizer.outputs.flags && ' ' }}${{ steps.sanitizer.outputs.flags }}"
export LD="${{ matrix.ld }}"
export ART_SAMPLES="${{ env.ART_SAMPLES }}"
./autogen.sh
./configure --enable-compare --enable-fuzz
- name: distcheck
if: matrix.do_distc == 'yes'
run: make -j '${{ steps.cpus.outputs.count }}' distcheck
- name: compile
run: make -j '${{ steps.cpus.outputs.count }}'
- name: ensure internal functions are namespaced
if: startsWith(matrix.os, 'ubuntu-')
run: |
test -f libass/.libs/libass.a || (echo "Static lib is missing!"; exit 1)
set +e
list="$(nm libass/.libs/libass.a | grep ' T ' | grep -v ' ass_')"
case "$?" in
1)
: # All good
;;
0)
echo "There are non-namespaced functions! Prefix them with 'ass_'."
echo "$list"
exit 1
;;
*)
echo "Internal grep error occured!"
echo "$list"
exit 2
;;
esac
- name: run tests
if: matrix.skip_tests != 'yes'
run: |
ART_REG_SKIP="${{ matrix.art_reg_skip }}" make -j 1 check \
|| (cat test-suite.log; exit 1;)
- name: Coverity scan
if: >
matrix.do_coverity == 'yes'
&& github.repository == 'libass/libass'
&& github.event_name != 'pull_request'
env:
COVERITY_SCAN_TOKEN: ${{ secrets.COVERITY_SCAN_TOKEN }}
PROJECT_NAME: libass/libass
NOTIFY_EMAIL: none@example.com
TOOL_URL: https://scan.coverity.com/download/
UPLOAD_URL: https://scan.coverity.com/builds?project=libass%2Flibass
SCAN_URL: https://scan.coverity.com
RES_DIR: cov-int
run: |
exit_code=0
echo "Running Coverity ..."
# Remove previous build output
make clean
# The upstream script is borked and always exits with 1 even on success
# To get meaningful success/error status we're using our own script
# but we still want to be informed about upstream script changes
if curl -s https://scan.coverity.com/scripts/travisci_build_coverity_scan.sh \
| shasum -a 256 \
| grep -Eq '^234d71b4a5257a79559e66dd3ba5765576d2af4845da83af4975b77b14ab536b '
then
: remote unchanged
else
echo "Coverity's travis script changed!"
exit_code=1
fi
# Check if we are within quoata
quota_res="$(curl -s --form project="$PROJECT_NAME" \
--form token="$COVERITY_SCAN_TOKEN" \
"$SCAN_URL"/api/upload_permitted)"
if [ "$?" -ne 0 ] || [ "x$quota_res" = "xAccess denied" ] ; then
echo "Coverity denied access or did not respond!"
exit 1
elif echo "$quota_res" | grep -Eq 'upload_permitted": *true' ; then
echo "Within Coverity quota."
else
echo "Exceeding Coverity quota! Try again later."
echo "$quota_res" | grep -Eo 'next_upload_permitted_at":[^,}]*'
exit 0
fi
# Download cov tool and make it available
wget -nv "$TOOL_URL""$(uname)" \
--post-data "project=$PROJECT_NAME&token=$COVERITY_SCAN_TOKEN" \
-O cov-analysis-tool.tar.gz
mkdir cov-analysis-tool
tar xzf cov-analysis-tool.tar.gz --strip 1 -C cov-analysis-tool
export PATH="$(pwd)/cov-analysis-tool/bin:$PATH"
# Coverity Build
echo "Starting Coverity build..."
#mkdir "$RES_DIR" # already done by cov-build
COVERITY_UNSUPPORTED=1 cov-build --dir "$RES_DIR" make -j '${{ steps.cpus.outputs.count }}'
cov-import-scm --dir "$RES_DIR" --scm git --log "$RES_DIR/scm_log.txt" 2>&1
# Submit results to Coverity's server
tar czf libass.tar.gz "$RES_DIR"
upstat="$(curl --silent --write-out "\n%{http_code}\n" \
--form project="PROJECT_NAME" \
--form token="$COVERITY_SCAN_TOKEN" \
--form email="$NOTIFY_EMAIL" \
--form file=@libass.tar.gz \
--form version="${{ github.sha }}" \
--form description="GitHubActions CI build" \
"$UPLOAD_URL")"
if [ "$?" -ne 0 ] ; then
echo "Upload failed (curl error)"
exit_code=1
elif echo "$upstat" | tail -n 1 | grep -Eq '^2[0-9]{2}$' ; then
echo "Upload successful."
else
echo "Upload failed (server error)"
exit_code=1
fi
echo "$upstat" | head
exit $exit_code
- name: Stop Docker
if: matrix.docker_image
shell: bash
run: |
docker rm --force dockerciimage