/*
 * CDDL HEADER START
 *
 * The contents of this file are subject to the terms of the
 * Common Development and Distribution License, Version 1.0 only
 * (the "License").  You may not use this file except in compliance
 * with the License.
 *
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
 * or http://www.opensolaris.org/os/licensing.
 * See the License for the specific language governing permissions
 * and limitations under the License.
 *
 * When distributing Covered Code, include this CDDL HEADER in each
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
 * If applicable, add the following below this CDDL HEADER, with the
 * fields enclosed by brackets "[]" replaced with your own identifying
 * information: Portions Copyright [yyyy] [name of copyright owner]
 *
 * CDDL HEADER END
 */
/*
 * Copyright 2004 Sun Microsystems, Inc.  All rights reserved.
 * Use is subject to license terms.
 */

/*	Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T	*/
/*	  All Rights Reserved  	*/


#pragma ident	"%Z%%M%	%I%	%E% SMI"


#include <stdlib.h>
#include <stdio.h>
#include <errno.h>
#include <termio.h>
#include <sys/stermio.h>
#include <sys/termiox.h>
#include <string.h>
#include <ctype.h>
#include <unistd.h>
#include <sys/types.h>
#include "sys/stropts.h"
#include "sys/signal.h"
#include "ttymon.h" 
#include "tmstruct.h" 
#include "tmextern.h" 

extern	void	mkargv();

/*
 *	set_termio	- set termio on device 
 *		fd	- fd for the device
 *		options - stty termio options 
 *		aspeed  - autobaud speed 
 *		clear	- if TRUE, current flags will be set to some defaults
 *			  before applying the options 
 *		    	- if FALSE, current flags will not be cleared
 *		mode	- terminal mode, CANON, RAW
 */



int
set_termio(fd,options,aspeed,clear,mode)
int	fd;
char	*options;
char	*aspeed;
int	clear;
long	mode;
{
	struct 	 termio termio;
	struct 	 termios termios;
	struct 	 stio stermio;
	struct 	 termiox termiox;
	struct 	 winsize winsize;
	struct 	 winsize owinsize;
	int	 term;
	int	 cnt = 1;
	char	 *uarg;	
	char	 *argvp[MAXARGS];	/* stty args */
	static   char	 *binstty = "/usr/bin/stty";
	static	 char	buf[BUFSIZ];
	extern 	 int get_ttymode(), set_ttymode();
	extern	 char	*sttyparse();

#ifdef	DEBUG
	debug("in set_termio");
#endif

	if ((term = get_ttymode(fd, &termio, &termios, &stermio, 
				&termiox, &winsize)) < 0) {
		log("set_termio: get_ttymode failed: %s", strerror(errno));
		return(-1);
	}
	owinsize = winsize;
	if (clear) {
		if (mode & CANON) {
			/* could have removed these too - rely on defaults */
			termios.c_cc[VEOF] = CEOF;
			termios.c_cc[VEOL] = CNUL;
		}
		else  {
			termios.c_lflag &= ECHO;
			termios.c_cc[VMIN] = 1;
			termios.c_cc[VTIME] = 0;
		}

	}

	if (options != NULL && *options != '\0') {
		/* just a place holder to make it look like invoking stty */
		argvp[0] = binstty;
		(void)strcpy(buf,options);
		mkargv(buf,&argvp[1],&cnt,MAXARGS-1);
		if (aspeed != NULL && *aspeed != '\0') {
			argvp[cnt++] = aspeed;
		}
		argvp[cnt] = (char *)0;
		if ((uarg = sttyparse(cnt, argvp, term, &termio, &termios, 
				&termiox, &winsize)) != NULL) {
			log("sttyparse unknown mode: %s", uarg);
			return(-1);
		}
	}


	if (set_ttymode(fd, term, &termio, &termios, &stermio, 
			&termiox, &winsize, &owinsize) != 0) {
		log("set_termio: set_ttymode failed", strerror(errno));
		return(-1);
	}

	return(0);
}

#ifdef	NOT_USE
/*
 *	turnon_canon	- turn on canonical processing
 *			- return 0 if succeeds, -1 if fails
 */
