-
Notifications
You must be signed in to change notification settings - Fork 17.8k
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
time: add ExternalNow, etc for external time and timers #36141
Comments
Making time sense a configuration option in From reading the discussion at #35482, it seems to me like the only cases where real time is mandatory are cases where the program interacts with the outside world, and furthermore that all cases where the program interacts with the outside world really want real time, even if they currently don't (or previously didn't) use it due to operating system behavior. That implies that the suggestion in #35482 (comment) of As a supporting anecdote: I have an IRC bot that I run using Go 1.13.0 on my local Windows machine, which I also put to sleep from time to time. Sometimes-but-not-always after waking my computer, the program will actively spin for 300 seconds while waiting for the network deadlines to "actually" happen. I haven't tried it yet with Go 1.13.3+, but from what I understand, that behavior would be fixed there – the deadlines would always occur immediately after the computer wakes. This indicates to me that real time is the correct time for all network protocols, not just for WireGuard. |
Yes, probably more so than not. Brad wanted "real time" on a networking CL recently too and settled for the unreliable "wall time" instead. |
We're expecting a separate proposal from the Go team along the lines described by @aclements in #35482 (comment). I posted some followup questions to it in #35482 (comment) See also #24595, #35012, #29485 (comment) @bradfitz mentions his wall (i.e. pseudo-real) timers in #35482 (comment) It would probably help if Go allowed us to set a suspend/resume handler. Note that network protocol use between apps on the same machine is also common, and would generally not want real-time timers. |
I agree with @zephyrtronium that a global setting as proposed in possibilities d and e is a non-starter. Large programs include many packages, and those packages may legitimately have different requirements. Possibility a is essentially #24595 as a choice made for all systems where possible. The Linux kernel attempted to make this choice, and reverted it, as discussed at https://www.spinics.net/lists/linux-tip-commits/msg43709.html. Watchdog timers failing on resume are a particular issue. So are timers for single user games. Very roughly, timers associated with behavior of other programs on the current system, or with behavior of the user of the system, seem likely to want "program time". Timers associated with behavior of other systems on the network seem likely to want "real time". Given that Go has historically used "program time" on all systems other than Windows, shifting all systems to "real time" seems likely to hurt more existing code than it helps. The distinction between possibilities b and c is subtle, and it's not yet clear to me that it matters. If we have a way to detect that a system was suspended, which is the only case where "program time" differs from "real time", then we can always implement timers that use "real time". |
From the API perspective I think the relevant functions and methods are:
If there are other relevant functions/methods, please point them out. We also need to pay particular attention to
If we are going to make changes in this area, it would be nice to eliminate this inconsistency if possible. |
As others have said, one way that an API can permit programs to distinguish between "real time" and "program time" is to use the difference between We could add a new function A disadvantage of this approach is that |
Turning to implementation, I believe that we can straightforwardly implement both "real time" and "program time" timers if we have a way to receive a notification that the system has woken up from a suspension. Unfortunately, as far as I know, GNU/Linux provides no such notification capability. We can detect a suspension after the fact by comparing the results of The problem is that for any sort of timer we need to have a way to sleep until the timer is ready to fire. To support both timers that work in "program time" and timers that work in "real time" we need to have a way to sleep in "program time" and in "real time", or a way to detect that we've woken up from suspend and need to recalculate sleep times. As far as I know, the only way to sleep in "real time" on GNU/Linux is via I don't know what is available on other systems. |
systemd sees suspend/resume events. Go could provide an optional resume hook that pings a unix socket or other pollable widget. |
There's I'll do some research on Linux sleep resumption notification mechanisms. Tailing dmesg sounds pretty bad. Hooking into systemd or android power binder misery or similar sounds less bad but still bad. I wonder if there's some magic sysfs file we can epoll on. I'll look around.
Yea, I was afraid of that. I guess runtime has a lot of global nobs, but I understand that you don't want to add more, especially important ones.
Sounds like you don't like that either. I'm biased based on my own needs of course, but something to consider is the slight misunderstanding implied by @aclements's original nomenclature we've been using. "Program time" isn't actually the time spent by a program, since the scheduler implies it might not actually be running that whole time, or might be SIGSTOP'd or something. That means that ordinary programs may very well see large leaps in "program time", just as they will for "real time" in the case of resuming from suspend. With large leaps being a potential either way, what's the point of the distinction? Seems like programs are better off coding around "real time" assumptions since "program time" assumptions don't actually apply in the case of SIGSTOP or scheduler contention and such. Maybe the distinction between the two is only important for something like watchdog timers? That seems like more of the exception than the rule though. But, as said, I'm a bit biased here.
Do you mean to say that we could perhaps add the new API without needing a function to determine whether or not we have a detection mechanism (or without returning an error), because in the case where we don't have a detection mechanism, there is no S3 sleep anyway, so we might as well consider them the same? That would be fine with me, I suppose.
time.Ticker is another one.
Right now a time object has internally both wall time and program time. We could slow it down further and have it always fetch real time. And then introduce more comparison functions. But that seems tricky. Alternatively, we could change the construction functions, for NowReal() and NowProgram() or something, but then there are hairy issues on what to do when comparing unlike objects. Or maybe we'll just want to introduce a whole new object all together?
But actually, isn't the net case the one area where we actually do want to prefer "real time"? For example, see Brad's CLs where he tries to approximate "real time" by using "wall time" (via Round(0)). So maybe that's okay. |
On laptops, programs are rarely stopped, and if they are, it affects network peers on the same machine. Suspend/resume is frequent, and does not affect peers on the same machine. |
As far as I know the runtime package does not have any knobs that change the meaning of any APIs. |
I did forget about |
In general, yes, but not always. For processes communicating on the same system, "program time" may be expected. We could go ahead and decide to change it anyhow, but it requires thought. |
It's true of course that "program time" (which I agree is a minomer) can see delays. But consider the case of an interactive game with no network connection that sets up dozens of timers for different times that cause events to occur in the game. If the user suspends a laptop while playing the game, those timers should also suspend; they shouldn't all fire when the laptop is resumed. |
Assuming the Windows runtime switches to program/monotonic time, it might help to provide a switch (e.g. env var) to return it to real/boot time. Conversely, it would help for 1.14 to have a switch to enable program time; the work for that is mostly done. |
Alright here's where we are with technical solutions for supporting "real time": Windows
Linux
macOS
I haven't looked into the BSDs yet. |
It might be essential to make real timers on Windows use a high-precision source; current resolution is 2ms, which isn't enough for some protocols, for example #29485 (comment). Instead of changing time.Sub() to compute real time, we could consider an API which provides any delta between program & real time, e.g. Re SetDeadline() computing a duration internally, I'd suggest this API, and deprecating Set*Deadline().
|
|
True; requiring a type-assert to call a new method. Maybe new fields in net.Dialer & net.ListenConfig should set the meaning of the Set*Deadline() time.Time argument. |
I would prefer a whole new API, in a separate package, 'time/realtime' that has all the functionality of the time package but that explicitly uses real time clocks on all platform. The current And @ianlancetaylor, as an aside, thanks of thinking about game programming using Go. I definitely appreciate that core Go developers are starting to take non-server applications for Go seriously. As you say, for single player games, timers of game events really need to be suspended when the program is suspended. As an aside, over 20 years ago, I learned programming because I wanted to make games. Likewise I want to teach my daughter how to program Go by letting her make games. If we want Go to become more popular, then we have to make it even easier to use for game programming as well. |
@ianlancetaylor is the Go team working on a concrete proposal? If so, is there an ETA? Presumably we'd want to see a draft implementation early in 1.15 cycle... |
There is no ETA. |
@networkimprov Perhaps this doesn't need to be in the Go standard library? I think it would not be that hard to write an external realtime module that uses real time timers with high precision. |
I looked at that discussion, and after that I'm not very happy with the current state of things. What I would like is consistency between platforms. The WireGuard project should not have to use a patched runtime on Linux, but to me it looks like they got a patch into the Windows Go runtime that worsened the inconsistency overall. As I said above, I would solve the problem by making 'time' use program time/wall time, and have a 'time/realtime' package for real time/monotonic clocks. The former can be done relatively easily in Go itself, the latter could be implemented as a third party library at first. |
I’m not sure about this case specifically, but sometimes crashing is much worse for users than slightly wrong results. |
In this case, you'd see the panic when testing; it wouldn't be intermittent. |
No. |
No change in consensus, so accepted. 🎉 |
That's a surprise. I thought that failing fast and loud is preferable to silently accumulating error. Especially considering that panics are not crashes and can be recovered from. |
If you make it fail-able then you make it harder to use, as devs need to consider handling that failure instead of just having a guaranteed successful function |
Hunter, what you suggest is akin to saying that out-of-bounds writes like |
I don’t think it’s black and white. I do think it’s undeniable that can-panic is part of an API and makes that API harder to use. It can still sometimes be the right decision, such as in the out of bounds array access case. |
Update May 5 2021: The current proposed API is in #36141 (comment). - rsc
Vocabulary:
(These vocabulary terms can be nitpicked - maybe program time should be cpu time or something - but we've been using them prior in discussion, so let's continue to use them so as not to introduce confusion.)
Proposal:
Motivation:
Landscape:
Possibilities:
a. Make the existing
time.
andtime.Timer.
functions use "real time" exclusively, when possible. Introduce a functionruntime.RealtimeTimers() bool
to indicate whether Go successfully enabled "real time" timers rather than "program time" timers, the fallback.b. Introduce additional duplicated functions to
time.
andtime.Timer.
that use "real time" rather than "program time". Introduce a functiontime.RealtimeTimersAreRealTime() bool
to indicate whether Go successfully enabled "real time" timers on this new set of functions, or if the new set of functions behave identically to the old.c. Introduce additional duplicated functions to
time.
andtime.Timer.
that use "real time" rather than "program time", and throw an error if "real time" capabilities aren't available, forcing users to introduce verbose fallback code if they only want to support "real time" opportunistically.d. Add a function
runtime.UseRealtimeTimers() error
that attempts to change the runtime to use "real time" timers everywhere, like (a).e. Add runtime function
runtime.UseRealtimeTimers(yes bool) error
that attempts to change the runtime to use "real time" or "program time" timers everywhere, like (a) but the ability to toggle. Add runtime functionruntime.RealtimeTimers() bool
to indicate the current state. The default start-state would be either OS-defined or "real time" or "program time", depending on what we decide.f. Other options?
My personal preference would be (a) or (e), but I'm open to discussion.
CC @ianlancetaylor @bradfitz @aclements @rsc
The text was updated successfully, but these errors were encountered: