2012-07-05 23:07:53 +00:00
|
|
|
// Functions for building ASTs, without having to fuss with spans.
|
|
|
|
//
|
|
|
|
// To start with, it will be use dummy spans, but it might someday do
|
|
|
|
// something smarter.
|
|
|
|
|
|
|
|
import ast::{ident, node_id};
|
|
|
|
import codemap::span;
|
|
|
|
import ext::base::mk_ctxt;
|
|
|
|
|
2012-07-24 01:50:53 +00:00
|
|
|
// Transitional reexports so qquote can find the paths it is looking for
|
|
|
|
mod syntax {
|
|
|
|
import ext;
|
|
|
|
export ext;
|
|
|
|
import parse;
|
|
|
|
export parse;
|
|
|
|
}
|
|
|
|
|
2012-07-18 23:18:02 +00:00
|
|
|
fn path(ids: ~[ident], span: span) -> @ast::path {
|
2012-07-24 23:58:48 +00:00
|
|
|
@{span: span,
|
2012-07-05 23:07:53 +00:00
|
|
|
global: false,
|
2012-07-18 23:18:02 +00:00
|
|
|
idents: ids,
|
2012-08-20 19:23:37 +00:00
|
|
|
rp: None,
|
2012-07-05 23:07:53 +00:00
|
|
|
types: ~[]}
|
|
|
|
}
|
|
|
|
|
2012-07-24 23:58:48 +00:00
|
|
|
fn empty_span() -> span {
|
2012-08-20 19:23:37 +00:00
|
|
|
{lo: 0, hi: 0, expn_info: None}
|
2012-07-24 23:58:48 +00:00
|
|
|
}
|
|
|
|
|
2012-07-11 22:00:40 +00:00
|
|
|
trait append_types {
|
|
|
|
fn add_ty(ty: @ast::ty) -> @ast::path;
|
|
|
|
fn add_tys(+tys: ~[@ast::ty]) -> @ast::path;
|
|
|
|
}
|
2012-07-05 23:07:53 +00:00
|
|
|
|
2012-08-08 01:10:06 +00:00
|
|
|
impl @ast::path: append_types {
|
2012-07-05 23:07:53 +00:00
|
|
|
fn add_ty(ty: @ast::ty) -> @ast::path {
|
|
|
|
@{types: vec::append_one(self.types, ty)
|
|
|
|
with *self}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn add_tys(+tys: ~[@ast::ty]) -> @ast::path {
|
|
|
|
@{types: vec::append(self.types, tys)
|
|
|
|
with *self}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-11 22:00:40 +00:00
|
|
|
trait ext_ctxt_ast_builder {
|
|
|
|
fn ty_param(id: ast::ident, +bounds: ~[ast::ty_param_bound])
|
|
|
|
-> ast::ty_param;
|
|
|
|
fn arg(name: ident, ty: @ast::ty) -> ast::arg;
|
|
|
|
fn arg_mode(name: ident, ty: @ast::ty, mode: ast::rmode) -> ast::arg;
|
|
|
|
fn expr_block(e: @ast::expr) -> ast::blk;
|
|
|
|
fn fn_decl(+inputs: ~[ast::arg], output: @ast::ty) -> ast::fn_decl;
|
2012-08-17 18:26:35 +00:00
|
|
|
fn item(name: ident, span: span, +node: ast::item_) -> @ast::item;
|
2012-07-11 22:00:40 +00:00
|
|
|
fn item_fn_poly(name: ident,
|
|
|
|
+inputs: ~[ast::arg],
|
|
|
|
output: @ast::ty,
|
|
|
|
+ty_params: ~[ast::ty_param],
|
|
|
|
+body: ast::blk) -> @ast::item;
|
|
|
|
fn item_fn(name: ident,
|
|
|
|
+inputs: ~[ast::arg],
|
|
|
|
output: @ast::ty,
|
|
|
|
+body: ast::blk) -> @ast::item;
|
|
|
|
fn item_enum_poly(name: ident,
|
2012-08-17 18:26:35 +00:00
|
|
|
span: span,
|
2012-08-09 00:14:25 +00:00
|
|
|
+enum_definition: ast::enum_def,
|
2012-07-11 22:00:40 +00:00
|
|
|
+ty_params: ~[ast::ty_param]) -> @ast::item;
|
2012-08-17 18:47:01 +00:00
|
|
|
fn item_enum(name: ident, span: span,
|
2012-08-17 18:26:35 +00:00
|
|
|
+enum_definition: ast::enum_def) -> @ast::item;
|
|
|
|
fn variant(name: ident, span: span, +tys: ~[@ast::ty]) -> ast::variant;
|
|
|
|
fn item_mod(name: ident, span: span, +items: ~[@ast::item]) -> @ast::item;
|
2012-07-11 22:00:40 +00:00
|
|
|
fn ty_path_ast_builder(path: @ast::path) -> @ast::ty;
|
|
|
|
fn item_ty_poly(name: ident,
|
2012-08-17 18:26:35 +00:00
|
|
|
span: span,
|
2012-07-11 22:00:40 +00:00
|
|
|
ty: @ast::ty,
|
|
|
|
+params: ~[ast::ty_param]) -> @ast::item;
|
2012-08-17 18:26:35 +00:00
|
|
|
fn item_ty(name: ident, span: span, ty: @ast::ty) -> @ast::item;
|
2012-07-11 22:00:40 +00:00
|
|
|
fn ty_vars(+ty_params: ~[ast::ty_param]) -> ~[@ast::ty];
|
2012-07-23 21:46:39 +00:00
|
|
|
fn ty_field_imm(name: ident, ty: @ast::ty) -> ast::ty_field;
|
|
|
|
fn ty_rec(+~[ast::ty_field]) -> @ast::ty;
|
2012-07-24 01:50:53 +00:00
|
|
|
fn field_imm(name: ident, e: @ast::expr) -> ast::field;
|
|
|
|
fn rec(+~[ast::field]) -> @ast::expr;
|
|
|
|
fn block(+stmts: ~[@ast::stmt], e: @ast::expr) -> ast::blk;
|
|
|
|
fn stmt_let(ident: ident, e: @ast::expr) -> @ast::stmt;
|
|
|
|
fn stmt_expr(e: @ast::expr) -> @ast::stmt;
|
|
|
|
fn block_expr(b: ast::blk) -> @ast::expr;
|
2012-07-24 23:58:48 +00:00
|
|
|
fn empty_span() -> span;
|
2012-08-07 18:46:52 +00:00
|
|
|
fn ty_option(ty: @ast::ty) -> @ast::ty;
|
2012-07-11 22:00:40 +00:00
|
|
|
}
|
|
|
|
|
2012-08-08 01:10:06 +00:00
|
|
|
impl ext_ctxt: ext_ctxt_ast_builder {
|
2012-08-07 18:46:52 +00:00
|
|
|
fn ty_option(ty: @ast::ty) -> @ast::ty {
|
2012-08-20 19:23:37 +00:00
|
|
|
self.ty_path_ast_builder(path(~[self.ident_of(~"Option")],
|
2012-07-18 23:18:02 +00:00
|
|
|
self.empty_span())
|
2012-08-07 18:46:52 +00:00
|
|
|
.add_ty(ty))
|
|
|
|
}
|
|
|
|
|
2012-07-24 23:58:48 +00:00
|
|
|
fn empty_span() -> span {
|
|
|
|
{lo: 0, hi: 0, expn_info: self.backtrace()}
|
|
|
|
}
|
|
|
|
|
2012-07-24 01:50:53 +00:00
|
|
|
fn block_expr(b: ast::blk) -> @ast::expr {
|
|
|
|
@{id: self.next_id(),
|
|
|
|
callee_id: self.next_id(),
|
|
|
|
node: ast::expr_block(b),
|
2012-07-24 23:58:48 +00:00
|
|
|
span: self.empty_span()}
|
2012-07-24 01:50:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn stmt_expr(e: @ast::expr) -> @ast::stmt {
|
|
|
|
@{node: ast::stmt_expr(e, self.next_id()),
|
2012-07-24 23:58:48 +00:00
|
|
|
span: self.empty_span()}
|
2012-07-24 01:50:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn stmt_let(ident: ident, e: @ast::expr) -> @ast::stmt {
|
|
|
|
// If the quasiquoter could interpolate idents, this is all
|
|
|
|
// we'd need.
|
|
|
|
//
|
|
|
|
//let ext_cx = self;
|
|
|
|
//#ast[stmt] { let $(ident) = $(e) }
|
|
|
|
|
|
|
|
@{node: ast::stmt_decl(@{node: ast::decl_local(~[
|
|
|
|
@{node: {is_mutbl: false,
|
|
|
|
ty: self.ty_infer(),
|
|
|
|
pat: @{id: self.next_id(),
|
2012-08-06 14:20:23 +00:00
|
|
|
node: ast::pat_ident(ast::bind_by_implicit_ref,
|
2012-07-18 23:18:02 +00:00
|
|
|
path(~[ident],
|
2012-08-01 02:25:24 +00:00
|
|
|
self.empty_span()),
|
2012-08-20 19:23:37 +00:00
|
|
|
None),
|
2012-07-24 23:58:48 +00:00
|
|
|
span: self.empty_span()},
|
2012-08-20 19:23:37 +00:00
|
|
|
init: Some({op: ast::init_move,
|
2012-07-24 01:50:53 +00:00
|
|
|
expr: e}),
|
|
|
|
id: self.next_id()},
|
2012-07-24 23:58:48 +00:00
|
|
|
span: self.empty_span()}]),
|
|
|
|
span: self.empty_span()}, self.next_id()),
|
|
|
|
span: self.empty_span()}
|
2012-07-24 01:50:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn field_imm(name: ident, e: @ast::expr) -> ast::field {
|
|
|
|
{node: {mutbl: ast::m_imm, ident: name, expr: e},
|
2012-07-24 23:58:48 +00:00
|
|
|
span: self.empty_span()}
|
2012-07-24 01:50:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn rec(+fields: ~[ast::field]) -> @ast::expr {
|
|
|
|
@{id: self.next_id(),
|
|
|
|
callee_id: self.next_id(),
|
2012-08-20 19:23:37 +00:00
|
|
|
node: ast::expr_rec(fields, None),
|
2012-07-24 23:58:48 +00:00
|
|
|
span: self.empty_span()}
|
2012-07-24 01:50:53 +00:00
|
|
|
}
|
|
|
|
|
2012-07-23 21:46:39 +00:00
|
|
|
fn ty_field_imm(name: ident, ty: @ast::ty) -> ast::ty_field {
|
|
|
|
{node: {ident: name, mt: { ty: ty, mutbl: ast::m_imm } },
|
2012-07-24 23:58:48 +00:00
|
|
|
span: self.empty_span()}
|
2012-07-23 21:46:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn ty_rec(+fields: ~[ast::ty_field]) -> @ast::ty {
|
|
|
|
@{id: self.next_id(),
|
|
|
|
node: ast::ty_rec(fields),
|
2012-07-24 23:58:48 +00:00
|
|
|
span: self.empty_span()}
|
2012-07-23 21:46:39 +00:00
|
|
|
}
|
|
|
|
|
2012-07-24 01:50:53 +00:00
|
|
|
fn ty_infer() -> @ast::ty {
|
|
|
|
@{id: self.next_id(),
|
|
|
|
node: ast::ty_infer,
|
2012-07-24 23:58:48 +00:00
|
|
|
span: self.empty_span()}
|
2012-07-24 01:50:53 +00:00
|
|
|
}
|
|
|
|
|
2012-07-05 23:07:53 +00:00
|
|
|
fn ty_param(id: ast::ident, +bounds: ~[ast::ty_param_bound])
|
|
|
|
-> ast::ty_param
|
|
|
|
{
|
|
|
|
{ident: id, id: self.next_id(), bounds: @bounds}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn arg(name: ident, ty: @ast::ty) -> ast::arg {
|
|
|
|
{mode: ast::infer(self.next_id()),
|
|
|
|
ty: ty,
|
|
|
|
ident: name,
|
|
|
|
id: self.next_id()}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn arg_mode(name: ident, ty: @ast::ty, mode: ast::rmode) -> ast::arg {
|
|
|
|
{mode: ast::expl(mode),
|
|
|
|
ty: ty,
|
|
|
|
ident: name,
|
|
|
|
id: self.next_id()}
|
|
|
|
}
|
|
|
|
|
2012-07-24 01:50:53 +00:00
|
|
|
fn block(+stmts: ~[@ast::stmt], e: @ast::expr) -> ast::blk {
|
2012-07-05 23:07:53 +00:00
|
|
|
let blk = {view_items: ~[],
|
2012-07-24 01:50:53 +00:00
|
|
|
stmts: stmts,
|
2012-08-20 19:23:37 +00:00
|
|
|
expr: Some(e),
|
2012-07-05 23:07:53 +00:00
|
|
|
id: self.next_id(),
|
|
|
|
rules: ast::default_blk};
|
|
|
|
|
|
|
|
{node: blk,
|
2012-07-24 23:58:48 +00:00
|
|
|
span: self.empty_span()}
|
2012-07-05 23:07:53 +00:00
|
|
|
}
|
|
|
|
|
2012-07-24 01:50:53 +00:00
|
|
|
fn expr_block(e: @ast::expr) -> ast::blk {
|
|
|
|
self.block(~[], e)
|
|
|
|
}
|
|
|
|
|
2012-07-05 23:07:53 +00:00
|
|
|
fn fn_decl(+inputs: ~[ast::arg],
|
|
|
|
output: @ast::ty) -> ast::fn_decl {
|
|
|
|
{inputs: inputs,
|
|
|
|
output: output,
|
2012-07-14 01:43:52 +00:00
|
|
|
cf: ast::return_val}
|
2012-07-05 23:07:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn item(name: ident,
|
2012-08-17 18:26:35 +00:00
|
|
|
span: span,
|
2012-07-05 23:07:53 +00:00
|
|
|
+node: ast::item_) -> @ast::item {
|
|
|
|
@{ident: name,
|
|
|
|
attrs: ~[],
|
|
|
|
id: self.next_id(),
|
|
|
|
node: node,
|
|
|
|
vis: ast::public,
|
2012-08-17 18:26:35 +00:00
|
|
|
span: span}
|
2012-07-05 23:07:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn item_fn_poly(name: ident,
|
|
|
|
+inputs: ~[ast::arg],
|
|
|
|
output: @ast::ty,
|
|
|
|
+ty_params: ~[ast::ty_param],
|
|
|
|
+body: ast::blk) -> @ast::item {
|
|
|
|
self.item(name,
|
2012-08-17 18:26:35 +00:00
|
|
|
self.empty_span(),
|
2012-07-05 23:07:53 +00:00
|
|
|
ast::item_fn(self.fn_decl(inputs, output),
|
2012-08-24 01:17:16 +00:00
|
|
|
ast::impure_fn,
|
2012-07-05 23:07:53 +00:00
|
|
|
ty_params,
|
|
|
|
body))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn item_fn(name: ident,
|
|
|
|
+inputs: ~[ast::arg],
|
|
|
|
output: @ast::ty,
|
|
|
|
+body: ast::blk) -> @ast::item {
|
|
|
|
self.item_fn_poly(name, inputs, output, ~[], body)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn item_enum_poly(name: ident,
|
2012-08-17 18:26:35 +00:00
|
|
|
span: span,
|
2012-08-09 00:14:25 +00:00
|
|
|
+enum_definition: ast::enum_def,
|
2012-07-05 23:07:53 +00:00
|
|
|
+ty_params: ~[ast::ty_param]) -> @ast::item {
|
2012-08-17 18:26:35 +00:00
|
|
|
self.item(name, span, ast::item_enum(enum_definition, ty_params))
|
2012-07-05 23:07:53 +00:00
|
|
|
}
|
|
|
|
|
2012-08-17 18:26:35 +00:00
|
|
|
fn item_enum(name: ident, span: span,
|
|
|
|
+enum_definition: ast::enum_def) -> @ast::item {
|
|
|
|
self.item_enum_poly(name, span, enum_definition, ~[])
|
2012-07-05 23:07:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn variant(name: ident,
|
2012-08-17 18:26:35 +00:00
|
|
|
span: span,
|
2012-07-05 23:07:53 +00:00
|
|
|
+tys: ~[@ast::ty]) -> ast::variant {
|
|
|
|
let args = tys.map(|ty| {ty: ty, id: self.next_id()});
|
|
|
|
|
2012-07-24 23:58:48 +00:00
|
|
|
{node: {name: name,
|
|
|
|
attrs: ~[],
|
2012-08-07 21:24:04 +00:00
|
|
|
kind: ast::tuple_variant_kind(args),
|
2012-07-24 23:58:48 +00:00
|
|
|
id: self.next_id(),
|
2012-08-20 19:23:37 +00:00
|
|
|
disr_expr: None,
|
2012-07-24 23:58:48 +00:00
|
|
|
vis: ast::public},
|
2012-08-17 18:26:35 +00:00
|
|
|
span: span}
|
2012-07-05 23:07:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn item_mod(name: ident,
|
2012-08-17 18:26:35 +00:00
|
|
|
span: span,
|
2012-07-05 23:07:53 +00:00
|
|
|
+items: ~[@ast::item]) -> @ast::item {
|
|
|
|
self.item(name,
|
2012-08-17 18:26:35 +00:00
|
|
|
span,
|
2012-07-05 23:07:53 +00:00
|
|
|
ast::item_mod({
|
|
|
|
view_items: ~[],
|
|
|
|
items: items}))
|
|
|
|
}
|
|
|
|
|
2012-07-11 22:00:40 +00:00
|
|
|
fn ty_path_ast_builder(path: @ast::path) -> @ast::ty {
|
2012-07-05 23:07:53 +00:00
|
|
|
@{id: self.next_id(),
|
|
|
|
node: ast::ty_path(path, self.next_id()),
|
2012-08-17 18:26:35 +00:00
|
|
|
span: path.span}
|
2012-07-05 23:07:53 +00:00
|
|
|
}
|
|
|
|
|
2012-07-17 23:49:54 +00:00
|
|
|
fn ty_nil_ast_builder() -> @ast::ty {
|
2012-07-16 21:44:27 +00:00
|
|
|
@{id: self.next_id(),
|
|
|
|
node: ast::ty_nil,
|
2012-07-24 23:58:48 +00:00
|
|
|
span: self.empty_span()}
|
2012-07-16 21:44:27 +00:00
|
|
|
}
|
|
|
|
|
2012-07-05 23:07:53 +00:00
|
|
|
fn item_ty_poly(name: ident,
|
2012-08-17 18:26:35 +00:00
|
|
|
span: span,
|
2012-07-05 23:07:53 +00:00
|
|
|
ty: @ast::ty,
|
|
|
|
+params: ~[ast::ty_param]) -> @ast::item {
|
2012-08-17 18:26:35 +00:00
|
|
|
self.item(name, span, ast::item_ty(ty, params))
|
2012-07-05 23:07:53 +00:00
|
|
|
}
|
|
|
|
|
2012-08-17 18:26:35 +00:00
|
|
|
fn item_ty(name: ident, span: span, ty: @ast::ty) -> @ast::item {
|
|
|
|
self.item_ty_poly(name, span, ty, ~[])
|
2012-07-05 23:07:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn ty_vars(+ty_params: ~[ast::ty_param]) -> ~[@ast::ty] {
|
2012-07-24 23:58:48 +00:00
|
|
|
ty_params.map(|p| self.ty_path_ast_builder(
|
2012-07-18 23:18:02 +00:00
|
|
|
path(~[p.ident], self.empty_span())))
|
2012-07-05 23:07:53 +00:00
|
|
|
}
|
|
|
|
}
|