upscheme/c/print.h

910 lines
26 KiB
C
Raw Normal View History

2009-07-05 23:56:48 -04:00
extern void *memrchr(const void *s, int c, size_t n);
2019-08-25 05:49:56 -04:00
// *print-readably* -- use `write` repr instead of `display` repr
static int print_readably;
2019-08-25 05:41:24 -04:00
// *print-pretty* -- indent instead of printing everything on one long line
static int print_pretty;
// *print-width* -- maximum line length when indenting, ignored when not
2019-08-25 05:45:48 -04:00
static int print_width = 80;
2019-08-25 05:41:24 -04:00
// *print-length* -- truncate lists after N items and write "..."
static fixnum_t print_length;
2019-08-25 05:41:24 -04:00
// *print-level* -- print only the outermost N levels of nested structures
static fixnum_t print_level;
2008-06-30 21:54:22 -04:00
2019-08-25 05:58:40 -04:00
static fixnum_t cur_line;
static fixnum_t cur_column = 0;
static fixnum_t cur_level;
static uint32_t cycle_used_labels;
static struct htable cycle_visited_pairs;
2019-08-09 12:26:20 -04:00
static void outc(char c, struct ios *f)
2008-06-30 21:54:22 -04:00
{
ios_putc(c, f);
2009-07-05 23:56:48 -04:00
if (c == '\n')
2019-08-25 05:58:40 -04:00
cur_column = 0;
2009-07-05 23:56:48 -04:00
else
2019-08-25 05:58:40 -04:00
cur_column++;
2008-06-30 21:54:22 -04:00
}
2019-08-09 12:26:20 -04:00
static void outs(char *s, struct ios *f)
2008-06-30 21:54:22 -04:00
{
ios_puts(s, f);
2019-08-25 05:58:40 -04:00
cur_column += u8_strwidth(s);
2008-06-30 21:54:22 -04:00
}
2019-08-09 12:26:20 -04:00
static void outsn(char *s, struct ios *f, size_t n)
{
ios_write(f, s, n);
2019-08-25 05:58:40 -04:00
cur_column += u8_strwidth(s);
}
2019-08-09 12:26:20 -04:00
static int outindent(int n, struct ios *f)
2008-06-30 21:54:22 -04:00
{
int n0;
// move back to left margin if we get too indented
2019-08-25 05:45:48 -04:00
if (n > print_width - 12)
n = 2;
n0 = n;
ios_putc('\n', f);
2019-08-25 05:58:40 -04:00
cur_line++;
cur_column = n;
2008-06-30 21:54:22 -04:00
while (n) {
ios_putc(' ', f);
2008-06-30 21:54:22 -04:00
n--;
}
return n0;
2008-06-30 21:54:22 -04:00
}
2019-08-09 12:26:20 -04:00
void fl_print_chr(char c, struct ios *f) { outc(c, f); }
2019-08-09 12:26:20 -04:00
void fl_print_str(char *s, struct ios *f) { outs(s, f); }
void print_traverse(value_t v)
2008-06-30 21:54:22 -04:00
{
value_t *bp;
2008-06-30 21:54:22 -04:00
while (iscons(v)) {
if (ismarked(v)) {
bp = (value_t *)ptrhash_bp(&cycle_visited_pairs, (void *)v);
if (*bp == (value_t)HT_NOTFOUND)
*bp = fixnum(cycle_used_labels++);
2008-06-30 21:54:22 -04:00
return;
}
mark_cons(v);
print_traverse(car_(v));
v = cdr_(v);
}
if (!ismanaged(v) || issymbol(v))
return;
if (ismarked(v)) {
bp = (value_t *)ptrhash_bp(&cycle_visited_pairs, (void *)v);
if (*bp == (value_t)HT_NOTFOUND)
*bp = fixnum(cycle_used_labels++);
return;
}
if (isvector(v)) {
unsigned int i;
if (vector_size(v) > 0)
mark_cons(v);
for (i = 0; i < vector_size(v); i++)
print_traverse(vector_elt(v, i));
} else if (iscprim(v)) {
// don't consider shared references to e.g. chars
} else if (isclosure(v)) {
struct function *f;
mark_cons(v);
f = (struct function *)ptr(v);
print_traverse(f->bcode);
print_traverse(f->vals);
print_traverse(f->env);
} else {
struct cvalue *cv;
struct fltype *t;
assert(iscvalue(v));
cv = (struct cvalue *)ptr(v);
// don't consider shared references to ""
if (!cv_isstr(cv) || cv_len(cv) != 0)
2008-06-30 21:54:22 -04:00
mark_cons(v);
t = cv_class(cv);
if (t->vtable != NULL && t->vtable->print_traverse != NULL)
t->vtable->print_traverse(v);
2008-06-30 21:54:22 -04:00
}
}
2019-08-09 12:26:20 -04:00
static void print_symbol_name(struct ios *f, char *name)
2008-06-30 21:54:22 -04:00
{
int i, escape, charescape;
2008-06-30 21:54:22 -04:00
escape = charescape = 0;
if ((name[0] == '\0') || (name[0] == '.' && name[1] == '\0') ||
(name[0] == '#') || isnumtok(name, NULL))
2008-06-30 21:54:22 -04:00
escape = 1;
i = 0;
2008-06-30 21:54:22 -04:00
while (name[i]) {
if (!symchar(name[i])) {
escape = 1;
if (name[i] == '|' || name[i] == '\\') {
2008-06-30 21:54:22 -04:00
charescape = 1;
break;
}
}
i++;
}
if (escape) {
if (charescape) {
outc('|', f);
i = 0;
2008-06-30 21:54:22 -04:00
while (name[i]) {
if (name[i] == '|' || name[i] == '\\')
2008-06-30 21:54:22 -04:00
outc('\\', f);
outc(name[i], f);
i++;
}
outc('|', f);
} else {
2008-06-30 21:54:22 -04:00
outc('|', f);
outs(name, f);
outc('|', f);
}
} else {
2008-06-30 21:54:22 -04:00
outs(name, f);
}
}
/*
The following implements a simple pretty-printing algorithm. This is
an unlimited-width approach that doesn't require an extra pass.
It uses some heuristics to guess whether an expression is "small",
and avoids wrapping symbols across lines. The result is high
performance and nice output for typical code. Quality is poor for
pathological or deeply-nested expressions, but those are difficult
to print anyway.
*/
#define SMALL_STR_LEN 20
static int tinyp(value_t v)
2008-06-30 21:54:22 -04:00
{
if (issymbol(v))
return (u8_strwidth(symbol_name(v)) < SMALL_STR_LEN);
if (fl_isstring(v))
2019-08-09 12:36:20 -04:00
return (cv_len((struct cvalue *)ptr(v)) < SMALL_STR_LEN);
return (isfixnum(v) || isbuiltin(v) || v == FL_F || v == FL_T ||
v == FL_NIL || v == FL_EOF || iscprim(v));
2008-06-30 21:54:22 -04:00
}
static int smallp(value_t v)
{
if (tinyp(v))
return 1;
if (fl_isnumber(v))
return 1;
2008-06-30 21:54:22 -04:00
if (iscons(v)) {
if (tinyp(car_(v)) &&
(tinyp(cdr_(v)) || (iscons(cdr_(v)) && tinyp(car_(cdr_(v))) &&
cdr_(cdr_(v)) == NIL)))
2008-06-30 21:54:22 -04:00
return 1;
return 0;
}
if (isvector(v)) {
size_t s = vector_size(v);
return (s == 0 || (tinyp(vector_elt(v, 0)) &&
(s == 1 || (s == 2 && tinyp(vector_elt(v, 1))))));
2008-06-30 21:54:22 -04:00
}
return 0;
}
static int specialindent(value_t head)
2008-06-30 21:54:22 -04:00
{
// indent these forms 2 spaces, not lined up with the first argument
if (head == LAMBDA || head == TRYCATCH || head == definesym ||
head == defmacrosym || head == forsym)
2008-06-30 21:54:22 -04:00
return 2;
return -1;
}
static int lengthestimate(value_t v)
{
// get the width of an expression if we can do so cheaply
if (issymbol(v))
return u8_strwidth(symbol_name(v));
2019-08-09 12:25:43 -04:00
if (iscprim(v) && cp_class((struct cprim *)ptr(v)) == wchartype)
return 4;
2008-06-30 21:54:22 -04:00
return -1;
}
static int allsmallp(value_t v)
{
int n;
n = 1;
2008-06-30 21:54:22 -04:00
while (iscons(v)) {
if (!smallp(car_(v)))
return 0;
v = cdr_(v);
n++;
if (n > 25)
return n;
}
return n;
}
static int indentafter3(value_t head, value_t v)
{
// for certain X always indent (X a b c) after b
return ((head == forsym) && !allsmallp(cdr_(v)));
}
static int indentafter2(value_t head, value_t v)
{
// for certain X always indent (X a b) after a
return ((head == definesym || head == defmacrosym) &&
!allsmallp(cdr_(v)));
}
2008-06-30 21:54:22 -04:00
static int indentevery(value_t v)
{
value_t c;
2008-06-30 21:54:22 -04:00
// indent before every subform of a special form, unless every
// subform is "small"
c = car_(v);
if (c == LAMBDA || c == setqsym)
2008-06-30 21:54:22 -04:00
return 0;
if (c == IF) // TODO: others
2008-06-30 21:54:22 -04:00
return !allsmallp(cdr_(v));
return 0;
}
static int blockindent(value_t v)
{
// in this case we switch to block indent mode, where the head
// is no longer considered special:
// (a b c d e
// f g h i j)
return (allsmallp(v) > 9);
}
2019-08-09 12:26:20 -04:00
static void print_pair(struct ios *f, value_t v)
2008-06-30 21:54:22 -04:00
{
value_t cd, head;
char *op;
2019-08-25 05:58:40 -04:00
fixnum_t last_line;
int startpos, newindent, blk, n_unindented, n, si, ind, est, always,
nextsmall, thistiny, after2, after3;
op = NULL;
2008-06-30 21:54:22 -04:00
if (iscons(cdr_(v)) && cdr_(cdr_(v)) == NIL &&
!ptrhash_has(&cycle_visited_pairs, (void *)cdr_(v)) &&
(((car_(v) == QUOTE) && (op = "'")) ||
((car_(v) == BACKQUOTE) && (op = "`")) ||
((car_(v) == COMMA) && (op = ",")) ||
((car_(v) == COMMAAT) && (op = ",@")) ||
((car_(v) == COMMADOT) && (op = ",.")))) {
2008-06-30 21:54:22 -04:00
// special prefix syntax
unmark_cons(v);
unmark_cons(cdr_(v));
outs(op, f);
fl_print_child(f, car_(cdr_(v)));
2008-06-30 21:54:22 -04:00
return;
}
2019-08-25 05:58:40 -04:00
startpos = cur_column;
2008-06-30 21:54:22 -04:00
outc('(', f);
2019-08-25 05:58:40 -04:00
newindent = cur_column;
blk = blockindent(v);
n = ind = always = 0;
if (!blk)
always = indentevery(v);
head = car_(v);
after3 = indentafter3(head, v);
after2 = indentafter2(head, v);
n_unindented = 1;
2008-06-30 21:54:22 -04:00
while (1) {
cd = cdr_(v);
if (print_length >= 0 && n >= print_length && cd != NIL) {
outsn("...)", f, 4);
break;
}
2019-08-25 05:58:40 -04:00
last_line = cur_line;
2008-06-30 21:54:22 -04:00
unmark_cons(v);
fl_print_child(f, car_(v));
if (!iscons(cd) || ptrhash_has(&cycle_visited_pairs, (void *)cd)) {
2008-06-30 21:54:22 -04:00
if (cd != NIL) {
outsn(" . ", f, 3);
fl_print_child(f, cd);
2008-06-30 21:54:22 -04:00
}
outc(')', f);
break;
}
if (!print_pretty || ((head == LAMBDA) && n == 0)) {
// never break line before lambda-list
2008-06-30 21:54:22 -04:00
ind = 0;
} else {
2008-06-30 21:54:22 -04:00
est = lengthestimate(car_(cd));
nextsmall = smallp(car_(cd));
thistiny = tinyp(car_(v));
2019-08-25 05:58:40 -04:00
ind =
(((cur_line > last_line) || (cur_column > print_width / 2 &&
!nextsmall && !thistiny && n > 0)) ||
2019-08-25 05:58:40 -04:00
(cur_column > print_width - 4) ||
2019-08-25 05:58:40 -04:00
(est != -1 && (cur_column + est > print_width - 2)) ||
2019-08-25 05:58:40 -04:00
((head == LAMBDA) && !nextsmall) ||
2019-08-25 05:58:40 -04:00
(n > 0 && always) ||
2019-08-25 05:58:40 -04:00
(n == 2 && after3) || (n == 1 && after2) ||
2019-08-25 05:58:40 -04:00
(n_unindented >= 3 && !nextsmall) ||
2019-08-25 05:58:40 -04:00
(n == 0 && !smallp(head)));
2008-06-30 21:54:22 -04:00
}
if (ind) {
newindent = outindent(newindent, f);
n_unindented = 1;
} else {
n_unindented++;
2008-06-30 21:54:22 -04:00
outc(' ', f);
if (n == 0) {
2008-06-30 21:54:22 -04:00
// set indent level after printing head
si = specialindent(head);
if (si != -1)
newindent = startpos + si;
else if (!blk)
2019-08-25 05:58:40 -04:00
newindent = cur_column;
2008-06-30 21:54:22 -04:00
}
}
n++;
v = cd;
}
}
2019-08-09 12:26:20 -04:00
static void cvalue_print(struct ios *f, value_t v);
2008-06-30 21:54:22 -04:00
static int write_cycle_prefix(struct ios *f, value_t v)
2008-06-30 21:54:22 -04:00
{
value_t label;
if ((label = (value_t)ptrhash_get(&cycle_visited_pairs, (void *)v)) !=
(value_t)HT_NOTFOUND) {
if (!ismarked(v)) {
2019-08-25 05:58:40 -04:00
cur_column += ios_printf(f, "#%ld#", numval(label));
return 1;
}
2019-08-25 05:58:40 -04:00
cur_column += ios_printf(f, "#%ld=", numval(label));
}
if (ismanaged(v))
unmark_cons(v);
return 0;
}
2019-08-09 12:26:20 -04:00
void fl_print_child(struct ios *f, value_t v)
{
2008-06-30 21:54:22 -04:00
char *name;
2019-08-25 05:58:40 -04:00
if (print_level >= 0 && cur_level >= print_level &&
(iscons(v) || isvector(v) || isclosure(v))) {
outc('#', f);
return;
}
2019-08-25 05:58:40 -04:00
cur_level++;
2008-06-30 21:54:22 -04:00
switch (tag(v)) {
case TAG_NUM:
case TAG_NUM1:
2019-08-25 05:58:40 -04:00
cur_column += ios_printf(f, "%ld", numval(v));
break;
2008-06-30 21:54:22 -04:00
case TAG_SYM:
name = symbol_name(v);
2019-08-25 05:49:56 -04:00
if (!print_readably)
2008-06-30 21:54:22 -04:00
outs(name, f);
else if (ismanaged(v)) {
outsn("#:", f, 2);
2008-06-30 21:54:22 -04:00
outs(name, f);
} else
2008-06-30 21:54:22 -04:00
print_symbol_name(f, name);
break;
case TAG_FUNCTION:
if (v == FL_T) {
outsn("#t", f, 2);
} else if (v == FL_F) {
outsn("#f", f, 2);
} else if (v == FL_NIL) {
outsn("()", f, 2);
} else if (v == FL_EOF) {
outsn("#<eof>", f, 6);
} else if (isbuiltin(v)) {
2019-08-25 05:49:56 -04:00
if (print_readably)
outsn("#.", f, 2);
2008-06-30 21:54:22 -04:00
outs(builtin_names[uintval(v)], f);
} else {
assert(isclosure(v));
2019-08-25 05:49:56 -04:00
if (print_readably) {
struct function *fn;
char *data;
size_t i, sz;
if (write_cycle_prefix(f, v))
break;
fn = (struct function *)ptr(v);
outs("#fn(", f);
data = cvalue_data(fn->bcode);
sz = cvalue_len(fn->bcode);
for (i = 0; i < sz; i++)
data[i] += 48;
2009-07-05 23:56:48 -04:00
fl_print_child(f, fn->bcode);
for (i = 0; i < sz; i++)
data[i] -= 48;
outc(' ', f);
2009-07-05 23:56:48 -04:00
fl_print_child(f, fn->vals);
if (fn->env != NIL) {
outc(' ', f);
fl_print_child(f, fn->env);
}
if (fn->name != LAMBDA) {
outc(' ', f);
fl_print_child(f, fn->name);
}
2009-07-05 23:56:48 -04:00
outc(')', f);
} else {
2009-07-05 23:56:48 -04:00
outs("#<function>", f);
}
}
break;
case TAG_CPRIM:
if (v == UNBOUND)
outs("#<undefined>", f);
else
cvalue_print(f, v);
break;
case TAG_CVALUE:
case TAG_VECTOR:
2008-06-30 21:54:22 -04:00
case TAG_CONS:
if (print_readably && write_cycle_prefix(f, v))
break;
2008-06-30 21:54:22 -04:00
if (isvector(v)) {
int newindent, est, sz, i;
2008-06-30 21:54:22 -04:00
outc('[', f);
2019-08-25 05:58:40 -04:00
newindent = cur_column;
sz = vector_size(v);
for (i = 0; i < sz; i++) {
if (print_length >= 0 && i >= print_length && i < sz - 1) {
outsn("...", f, 3);
break;
}
fl_print_child(f, vector_elt(v, i));
if (i < sz - 1) {
if (!print_pretty) {
2008-06-30 21:54:22 -04:00
outc(' ', f);
} else {
est = lengthestimate(vector_elt(v, i + 1));
2019-08-25 05:58:40 -04:00
if (cur_column > print_width - 4 ||
(est != -1 &&
(cur_column + est > print_width - 2)) ||
(cur_column > print_width / 2 &&
!smallp(vector_elt(v, i + 1)) &&
!tinyp(vector_elt(v, i))))
newindent = outindent(newindent, f);
2008-06-30 21:54:22 -04:00
else
outc(' ', f);
}
}
}
outc(']', f);
break;
}
if (iscvalue(v))
cvalue_print(f, v);
else
print_pair(f, v);
2008-06-30 21:54:22 -04:00
break;
}
2019-08-25 05:58:40 -04:00
cur_level--;
2008-06-30 21:54:22 -04:00
}
2019-08-09 12:26:20 -04:00
static void print_string(struct ios *f, char *str, size_t sz)
2008-06-30 21:54:22 -04:00
{
char buf[512];
size_t i = 0;
2009-04-22 20:22:03 -04:00
uint8_t c;
static char hexdig[] = "0123456789abcdef";
2008-06-30 21:54:22 -04:00
outc('"', f);
2009-04-22 20:22:03 -04:00
if (!u8_isvalid(str, sz)) {
// alternate print algorithm that preserves data if it's not UTF-8
for (i = 0; i < sz; i++) {
2009-04-22 20:22:03 -04:00
c = str[i];
if (c == '\\')
outsn("\\\\", f, 2);
else if (c == '"')
outsn("\\\"", f, 2);
else if (c >= 32 && c < 0x7f)
outc(c, f);
else {
outsn("\\x", f, 2);
outc(hexdig[c >> 4], f);
outc(hexdig[c & 0xf], f);
}
2009-04-22 20:22:03 -04:00
}
} else {
2009-04-22 20:22:03 -04:00
while (i < sz) {
size_t n = u8_escape(buf, sizeof(buf), str, &i, sz, 1, 0);
outsn(buf, f, n - 1);
2009-04-22 20:22:03 -04:00
}
2008-06-30 21:54:22 -04:00
}
outc('"', f);
}
int double_exponent(double d)
{
union ieee754_double dl;
dl.d = d;
return dl.ieee.exponent - IEEE754_DOUBLE_BIAS;
}
void snprint_real(char *s, size_t cnt, double r,
int width, // printf field width, or 0
int dec, // # decimal digits desired, recommend 16
// # of zeros in .00...0x before using scientific notation
// recommend 3-4 or so
int max_digs_rt,
// # of digits left of decimal before scientific notation
// recommend 10
int max_digs_lf)
{
int mag;
double fpart, temp;
char format[8];
char num_format[3];
int sz, keepz = 0;
s[0] = '\0';
if (width == -1) {
width = 0;
keepz = 1;
}
if (isnan(r)) {
if (sign_bit(r))
strncpy(s, "-nan", cnt);
else
strncpy(s, "nan", cnt);
return;
}
if (r == 0) {
strncpy(s, "0", cnt);
return;
}
num_format[0] = 'l';
num_format[2] = '\0';
mag = double_exponent(r);
mag = (int)(((double)mag) / LOG2_10 + 0.5);
if (r == 0)
mag = 0;
if ((mag > max_digs_lf - 1) || (mag < -max_digs_rt)) {
num_format[1] = 'e';
temp = r / pow(10, mag); /* see if number will have a decimal */
fpart = temp - floor(temp); /* when written in scientific notation */
} else {
num_format[1] = 'f';
fpart = r - floor(r);
}
if (fpart == 0)
dec = 0;
if (width == 0) {
snprintf(format, 8, "%%.%d%s", dec, num_format);
} else {
snprintf(format, 8, "%%%d.%d%s", width, dec, num_format);
}
sz = snprintf(s, cnt, format, r);
/* trim trailing zeros from fractions. not when using scientific
notation, since we might have e.g. 1.2000e+100. also not when we
need a specific output width */
if (width == 0 && !keepz) {
if (sz > 2 && fpart && num_format[1] != 'e') {
while (s[sz - 1] == '0') {
s[sz - 1] = '\0';
sz--;
}
// don't need trailing .
if (s[sz - 1] == '.') {
s[sz - 1] = '\0';
sz--;
}
}
}
// TODO. currently 1.1e20 prints as 1.1000000000000000e+20; be able to
// get rid of all those zeros.
}
2008-06-30 21:54:22 -04:00
static numerictype_t sym_to_numtype(value_t type);
// 'weak' means we don't need to accurately reproduce the type, so
// for example #int32(0) can be printed as just 0. this is used
// printing in a context where a type is already implied, e.g. inside
// an array.
2019-08-09 12:26:20 -04:00
static void cvalue_printdata(struct ios *f, void *data, size_t len,
value_t type, int weak)
2008-06-30 21:54:22 -04:00
{
if (type == bytesym) {
unsigned char ch = *(unsigned char *)data;
2019-08-25 05:49:56 -04:00
if (!print_readably)
2008-06-30 21:54:22 -04:00
outc(ch, f);
else if (weak)
2019-08-25 05:58:40 -04:00
cur_column += ios_printf(f, "#x%hhx", ch);
2008-06-30 21:54:22 -04:00
else
2019-08-25 05:58:40 -04:00
cur_column += ios_printf(f, "#byte(#x%hhx)", ch);
} else if (type == wcharsym) {
2008-06-30 21:54:22 -04:00
char seq[8];
uint32_t wc = *(uint32_t *)data;
size_t nb = u8_toutf8(seq, sizeof(seq), &wc, 1);
seq[nb] = '\0';
2019-08-25 05:49:56 -04:00
if (!print_readably) {
2008-06-30 21:54:22 -04:00
// TODO: better multibyte handling
2017-08-19 14:18:32 -04:00
if (wc == 0)
ios_putc(0, f);
else
outs(seq, f);
} else {
outsn("#\\", f, 2);
if (wc == 0x00)
outsn("nul", f, 3);
else if (wc == 0x07)
outsn("alarm", f, 5);
else if (wc == 0x08)
outsn("backspace", f, 9);
else if (wc == 0x09)
outsn("tab", f, 3);
// else if (wc == 0x0A) outsn("linefeed", f, 8);
else if (wc == 0x0A)
outsn("newline", f, 7);
else if (wc == 0x0B)
outsn("vtab", f, 4);
else if (wc == 0x0C)
outsn("page", f, 4);
else if (wc == 0x0D)
outsn("return", f, 6);
else if (wc == 0x1B)
outsn("esc", f, 3);
// else if (wc == 0x20) outsn("space", f, 5);
else if (wc == 0x7F)
outsn("delete", f, 6);
else if (iswprint(wc))
outs(seq, f);
else
2019-08-25 05:58:40 -04:00
cur_column += ios_printf(f, "x%04x", (int)wc);
2008-06-30 21:54:22 -04:00
}
} else if (type == floatsym || type == doublesym) {
2008-06-30 21:54:22 -04:00
char buf[64];
double d;
int ndec;
if (type == floatsym) {
d = (double)*(float *)data;
ndec = 8;
} else {
d = *(double *)data;
ndec = 16;
}
if (!DFINITE(d)) {
char *rep;
if (isnan(d))
rep = sign_bit(d) ? "-nan.0" : "+nan.0";
else
rep = sign_bit(d) ? "-inf.0" : "+inf.0";
2019-08-25 05:49:56 -04:00
if (type == floatsym && print_readably && !weak)
2019-08-25 05:58:40 -04:00
cur_column +=
ios_printf(f, "#%s(%s)", symbol_name(type), rep);
else
outs(rep, f);
} else if (d == 0) {
if (1 / d < 0)
outsn("-0.0", f, 4);
else
outsn("0.0", f, 3);
2019-08-25 05:49:56 -04:00
if (type == floatsym && print_readably && !weak)
outc('f', f);
} else {
int hasdec;
snprint_real(buf, sizeof(buf), d, 0, ndec, 3, 10);
hasdec = (strpbrk(buf, ".eE") != NULL);
outs(buf, f);
if (!hasdec)
outsn(".0", f, 2);
2019-08-25 05:49:56 -04:00
if (type == floatsym && print_readably && !weak)
outc('f', f);
2008-06-30 21:54:22 -04:00
}
} else if (type == uint64sym
#ifdef BITS64
|| type == ulongsym
#endif
) {
uint64_t ui64 = *(uint64_t *)data;
2019-08-25 05:49:56 -04:00
if (weak || !print_readably)
2019-08-25 05:58:40 -04:00
cur_column += ios_printf(f, "%llu", ui64);
else
2019-08-25 05:58:40 -04:00
cur_column += ios_printf(f, "#%s(%llu)", symbol_name(type), ui64);
} else if (issymbol(type)) {
// handle other integer prims. we know it's smaller than uint64
2008-06-30 21:54:22 -04:00
// at this point, so int64 is big enough to capture everything.
numerictype_t nt = sym_to_numtype(type);
if (nt == N_NUMTYPES) {
2019-08-25 05:58:40 -04:00
cur_column += ios_printf(f, "#<%s>", symbol_name(type));
} else {
int64_t i64 = conv_to_int64(data, nt);
2019-08-25 05:49:56 -04:00
if (weak || !print_readably)
2019-08-25 05:58:40 -04:00
cur_column += ios_printf(f, "%lld", i64);
else
2019-08-25 05:58:40 -04:00
cur_column +=
ios_printf(f, "#%s(%lld)", symbol_name(type), i64);
}
} else if (iscons(type)) {
2008-06-30 21:54:22 -04:00
if (car_(type) == arraysym) {
value_t eltype = car(cdr_(type));
size_t cnt, elsize, i;
2008-06-30 21:54:22 -04:00
if (iscons(cdr_(cdr_(type)))) {
cnt = toulong(car_(cdr_(cdr_(type))), "length");
elsize = cnt ? len / cnt : 0;
} else {
2008-06-30 21:54:22 -04:00
// incomplete array type
int junk;
elsize = ctype_sizeof(eltype, &junk);
cnt = elsize ? len / elsize : 0;
2008-06-30 21:54:22 -04:00
}
if (eltype == bytesym) {
2019-08-25 05:49:56 -04:00
if (!print_readably) {
ios_write(f, data, len);
2009-07-05 23:56:48 -04:00
/*
char *nl = memrchr(data, '\n', len);
if (nl)
2019-08-25 05:58:40 -04:00
cur_column = u8_strwidth(nl+1);
2009-07-05 23:56:48 -04:00
else
2019-08-25 05:58:40 -04:00
cur_column += u8_strwidth(data);
2009-07-05 23:56:48 -04:00
*/
} else {
print_string(f, (char *)data, len);
2008-06-30 21:54:22 -04:00
}
return;
} else if (eltype == wcharsym) {
2008-06-30 21:54:22 -04:00
// TODO wchar
} else {
2008-06-30 21:54:22 -04:00
}
if (!weak) {
if (eltype == uint8sym) {
outsn("#vu8(", f, 5);
} else {
outsn("#array(", f, 7);
fl_print_child(f, eltype);
if (cnt > 0)
outc(' ', f);
}
} else {
outc('[', f);
2008-06-30 21:54:22 -04:00
}
for (i = 0; i < cnt; i++) {
if (i > 0)
2008-06-30 21:54:22 -04:00
outc(' ', f);
cvalue_printdata(f, data, elsize, eltype, 1);
data = (char *)data + elsize;
2008-06-30 21:54:22 -04:00
}
if (!weak)
outc(')', f);
else
outc(']', f);
} else if (car_(type) == enumsym) {
int n = *(int *)data;
value_t syms = car(cdr_(type));
assert(isvector(syms));
2008-06-30 21:54:22 -04:00
if (!weak) {
outsn("#enum(", f, 6);
fl_print_child(f, syms);
2008-06-30 21:54:22 -04:00
outc(' ', f);
}
if (n >= (int)vector_size(syms)) {
cvalue_printdata(f, data, len, int32sym, 1);
} else {
fl_print_child(f, vector_elt(syms, n));
2008-06-30 21:54:22 -04:00
}
if (!weak)
outc(')', f);
}
}
}
2019-08-09 12:26:20 -04:00
static void cvalue_print(struct ios *f, value_t v)
2008-06-30 21:54:22 -04:00
{
2019-08-09 12:36:20 -04:00
struct cvalue *cv = (struct cvalue *)ptr(v);
void *data = cptr(v);
value_t label;
2008-06-30 21:54:22 -04:00
if (cv_class(cv) == builtintype) {
void *fptr = *(void **)data;
label = (value_t)ptrhash_get(&reverse_dlsym_lookup_table, cv);
if (label == (value_t)HT_NOTFOUND) {
2019-08-25 05:58:40 -04:00
cur_column +=
ios_printf(f, "#<builtin @#x%08zx>", (size_t)(builtin_t)fptr);
} else {
2019-08-25 05:49:56 -04:00
if (!print_readably) {
outs(symbol_name(label), f);
} else {
outsn("#fn(", f, 4);
outs(symbol_name(label), f);
outc(')', f);
}
}
} else if (cv_class(cv)->vtable != NULL &&
cv_class(cv)->vtable->print != NULL) {
cv_class(cv)->vtable->print(v, f);
} else {
value_t type = cv_type(cv);
size_t len = iscprim(v) ? cv_class(cv)->size : cv_len(cv);
cvalue_printdata(f, data, len, type, 0);
}
2008-06-30 21:54:22 -04:00
}
static void set_print_width(void)
{
value_t pw;
pw = symbol_value(printwidthsym);
if (!isfixnum(pw))
return;
2019-08-25 05:45:48 -04:00
print_width = numval(pw);
}
2019-08-09 12:26:20 -04:00
void fl_print(struct ios *f, value_t v)
2008-06-30 21:54:22 -04:00
{
value_t pl;
print_pretty = (symbol_value(printprettysym) != FL_F);
if (print_pretty)
set_print_width();
2019-08-25 05:49:56 -04:00
print_readably = (symbol_value(printreadablysym) != FL_F);
pl = symbol_value(printlengthsym);
if (isfixnum(pl))
print_length = numval(pl);
else
print_length = -1;
pl = symbol_value(printlevelsym);
if (isfixnum(pl))
print_level = numval(pl);
else
print_level = -1;
2019-08-25 05:58:40 -04:00
cur_level = 0;
cycle_used_labels = 0;
2019-08-25 05:49:56 -04:00
if (print_readably)
print_traverse(v);
2019-08-25 05:58:40 -04:00
cur_line = cur_column = 0;
fl_print_child(f, v);
if (print_level >= 0 || print_length >= 0) {
2019-08-09 12:28:14 -04:00
memset(consflags, 0,
4 * bitvector_nwords(heapsize / sizeof(struct cons)));
}
2010-05-04 19:54:07 -04:00
if ((iscons(v) || isvector(v) || isfunction(v) || iscvalue(v)) &&
!fl_isstring(v) && v != FL_T && v != FL_F && v != FL_NIL) {
htable_reset(&cycle_visited_pairs, 32);
2010-05-04 19:54:07 -04:00
}
2008-06-30 21:54:22 -04:00
}