locking macro creates monitor bytecode difficult to analyze in Graal native-image and ART runtime

Description

Android ART runs compile time verification on bytecode and fails on some usages of the locking macro. Example:

Android error:

Graal fails with this error:

Cause: The locking macro emits a try block. When not in tail position, try blocks get lifted into lambdas, and closed over variables (including, importantly, the lockee) become ‘fields’ of the lambda object. Thus in that situation the monitor enter/exit calls were being made on a field. Graal doesn’t trust that the value of that field will be the same in the enter and exit calls, so it balks that they might not be a balanced pair. The solution is to get the lockee onto the stack so Graal can understand that it is the same target. The problem has nothing to do with exception handling blocks or failure to emulate javac’s infinite loop trick.

Approach: We need to get the lockee on the stack, even when lifting occurs, and use the same stack variable in try/finally. Thus we need a let around try/finally. But lifting can still happen, so that let needs to be in what’s lifted. That’s the role of the outer try. The outer try will be lifted, and the let with it, the let will put the field on the stack, Graal sees enter/exit on the same stack value and its verifier is thus happy. The inner try will always be in tail position.

Patch: clj-1472-5.patch

Environment

Android ART runtime

Attachments

8

Activity

Show:

Daniel Compton March 31, 2020 at 1:11 AM

Alex Miller March 3, 2020 at 7:20 PM

Applied for 1.10.2

Alex Miller March 2, 2020 at 8:03 PM
Edited

I have verified that the new clj-1472-5.patch resolves the issue per instructions by Lee Read with Graal 19, existing Clojure tests all pass, core.memoize code is ok.

The latest Graal (20) works under Java 8, but fails with different errors under Java 11.

Alex Miller February 28, 2020 at 11:02 PM

Alex Miller February 28, 2020 at 4:46 PM

This code in core.memoize is broken with CLJ-1472-4 as the locking thunk is pulled out of the context where set! on mutable volatile deftype fields is allowed:

Completed

Details

Assignee

Reporter

Approval

Ok

Patch

Code

Priority

Affects versions

Fix versions

Created July 23, 2014 at 6:39 PM
Updated March 31, 2020 at 1:11 AM
Resolved March 3, 2020 at 7:20 PM