rust/src/test/bench/shootout-k-nucleotide-pipes.rs

216 lines
5.7 KiB
Rust
Raw Normal View History

// xfail-pretty
// multi tasking k-nucleotide
#[legacy_modes];
extern mod std;
2012-09-05 12:32:05 -07:00
use std::map;
2012-09-10 15:38:28 -07:00
use std::map::HashMap;
2012-09-05 12:32:05 -07:00
use std::sort;
use io::ReaderUtil;
use pipes::{stream, Port, Chan};
2012-09-10 15:35:37 -07:00
use cmp::Ord;
// given a map, print a sorted version of it
2012-09-10 15:38:28 -07:00
fn sort_and_fmt(mm: HashMap<~[u8], uint>, total: uint) -> ~str {
fn pct(xx: uint, yy: uint) -> float {
2012-08-01 17:30:05 -07:00
return (xx as float) * 100f / (yy as float);
}
2012-09-10 15:35:37 -07:00
pure fn le_by_val<TT: Copy, UU: Copy Ord>(kv0: &(TT,UU),
kv1: &(TT,UU)) -> bool {
let (_, v0) = *kv0;
let (_, v1) = *kv1;
2012-08-01 17:30:05 -07:00
return v0 >= v1;
}
2012-09-10 15:35:37 -07:00
pure fn le_by_key<TT: Copy Ord, UU: Copy>(kv0: &(TT,UU),
kv1: &(TT,UU)) -> bool {
let (k0, _) = *kv0;
let (k1, _) = *kv1;
2012-08-01 17:30:05 -07:00
return k0 <= k1;
}
// sort by key, then by value
2012-09-10 15:35:37 -07:00
fn sortKV<TT: Copy Ord, UU: Copy Ord>(orig: ~[(TT,UU)]) -> ~[(TT,UU)] {
2012-08-01 17:30:05 -07:00
return sort::merge_sort(le_by_val, sort::merge_sort(le_by_key, orig));
}
let mut pairs = ~[];
// map -> [(k,%)]
mm.each(fn&(key: ~[u8], val: uint) -> bool {
pairs.push((key, pct(val, total)));
2012-08-01 17:30:05 -07:00
return true;
});
let pairs_sorted = sortKV(pairs);
let mut buffer = ~"";
for pairs_sorted.each |kv| {
let (k,v) = *kv;
unsafe {
buffer += (fmt!("%s %0.3f\n", str::to_upper(str::raw::from_bytes(k)), v));
}
}
2012-08-01 17:30:05 -07:00
return buffer;
}
// given a map, search for the frequency of a pattern
2012-09-10 15:38:28 -07:00
fn find(mm: HashMap<~[u8], uint>, key: ~str) -> uint {
match mm.find(str::to_bytes(str::to_lower(key))) {
2012-08-20 12:23:37 -07:00
option::None => { return 0u; }
option::Some(num) => { return num; }
}
}
// given a map, increment the counter for a key
2012-09-10 15:38:28 -07:00
fn update_freq(mm: HashMap<~[u8], uint>, key: &[u8]) {
let key = vec::slice(key, 0, key.len());
2012-08-06 12:34:08 -07:00
match mm.find(key) {
2012-08-20 12:23:37 -07:00
option::None => { mm.insert(key, 1u ); }
option::Some(val) => { mm.insert(key, 1u + val); }
}
}
// given a ~[u8], for each window call a function
// i.e., for "hello" and windows of size four,
// run it("hell") and it("ello"), then return "llo"
fn windows_with_carry(bb: &[u8], nn: uint,
it: fn(window: &[u8])) -> ~[u8] {
let mut ii = 0u;
let len = vec::len(bb);
while ii < len - (nn - 1u) {
it(vec::view(bb, ii, ii+nn));
ii += 1u;
}
2012-08-01 17:30:05 -07:00
return vec::slice(bb, len - (nn - 1u), len);
}
2012-08-28 11:11:15 -07:00
fn make_sequence_processor(sz: uint, from_parent: pipes::Port<~[u8]>,
to_parent: pipes::Chan<~str>) {
let freqs: HashMap<~[u8], uint> = map::HashMap();
let mut carry: ~[u8] = ~[];
let mut total: uint = 0u;
let mut line: ~[u8];
loop {
line = from_parent.recv();
if line == ~[] { break; }
carry = windows_with_carry(carry + line, sz, |window| {
update_freq(freqs, window);
total += 1u;
});
}
2012-08-06 12:34:08 -07:00
let buffer = match sz {
2012-08-03 19:59:04 -07:00
1u => { sort_and_fmt(freqs, total) }
2u => { sort_and_fmt(freqs, total) }
2012-08-22 17:24:52 -07:00
3u => { fmt!("%u\t%s", find(freqs, ~"GGT"), ~"GGT") }
4u => { fmt!("%u\t%s", find(freqs, ~"GGTA"), ~"GGTA") }
6u => { fmt!("%u\t%s", find(freqs, ~"GGTATT"), ~"GGTATT") }
12u => { fmt!("%u\t%s", find(freqs, ~"GGTATTTTAATT"), ~"GGTATTTTAATT") }
18u => { fmt!("%u\t%s", find(freqs, ~"GGTATTTTAATTTATAGT"), ~"GGTATTTTAATTTATAGT") }
2012-08-03 19:59:04 -07:00
_ => { ~"" }
};
2012-08-22 17:24:52 -07:00
//comm::send(to_parent, fmt!("yay{%u}", sz));
to_parent.send(buffer);
}
// given a FASTA file on stdin, process sequence THREE
fn main() {
let args = os::args();
let rdr = if os::getenv(~"RUST_BENCH").is_some() {
// FIXME: Using this compile-time env variable is a crummy way to
// get to this massive data set, but #include_bin chokes on it (#2598)
let path = Path(env!("CFG_SRC_DIR"))
.push_rel(&Path("src/test/bench/shootout-k-nucleotide.data"));
2012-09-25 16:23:04 -07:00
result::get(&io::file_reader(&path))
} else {
io::stdin()
};
// initialize each sequence sorter
let sizes = ~[1u,2u,3u,4u,6u,12u,18u];
2012-08-20 12:23:37 -07:00
let streams = vec::map(sizes, |_sz| Some(stream()));
let streams = vec::to_mut(streams);
let mut from_child = ~[];
let to_child = vec::mapi(sizes, |ii, sz| {
let sz = *sz;
2012-08-20 12:23:37 -07:00
let mut stream = None;
stream <-> streams[ii];
let (to_parent_, from_child_) = option::unwrap(stream);
from_child.push(from_child_);
2012-07-10 11:58:43 -07:00
let (to_child, from_parent) = pipes::stream();
do task::spawn_with(from_parent) |from_parent| {
make_sequence_processor(sz, from_parent, to_parent_);
};
to_child
});
// latch stores true after we've started
// reading the sequence of interest
let mut proc_mode = false;
while !rdr.eof() {
let line: ~str = rdr.read_line();
if str::len(line) == 0u { loop; }
2012-08-06 12:34:08 -07:00
match (line[0], proc_mode) {
// start processing if this is the one
2012-08-03 19:59:04 -07:00
('>' as u8, false) => {
2012-08-06 12:34:08 -07:00
match str::find_str_from(line, ~"THREE", 1u) {
2012-08-20 12:23:37 -07:00
option::Some(_) => { proc_mode = true; }
option::None => { }
}
}
// break our processing
2012-08-03 19:59:04 -07:00
('>' as u8, true) => { break; }
// process the sequence for k-mers
2012-08-03 19:59:04 -07:00
(_, true) => {
let line_bytes = str::to_bytes(line);
for sizes.eachi |ii, _sz| {
let mut lb = line_bytes;
to_child[ii].send(lb);
}
}
// whatever
2012-08-03 19:59:04 -07:00
_ => { }
}
}
// finish...
for sizes.eachi |ii, _sz| {
to_child[ii].send(~[]);
}
// now fetch and print result messages
for sizes.eachi |ii, _sz| {
io::println(from_child[ii].recv());
}
}