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

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


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

commit da9a751745fe90e4e6aaddb18aba05aa3ac2b627
Author: Eric Milles <[email protected]>
AuthorDate: Fri Oct 10 10:31:40 2025 -0500

    GROOVY-11776: set method target to trait helper for static dispatch
    
    4_0_X backport
---
 .../transform/trait/TraitASTTransformation.java    | 65 +++++++++------
 .../groovy/transform/trait/TraitComposer.java      |  1 +
 .../groovy/transform/traitx/Groovy11776.groovy     | 96 ++++++++++++++++++++++
 3 files changed, 136 insertions(+), 26 deletions(-)

diff --git 
a/src/main/java/org/codehaus/groovy/transform/trait/TraitASTTransformation.java 
b/src/main/java/org/codehaus/groovy/transform/trait/TraitASTTransformation.java
index ac06a928fb..554d0f4f58 100644
--- 
a/src/main/java/org/codehaus/groovy/transform/trait/TraitASTTransformation.java
+++ 
b/src/main/java/org/codehaus/groovy/transform/trait/TraitASTTransformation.java
@@ -87,6 +87,7 @@ import static org.objectweb.asm.Opcodes.ACC_ABSTRACT;
 import static org.objectweb.asm.Opcodes.ACC_FINAL;
 import static org.objectweb.asm.Opcodes.ACC_INTERFACE;
 import static org.objectweb.asm.Opcodes.ACC_PRIVATE;
+import static org.objectweb.asm.Opcodes.ACC_PROTECTED;
 import static org.objectweb.asm.Opcodes.ACC_PUBLIC;
 import static org.objectweb.asm.Opcodes.ACC_STATIC;
 import static org.objectweb.asm.Opcodes.ACC_SYNTHETIC;
@@ -569,43 +570,55 @@ public class TraitASTTransformation extends 
AbstractASTTransformation implements
     }
 
     private MethodNode processMethod(final ClassNode traitClass, final 
ClassNode traitHelperClass, final MethodNode methodNode, final ClassNode 
fieldHelper, final Collection<String> knownFields) {
-        Parameter[] initialParams = methodNode.getParameters();
-        Parameter[] newParams = new Parameter[initialParams.length + 1];
-        newParams[0] = createSelfParameter(traitClass, methodNode.isStatic());
-        System.arraycopy(initialParams, 0, newParams, 1, initialParams.length);
-        final int mod = methodNode.isPrivate() ? ACC_PRIVATE : ACC_PUBLIC | 
(methodNode.isFinal() ? ACC_FINAL : 0);
+        boolean isAbstractMethod = methodNode.isAbstract();
+        boolean isPrivateMethod = methodNode.isPrivate();
+        boolean isStaticMethod = methodNode.isStatic();
+
+        int modifiers = ACC_PUBLIC;
+        if (isAbstractMethod) {
+            modifiers |= ACC_ABSTRACT;
+        } else {
+            // public or private
+            if (isPrivateMethod) {
+                modifiers ^= ACC_PUBLIC | (!isStaticMethod ? ACC_PRIVATE : 
ACC_PROTECTED); // GROOVY-11776
+            }
+            // static or final (maybe)
+            if (methodNode.isFinal() && !isStaticMethod) {
+                modifiers |= ACC_FINAL;
+            }
+            modifiers |= ACC_STATIC;
+        }
+
+        Parameter[] methodParams = methodNode.getParameters();
+        Parameter[] helperParams = new Parameter[methodParams.length + 1];
+        helperParams[0] = createSelfParameter(traitClass, isStaticMethod);
+        System.arraycopy(methodParams, 0, helperParams, 1, 
methodParams.length);
+
         MethodNode mNode = new MethodNode(
                 methodNode.getName(),
-                mod | ACC_STATIC,
+                modifiers,
                 methodNode.getReturnType(),
-                newParams,
+                helperParams,
                 methodNode.getExceptions(),
-                processBody(varX(newParams[0]), methodNode.getCode(), 
traitClass, traitHelperClass, fieldHelper, knownFields)
+                processBody(varX(helperParams[0]), methodNode.getCode(), 
traitClass, traitHelperClass, fieldHelper, knownFields)
         );
-        mNode.setSourcePosition(methodNode);
-        mNode.addAnnotations(filterAnnotations(methodNode.getAnnotations()));
+        for (AnnotationNode annotation : methodNode.getAnnotations()) {
+            if (!annotation.getClassNode().equals(OVERRIDE_TYPE)) {
+                mNode.addAnnotation(annotation);
+            }
+        }
         mNode.setGenericsTypes(methodNode.getGenericsTypes());
-        if (methodNode.isAbstract()) {
-            mNode.setModifiers(ACC_PUBLIC | ACC_ABSTRACT);
-        } else {
+        mNode.setSourcePosition(methodNode);
+
+        if (!isAbstractMethod) {
             methodNode.addAnnotation(new 
AnnotationNode(Traits.IMPLEMENTED_CLASSNODE));
         }
-        methodNode.setCode(null);
-
-        if (!methodNode.isPrivate() && !methodNode.isStatic()) {
+        if (!isPrivateMethod && !isStaticMethod) {
             methodNode.setModifiers(ACC_PUBLIC | ACC_ABSTRACT);
         }
-        return mNode;
-    }
+        methodNode.setCode(null);
 
-    private static List<AnnotationNode> filterAnnotations(final 
List<AnnotationNode> annotations) {
-        List<AnnotationNode> result = new ArrayList<>(annotations.size());
-        for (AnnotationNode annotation : annotations) {
-            if (!annotation.getClassNode().equals(OVERRIDE_TYPE)) {
-                result.add(annotation);
-            }
-        }
-        return result;
+        return mNode;
     }
 
     private static Parameter createSelfParameter(final ClassNode traitClass, 
boolean isStatic) {
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 f569980f4b..e5b531a5eb 100644
--- a/src/main/java/org/codehaus/groovy/transform/trait/TraitComposer.java
+++ b/src/main/java/org/codehaus/groovy/transform/trait/TraitComposer.java
@@ -325,6 +325,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