/*
// Full copyright information is available in the file ../doc/CREDITS
//
// Write and retrieve objects to disk.
*/
#include "defs.h"
#include <string.h>
#include <limits.h>
#include "cdc_db.h"
#include "macros.h"
#define COMPRESS ENABLED
#define ORDER_BYTES DISABLED
#ifndef LONG_MIN
#error "Your OS doesnt define LONG_MIN!! You can try to define it as"
#error "(-2147483647-1), which is the correct value for a 32 bit long."
#endif
/* Write a four-byte number to fp in a consistent byte-order. */
void write_long(Long n, FILE *fp)
{
#if COMPRESS
Int sign, i, h, buf[5];
if (n == LONG_MIN)
{
fputc((int) 0xA0, fp);
fputc((int) 0x00, fp);
fputc((int) 0x00, fp);
fputc((int) 0x00, fp);
fputc((int) 0x08, fp);
return;
}
sign = n<0 ? 1 : 0;
n = abs(n);
h = 1;
buf[0] = n&15;
n >>= 4;
while (n) {
buf[h++] = n & 255;
n >>= 8;
}
buf[0] += (h << 5) + (sign << 4);
for (i=0; i<h; i++)
fputc((int) buf[i], fp);
#else
# if ORDER_BYTES
/* Since first byte is special, special-case 0 as well. */
if (!n) {
fputc((int) 96, fp);
return;
}
/* First byte depends on sign. */
fputc((int) (n > 0) ? 64 + (n % 32) : 32 + (-n % 32), fp);
n = (n > 0) ? n / 32 : -n / 32;
while (n) {
fputc((int) 32 + (n % 64), fp);
n /= 64;
}
fputc((int) 96, fp);
# else
fwrite(&n, sizeof(Long), 1, fp);
# endif
#endif
}
/* Read a four-byte number in a consistent byte-order. */
Long read_long(FILE *fp)
{
#if COMPRESS
Int sign, i, h, n, k;
h = (unsigned)getc(fp) & 255;
sign = h & 16;
n = h & 15;
k = 4;
h >>= 5;
h--;
for (i=0; i<h; i++) {
n += ((unsigned)getc(fp) & 255) << k;
k += 8;
}
if (sign) n=-n;
return n;
#else
# if ORDER_BYTES
Int c;
Long n, place;
/* Check for initial terminator, meaning 0. */
c = getc(fp);
if (c == 96)
return 0;
/* Initial byte determines sign. */
n = (c < 64) ? -((c - 32) % 32) : ((c - 64) % 32);
place = (c < 64) ? -32 : 32;
forever {
c = getc(fp);
if (c == 96)
return n;
n += place * (c - 32);
place *= 64;
}
# else
Long l;
fread(&l, sizeof(Long), 1, fp);
return l;
# endif
#endif
}
Int size_long(Long n)
{
#if COMPRESS
Int h;
if (n == LONG_MIN)
return 5;
n = abs(n);
h = 1;
n >>= 4;
while (n) {
h++;
n >>= 8;
}
return h;
#else
# if ORDER_BYTES
Int count = 2;
if (!n)
return 1;
n /= 32;
while (n) {
n /= 64;
count++;
}
return count;
# else
return sizeof(n);
# endif
#endif
}
void write_ident(Long id, FILE *fp)
{
Char *s;
Int len;
if (id == NOT_AN_IDENT) {
write_long(NOT_AN_IDENT, fp);
return;
}
s = ident_name(id);
len = strlen(s);
write_long(len, fp);
fwrite(s, sizeof(Char), len, fp);
}
Long read_ident(FILE *fp)
{
Int len;
Char *s;
Long id;
/* Read the length of the identifier. */
len = read_long(fp);
/* If the length is -1, it's not really an identifier, but a -1 signalling
* a blank variable or method. */
if (len == NOT_AN_IDENT)
return NOT_AN_IDENT;
/* Otherwise, it's an identifier. Read it into temporary storage. */
s = TMALLOC(Char, len + 1);
fread(s, sizeof(Char), len, fp);
s[len] = 0;
/* Get the index for the identifier and free the temporary memory. */
id = ident_get(s);
tfree_chars(s);
return id;
}
Long size_ident(Long id)
{
Int len = strlen(ident_name(id));
return size_long(len) + (len * sizeof(Char));
}
INTERNAL void pack_list(cList *list, FILE *fp) {
cData *d;
write_long(list_length(list), fp);
for (d = list_first(list); d; d = list_next(list, d))
pack_data(d, fp);
}
INTERNAL cList *unpack_list(FILE *fp) {
Int len, i;
cList *list;
cData *d;
len = read_long(fp);
list = list_new(len);
d = list_empty_spaces(list, len);
for (i = 0; i < len; i++)
unpack_data(d++, fp);
return list;
}
INTERNAL Int size_list(cList *list)
{
cData *d;
Int size = 0;
size += size_long(list_length(list));
for (d = list_first(list); d; d = list_next(list, d))
size += size_data(d);
return size;
}
INTERNAL void pack_dict(cDict *dict, FILE *fp)
{
Int i;
pack_list(dict->keys, fp);
pack_list(dict->values, fp);
write_long(dict->hashtab_size, fp);
for (i = 0; i < dict->hashtab_size; i++) {
write_long(dict->links[i], fp);
write_long(dict->hashtab[i], fp);
}
}
INTERNAL cDict *unpack_dict(FILE *fp)
{
cDict *dict;
Int i;
dict = EMALLOC(cDict, 1);
dict->keys = unpack_list(fp);
dict->values = unpack_list(fp);
dict->hashtab_size = read_long(fp);
dict->links = EMALLOC(Int, dict->hashtab_size);
dict->hashtab = EMALLOC(Int, dict->hashtab_size);
for (i = 0; i < dict->hashtab_size; i++) {
dict->links[i] = read_long(fp);
dict->hashtab[i] = read_long(fp);
}
dict->refs = 1;
return dict;
}
INTERNAL Int size_dict(cDict *dict)
{
Int size = 0, i;
size += size_list(dict->keys);
size += size_list(dict->values);
size += size_long(dict->hashtab_size);
for (i = 0; i < dict->hashtab_size; i++) {
size += size_long(dict->links[i]);
size += size_long(dict->hashtab[i]);
}
return size;
}
INTERNAL void pack_vars(Obj *obj, FILE *fp)
{
Int i;
write_long(obj->vars.size, fp);
write_long(obj->vars.blanks, fp);
for (i = 0; i < obj->vars.size; i++) {
write_long(obj->vars.hashtab[i], fp);
if (obj->vars.tab[i].name != NOT_AN_IDENT) {
write_ident(obj->vars.tab[i].name, fp);
write_long(obj->vars.tab[i].cclass, fp);
pack_data(&obj->vars.tab[i].val, fp);
} else {
write_long(NOT_AN_IDENT, fp);
}
write_long(obj->vars.tab[i].next, fp);
}
}
INTERNAL void unpack_vars(Obj *obj, FILE *fp)
{
Int i;
obj->vars.size = read_long(fp);
obj->vars.blanks = read_long(fp);
obj->vars.hashtab = EMALLOC(Int, obj->vars.size);
obj->vars.tab = EMALLOC(Var, obj->vars.size);
for (i = 0; i < obj->vars.size; i++) {
obj->vars.hashtab[i] = read_long(fp);
obj->vars.tab[i].name = read_ident(fp);
if (obj->vars.tab[i].name != NOT_AN_IDENT) {
obj->vars.tab[i].cclass = read_long(fp);
unpack_data(&obj->vars.tab[i].val, fp);
}
obj->vars.tab[i].next = read_long(fp);
}
}
INTERNAL Int size_vars(Obj *obj)
{
Int size = 0, i;
size += size_long(obj->vars.size);
size += size_long(obj->vars.blanks);
for (i = 0; i < obj->vars.size; i++) {
size += size_long(obj->vars.hashtab[i]);
if (obj->vars.tab[i].name != NOT_AN_IDENT) {
size += size_ident(obj->vars.tab[i].name);
size += size_long(obj->vars.tab[i].cclass);
size += size_data(&obj->vars.tab[i].val);
} else {
size += size_long(NOT_AN_IDENT);
}
size += size_long(obj->vars.tab[i].next);
}
return size;
}
INTERNAL void pack_method(Method *method, FILE *fp)
{
Int i, j;
write_ident(method->name, fp);
write_long(method->num_args, fp);
for (i = 0; i < method->num_args; i++)
write_long(method->argnames[i], fp);
write_long(method->rest, fp);
write_long(method->num_vars, fp);
for (i = 0; i < method->num_vars; i++)
write_long(method->varnames[i], fp);
write_long(method->num_opcodes, fp);
for (i = 0; i < method->num_opcodes; i++)
write_long(method->opcodes[i], fp);
write_long(method->num_error_lists, fp);
for (i = 0; i < method->num_error_lists; i++) {
write_long(method->error_lists[i].num_errors, fp);
for (j = 0; j < method->error_lists[i].num_errors; j++)
write_ident(method->error_lists[i].error_ids[j], fp);
}
write_long(method->m_access, fp);
write_long(method->m_flags, fp);
write_long(method->native, fp);
}
INTERNAL Method *unpack_method(FILE *fp)
{
Int name, i, j, n;
Method *method;
/* Read in the name. If this is -1, it was a marker for a blank entry. */
name = read_ident(fp);
if (name == NOT_AN_IDENT)
return NULL;
method = EMALLOC(Method, 1);
method->name = name;
method->num_args = read_long(fp);
if (method->num_args) {
method->argnames = TMALLOC(Int, method->num_args);
for (i = 0; i < method->num_args; i++)
method->argnames[i] = read_long(fp);
}
method->rest = read_long(fp);
method->num_vars = read_long(fp);
if (method->num_vars) {
method->varnames = TMALLOC(Int, method->num_vars);
for (i = 0; i < method->num_vars; i++)
method->varnames[i] = read_long(fp);
}
method->num_opcodes = read_long(fp);
method->opcodes = TMALLOC(Long, method->num_opcodes);
for (i = 0; i < method->num_opcodes; i++)
method->opcodes[i] = read_long(fp);
method->num_error_lists = read_long(fp);
if (method->num_error_lists) {
method->error_lists = TMALLOC(Error_list, method->num_error_lists);
for (i = 0; i < method->num_error_lists; i++) {
n = read_long(fp);
method->error_lists[i].num_errors = n;
method->error_lists[i].error_ids = TMALLOC(Int, n);
for (j = 0; j < n; j++)
method->error_lists[i].error_ids[j] = read_ident(fp);
}
}
method->m_access = read_long(fp);
method->m_flags = read_long(fp);
method->native = read_long(fp);
method->refs = 1;
return method;
}
INTERNAL Int size_method(Method *method)
{
Int size = 0, i, j;
size += size_ident(method->name);
size += size_long(method->num_args);
for (i = 0; i < method->num_args; i++)
size += size_long(method->argnames[i]);
size += size_long(method->rest);
size += size_long(method->num_vars);
for (i = 0; i < method->num_vars; i++)
size += size_long(method->varnames[i]);
size += size_long(method->num_opcodes);
for (i = 0; i < method->num_opcodes; i++)
size += size_long(method->opcodes[i]);
size += size_long(method->num_error_lists);
for (i = 0; i < method->num_error_lists; i++) {
size += size_long(method->error_lists[i].num_errors);
for (j = 0; j < method->error_lists[i].num_errors; j++)
size += size_ident(method->error_lists[i].error_ids[j]);
}
size += size_long(method->native);
size += size_long(method->m_access);
size += size_long(method->m_flags);
return size;
}
INTERNAL void pack_methods(Obj *obj, FILE *fp)
{
Int i;
write_long(obj->methods.size, fp);
write_long(obj->methods.blanks, fp);
for (i = 0; i < obj->methods.size; i++) {
write_long(obj->methods.hashtab[i], fp);
if (obj->methods.tab[i].m) {
pack_method(obj->methods.tab[i].m, fp);
} else {
/* Method begins with name identifier; write NOT_AN_IDENT. */
write_long(NOT_AN_IDENT, fp);
}
write_long(obj->methods.tab[i].next, fp);
}
}
INTERNAL void unpack_methods(Obj *obj, FILE *fp)
{
Int i;
obj->methods.size = read_long(fp);
obj->methods.blanks = read_long(fp);
obj->methods.hashtab = EMALLOC(Int, obj->methods.size);
obj->methods.tab = EMALLOC(struct mptr, obj->methods.size);
for (i = 0; i < obj->methods.size; i++) {
obj->methods.hashtab[i] = read_long(fp);
obj->methods.tab[i].m = unpack_method(fp);
if (obj->methods.tab[i].m)
obj->methods.tab[i].m->object = obj;
obj->methods.tab[i].next = read_long(fp);
}
}
INTERNAL Int size_methods(Obj *obj)
{
Int size = 0, i;
size += size_long(obj->methods.size);
size += size_long(obj->methods.blanks);
for (i = 0; i < obj->methods.size; i++) {
size += size_long(obj->methods.hashtab[i]);
if (obj->methods.tab[i].m)
size += size_method(obj->methods.tab[i].m);
else
size += size_long(NOT_AN_IDENT);
size += size_long(obj->methods.tab[i].next);
}
return size;
}
INTERNAL void pack_strings(Obj *obj, FILE *fp)
{
Int i;
write_long(obj->strings_size, fp);
write_long(obj->num_strings, fp);
for (i = 0; i < obj->num_strings; i++) {
string_pack(obj->strings[i].str, fp);
if (obj->strings[i].str)
write_long(obj->strings[i].refs, fp);
}
}
INTERNAL void unpack_strings(Obj *obj, FILE *fp)
{
Int i;
obj->strings_size = read_long(fp);
obj->num_strings = read_long(fp);
obj->strings = EMALLOC(String_entry, obj->strings_size);
for (i = 0; i < obj->num_strings; i++) {
obj->strings[i].str = string_unpack(fp);
if (obj->strings[i].str)
obj->strings[i].refs = read_long(fp);
}
}
INTERNAL Int size_strings(Obj *obj)
{
Int size = 0, i;
size += size_long(obj->strings_size);
size += size_long(obj->num_strings);
for (i = 0; i < obj->num_strings; i++) {
size += string_packed_size(obj->strings[i].str);
if (obj->strings[i].str)
size += size_long(obj->strings[i].refs);
}
return size;
}
INTERNAL void pack_idents(Obj *obj, FILE *fp)
{
Int i;
write_long(obj->idents_size, fp);
write_long(obj->num_idents, fp);
for (i = 0; i < obj->num_idents; i++) {
if (obj->idents[i].id != NOT_AN_IDENT) {
write_ident(obj->idents[i].id, fp);
write_long(obj->idents[i].refs, fp);
} else {
write_long(NOT_AN_IDENT, fp);
}
}
}
INTERNAL void unpack_idents(Obj *obj, FILE *fp)
{
Int i;
obj->idents_size = read_long(fp);
obj->num_idents = read_long(fp);
obj->idents = EMALLOC(Ident_entry, obj->idents_size);
for (i = 0; i < obj->num_idents; i++) {
obj->idents[i].id = read_ident(fp);
if (obj->idents[i].id != NOT_AN_IDENT)
obj->idents[i].refs = read_long(fp);
}
}
INTERNAL Int size_idents(Obj *obj)
{
Int size = 0, i;
size += size_long(obj->idents_size);
size += size_long(obj->num_idents);
for (i = 0; i < obj->num_idents; i++) {
if (obj->idents[i].id != NOT_AN_IDENT) {
size += size_ident(obj->idents[i].id);
size += size_long(obj->idents[i].refs);
} else {
size += size_long(NOT_AN_IDENT);
}
}
return size;
}
void pack_data(cData *data, FILE *fp)
{
write_long(data->type, fp);
switch (data->type) {
case INTEGER:
write_long(data->u.val, fp);
break;
case FLOAT:
write_long(*((Long*)(&data->u.fval)), fp);
break;
case STRING:
string_pack(data->u.str, fp);
break;
case OBJNUM:
write_long(data->u.objnum, fp);
break;
case LIST:
pack_list(data->u.list, fp);
break;
case SYMBOL:
write_ident(data->u.symbol, fp);
break;
case T_ERROR:
write_ident(data->u.error, fp);
break;
case FROB:
write_long(data->u.frob->cclass, fp);
pack_data(&data->u.frob->rep, fp);
break;
case DICT:
pack_dict(data->u.dict, fp);
break;
case BUFFER: {
Int i;
write_long(data->u.buffer->len, fp);
for (i = 0; i < data->u.buffer->len; i++)
write_long(data->u.buffer->s[i], fp);
break;
}
default: {
INSTANCE_RECORD(data->type, r);
r->pack(data, fp);
}
}
}
void unpack_data(cData *data, FILE *fp)
{
data->type = read_long(fp);
switch (data->type) {
case INTEGER:
data->u.val = read_long(fp);
break;
case FLOAT: {
Long k = read_long(fp);
data->u.fval = *((cFloat*)(&k));
break;
}
case STRING:
data->u.str = string_unpack(fp);
break;
case OBJNUM:
data->u.objnum = read_long(fp);
break;
case LIST:
data->u.list = unpack_list(fp);
break;
case SYMBOL:
data->u.symbol = read_ident(fp);
break;
case T_ERROR:
data->u.error = read_ident(fp);
break;
case FROB:
data->u.frob = TMALLOC(cFrob, 1);
data->u.frob->cclass = read_long(fp);
unpack_data(&data->u.frob->rep, fp);
break;
case DICT:
data->u.dict = unpack_dict(fp);
break;
case BUFFER: {
Int len, i;
len = read_long(fp);
data->u.buffer = buffer_new(len);
for (i = 0; i < len; i++)
data->u.buffer->s[i] = read_long(fp);
break;
}
default: {
INSTANCE_RECORD(data->type, r);
r->unpack(data, fp);
}
}
}
Int size_data(cData *data) {
Int size = 0;
size += size_long(data->type);
switch (data->type) {
case INTEGER:
size += size_long(data->u.val);
break;
case FLOAT:
size += size_long(*((Long*)(&data->u.fval)));
break;
case STRING:
size += string_packed_size(data->u.str);
break;
case OBJNUM:
size += size_long(data->u.objnum);
break;
case LIST:
size += size_list(data->u.list);
break;
case SYMBOL:
size += size_ident(data->u.symbol);
break;
case T_ERROR:
size += size_ident(data->u.error);
break;
case FROB:
size += size_long(data->u.frob->cclass);
size += size_data(&data->u.frob->rep);
break;
case DICT:
size += size_dict(data->u.dict);
break;
case BUFFER: {
Int i;
size += size_long(data->u.buffer->len);
for (i = 0; i < data->u.buffer->len; i++)
size += size_long(data->u.buffer->s[i]);
break;
}
default: {
INSTANCE_RECORD(data->type, r);
size += r->size(data);
}
}
return size;
}
void pack_object(Obj *obj, FILE *fp)
{
pack_list(obj->parents, fp);
pack_list(obj->children, fp);
pack_vars(obj, fp);
pack_methods(obj, fp);
pack_strings(obj, fp);
pack_idents(obj, fp);
write_ident(obj->objname, fp);
#if 0
write_long(obj->search, fp);
#endif
}
void unpack_object(Obj *obj, FILE *fp)
{
obj->parents = unpack_list(fp);
obj->children = unpack_list(fp);
unpack_vars(obj, fp);
unpack_methods(obj, fp);
unpack_strings(obj, fp);
unpack_idents(obj, fp);
obj->objname = read_ident(fp);
#if 0
obj->search = read_long(fp);
#endif
}
Int size_object(Obj *obj)
{
Int size = 0;
size = size_list(obj->parents);
size += size_list(obj->children);
size += size_vars(obj);
size += size_methods(obj);
size += size_strings(obj);
size += size_idents(obj);
if (obj->objname != -1)
size += size_ident(obj->objname);
#if 0
size += size_long(obj->search);
#endif
return size;
}