/**************************************************************************/ /* */ /* OCaml */ /* */ /* Xavier Leroy, projet Cristal, INRIA Rocquencourt */ /* */ /* Copyright 1996 Institut National de Recherche en Informatique et */ /* en Automatique. */ /* */ /* All rights reserved. This file is distributed under the terms of */ /* the GNU Lesser General Public License version 2.1, with the */ /* special exception on linking described in the file LICENSE. */ /* */ /**************************************************************************/ /* Read and output terminal commands */ #include "caml/config.h" #include "caml/alloc.h" #include "caml/fail.h" #include "caml/io.h" #include "caml/mlvalues.h" #define Uninitialised (Val_int(0)) #define Bad_term (Val_int(1)) #define Good_term_tag 0 #if defined (HAS_TERMCAP) && !defined (NATIVE_CODE) extern int tgetent (char * buffer, char * name); extern char * tgetstr (char * id, char ** area); extern int tgetnum (char * id); extern int tputs (char * str, int count, int (*outchar)(int c)); static struct channel *chan; static char area [1024]; static char *area_p = area; static int num_lines; static char *up = NULL; static char *down = NULL; static char *standout = NULL; static char *standend = NULL; CAMLprim value caml_terminfo_setup (value vchan) { value result; static char buffer[1024]; char *term; chan = Channel (vchan); term = getenv ("TERM"); if (term == NULL) return Bad_term; if (tgetent(buffer, term) != 1) return Bad_term; num_lines = tgetnum ("li"); up = tgetstr ("up", &area_p); down = tgetstr ("do", &area_p); standout = tgetstr ("us", &area_p); standend = tgetstr ("ue", &area_p); if (standout == NULL || standend == NULL){ standout = tgetstr ("so", &area_p); standend = tgetstr ("se", &area_p); } Assert (area_p <= area + 1024); if (num_lines == -1 || up == NULL || down == NULL || standout == NULL || standend == NULL){ return Bad_term; } result = caml_alloc_small (1, Good_term_tag); Field (result, 0) = Val_int (num_lines); return result; } static int terminfo_putc (int c) { putch (chan, c); return c; } CAMLprim value caml_terminfo_backup (value lines) { int i; for (i = 0; i < Int_val (lines); i++){ tputs (up, 1, terminfo_putc); } return Val_unit; } CAMLprim value caml_terminfo_standout (value start) { tputs (Bool_val (start) ? standout : standend, 1, terminfo_putc); return Val_unit; } CAMLprim value caml_terminfo_resume (value lines) { int i; for (i = 0; i < Int_val (lines); i++){ tputs (down, 1, terminfo_putc); } return Val_unit; } #else /* defined (HAS_TERMCAP) && !defined (NATIVE_CODE) */ CAMLexport value caml_terminfo_setup (value vchan) { return Bad_term; } CAMLexport value caml_terminfo_backup (value lines) { caml_invalid_argument("Terminfo.backup"); return Val_unit; } CAMLexport value caml_terminfo_standout (value start) { caml_invalid_argument("Terminfo.standout"); return Val_unit; } CAMLexport value caml_terminfo_resume (value lines) { caml_invalid_argument("Terminfo.resume"); return Val_unit; } #endif /* defined (HAS_TERMCAP) && !defined (NATIVE_CODE) */