aseprite/build.sh

509 lines
16 KiB
Bash
Raw Normal View History

2024-10-28 21:49:43 +00:00
#! /bin/bash
#
# This is a script to help users and developers to build Aseprite.
# Usage:
#
# ./build.sh
# ./build.sh --reset
# ./build.sh --auto [--norun]
#
# If you run this script without parameters, you will be able to
# follows the instructions and a configuration will be stored in a
# ".build" directory.
#
# Options:
#
# --reset Deletes the configuration and you can start over
# --auto Tries to build the default user configuration (release mode)
# --norun Doesn't auto-run when using --auto
#
echo "======================= BUILD ASEPRITE HELPER ========================"
# Check that we are running the script from the Aseprite clone directory.
pwd=$(pwd)
2024-11-11 16:06:03 +00:00
if [[ ! -f "$pwd/EULA.txt" || ! -f "$pwd/.gitmodules" ]] ; then
2024-10-28 21:49:43 +00:00
echo ""
echo "Run build script from the Aseprite directory"
exit 1
fi
# Use "./build.sh --reset" to reset all the configuration (deletes
# .build directory).
if [ "$1" == "--reset" ] ; then
echo ""
echo "Resetting $pwd/.build directory"
if [ -f "$pwd/.build/builds_dir" ] ; then rm $pwd/.build/builds_dir ; fi
if [ -f "$pwd/.build/log" ] ; then rm $pwd/.build/log ; fi
if [ -f "$pwd/.build/main_skia_dir" ] ; then rm $pwd/.build/main_skia_dir ; fi
if [ -f "$pwd/.build/beta_skia_dir" ] ; then rm $pwd/.build/beta_skia_dir ; fi
if [ -f "$pwd/.build/userkind" ] ; then rm $pwd/.build/userkind ; fi
if [ -d "$pwd/.build" ] ; then rmdir $pwd/.build ; fi
echo "Done"
exit 0
fi
# Use "./build.sh --auto" to build the user configuration without
# questions (downloading Skia/release mode).
auto=
if [ "$1" == "--auto" ] ; then
shift
auto=1
fi
norun=
if [ "$1" == "--norun" ] ; then
shift
norun=1
fi
# Platform.
if [[ "$(uname)" =~ "MINGW32" ]] || [[ "$(uname)" =~ "MINGW64" ]] || [[ "$(uname)" =~ "MSYS_NT-10.0" ]] ; then
is_win=1
cpu=x64
if ! cl.exe >/dev/null 2>/dev/null ; then
echo ""
echo "MSVC compiler (cl.exe) not found in PATH"
echo ""
echo " PATH=$PATH"
echo ""
exit 1
fi
elif [[ "$(uname)" == "Linux" ]] ; then
is_linux=1
cpu=x64
elif [[ "$(uname)" =~ "Darwin" ]] ; then
is_macos=1
if [[ $(uname -m) == "arm64" ]]; then
cpu=arm64
else
cpu=x64
fi
fi
# Check utilities.
if ! cmake --version >/dev/null ; then
echo ""
echo "cmake utility is not available. You can get cmake from:"
echo ""
echo " https://cmake.org/download/"
echo ""
exit 1
fi
if ! ninja --version >/dev/null ; then
echo ""
echo "ninja utility is not available. You can get ninja from:"
echo ""
echo " https://github.com/ninja-build/ninja/releases"
echo ""
exit 1
fi
2024-11-11 16:06:03 +00:00
# Check that all submodules are checked out.
run_submodule_update=
for module in $(cat "$pwd/.gitmodules" | \
grep '^\[submodule' | \
sed -e 's/^\[.*\"\(.*\)\"\]/\1/') \
$(cat "$pwd/laf/.gitmodules" | \
grep '^\[submodule' | \
sed -e 's/^\[.*\"\(.*\)\"\]/laf\/\1/') ; do
if [[ ! -f "$module/CMakeLists.txt" &&
! -f "$module/Makefile" &&
! -f "$module/makefile" &&
! -f "$module/Makefile.am" ]] ; then
echo ""
echo "Module $module doesn't exist."
if [ $auto ] ; then
run_submodule_update=1
break
else
echo "Run:"
echo ""
echo " git submodule update --init --recursive"
echo ""
exit 1
fi
fi
done
if [ $run_submodule_update ] ; then
echo "Running:"
echo ""
echo " git submodule update --init --recursive"
echo ""
if ! git submodule update --init --recursive ; then
echo "Failed, try again"
exit 1
fi
echo "Done"
fi
2024-10-28 21:49:43 +00:00
# Create the directory to store the configuration.
if [ ! -d "$pwd/.build" ] ; then
mkdir "$pwd/.build"
fi
# Kind of user (User or Developer).
# For users we simplify the process (no multiple builds), for
# developers we have more options (debug mode, etc.).
if [ ! -f "$pwd/.build/userkind" ] ; then
if [ $auto ] ; then
echo "user" > $pwd/.build/userkind
else
echo ""
echo "Select what kind of user you are (press U or D keys):"
echo ""
echo " [U]ser: give a try to Aseprite"
echo " [D]eveloper: develop/modify Aseprite"
echo ""
read -sN 1 -p "[U/D]? "
echo ""
if [[ "$REPLY" == "d" || "$REPLY" == "D" ]] ; then
echo "developer" > $pwd/.build/userkind
elif [[ "$REPLY" == "u" || "$REPLY" == "U" ]] ; then
echo "user" > $pwd/.build/userkind
else
echo "Use U or D keys to select kind of user/build process"
exit 1
fi
fi
fi
userkind=$(echo -n $(cat $pwd/.build/userkind))
if [ "$userkind" == "developer" ] ; then
echo "======================= BUILDING FOR DEVELOPER ======================="
else
echo "========================= BUILDING FOR USER =========================="
fi
# Get the builds_dir location.
if [ ! -f "$pwd/.build/builds_dir" ] ; then
if [ "$userkind" == "developer" ] ; then
# The "builds" folder is a place where all possible combination/builds
# will be stored. If the ASEPRITE_BUILD environment variable is
# defined, that's the directory, in other case for a regular "user"
# the folder will be this same directory where Aseprite was cloned.
if [[ "$ASEPRITE_BUILD" != "" ]] ; then
if [ $is_win ] ; then
builds_dir=$(cygpath "$ASEPRITE_BUILD")
else
builds_dir="$ASEPRITE_BUILD"
fi
if [ -d "$builds_dir" ] ; then
echo ""
echo "Using ASEPRITE_BUILD environment variable for builds directory."
else
if ! mkdir "$builds_dir" ; then
echo ""
echo "The ASEPRITE_BUILD is defined but we weren't able to create the directory:"
echo ""
echo " ASEPRITE_BUILD=$builds_dir"
echo ""
echo "To solve this issue delete the ASEPRITE_BUILD variable or point it to a valid path."
exit 1
fi
fi
else
# Default location for developers
builds_dir=$HOME/builds
echo ""
echo "Select a folder where to leave all builds:"
echo " builds_dir/"
echo " release-x64/..."
echo " debug-x64/..."
echo ""
echo "builds_dir [$builds_dir]? "
read builds_dir_read
if [ "$builds_dir_read" != "" ] ; then
builds_dir="$builds_dir_read"
fi
fi
else
# Default location for users
builds_dir="$pwd"
echo ""
echo "We'll build Aseprite in $builds_dir/build directory"
fi
echo "$builds_dir" > "$pwd/.build/builds_dir"
fi
# Overwrite $builds_dir variable from the config content.
builds_dir="$(echo -n $(cat $pwd/.build/builds_dir))"
# List all builds.
builds_list="$(mktemp)"
n=1
for file in $(ls $builds_dir/*/CMakeCache.txt 2>/dev/null | sort) ; do
if cat "$file" | grep -q "CMAKE_PROJECT_NAME:STATIC=aseprite" ; then
if [ $n -eq 1 ] ; then
echo "-- AVAILABLE BUILDS --"
fi
echo "$file" >> $builds_list
echo "$n. $file"
n=$(($n+1))
fi
done
# New build configuration.
build_type=RelWithDebInfo
# No builds, so this is the first build.
if [[ $n -eq 1 ]] ; then
echo "-- FIRST BUILD --"
if [ "$userkind" == "developer" ] ; then
active_build_dir="$builds_dir/aseprite-release"
else
active_build_dir="$builds_dir/build"
fi
echo "First build directory: $active_build_dir"
else
echo "N. New build (N key)"
echo "U. Update Visual Studio/Windows Kit/macOS SDK version (U key)"
2024-10-28 21:49:43 +00:00
read -p "Select an option or number to build? " build_n
# New build
if [[ "$build_n" == "n" || "$build_n" == "N" ]] ; then
read -p "Select build type [RELEASE/debug]? "
if [[ "${REPLY,,}" == "debug" ]] ; then
build_type=Debug
new_build_name=aseprite-debug
else
build_type=RelWithDebInfo
new_build_name=aseprite-release
fi
read -p "Select a name [$new_build_name]? "
if [[ "$REPLY" != "" ]] ; then
new_build_name=$REPLY
fi
active_build_dir="$builds_dir/$new_build_name"
# Update SDK
elif [[ "$build_n" == "u" || "$build_n" == "U" ]] ; then
echo "Update SDK dirs..."
if [ $is_win ] ; then
newclver=$(echo $VCToolsInstallDir | sed -e 's_^.*\\\([0-9\.]*\)\\$_\1_')
function update_file {
file=$1
echo "--- Updating $file ---" | tee -a "$pwd/.build/log"
mv "$file" "$file-old"
cat "$file-old" | sed -e 's_^\(.*/VC/Tools/MSVC/\)\([0-9\.]*\)\(.*$\)_\1'$newclver'\3_' > "$file"
diff -w -u3 "$file-old" "$file" >> "$pwd/.build/log"
echo "--- End $file ---" >> "$pwd/.build/log"
}
echo "New VC version: $newclver"
for file in $(cat $builds_list) ; do
build_dir=$(dirname $file)
echo "--- Updating $build_dir ---"
update_file "$file"
for other_file in "$build_dir/CMakeFiles/rules.ninja" \
"$build_dir"/CMakeFiles/*/*.cmake \
"$build_dir/third_party/libpng/scripts/genout.cmake" ; do
update_file "$other_file"
done
done
fi
echo "Done"
exit
else # Build the selected dir
2024-10-28 21:49:43 +00:00
n=1
for file in $(cat $builds_list) ; do
if [ "$n" == "$build_n" ] ; then
active_build_dir=$(dirname $file)
build_type=$(cat $active_build_dir/CMakeCache.txt | grep CMAKE_BUILD_TYPE | cut -d "=" -f2)
break
fi
n=$(($n+1))
done
fi
fi
if [ "$active_build_dir" == "" ] ; then
echo "No build selected"
exit 1
fi
if [ -f "$active_build_dir/CMakeCache.txt" ] ; then
source_dir=$(cat $active_build_dir/CMakeCache.txt | grep aseprite_SOURCE_DIR | cut -d "=" -f2)
else
source_dir="$pwd"
fi
branch_name=$(git --git-dir="$source_dir/.git" rev-parse --abbrev-ref HEAD)
if [[ "$branch_name" == "main" || "$branch_name" == "beta" ]] ; then
base_branch_name="$branch_name"
else
# Get the origin (or first remote) name
remote=$(git --git-dir="$source_dir/.git" remote | grep origin)
if [ "$remote" == "" ] ; then
remote=$(git --git-dir="$source_dir/.git" remote | head -n1)
fi
if git --git-dir="$source_dir/.git" branch --contains "$remote/beta" | grep -q "^\* $branch_name\$" ; then
base_branch_name=beta
elif git --git-dir="$source_dir/.git" branch --contains "$remote/main" | grep -q "^\* $branch_name\$" ; then
base_branch_name=main
else
echo ""
echo "Error: Branch $branch_name looks like doesn't belong to main or beta"
echo ""
exit 1
fi
fi
echo "=========================== CONFIGURATION ============================"
echo "Build type: $build_type"
echo "Build dir: \"$active_build_dir\""
echo "Source dir: \"$source_dir\""
if [ "$branch_name" != "$base_branch_name" ] ; then
echo "Branch name: $base_branch_name > $branch_name"
else
echo "Branch name: $base_branch_name"
fi
# Required Skia for the base branch.
if [ "$base_branch_name" == "beta" ] ; then
skia_tag=m124-08a5439a6b
file_skia_dir=beta_skia_dir
possible_skia_dir_name=skia-m124
else
skia_tag=m102-861e4743af
file_skia_dir=main_skia_dir
possible_skia_dir_name=skia
fi
# Check Skia dependency.
if [ ! -f "$pwd/.build/$file_skia_dir" ] ; then
# Try "C:/deps/skia" or "$HOME/deps/skia"
if [[ $is_win ]] ; then
skia_dir="C:/deps/$possible_skia_dir_name"
else
skia_dir="$HOME/deps/$possible_skia_dir_name"
fi
if [ ! -d "$skia_dir" ] ; then
echo ""
echo "Skia directory wasn't found."
echo ""
echo "Select Skia directory to create [$skia_dir]? "
if [ ! $auto ] ; then
read skia_dir_read
if [ "$skia_dir_read" != "" ] ; then
skia_dir="$skia_dir_read"
fi
fi
mkdir -p $skia_dir || exit 1
fi
echo $skia_dir > "$pwd/.build/$file_skia_dir"
fi
skia_dir=$(echo -n $(cat $pwd/.build/$file_skia_dir))
if [ ! -d "$skia_dir" ] ; then
mkdir "$skia_dir"
fi
# Only on Windows we need the Debug version of Skia to compile the
# Debug version of Aseprite.
if [[ $is_win && "$build_type" == "Debug" ]] ; then
skia_library_dir="$skia_dir/out/Debug-x64"
else
skia_library_dir="$skia_dir/out/Release-$cpu"
fi
# If the library directory is not available, we can try to download
# the pre-built package.
if [ ! -d "$skia_library_dir" ] ; then
echo ""
echo "Skia library wasn't found."
echo ""
if [ ! $auto ] ; then
read -sN 1 -p "Download pre-compiled Skia automatically [Y/n]? "
fi
if [[ $auto || "$REPLY" == "" || "$REPLY" == "y" || "$REPLY" == "Y" ]] ; then
if [[ $is_win && "$build_type" == "Debug" ]] ; then
skia_build=Debug
else
skia_build=Release
fi
if [ $is_win ] ; then
skia_file=Skia-Windows-$skia_build-$cpu.zip
elif [ $is_macos ] ; then
skia_file=Skia-macOS-$skia_build-$cpu.zip
else
skia_file=Skia-Linux-$skia_build-$cpu-libstdc++.zip
fi
skia_url=https://github.com/aseprite/skia/releases/download/$skia_tag/$skia_file
if [ ! -f "$skia_dir/$skia_file" ] ; then
curl -L -o "$skia_dir/$skia_file" "$skia_url"
fi
if [ ! -d "$skia_library_dir" ] ; then
unzip -n -d "$skia_dir" "$skia_dir/$skia_file"
fi
else
echo "Please follow these instructions to compile Skia manually:"
echo ""
echo " https://github.com/aseprite/skia?tab=readme-ov-file#skia-for-aseprite-and-laf"
echo ""
exit 1
fi
fi
echo "================================ SKIA ================================"
echo "Skia directory: \"$skia_dir\""
echo "Skia library directory: \"$skia_library_dir\""
if [ ! -d "$skia_library_dir" ] ; then
echo " But the Skia library directory wasn't found."
exit 1
fi
# Building
echo "=============================== CMAKE ================================"
if [ ! -f "$active_build_dir/ninja.build" ] ; then
echo ""
echo "We are going to run cmake and then build the project."
echo "This will take some minutes."
echo ""
if [ ! $auto ] ; then
read -sN 1 -p "Press any key to continue. "
fi
if [ $is_macos ] ; then
osx_deployment_target="-DCMAKE_OSX_DEPLOYMENT_TARGET=11.0"
else
osx_deployment_target=
fi
echo "Configuring Aseprite..."
if ! cmake -B "$active_build_dir" -S "$source_dir" -G Ninja \
-DCMAKE_BUILD_TYPE=$build_type \
$osx_deployment_target \
-DLAF_BACKEND=skia \
-DSKIA_DIR="$skia_dir" \
-DSKIA_LIBRARY_DIR="$skia_library_dir" | \
tee -a "$pwd/.build/log" ; then
2024-10-28 21:49:43 +00:00
echo "Error running cmake."
exit 1
fi
fi
echo "============================== BUILDING =============================="
if ! cmake --build "$active_build_dir" -- aseprite | tee -a "$pwd/.build/log" ; then
2024-10-28 21:49:43 +00:00
echo "Error building Aseprite."
exit 1
fi
echo "================================ DONE ================================"
if [ $is_win ] ; then exe=.exe ; else exe= ; fi
echo "Run Aseprite with the following command:"
echo ""
echo " $active_build_dir/bin/aseprite$exe"
echo ""
# Run Aseprite in --auto mode
if [[ $auto && ! $norun ]] ; then
$active_build_dir/bin/aseprite$exe
fi