2011-06-15 13:19:50 -05:00
|
|
|
|
2011-05-12 10:24:54 -05:00
|
|
|
import std::map;
|
|
|
|
import std::map::hashmap;
|
2011-05-17 13:41:41 -05:00
|
|
|
import std::uint;
|
|
|
|
import std::int;
|
|
|
|
import std::vec;
|
2011-05-12 10:24:54 -05:00
|
|
|
import std::option::none;
|
2011-05-18 17:38:38 -05:00
|
|
|
import std::option::some;
|
2011-05-12 10:24:54 -05:00
|
|
|
import front::ast;
|
2011-05-14 21:02:30 -05:00
|
|
|
import front::ast::ty;
|
|
|
|
import front::ast::pat;
|
2011-05-30 18:22:58 -05:00
|
|
|
import front::codemap::codemap;
|
2011-06-01 20:10:10 -05:00
|
|
|
import front::ast::lit;
|
|
|
|
import front::ast::path;
|
2011-05-26 19:16:54 -05:00
|
|
|
import middle::walk;
|
2011-05-12 10:24:54 -05:00
|
|
|
import std::io::stdout;
|
|
|
|
import std::io::str_writer;
|
|
|
|
import std::io::string_writer;
|
|
|
|
import pretty::pprust::print_block;
|
2011-05-18 17:38:38 -05:00
|
|
|
import pretty::pprust::print_item;
|
2011-05-12 10:24:54 -05:00
|
|
|
import pretty::pprust::print_expr;
|
2011-06-01 20:10:10 -05:00
|
|
|
import pretty::pprust::print_path;
|
2011-05-12 10:24:54 -05:00
|
|
|
import pretty::pprust::print_decl;
|
|
|
|
import pretty::pprust::print_fn;
|
2011-05-14 21:02:30 -05:00
|
|
|
import pretty::pprust::print_type;
|
2011-06-09 11:48:16 -05:00
|
|
|
import pretty::ppaux::print_literal;
|
|
|
|
import pretty::ppaux::mo_untyped;
|
2011-05-28 21:16:18 -05:00
|
|
|
import pretty::pp::mk_printer;
|
2011-02-23 16:37:39 -06:00
|
|
|
|
|
|
|
type filename = str;
|
2011-06-15 13:19:50 -05:00
|
|
|
|
2011-04-08 11:44:20 -05:00
|
|
|
type span = rec(uint lo, uint hi);
|
2011-06-15 13:19:50 -05:00
|
|
|
|
2010-10-05 20:21:44 -05:00
|
|
|
type spanned[T] = rec(T node, span span);
|
2011-06-15 13:19:50 -05:00
|
|
|
|
Support all expression forms in typestate
Added support for self_method, cont, chan, port, recv, send, be,
do_while, spawn, and ext; handled break and cont correctly.
(However, there are no non-xfailed test cases for ext or spawn in
stage0 currently.)
Although the standard library compiles and all test cases pass with
typestate enabled, I left typestate checking disabled as rustc
terminates abnormally when building the standard library if so,
even though it does generate code correctly.
2011-04-21 19:39:04 -05:00
|
|
|
type flag = hashmap[str, ()];
|
2010-08-18 13:34:47 -05:00
|
|
|
|
2010-09-09 17:59:29 -05:00
|
|
|
tag ty_mach {
|
2010-09-21 01:56:43 -05:00
|
|
|
ty_i8;
|
|
|
|
ty_i16;
|
|
|
|
ty_i32;
|
|
|
|
ty_i64;
|
|
|
|
ty_u8;
|
|
|
|
ty_u16;
|
|
|
|
ty_u32;
|
|
|
|
ty_u64;
|
|
|
|
ty_f32;
|
|
|
|
ty_f64;
|
2010-09-09 17:59:29 -05:00
|
|
|
}
|
2010-08-18 13:34:47 -05:00
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
tag ty_or_bang[T] { a_ty(T); a_bang; }
|
2011-05-20 16:15:39 -05:00
|
|
|
|
2010-08-20 13:41:34 -05:00
|
|
|
fn ty_mach_to_str(ty_mach tm) -> str {
|
|
|
|
alt (tm) {
|
2010-09-21 01:56:43 -05:00
|
|
|
case (ty_u8) { ret "u8"; }
|
|
|
|
case (ty_u16) { ret "u16"; }
|
|
|
|
case (ty_u32) { ret "u32"; }
|
|
|
|
case (ty_u64) { ret "u64"; }
|
|
|
|
case (ty_i8) { ret "i8"; }
|
|
|
|
case (ty_i16) { ret "i16"; }
|
|
|
|
case (ty_i32) { ret "i32"; }
|
|
|
|
case (ty_i64) { ret "i64"; }
|
|
|
|
case (ty_f32) { ret "f32"; }
|
|
|
|
case (ty_f64) { ret "f64"; }
|
2010-08-20 13:41:34 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn new_str_hash[V]() -> std::map::hashmap[str, V] {
|
2011-05-17 13:41:41 -05:00
|
|
|
let std::map::hashfn[str] hasher = std::str::hash;
|
|
|
|
let std::map::eqfn[str] eqer = std::str::eq;
|
2011-06-15 13:19:50 -05:00
|
|
|
ret std::map::mk_hashmap[str, V](hasher, eqer);
|
2010-09-21 18:22:32 -05:00
|
|
|
}
|
|
|
|
|
2011-05-12 10:24:54 -05:00
|
|
|
fn def_eq(&ast::def_id a, &ast::def_id b) -> bool {
|
2011-03-08 13:59:38 -06:00
|
|
|
ret a._0 == b._0 && a._1 == b._1;
|
|
|
|
}
|
|
|
|
|
2011-05-12 10:24:54 -05:00
|
|
|
fn hash_def(&ast::def_id d) -> uint {
|
2011-03-31 13:55:28 -05:00
|
|
|
auto h = 5381u;
|
2011-06-15 13:19:50 -05:00
|
|
|
h = (h << 5u) + h ^ (d._0 as uint);
|
|
|
|
h = (h << 5u) + h ^ (d._1 as uint);
|
2011-03-31 13:55:28 -05:00
|
|
|
ret h;
|
|
|
|
}
|
2010-10-19 16:54:10 -05:00
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn new_def_hash[V]() -> std::map::hashmap[ast::def_id, V] {
|
2011-05-12 10:24:54 -05:00
|
|
|
let std::map::hashfn[ast::def_id] hasher = hash_def;
|
|
|
|
let std::map::eqfn[ast::def_id] eqer = def_eq;
|
2011-06-15 13:19:50 -05:00
|
|
|
ret std::map::mk_hashmap[ast::def_id, V](hasher, eqer);
|
2010-10-19 16:54:10 -05:00
|
|
|
}
|
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn new_int_hash[V]() -> std::map::hashmap[int, V] {
|
2011-03-25 12:42:57 -05:00
|
|
|
fn hash_int(&int x) -> uint { ret x as uint; }
|
|
|
|
fn eq_int(&int a, &int b) -> bool { ret a == b; }
|
|
|
|
auto hasher = hash_int;
|
|
|
|
auto eqer = eq_int;
|
2011-06-15 13:19:50 -05:00
|
|
|
ret std::map::mk_hashmap[int, V](hasher, eqer);
|
2011-03-25 12:42:57 -05:00
|
|
|
}
|
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn new_uint_hash[V]() -> std::map::hashmap[uint, V] {
|
2011-04-11 18:54:05 -05:00
|
|
|
fn hash_uint(&uint x) -> uint { ret x; }
|
|
|
|
fn eq_uint(&uint a, &uint b) -> bool { ret a == b; }
|
|
|
|
auto hasher = hash_uint;
|
|
|
|
auto eqer = eq_uint;
|
2011-06-15 13:19:50 -05:00
|
|
|
ret std::map::mk_hashmap[uint, V](hasher, eqer);
|
2011-04-11 18:54:05 -05:00
|
|
|
}
|
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn istr(int i) -> str { ret int::to_str(i, 10u); }
|
2010-09-23 19:16:34 -05:00
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn uistr(uint i) -> str { ret uint::to_str(i, 10u); }
|
2011-04-07 20:15:56 -05:00
|
|
|
|
2011-05-12 10:24:54 -05:00
|
|
|
fn elt_expr(&ast::elt e) -> @ast::expr { ret e.expr; }
|
2011-04-07 20:15:56 -05:00
|
|
|
|
2011-05-12 10:24:54 -05:00
|
|
|
fn elt_exprs(&vec[ast::elt] elts) -> vec[@ast::expr] {
|
2011-04-07 20:15:56 -05:00
|
|
|
auto f = elt_expr;
|
2011-05-17 13:41:41 -05:00
|
|
|
ret vec::map[ast::elt, @ast::expr](f, elts);
|
2011-04-07 20:15:56 -05:00
|
|
|
}
|
|
|
|
|
2011-05-30 17:56:01 -05:00
|
|
|
fn field_expr(&ast::field f) -> @ast::expr { ret f.node.expr; }
|
2011-04-12 14:16:21 -05:00
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn field_exprs(vec[ast::field] fields) -> vec[@ast::expr] {
|
2011-04-12 14:16:21 -05:00
|
|
|
auto f = field_expr;
|
2011-05-17 13:41:41 -05:00
|
|
|
ret vec::map[ast::field, @ast::expr](f, fields);
|
2011-04-12 14:16:21 -05:00
|
|
|
}
|
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn log_expr(&ast::expr e) { log pretty::pprust::expr_to_str(@e); }
|
Further work on typestate_check
Lots of work on typestate_check, seems to get a lot of the way
through checking the standard library.
* Added for, for_each, assign_op, bind, cast, put, check, break,
and cont. (I'm not sure break and cont are actually handled correctly.)
* Fixed side-effect bug in seq_preconds so that unioning the
preconditions of a sequence of statements or expressions
is handled correctly.
* Pass poststate correctly through a stmt_decl.
* Handle expr_ret and expr_fail properly (after execution of a ret
or fail, everything is true -- this is needed to handle ifs and alts
where one branch is a ret or fail)
* Fixed bug in set_prestate_ann where a thing that needed to be
mutated wasn't getting passed as an alias
* Fixed bug in how expr_alt was treated (zero is not the identity
for intersect, who knew, right?)
* Update logging to reflect log_err vs. log
* Fixed find_locals so as to return all local decls and exclude
function arguments.
* Make union_postconds work on an empty vector (needed to handle
empty blocks correctly)
* Added _vec.cat_options, which takes a list of option[T] to a list
of T, ignoring any Nones
* Added two test cases.
2011-04-20 14:11:01 -05:00
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn log_expr_err(&ast::expr e) { log_err pretty::pprust::expr_to_str(@e); }
|
Further work on typestate_check
Lots of work on typestate_check, seems to get a lot of the way
through checking the standard library.
* Added for, for_each, assign_op, bind, cast, put, check, break,
and cont. (I'm not sure break and cont are actually handled correctly.)
* Fixed side-effect bug in seq_preconds so that unioning the
preconditions of a sequence of statements or expressions
is handled correctly.
* Pass poststate correctly through a stmt_decl.
* Handle expr_ret and expr_fail properly (after execution of a ret
or fail, everything is true -- this is needed to handle ifs and alts
where one branch is a ret or fail)
* Fixed bug in set_prestate_ann where a thing that needed to be
mutated wasn't getting passed as an alias
* Fixed bug in how expr_alt was treated (zero is not the identity
for intersect, who knew, right?)
* Update logging to reflect log_err vs. log
* Fixed find_locals so as to return all local decls and exclude
function arguments.
* Make union_postconds work on an empty vector (needed to handle
empty blocks correctly)
* Added _vec.cat_options, which takes a list of option[T] to a list
of T, ignoring any Nones
* Added two test cases.
2011-04-20 14:11:01 -05:00
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn log_ty_err(&ty t) { log_err pretty::pprust::ty_to_str(t); }
|
2011-05-14 21:02:30 -05:00
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn log_pat_err(&@pat p) { log_err pretty::pprust::pat_to_str(p); }
|
2011-05-14 21:02:30 -05:00
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn log_block(&ast::block b) { log pretty::pprust::block_to_str(b); }
|
Further work on typestate_check
Lots of work on typestate_check, seems to get a lot of the way
through checking the standard library.
* Added for, for_each, assign_op, bind, cast, put, check, break,
and cont. (I'm not sure break and cont are actually handled correctly.)
* Fixed side-effect bug in seq_preconds so that unioning the
preconditions of a sequence of statements or expressions
is handled correctly.
* Pass poststate correctly through a stmt_decl.
* Handle expr_ret and expr_fail properly (after execution of a ret
or fail, everything is true -- this is needed to handle ifs and alts
where one branch is a ret or fail)
* Fixed bug in set_prestate_ann where a thing that needed to be
mutated wasn't getting passed as an alias
* Fixed bug in how expr_alt was treated (zero is not the identity
for intersect, who knew, right?)
* Update logging to reflect log_err vs. log
* Fixed find_locals so as to return all local decls and exclude
function arguments.
* Make union_postconds work on an empty vector (needed to handle
empty blocks correctly)
* Added _vec.cat_options, which takes a list of option[T] to a list
of T, ignoring any Nones
* Added two test cases.
2011-04-20 14:11:01 -05:00
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn log_block_err(&ast::block b) { log_err pretty::pprust::block_to_str(b); }
|
2011-04-12 14:16:21 -05:00
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn log_item_err(&@ast::item i) { log_err pretty::pprust::item_to_str(i); }
|
2011-04-22 13:08:47 -05:00
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn log_fn(&ast::_fn f, str name, vec[ast::ty_param] params) {
|
|
|
|
log pretty::pprust::fun_to_str(f, name, params);
|
2011-04-22 13:08:47 -05:00
|
|
|
}
|
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn log_fn_err(&ast::_fn f, str name, vec[ast::ty_param] params) {
|
|
|
|
log_err pretty::pprust::fun_to_str(f, name, params);
|
Further work on typestate_check
Lots of work on typestate_check, seems to get a lot of the way
through checking the standard library.
* Added for, for_each, assign_op, bind, cast, put, check, break,
and cont. (I'm not sure break and cont are actually handled correctly.)
* Fixed side-effect bug in seq_preconds so that unioning the
preconditions of a sequence of statements or expressions
is handled correctly.
* Pass poststate correctly through a stmt_decl.
* Handle expr_ret and expr_fail properly (after execution of a ret
or fail, everything is true -- this is needed to handle ifs and alts
where one branch is a ret or fail)
* Fixed bug in set_prestate_ann where a thing that needed to be
mutated wasn't getting passed as an alias
* Fixed bug in how expr_alt was treated (zero is not the identity
for intersect, who knew, right?)
* Update logging to reflect log_err vs. log
* Fixed find_locals so as to return all local decls and exclude
function arguments.
* Make union_postconds work on an empty vector (needed to handle
empty blocks correctly)
* Added _vec.cat_options, which takes a list of option[T] to a list
of T, ignoring any Nones
* Added two test cases.
2011-04-20 14:11:01 -05:00
|
|
|
}
|
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn log_stmt(&ast::stmt st) { log pretty::pprust::stmt_to_str(st); }
|
Further work on typestate_check
Lots of work on typestate_check, seems to get a lot of the way
through checking the standard library.
* Added for, for_each, assign_op, bind, cast, put, check, break,
and cont. (I'm not sure break and cont are actually handled correctly.)
* Fixed side-effect bug in seq_preconds so that unioning the
preconditions of a sequence of statements or expressions
is handled correctly.
* Pass poststate correctly through a stmt_decl.
* Handle expr_ret and expr_fail properly (after execution of a ret
or fail, everything is true -- this is needed to handle ifs and alts
where one branch is a ret or fail)
* Fixed bug in set_prestate_ann where a thing that needed to be
mutated wasn't getting passed as an alias
* Fixed bug in how expr_alt was treated (zero is not the identity
for intersect, who knew, right?)
* Update logging to reflect log_err vs. log
* Fixed find_locals so as to return all local decls and exclude
function arguments.
* Make union_postconds work on an empty vector (needed to handle
empty blocks correctly)
* Added _vec.cat_options, which takes a list of option[T] to a list
of T, ignoring any Nones
* Added two test cases.
2011-04-20 14:11:01 -05:00
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn log_stmt_err(&ast::stmt st) { log_err pretty::pprust::stmt_to_str(st); }
|
Further work on typestate_check
Lots of work on typestate_check, seems to get a lot of the way
through checking the standard library.
* Added for, for_each, assign_op, bind, cast, put, check, break,
and cont. (I'm not sure break and cont are actually handled correctly.)
* Fixed side-effect bug in seq_preconds so that unioning the
preconditions of a sequence of statements or expressions
is handled correctly.
* Pass poststate correctly through a stmt_decl.
* Handle expr_ret and expr_fail properly (after execution of a ret
or fail, everything is true -- this is needed to handle ifs and alts
where one branch is a ret or fail)
* Fixed bug in set_prestate_ann where a thing that needed to be
mutated wasn't getting passed as an alias
* Fixed bug in how expr_alt was treated (zero is not the identity
for intersect, who knew, right?)
* Update logging to reflect log_err vs. log
* Fixed find_locals so as to return all local decls and exclude
function arguments.
* Make union_postconds work on an empty vector (needed to handle
empty blocks correctly)
* Added _vec.cat_options, which takes a list of option[T] to a list
of T, ignoring any Nones
* Added two test cases.
2011-04-20 14:11:01 -05:00
|
|
|
|
2011-05-12 10:24:54 -05:00
|
|
|
fn has_nonlocal_exits(&ast::block b) -> bool {
|
2011-06-15 13:19:50 -05:00
|
|
|
auto has_exits = @mutable false;
|
|
|
|
fn visit_expr(@mutable bool flag, &@ast::expr e) {
|
|
|
|
alt (e.node) {
|
2011-06-21 15:16:40 -05:00
|
|
|
case (ast::expr_break) { *flag = true; }
|
|
|
|
case (ast::expr_cont) { *flag = true; }
|
2011-06-15 13:19:50 -05:00
|
|
|
case (_) { }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
auto v =
|
|
|
|
rec(visit_expr_pre=bind visit_expr(has_exits, _)
|
|
|
|
with walk::default_visitor());
|
2011-05-26 19:16:54 -05:00
|
|
|
walk::walk_block(v, b);
|
|
|
|
ret *has_exits;
|
Support all expression forms in typestate
Added support for self_method, cont, chan, port, recv, send, be,
do_while, spawn, and ext; handled break and cont correctly.
(However, there are no non-xfailed test cases for ext or spawn in
stage0 currently.)
Although the standard library compiles and all test cases pass with
typestate enabled, I left typestate checking disabled as rustc
terminates abnormally when building the standard library if so,
even though it does generate code correctly.
2011-04-21 19:39:04 -05:00
|
|
|
}
|
2011-05-14 21:02:30 -05:00
|
|
|
|
2011-06-16 17:58:25 -05:00
|
|
|
fn local_rhs_span(&@ast::local l, &span def) -> span {
|
|
|
|
alt (l.node.init) {
|
2011-05-30 23:39:19 -05:00
|
|
|
case (some(?i)) { ret i.expr.span; }
|
2011-05-18 17:38:38 -05:00
|
|
|
case (_) { ret def; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-06-01 20:10:10 -05:00
|
|
|
fn lit_eq(&@ast::lit l, &@ast::lit m) -> bool {
|
|
|
|
alt (l.node) {
|
2011-06-15 13:19:50 -05:00
|
|
|
case (ast::lit_str(?s, ?kind_s)) {
|
2011-06-01 20:10:10 -05:00
|
|
|
alt (m.node) {
|
2011-06-15 13:19:50 -05:00
|
|
|
case (ast::lit_str(?t, ?kind_t)) {
|
2011-06-09 19:11:21 -05:00
|
|
|
ret s == t && kind_s == kind_t;
|
|
|
|
}
|
2011-06-15 13:19:50 -05:00
|
|
|
case (_) { ret false; }
|
2011-06-01 20:10:10 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ast::lit_char(?c)) {
|
|
|
|
alt (m.node) {
|
|
|
|
case (ast::lit_char(?d)) { ret c == d; }
|
2011-06-15 13:19:50 -05:00
|
|
|
case (_) { ret false; }
|
2011-06-01 20:10:10 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ast::lit_int(?i)) {
|
|
|
|
alt (m.node) {
|
|
|
|
case (ast::lit_int(?j)) { ret i == j; }
|
2011-06-15 13:19:50 -05:00
|
|
|
case (_) { ret false; }
|
2011-06-01 20:10:10 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ast::lit_uint(?i)) {
|
|
|
|
alt (m.node) {
|
|
|
|
case (ast::lit_uint(?j)) { ret i == j; }
|
2011-06-15 13:19:50 -05:00
|
|
|
case (_) { ret false; }
|
2011-06-01 20:10:10 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ast::lit_mach_int(_, ?i)) {
|
|
|
|
alt (m.node) {
|
|
|
|
case (ast::lit_mach_int(_, ?j)) { ret i == j; }
|
2011-06-15 13:19:50 -05:00
|
|
|
case (_) { ret false; }
|
2011-06-01 20:10:10 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ast::lit_float(?s)) {
|
|
|
|
alt (m.node) {
|
|
|
|
case (ast::lit_float(?t)) { ret s == t; }
|
2011-06-15 13:19:50 -05:00
|
|
|
case (_) { ret false; }
|
2011-06-01 20:10:10 -05:00
|
|
|
}
|
|
|
|
}
|
2011-06-15 13:19:50 -05:00
|
|
|
case (ast::lit_mach_float(_, ?s)) {
|
2011-06-01 20:10:10 -05:00
|
|
|
alt (m.node) {
|
2011-06-15 13:19:50 -05:00
|
|
|
case (ast::lit_mach_float(_, ?t)) { ret s == t; }
|
|
|
|
case (_) { ret false; }
|
2011-06-01 20:10:10 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ast::lit_nil) {
|
|
|
|
alt (m.node) {
|
|
|
|
case (ast::lit_nil) { ret true; }
|
2011-06-15 13:19:50 -05:00
|
|
|
case (_) { ret false; }
|
2011-06-01 20:10:10 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
case (ast::lit_bool(?b)) {
|
|
|
|
alt (m.node) {
|
|
|
|
case (ast::lit_bool(?c)) { ret b == c; }
|
2011-06-15 13:19:50 -05:00
|
|
|
case (_) { ret false; }
|
2011-06-01 20:10:10 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn respan[T](&span sp, &T t) -> spanned[T] { ret rec(node=t, span=sp); }
|
2011-05-26 19:16:54 -05:00
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn may_begin_ident(char c) -> bool { ret is_alpha(c) || c == '_'; }
|
2011-06-09 11:48:16 -05:00
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn in_range(char c, char lo, char hi) -> bool { ret lo <= c && c <= hi; }
|
2011-06-09 11:48:16 -05:00
|
|
|
|
|
|
|
fn is_alpha(char c) -> bool {
|
2011-06-15 13:19:50 -05:00
|
|
|
ret in_range(c, 'a', 'z') || in_range(c, 'A', 'Z');
|
2011-06-09 11:48:16 -05:00
|
|
|
}
|
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn is_dec_digit(char c) -> bool { ret in_range(c, '0', '9'); }
|
2011-06-09 11:48:16 -05:00
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn is_alnum(char c) -> bool { ret is_alpha(c) || is_dec_digit(c); }
|
2011-06-09 11:48:16 -05:00
|
|
|
|
|
|
|
fn is_hex_digit(char c) -> bool {
|
2011-06-15 13:19:50 -05:00
|
|
|
ret in_range(c, '0', '9') || in_range(c, 'a', 'f') ||
|
|
|
|
in_range(c, 'A', 'F');
|
2011-06-09 11:48:16 -05:00
|
|
|
}
|
|
|
|
|
2011-06-15 13:19:50 -05:00
|
|
|
fn is_bin_digit(char c) -> bool { ret c == '0' || c == '1'; }
|
2010-08-18 13:34:47 -05:00
|
|
|
//
|
|
|
|
// Local Variables:
|
|
|
|
// mode: rust
|
|
|
|
// fill-column: 78;
|
|
|
|
// indent-tabs-mode: nil
|
|
|
|
// c-basic-offset: 4
|
|
|
|
// buffer-file-coding-system: utf-8-unix
|
2011-03-25 17:07:27 -05:00
|
|
|
// compile-command: "make -k -C $RBUILD 2>&1 | sed -e 's/\\/x\\//x:\\//g'";
|
2010-08-18 13:34:47 -05:00
|
|
|
// End:
|
|
|
|
//
|