345 lines
		
	
	
	
		
			7.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			345 lines
		
	
	
	
		
			7.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* tparam.c - merge parameters into a termcap entry string. */
 | 
						||
 | 
						||
/* Copyright (C) 1985, 1986, 1993,1994, 1995, 1998, 2001,2003,2005,2006,2008,2009 Free Software Foundation, Inc.
 | 
						||
 | 
						||
   This file is part of GNU Bash, the Bourne Again SHell.
 | 
						||
 | 
						||
   Bash 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 3 of the License, or
 | 
						||
   (at your option) any later version.
 | 
						||
 | 
						||
   Bash 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 Bash.  If not, see <http://www.gnu.org/licenses/>.
 | 
						||
*/
 | 
						||
 | 
						||
/* Emacs config.h may rename various library functions such as malloc.  */
 | 
						||
#ifdef HAVE_CONFIG_H
 | 
						||
#include <config.h>
 | 
						||
 | 
						||
#ifdef HAVE_STDLIB_H 
 | 
						||
#  include <stdlib.h>
 | 
						||
#else
 | 
						||
extern char *getenv ();
 | 
						||
extern char *malloc ();
 | 
						||
extern char *realloc ();
 | 
						||
#endif
 | 
						||
 | 
						||
#if defined (HAVE_STRING_H)
 | 
						||
#include <string.h>
 | 
						||
#endif
 | 
						||
 | 
						||
#if !defined (HAVE_BCOPY) && (defined (HAVE_STRING_H) || defined (STDC_HEADERS))
 | 
						||
#  define bcopy(s, d, n)	memcpy ((d), (s), (n))
 | 
						||
#endif
 | 
						||
 | 
						||
#else /* not HAVE_CONFIG_H */
 | 
						||
 | 
						||
#if defined(HAVE_STRING_H) || defined(STDC_HEADERS)
 | 
						||
#define bcopy(s, d, n) memcpy ((d), (s), (n))
 | 
						||
#endif
 | 
						||
 | 
						||
#ifdef STDC_HEADERS
 | 
						||
#include <stdlib.h>
 | 
						||
#include <string.h>
 | 
						||
#else
 | 
						||
char *malloc ();
 | 
						||
char *realloc ();
 | 
						||
#endif
 | 
						||
 | 
						||
#endif /* not HAVE_CONFIG_H */
 | 
						||
 | 
						||
#include "ltcap.h"
 | 
						||
 | 
						||
#ifndef NULL
 | 
						||
#define NULL (char *) 0
 | 
						||
#endif
 | 
						||
 | 
						||
#ifndef emacs
 | 
						||
static void
 | 
						||
memory_out ()
 | 
						||
{
 | 
						||
  write (2, "virtual memory exhausted\n", 25);
 | 
						||
  exit (1);
 | 
						||
}
 | 
						||
 | 
						||
static char *
 | 
						||
xmalloc (size)
 | 
						||
     unsigned size;
 | 
						||
{
 | 
						||
  register char *tem = malloc (size);
 | 
						||
 | 
						||
  if (!tem)
 | 
						||
    memory_out ();
 | 
						||
  return tem;
 | 
						||
}
 | 
						||
 | 
						||
static char *
 | 
						||
xrealloc (ptr, size)
 | 
						||
     char *ptr;
 | 
						||
     unsigned size;
 | 
						||
{
 | 
						||
  register char *tem = realloc (ptr, size);
 | 
						||
 | 
						||
  if (!tem)
 | 
						||
    memory_out ();
 | 
						||
  return tem;
 | 
						||
}
 | 
						||
#endif /* not emacs */
 | 
						||
 | 
						||
/* Assuming STRING is the value of a termcap string entry
 | 
						||
   containing `%' constructs to expand parameters,
 | 
						||
   merge in parameter values and store result in block OUTSTRING points to.
 | 
						||
   LEN is the length of OUTSTRING.  If more space is needed,
 | 
						||
   a block is allocated with `malloc'.
 | 
						||
 | 
						||
   The value returned is the address of the resulting string.
 | 
						||
   This may be OUTSTRING or may be the address of a block got with `malloc'.
 | 
						||
   In the latter case, the caller must free the block.
 | 
						||
 | 
						||
   The fourth and following args to tparam serve as the parameter values.  */
 | 
						||
 | 
						||
static char *tparam1 ();
 | 
						||
 | 
						||
/* VARARGS 2 */
 | 
						||
char *
 | 
						||
tparam (string, outstring, len, arg0, arg1, arg2, arg3)
 | 
						||
     char *string;
 | 
						||
     char *outstring;
 | 
						||
     int len;
 | 
						||
     int arg0, arg1, arg2, arg3;
 | 
						||
{
 | 
						||
  int arg[4];
 | 
						||
 | 
						||
  arg[0] = arg0;
 | 
						||
  arg[1] = arg1;
 | 
						||
  arg[2] = arg2;
 | 
						||
  arg[3] = arg3;
 | 
						||
  return tparam1 (string, outstring, len, NULL, NULL, arg);
 | 
						||
}
 | 
						||
 | 
						||
