Blame support/cltkVar.c

Packit bd2e5d
/***********************************************************************/
Packit bd2e5d
/*                                                                     */
Packit bd2e5d
/*                 MLTk, Tcl/Tk interface of OCaml                     */
Packit bd2e5d
/*                                                                     */
Packit bd2e5d
/*    Francois Rouaix, Francois Pessaux, Jun Furuse and Pierre Weis    */
Packit bd2e5d
/*               projet Cristal, INRIA Rocquencourt                    */
Packit bd2e5d
/*            Jacques Garrigue, Kyoto University RIMS                  */
Packit bd2e5d
/*                                                                     */
Packit bd2e5d
/*  Copyright 2002 Institut National de Recherche en Informatique et   */
Packit bd2e5d
/*  en Automatique and Kyoto University.  All rights reserved.         */
Packit bd2e5d
/*  This file is distributed under the terms of the GNU Library        */
Packit bd2e5d
/*  General Public License, with the special exception on linking      */
Packit bd2e5d
/*  described in file LICENSE found in the OCaml source tree.          */
Packit bd2e5d
/*                                                                     */
Packit bd2e5d
/***********************************************************************/
Packit bd2e5d
Packit bd2e5d
/* $Id$ */
Packit bd2e5d
Packit bd2e5d
/* Alternative to tkwait variable */
Packit bd2e5d
#include <string.h>
Packit bd2e5d
#include <tcl.h>
Packit bd2e5d
#include <tk.h>
Packit bd2e5d
#include <mlvalues.h>
Packit bd2e5d
#include <memory.h>
Packit bd2e5d
#include <alloc.h>
Packit bd2e5d
#include <callback.h>
Packit bd2e5d
#include "camltk.h"
Packit bd2e5d
Packit bd2e5d
CAMLprim value camltk_getvar(value var)
Packit bd2e5d
{
Packit bd2e5d
  char *s;
Packit bd2e5d
  char *stable_var = NULL;
Packit bd2e5d
  CheckInit();
Packit bd2e5d
Packit bd2e5d
  stable_var = string_to_c(var);
Packit bd2e5d
  s = (char *)Tcl_GetVar(cltclinterp,stable_var,
Packit bd2e5d
                         TCL_GLOBAL_ONLY|TCL_LEAVE_ERR_MSG);
Packit bd2e5d
  caml_stat_free(stable_var);
Packit bd2e5d
Packit bd2e5d
  if (s == NULL)
Packit bd2e5d
    tk_error(Tcl_GetStringResult(cltclinterp));
Packit bd2e5d
  else
Packit bd2e5d
    return(tcl_string_to_caml(s));
Packit bd2e5d
}
Packit bd2e5d
Packit bd2e5d
CAMLprim value camltk_setvar(value var, value contents)
Packit bd2e5d
{
Packit bd2e5d
  char *s;
Packit bd2e5d
  char *stable_var = NULL;
Packit bd2e5d
  char *utf_contents;
Packit bd2e5d
  CheckInit();
Packit bd2e5d
Packit bd2e5d
  /* SetVar makes a copy of the contents. */
Packit bd2e5d
  /* In case we have write traces in OCaml, it's better to make sure that
Packit bd2e5d
     var doesn't move... */
Packit bd2e5d
  stable_var = string_to_c(var);
Packit bd2e5d
  utf_contents = caml_string_to_tcl(contents);
Packit bd2e5d
  s = (char *)Tcl_SetVar(cltclinterp,stable_var, utf_contents,
Packit bd2e5d
                         TCL_GLOBAL_ONLY|TCL_LEAVE_ERR_MSG);
Packit bd2e5d
  caml_stat_free(stable_var);
Packit bd2e5d
  if( s == utf_contents ){
Packit bd2e5d
    tk_error("camltk_setvar: Tcl_SetVar returned strange result. Call the author of mlTk!");
Packit bd2e5d
  }
Packit bd2e5d
  caml_stat_free(utf_contents);
Packit bd2e5d
Packit bd2e5d
  if (s == NULL)
Packit bd2e5d
    tk_error(Tcl_GetStringResult(cltclinterp));
Packit bd2e5d
  else
Packit bd2e5d
    return(Val_unit);
Packit bd2e5d
}
Packit bd2e5d
Packit bd2e5d
Packit bd2e5d
/* The appropriate type is
Packit bd2e5d
typedef char *(Tcl_VarTraceProc) _ANSI_ARGS_((ClientData clientData,
Packit bd2e5d
        Tcl_Interp *interp, char *part1, char *part2, int flags));
Packit bd2e5d
 */
