Skip to content
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

segfault when trying to write multiple times to a socket moved across a task boundary #12058

Closed
vogon opened this issue Feb 6, 2014 · 3 comments

Comments

@vogon
Copy link

vogon commented Feb 6, 2014

hey, I'm working on trying to make a multi-client network server as my first project in Rust, so I've got this small hello-worldish program:

extern mod extra;

use std::io::{Listener, Acceptor};
use std::io::net::ip::SocketAddr;
use std::io::net::tcp::{TcpListener, TcpStream};
use std::io::timer;

fn main() {
    let addr = from_str::<SocketAddr>("0.0.0.0:3569").unwrap();
    let listener = TcpListener::bind(addr).unwrap();

    let mut acceptor = listener.listen().unwrap();

    loop {
        let mut stream: TcpStream = acceptor.accept().unwrap();

        stream.write_str("hey hi\r\n");

        spawn(proc() {
            loop {
                let mut stream = stream;

                stream.write_str("ping\r\n");
                timer::sleep(1000);
            }
        });
    }
}

it compiles fine, but when I run this program, and connect to it by telnetting to localhost:3569, it prints "hey hi", then one "ping", then crashes.

here's a stack trace:

(gdb) bt
#0  0x00000000004629b4 in io::net::tcp::Writer$TcpStream::write::h3e42059c65c3f43f05au::v0.10.pre ()
#1  0x00000000004064d1 in io::Writer::write_str::hb41c95d4c4ae4cb1aY::v0.0 ()
#2  0x000000000040663e in main::anon::expr_fn::aa ()
#3  0x00000000004c9368 in task::__extensions__::build_start_wrapper::anon::anon::expr_fn::aI ()
#4  0x00000000004a1e33 in rt::task::__extensions__::run::anon::expr_fn::aG ()
#5  0x00000000004a9bec in rust_try ()
#6  0x00000000004a1da2 in rt::task::Task::run::hc9c9660d47e409e7TWaD::v0.10.pre
    ()
#7  0x00000000004c9039 in task::__extensions__::build_start_wrapper::anon::expr_fn::aw ()
#8  0x0000000000000000 in ?? ()

I'm using master as of 3e39e3e.

@vogon
Copy link
Author

vogon commented Feb 6, 2014

I also just updated to a newer Ubuntu rust-nightly (4509b49) and it still happens.

@alexcrichton
Copy link
Member

Closing as a dupe of #12041. You'll need to lift let mut stream = stream; outside of the loop (you're moving multiple times). The compiler should catch this for you, but that's the bug of #12041

@vogon
Copy link
Author

vogon commented Feb 6, 2014

cool, thanks so much!

bors added a commit to rust-lang-ci/rust that referenced this issue Jul 25, 2022
…-stack-please, r=jonas-schievink

fix: Spawn a new thread with a larger stack for the LSP and proc-macro server

This runs the server and proc-macro process in dedicated threads with 8 MB of stack space to paper over OS differences and fix occasional stack overflows.

This hopefully resolves rust-lang/rust-analyzer#11669
flip1995 pushed a commit to flip1995/rust that referenced this issue Jan 11, 2024
Fixed ICE introduced in rust-lang#12004

Issue: in rust-lang/rust-clippy#12004, we emit a lint for `filter(Option::is_some)`. If the
parent expression is a `.map` we don't emit that lint as there exists a
more specialized lint for that.

The ICE introduced in rust-lang/rust-clippy#12004 is a consequence of the assumption that a
parent expression after a filter would be a method call with the filter
call being the receiver. However, it is entirely possible to have a
closure of the form

```
|| { vec![Some(1), None].into_iter().filter(Option::is_some) }
```
The previous implementation looked at the parent expression; namely the
closure, and tried to check the parameters by indexing [0] on an empty
list.

This commit is an overhaul of the lint with significantly more FP tests
and checks.

Impl details:

1. We verify that the filter method we are in is a proper trait method
   to avoid FPs.
2. We check that the parent expression is not a map by checking whether
   it exists; if is a trait method; and then a method call.
3. We check that we don't have comments in the span.
4. We verify that we are in an Iterator of Option and Result.
5. We check the contents of the filter.
   1. For closures we peel it. If it is not a single expression, we don't
     lint. We then try again by checking the peeled expression.
   2. For paths, we do a typecheck to avoid FPs for types that impl
     functions with the same names.
   3. For calls, we verify the type, via the path, and that the param of
     the closure is the single argument to the call.
   4. For method calls we verify that the receiver is the parameter of
     the closure. Since we handle single, non-block exprs, the
     parameter can't be shadowed, so no FP.

This commit also adds additional FP tests.

Fixes: rust-lang#12058

Adding `@xFrednet` as you've the most context for this as you reviewed it last time.

`@rustbot` r? `@xFrednet`

---

changelog: none
(Will be backported and therefore don't effect stable)
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

2 participants