2014-02-10 15:36:31 +01:00
|
|
|
// Copyright 2012-2014 The Rust Project Developers. See the COPYRIGHT
|
2012-12-03 16:48:01 -08:00
|
|
|
// 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.
|
|
|
|
|
2014-03-21 18:05:05 -07:00
|
|
|
#![allow(non_camel_case_types)]
|
2014-10-27 15:37:07 -07:00
|
|
|
#![allow(unsigned_negation)]
|
2013-05-17 15:28:44 -07:00
|
|
|
|
2014-11-06 00:05:53 -08:00
|
|
|
pub use self::const_val::*;
|
|
|
|
pub use self::constness::*;
|
|
|
|
|
2013-03-21 00:27:26 -07:00
|
|
|
use metadata::csearch;
|
2014-11-23 12:14:35 +01:00
|
|
|
use middle::{astencode, def};
|
2014-07-13 15:12:47 +02:00
|
|
|
use middle::pat_util::def_to_path;
|
2014-11-26 04:52:02 -05:00
|
|
|
use middle::ty::{mod};
|
|
|
|
use middle::astconv_util::{ast_ty_to_prim_ty};
|
2014-11-23 12:14:35 +01:00
|
|
|
use util::nodemap::DefIdMap;
|
2012-12-23 17:41:37 -05:00
|
|
|
|
2014-09-13 20:10:34 +03:00
|
|
|
use syntax::ast::{mod, Expr};
|
2014-01-10 14:02:36 -08:00
|
|
|
use syntax::parse::token::InternedString;
|
2014-09-07 20:09:06 +03:00
|
|
|
use syntax::ptr::P;
|
2014-11-23 12:14:35 +01:00
|
|
|
use syntax::visit::{mod, Visitor};
|
2014-09-13 20:10:34 +03:00
|
|
|
use syntax::{ast_map, ast_util, codemap};
|
2012-07-30 19:05:56 -07:00
|
|
|
|
2014-02-01 15:57:59 +11:00
|
|
|
use std::rc::Rc;
|
2014-10-30 21:25:08 -04:00
|
|
|
use std::collections::hash_map::Vacant;
|
2013-03-21 00:27:26 -07:00
|
|
|
|
2012-07-30 19:05:56 -07:00
|
|
|
//
|
|
|
|
// This pass classifies expressions by their constant-ness.
|
|
|
|
//
|
|
|
|
// Constant-ness comes in 3 flavours:
|
|
|
|
//
|
|
|
|
// - Integer-constants: can be evaluated by the frontend all the way down
|
|
|
|
// to their actual value. They are used in a few places (enum
|
|
|
|
// discriminants, switch arms) and are a subset of
|
|
|
|
// general-constants. They cover all the integer and integer-ish
|
|
|
|
// literals (nil, bool, int, uint, char, iNN, uNN) and all integer
|
|
|
|
// operators and copies applied to them.
|
|
|
|
//
|
|
|
|
// - General-constants: can be evaluated by LLVM but not necessarily by
|
|
|
|
// the frontend; usually due to reliance on target-specific stuff such
|
|
|
|
// as "where in memory the value goes" or "what floating point mode the
|
|
|
|
// target uses". This _includes_ integer-constants, plus the following
|
|
|
|
// constructors:
|
|
|
|
//
|
2012-10-10 00:28:04 -04:00
|
|
|
// fixed-size vectors and strings: [] and ""/_
|
2012-07-30 19:05:56 -07:00
|
|
|
// vector and string slices: &[] and &""
|
|
|
|
// tuples: (,)
|
|
|
|
// enums: foo(...)
|
|
|
|
// floating point literals and operators
|
|
|
|
// & and * pointers
|
|
|
|
// copies of general constants
|
|
|
|
//
|
2013-01-04 09:52:07 -05:00
|
|
|
// (in theory, probably not at first: if/match on integer-const
|
2014-09-02 01:35:58 -04:00
|
|
|
// conditions / discriminants)
|
2012-07-30 19:05:56 -07:00
|
|
|
//
|
|
|
|
// - Non-constants: everything else.
|
|
|
|
//
|
|
|
|
|
2013-01-30 13:44:24 -08:00
|
|
|
pub enum constness {
|
2012-07-30 19:05:56 -07:00
|
|
|
integral_const,
|
|
|
|
general_const,
|
|
|
|
non_const
|
|
|
|
}
|
|
|
|
|
2014-02-28 14:34:26 -08:00
|
|
|
type constness_cache = DefIdMap<constness>;
|
2013-10-04 15:04:11 -07:00
|
|
|
|
2013-01-30 13:44:24 -08:00
|
|
|
pub fn join(a: constness, b: constness) -> constness {
|
2012-09-27 22:20:47 -07:00
|
|
|
match (a, b) {
|
2012-08-03 19:59:04 -07:00
|
|
|
(integral_const, integral_const) => integral_const,
|
2012-07-30 19:05:56 -07:00
|
|
|
(integral_const, general_const)
|
|
|
|
| (general_const, integral_const)
|
2012-08-03 19:59:04 -07:00
|
|
|
| (general_const, general_const) => general_const,
|
|
|
|
_ => non_const
|
2012-07-30 19:05:56 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-06 09:32:37 -08:00
|
|
|
pub fn join_all<It: Iterator<constness>>(cs: It) -> constness {
|
2013-06-29 15:05:50 +10:00
|
|
|
cs.fold(integral_const, |a, b| join(a, b))
|
2012-07-30 19:05:56 -07:00
|
|
|
}
|
|
|
|
|
2014-09-07 20:09:06 +03:00
|
|
|
fn lookup_const<'a>(tcx: &'a ty::ctxt, e: &Expr) -> Option<&'a Expr> {
|
2014-11-07 14:35:18 -05:00
|
|
|
let opt_def = tcx.def_map.borrow().get(&e.id).cloned();
|
2013-12-23 11:15:16 -08:00
|
|
|
match opt_def {
|
rustc: Add `const` globals to the language
This change is an implementation of [RFC 69][rfc] which adds a third kind of
global to the language, `const`. This global is most similar to what the old
`static` was, and if you're unsure about what to use then you should use a
`const`.
The semantics of these three kinds of globals are:
* A `const` does not represent a memory location, but only a value. Constants
are translated as rvalues, which means that their values are directly inlined
at usage location (similar to a #define in C/C++). Constant values are, well,
constant, and can not be modified. Any "modification" is actually a
modification to a local value on the stack rather than the actual constant
itself.
Almost all values are allowed inside constants, whether they have interior
mutability or not. There are a few minor restrictions listed in the RFC, but
they should in general not come up too often.
* A `static` now always represents a memory location (unconditionally). Any
references to the same `static` are actually a reference to the same memory
location. Only values whose types ascribe to `Sync` are allowed in a `static`.
This restriction is in place because many threads may access a `static`
concurrently. Lifting this restriction (and allowing unsafe access) is a
future extension not implemented at this time.
* A `static mut` continues to always represent a memory location. All references
to a `static mut` continue to be `unsafe`.
This is a large breaking change, and many programs will need to be updated
accordingly. A summary of the breaking changes is:
* Statics may no longer be used in patterns. Statics now always represent a
memory location, which can sometimes be modified. To fix code, repurpose the
matched-on-`static` to a `const`.
static FOO: uint = 4;
match n {
FOO => { /* ... */ }
_ => { /* ... */ }
}
change this code to:
const FOO: uint = 4;
match n {
FOO => { /* ... */ }
_ => { /* ... */ }
}
* Statics may no longer refer to other statics by value. Due to statics being
able to change at runtime, allowing them to reference one another could
possibly lead to confusing semantics. If you are in this situation, use a
constant initializer instead. Note, however, that statics may reference other
statics by address, however.
* Statics may no longer be used in constant expressions, such as array lengths.
This is due to the same restrictions as listed above. Use a `const` instead.
[breaking-change]
[rfc]: https://github.com/rust-lang/rfcs/pull/246
2014-10-06 08:17:01 -07:00
|
|
|
Some(def::DefConst(def_id)) => {
|
2013-12-23 11:15:16 -08:00
|
|
|
lookup_const_by_id(tcx, def_id)
|
|
|
|
}
|
2014-05-14 15:31:30 -04:00
|
|
|
Some(def::DefVariant(enum_def, variant_def, _)) => {
|
2013-12-23 11:15:16 -08:00
|
|
|
lookup_variant_by_id(tcx, enum_def, variant_def)
|
|
|
|
}
|
2012-11-12 22:10:15 -08:00
|
|
|
_ => None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-07 20:09:06 +03:00
|
|
|
fn lookup_variant_by_id<'a>(tcx: &'a ty::ctxt,
|
2013-09-02 03:45:37 +02:00
|
|
|
enum_def: ast::DefId,
|
|
|
|
variant_def: ast::DefId)
|
2014-09-07 20:09:06 +03:00
|
|
|
-> Option<&'a Expr> {
|
|
|
|
fn variant_expr<'a>(variants: &'a [P<ast::Variant>], id: ast::NodeId)
|
|
|
|
-> Option<&'a Expr> {
|
2013-08-03 12:45:23 -04:00
|
|
|
for variant in variants.iter() {
|
2013-07-16 04:27:54 -04:00
|
|
|
if variant.node.id == id {
|
2014-09-07 20:09:06 +03:00
|
|
|
return variant.node.disr_expr.as_ref().map(|e| &**e);
|
2013-07-16 04:27:54 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
None
|
|
|
|
}
|
|
|
|
|
|
|
|
if ast_util::is_local(enum_def) {
|
2014-09-07 20:09:06 +03:00
|
|
|
match tcx.map.find(enum_def.node) {
|
|
|
|
None => None,
|
|
|
|
Some(ast_map::NodeItem(it)) => match it.node {
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::ItemEnum(ast::EnumDef { ref variants }, _) => {
|
2014-09-07 20:09:06 +03:00
|
|
|
variant_expr(variants.as_slice(), variant_def.node)
|
|
|
|
}
|
|
|
|
_ => None
|
|
|
|
},
|
|
|
|
Some(_) => None
|
2013-07-16 04:27:54 -04:00
|
|
|
}
|
|
|
|
} else {
|
2014-11-06 12:25:16 -05:00
|
|
|
match tcx.extern_const_variants.borrow().get(&variant_def) {
|
2014-09-07 20:09:06 +03:00
|
|
|
Some(&ast::DUMMY_NODE_ID) => return None,
|
|
|
|
Some(&expr_id) => {
|
|
|
|
return Some(tcx.map.expect_expr(expr_id));
|
|
|
|
}
|
2014-03-15 22:29:34 +02:00
|
|
|
None => {}
|
2013-10-04 15:04:11 -07:00
|
|
|
}
|
2014-09-07 20:09:06 +03:00
|
|
|
let expr_id = match csearch::maybe_get_item_ast(tcx, enum_def,
|
2014-04-21 19:21:53 -04:00
|
|
|
|a, b, c, d| astencode::decode_inlined_item(a, b, c, d)) {
|
2014-09-07 20:09:06 +03:00
|
|
|
csearch::found(&ast::IIItem(ref item)) => match item.node {
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::ItemEnum(ast::EnumDef { ref variants }, _) => {
|
2014-09-07 20:09:06 +03:00
|
|
|
// NOTE this doesn't do the right thing, it compares inlined
|
|
|
|
// NodeId's to the original variant_def's NodeId, but they
|
|
|
|
// come from different crates, so they will likely never match.
|
|
|
|
variant_expr(variants.as_slice(), variant_def.node).map(|e| e.id)
|
2013-07-16 04:27:54 -04:00
|
|
|
}
|
|
|
|
_ => None
|
|
|
|
},
|
|
|
|
_ => None
|
2013-10-04 15:04:11 -07:00
|
|
|
};
|
2014-09-07 20:09:06 +03:00
|
|
|
tcx.extern_const_variants.borrow_mut().insert(variant_def,
|
|
|
|
expr_id.unwrap_or(ast::DUMMY_NODE_ID));
|
|
|
|
expr_id.map(|id| tcx.map.expect_expr(id))
|
2013-07-16 04:27:54 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-07 20:09:06 +03:00
|
|
|
pub fn lookup_const_by_id<'a>(tcx: &'a ty::ctxt, def_id: ast::DefId)
|
|
|
|
-> Option<&'a Expr> {
|
2012-11-12 22:10:15 -08:00
|
|
|
if ast_util::is_local(def_id) {
|
2014-09-07 20:09:06 +03:00
|
|
|
match tcx.map.find(def_id.node) {
|
|
|
|
None => None,
|
|
|
|
Some(ast_map::NodeItem(it)) => match it.node {
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::ItemConst(_, ref const_expr) => {
|
2014-09-07 20:09:06 +03:00
|
|
|
Some(&**const_expr)
|
|
|
|
}
|
|
|
|
_ => None
|
|
|
|
},
|
|
|
|
Some(_) => None
|
2012-10-15 12:27:09 -07:00
|
|
|
}
|
2012-11-12 22:10:15 -08:00
|
|
|
} else {
|
2014-11-06 12:25:16 -05:00
|
|
|
match tcx.extern_const_statics.borrow().get(&def_id) {
|
2014-09-07 20:09:06 +03:00
|
|
|
Some(&ast::DUMMY_NODE_ID) => return None,
|
|
|
|
Some(&expr_id) => {
|
|
|
|
return Some(tcx.map.expect_expr(expr_id));
|
|
|
|
}
|
2014-03-20 19:49:20 -07:00
|
|
|
None => {}
|
2013-10-04 15:04:11 -07:00
|
|
|
}
|
2014-09-07 20:09:06 +03:00
|
|
|
let expr_id = match csearch::maybe_get_item_ast(tcx, def_id,
|
2014-04-21 19:21:53 -04:00
|
|
|
|a, b, c, d| astencode::decode_inlined_item(a, b, c, d)) {
|
2014-09-07 20:09:06 +03:00
|
|
|
csearch::found(&ast::IIItem(ref item)) => match item.node {
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::ItemConst(_, ref const_expr) => Some(const_expr.id),
|
2013-03-21 00:27:26 -07:00
|
|
|
_ => None
|
|
|
|
},
|
|
|
|
_ => None
|
2013-10-04 15:04:11 -07:00
|
|
|
};
|
2014-09-07 20:09:06 +03:00
|
|
|
tcx.extern_const_statics.borrow_mut().insert(def_id,
|
|
|
|
expr_id.unwrap_or(ast::DUMMY_NODE_ID));
|
|
|
|
expr_id.map(|id| tcx.map.expect_expr(id))
|
2012-10-15 12:27:09 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-22 15:56:37 +03:00
|
|
|
struct ConstEvalVisitor<'a, 'tcx: 'a> {
|
|
|
|
tcx: &'a ty::ctxt<'tcx>,
|
2013-10-04 15:04:11 -07:00
|
|
|
ccache: constness_cache,
|
|
|
|
}
|
|
|
|
|
2014-04-22 15:56:37 +03:00
|
|
|
impl<'a, 'tcx> ConstEvalVisitor<'a, 'tcx> {
|
2013-10-04 15:04:11 -07:00
|
|
|
fn classify(&mut self, e: &Expr) -> constness {
|
|
|
|
let did = ast_util::local_def(e.id);
|
2014-11-06 12:25:16 -05:00
|
|
|
match self.ccache.get(&did) {
|
2013-10-04 15:04:11 -07:00
|
|
|
Some(&x) => return x,
|
|
|
|
None => {}
|
|
|
|
}
|
|
|
|
let cn = match e.node {
|
2014-05-16 10:15:33 -07:00
|
|
|
ast::ExprLit(ref lit) => {
|
2013-10-04 15:04:11 -07:00
|
|
|
match lit.node {
|
2014-01-09 15:05:33 +02:00
|
|
|
ast::LitStr(..) | ast::LitFloat(..) => general_const,
|
2013-10-04 15:04:11 -07:00
|
|
|
_ => integral_const
|
|
|
|
}
|
2012-10-15 12:27:09 -07:00
|
|
|
}
|
2013-10-04 15:04:11 -07:00
|
|
|
|
2014-05-16 10:15:33 -07:00
|
|
|
ast::ExprUnary(_, ref inner) | ast::ExprParen(ref inner) =>
|
|
|
|
self.classify(&**inner),
|
2013-10-04 15:04:11 -07:00
|
|
|
|
2014-05-16 10:15:33 -07:00
|
|
|
ast::ExprBinary(_, ref a, ref b) =>
|
|
|
|
join(self.classify(&**a), self.classify(&**b)),
|
2013-10-04 15:04:11 -07:00
|
|
|
|
|
|
|
ast::ExprTup(ref es) |
|
2014-04-04 13:12:18 +03:00
|
|
|
ast::ExprVec(ref es) =>
|
2014-05-16 10:15:33 -07:00
|
|
|
join_all(es.iter().map(|e| self.classify(&**e))),
|
2013-10-04 15:04:11 -07:00
|
|
|
|
|
|
|
ast::ExprStruct(_, ref fs, None) => {
|
2014-05-16 10:15:33 -07:00
|
|
|
let cs = fs.iter().map(|f| self.classify(&*f.expr));
|
2013-10-04 15:04:11 -07:00
|
|
|
join_all(cs)
|
|
|
|
}
|
|
|
|
|
2014-05-16 10:15:33 -07:00
|
|
|
ast::ExprCast(ref base, _) => {
|
2013-10-04 15:04:11 -07:00
|
|
|
let ty = ty::expr_ty(self.tcx, e);
|
2014-05-16 10:15:33 -07:00
|
|
|
let base = self.classify(&**base);
|
2013-10-04 15:04:11 -07:00
|
|
|
if ty::type_is_integral(ty) {
|
|
|
|
join(integral_const, base)
|
|
|
|
} else if ty::type_is_fp(ty) {
|
|
|
|
join(general_const, base)
|
|
|
|
} else {
|
|
|
|
non_const
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-23 12:14:35 +01:00
|
|
|
ast::ExprField(ref base, _) => self.classify(&**base),
|
2013-10-04 15:04:11 -07:00
|
|
|
|
2014-11-23 12:14:35 +01:00
|
|
|
ast::ExprTupField(ref base, _) => self.classify(&**base),
|
2014-08-10 15:54:33 +12:00
|
|
|
|
2014-05-16 10:15:33 -07:00
|
|
|
ast::ExprIndex(ref base, ref idx) =>
|
|
|
|
join(self.classify(&**base), self.classify(&**idx)),
|
2013-10-04 15:04:11 -07:00
|
|
|
|
2014-05-16 10:15:33 -07:00
|
|
|
ast::ExprAddrOf(ast::MutImmutable, ref base) =>
|
|
|
|
self.classify(&**base),
|
2013-10-04 15:04:11 -07:00
|
|
|
|
|
|
|
// FIXME: (#3728) we can probably do something CCI-ish
|
|
|
|
// surrounding nonlocal constants. But we don't yet.
|
|
|
|
ast::ExprPath(_) => self.lookup_constness(e),
|
|
|
|
|
2013-11-28 12:22:53 -08:00
|
|
|
ast::ExprRepeat(..) => general_const,
|
2013-10-04 15:04:11 -07:00
|
|
|
|
2014-05-04 10:39:11 +02:00
|
|
|
ast::ExprBlock(ref block) => {
|
|
|
|
match block.expr {
|
|
|
|
Some(ref e) => self.classify(&**e),
|
|
|
|
None => integral_const
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-04 15:04:11 -07:00
|
|
|
_ => non_const
|
|
|
|
};
|
|
|
|
self.ccache.insert(did, cn);
|
|
|
|
cn
|
|
|
|
}
|
|
|
|
|
|
|
|
fn lookup_constness(&self, e: &Expr) -> constness {
|
|
|
|
match lookup_const(self.tcx, e) {
|
|
|
|
Some(rhs) => {
|
2014-05-16 10:15:33 -07:00
|
|
|
let ty = ty::expr_ty(self.tcx, &*rhs);
|
2013-10-04 15:04:11 -07:00
|
|
|
if ty::type_is_integral(ty) {
|
|
|
|
integral_const
|
|
|
|
} else {
|
|
|
|
general_const
|
|
|
|
}
|
|
|
|
}
|
|
|
|
None => non_const
|
2012-10-15 12:27:09 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-04 15:04:11 -07:00
|
|
|
}
|
2013-08-13 03:37:40 +02:00
|
|
|
|
2014-09-10 01:54:36 +03:00
|
|
|
impl<'a, 'tcx, 'v> Visitor<'v> for ConstEvalVisitor<'a, 'tcx> {
|
2014-09-12 13:10:30 +03:00
|
|
|
fn visit_expr_post(&mut self, e: &Expr) {
|
2013-10-04 15:04:11 -07:00
|
|
|
self.classify(e);
|
2013-08-13 03:37:40 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-07 20:09:06 +03:00
|
|
|
pub fn process_crate(tcx: &ty::ctxt) {
|
|
|
|
visit::walk_crate(&mut ConstEvalVisitor {
|
2013-10-04 15:04:11 -07:00
|
|
|
tcx: tcx,
|
2014-02-28 14:34:26 -08:00
|
|
|
ccache: DefIdMap::new(),
|
2014-09-07 20:09:06 +03:00
|
|
|
}, tcx.map.krate());
|
2012-07-30 19:05:56 -07:00
|
|
|
tcx.sess.abort_if_errors();
|
|
|
|
}
|
|
|
|
|
2012-03-22 14:56:56 -07:00
|
|
|
|
2012-06-21 16:44:10 -07:00
|
|
|
// FIXME (#33): this doesn't handle big integer/float literals correctly
|
|
|
|
// (nor does the rest of our literal handling).
|
2014-05-29 17:45:07 -07:00
|
|
|
#[deriving(Clone, PartialEq)]
|
2013-01-30 13:44:24 -08:00
|
|
|
pub enum const_val {
|
2012-06-04 17:26:17 -07:00
|
|
|
const_float(f64),
|
2012-03-22 14:56:56 -07:00
|
|
|
const_int(i64),
|
|
|
|
const_uint(u64),
|
2014-01-10 14:02:36 -08:00
|
|
|
const_str(InternedString),
|
2014-03-04 10:02:49 -08:00
|
|
|
const_binary(Rc<Vec<u8> >),
|
2014-11-09 16:14:15 +01:00
|
|
|
const_bool(bool)
|
2012-03-22 14:56:56 -07:00
|
|
|
}
|
|
|
|
|
2014-09-13 20:10:34 +03:00
|
|
|
pub fn const_expr_to_pat(tcx: &ty::ctxt, expr: &Expr) -> P<ast::Pat> {
|
2014-07-13 15:12:47 +02:00
|
|
|
let pat = match expr.node {
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::ExprTup(ref exprs) =>
|
|
|
|
ast::PatTup(exprs.iter().map(|expr| const_expr_to_pat(tcx, &**expr)).collect()),
|
2014-07-13 15:12:47 +02:00
|
|
|
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::ExprCall(ref callee, ref args) => {
|
2014-11-07 14:35:18 -05:00
|
|
|
let def = tcx.def_map.borrow()[callee.id].clone();
|
2014-11-29 16:41:21 -05:00
|
|
|
if let Vacant(entry) = tcx.def_map.borrow_mut().entry(expr.id) {
|
|
|
|
entry.set(def);
|
|
|
|
}
|
2014-07-13 15:12:47 +02:00
|
|
|
let path = match def {
|
|
|
|
def::DefStruct(def_id) => def_to_path(tcx, def_id),
|
|
|
|
def::DefVariant(_, variant_did, _) => def_to_path(tcx, variant_did),
|
|
|
|
_ => unreachable!()
|
|
|
|
};
|
2014-09-07 20:09:06 +03:00
|
|
|
let pats = args.iter().map(|expr| const_expr_to_pat(tcx, &**expr)).collect();
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::PatEnum(path, Some(pats))
|
2014-07-13 15:12:47 +02:00
|
|
|
}
|
|
|
|
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::ExprStruct(ref path, ref fields, None) => {
|
2014-10-06 13:36:53 +13:00
|
|
|
let field_pats = fields.iter().map(|field| codemap::Spanned {
|
|
|
|
span: codemap::DUMMY_SP,
|
2014-09-13 20:10:34 +03:00
|
|
|
node: ast::FieldPat {
|
2014-10-06 13:36:53 +13:00
|
|
|
ident: field.ident.node,
|
|
|
|
pat: const_expr_to_pat(tcx, &*field.expr),
|
2014-10-27 00:11:26 -07:00
|
|
|
is_shorthand: false,
|
2014-10-06 13:36:53 +13:00
|
|
|
},
|
2014-07-13 15:12:47 +02:00
|
|
|
}).collect();
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::PatStruct(path.clone(), field_pats, false)
|
2014-07-13 15:12:47 +02:00
|
|
|
}
|
|
|
|
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::ExprVec(ref exprs) => {
|
2014-09-07 20:09:06 +03:00
|
|
|
let pats = exprs.iter().map(|expr| const_expr_to_pat(tcx, &**expr)).collect();
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::PatVec(pats, None, vec![])
|
2014-07-13 15:12:47 +02:00
|
|
|
}
|
|
|
|
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::ExprPath(ref path) => {
|
2014-11-07 14:35:18 -05:00
|
|
|
let opt_def = tcx.def_map.borrow().get(&expr.id).cloned();
|
2014-07-13 15:12:47 +02:00
|
|
|
match opt_def {
|
|
|
|
Some(def::DefStruct(..)) =>
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::PatStruct(path.clone(), vec![], false),
|
2014-07-13 15:12:47 +02:00
|
|
|
Some(def::DefVariant(..)) =>
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::PatEnum(path.clone(), None),
|
2014-07-13 15:12:47 +02:00
|
|
|
_ => {
|
2014-09-07 20:09:06 +03:00
|
|
|
match lookup_const(tcx, expr) {
|
2014-07-13 15:12:47 +02:00
|
|
|
Some(actual) => return const_expr_to_pat(tcx, actual),
|
|
|
|
_ => unreachable!()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-13 20:10:34 +03:00
|
|
|
_ => ast::PatLit(P(expr.clone()))
|
2014-07-13 15:12:47 +02:00
|
|
|
};
|
2014-09-13 20:10:34 +03:00
|
|
|
P(ast::Pat { id: expr.id, node: pat, span: expr.span })
|
2014-07-13 15:12:47 +02:00
|
|
|
}
|
|
|
|
|
2014-03-06 05:07:47 +02:00
|
|
|
pub fn eval_const_expr(tcx: &ty::ctxt, e: &Expr) -> const_val {
|
|
|
|
match eval_const_expr_partial(tcx, e) {
|
2013-06-13 03:02:55 +10:00
|
|
|
Ok(r) => r,
|
2014-05-09 18:45:36 -07:00
|
|
|
Err(s) => tcx.sess.span_fatal(e.span, s.as_slice())
|
2012-10-15 12:27:09 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-22 15:56:37 +03:00
|
|
|
pub fn eval_const_expr_partial(tcx: &ty::ctxt, e: &Expr) -> Result<const_val, String> {
|
2014-05-22 16:57:53 -07:00
|
|
|
fn fromb(b: bool) -> Result<const_val, String> { Ok(const_int(b as i64)) }
|
2012-08-23 15:51:23 -07:00
|
|
|
match e.node {
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::ExprUnary(ast::UnNeg, ref inner) => {
|
2014-05-16 10:15:33 -07:00
|
|
|
match eval_const_expr_partial(tcx, &**inner) {
|
2012-10-15 12:27:09 -07:00
|
|
|
Ok(const_float(f)) => Ok(const_float(-f)),
|
|
|
|
Ok(const_int(i)) => Ok(const_int(-i)),
|
|
|
|
Ok(const_uint(i)) => Ok(const_uint(-i)),
|
2014-05-25 03:17:19 -07:00
|
|
|
Ok(const_str(_)) => Err("negate on string".to_string()),
|
|
|
|
Ok(const_bool(_)) => Err("negate on boolean".to_string()),
|
2013-07-02 12:47:32 -07:00
|
|
|
ref err => ((*err).clone())
|
2012-03-22 14:56:56 -07:00
|
|
|
}
|
|
|
|
}
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::ExprUnary(ast::UnNot, ref inner) => {
|
2014-05-16 10:15:33 -07:00
|
|
|
match eval_const_expr_partial(tcx, &**inner) {
|
2012-10-15 12:27:09 -07:00
|
|
|
Ok(const_int(i)) => Ok(const_int(!i)),
|
|
|
|
Ok(const_uint(i)) => Ok(const_uint(!i)),
|
|
|
|
Ok(const_bool(b)) => Ok(const_bool(!b)),
|
2014-05-25 03:17:19 -07:00
|
|
|
_ => Err("not on float or string".to_string())
|
2012-03-22 14:56:56 -07:00
|
|
|
}
|
|
|
|
}
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::ExprBinary(op, ref a, ref b) => {
|
2014-05-16 10:15:33 -07:00
|
|
|
match (eval_const_expr_partial(tcx, &**a),
|
|
|
|
eval_const_expr_partial(tcx, &**b)) {
|
2012-10-15 12:27:09 -07:00
|
|
|
(Ok(const_float(a)), Ok(const_float(b))) => {
|
2012-08-23 15:51:23 -07:00
|
|
|
match op {
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::BiAdd => Ok(const_float(a + b)),
|
|
|
|
ast::BiSub => Ok(const_float(a - b)),
|
|
|
|
ast::BiMul => Ok(const_float(a * b)),
|
|
|
|
ast::BiDiv => Ok(const_float(a / b)),
|
|
|
|
ast::BiRem => Ok(const_float(a % b)),
|
|
|
|
ast::BiEq => fromb(a == b),
|
|
|
|
ast::BiLt => fromb(a < b),
|
|
|
|
ast::BiLe => fromb(a <= b),
|
|
|
|
ast::BiNe => fromb(a != b),
|
|
|
|
ast::BiGe => fromb(a >= b),
|
|
|
|
ast::BiGt => fromb(a > b),
|
2014-05-25 03:17:19 -07:00
|
|
|
_ => Err("can't do this op on floats".to_string())
|
2012-03-22 14:56:56 -07:00
|
|
|
}
|
|
|
|
}
|
2012-10-15 12:27:09 -07:00
|
|
|
(Ok(const_int(a)), Ok(const_int(b))) => {
|
2012-08-23 15:51:23 -07:00
|
|
|
match op {
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::BiAdd => Ok(const_int(a + b)),
|
|
|
|
ast::BiSub => Ok(const_int(a - b)),
|
|
|
|
ast::BiMul => Ok(const_int(a * b)),
|
|
|
|
ast::BiDiv if b == 0 => {
|
2014-05-25 03:17:19 -07:00
|
|
|
Err("attempted to divide by zero".to_string())
|
2014-05-09 18:45:36 -07:00
|
|
|
}
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::BiDiv => Ok(const_int(a / b)),
|
|
|
|
ast::BiRem if b == 0 => {
|
2014-05-09 18:45:36 -07:00
|
|
|
Err("attempted remainder with a divisor of \
|
2014-05-25 03:17:19 -07:00
|
|
|
zero".to_string())
|
2014-05-09 18:45:36 -07:00
|
|
|
}
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::BiRem => Ok(const_int(a % b)),
|
|
|
|
ast::BiAnd | ast::BiBitAnd => Ok(const_int(a & b)),
|
|
|
|
ast::BiOr | ast::BiBitOr => Ok(const_int(a | b)),
|
|
|
|
ast::BiBitXor => Ok(const_int(a ^ b)),
|
|
|
|
ast::BiShl => Ok(const_int(a << b as uint)),
|
|
|
|
ast::BiShr => Ok(const_int(a >> b as uint)),
|
|
|
|
ast::BiEq => fromb(a == b),
|
|
|
|
ast::BiLt => fromb(a < b),
|
|
|
|
ast::BiLe => fromb(a <= b),
|
|
|
|
ast::BiNe => fromb(a != b),
|
|
|
|
ast::BiGe => fromb(a >= b),
|
|
|
|
ast::BiGt => fromb(a > b)
|
2012-03-22 14:56:56 -07:00
|
|
|
}
|
|
|
|
}
|
2012-10-15 12:27:09 -07:00
|
|
|
(Ok(const_uint(a)), Ok(const_uint(b))) => {
|
2012-08-23 15:51:23 -07:00
|
|
|
match op {
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::BiAdd => Ok(const_uint(a + b)),
|
|
|
|
ast::BiSub => Ok(const_uint(a - b)),
|
|
|
|
ast::BiMul => Ok(const_uint(a * b)),
|
|
|
|
ast::BiDiv if b == 0 => {
|
2014-05-25 03:17:19 -07:00
|
|
|
Err("attempted to divide by zero".to_string())
|
2014-05-09 18:45:36 -07:00
|
|
|
}
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::BiDiv => Ok(const_uint(a / b)),
|
|
|
|
ast::BiRem if b == 0 => {
|
2014-05-09 18:45:36 -07:00
|
|
|
Err("attempted remainder with a divisor of \
|
2014-05-25 03:17:19 -07:00
|
|
|
zero".to_string())
|
2014-05-09 18:45:36 -07:00
|
|
|
}
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::BiRem => Ok(const_uint(a % b)),
|
|
|
|
ast::BiAnd | ast::BiBitAnd => Ok(const_uint(a & b)),
|
|
|
|
ast::BiOr | ast::BiBitOr => Ok(const_uint(a | b)),
|
|
|
|
ast::BiBitXor => Ok(const_uint(a ^ b)),
|
|
|
|
ast::BiShl => Ok(const_uint(a << b as uint)),
|
|
|
|
ast::BiShr => Ok(const_uint(a >> b as uint)),
|
|
|
|
ast::BiEq => fromb(a == b),
|
|
|
|
ast::BiLt => fromb(a < b),
|
|
|
|
ast::BiLe => fromb(a <= b),
|
|
|
|
ast::BiNe => fromb(a != b),
|
|
|
|
ast::BiGe => fromb(a >= b),
|
|
|
|
ast::BiGt => fromb(a > b),
|
2012-03-22 14:56:56 -07:00
|
|
|
}
|
|
|
|
}
|
2012-05-30 15:03:24 -07:00
|
|
|
// shifts can have any integral type as their rhs
|
2012-10-15 12:27:09 -07:00
|
|
|
(Ok(const_int(a)), Ok(const_uint(b))) => {
|
2012-08-23 15:51:23 -07:00
|
|
|
match op {
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::BiShl => Ok(const_int(a << b as uint)),
|
|
|
|
ast::BiShr => Ok(const_int(a >> b as uint)),
|
2014-05-25 03:17:19 -07:00
|
|
|
_ => Err("can't do this op on an int and uint".to_string())
|
2012-05-30 15:03:24 -07:00
|
|
|
}
|
|
|
|
}
|
2012-10-15 12:27:09 -07:00
|
|
|
(Ok(const_uint(a)), Ok(const_int(b))) => {
|
2012-08-23 15:51:23 -07:00
|
|
|
match op {
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::BiShl => Ok(const_uint(a << b as uint)),
|
|
|
|
ast::BiShr => Ok(const_uint(a >> b as uint)),
|
2014-05-25 03:17:19 -07:00
|
|
|
_ => Err("can't do this op on a uint and int".to_string())
|
2012-05-30 15:03:24 -07:00
|
|
|
}
|
|
|
|
}
|
2012-10-15 12:27:09 -07:00
|
|
|
(Ok(const_bool(a)), Ok(const_bool(b))) => {
|
|
|
|
Ok(const_bool(match op {
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::BiAnd => a && b,
|
|
|
|
ast::BiOr => a || b,
|
|
|
|
ast::BiBitXor => a ^ b,
|
|
|
|
ast::BiBitAnd => a & b,
|
|
|
|
ast::BiBitOr => a | b,
|
|
|
|
ast::BiEq => a == b,
|
|
|
|
ast::BiNe => a != b,
|
2014-05-25 03:17:19 -07:00
|
|
|
_ => return Err("can't do this op on bools".to_string())
|
2012-10-15 12:27:09 -07:00
|
|
|
}))
|
2012-08-23 15:51:23 -07:00
|
|
|
}
|
2014-05-25 03:17:19 -07:00
|
|
|
_ => Err("bad operands for binary".to_string())
|
2012-03-22 14:56:56 -07:00
|
|
|
}
|
|
|
|
}
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::ExprCast(ref base, ref target_ty) => {
|
2014-01-29 00:05:11 -05:00
|
|
|
// This tends to get called w/o the type actually having been
|
|
|
|
// populated in the ctxt, which was causing things to blow up
|
|
|
|
// (#5900). Fall back to doing a limited lookup to get past it.
|
2014-04-22 15:56:37 +03:00
|
|
|
let ety = ty::expr_ty_opt(tcx, e)
|
2014-11-26 04:52:02 -05:00
|
|
|
.or_else(|| ast_ty_to_prim_ty(tcx, &**target_ty))
|
2014-05-16 10:45:16 -07:00
|
|
|
.unwrap_or_else(|| {
|
2014-04-22 15:56:37 +03:00
|
|
|
tcx.sess.span_fatal(target_ty.span,
|
|
|
|
"target type not found for const cast")
|
2014-05-16 10:45:16 -07:00
|
|
|
});
|
2014-01-29 00:05:11 -05:00
|
|
|
|
2014-09-07 20:45:12 +02:00
|
|
|
macro_rules! define_casts(
|
|
|
|
($val:ident, {
|
|
|
|
$($ty_pat:pat => (
|
|
|
|
$intermediate_ty:ty,
|
|
|
|
$const_type:ident,
|
|
|
|
$target_ty:ty
|
|
|
|
)),*
|
2014-10-31 10:51:16 +02:00
|
|
|
}) => (match ety.sty {
|
2014-09-07 20:45:12 +02:00
|
|
|
$($ty_pat => {
|
|
|
|
match $val {
|
|
|
|
const_bool(b) => Ok($const_type(b as $intermediate_ty as $target_ty)),
|
|
|
|
const_uint(u) => Ok($const_type(u as $intermediate_ty as $target_ty)),
|
|
|
|
const_int(i) => Ok($const_type(i as $intermediate_ty as $target_ty)),
|
|
|
|
const_float(f) => Ok($const_type(f as $intermediate_ty as $target_ty)),
|
|
|
|
_ => Err(concat!(
|
|
|
|
"can't cast this type to ", stringify!($const_type)
|
|
|
|
).to_string())
|
2013-07-02 12:47:32 -07:00
|
|
|
}
|
2014-09-07 20:45:12 +02:00
|
|
|
},)*
|
|
|
|
_ => Err("can't cast this type".to_string())
|
|
|
|
})
|
|
|
|
)
|
|
|
|
|
|
|
|
eval_const_expr_partial(tcx, &**base)
|
|
|
|
.and_then(|val| define_casts!(val, {
|
|
|
|
ty::ty_int(ast::TyI) => (int, const_int, i64),
|
|
|
|
ty::ty_int(ast::TyI8) => (i8, const_int, i64),
|
|
|
|
ty::ty_int(ast::TyI16) => (i16, const_int, i64),
|
|
|
|
ty::ty_int(ast::TyI32) => (i32, const_int, i64),
|
|
|
|
ty::ty_int(ast::TyI64) => (i64, const_int, i64),
|
|
|
|
ty::ty_uint(ast::TyU) => (uint, const_uint, u64),
|
|
|
|
ty::ty_uint(ast::TyU8) => (u8, const_uint, u64),
|
|
|
|
ty::ty_uint(ast::TyU16) => (u16, const_uint, u64),
|
|
|
|
ty::ty_uint(ast::TyU32) => (u32, const_uint, u64),
|
|
|
|
ty::ty_uint(ast::TyU64) => (u64, const_uint, u64),
|
|
|
|
ty::ty_float(ast::TyF32) => (f32, const_float, f64),
|
|
|
|
ty::ty_float(ast::TyF64) => (f64, const_float, f64)
|
|
|
|
}))
|
2012-03-22 14:56:56 -07:00
|
|
|
}
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::ExprPath(_) => {
|
2014-04-22 15:56:37 +03:00
|
|
|
match lookup_const(tcx, e) {
|
|
|
|
Some(actual_e) => eval_const_expr_partial(tcx, &*actual_e),
|
2014-05-25 03:17:19 -07:00
|
|
|
None => Err("non-constant path in constant expr".to_string())
|
2012-10-15 12:27:09 -07:00
|
|
|
}
|
|
|
|
}
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::ExprLit(ref lit) => Ok(lit_to_const(&**lit)),
|
|
|
|
ast::ExprParen(ref e) => eval_const_expr_partial(tcx, &**e),
|
|
|
|
ast::ExprBlock(ref block) => {
|
2014-05-04 10:39:11 +02:00
|
|
|
match block.expr {
|
|
|
|
Some(ref expr) => eval_const_expr_partial(tcx, &**expr),
|
|
|
|
None => Ok(const_int(0i64))
|
|
|
|
}
|
|
|
|
}
|
2014-11-24 12:46:02 +01:00
|
|
|
ast::ExprTupField(ref base, index) => {
|
|
|
|
// Get the base tuple if it is constant
|
|
|
|
if let Some(&ast::ExprTup(ref fields)) = lookup_const(tcx, &**base).map(|s| &s.node) {
|
|
|
|
// Check that the given index is within bounds and evaluate its value
|
|
|
|
if fields.len() > index.node {
|
|
|
|
return eval_const_expr_partial(tcx, &*fields[index.node])
|
|
|
|
} else {
|
|
|
|
return Err("tuple index out of bounds".to_string())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Err("non-constant struct in constant expr".to_string())
|
|
|
|
}
|
|
|
|
ast::ExprField(ref base, field_name) => {
|
|
|
|
// Get the base expression if it is a struct and it is constant
|
|
|
|
if let Some(&ast::ExprStruct(_, ref fields, _)) = lookup_const(tcx, &**base)
|
|
|
|
.map(|s| &s.node) {
|
|
|
|
// Check that the given field exists and evaluate it
|
|
|
|
if let Some(f) = fields.iter().find(|f|
|
|
|
|
f.ident.node.as_str() == field_name.node.as_str()) {
|
|
|
|
return eval_const_expr_partial(tcx, &*f.expr)
|
|
|
|
} else {
|
|
|
|
return Err("nonexistent struct field".to_string())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Err("non-constant struct in constant expr".to_string())
|
|
|
|
}
|
2014-05-25 03:17:19 -07:00
|
|
|
_ => Err("unsupported constant expr".to_string())
|
2012-03-22 14:56:56 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-13 20:10:34 +03:00
|
|
|
pub fn lit_to_const(lit: &ast::Lit) -> const_val {
|
2012-08-06 12:34:08 -07:00
|
|
|
match lit.node {
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::LitStr(ref s, _) => const_str((*s).clone()),
|
|
|
|
ast::LitBinary(ref data) => {
|
2014-03-20 22:10:44 -07:00
|
|
|
const_binary(Rc::new(data.iter().map(|x| *x).collect()))
|
2014-02-28 15:25:15 -08:00
|
|
|
}
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::LitByte(n) => const_uint(n as u64),
|
|
|
|
ast::LitChar(n) => const_uint(n as u64),
|
|
|
|
ast::LitInt(n, ast::SignedIntLit(_, ast::Plus)) |
|
|
|
|
ast::LitInt(n, ast::UnsuffixedIntLit(ast::Plus)) => const_int(n as i64),
|
|
|
|
ast::LitInt(n, ast::SignedIntLit(_, ast::Minus)) |
|
|
|
|
ast::LitInt(n, ast::UnsuffixedIntLit(ast::Minus)) => const_int(-(n as i64)),
|
|
|
|
ast::LitInt(n, ast::UnsignedIntLit(_)) => const_uint(n),
|
|
|
|
ast::LitFloat(ref n, _) |
|
|
|
|
ast::LitFloatUnsuffixed(ref n) => {
|
2014-01-15 17:15:39 -08:00
|
|
|
const_float(from_str::<f64>(n.get()).unwrap() as f64)
|
|
|
|
}
|
2014-09-13 20:10:34 +03:00
|
|
|
ast::LitBool(b) => const_bool(b)
|
2012-03-22 14:56:56 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-29 17:45:07 -07:00
|
|
|
fn compare_vals<T: PartialOrd>(a: T, b: T) -> Option<int> {
|
2013-06-22 18:58:41 -07:00
|
|
|
Some(if a == b { 0 } else if a < b { -1 } else { 1 })
|
|
|
|
}
|
2013-05-21 18:04:55 +09:00
|
|
|
pub fn compare_const_vals(a: &const_val, b: &const_val) -> Option<int> {
|
2013-06-22 18:58:41 -07:00
|
|
|
match (a, b) {
|
|
|
|
(&const_int(a), &const_int(b)) => compare_vals(a, b),
|
|
|
|
(&const_uint(a), &const_uint(b)) => compare_vals(a, b),
|
|
|
|
(&const_float(a), &const_float(b)) => compare_vals(a, b),
|
2014-01-10 14:02:36 -08:00
|
|
|
(&const_str(ref a), &const_str(ref b)) => compare_vals(a, b),
|
2013-06-22 18:58:41 -07:00
|
|
|
(&const_bool(a), &const_bool(b)) => compare_vals(a, b),
|
2014-06-07 15:32:01 +01:00
|
|
|
(&const_binary(ref a), &const_binary(ref b)) => compare_vals(a, b),
|
2013-06-22 18:58:41 -07:00
|
|
|
_ => None
|
2013-05-21 18:04:55 +09:00
|
|
|
}
|
2012-03-22 14:56:56 -07:00
|
|
|
}
|
|
|
|
|
2014-03-06 05:07:47 +02:00
|
|
|
pub fn compare_lit_exprs(tcx: &ty::ctxt, a: &Expr, b: &Expr) -> Option<int> {
|
2013-05-21 18:04:55 +09:00
|
|
|
compare_const_vals(&eval_const_expr(tcx, a), &eval_const_expr(tcx, b))
|
2012-03-22 14:56:56 -07:00
|
|
|
}
|