From e62e77608f449b2cfbe5edfcc836789fb13619a7 Mon Sep 17 00:00:00 2001 From: Bindea Cristian Date: Tue, 23 Apr 2024 17:56:12 +0300 Subject: [PATCH] ci/ubuntu: add Scopy x86-64 appimage workflow Signed-off-by: Bindea Cristian --- .github/workflows/appimageubuntu.yml | 44 +++ .gitignore | 1 + CI/appimage/AppRun | 10 + CI/appimage/docker/Dockerfile | 35 +++ CI/appimage/scopy.desktop | 10 + CI/appimage/ubuntu_appimage_process.sh | 393 +++++++++++++++++++++++++ 6 files changed, 493 insertions(+) create mode 100644 .github/workflows/appimageubuntu.yml create mode 100755 CI/appimage/AppRun create mode 100644 CI/appimage/docker/Dockerfile create mode 100644 CI/appimage/scopy.desktop create mode 100755 CI/appimage/ubuntu_appimage_process.sh diff --git a/.github/workflows/appimageubuntu.yml b/.github/workflows/appimageubuntu.yml new file mode 100644 index 0000000000..15fd346200 --- /dev/null +++ b/.github/workflows/appimageubuntu.yml @@ -0,0 +1,44 @@ +name: Ubuntu Scopy Appimage Build + +on: [push, pull_request] + +env: + BUILD_HOST: ubuntu-20.04 + USERNAME: github-actions + +jobs: + + build_scopy1_ubuntu_appimage: + runs-on: ubuntu-20.04 + + steps: + - uses: actions/checkout@v4 + with: + set-safe-directory: 'true' + + - name: Pull the Docker Image + run: docker pull cristianbindea/scopy1-ubuntu-appimage:latest + + - name: Create Scopy AppImage + shell: bash + run: | + cd $GITHUB_WORKSPACE + ./CI/appimage/ubuntu_appimage_process.sh generate_ci_envs + + docker run \ + --mount type=bind,source="$GITHUB_WORKSPACE",target=/home/runner/scopy \ + --env-file $GITHUB_WORKSPACE/CI/appimage/gh-actions.envs \ + cristianbindea/scopy1-ubuntu-appimage:latest \ + /bin/bash -c 'cd $HOME && \ + sudo chown -R runner:runner scopy && \ + cd $HOME/scopy && \ + ./CI/appimage/ubuntu_appimage_process.sh download_tools build_iio-emu build_scopy create_appdir create_appimage move_appimage + ' + - name: Set short git commit SHA + shell: bash + run: echo "commit_sha=$(git rev-parse --short ${{ github.sha }})" >> "$GITHUB_ENV" + + - uses: actions/upload-artifact@v4 + with: + name: scopy-linux-armhf-${{ env.commit_sha }} + path: ${{ github.workspace }}/Scopy1-x86_64.AppImage diff --git a/.gitignore b/.gitignore index b3fa412766..8a846466b4 100644 --- a/.gitignore +++ b/.gitignore @@ -22,5 +22,6 @@ android*.sh !CI/android/* CI/kuiper/staging CI/kuiper/scopy.AppDir +CI/appimage/staging html/ build_arm64-v8a/ diff --git a/CI/appimage/AppRun b/CI/appimage/AppRun new file mode 100755 index 0000000000..9a575a5c17 --- /dev/null +++ b/CI/appimage/AppRun @@ -0,0 +1,10 @@ +#!/bin/bash +set -e +SELF=$(readlink -f "$0") +HERE=${SELF%/*} + +export QT_PLUGIN_PATH=$HERE/usr/plugins +export QT_QPA_PLATFORM_PLUGIN_PATH=$HERE/usr/plugins/platforms +export LD_LIBRARY_PATH=$HERE/usr/lib + +exec $HERE/usr/bin/scopy diff --git a/CI/appimage/docker/Dockerfile b/CI/appimage/docker/Dockerfile new file mode 100644 index 0000000000..d2518b6f70 --- /dev/null +++ b/CI/appimage/docker/Dockerfile @@ -0,0 +1,35 @@ +#docker build -t cristianbindea/scopy1-ubuntu-appimage . +#DOCKER_BUILDKIT=0 +FROM ubuntu:20.04 +SHELL ["/bin/bash", "-c"] + +ARG USER=runner +ARG DEBIAN_FRONTEND=noninteractive + +ENV TZ=Europe/Bucharest +RUN ln -snf /usr/share/zoneinfo/$TZ /etc/localtime && echo $TZ > /etc/timezone +RUN apt-get update && apt-get -y upgrade && apt-get install -y apt-utils sudo git tzdata keyboard-configuration + +RUN groupadd -g 1000 -r $USER && \ + useradd -u 1000 -g 1000 --create-home -r $USER + +#Change password +RUN echo "$USER:$USER" | chpasswd + +#Make sudo passwordless +RUN echo "${USER} ALL=(ALL) NOPASSWD:ALL" > /etc/sudoers.d/90-$USER && \ + usermod -aG sudo $USER && \ + usermod -aG plugdev $USER + +USER $USER +WORKDIR /home/${USER} +RUN git clone https://github.com/analogdevicesinc/scopy --branch appimage-scopy1 +WORKDIR /home/${USER}/scopy + +RUN ./CI/appimage/ubuntu_appimage_process.sh install_packages +RUN sudo pip3 install aqtinstall && sudo python3 -m aqt install-qt --outputdir /opt/Qt linux desktop 5.15.2 +RUN ./CI/appimage/ubuntu_appimage_process.sh clone download_tools build_deps + +# to be set at runtime CI_SCRIPT=ON +WORKDIR /home/${USER} +# RUN rm -rf scopy diff --git a/CI/appimage/scopy.desktop b/CI/appimage/scopy.desktop new file mode 100644 index 0000000000..38aaa876d6 --- /dev/null +++ b/CI/appimage/scopy.desktop @@ -0,0 +1,10 @@ +[Desktop Entry] +Version=1.0 +Icon=Scopy +Exec=scopy +Terminal=false +Type=Application +Categories=Science +Name=Scopy +GenericName=Oscilloscope +Comment=A software oscilloscope diff --git a/CI/appimage/ubuntu_appimage_process.sh b/CI/appimage/ubuntu_appimage_process.sh new file mode 100755 index 0000000000..5495c71b08 --- /dev/null +++ b/CI/appimage/ubuntu_appimage_process.sh @@ -0,0 +1,393 @@ +#!/bin/bash +set -ex + +if [ "$CI_SCRIPT" == "ON" ] + then + SRC_DIR=$GITHUB_WORKSPACE + git config --global --add safe.directory '*' + else + SRC_DIR=$(git rev-parse --show-toplevel) +fi + +## Set STAGING here: +USE_STAGING=OFF +## + +LIBIIO_VERSION=v0.25 +LIBAD9361_BRANCH=main +LIBM2K_BRANCH=main +SPDLOG_BRANCH=v1.x +VOLK_BRANCH=main +GNURADIO_BRANCH=maint-3.10 +GRSCOPY_BRANCH=3.10 +GRM2K_BRANCH=main +LIBSIGROKDECODE_BRANCH=master +QWT_BRANCH=qwt-multiaxes-updated +LIBTINYIIOD_BRANCH=master +IIOEMU_BRANCH=master + +QT_LOCATION=/opt/Qt/5.15.2/gcc_64 + +STAGING_AREA=$SRC_DIR/CI/appimage/staging +QMAKE_BIN=$QT_LOCATION/bin/qmake +CMAKE_BIN=${STAGING_AREA}/cmake/bin/cmake +JOBS=-j14 + +APP_DIR_NAME=scopy.AppDir +APP_DIR=$SRC_DIR/CI/appimage/$APP_DIR_NAME +APP_IMAGE=$SRC_DIR/CI/appimage/Scopy1-x86_64.AppImage + +if [ "$USE_STAGING" == "ON" ] + then + echo -- USING STAGING + STAGING_AREA_DEPS=$STAGING_AREA/dependencies + export LD_LIBRARY_PATH=$STAGING_AREA_DEPS/lib:$QT_LOCATION/lib:$LD_LIBRARY_PATH + CMAKE_OPTS=(\ + -DCMAKE_LIBRARY_PATH=$STAGING_AREA_DEPS \ + -DCMAKE_INSTALL_PREFIX=$STAGING_AREA_DEPS \ + -DCMAKE_PREFIX_PATH=$QT_LOCATION\;$STAGING_AREA_DEPS \ + -DCMAKE_EXE_LINKER_FLAGS=-L$STAGING_AREA_DEPS\;-L$STAGING_AREA_DEPS/lib \ + -DCMAKE_SHARED_LINKER_FLAGS=-L$STAGING_AREA_DEPS\;-L$STAGING_AREA_DEPS/lib \ + -DCMAKE_BUILD_TYPE=RelWithDebInfo \ + -DCMAKE_VERBOSE_MAKEFILE=ON \ + ) + echo -- STAGING_DIR $STAGING_AREA_DEPS + else + echo -- NO STAGING + STAGING_AREA_DEPS=/usr/local + export LD_LIBRARY_PATH=$QT_LOCATION/lib:$LD_LIBRARY_PATH: + CMAKE_OPTS=(\ + -DCMAKE_PREFIX_PATH=$QT_LOCATION \ + -DCMAKE_BUILD_TYPE=RelWithDebInfo \ + -DCMAKE_VERBOSE_MAKEFILE=ON \ + ) +fi + +CMAKE="$CMAKE_BIN ${CMAKE_OPTS[*]}" +echo -- USING CMAKE COMMAND: +echo $CMAKE +echo -- USING QT: $QT_LOCATION +echo -- USING QMAKE: $QMAKE_BIN + +clone() { + echo "#######CLONE#######" + mkdir -p $STAGING_AREA + pushd $STAGING_AREA + [ -d 'libiio' ] || git clone --recursive https://github.com/analogdevicesinc/libiio.git -b $LIBIIO_VERSION libiio + [ -d 'libad9361' ] || git clone --recursive https://github.com/analogdevicesinc/libad9361-iio.git -b $LIBAD9361_BRANCH libad9361 + [ -d 'libm2k' ] || git clone --recursive https://github.com/analogdevicesinc/libm2k.git -b $LIBM2K_BRANCH libm2k + [ -d 'spdlog' ] || git clone --recursive https://github.com/gabime/spdlog.git -b $SPDLOG_BRANCH spdlog + [ -d 'gr-scopy' ] || git clone --recursive https://github.com/analogdevicesinc/gr-scopy.git -b $GRSCOPY_BRANCH gr-scopy + [ -d 'gr-m2k' ] || git clone --recursive https://github.com/analogdevicesinc/gr-m2k.git -b $GRM2K_BRANCH gr-m2k + [ -d 'volk' ] || git clone --recursive https://github.com/gnuradio/volk.git -b $VOLK_BRANCH volk + [ -d 'gnuradio' ] || git clone --recursive https://github.com/gnuradio/gnuradio.git -b $GNURADIO_BRANCH gnuradio + [ -d 'qwt' ] || git clone --recursive https://github.com/cseci/qwt.git -b $QWT_BRANCH qwt + [ -d 'libsigrokdecode' ] || git clone --recursive https://github.com/sigrokproject/libsigrokdecode.git -b $LIBSIGROKDECODE_BRANCH libsigrokdecode + [ -d 'libtinyiiod' ] || git clone --recursive https://github.com/analogdevicesinc/libtinyiiod.git -b $LIBTINYIIOD_BRANCH libtinyiiod + popd +} + +download_tools() { + mkdir -p ${STAGING_AREA} + pushd ${STAGING_AREA} + + if [ ! -d cmake ];then + wget https://github.com/Kitware/CMake/releases/download/v3.29.0-rc2/cmake-3.29.0-rc2-linux-x86_64.tar.gz + tar -xvf cmake*.tar.gz && rm cmake*.tar.gz && mv cmake* cmake # unzip and rename + fi + + # download tools for creating the AppDir and the AppImage + if [ ! -f linuxdeploy-x86_64.AppImage ];then + wget https://github.com/linuxdeploy/linuxdeploy/releases/download/1-alpha-20231026-1/linuxdeploy-x86_64.AppImage + chmod +x linuxdeploy-x86_64.AppImage + fi + + if [ ! -f linuxdeploy-plugin-qt-x86_64.AppImage ];then + wget https://github.com/linuxdeploy/linuxdeploy-plugin-qt/releases/download/continuous/linuxdeploy-plugin-qt-x86_64.AppImage + chmod +x linuxdeploy-plugin-qt-x86_64.AppImage + fi + + if [ ! -f linuxdeploy-plugin-appimage-x86_64.AppImage ];then + wget https://github.com/linuxdeploy/linuxdeploy-plugin-appimage/releases/download/1-alpha-20230713-1/linuxdeploy-plugin-appimage-x86_64.AppImage + chmod +x linuxdeploy-plugin-appimage-x86_64.AppImage + fi + + popd +} + +build_with_cmake() { + INSTALL=$1 + [ -z $INSTALL ] && INSTALL=ON + BUILD_FOLDER=$PWD/build + rm -rf $BUILD_FOLDER + mkdir -p $BUILD_FOLDER + cd $BUILD_FOLDER + $CMAKE $CURRENT_BUILD_CMAKE_OPTS ../ + make $JOBS + if [ "$INSTALL" == "ON" ];then + if [ "$USE_STAGING" == "ON" ]; then make install; else sudo make install; fi + fi + CURRENT_BUILD_CMAKE_OPTS="" +} + +install_packages() { + sudo DEBIAN_FRONTEND=noninteractive apt-get -y install + + sudo apt-get update + sudo apt-get -y upgrade + sudo apt-get -y install \ + python3 python-dev python3-pip libpython3-all-dev python3-numpy \ + keyboard-configuration vim git wget unzip\ + g++ build-essential cmake curl autogen autoconf autoconf-archive pkg-config flex bison swig \ + subversion mesa-common-dev graphviz xserver-xorg gettext texinfo mm-common doxygen \ + libboost-all-dev libfftw3-bin libfftw3-dev libfftw3-3 liblog4cpp5v5 liblog4cpp5-dev \ + libxcb-xinerama0 libgmp3-dev libzip-dev libglib2.0-dev libglibmm-2.4-dev libsigc++-2.0-dev \ + libclang1-9 libmatio-dev liborc-0.4-dev libgl1-mesa-dev libserialport0 libserialport-dev \ + libusb-1.0 libusb-1.0-0 libusb-1.0-0-dev libavahi-client-dev libsndfile1-dev \ + libxkbcommon-x11-0 libqt5gui5 libncurses5 libtool libaio-dev libzmq3-dev libxml2-dev + + pip3 install mako + pip3 install packaging +} + +build_libiio() { + echo "### Building libiio - version $LIBIIO_VERSION" + pushd $STAGING_AREA/libiio + CURRENT_BUILD_CMAKE_OPTS="\ + -DWITH_TESTS:BOOL=OFF \ + -DWITH_DOC:BOOL=OFF \ + -DHAVE_DNS_SD:BOOL=OFF\ + -DWITH_MATLAB_BINDINGS:BOOL=OFF \ + -DCSHARP_BINDINGS:BOOL=OFF \ + -DPYTHON_BINDINGS:BOOL=OFF \ + -DWITH_SERIAL_BACKEND:BOOL=ON \ + -DENABLE_IPV6:BOOL=OFF \ + -DINSTALL_UDEV_RULE:BOOL=OFF + " + build_with_cmake $1 + popd +} + +build_libad9361() { + echo "### Building libad9361 - branch $LIBAD9361_BRANCH" + pushd $STAGING_AREA/libad9361 + build_with_cmake $1 + popd +} + +build_spdlog() { + echo "### Building spdlog - branch $SPDLOG_BRANCH" + pushd $STAGING_AREA/spdlog + CURRENT_BUILD_CMAKE_OPTS="-DSPDLOG_BUILD_SHARED=ON" + build_with_cmake $1 + popd +} + +build_libm2k() { + echo "### Building libm2k - branch $LIBM2K_BRANCH" + pushd $STAGING_AREA/libm2k + CURRENT_BUILD_CMAKE_OPTS="\ + -DENABLE_PYTHON=OFF \ + -DENABLE_CSHARP=OFF \ + -DBUILD_EXAMPLES=OFF \ + -DENABLE_TOOLS=OFF \ + -DINSTALL_UDEV_RULES=OFF \ + " + build_with_cmake $1 + popd +} + +build_volk() { + echo "### Building volk - branch $VOLK_BRANCH" + pushd $STAGING_AREA/volk + CURRENT_BUILD_CMAKE_OPTS="-DPYTHON_EXECUTABLE=/usr/bin/python3" + build_with_cmake $1 + popd +} + +build_gnuradio() { + echo "### Building gnuradio - branch $GNURADIO_BRANCH" + pushd $STAGING_AREA/gnuradio + CURRENT_BUILD_CMAKE_OPTS="\ + -DPYTHON_EXECUTABLE=/usr/bin/python3 \ + -DENABLE_DEFAULT=OFF \ + -DENABLE_GNURADIO_RUNTIME=ON \ + -DENABLE_GR_ANALOG=ON \ + -DENABLE_GR_BLOCKS=ON \ + -DENABLE_GR_FFT=ON \ + -DENABLE_GR_FILTER=ON \ + -DENABLE_GR_IIO=ON \ + -DENABLE_POSTINSTALL=OFF + " + build_with_cmake $1 + popd +} + +build_grscopy() { + echo "### Building gr-scopy - branch $GRSCOPY_BRANCH" + pushd $STAGING_AREA/gr-scopy + build_with_cmake $1 + popd +} + +build_grm2k() { + echo "### Building gr-m2k - branch $GRM2K_BRANCH" + pushd $STAGING_AREA/gr-m2k + CURRENT_BUILD_CMAKE_OPTS="\ + -DENABLE_PYTHON=OFF \ + -DDIGITAL=OFF + " + build_with_cmake $1 + popd +} + +build_qwt() { + echo "### Building qwt - branch $QWT_BRANCH" + pushd $STAGING_AREA/qwt + git clean -xdf + sed -i 's|/usr/local/qwt-$$QWT_VERSION-ma|/usr/local|g' qwtconfig.pri + if [ "$USE_STAGING" == "ON" ] + then + $QMAKE_BIN INCLUDEPATH=$STAGING_AREA_DEPS/include LIBS=-L$STAGING_AREA_DEPS/lib qwt.pro + make $JOBS + make INSTALL_ROOT=$STAGING_AREA_DEPS install + cp -r $STAGING_AREA_DEPS/usr/local/* $STAGING_AREA_DEPS/ + else + $QMAKE_BIN qwt.pro + make $JOBS + sudo make install + fi + + popd +} + +build_libsigrokdecode() { + echo "### Building libsigrokdecode - branch $LIBSIGROKDECODE_BRANCH" + pushd $STAGING_AREA/libsigrokdecode + git clean -xdf + ./autogen.sh + if [ "$USE_STAGING" == "ON" ] + then + ./configure --prefix $STAGING_AREA_DEPS + else + ./configure + fi + make $JOBS + if [ "$USE_STAGING" == "ON" ]; then make install; else sudo make install; fi + popd +} + +build_libtinyiiod() { + echo "### Building libtinyiiod - branch $LIBTINYIIOD_BRANCH" + pushd $STAGING_AREA/libtinyiiod + CURRENT_BUILD_CMAKE_OPTS="-DBUILD_EXAMPLES=OFF" + build_with_cmake $1 + popd +} + +build_iio-emu() { + echo "### Building iio-emu - branch $IIOEMU_BRANCH" + mkdir -p $STAGING_AREA + pushd $STAGING_AREA + [ -d 'iio-emu' ] || git clone --recursive https://github.com/analogdevicesinc/iio-emu -b $IIOEMU_BRANCH iio-emu + pushd $STAGING_AREA/iio-emu + build_with_cmake OFF + popd + popd +} + +build_scopy() { + echo "### Building scopy" + pushd $SRC_DIR + CURRENT_BUILD_CMAKE_OPTS="\ + -DCLONE_IIO_EMU=OFF + " + build_with_cmake OFF + popd +} + +build_deps(){ + build_libiio ON + build_libad9361 ON + build_spdlog ON + build_libm2k ON + build_volk ON + build_gnuradio ON + build_grscopy ON + build_grm2k ON + build_qwt + build_libsigrokdecode + build_libtinyiiod ON +} + +create_appdir(){ + pushd ${STAGING_AREA} + LD_LIBRARY_PATH=$LD_LIBRARY_PATH:$STAGING_AREA_DEPS/lib:$QT_LOCATION/lib + export PATH=$QT_LOCATION:$PATH + + rm -rf $APP_DIR + export QMAKE=$QMAKE_BIN # this is needed for deploy-plugin-qt.AppImage + # inside a docker image you can't run an appimage executable without privileges + # so the solution is to extract the appimage first and only then to run it + export APPIMAGE_EXTRACT_AND_RUN=1 + ${STAGING_AREA}/linuxdeploy-x86_64.AppImage \ + --appdir $APP_DIR\ + --executable $SRC_DIR/build/scopy \ + --custom-apprun $SRC_DIR/CI/appimage/AppRun \ + --desktop-file $SRC_DIR/CI/appimage/scopy.desktop \ + --icon-file $SRC_DIR/resources/Scopy.png \ + --plugin qt + + cp $STAGING_AREA/iio-emu/build/iio-emu $APP_DIR/usr/bin + cp -r /lib/python3.9 $APP_DIR/usr/lib + cp -r $QT_LOCATION/plugins $APP_DIR/usr + + if [ -d $STAGING_AREA_DEPS/share/libsigrokdecode/decoders ]; then + cp -r $STAGING_AREA_DEPS/share/libsigrokdecode/decoders $APP_DIR/usr/lib + elif [ -d $STAGING_AREA/libsigrokdecode/decoders ];then + cp -r $STAGING_AREA/libsigrokdecode/decoders $APP_DIR/usr/lib + else + echo "No decoders for libsigrokdecode found" + fi + + cp $QT_LOCATION/lib/libQt5XcbQpa.so* $APP_DIR/usr/lib + cp $QT_LOCATION/lib/libQt5EglFSDeviceIntegration.so* $APP_DIR/usr/lib + cp $QT_LOCATION/lib/libQt5DBus.so* $APP_DIR/usr/lib + + cp /usr/lib/x86_64-linux-gnu/libGLESv2.so* $APP_DIR/usr/lib + cp /usr/lib/x86_64-linux-gnu/libstdc++.so* $APP_DIR/usr/lib + cp /usr/lib/x86_64-linux-gnu/libc.so* $APP_DIR/usr/lib + cp /usr/lib/x86_64-linux-gnu/libdl.so* $APP_DIR/usr/lib + cp /usr/lib/x86_64-linux-gnu/libpthread.so* $APP_DIR/usr/lib + cp /usr/lib/x86_64-linux-gnu/libXdmcp.so* $APP_DIR/usr/lib + cp /usr/lib/x86_64-linux-gnu/libgcc_s.so* $APP_DIR/usr/lib + + cp /usr/lib/x86_64-linux-gnu/libm.so* $APP_DIR/usr/lib + cp /usr/lib/x86_64-linux-gnu/libbsd.so* $APP_DIR/usr/lib + cp /usr/lib/x86_64-linux-gnu/libXau.so* $APP_DIR/usr/lib + cp /usr/lib/x86_64-linux-gnu/libxcb.so* $APP_DIR/usr/lib + popd +} + +create_appimage(){ + rm -rf $APP_IMAGE + + pushd ${STAGING_AREA} + export APPIMAGE_EXTRACT_AND_RUN=1 + ${STAGING_AREA}/linuxdeploy-plugin-appimage-x86_64.AppImage --appdir $APP_DIR + chmod +x Scopy*.AppImage + mv Scopy*.AppImage $APP_IMAGE + popd +} + +generate_ci_envs(){ + $SRC_DIR/CI/appveyor/gen_ci_envs.sh > $SRC_DIR/CI/appimage/gh-actions.envs +} + +move_appimage(){ + mv $APP_IMAGE $SRC_DIR +} + +for arg in $@; do + $arg +done