|  | #!/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" | 
|  | } | 
|  |  | 
|  | #meta_is_implicitly_relaxed(meta) | 
|  | meta_is_implicitly_relaxed() | 
|  | { | 
|  | meta_in "$1" "vls" | 
|  | } | 
|  |  | 
|  | #find_template(tmpltype, pfx, name, sfx, order) | 
|  | find_template() | 
|  | { | 
|  | local tmpltype="$1"; shift | 
|  | 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}" "${pfx}${name}${sfx}" "${name}"; do | 
|  | file="${ATOMICDIR}/${tmpltype}/${base}" | 
|  |  | 
|  | if [ -f "${file}" ]; then | 
|  | printf "${file}" | 
|  | break | 
|  | fi | 
|  | done | 
|  | } | 
|  |  | 
|  | #find_fallback_template(pfx, name, sfx, order) | 
|  | find_fallback_template() | 
|  | { | 
|  | find_template "fallbacks" "$@" | 
|  | } | 
|  |  | 
|  | #find_kerneldoc_template(pfx, name, sfx, order) | 
|  | find_kerneldoc_template() | 
|  | { | 
|  | find_template "kerneldoc" "$@" | 
|  | } | 
|  |  | 
|  | #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_desc_return(meta) | 
|  | gen_desc_return() | 
|  | { | 
|  | local meta="$1"; shift | 
|  |  | 
|  | case "${meta}" in | 
|  | [v]) | 
|  | printf "Return: Nothing." | 
|  | ;; | 
|  | [Ff]) | 
|  | printf "Return: The original value of @v." | 
|  | ;; | 
|  | [R]) | 
|  | printf "Return: The updated value of @v." | 
|  | ;; | 
|  | [l]) | 
|  | printf "Return: The value of @v." | 
|  | ;; | 
|  | esac | 
|  | } | 
|  |  | 
|  | #gen_template_kerneldoc(template, class, meta, pfx, name, sfx, order, atomic, int, args...) | 
|  | gen_template_kerneldoc() | 
|  | { | 
|  | local template="$1"; shift | 
|  | local class="$1"; shift | 
|  | local meta="$1"; shift | 
|  | local pfx="$1"; shift | 
|  | local name="$1"; shift | 
|  | local sfx="$1"; shift | 
|  | local order="$1"; shift | 
|  | local atomic="$1"; shift | 
|  | local int="$1"; shift | 
|  |  | 
|  | local atomicname="${atomic}_${pfx}${name}${sfx}${order}" | 
|  |  | 
|  | local ret="$(gen_ret_type "${meta}" "${int}")" | 
|  | local retstmt="$(gen_ret_stmt "${meta}")" | 
|  | local params="$(gen_params "${int}" "${atomic}" "$@")" | 
|  | local args="$(gen_args "$@")" | 
|  | local desc_order="" | 
|  | local desc_instrumentation="" | 
|  | local desc_return="" | 
|  |  | 
|  | if [ ! -z "${order}" ]; then | 
|  | desc_order="${order##_}" | 
|  | elif meta_is_implicitly_relaxed "${meta}"; then | 
|  | desc_order="relaxed" | 
|  | else | 
|  | desc_order="full" | 
|  | fi | 
|  |  | 
|  | if [ -z "${class}" ]; then | 
|  | desc_noinstr="Unsafe to use in noinstr code; use raw_${atomicname}() there." | 
|  | else | 
|  | desc_noinstr="Safe to use in noinstr code; prefer ${atomicname}() elsewhere." | 
|  | fi | 
|  |  | 
|  | desc_return="$(gen_desc_return "${meta}")" | 
|  |  | 
|  | . ${template} | 
|  | } | 
|  |  | 
|  | #gen_kerneldoc(class, meta, pfx, name, sfx, order, atomic, int, args...) | 
|  | gen_kerneldoc() | 
|  | { | 
|  | local class="$1"; shift | 
|  | local meta="$1"; shift | 
|  | local pfx="$1"; shift | 
|  | local name="$1"; shift | 
|  | local sfx="$1"; shift | 
|  | local order="$1"; shift | 
|  |  | 
|  | local atomicname="${atomic}_${pfx}${name}${sfx}${order}" | 
|  |  | 
|  | local tmpl="$(find_kerneldoc_template "${pfx}" "${name}" "${sfx}" "${order}")" | 
|  | if [ -z "${tmpl}" ]; then | 
|  | printf "/*\n" | 
|  | printf " * No kerneldoc available for ${class}${atomicname}\n" | 
|  | printf " */\n" | 
|  | else | 
|  | gen_template_kerneldoc "${tmpl}" "${class}" "${meta}" "${pfx}" "${name}" "${sfx}" "${order}" "$@" | 
|  | fi | 
|  | } | 
|  |  | 
|  | #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 | 
|  | } |