-
Notifications
You must be signed in to change notification settings - Fork 2.1k
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
Use go 1.22 #4278
Use go 1.22 #4278
Conversation
Signed-off-by: lifubang <lifubang@acmcoder.com> Signed-off-by: lfbzhm <lifubang@acmcoder.com>
This reverts commit ac31da6. Signed-off-by: lifubang <lifubang@acmcoder.com> Signed-off-by: lfbzhm <lifubang@acmcoder.com>
This reverts commit e377e16. Signed-off-by: lifubang <lifubang@acmcoder.com> Signed-off-by: lfbzhm <lifubang@acmcoder.com>
2339a19
to
19cd152
Compare
// Since Go 1.21 </~https://github.com/golang/go/commit/c426c87012b5e>, the Go | ||
// runtime will try to call pthread_getattr_np(pthread_self()). This causes | ||
// issues with nsexec and requires backward compatibility with old versions | ||
// of the glibc. |
This comment was marked as off-topic.
This comment was marked as off-topic.
Sorry, something went wrong.
As pointed out by @cyphar /~https://github.com/golang/go/issues/65625#issuecomment-2036320298:
Go maintainers (@cherrymui @prattmic) concur:
If @cyphar still thinks the same way, maybe we should try switching to fork? |
I’ll reply in go PR thread later after I finished read all cgo code. Before that I want to say that tid dirty cache issue is only influence someone who writing multi thread programs(with clone(2)), but both golang and runc are not, golang only use pthread to get stack address and size, we have another way to do this, not just only using pthread. BTW, I also don’t know what’s the purpose of this commit, it seems that there’s no detail comment in the commit message: golang/go@52e17c2 |
I think in this time, if we switch to fork, we should remove So I strongly recommend this PR(#4278), it's easy to review. Though we don't solve any
|
This might not be a bad thing. There is some runc functionality that is plain wrong (I'm not saying
I totally agree we have to have something sooner than later. |
ea2580a
to
7c6452d
Compare
} | ||
|
||
pthread_attr_getstack(&attr, &addr, &size); // low address | ||
#elif defined(__illumos__) |
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.
As I said earlier, it makes sense to remove solaris/illumos support, for clarity.
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.
I think if the change is small, I want to keep it as is, because we just have only added/changed about 5 lines code.
Then people will read the code more clearly.
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.
Makes sense
// pthread_getattr_np is a GNU extension supported in glibc. | ||
// Solaris is not glibc but does support pthread_getattr_np | ||
// (and the fallback doesn't work...). Illumos does not. | ||
err = pthread_getattr_np(pthread_self(), &attr); // GNU extension |
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.
attr
is not intialized?
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.
Yes, before glibc 2.32, attr
are not initialized with all the field filled with zero!
After glibc 2.31, it is initialized in pthread_getattr_np
.
/~https://github.com/bminor/glibc/blob/glibc-2.32/nptl/pthread_getattr_np.c#L38
I have no time to find out which commit include this change.
7c6452d
to
45d37ac
Compare
For the sake of review, here's the diff against the upstream source: --- ../golang/go/src/runtime/cgo/gcc_stack_unix.c 2024-05-21 13:14:57.977264422 -0700
+++ libcontainer/nsenter/nsenter_go122.go 2024-05-21 13:09:38.552902378 -0700
@@ -1,28 +1,47 @@
-// Copyright 2023 The Go Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style
-// license that can be found in the LICENSE file.
+//go:build linux && !gccgo && go1.22
+// +build linux,!gccgo,go1.22
-//go:build unix && !darwin
+package nsenter
-#ifndef _GNU_SOURCE // pthread_getattr_np
+/*
+#cgo LDFLAGS: -Wl,--wrap=x_cgo_getstackbound
#define _GNU_SOURCE
-#endif
-
#include <pthread.h>
-#include "libcgo.h"
+#include <stdint.h>
-void
-x_cgo_getstackbound(uintptr bounds[2])
+typedef uintptr_t uintptr;
+
+// Since Go 1.22 </~https://github.com/golang/go/commit/52e17c2>, the
+// go runtime will try to call pthread_getattr_np(pthread_self()) in cgo.
+// This will cause issues in nsexec with glibc < 2.32. so it requires
+// us to provide a backward compatibility with old versions of glibc.
+// The core issue of glibc is that, before 2.32, `pthread_getattr_np`
+// did not call `__pthread_attr_init (attr)`, we need to init the attr
+// in go runtime. Fortunately, cgo exports a function named `x_cgo_getstackbound`,
+// we can wrap it in the compile phrase. Please see:
+// /~https://github.com/golang/go/blob/52e17c2/src/runtime/cgo/gcc_stack_unix.c
+// Fix me: this hack looks ugly, once we have removed `clone(2)` in nsexec,
+// please remember to remove this hack.
+void __wrap_x_cgo_getstackbound(uintptr bounds[2])
{
pthread_attr_t attr;
void *addr;
size_t size;
+ int err;
#if defined(__GLIBC__) || (defined(__sun) && !defined(__illumos__))
// pthread_getattr_np is a GNU extension supported in glibc.
// Solaris is not glibc but does support pthread_getattr_np
// (and the fallback doesn't work...). Illumos does not.
- pthread_getattr_np(pthread_self(), &attr); // GNU extension
+ err = pthread_getattr_np(pthread_self(), &attr); // GNU extension
+ // The main change begin
+ if (err != 0) {
+ // This is to have a backward compatibility with glibc < 2.32
+ pthread_attr_init(&attr);
+ pthread_getattr_np(pthread_self(), &attr);
+ }
+ // The main change end
+
pthread_attr_getstack(&attr, &addr, &size); // low address
#elif defined(__illumos__)
pthread_attr_init(&attr);
@@ -37,10 +56,8 @@
#endif
pthread_attr_destroy(&attr);
- // bounds points into the Go stack. TSAN can't see the synchronization
- // in Go around stack reuse.
- _cgo_tsan_acquire();
bounds[0] = (uintptr)addr;
bounds[1] = (uintptr)addr + size;
- _cgo_tsan_release();
}
+*/
+import "C" |
Because for some old versions glibc, there is a tid cache update issue, it will cause cgo with clone(2) in go1.21 and above error when calling `pthread_getattr_np`, so we need to add a backward compatibility with old versions glibc. Signed-off-by: lifubang <lifubang@acmcoder.com>
45d37ac
to
741b045
Compare
Here's the alternative: https://go-review.googlesource.com/c/go/+/587919 |
I think only adding init is not enough: I suggest to handle this error, or else other contributors will introduce this error again when working around this. https://go.dev/cl/563379 At least I think we should add a comment about this bug from glibc to let other people know we can’t just only check err!=0 when adding error assert. |
An alternative to #4193.
Fixes #4233
As mentioned in golang/go#65625 (comment), the commit golang/go@52e17c2 (CL 519457) uses
x_cgo_getstackbound
to setg->stacklo
based on the stack size. It will cause some backward compatibility issues with older versions of the glibc because of the tid cache update issue.I think golang runtime should provide this backward compatibility(I think we should submit a PR to golang), but unfortunately they didn't implement it. Before golang support this feature, I think we can wrap
x_cgo_getstackbound
function to ensure we can have this backward compatibility.