build: Don't include <config.h> in native programs when cross-compiling.
[bpt/guile.git] / libguile / root.c
index dfe0ae3..8c8fd1a 100644 (file)
@@ -1,24 +1,30 @@
-/* Copyright (C) 1995,1996,1997,1998,1999,2000, 2001, 2002 Free Software Foundation, Inc.
+/* Copyright (C) 1995,1996,1997,1998,1999,2000, 2001, 2002, 2006, 2008, 2009 Free Software Foundation, Inc.
  * 
  * This library is free software; you can redistribute it and/or
- * modify it under the terms of the GNU Lesser General Public
- * License as published by the Free Software Foundation; either
- * version 2.1 of the License, or (at your option) any later version.
+ * modify it under the terms of the GNU Lesser General Public License
+ * as published by the Free Software Foundation; either version 3 of
+ * the License, or (at your option) any later version.
  *
- * This library is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * This library is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
  * Lesser General Public License for more details.
  *
  * You should have received a copy of the GNU Lesser General Public
  * License along with this library; if not, write to the Free Software
- * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
+ * 02110-1301 USA
  */
 
 
 \f
+#ifdef HAVE_CONFIG_H
+# include <config.h>
+#endif
 
 #include <string.h>
+#include <stdio.h>
+
 #include "libguile/_scm.h"
 #include "libguile/stackchk.h"
 #include "libguile/dynwind.h"
 #include "libguile/root.h"
 \f
 
-SCM scm_sys_protects[SCM_NUM_PROTECTS];
-
-scm_t_bits scm_tc16_root;
-
-\f
-
-static SCM
-root_mark (SCM root)
-{
-  scm_root_state *s = SCM_ROOT_STATE (root);
-
-  scm_gc_mark (s->rootcont);
-  scm_gc_mark (s->dynwinds);
-  scm_gc_mark (s->progargs);
-  scm_gc_mark (s->exitval);
-  scm_gc_mark (s->cur_inp);
-  scm_gc_mark (s->cur_outp);
-  scm_gc_mark (s->cur_errp);
-  /* No need to gc mark def_loadp */
-  scm_gc_mark (s->fluids);
-  scm_gc_mark (s->active_asyncs);
-  scm_gc_mark (s->signal_asyncs);
-  return SCM_ROOT_STATE (root) -> parent;
-}
-
-
-static int
-root_print (SCM exp, SCM port, scm_print_state *pstate SCM_UNUSED)
-{
-  scm_puts ("#<root ", port);
-  scm_uintprint(SCM_SEQ (SCM_ROOT_STATE (exp) -> rootcont), 16, port);
-  scm_putc('>', port);
-  return 1;
-}
-
-
-\f
-
-SCM
-scm_make_root (SCM parent)
-{
-  SCM root;
-  scm_root_state *root_state;
-
-  root_state = (scm_root_state *) scm_gc_malloc (sizeof (scm_root_state),
-                                                "root state");
-  if (SCM_ROOTP (parent))
-    {
-      memcpy (root_state, SCM_ROOT_STATE (parent), sizeof (scm_root_state));
-      root_state->parent = parent;
-    }
-  else
-    {
-      root_state->parent = SCM_BOOL_F;
-
-      /* Initialize everything right now, in case a GC happens early.  */
-      root_state->rootcont
-       = root_state->dynwinds
-       = root_state->progargs
-       = root_state->exitval
-       = root_state->cur_inp
-       = root_state->cur_outp
-       = root_state->cur_errp
-       = root_state->cur_loadp
-       = root_state->fluids
-       = root_state->handle
-       = root_state->parent
-       = SCM_BOOL_F;
-    }
-  
-  root_state->active_asyncs = SCM_EOL;
-  root_state->signal_asyncs = SCM_EOL;
-  root_state->block_asyncs = 0;
-  root_state->pending_asyncs = 1;
-
-  SCM_NEWSMOB (root, scm_tc16_root, root_state);
-  root_state->handle = root;
-  
-  if (SCM_ROOTP (parent))
-    /* Must be done here so that fluids are GC protected */
-    scm_i_copy_fluids (root_state);
-  
-  return root;
-}
-
 /* {call-with-dynamic-root}
  *
  * Suspending the current thread to evaluate a thunk on the
@@ -125,25 +46,6 @@ scm_make_root (SCM parent)
  * Calls to call-with-dynamic-root return exactly once (unless
  * the process is somehow exitted).  */
 
