#!/usr/bin/env sh
# SPDX-License-Identifier: GPL-3.0-or-later
# SPDX-FileCopyrightText: 2014-2016,2020,2021,2023 Leah Rowe <leah@libreboot.org>
# SPDX-FileCopyrightText: 2022 Alper Nebi Yasak <alpernebiyasak@gmail.com>
# SPDX-FileCopyrightText: 2022 Ferass El Hafidi <vitali64pmemail@protonmail.com>

[ "x${DEBUG+set}" = 'xset' ] && set -v
set -u -e

. "include/err.sh"
. "include/git.sh"
. "include/option.sh"

_target=""
tree=""
rev=""
project=""
cfgsdir=""

main()
{
	rm -f "${cfgsdir}"/*/seen || err_rm_seen "main 1"

	printf "Downloading %s and (if available) applying patches\n" \
	    ${project}

	[ -z "${1}" ] && err "project name not specified"
	project="${1}"
	cfgsdir="config/${project}"
	[ -d "${cfgsdir}" ] || err "unsupported project name"
	shift 1

	targets=$(listitems "${cfgsdir}")
	[ $# -gt 0 ] && targets=$@
	[ -z "${targets}" ] && \
	    err "No targets available for project: ${project}"

	for x in ${targets}; do
		rm -f "${cfgsdir}"/*/seen || err_rm_seen "main 2"
		download_for_target "${x}" || \
		    err "${project}/${target}: cannot download source tree"
	done

	rm -f "${cfgsdir}"/*/seen || err_rm_seen "main 3"
}

download_for_target()
{
	_target="${1}"
	tree="undefined"
	rev="undefined"

	fetch_config "${_target}" || \
	    err "download_for_target: ${project}/${_target}: bad target.cfg"

	rm -f "${cfgsdir}"/*/seen || err_rm_seen "download_for_target"

	if [ -d "${project}/${tree}" ]; then
		printf "REMARK: download/%s %s: exists. Skipping.\n" \
		    "${project}" "${tree}" 1>&2
		[ "${tree}" != "${_target}" ] && \
			printf "(for target: '%s}')\n" "${_target}" 1>&2
		return 0
	fi

	fetch_from_upstream || \
	    err "download_for_target: cannot fetch: ${project}"
	prepare_new_tree "${_target}" "${tree}" "${rev}" || \
	    err "download_for_target: cannot create tree: ${project}/${tree}"
}

fetch_config()
{
	_target=${1}

	while true; do
		rev="undefined"
		tree="undefined"

		check_config_for_target "${_target}" || return 1

		# This is to override $rev and $tree
		. "${cfgsdir}/${_target}/target.cfg" || \
		    err "fetch_config: no \"${cfgsdir}/${_target}/target.cfg\""

		if [ "${_target}" != "${tree}" ]; then
			_target="${tree}"
			continue
		elif [ "${tree}" = "undefined" ]; then
			printf "ERROR (fetch_config): download/%s:" 1>&2
			printf " tree name undefined for '%s\n'" \
			    "${project}" "${_target}" 1>&2
			return 1
		elif [ "${rev}" = "undefined" ]; then
			printf "ERROR (fetch_config): download/%s:" 1>&2
			printf " commit ID undefined for '%s'\n" \
			    "${project}" "${_target}" 1>&2
			return 1
		else
			break
		fi
	done
}

check_config_for_target()
{
	_target=${1}

	if [ ! -f "${cfgsdir}/${_target}/target.cfg" ]; then
		printf "ERROR: download/%s: target.cfg does not" \
		    "${project}" 1>&2
		printf " exist for '%s'\n" "${_target}" 1>&2
		return 1
	elif [ -f "${cfgsdir}/${_target}/seen" ]; then
		printf "ERROR: download/%s: logical loop:" "${project}" 1>&2
		printf " '%s' target.cfg refers to another tree," "${_target}" \
		    1>&2
		printf " which ultimately refers back to '%s'." "${_target}" \
		    1>&2
		return 1
	fi
	touch "${cfgsdir}/${_target}/seen" || \
	    err "${project}/${_target}: touch \"${cfgsdir}/${_target}/seen\""
}

fetch_from_upstream()
{
	[ -d "${project}" ] || mkdir -p "${project}" || return 1
	[ -d "${project}" ] || return 1
	[ -d "${project}/${project}" ] && return 0

	./update project repo ${project} || return 1
}

prepare_new_tree()
{
	target=${1}
	tree=${2}
	rev=${3}

	printf "Preparing %s tree: %s\n" ${project} ${tree}
	[ "${tree}" != "${target}" ] && \
		printf "(for target, %s)\n" "${target}"

	cp -R "${project}/${project}" "${project}/${tree}" || \
	    err "${project}/${tree}: cannot copy source tree"
	git_reset_rev "${project}/${tree}" "${rev}" "err" || \
	    err "prepare_new_trees ${project}/${tree}: cannot reset <- ${rev}"

	(
	cd "${project}/${tree}" || \
	    err "prepare_new_tree: !cd \"${project}/${tree}\""
	git submodule update --init --checkout || \
	    err "prepare_new_tree ${project}/${tree}: can't update git modules"
	)

	git_am_patches "${PWD}/${project}/${tree}" \
	    "${PWD}/${cfgsdir}/${tree}/patches" "err" || \
	    err "prepare_new_trees ${project}/${tree}: cannot apply patches"
}

err_rm_seen()
{
	err "${1}: ${project}/${target}: cannot rm: \"${cfgsdir}/*/seen\""
}

main $@