Virtual thread memory leak because TrackingRootContainer keeps threads
Brian S O'Neill
bronee at gmail.com
Mon Jul 1 23:59:01 UTC 2024
I was confused when I first discovered that virtual threads could be
GC'd when blocked with an infinite timeout. This was before the
trackAllThreads option was enabled by default.
If the virtual thread is blocked with an effectively infinite timeout
(100 years), it won't be GC'd even when trackAllThreads is false. For
consistency, I think it makes sense for virtual threads to never
actually be GC'd except until after they exit.
If you depended on virtual threads being GC'd when blocked with a real
infinite timeout, then switching to platform threads would introduce a
memory leak that didn't exist before. Ideally, the two thread types
should be interchangeable.
On 2024-07-01 12:08 PM, Michal Domagala wrote:
> According to https://openjdk.org/jeps/444 <https://openjdk.org/jeps/444>,
>
> "Unlike platform thread stacks, virtual thread stacks are not GC roots."
>
> But jdk.internal.vm.ThreadContainers.RootContainer.TrackingRootContainer#VTHREADS keeps (hard) references to each virtual thread. Effectively, virtual threads are GC roots.
>
> I described full example here:
> https://stackoverflow.com/questions/78596905/why-virtual-thread-is-not-garbage-collected-when-not-reachable <https://stackoverflow.com/questions/78596905/why-virtual-thread-is-not-garbage-collected-when-not-reachable>
>
> The problem was not visible since
> https://bugs.openjdk.org/browse/JDK-8309406
> <https://bugs.openjdk.org/browse/JDK-8309406>, because before the
> change, VTHREADS "keep" was not enabled and i guess nobody cares memory
> leak.
>
> But after the change every JVM is affected.
>
> A workaround for memory leak is set jdk.trackAllThreads=false
>
> Best regards
> Michal Domagala
>
More information about the loom-dev
mailing list