Skip to content

Commit 5951b3f

Browse files
authored
Fix infinite loop when resolving method call with superclass cycle (#656)
1 parent dcbf6cb commit 5951b3f

File tree

2 files changed

+11
-1
lines changed

2 files changed

+11
-1
lines changed

php-frontend/src/main/java/org/sonar/php/tree/symbols/SymbolUsageVisitor.java

+4-1
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,9 @@
2020
package org.sonar.php.tree.symbols;
2121

2222
import java.util.ArrayDeque;
23+
import java.util.HashSet;
2324
import java.util.Optional;
25+
import java.util.Set;
2426
import org.sonar.php.symbols.ClassSymbol;
2527
import org.sonar.php.symbols.ClassSymbolIndex;
2628
import org.sonar.php.symbols.FunctionSymbol;
@@ -99,7 +101,8 @@ private void resolveMethodCall(FunctionCallTree tree, MemberAccessTree callee) {
99101
String methodName = ((NameIdentifierTree) callee.member()).text();
100102
MethodSymbol methodSymbol = receiverSymbol.getDeclaredMethod(methodName);
101103
Optional<ClassSymbol> superClass = receiverSymbol.superClass();
102-
while (superClass.isPresent() && methodSymbol.isUnknownSymbol()) {
104+
Set<ClassSymbol> processedClasses = new HashSet<>();
105+
while (superClass.isPresent() && methodSymbol.isUnknownSymbol() && processedClasses.add(superClass.get())) {
103106
methodSymbol = superClass.get().getDeclaredMethod(methodName);
104107
superClass = superClass.get().superClass();
105108
}

php-frontend/src/test/java/org/sonar/php/symbols/MethodCallResolutionTest.java

+7
Original file line numberDiff line numberDiff line change
@@ -50,6 +50,13 @@ public void resolve_this_with_inheritance() {
5050
"class B extends A { function f(){ $this->f(); } }")).isKnown("B::f");
5151
}
5252

53+
@Test
54+
public void resolve_with_superclass_cycle() {
55+
assertThat(callSymbol("<?php ",
56+
"class A extends B { }",
57+
"class B extends A { function g(){ $this->f(); } }")).isUnknown();
58+
}
59+
5360
@Test
5461
public void resolve_self_and_static() {
5562
assertThat(callSymbol("<?php class A { function f(){} function g(){ self::f(); } }")).isKnown("A::f");

0 commit comments

Comments
 (0)