Logo Search packages:      
Sourcecode: afnix version File versions

InputTerm.cpp

// ---------------------------------------------------------------------------
// - InputTerm.cpp                                                           -
// - standard object library - terminal input stream class implementation    -
// ---------------------------------------------------------------------------
// - This program is free software;  you can redistribute it  and/or  modify -
// - it provided that this copyright notice is kept intact.                  -
// -                                                                         -
// - This program  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.  In no event shall -
// - the copyright holder be liable for any  direct, indirect, incidental or -
// - special damages arising in any way out of the use of this software.     -
// ---------------------------------------------------------------------------
// - copyright (c) 1999-2007 amaury darsch                                   -
// ---------------------------------------------------------------------------

#include "Vector.hpp"
#include "Unicode.hpp"
#include "Boolean.hpp"
#include "InputTerm.hpp"
#include "QuarkZone.hpp"
#include "Exception.hpp"
#include "csio.hpp"
#include "ctrm.hpp"
#include "cstr.hpp"
#include "cerr.hpp"

namespace afnix {

  // -------------------------------------------------------------------------
  // - private section                                                       -
  // -------------------------------------------------------------------------

  // this function computes the maximum size of the temporary buffer we must
  // allocate when reading wide characters and special functions
  static long get_mtlen (char** tinfo) {
    if (!tinfo) return 0;
    long max = 0;
    for (long i = 0; i < ITERM_PARMS_MAX; i++) {
      long len = c_strlen (tinfo[i]);
      max = (max < len) ? len : max;
    }
    return max;
  }
  
  // this function check if we have a match between the buffer and the term
  // info sequence. it returns the number of matches
  static long match_tinfo (char** tinfo, char* buffer, long index, 
                     long* match) {
    if (!tinfo) return 0;
    long count = 0;
    for (long i = 0; i < ITERM_PARMS_MAX; i++) {
      if (c_strncmp (tinfo[i], buffer, index) == true) {
      count++;
      *match = i;
      }
    }
    return count;
  }

  // -------------------------------------------------------------------------
  // - public section                                                        -
  // -------------------------------------------------------------------------

  // return true if we have a wide character
    
00067   bool InputTerm::iswide (const t_quad w) {
    if ((w & 0x80000000) == 0x80000000) return true;
    return false;
  }

  // return true if we have a backspace character
  
00074   bool InputTerm::isbs (const t_quad w) {
    if (w == InputTerm::WIDE_BS)    return true;
    if (w == InputTerm::WIDE_STDBS) return true;
    return false;
  }

  // return true if we have a arrow up character
  
00082   bool InputTerm::isup (const t_quad w) {
    if (w == InputTerm::WIDE_UP)    return true;
    if (w == InputTerm::WIDE_STDUP) return true;
    return false;
  }
  
  // return true if we have a arrow down character
  
00090   bool InputTerm::isdo (const t_quad w) {
    if (w == InputTerm::WIDE_DO)    return true;
    if (w == InputTerm::WIDE_STDDO) return true;
    return false;
  }
  
  // return true if we have a arrow left character
  
00098   bool InputTerm::isle (const t_quad w) {
    if (w == InputTerm::WIDE_LE)    return true;
    if (w == InputTerm::WIDE_STDLE) return true;
    return false;
  }
    
  // return true if we have a arrow right character
  
00106   bool InputTerm::isri (const t_quad w) {
    if (w == InputTerm::WIDE_RI)    return true;
    if (w == InputTerm::WIDE_STDRI) return true;
    return false;
  }
  
  // return true if we have a printable character
  
00114   bool InputTerm::ispr (const t_quad w) {
    // accept eof/eol
    if (w == InputTerm::WIDE_STDNL) return true;
    if (w == InputTerm::WIDE_STDEF) return true;
    // reject wide characters
    if (InputTerm::iswide (w) == true) return false;
    // everything else is valid - so far !
    return true;
  }

  // return true if we have a delete character

00126   bool InputTerm::isdel (const t_quad w) {
    if (w == InputTerm::WIDE_DEL)   return true;
    if (w == InputTerm::WIDE_STDDE) return true;
    return false;
  }
  
