2022-11-01 18:10:01 +00:00
|
|
|
name: Build sub target
|
|
|
|
|
|
|
|
on:
|
|
|
|
workflow_call:
|
|
|
|
inputs:
|
|
|
|
target:
|
|
|
|
required: true
|
|
|
|
type: string
|
2022-12-03 02:04:40 +00:00
|
|
|
testing:
|
|
|
|
type: boolean
|
2022-11-05 13:27:11 +00:00
|
|
|
build_toolchain:
|
|
|
|
type: boolean
|
2022-11-01 18:10:01 +00:00
|
|
|
include_feeds:
|
|
|
|
type: boolean
|
|
|
|
build_full:
|
|
|
|
type: boolean
|
|
|
|
build_all_modules:
|
|
|
|
type: boolean
|
|
|
|
build_all_kmods:
|
|
|
|
type: boolean
|
2022-12-03 02:04:40 +00:00
|
|
|
build_all_boards:
|
|
|
|
type: boolean
|
2022-11-01 18:10:01 +00:00
|
|
|
|
|
|
|
permissions:
|
|
|
|
contents: read
|
|
|
|
|
|
|
|
jobs:
|
|
|
|
setup_build:
|
|
|
|
name: Setup build
|
|
|
|
runs-on: ubuntu-latest
|
|
|
|
outputs:
|
|
|
|
owner_lc: ${{ steps.lower_owner.outputs.owner_lc }}
|
|
|
|
ccache_hash: ${{ steps.ccache_hash.outputs.ccache_hash }}
|
2022-11-27 18:53:08 +00:00
|
|
|
container_tag: ${{ steps.determine_tools_container.outputs.container_tag }}
|
2022-11-01 18:10:01 +00:00
|
|
|
|
|
|
|
steps:
|
|
|
|
- name: Checkout
|
|
|
|
uses: actions/checkout@v3
|
|
|
|
|
|
|
|
- name: Set lower case owner name
|
|
|
|
id: lower_owner
|
|
|
|
run: |
|
|
|
|
OWNER_LC=$(echo "${{ github.repository_owner }}" \
|
|
|
|
| tr '[:upper:]' '[:lower:]')
|
|
|
|
echo "owner_lc=$OWNER_LC" >> $GITHUB_OUTPUT
|
|
|
|
|
|
|
|
- name: Generate ccache hash
|
|
|
|
id: ccache_hash
|
|
|
|
run: |
|
|
|
|
CCACHE_HASH=$(md5sum include/kernel-* | awk '{ print $1 }' \
|
|
|
|
| md5sum | awk '{ print $1 }')
|
|
|
|
echo "ccache_hash=$CCACHE_HASH" >> $GITHUB_OUTPUT
|
|
|
|
|
2022-11-27 18:53:08 +00:00
|
|
|
# Per branch tools container tag
|
|
|
|
# By default stick to latest
|
|
|
|
# For official test targetting openwrt stable branch
|
|
|
|
# Get the branch or parse the tag and push dedicated tools containers
|
|
|
|
# For local test to use the correct container for stable release testing
|
|
|
|
# you need to use for the branch name a prefix of openwrt-[0-9][0-9].[0-9][0-9]-
|
|
|
|
- name: Determine tools container tag
|
|
|
|
id: determine_tools_container
|
|
|
|
run: |
|
|
|
|
CONTAINER_TAG=latest
|
|
|
|
if [ -n "${{ github.base_ref }}" ]; then
|
2022-12-04 19:28:28 +00:00
|
|
|
if echo "${{ github.base_ref }}" | grep -q -E '^openwrt-[0-9][0-9]\.[0-9][0-9]$'; then
|
2022-11-27 18:53:08 +00:00
|
|
|
CONTAINER_TAG="${{ github.base_ref }}"
|
|
|
|
fi
|
|
|
|
elif [ ${{ github.ref_type }} == "branch" ]; then
|
2022-12-04 19:28:28 +00:00
|
|
|
if echo "${{ github.ref_name }}" | grep -q -E '^openwrt-[0-9][0-9]\.[0-9][0-9]$'; then
|
|
|
|
CONTAINER_TAG=${{ github.ref_name }}
|
|
|
|
elif echo "${{ github.ref_name }}" | grep -q -E '^openwrt-[0-9][0-9]\.[0-9][0-9]-'; then
|
2022-11-27 18:53:08 +00:00
|
|
|
CONTAINER_TAG="$(echo ${{ github.ref_name }} | sed 's/^\(openwrt-[0-9][0-9]\.[0-9][0-9]\)-.*/\1/')"
|
|
|
|
fi
|
|
|
|
elif [ ${{ github.ref_type }} == "tag" ]; then
|
2022-12-04 19:28:28 +00:00
|
|
|
if echo "${{ github.ref_name }}" | grep -q -E '^v[0-9][0-9]\.[0-9][0-9]\..+'; then
|
|
|
|
CONTAINER_TAG=openwrt-"$(echo ${{ github.ref_name }} | sed 's/^v\([0-9][0-9]\.[0-9][0-9]\)\..\+/\1/')"
|
2022-11-27 18:53:08 +00:00
|
|
|
fi
|
|
|
|
fi
|
|
|
|
echo "Tools container to use tools:$CONTAINER_TAG"
|
|
|
|
echo "container_tag=$CONTAINER_TAG" >> $GITHUB_OUTPUT
|
|
|
|
|
2022-11-01 18:10:01 +00:00
|
|
|
build:
|
|
|
|
name: Build with external toolchain
|
|
|
|
needs: setup_build
|
|
|
|
runs-on: ubuntu-latest
|
|
|
|
|
2022-11-27 18:53:08 +00:00
|
|
|
container: ghcr.io/${{ needs.setup_build.outputs.owner_lc }}/tools:${{ needs.setup_build.outputs.container_tag }}
|
2022-11-01 18:10:01 +00:00
|
|
|
|
|
|
|
permissions:
|
|
|
|
contents: read
|
|
|
|
packages: read
|
|
|
|
|
|
|
|
steps:
|
|
|
|
- name: Checkout master directory
|
|
|
|
uses: actions/checkout@v3
|
|
|
|
with:
|
|
|
|
path: openwrt
|
|
|
|
|
|
|
|
- name: Checkout packages feed
|
|
|
|
if: inputs.include_feeds == true
|
|
|
|
uses: actions/checkout@v3
|
|
|
|
with:
|
|
|
|
repository: openwrt/packages
|
|
|
|
path: openwrt/feeds/packages
|
|
|
|
|
|
|
|
- name: Checkout luci feed
|
|
|
|
if: inputs.include_feeds == true
|
|
|
|
uses: actions/checkout@v3
|
|
|
|
with:
|
|
|
|
repository: openwrt/luci
|
|
|
|
path: openwrt/feeds/luci
|
|
|
|
|
|
|
|
- name: Checkout routing feed
|
|
|
|
if: inputs.include_feeds == true
|
|
|
|
uses: actions/checkout@v3
|
|
|
|
with:
|
|
|
|
repository: openwrt/routing
|
|
|
|
path: openwrt/feeds/routing
|
|
|
|
|
|
|
|
- name: Checkout telephony feed
|
|
|
|
if: inputs.include_feeds == true
|
|
|
|
uses: actions/checkout@v3
|
|
|
|
with:
|
|
|
|
repository: openwrt/telephony
|
|
|
|
path: openwrt/feeds/telephony
|
|
|
|
|
|
|
|
- name: Fix permission
|
|
|
|
run: |
|
|
|
|
chown -R buildbot:buildbot openwrt
|
|
|
|
|
|
|
|
- name: Initialization environment
|
|
|
|
run: |
|
|
|
|
TARGET=$(echo ${{ inputs.target }} | cut -d "/" -f 1)
|
|
|
|
SUBTARGET=$(echo ${{ inputs.target }} | cut -d "/" -f 2)
|
|
|
|
echo "TARGET=$TARGET" >> "$GITHUB_ENV"
|
|
|
|
echo "SUBTARGET=$SUBTARGET" >> "$GITHUB_ENV"
|
|
|
|
|
|
|
|
- name: Update & Install feeds
|
|
|
|
if: inputs.include_feeds == true
|
|
|
|
shell: su buildbot -c "sh -e {0}"
|
|
|
|
working-directory: openwrt
|
|
|
|
run: |
|
|
|
|
./scripts/feeds update -a
|
|
|
|
./scripts/feeds install -a
|
|
|
|
|
|
|
|
- name: Parse toolchain file
|
2022-11-05 13:27:11 +00:00
|
|
|
if: inputs.build_toolchain == false
|
2022-11-28 15:12:13 +00:00
|
|
|
id: parse-toolchain
|
2022-11-01 18:10:01 +00:00
|
|
|
working-directory: openwrt
|
|
|
|
run: |
|
2022-11-28 11:44:01 +00:00
|
|
|
TOOLCHAIN_PATH=snapshots
|
|
|
|
|
|
|
|
if [ -n "${{ github.base_ref }}" ]; then
|
2022-12-04 19:36:11 +00:00
|
|
|
if echo "${{ github.base_ref }}" | grep -q -E '^openwrt-[0-9][0-9]\.[0-9][0-9]$'; then
|
|
|
|
major_ver="$(echo ${{ github.base_ref }} | sed 's/^openwrt-/v/')"
|
2022-11-28 11:44:01 +00:00
|
|
|
fi
|
|
|
|
elif [ "${{ github.ref_type }}" = "branch" ]; then
|
2022-12-04 19:36:11 +00:00
|
|
|
if echo "${{ github.ref_name }}" | grep -q -E '^openwrt-[0-9][0-9]\.[0-9][0-9]$'; then
|
|
|
|
major_ver="$(echo ${{ github.ref_name }} | sed 's/^openwrt-/v/')"
|
|
|
|
elif echo "${{ github.ref_name }}" | grep -q -E '^openwrt-[0-9][0-9]\.[0-9][0-9]-'; then
|
|
|
|
major_ver="$(echo ${{ github.ref_name }} | sed 's/^openwrt-\([0-9][0-9]\.[0-9][0-9]\)-.*/v\1/')"
|
2022-11-28 11:44:01 +00:00
|
|
|
fi
|
|
|
|
elif [ "${{ github.ref_type }}" = "tag" ]; then
|
2022-12-04 19:36:11 +00:00
|
|
|
if echo "${{ github.ref_name }}" | grep -q -E '^v[0-9][0-9]\.[0-9][0-9]\..+'; then
|
|
|
|
major_ver="$(echo ${{ github.ref_name }} | sed 's/^\(v[0-9][0-9]\.[0-9][0-9]\)\..\+/\1/')"
|
2022-11-28 11:44:01 +00:00
|
|
|
fi
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [ -n "$major_ver" ]; then
|
|
|
|
git fetch --tags
|
|
|
|
latest_tag="$(git tag --sort=-creatordate -l $major_ver* | head -n1)"
|
|
|
|
if [ -n "$latest_tag" ]; then
|
|
|
|
TOOLCHAIN_PATH=releases/$(echo $latest_tag | sed 's/^v//')
|
|
|
|
fi
|
|
|
|
fi
|
|
|
|
|
2022-11-28 15:12:13 +00:00
|
|
|
SUMS_FILE="https://downloads.cdn.openwrt.org/$TOOLCHAIN_PATH/targets/${{ env.TARGET }}/${{ env.SUBTARGET }}/sha256sums"
|
|
|
|
if curl $SUMS_FILE | grep -q ".*openwrt-toolchain.*tar.xz"; then
|
|
|
|
TOOLCHAIN_STRING="$( curl $SUMS_FILE | grep ".*openwrt-toolchain.*tar.xz")"
|
|
|
|
TOOLCHAIN_FILE=$(echo "$TOOLCHAIN_STRING" | sed -n -e 's/.*\(openwrt-toolchain.*\).tar.xz/\1/p')
|
|
|
|
TOOLCHAIN_SHA256=$(echo "$TOOLCHAIN_STRING" | cut -d ' ' -f 1)
|
|
|
|
|
|
|
|
echo "toolchain-type=external_toolchain" >> $GITHUB_OUTPUT
|
|
|
|
elif curl $SUMS_FILE | grep -q ".*openwrt-sdk.*tar.xz"; then
|
|
|
|
TOOLCHAIN_STRING="$( curl $SUMS_FILE | grep ".*openwrt-sdk.*tar.xz")"
|
|
|
|
TOOLCHAIN_FILE=$(echo "$TOOLCHAIN_STRING" | sed -n -e 's/.*\(openwrt-sdk.*\).tar.xz/\1/p')
|
|
|
|
TOOLCHAIN_SHA256=$(echo "$TOOLCHAIN_STRING" | cut -d ' ' -f 1)
|
|
|
|
|
|
|
|
echo "toolchain-type=external_sdk" >> $GITHUB_OUTPUT
|
|
|
|
fi
|
2022-11-01 18:10:01 +00:00
|
|
|
|
|
|
|
echo "TOOLCHAIN_FILE=$TOOLCHAIN_FILE" >> "$GITHUB_ENV"
|
|
|
|
echo "TOOLCHAIN_SHA256=$TOOLCHAIN_SHA256" >> "$GITHUB_ENV"
|
2022-11-28 11:44:01 +00:00
|
|
|
echo "TOOLCHAIN_PATH=$TOOLCHAIN_PATH" >> "$GITHUB_ENV"
|
2022-11-01 18:10:01 +00:00
|
|
|
|
2022-11-28 15:12:13 +00:00
|
|
|
- name: Cache external toolchain/sdk
|
2022-11-05 13:27:11 +00:00
|
|
|
if: inputs.build_toolchain == false
|
2022-11-01 18:10:01 +00:00
|
|
|
id: cache-external-toolchain
|
|
|
|
uses: actions/cache@v3
|
|
|
|
with:
|
|
|
|
path: openwrt/${{ env.TOOLCHAIN_FILE }}
|
2022-11-28 15:12:13 +00:00
|
|
|
key: ${{ env.TOOLCHAIN_FILE }}-${{ steps.parse-toolchain.outputs.toolchain-type }}-${{ env.TOOLCHAIN_SHA256 }}
|
2022-11-01 18:10:01 +00:00
|
|
|
|
|
|
|
- name: Cache ccache
|
|
|
|
uses: actions/cache@v3
|
|
|
|
with:
|
|
|
|
path: openwrt/.ccache
|
|
|
|
key: ccache-kernel-${{ env.TARGET }}/${{ env.SUBTARGET }}-${{ needs.setup_build.outputs.ccache_hash }}
|
|
|
|
restore-keys: |
|
|
|
|
ccache-kernel-${{ env.TARGET }}/${{ env.SUBTARGET }}-
|
|
|
|
|
2022-11-28 15:12:13 +00:00
|
|
|
- name: Download external toolchain/sdk
|
2022-11-05 13:27:11 +00:00
|
|
|
if: inputs.build_toolchain == false && steps.cache-external-toolchain.outputs.cache-hit != 'true'
|
2022-11-01 18:10:01 +00:00
|
|
|
shell: su buildbot -c "sh -e {0}"
|
|
|
|
working-directory: openwrt
|
|
|
|
run: |
|
2022-11-28 11:44:01 +00:00
|
|
|
wget -O - https://downloads.cdn.openwrt.org/${{ env.TOOLCHAIN_PATH }}/targets/${{ env.TARGET }}/${{ env.SUBTARGET }}/${{ env.TOOLCHAIN_FILE }}.tar.xz \
|
2022-11-01 18:10:01 +00:00
|
|
|
| tar --xz -xf -
|
|
|
|
|
|
|
|
- name: Extract prebuilt tools
|
|
|
|
shell: su buildbot -c "sh -e {0}"
|
|
|
|
working-directory: openwrt
|
|
|
|
run: ./scripts/ext-tools.sh --tools /tools.tar
|
|
|
|
|
2022-12-03 02:04:40 +00:00
|
|
|
- name: Configure testing kernel
|
|
|
|
if: inputs.testing == true
|
|
|
|
shell: su buildbot -c "sh -e {0}"
|
|
|
|
working-directory: openwrt
|
|
|
|
run: |
|
|
|
|
echo CONFIG_TESTING_KERNEL=y >> .config
|
|
|
|
|
2022-11-01 18:10:01 +00:00
|
|
|
- name: Configure all kernel modules
|
|
|
|
if: inputs.build_all_kmods == true
|
|
|
|
shell: su buildbot -c "sh -e {0}"
|
|
|
|
working-directory: openwrt
|
|
|
|
run: |
|
|
|
|
echo CONFIG_ALL_KMODS=y >> .config
|
|
|
|
|
|
|
|
- name: Configure all modules
|
|
|
|
if: inputs.build_all_modules == true
|
|
|
|
shell: su buildbot -c "sh -e {0}"
|
|
|
|
working-directory: openwrt
|
|
|
|
run: |
|
|
|
|
echo CONFIG_ALL=y >> .config
|
|
|
|
|
2022-12-03 02:04:40 +00:00
|
|
|
- name: Configure all boards
|
|
|
|
if: inputs.build_all_boards == true
|
|
|
|
shell: su buildbot -c "sh -e {0}"
|
|
|
|
working-directory: openwrt
|
|
|
|
run: |
|
|
|
|
echo CONFIG_TARGET_MULTI_PROFILE=y >> .config
|
|
|
|
echo CONFIG_TARGET_PER_DEVICE_ROOTFS=y >> .config
|
|
|
|
echo CONFIG_TARGET_ALL_PROFILES=y >> .config
|
|
|
|
|
2022-11-01 18:10:01 +00:00
|
|
|
- name: Configure external toolchain
|
2022-11-28 15:12:13 +00:00
|
|
|
if: inputs.build_toolchain == false && steps.parse-toolchain.outputs.toolchain-type == 'external_toolchain'
|
2022-11-01 18:10:01 +00:00
|
|
|
shell: su buildbot -c "sh -e {0}"
|
|
|
|
working-directory: openwrt
|
|
|
|
run: |
|
|
|
|
echo CONFIG_DEVEL=y >> .config
|
|
|
|
echo CONFIG_AUTOREMOVE=y >> .config
|
|
|
|
echo CONFIG_CCACHE=y >> .config
|
|
|
|
|
|
|
|
./scripts/ext-toolchain.sh \
|
|
|
|
--toolchain ${{ env.TOOLCHAIN_FILE }}/toolchain-* \
|
|
|
|
--overwrite-config \
|
2022-11-28 15:12:13 +00:00
|
|
|
--config ${{ env.TARGET }}/${{ env.SUBTARGET }}
|
|
|
|
|
2022-12-05 22:23:04 +00:00
|
|
|
- name: Adapt external sdk to external toolchain format
|
2022-12-07 17:09:18 +00:00
|
|
|
if: inputs.build_toolchain == false && steps.parse-toolchain.outputs.toolchain-type == 'external_sdk' && steps.cache-external-toolchain.outputs.cache-hit != 'true'
|
2022-12-05 22:23:04 +00:00
|
|
|
shell: su buildbot -c "sh -e {0}"
|
|
|
|
working-directory: openwrt
|
|
|
|
run: |
|
|
|
|
TOOLCHAIN_DIR=${{ env.TOOLCHAIN_FILE }}/staging_dir/$(ls ${{ env.TOOLCHAIN_FILE }}/staging_dir | grep toolchain)
|
|
|
|
TOOLCHAIN_BIN=$TOOLCHAIN_DIR/bin
|
|
|
|
OPENWRT_DIR=$(pwd)
|
|
|
|
|
|
|
|
# Find target name from toolchain info.mk
|
|
|
|
GNU_TARGET_NAME=$(cat $TOOLCHAIN_DIR/info.mk | grep TARGET_CROSS | sed 's/^TARGET_CROSS=\(.*\)-$/\1/')
|
|
|
|
|
|
|
|
cd $TOOLCHAIN_BIN
|
|
|
|
|
|
|
|
# Revert sdk wrapper scripts applied to all the bins
|
|
|
|
for app in $(find . -name "*.bin"); do
|
|
|
|
TARGET_APP=$(echo $app | sed 's/\.\/\.\(.*\)\.bin/\1/')
|
|
|
|
rm $TARGET_APP
|
|
|
|
mv .$TARGET_APP.bin $TARGET_APP
|
|
|
|
done
|
|
|
|
|
|
|
|
# Setup the wrapper script in the sdk toolchain dir simulating an external toolchain build
|
|
|
|
cp $OPENWRT_DIR/target/toolchain/files/wrapper.sh $GNU_TARGET_NAME-wrapper.sh
|
|
|
|
for app in cc gcc g++ c++ cpp ld as ; do
|
|
|
|
[ -f $GNU_TARGET_NAME-$app ] && mv $GNU_TARGET_NAME-$app $GNU_TARGET_NAME-$app.bin
|
|
|
|
ln -sf $GNU_TARGET_NAME-wrapper.sh $GNU_TARGET_NAME-$app
|
|
|
|
done
|
|
|
|
|
2022-11-28 15:12:13 +00:00
|
|
|
- name: Configure external toolchain with sdk
|
|
|
|
if: inputs.build_toolchain == false && steps.parse-toolchain.outputs.toolchain-type == 'external_sdk'
|
|
|
|
shell: su buildbot -c "sh -e {0}"
|
|
|
|
working-directory: openwrt
|
|
|
|
run: |
|
|
|
|
echo CONFIG_DEVEL=y >> .config
|
|
|
|
echo CONFIG_AUTOREMOVE=y >> .config
|
|
|
|
echo CONFIG_CCACHE=y >> .config
|
|
|
|
|
|
|
|
./scripts/ext-toolchain.sh \
|
|
|
|
--toolchain ${{ env.TOOLCHAIN_FILE }}/staging_dir/toolchain-* \
|
|
|
|
--overwrite-config \
|
2022-11-01 18:10:01 +00:00
|
|
|
--config ${{ env.TARGET }}/${{ env.SUBTARGET }}
|
|
|
|
|
2022-11-05 13:27:11 +00:00
|
|
|
- name: Configure internal toolchain
|
|
|
|
if: inputs.build_toolchain == true
|
|
|
|
shell: su buildbot -c "sh -e {0}"
|
|
|
|
working-directory: openwrt
|
|
|
|
run: |
|
|
|
|
echo CONFIG_DEVEL=y >> .config
|
|
|
|
echo CONFIG_AUTOREMOVE=y >> .config
|
|
|
|
echo CONFIG_CCACHE=y >> .config
|
|
|
|
|
|
|
|
echo "CONFIG_TARGET_${{ env.TARGET }}=y" >> .config
|
|
|
|
echo "CONFIG_TARGET_${{ env.TARGET }}_${{ env.SUBTARGET }}=y" >> .config
|
|
|
|
|
|
|
|
make defconfig
|
|
|
|
|
2022-11-01 18:10:01 +00:00
|
|
|
- name: Show configuration
|
|
|
|
shell: su buildbot -c "sh -e {0}"
|
|
|
|
working-directory: openwrt
|
|
|
|
run: ./scripts/diffconfig.sh
|
|
|
|
|
|
|
|
- name: Build tools
|
|
|
|
shell: su buildbot -c "sh -e {0}"
|
|
|
|
working-directory: openwrt
|
|
|
|
run: make tools/install -j$(nproc) BUILD_LOG=1 || ret=$? .github/workflows/scripts/show_build_failures.sh
|
|
|
|
|
|
|
|
- name: Build toolchain
|
|
|
|
shell: su buildbot -c "sh -e {0}"
|
|
|
|
working-directory: openwrt
|
|
|
|
run: make toolchain/install -j$(nproc) BUILD_LOG=1 || ret=$? .github/workflows/scripts/show_build_failures.sh
|
|
|
|
|
|
|
|
- name: Build Kernel
|
|
|
|
shell: su buildbot -c "sh -e {0}"
|
|
|
|
working-directory: openwrt
|
|
|
|
run: make target/compile -j$(nproc) BUILD_LOG=1 || ret=$? .github/workflows/scripts/show_build_failures.sh
|
|
|
|
|
|
|
|
- name: Build Kernel Kmods
|
|
|
|
shell: su buildbot -c "sh -e {0}"
|
|
|
|
working-directory: openwrt
|
|
|
|
run: make package/linux/compile -j$(nproc) BUILD_LOG=1 || ret=$? .github/workflows/scripts/show_build_failures.sh
|
|
|
|
|
|
|
|
- name: Build everything
|
|
|
|
if: inputs.build_full == true
|
|
|
|
shell: su buildbot -c "sh -e {0}"
|
|
|
|
working-directory: openwrt
|
|
|
|
run: make -j$(nproc) BUILD_LOG=1 || ret=$? .github/workflows/scripts/show_build_failures.sh
|
|
|
|
|
|
|
|
- name: Upload logs
|
|
|
|
if: failure()
|
|
|
|
uses: actions/upload-artifact@v3
|
|
|
|
with:
|
|
|
|
name: ${{ env.TARGET }}-${{ env.SUBTARGET }}-logs
|
|
|
|
path: "openwrt/logs"
|