mirror of
				https://github.com/flatcar/scripts.git
				synced 2025-10-25 14:21:36 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			1178 lines
		
	
	
		
			35 KiB
		
	
	
	
		
			Bash
		
	
	
	
	
	
			
		
		
	
	
			1178 lines
		
	
	
		
			35 KiB
		
	
	
	
		
			Bash
		
	
	
	
	
	
| #!/bin/bash
 | |
| 
 | |
| # This file implements parsing the md5-metadata cache files and
 | |
| # accessing the parsed results. Not the entirety of the cache file is
 | |
| # parsed, only the parts that were needed at the time of writing
 | |
| # it. So currently the exposed parts of parsed cache files are EAPI,
 | |
| # IUSE, KEYWORDS, LICENSE, {B,R,P,I,}DEPEND and _eclasses_. The
 | |
| # _eclasses_ part discards the checksums, though, so only names are
 | |
| # available.
 | |
| 
 | |
| if [[ -z ${__MD5_CACHE_LIB_SH_INCLUDED__:-} ]]; then
 | |
| __MD5_CACHE_LIB_SH_INCLUDED__=x
 | |
| 
 | |
| source "$(dirname "${BASH_SOURCE[0]}")/util.sh"
 | |
| source "${PKG_AUTO_IMPL_DIR}/debug.sh"
 | |
| source "${PKG_AUTO_IMPL_DIR}/gentoo_ver.sh"
 | |
| 
 | |
| #
 | |
| # Cache file
 | |
| #
 | |
| 
 | |
| # Indices to access fields of the cache file:
 | |
| # PCF_EAPI_IDX     - a string/number describing EAPI
 | |
| # PCF_KEYWORDS_IDX - a name of an array containing keyword objects
 | |
| # PCF_IUSE_IDX     - a name of an array containing iuse objects
 | |
| # PCF_BDEPEND_IDX  - a name of a group object with build dependencies
 | |
| # PCF_DEPEND_IDX   - a name of a group object with dependencies
 | |
| # PCF_IDEPEND_IDX  - a name of a group object with install dependencies
 | |
| # PCF_PDEPEND_IDX  - a name of a group object with post dependencies
 | |
| # PCF_RDEPEND_IDX  - a name of a group object with runtime dependencies
 | |
| # PCF_LICENSE_IDX  - a name of a group object with licenses
 | |
| # PCF_ECLASSES_IDX - a name of an array with used eclasses
 | |
| declare -gri PCF_EAPI_IDX=0 PCF_KEYWORDS_IDX=1 PCF_IUSE_IDX=2 PCF_BDEPEND_IDX=3 PCF_DEPEND_IDX=4 PCF_IDEPEND_IDX=5 PCF_PDEPEND_IDX=6 PCF_RDEPEND_IDX=7 PCF_LICENSE_IDX=8 PCF_ECLASSES_IDX=9
 | |
| 
 | |
| # Declares empty cache files. Can take flags that are passed to
 | |
| # declare. Usually only -r or -t make sense to pass. Can take several
 | |
| # names, just like declare. Takes no initializers - this is hardcoded.
 | |
| function cache_file_declare() {
 | |
|     struct_declare -ga "${@}" "( '0' 'EMPTY_ARRAY' 'EMPTY_ARRAY' 'EMPTY_GROUP' 'EMPTY_GROUP' 'EMPTY_GROUP' 'EMPTY_GROUP' 'EMPTY_GROUP' 'EMPTY_GROUP' 'EMPTY_ARRAY' )"
 | |
| }
 | |
| 
 | |
| # Unsets cache files, can take several names, just like unset.
 | |
| function cache_file_unset() {
 | |
|     local name
 | |
|     for name; do
 | |
|         local -n cache_file_ref=${name}
 | |
| 
 | |
|         __mcl_unset_array "${cache_file_ref[PCF_KEYWORDS_IDX]}" kw_unset
 | |
|         __mcl_unset_array "${cache_file_ref[PCF_IUSE_IDX]}" iuse_unset
 | |
|         __mcl_unset_array "${cache_file_ref[PCF_ECLASSES_IDX]}" unset
 | |
| 
 | |
|         group_unset \
 | |
|             "${cache_file_ref[PCF_BDEPEND_IDX]}" \
 | |
|             "${cache_file_ref[PCF_DEPEND_IDX]}" \
 | |
|             "${cache_file_ref[PCF_IDEPEND_IDX]}" \
 | |
|             "${cache_file_ref[PCF_PDEPEND_IDX]}" \
 | |
|             "${cache_file_ref[PCF_RDEPEND_IDX]}" \
 | |
|             "${cache_file_ref[PCF_LICENSE_IDX]}"
 | |
| 
 | |
|         unset -n cache_file_ref
 | |
|     done
 | |
|     unset "${@}"
 | |
| }
 | |
| 
 | |
| # Parses a cache file under the passed path.
 | |
| #
 | |
| # Params:
 | |
| #
 | |
| # 1 - cache file
 | |
| # 2 - path to the cache file
 | |
| function parse_cache_file() {
 | |
|     local -n cache_file_ref=${1}; shift
 | |
|     local path=${1}; shift
 | |
|     # rest are architectures
 | |
| 
 | |
|     if pkg_debug_enabled; then
 | |
|         local -a file_lines
 | |
|         mapfile -t file_lines <"${path}"
 | |
|         pkg_debug_print_lines "parsing ${path@Q}" "${file_lines[@]}"
 | |
|         unset file_lines
 | |
|     fi
 | |
| 
 | |
|     local -n pkg_eapi_ref='cache_file_ref[PCF_EAPI_IDX]'
 | |
|     local -n pkg_keywords_ref='cache_file_ref[PCF_KEYWORDS_IDX]'
 | |
|     local -n pkg_iuse_ref='cache_file_ref[PCF_IUSE_IDX]'
 | |
|     local -n pkg_bdepend_group_name_ref='cache_file_ref[PCF_BDEPEND_IDX]'
 | |
|     local -n pkg_depend_group_name_ref='cache_file_ref[PCF_DEPEND_IDX]'
 | |
|     local -n pkg_idepend_group_name_ref='cache_file_ref[PCF_IDEPEND_IDX]'
 | |
|     local -n pkg_pdepend_group_name_ref='cache_file_ref[PCF_PDEPEND_IDX]'
 | |
|     local -n pkg_rdepend_group_name_ref='cache_file_ref[PCF_RDEPEND_IDX]'
 | |
|     local -n pkg_license_group_name_ref='cache_file_ref[PCF_LICENSE_IDX]'
 | |
|     local -n pkg_eclasses_ref='cache_file_ref[PCF_ECLASSES_IDX]'
 | |
| 
 | |
|     local l key
 | |
|     while read -r l; do
 | |
|         key=${l%%=*}
 | |
|         pkg_debug "parsing ${key@Q}"
 | |
|         case ${key} in
 | |
|             'EAPI')
 | |
|                 pkg_eapi_ref=${l#*=}
 | |
|                 pkg_debug "EAPI: ${pkg_eapi_ref}"
 | |
|                 ;;
 | |
|             'KEYWORDS')
 | |
