From 57ac810951a4309bfddddcc9892531366d8769c3 Mon Sep 17 00:00:00 2001 From: Stuart Tettemer Date: Tue, 4 Jan 2022 15:14:51 -0600 Subject: [PATCH] Script: track this pointer capture from blocks within lambdas (#82228) * Script: track this pointer capture from blocks within lambdas If a lambda contains a block that calls into a user function, the painless compiler was not tracking that the lambda needs to capture the `this` pointer. Scripts that attempted to use a lambda that calls a user function from inside a block would trigger an `illegal_state_exception`: > no 'this' pointer within static method `BlockScope` now forwards `setUsesInstanceMethod` calls to it's parent, which may be a `LambdaScope`. `LambdaScope` will also forward `setUsesInstanceMethod` to it's parent after setting it's own `usesInstanceMethod` flag, propagating `this` pointer capture in the case of nested lambdas. Fixes: #82224 --- docs/changelog/82228.yaml | 5 +++ .../painless/symbol/SemanticScope.java | 12 ++++++- .../painless/UserFunctionTests.java | 36 +++++++++++++++++++ 3 files changed, 52 insertions(+), 1 deletion(-) create mode 100644 docs/changelog/82228.yaml diff --git a/docs/changelog/82228.yaml b/docs/changelog/82228.yaml new file mode 100644 index 0000000000000..5bf707b6dddde --- /dev/null +++ b/docs/changelog/82228.yaml @@ -0,0 +1,5 @@ +pr: 82228 +summary: "Script: track this pointer capture from blocks within lambdas" +area: Infra/Scripting +type: bug +issues: [] diff --git a/modules/lang-painless/src/main/java/org/elasticsearch/painless/symbol/SemanticScope.java b/modules/lang-painless/src/main/java/org/elasticsearch/painless/symbol/SemanticScope.java index 4f05606dffa9d..3e3c4b9e8ad46 100644 --- a/modules/lang-painless/src/main/java/org/elasticsearch/painless/symbol/SemanticScope.java +++ b/modules/lang-painless/src/main/java/org/elasticsearch/painless/symbol/SemanticScope.java @@ -198,6 +198,9 @@ public void setUsesInstanceMethod() { return; } usesInstanceMethod = true; + if (parent != null) { + parent.setUsesInstanceMethod(); + } } @Override @@ -261,6 +264,12 @@ public Class getReturnType() { public String getReturnCanonicalTypeName() { return parent.getReturnCanonicalTypeName(); } + + @Override + // If the parent scope is a lambda, we want to track this usage, so forward call to parent. + public void setUsesInstanceMethod() { + parent.setUsesInstanceMethod(); + } } /** @@ -356,7 +365,8 @@ public Variable defineVariable(Location location, Class type, String name, bo public abstract Variable getVariable(Location location, String name); - // We only want to track instance method use inside of lambdas for "this" injection. It's a noop for other scopes. + // We only want to track instance method use inside of lambdas (and blocks inside lambdas) for "this" injection. + // It's a noop for other scopes. public void setUsesInstanceMethod() {} public boolean usesInstanceMethod() { diff --git a/modules/lang-painless/src/test/java/org/elasticsearch/painless/UserFunctionTests.java b/modules/lang-painless/src/test/java/org/elasticsearch/painless/UserFunctionTests.java index 0bd074eb698ac..cb84750f2d096 100644 --- a/modules/lang-painless/src/test/java/org/elasticsearch/painless/UserFunctionTests.java +++ b/modules/lang-painless/src/test/java/org/elasticsearch/painless/UserFunctionTests.java @@ -163,4 +163,40 @@ public void testLambdaCapture() { assertEquals(List.of(100, 1, -100), exec(source, Map.of("a", 1), false)); assertBytecodeExists(source, "public static synthetic lambda$synthetic$0(ILjava/lang/Object;Ljava/lang/Object;)I"); } + + public void testCallUserMethodFromStatementWithinLambda() { + String source = "" + + "int test1() { return 1; }" + + "void test(Map params) { " + + " int i = 0;" + + " params.forEach(" + + " (k, v) -> { if (i == 0) { test1() } else { 20 } }" + + " );" + + "}" + + "test(params)"; + assertNull(exec(source, Map.of("a", 5), false)); + assertBytecodeExists(source, "public synthetic lambda$synthetic$0(ILjava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;"); + } + + public void testCallUserMethodFromStatementWithinNestedLambda() { + String source = "" + + "int test1() { return 1; }" + + "void test(Map params) { " + + " int i = 0;" + + " int j = 5;" + + " params.replaceAll( " + + " (n, m) -> {" + + " m.forEach(" + + " (k, v) -> { if (i == 0) { test1() } else { 20 } }" + + " );" + + " return ['aaa': j];" + + " }" + + " );" + + "}" + + "Map myParams = new HashMap(params);" + + "test(myParams);" + + "myParams['a']['aaa']"; + assertEquals(5, exec(source, Map.of("a", Map.of("b", 1)), false)); + assertBytecodeExists(source, "public synthetic lambda$synthetic$1(IILjava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;"); + } }