  // return true if we have an insert character

00134   bool InputTerm::isins (const t_quad w) {
    if (w == InputTerm::WIDE_STDIM) return true;
    if (w == InputTerm::WIDE_INSCH) return true;
    if (w == InputTerm::WIDE_STDIN) return true;
    return false;
  }

  // -------------------------------------------------------------------------
  // - class section                                                         -
  // -------------------------------------------------------------------------

  // create a new default input stream. By default the input stream is
  // mapped to the default standard input
  
00148   InputTerm::InputTerm (void)  {
    // set the transcoder and stream modes
    settmod (System::getstm ());
    setemod (System::getsem ());
    // initialize the terminal
    d_sid    = c_stdin ();
    d_eof    = false;
    p_attr   = nilp;
    p_tinfo  = c_tinfo (true);
    d_mtlen  = get_mtlen (p_tinfo);
    d_igneof = false;
    d_eofmap = eolc;
  }
  
  // destroy this terminal class
  
00164   InputTerm::~InputTerm (void) {
    c_stattr (d_sid, p_attr);
    c_ftattr (p_attr);
    if (p_tinfo != nilp) {
      for (long i = 0; i < ITERM_PARMS_MAX; i++) delete [] p_tinfo[i];
      delete [] p_tinfo;
    }
  }

  // return the class name

00175   String InputTerm::repr (void) const {
    return "InputTerm";
  }

  // return the terminal descriptor

00181   int InputTerm::getsid (void) const {
    rdlock ();
    int result = d_sid;
    unlock ();
    return result;
  }

  // read the next available character

00190   char InputTerm::read (void) {
    wrlock ();
    try {
      // check for the pushback buffer
      if (d_buffer.empty () == false) {
      char result = d_buffer.read ();
      unlock ();
      return result;
      }
      // check if we are at the end of file
      if (d_eof == true) {
      unlock ();
      return eofc;
      }
      // read the next character on the stream
      char byte  = nilc;
      long count = 0;
      if ((count = c_read (d_sid, &byte,1)) < 0) {
      throw Exception ("read-error", c_errmsg (count));
      }
      // check for ctrl-d against ignore flag - remap to eolc
      if ((byte == eofc) && (d_igneof == true)) {
      char result = d_eofmap;
      unlock ();
      return result;
      }
      // check for eof
      if ((count == 0) || (byte == eofc)) {
      d_eof = true;
      unlock ();
      return eofc;
      }
      unlock ();
      return byte;
    } catch (...) {
      unlock ();
      throw;
    }
  }

  // return a wide character 

00232   t_quad InputTerm::wread (void) {
    long match = 0;
    long count = 0;
    long index = 0;
    char buffer[256];
    
    // make sure we have allocated enough character for the buffer
    // thanks to Reggie Seagraves for this one
    if (d_mtlen+1 > 256)
      throw Exception ("internal-error", "wide char buffer is not big enough");

    // read the first character
    wrlock ();
    try {
      // try to read an ascii character
      t_quad data = rduc ();
      if (Unicode::isascii (data) == false) {
      unlock ();
      return data;
      }
      // here we have an ascii character - go for the sequence
      buffer[index]   = data;
      buffer[index+1] = nilc;

      // try to match it
      if ((count = match_tinfo (p_tinfo, buffer, 1, &match)) == 0) {
      t_quad result = buffer[0];
      unlock ();
      return result;
      }
      // may be we have a match
      if (count == 1) {
      t_quad result = 0x80000000 | (t_quad) match;
      unlock ();
      return result;
      }
      // now we have a match but it is not complete - let's continue
      // we install a timeout for the read - if we timeout we pushback the 
      // buffer sequence if not we continue to match
      while (++index <= d_mtlen) {
      count = 0;
      if (c_rdwait (d_sid, 500) == false) break;
      buffer[index]   = read ();
      buffer[index+1] = nilc;
      count = match_tinfo (p_tinfo, buffer, index + 1, &match);
      if ((count == 0) || 
          ((count == 1) && c_strcmp (p_tinfo[match],buffer))) break;
      }      
      // at this stage, we either have a match, or we timed-out. If we timed
      // out count is null and index indicates the amount of characters to 
      // pushback if count is 1, this is the match, else we pushback the 
      // buffer and return the first character
      if (count != 1) {
      pushback (buffer);
      t_quad result = 0x000000ff & read ();
      unlock ();
      return result;
      }
      t_quad result = 0x80000000 + (t_quad) match;
      unlock ();
      return result;
    } catch (...) {
      unlock ();
      throw;
    }
  }
  