|                 __mcl_parse_keywords "${l#*=}" pkg_keywords_ref "${@}"
 | |
|                 ;;
 | |
|             'IUSE')
 | |
|                 __mcl_parse_iuse "${l#*=}" pkg_iuse_ref
 | |
|                 ;;
 | |
|             'BDEPEND')
 | |
|                 __mcl_parse_dsf "${__MCL_DSF_DEPEND}" "${l#*=}" pkg_bdepend_group_name_ref
 | |
|                 ;;
 | |
|             'DEPEND')
 | |
|                 __mcl_parse_dsf "${__MCL_DSF_DEPEND}" "${l#*=}" pkg_depend_group_name_ref
 | |
|                 ;;
 | |
|             'IDEPEND')
 | |
|                 __mcl_parse_dsf "${__MCL_DSF_DEPEND}" "${l#*=}" pkg_idepend_group_name_ref
 | |
|                 ;;
 | |
|             'PDEPEND')
 | |
|                 __mcl_parse_dsf "${__MCL_DSF_DEPEND}" "${l#*=}" pkg_pdepend_group_name_ref
 | |
|                 ;;
 | |
|             'RDEPEND')
 | |
|                 __mcl_parse_dsf "${__MCL_DSF_DEPEND}" "${l#*=}" pkg_rdepend_group_name_ref
 | |
|                 ;;
 | |
|             'LICENSE')
 | |
|                 __mcl_parse_dsf "${__MCL_DSF_LICENSE}" "${l#*=}" pkg_license_group_name_ref
 | |
|                 ;;
 | |
|             '_eclasses_')
 | |
|                 __mcl_parse_eclasses "${l#*=}" pkg_eclasses_ref
 | |
|                 ;;
 | |
|             *)
 | |
|                 pkg_debug "Not parsing ${key@Q}, ignoring"
 | |
|                 ;;
 | |
|         esac
 | |
|     done <"${path}"
 | |
| }
 | |
| 
 | |
| #
 | |
| # Use requirement (the part in the square brackets in the package
 | |
| # dependency specification).
 | |
| #
 | |
| 
 | |
| # Indices to access fields of the use requirement:
 | |
| # UR_NAME_IDX    - a use name
 | |
| # UR_MODE_IDX    - a string describing the mode: "+" is enabled, "="
 | |
| #                  is strict relation, "!=" reversed strict relation,
 | |
| #                  "?"  is "enabled if enabled in the ebuild", "!?" is
 | |
| #                  "disabled if disabled in the ebuild", and "-" is
 | |
| #                  disabled)
 | |
| # UR_PRETEND_IDX - a string describing pretend mode: it can be either
 | |
| #                  empty (no pretending if use is missing in the
 | |
| #                  package), "+" (pretend enabled if missing in the
 | |
| #                  package), or "-" (pretend disabled if missing in
 | |
| #                  the package)
 | |
| declare -gri UR_NAME_IDX=0 UR_MODE_IDX=1 UR_PRETEND_IDX=2
 | |
| 
 | |
| # Declares empty use requirements. Can take flags that are passed to
 | |
| # declare. Usually only -r or -t make sense to pass. Can take several
 | |
| # names, just like declare. Takes no initializers - this is hardcoded.
 | |
| function ur_declare() {
 | |
|     struct_declare -ga "${@}" "( 'ITS_UNSET' '+' '' )"
 | |
| }
 | |
| 
 | |
| # Unsets use requirements, can take several names, just like unset.
 | |
| function ur_unset() {
 | |
|     unset "${@}"
 | |
| }
 | |
| 
 | |
| # Copies use requirement into another.
 | |
| #
 | |
| # Params:
 | |
| #
 | |
| # 1 - use requirement to be clobbered
 | |
| # 2 - the source use requirement
 | |
| function ur_copy() {
 | |
|     local -n to_clobber_ref=${1}; shift
 | |
|     local -n to_copy_ref=${1}; shift
 | |
| 
 | |
|     local -i idx
 | |
|     for idx in UR_NAME_IDX UR_MODE_IDX UR_PRETEND_IDX; do
 | |
|         to_clobber_ref[idx]=${to_copy_ref[idx]}
 | |
|     done
 | |
| }
 | |
| 
 | |
| # Stringifies use requirement.
 | |
| #
 | |
| # Params:
 | |
| #
 | |
| # 1 - use requirement
 | |
| # 2 - name of a variable where the string form will be stored
 | |
| function ur_to_string() {
 | |
|     local -n ur_ref=${1}; shift
 | |
|     local -n str_ref=${1}; shift
 | |
| 
 | |
|     case ${ur_ref[UR_MODE_IDX]} in
 | |
|         '!'*)
 | |
|             str_ref='!'
 | |
|             ;;
 | |
|         '-'*)
 | |
|             str_ref='-'
 | |
|             ;;
 | |
|         *)
 | |
|             str_ref=''
 | |
|             ;;
 | |
|     esac
 | |
|     str_ref+=${ur_ref[UR_NAME_IDX]}
 | |
| 
 | |
|     local p=${ur_ref[UR_PRETEND_IDX]}
 | |
|     if [[ -n ${p} ]]; then
 | |
|         str_ref+="(${p})"
 | |
|     fi
 | |
|     case ${ur_ref[UR_MODE_IDX]} in
 | |
|         *'=')
 | |
|             str_ref+='='
 | |
|             ;;
 | |
|         *'?')
 | |
|             str_ref+='?'
 | |
|             ;;
 | |
|     esac
 | |
| }
 | |
| 
 | |
| #
 | |
| # Package depedency specification (or PDS)
 | |
| #
 | |
| 
 | |
| # Enumeration describing blocker mode of a package. Self-describing.
 | |
| declare -gri PDS_NO_BLOCK=0 PDS_WEAK_BLOCK=1 PDS_STRONG_BLOCK=2
 | |
| 
 | |
| # Indices to access fields of the package dependency specification:
 | |
| # PDS_BLOCKS_IDX - a number describing blocker mode (use PDS_NO_BLOCK,
 | |
| #                  PDS_WEAK_BLOCK and PDS_STRONG_BLOCK)
 | |
| # PDS_OP_IDX     - a string describing the relational operator, can be
 | |
| #                  empty or one of "<", "<=", "=", "=*", "~", ">=", or
 | |
| #                  ">" ("=*" is same as "=" with asterisk appended to
 | |
| #                  version)
 | |
| #                  if empty, the version will also be empty
 | |
| # PDS_NAME_IDX   - a qualified package name (so category/name)
 | |
| # PDS_VER_IDX    - a version of the package, may be empty (if so,
 | |
| #                  operator is also empty)
 | |
| # PDS_SLOT_IDX   - a string describing the slot operator, without the
 | |
| #                  preceding colon
 | |
| # PDS_UR_IDX     - a name of an array variable containing use
 | |
| #                  requirements
 | |
| declare -gri PDS_BLOCKS_IDX=0 PDS_OP_IDX=1 PDS_NAME_IDX=2 PDS_VER_IDX=3 PDS_SLOT_IDX=4 PDS_UR_IDX=5
 | |