-/* Some questions about cwdr:
-
-   Couldn't the body just be a closure?  Do we really need to pass
-   args through to it?
-
-   The semantics are a lot like catch's; in fact, we call
-   scm_internal_catch to take care of that part of things.  Wouldn't
-   it be cleaner to say that uncaught throws just disappear into the
-   ether (or print a message to stderr), and let the caller use catch
-   themselves if they want to?
-
-   -JimB */
-
-#if 0
-SCM scm_exitval;               /* INUM with return value */
-#endif
-static long n_dynamic_roots = 0;
-
-
 /* cwdr fills out both of these structures, and then passes a pointer
    to them through scm_internal_catch to the cwdr_body and
    cwdr_handler functions, to tell them how to behave and to get
@@ -201,62 +103,32 @@ cwdr_handler (void *data, SCM tag, SCM args)
   return SCM_UNSPECIFIED;
 }
 
-/* This is the basic code for new root creation.
- *
- * WARNING!  The order of actions in this routine is in many ways
- * critical.  E. g., it is essential that an error doesn't leave Guile
- * in a messed up state.  */
-
 SCM 
 scm_internal_cwdr (scm_t_catch_body body, void *body_data,
                   scm_t_catch_handler handler, void *handler_data,
                   SCM_STACKITEM *stack_start)
 {
-  SCM old_rootcont, old_winds;
   struct cwdr_handler_data my_handler_data;
-  SCM answer;
-
-  /* Create a fresh root continuation.  */
-  {
-    SCM new_rootcont;
-
-    SCM_REDEFER_INTS;
-    {
-      scm_t_contregs *contregs = scm_gc_malloc (sizeof (scm_t_contregs),
-                                               "continuation");
-
-      contregs->num_stack_items = 0;
-      contregs->dynenv = SCM_EOL;
-      contregs->base = stack_start;
-      contregs->seq = ++n_dynamic_roots;
-      contregs->throw_value = SCM_BOOL_F;
-      contregs->dframe = 0;
-      SCM_NEWSMOB (new_rootcont, scm_tc16_continuation, contregs);
-    }
-    old_rootcont = scm_rootcont;
-    scm_rootcont = new_rootcont;
-    SCM_REALLOW_INTS;
-  }
+  SCM answer, old_winds;
 
   /* Exit caller's dynamic state.
    */
-  old_winds = scm_dynwinds;
-  scm_dowinds (SCM_EOL, scm_ilength (scm_dynwinds));
-  SCM_DFRAME (old_rootcont) = scm_last_debug_frame;
-  scm_last_debug_frame = 0;
-
-  {
-    my_handler_data.run_handler = 0;
-    answer = scm_internal_catch (SCM_BOOL_T,
-                                body, body_data,
-                                cwdr_handler, &my_handler_data);
-  }
+  old_winds = scm_i_dynwinds ();
+  scm_dowinds (SCM_EOL, scm_ilength (old_winds));
 
+  scm_dynwind_begin (SCM_F_DYNWIND_REWINDABLE);
+  scm_dynwind_current_dynamic_state (scm_make_dynamic_state (SCM_UNDEFINED));
+
+  my_handler_data.run_handler = 0;
+  answer = scm_i_with_continuation_barrier (body, body_data,
+                                           cwdr_handler, &my_handler_data,
+                                           NULL, NULL);
+
+  scm_dynwind_end ();
+
+  /* Enter caller's dynamic state.
+   */
   scm_dowinds (old_winds, - scm_ilength (old_winds));
-  SCM_REDEFER_INTS;
-  scm_last_debug_frame = SCM_DFRAME (old_rootcont);
-  scm_rootcont = old_rootcont;
-  SCM_REALLOW_INTS;
 
   /* Now run the real handler iff the body did a throw. */
   if (my_handler_data.run_handler)
@@ -283,41 +155,10 @@ cwdr (SCM proc, SCM a1, SCM args, SCM handler, SCM_STACKITEM *stack_start)
 
 SCM_DEFINE (scm_call_with_dynamic_root, "call-with-dynamic-root", 2, 0, 0,
            (SCM thunk, SCM handler),
-           "Evaluate @code{(thunk)} in a new dynamic context, returning its value.\n\n"
-           "If an error occurs during evaluation, apply @var{handler} to the\n"
-           "arguments to the throw, just as @code{throw} would.  If this happens,\n"
-           "@var{handler} is called outside the scope of the new root -- it is\n"
-           "called in the same dynamic context in which\n"
-           "@code{call-with-dynamic-root} was evaluated.\n\n"
-           "If @var{thunk} captures a continuation, the continuation is rooted at\n"
-           "the call to @var{thunk}.  In particular, the call to\n"
-           "@code{call-with-dynamic-root} is not captured.  Therefore,\n"
-           "@code{call-with-dynamic-root} always returns at most one time.\n\n"
-           "Before calling @var{thunk}, the dynamic-wind chain is un-wound back to\n"
-           "the root and a new chain started for @var{thunk}.  Therefore, this call\n"
-           "may not do what you expect:\n\n"
-           "@lisp\n"
-           ";; Almost certainly a bug:\n"
-           "(with-output-to-port\n"
-           " some-port\n\n"
-           " (lambda ()\n"
-           "   (call-with-dynamic-root\n"
-           "    (lambda ()\n"
-           "      (display 'fnord)\n"
-           "      (newline))\n"
-           "    (lambda (errcode) errcode))))\n"
-           "@end lisp\n\n"
-           "The problem is, on what port will @samp{fnord} be displayed?  You\n"
-           "might expect that because of the @code{with-output-to-port} that\n"
-           "it will be displayed on the port bound to @code{some-port}.  But it\n"
-           "probably won't -- before evaluating the thunk, dynamic winds are\n"
-           "unwound, including those created by @code{with-output-to-port}.\n"
-           "So, the standard output port will have been re-set to its default value\n"
-           "before @code{display} is evaluated.\n\n"
-           "(This function was added to Guile mostly to help calls to functions in C\n"
-           "libraries that can not tolerate non-local exits or calls that return\n"
-           "multiple times.  If such functions call back to the interpreter, it should\n"
-           "be under a new dynamic root.)")
+           "Call @var{thunk} with a new dynamic state and within\n"
+           "a continuation barrier.  The @var{handler} catches all\n"
+           "otherwise uncaught throws and executes within the same\n"
+           "dynamic context as @var{thunk}.")
 #define FUNC_NAME s_scm_call_with_dynamic_root
 {
   SCM_STACKITEM stack_place;
@@ -328,12 +169,10 @@ SCM_DEFINE (scm_call_with_dynamic_root, "call-with-dynamic-root", 2, 0, 0,
 SCM_DEFINE (scm_dynamic_root, "dynamic-root", 0, 0, 0, 
            (),
            "Return an object representing the current dynamic root.\n\n"
-           "These objects are only useful for comparison using @code{eq?}.\n"
-           "They are currently represented as numbers, but your code should\n"
-           "in no way depend on this.")
+           "These objects are only useful for comparison using @code{eq?}.\n")
 #define FUNC_NAME s_scm_dynamic_root
 {
-  return scm_from_ulong (SCM_SEQ (scm_root->rootcont));
+  return SCM_I_CURRENT_THREAD->continuation_root;
 }
 #undef FUNC_NAME
 
@@ -349,10 +188,6 @@ scm_apply_with_dynamic_root (SCM proc, SCM a1, SCM args, SCM handler)
 void
 scm_init_root ()
 {
-  scm_tc16_root = scm_make_smob_type ("root", sizeof (struct scm_root_state));
-  scm_set_smob_mark (scm_tc16_root, root_mark);
-  scm_set_smob_print (scm_tc16_root, root_print);
-
 #include "libguile/root.x"
 }