feat(kernel/expr): compress application serialization
Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
parent
63a4a2f288
commit
d0fdc3619b
3 changed files with 27 additions and 1 deletions
Binary file not shown.
Binary file not shown.
|
@ -333,6 +333,16 @@ local_context read_local_context(deserializer & d) {
|
||||||
return to_list(entries.begin(), entries.end());
|
return to_list(entries.begin(), entries.end());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// To save space, we pack the number of arguments in small applications in the kind byte.
|
||||||
|
// The extra kinds start at g_first_app_size_kind. This value should be bigger than the
|
||||||
|
// real kinds. Moreover g_first_app_size_kind + g_small_app_num_args should fit in a byte.
|
||||||
|
constexpr char g_first_app_size_kind = 32;
|
||||||
|
constexpr char g_small_app_num_args = 32;
|
||||||
|
constexpr bool is_small(expr_kind k) { return 0 <= static_cast<char>(k) && static_cast<char>(k) < g_first_app_size_kind; }
|
||||||
|
static_assert(is_small(expr_kind::Var) && is_small(expr_kind::Constant) && is_small(expr_kind::Value) && is_small(expr_kind::App) &&
|
||||||
|
is_small(expr_kind::Lambda) && is_small(expr_kind::Pi) && is_small(expr_kind::Type) && is_small(expr_kind::Eq) &&
|
||||||
|
is_small(expr_kind::Let) && is_small(expr_kind::MetaVar), "expr_kind is too big");
|
||||||
|
|
||||||
class expr_serializer : public object_serializer<expr, expr_hash_alloc, expr_eqp> {
|
class expr_serializer : public object_serializer<expr, expr_hash_alloc, expr_eqp> {
|
||||||
typedef object_serializer<expr, expr_hash_alloc, expr_eqp> super;
|
typedef object_serializer<expr, expr_hash_alloc, expr_eqp> super;
|
||||||
public:
|
public:
|
||||||
|
@ -350,6 +360,13 @@ public:
|
||||||
super::write(a, [&]() {
|
super::write(a, [&]() {
|
||||||
serializer & s = get_owner();
|
serializer & s = get_owner();
|
||||||
auto k = a.kind();
|
auto k = a.kind();
|
||||||
|
if (k == expr_kind::App && num_args(a) < g_small_app_num_args) {
|
||||||
|
// compressed application
|
||||||
|
s << static_cast<char>(g_first_app_size_kind + num_args(a));
|
||||||
|
for (unsigned i = 0; i < num_args(a); i++)
|
||||||
|
write(arg(a, i));
|
||||||
|
return;
|
||||||
|
}
|
||||||
s << static_cast<char>(k);
|
s << static_cast<char>(k);
|
||||||
switch (k) {
|
switch (k) {
|
||||||
case expr_kind::Var: s << var_idx(a); break;
|
case expr_kind::Var: s << var_idx(a); break;
|
||||||
|
@ -386,7 +403,16 @@ public:
|
||||||
expr read() {
|
expr read() {
|
||||||
return super::read([&]() {
|
return super::read([&]() {
|
||||||
deserializer & d = get_owner();
|
deserializer & d = get_owner();
|
||||||
auto k = static_cast<expr_kind>(d.read_char());
|
char c = d.read_char();
|
||||||
|
if (c >= g_first_app_size_kind) {
|
||||||
|
// compressed application
|
||||||
|
unsigned num = c - g_first_app_size_kind;
|
||||||
|
buffer<expr> args;
|
||||||
|
for (unsigned i = 0; i < num; i++)
|
||||||
|
args.push_back(read());
|
||||||
|
return mk_app(args);
|
||||||
|
}
|
||||||
|
auto k = static_cast<expr_kind>(c);
|
||||||
switch (k) {
|
switch (k) {
|
||||||
case expr_kind::Var:
|
case expr_kind::Var:
|
||||||
return mk_var(d.read_unsigned());
|
return mk_var(d.read_unsigned());
|
||||||
|
|
Loading…
Reference in a new issue