| 
 | |
| # Declares empty package definition specifications. Can take flags
 | |
| # that are passed to declare. Usually only -r or -t make sense to
 | |
| # pass. Can take several names, just like declare. Takes no
 | |
| # initializers - this is hardcoded.
 | |
| function pds_declare() {
 | |
|     struct_declare -ga "${@}" "( ${PDS_NO_BLOCK@Q} '' 'ITS_UNSET' '' '' 'EMPTY_ARRAY' )"
 | |
| }
 | |
| 
 | |
| # Unsets package definition specifications, can take several names,
 | |
| # just like unset.
 | |
| function pds_unset() {
 | |
|     local use_reqs_name name
 | |
|     for name; do
 | |
|         local -n pds_ref=${name}
 | |
| 
 | |
|         use_reqs_name=${pds_ref[PDS_UR_IDX]}
 | |
|         local -n use_reqs_ref=${use_reqs_name}
 | |
|         ur_unset "${use_reqs_ref[@]}"
 | |
|         unset -n use_reqs_ref
 | |
|         if [[ ${use_reqs_name} != 'EMPTY_ARRAY' ]]; then
 | |
|             unset "${use_reqs_name}"
 | |
|         fi
 | |
| 
 | |
|         unset -n pds_ref
 | |
|     done
 | |
|     unset "${@}"
 | |
| }
 | |
| 
 | |
| # Copies package dependency specification into another.
 | |
| #
 | |
| # Params:
 | |
| #
 | |
| # 1 - package dependency specification to be clobbered
 | |
| # 2 - the source package dependency specification
 | |