  // return true if we are at the eof

00301   bool InputTerm::iseof (void) const {
    wrlock ();
    // check for the pushback buffer
    if (d_buffer.empty () == false) {
      unlock ();
      return false;
    }
    bool result = d_eof;
    unlock ();
    return result;
  }

  // return true if we can read a character

00315   bool InputTerm::valid (const long tout) const {
    wrlock ();
    // check for the pushback buffer
    if (d_buffer.empty () == false) {
      unlock ();
      return true;
    }
    // check if we can read one character
    bool status = c_rdwait (d_sid, tout);
    if (status == false) {
      unlock ();
      return false;
    }
    // here the eof flag is used
    return (!d_eof);
  }

  // return true if the terminal is a tty
  
00334   bool InputTerm::istty (void) const {
    rdlock ();
    bool result = c_istty (d_sid);
    unlock ();
    return result;
  }
  
  // save the terminal state

00343   void InputTerm::save (void) {
    wrlock ();
    p_attr = c_gtattr (d_sid);
    unlock ();
  }
  
  // restore the terminal state
  
00351   void InputTerm::restore (void) {
    wrlock ();
    c_stattr (d_sid, p_attr);
    unlock ();
  }
  
  // put the terminal in non canonical mode
  
00359   bool InputTerm::nocanon (void) {
    wrlock ();
    bool result = c_stcanon (d_sid);
    unlock ();
    return result;
  }

  // set the ignore eof flag

00368   void InputTerm::setigneof (const bool flag) {
    wrlock ();
    d_igneof = flag;
    unlock ();
  }

  // set the eof remapped character

00376   void InputTerm::seteofmap (const char c) {
    wrlock ();
    d_eofmap = c;
    unlock ();
  }

  // -------------------------------------------------------------------------
  // - object section                                                        -
  // -------------------------------------------------------------------------

  // the quark zone
  static const long QUARK_ZONE_LENGTH = 2;
  static QuarkZone  zone (QUARK_ZONE_LENGTH);
  
  // the object supported quarks
  static const long QUARK_SETEOFIGN = zone.intern ("set-eof-ignore");
  static const long QUARK_SETEOFMAP = zone.intern ("set-eof-character");

  // create a new object in a generic way

00396   Object* InputTerm::mknew (Vector* argv) {
    long argc = (argv == nilp) ? 0 : argv->length ();
    if (argc != 0) 
      throw Exception ("argument-error", 
                   "invalid arguments with with input term"); 
    return new InputTerm;
  }

  // return true if the given quark is defined

00406   bool InputTerm::isquark (const long quark, const bool hflg) const {
    rdlock ();
    if (zone.exists (quark) == true) {
      unlock ();
      return true;
    }
    bool result = hflg ? Input::isquark (quark, hflg) : false;
    unlock ();
    return result;
  }
  
  // apply this object with a set of arguments and a quark

00419   Object* InputTerm::apply (Runnable* robj, Nameset* nset, const long quark,
                      Vector* argv) {
    // get the number of arguments
    long argc = (argv == nilp) ? 0 : argv->length ();

    // check for 1 argument
    if (argc == 1) {
      if (quark == QUARK_SETEOFIGN) {
      bool flag = argv->getbool (0);
      setigneof (flag);
      return nilp;
      }
      if (quark == QUARK_SETEOFMAP) {
      char c = argv->getbyte (0);
      seteofmap (c);
      return nilp;
      }
    }
    // call the input method
    return Input::apply (robj, nset, quark, argv);
  }
}

Generated by  Doxygen 1.6.0   Back to index