ftp.nice.ch/pub/next/unix/editor/xemacs.19.13.s.tar.gz#/xemacs-19.13/src/device-tty.c

This is device-tty.c in view mode; [Download] [Up]

/* TTY device functions.
   Copyright (C) 1994, 1995 Board of Trustees, University of Illinois
   Copyright (C) 1994, 1995 Amdahl Corporation

This file is part of XEmacs.

XEmacs is free software; you can redistribute it and/or modify it
under the terms of the GNU General Public License as published by the
Free Software Foundation; either version 2, or (at your option) any
later version.

XEmacs 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 General Public License
for more details.

You should have received a copy of the GNU General Public License
along with XEmacs; see the file COPYING.  If not, write to the Free
Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.  */

/* Synched up with: Not in FSF. */

/* Authors: Chuck Thompson and Ben Wing. */

#include <config.h>
#include "lisp.h"

#include "device-tty.h"
#include "device-stream.h"
#include "events.h"
#include "faces.h"
#include "frame.h"
#include "redisplay.h"
#include "sysdep.h"

#include "syssignal.h" /* for SIGWINCH */

#include <errno.h>

DEFINE_DEVICE_TYPE (tty);

Lisp_Object Qterminal_type;


static void
allocate_tty_device_struct (struct device *d)
{
  d->device_data =
    (struct tty_device *) xmalloc (sizeof (struct tty_device));

  /* zero out all slots. */
  memset (d->device_data, 0, sizeof (struct tty_device));
  /* except the lisp ones ... */
  DEVICE_TTY_DATA (d)->tty = Qnil;
  DEVICE_TTY_DATA (d)->terminal_type = Qnil;
}

static void
tty_init_device (struct device *d, Lisp_Object params)
{
  Lisp_Object tty = Qnil, terminal_type = Qnil;
  FILE *infd, *outfd;

  tty = Fcdr_safe (Fassq (Qtty, params));
  terminal_type = Fcdr_safe (Fassq (Qterminal_type, params));

  /* Open the specified device */

  if (NILP (tty))
    {
      infd = stdin;
      outfd = stdout;
    }
  else
    {
      CHECK_STRING (tty, 0);
      infd = outfd = fopen ((char *) string_data (XSTRING (tty)), "r+");
      if (!infd)
	error ("Unable to open tty %s", string_data (XSTRING (tty)));
    }
  
  /* Determine the terminal type */

  if (!NILP (terminal_type))
    CHECK_STRING (terminal_type, 0);
  else
    {
      char *temp_type = (char *) getenv ("TERM");

      if (!temp_type)
	{
	  if (infd != stdin)
	    fclose (infd);
	  error ("Cannot determine terminal type");
	}
      else
	terminal_type = build_string (temp_type);
    }

  allocate_tty_device_struct (d);
  DEVICE_TTY_DATA (d)->infd = infd;
  DEVICE_TTY_DATA (d)->outfd = outfd;
  DEVICE_TTY_DATA (d)->terminal_type = terminal_type;
  if (NILP (tty))
    tty = build_string ("stdio");
  DEVICE_TTY_DATA (d)->tty = tty;
  if (NILP (DEVICE_NAME (d)))
    DEVICE_NAME (d) = Ffile_name_nondirectory (tty);
  DEVICE_INFD (d) = fileno (infd);
  DEVICE_OUTFD (d) = fileno (outfd);
  {
    int tty_pg;
    int controlling_tty_pg;
    int cfd;

    /* OK, the only sure-fire way I can think of to determine
       whether a particular TTY is our controlling TTY is to check
       if it has the same foreground process group as our controlling
       TTY.  This is OK because a process group can never simultaneously
       be the foreground process group of two TTY's (in that case it
       would have two controlling TTY's, which is not allowed). */

    EMACS_GET_TTY_PROCESS_GROUP (fileno (infd), &tty_pg);
    cfd = open ("/dev/tty", O_RDWR, 0);
    EMACS_GET_TTY_PROCESS_GROUP (cfd, &controlling_tty_pg);
    close (cfd);
    if (tty_pg == controlling_tty_pg)
      {
	DEVICE_TTY_DATA (d)->controlling_terminal = 1;
	XSETDEVICE (Vcontrolling_terminal, d);
	munge_tty_process_group ();
      }
    else
      DEVICE_TTY_DATA (d)->controlling_terminal = 0;
  }

  init_baud_rate (d);

  switch (init_tty_for_redisplay
	  (d, (char *) string_data (XSTRING (terminal_type))))
    {
    case TTY_UNABLE_OPEN_DATABASE:
      error ("Can't access terminal information database");
      break;
    case TTY_TYPE_UNDEFINED:
      error ("Terminal type `%s' undefined",
	     string_data (XSTRING (terminal_type)));
      break;
    case TTY_TYPE_INSUFFICIENT:
      error ("Terminal type `%s' not powerful enough to run Emacs",
	     string_data (XSTRING (terminal_type)));
      break;
    case TTY_SIZE_UNSPECIFIED:
      error ("Can't determine window size of terminal");
      break;
    case TTY_INIT_SUCCESS:
      break;
    default:
      abort ();
    }

  init_one_device (d);

  /* This will cause the elisp side of the TTY device intialization to
     be loaded.  At this point we consider the two parts to be one.
     They are split up because of requirements in lisp/prim/startup.el */
  Frequire (intern ("pre-tty-win"), Qnil);
  Frequire (intern ("post-tty-win"), Qnil);
}