| function pds_copy() {
 | |
|     local -n to_clobber_ref=${1}; shift
 | |
|     local -n to_copy_ref=${1}; shift
 | |
| 
 | |
|     local -i idx
 | |
|     for idx in PDS_BLOCKS_IDX PDS_OP_IDX PDS_NAME_IDX PDS_VER_IDX PDS_SLOT_IDX; do
 | |
|         to_clobber_ref[idx]=${to_copy_ref[idx]}
 | |
|     done
 | |
| 
 | |
|     if [[ ${to_copy_ref[PDS_UR_IDX]} = 'EMPTY_ARRAY' || ${#to_copy_ref[PDS_UR_IDX]} -eq 0 ]]; then
 | |
|         to_clobber_ref[PDS_UR_IDX]='EMPTY_ARRAY'
 | |
|     else
 | |
|         local pc_ur_array_name
 | |
|         gen_varname pc_ur_array_name
 | |
|         declare -ga "${pc_ur_array_name}=()"
 | |
| 
 | |
|         local -n urs_to_copy_ref=${to_copy_ref[PDS_UR_IDX]} urs_ref=${pc_ur_array_name}
 | |
|         local ur_name pc_ur_name
 | |
|         for ur_name in "${urs_to_copy_ref[@]}"; do
 | |
|             gen_varname pc_ur_name
 | |
|             ur_declare "${pc_ur_name}"
 | |
|             ur_copy "${pc_ur_name}" "${ur_name}"
 | |
|             urs_ref+=( "${pc_ur_name}" )
 | |
|         done
 | |
| 
 | |
|         to_clobber_ref[PDS_UR_IDX]=${pc_ur_array_name}
 | |
|     fi
 | |
| }
 | |
| 
 | |
| # Adds use requirements to the package dependency specification.
 | |
| #
 | |
| # Params:
 | |
| #
 | |
| # 1 - package dependency specification
 | |
| # @ - use requirements
 | |
| function pds_add_urs() {
 | |
|     local -n pds_ref=${1}; shift
 | |
|     # rest are use requirements
 | |
| 
 | |
|     if [[ ${#} -eq 0 ]]; then
 | |
|         return 0
 | |
|     fi
 | |
| 
 | |
|     local use_reqs_name=${pds_ref[PDS_UR_IDX]}
 | |
|     if [[ ${use_reqs_name} = 'EMPTY_ARRAY' ]]; then
 | |
|         local ura_name
 | |
|         gen_varname ura_name
 | |
|         declare -ga "${ura_name}=()"
 | |
|         pds_ref[PDS_UR_IDX]=${ura_name}
 | |
|         use_reqs_name=${ura_name}
 | |
|         unset ura_name
 | |
|     fi
 | |
| 
 | |
|     local -n use_reqs_ref=${use_reqs_name}
 | |
|     use_reqs_ref+=( "${@}" )
 | |
| }
 | |
| 
 | |
| # Stringifies package dependency specification.
 | |
| #
 | |
| # Params:
 | |
| #
 | |
| # 1 - package dependency specification
 | |
| # 2 - name of a variable where the string form will be stored
 | |
| function pds_to_string() {
 | |
|     local -n pds_ref=${1}; shift
 | |
|     local -n str_ref=${1}; shift
 | |
| 
 | |
|     case ${pds_ref[PDS_BLOCKS_IDX]} in
 | |
|         "${PDS_NO_BLOCK}")
 | |
|             str_ref=''
 | |
|             ;;
 | |
|         "${PDS_WEAK_BLOCK}")
 | |
|             str_ref='!'
 | |
|             ;;
 | |
|         "${PDS_STRONG_BLOCK}")
 | |
|             str_ref='!!'
 | |
|             ;;
 | |
|     esac
 | |
|     local op=${pds_ref[PDS_OP_IDX]}
 | |
|     local v=${pds_ref[PDS_VER_IDX]}
 | |
|     if [[ ${op} = '=*' ]]; then
 | |
|         op='='
 | |
|         # if there's an op, then we assume version is not empty - so
 | |
|         # version will never end up being just *
 | |
|         v+='*'
 | |
|     fi
 | |
|     str_ref+=${op}${pds_ref[PDS_NAME_IDX]}
 | |
|     if [[ -n ${v} ]]; then
 | |
|         str_ref+=-${v}
 | |
|     fi
 | |
|     local s=${pds_ref[PDS_SLOT_IDX]}
 | |
|     if [[ -n ${s} ]]; then
 | |
|         str_ref+=:${s}
 | |
|     fi
 | |
|     local -n urs_ref=${pds_ref[PDS_UR_IDX]}
 | |
|     if [[ ${#urs_ref[@]} -gt 0 ]]; then
 | |
|         str_ref+='['
 | |
|         local u ur_str
 | |
|         for u in "${urs_ref[@]}"; do
 | |
|             ur_to_string "${u}" ur_str
 | |
|             str_ref+=${ur_str},
 | |
|         done
 | |
|         unset ur_str u
 | |
|         str_ref=${str_ref:0:$(( ${#str_ref} - 1 ))}']'
 | |
|     fi
 | |
| }
 | |
| 
 | |
| #
 | |
| # Group. A structure for describing {B,R,I,P,}DEPEND and LICENSE
 | |
| # fields. Contains items, which can be either package dependency
 | |
| # specifications or another groups. So it is a recursive structure.
 | |
| #
 | |
| 
 | |
| # Enumeration describing type of a group. Self-describing.
 | |
| declare -gri GROUP_ALL_OF=0 GROUP_ANY_OF=1
 | |
| 
 | |
| # Enumeration describing whether items in group are for enabled or disabled USE. Self-describing.
 | |
| declare -gri GROUP_USE_ENABLED=0 GROUP_USE_DISABLED=1
 | |
| 
 | |
| # Indices to access fields of the group:
 | |
| # GROUP_TYPE_IDX    - a number describing a type of the group (use
 | |
| #                     GROUP_ALL_OF and GROUP_ANY_OF)
 | |
| # GROUP_USE_IDX     - a USE name of the group, may be empty, and must
 | |
| #                     be empty for GROUP_ANY_OF groups
 | |
| # GROUP_ENABLED_IDX - a number describing mode of the USE, should be
 | |
| #                     ignored if USE name is empty (use
 | |
| #                     GROUP_USE_ENABLED and GROUP_USE_DISABLED)
 | |
| # GROUP_ITEMS_IDX   - a name of an array containing items
 | |
| declare -gri GROUP_TYPE_IDX=0 GROUP_USE_IDX=1 GROUP_ENABLED_IDX=2 GROUP_ITEMS_IDX=3
 | |
| 
 | |
| # Declares empty groups. Can take flags that are passed to
 | |
| # declare. Usually only -r or -t make sense to pass. Can take several
 | |
| # names, just like declare. Takes no initializers - this is hardcoded.
 | |
| function group_declare() {
 | |
|     struct_declare -ga "${@}" "( ${GROUP_ALL_OF@Q} '' ${GROUP_USE_ENABLED@Q} 'EMPTY_ARRAY' )"
 | |
| }
 | |
| 
 | |
| # An empty readonly group.
 | |
| group_declare -r EMPTY_GROUP
 | |
| 
 | |
| # Unsets groups, can take several names, just like unset.
 | |
| function group_unset() {
 | |
|     local -a to_unset=()
 | |
|     local name items_name
 | |
|     for name; do
 | |
|         if [[ ${name} == 'EMPTY_GROUP' ]]; then
 | |
|             continue
 | |
|         fi
 | |
| 
 | |
|         local -n group_ref=${name}
 | |
|         items_name=${group_ref[GROUP_ITEMS_IDX]}
 | |
|         unset -n group_ref
 | |
| 
 | |
|         to_unset+=( "${name}" )
 | |
|         if [[ ${items_name} == 'EMPTY_ARRAY' ]]; then
 | |
|             continue
 | |
|         fi
 | |
|         to_unset+=( "${items_name}" )
 | |
| 
 | |
|         local -n items_ref=${items_name}
 | |
|         item_unset "${items_ref[@]}"
 | |
|         unset -n items_ref
 | |
|     done
 | |
|     unset "${to_unset[@]}"
 | |
| }
 | |
| 
 | |
| # Copies group into another.
 | |
| #
 | |
| # Params:
 | |
| #
 | |
| # 1 - group to be clobbered
 | |
| # 2 - the source group
 | |
| function group_copy() {
 | |
|     local -n to_clobber_ref=${1}; shift
 | |
|     local -n to_copy_ref=${1}; shift
 | |
| 
 | |
|     local -i idx
 | |
|     for idx in GROUP_TYPE_IDX GROUP_USE_IDX GROUP_ENABLED_IDX; do
 | |
|         to_clobber_ref[idx]=${to_copy_ref[idx]}
 | |
|     done
 | |
| 
 | |
|     if [[ ${to_copy_ref[GROUP_ITEMS_IDX]} = 'EMPTY_ARRAY' || ${#to_copy_ref[GROUP_ITEMS_IDX]} -eq 0 ]]; then
 | |
|         to_clobber_ref[GROUP_ITEMS_IDX]='EMPTY_ARRAY'
 | |
|     else
 | |
|         local gc_items_name
 | |
|         gen_varname gc_items_name
 | |
|         declare -ga "${gc_items_name}=()"
 | |
| 
 | |
|         local -n items_to_copy_ref=${to_copy_ref[GROUP_ITEMS_IDX]}
 | |
|         local -n items_ref=${gc_items_name}
 | |
|         local item_name_to_copy gc_item_name
 | |
|         for item_name_to_copy in "${items_to_copy_ref[@]}"; do
 | |
|             gen_varname gc_item_name
 | |
|             item_declare "${gc_item_name}"
 | |
|             item_copy "${gc_item_name}" "${item_name_to_copy}"
 | |
|             items_ref+=( "${gc_item_name}" )
 | |
|         done
 | |
|         unset -n items_ref items_to_copy_ref
 | |
|         to_clobber_ref[GROUP_ITEMS_IDX]="${gc_items_name}"
 | |
|     fi
 | |
| }
 | |
| 
 | |
| # Adds items to the group.
 | |
| #
 | |
| # Params:
 | |
| #
 | |
| # 1 - group
 | |
| # @ - items
 | |
| function group_add_items() {
 | |
|     local -n group_ref=${1}; shift
 | |
|     # rest are items to add
 | |
| 
 | |
|     if [[ ${#} -eq 0 ]]; then
 | |
|         return 0
 | |
|     fi
 | |
| 
 | |
|     local items_name=${group_ref[GROUP_ITEMS_IDX]}
 | |
|     if [[ ${items_name} = 'EMPTY_ARRAY' ]]; then
 | |
|         local ia_name
 | |
|         gen_varname ia_name
 | |
|         declare -ga "${ia_name}=()"
 | |
|         group_ref[GROUP_ITEMS_IDX]=${ia_name}
 | |
|         items_name=${ia_name}
 | |
|         unset ia_name
 | |
|     fi
 | |
| 
 | |
|     local -n items_ref=${items_name}
 | |
|     items_ref+=( "${@}" )
 | |
| }
 | |
| 
 | |
| # Stringifies group.
 | |
| #
 | |
| # Params:
 | |
| #
 | |
| # 1 - group
 | |
| # 2 - name of a variable where the string form will be stored
 | |
| function group_to_string() {
 | |
|     local -n group_ref=${1}; shift
 | |
|     local -n str_ref=${1}; shift
 | |
| 
 | |
|     local t=${group_ref[GROUP_TYPE_IDX]}
 | |
|     case ${t} in
 | |
|         "${GROUP_ALL_OF}")
 | |
|             local u=${group_ref[GROUP_USE_IDX]}
 | |
|             if [[ -n ${u} ]]; then
 | |
|                 local e=${group_ref[GROUP_ENABLED_IDX]}
 | |
|                 case ${e} in
 | |
|                     "${GROUP_USE_ENABLED}")
 | |
|                         str_ref=''
 | |
|                         ;;
 | |
|                     "${GROUP_USE_DISABLED}")
 | |
|                         str_ref='!'
 | |
|                 esac
 | |
|                 unset e
 | |
|                 str_ref+="${u}? "
 | |
|             else
 | |
|                 str_ref=''
 | |
|             fi
 | |
|             unset u
 | |
|             ;;
 | |
|         "${GROUP_ANY_OF}")
 | |
|             str_ref='|| '
 | |
|             ;;
 | |
|     esac
 | |
| 
 | |
|     str_ref+='( '
 | |
|     local -n item_names_ref=${group_ref[GROUP_ITEMS_IDX]}
 | |
|     if [[ ${#item_names_ref[@]} -gt 0 ]]; then
 | |
|         local item_name item_str
 | |
|         for item_name in "${item_names_ref[@]}"; do
 | |
|             item_to_string "${item_name}" item_str
 | |
|             str_ref+="${item_str} "
 | |
|         done
 | |
|         unset item_str item_name
 | |
|     fi
 | |
|     str_ref+=')'
 | |
| }
 | |
| 
 | |
| #
 | |
| # Item. A string of <mode>:<data>. Mode is a single char and describes
 | |
| # the data.
 | |
| #
 | |
| # Modes:
 | |
| # "e" - empty, data is meaningless and should be just empty.
 | |
| # "g" - group, data is a group (a name of a variable containing a
 | |
| #       group)
 | |
| # "l" - license, data is a name of a license (plain string)
 | |
| # "p" - package dependency specification, data is pds (a name of a
 | |
| #       variable containing a pds)
 | |
| 
 | |
| # Declares empty items. Can take flags that are passed to
 | |
| # declare. Usually only -r or -t make sense to pass. Can take several
 | |
| # names, just like declare. Takes no initializers - this is hardcoded.
 | |
| function item_declare() {
 | |
|     struct_declare -g "${@}" 'e:'
 | |
| }
 | |
| 
 | |
| # Unsets items, can take several names, just like unset.
 | |
| function item_unset() {
 | |
|     local name
 | |
|     for name; do
 | |
|         local -n item_ref=${name}
 | |
| 
 | |
|         case ${item_ref} in
 | |
|             e:*)
 | |
|                 # noop
 | |
|                 :
 | |
|                 ;;
 | |
|             g:*)
 | |
|                 group_unset "${item_ref:2}"
 | |
|                 ;;
 | |
|             l:*)
 | |
|                 # noop, license is just a string
 | |
|                 ;;
 | |
|             p:*)
 | |
|                 pds_unset "${item_ref:2}"
 | |
|                 ;;
 | |
|         esac
 | |
|         unset -n item_ref
 | |
|     done
 | |
| 
 | |
|     unset "${@}"
 | |
| }
 | |
| 
 | |
| # Copies item into another.
 | |
| #
 | |
| # Params:
 | |
| #
 | |
| # 1 - item to be clobbered
 | |
| # 2 - the source item
 | |
| function item_copy() {
 | |
|     local -n to_clobber_ref=${1}; shift
 | |
|     local -n to_copy_ref=${1}; shift
 | |
| 
 | |
|     local ic_name
 | |
|     local t=${to_copy_ref:0:1} v=${to_copy_ref:2}
 | |
|     case ${t} in
 | |
|         'e'|'l')
 | |
|             to_clobber_ref=${to_copy_ref}
 | |
|             ;;
 | |
|         'g')
 | |
|             gen_varname ic_name
 | |
|             group_declare "${ic_name}"
 | |
|             group_copy "${ic_name}" "${v}"
 | |
|             to_clobber_ref="g:${ic_name}"
 | |
|             ;;
 | |
|         'p')
 | |
|             gen_varname ic_name
 | |
|             pds_declare "${ic_name}"
 | |
|             pds_copy "${ic_name}" "${v}"
 | |
|             to_clobber_ref="p:${ic_name}"
 | |
|             ;;
 | |
|     esac
 | |
| }
 | |
| 
 | |
| # Stringifies item.
 | |
| #
 | |
| # Params:
 | |
| #
 | |
| # 1 - item
 | |
| # 2 - name of a variable where the string form will be stored
 | |
| function item_to_string() {
 | |
|     local -n item_ref=${1}; shift
 | |
|     local -n str_ref=${1}; shift
 | |
| 
 | |
|     local t=${item_ref:0:1}
 | |
|     case ${t} in
 | |
|         e)
 | |
|             str_ref=''
 | |
|             ;;
 | |
|         g)
 | |
|             local group_name=${item_ref:2}
 | |
|             local group_str
 | |
|             group_to_string "${group_name}" group_str
 | |
|             str_ref=${group_str}
 | |
|             unset group_str group_name
 | |
|             ;;
 | |
|         l)
 | |
|             str_ref=${item_ref:2}
 | |
|             ;;
 | |
|         p)
 | |
|             local pds_name=${item_ref:2}
 | |
|             local pds_str=''
 | |
|             pds_to_string "${pds_name}" pds_str
 | |
|             str_ref=${pds_str}
 | |
|             unset pds_str pds_name
 | |
|             ;;
 | |
|     esac
 | |
| }
 | |
| 
 | |
| #
 | |
| # Keyword
 | |
| #
 | |
| # n - name of keyword
 | |
| # m - stable (amd64), unstable (~amd64), broken (-amd64), unknown (absent in KEYWORDS)
 | |
| 
 | |
| # Enumeration describing mode of the keyword.
 | |
| # KW_STABLE   - like "amd64"
 | |
| # KW_UNSTABLE - like "~amd64"
 | |
| # KW_BROKEN   - like "-amd64"
 | |
| # KW_UNKNOWN  - missing from KEYWORDS entirely
 | |
| declare -gri KW_STABLE=0 KW_UNSTABLE=1 KW_BROKEN=2 KW_UNKNOWN=3
 | |
| 
 | |
| # Indices to access fields of the keyword:
 | |
| # KW_NAME_IDX  - name of the architecture
 | |
| # KW_LEVEL_IDX - mode of the keyword (use KW_STABLE, KW_UNSTABLE,
 | |
| #                KW_BROKEN and KW_UNKNOWN)
 | |
| declare -gri KW_NAME_IDX=0 KW_LEVEL_IDX=1
 | |
| 
 | |
| # Declares empty keywords. Can take flags that are passed to
 | |
| # declare. Usually only -r or -t make sense to pass. Can take several
 | |
| # names, just like declare. Takes no initializers - this is hardcoded.
 | |
| function kw_declare() {
 | |
|     struct_declare -ga "${@}" "( 'ITS_UNSET' ${KW_UNSTABLE@Q} )"
 | |
| }
 | |
| 
 | |
| # Unsets keywords, can take several names, just like unset.
 | |
| function kw_unset() {
 | |
|     unset "${@}"
 | |
| }
 | |
| 
 | |
| # Stringifies keyword.
 | |
| #
 | |
| # Params:
 | |
| #
 | |
| # 1 - keyword
 | |
| # 2 - name of a variable where the string form will be stored
 | |
| function kw_to_string() {
 | |
|     local -n kw_ref=${1}; shift
 | |
|     local -n str_ref=${1}; shift
 | |
| 
 | |
|     local n=${kw_ref[KW_NAME_IDX]}
 | |
|     case ${kw_ref[KW_LEVEL_IDX]} in
 | |
|         "${KW_STABLE}")
 | |
|             str_ref=${n}
 | |
|             ;;
 | |
|         "${KW_UNSTABLE}")
 | |
|             str_ref="~${n}"
 | |
|             ;;
 | |
|         "${KW_BROKEN}")
 | |
|             str_ref="-${n}"
 | |
|             ;;
 | |
|         "${KW_UNKNOWN}")
 | |
|             str_ref=''
 | |
|             ;;
 | |
|     esac
 | |
| }
 | |