Packit bd2e5d
static char * tracevar(clientdata, interp, name1, name2, flags)
Packit bd2e5d
     ClientData clientdata;
Packit bd2e5d
     Tcl_Interp *interp;        /* Interpreter containing variable. */
Packit bd2e5d
     char *name1;               /* Name of variable. */
Packit bd2e5d
     char *name2;               /* Second part of variable name. */
Packit bd2e5d
     int flags;                 /* Information about what happened. */
Packit bd2e5d
{
Packit bd2e5d
  Tcl_UntraceVar2(interp, name1, name2,
Packit bd2e5d
                TCL_GLOBAL_ONLY|TCL_TRACE_WRITES|TCL_TRACE_UNSETS,
Packit bd2e5d
                tracevar, clientdata);
Packit bd2e5d
  caml_callback2(*handler_code,Val_int(clientdata),Val_unit);
Packit bd2e5d
  return (char *)NULL;
Packit bd2e5d
}
Packit bd2e5d
Packit bd2e5d
/* Sets up a callback upon modification of a variable */
Packit bd2e5d
CAMLprim value camltk_trace_var(value var, value cbid)
Packit bd2e5d
{
Packit bd2e5d
  char *cvar = NULL;
Packit bd2e5d
Packit bd2e5d
  CheckInit();
Packit bd2e5d
  /* Make a copy of var, since Tcl will modify it in place, and we
Packit bd2e5d
   * don't trust that much what it will do here
Packit bd2e5d
   */
Packit bd2e5d
  cvar = string_to_c(var);
Packit bd2e5d
  if (Tcl_TraceVar(cltclinterp, cvar,
Packit bd2e5d
                   TCL_GLOBAL_ONLY|TCL_TRACE_WRITES|TCL_TRACE_UNSETS,
Packit bd2e5d
                   tracevar,
Packit bd2e5d
                   (ClientData) (Long_val(cbid)))
Packit bd2e5d
                   != TCL_OK) {
Packit bd2e5d
    caml_stat_free(cvar);
Packit bd2e5d
    tk_error(Tcl_GetStringResult(cltclinterp));
Packit bd2e5d
  };
Packit bd2e5d
  caml_stat_free(cvar);
Packit bd2e5d
  return Val_unit;
Packit bd2e5d
}
Packit bd2e5d
Packit bd2e5d
CAMLprim value camltk_untrace_var(value var, value cbid)
Packit bd2e5d
{
Packit bd2e5d
  char *cvar = NULL;
Packit bd2e5d
Packit bd2e5d
  CheckInit();
Packit bd2e5d
  /* Make a copy of var, since Tcl will modify it in place, and we
Packit bd2e5d
   * don't trust that much what it will do here
Packit bd2e5d
   */
Packit bd2e5d
  cvar = string_to_c(var);
Packit bd2e5d
  Tcl_UntraceVar(cltclinterp, cvar,
Packit bd2e5d
                 TCL_GLOBAL_ONLY|TCL_TRACE_WRITES|TCL_TRACE_UNSETS,
Packit bd2e5d
                 tracevar,
Packit bd2e5d
                 (ClientData) (Long_val(cbid)));
Packit bd2e5d
  caml_stat_free(cvar);
Packit bd2e5d
  return Val_unit;
Packit bd2e5d
}