__private_extern__ char *BC;
 | 
						||
__private_extern__ char *UP;
 | 
						||
 | 
						||
static char tgoto_buf[50];
 | 
						||
 | 
						||
__private_extern__
 | 
						||
char *
 | 
						||
tgoto (cm, hpos, vpos)
 | 
						||
     char *cm;
 | 
						||
     int hpos, vpos;
 | 
						||
{
 | 
						||
  int args[2];
 | 
						||
  if (!cm)
 | 
						||
    return NULL;
 | 
						||
  args[0] = vpos;
 | 
						||
  args[1] = hpos;
 | 
						||
  return tparam1 (cm, tgoto_buf, 50, UP, BC, args);
 | 
						||
}
 | 
						||
 | 
						||
static char *
 | 
						||
tparam1 (string, outstring, len, up, left, argp)
 | 
						||
     char *string;
 | 
						||
     char *outstring;
 | 
						||
     int len;
 | 
						||
     char *up, *left;
 | 
						||
     register int *argp;
 | 
						||
{
 | 
						||
  register int c;
 | 
						||
  register char *p = string;
 | 
						||
  register char *op = outstring;
 | 
						||
  char *outend;
 | 
						||
  int outlen = 0;
 | 
						||
 | 
						||
  register int tem;
 | 
						||
  int *old_argp = argp;
 | 
						||
  int doleft = 0;
 | 
						||
  int doup = 0;
 | 
						||
 | 
						||
  outend = outstring + len;
 | 
						||
 | 
						||
  while (1)
 | 
						||
    {
 | 
						||
      /* If the buffer might be too short, make it bigger.  */
 | 
						||
      if (op + 5 >= outend)
 | 
						||
	{
 | 
						||
	  register char *new;
 | 
						||
	  if (outlen == 0)
 | 
						||
	    {
 | 
						||
	      outlen = len + 40;
 | 
						||
	      new = (char *) xmalloc (outlen);
 | 
						||
	      outend += 40;
 | 
						||
	      bcopy (outstring, new, op - outstring);
 | 
						||
	    }
 | 
						||
	  else
 | 
						||
	    {
 | 
						||
	      outend += outlen;
 | 
						||
	      outlen *= 2;
 | 
						||
	      new = (char *) xrealloc (outstring, outlen);
 | 
						||
	    }
 | 
						||
	  op += new - outstring;
 | 
						||
	  outend += new - outstring;
 | 
						||
	  outstring = new;
 | 
						||
	}
 | 
						||
      c = *p++;
 | 
						||
      if (!c)
 | 
						||
	break;
 | 
						||
      if (c == '%')
 | 
						||
	{
 | 
						||
	  c = *p++;
 | 
						||
	  tem = *argp;
 | 
						||
	  switch (c)
 | 
						||
	    {
 | 
						||
	    case 'd':		/* %d means output in decimal.  */
 | 
						||
	      if (tem < 10)
 | 
						||
		goto onedigit;
 | 
						||
	      if (tem < 100)
 | 
						||
		goto twodigit;
 | 
						||
	    case '3':		/* %3 means output in decimal, 3 digits.  */
 | 
						||
	      if (tem > 999)
 | 
						||
		{
 | 
						||
		  *op++ = tem / 1000 + '0';
 | 
						||
		  tem %= 1000;
 | 
						||
		}
 | 
						||
	      *op++ = tem / 100 + '0';
 | 
						||
	    case '2':		/* %2 means output in decimal, 2 digits.  */
 | 
						||
	    twodigit:
 | 
						||
	      tem %= 100;
 | 
						||
	      *op++ = tem / 10 + '0';
 | 
						||
	    onedigit:
 | 
						||
	      *op++ = tem % 10 + '0';
 | 
						||
	      argp++;
 | 
						||
	      break;
 | 
						||
 | 
						||
	    case 'C':
 | 
						||
	      /* For c-100: print quotient of value by 96, if nonzero,
 | 
						||
		 then do like %+.  */
 | 
						||
	      if (tem >= 96)
 | 
						||
		{
 | 
						||
		  *op++ = tem / 96;
 | 
						||
		  tem %= 96;
 | 
						||
		}
 | 
						||
	    case '+':		/* %+x means add character code of char x.  */
 | 
						||
	      tem += *p++;
 | 
						||
	    case '.':		/* %. means output as character.  */
 | 
						||
	      if (left)
 | 
						||
		{
 | 
						||
		  /* If want to forbid output of 0 and \n and \t,
 | 
						||
		     and this is one of them, increment it.  */
 | 
						||
		  while (tem == 0 || tem == '\n' || tem == '\t')
 | 
						||
		    {
 | 
						||
		      tem++;
 | 
						||
		      if (argp == old_argp)
 | 
						||
			doup++, outend -= strlen (up);
 | 
						||
		      else
 | 
						||
			doleft++, outend -= strlen (left);
 | 
						||
		    }
 | 
						||
		}
 | 
						||
	      *op++ = tem ? tem : 0200;
 | 
						||
	    case 'f':		/* %f means discard next arg.  */
 | 
						||
	      argp++;
 | 
						||
	      break;
 | 
						||
 | 
						||
	    case 'b':		/* %b means back up one arg (and re-use it).  */
 | 
						||
	      argp--;
 | 
						||
	      break;
 | 
						||
 | 
						||
	    case 'r':		/* %r means interchange following two args.  */
 | 
						||
	      argp[0] = argp[1];
 | 
						||
	      argp[1] = tem;
 | 
						||
	      old_argp++;
 | 
						||
	      break;
 | 
						||
 | 
						||
	    case '>':		/* %>xy means if arg is > char code of x, */
 | 
						||
	      if (argp[0] > *p++) /* then add char code of y to the arg, */
 | 
						||
		argp[0] += *p;	/* and in any case don't output.  */
 | 
						||
	      p++;		/* Leave the arg to be output later.  */
 | 
						||
	      break;
 | 
						||
 | 
						||
	    case 'a':		/* %a means arithmetic.  */
 | 
						||
	      /* Next character says what operation.
 | 
						||
		 Add or subtract either a constant or some other arg.  */
 | 
						||
	      /* First following character is + to add or - to subtract
 | 
						||
		 or = to assign.  */
 | 
						||
	      /* Next following char is 'p' and an arg spec
 | 
						||
		 (0100 plus position of that arg relative to this one)
 | 
						||
		 or 'c' and a constant stored in a character.  */
 | 
						||
	      tem = p[2] & 0177;
 | 
						||
	      if (p[1] == 'p')
 | 
						||
		tem = argp[tem - 0100];
 | 
						||
	      if (p[0] == '-')
 | 
						||
		argp[0] -= tem;
 | 
						||
	      else if (p[0] == '+')
 | 
						||
		argp[0] += tem;
 | 
						||
	      else if (p[0] == '*')
 | 
						||
		argp[0] *= tem;
 | 
						||
	      else if (p[0] == '/')
 | 
						||
		argp[0] /= tem;
 | 
						||
	      else
 | 
						||
		argp[0] = tem;
 | 
						||
 | 
						||
	      p += 3;
 | 
						||
	      break;
 | 
						||
 | 
						||
	    case 'i':		/* %i means add one to arg, */
 | 
						||
	      argp[0] ++;	/* and leave it to be output later.  */
 | 
						||
	      argp[1] ++;	/* Increment the following arg, too!  */
 | 
						||
	      break;
 | 
						||
 | 
						||
	    case '%':		/* %% means output %; no arg.  */
 | 
						||
	      goto ordinary;
 | 
						||
 | 
						||
	    case 'n':		/* %n means xor each of next two args with 140.  */
 | 
						||
	      argp[0] ^= 0140;
 | 
						||
	      argp[1] ^= 0140;
 | 
						||
	      break;
 | 
						||
 | 
						||
	    case 'm':		/* %m means xor each of next two args with 177.  */
 | 
						||
	      argp[0] ^= 0177;
 | 
						||
	      argp[1] ^= 0177;
 | 
						||
	      break;
 | 
						||
 | 
						||
	    case 'B':		/* %B means express arg as BCD char code.  */
 | 
						||
	      argp[0] += 6 * (tem / 10);
 | 
						||
	      break;
 | 
						||
 | 
						||
	    case 'D':		/* %D means weird Delta Data transformation.  */
 | 
						||
	      argp[0] -= 2 * (tem % 16);
 | 
						||
	      break;
 | 
						||
	    }
 | 
						||
	}
 | 
						||
      else
 | 
						||
	/* Ordinary character in the argument string.  */
 | 
						||
      ordinary:
 | 
						||
	*op++ = c;
 | 
						||
    }
 | 
						||
  *op = 0;
 | 
						||
  while (doup-- > 0)
 | 
						||
    strcat (op, up);
 | 
						||
  while (doleft-- > 0)
 | 
						||
    strcat (op, left);
 | 
						||
  return outstring;
 | 
						||
}
 | 
						||
 | 
						||
#ifdef DEBUG
 | 
						||
 | 
						||
main (argc, argv)
 | 
						||
     int argc;
 | 
						||
     char **argv;
 | 
						||
{
 | 
						||
  char buf[50];
 | 
						||
  int args[3];
 | 
						||
  args[0] = atoi (argv[2]);
 | 
						||
  args[1] = atoi (argv[3]);
 | 
						||
  args[2] = atoi (argv[4]);
 | 
						||
  tparam1 (argv[1], buf, "LEFT", "UP", args);
 | 
						||
  printf ("%s\n", buf);
 | 
						||
  return 0;
 | 
						||
}
 | 
						||
 | 
						||
#endif /* DEBUG */
 |