| 
 | |
| #
 | |
| # IUSE
 | |
| #
 | |
| 
 | |
| # n - name of IUSE flag
 | |
| # m - 0 or 1 (0 disabled, 1 enabled)
 | |
| 
 | |
| # Enumeration describing mode of the IUSE. Self-describing.
 | |
| declare -gri IUSE_DISABLED=0 IUSE_ENABLED=1
 | |
| 
 | |
| # Indices to access fields of the IUSE:
 | |
| # IUSE_NAME_IDX - IUSE name
 | |
| # IUSE_MODE_IDX - IUSE mode (use IUSE_DISABLED and IUSE_ENABLED)
 | |
| declare -gri IUSE_NAME_IDX=0 IUSE_MODE_IDX=1
 | |
| 
 | |
| # Declares empty IUSEs. Can take flags that are passed to
 | |
| # declare. Usually only -r or -t make sense to pass. Can take several
 | |
| # names, just like declare. Takes no initializers - this is hardcoded.
 | |
| function iuse_declare() {
 | |
|     struct_declare -ga "${@}" "( 'ITS_UNSET' ${IUSE_DISABLED@Q} )"
 | |
| }
 | |
| 
 | |
| # Unsets IUSEs, can take several names, just like unset.
 | |
| function iuse_unset() {
 | |
|     unset "${@}"
 | |
| }
 | |
