This is an automated email from the ASF dual-hosted git repository.

emilles pushed a commit to branch GROOVY_5_0_X
in repository https://gitbox.apache.org/repos/asf/groovy.git


The following commit(s) were added to refs/heads/GROOVY_5_0_X by this push:
     new fbbc7a2d50 GROOVY-11776: set method target to trait helper for static 
dispatch
fbbc7a2d50 is described below

commit fbbc7a2d50d025deea6fff41b31d6d0c1f5f7e88
Author: Eric Milles <[email protected]>
AuthorDate: Thu Oct 9 14:28:57 2025 -0500

    GROOVY-11776: set method target to trait helper for static dispatch
---
 .../groovy/transform/trait/TraitComposer.java      |  1 +
 .../groovy/transform/traitx/Groovy11776.groovy     | 96 ++++++++++++++++++++++
 2 files changed, 97 insertions(+)

diff --git 
a/src/main/java/org/codehaus/groovy/transform/trait/TraitComposer.java 
b/src/main/java/org/codehaus/groovy/transform/trait/TraitComposer.java
index a8e6f10769..ebca5894fe 100644
--- a/src/main/java/org/codehaus/groovy/transform/trait/TraitComposer.java
+++ b/src/main/java/org/codehaus/groovy/transform/trait/TraitComposer.java
@@ -327,6 +327,7 @@ public abstract class TraitComposer {
                 helperMethodArgList
         );
         mce.setImplicitThis(false);
+        mce.setMethodTarget(helperMethod); // GROOVY-11776
 
         ClassNode[] exceptionTypes = 
GenericsUtils.correctToGenericsSpecRecurse(genericsSpec, 
copyExceptions(helperMethod.getExceptions()));
         ClassNode returnType = 
GenericsUtils.correctToGenericsSpecRecurse(genericsSpec, 
helperMethod.getReturnType());
diff --git 
a/src/test/groovy/org/codehaus/groovy/transform/traitx/Groovy11776.groovy 
b/src/test/groovy/org/codehaus/groovy/transform/traitx/Groovy11776.groovy
new file mode 100644
index 0000000000..f26bbdc7b1
--- /dev/null
+++ b/src/test/groovy/org/codehaus/groovy/transform/traitx/Groovy11776.groovy
@@ -0,0 +1,96 @@
+/*
+ *  Licensed to the Apache Software Foundation (ASF) under one
+ *  or more contributor license agreements.  See the NOTICE file
+ *  distributed with this work for additional information
+ *  regarding copyright ownership.  The ASF licenses this file
+ *  to you under the Apache License, Version 2.0 (the
+ *  "License"); you may not use this file except in compliance
+ *  with the License.  You may obtain a copy of the License at
+ *
+ *    http://www.apache.org/licenses/LICENSE-2.0
+ *
+ *  Unless required by applicable law or agreed to in writing,
+ *  software distributed under the License is distributed on an
+ *  "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
+ *  KIND, either express or implied.  See the License for the
+ *  specific language governing permissions and limitations
+ *  under the License.
+ */
+package org.codehaus.groovy.transform.traitx
+
+import org.codehaus.groovy.control.CompilationUnit
+import org.codehaus.groovy.control.CompilerConfiguration
+import org.junit.Test
+import org.objectweb.asm.ClassReader
+import org.objectweb.asm.util.CheckClassAdapter
+
+final class Groovy11776 {
+
+    @Test
+    void testTraitMethodOverloads() {
+        File sourceDir = File.createTempDir()
+        File targetDir = File.createTempDir()
+        try {
+            def a = new File(sourceDir, 'A.groovy')
+            a.write '''
+                trait A {
+                    def foo(Object o) {
+                        return 'foo(o)'
+                    }
+                    def foo(Map<String,Object> m) {
+                        return 'foo(m)'
+                    }
+                }
+            '''
+            def b = new File(sourceDir, 'B.groovy')
+            b.write '''
+                class B implements A {
+                    def bar(Object o) {
+                        return 'bar(o)'
+                    }
+                    def bar(Map<String,Object> m) {
+                        return 'bar(m)'
+                    }
+                }
+            '''
+            def c = new File(sourceDir, 'C.groovy')
+            c.write '''
+                new B().with {
+                    assert bar( (Object) null) == 'bar(o)'
+                    assert bar(null as Object) == 'bar(o)'
+                    assert foo( (Object) null) == 'foo(o)'
+                    assert foo(null as Object) == 'foo(o)'
+                }
+                (new Object() as A).with {
+                    assert foo( (Object) null) == 'foo(o)'
+                    assert foo(null as Object) == 'foo(o)'
+                }
+            '''
+
+            def config = new CompilerConfiguration(targetDirectory: targetDir)
+            def loader = new GroovyClassLoader(this.class.classLoader)
+            def unit = new CompilationUnit(config, null, loader)
+            unit.addSources(a, b, c)
+            unit.compile()
+
+            loader.addClasspath(targetDir.absolutePath)
+            loader.loadClass('C', true).main()
+
+            // produce bytecode for class B
+            def writer = new StringWriter()
+            def reader = new ClassReader(unit.classes.find{ it.name == 'B' 
}.bytes)
+            CheckClassAdapter.verify(reader, loader, true, new 
PrintWriter(writer))
+
+            def string = writer.toString().with {
+                int start = indexOf('foo(Ljava/lang/Object;)')
+                int until = indexOf('ARETURN', start) + 8
+                substring(start, until)
+            }
+            assert !string.contains('INVOKEDYNAMIC invoke(Ljava/lang/Class;')
+            assert  string.contains('INVOKESTATIC A$Trait$Helper.foo')
+        } finally {
+            sourceDir.deleteDir()
+            targetDir.deleteDir()
+        }
+    }
+}

Reply via email to