2012-12-03 16:48:01 -08: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.
|
|
|
|
|
2011-08-27 21:27:39 -07:00
|
|
|
|
|
|
|
|
|
|
|
/*
|
2012-10-12 12:32:36 -07:00
|
|
|
* The compiler code necessary to support the fmt! extension. Eventually this
|
2011-08-27 21:27:39 -07:00
|
|
|
* should all get sucked into either the standard library extfmt module or the
|
|
|
|
* compiler syntax extension plugin interface.
|
|
|
|
*/
|
2012-12-23 17:41:37 -05:00
|
|
|
|
2013-01-08 19:37:25 -08:00
|
|
|
use core::prelude::*;
|
|
|
|
|
2012-12-23 17:41:37 -05:00
|
|
|
use ast;
|
2012-09-04 11:37:29 -07:00
|
|
|
use codemap::span;
|
2012-12-23 17:41:37 -05:00
|
|
|
use ext::base::*;
|
|
|
|
use ext::base;
|
2013-02-25 14:11:21 -05:00
|
|
|
use ext::build;
|
2012-09-04 11:37:29 -07:00
|
|
|
use ext::build::*;
|
2013-03-01 10:44:43 -08:00
|
|
|
|
|
|
|
use core::unstable::extfmt::ct::*;
|
2012-12-23 17:41:37 -05:00
|
|
|
|
2013-03-12 13:00:50 -07:00
|
|
|
pub fn expand_syntax_ext(cx: @ext_ctxt, sp: span, tts: &[ast::token_tree])
|
2013-01-22 16:45:27 -08:00
|
|
|
-> base::MacResult {
|
2013-02-18 21:25:44 -08:00
|
|
|
let args = get_exprs_from_tts(cx, tts);
|
2012-12-12 17:08:09 -08:00
|
|
|
if args.len() == 0 {
|
|
|
|
cx.span_fatal(sp, "fmt! takes at least 1 argument.");
|
|
|
|
}
|
2011-08-27 21:27:39 -07:00
|
|
|
let fmt =
|
|
|
|
expr_to_str(cx, args[0],
|
2012-10-12 12:32:36 -07:00
|
|
|
~"first argument to fmt! must be a string literal.");
|
2011-08-27 21:27:39 -07:00
|
|
|
let fmtspan = args[0].span;
|
2013-03-08 12:39:42 -08:00
|
|
|
debug!("Format string: %s", fmt);
|
2013-03-12 13:00:50 -07:00
|
|
|
fn parse_fmt_err_(cx: @ext_ctxt, sp: span, msg: &str) -> ! {
|
2011-08-27 21:27:39 -07:00
|
|
|
cx.span_fatal(sp, msg);
|
|
|
|
}
|
2013-03-01 13:30:06 -08:00
|
|
|
let parse_fmt_err: @fn(&str) -> ! = |s| parse_fmt_err_(cx, fmtspan, s);
|
2011-08-27 21:27:39 -07:00
|
|
|
let pieces = parse_fmt_string(fmt, parse_fmt_err);
|
2013-01-22 16:45:27 -08:00
|
|
|
MRExpr(pieces_to_expr(cx, sp, pieces, args))
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
|
|
|
|
2012-06-21 16:44:10 -07: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.
|
2013-01-29 16:18:59 -08:00
|
|
|
// Note: Moved many of the common ones to build.rs --kevina
|
2013-03-12 13:00:50 -07:00
|
|
|
fn pieces_to_expr(cx: @ext_ctxt, sp: span,
|
2012-09-11 19:37:29 -07:00
|
|
|
pieces: ~[Piece], args: ~[@ast::expr])
|
2011-09-12 11:27:30 +02:00
|
|
|
-> @ast::expr {
|
2013-03-12 13:00:50 -07:00
|
|
|
fn make_path_vec(cx: @ext_ctxt, ident: @~str) -> ~[ast::ident] {
|
2013-01-24 10:33:20 -08:00
|
|
|
let intr = cx.parse_sess().interner;
|
2013-02-27 19:53:31 -08:00
|
|
|
return ~[intr.intern(@~"unstable"), intr.intern(@~"extfmt"),
|
2013-02-20 21:08:25 +01:00
|
|
|
intr.intern(@~"rt"), intr.intern(ident)];
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
2013-03-12 13:00:50 -07:00
|
|
|
fn make_rt_path_expr(cx: @ext_ctxt, sp: span, nm: @~str) -> @ast::expr {
|
2012-07-18 16:18:02 -07:00
|
|
|
let path = make_path_vec(cx, nm);
|
2012-12-23 17:41:37 -05:00
|
|
|
return mk_path_global(cx, sp, path);
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
|
|
|
// Produces an AST expression that represents a RT::conv record,
|
|
|
|
// which tells the RT::conv* functions how to perform the conversion
|
|
|
|
|
2013-03-19 21:24:01 -04:00
|
|
|
fn make_rt_conv_expr(cx: @ext_ctxt, sp: span, cnv: &Conv) -> @ast::expr {
|
2013-03-12 13:00:50 -07:00
|
|
|
fn make_flags(cx: @ext_ctxt, sp: span, flags: ~[Flag]) -> @ast::expr {
|
2012-07-13 22:57:48 -07:00
|
|
|
let mut tmp_expr = make_rt_path_expr(cx, sp, @~"flag_none");
|
2012-06-30 16:19:07 -07:00
|
|
|
for flags.each |f| {
|
2012-09-19 16:55:01 -07:00
|
|
|
let fstr = match *f {
|
2012-09-11 19:37:29 -07:00
|
|
|
FlagLeftJustify => ~"flag_left_justify",
|
|
|
|
FlagLeftZeroPad => ~"flag_left_zero_pad",
|
|
|
|
FlagSpaceForSign => ~"flag_space_for_sign",
|
|
|
|
FlagSignAlways => ~"flag_sign_always",
|
|
|
|
FlagAlternate => ~"flag_alternate"
|
2012-06-28 23:36:00 -07:00
|
|
|
};
|
|
|
|
tmp_expr = mk_binary(cx, sp, ast::bitor, tmp_expr,
|
|
|
|
make_rt_path_expr(cx, sp, @fstr));
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
2012-08-01 17:30:05 -07:00
|
|
|
return tmp_expr;
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
2013-03-12 13:00:50 -07:00
|
|
|
fn make_count(cx: @ext_ctxt, sp: span, cnt: Count) -> @ast::expr {
|
2012-08-06 12:34:08 -07:00
|
|
|
match cnt {
|
2012-09-11 19:37:29 -07:00
|
|
|
CountImplied => {
|
|
|
|
return make_rt_path_expr(cx, sp, @~"CountImplied");
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
2012-09-11 19:37:29 -07:00
|
|
|
CountIs(c) => {
|
2012-11-01 16:47:24 -07:00
|
|
|
let count_lit = mk_uint(cx, sp, c as uint);
|
2012-09-11 19:37:29 -07:00
|
|
|
let count_is_path = make_path_vec(cx, @~"CountIs");
|
2012-06-29 16:26:56 -07:00
|
|
|
let count_is_args = ~[count_lit];
|
2012-12-23 17:41:37 -05:00
|
|
|
return mk_call_global(cx, sp, count_is_path, count_is_args);
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
2012-10-12 12:32:36 -07:00
|
|
|
_ => cx.span_unimpl(sp, ~"unimplemented fmt! conversion")
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
|
|
|
}
|
2013-03-12 13:00:50 -07:00
|
|
|
fn make_ty(cx: @ext_ctxt, sp: span, t: Ty) -> @ast::expr {
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut rt_type;
|
2012-08-06 12:34:08 -07:00
|
|
|
match t {
|
2012-09-11 19:37:29 -07:00
|
|
|
TyHex(c) => match c {
|
|
|
|
CaseUpper => rt_type = ~"TyHexUpper",
|
|
|
|
CaseLower => rt_type = ~"TyHexLower"
|
2012-08-06 17:14:32 -07:00
|
|
|
},
|
2012-09-11 19:37:29 -07:00
|
|
|
TyBits => rt_type = ~"TyBits",
|
|
|
|
TyOctal => rt_type = ~"TyOctal",
|
|
|
|
_ => rt_type = ~"TyDefault"
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
2012-08-01 17:30:05 -07:00
|
|
|
return make_rt_path_expr(cx, sp, @rt_type);
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
2013-03-12 13:00:50 -07:00
|
|
|
fn make_conv_struct(cx: @ext_ctxt, sp: span, flags_expr: @ast::expr,
|
2011-08-27 21:27:39 -07:00
|
|
|
width_expr: @ast::expr, precision_expr: @ast::expr,
|
|
|
|
ty_expr: @ast::expr) -> @ast::expr {
|
2012-07-18 16:18:02 -07:00
|
|
|
let intr = cx.parse_sess().interner;
|
2013-01-24 10:33:20 -08:00
|
|
|
mk_global_struct_e(
|
|
|
|
cx,
|
|
|
|
sp,
|
|
|
|
make_path_vec(cx, @~"Conv"),
|
|
|
|
~[
|
2013-02-21 00:16:31 -08:00
|
|
|
build::Field {
|
|
|
|
ident: intr.intern(@~"flags"), ex: flags_expr
|
|
|
|
},
|
|
|
|
build::Field {
|
|
|
|
ident: intr.intern(@~"width"), ex: width_expr
|
|
|
|
},
|
|
|
|
build::Field {
|
|
|
|
ident: intr.intern(@~"precision"), ex: precision_expr
|
|
|
|
},
|
|
|
|
build::Field {
|
|
|
|
ident: intr.intern(@~"ty"), ex: ty_expr
|
|
|
|
},
|
2013-01-24 10:33:20 -08:00
|
|
|
]
|
|
|
|
)
|
2011-08-27 21:27:39 -07: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);
|
2013-01-24 10:33:20 -08:00
|
|
|
make_conv_struct(cx, sp, rt_conv_flags, rt_conv_width,
|
|
|
|
rt_conv_precision, rt_conv_ty)
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
2013-03-19 21:24:01 -04:00
|
|
|
fn make_conv_call(cx: @ext_ctxt, sp: span, conv_type: ~str, cnv: &Conv,
|
2011-09-02 15:34:58 -07:00
|
|
|
arg: @ast::expr) -> @ast::expr {
|
2012-07-13 22:57:48 -07:00
|
|
|
let fname = ~"conv_" + conv_type;
|
2012-06-10 00:49:59 -07:00
|
|
|
let path = make_path_vec(cx, @fname);
|
2011-08-27 21:27:39 -07:00
|
|
|
let cnv_expr = make_rt_conv_expr(cx, sp, cnv);
|
2012-06-29 16:26:56 -07:00
|
|
|
let args = ~[cnv_expr, arg];
|
2012-12-23 17:41:37 -05:00
|
|
|
return mk_call_global(cx, arg.span, path, args);
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
2012-06-28 16:23:12 -07:00
|
|
|
|
2013-03-19 21:24:01 -04:00
|
|
|
fn make_new_conv(cx: @ext_ctxt, sp: span, cnv: &Conv, arg: @ast::expr) ->
|
2011-08-27 21:27:39 -07:00
|
|
|
@ast::expr {
|
2012-04-19 15:20:11 -07:00
|
|
|
// FIXME: Move validation code into core::extfmt (Issue #2249)
|
2011-08-27 21:27:39 -07:00
|
|
|
|
2013-03-19 21:24:01 -04:00
|
|
|
fn is_signed_type(cnv: &Conv) -> bool {
|
2012-08-06 12:34:08 -07:00
|
|
|
match cnv.ty {
|
2012-09-11 19:37:29 -07:00
|
|
|
TyInt(s) => match s {
|
|
|
|
Signed => return true,
|
|
|
|
Unsigned => return false
|
2012-08-06 17:14:32 -07:00
|
|
|
},
|
2012-09-11 19:37:29 -07:00
|
|
|
TyFloat => return true,
|
2012-08-03 19:59:04 -07:00
|
|
|
_ => return false
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
|
|
|
}
|
2012-10-12 12:32:36 -07:00
|
|
|
let unsupported = ~"conversion not supported in fmt! string";
|
2012-08-06 12:34:08 -07:00
|
|
|
match cnv.param {
|
2012-08-20 12:23:37 -07:00
|
|
|
option::None => (),
|
2012-08-03 19:59:04 -07:00
|
|
|
_ => cx.span_unimpl(sp, unsupported)
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
2012-06-30 16:19:07 -07:00
|
|
|
for cnv.flags.each |f| {
|
2012-09-19 16:55:01 -07:00
|
|
|
match *f {
|
2012-09-11 19:37:29 -07:00
|
|
|
FlagLeftJustify => (),
|
|
|
|
FlagSignAlways => {
|
2011-08-27 21:27:39 -07:00
|
|
|
if !is_signed_type(cnv) {
|
|
|
|
cx.span_fatal(sp,
|
2012-07-13 22:57:48 -07:00
|
|
|
~"+ flag only valid in " +
|
2012-10-12 12:32:36 -07:00
|
|
|
~"signed fmt! conversion");
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
|
|
|
}
|
2012-09-11 19:37:29 -07:00
|
|
|
FlagSpaceForSign => {
|
2011-08-27 21:27:39 -07:00
|
|
|
if !is_signed_type(cnv) {
|
|
|
|
cx.span_fatal(sp,
|
2012-07-13 22:57:48 -07:00
|
|
|
~"space flag only valid in " +
|
2012-10-12 12:32:36 -07:00
|
|
|
~"signed fmt! conversions");
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
|
|
|
}
|
2012-09-11 19:37:29 -07:00
|
|
|
FlagLeftZeroPad => (),
|
2012-08-03 19:59:04 -07:00
|
|
|
_ => cx.span_unimpl(sp, unsupported)
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
|
|
|
}
|
2012-08-06 12:34:08 -07:00
|
|
|
match cnv.width {
|
2012-09-11 19:37:29 -07:00
|
|
|
CountImplied => (),
|
|
|
|
CountIs(_) => (),
|
2012-08-03 19:59:04 -07:00
|
|
|
_ => cx.span_unimpl(sp, unsupported)
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
2012-08-06 12:34:08 -07:00
|
|
|
match cnv.precision {
|
2012-09-11 19:37:29 -07:00
|
|
|
CountImplied => (),
|
|
|
|
CountIs(_) => (),
|
2012-08-03 19:59:04 -07:00
|
|
|
_ => cx.span_unimpl(sp, unsupported)
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
2012-08-06 12:34:08 -07:00
|
|
|
match cnv.ty {
|
2012-09-11 19:37:29 -07:00
|
|
|
TyStr => return make_conv_call(cx, arg.span, ~"str", cnv, arg),
|
|
|
|
TyInt(sign) => match sign {
|
|
|
|
Signed => return make_conv_call(cx, arg.span, ~"int", cnv, arg),
|
|
|
|
Unsigned => {
|
2012-08-03 19:59:04 -07:00
|
|
|
return make_conv_call(cx, arg.span, ~"uint", cnv, arg)
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
2012-08-06 17:14:32 -07:00
|
|
|
},
|
2012-09-11 19:37:29 -07:00
|
|
|
TyBool => return make_conv_call(cx, arg.span, ~"bool", cnv, arg),
|
|
|
|
TyChar => return make_conv_call(cx, arg.span, ~"char", cnv, arg),
|
|
|
|
TyHex(_) => {
|
2012-08-01 17:30:05 -07:00
|
|
|
return make_conv_call(cx, arg.span, ~"uint", cnv, arg);
|
|
|
|
}
|
2012-09-11 19:37:29 -07:00
|
|
|
TyBits => return make_conv_call(cx, arg.span, ~"uint", cnv, arg),
|
|
|
|
TyOctal => return make_conv_call(cx, arg.span, ~"uint", cnv, arg),
|
|
|
|
TyFloat => {
|
2012-08-01 17:30:05 -07:00
|
|
|
return make_conv_call(cx, arg.span, ~"float", cnv, arg);
|
|
|
|
}
|
2012-09-28 12:22:33 -07:00
|
|
|
TyPoly => return make_conv_call(cx, arg.span, ~"poly", cnv,
|
2012-10-01 12:47:02 -07:00
|
|
|
mk_addr_of(cx, sp, arg))
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
|
|
|
}
|
2013-03-19 21:24:01 -04:00
|
|
|
fn log_conv(c: &Conv) {
|
2012-08-06 12:34:08 -07:00
|
|
|
match c.param {
|
2013-03-08 12:39:42 -08:00
|
|
|
Some(p) => { debug!("param: %s", p.to_str()); }
|
2012-08-22 17:24:52 -07:00
|
|
|
_ => debug!("param: none")
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
2012-06-30 16:19:07 -07:00
|
|
|
for c.flags.each |f| {
|
2012-09-19 16:55:01 -07:00
|
|
|
match *f {
|
2012-09-11 19:37:29 -07:00
|
|
|
FlagLeftJustify => debug!("flag: left justify"),
|
|
|
|
FlagLeftZeroPad => debug!("flag: left zero pad"),
|
|
|
|
FlagSpaceForSign => debug!("flag: left space pad"),
|
|
|
|
FlagSignAlways => debug!("flag: sign always"),
|
|
|
|
FlagAlternate => debug!("flag: alternate")
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
|
|
|
}
|
2012-08-06 12:34:08 -07:00
|
|
|
match c.width {
|
2013-03-08 12:39:42 -08:00
|
|
|
CountIs(i) =>
|
|
|
|
debug!("width: count is %s", i.to_str()),
|
|
|
|
CountIsParam(i) =>
|
|
|
|
debug!("width: count is param %s", i.to_str()),
|
2012-09-11 19:37:29 -07:00
|
|
|
CountIsNextParam => debug!("width: count is next param"),
|
|
|
|
CountImplied => debug!("width: count is implied")
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
2012-08-06 12:34:08 -07:00
|
|
|
match c.precision {
|
2013-03-08 12:39:42 -08:00
|
|
|
CountIs(i) =>
|
|
|
|
debug!("prec: count is %s", i.to_str()),
|
|
|
|
CountIsParam(i) =>
|
|
|
|
debug!("prec: count is param %s", i.to_str()),
|
2012-09-11 19:37:29 -07:00
|
|
|
CountIsNextParam => debug!("prec: count is next param"),
|
|
|
|
CountImplied => debug!("prec: count is implied")
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
2012-08-06 12:34:08 -07:00
|
|
|
match c.ty {
|
2012-09-11 19:37:29 -07:00
|
|
|
TyBool => debug!("type: bool"),
|
|
|
|
TyStr => debug!("type: str"),
|
|
|
|
TyChar => debug!("type: char"),
|
|
|
|
TyInt(s) => match s {
|
|
|
|
Signed => debug!("type: signed"),
|
|
|
|
Unsigned => debug!("type: unsigned")
|
2012-08-06 17:14:32 -07:00
|
|
|
},
|
2012-09-11 19:37:29 -07:00
|
|
|
TyBits => debug!("type: bits"),
|
|
|
|
TyHex(cs) => match cs {
|
|
|
|
CaseUpper => debug!("type: uhex"),
|
|
|
|
CaseLower => debug!("type: lhex"),
|
2012-08-06 17:14:32 -07:00
|
|
|
},
|
2012-09-11 19:37:29 -07:00
|
|
|
TyOctal => debug!("type: octal"),
|
|
|
|
TyFloat => debug!("type: float"),
|
|
|
|
TyPoly => debug!("type: poly")
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
let fmt_sp = args[0].span;
|
2012-03-15 09:47:03 -04:00
|
|
|
let mut n = 0u;
|
2012-06-29 16:26:56 -07:00
|
|
|
let mut piece_exprs = ~[];
|
2012-06-28 16:23:12 -07:00
|
|
|
let nargs = args.len();
|
2012-06-30 16:19:07 -07:00
|
|
|
for pieces.each |pc| {
|
2012-09-19 16:55:01 -07:00
|
|
|
match *pc {
|
2012-12-04 10:50:00 -08:00
|
|
|
PieceString(ref s) => {
|
2013-02-24 21:27:51 -08:00
|
|
|
piece_exprs.push(mk_uniq_str(cx, fmt_sp, copy *s))
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
2013-02-24 21:27:51 -08:00
|
|
|
PieceConv(ref conv) => {
|
2011-08-27 21:27:39 -07:00
|
|
|
n += 1u;
|
|
|
|
if n >= nargs {
|
|
|
|
cx.span_fatal(sp,
|
2012-10-12 12:32:36 -07:00
|
|
|
~"not enough arguments to fmt! " +
|
2012-07-13 22:57:48 -07:00
|
|
|
~"for the given format string");
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
2012-08-22 17:24:52 -07:00
|
|
|
debug!("Building conversion:");
|
2013-03-19 21:24:01 -04:00
|
|
|
log_conv(conv);
|
2011-08-27 21:27:39 -07:00
|
|
|
let arg_expr = args[n];
|
2013-02-24 21:27:51 -08:00
|
|
|
let c_expr = make_new_conv(
|
|
|
|
cx,
|
|
|
|
fmt_sp,
|
2013-03-19 21:24:01 -04:00
|
|
|
conv,
|
2013-02-24 21:27:51 -08:00
|
|
|
arg_expr
|
|
|
|
);
|
2012-09-26 17:33:34 -07:00
|
|
|
piece_exprs.push(c_expr);
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let expected_nargs = n + 1u; // n conversions + the fmt string
|
|
|
|
|
|
|
|
if expected_nargs < nargs {
|
2011-09-12 12:39:38 +02:00
|
|
|
cx.span_fatal
|
2012-10-12 12:32:36 -07:00
|
|
|
(sp, fmt!("too many arguments to fmt!. found %u, expected %u",
|
2012-08-22 17:24:52 -07:00
|
|
|
nargs, expected_nargs));
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
2012-06-28 16:23:12 -07:00
|
|
|
|
|
|
|
let arg_vec = mk_fixed_vec_e(cx, fmt_sp, piece_exprs);
|
2012-12-23 17:41:37 -05:00
|
|
|
return mk_call_global(cx,
|
|
|
|
fmt_sp,
|
|
|
|
~[cx.parse_sess().interner.intern(@~"str"),
|
|
|
|
cx.parse_sess().interner.intern(@~"concat")],
|
|
|
|
~[arg_vec]);
|
2011-08-27 21:27:39 -07:00
|
|
|
}
|
|
|
|
//
|
|
|
|
// Local Variables:
|
|
|
|
// mode: rust
|
|
|
|
// fill-column: 78;
|
|
|
|
// indent-tabs-mode: nil
|
|
|
|
// c-basic-offset: 4
|
|
|
|
// buffer-file-coding-system: utf-8-unix
|
|
|
|
// End:
|
|
|
|
//
|