| 
 | |
| # Stringifies IUSE.
 | |
| #
 | |
| # Params:
 | |
| #
 | |
| # 1 - IUSE
 | |
| # 2 - name of a variable where the string form will be stored
 | |
| function iuse_to_string() {
 | |
|     local -n iuse_ref=${1}; shift
 | |
|     local -n str_ref=${1}; shift
 | |
| 
 | |
|     case ${iuse_ref[IUSE_MODE_IDX]} in
 | |
|         "${IUSE_ENABLED}")
 | |
|             str_ref='+'
 | |
|             ;;
 | |
|         "${IUSE_DISABLED}")
 | |
|             str_ref=''
 | |
|             ;;
 | |
|     esac
 | |
|     str_ref+=${iuse_ref[IUSE_NAME_IDX]}
 | |
| }
 | |
| 
 | |
| #
 | |
| # Implementation details
 | |
| #
 | |
| 
 | |
| # parse dependency specification format (DSF)
 | |
| 
 | |
| declare -gri __MCL_DSF_DEPEND=0 __MCL_DSF_LICENSE=1
 | |
| 
 | |
| function __mcl_parse_dsf() {
 | |
|     local -i dsf_type=${1}; shift
 | |
|     local dep=${1}; shift
 | |
|     local -n top_group_out_var_name_ref=${1}; shift
 | |
| 
 | |
|     local -a group_stack
 | |
|     local pd_group pd_item pd_group_created='' pd_pds
 | |
| 
 | |
|     gen_varname pd_group
 | |
|     group_declare "${pd_group}"
 | |
|     group_stack+=( "${pd_group}" )
 | |
| 
 | |
|     local -a tokens
 | |
|     mapfile -t tokens <<<"${dep// /$'\n'}"
 | |
|     local -i last_index
 | |
| 
 | |
|     local token
 | |
|     for token in "${tokens[@]}"; do
 | |
|         if [[ ${token} = '||' ]]; then
 | |
|             # "any of" group, so create the group, make it an item, add
 | |
|             # to current group and mark the new group as current
 | |
|             gen_varname pd_group
 | |
|             group_declare "${pd_group}"
 | |
|             local -n g_ref=${pd_group}
 | |
|             g_ref[GROUP_TYPE_IDX]=${GROUP_ANY_OF}
 | |
|             unset -n g_ref
 | |
| 
 | |
|             gen_varname pd_item
 | |
|             item_declare "${pd_item}"
 | |
|             local -n i_ref=${pd_item}
 | |
|             i_ref="g:${pd_group}"
 | |
|             unset -n i_ref
 | |
| 
 | |
|             group_add_items "${group_stack[-1]}" "${pd_item}"
 | |
| 
 | |
|             group_stack+=( "${pd_group}" )
 | |
|             pd_group_created=x
 | |
|         elif [[ ${token} =~ ^!?[A-Za-z0-9][A-Za-z0-9+_-]*\?$ ]]; then
 | |
|             # "use" group, so create the group, make it an item, add
 | |
|             # to current group and mark the new group as current
 | |
|             local -i disabled=GROUP_USE_ENABLED
 | |
|             local use=${token%?}
 | |
| 
 | |
|             if [[ ${use} = '!'* ]]; then
 | |
|                 disabled=GROUP_USE_DISABLED
 | |
|                 use=${use:1}
 | |
|             fi
 | |
| 
 | |
|             gen_varname pd_group
 | |
|             group_declare "${pd_group}"
 | |
|             local -n g_ref=${pd_group}
 | |
|             g_ref[GROUP_TYPE_IDX]=${GROUP_ALL_OF}
 | |
|             g_ref[GROUP_USE_IDX]=${use}
 | |
|             g_ref[GROUP_ENABLED_IDX]=${disabled}
 | |
|             unset -n g_ref
 | |
| 
 | |
|             unset use disabled
 | |
| 
 | |
|             gen_varname pd_item
 | |
|             item_declare "${pd_item}"
 | |
|             local -n i_ref=${pd_item}
 | |
|             i_ref="g:${pd_group}"
 | |
|             unset -n i_ref
 | |
| 
 | |
|             group_add_items "${group_stack[-1]}" "${pd_item}"
 | |
| 
 | |
|             group_stack+=( "${pd_group}" )
 | |
|             pd_group_created=x
 | |
|         elif [[ ${token} = '(' ]]; then
 | |
|             # beginning of a group; usually it is already created,
 | |
|             # because it is an "any of" or "use" group, but it is
 | |
|             # legal to specify just a "all of" group
 | |
|             if [[ -n ${pd_group_created} ]]; then
 | |
|                 pd_group_created=
 | |
|             else
 | |
|                 gen_varname pd_group
 | |
|                 group_declare "${pd_group}"
 | |
| 
 | |
|                 gen_varname pd_item
 | |
|                 item_declare "${pd_item}"
 | |
|                 local -n i_ref=${pd_item}
 | |
|                 i_ref="g:${pd_group}"
 | |
|                 unset -n i_ref
 | |
| 
 | |
|                 group_add_items "${group_stack[-1]}" "${pd_item}"
 | |
| 
 | |
|                 group_stack+=( "${pd_group}" )
 | |
|             fi
 | |
|         elif [[ ${token} = ')' ]]; then
 | |
|             # end of a group, pop it from the stack
 | |
|             last_index=$(( ${#group_stack[@]} - 1 ))
 | |
|             unset "group_stack[${last_index}]"
 | |
|         elif [[ ${token} =~ ^[A-Za-z0-9_][A-Za-z0-9+_.-]*$ ]]; then
 | |
|             # license
 | |
|             if [[ dsf_type -ne __MCL_DSF_LICENSE ]]; then
 | |
|                 fail "license tokens are only allowed for LICENSE keys (token: ${token@Q})"
 | |
|             fi
 | |
| 
 | |
|             gen_varname pd_item
 | |
|             item_declare "${pd_item}"
 | |
|             local -n i_ref=${pd_item}
 | |
|             i_ref="l:${token}"
 | |
|             unset -n i_ref
 | |
|             group_add_items "${group_stack[-1]}" "${pd_item}"
 | |
|         elif [[ ${token} =~ ^!?!?(<|<=|=|~|>=|>)?[A-Za-z0-9_][A-Za-z0-9+_.-]*/[A-Za-z0-9_] ]]; then
 | |
|             # pds
 | |
|             if [[ dsf_type -ne __MCL_DSF_DEPEND ]]; then
 | |
|                 fail "package dependency specification is only allowed for DEPEND-like keys (token: ${token@Q})"
 | |
|             fi
 | |
| 
 | |
|             local -i blocks=PDS_NO_BLOCK
 | |
|             local operator='' name='' version='' slot=''
 | |
|             local -a use_requirements=()
 | |
| 
 | |
|             case ${token} in
 | |
|                 '!!'*)
 | |
|                     blocks=PDS_STRONG_BLOCK
 | |
|                     token=${token:2}
 | |
|                     ;;
 | |
|                 '!'*)
 | |
|                     blocks=PDS_WEAK_BLOCK
 | |
|                     token=${token:1}
 | |
|                     ;;
 | |
|             esac
 | |
| 
 | |
|             case ${token} in
 | |
|                 '<='*|'>='*)
 | |
|                     operator=${token:0:2}
 | |
|                     token=${token:2}
 | |
|                     ;;
 | |
|                 '<'*|'='*|'~'*|'>'*)
 | |
