2011-08-03 20:25:44 -07:00
|
|
|
// Functions that interpret the shape of a type to perform various low-level
|
|
|
|
// actions, such as copying, freeing, comparing, and so on.
|
|
|
|
|
|
|
|
#include <algorithm>
|
2011-08-10 14:30:09 -07:00
|
|
|
#include <iomanip>
|
2011-08-09 19:01:15 -07:00
|
|
|
#include <iostream>
|
2011-08-10 12:55:29 -07:00
|
|
|
#include <sstream>
|
2011-08-03 20:25:44 -07:00
|
|
|
#include <utility>
|
|
|
|
#include <cassert>
|
2011-08-10 14:30:09 -07:00
|
|
|
#include <cctype>
|
2011-08-03 20:25:44 -07:00
|
|
|
#include <cstdio>
|
|
|
|
#include <cstdlib>
|
|
|
|
#include "rust_internal.h"
|
2011-08-19 18:51:09 -07:00
|
|
|
#include "rust_shape.h"
|
2011-08-03 20:25:44 -07:00
|
|
|
|
|
|
|
namespace shape {
|
|
|
|
|
|
|
|
using namespace shape;
|
|
|
|
|
|
|
|
// Constants
|
|
|
|
|
|
|
|
const uint8_t CMP_EQ = 0u;
|
|
|
|
const uint8_t CMP_LT = 1u;
|
|
|
|
const uint8_t CMP_LE = 2u;
|
|
|
|
|
2011-08-17 18:14:47 -07:00
|
|
|
|
2011-08-24 15:18:57 -07:00
|
|
|
// Type parameters
|
|
|
|
|
|
|
|
type_param *
|
|
|
|
type_param::make(const type_desc **tydescs, unsigned n_tydescs,
|
|
|
|
arena &arena) {
|
|
|
|
if (!n_tydescs)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
type_param *ptrs = arena.alloc<type_param>(n_tydescs);
|
|
|
|
for (uint32_t i = 0; i < n_tydescs; i++) {
|
|
|
|
const type_desc *subtydesc = tydescs[i];
|
|
|
|
ptrs[i].shape = subtydesc->shape;
|
|
|
|
ptrs[i].tables = subtydesc->shape_tables;
|
|
|
|
ptrs[i].params = from_tydesc(subtydesc, arena);
|
|
|
|
}
|
|
|
|
return ptrs;
|
|
|
|
}
|
|
|
|
|
2011-08-26 17:05:05 -07:00
|
|
|
// Constructs type parameters from a function shape. This is a bit messy,
|
|
|
|
// because it requires that the function shape have a specific format.
|
|
|
|
type_param *
|
|
|
|
type_param::from_fn_shape(const uint8_t *sp, ptr dp, arena &arena) {
|
|
|
|
const type_desc *tydesc = bump_dp<const type_desc *>(dp);
|
|
|
|
const type_desc **descs = (const type_desc **)(dp + tydesc->size);
|
|
|
|
unsigned n_tydescs = tydesc->n_obj_params & 0x7fffffff;
|
|
|
|
return make(descs, n_tydescs, arena);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Constructs type parameters from an object shape. This is also a bit messy,
|
2011-08-24 16:54:10 -07:00
|
|
|
// because it requires that the object shape have a specific format.
|
2011-08-24 15:18:57 -07:00
|
|
|
type_param *
|
2011-08-24 16:54:10 -07:00
|
|
|
type_param::from_obj_shape(const uint8_t *sp, ptr dp, arena &arena) {
|
|
|
|
uint8_t shape = *sp++; assert(shape == SHAPE_STRUCT);
|
|
|
|
get_u16_bump(sp); // Skip over the size.
|
|
|
|
shape = *sp++; assert(shape == SHAPE_PTR);
|
|
|
|
shape = *sp++; assert(shape == SHAPE_STRUCT);
|
|
|
|
|
|
|
|
unsigned n_tydescs = get_u16_bump(sp);
|
|
|
|
|
|
|
|
// Type descriptors start right after the reference count.
|
|
|
|
const type_desc **descs = (const type_desc **)(dp + sizeof(uintptr_t));
|
|
|
|
|
|
|
|
return make(descs, n_tydescs, arena);
|
2011-08-24 15:18:57 -07:00
|
|
|
}
|
2011-08-17 18:14:47 -07:00
|
|
|
|
|
|
|
|
2011-08-03 20:25:44 -07:00
|
|
|
// A shape printer, useful for debugging
|
|
|
|
|
|
|
|
void
|
2011-08-05 16:46:51 -07:00
|
|
|
print::walk_tag(bool align, tag_info &tinfo) {
|
|
|
|
DPRINT("tag%u", tinfo.tag_id);
|
|
|
|
if (!tinfo.n_params)
|
2011-08-03 20:25:44 -07:00
|
|
|
return;
|
|
|
|
|
|
|
|
DPRINT("<");
|
|
|
|
|
|
|
|
bool first = true;
|
2011-08-05 16:46:51 -07:00
|
|
|
for (uint16_t i = 0; i < tinfo.n_params; i++) {
|
2011-08-03 20:25:44 -07:00
|
|
|
if (!first)
|
|
|
|
DPRINT(",");
|
|
|
|
first = false;
|
|
|
|
|
2011-08-05 16:46:51 -07:00
|
|
|
ctxt<print> sub(*this, tinfo.params[i].shape);
|
2011-08-03 20:25:44 -07:00
|
|
|
sub.walk(align);
|
|
|
|
}
|
|
|
|
|
|
|
|
DPRINT(">");
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
print::walk_struct(bool align, const uint8_t *end_sp) {
|
|
|
|
DPRINT("(");
|
|
|
|
|
|
|
|
bool first = true;
|
|
|
|
while (sp != end_sp) {
|
|
|
|
if (!first)
|
|
|
|
DPRINT(",");
|
|
|
|
first = false;
|
|
|
|
|
|
|
|
walk(align);
|
|
|
|
}
|
|
|
|
|
|
|
|
DPRINT(")");
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
print::walk_res(bool align, const rust_fn *dtor, uint16_t n_ty_params,
|
|
|
|
const uint8_t *ty_params_sp) {
|
|
|
|
DPRINT("res@%p", dtor);
|
|
|
|
if (!n_ty_params)
|
|
|
|
return;
|
|
|
|
|
|
|
|
DPRINT("<");
|
|
|
|
|
|
|
|
bool first = true;
|
|
|
|
for (uint16_t i = 0; i < n_ty_params; i++) {
|
|
|
|
if (!first)
|
|
|
|
DPRINT(",");
|
|
|
|
first = false;
|
|
|
|
get_u16_bump(sp); // Skip over the size.
|
|
|
|
walk(align);
|
|
|
|
}
|
|
|
|
|
|
|
|
DPRINT(">");
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
print::walk_var(bool align, uint8_t param_index) {
|
|
|
|
DPRINT("%c=", 'T' + param_index);
|
|
|
|
|
|
|
|
const type_param *param = ¶ms[param_index];
|
|
|
|
print sub(*this, param->shape, param->params, param->tables);
|
|
|
|
sub.walk(align);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<>
|
|
|
|
void print::walk_number<uint8_t>(bool align) { DPRINT("u8"); }
|
|
|
|
template<>
|
|
|
|
void print::walk_number<uint16_t>(bool align) { DPRINT("u16"); }
|
|
|
|
template<>
|
|
|
|
void print::walk_number<uint32_t>(bool align) { DPRINT("u32"); }
|
|
|
|
template<>
|
|
|
|
void print::walk_number<uint64_t>(bool align) { DPRINT("u64"); }
|
|
|
|
template<>
|
|
|
|
void print::walk_number<int8_t>(bool align) { DPRINT("i8"); }
|
|
|
|
template<>
|
|
|
|
void print::walk_number<int16_t>(bool align) { DPRINT("i16"); }
|
|
|
|
template<>
|
|
|
|
void print::walk_number<int32_t>(bool align) { DPRINT("i32"); }
|
|
|
|
template<>
|
|
|
|
void print::walk_number<int64_t>(bool align) { DPRINT("i64"); }
|
|
|
|
template<>
|
|
|
|
void print::walk_number<float>(bool align) { DPRINT("f32"); }
|
|
|
|
template<>
|
|
|
|
void print::walk_number<double>(bool align) { DPRINT("f64"); }
|
|
|
|
|
|
|
|
|
|
|
|
void
|
2011-08-05 17:50:12 -07:00
|
|
|
size_of::compute_tag_size(tag_info &tinfo) {
|
2011-08-03 20:25:44 -07:00
|
|
|
// If the precalculated size and alignment are good, use them.
|
2011-08-05 17:50:12 -07:00
|
|
|
if (tinfo.tag_sa.is_set())
|
2011-08-03 20:25:44 -07:00
|
|
|
return;
|
|
|
|
|
2011-08-05 16:46:51 -07:00
|
|
|
uint16_t n_largest_variants = get_u16_bump(tinfo.largest_variants_ptr);
|
2011-08-05 17:50:12 -07:00
|
|
|
tinfo.tag_sa.set(0, 0);
|
2011-08-03 20:25:44 -07:00
|
|
|
for (uint16_t i = 0; i < n_largest_variants; i++) {
|
2011-08-05 16:46:51 -07:00
|
|
|
uint16_t variant_id = get_u16_bump(tinfo.largest_variants_ptr);
|
2011-08-08 18:29:20 -07:00
|
|
|
std::pair<const uint8_t *,const uint8_t *> variant_ptr_and_end =
|
|
|
|
get_variant_sp(tinfo, variant_id);
|
|
|
|
const uint8_t *variant_ptr = variant_ptr_and_end.first;
|
|
|
|
const uint8_t *variant_end = variant_ptr_and_end.second;
|
2011-08-03 20:25:44 -07:00
|
|
|
|
2011-08-09 12:12:49 -07:00
|
|
|
size_of sub(*this, variant_ptr, tinfo.params, NULL);
|
2011-08-03 20:25:44 -07:00
|
|
|
|
|
|
|
// Compute the size of this variant.
|
|
|
|
size_align variant_sa;
|
|
|
|
bool first = true;
|
|
|
|
while (sub.sp != variant_end) {
|
|
|
|
if (!first)
|
2011-08-08 14:26:58 -07:00
|
|
|
variant_sa.size = align_to(variant_sa.size, sub.sa.alignment);
|
2011-08-03 20:25:44 -07:00
|
|
|
sub.walk(!first);
|
|
|
|
first = false;
|
|
|
|
|
|
|
|
variant_sa.add(sub.sa.size, sub.sa.alignment);
|
|
|
|
}
|
|
|
|
|
2011-08-05 17:50:12 -07:00
|
|
|
if (tinfo.tag_sa.size < variant_sa.size)
|
|
|
|
tinfo.tag_sa = variant_sa;
|
2011-08-03 20:25:44 -07:00
|
|
|
}
|
|
|
|
|
2011-08-05 16:46:51 -07:00
|
|
|
if (tinfo.variant_count == 1) {
|
2011-08-05 17:50:12 -07:00
|
|
|
if (!tinfo.tag_sa.size)
|
|
|
|
tinfo.tag_sa.set(1, 1);
|
2011-08-03 20:25:44 -07:00
|
|
|
} else {
|
|
|
|
// Add in space for the tag.
|
2011-08-05 17:50:12 -07:00
|
|
|
tinfo.tag_sa.add(sizeof(uint32_t), ALIGNOF(uint32_t));
|
2011-08-03 20:25:44 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-05 17:50:12 -07:00
|
|
|
void
|
|
|
|
size_of::walk_tag(bool align, tag_info &tinfo) {
|
|
|
|
compute_tag_size(*this, tinfo);
|
|
|
|
sa = tinfo.tag_sa;
|
|
|
|
}
|
|
|
|
|
2011-08-03 20:25:44 -07:00
|
|
|
void
|
|
|
|
size_of::walk_struct(bool align, const uint8_t *end_sp) {
|
|
|
|
size_align struct_sa(0, 1);
|
|
|
|
|
|
|
|
bool first = true;
|
|
|
|
while (sp != end_sp) {
|
|
|
|
if (!first)
|
2011-08-08 14:26:58 -07:00
|
|
|
struct_sa.size = align_to(struct_sa.size, sa.alignment);
|
2011-08-03 20:25:44 -07:00
|
|
|
walk(!first);
|
|
|
|
first = false;
|
|
|
|
|
|
|
|
struct_sa.add(sa);
|
|
|
|
}
|
|
|
|
|
|
|
|
sa = struct_sa;
|
|
|
|
}
|
|
|
|
|
2011-08-05 16:46:51 -07:00
|
|
|
|
|
|
|
// Copy constructors
|
|
|
|
|
2011-08-05 17:50:12 -07:00
|
|
|
#if 0
|
|
|
|
|
2011-08-08 14:26:58 -07:00
|
|
|
class copy : public data<copy,uint8_t *> {
|
2011-08-05 16:46:51 -07:00
|
|
|
// TODO
|
|
|
|
};
|
|
|
|
|
2011-08-05 17:50:12 -07:00
|
|
|
#endif
|
|
|
|
|
2011-08-05 18:47:12 -07:00
|
|
|
|
|
|
|
// Structural comparison glue.
|
|
|
|
|
|
|
|
class cmp : public data<cmp,ptr_pair> {
|
2011-08-09 12:12:49 -07:00
|
|
|
friend class data<cmp,ptr_pair>;
|
|
|
|
|
2011-08-08 14:26:58 -07:00
|
|
|
private:
|
2011-08-09 12:49:04 -07:00
|
|
|
void walk_vec(bool align, bool is_pod,
|
|
|
|
const std::pair<ptr_pair,ptr_pair> &data_range);
|
2011-08-08 14:26:58 -07:00
|
|
|
|
2011-08-09 16:26:03 -07:00
|
|
|
inline void walk_subcontext(bool align, cmp &sub) {
|
2011-08-09 12:12:49 -07:00
|
|
|
sub.walk(align);
|
|
|
|
result = sub.result;
|
|
|
|
}
|
|
|
|
|
2011-08-10 15:36:15 -07:00
|
|
|
inline void walk_box_contents(bool align, cmp &sub,
|
|
|
|
ptr_pair &ref_count_dp) {
|
|
|
|
sub.walk(true);
|
|
|
|
result = sub.result;
|
|
|
|
}
|
|
|
|
|
2011-08-09 16:26:03 -07:00
|
|
|
inline void cmp_two_pointers(bool align) {
|
|
|
|
if (align) dp = align_to(dp, ALIGNOF(uint8_t *) * 2);
|
|
|
|
data_pair<uint8_t *> fst = bump_dp<uint8_t *>(dp);
|
|
|
|
data_pair<uint8_t *> snd = bump_dp<uint8_t *>(dp);
|
|
|
|
cmp_number(fst);
|
|
|
|
if (!result)
|
|
|
|
cmp_number(snd);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void cmp_pointer(bool align) {
|
|
|
|
if (align) dp = align_to(dp, ALIGNOF(uint8_t *));
|
|
|
|
cmp_number(bump_dp<uint8_t *>(dp));
|
|
|
|
}
|
|
|
|
|
2011-08-09 12:49:04 -07:00
|
|
|
template<typename T>
|
|
|
|
void cmp_number(const data_pair<T> &nums) {
|
|
|
|
result = (nums.fst < nums.snd) ? -1 : (nums.fst == nums.snd) ? 0 : 1;
|
|
|
|
}
|
|
|
|
|
2011-08-08 14:26:58 -07:00
|
|
|
public:
|
|
|
|
int result;
|
|
|
|
|
|
|
|
cmp(rust_task *in_task,
|
|
|
|
const uint8_t *in_sp,
|
|
|
|
const type_param *in_params,
|
|
|
|
const rust_shape_tables *in_tables,
|
|
|
|
uint8_t *in_data_0,
|
|
|
|
uint8_t *in_data_1)
|
|
|
|
: data<cmp,ptr_pair>(in_task, in_sp, in_params, in_tables,
|
2011-08-08 16:51:32 -07:00
|
|
|
ptr_pair::make(in_data_0, in_data_1)),
|
|
|
|
result(0) {}
|
2011-08-08 14:26:58 -07:00
|
|
|
|
2011-08-08 18:29:20 -07:00
|
|
|
cmp(const cmp &other,
|
|
|
|
const uint8_t *in_sp = NULL,
|
|
|
|
const type_param *in_params = NULL,
|
|
|
|
const rust_shape_tables *in_tables = NULL)
|
|
|
|
: data<cmp,ptr_pair>(other.task,
|
|
|
|
in_sp ? in_sp : other.sp,
|
|
|
|
in_params ? in_params : other.params,
|
|
|
|
in_tables ? in_tables : other.tables,
|
2011-08-09 15:15:43 -07:00
|
|
|
other.dp),
|
|
|
|
result(0) {}
|
2011-08-08 18:29:20 -07:00
|
|
|
|
2011-08-08 16:51:32 -07:00
|
|
|
cmp(const cmp &other, const ptr_pair &in_dp)
|
|
|
|
: data<cmp,ptr_pair>(other.task, other.sp, other.params, other.tables,
|
|
|
|
in_dp),
|
|
|
|
result(0) {}
|
|
|
|
|
2011-08-09 12:49:04 -07:00
|
|
|
void walk_evec(bool align, bool is_pod, uint16_t sp_size) {
|
2011-08-09 19:01:15 -07:00
|
|
|
walk_vec(align, is_pod, get_evec_data_range(dp));
|
2011-08-09 12:49:04 -07:00
|
|
|
}
|
|
|
|
|
2011-08-25 10:18:02 +02:00
|
|
|
void walk_vec(bool align, bool is_pod, uint16_t sp_size) {
|
|
|
|
walk_vec(align, is_pod, get_vec_data_range(dp));
|
2011-08-09 12:49:04 -07:00
|
|
|
}
|
|
|
|
|
2011-08-10 15:36:15 -07:00
|
|
|
void walk_box(bool align) {
|
|
|
|
data<cmp,ptr_pair>::walk_box_contents(align);
|
|
|
|
}
|
|
|
|
|
2011-08-09 16:26:03 -07:00
|
|
|
void walk_fn(bool align) { return cmp_two_pointers(align); }
|
|
|
|
void walk_obj(bool align) { return cmp_two_pointers(align); }
|
|
|
|
void walk_port(bool align) { return cmp_pointer(align); }
|
|
|
|
void walk_chan(bool align) { return cmp_pointer(align); }
|
|
|
|
void walk_task(bool align) { return cmp_pointer(align); }
|
|
|
|
|
2011-08-08 14:26:58 -07:00
|
|
|
void walk_tag(bool align, tag_info &tinfo,
|
2011-08-08 17:29:21 -07:00
|
|
|
const data_pair<uint32_t> &tag_variants);
|
2011-08-08 19:16:33 -07:00
|
|
|
void walk_struct(bool align, const uint8_t *end_sp);
|
2011-08-08 14:26:58 -07:00
|
|
|
void walk_res(bool align, const rust_fn *dtor, uint16_t n_ty_params,
|
|
|
|
const uint8_t *ty_params_sp);
|
2011-08-08 18:29:20 -07:00
|
|
|
void walk_variant(bool align, tag_info &tinfo, uint32_t variant_id,
|
|
|
|
const std::pair<const uint8_t *,const uint8_t *>
|
|
|
|
variant_ptr_and_end);
|
2011-08-08 14:26:58 -07:00
|
|
|
|
|
|
|
template<typename T>
|
2011-08-09 12:10:40 -07:00
|
|
|
void walk_number() { cmp_number(get_dp<T>(dp)); }
|
2011-08-05 18:47:12 -07:00
|
|
|
};
|
|
|
|
|
2011-08-09 12:10:40 -07:00
|
|
|
template<>
|
|
|
|
void cmp::cmp_number<int32_t>(const data_pair<int32_t> &nums) {
|
|
|
|
result = (nums.fst < nums.snd) ? -1 : (nums.fst == nums.snd) ? 0 : 1;
|
|
|
|
}
|
|
|
|
|
2011-08-08 16:51:32 -07:00
|
|
|
void
|
2011-08-09 12:49:04 -07:00
|
|
|
cmp::walk_vec(bool align, bool is_pod,
|
|
|
|
const std::pair<ptr_pair,ptr_pair> &data_range) {
|
2011-08-08 19:16:33 -07:00
|
|
|
cmp sub(*this, data_range.first);
|
|
|
|
ptr_pair data_end = data_range.second;
|
|
|
|
while (!result && sub.dp < data_end) {
|
2011-08-09 12:10:40 -07:00
|
|
|
sub.walk_reset(align);
|
2011-08-08 19:16:33 -07:00
|
|
|
result = sub.result;
|
2011-08-08 16:51:32 -07:00
|
|
|
align = true;
|
2011-08-09 12:10:40 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!result) {
|
|
|
|
// If we hit the end, the result comes down to length comparison.
|
|
|
|
int len_fst = data_range.second.fst - data_range.first.fst;
|
|
|
|
int len_snd = data_range.second.snd - data_range.first.snd;
|
|
|
|
cmp_number(data_pair<int>::make(len_fst, len_snd));
|
2011-08-08 16:51:32 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-08 14:26:58 -07:00
|
|
|
void
|
|
|
|
cmp::walk_tag(bool align, tag_info &tinfo,
|
2011-08-08 17:29:21 -07:00
|
|
|
const data_pair<uint32_t> &tag_variants) {
|
|
|
|
cmp_number(tag_variants);
|
|
|
|
if (result != 0)
|
|
|
|
return;
|
2011-08-08 18:29:20 -07:00
|
|
|
data<cmp,ptr_pair>::walk_variant(align, tinfo, tag_variants.fst);
|
2011-08-08 14:26:58 -07:00
|
|
|
}
|
|
|
|
|
2011-08-08 19:16:33 -07:00
|
|
|
void
|
|
|
|
cmp::walk_struct(bool align, const uint8_t *end_sp) {
|
|
|
|
while (!result && this->sp != end_sp) {
|
|
|
|
this->walk(align);
|
|
|
|
align = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-08 14:26:58 -07:00
|
|
|
void
|
|
|
|
cmp::walk_res(bool align, const rust_fn *dtor, uint16_t n_ty_params,
|
|
|
|
const uint8_t *ty_params_sp) {
|
|
|
|
abort(); // TODO
|
|
|
|
}
|
|
|
|
|
2011-08-08 18:29:20 -07:00
|
|
|
void
|
|
|
|
cmp::walk_variant(bool align, tag_info &tinfo, uint32_t variant_id,
|
|
|
|
const std::pair<const uint8_t *,const uint8_t *>
|
|
|
|
variant_ptr_and_end) {
|
2011-08-09 12:12:49 -07:00
|
|
|
cmp sub(*this, variant_ptr_and_end.first, tinfo.params);
|
2011-08-08 18:29:20 -07:00
|
|
|
|
|
|
|
const uint8_t *variant_end = variant_ptr_and_end.second;
|
2011-08-08 19:16:33 -07:00
|
|
|
while (!result && sub.sp < variant_end) {
|
2011-08-08 18:29:20 -07:00
|
|
|
sub.walk(align);
|
2011-08-08 19:16:33 -07:00
|
|
|
result = sub.result;
|
2011-08-08 18:29:20 -07:00
|
|
|
align = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-09 19:01:15 -07:00
|
|
|
|
|
|
|
// Polymorphic logging, for convenience
|
|
|
|
|
2011-08-10 14:30:09 -07:00
|
|
|
void
|
|
|
|
log::walk_string(const std::pair<ptr,ptr> &data) {
|
|
|
|
out << "\"" << std::hex;
|
|
|
|
|
|
|
|
ptr subdp = data.first;
|
|
|
|
while (subdp < data.second) {
|
|
|
|
char ch = *subdp;
|
|
|
|
if (isprint(ch))
|
|
|
|
out << ch;
|
|
|
|
else if (ch)
|
|
|
|
out << "\\x" << std::setw(2) << std::setfill('0') << (int)ch;
|
|
|
|
++subdp;
|
|
|
|
}
|
|
|
|
|
|
|
|
out << "\"" << std::dec;
|
|
|
|
}
|
|
|
|
|
2011-08-10 15:36:15 -07:00
|
|
|
void
|
|
|
|
log::walk_struct(bool align, const uint8_t *end_sp) {
|
|
|
|
out << "(";
|
|
|
|
|
|
|
|
bool first = true;
|
|
|
|
while (sp != end_sp) {
|
|
|
|
if (!first)
|
|
|
|
out << ", ";
|
|
|
|
walk(align);
|
|
|
|
align = true, first = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
out << ")";
|
|
|
|
}
|
|
|
|
|
2011-08-09 19:01:15 -07:00
|
|
|
void
|
2011-08-10 12:55:29 -07:00
|
|
|
log::walk_vec(bool align, bool is_pod, const std::pair<ptr,ptr> &data) {
|
2011-08-10 14:30:09 -07:00
|
|
|
if (peek() == SHAPE_U8) {
|
|
|
|
sp++; // It's a string. We handle this ourselves.
|
|
|
|
walk_string(data);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2011-08-10 14:42:06 -07:00
|
|
|
out << "[";
|
|
|
|
|
|
|
|
log sub(*this, data.first);
|
|
|
|
|
|
|
|
bool first = true;
|
|
|
|
while (sub.dp < data.second) {
|
|
|
|
if (!first)
|
|
|
|
out << ", ";
|
|
|
|
|
|
|
|
sub.walk_reset(align);
|
|
|
|
|
|
|
|
align = true;
|
|
|
|
first = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
out << "]";
|
2011-08-09 19:01:15 -07:00
|
|
|
}
|
|
|
|
|
2011-08-10 14:54:29 -07:00
|
|
|
void
|
|
|
|
log::walk_variant(bool align, tag_info &tinfo, uint32_t variant_id,
|
|
|
|
const std::pair<const uint8_t *,const uint8_t *>
|
|
|
|
variant_ptr_and_end) {
|
|
|
|
log sub(*this, variant_ptr_and_end.first, tinfo.params);
|
|
|
|
const uint8_t *variant_end = variant_ptr_and_end.second;
|
|
|
|
|
|
|
|
bool first = true;
|
|
|
|
while (sub.sp < variant_end) {
|
|
|
|
out << (first ? "(" : ", ");
|
|
|
|
|
|
|
|
sub.walk(align);
|
|
|
|
|
|
|
|
align = true;
|
|
|
|
first = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!first)
|
|
|
|
out << ")";
|
|
|
|
}
|
|
|
|
|
2011-08-03 20:25:44 -07:00
|
|
|
} // end namespace shape
|
|
|
|
|
2011-08-08 14:26:58 -07:00
|
|
|
extern "C" void
|
|
|
|
upcall_cmp_type(int8_t *result, rust_task *task, type_desc *tydesc,
|
|
|
|
const type_desc **subtydescs, uint8_t *data_0,
|
|
|
|
uint8_t *data_1, uint8_t cmp_type) {
|
|
|
|
shape::arena arena;
|
2011-08-24 15:18:57 -07:00
|
|
|
shape::type_param *params = shape::type_param::from_tydesc(tydesc, arena);
|
2011-08-08 14:26:58 -07:00
|
|
|
shape::cmp cmp(task, tydesc->shape, params, tydesc->shape_tables, data_0,
|
|
|
|
data_1);
|
|
|
|
cmp.walk(true);
|
|
|
|
|
|
|
|
switch (cmp_type) {
|
|
|
|
case shape::CMP_EQ: *result = cmp.result == 0; break;
|
|
|
|
case shape::CMP_LT: *result = cmp.result < 0; break;
|
|
|
|
case shape::CMP_LE: *result = cmp.result <= 0; break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-10 12:55:29 -07:00
|
|
|
extern "C" void
|
|
|
|
upcall_log_type(rust_task *task, type_desc *tydesc, uint8_t *data,
|
|
|
|
uint32_t level) {
|
|
|
|
if (task->sched->log_lvl < level)
|
|
|
|
return; // TODO: Don't evaluate at all?
|
|
|
|
|
|
|
|
shape::arena arena;
|
2011-08-24 15:18:57 -07:00
|
|
|
shape::type_param *params = shape::type_param::from_tydesc(tydesc, arena);
|
2011-08-10 12:55:29 -07:00
|
|
|
|
|
|
|
std::stringstream ss;
|
|
|
|
shape::log log(task, tydesc->shape, params, tydesc->shape_tables, data,
|
|
|
|
ss);
|
|
|
|
|
|
|
|
log.walk(true);
|
|
|
|
|
|
|
|
task->sched->log(task, level, "%s", ss.str().c_str());
|
|
|
|
}
|
|
|
|
|