rust/src/lib/fs.rs
Marijn Haverbeke 3bdbf74d47 Make moving of temporaries do the right thing, use it to optimize
This adds support for dropping cleanups for temporary values when they
are moved somewhere else. It then adds wraps most copy operations
(return, put in data structure, box, etc) in a way that will fall back
to a move when it is safe.

This saves a lot of taking/dropping, shaving over a megabyte off the
stage2/rustc binary size.

In some cases, most notably function returns, we could detect that the
returned value is a local variable, and can thus be safely moved even
though it is not a temporary. This will require putting some more
information in lvals.

I did not yet handle function arguments, since the logic for passing
them looked too convoluted to touch. I'll probably try that in the
near future, since it's bound to be a big win.
2011-07-07 15:54:01 +02:00

85 lines
2.1 KiB
Rust

import os::getcwd;
import os_fs;
import str;
native "rust" mod rustrt {
fn rust_file_is_dir(str path) -> int;
}
fn path_sep() -> str { ret str::from_char(os_fs::path_sep); }
type path = str;
fn dirname(path p) -> path {
let int i = str::rindex(p, os_fs::path_sep as u8);
if (i == -1) {
i = str::rindex(p, os_fs::alt_path_sep as u8);
if (i == -1) { ret "."; }
}
ret str::substr(p, 0u, i as uint);
}
fn basename(path p) -> path {
let int i = str::rindex(p, os_fs::path_sep as u8);
if (i == -1) {
i = str::rindex(p, os_fs::alt_path_sep as u8);
if (i == -1) { ret p; }
}
auto len = str::byte_len(p);
if (i + 1 as uint >= len) { ret p; }
ret str::slice(p, i + 1 as uint, len);
}
// FIXME: Need some typestate to avoid bounds check when len(pre) == 0
fn connect(path pre, path post) -> path {
auto len = str::byte_len(pre);
ret if (pre.(len - 1u) == os_fs::path_sep as u8) {
// Trailing '/'?
pre + post
} else { pre + path_sep() + post };
}
fn file_is_dir(path p) -> bool { ret rustrt::rust_file_is_dir(p) != 0; }
fn list_dir(path p) -> vec[str] {
auto pl = str::byte_len(p);
if (pl == 0u || p.(pl - 1u) as char != os_fs::path_sep) {
p += path_sep();
}
let vec[str] full_paths = [];
for (str filename in os_fs::list_dir(p)) {
if (!str::eq(filename, ".")) {
if (!str::eq(filename, "..")) {
vec::push[str](full_paths, p + filename);
}
}
}
ret full_paths;
}
fn path_is_absolute(path p) -> bool {
ret os_fs::path_is_absolute(p);
}
// FIXME: under Windows, we should prepend the current drive letter to paths
// that start with a slash.
fn make_absolute(path p) -> path {
if(path_is_absolute(p)) {
ret p;
}
else {
ret connect(getcwd(), p);
}
}
// Local Variables:
// mode: rust;
// fill-column: 78;
// indent-tabs-mode: nil
// c-basic-offset: 4
// buffer-file-coding-system: utf-8-unix
// compile-command: "make -k -C $RBUILD 2>&1 | sed -e 's/\\/x\\//x:\\//g'";
// End: