/*
 * Copyright (c) 2022 Arm Limited.
 *
 * SPDX-License-Identifier: MIT
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to
 * deal in the Software without restriction, including without limitation the
 * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
 * sell copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in all
 * copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 * SOFTWARE.
 */

#if defined(__ARM_FEATURE_FP16_VECTOR_ARITHMETIC) && defined(ENABLE_FP16_KERNELS)

#include "arm_compute/core/Helpers.h"

#include "src/cpu/kernels/elementwise_binary/generic/sve/impl.h"
namespace arm_compute
{
namespace cpu
{
template <ArithmeticOperation op>
void sve_fp16_elementwise_binary(const ITensor *in1, const ITensor *in2, ITensor *out, const Window &window)
{
    return elementwise_arithmetic_op<float16_t>(in1, in2, out, op, window);
}

template void sve_fp16_elementwise_binary<ArithmeticOperation::ADD>(const ITensor *in1,
                                                                    const ITensor *in2,
                                                                    ITensor       *out,
                                                                    const Window  &window);
template void sve_fp16_elementwise_binary<ArithmeticOperation::SUB>(const ITensor *in1,
                                                                    const ITensor *in2,
                                                                    ITensor       *out,
                                                                    const Window  &window);
template void sve_fp16_elementwise_binary<ArithmeticOperation::DIV>(const ITensor *in1,
                                                                    const ITensor *in2,
                                                                    ITensor       *out,
                                                                    const Window  &window);
template void sve_fp16_elementwise_binary<ArithmeticOperation::MIN>(const ITensor *in1,
                                                                    const ITensor *in2,
                                                                    ITensor       *out,
                                                                    const Window  &window);
template void sve_fp16_elementwise_binary<ArithmeticOperation::MAX>(const ITensor *in1,
                                                                    const ITensor *in2,
                                                                    ITensor       *out,
                                                                    const Window  &window);
template void sve_fp16_elementwise_binary<ArithmeticOperation::SQUARED_DIFF>(const ITensor *in1,
                                                                             const ITensor *in2,
                                                                             ITensor       *out,
                                                                             const Window  &window);
template void sve_fp16_elementwise_binary<ArithmeticOperation::POWER>(const ITensor *in1,
                                                                      const ITensor *in2,
                                                                      ITensor       *out,
                                                                      const Window  &window);
template void sve_fp16_elementwise_binary<ArithmeticOperation::PRELU>(const ITensor *in1,
                                                                      const ITensor *in2,
                                                                      ITensor       *out,
                                                                      const Window  &window);

template <ComparisonOperation op>
void sve_fp16_comparison_elementwise_binary(const ITensor *in1, const ITensor *in2, ITensor *out, const Window &window)
{
    return elementwise_comparison_op<float16_t>(in1, in2, out, op, window);
}

template void sve_fp16_comparison_elementwise_binary<ComparisonOperation::Equal>(const ITensor *in1,
                                                                                 const ITensor *in2,
                                                                                 ITensor       *out,
                                                                                 const Window  &window);
template void sve_fp16_comparison_elementwise_binary<ComparisonOperation::NotEqual>(const ITensor *in1,
                                                                                    const ITensor *in2,
                                                                                    ITensor       *out,
                                                                                    const Window  &window);
template void sve_fp16_comparison_elementwise_binary<ComparisonOperation::Greater>(const ITensor *in1,
                                                                                   const ITensor *in2,
                                                                                   ITensor       *out,
                                                                                   const Window  &window);
template void sve_fp16_comparison_elementwise_binary<ComparisonOperation::GreaterEqual>(const ITensor *in1,
                                                                                        const ITensor *in2,
                                                                                        ITensor       *out,
                                                                                        const Window  &window);
template void sve_fp16_comparison_elementwise_binary<ComparisonOperation::Less>(const ITensor *in1,
                                                                                const ITensor *in2,
                                                                                ITensor       *out,
                                                                                const Window  &window);
template void sve_fp16_comparison_elementwise_binary<ComparisonOperation::LessEqual>(const ITensor *in1,
                                                                                     const ITensor *in2,
                                                                                     ITensor       *out,
                                                                                     const Window  &window);

} // namespace cpu
} // namespace arm_compute

#endif /* defined(__ARM_FEATURE_FP16_VECTOR_ARITHMETIC) && defined(ENABLE_FP16_KERNELS) */
