From 26ca9c5b0657489f82a221c62903cbc802aaa6a0 Mon Sep 17 00:00:00 2001 From: Yong He Date: Thu, 19 Sep 2024 15:44:32 -0700 Subject: Synthesize conformance for generic requirements. (#5111) * Synthesize conformance for generic requirements. * Fix. * Fix build error. * address code review. --- .../interfaces/generic-requirement-synth-2.slang | 35 +++++++++ .../interfaces/generic-requirement-synth.slang | 90 ++++++++++++++++++++++ 2 files changed, 125 insertions(+) create mode 100644 tests/language-feature/interfaces/generic-requirement-synth-2.slang create mode 100644 tests/language-feature/interfaces/generic-requirement-synth.slang (limited to 'tests') diff --git a/tests/language-feature/interfaces/generic-requirement-synth-2.slang b/tests/language-feature/interfaces/generic-requirement-synth-2.slang new file mode 100644 index 000000000..9066fc5b7 --- /dev/null +++ b/tests/language-feature/interfaces/generic-requirement-synth-2.slang @@ -0,0 +1,35 @@ +// Test that we allow type conformances whose base interface is generic. + +//TEST(compute):COMPARE_COMPUTE_EX(filecheck-buffer=CHECK):-dx11 -compute -output-using-type +//TEST(compute):COMPARE_COMPUTE_EX(filecheck-buffer=CHECK):-vk -compute -output-using-type + +interface IStack +{ + IStack popN(); + + int get(); +} +struct StackImpl : IStack +{ + // member 'popN' does not match interface requirement. + StackImpl popN() { return StackImpl(); } + + int get() { return D; } +} + +int helper>(T stack) +{ + return stack.popN<2>().get(); +} + +//TEST_INPUT: set outputBuffer = out ubuffer(data=[0 0 0 0], stride=4); +RWStructuredBuffer outputBuffer; + +[numthreads(1, 1, 1)] +void computeMain() +{ + StackImpl<5> obj = StackImpl<5>(); + + // CHECK: 3 + outputBuffer[0] = helper(obj); +} \ No newline at end of file diff --git a/tests/language-feature/interfaces/generic-requirement-synth.slang b/tests/language-feature/interfaces/generic-requirement-synth.slang new file mode 100644 index 000000000..1ac118014 --- /dev/null +++ b/tests/language-feature/interfaces/generic-requirement-synth.slang @@ -0,0 +1,90 @@ +// Test that we can synthesize requirements for generic methods. + +//TEST(compute):COMPARE_COMPUTE_EX(filecheck-buffer=CHECK):-dx11 -compute -output-using-type +//TEST(compute):COMPARE_COMPUTE_EX(filecheck-buffer=CHECK):-vk -compute -output-using-type + +interface IBase +{ + static float get(); +} +interface IBar : IBase +{ + float derivedMethod(); +} + +struct Bar : IBar +{ + static float get() { return 1.0f; } + float derivedMethod() { return 2.0f; } +} + +interface ITestInterface +{ + Real sample(T t); + + __init(T t); + + __generic + __subscript(T t)->Real { get; } +} + +struct TestInterfaceImpl : ITestInterface +{ + // The signature of this sample method is different from the one in the + // interface. However, we should be able to form a call into this method + // from the synthesized implementation matching the interface definition, + // so the conformance should hold. + Real sample(T t) + { + return x + Real(T.get()); + } + + // Test the same thing for constructors. + __init(T t) + { + x = Real(T.get()); + } + + // Test the same thing for subscript operators. + __generic + __subscript(T t)->Real { get { return x + Real(T.get()); } } + Real x; +} + +float test(ITestInterface obj) +{ + Bar b = {}; + return obj.sample(b); +} + +float test1(ITestInterface obj) +{ + Bar b = {}; + return obj[b]; +} + +float test2>() +{ + Bar b = {}; + T obj = T(b); + return obj[b]; +} + +//TEST_INPUT: set outputBuffer = out ubuffer(data=[0 0 0 0], stride=4); +RWStructuredBuffer outputBuffer; + +[numthreads(1, 1, 1)] +void computeMain() +{ + TestInterfaceImpl obj; + obj.x = 1.0f; + + // CHECK: 2 + outputBuffer[0] = int(test(obj)); + + // CHECK: 2 + outputBuffer[1] = int(test1(obj)); + + // CHECK: 2 + outputBuffer[3] = int(test2>()); +} \ No newline at end of file -- cgit v1.2.3