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

Unsafe behaviour without unsafe code, bug in lifetime checker / shared memory restrictions? #19552

Closed
ztdwu opened this issue Dec 5, 2014 · 10 comments
Assignees
Milestone

Comments

@ztdwu
Copy link

ztdwu commented Dec 5, 2014

#![feature( if_let )]
use std::io::BufReader;

fn main() {
    let text = b"the
                 quick
                 brown
                 fox
                 jumps
                 over";
    let mut reader = BufReader::new(text);

    let (tx, rx) = channel();

    spawn(proc() {
        for line in reader.lines() {
            let line = line.unwrap();
            let line = line.trim_chars(|c: char| c.is_whitespace());
            let list = vec![ line ];
            if let [ word ] = &*list {
                tx.send(word);
            }
        }
    });

    for word in rx.iter() {
        println!("{}", word);
    }
}

One would expect the output to be

the
quick
brown
fox
jumps
over

But the actual output is:

the
overs
overs
ove
overs
over

It looks like the memory location at word is overwritten.
The program will behave correctly if word is deep copied before being sent, i.e. changing tx.send(word) to tx.send(word.to_string())

@ztdwu ztdwu changed the title Unsafe behaviour without unsafe code, bug in lifetime scope / shared memory restrictions? Unsafe behaviour without unsafe code, bug in lifetime checker / shared memory restrictions? Dec 5, 2014
@sfackler
Copy link
Member

sfackler commented Dec 5, 2014

cc @luqmana looks like it could be that by-val match optimization issue?

@nikomatsakis
Copy link
Contributor

cc me

@ftxqxd
Copy link
Contributor

ftxqxd commented Dec 5, 2014

On 0.12.0 this fails to compile because line does not fulfil the 'static lifetime required by Sender/Receiver. I think that this might be #19261?

@nikomatsakis
Copy link
Contributor

I'm not sure. This program runs fine for me. The byte literal is typed as &'static [u8] and, from inspecting the output, does appear to be placed into a static constant. Basically everything seems to be...going correctly here?

@ztdwu
Copy link
Author

ztdwu commented Dec 6, 2014

This is most likely a race condition. The output always seems correct in the playpen, and it's sometimes correct on my 1-core VM. But running it on a multicore machine always gives me jumbled up output. I tested it on the latest nightly on both Windows and Linux.

@brson brson removed the I-nominated label Dec 11, 2014
@brson brson added this to the 1.0 milestone Dec 11, 2014
@brson
Copy link
Contributor

brson commented Dec 11, 2014

P-backcompat-lang 1.0

@nikomatsakis
Copy link
Contributor

@alexcrichton and I need to look more closely. I'm not yet sure what's going wrong.

@alexcrichton
Copy link
Member

Minimized version:

fn assert_send<T: Send>(_t: T) {}

fn main() {
    let line = String::new();
    match [line.as_slice()] {
        [ word ] => { assert_send(word); }
    }
}

@nikomatsakis
Copy link
Contributor

Seems to be tied to vector patterns. Should have guessed.

@nikomatsakis
Copy link
Contributor

Got a fix underway. Turns out to be a dup of #19997

@nikomatsakis nikomatsakis self-assigned this Jan 4, 2015
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

7 participants