2011-06-15 13:19:50 -05:00
|
|
|
|
|
|
|
|
2011-04-13 19:51:24 -05:00
|
|
|
/*
|
|
|
|
* The compiler code necessary to support the #fmt extension. Eventually this
|
2011-05-12 10:24:54 -05:00
|
|
|
* should all get sucked into either the standard library extfmt module or the
|
2011-04-13 19:51:24 -05:00
|
|
|
* compiler syntax extension plugin interface.
|
2011-02-23 22:48:01 -06:00
|
|
|
*/
|
2011-07-06 21:00:00 -05:00
|
|
|
import std::ivec;
|
2011-05-17 13:41:41 -05:00
|
|
|
import std::str;
|
|
|
|
import std::vec;
|
2011-05-12 10:24:54 -05:00
|
|
|
import std::option;
|
|
|
|
import std::option::none;
|
|
|
|
import std::option::some;
|
2011-06-02 20:58:23 -05:00
|
|
|
import std::extfmt::ct::*;
|
2011-07-05 04:48:19 -05:00
|
|
|
import base::*;
|
|
|
|
import codemap::span;
|
2011-04-10 13:09:47 -05:00
|
|
|
export expand_syntax_ext;
|
2011-02-24 22:22:36 -06:00
|
|
|
|
2011-07-27 07:19:39 -05:00
|
|
|
fn expand_syntax_ext(cx: &ext_ctxt, sp: span, args: &(@ast::expr)[],
|
|
|
|
body: option::t[str]) -> @ast::expr {
|
|
|
|
if ivec::len[@ast::expr](args) == 0u {
|
2011-06-19 00:41:20 -05:00
|
|
|
cx.span_fatal(sp, "#fmt requires a format string");
|
2011-02-24 22:22:36 -06:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
let fmt =
|
|
|
|
expr_to_str(cx, args.(0),
|
|
|
|
"first argument to #fmt must be a " + "string literal.");
|
|
|
|
let fmtspan = args.(0).span;
|
2011-06-04 16:53:17 -05:00
|
|
|
log "Format string:";
|
|
|
|
log fmt;
|
2011-07-27 07:19:39 -05:00
|
|
|
fn parse_fmt_err_(cx: &ext_ctxt, sp: span, msg: str) -> ! {
|
2011-06-19 00:41:20 -05:00
|
|
|
cx.span_fatal(sp, msg);
|
2011-06-04 17:23:06 -05:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
let parse_fmt_err = bind parse_fmt_err_(cx, fmtspan, _);
|
|
|
|
let pieces = parse_fmt_string(fmt, parse_fmt_err);
|
2011-06-04 16:47:36 -05:00
|
|
|
ret pieces_to_expr(cx, sp, pieces, args);
|
2011-02-24 22:22:36 -06:00
|
|
|
}
|
|
|
|
|
2011-04-16 18:43:29 -05:00
|
|
|
// FIXME: A lot of these functions for producing expressions can probably
|
|
|
|
// be factored out in common with other code that builds expressions.
|
|
|
|
// FIXME: Cleanup the naming of these functions
|
2011-07-27 07:19:39 -05:00
|
|
|
fn pieces_to_expr(cx: &ext_ctxt, sp: span, pieces: vec[piece],
|
|
|
|
args: &(@ast::expr)[]) -> @ast::expr {
|
|
|
|
fn make_new_lit(cx: &ext_ctxt, sp: span, lit: ast::lit_) -> @ast::expr {
|
|
|
|
let sp_lit = @{node: lit, span: sp};
|
|
|
|
ret @{id: cx.next_id(), node: ast::expr_lit(sp_lit), span: sp};
|
2011-02-27 17:23:16 -06:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
fn make_new_str(cx: &ext_ctxt, sp: span, s: str) -> @ast::expr {
|
|
|
|
let lit = ast::lit_str(s, ast::sk_rc);
|
2011-06-04 16:39:55 -05:00
|
|
|
ret make_new_lit(cx, sp, lit);
|
2011-02-27 18:44:57 -06:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
fn make_new_int(cx: &ext_ctxt, sp: span, i: int) -> @ast::expr {
|
|
|
|
let lit = ast::lit_int(i);
|
2011-06-04 16:39:55 -05:00
|
|
|
ret make_new_lit(cx, sp, lit);
|
2011-04-16 17:00:52 -05:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
fn make_new_uint(cx: &ext_ctxt, sp: span, u: uint) -> @ast::expr {
|
|
|
|
let lit = ast::lit_uint(u);
|
2011-06-04 16:39:55 -05:00
|
|
|
ret make_new_lit(cx, sp, lit);
|
2011-02-27 18:44:57 -06:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
fn make_add_expr(cx: &ext_ctxt, sp: span, lhs: @ast::expr,
|
|
|
|
rhs: @ast::expr) -> @ast::expr {
|
|
|
|
let binexpr = ast::expr_binary(ast::add, lhs, rhs);
|
|
|
|
ret @{id: cx.next_id(), node: binexpr, span: sp};
|
2011-02-27 17:23:16 -06:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
fn make_path_expr(cx: &ext_ctxt, sp: span, idents: &ast::ident[]) ->
|
|
|
|
@ast::expr {
|
|
|
|
let path = {global: false, idents: idents, types: ~[]};
|
|
|
|
let sp_path = {node: path, span: sp};
|
|
|
|
let pathexpr = ast::expr_path(sp_path);
|
|
|
|
ret @{id: cx.next_id(), node: pathexpr, span: sp};
|
2011-04-13 19:14:59 -05:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
fn make_vec_expr(cx: &ext_ctxt, sp: span, exprs: &(@ast::expr)[]) ->
|
2011-06-15 13:19:50 -05:00
|
|
|
@ast::expr {
|
2011-07-27 07:19:39 -05:00
|
|
|
let vecexpr = ast::expr_vec(exprs, ast::imm, ast::sk_rc);
|
|
|
|
ret @{id: cx.next_id(), node: vecexpr, span: sp};
|
2011-04-16 18:43:29 -05:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
fn make_call(cx: &ext_ctxt, sp: span, fn_path: &ast::ident[],
|
|
|
|
args: &(@ast::expr)[]) -> @ast::expr {
|
|
|
|
let pathexpr = make_path_expr(cx, sp, fn_path);
|
|
|
|
let callexpr = ast::expr_call(pathexpr, args);
|
|
|
|
ret @{id: cx.next_id(), node: callexpr, span: sp};
|
2011-02-27 18:44:57 -06:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
fn make_rec_expr(cx: &ext_ctxt, sp: span,
|
|
|
|
fields: vec[{ident: ast::ident, ex: @ast::expr}]) ->
|
|
|
|
@ast::expr {
|
|
|
|
let astfields: ast::field[] = ~[];
|
|
|
|
for field: {ident: ast::ident, ex: @ast::expr} in fields {
|
|
|
|
let ident = field.ident;
|
|
|
|
let val = field.ex;
|
|
|
|
let astfield =
|
|
|
|
{node: {mut: ast::imm, ident: ident, expr: val}, span: sp};
|
2011-07-06 21:00:00 -05:00
|
|
|
astfields += ~[astfield];
|
2011-04-13 19:14:59 -05:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
let recexpr = ast::expr_rec(astfields, option::none[@ast::expr]);
|
|
|
|
ret @{id: cx.next_id(), node: recexpr, span: sp};
|
2011-04-13 19:14:59 -05:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
fn make_path_vec(cx: &ext_ctxt, ident: str) -> str[] {
|
|
|
|
fn compiling_std(cx: &ext_ctxt) -> bool {
|
2011-07-10 19:00:28 -05:00
|
|
|
ret str::find(cx.crate_file_name_hack, "std.rc") >= 0;
|
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
if compiling_std(cx) {
|
2011-07-10 19:00:28 -05:00
|
|
|
ret ~["extfmt", "rt", ident];
|
2011-07-27 07:19:39 -05:00
|
|
|
} else { ret ~["std", "extfmt", "rt", ident]; }
|
2011-04-13 19:14:59 -05:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
fn make_rt_path_expr(cx: &ext_ctxt, sp: span, ident: str) -> @ast::expr {
|
|
|
|
let path = make_path_vec(cx, ident);
|
2011-06-04 16:39:55 -05:00
|
|
|
ret make_path_expr(cx, sp, path);
|
2011-04-16 18:43:29 -05:00
|
|
|
}
|
2011-05-12 10:24:54 -05:00
|
|
|
// Produces an AST expression that represents a RT::conv record,
|
|
|
|
// which tells the RT::conv* functions how to perform the conversion
|
2011-04-16 17:00:52 -05:00
|
|
|
|
2011-07-27 07:19:39 -05:00
|
|
|
fn make_rt_conv_expr(cx: &ext_ctxt, sp: span, cnv: &conv) -> @ast::expr {
|
|
|
|
fn make_flags(cx: &ext_ctxt, sp: span, flags: vec[flag]) ->
|
2011-06-15 13:19:50 -05:00
|
|
|
@ast::expr {
|
2011-07-27 07:19:39 -05:00
|
|
|
let flagexprs: (@ast::expr)[] = ~[];
|
|
|
|
for f: flag in flags {
|
|
|
|
let fstr;
|
|
|
|
alt f {
|
|
|
|
flag_left_justify. { fstr = "flag_left_justify"; }
|
|
|
|
flag_left_zero_pad. { fstr = "flag_left_zero_pad"; }
|
|
|
|
flag_space_for_sign. { fstr = "flag_space_for_sign"; }
|
|
|
|
flag_sign_always. { fstr = "flag_sign_always"; }
|
|
|
|
flag_alternate. { fstr = "flag_alternate"; }
|
2011-04-16 18:43:29 -05:00
|
|
|
}
|
2011-07-06 21:00:00 -05:00
|
|
|
flagexprs += ~[make_rt_path_expr(cx, sp, fstr)];
|
2011-04-16 18:43:29 -05:00
|
|
|
}
|
|
|
|
// FIXME: 0-length vectors can't have their type inferred
|
|
|
|
// through the rec that these flags are a member of, so
|
|
|
|
// this is a hack placeholder flag
|
2011-06-15 13:19:50 -05:00
|
|
|
|
2011-07-27 07:19:39 -05:00
|
|
|
if ivec::len[@ast::expr](flagexprs) == 0u {
|
2011-07-06 21:00:00 -05:00
|
|
|
flagexprs += ~[make_rt_path_expr(cx, sp, "flag_none")];
|
2011-04-16 18:43:29 -05:00
|
|
|
}
|
2011-06-04 16:39:55 -05:00
|
|
|
ret make_vec_expr(cx, sp, flagexprs);
|
2011-04-16 18:43:29 -05:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
fn make_count(cx: &ext_ctxt, sp: span, cnt: &count) -> @ast::expr {
|
|
|
|
alt cnt {
|
|
|
|
count_implied. {
|
|
|
|
ret make_rt_path_expr(cx, sp, "count_implied");
|
|
|
|
}
|
|
|
|
count_is(c) {
|
|
|
|
let count_lit = make_new_int(cx, sp, c);
|
|
|
|
let count_is_path = make_path_vec(cx, "count_is");
|
|
|
|
let count_is_args = ~[count_lit];
|
|
|
|
ret make_call(cx, sp, count_is_path, count_is_args);
|
|
|
|
}
|
|
|
|
_ { cx.span_unimpl(sp, "unimplemented #fmt conversion"); }
|
2011-04-16 17:00:52 -05:00
|
|
|
}
|
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
fn make_ty(cx: &ext_ctxt, sp: span, t: &ty) -> @ast::expr {
|
|
|
|
let rt_type;
|
|
|
|
alt t {
|
|
|
|
ty_hex(c) {
|
|
|
|
alt c {
|
|
|
|
case_upper. { rt_type = "ty_hex_upper"; }
|
|
|
|
case_lower. { rt_type = "ty_hex_lower"; }
|
2011-04-13 19:14:59 -05:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
}
|
|
|
|
ty_bits. { rt_type = "ty_bits"; }
|
|
|
|
ty_octal. { rt_type = "ty_octal"; }
|
|
|
|
_ { rt_type = "ty_default"; }
|
2011-04-13 19:14:59 -05:00
|
|
|
}
|
2011-06-04 16:39:55 -05:00
|
|
|
ret make_rt_path_expr(cx, sp, rt_type);
|
2011-04-13 19:14:59 -05:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
fn make_conv_rec(cx: &ext_ctxt, sp: span, flags_expr: @ast::expr,
|
|
|
|
width_expr: @ast::expr, precision_expr: @ast::expr,
|
|
|
|
ty_expr: @ast::expr) -> @ast::expr {
|
2011-06-15 13:19:50 -05:00
|
|
|
ret make_rec_expr(cx, sp,
|
2011-07-27 07:19:39 -05:00
|
|
|
[{ident: "flags", ex: flags_expr},
|
|
|
|
{ident: "width", ex: width_expr},
|
|
|
|
{ident: "precision", ex: precision_expr},
|
|
|
|
{ident: "ty", ex: ty_expr}]);
|
2011-04-13 19:14:59 -05:00
|
|
|
}
|
2011-07-27 07:19: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);
|
2011-06-15 13:19:50 -05:00
|
|
|
ret make_conv_rec(cx, sp, rt_conv_flags, rt_conv_width,
|
|
|
|
rt_conv_precision, rt_conv_ty);
|
2011-04-13 19:14:59 -05:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
fn make_conv_call(cx: &ext_ctxt, sp: span, conv_type: str, cnv: &conv,
|
|
|
|
arg: @ast::expr) -> @ast::expr {
|
|
|
|
let fname = "conv_" + conv_type;
|
|
|
|
let path = make_path_vec(cx, fname);
|
|
|
|
let cnv_expr = make_rt_conv_expr(cx, sp, cnv);
|
|
|
|
let args = ~[cnv_expr, arg];
|
2011-06-04 16:39:55 -05:00
|
|
|
ret make_call(cx, arg.span, path, args);
|
2011-04-11 20:55:03 -05:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
fn make_new_conv(cx: &ext_ctxt, sp: span, cnv: conv, arg: @ast::expr) ->
|
|
|
|
@ast::expr {
|
2011-05-12 18:51:22 -05:00
|
|
|
// FIXME: Extract all this validation into extfmt::ct
|
2011-06-15 13:19:50 -05:00
|
|
|
|
2011-07-27 07:19:39 -05:00
|
|
|
fn is_signed_type(cnv: conv) -> bool {
|
|
|
|
alt cnv.ty {
|
|
|
|
ty_int(s) {
|
|
|
|
alt s { signed. { ret true; } unsigned. { ret false; } }
|
|
|
|
}
|
|
|
|
_ { ret false; }
|
2011-04-17 18:50:58 -05:00
|
|
|
}
|
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
let unsupported = "conversion not supported in #fmt string";
|
|
|
|
alt cnv.param {
|
|
|
|
option::none. { }
|
|
|
|
_ { cx.span_unimpl(sp, unsupported); }
|
2011-02-27 18:44:57 -06:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
for f: flag in cnv.flags {
|
|
|
|
alt f {
|
|
|
|
flag_left_justify. { }
|
|
|
|
flag_sign_always. {
|
|
|
|
if !is_signed_type(cnv) {
|
|
|
|
cx.span_fatal(sp,
|
|
|
|
"+ flag only valid in " +
|
|
|
|
"signed #fmt conversion");
|
2011-04-17 18:50:58 -05:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
}
|
|
|
|
flag_space_for_sign. {
|
|
|
|
if !is_signed_type(cnv) {
|
|
|
|
cx.span_fatal(sp,
|
|
|
|
"space flag only valid in " +
|
|
|
|
"signed #fmt conversions");
|
2011-04-17 17:19:26 -05:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
}
|
|
|
|
flag_left_zero_pad. { }
|
|
|
|
_ { cx.span_unimpl(sp, unsupported); }
|
2011-04-16 18:43:29 -05:00
|
|
|
}
|
2011-02-27 18:44:57 -06:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
alt cnv.width {
|
|
|
|
count_implied. { }
|
|
|
|
count_is(_) { }
|
|
|
|
_ { cx.span_unimpl(sp, unsupported); }
|
2011-02-27 18:44:57 -06:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
alt cnv.precision {
|
|
|
|
count_implied. { }
|
|
|
|
count_is(_) { }
|
|
|
|
_ { cx.span_unimpl(sp, unsupported); }
|
2011-02-27 18:44:57 -06:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
alt cnv.ty {
|
|
|
|
ty_str. { ret make_conv_call(cx, arg.span, "str", cnv, arg); }
|
|
|
|
ty_int(sign) {
|
|
|
|
alt sign {
|
|
|
|
signed. { ret make_conv_call(cx, arg.span, "int", cnv, arg); }
|
|
|
|
unsigned. {
|
2011-06-04 16:39:55 -05:00
|
|
|
ret make_conv_call(cx, arg.span, "uint", cnv, arg);
|
2011-07-27 07:19:39 -05:00
|
|
|
}
|
2011-04-26 19:13:23 -05:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
}
|
|
|
|
ty_bool. { ret make_conv_call(cx, arg.span, "bool", cnv, arg); }
|
|
|
|
ty_char. { ret make_conv_call(cx, arg.span, "char", cnv, arg); }
|
|
|
|
ty_hex(_) { ret make_conv_call(cx, arg.span, "uint", cnv, arg); }
|
|
|
|
ty_bits. { ret make_conv_call(cx, arg.span, "uint", cnv, arg); }
|
|
|
|
ty_octal. { ret make_conv_call(cx, arg.span, "uint", cnv, arg); }
|
|
|
|
_ { cx.span_unimpl(sp, unsupported); }
|
2011-02-27 18:44:57 -06:00
|
|
|
}
|
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
fn log_conv(c: conv) {
|
|
|
|
alt c.param {
|
|
|
|
some(p) { log "param: " + std::int::to_str(p, 10u); }
|
|
|
|
_ { log "param: none"; }
|
2011-03-01 20:03:44 -06:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
for f: flag in c.flags {
|
|
|
|
alt f {
|
|
|
|
flag_left_justify. { log "flag: left justify"; }
|
|
|
|
flag_left_zero_pad. { log "flag: left zero pad"; }
|
|
|
|
flag_space_for_sign. { log "flag: left space pad"; }
|
|
|
|
flag_sign_always. { log "flag: sign always"; }
|
|
|
|
flag_alternate. { log "flag: alternate"; }
|
2011-03-01 20:03:44 -06:00
|
|
|
}
|
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
alt c.width {
|
|
|
|
count_is(i) { log "width: count is " + std::int::to_str(i, 10u); }
|
|
|
|
count_is_param(i) {
|
|
|
|
log "width: count is param " + std::int::to_str(i, 10u);
|
|
|
|
}
|
|
|
|
count_is_next_param. { log "width: count is next param"; }
|
|
|
|
count_implied. { log "width: count is implied"; }
|
2011-03-01 20:03:44 -06:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
alt c.precision {
|
|
|
|
count_is(i) { log "prec: count is " + std::int::to_str(i, 10u); }
|
|
|
|
count_is_param(i) {
|
|
|
|
log "prec: count is param " + std::int::to_str(i, 10u);
|
|
|
|
}
|
|
|
|
count_is_next_param. { log "prec: count is next param"; }
|
|
|
|
count_implied. { log "prec: count is implied"; }
|
2011-03-01 20:03:44 -06:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
alt c.ty {
|
|
|
|
ty_bool. { log "type: bool"; }
|
|
|
|
ty_str. { log "type: str"; }
|
|
|
|
ty_char. { log "type: char"; }
|
|
|
|
ty_int(s) {
|
|
|
|
alt s {
|
|
|
|
signed. { log "type: signed"; }
|
|
|
|
unsigned. { log "type: unsigned"; }
|
2011-03-01 20:03:44 -06:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
}
|
|
|
|
ty_bits. { log "type: bits"; }
|
|
|
|
ty_hex(cs) {
|
|
|
|
alt cs {
|
|
|
|
case_upper. { log "type: uhex"; }
|
|
|
|
case_lower. { log "type: lhex"; }
|
2011-03-01 20:03:44 -06:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
}
|
|
|
|
ty_octal. { log "type: octal"; }
|
2011-03-01 20:03:44 -06:00
|
|
|
}
|
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
let fmt_sp = args.(0).span;
|
|
|
|
let n = 0u;
|
|
|
|
let tmp_expr = make_new_str(cx, sp, "");
|
|
|
|
let nargs = ivec::len[@ast::expr](args);
|
|
|
|
for pc: piece in pieces {
|
|
|
|
alt pc {
|
|
|
|
piece_string(s) {
|
|
|
|
let s_expr = make_new_str(cx, fmt_sp, s);
|
|
|
|
tmp_expr = make_add_expr(cx, fmt_sp, tmp_expr, s_expr);
|
|
|
|
}
|
|
|
|
piece_conv(conv) {
|
|
|
|
n += 1u;
|
|
|
|
if n >= nargs {
|
|
|
|
cx.span_fatal(sp,
|
|
|
|
"not enough arguments to #fmt " +
|
|
|
|
"for the given format string");
|
2011-02-27 17:23:16 -06:00
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
log "Building conversion:";
|
|
|
|
log_conv(conv);
|
|
|
|
let arg_expr = args.(n);
|
|
|
|
let c_expr = make_new_conv(cx, fmt_sp, conv, arg_expr);
|
|
|
|
tmp_expr = make_add_expr(cx, fmt_sp, tmp_expr, c_expr);
|
|
|
|
}
|
2011-02-27 17:23:16 -06:00
|
|
|
}
|
|
|
|
}
|
2011-07-27 07:19:39 -05:00
|
|
|
let expected_nargs = n + 1u; // n conversions + the fmt string
|
2011-06-15 13:19:50 -05:00
|
|
|
|
2011-07-27 07:19:39 -05:00
|
|
|
if expected_nargs < nargs {
|
2011-07-27 07:48:34 -05:00
|
|
|
cx.span_fatal
|
|
|
|
(sp, #fmt("too many arguments to #fmt. found %u, expected %u",
|
|
|
|
nargs, expected_nargs));
|
2011-05-25 22:42:50 -05:00
|
|
|
}
|
2011-03-01 20:03:44 -06:00
|
|
|
ret tmp_expr;
|
2011-02-24 22:22:36 -06:00
|
|
|
}
|
2011-02-23 22:48:01 -06: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'";
|
2011-02-23 22:48:01 -06:00
|
|
|
// End:
|
|
|
|
//
|