-
Notifications
You must be signed in to change notification settings - Fork 525
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
IOLocal
propagation for unsafe access
#3636
Changes from 4 commits
0b88c01
db743e2
716ef32
0a69caf
2775064
270764f
d55489d
2cf72a5
cb3859d
7dce01c
5e171ac
c2f312d
638930d
9174c6a
1987e3a
02a43a6
a7bf748
145fc0e
fa99a5c
6cad03c
bb5d4b1
7517755
8d8e004
3589db4
522677e
6cc4d38
ac88480
49e5c30
925f504
d63a6ff
d4549fb
2502045
535fc8a
d854799
f070552
2cf1d8a
0eec9dd
af84973
1adf368
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -1009,7 +1009,7 @@ sealed abstract class IO[+A] private () extends IOPlatform[A] { | |
implicit runtime: unsafe.IORuntime): IOFiber[A @uncheckedVariance] = { | ||
|
||
val fiber = new IOFiber[A]( | ||
Map.empty, | ||
if (IOFiberConstants.dumpLocals) unsafe.IOLocals.getState else Map.empty, | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. We can even go in the opposite direction for It's less clear if/how to do this for fibers started in a |
||
oc => | ||
oc.fold( | ||
{ | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -250,6 +250,10 @@ private final class IOFiber[A]( | |
pushTracingEvent(cur.event) | ||
} | ||
|
||
if (dumpLocals) { | ||
IOLocals.setState(localState) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Dumb question: can't we simply do this when we get scheduled on a thread? We know when we're on a thread and we know when we get off of it, so can't we simply set and clear the state respectively at those points? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. No we can't, unless we unify how the state is represented. Currently it's a var to an immutable map in the fiber and also in the thread. While the fiber is running its copy of the var may be updated effectually in the runloop so the thread-local copy would need to be kept in sync with that. Or we could drive all updates through the thread-local copy of the var, but then there would be a penalty for accessing it esp. if we are not running on a worker thread. Putting aside technical issues, nobody should be unsafely messing about with There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. What we can do is set the current fiber in a thread local every time we get scheduled on a thread. Then the unsafe Note this would leave the fiber's |
||
} | ||
armanbilge marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
var error: Throwable = null | ||
val r = | ||
try cur.thunk() | ||
|
@@ -260,6 +264,10 @@ private final class IOFiber[A]( | |
onFatalFailure(t) | ||
} | ||
|
||
if (dumpLocals) { | ||
localState = IOLocals.getAndClearState() | ||
} | ||
|
||
val next = | ||
if (error == null) succeeded(r, 0) | ||
else failed(error, 0) | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,68 @@ | ||
package cats.effect | ||
package unsafe | ||
|
||
// TODO handle defaults and lenses. all do-able, just needs refactoring ... | ||
object IOLocals { | ||
djspiewak marked this conversation as resolved.
Show resolved
Hide resolved
djspiewak marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
def get[A](iol: IOLocal[A]): A = { | ||
val thread = Thread.currentThread() | ||
val state = | ||
if (thread.isInstanceOf[WorkerThread]) | ||
thread.asInstanceOf[WorkerThread].ioLocalState | ||
else | ||
threadLocal.get | ||
state(iol).asInstanceOf[A] | ||
} | ||
armanbilge marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
def set[A](iol: IOLocal[A], value: A): Unit = { | ||
val thread = Thread.currentThread() | ||
if (thread.isInstanceOf[WorkerThread]) | ||
thread.asInstanceOf[WorkerThread].ioLocalState += (iol -> value) | ||
else | ||
threadLocal.set(threadLocal.get() + (iol -> value)) | ||
armanbilge marked this conversation as resolved.
Show resolved
Hide resolved
|
||
} | ||
|
||
def reset[A](iol: IOLocal[A]): Unit = { | ||
val thread = Thread.currentThread() | ||
if (thread.isInstanceOf[WorkerThread]) | ||
thread.asInstanceOf[WorkerThread].ioLocalState -= iol | ||
else | ||
threadLocal.set(threadLocal.get() - iol) | ||
} | ||
|
||
// TODO other ops from IOLocal | ||
|
||
private[this] val threadLocal = new ThreadLocal[IOLocalState] { | ||
override def initialValue() = IOLocalState.empty | ||
} | ||
|
||
private[effect] def getState = { | ||
val thread = Thread.currentThread() | ||
if (thread.isInstanceOf[WorkerThread]) | ||
thread.asInstanceOf[WorkerThread].ioLocalState | ||
else | ||
threadLocal.get() | ||
} | ||
|
||
private[effect] def setState(state: IOLocalState) = { | ||
val thread = Thread.currentThread() | ||
if (thread.isInstanceOf[WorkerThread]) | ||
thread.asInstanceOf[WorkerThread].ioLocalState = state | ||
else | ||
threadLocal.set(state) | ||
} | ||
|
||
private[effect] def getAndClearState() = { | ||
val thread = Thread.currentThread() | ||
if (thread.isInstanceOf[WorkerThread]) { | ||
val worker = thread.asInstanceOf[WorkerThread] | ||
val state = worker.ioLocalState | ||
worker.ioLocalState = IOLocalState.empty | ||
state | ||
} else { | ||
val state = threadLocal.get() | ||
threadLocal.set(IOLocalState.empty) | ||
state | ||
} | ||
} | ||
} |
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.
Bikesheddable configuration for opting-in. So the rest of us don't have to pay the penalty 😇
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.
Is this specifically "tracing", even if that's the most obvious use case?
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 woops, this was a very lazy copy-pasta. I copied it from the system properties we use to configure fiber tracing. We should rename it anyway,
dumpLocals
is not quite right I think 😅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.
What about
cats.effect.localContextPropagation
similar to Monix'smonix.environment.localContextPropagation
?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.
Thanks, I liked that! I went with
cats.effect.ioLocalPropagation
.