turnon_canon(fd)
int	fd;
{
	struct termio termio;

#ifdef	DEBUG
	debug("in turnon_canon");
#endif
	if (ioctl(fd, TCGETA, &termio) != 0) {
		log("turnon_canon: TCGETA failed, fd = %d: %s", fd,
		    strerror(errno));
		return(-1);
	}
	termio.c_lflag |= (ISIG|ICANON|ECHO|ECHOE|ECHOK); 
	termio.c_cc[VEOF] = CEOF;
	termio.c_cc[VEOL] = CNUL;
	if (ioctl(fd, TCSETA, &termio) != 0) {
		log("turnon_canon: TCSETA failed, fd = %d: %s", fd,
		    strerror(errno));
		return(-1);
	}
	return(0);
}
#endif

/*
 *	flush_input	- flush the input queue
 */
void
flush_input(fd)
int	fd;
{
	if (ioctl(fd, I_FLUSH, FLUSHR) == -1)
		log("flush_input failed, fd = %d: %s", fd, strerror(errno));

	if (ioctl(fd, TCSBRK, 1) == -1)
		log("drain of ouput failed, fd = %d: %s", fd, strerror(errno));

	return;
}

/*
 * push_linedisc	- if modules is not NULL, pop everything
 *			- then push modules specified by "modules"
 */
int
push_linedisc(
	int	fd,	/* fd to push modules on */
	char	*modules, /* ptr to a list of comma separated module names */
	char	*device) /* device name for printing msg */
{
	char	*p, *tp;
	char	buf[BUFSIZ];

#ifdef	DEBUG
	debug("in push_linedisc");
#endif
	/*
	 * copy modules into buf so we won't mess up the original buffer
	 * because strtok will chop the string
	 */
	p = strcpy(buf,modules);

	while(ioctl(fd, I_POP, 0) >= 0)  /* pop everything */ 
		;
	for (p=(char *)strtok(p,","); p!=(char *)NULL; 
		p=(char *)strtok(NULL,",")) {
		for (tp = p + strlen(p) - 1; tp >= p && isspace(*tp); --tp)
			*tp = '\0';
		if (ioctl(fd, I_PUSH, p) == -1) {
			log("push (%s) on %s failed: %s", p, device,
			    strerror(errno));
			return(-1);
		}  
	}
	return(0);
}

/*
 *	hang_up_line	- set speed to B0. This will drop DTR
 */
int
hang_up_line(int fd)
{
	struct termio termio;
	struct termios termios;

#ifdef	DEBUG
	debug("in hang_up_line");
#endif
	if (ioctl(fd,TCGETS,&termios) < 0) {
	    if (ioctl(fd,TCGETA,&termio) < 0) {
		log("hang_up_line: TCGETA failed: %s", strerror(errno));
		return(-1);
	    }
	    termio.c_cflag &= ~CBAUD;
	    termio.c_cflag |= B0;

	    if (ioctl(fd,TCSETA,&termio) < 0) {
		log("hang_up_line: TCSETA failed: %s", strerror(errno));
		return(-1);
	    }
	} else {
	    cfsetospeed(&termios, B0);

	    if (ioctl(fd,TCSETS,&termios) < 0) {
		log("hang_up_line: TCSETS failed: %s", strerror(errno));
		return(-1);
	    }
	}
	return(0);
}

/*
 * initial_termio	- set initial termios
 *			- return 0 if successful, -1 if failed.
 */
int
initial_termio(fd,pmptr)
int	fd;
struct	pmtab	*pmptr;
{
	int	ret;
	struct	Gdef *speedef;
	struct	Gdef *get_speed();
	extern	int  auto_termio();

	speedef = get_speed(pmptr->p_ttylabel);
	if (speedef->g_autobaud & A_FLAG) {
		pmptr->p_ttyflags |= A_FLAG;
		if (auto_termio(fd) == -1) {
			(void)close(fd);
			return(-1);
		}
	}
	else {
		if (pmptr->p_ttyflags & R_FLAG)
			ret = set_termio(fd,speedef->g_iflags,
				(char *)NULL, TRUE, (long)RAW);
		else 
			ret = set_termio(fd,speedef->g_iflags,
				(char *)NULL, TRUE, (long)CANON);
		if (ret == -1) {
			log("initial termio on (%s) failed", pmptr->p_device);
			(void)close(fd);
			return(-1);
		}
	}
	return(0);
}