static void
tty_mark_device (struct device *d, void (*markobj) (Lisp_Object))
{
  ((markobj) (DEVICE_TTY_DATA (d)->tty));
  ((markobj) (DEVICE_TTY_DATA (d)->terminal_type));
}


static int
tty_initially_selected_for_input (struct device *d)
{
  return 1;
}

static void
free_tty_device_struct (struct device *d)
{
  struct tty_device *td = (struct tty_device *) d->device_data;
  if (td && td->term_entry_buffer) /* allocated in term_init () */
    xfree (td->term_entry_buffer);
  if (td)
    xfree (td);
}

static void
tty_delete_device (struct device *d)
{
  if (DEVICE_TTY_DATA (d)->infd != stdin)
    fclose (DEVICE_TTY_DATA (d)->infd);
  if (DEVICE_TTY_DATA (d)->controlling_terminal)
    {
      Vcontrolling_terminal = Qnil;
      unmunge_tty_process_group ();
    }
  free_tty_device_struct (d);
}

#ifdef SIGWINCH

static SIGTYPE
tty_device_size_change_signal (int signo)
{
  int old_errno = errno;
  asynch_device_change_pending++;
  signal_fake_event ();
  EMACS_REESTABLISH_SIGNAL (SIGWINCH, tty_device_size_change_signal);
  errno = old_errno;
  SIGRETURN;
}

/* frame_change_signal does nothing but set a flag that it was called.
   When redisplay is called, it will notice that the flag is set and
   call handle_pending_device_size_change to do the actual work. */
static void
tty_asynch_device_change (void)
{
  Lisp_Object device;

  for (device = Vdevice_list ; !NILP (device) ; device = XCDR (device))
    {
      int width, height;
      Lisp_Object tail;
      struct device *d = XDEVICE (XCAR (device));

      if (!DEVICE_IS_TTY (d))
	continue;

      get_tty_device_size (d, &width, &height);
      if (width > 0 && height > 0)
	{
	  DEVICE_TTY_DATA (d)->width = width;
	  DEVICE_TTY_DATA (d)->height = height;

	  for (tail = DEVICE_FRAME_LIST (d);
	       !NILP (tail);
	       tail = XCDR (tail))
	    {
	      struct frame *f = XFRAME (XCAR (tail));
	      
	      /* We know the frame is tty because we made sure that the
		 device is tty. */
	      change_frame_size (f, height, width, 0, 1);
	    }
	}
    }
}

#endif /* SIGWINCH */

static int
tty_device_pixel_width (struct device *d)
{
  return DEVICE_TTY_DATA (d)->width;
}

static int
tty_device_pixel_height (struct device *d)
{
  return DEVICE_TTY_DATA (d)->height;
}


static struct device *
get_tty_device (Lisp_Object device)
{
  if (NILP (device))
    device = Fselected_device ();
  /* quietly accept frames for the device arg */
  if (FRAMEP (device))
    device = XFRAME (device)->device;
  CHECK_TTY_DEVICE (device, 0);
  return XDEVICE (device);
}

DEFUN ("device-tty-tty", Fdevice_tty_tty, Sdevice_tty_tty, 0, 1, 0,
       "Return the tty filename which DEVICE is connected to, as a string.")
     (device)
     Lisp_Object device;
{
  return DEVICE_TTY_DATA (get_tty_device (device))->tty;
}

DEFUN ("device-tty-terminal-type", Fdevice_tty_terminal_type,
       Sdevice_tty_terminal_type, 0, 1, 0,
  "Return the terminal type of TTY device DEVICE.")
     (device)
     Lisp_Object device;
{
  return DEVICE_TTY_DATA (get_tty_device (device))->terminal_type;
}


/************************************************************************/
/*                            initialization                            */
/************************************************************************/

void
syms_of_device_tty (void)
{
  defsubr (&Sdevice_tty_tty);
  defsubr (&Sdevice_tty_terminal_type);
  defsymbol (&Qterminal_type, "terminal-type");
}

void
device_type_create_tty (void)
{
  INITIALIZE_DEVICE_TYPE (tty, "tty", "device-tty-p");

  DEVICE_HAS_METHOD (tty, init_device);
  DEVICE_HAS_METHOD (tty, mark_device);
  DEVICE_HAS_METHOD (tty, initially_selected_for_input);
  DEVICE_HAS_METHOD (tty, delete_device);
#ifdef SIGWINCH
  DEVICE_HAS_METHOD (tty, asynch_device_change);
#endif /* SIGWINCH */
  DEVICE_HAS_METHOD (tty, device_pixel_width);
  DEVICE_HAS_METHOD (tty, device_pixel_height);
}

void
vars_of_device_tty (void)
{
  Fprovide (Qtty);
}

void
init_device_tty (void)
{
#ifdef SIGWINCH
  if (initialized && !noninteractive)
    signal (SIGWINCH, tty_device_size_change_signal);
#endif /* SIGWINCH */
}

These are the contents of the former NiCE NeXT User Group NeXTSTEP/OpenStep software archive, currently hosted by Netfuture.ch.