| #!/bin/sh | 
 | # SPDX-License-Identifier: GPL-2.0 | 
 | # helpers for dealing with atomics.tbl | 
 |  | 
 | #meta_in(meta, match) | 
 | meta_in() | 
 | { | 
 | 	case "$1" in | 
 | 	[$2]) return 0;; | 
 | 	esac | 
 |  | 
 | 	return 1 | 
 | } | 
 |  | 
 | #meta_has_ret(meta) | 
 | meta_has_ret() | 
 | { | 
 | 	meta_in "$1" "bBiIfFlR" | 
 | } | 
 |  | 
 | #meta_has_acquire(meta) | 
 | meta_has_acquire() | 
 | { | 
 | 	meta_in "$1" "BFIlR" | 
 | } | 
 |  | 
 | #meta_has_release(meta) | 
 | meta_has_release() | 
 | { | 
 | 	meta_in "$1" "BFIRs" | 
 | } | 
 |  | 
 | #meta_has_relaxed(meta) | 
 | meta_has_relaxed() | 
 | { | 
 | 	meta_in "$1" "BFIR" | 
 | } | 
 |  | 
 | #find_fallback_template(pfx, name, sfx, order) | 
 | find_fallback_template() | 
 | { | 
 | 	local pfx="$1"; shift | 
 | 	local name="$1"; shift | 
 | 	local sfx="$1"; shift | 
 | 	local order="$1"; shift | 
 |  | 
 | 	local base="" | 
 | 	local file="" | 
 |  | 
 | 	# We may have fallbacks for a specific case (e.g. read_acquire()), or | 
 | 	# an entire class, e.g. *inc*(). | 
 | 	# | 
 | 	# Start at the most specific, and fall back to the most general. Once | 
 | 	# we find a specific fallback, don't bother looking for more. | 
 | 	for base in "${pfx}${name}${sfx}${order}" "${name}"; do | 
 | 		file="${ATOMICDIR}/fallbacks/${base}" | 
 |  | 
 | 		if [ -f "${file}" ]; then | 
 | 			printf "${file}" | 
 | 			break | 
 | 		fi | 
 | 	done | 
 | } | 
 |  | 
 | #gen_ret_type(meta, int) | 
 | gen_ret_type() { | 
 | 	local meta="$1"; shift | 
 | 	local int="$1"; shift | 
 |  | 
 | 	case "${meta}" in | 
 | 	[sv]) printf "void";; | 
 | 	[bB]) printf "bool";; | 
 | 	[aiIfFlR]) printf "${int}";; | 
 | 	esac | 
 | } | 
 |  | 
 | #gen_ret_stmt(meta) | 
 | gen_ret_stmt() | 
 | { | 
 | 	if meta_has_ret "${meta}"; then | 
 | 		printf "return "; | 
 | 	fi | 
 | } | 
 |  | 
 | # gen_param_name(arg) | 
 | gen_param_name() | 
 | { | 
 | 	# strip off the leading 'c' for 'cv' | 
 | 	local name="${1#c}" | 
 | 	printf "${name#*:}" | 
 | } | 
 |  | 
 | # gen_param_type(arg, int, atomic) | 
 | gen_param_type() | 
 | { | 
 | 	local type="${1%%:*}"; shift | 
 | 	local int="$1"; shift | 
 | 	local atomic="$1"; shift | 
 |  | 
 | 	case "${type}" in | 
 | 	i) type="${int} ";; | 
 | 	p) type="${int} *";; | 
 | 	v) type="${atomic}_t *";; | 
 | 	cv) type="const ${atomic}_t *";; | 
 | 	esac | 
 |  | 
 | 	printf "${type}" | 
 | } | 
 |  | 
 | #gen_param(arg, int, atomic) | 
 | gen_param() | 
 | { | 
 | 	local arg="$1"; shift | 
 | 	local int="$1"; shift | 
 | 	local atomic="$1"; shift | 
 | 	local name="$(gen_param_name "${arg}")" | 
 | 	local type="$(gen_param_type "${arg}" "${int}" "${atomic}")" | 
 |  | 
 | 	printf "${type}${name}" | 
 | } | 
 |  | 
 | #gen_params(int, atomic, arg...) | 
 | gen_params() | 
 | { | 
 | 	local int="$1"; shift | 
 | 	local atomic="$1"; shift | 
 |  | 
 | 	while [ "$#" -gt 0 ]; do | 
 | 		gen_param "$1" "${int}" "${atomic}" | 
 | 		[ "$#" -gt 1 ] && printf ", " | 
 | 		shift; | 
 | 	done | 
 | } | 
 |  | 
 | #gen_args(arg...) | 
 | gen_args() | 
 | { | 
 | 	while [ "$#" -gt 0 ]; do | 
 | 		printf "$(gen_param_name "$1")" | 
 | 		[ "$#" -gt 1 ] && printf ", " | 
 | 		shift; | 
 | 	done | 
 | } | 
 |  | 
 | #gen_proto_order_variants(meta, pfx, name, sfx, ...) | 
 | gen_proto_order_variants() | 
 | { | 
 | 	local meta="$1"; shift | 
 | 	local pfx="$1"; shift | 
 | 	local name="$1"; shift | 
 | 	local sfx="$1"; shift | 
 |  | 
 | 	gen_proto_order_variant "${meta}" "${pfx}" "${name}" "${sfx}" "" "$@" | 
 |  | 
 | 	if meta_has_acquire "${meta}"; then | 
 | 		gen_proto_order_variant "${meta}" "${pfx}" "${name}" "${sfx}" "_acquire" "$@" | 
 | 	fi | 
 | 	if meta_has_release "${meta}"; then | 
 | 		gen_proto_order_variant "${meta}" "${pfx}" "${name}" "${sfx}" "_release" "$@" | 
 | 	fi | 
 | 	if meta_has_relaxed "${meta}"; then | 
 | 		gen_proto_order_variant "${meta}" "${pfx}" "${name}" "${sfx}" "_relaxed" "$@" | 
 | 	fi | 
 | } | 
 |  | 
 | #gen_proto_variants(meta, name, ...) | 
 | gen_proto_variants() | 
 | { | 
 | 	local meta="$1"; shift | 
 | 	local name="$1"; shift | 
 | 	local pfx="" | 
 | 	local sfx="" | 
 |  | 
 | 	meta_in "${meta}" "fF" && pfx="fetch_" | 
 | 	meta_in "${meta}" "R" && sfx="_return" | 
 |  | 
 | 	gen_proto_order_variants "${meta}" "${pfx}" "${name}" "${sfx}" "$@" | 
 | } | 
 |  | 
 | #gen_proto(meta, ...) | 
 | gen_proto() { | 
 | 	local meta="$1"; shift | 
 | 	for m in $(echo "${meta}" | grep -o .); do | 
 | 		gen_proto_variants "${m}" "$@" | 
 | 	done | 
 | } |