*** empty log message ***
[bpt/guile.git] / libguile / print.c
1 /* Copyright (C) 1995-1999,2000,2001, 2002, 2003, 2004 Free Software Foundation, Inc.
2 *
3 * This library is free software; you can redistribute it and/or
4 * modify it under the terms of the GNU Lesser General Public
5 * License as published by the Free Software Foundation; either
6 * version 2.1 of the License, or (at your option) any later version.
7 *
8 * This library is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
11 * Lesser General Public License for more details.
12 *
13 * You should have received a copy of the GNU Lesser General Public
14 * License along with this library; if not, write to the Free Software
15 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
16 */
17
18
19 \f
20
21 #include <errno.h>
22
23 #include "libguile/_scm.h"
24 #include "libguile/chars.h"
25 #include "libguile/continuations.h"
26 #include "libguile/smob.h"
27 #include "libguile/eval.h"
28 #include "libguile/macros.h"
29 #include "libguile/procprop.h"
30 #include "libguile/read.h"
31 #include "libguile/weaks.h"
32 #include "libguile/unif.h"
33 #include "libguile/alist.h"
34 #include "libguile/struct.h"
35 #include "libguile/objects.h"
36 #include "libguile/ports.h"
37 #include "libguile/root.h"
38 #include "libguile/strings.h"
39 #include "libguile/strports.h"
40 #include "libguile/vectors.h"
41 #include "libguile/lang.h"
42 #include "libguile/numbers.h"
43
44 #include "libguile/validate.h"
45 #include "libguile/print.h"
46 \f
47
48 /* {Names of immediate symbols}
49 *
50 * This table must agree with the declarations in scm.h: {Immediate Symbols}.
51 */
52
53 /* This table must agree with the list of flags in tags.h. */
54 static const char *iflagnames[] =
55 {
56 "#f",
57 "#t",
58 "#<undefined>",
59 "#<eof>",
60 "()",
61 "#<unspecified>",
62
63 /* Unbound slot marker for GOOPS. For internal use in GOOPS only. */
64 "#<unbound>",
65
66 /* Elisp nil value. This is its Scheme name; whenever it's printed in
67 * Elisp, it should appear as the symbol `nil'. */
68 "#nil"
69 };
70
71 scm_t_option scm_print_opts[] = {
72 { SCM_OPTION_SCM, "closure-hook", SCM_UNPACK (SCM_BOOL_F),
73 "Hook for printing closures (should handle macros as well)." },
74 { SCM_OPTION_BOOLEAN, "source", 0,
75 "Print closures with source." },
76 { SCM_OPTION_SCM, "highlight-prefix", (unsigned long)SCM_BOOL_F,
77 "The string to print before highlighted values." },
78 { SCM_OPTION_SCM, "highlight-suffix", (unsigned long)SCM_BOOL_F,
79 "The string to print after highlighted values." }
80 };
81
82 SCM_DEFINE (scm_print_options, "print-options-interface", 0, 1, 0,
83 (SCM setting),
84 "Option interface for the print options. Instead of using\n"
85 "this procedure directly, use the procedures\n"
86 "@code{print-enable}, @code{print-disable}, @code{print-set!}\n"
87 "and @code{print-options}.")
88 #define FUNC_NAME s_scm_print_options
89 {
90 SCM ans = scm_options (setting,
91 scm_print_opts,
92 SCM_N_PRINT_OPTIONS,
93 FUNC_NAME);
94 return ans;
95 }
96 #undef FUNC_NAME
97
98 \f
99 /* {Printing of Scheme Objects}
100 */
101
102 /* Detection of circular references.
103 *
104 * Due to other constraints in the implementation, this code has bad
105 * time complexity (O (depth * N)), The printer code can be
106 * rewritten to be O(N).
107 */
108 #define PUSH_REF(pstate, obj) \
109 do { \
110 PSTATE_STACK_SET (pstate, pstate->top++, obj); \
111 if (pstate->top == pstate->ceiling) \
112 grow_ref_stack (pstate); \
113 } while(0)
114
115 #define ENTER_NESTED_DATA(pstate, obj, label) \
116 do { \
117 register unsigned long i; \
118 for (i = 0; i < pstate->top; ++i) \
119 if (scm_is_eq (PSTATE_STACK_REF (pstate, i), (obj))) \
120 goto label; \
121 if (pstate->fancyp) \
122 { \
123 if (pstate->top - pstate->list_offset >= pstate->level) \
124 { \
125 scm_putc ('#', port); \
126 return; \
127 } \
128 } \
129 PUSH_REF(pstate, obj); \
130 } while(0)
131
132 #define EXIT_NESTED_DATA(pstate) { --pstate->top; }
133
134 SCM scm_print_state_vtable = SCM_BOOL_F;
135 static SCM print_state_pool = SCM_EOL;
136 scm_i_pthread_mutex_t print_state_mutex = SCM_I_PTHREAD_MUTEX_INITIALIZER;
137
138 #ifdef GUILE_DEBUG /* Used for debugging purposes */
139
140 SCM_DEFINE (scm_current_pstate, "current-pstate", 0, 0, 0,
141 (),
142 "Return the current-pstate -- the car of the\n"
143 "@code{print_state_pool}. @code{current-pstate} is only\n"
144 "included in @code{--enable-guile-debug} builds.")
145 #define FUNC_NAME s_scm_current_pstate
146 {
147 if (!scm_is_null (print_state_pool))
148 return SCM_CAR (print_state_pool);
149 else
150 return SCM_BOOL_F;
151 }
152 #undef FUNC_NAME
153
154 #endif
155
156 #define PSTATE_SIZE 50L
157
158 static SCM
159 make_print_state (void)
160 {
161 SCM print_state
162 = scm_make_struct (scm_print_state_vtable, SCM_INUM0, SCM_EOL);
163 scm_print_state *pstate = SCM_PRINT_STATE (print_state);
164 pstate->ref_vect = scm_c_make_vector (PSTATE_SIZE, SCM_UNDEFINED);
165 pstate->ceiling = SCM_SIMPLE_VECTOR_LENGTH (pstate->ref_vect);
166 pstate->highlight_objects = SCM_EOL;
167 return print_state;
168 }
169
170 SCM
171 scm_make_print_state ()
172 {
173 SCM answer = SCM_BOOL_F;
174
175 /* First try to allocate a print state from the pool */
176 scm_i_pthread_mutex_lock (&print_state_mutex);
177 if (!scm_is_null (print_state_pool))
178 {
179 answer = SCM_CAR (print_state_pool);
180 print_state_pool = SCM_CDR (print_state_pool);
181 }
182 scm_i_pthread_mutex_unlock (&print_state_mutex);
183
184 return scm_is_false (answer) ? make_print_state () : answer;
185 }
186
187 void
188 scm_free_print_state (SCM print_state)
189 {
190 SCM handle;
191 scm_print_state *pstate = SCM_PRINT_STATE (print_state);
192 /* Cleanup before returning print state to pool.
193 * It is better to do it here. Doing it in scm_prin1
194 * would cost more since that function is called much more
195 * often.
196 */
197 pstate->fancyp = 0;
198 pstate->revealed = 0;
199 pstate->highlight_objects = SCM_EOL;
200 scm_i_pthread_mutex_lock (&print_state_mutex);
201 handle = scm_cons (print_state, print_state_pool);
202 print_state_pool = handle;
203 scm_i_pthread_mutex_unlock (&print_state_mutex);
204 }
205
206 SCM
207 scm_i_port_with_print_state (SCM port, SCM print_state)
208 {
209 if (SCM_UNBNDP (print_state))
210 {
211 if (SCM_PORT_WITH_PS_P (port))
212 return port;
213 else
214 print_state = scm_make_print_state ();
215 /* port does not need to be coerced since it doesn't have ps */
216 }
217 else
218 port = SCM_COERCE_OUTPORT (port);
219 SCM_RETURN_NEWSMOB (scm_tc16_port_with_ps,
220 SCM_UNPACK (scm_cons (port, print_state)));
221 }
222
223 static void
224 grow_ref_stack (scm_print_state *pstate)
225 {
226 SCM old_vect = pstate->ref_vect;
227 size_t old_size = SCM_SIMPLE_VECTOR_LENGTH (old_vect);
228 size_t new_size = 2 * pstate->ceiling;
229 SCM new_vect = scm_c_make_vector (new_size, SCM_UNDEFINED);
230 unsigned long int i;
231
232 for (i = 0; i != old_size; ++i)
233 SCM_SIMPLE_VECTOR_SET (new_vect, i, SCM_SIMPLE_VECTOR_REF (old_vect, i));
234
235 pstate->ref_vect = new_vect;
236 pstate->ceiling = new_size;
237 }
238
239 #define PSTATE_STACK_REF(p,i) SCM_SIMPLE_VECTOR_REF((p)->ref_vect, (i))
240 #define PSTATE_STACK_SET(p,i,v) SCM_SIMPLE_VECTOR_SET((p)->ref_vect, (i), (v))
241
242 static void
243 print_circref (SCM port, scm_print_state *pstate, SCM ref)
244 {
245 register long i;
246 long self = pstate->top - 1;
247 i = pstate->top - 1;
248 if (scm_is_pair (PSTATE_STACK_REF (pstate, i)))
249 {
250 while (i > 0)
251 {
252 if (!scm_is_pair (PSTATE_STACK_REF (pstate, i-1))
253 || !scm_is_eq (SCM_CDR (PSTATE_STACK_REF (pstate, i-1)),
254 SCM_CDR (PSTATE_STACK_REF (pstate, i))))
255 break;
256 --i;
257 }
258 self = i;
259 }
260 for (i = pstate->top - 1; 1; --i)
261 if (scm_is_eq (PSTATE_STACK_REF(pstate, i), ref))
262 break;
263 scm_putc ('#', port);
264 scm_intprint (i - self, 10, port);
265 scm_putc ('#', port);
266 }
267
268 /* Print the name of a symbol. */
269
270 void
271 scm_print_symbol_name (const char *str, size_t len, SCM port)
272 {
273 /* This points to the first character that has not yet been written to the
274 * port. */
275 size_t pos = 0;
276 /* This points to the character we're currently looking at. */
277 size_t end;
278 /* If the name contains weird characters, we'll escape them with
279 * backslashes and set this flag; it indicates that we should surround the
280 * name with "#{" and "}#". */
281 int weird = 0;
282 /* Backslashes are not sufficient to make a name weird, but if a name is
283 * weird because of other characters, backslahes need to be escaped too.
284 * The first time we see a backslash, we set maybe_weird, and mw_pos points
285 * to the backslash. Then if the name turns out to be weird, we re-process
286 * everything starting from mw_pos.
287 * We could instead make backslashes always weird. This is not necessary
288 * to ensure that the output is (read)-able, but it would make this code
289 * simpler and faster. */
290 int maybe_weird = 0;
291 size_t mw_pos = 0;
292
293 if (len == 0 || str[0] == '\'' || str[0] == '`' || str[0] == ',' ||
294 str[0] == ':' || str[len-1] == ':' || (str[0] == '.' && len == 1) ||
295 scm_is_true (scm_i_mem2number(str, len, 10)))
296 {
297 scm_lfwrite ("#{", 2, port);
298 weird = 1;
299 }
300
301 for (end = pos; end < len; ++end)
302 switch (str[end])
303 {
304 #ifdef BRACKETS_AS_PARENS
305 case '[':
306 case ']':
307 #endif
308 case '(':
309 case ')':
310 case '"':
311 case ';':
312 case '#':
313 case SCM_WHITE_SPACES:
314 case SCM_LINE_INCREMENTORS:
315 weird_handler:
316 if (maybe_weird)
317 {
318 end = mw_pos;
319 maybe_weird = 0;
320 }
321 if (!weird)
322 {
323 scm_lfwrite ("#{", 2, port);
324 weird = 1;
325 }
326 if (pos < end)
327 scm_lfwrite (str + pos, end - pos, port);
328 {
329 char buf[2];
330 buf[0] = '\\';
331 buf[1] = str[end];
332 scm_lfwrite (buf, 2, port);
333 }
334 pos = end + 1;
335 break;
336 case '\\':
337 if (weird)
338 goto weird_handler;
339 if (!maybe_weird)
340 {
341 maybe_weird = 1;
342 mw_pos = pos;
343 }
344 break;
345 default:
346 break;
347 }
348 if (pos < end)
349 scm_lfwrite (str + pos, end - pos, port);
350 if (weird)
351 scm_lfwrite ("}#", 2, port);
352 }
353
354 /* Print generally. Handles both write and display according to PSTATE.
355 */
356 SCM_GPROC(s_write, "write", 1, 1, 0, scm_write, g_write);
357 SCM_GPROC(s_display, "display", 1, 1, 0, scm_display, g_display);
358
359 static void iprin1 (SCM exp, SCM port, scm_print_state *pstate);
360
361 void
362 scm_iprin1 (SCM exp, SCM port, scm_print_state *pstate)
363 {
364 if (pstate->fancyp
365 && scm_is_true (scm_memq (exp, pstate->highlight_objects)))
366 {
367 scm_display (SCM_PRINT_HIGHLIGHT_PREFIX, port);
368 iprin1 (exp, port, pstate);
369 scm_display (SCM_PRINT_HIGHLIGHT_SUFFIX, port);
370 }
371 else
372 iprin1 (exp, port, pstate);
373 }
374
375 static void
376 iprin1 (SCM exp, SCM port, scm_print_state *pstate)
377 {
378 switch (SCM_ITAG3 (exp))
379 {
380 case scm_tc3_closure:
381 case scm_tc3_tc7_1:
382 case scm_tc3_tc7_2:
383 /* These tc3 tags should never occur in an immediate value. They are
384 * only used in cell types of non-immediates, i. e. the value returned
385 * by SCM_CELL_TYPE (exp) can use these tags.
386 */
387 scm_ipruk ("immediate", exp, port);
388 break;
389 case scm_tc3_int_1:
390 case scm_tc3_int_2:
391 scm_intprint (SCM_I_INUM (exp), 10, port);
392 break;
393 case scm_tc3_imm24:
394 if (SCM_CHARP (exp))
395 {
396 long i = SCM_CHAR (exp);
397
398 if (SCM_WRITINGP (pstate))
399 {
400 scm_puts ("#\\", port);
401 if ((i >= 0) && (i <= ' ') && scm_charnames[i])
402 scm_puts (scm_charnames[i], port);
403 #ifndef EBCDIC
404 else if (i == '\177')
405 scm_puts (scm_charnames[scm_n_charnames - 1], port);
406 #endif
407 else if (i < 0 || i > '\177')
408 scm_intprint (i, 8, port);
409 else
410 scm_putc (i, port);
411 }
412 else
413 scm_putc (i, port);
414 }
415 else if (SCM_IFLAGP (exp)
416 && ((size_t) SCM_IFLAGNUM (exp) < (sizeof iflagnames / sizeof (char *))))
417 {
418 scm_puts (iflagnames [SCM_IFLAGNUM (exp)], port);
419 }
420 else if (SCM_ISYMP (exp))
421 {
422 scm_i_print_isym (exp, port);
423 }
424 else if (SCM_ILOCP (exp))
425 {
426 scm_i_print_iloc (exp, port);
427 }
428 else
429 {
430 /* unknown immediate value */
431 scm_ipruk ("immediate", exp, port);
432 }
433 break;
434 case scm_tc3_cons:
435 switch (SCM_TYP7 (exp))
436 {
437 case scm_tcs_struct:
438 {
439 ENTER_NESTED_DATA (pstate, exp, circref);
440 if (SCM_OBJ_CLASS_FLAGS (exp) & SCM_CLASSF_GOOPS)
441 {
442 SCM pwps, print = pstate->writingp ? g_write : g_display;
443 if (!print)
444 goto print_struct;
445 pwps = scm_i_port_with_print_state (port, pstate->handle);
446 pstate->revealed = 1;
447 scm_call_generic_2 (print, exp, pwps);
448 }
449 else
450 {
451 print_struct:
452 scm_print_struct (exp, port, pstate);
453 }
454 EXIT_NESTED_DATA (pstate);
455 }
456 break;
457 case scm_tcs_cons_imcar:
458 case scm_tcs_cons_nimcar:
459 ENTER_NESTED_DATA (pstate, exp, circref);
460 scm_iprlist ("(", exp, ')', port, pstate);
461 EXIT_NESTED_DATA (pstate);
462 break;
463 circref:
464 print_circref (port, pstate, exp);
465 break;
466 case scm_tcs_closures:
467 if (scm_is_false (scm_procedure_p (SCM_PRINT_CLOSURE))
468 || scm_is_false (scm_printer_apply (SCM_PRINT_CLOSURE,
469 exp, port, pstate)))
470 {
471 SCM formals = SCM_CLOSURE_FORMALS (exp);
472 scm_puts ("#<procedure", port);
473 scm_putc (' ', port);
474 scm_iprin1 (scm_procedure_name (exp), port, pstate);
475 scm_putc (' ', port);
476 if (SCM_PRINT_SOURCE_P)
477 {
478 SCM env = SCM_ENV (exp);
479 SCM xenv = SCM_EXTEND_ENV (formals, SCM_EOL, env);
480 SCM src = scm_i_unmemocopy_body (SCM_CODE (exp), xenv);
481 ENTER_NESTED_DATA (pstate, exp, circref);
482 scm_iprin1 (src, port, pstate);
483 EXIT_NESTED_DATA (pstate);
484 }
485 else
486 scm_iprin1 (formals, port, pstate);
487 scm_putc ('>', port);
488 }
489 break;
490 case scm_tc7_number:
491 switch SCM_TYP16 (exp) {
492 case scm_tc16_big:
493 scm_bigprint (exp, port, pstate);
494 break;
495 case scm_tc16_real:
496 scm_print_real (exp, port, pstate);
497 break;
498 case scm_tc16_complex:
499 scm_print_complex (exp, port, pstate);
500 break;
501 case scm_tc16_fraction:
502 scm_i_print_fraction (exp, port, pstate);
503 break;
504 }
505 break;
506 case scm_tc7_string:
507 if (SCM_WRITINGP (pstate))
508 {
509 size_t i, j, len;
510 const char *data;
511
512 scm_putc ('"', port);
513 len = scm_i_string_length (exp);
514 data = scm_i_string_chars (exp);
515 for (i = 0, j = 0; i < len; ++i)
516 {
517 unsigned char ch = data[i];
518 if ((ch < 32 && ch != '\n') || (127 <= ch && ch < 148))
519 {
520 static char const hex[]="0123456789abcdef";
521 char buf[4];
522
523 scm_lfwrite (data+j, i-j, port);
524 buf[0] = '\\';
525 buf[1] = 'x';
526 buf[2] = hex [ch / 16];
527 buf[3] = hex [ch % 16];
528 scm_lfwrite (buf, 4, port);
529 data = scm_i_string_chars (exp);
530 j = i+1;
531 }
532 else if (ch == '"' || ch == '\\')
533 {
534 scm_lfwrite (data+j, i-j, port);
535 scm_putc ('\\', port);
536 data = scm_i_string_chars (exp);
537 j = i;
538 }
539 }
540 scm_lfwrite (data+j, i-j, port);
541 scm_putc ('"', port);
542 scm_remember_upto_here_1 (exp);
543 }
544 else
545 scm_lfwrite (scm_i_string_chars (exp), scm_i_string_length (exp),
546 port);
547 scm_remember_upto_here_1 (exp);
548 break;
549 case scm_tc7_symbol:
550 if (scm_i_symbol_is_interned (exp))
551 {
552 scm_print_symbol_name (scm_i_symbol_chars (exp),
553 scm_i_symbol_length (exp),
554 port);
555 scm_remember_upto_here_1 (exp);
556 }
557 else
558 {
559 scm_puts ("#<uninterned-symbol ", port);
560 scm_print_symbol_name (scm_i_symbol_chars (exp),
561 scm_i_symbol_length (exp),
562 port);
563 scm_putc (' ', port);
564 scm_uintprint (SCM_UNPACK (exp), 16, port);
565 scm_putc ('>', port);
566 }
567 break;
568 case scm_tc7_variable:
569 scm_i_variable_print (exp, port, pstate);
570 break;
571 case scm_tc7_wvect:
572 ENTER_NESTED_DATA (pstate, exp, circref);
573 if (SCM_IS_WHVEC (exp))
574 scm_puts ("#wh(", port);
575 else
576 scm_puts ("#w(", port);
577 goto common_vector_printer;
578
579 case scm_tc7_vector:
580 ENTER_NESTED_DATA (pstate, exp, circref);
581 scm_puts ("#(", port);
582 common_vector_printer:
583 {
584 register long i;
585 long last = SCM_SIMPLE_VECTOR_LENGTH (exp) - 1;
586 int cutp = 0;
587 if (pstate->fancyp
588 && SCM_SIMPLE_VECTOR_LENGTH (exp) > pstate->length)
589 {
590 last = pstate->length - 1;
591 cutp = 1;
592 }
593 for (i = 0; i < last; ++i)
594 {
595 /* CHECK_INTS; */
596 scm_iprin1 (SCM_SIMPLE_VECTOR_REF (exp, i), port, pstate);
597 scm_putc (' ', port);
598 }
599 if (i == last)
600 {
601 /* CHECK_INTS; */
602 scm_iprin1 (SCM_SIMPLE_VECTOR_REF (exp, i), port, pstate);
603 }
604 if (cutp)
605 scm_puts (" ...", port);
606 scm_putc (')', port);
607 }
608 EXIT_NESTED_DATA (pstate);
609 break;
610 case scm_tcs_subrs:
611 scm_puts (SCM_SUBR_GENERIC (exp)
612 ? "#<primitive-generic "
613 : "#<primitive-procedure ",
614 port);
615 scm_puts (scm_i_symbol_chars (SCM_SNAME (exp)), port);
616 scm_putc ('>', port);
617 break;
618 #ifdef CCLO
619 case scm_tc7_cclo:
620 {
621 SCM proc = SCM_CCLO_SUBR (exp);
622 if (scm_is_eq (proc, scm_f_gsubr_apply))
623 {
624 /* Print gsubrs as primitives */
625 SCM name = scm_procedure_name (exp);
626 scm_puts ("#<primitive-procedure", port);
627 if (scm_is_true (name))
628 {
629 scm_putc (' ', port);
630 scm_puts (scm_i_symbol_chars (name), port);
631 }
632 }
633 else
634 {
635 scm_puts ("#<compiled-closure ", port);
636 scm_iprin1 (proc, port, pstate);
637 }
638 scm_putc ('>', port);
639 }
640 break;
641 #endif
642 case scm_tc7_pws:
643 scm_puts ("#<procedure-with-setter", port);
644 {
645 SCM name = scm_procedure_name (exp);
646 if (scm_is_true (name))
647 {
648 scm_putc (' ', port);
649 scm_display (name, port);
650 }
651 }
652 scm_putc ('>', port);
653 break;
654 case scm_tc7_port:
655 {
656 register long i = SCM_PTOBNUM (exp);
657 if (i < scm_numptob
658 && scm_ptobs[i].print
659 && (scm_ptobs[i].print) (exp, port, pstate))
660 break;
661 goto punk;
662 }
663 case scm_tc7_smob:
664 ENTER_NESTED_DATA (pstate, exp, circref);
665 SCM_SMOB_DESCRIPTOR (exp).print (exp, port, pstate);
666 EXIT_NESTED_DATA (pstate);
667 break;
668 default:
669 punk:
670 scm_ipruk ("type", exp, port);
671 }
672 }
673 }
674
675 /* Print states are necessary for circular reference safe printing.
676 * They are also expensive to allocate. Therefore print states are
677 * kept in a pool so that they can be reused.
678 */
679
680 /* The PORT argument can also be a print-state/port pair, which will
681 * then be used instead of allocating a new print state. This is
682 * useful for continuing a chain of print calls from Scheme. */
683
684 void
685 scm_prin1 (SCM exp, SCM port, int writingp)
686 {
687 SCM handle = SCM_BOOL_F; /* Will GC protect the handle whilst unlinked */
688 SCM pstate_scm;
689 scm_print_state *pstate;
690 int old_writingp;
691
692 /* If PORT is a print-state/port pair, use that. Else create a new
693 print-state. */
694
695 if (SCM_PORT_WITH_PS_P (port))
696 {
697 pstate_scm = SCM_PORT_WITH_PS_PS (port);
698 port = SCM_PORT_WITH_PS_PORT (port);
699 }
700 else
701 {
702 /* First try to allocate a print state from the pool */
703 scm_i_pthread_mutex_lock (&print_state_mutex);
704 if (!scm_is_null (print_state_pool))
705 {
706 handle = print_state_pool;
707 print_state_pool = SCM_CDR (print_state_pool);
708 }
709 scm_i_pthread_mutex_unlock (&print_state_mutex);
710 if (scm_is_false (handle))
711 handle = scm_list_1 (make_print_state ());
712 pstate_scm = SCM_CAR (handle);
713 }
714
715 pstate = SCM_PRINT_STATE (pstate_scm);
716 old_writingp = pstate->writingp;
717 pstate->writingp = writingp;
718 scm_iprin1 (exp, port, pstate);
719 pstate->writingp = old_writingp;
720
721 /* Return print state to pool if it has been created above and
722 hasn't escaped to Scheme. */
723
724 if (scm_is_true (handle) && !pstate->revealed)
725 {
726 scm_i_pthread_mutex_lock (&print_state_mutex);
727 SCM_SETCDR (handle, print_state_pool);
728 print_state_pool = handle;
729 scm_i_pthread_mutex_unlock (&print_state_mutex);
730 }
731 }
732
733
734 /* Print an integer.
735 */
736
737 void
738 scm_intprint (scm_t_intmax n, int radix, SCM port)
739 {
740 char num_buf[SCM_INTBUFLEN];
741 scm_lfwrite (num_buf, scm_iint2str (n, radix, num_buf), port);
742 }
743
744 void
745 scm_uintprint (scm_t_uintmax n, int radix, SCM port)
746 {
747 char num_buf[SCM_INTBUFLEN];
748 scm_lfwrite (num_buf, scm_iuint2str (n, radix, num_buf), port);
749 }
750
751 /* Print an object of unrecognized type.
752 */
753
754 void
755 scm_ipruk (char *hdr, SCM ptr, SCM port)
756 {
757 scm_puts ("#<unknown-", port);
758 scm_puts (hdr, port);
759 if (scm_in_heap_p (ptr))
760 {
761 scm_puts (" (0x", port);
762 scm_uintprint (SCM_CELL_WORD_0 (ptr), 16, port);
763 scm_puts (" . 0x", port);
764 scm_uintprint (SCM_CELL_WORD_1 (ptr), 16, port);
765 scm_puts (") @", port);
766 }
767 scm_puts (" 0x", port);
768 scm_uintprint (SCM_UNPACK (ptr), 16, port);
769 scm_putc ('>', port);
770 }
771
772
773 /* Print a list.
774 */
775 void
776 scm_iprlist (char *hdr, SCM exp, int tlr, SCM port, scm_print_state *pstate)
777 {
778 register SCM hare, tortoise;
779 long floor = pstate->top - 2;
780 scm_puts (hdr, port);
781 /* CHECK_INTS; */
782 if (pstate->fancyp)
783 goto fancy_printing;
784
785 /* Run a hare and tortoise so that total time complexity will be
786 O(depth * N) instead of O(N^2). */
787 hare = SCM_CDR (exp);
788 tortoise = exp;
789 while (scm_is_pair (hare))
790 {
791 if (scm_is_eq (hare, tortoise))
792 goto fancy_printing;
793 hare = SCM_CDR (hare);
794 if (!scm_is_pair (hare))
795 break;
796 hare = SCM_CDR (hare);
797 tortoise = SCM_CDR (tortoise);
798 }
799
800 /* No cdr cycles intrinsic to this list */
801 scm_iprin1 (SCM_CAR (exp), port, pstate);
802 for (exp = SCM_CDR (exp); scm_is_pair (exp); exp = SCM_CDR (exp))
803 {
804 register long i;
805
806 for (i = floor; i >= 0; --i)
807 if (scm_is_eq (PSTATE_STACK_REF(pstate, i), exp))
808 goto circref;
809 PUSH_REF (pstate, exp);
810 scm_putc (' ', port);
811 /* CHECK_INTS; */
812 scm_iprin1 (SCM_CAR (exp), port, pstate);
813 }
814 if (!SCM_NULL_OR_NIL_P (exp))
815 {
816 scm_puts (" . ", port);
817 scm_iprin1 (exp, port, pstate);
818 }
819
820 end:
821 scm_putc (tlr, port);
822 pstate->top = floor + 2;
823 return;
824
825 fancy_printing:
826 {
827 long n = pstate->length;
828
829 scm_iprin1 (SCM_CAR (exp), port, pstate);
830 exp = SCM_CDR (exp); --n;
831 for (; scm_is_pair (exp); exp = SCM_CDR (exp))
832 {
833 register unsigned long i;
834
835 for (i = 0; i < pstate->top; ++i)
836 if (scm_is_eq (PSTATE_STACK_REF(pstate, i), exp))
837 goto fancy_circref;
838 if (pstate->fancyp)
839 {
840 if (n == 0)
841 {
842 scm_puts (" ...", port);
843 goto skip_tail;
844 }
845 else
846 --n;
847 }
848 PUSH_REF(pstate, exp);
849 ++pstate->list_offset;
850 scm_putc (' ', port);
851 /* CHECK_INTS; */
852 scm_iprin1 (SCM_CAR (exp), port, pstate);
853 }
854 }
855 if (!SCM_NULL_OR_NIL_P (exp))
856 {
857 scm_puts (" . ", port);
858 scm_iprin1 (exp, port, pstate);
859 }
860 skip_tail:
861 pstate->list_offset -= pstate->top - floor - 2;
862 goto end;
863
864 fancy_circref:
865 pstate->list_offset -= pstate->top - floor - 2;
866
867 circref:
868 scm_puts (" . ", port);
869 print_circref (port, pstate, exp);
870 goto end;
871 }
872
873 \f
874
875 int
876 scm_valid_oport_value_p (SCM val)
877 {
878 return (SCM_OPOUTPORTP (val)
879 || (SCM_PORT_WITH_PS_P (val)
880 && SCM_OPOUTPORTP (SCM_PORT_WITH_PS_PORT (val))));
881 }
882
883 /* SCM_GPROC(s_write, "write", 1, 1, 0, scm_write, g_write); */
884
885 SCM
886 scm_write (SCM obj, SCM port)
887 {
888 if (SCM_UNBNDP (port))
889 port = scm_current_output_port ();
890
891 SCM_ASSERT (scm_valid_oport_value_p (port), port, SCM_ARG2, s_write);
892
893 scm_prin1 (obj, port, 1);
894 #ifdef HAVE_PIPE
895 # ifdef EPIPE
896 if (EPIPE == errno)
897 scm_close_port (port);
898 # endif
899 #endif
900 return SCM_UNSPECIFIED;
901 }
902
903
904 /* SCM_GPROC(s_display, "display", 1, 1, 0, scm_display, g_display); */
905
906 SCM
907 scm_display (SCM obj, SCM port)
908 {
909 if (SCM_UNBNDP (port))
910 port = scm_current_output_port ();
911
912 SCM_ASSERT (scm_valid_oport_value_p (port), port, SCM_ARG2, s_display);
913
914 scm_prin1 (obj, port, 0);
915 #ifdef HAVE_PIPE
916 # ifdef EPIPE
917 if (EPIPE == errno)
918 scm_close_port (port);
919 # endif
920 #endif
921 return SCM_UNSPECIFIED;
922 }
923
924
925 SCM_DEFINE (scm_simple_format, "simple-format", 2, 0, 1,
926 (SCM destination, SCM message, SCM args),
927 "Write @var{message} to @var{destination}, defaulting to\n"
928 "the current output port.\n"
929 "@var{message} can contain @code{~A} (was @code{%s}) and\n"
930 "@code{~S} (was @code{%S}) escapes. When printed,\n"
931 "the escapes are replaced with corresponding members of\n"
932 "@var{ARGS}:\n"
933 "@code{~A} formats using @code{display} and @code{~S} formats\n"
934 "using @code{write}.\n"
935 "If @var{destination} is @code{#t}, then use the current output\n"
936 "port, if @var{destination} is @code{#f}, then return a string\n"
937 "containing the formatted text. Does not add a trailing newline.")
938 #define FUNC_NAME s_scm_simple_format
939 {
940 SCM port, answer = SCM_UNSPECIFIED;
941 int fReturnString = 0;
942 int writingp;
943 const char *start;
944 const char *end;
945 const char *p;
946
947 if (scm_is_eq (destination, SCM_BOOL_T))
948 {
949 destination = port = scm_current_output_port ();
950 }
951 else if (scm_is_false (destination))
952 {
953 fReturnString = 1;
954 port = scm_mkstrport (SCM_INUM0,
955 scm_make_string (SCM_INUM0, SCM_UNDEFINED),
956 SCM_OPN | SCM_WRTNG,
957 FUNC_NAME);
958 destination = port;
959 }
960 else
961 {
962 SCM_VALIDATE_OPORT_VALUE (1, destination);
963 port = SCM_COERCE_OUTPORT (destination);
964 }
965 SCM_VALIDATE_STRING (2, message);
966 SCM_VALIDATE_REST_ARGUMENT (args);
967
968 start = scm_i_string_chars (message);
969 end = start + scm_i_string_length (message);
970 for (p = start; p != end; ++p)
971 if (*p == '~')
972 {
973 if (++p == end)
974 break;
975
976 switch (*p)
977 {
978 case 'A': case 'a':
979 writingp = 0;
980 break;
981 case 'S': case 's':
982 writingp = 1;
983 break;
984 case '~':
985 scm_lfwrite (start, p - start, port);
986 start = p + 1;
987 continue;
988 case '%':
989 scm_lfwrite (start, p - start - 1, port);
990 scm_newline (port);
991 start = p + 1;
992 continue;
993 default:
994 SCM_MISC_ERROR ("FORMAT: Unsupported format option ~~~A - use (ice-9 format) instead",
995 scm_list_1 (SCM_MAKE_CHAR (*p)));
996
997 }
998
999
1000 if (!scm_is_pair (args))
1001 SCM_MISC_ERROR ("FORMAT: Missing argument for ~~~A",
1002 scm_list_1 (SCM_MAKE_CHAR (*p)));
1003
1004 scm_lfwrite (start, p - start - 1, port);
1005 /* we pass destination here */
1006 scm_prin1 (SCM_CAR (args), destination, writingp);
1007 args = SCM_CDR (args);
1008 start = p + 1;
1009 }
1010
1011 scm_lfwrite (start, p - start, port);
1012 if (!scm_is_eq (args, SCM_EOL))
1013 SCM_MISC_ERROR ("FORMAT: ~A superfluous arguments",
1014 scm_list_1 (scm_length (args)));
1015
1016 if (fReturnString)
1017 answer = scm_strport_to_string (destination);
1018
1019 return scm_return_first (answer, message);
1020 }
1021 #undef FUNC_NAME
1022
1023
1024 SCM_DEFINE (scm_newline, "newline", 0, 1, 0,
1025 (SCM port),
1026 "Send a newline to @var{port}.\n"
1027 "If @var{port} is omitted, send to the current output port.")
1028 #define FUNC_NAME s_scm_newline
1029 {
1030 if (SCM_UNBNDP (port))
1031 port = scm_current_output_port ();
1032
1033 SCM_VALIDATE_OPORT_VALUE (1, port);
1034
1035 scm_putc ('\n', SCM_COERCE_OUTPORT (port));
1036 return SCM_UNSPECIFIED;
1037 }
1038 #undef FUNC_NAME
1039
1040 SCM_DEFINE (scm_write_char, "write-char", 1, 1, 0,
1041 (SCM chr, SCM port),
1042 "Send character @var{chr} to @var{port}.")
1043 #define FUNC_NAME s_scm_write_char
1044 {
1045 if (SCM_UNBNDP (port))
1046 port = scm_current_output_port ();
1047
1048 SCM_VALIDATE_CHAR (1, chr);
1049 SCM_VALIDATE_OPORT_VALUE (2, port);
1050
1051 scm_putc ((int) SCM_CHAR (chr), SCM_COERCE_OUTPORT (port));
1052 #ifdef HAVE_PIPE
1053 # ifdef EPIPE
1054 if (EPIPE == errno)
1055 scm_close_port (port);
1056 # endif
1057 #endif
1058 return SCM_UNSPECIFIED;
1059 }
1060 #undef FUNC_NAME
1061
1062 \f
1063
1064 /* Call back to Scheme code to do the printing of special objects
1065 * (like structs). SCM_PRINTER_APPLY applies PROC to EXP and a smob
1066 * containing PORT and PSTATE. This object can be used as the port for
1067 * display/write etc to continue the current print chain. The REVEALED
1068 * field of PSTATE is set to true to indicate that the print state has
1069 * escaped to Scheme and thus has to be freed by the GC.
1070 */
1071
1072 scm_t_bits scm_tc16_port_with_ps;
1073
1074 /* Print exactly as the port itself would */
1075
1076 static int
1077 port_with_ps_print (SCM obj, SCM port, scm_print_state *pstate)
1078 {
1079 obj = SCM_PORT_WITH_PS_PORT (obj);
1080 return scm_ptobs[SCM_PTOBNUM (obj)].print (obj, port, pstate);
1081 }
1082
1083 SCM
1084 scm_printer_apply (SCM proc, SCM exp, SCM port, scm_print_state *pstate)
1085 {
1086 pstate->revealed = 1;
1087 return scm_call_2 (proc, exp,
1088 scm_i_port_with_print_state (port, pstate->handle));
1089 }
1090
1091 SCM_DEFINE (scm_port_with_print_state, "port-with-print-state", 1, 1, 0,
1092 (SCM port, SCM pstate),
1093 "Create a new port which behaves like @var{port}, but with an\n"
1094 "included print state @var{pstate}. @var{pstate} is optional.\n"
1095 "If @var{pstate} isn't supplied and @var{port} already has\n"
1096 "a print state, the old print state is reused.")
1097 #define FUNC_NAME s_scm_port_with_print_state
1098 {
1099 SCM_VALIDATE_OPORT_VALUE (1, port);
1100 if (!SCM_UNBNDP (pstate))
1101 SCM_VALIDATE_PRINTSTATE (2, pstate);
1102 return scm_i_port_with_print_state (port, pstate);
1103 }
1104 #undef FUNC_NAME
1105
1106 SCM_DEFINE (scm_get_print_state, "get-print-state", 1, 0, 0,
1107 (SCM port),
1108 "Return the print state of the port @var{port}. If @var{port}\n"
1109 "has no associated print state, @code{#f} is returned.")
1110 #define FUNC_NAME s_scm_get_print_state
1111 {
1112 if (SCM_PORT_WITH_PS_P (port))
1113 return SCM_PORT_WITH_PS_PS (port);
1114 if (SCM_OUTPUT_PORT_P (port))
1115 return SCM_BOOL_F;
1116 SCM_WRONG_TYPE_ARG (1, port);
1117 }
1118 #undef FUNC_NAME
1119
1120 \f
1121
1122 void
1123 scm_init_print ()
1124 {
1125 SCM vtable, layout, type;
1126
1127 scm_init_opts (scm_print_options, scm_print_opts, SCM_N_PRINT_OPTIONS);
1128
1129 scm_print_options (scm_list_4 (scm_from_locale_symbol ("highlight-prefix"),
1130 scm_from_locale_string ("{"),
1131 scm_from_locale_symbol ("highlight-suffix"),
1132 scm_from_locale_string ("}")));
1133
1134 scm_gc_register_root (&print_state_pool);
1135 scm_gc_register_root (&scm_print_state_vtable);
1136 vtable = scm_make_vtable_vtable (scm_nullstr, SCM_INUM0, SCM_EOL);
1137 layout =
1138 scm_make_struct_layout (scm_from_locale_string (SCM_PRINT_STATE_LAYOUT));
1139 type = scm_make_struct (vtable, SCM_INUM0, scm_list_1 (layout));
1140 scm_set_struct_vtable_name_x (type, scm_from_locale_symbol ("print-state"));
1141 scm_print_state_vtable = type;
1142
1143 /* Don't want to bind a wrapper class in GOOPS, so pass 0 as arg1. */
1144 scm_tc16_port_with_ps = scm_make_smob_type (0, 0);
1145 scm_set_smob_mark (scm_tc16_port_with_ps, scm_markcdr);
1146 scm_set_smob_print (scm_tc16_port_with_ps, port_with_ps_print);
1147
1148 #include "libguile/print.x"
1149 }
1150
1151 /*
1152 Local Variables:
1153 c-file-style: "gnu"
1154 End:
1155 */