// xfail-pretty // multi tasking k-nucleotide import io::reader_util; use std; import std::map; import std::map::hashmap; import std::sort; // 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) -> float { return (xx as float) * 100f / (yy as float); } pure fn le_by_val(kv0: &(TT,UU), kv1: &(TT,UU)) -> bool { let (_, v0) = *kv0; let (_, v1) = *kv1; return v0 >= v1; } pure fn le_by_key(kv0: &(TT,UU), kv1: &(TT,UU)) -> bool { let (k0, _) = *kv0; let (k1, _) = *kv1; return k0 <= k1; } // sort by key, then by value fn sortKV(orig: ~[(TT,UU)]) -> ~[(TT,UU)] { 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 { vec::push(pairs, (key, pct(val, total))); return true; }); let pairs_sorted = sortKV(pairs); let mut buffer = ~""; pairs_sorted.each(fn&(kv: (~[u8], float)) -> bool unsafe { let (k,v) = kv; buffer += (fmt!{"%s %0.3f\n", str::to_upper(str::unsafe::from_bytes(k)), v}); return true; }); return buffer; } // given a map, search for the frequency of a pattern fn find(mm: hashmap<~[u8], uint>, key: ~str) -> uint { alt mm.find(str::bytes(str::to_lower(key))) { option::none { return 0u; } option::some(num) { return num; } } } // given a map, increment the counter for a key fn update_freq(mm: hashmap<~[u8], uint>, key: &[u8]) { let key = vec::slice(key, 0, key.len()); alt mm.find(key) { 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; } return vec::slice(bb, len - (nn - 1u), len); } fn make_sequence_processor(sz: uint, from_parent: comm::port<~[u8]>, to_parent: comm::chan<~str>) { let freqs: hashmap<~[u8], uint> = map::bytes_hash(); let mut carry: ~[u8] = ~[]; let mut total: uint = 0u; let mut line: ~[u8]; loop { line = comm::recv(from_parent); if line == ~[] { break; } carry = windows_with_carry(carry + line, sz, |window| { update_freq(freqs, window); total += 1u; }); } let buffer = alt sz { 1u { sort_and_fmt(freqs, total) } 2u { sort_and_fmt(freqs, total) } 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"} } _ { ~"" } }; //comm::send(to_parent, fmt!{"yay{%u}", sz}); comm::send(to_parent, buffer); } // given a FASTA file on stdin, process sequence THREE fn main(args: ~[~str]) { 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::connect( env!{"CFG_SRC_DIR"}, ~"src/test/bench/shootout-k-nucleotide.data" ); result::get(io::file_reader(path)) } else { io::stdin() }; // initialize each sequence sorter let sizes = ~[1u,2u,3u,4u,6u,12u,18u]; let from_child = vec::map (sizes, |_sz| comm::port() ); let to_parent = vec::mapi(sizes, |ii, _sz| comm::chan(from_child[ii]) ); let to_child = vec::mapi(sizes, fn@(ii: uint, sz: uint) -> comm::chan<~[u8]> { return do task::spawn_listener |from_parent| { make_sequence_processor(sz, from_parent, to_parent[ii]); }; }); // 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 { again; } alt (line[0], proc_mode) { // start processing if this is the one ('>' as u8, false) { alt str::find_str_from(line, ~"THREE", 1u) { option::some(_) { proc_mode = true; } option::none { } } } // break our processing ('>' as u8, true) { break; } // process the sequence for k-mers (_, true) { let line_bytes = str::bytes(line); for sizes.eachi |ii, _sz| { let mut lb = line_bytes; comm::send(to_child[ii], lb); } } // whatever _ { } } } // finish... for sizes.eachi |ii, _sz| { comm::send(to_child[ii], ~[]); } // now fetch and print result messages for sizes.eachi |ii, _sz| { io::println(comm::recv(from_child[ii])); } }