|                     operator=${token:0:1}
 | |
|                     token=${token:1}
 | |
|                     ;;
 | |
|             esac
 | |
| 
 | |
|             if [[ ${token} = *']' ]]; then
 | |
|                 local use_reqs_string=${token#*'['}
 | |
|                 use_reqs_string=${use_reqs_string%']'}
 | |
|                 token=${token%"[${use_reqs_string}]"}
 | |
|                 local -a use_reqs
 | |
|                 mapfile -t use_reqs <<<"${use_reqs_string//,/$'\n'}"
 | |
| 
 | |
|                 unset use_reqs_string
 | |
| 
 | |
|                 local ur name mode pretend pd_ur
 | |
|                 for ur in "${use_reqs[@]}"; do
 | |
|                     name=''
 | |
|                     mode=''
 | |
|                     pretend=''
 | |
|                     case ${ur} in
 | |
|                         '-'*)
 | |
|                             mode='-'
 | |
|                             ur=${ur:1}
 | |
|                             ;;
 | |
|                         '!'*)
 | |
|                             mode='!'
 | |
|                             ur=${ur:1}
 | |
|                             ;;
 | |
|                     esac
 | |
|                     if [[ ${mode} != '-' ]]; then
 | |
|                         case ${ur} in
 | |
|                             *'=')
 | |
|                                 mode+='='
 | |
|                                 ur=${ur%'='}
 | |
|                                 ;;
 | |
|                             *'?')
 | |
|                                 mode+='?'
 | |
|                                 ur=${ur%'?'}
 | |
|                                 ;;
 | |
|                         esac
 | |
|                     fi
 | |
|                     if [[ -z ${mode} ]]; then
 | |
|                         mode='+'
 | |
|                     fi
 | |
|                     if [[ ${ur} =~ \(([+-])\)$ ]]; then
 | |
|                         pretend=${BASH_REMATCH[1]}
 | |
|                         ur=${ur%"(${pretend})"}
 | |
|                     fi
 | |
|                     name=${ur}
 | |
|                     gen_varname pd_ur
 | |
|                     ur_declare "${pd_ur}"
 | |
|                     local -n u_ref=${pd_ur}
 | |
|                     u_ref[UR_NAME_IDX]=${name}
 | |
|                     u_ref[UR_MODE_IDX]=${mode}
 | |
|                     u_ref[UR_PRETEND_IDX]=${pretend}
 | |
|                     unset -n u_ref
 | |
|                     use_requirements+=( "${pd_ur}" )
 | |
|                 done
 | |
|                 unset pd_ur pretend mode name ur use_reqs
 | |
|             fi
 | |
| 
 | |
|             if [[ ${token} = *:* ]]; then
 | |
