2011-08-27 23:27:39 -05:00
|
|
|
|
|
|
|
|
|
|
|
/*
|
2011-09-01 19:49:29 -05:00
|
|
|
* The compiler code necessary to support the #fmt extension. Eventually this
|
2011-08-27 23:27:39 -05:00
|
|
|
* should all get sucked into either the standard library extfmt module or the
|
|
|
|
* compiler syntax extension plugin interface.
|
|
|
|
*/
|
2011-12-13 18:25:51 -06:00
|
|
|
import extfmt::ct::*;
|
2011-08-27 23:27:39 -05:00
|
|
|
import base::*;
|
|
|
|
import codemap::span;
|
2012-03-29 15:48:05 -05:00
|
|
|
import ext::build::*;
|
2011-08-27 23:27:39 -05:00
|
|
|
export expand_syntax_ext;
|
|
|
|
|
2012-02-01 00:50:12 -06:00
|
|
|
fn expand_syntax_ext(cx: ext_ctxt, sp: span, arg: ast::mac_arg,
|
2012-01-31 21:30:26 -06:00
|
|
|
_body: ast::mac_body) -> @ast::expr {
|
2012-07-14 00:57:48 -05:00
|
|
|
let args = get_mac_args_no_max(cx, sp, arg, 1u, ~"fmt");
|
2011-08-27 23:27:39 -05:00
|
|
|
let fmt =
|
|
|
|
expr_to_str(cx, args[0],
|
2012-07-14 00:57:48 -05:00
|
|
|
~"first argument to #fmt must be a string literal.");
|
2011-08-27 23:27:39 -05:00
|
|
|
let fmtspan = args[0].span;
|
2012-07-30 18:01:07 -05:00
|
|
|
debug!{"Format string:"};
|
2011-12-22 19:53:53 -06:00
|
|
|
log(debug, fmt);
|
2012-07-14 00:57:48 -05:00
|
|
|
fn parse_fmt_err_(cx: ext_ctxt, sp: span, msg: ~str) -> ! {
|
2011-08-27 23:27:39 -05:00
|
|
|
cx.span_fatal(sp, msg);
|
|
|
|
}
|
2012-07-14 00:57:48 -05:00
|
|
|
let parse_fmt_err = fn@(s: ~str) -> ! {
|
2012-06-19 21:34:01 -05:00
|
|
|
parse_fmt_err_(cx, fmtspan, s)
|
|
|
|
};
|
2011-08-27 23:27:39 -05:00
|
|
|
let pieces = parse_fmt_string(fmt, parse_fmt_err);
|
2012-08-01 19:30:05 -05:00
|
|
|
return pieces_to_expr(cx, sp, pieces, args);
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
|
|
|
|
2012-06-21 18:44:10 -05:00
|
|
|
// FIXME (#2249): A lot of these functions for producing expressions can
|
|
|
|
// probably be factored out in common with other code that builds
|
|
|
|
// expressions. Also: Cleanup the naming of these functions.
|
2012-01-25 15:47:56 -06:00
|
|
|
// NOTE: Moved many of the common ones to build.rs --kevina
|
2012-06-25 22:00:46 -05:00
|
|
|
fn pieces_to_expr(cx: ext_ctxt, sp: span,
|
2012-06-29 18:26:56 -05:00
|
|
|
pieces: ~[piece], args: ~[@ast::expr])
|
2011-09-12 04:27:30 -05:00
|
|
|
-> @ast::expr {
|
2012-06-29 18:26:56 -05:00
|
|
|
fn make_path_vec(_cx: ext_ctxt, ident: ast::ident) -> ~[ast::ident] {
|
2012-08-01 19:30:05 -05:00
|
|
|
return ~[@~"extfmt", @~"rt", ident];
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
2012-06-10 02:49:59 -05:00
|
|
|
fn make_rt_path_expr(cx: ext_ctxt, sp: span,
|
|
|
|
ident: ast::ident) -> @ast::expr {
|
2011-08-27 23:27:39 -05:00
|
|
|
let path = make_path_vec(cx, ident);
|
2012-08-01 19:30:05 -05:00
|
|
|
return mk_path(cx, sp, path);
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
|
|
|
// Produces an AST expression that represents a RT::conv record,
|
|
|
|
// which tells the RT::conv* functions how to perform the conversion
|
|
|
|
|
2011-09-12 04:27:30 -05:00
|
|
|
fn make_rt_conv_expr(cx: ext_ctxt, sp: span, cnv: conv) -> @ast::expr {
|
2012-06-29 18:26:56 -05:00
|
|
|
fn make_flags(cx: ext_ctxt, sp: span, flags: ~[flag]) -> @ast::expr {
|
2012-07-14 00:57:48 -05:00
|
|
|
let mut tmp_expr = make_rt_path_expr(cx, sp, @~"flag_none");
|
2012-06-30 18:19:07 -05:00
|
|
|
for flags.each |f| {
|
2012-06-29 01:36:00 -05:00
|
|
|
let fstr = alt f {
|
2012-08-03 21:59:04 -05:00
|
|
|
flag_left_justify => ~"flag_left_justify",
|
|
|
|
flag_left_zero_pad => ~"flag_left_zero_pad",
|
|
|
|
flag_space_for_sign => ~"flag_space_for_sign",
|
|
|
|
flag_sign_always => ~"flag_sign_always",
|
|
|
|
flag_alternate => ~"flag_alternate"
|
2012-06-29 01:36:00 -05:00
|
|
|
};
|
|
|
|
tmp_expr = mk_binary(cx, sp, ast::bitor, tmp_expr,
|
|
|
|
make_rt_path_expr(cx, sp, @fstr));
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
2012-08-01 19:30:05 -05:00
|
|
|
return tmp_expr;
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
2011-09-12 04:27:30 -05:00
|
|
|
fn make_count(cx: ext_ctxt, sp: span, cnt: count) -> @ast::expr {
|
2011-08-27 23:27:39 -05:00
|
|
|
alt cnt {
|
2012-08-03 21:59:04 -05:00
|
|
|
count_implied => {
|
2012-08-01 19:30:05 -05:00
|
|
|
return make_rt_path_expr(cx, sp, @~"count_implied");
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
count_is(c) => {
|
2012-01-26 17:34:05 -06:00
|
|
|
let count_lit = mk_int(cx, sp, c);
|
2012-07-14 00:57:48 -05:00
|
|
|
let count_is_path = make_path_vec(cx, @~"count_is");
|
2012-06-29 18:26:56 -05:00
|
|
|
let count_is_args = ~[count_lit];
|
2012-08-01 19:30:05 -05:00
|
|
|
return mk_call(cx, sp, count_is_path, count_is_args);
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
_ => cx.span_unimpl(sp, ~"unimplemented #fmt conversion")
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
|
|
|
}
|
2011-09-12 04:27:30 -05:00
|
|
|
fn make_ty(cx: ext_ctxt, sp: span, t: ty) -> @ast::expr {
|
2012-03-15 08:47:03 -05:00
|
|
|
let mut rt_type;
|
2011-08-27 23:27:39 -05:00
|
|
|
alt t {
|
2012-08-03 21:59:04 -05:00
|
|
|
ty_hex(c) => alt c {
|
|
|
|
case_upper => rt_type = ~"ty_hex_upper",
|
|
|
|
case_lower => rt_type = ~"ty_hex_lower"
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
ty_bits => rt_type = ~"ty_bits",
|
|
|
|
ty_octal => rt_type = ~"ty_octal",
|
|
|
|
_ => rt_type = ~"ty_default"
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
2012-08-01 19:30:05 -05:00
|
|
|
return make_rt_path_expr(cx, sp, @rt_type);
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
2011-09-12 04:27:30 -05:00
|
|
|
fn make_conv_rec(cx: ext_ctxt, sp: span, flags_expr: @ast::expr,
|
2011-08-27 23:27:39 -05:00
|
|
|
width_expr: @ast::expr, precision_expr: @ast::expr,
|
|
|
|
ty_expr: @ast::expr) -> @ast::expr {
|
2012-08-01 19:30:05 -05:00
|
|
|
return mk_rec_e(cx, sp,
|
2012-07-14 00:57:48 -05:00
|
|
|
~[{ident: @~"flags", ex: flags_expr},
|
|
|
|
{ident: @~"width", ex: width_expr},
|
|
|
|
{ident: @~"precision", ex: precision_expr},
|
|
|
|
{ident: @~"ty", ex: ty_expr}]);
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
|
|
|
let rt_conv_flags = make_flags(cx, sp, cnv.flags);
|
|
|
|
let rt_conv_width = make_count(cx, sp, cnv.width);
|
|
|
|
let rt_conv_precision = make_count(cx, sp, cnv.precision);
|
|
|
|
let rt_conv_ty = make_ty(cx, sp, cnv.ty);
|
2012-08-01 19:30:05 -05:00
|
|
|
return make_conv_rec(cx, sp, rt_conv_flags, rt_conv_width,
|
2011-08-27 23:27:39 -05:00
|
|
|
rt_conv_precision, rt_conv_ty);
|
|
|
|
}
|
2012-07-14 00:57:48 -05:00
|
|
|
fn make_conv_call(cx: ext_ctxt, sp: span, conv_type: ~str, cnv: conv,
|
2011-09-02 17:34:58 -05:00
|
|
|
arg: @ast::expr) -> @ast::expr {
|
2012-07-14 00:57:48 -05:00
|
|
|
let fname = ~"conv_" + conv_type;
|
2012-06-10 02:49:59 -05:00
|
|
|
let path = make_path_vec(cx, @fname);
|
2011-08-27 23:27:39 -05:00
|
|
|
let cnv_expr = make_rt_conv_expr(cx, sp, cnv);
|
2012-06-29 18:26:56 -05:00
|
|
|
let args = ~[cnv_expr, arg];
|
2012-08-01 19:30:05 -05:00
|
|
|
return mk_call(cx, arg.span, path, args);
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
2012-06-28 18:23:12 -05:00
|
|
|
|
2011-09-12 04:27:30 -05:00
|
|
|
fn make_new_conv(cx: ext_ctxt, sp: span, cnv: conv, arg: @ast::expr) ->
|
2011-08-27 23:27:39 -05:00
|
|
|
@ast::expr {
|
2012-04-19 17:20:11 -05:00
|
|
|
// FIXME: Move validation code into core::extfmt (Issue #2249)
|
2011-08-27 23:27:39 -05:00
|
|
|
|
|
|
|
fn is_signed_type(cnv: conv) -> bool {
|
|
|
|
alt cnv.ty {
|
2012-08-03 21:59:04 -05:00
|
|
|
ty_int(s) => alt s {
|
|
|
|
signed => return true,
|
|
|
|
unsigned => return false
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
ty_float => return true,
|
|
|
|
_ => return false
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
|
|
|
}
|
2012-07-14 00:57:48 -05:00
|
|
|
let unsupported = ~"conversion not supported in #fmt string";
|
2011-08-27 23:27:39 -05:00
|
|
|
alt cnv.param {
|
2012-08-03 21:59:04 -05:00
|
|
|
option::none => (),
|
|
|
|
_ => cx.span_unimpl(sp, unsupported)
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
2012-06-30 18:19:07 -05:00
|
|
|
for cnv.flags.each |f| {
|
2011-08-27 23:27:39 -05:00
|
|
|
alt f {
|
2012-08-03 21:59:04 -05:00
|
|
|
flag_left_justify => (),
|
|
|
|
flag_sign_always => {
|
2011-08-27 23:27:39 -05:00
|
|
|
if !is_signed_type(cnv) {
|
|
|
|
cx.span_fatal(sp,
|
2012-07-14 00:57:48 -05:00
|
|
|
~"+ flag only valid in " +
|
|
|
|
~"signed #fmt conversion");
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
flag_space_for_sign => {
|
2011-08-27 23:27:39 -05:00
|
|
|
if !is_signed_type(cnv) {
|
|
|
|
cx.span_fatal(sp,
|
2012-07-14 00:57:48 -05:00
|
|
|
~"space flag only valid in " +
|
|
|
|
~"signed #fmt conversions");
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
flag_left_zero_pad => (),
|
|
|
|
_ => cx.span_unimpl(sp, unsupported)
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
alt cnv.width {
|
2012-08-03 21:59:04 -05:00
|
|
|
count_implied => (),
|
|
|
|
count_is(_) => (),
|
|
|
|
_ => cx.span_unimpl(sp, unsupported)
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
|
|
|
alt cnv.precision {
|
2012-08-03 21:59:04 -05:00
|
|
|
count_implied => (),
|
|
|
|
count_is(_) => (),
|
|
|
|
_ => cx.span_unimpl(sp, unsupported)
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
|
|
|
alt cnv.ty {
|
2012-08-03 21:59:04 -05:00
|
|
|
ty_str => return make_conv_call(cx, arg.span, ~"str", cnv, arg),
|
|
|
|
ty_int(sign) => alt sign {
|
|
|
|
signed => return make_conv_call(cx, arg.span, ~"int", cnv, arg),
|
|
|
|
unsigned => {
|
|
|
|
return make_conv_call(cx, arg.span, ~"uint", cnv, arg)
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
ty_bool => return make_conv_call(cx, arg.span, ~"bool", cnv, arg),
|
|
|
|
ty_char => return make_conv_call(cx, arg.span, ~"char", cnv, arg),
|
|
|
|
ty_hex(_) => {
|
2012-08-01 19:30:05 -05:00
|
|
|
return make_conv_call(cx, arg.span, ~"uint", cnv, arg);
|
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
ty_bits => return make_conv_call(cx, arg.span, ~"uint", cnv, arg),
|
|
|
|
ty_octal => return make_conv_call(cx, arg.span, ~"uint", cnv, arg),
|
|
|
|
ty_float => {
|
2012-08-01 19:30:05 -05:00
|
|
|
return make_conv_call(cx, arg.span, ~"float", cnv, arg);
|
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
ty_poly => return make_conv_call(cx, arg.span, ~"poly", cnv, arg)
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
fn log_conv(c: conv) {
|
|
|
|
alt c.param {
|
2012-08-03 21:59:04 -05:00
|
|
|
some(p) => { log(debug, ~"param: " + int::to_str(p, 10u)); }
|
|
|
|
_ => debug!{"param: none"}
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
2012-06-30 18:19:07 -05:00
|
|
|
for c.flags.each |f| {
|
2011-08-27 23:27:39 -05:00
|
|
|
alt f {
|
2012-08-03 21:59:04 -05:00
|
|
|
flag_left_justify => debug!{"flag: left justify"},
|
|
|
|
flag_left_zero_pad => debug!{"flag: left zero pad"},
|
|
|
|
flag_space_for_sign => debug!{"flag: left space pad"},
|
|
|
|
flag_sign_always => debug!{"flag: sign always"},
|
|
|
|
flag_alternate => debug!{"flag: alternate"}
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
alt c.width {
|
2012-08-03 21:59:04 -05:00
|
|
|
count_is(i) => log(
|
|
|
|
debug, ~"width: count is " + int::to_str(i, 10u)),
|
|
|
|
count_is_param(i) => log(
|
|
|
|
debug, ~"width: count is param " + int::to_str(i, 10u)),
|
|
|
|
count_is_next_param => debug!{"width: count is next param"},
|
|
|
|
count_implied => debug!{"width: count is implied"}
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
|
|
|
alt c.precision {
|
2012-08-03 21:59:04 -05:00
|
|
|
count_is(i) => log(
|
|
|
|
debug, ~"prec: count is " + int::to_str(i, 10u)),
|
|
|
|
count_is_param(i) => log(
|
|
|
|
debug, ~"prec: count is param " + int::to_str(i, 10u)),
|
|
|
|
count_is_next_param => debug!{"prec: count is next param"},
|
|
|
|
count_implied => debug!{"prec: count is implied"}
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
|
|
|
alt c.ty {
|
2012-08-03 21:59:04 -05:00
|
|
|
ty_bool => debug!{"type: bool"},
|
|
|
|
ty_str => debug!{"type: str"},
|
|
|
|
ty_char => debug!{"type: char"},
|
|
|
|
ty_int(s) => alt s {
|
|
|
|
signed => debug!{"type: signed"},
|
|
|
|
unsigned => debug!{"type: unsigned"}
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
ty_bits => debug!{"type: bits"},
|
|
|
|
ty_hex(cs) => alt cs {
|
|
|
|
case_upper => debug!{"type: uhex"},
|
|
|
|
case_lower => debug!{"type: lhex"},
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
ty_octal => debug!{"type: octal"},
|
|
|
|
ty_float => debug!{"type: float"},
|
|
|
|
ty_poly => debug!{"type: poly"}
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
let fmt_sp = args[0].span;
|
2012-03-15 08:47:03 -05:00
|
|
|
let mut n = 0u;
|
2012-06-29 18:26:56 -05:00
|
|
|
let mut piece_exprs = ~[];
|
2012-06-28 18:23:12 -05:00
|
|
|
let nargs = args.len();
|
2012-06-30 18:19:07 -05:00
|
|
|
for pieces.each |pc| {
|
2011-08-27 23:27:39 -05:00
|
|
|
alt pc {
|
2012-08-03 21:59:04 -05:00
|
|
|
piece_string(s) => {
|
|
|
|
vec::push(piece_exprs, mk_uniq_str(cx, fmt_sp, s))
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
2012-08-03 21:59:04 -05:00
|
|
|
piece_conv(conv) => {
|
2011-08-27 23:27:39 -05:00
|
|
|
n += 1u;
|
|
|
|
if n >= nargs {
|
|
|
|
cx.span_fatal(sp,
|
2012-07-14 00:57:48 -05:00
|
|
|
~"not enough arguments to #fmt " +
|
|
|
|
~"for the given format string");
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
2012-07-30 18:01:07 -05:00
|
|
|
debug!{"Building conversion:"};
|
2011-08-27 23:27:39 -05:00
|
|
|
log_conv(conv);
|
|
|
|
let arg_expr = args[n];
|
|
|
|
let c_expr = make_new_conv(cx, fmt_sp, conv, arg_expr);
|
2012-06-28 18:23:12 -05:00
|
|
|
vec::push(piece_exprs, c_expr);
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let expected_nargs = n + 1u; // n conversions + the fmt string
|
|
|
|
|
|
|
|
if expected_nargs < nargs {
|
2011-09-12 05:39:38 -05:00
|
|
|
cx.span_fatal
|
2012-07-30 18:01:07 -05:00
|
|
|
(sp, fmt!{"too many arguments to #fmt. found %u, expected %u",
|
|
|
|
nargs, expected_nargs});
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
2012-06-28 18:23:12 -05:00
|
|
|
|
|
|
|
let arg_vec = mk_fixed_vec_e(cx, fmt_sp, piece_exprs);
|
2012-08-01 19:30:05 -05:00
|
|
|
return mk_call(cx, fmt_sp, ~[@~"str", @~"concat"], ~[arg_vec]);
|
2011-08-27 23:27:39 -05:00
|
|
|
}
|
|
|
|
//
|
|
|
|
// Local Variables:
|
|
|
|
// mode: rust
|
|
|
|
// fill-column: 78;
|
|
|
|
// indent-tabs-mode: nil
|
|
|
|
// c-basic-offset: 4
|
|
|
|
// buffer-file-coding-system: utf-8-unix
|
|
|
|
// End:
|
|
|
|
//
|