// Part of the Carbon Language project, under the Apache License v2.0 with LLVM // Exceptions. See /LICENSE for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // // INCLUDE-FILE: toolchain/testing/testdata/min_prelude/primitives.carbon // // AUTOUPDATE // TIP: To test this file alone, run: // TIP: bazel test //toolchain/testing:file_test --test_arg=--file_tests=toolchain/check/testdata/interop/cpp/function/import/template.carbon // TIP: To dump output, run: // TIP: bazel run //toolchain/testing:file_test -- --dump_output --file_tests=toolchain/check/testdata/interop/cpp/function/import/template.carbon // --- deduced_or_explicit.carbon library "[[@TEST_NAME]]"; import Cpp inline ''' template T min(T a, T b) { return a < b ? a : b; } '''; let a: i32 = 1; let b: i32 = 2; let min_implicit: i32 = Cpp.min(a, b); let min_explicit: i32 = Cpp.min(i32, a, b); let min_explicit_literal: i32 = Cpp.min(i32, 1, 2); // --- template_template_arg.carbon library "[[@TEST_NAME]]"; import Cpp inline ''' template struct X; template struct Y; template typename... TT, typename... T> void f(T ...a) {} '''; fn Call() { let a: i32 = 0; let b: i32 = 0; // `a` and `b` are not classified as template arguments, so are passed as the // function parameter pack `a...`. Cpp.f(a, b); // `Cpp.X` and `Cpp.Y` are classified as template template arguments, so they // are passed as the template parameter pack `TT...`. Cpp.f(Cpp.X, Cpp.Y); // Pass some template arguments followed by some function arguments. Cpp.f(Cpp.X, Cpp.Y, a, b); Cpp.f(Cpp.X, a, b, a); } // --- nontype_template_arg.carbon library "[[@TEST_NAME]]"; import Cpp inline ''' template void f(T ...a) {} template void g(T ...a) {} '''; fn Call() { let a: i32 = 0; let b: i32 = 0; Cpp.f(a, b); // TODO: There should be a way to indicate that the arguments are template // arguments, not function arguments, when calling `f`. Cpp.f(1, 2); // Two function arguments. Cpp.g(1, 2); // Three template arguments. Cpp.g(1, 2, i32); // Three template arguments and two function arguments. Cpp.g(1, 2, i32, 3, 4); } // --- fail_function_arg_not_template_arg.carbon library "[[@TEST_NAME]]"; import Cpp inline ''' template void f(); '''; fn Call() { // We treat this as a function argument, even though that makes the call fail. // CHECK:STDERR: fail_function_arg_not_template_arg.carbon:[[@LINE+7]]:10: error: no matching function for call to 'f' [CppInteropParseError] // CHECK:STDERR: 17 | Cpp.f(1); // CHECK:STDERR: | ^ // CHECK:STDERR: fail_function_arg_not_template_arg.carbon:[[@LINE-8]]:22: note: candidate function template not viable: requires 0 arguments, but 1 was provided [CppInteropParseNote] // CHECK:STDERR: 5 | template void f(); // CHECK:STDERR: | ^ // CHECK:STDERR: Cpp.f(1); } // --- function_arg_not_template_arg_overloaded.carbon library "[[@TEST_NAME]]"; import Cpp inline ''' template void f(); void f(int n); '''; fn Call() { Cpp.f(1); } // --- overloaded_with_different_template_params.carbon library "[[@TEST_NAME]]"; import Cpp inline ''' template struct X; struct Y; template void f(); template void f(); template typename T> void f(); '''; fn Call() { Cpp.f(1, i32); Cpp.f(Cpp.X); Cpp.f(Cpp.Y); } // --- only_call_templates_when_given_template_args.carbon library "[[@TEST_NAME]]"; import Cpp inline ''' struct NonTemplate {}; struct Template {}; NonTemplate f(unsigned char, unsigned char); template Template f(unsigned char, int = 0); '''; // Prefers the non-template because it has a better match for the second parameter. let x: Cpp.NonTemplate = Cpp.f(1 as u8, 2 as u8); // Calls the template; the non-template is not considered due to the template // argument `i32`. let y: Cpp.Template = Cpp.f(i32, 1 as u8);