Blame src/lftp_pty.c

Packit 8f70b4
/*
Packit 8f70b4
 * lftp - file transfer program
Packit 8f70b4
 *
Packit 8f70b4
 * Copyright (c) 1996-2012 by Alexander V. Lukyanov (lav@yars.free.net)
Packit 8f70b4
 *
Packit 8f70b4
 * This program is free software; you can redistribute it and/or modify
Packit 8f70b4
 * it under the terms of the GNU General Public License as published by
Packit 8f70b4
 * the Free Software Foundation; either version 3 of the License, or
Packit 8f70b4
 * (at your option) any later version.
Packit 8f70b4
 *
Packit 8f70b4
 * This program is distributed in the hope that it will be useful,
Packit 8f70b4
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 8f70b4
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 8f70b4
 * GNU General Public License for more details.
Packit 8f70b4
 *
Packit 8f70b4
 * You should have received a copy of the GNU General Public License
Packit 8f70b4
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
Packit 8f70b4
 */
Packit 8f70b4
Packit 8f70b4
/* Based on sshpty.c from openssh written by Tatu Ylonen <ylo@cs.hut.fi> */
Packit 8f70b4
Packit 8f70b4
#include <config.h>
Packit 8f70b4
Packit 8f70b4
#include <sys/types.h>
Packit 8f70b4
#include <unistd.h>
Packit 8f70b4
#include <string.h>
Packit 8f70b4
#include <signal.h>
Packit 8f70b4
#include <fcntl.h>
Packit 8f70b4
#include <stdio.h>
Packit 8f70b4
#include <stdlib.h>
Packit 8f70b4
#include <sys/stat.h>
Packit 8f70b4
Packit 8f70b4
#ifdef HAVE_TERMIOS_H
Packit 8f70b4
# include <termios.h>
Packit 8f70b4
#endif
Packit 8f70b4
#ifdef HAVE_UTIL_H
Packit 8f70b4
# include <util.h>
Packit 8f70b4
#endif
Packit 8f70b4
#ifdef HAVE_PTY_H
Packit 8f70b4
# include <pty.h>
Packit 8f70b4
#endif
Packit 8f70b4
#ifdef HAVE_SYS_STROPTS_H
Packit 8f70b4
# include <sys/stropts.h>
Packit 8f70b4
#endif
Packit 8f70b4
Packit 8f70b4
#include "lftp_pty.h"
Packit 8f70b4
Packit 8f70b4
const char *open_pty(int *ptyfd, int *ttyfd)
Packit 8f70b4
{
Packit 8f70b4
   const char *name=0;
Packit 8f70b4
   void *old_sigchld=signal(SIGCHLD,SIG_DFL);
Packit 8f70b4
   *ptyfd=*ttyfd=-1;
Packit 8f70b4
Packit 8f70b4
#if defined(HAVE_OPENPTY) || defined(BSD4_4)
Packit 8f70b4
   /* openpty(3) exists in OSF/1 and some other os'es */
Packit 8f70b4
   openpty(ptyfd, ttyfd, NULL, NULL, NULL);
Packit 8f70b4
#else /* HAVE_OPENPTY */
Packit 8f70b4
#ifdef HAVE__GETPTY
Packit 8f70b4
   /*
Packit 8f70b4
    * _getpty(3) exists in SGI Irix 4.x, 5.x & 6.x -- it generates more
Packit 8f70b4
    * pty's automagically when needed
Packit 8f70b4
    */
Packit 8f70b4
   name=_getpty(ptyfd,O_RDWR|O_NOCTTY,0600,0);
Packit 8f70b4
#else /* HAVE__GETPTY */
Packit 8f70b4
#if defined(HAVE_DEV_PTMX)
Packit 8f70b4
   /*
Packit 8f70b4
    * This code is used e.g. on Solaris 2.x.  (Note that Solaris 2.3
Packit 8f70b4
    * also has bsd-style ptys, but they simply do not work.)
Packit 8f70b4
    */
Packit 8f70b4
   *ptyfd=open("/dev/ptmx",O_RDWR|O_NOCTTY);
Packit 8f70b4
   if(*ptyfd<0)
Packit 8f70b4
      goto fail;
Packit 8f70b4
   if(grantpt(*ptyfd)<0)
Packit 8f70b4
      goto fail;
Packit 8f70b4
   if(unlockpt(*ptyfd)<0)
Packit 8f70b4
      goto fail;
Packit 8f70b4
   name=ptsname(*ptyfd);
Packit 8f70b4
#else /* HAVE_DEV_PTMX */
Packit 8f70b4
#ifdef HAVE_DEV_PTS_AND_PTC
Packit 8f70b4
{
Packit 8f70b4
   /* AIX-style pty code. */
Packit 8f70b4
   *ptyfd=open("/dev/ptc",O_RDWR|O_NOCTTY);
Packit 8f70b4
   if(*ptyfd<0)
Packit 8f70b4
      goto fail;
Packit 8f70b4
   name=ttyname(*ptyfd);
Packit 8f70b4
}
Packit 8f70b4
#else /* HAVE_DEV_PTS_AND_PTC */
Packit 8f70b4
{
Packit 8f70b4
   /* BSD-style pty code. */
Packit 8f70b4
   char master[64];
Packit 8f70b4
   static char slave[64];
Packit 8f70b4
   int i;
Packit 8f70b4
   const char *ptymajors="pqrstuvwxyzabcdefghijklmnoABCDEFGHIJKLMNOPQRSTUVWXYZ";
Packit 8f70b4
   const char *ptyminors="0123456789abcdef";
Packit 8f70b4
   int num_minors=strlen(ptyminors);
Packit 8f70b4
   int num_ptys=strlen(ptymajors)*num_minors;
Packit 8f70b4
Packit 8f70b4
   for(i=0; i
Packit 8f70b4
   {
Packit 8f70b4
      sprintf(master,"/dev/pty%c%c",ptymajors[i/num_minors],ptyminors[i%num_minors]);
Packit 8f70b4
      sprintf(slave, "/dev/tty%c%c",ptymajors[i/num_minors],ptyminors[i%num_minors]);
Packit 8f70b4
      *ptyfd=open(master,O_RDWR|O_NOCTTY);
Packit 8f70b4
      if(*ptyfd>=0)
Packit 8f70b4
	 break;
Packit 8f70b4
      /* Try SCO style naming */
Packit 8f70b4
      sprintf(master,"/dev/ptyp%d",i);
Packit 8f70b4
      sprintf(slave, "/dev/ttyp%d",i);
Packit 8f70b4
      *ptyfd=open(master,O_RDWR|O_NOCTTY);
Packit 8f70b4
      if(*ptyfd>=0)
Packit 8f70b4
	 break;
Packit 8f70b4
   }
Packit 8f70b4
   name=slave;
Packit 8f70b4
}
Packit 8f70b4
#endif /* HAVE_DEV_PTS_AND_PTC */
Packit 8f70b4
#endif /* HAVE_DEV_PTMX */
Packit 8f70b4
#endif /* HAVE__GETPTY */
Packit 8f70b4
#endif /* HAVE_OPENPTY */
Packit 8f70b4
Packit 8f70b4
   if(*ptyfd<0)
Packit 8f70b4
      goto fail;
Packit 8f70b4
   if(name && *ttyfd<0)
Packit 8f70b4
      *ttyfd=open(name,O_RDWR|O_NOCTTY);
Packit 8f70b4
   if(!name && *ttyfd>=0)
Packit 8f70b4
      name=ttyname(*ttyfd);
Packit 8f70b4
Packit 8f70b4
   if(!name || *ttyfd<0)
Packit 8f70b4
      goto fail;
Packit 8f70b4
Packit 8f70b4
#if defined(HAVE_DEV_PTMX) && defined(I_PUSH) \
Packit 8f70b4
 && !defined(HAVE__GETPTY) && !defined(HAVE_OPENPTY)
Packit 8f70b4
   ioctl(*ttyfd, I_PUSH, "ptem");
Packit 8f70b4
   ioctl(*ttyfd, I_PUSH, "ldterm");
Packit 8f70b4
   ioctl(*ttyfd, I_PUSH, "ttcompat");
Packit 8f70b4
#endif
Packit 8f70b4
#ifdef HAVE_FCHMOD
Packit 8f70b4
   fchmod(*ttyfd,0600);
Packit 8f70b4
#else
Packit 8f70b4
   chmod(name,0600);
Packit 8f70b4
#endif
Packit 8f70b4
   signal(SIGCHLD,old_sigchld);
Packit 8f70b4
   return name;
Packit 8f70b4
Packit 8f70b4
fail:
Packit 8f70b4
   if(*ttyfd>=0)
Packit 8f70b4
      close(*ttyfd);
Packit 8f70b4
   if(*ptyfd>=0)
Packit 8f70b4
      close(*ptyfd);
Packit 8f70b4
   signal(SIGCHLD,old_sigchld);
Packit 8f70b4
   return 0;
Packit 8f70b4
}