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

239 lines
6.7 KiB
Rust
Raw Normal View History

// Copyright 2012 The Rust Project Developers. See the COPYRIGHT
// file at the top-level directory of this distribution and at
// http://rust-lang.org/COPYRIGHT.
//
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
// option. This file may not be copied, modified, or distributed
// except according to those terms.
// xfail-pretty the `let to_child` line gets an extra newline
// multi tasking k-nucleotide
extern mod extra;
use extra::sort;
use std::cmp::Ord;
use std::comm::{stream, Port, Chan};
use std::comm;
use std::hashmap::HashMap;
use std::io::ReaderUtil;
use std::io;
use std::option;
use std::os;
use std::str;
use std::task;
use std::util;
use std::vec;
// given a map, print a sorted version of it
fn sort_and_fmt(mm: &HashMap<~[u8], uint>, total: uint) -> ~str {
fn pct(xx: uint, yy: uint) -> f64 {
return (xx as f64) * 100.0 / (yy as f64);
}
fn le_by_val<TT:Clone,
UU:Clone + Ord>(
2013-07-02 14:47:32 -05:00
kv0: &(TT,UU),
kv1: &(TT,UU))
-> bool {
let (_, v0) = (*kv0).clone();
let (_, v1) = (*kv1).clone();
2012-08-01 19:30:05 -05:00
return v0 >= v1;
}
fn le_by_key<TT:Clone + Ord,
UU:Clone>(
2013-07-02 14:47:32 -05:00
kv0: &(TT,UU),
kv1: &(TT,UU))
-> bool {
let (k0, _) = (*kv0).clone();
let (k1, _) = (*kv1).clone();
2012-08-01 19:30:05 -05:00
return k0 <= k1;
}
// sort by key, then by value
fn sortKV<TT:Clone + Ord, UU:Clone + Ord>(orig: ~[(TT,UU)]) -> ~[(TT,UU)] {
return sort::merge_sort(sort::merge_sort(orig, le_by_key), le_by_val);
}
let mut pairs = ~[];
// map -> [(k,%)]
for (key, &val) in mm.iter() {
pairs.push(((*key).clone(), pct(val, total)));
2013-02-02 13:47:41 -06:00
}
let pairs_sorted = sortKV(pairs);
let mut buffer = ~"";
for kv in pairs_sorted.iter() {
2013-07-02 14:47:32 -05:00
let (k,v) = (*kv).clone();
unsafe {
let b = str::raw::from_utf8(k);
// FIXME: #4318 Instead of to_ascii and to_str_ascii, could use
// to_ascii_move and to_str_move to not do a unnecessary copy.
2013-09-30 01:13:47 -05:00
buffer.push_str(format!("{} {:0.3f}\n",
b.to_ascii().to_upper().to_str_ascii(), v));
}
}
2012-08-01 19:30:05 -05:00
return buffer;
}
// given a map, search for the frequency of a pattern
fn find(mm: &HashMap<~[u8], uint>, key: ~str) -> uint {
// FIXME: #4318 Instead of to_ascii and to_str_ascii, could use
// to_ascii_move and to_str_move to not do a unnecessary copy.
let key = key.to_ascii().to_lower().to_str_ascii();
match mm.find_equiv(&key.as_bytes()) {
2012-08-20 14:23:37 -05:00
option::None => { return 0u; }
2013-03-23 20:22:00 -05:00
option::Some(&num) => { return num; }
}
}
// given a map, increment the counter for a key
fn update_freq(mm: &mut HashMap<~[u8], uint>, key: &[u8]) {
let key = key.to_owned();
2013-03-23 20:22:00 -05:00
let newval = match mm.pop(&key) {
Some(v) => v + 1,
None => 1
};
mm.insert(key, newval);
}
// 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;
2013-05-14 04:52:12 -05:00
let len = bb.len();
while ii < len - (nn - 1u) {
it(bb.slice(ii, ii+nn));
ii += 1u;
}
return bb.slice(len - (nn - 1u), len).to_owned();
}
fn make_sequence_processor(sz: uint,
from_parent: &Port<~[u8]>,
to_parent: &Chan<~str>) {
let mut freqs: HashMap<~[u8], uint> = HashMap::new();
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| {
2013-03-23 20:22:00 -05:00
update_freq(&mut freqs, window);
total += 1u;
});
}
let buffer = match sz {
2013-03-23 20:22:00 -05:00
1u => { sort_and_fmt(&freqs, total) }
2u => { sort_and_fmt(&freqs, total) }
2013-09-30 01:13:47 -05:00
3u => { format!("{}\t{}", find(&freqs, ~"GGT"), "GGT") }
4u => { format!("{}\t{}", find(&freqs, ~"GGTA"), "GGTA") }
6u => { format!("{}\t{}", find(&freqs, ~"GGTATT"), "GGTATT") }
12u => { format!("{}\t{}", find(&freqs, ~"GGTATTTTAATT"), "GGTATTTTAATT") }
18u => { format!("{}\t{}", find(&freqs, ~"GGTATTTTAATTTATAGT"), "GGTATTTTAATTTATAGT") }
2012-08-03 21:59:04 -05:00
_ => { ~"" }
};
2013-02-15 04:44:18 -06:00
to_parent.send(buffer);
}
// given a FASTA file on stdin, process sequence THREE
fn main() {
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"));
2013-07-26 20:36:51 -05:00
io::file_reader(&path).unwrap()
} else {
io::stdin()
};
// initialize each sequence sorter
let sizes = ~[1u,2,3,4,6,12,18];
let mut streams = vec::from_fn(sizes.len(), |_| Some(stream::<~str>()));
let mut from_child = ~[];
let to_child = do sizes.iter().zip(streams.mut_iter()).map |(sz, stream_ref)| {
let sz = *sz;
let stream = util::replace(stream_ref, None);
let (from_child_, to_parent_) = stream.unwrap();
2013-02-15 04:44:18 -06:00
from_child.push(from_child_);
2013-02-02 05:10:12 -06:00
let (from_parent, to_child) = comm::stream();
2013-02-15 04:44:18 -06:00
do task::spawn_with(from_parent) |from_parent| {
2013-04-26 16:04:39 -05:00
make_sequence_processor(sz, &from_parent, &to_parent_);
};
2013-02-15 04:44:18 -06:00
to_child
}.collect::<~[Chan<~[u8]>]>();
// 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 line.len() == 0u { continue; }
2013-05-13 13:26:15 -05:00
match (line[0] as char, proc_mode) {
// start processing if this is the one
2013-05-13 13:26:15 -05:00
('>', false) => {
match line.slice_from(1).find_str("THREE") {
2012-08-20 14:23:37 -05:00
option::Some(_) => { proc_mode = true; }
option::None => { }
}
}
// break our processing
2013-05-13 13:26:15 -05:00
('>', true) => { break; }
// process the sequence for k-mers
2012-08-03 21:59:04 -05:00
(_, true) => {
let line_bytes = line.as_bytes();
for (ii, _sz) in sizes.iter().enumerate() {
let lb = line_bytes.to_owned();
to_child[ii].send(lb);
}
}
// whatever
2012-08-03 21:59:04 -05:00
_ => { }
}
}
// finish...
for (ii, _sz) in sizes.iter().enumerate() {
to_child[ii].send(~[]);
}
// now fetch and print result messages
for (ii, _sz) in sizes.iter().enumerate() {
io::println(from_child[ii].recv());
}
}