-
-
Notifications
You must be signed in to change notification settings - Fork 368
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Overhaul worker concurrency helpers with mill.api.CacheFactory
#4001
Conversation
mill.api.CacheFactory
mill.api.CacheFactory
CC @jodersky @alexarchambault if you guys want to take a look |
@@ -30,20 +36,25 @@ class CompressWorker(dest: os.Path) { | |||
val cache = collection.mutable.Map.empty[Int, Array[Byte]] | |||
def compress(name: String, bytes: Array[Byte]): Array[Byte] = { | |||
val hash = Arrays.hashCode(bytes) | |||
if (!cache.contains(hash)) { | |||
if (!synchronized(cache.contains(hash))) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Naive 'user level' complaint:
I'm not a fan the 'synchronized' in an example. Its so easy to get wrong and I'm always having a hard time to convince myself that it is correct.
What I would love instead is having the concurrent map operations on it:
cache.computeIfAbsent(key, (key)=>createTheExpensiveThing())
Note again: This is from a doc/end user perspective and not the internal use.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Oh...this is a negative example / problem statement =). So,yes, then it makes sense.
Fixes #3730
The old
FixSizedCache
/KeyedLockedCache
/RandomBoundedCache
were all a bit of a mess, and none of them provided one crucial component to long-lived cache entries: the ability to tear them down after you are done with them. This is necessary for a lot of long-lived components you may want to cache: classloaders, subprocesses, open file handles, etc. all need to be torn down after use. Also, their implementation was weird, probably overly convoluted, and not used consistently throughout MillThis PR introduces a new
CachedFactory
class meant to replace the old ones. I use it in four places (ZincWorker Scala and Java compilers, ScalaJsWorker, ScalanativeWorker), and added docs so users can discover and use it too. It provides a hook fordef teardown
, which should hopefully fix some of the classloader leaks I've seen in our prior implementations (these normally manifest as CODE CACHE FULL, DISABLING JIT COMPILER warnings when doing large builds). The last previously-problematic worker defsiteVisualizeModule.worker
cannot be changed without breaking bincompat so I left it in place for nowCachedFactory
also differs from the original implementation is that the cache contains the logic fordef setup
, rather than relying on the callsite of.withCachedValue
to provide the setup and key together. This should be easier to understand and less likely to be mis-used, as the mapping fromkey => value
is kept in one place, whereas previously everywithCachedValue
could have a different mapping.Hopefully this new
CachedFactory
helper class can form the basis of how people define concurrency-friendly stateful workers going forwardCovered by some unit tests and example tests, as well as usage in the main
ZincWorker
/ScalaJSWorker
/ScalaNativeWorker
code paths