I have stumbled somewhat happily into Constable
and the like in JDK 15. I mostly understand.
After frolicking through all the compiler theory and even understanding a little bit of it, I find I still have this question: Who calls a Constable
's describeConstable()
method, and when? Brian's presentation seemed to imply it is somehow accessed at compile time. Being naïve about such things, I was expecting it to show up in the usage page under jdk.compiler
or something. Instead, the only consumption seems to be in the jdk.incubator.foreign
package. (Obviously I understand it may be used by some private machinery somewhere that isn't exposed by the usage page; hence my question.)
I placed a Thread.dumpStack()
in a describeConstable()
implementation of a dumb class that implements Constable
and that returns Optional.ofNullable(null)
just to see what would happen and…nothing happened at compile- or runtime.
(I do know that until JDK 12 if you wanted to write dynamic constants you had to use ASM or ByteBuddy or something similar. To my naïve eyes it looks, though, like Constable
is there to allow your user class to "plug into" the Java compiler and allow it to do the constant writing for you. I am also aware that the classes and methods in java.lang.constant
are primarily intended for compiler writers, but Constable
seemed to me to be a bit of an exception. Finally, I obviously understand that I can call this method any time I wish, but that's clearly not what it's intended for.)
EDIT: Thanks (very much) to some of the extremely helpful and patient answers and comments below, I think I'm starting to get it (I'm not a compiler guy which by this point should be quite obvious). While I understand that once an instance of X implements Constable
exists then the ContantDesc
it returns from its describeConstable()
must be made (itself) of other constant descriptors, and while I understand that "constant factories" (such as ClassDesc#of()
and so on) may be called at compile time and obviously must accept only other constants as any arguments they might require, I'm still not clear on how an arbitrary X implements Constable
is instantiated during compilation in the first place while…it is being compiled (!) such that describeConstable()
can be called on it at compile time.
Please kindly bear in mind the answer to this question may be something rudimentary that I'm missing about compilers in general, or the sorts of hijinks they get up to during static analysis. I just see an instance method (describeConstable()
) that needs to be invoked on an instance of an object (X implements Constable
) and in order to have an instance of an object someone has to call its constructor. It's unclear to me how the Java compiler could know how to construct my X implements Constable
with its arbitrary, possibly multi-argument constructor so that it could then call describeConstable()
on it.