|                 slot=${token#*:}
 | |
|                 token=${token%:"${slot}"}
 | |
|             fi
 | |
| 
 | |
|             if [[ ${token} = *'*' ]]; then
 | |
|                 operator='=*'
 | |
|                 token=${token%'*'}
 | |
|             fi
 | |
|             local ver_ere_with_dash="-(${VER_ERE_UNBOUNDED})$"
 | |
|             if [[ ${token} =~ ${ver_ere_with_dash} ]]; then
 | |
|                 version=${BASH_REMATCH[1]}
 | |
|                 token=${token%"-${version}"}
 | |
|             fi
 | |
|             name=${token}
 | |
| 
 | |
|             gen_varname pd_pds
 | |
|             pds_declare "${pd_pds}"
 | |
|             local -n p_ref=${pd_pds}
 | |
|             p_ref[PDS_BLOCKS_IDX]=${blocks}
 | |
|             p_ref[PDS_OP_IDX]=${operator}
 | |
|             p_ref[PDS_NAME_IDX]=${name}
 | |
|             p_ref[PDS_VER_IDX]=${version}
 | |
|             p_ref[PDS_SLOT_IDX]=${slot}
 | |
|             unset -n p_ref
 | |
|             pds_add_urs "${pd_pds}" "${use_requirements[@]}"
 | |
|             unset use_requirements slot version name operator blocks
 | |
| 
 | |
|             gen_varname pd_item
 | |
|             item_declare "${pd_item}"
 | |
|             local -n i_ref=${pd_item}
 | |
|             i_ref="p:${pd_pds}"
 | |
|             unset -n i_ref
 | |
| 
 | |
|             group_add_items "${group_stack[-1]}" "${pd_item}"
 | |
|         else
 | |
|             fail "unknown token ${token@Q}"
 | |
|         fi
 | |
|     done
 | |
| 
 | |
|     if [[ ${#group_stack[@]} -ne 1 ]]; then
 | |
|         fail "botched parsing, group stack has ${#group_stack[@]} groups instead of 1"
 | |
|     fi
 | |
| 
 | |
|     top_group_out_var_name_ref=${group_stack[0]}
 | |
| 
 | |
|     if pkg_debug_enabled; then
 | |
|         local pd_group_str
 | |
|         group_to_string "${group_stack[0]}" pd_group_str
 | |
|         pkg_debug_print "dsf: ${pd_group_str}"
 | |
|     fi
 | |
| }
 | |
| 
 | |
| function __mcl_parse_eclasses() {
 | |
|     local eclasses_string=${1}; shift
 | |
|     local -n eclasses_out_var_name_ref=${1}; shift
 | |
| 
 | |
|     local eclasses_var_name
 | |
|     gen_varname eclasses_var_name
 | |
|     declare -ga "${eclasses_var_name}=()"
 | |
|     local -n eclasses_ref=${eclasses_var_name}
 | |
| 
 | |
|     local -a tokens
 | |
|     mapfile -t tokens <<<"${eclasses_string//$'\t'/$'\n'}"
 | |
| 
 | |
|     local token
 | |
|     local -i eclass_name_now=1
 | |
|     for token in "${tokens[@]}"; do
 | |
|         if [[ eclass_name_now -eq 1 ]]; then
 | |
|             eclasses_ref+=( "${token}" )
 | |
|         fi
 | |
|         eclass_name_now=$((eclass_name_now ^ 1))
 | |
|     done
 | |
|     eclasses_out_var_name_ref=${eclasses_var_name}
 | |
| 
 | |
|     if pkg_debug_enabled; then
 | |
|         local joined_eclasses_string
 | |
|         join_by joined_eclasses_string ' ' "${eclasses_ref[@]}"
 | |
|         pkg_debug_print "eclasses: ${joined_eclasses_string}"
 | |
|     fi
 | |
| }
 | |
| 
 | |
| function __mcl_parse_keywords() {
 | |
|     local keywords_string=${1}; shift
 | |
|     local -n keywords_out_var_name_ref=${1}; shift
 | |
|     # rest are architectures
 | |
| 
 | |
|     local keywords_var_name
 | |
|     gen_varname keywords_var_name
 | |
|     declare -ga "${keywords_var_name}=()"
 | |
|     local -n keywords_ref=${keywords_var_name}
 | |
| 
 | |
|     local -A keywords_set=()
 | |
| 
 | |
|     local -a tokens
 | |
|     mapfile -t tokens <<<"${keywords_string// /$'\n'}"
 | |
|     local token
 | |
|     for token in "${tokens[@]}"; do
 | |
|         keywords_set["${token}"]=x
 | |
|     done
 | |
| 
 | |
|     local has_hyphen_star=${keywords_set['-*']:-}
 | |
|     local arch mark kw_level_pair kw kw_name
 | |
|     local -i level
 | |
|     for arch; do
 | |
|         for kw_level_pair in "${arch}@${KW_STABLE}" "~${arch}@${KW_UNSTABLE}" "-${arch}@${KW_BROKEN}"; do
 | |
|             kw=${kw_level_pair%@*}
 | |
|             level=${kw_level_pair#*@}
 | |
|             mark=${keywords_set["${kw}"]:-}
 | |
|             if [[ -n ${mark} ]]; then
 | |
|                 gen_varname kw_name
 | |
|                 kw_declare "${kw_name}"
 | |
|                 local -n k_ref=${kw_name}
 | |
|                 k_ref[KW_NAME_IDX]=${arch}
 | |
|                 k_ref[KW_LEVEL_IDX]=${level}
 | |
|                 unset -n k_ref
 | |
|                 keywords_ref+=( "${kw_name}" )
 | |
|                 break
 | |
|             fi
 | |
|         done
 | |
|         if [[ -z ${mark} ]]; then
 | |
|             gen_varname kw_name
 | |
|             kw_declare "${kw_name}"
 | |
|             local -n k_ref=${kw_name}
 | |
|             k_ref[KW_NAME_IDX]=${arch}
 | |
|             if [[ -n ${has_hyphen_star} ]]; then
 | |
|                 k_ref[KW_LEVEL_IDX]=${KW_BROKEN}
 | |
|             else
 | |
|                 k_ref[KW_LEVEL_IDX]=${KW_UNKNOWN}
 | |
|             fi
 | |
|             unset -n k_ref
 | |
|             keywords_ref+=( "${kw_name}" )
 | |
|         fi
 | |
|     done
 | |
|     keywords_out_var_name_ref=${keywords_var_name}
 | |
| 
 | |
|     if pkg_debug_enabled; then
 | |
|         local -a all_kws_strings=()
 | |
|         local kw_name pk_kw_str
 | |
|         local joined_kws_string
 | |
|         for kw_name in "${keywords_ref[@]}"; do
 | |
|             kw_to_string "${kw_name}" pk_kw_str
 | |
|             all_kws_strings+=( "${pk_kw_str}" )
 | |
|         done
 | |
|         join_by joined_kws_string ' ' "${all_kws_strings[@]}"
 | |
|         pkg_debug_print "keywords: ${joined_kws_string}"
 | |
|     fi
 | |
| }
 | |
| 
 | |
| function __mcl_parse_iuse() {
 | |
|     local iuse_string=${1}; shift
 | |
|     local -n iuse_out_var_name_ref=${1}; shift
 | |
| 
 | |
|     local iuse_var_name
 | |
|     gen_varname iuse_var_name
 | |
|     declare -ga "${iuse_var_name}=()"
 | |
|     local -n iuse_ref=${iuse_var_name}
 | |
| 
 | |
|     local -a tokens
 | |
|     mapfile -t tokens <<<"${iuse_string// /$'\n'}"
 | |
|     local token pi_iuse
 | |
|     for token in "${tokens[@]}"; do
 | |
|         gen_varname pi_iuse
 | |
|         iuse_declare "${pi_iuse}"
 | |
|         local -n i_ref=${pi_iuse}
 | |
|         if [[ ${token} = '+'* ]]; then
 | |
|             i_ref[IUSE_MODE_IDX]=${IUSE_ENABLED}
 | |
|             token=${token:1}
 | |
|         fi
 | |
|         i_ref[IUSE_NAME_IDX]=${token}
 | |
|         unset -n i_ref
 | |
|         iuse_ref+=( "${pi_iuse}" )
 | |
|     done
 | |
| 
 | |
|     iuse_out_var_name_ref=${iuse_var_name}
 | |
| 
 | |
|     if pkg_debug_enabled; then
 | |
|         local -a all_iuse_strings=()
 | |
|         local iuse_name pi_iuse_str
 | |
|         local joined_iuse_string
 | |
|         for iuse_name in "${iuse_ref[@]}"; do
 | |
|             iuse_to_string "${iuse_name}" pi_iuse_str
 | |
|             all_iuse_strings+=( "${pi_iuse_str}" )
 | |
|         done
 | |
|         join_by joined_iuse_string ' ' "${all_iuse_strings[@]}"
 | |
|         pkg_debug_print "IUSE: ${joined_iuse_string}"
 | |
|     fi
 | |
| }
 | |
| 
 | |
| function __mcl_unset_array() {
 | |
|     local array_name=${1}; shift
 | |
|     local item_unset_func=${1}; shift
 | |
| 
 | |
|     if [[ ${array_name} = EMPTY_ARRAY ]]; then
 | |
|         return 0
 | |
|     fi
 | |
|     local -n array_ref=${array_name}
 | |
|     "${item_unset_func}" "${array_ref[@]}"
 | |
|     unset -n array_ref
 | |
|     unset "${array_name}"
 | |
| }
 | |
| 
 | |
| fi
 |