You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
This is a continuation of #62654, in particular its item 6, described in there as:
(Optional) Add (*Process).Handle() uintptr method to return process handle on Windows and pidfd on Linux. This might be useful for low-level operations that require handle/pidfd (e.g. pidfd_getfd on Linux), or to ensure that pidfd (rather than pid) is being used for kill/wait.
A similar thing was proposed earlier here by @prattmic:
A new os.Process.Fd() could return the pid FD for additional direct use.
Proposal
Add a new Handle method for os.Process, which returns a process handle, if available, and a boolean flag telling if a handle is valid. On Linux, handle is a file descriptor of referring to the process (a pidfd). On Windows, handle is a handle to the process.
func (p*Process) Handle() (uintptr, bool)
Use cases
1. Check if pidfd is being used on Linux.
Since Go 1.23, pidfd is used for os.Process-related operations instead of pid, if supported by the Linux kernel. This includes os.StartProcess and os.FindProcess (they obtain a pidfd), as well as (*Process).Wait, (*Process).Signal, and (*Process).Kill (they use pidfd). The main benefit of pidfd in the use cases above is a guarantee we're referring to the same process (i.e. there's no pid reuse issue).
However, since this is done in a fully transparent way, there is no way for a user to know if pidfd is being used or not. Some programs implement some protection against pid reuse (for example, runc and cri-o obtain and check process start time from /proc/<pid>/stat). They can benefit from being able to know if Go is using pidfd internally.
Another example is containerd which relies on Go 1.23 using pidfd internally, but since there's no way to check they had to recreate all the functionality checking for pidfd support here (which is still not 100% correct since the checks are slightly different from those in Go's checkPidfd, and Go checks may change over time ). Cc @fuweid.
With the proposed interface, a user can easily check if pidfd is being used:
p, err:=os.FindProcess()
...if_, ok:=p.Handle(); ok {
// pidfd is used internally by os.Process methods.
}
2. Obtain a pidfd for additional direct use.
Aside from use cases already covered by existing os.Process methods, pidfd can also be used to:
obtain a duplicate of a file descriptor of another process (pidfd_getfd(2));
select/poll/epoll on a pidfd to know when a process is terminated;
move into one or more of the same namespaces as the process referred to by the file descriptor (setns(2)).
Other use cases may emerge in the future.
Currently, the only way to obtain a pidfd on Linux is to execute a new process (via os.StartProcess or os/exec) with process' Attr.SysAttr.PidFD field set. This works if we're starting the process, but not in any other case (someone else starts a process for us, or it is already running).
Questions
1. What are (could be) the additional direct use cases of Windows process handle?
A few are listed here. Apparently some git grep (GetPriorityClass, SetPriorityClass, AssignProcessToJobObject) are implemented in golang.org/x/sys/windows.
2. Should a duplicate of a handle be returned, or the original handle?
Return the original one, ensuring that Handle documentation describes when pidfd may become invalid.
Arguments against duplicated handle:
a duplicated pidfd makes the "check if pidfd is being used" use case above more complicated as the user will need to close the returned pidfd;
a user can always use dupfd if/when needed;
returned handle won't leak as it is still part of os.Process struct, which have a Release method and a proper finalizer;
os.File.Fd returns the original underlying fd, pidfd is similar.
3. Should Handle return *os.File rather than uintptr?
Raw handle makes more sense in this case, and the finalizer set by NewFile does not make sense if the original handle is returned. Also, this won't work for Windows handle is not a file.
4. Should this be Linux-specific?
Probably not. Since we have a boolean flag returned, we can implement it for all platforms and return 0, false for those that do not support process handle (other than Windows and Linux).
The text was updated successfully, but these errors were encountered:
Proposal Details
This is a continuation of #62654, in particular its item 6, described in there as:
A similar thing was proposed earlier here by @prattmic:
Proposal
Add a new
Handle
method foros.Process
, which returns a process handle, if available, and a boolean flag telling if a handle is valid. On Linux, handle is a file descriptor of referring to the process (a pidfd). On Windows, handle is a handle to the process.Use cases
1. Check if pidfd is being used on Linux.
Since Go 1.23, pidfd is used for os.Process-related operations instead of pid, if supported by the Linux kernel. This includes
os.StartProcess
andos.FindProcess
(they obtain a pidfd), as well as(*Process).Wait
,(*Process).Signal
, and(*Process).Kill
(they use pidfd). The main benefit of pidfd in the use cases above is a guarantee we're referring to the same process (i.e. there's no pid reuse issue).However, since this is done in a fully transparent way, there is no way for a user to know if pidfd is being used or not. Some programs implement some protection against pid reuse (for example,
runc
andcri-o
obtain and check process start time from/proc/<pid>/stat
). They can benefit from being able to know if Go is using pidfd internally.Another example is containerd which relies on Go 1.23 using pidfd internally, but since there's no way to check they had to recreate all the functionality checking for pidfd support here (which is still not 100% correct since the checks are slightly different from those in Go's checkPidfd, and Go checks may change over time ). Cc @fuweid.
With the proposed interface, a user can easily check if pidfd is being used:
2. Obtain a pidfd for additional direct use.
Aside from use cases already covered by existing
os.Process
methods, pidfd can also be used to:Other use cases may emerge in the future.
Currently, the only way to obtain a pidfd on Linux is to execute a new process (via
os.StartProcess
oros/exec
) with process'Attr.SysAttr.PidFD
field set. This works if we're starting the process, but not in any other case (someone else starts a process for us, or it is already running).Questions
1. What are (could be) the additional direct use cases of Windows process handle?
A few are listed here. Apparently some git grep (GetPriorityClass, SetPriorityClass, AssignProcessToJobObject) are implemented in golang.org/x/sys/windows.
2. Should a duplicate of a handle be returned, or the original handle?
Return the original one, ensuring that
Handle
documentation describes when pidfd may become invalid.Arguments against duplicated handle:
Release
method and a proper finalizer;Arguments for duplicated handle:
3. Should Handle return
*os.File
rather thanuintptr
?Raw handle makes more sense in this case, and the finalizer set by
NewFile
does not make sense if the original handle is returned. Also, this won't work for Windows handle is not a file.4. Should this be Linux-specific?
Probably not. Since we have a boolean flag returned, we can implement it for all platforms and return
0, false
for those that do not support process handle (other than Windows and Linux).The text was updated successfully, but these errors were encountered: