OSDN Git Service

[fix] : No output in kernel module check
[alterlinux/alterlinux.git] / build.sh
1 #!/usr/bin/env bash
2 #
3 # Yamada Hayao
4 # Twitter: @Hayao0819
5 # Email  : hayao@fascode.net
6 #
7 # (c) 2019-2021 Fascode Network.
8 #
9 # build.sh
10 #
11 # The main script that runs the build
12 #
13
14 set -Eeu
15
16 # Internal config
17 # Do not change these values.
18 script_path="$( cd -P "$( dirname "$(readlink -f "${0}")" )" && pwd )"
19 defaultconfig="${script_path}/default.conf"
20 tools_dir="${script_path}/tools"
21 module_dir="${script_path}/modules"
22 customized_username=false
23 customized_password=false
24 customized_kernel=false
25 customized_logpath=false
26 pkglist_args=()
27 makepkg_script_args=()
28 modules=()
29 DEFAULT_ARGUMENT=""
30 ARGUMENT=("${@}")
31 alteriso_version="3.1"
32 norepopkg=()
33
34 # Load config file
35 [[ ! -f "${defaultconfig}" ]] && "${tools_dir}/msg.sh" -a 'build.sh' error "${defaultconfig} was not found." && exit 1
36 for config in "${defaultconfig}" "${script_path}/custom.conf"; do
37     [[ -f "${config}" ]] && source "${config}"
38 done
39
40 umask 0022
41
42 # Message common function
43 # msg_common [type] [-n] [string]
44 msg_common(){
45     local _msg_opts=("-a" "build.sh") _type="${1}"
46     shift 1
47     [[ "${1}" = "-n" ]] && _msg_opts+=("-o" "-n") && shift 1
48     [[ "${msgdebug}" = true ]] && _msg_opts+=("-x")
49     [[ "${nocolor}"  = true ]] && _msg_opts+=("-n")
50     _msg_opts+=("${_type}" "${@}")
51     "${tools_dir}/msg.sh" "${_msg_opts[@]}"
52 }
53
54 # Show an INFO message
55 # ${1}: message string
56 msg_info() { msg_common info "${@}"; }
57
58 # Show an Warning message
59 # ${1}: message string
60 msg_warn() { msg_common warn "${@}"; }
61
62 # Show an debug message
63 # ${1}: message string
64 msg_debug() { 
65     [[ "${debug}" = true ]] && msg_common debug "${@}" || return 0
66 }
67
68 # Show an ERROR message then exit with status
69 # ${1}: message string
70 # ${2}: exit code number (with 0 does not exit)
71 msg_error() {
72     msg_common error "${1}"
73     [[ -n "${2:-""}" ]] && exit "${2}" || return 0
74 }
75
76
77 # Usage: getclm <number>
78 # 標準入力から値を受けとり、引数で指定された列を抽出します。
79 getclm() { cut -d " " -f "${1}"; }
80
81 # Usage: echo_blank <number>
82 # 指定されたぶんの半角空白文字を出力します
83 echo_blank(){ yes " " 2> /dev/null  | head -n "${1}" | tr -d "\n"; }
84
85 _usage () {
86     echo "usage ${0} [options] [channel]"
87     echo
88     echo "A channel is a profile of AlterISO settings."
89     echo
90     echo " General options:"
91     echo "    -b | --boot-splash           Enable boot splash"
92     echo "    -e | --cleanup | --cleaning  Enable post-build cleaning"
93     echo "    -r |  --tarball               Build rootfs in tar.xz format"
94     echo "    -h | --help                  This help message and exit"
95     echo
96     echo "    -a | --arch <arch>           Set iso architecture"
97     echo "    -c | --comp-type <comp_type> Set SquashFS compression type (gzip, lzma, lzo, xz, zstd)"
98     echo "    -g | --gpgkey <key>          Set gpg key"
99     echo "    -l | --lang <lang>           Specifies the default language for the live environment"
100     echo "    -k | --kernel <kernel>       Set special kernel type. See below for available kernels"
101     echo "    -o | --out <out_dir>         Set the output directory"
102     echo "    -p | --password <password>   Set a live user password"
103     echo "    -t | --comp-opts <options>   Set compressor-specific options."
104     echo "    -u | --user <username>       Set user name"
105     echo "    -w | --work <work_dir>       Set the working directory"
106     echo
107
108     local blank="29" _arch _dirname _type _output _first
109     for _type in "locale" "kernel"; do
110         echo " ${_type} for each architecture:"
111         for _arch in $(find "${script_path}/system/" -maxdepth 1 -mindepth 1 -name "${_type}-*" -print0 | xargs -I{} -0 basename {} | sed "s|${_type}-||g"); do
112             echo -n "    ${_arch}$(echo_blank "$(( "${blank}" - "${#_arch}" ))")"
113             "${tools_dir}/${_type}.sh" -a "${_arch}" show
114         done
115         echo
116     done
117
118     echo " Channel:"
119     for _dirname in $(bash "${tools_dir}/channel.sh" --version "${alteriso_version}" -d -b -n --line show | sed "s|.add$||g"); do
120         readarray -t _output < <("${tools_dir}/channel.sh" --version "${alteriso_version}" --nocheck desc "${_dirname}")
121         _first=true
122         echo -n "    ${_dirname}"
123         for _out in "${_output[@]}"; do
124             "${_first}" && echo -e "    $(echo_blank "$(( "${blank}" - 4 - "${#_dirname}" ))")${_out}" || echo -e "    $(echo_blank "$(( "${blank}" + 5 - "${#_dirname}" ))")${_out}"
125             _first=false
126         done
127     done
128
129     echo
130     echo " Debug options: Please use at your own risk."
131     echo "    -d | --debug                 Enable debug messages"
132     echo "    -x | --bash-debug            Enable bash debug mode(set -xv)"
133     echo "         --channellist           Output the channel list and exit"
134     echo "         --config                Load additional config file"
135     echo "         --gitversion            Add Git commit hash to image file version"
136     echo "         --logpath <file>        Set log file path (use with --log)"
137     echo "         --[no]log               (No) log ;re-run script with tee"
138     echo "         --msgdebug              Enables output debugging"
139     echo "         --noaur                 Ignore aur packages (Use only for debugging)"
140     echo "         --nocolor               No output colored output"
141     echo "         --[no]confirm           (No) check the settings before building"
142     echo "         --nochkver              No check the version of the channel"
143     echo "         --nodebug               Disable all debug messages"
144     echo "         --noefi                 No efi boot (Use only for debugging)"
145     echo "         --noloopmod             No check and load kernel module automatically"
146     echo "         --nodepend              No check package dependencies before building"
147     echo "         --noiso                 No build iso image (Use with --tarball)"
148     echo "         --nosigcheck            No pacman signature check"
149     echo "         --pacman-debug          Enable pacman debug mode"
150     echo "         --normwork              No remove working dir"
151     echo "         --nopkgbuild            Ignore PKGBUILD (Use only for debugging)"
152     echo "         --tar-type <comp_type>  Set compression type (gzip, lzma, lzo, xz, zstd)"
153     echo "         --tar-opts <option>     Set tar command argument (Use with --tarball)"
154     echo "         --add-module <module>   Load additional modules (Separated by \",\")"
155     echo
156     echo " Many packages are installed from AUR, so specifying --noaur can cause problems."
157     echo
158     [[ -n "${1:-}" ]] && exit "${1}"
159 }
160
161 # Unmount helper Usage: _umount <target>
162 _umount() { if mountpoint -q "${1}"; then umount -lf "${1}"; fi; }
163
164 # Mount helper Usage: _mount <source> <target>
165 _mount() { ! mountpoint -q "${2}" && [[ -f "${1}" ]] && [[ -d "${2}" ]] && mount "${1}" "${2}"; return 0; }
166
167 # Unmount work dir
168 umount_work () {
169     local _args=("${build_dir}")
170     [[ "${debug}" = true ]] && _args=("-d" "${_args[@]}")
171     [[ "${nocolor}" = true ]] && _args+=("--nocolor")
172     "${tools_dir}/umount.sh" "${_args[@]}"
173 }
174
175 # Mount airootfs on "${airootfs_dir}"
176 mount_airootfs () {
177     mkdir -p "${airootfs_dir}"
178     _mount "${airootfs_dir}.img" "${airootfs_dir}"
179 }
180
181
182 # Helper function to run make_*() only one time.
183 run_once() {
184     set -eu
185     if [[ ! -e "${lockfile_dir}/build.${1}" ]]; then
186         msg_debug "Running ${1} ..."
187         mount_airootfs
188         eval "${@}"
189         mkdir -p "${lockfile_dir}"; touch "${lockfile_dir}/build.${1}"
190         umount_work
191     else
192         msg_debug "Skipped because ${1} has already been executed."
193     fi
194 }
195
196 # Show message when file is removed
197 # remove <file> <file> ...
198 remove() {
199     local _file
200     for _file in "${@}"; do msg_debug "Removing ${_file}"; rm -rf "${_file}"; done
201 }
202
203 # 強制終了時にアンマウント
204 umount_trap() {
205     local _status="${?}"
206     umount_work
207     msg_error "It was killed by the user.\nThe process may not have completed successfully."
208     exit "${_status}"
209 }
210
211 # 設定ファイルを読み込む
212 # load_config [file1] [file2] ...
213 load_config() {
214     local _file
215     for _file in "${@}"; do [[ -f "${_file}" ]] && source "${_file}" && msg_debug "The settings have been overwritten by the ${_file}"; done
216     return 0
217 }
218
219 # Display channel list
220 show_channel_list() {
221     local _args=("-v" "${alteriso_version}" show)
222     [[ "${nochkver}" = true ]] && _args+=("-n")
223     bash "${tools_dir}/channel.sh" "${_args[@]}"
224 }
225
226 # Execute command for each module. It will be executed with {} replaced with the module name.
227 # for_module <command>
228 for_module(){ local module && for module in "${modules[@]}"; do eval "${@//"{}"/${module}}"; done; }
229
230 # pacstrapを実行
231 _pacstrap(){
232     msg_info "Installing packages to ${airootfs_dir}/'..."
233     local _args=("-c" "-G" "-M" "--" "${airootfs_dir}" "${@}")
234     [[ "${pacman_debug}" = true ]] && _args+=(--debug)
235     pacstrap -C "${build_dir}/pacman.conf" "${_args[@]}"
236     msg_info "Packages installed successfully!"
237 }
238
239 # chroot環境でpacmanコマンドを実行
240 # /etc/alteriso-pacman.confを準備してコマンドを実行します
241 _run_with_pacmanconf(){
242     sed "s|^CacheDir     =|#CacheDir    =|g" "${build_dir}/pacman.conf" > "${airootfs_dir}/etc/alteriso-pacman.conf"
243     eval -- "${@}"
244     remove "${airootfs_dir}/etc/alteriso-pacman.conf"
245 }
246
247 # コマンドをchrootで実行する
248 _chroot_run() {
249     msg_debug "Run command in chroot\nCommand: ${*}"
250     eval -- arch-chroot "${airootfs_dir}" "${@}"
251 }
252
253 _cleanup_common () {
254     msg_info "Cleaning up what we can on airootfs..."
255
256     # Delete pacman database sync cache files (*.tar.gz)
257     [[ -d "${airootfs_dir}/var/lib/pacman" ]] && find "${airootfs_dir}/var/lib/pacman" -maxdepth 1 -type f -delete
258
259     # Delete pacman database sync cache
260     [[ -d "${airootfs_dir}/var/lib/pacman/sync" ]] && find "${airootfs_dir}/var/lib/pacman/sync" -delete
261
262     # Delete pacman package cache
263     [[ -d "${airootfs_dir}/var/cache/pacman/pkg" ]] && find "${airootfs_dir}/var/cache/pacman/pkg" -type f -delete
264
265     # Delete all log files, keeps empty dirs.
266     [[ -d "${airootfs_dir}/var/log" ]] && find "${airootfs_dir}/var/log" -type f -delete
267
268     # Delete all temporary files and dirs
269     [[ -d "${airootfs_dir}/var/tmp" ]] && find "${airootfs_dir}/var/tmp" -mindepth 1 -delete
270
271     # Delete package pacman related files.
272     find "${build_dir}" \( -name '*.pacnew' -o -name '*.pacsave' -o -name '*.pacorig' \) -delete
273
274     # Delete all cache file
275     [[ -d "${airootfs_dir}/var/cache" ]] && find "${airootfs_dir}/var/cache" -mindepth 1 -delete
276
277     # Create an empty /etc/machine-id
278     printf '' > "${airootfs_dir}/etc/machine-id"
279
280     msg_info "Done!"
281 }
282
283 _cleanup_airootfs(){
284     _cleanup_common
285     # Delete all files in /boot
286     [[ -d "${airootfs_dir}/boot" ]] && find "${airootfs_dir}/boot" -mindepth 1 -delete
287 }
288
289 _mkchecksum() {
290     msg_info "Creating md5 checksum ..."
291     echo "$(md5sum "${1}" | getclm 1) $(basename "${1}")" > "${1}.md5"
292     msg_info "Creating sha256 checksum ..."
293     echo "$(sha256sum "${1}" | getclm 1) $(basename "${1}")" > "${1}.sha256"
294 }
295
296 # Check the value of a variable that can only be set to true or false.
297 check_bool() {
298     local _value _variable
299     for _variable in "${@}"; do
300         msg_debug -n "Checking ${_variable}..."
301         eval ": \${${_variable}:=''}"
302         _value="$(eval echo "\$${_variable}")"
303         if [[ ! -v "${1}" ]] || [[ "${_value}"  = "" ]]; then
304             [[ "${debug}" = true ]] && echo ; msg_error "The variable name ${_variable} is empty." "1"
305         elif [[ ! "${_value}" = "true" ]] && [[ ! "${_value}" = "false" ]]; then
306             [[ "${debug}" = true ]] && echo ; msg_error "The variable name ${_variable} is not of bool type (${_variable} = ${_value})" "1"
307         elif [[ "${debug}" = true ]]; then
308             echo -e " ${_value}"
309         fi
310     done
311 }
312
313 _run_cleansh(){
314     bash "$([[ "${bash_debug}" = true ]] && echo -n "-x" || echo -n "+x")" "${tools_dir}/clean.sh" -o -w "$(realpath "${build_dir}")" "$([[ "${debug}" = true ]] && printf "%s" "-d")" "$([[ "${noconfirm}" = true ]] && printf "%s" "-n")" "$([[ "${nocolor}" = true ]] && printf "%s" "--nocolor")"
315 }
316
317
318 # Check the build environment and create a directory.
319 prepare_env() {
320     # Check packages
321     if [[ "${nodepend}" = false ]]; then
322         local _check_failed=false _pkg _result=0
323         msg_info "Checking dependencies ..."
324         ! pacman -Qq pyalpm > /dev/null 2>&1 && msg_error "pyalpm is not installed." 1
325         for _pkg in "${dependence[@]}"; do
326             eval "${tools_dir}/package.py" "${_pkg}" "$( [[ "${debug}" = false ]] && echo "> /dev/null")" || _result="${?}"
327             if (( _result == 3 )) || (( _result == 4 )); then
328                 _check_failed=true
329             fi
330             _result=0
331         done
332         [[ "${_check_failed}" = true ]] && exit 1
333     fi
334
335     # Load loop kernel module
336     if [[ "${noloopmod}" = false ]]; then
337         [[ ! -d "/usr/lib/modules/$(uname -r)" ]] && msg_error "The currently running kernel module could not be found.\nProbably the system kernel has been updated.\nReboot your system to run the latest kernel." "1"
338         lsmod | getclm 1 | grep -qx "loop" || modprobe loop
339     fi
340
341     # Check work dir
342     if [[ "${normwork}" = false ]]; then
343         msg_info "Deleting the contents of ${build_dir}..."
344         _run_cleansh
345     fi
346
347     # Set gpg key
348     if [[ -n "${gpg_key}" ]]; then
349         gpg --batch --output "${work_dir}/pubkey.gpg" --export "${gpg_key}"
350         exec {ARCHISO_GNUPG_FD}<>"${build_dir}/pubkey.gpg"
351         export ARCHISO_GNUPG_FD
352     fi
353
354     # 強制終了時に作業ディレクトリを削除する
355     local _trap_remove_work
356     _trap_remove_work() {
357         local status="${?}"
358         [[ "${normwork}" = false ]] && echo && _run_cleansh
359         exit "${status}"
360     }
361     trap '_trap_remove_work' HUP INT QUIT TERM
362
363     return 0
364 }
365
366 # Error message
367 error_exit_trap(){
368     local _exit="${?}" _line="${1}" && shift 1
369     msg_error "An exception error occurred in the function"
370     msg_error "Exit Code: ${_exit}\nLine: ${_line}\nArgument: ${ARGUMENT[*]}"
371     exit "${_exit}"
372 }
373
374 # Show settings.
375 show_settings() {
376     if [[ "${boot_splash}" = true ]]; then
377         msg_info "Boot splash is enabled."
378         msg_info "Theme is used ${theme_name}."
379     fi
380     msg_info "Language is ${locale_fullname}."
381     msg_info "Use the ${kernel} kernel."
382     msg_info "Live username is ${username}."
383     msg_info "Live user password is ${password}."
384     msg_info "The compression method of squashfs is ${sfs_comp}."
385     msg_info "Use the ${channel_name%.add} channel."
386     msg_info "Build with architecture ${arch}."
387     (( "${#additional_exclude_pkg[@]}" != 0 )) && msg_info "Excluded packages: ${additional_exclude_pkg[*]}"
388     if [[ "${noconfirm}" = false ]]; then
389         echo -e "\nPress Enter to continue or Ctrl + C to cancel."
390         read -r
391     fi
392     trap HUP INT QUIT TERM
393     trap 'umount_trap' HUP INT QUIT TERM
394     trap 'error_exit_trap $LINENO' ERR
395
396     return 0
397 }
398
399
400 # Preparation for build
401 prepare_build() {
402     # Debug mode
403     [[ "${bash_debug}" = true ]] && set -x -v
404
405     # Show alteriso version
406     [[ -n "${gitrev-""}" ]] && msg_debug "The version of alteriso is ${gitrev}"
407
408     # Load configs
409     load_config "${channel_dir}/config.any" "${channel_dir}/config.${arch}"
410
411     # Additional modules
412     modules+=("${additional_modules[@]}")
413
414     # Legacy mode
415     if [[ "$(bash "${tools_dir}/channel.sh" --version "${alteriso_version}" ver "${channel_name}")" = "3.0" ]]; then
416         msg_warn "The module cannot be used because it works with Alter ISO3.0 compatibility."
417         modules=("legacy")
418         [[ "${include_extra-"unset"}" = true ]] && modules=("legacy-extra")
419     fi
420
421     # Load presets
422     local _modules=() module_check
423     for_module '[[ -f "${preset_dir}/{}" ]] && readarray -t -O "${#_modules[@]}" _modules < <(cat "${preset_dir}/{}") || _modules+=("{}")'
424     modules=("${_modules[@]}")
425     unset _modules
426
427     # Check modules
428     module_check(){
429         msg_debug "Checking ${1} module ..."
430         bash "${tools_dir}/module.sh" check "${1}" || msg_error "Module ${1} is not available." "1";
431     }
432     readarray -t modules < <(printf "%s\n" "${modules[@]}" | awk '!a[$0]++')
433     for_module "module_check {}"
434
435     # Load modules
436     for_module load_config "${module_dir}/{}/config.any" "${module_dir}/{}/config.${arch}"
437     msg_debug "Loaded modules: ${modules[*]}"
438     ! printf "%s\n" "${modules[@]}" | grep -x "share" >/dev/null 2>&1 && msg_warn "The share module is not loaded."
439
440     # Set kernel
441     [[ "${customized_kernel}" = false ]] && kernel="${defaultkernel}"
442
443     # Parse files
444     eval "$(bash "${tools_dir}/locale.sh" -s -a "${arch}" get "${locale_name}")"
445     eval "$(bash "${tools_dir}/kernel.sh" -s -c "${channel_name}" -a "${arch}" get "${kernel}")"
446
447     # Set username and password
448     [[ "${customized_username}" = false ]] && username="${defaultusername}"
449     [[ "${customized_password}" = false ]] && password="${defaultpassword}"
450
451     # gitversion
452     [[ "${gitversion}" = true ]] && iso_version="${iso_version}-${gitrev}"
453
454     # Generate tar file name
455     tar_ext=""
456     case "${tar_comp}" in
457         "gzip" ) tar_ext="gz"                        ;;
458         "zstd" ) tar_ext="zst"                       ;;
459         "xz" | "lzo" | "lzma") tar_ext="${tar_comp}" ;;
460     esac
461
462     # Generate iso file name
463     local _channel_name="${channel_name%.add}-${locale_version}" 
464     iso_filename="${iso_name}-${_channel_name}-${iso_version}-${arch}.iso"
465     tar_filename="${iso_filename%.iso}.tar.${tar_ext}"
466     [[ "${nochname}" = true ]] && iso_filename="${iso_name}-${iso_version}-${arch}.iso"
467     msg_debug "Iso filename is ${iso_filename}"
468
469     # check bool
470     check_bool boot_splash cleaning noconfirm nodepend customized_username customized_password noloopmod nochname tarball noiso noaur customized_syslinux norescue_entry debug bash_debug nocolor msgdebug noefi nosigcheck gitversion
471
472     # Check architecture for each channel
473     local _exit=0
474     bash "${tools_dir}/channel.sh" --version "${alteriso_version}" -a "${arch}" -n -b check "${channel_name}" || _exit="${?}"
475     ( (( "${_exit}" != 0 )) && (( "${_exit}" != 1 )) ) && msg_error "${channel_name} channel does not support current architecture (${arch})." "1"
476
477     # Run with tee
478     if [[ ! "${logging}" = false ]]; then
479         [[ "${customized_logpath}" = false ]] && logging="${out_dir}/${iso_filename%.iso}.log"
480         mkdir -p "$(dirname "${logging}")" && touch "${logging}"
481         msg_warn "Re-run sudo ${0} ${ARGUMENT[*]} --nodepend --nolog --nocolor 2>&1 | tee ${logging}"
482         sudo "${0}" "${ARGUMENT[@]}" --nolog --nocolor --nodepend 2>&1 | tee "${logging}"
483         exit "${?}"
484     fi
485
486     # Set argument of pkglist.sh
487     pkglist_args=("-a" "${arch}" "-k" "${kernel}" "-c" "${channel_dir}" "-l" "${locale_name}" --line)
488     [[ "${boot_splash}"              = true ]] && pkglist_args+=("-b")
489     [[ "${debug}"                    = true ]] && pkglist_args+=("-d")
490     [[ "${memtest86}"                = true ]] && pkglist_args+=("-m")
491     [[ "${nocolor}"                  = true ]] && pkglist_args+=("--nocolor")
492     (( "${#additional_exclude_pkg[@]}" >= 1 )) && pkglist_args+=("-e" "${additional_exclude_pkg[*]}")
493     pkglist_args+=("${modules[@]}")
494
495     # Set argument of aur.sh and pkgbuild.sh
496     [[ "${bash_debug}"   = true ]] && makepkg_script_args+=("-x")
497     [[ "${pacman_debug}" = true ]] && makepkg_script_args+=("-d")
498
499     return 0
500 }
501
502
503 # Setup custom pacman.conf with current cache directories.
504 make_pacman_conf() {
505     # Pacman configuration file used only when building
506     # If there is pacman.conf for each channel, use that for building
507     local _pacman_conf _pacman_conf_list=("${script_path}/pacman-${arch}.conf" "${channel_dir}/pacman-${arch}.conf" "${script_path}/system/pacman-${arch}.conf")
508     for _pacman_conf in "${_pacman_conf_list[@]}"; do
509         if [[ -f "${_pacman_conf}" ]]; then
510             build_pacman_conf="${_pacman_conf}"
511             break
512         fi
513     done
514
515     msg_debug "Use ${build_pacman_conf}"
516     sed -r "s|^#?\\s*CacheDir.+|CacheDir     = ${cache_dir}|g" "${build_pacman_conf}" > "${build_dir}/pacman.conf"
517
518     [[ "${nosigcheck}" = true ]] && sed -ir "s|^s*SigLevel.+|SigLevel = Never|g" "${build_pacman_conf}"
519
520     [[ -n "$(find "${cache_dir}" -maxdepth 1 -name '*.pkg.tar.*' 2> /dev/null)" ]] && msg_info "Use cached package files in ${cache_dir}"
521
522     # Share any architecture packages
523     #while read -r _pkg; do
524     #    if [[ ! -f "${cache_dir}/$(basename "${_pkg}")" ]]; then
525     #        ln -s "${_pkg}" "${cache_dir}"
526     #    fi
527     #done < <(find "${cache_dir}/../" -type d -name "$(basename "${cache_dir}")" -prune -o -type f -name "*-any.pkg.tar.*" -printf "%p\n")
528
529     return 0
530 }
531
532 # Base installation (airootfs)
533 make_basefs() {
534     msg_info "Creating ext4 image of 32GiB..."
535     truncate -s 32G -- "${airootfs_dir}.img"
536     mkfs.ext4 -O '^has_journal,^resize_inode' -E 'lazy_itable_init=0' -m 0 -F -- "${airootfs_dir}.img" 32G
537     tune2fs -c "0" -i "0" "${airootfs_dir}.img"
538     msg_info "Done!"
539
540     msg_info "Mounting ${airootfs_dir}.img on ${airootfs_dir}"
541     mount_airootfs
542     msg_info "Done!"
543     return 0
544 }
545
546 # Additional packages (airootfs)
547 make_packages_repo() {
548     msg_debug "pkglist.sh ${pkglist_args[*]}"
549     readarray -t _pkglist_install < <("${tools_dir}/pkglist.sh" "${pkglist_args[@]}")
550
551     # Package check
552     #readarray -t _pkglist < <("${tools_dir}/pkglist.sh" "${pkglist_args[@]}")
553     #readarray -t repopkgs < <(pacman-conf -c "${build_pacman_conf}" -l | xargs -I{} pacman -Sql --config "${build_pacman_conf}" --color=never {} && pacman -Sg)
554     #local _pkg
555     #for _pkg in "${_pkglist[@]}"; do
556     #    msg_info "Checking ${_pkg}..."
557     #    if printf "%s\n" "${repopkgs[@]}" | grep -qx "${_pkg}"; then
558     #        _pkglist_install+=("${_pkg}")
559     #    else
560     #        msg_info "${_pkg} was not found. Install it with yay from AUR"
561     #        norepopkg+=("${_pkg}")
562     #    fi
563     #done
564
565     # Create a list of packages to be finally installed as packages.list directly under the working directory.
566     echo -e "# The list of packages that is installed in live cd.\n#\n" > "${build_dir}/packages.list"
567     printf "%s\n" "${_pkglist_install[@]}" >> "${build_dir}/packages.list"
568
569     # Install packages on airootfs
570     _pacstrap "${_pkglist_install[@]}"
571
572     return 0
573 }
574
575 make_packages_aur() {
576     readarray -t _pkglist_aur < <("${tools_dir}/pkglist.sh" --aur "${pkglist_args[@]}")
577     _pkglist_aur=("${_pkglist_aur[@]}" "${norepopkg[@]}")
578
579     # Create a list of packages to be finally installed as packages.list directly under the working directory.
580     echo -e "\n# AUR packages.\n#\n" >> "${build_dir}/packages.list"
581     printf "%s\n" "${_pkglist_aur[@]}" >> "${build_dir}/packages.list"
582
583     # prepare for yay
584     cp -rf --preserve=mode "${script_path}/system/aur.sh" "${airootfs_dir}/root/aur.sh"
585     _pacstrap --asdeps --needed "go" # --asdepsをつけているのでaur.shで削除される --neededをつけているので明示的にインストールされている場合削除されない
586
587     # Run aur script
588     _run_with_pacmanconf _chroot_run "bash" "/root/aur.sh" "${makepkg_script_args[@]}" "${_pkglist_aur[@]}"
589
590     # Remove script
591     remove "${airootfs_dir}/root/aur.sh"
592
593     return 0
594 }
595
596 make_pkgbuild() {
597     # Get PKGBUILD List
598     local _pkgbuild_dirs=("${channel_dir}/pkgbuild.any" "${channel_dir}/pkgbuild.${arch}")
599     for_module '_pkgbuild_dirs+=("${module_dir}/{}/pkgbuild.any" "${module_dir}/{}/pkgbuild.${arch}")'
600
601     # Copy PKGBUILD to work
602     mkdir -p "${airootfs_dir}/pkgbuilds/"
603     for _dir in $(find "${_pkgbuild_dirs[@]}" -type f -name "PKGBUILD" -print0 2>/dev/null | xargs -0 -I{} realpath {} | xargs -I{} dirname {}); do
604         msg_info "Find $(basename "${_dir}")"
605         cp -r "${_dir}" "${airootfs_dir}/pkgbuilds/"
606     done
607     
608     # copy buold script
609     cp -rf --preserve=mode "${script_path}/system/pkgbuild.sh" "${airootfs_dir}/root/pkgbuild.sh"
610
611     # Run build script
612     _run_with_pacmanconf _chroot_run "bash" "/root/pkgbuild.sh" "${makepkg_script_args[@]}" "/pkgbuilds"
613
614     # Remove script
615     remove "${airootfs_dir}/root/pkgbuild.sh"
616
617     return 0
618 }
619
620 # Customize installation (airootfs)
621 make_customize_airootfs() {
622     # Overwrite airootfs with customize_airootfs.
623     local _airootfs _airootfs_script_options _script _script_list _airootfs_list=() _main_script
624
625     for_module '_airootfs_list+=("${module_dir}/{}/airootfs.any" "${module_dir}/{}/airootfs.${arch}")'
626     _airootfs_list+=("${channel_dir}/airootfs.any" "${channel_dir}/airootfs.${arch}")
627
628     for _airootfs in "${_airootfs_list[@]}";do
629         if [[ -d "${_airootfs}" ]]; then
630             msg_debug "Copying airootfs ${_airootfs} ..."
631             cp -af "${_airootfs}"/* "${airootfs_dir}"
632         fi
633     done
634
635     # Replace /etc/mkinitcpio.conf if Plymouth is enabled.
636     if [[ "${boot_splash}" = true ]]; then
637         cp -f "${script_path}/mkinitcpio/mkinitcpio-plymouth.conf" "${airootfs_dir}/etc/mkinitcpio.conf"
638     else
639         cp -f "${script_path}/mkinitcpio/mkinitcpio.conf" "${airootfs_dir}/etc/mkinitcpio.conf"
640     fi
641     
642     # customize_airootfs options
643     # -b                        : Enable boot splash.
644     # -d                        : Enable debug mode.
645     # -g <locale_gen_name>      : Set locale-gen.
646     # -i <inst_dir>             : Set install dir
647     # -k <kernel config line>   : Set kernel name.
648     # -o <os name>              : Set os name.
649     # -p <password>             : Set password.
650     # -s <shell>                : Set user shell.
651     # -t                        : Set plymouth theme.
652     # -u <username>             : Set live user name.
653     # -x                        : Enable bash debug mode.
654     # -z <locale_time>          : Set the time zone.
655     # -l <locale_name>          : Set language.
656     #
657     # -j is obsolete in AlterISO3 and cannot be used.
658     # -r is obsolete due to the removal of rebuild.
659     # -k changed in AlterISO3 from passing kernel name to passing kernel configuration.
660
661     # Generate options of customize_airootfs.sh.
662     _airootfs_script_options="-p '${password}' -k '${kernel} ${kernel_filename} ${kernel_mkinitcpio_profile}' -u '${username}' -o '${os_name}' -i '${install_dir}' -s '${usershell}' -a '${arch}' -g '${locale_gen_name}' -l '${locale_name}' -z '${locale_time}' -t ${theme_name}"
663     [[ "${boot_splash}" = true ]] && _airootfs_script_options="${_airootfs_script_options} -b"
664     [[ "${debug}" = true       ]] && _airootfs_script_options="${_airootfs_script_options} -d"
665     [[ "${bash_debug}" = true  ]] && _airootfs_script_options="${_airootfs_script_options} -x"
666
667     _main_script="root/customize_airootfs.sh"
668
669     _script_list=(
670         "${airootfs_dir}/root/customize_airootfs_${channel_name}.sh"
671         "${airootfs_dir}/root/customize_airootfs_${channel_name%.add}.sh"
672     )
673
674     for_module '_script_list+=("${airootfs_dir}/root/customize_airootfs_{}.sh")'
675
676     # Create script
677     for _script in "${_script_list[@]}"; do
678         if [[ -f "${_script}" ]]; then
679             (echo && cat "${_script}")  >> "${airootfs_dir}/${_main_script}"
680             remove "${_script}"
681         else
682             msg_debug "${_script} was not found."
683         fi
684     done
685
686     chmod 755 "${airootfs_dir}/${_main_script}"
687     cp "${airootfs_dir}/${_main_script}" "${build_dir}/$(basename ${_main_script})"
688     _chroot_run "${_main_script} ${_airootfs_script_options}"
689     remove "${airootfs_dir}/${_main_script}"
690
691     # /root permission https://github.com/archlinux/archiso/commit/d39e2ba41bf556674501062742190c29ee11cd59
692     chmod -f 750 "${airootfs_dir}/root"
693
694     return 0
695 }
696
697 # Copy mkinitcpio archiso hooks and build initramfs (airootfs)
698 make_setup_mkinitcpio() {
699     local _hook
700     mkdir -p "${airootfs_dir}/etc/initcpio/hooks" "${airootfs_dir}/etc/initcpio/install"
701
702     for _hook in "archiso" "archiso_shutdown" "archiso_pxe_common" "archiso_pxe_nbd" "archiso_pxe_http" "archiso_pxe_nfs" "archiso_loop_mnt"; do
703         cp "${script_path}/system/initcpio/hooks/${_hook}" "${airootfs_dir}/etc/initcpio/hooks"
704         cp "${script_path}/system/initcpio/install/${_hook}" "${airootfs_dir}/etc/initcpio/install"
705     done
706
707     sed -i "s|/usr/lib/initcpio/|/etc/initcpio/|g" "${airootfs_dir}/etc/initcpio/install/archiso_shutdown"
708     cp "${script_path}/system/initcpio/install/archiso_kms" "${airootfs_dir}/etc/initcpio/install"
709     cp "${script_path}/system/initcpio/archiso_shutdown" "${airootfs_dir}/etc/initcpio"
710     if [[ "${boot_splash}" = true ]]; then
711         cp "${script_path}/mkinitcpio/mkinitcpio-archiso-plymouth.conf" "${airootfs_dir}/etc/mkinitcpio-archiso.conf"
712     else
713         cp "${script_path}/mkinitcpio/mkinitcpio-archiso.conf" "${airootfs_dir}/etc/mkinitcpio-archiso.conf"
714     fi
715     if [[ "${gpg_key}" ]]; then
716       gpg --export "${gpg_key}" >"${build_dir}/gpgkey"
717       exec 17<>"${build_dir}/gpgkey"
718     fi
719
720     _chroot_run "mkinitcpio -c /etc/mkinitcpio-archiso.conf -k /boot/${kernel_filename} -g /boot/archiso.img"
721
722     [[ "${gpg_key}" ]] && exec 17<&-
723     
724     return 0
725 }
726
727 # Prepare kernel/initramfs ${install_dir}/boot/
728 make_boot() {
729     mkdir -p "${isofs_dir}/${install_dir}/boot/${arch}"
730     cp "${airootfs_dir}/boot/archiso.img" "${isofs_dir}/${install_dir}/boot/${arch}/archiso.img"
731     cp "${airootfs_dir}/boot/${kernel_filename}" "${isofs_dir}/${install_dir}/boot/${arch}/${kernel_filename}"
732
733     return 0
734 }
735
736 # Add other aditional/extra files to ${install_dir}/boot/
737 make_boot_extra() {
738     if [[ -e "${airootfs_dir}/boot/memtest86+/memtest.bin" ]]; then
739         install -m 0644 -- "${airootfs_dir}/boot/memtest86+/memtest.bin" "${isofs_dir}/${install_dir}/boot/memtest"
740         install -d -m 0755 -- "${isofs_dir}/${install_dir}/boot/licenses/memtest86+/"
741         install -m 0644 -- "${airootfs_dir}/usr/share/licenses/common/GPL2/license.txt" "${isofs_dir}/${install_dir}/boot/licenses/memtest86+/"
742     fi
743
744     local _ucode_image
745     msg_info "Preparing microcode for the ISO 9660 file system..."
746
747     for _ucode_image in {intel-uc.img,intel-ucode.img,amd-uc.img,amd-ucode.img,early_ucode.cpio,microcode.cpio}; do
748         if [[ -e "${airootfs_dir}/boot/${_ucode_image}" ]]; then
749             msg_info "Installimg ${_ucode_image} ..."
750             install -m 0644 -- "${airootfs_dir}/boot/${_ucode_image}" "${isofs_dir}/${install_dir}/boot/"
751             if [[ -e "${airootfs_dir}/usr/share/licenses/${_ucode_image%.*}/" ]]; then
752                 install -d -m 0755 -- "${isofs_dir}/${install_dir}/boot/licenses/${_ucode_image%.*}/"
753                 install -m 0644 -- "${airootfs_dir}/usr/share/licenses/${_ucode_image%.*}/"* "${isofs_dir}/${install_dir}/boot/licenses/${_ucode_image%.*}/"
754             fi
755         fi
756     done
757     msg_info "Done!"
758
759     return 0
760 }
761
762 # Prepare /${install_dir}/boot/syslinux
763 make_syslinux() {
764     mkdir -p "${isofs_dir}/syslinux"
765
766     # 一時ディレクトリに設定ファイルをコピー
767     mkdir -p "${build_dir}/syslinux/"
768     cp -a "${script_path}/syslinux/"* "${build_dir}/syslinux/"
769     if [[ -d "${channel_dir}/syslinux" ]] && [[ "${customized_syslinux}" = true ]]; then
770         cp -af "${channel_dir}/syslinux"* "${build_dir}/syslinux/"
771     fi
772
773     # copy all syslinux config to work dir
774     for _cfg in "${build_dir}/syslinux/"*.cfg; do
775         sed "s|%ARCHISO_LABEL%|${iso_label}|g;
776              s|%OS_NAME%|${os_name}|g;
777              s|%KERNEL_FILENAME%|${kernel_filename}|g;
778              s|%ARCH%|${arch}|g;
779              s|%INSTALL_DIR%|${install_dir}|g" "${_cfg}" > "${isofs_dir}/syslinux/${_cfg##*/}"
780     done
781
782     # Replace the SYSLINUX configuration file with or without boot splash.
783     local _use_config_name _no_use_config_name _pxe_or_sys
784     if [[ "${boot_splash}" = true ]]; then
785         _use_config_name=splash
786         _no_use_config_name=nosplash
787     else
788         _use_config_name=nosplash
789         _no_use_config_name=splash
790     fi
791     for _pxe_or_sys in "sys" "pxe"; do
792         remove "${isofs_dir}/syslinux/archiso_${_pxe_or_sys}_${_no_use_config_name}.cfg"
793         mv "${isofs_dir}/syslinux/archiso_${_pxe_or_sys}_${_use_config_name}.cfg" "${isofs_dir}/syslinux/archiso_${_pxe_or_sys}.cfg"
794     done
795
796     # Set syslinux wallpaper
797     if [[ -f "${channel_dir}/splash.png" ]]; then
798         cp "${channel_dir}/splash.png" "${isofs_dir}/syslinux"
799     else
800         cp "${script_path}/syslinux/splash.png" "${isofs_dir}/syslinux"
801     fi
802
803     # remove config
804     local _remove_config
805     function _remove_config() {
806         remove "${isofs_dir}/syslinux/${1}"
807         sed -i "s|$(grep "${1}" "${isofs_dir}/syslinux/archiso_sys_load.cfg")||g" "${isofs_dir}/syslinux/archiso_sys_load.cfg" 
808     }
809
810     [[ "${norescue_entry}" = true  ]] && _remove_config archiso_sys_rescue.cfg
811     [[ "${memtest86}"      = false ]] && _remove_config memtest86.cfg
812
813     # copy files
814     cp "${airootfs_dir}/usr/lib/syslinux/bios/"*.c32 "${isofs_dir}/syslinux"
815     cp "${airootfs_dir}/usr/lib/syslinux/bios/lpxelinux.0" "${isofs_dir}/syslinux"
816     cp "${airootfs_dir}/usr/lib/syslinux/bios/memdisk" "${isofs_dir}/syslinux"
817
818
819     if [[ -e "${isofs_dir}/syslinux/hdt.c32" ]]; then
820         install -d -m 0755 -- "${isofs_dir}/syslinux/hdt"
821         if [[ -e "${airootfs_dir}/usr/share/hwdata/pci.ids" ]]; then
822             gzip -c -9 "${airootfs_dir}/usr/share/hwdata/pci.ids" > "${isofs_dir}/syslinux/hdt/pciids.gz"
823         fi
824         find "${airootfs_dir}/usr/lib/modules" -name 'modules.alias' -print -exec gzip -c -9 '{}' ';' -quit > "${isofs_dir}/syslinux/hdt/modalias.gz"
825     fi
826
827     return 0
828 }
829
830 # Prepare /isolinux
831 make_isolinux() {
832     install -d -m 0755 -- "${isofs_dir}/syslinux"
833     sed "s|%INSTALL_DIR%|${install_dir}|g" "${script_path}/system/isolinux.cfg" > "${isofs_dir}/syslinux/isolinux.cfg"
834     install -m 0644 -- "${airootfs_dir}/usr/lib/syslinux/bios/isolinux.bin" "${isofs_dir}/syslinux/"
835     install -m 0644 -- "${airootfs_dir}/usr/lib/syslinux/bios/isohdpfx.bin" "${isofs_dir}/syslinux/"
836
837     return 0
838 }
839
840 # Prepare /EFI
841 make_efi() {
842     local _bootfile _use_config_name="nosplash" _efi_config_list=() _efi_config
843     [[ "${boot_splash}" = true ]] && _use_config_name="splash"
844     _bootfile="$(basename "$(ls "${airootfs_dir}/usr/lib/systemd/boot/efi/systemd-boot"*".efi" )")"
845
846     install -d -m 0755 -- "${isofs_dir}/EFI/boot"
847     install -m 0644 -- "${airootfs_dir}/usr/lib/systemd/boot/efi/${_bootfile}" "${isofs_dir}/EFI/boot/${_bootfile#systemd-}"
848
849     install -d -m 0755 -- "${isofs_dir}/loader/entries"
850     sed "s|%ARCH%|${arch}|g;" "${script_path}/efiboot/${_use_config_name}/loader.conf" > "${isofs_dir}/loader/loader.conf"
851
852     readarray -t _efi_config_list < <(find "${script_path}/efiboot/${_use_config_name}/" -mindepth 1 -maxdepth 1 -type f -name "archiso-usb*.conf" -printf "%f\n" | grep -v "rescue")
853     [[ "${norescue_entry}" = false ]] && readarray -t _efi_config_list < <(find "${script_path}/efiboot/${_use_config_name}/" -mindepth 1 -maxdepth 1 -type f  -name "archiso-usb*.conf" -printf "%f\n")
854
855     for _efi_config in "${_efi_config_list[@]}"; do
856         sed "s|%ARCHISO_LABEL%|${iso_label}|g;
857             s|%OS_NAME%|${os_name}|g;
858             s|%KERNEL_FILENAME%|${kernel_filename}|g;
859             s|%ARCH%|${arch}|g;
860             s|%INSTALL_DIR%|${install_dir}|g" \
861         "${script_path}/efiboot/${_use_config_name}/${_efi_config}" > "${isofs_dir}/loader/entries/$(basename "${_efi_config}" | sed "s|usb|${arch}|g")"
862     done
863
864     # edk2-shell based UEFI shell
865     local _efi_shell_arch
866     if [[ -d "${airootfs_dir}/usr/share/edk2-shell" ]]; then
867         for _efi_shell_arch in $(find "${airootfs_dir}/usr/share/edk2-shell" -mindepth 1 -maxdepth 1 -type d -print0 | xargs -0 -I{} basename {}); do
868             if [[ -f "${airootfs_dir}/usr/share/edk2-shell/${_efi_shell_arch}/Shell_Full.efi" ]]; then
869                 cp "${airootfs_dir}/usr/share/edk2-shell/${_efi_shell_arch}/Shell_Full.efi" "${isofs_dir}/EFI/shell_${_efi_shell_arch}.efi"
870             elif [[ -f "${airootfs_dir}/usr/share/edk2-shell/${_efi_shell_arch}/Shell.efi" ]]; then
871                 cp "${airootfs_dir}/usr/share/edk2-shell/${_efi_shell_arch}/Shell.efi" "${isofs_dir}/EFI/shell_${_efi_shell_arch}.efi"
872             else
873                 continue
874             fi
875             echo -e "title  UEFI Shell ${_efi_shell_arch}\nefi    /EFI/shell_${_efi_shell_arch}.efi" > "${isofs_dir}/loader/entries/uefi-shell-${_efi_shell_arch}.conf"
876         done
877     fi
878
879     return 0
880 }
881
882 # Prepare efiboot.img::/EFI for "El Torito" EFI boot mode
883 make_efiboot() {
884     truncate -s 128M "${build_dir}/efiboot.img"
885     mkfs.fat -n ARCHISO_EFI "${build_dir}/efiboot.img"
886
887     mkdir -p "${build_dir}/efiboot"
888     mount "${build_dir}/efiboot.img" "${build_dir}/efiboot"
889
890     mkdir -p "${build_dir}/efiboot/EFI/alteriso/${arch}" "${build_dir}/efiboot/EFI/boot" "${build_dir}/efiboot/loader/entries"
891     cp "${isofs_dir}/${install_dir}/boot/${arch}/${kernel_filename}" "${build_dir}/efiboot/EFI/alteriso/${arch}/${kernel_filename}.efi"
892     cp "${isofs_dir}/${install_dir}/boot/${arch}/archiso.img" "${build_dir}/efiboot/EFI/alteriso/${arch}/archiso.img"
893
894     local _ucode_image _efi_config _use_config_name="nosplash" _bootfile
895     for _ucode_image in "${airootfs_dir}/boot/"{intel-uc.img,intel-ucode.img,amd-uc.img,amd-ucode.img,early_ucode.cpio,microcode.cpio}; do
896         [[ -e "${_ucode_image}" ]] && cp "${_ucode_image}" "${build_dir}/efiboot/EFI/alteriso/"
897     done
898
899     cp "${airootfs_dir}/usr/share/efitools/efi/HashTool.efi" "${build_dir}/efiboot/EFI/boot/"
900
901     _bootfile="$(basename "$(ls "${airootfs_dir}/usr/lib/systemd/boot/efi/systemd-boot"*".efi" )")"
902     cp "${airootfs_dir}/usr/lib/systemd/boot/efi/${_bootfile}" "${build_dir}/efiboot/EFI/boot/${_bootfile#systemd-}"
903
904     [[ "${boot_splash}" = true ]] && _use_config_name="splash"
905     sed "s|%ARCH%|${arch}|g;" "${script_path}/efiboot/${_use_config_name}/loader.conf" > "${build_dir}/efiboot/loader/loader.conf"
906
907     find "${isofs_dir}/loader/entries/" -maxdepth 1 -mindepth 1 -name "uefi-shell*" -type f -printf "%p\0" | xargs -0 -I{} cp {} "${build_dir}/efiboot/loader/entries/"
908
909     readarray -t _efi_config_list < <(find "${script_path}/efiboot/${_use_config_name}/" -mindepth 1 -maxdepth 1 -type f -name "archiso-cd*.conf" -printf "%f\n" | grep -v "rescue")
910     [[ "${norescue_entry}" = false ]] && readarray -t _efi_config_list < <(find "${script_path}/efiboot/${_use_config_name}/" -mindepth 1 -maxdepth 1 -type f  -name "archiso-cd*.conf" -printf "%f\n")
911
912     for _efi_config in "${_efi_config_list[@]}"; do
913         sed "s|%ARCHISO_LABEL%|${iso_label}|g;
914             s|%OS_NAME%|${os_name}|g;
915             s|%KERNEL_FILENAME%|${kernel_filename}|g;
916             s|%ARCH%|${arch}|g;
917             s|%INSTALL_DIR%|${install_dir}|g" \
918         "${script_path}/efiboot/${_use_config_name}/${_efi_config}" > "${build_dir}/efiboot/loader/entries/$(basename "${_efi_config}" | sed "s|cd|${arch}|g")"
919     done
920
921     find "${isofs_dir}/EFI" -maxdepth 1 -mindepth 1 -name "shell*.efi" -printf "%p\0" | xargs -0 -I{} cp {} "${build_dir}/efiboot/EFI/"
922     umount -d "${build_dir}/efiboot"
923
924     return 0
925 }
926
927 # Compress tarball
928 make_tarball() {
929     # backup airootfs.img for tarball
930     msg_debug "Tarball filename is ${tar_filename}"
931     msg_info "Copying airootfs.img ..."
932     cp "${airootfs_dir}.img" "${airootfs_dir}.img.org"
933
934     # Run script
935     mount_airootfs
936     [[ -f "${airootfs_dir}/root/optimize_for_tarball.sh" ]] && _chroot_run "bash /root/optimize_for_tarball.sh -u ${username}"
937     _cleanup_common
938     _chroot_run "mkinitcpio -P"
939     remove "${airootfs_dir}/root/optimize_for_tarball.sh"
940
941     # make
942     tar_comp_opt+=("--${tar_comp}")
943     mkdir -p "${out_dir}"
944     msg_info "Creating tarball..."
945     cd -- "${airootfs_dir}"
946     msg_debug "Run tar -c -v -p -f \"${out_dir}/${tar_filename}\" ${tar_comp_opt[*]} ./*"
947     tar -c -v -p -f "${out_dir}/${tar_filename}" "${tar_comp_opt[@]}" ./*
948     cd -- "${OLDPWD}"
949
950     # checksum
951     _mkchecksum "${out_dir}/${tar_filename}"
952     msg_info "Done! | $(ls -sh "${out_dir}/${tar_filename}")"
953
954     remove "${airootfs_dir}.img"
955     mv "${airootfs_dir}.img.org" "${airootfs_dir}.img"
956
957     [[ "${noiso}" = true ]] && msg_info "The password for the live user and root is ${password}."
958     
959     return 0
960 }
961
962
963 # Build airootfs filesystem image
964 make_prepare() {
965     mount_airootfs
966
967     # Create packages list
968     msg_info "Creating a list of installed packages on live-enviroment..."
969     pacman-key --init
970     pacman -Q --sysroot "${airootfs_dir}" | tee "${isofs_dir}/${install_dir}/pkglist.${arch}.txt" "${build_dir}/packages-full.list" > /dev/null
971
972     # Cleanup
973     remove "${airootfs_dir}/root/optimize_for_tarball.sh"
974     _cleanup_airootfs
975
976     # Create squashfs
977     mkdir -p -- "${isofs_dir}/${install_dir}/${arch}"
978     msg_info "Creating SquashFS image, this may take some time..."
979     mksquashfs "${airootfs_dir}" "${build_dir}/iso/${install_dir}/${arch}/airootfs.sfs" -noappend -comp "${sfs_comp}" "${sfs_comp_opt[@]}"
980
981     # Create checksum
982     msg_info "Creating checksum file for self-test..."
983     echo "$(sha512sum "${isofs_dir}/${install_dir}/${arch}/airootfs.sfs" | getclm 1) airootfs.sfs" > "${isofs_dir}/${install_dir}/${arch}/airootfs.sha512"
984     msg_info "Done!"
985
986     # Sign with gpg
987     if [[ -v gpg_key ]] && (( "${#gpg_key}" != 0 )); then
988         msg_info "Creating signature file ($gpg_key) ..."
989         cd -- "${isofs_dir}/${install_dir}/${arch}"
990         gpg --detach-sign --default-key "${gpg_key}" "airootfs.sfs"
991         cd -- "${OLDPWD}"
992         msg_info "Done!"
993     fi
994
995     umount_work
996
997     [[ "${cleaning}" = true ]] && remove "${airootfs_dir}" "${airootfs_dir}.img"
998
999     return 0
1000 }
1001
1002 make_alteriso_info(){
1003     # iso version info
1004     if [[ "${include_info}" = true ]]; then
1005         local _info_file="${isofs_dir}/alteriso-info" _version="${iso_version}"
1006         remove "${_info_file}"; touch "${_info_file}"
1007         [[ -d "${script_path}/.git" ]] && [[ "${gitversion}" = false ]] && _version="${iso_version}-${gitrev}"
1008         "${tools_dir}/alteriso-info.sh" -a "${arch}" -b "${boot_splash}" -c "${channel_name%.add}" -d "${iso_publisher}" -k "${kernel}" -o "${os_name}" -p "${password}" -u "${username}" -v "${_version}" > "${_info_file}"
1009     fi
1010
1011     return 0
1012 }
1013
1014 # Add files to the root of isofs
1015 make_overisofs() {
1016     local _over_isofs_list _isofs
1017     _over_isofs_list=("${channel_dir}/over_isofs.any""${channel_dir}/over_isofs.${arch}")
1018     for_module '_over_isofs_list+=("${module_dir}/{}/over_isofs.any""${module_dir}/{}/over_isofs.${arch}")'
1019     for _isofs in "${_over_isofs_list[@]}"; do
1020         [[ -d "${_isofs}" ]] && cp -af "${_isofs}"/* "${isofs_dir}"
1021     done
1022
1023     return 0
1024 }
1025
1026 # Build ISO
1027 make_iso() {
1028     local _iso_efi_boot_args=()
1029     # If exists, add an EFI "El Torito" boot image (FAT filesystem) to ISO-9660 image.
1030     [[ -f "${build_dir}/efiboot.img" ]] && _iso_efi_boot_args=(-append_partition 2 C12A7328-F81F-11D2-BA4B-00A0C93EC93B "${build_dir}/efiboot.img" -appended_part_as_gpt -eltorito-alt-boot -e --interval:appended_partition_2:all:: -no-emul-boot -isohybrid-gpt-basdat)
1031
1032     mkdir -p -- "${out_dir}"
1033     msg_info "Creating ISO image..."
1034     xorriso -as mkisofs \
1035         -iso-level 3 \
1036         -full-iso9660-filenames \
1037         -joliet \
1038         -joliet-long \
1039         -rational-rock \
1040         -volid "${iso_label}" \
1041         -appid "${iso_application}" \
1042         -publisher "${iso_publisher}" \
1043         -preparer "prepared by AlterISO" \
1044         -eltorito-boot syslinux/isolinux.bin \
1045         -eltorito-catalog syslinux/boot.cat \
1046         -no-emul-boot -boot-load-size 4 -boot-info-table \
1047         -isohybrid-mbr "${build_dir}/iso/syslinux/isohdpfx.bin" \
1048         "${_iso_efi_boot_args[@]}" \
1049         -output "${out_dir}/${iso_filename}" \
1050         "${build_dir}/iso/"
1051     _mkchecksum "${out_dir}/${iso_filename}"
1052     msg_info "Done! | $(ls -sh -- "${out_dir}/${iso_filename}")"
1053
1054     msg_info "The password for the live user and root is ${password}."
1055
1056     return 0
1057 }
1058
1059
1060 # Parse options
1061 ARGUMENT=("${DEFAULT_ARGUMENT[@]}" "${@}")
1062 OPTS=("a:" "b" "c:" "d" "e" "g:" "h" "j" "k:" "l:" "o:" "p:" "r" "t:" "u:" "w:" "x")
1063 OPTL=("arch:" "boot-splash" "comp-type:" "debug" "cleaning" "cleanup" "gpgkey:" "help" "lang:" "japanese" "kernel:" "out:" "password:" "comp-opts:" "user:" "work:" "bash-debug" "nocolor" "noconfirm" "nodepend" "gitversion" "msgdebug" "noloopmod" "tarball" "noiso" "noaur" "nochkver" "channellist" "config:" "noefi" "nodebug" "nosigcheck" "normwork" "log" "logpath:" "nolog" "nopkgbuild" "pacman-debug" "confirm" "tar-type:" "tar-opts:" "add-module:")
1064 OPT="$(getopt -o "$(printf "%s," "${OPTS[@]}")" -l "$(printf "%s," "${OPTL[@]}")" --  "${ARGUMENT[@]}")" || exit 1
1065
1066 eval set -- "${OPT}"
1067 msg_debug "Argument: ${OPT}"
1068 unset OPT OPTS OPTL DEFAULT_ARGUMENT
1069
1070 while true; do
1071     case "${1}" in
1072         -a | --arch)
1073             arch="${2}"
1074             shift 2
1075             ;;
1076         -b | --boot-splash)
1077             boot_splash=true
1078             shift 1
1079             ;;
1080         -c | --comp-type)
1081             case "${2}" in
1082                 "gzip" | "lzma" | "lzo" | "lz4" | "xz" | "zstd") sfs_comp="${2}" ;;
1083                 *) msg_error "Invaild compressors '${2}'" '1' ;;
1084             esac
1085             shift 2
1086             ;;
1087         -d | --debug)
1088             debug=true
1089             shift 1
1090             ;;
1091         -e | --cleaning | --cleanup)
1092             cleaning=true
1093             shift 1
1094             ;;
1095         -g | --gpgkey)
1096             gpg_key="${2}"
1097             shift 2
1098             ;;
1099         -h | --help)
1100             _usage 0
1101             ;;
1102         -j | --japanese)
1103             msg_error "This option is obsolete in AlterISO 3. To use Japanese, use \"-l ja\"." "1"
1104             ;;
1105         -k | --kernel)
1106             customized_kernel=true
1107             kernel="${2}"
1108             shift 2
1109             ;;
1110         -l | --lang)
1111             locale_name="${2}"
1112             shift 2
1113             ;;
1114         -o | --out)
1115             out_dir="${2}"
1116             shift 2
1117             ;;
1118         -p | --password)
1119             customized_password=true
1120             password="${2}"
1121             shift 2
1122             ;;
1123         -r | --tarball)
1124             tarball=true
1125             shift 1
1126             ;;
1127         -t | --comp-opts)
1128             if [[ "${2}" = "reset" ]]; then
1129                 sfs_comp_opt=()
1130             else
1131                 IFS=" " read -r -a sfs_comp_opt <<< "${2}"
1132             fi
1133             shift 2
1134             ;;
1135         -u | --user)
1136             customized_username=true
1137             username="$(echo -n "${2}" | sed 's/ //g' | tr '[:upper:]' '[:lower:]')"
1138             shift 2
1139             ;;
1140         -w | --work)
1141             work_dir="${2}"
1142             shift 2
1143             ;;
1144         -x | --bash-debug)
1145             debug=true
1146             bash_debug=true
1147             shift 1
1148             ;;
1149         --noconfirm)
1150             noconfirm=true
1151             shift 1
1152             ;;
1153         --confirm)
1154             noconfirm=false
1155             shift 1
1156             ;;
1157         --nodepend)
1158             nodepend=true
1159             shift 1
1160             ;;
1161         --nocolor)
1162             nocolor=true
1163             shift 1
1164             ;;
1165         --gitversion)
1166             if [[ -d "${script_path}/.git" ]]; then
1167                 gitversion=true
1168             else
1169                 msg_error "There is no git directory. You need to use git clone to use this feature." "1"
1170             fi
1171             shift 1
1172             ;;
1173         --msgdebug)
1174             msgdebug=true;
1175             shift 1
1176             ;;
1177         --noloopmod)
1178             noloopmod=true
1179             shift 1
1180             ;;
1181         --noiso)
1182             noiso=true
1183             shift 1
1184             ;;
1185         --noaur)
1186             noaur=true
1187             shift 1
1188             ;;
1189         --nochkver)
1190             nochkver=true
1191             shift 1
1192             ;;
1193         --nodebug)
1194             debug=false
1195             msgdebug=false
1196             bash_debug=false
1197             shift 1
1198             ;;
1199         --noefi)
1200             noefi=true
1201             shift 1
1202             ;;
1203         --channellist)
1204             show_channel_list
1205             exit 0
1206             ;;
1207         --config)
1208             source "${2}"
1209             shift 2
1210             ;;
1211         --pacman-debug)
1212             pacman_debug=true
1213             shift 1
1214             ;;
1215         --nosigcheck)
1216             nosigcheck=true
1217             shift 1
1218             ;;
1219         --normwork)
1220             normwork=true
1221             shift 1
1222             ;;
1223         --log)
1224             logging=true
1225             shift 1
1226             ;;
1227         --logpath)
1228             logging="${2}"
1229             customized_logpath=true
1230             shift 2
1231             ;;
1232         --nolog)
1233             logging=false
1234             shift 1
1235             ;;
1236         --nopkgbuild)
1237             nopkgbuild=true
1238             shift 1
1239             ;;
1240         --tar-type)
1241             case "${2}" in
1242                 "gzip" | "lzma" | "lzo" | "lz4" | "xz" | "zstd") tar_comp="${2}" ;;
1243                 *) msg_error "Invaild compressors '${2}'" '1' ;;
1244             esac
1245             shift 2
1246             ;;
1247         --tar-opts)
1248             IFS=" " read -r -a tar_comp_opt <<< "${2}"
1249             shift 2
1250             ;;
1251         --add-module)
1252             readarray -t -O "${#additional_modules[@]}" additional_modules < <(echo "${2}" | tr "," "\n")
1253             msg_debug "Added modules: ${additional_modules[*]}"
1254             shift 2
1255             ;;
1256         --)
1257             shift
1258             break
1259             ;;
1260         *)
1261             msg_error "Argument exception error '${1}'"
1262             msg_error "Please report this error to the developer." 1
1263             ;;
1264     esac
1265 done
1266
1267 # Check root.
1268 if (( ! "${EUID}" == 0 )); then
1269     msg_warn "This script must be run as root." >&2
1270     msg_warn "Re-run 'sudo ${0} ${ARGUMENT[*]}'"
1271     sudo "${0}" "${ARGUMENT[@]}"
1272     exit "${?}"
1273 fi
1274
1275 # Show config message
1276 msg_debug "Use the default configuration file (${defaultconfig})."
1277 [[ -f "${script_path}/custom.conf" ]] && msg_debug "The default settings have been overridden by custom.conf"
1278
1279 # Debug mode
1280 [[ "${bash_debug}" = true ]] && set -x -v
1281
1282 # Check for a valid channel name
1283 if [[ -n "${1+SET}" ]]; then
1284     case "$(bash "${tools_dir}/channel.sh" --version "${alteriso_version}" -n check "${1}"; printf "%d" "${?}")" in
1285         "2")
1286             msg_error "Invalid channel ${1}" "1"
1287             ;;
1288         "1")
1289             channel_dir="${1}"
1290             channel_name="$(basename "${1%/}")"
1291             ;;
1292         "0")
1293             channel_dir="${script_path}/channels/${1}"
1294             channel_name="${1}"
1295             ;;
1296     esac
1297 else
1298     channel_dir="${script_path}/channels/${channel_name}"
1299 fi
1300
1301 # Set vars
1302 build_dir="${work_dir}/build/${arch}" cache_dir="${work_dir}/cache/${arch}" airootfs_dir="${build_dir}/airootfs" isofs_dir="${build_dir}/iso" lockfile_dir="${build_dir}/lockfile" gitrev="$(cd "${script_path}"; git rev-parse --short HEAD)" preset_dir="${script_path}/presets"
1303
1304 # Create dir
1305 for _dir in build_dir cache_dir airootfs_dir isofs_dir lockfile_dir out_dir; do
1306     mkdir -p "$(eval "echo \$${_dir}")"
1307     msg_debug "${_dir} is $(realpath "$(eval "echo \$${_dir}")")"
1308     eval "${_dir}=\"$(realpath "$(eval "echo \$${_dir}")")\""
1309 done
1310
1311
1312 # Set for special channels
1313 if [[ -d "${channel_dir}.add" ]]; then
1314     channel_name="${1}"
1315     channel_dir="${channel_dir}.add"
1316 elif [[ "${channel_name}" = "clean" ]]; then
1317    _run_cleansh
1318     exit 0
1319 fi
1320
1321 # Check channel version
1322 msg_debug "channel path is ${channel_dir}"
1323 if [[ ! "$(bash "${tools_dir}/channel.sh" --version "${alteriso_version}" ver "${channel_name}" | cut -d "." -f 1)" = "$(echo "${alteriso_version}" | cut -d "." -f 1)" ]] && [[ "${nochkver}" = false ]]; then
1324     msg_error "This channel does not support Alter ISO 3."
1325     if [[ -d "${script_path}/.git" ]]; then
1326         msg_error "Please run \"git checkout alteriso-2\"" "1"
1327     else
1328         msg_error "Please download old version here.\nhttps://github.com/FascodeNet/alterlinux/releases" "1"
1329     fi
1330 fi
1331
1332 set -eu
1333
1334 prepare_env
1335 prepare_build
1336 show_settings
1337 run_once make_pacman_conf
1338 run_once make_basefs # Mount airootfs
1339 run_once make_packages_repo
1340 [[ "${noaur}" = false ]] && run_once make_packages_aur
1341 [[ "${nopkgbuild}" = false ]] && run_once make_pkgbuild
1342 run_once make_customize_airootfs
1343 run_once make_setup_mkinitcpio
1344 [[ "${tarball}" = true ]] && run_once make_tarball
1345 if [[ "${noiso}" = false ]]; then
1346     run_once make_syslinux
1347     run_once make_isolinux
1348     run_once make_boot
1349     run_once make_boot_extra
1350     if [[ "${noefi}" = false ]]; then
1351         run_once make_efi
1352         run_once make_efiboot
1353     fi
1354     run_once make_alteriso_info
1355     run_once make_prepare
1356     run_once make_overisofs
1357     run_once make_iso
1358 fi
1359
1360 [[ "${cleaning}" = true ]] && _run_cleansh
1361
1362 exit 0