2012-12-10 19:32:48 -06:00
|
|
|
// 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.
|
|
|
|
|
2012-02-03 19:46:17 -06:00
|
|
|
// based on:
|
2012-02-03 21:50:32 -06:00
|
|
|
// http://shootout.alioth.debian.org/
|
|
|
|
// u64q/program.php?test=mandelbrot&lang=python3&id=2
|
2012-02-03 19:46:17 -06:00
|
|
|
//
|
2012-12-20 11:35:12 -06:00
|
|
|
// takes 2 optional args:
|
2012-02-03 22:43:48 -06:00
|
|
|
// square image size, defaults to 80_u
|
|
|
|
// output path, default is "" (no output), "-" means stdout
|
|
|
|
//
|
2012-02-03 19:46:17 -06:00
|
|
|
// in the shootout, they use 16000 as image size
|
|
|
|
//
|
2012-02-03 22:43:48 -06:00
|
|
|
// writes pbm image to output path
|
2012-02-03 19:46:17 -06:00
|
|
|
|
2012-09-11 19:46:20 -05:00
|
|
|
extern mod std;
|
2012-09-05 14:32:05 -05:00
|
|
|
use io::WriterUtil;
|
2013-02-01 01:13:36 -06:00
|
|
|
use std::oldmap::HashMap;
|
2012-02-03 19:46:17 -06:00
|
|
|
|
2012-07-27 21:32:42 -05:00
|
|
|
struct cmplx {
|
2012-09-07 16:50:47 -05:00
|
|
|
re: f64,
|
|
|
|
im: f64
|
2012-07-11 17:00:40 -05:00
|
|
|
}
|
|
|
|
|
2012-09-19 20:00:26 -05:00
|
|
|
impl cmplx : ops::Mul<cmplx,cmplx> {
|
2012-11-21 14:25:35 -06:00
|
|
|
pure fn mul(&self, x: &cmplx) -> cmplx {
|
2012-09-19 20:00:26 -05:00
|
|
|
cmplx {
|
|
|
|
re: self.re*(*x).re - self.im*(*x).im,
|
|
|
|
im: self.re*(*x).im + self.im*(*x).re
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-02-03 19:46:17 -06:00
|
|
|
|
2012-09-19 20:00:26 -05:00
|
|
|
impl cmplx : ops::Add<cmplx,cmplx> {
|
2012-12-06 17:36:26 -06:00
|
|
|
pure fn add(&self, x: &cmplx) -> cmplx {
|
2012-09-19 20:00:26 -05:00
|
|
|
cmplx {
|
|
|
|
re: self.re + (*x).re,
|
|
|
|
im: self.im + (*x).im
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-02-03 19:46:17 -06:00
|
|
|
|
2012-12-20 11:35:12 -06:00
|
|
|
struct Line {i: uint, b: ~[u8]}
|
2012-07-27 21:32:42 -05:00
|
|
|
|
2012-02-03 19:46:17 -06:00
|
|
|
pure fn cabs(x: cmplx) -> f64
|
|
|
|
{
|
2012-02-03 21:50:32 -06:00
|
|
|
x.re*x.re + x.im*x.im
|
2012-02-03 19:46:17 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
fn mb(x: cmplx) -> bool
|
|
|
|
{
|
2012-07-27 21:32:42 -05:00
|
|
|
let mut z = cmplx {re: 0f64, im: 0f64};
|
2012-03-22 10:39:41 -05:00
|
|
|
let mut i = 0;
|
|
|
|
let mut in = true;
|
2012-02-03 21:50:32 -06:00
|
|
|
while i < 50 {
|
|
|
|
z = z*z + x;
|
2012-06-04 19:26:17 -05:00
|
|
|
if cabs(z) >= 4f64 {
|
2012-02-03 21:50:32 -06:00
|
|
|
in = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
i += 1;
|
|
|
|
}
|
|
|
|
in
|
2012-02-03 19:46:17 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
fn fillbyte(x: cmplx, incr: f64) -> u8 {
|
2012-03-22 10:39:41 -05:00
|
|
|
let mut rv = 0_u8;
|
|
|
|
let mut i = 0_u8;
|
2012-02-03 21:50:32 -06:00
|
|
|
while i < 8_u8 {
|
2012-07-27 21:32:42 -05:00
|
|
|
let z = cmplx {re: x.re + (i as f64)*incr, im: x.im};
|
2012-02-03 21:50:32 -06:00
|
|
|
if mb(z) {
|
|
|
|
rv += 1_u8 << (7_u8 - i);
|
|
|
|
}
|
|
|
|
i += 1_u8;
|
|
|
|
}
|
|
|
|
rv
|
2012-02-03 19:46:17 -06:00
|
|
|
}
|
|
|
|
|
2012-12-27 13:42:50 -06:00
|
|
|
fn chanmb(i: uint, size: uint) -> Line
|
2012-02-03 19:46:17 -06:00
|
|
|
{
|
2012-06-29 18:26:56 -05:00
|
|
|
let mut crv = ~[];
|
2012-06-04 19:26:17 -05:00
|
|
|
let incr = 2f64/(size as f64);
|
|
|
|
let y = incr*(i as f64) - 1f64;
|
|
|
|
let xincr = 8f64*incr;
|
2012-06-30 18:19:07 -05:00
|
|
|
for uint::range(0_u, size/8_u) |j| {
|
2012-07-27 21:32:42 -05:00
|
|
|
let x = cmplx {re: xincr*(j as f64) - 1.5f64, im: y};
|
2012-09-26 19:33:34 -05:00
|
|
|
crv.push(fillbyte(x, incr));
|
2012-02-03 21:50:32 -06:00
|
|
|
};
|
2012-12-27 13:42:50 -06:00
|
|
|
Line {i:i, b:crv}
|
2012-02-03 19:46:17 -06:00
|
|
|
}
|
|
|
|
|
2012-12-30 11:54:44 -06:00
|
|
|
struct Devnull();
|
2012-02-03 22:43:48 -06:00
|
|
|
|
2012-12-30 11:54:44 -06:00
|
|
|
impl Devnull: io::Writer {
|
2012-12-24 15:29:36 -06:00
|
|
|
fn write(&self, _b: &[const u8]) {}
|
|
|
|
fn seek(&self, _i: int, _s: io::SeekStyle) {}
|
|
|
|
fn tell(&self) -> uint {0_u}
|
|
|
|
fn flush(&self) -> int {0}
|
|
|
|
fn get_type(&self) -> io::WriterType { io::File }
|
2012-02-03 22:43:48 -06:00
|
|
|
}
|
|
|
|
|
2012-12-27 13:42:50 -06:00
|
|
|
fn writer(path: ~str, pport: pipes::Port<Line>, size: uint)
|
2012-02-03 19:46:17 -06:00
|
|
|
{
|
2012-08-14 15:38:35 -05:00
|
|
|
let cout: io::Writer = match path {
|
2012-08-03 21:59:04 -05:00
|
|
|
~"" => {
|
2012-12-30 11:54:44 -06:00
|
|
|
Devnull as io::Writer
|
2012-02-03 22:43:48 -06:00
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
~"-" => {
|
2012-03-12 22:04:27 -05:00
|
|
|
io::stdout()
|
2012-02-03 22:43:48 -06:00
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
_ => {
|
2012-02-03 22:43:48 -06:00
|
|
|
result::get(
|
2012-09-25 18:23:04 -05:00
|
|
|
&io::file_writer(&Path(path),
|
2012-08-14 15:38:35 -05:00
|
|
|
~[io::Create, io::Truncate]))
|
2012-02-03 22:43:48 -06:00
|
|
|
}
|
|
|
|
};
|
2012-07-14 00:57:48 -05:00
|
|
|
cout.write_line(~"P4");
|
2012-08-22 19:24:52 -05:00
|
|
|
cout.write_line(fmt!("%u %u", size, size));
|
2012-09-19 17:13:04 -05:00
|
|
|
let lines: HashMap<uint, ~[u8]> = HashMap();
|
2012-03-22 10:39:41 -05:00
|
|
|
let mut done = 0_u;
|
|
|
|
let mut i = 0_u;
|
2012-02-03 21:50:32 -06:00
|
|
|
while i < size {
|
2012-12-27 13:42:50 -06:00
|
|
|
let aline = pport.recv();
|
2012-02-03 21:50:32 -06:00
|
|
|
if aline.i == done {
|
2012-08-22 19:24:52 -05:00
|
|
|
debug!("W %u", aline.i);
|
2012-02-03 21:50:32 -06:00
|
|
|
cout.write(aline.b);
|
|
|
|
done += 1_u;
|
2012-03-22 10:39:41 -05:00
|
|
|
let mut prev = done;
|
2012-02-03 21:50:32 -06:00
|
|
|
while prev <= i {
|
2013-02-02 01:04:22 -06:00
|
|
|
if lines.contains_key_ref(&prev) {
|
2012-08-22 19:24:52 -05:00
|
|
|
debug!("WS %u", prev);
|
2012-09-12 22:22:56 -05:00
|
|
|
cout.write(lines.get(prev));
|
2012-02-03 21:50:32 -06:00
|
|
|
done += 1_u;
|
|
|
|
lines.remove(prev);
|
|
|
|
prev += 1_u;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
else {
|
2012-08-22 19:24:52 -05:00
|
|
|
debug!("S %u", aline.i);
|
2012-12-06 17:36:26 -06:00
|
|
|
lines.insert(aline.i, copy aline.b); // FIXME: bad for perf
|
2012-02-03 21:50:32 -06:00
|
|
|
};
|
|
|
|
i += 1_u;
|
|
|
|
}
|
2012-02-03 19:46:17 -06:00
|
|
|
}
|
|
|
|
|
2012-10-03 21:16:27 -05:00
|
|
|
fn main() {
|
|
|
|
let args = os::args();
|
2012-07-14 00:57:48 -05:00
|
|
|
let args = if os::getenv(~"RUST_BENCH").is_some() {
|
2012-12-20 11:35:12 -06:00
|
|
|
~[~"", ~"4000"]
|
2012-05-24 00:53:50 -05:00
|
|
|
} else {
|
|
|
|
args
|
|
|
|
};
|
|
|
|
|
2012-12-20 11:35:12 -06:00
|
|
|
let path = if vec::len(args) < 3_u { ~"" }
|
|
|
|
else { copy args[2] }; // FIXME: bad for perf
|
2012-06-03 09:45:23 -05:00
|
|
|
|
|
|
|
let size = if vec::len(args) < 2_u { 80_u }
|
|
|
|
else { uint::from_str(args[1]).get() };
|
2012-05-24 00:53:50 -05:00
|
|
|
|
2012-12-27 13:42:50 -06:00
|
|
|
let (pport, pchan) = pipes::stream();
|
|
|
|
let pchan = pipes::SharedChan(pchan);
|
2012-06-30 18:19:07 -05:00
|
|
|
for uint::range(0_u, size) |j| {
|
2012-12-27 13:42:50 -06:00
|
|
|
let cchan = pchan.clone();
|
|
|
|
do task::spawn |move cchan| { cchan.send(chanmb(j, size)) };
|
2012-02-03 21:50:32 -06:00
|
|
|
};
|
2012-12-27 13:42:50 -06:00
|
|
|
writer(path, pport, size);
|
2012-02-03 19:46:17 -06:00
|
|
|
}
|