-
Notifications
You must be signed in to change notification settings - Fork 13.2k
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
calling libc::res_init from multiple threads is unsafe on at least OSX #43592
Comments
Your proposed workaround of only calling |
Can we detect whether the symbol |
This kinda sorta seems to work (in libstd we'd probably replace most of the machinery with the existing extern crate libc;
use std::ffi::CStr;
use std::mem;
fn glibc_version() -> Option<&'static CStr> {
unsafe {
let funcptr = libc::dlsym(
libc::RTLD_DEFAULT,
"gnu_get_libc_version\0".as_ptr() as *const libc::c_char,
);
if !funcptr.is_null() {
type CharstarFn = extern "C" fn() -> *const libc::c_char;
let gnu_get_libc_version: CharstarFn = mem::transmute(funcptr);
Some(CStr::from_ptr(gnu_get_libc_version()))
} else {
None
}
}
}
fn main() {
println!("{:?}", glibc_version());
} For me, |
FWIW, I just hit a crash on macOS that appears to possibly be this. Edit: Here's the error message:
|
The previous workaround for gibc's res_init bug is not thread-safe on other implementations of libc, and it can cause crashes. Use a runtime check to make sure we only call res_init when we need to, which is also when it's safe. See rust-lang#43592.
replace libc::res_init with res_init_if_glibc_before_2_26 The previous workaround for gibc's res_init bug is not thread-safe on other implementations of libc, and it can cause crashes. Use a runtime check to make sure we only call res_init when we need to, which is also when it's safe. See #43592. ~This PR is returning an InvalidData IO error if the glibc version string fails to parse. We could also have treated that case as "not glibc", and gotten rid of the idea that these functions could return an error. (Though I'm not a huge fan of ignoring error returns from `res_init` in any case.) Do other folks agree with these design choices?~ I'm pretty new to hacking on libstd. Is there an easy way to build a toy rust program against my changes to test this, other than doing an entire `sudo make install` on my system? What's the usual workflow?
The fix just landed in master. @alex, any chance you can test it after the next nighty comes out? |
Will do. (Although I haven't seen this error in a few days, probably because race conditions :-/) |
Btw, n00b question for you @alex. How do you figure out from |
I didn't, at least not conclusively. I hit this crash, @Manishearth pointed me to this bug, and since it was crashing in "the right place", it seemed that with high probability this was the right diagnosis :-) If I hadn't been pointed at this issue, I probably would have compiled with ASAN and seen what I hit. |
Closing this since #44965 landed a while ago. |
We recently started calling
res_init
on DNS failure to fix a glibc bug, and there was some worry that we might be calling a function that isn't threadsafe. It looked like we might not have any thread safety issues with glibc specifically, but now I'm convinced that we do have them in libc on OSX. I haven't repro'd the issue in Rust yet, but I'm able to produce very scary crashes with the following Go program if I run it on OSX (but not on Linux):The result is inconsistent, but here are a couple examples:
It might be that the best workaround is to limit our calls to
res_init
to when we know we're linking against glibc? That way we could still fix the original bug (stale/etc/resolv.conf
data in glibc specifically), take advantage of the thread safety that glibc seems to have here (we might want to audit it more carefully than I'm able to), and not worry about breaking any other platforms.The text was updated successfully, but these errors were encountered: