| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149 |
- // 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<typename T> 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<typename> struct X;
- template<typename> struct Y;
- template<template<typename> 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<int... I, typename... T> void f(T ...a) {}
- template<int I = 0, int J = 0, typename = void, typename... T> 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<int N> 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<int N> void f();
- // CHECK:STDERR: | ^
- // CHECK:STDERR:
- Cpp.f(1);
- }
- // --- function_arg_not_template_arg_overloaded.carbon
- library "[[@TEST_NAME]]";
- import Cpp inline '''
- template<int N> void f();
- void f(int n);
- ''';
- fn Call() {
- Cpp.f(1);
- }
- // --- overloaded_with_different_template_params.carbon
- library "[[@TEST_NAME]]";
- import Cpp inline '''
- template<typename T> struct X;
- struct Y;
- template<int N, typename T> void f();
- template<typename T> void f();
- template<template<typename> 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<typename = void> 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);
|