/* shell.c -- readline utility functions that are normally provided by
|
/* shell.c -- readline utility functions that are normally provided by
|
bash when readline is linked as part of the shell. */
|
bash when readline is linked as part of the shell. */
|
|
|
/* Copyright (C) 1997 Free Software Foundation, Inc.
|
/* Copyright (C) 1997 Free Software Foundation, Inc.
|
|
|
This file is part of the GNU Readline Library, a library for
|
This file is part of the GNU Readline Library, a library for
|
reading lines of text with interactive input and history editing.
|
reading lines of text with interactive input and history editing.
|
|
|
The GNU Readline Library is free software; you can redistribute it
|
The GNU Readline Library is free software; you can redistribute it
|
and/or modify it under the terms of the GNU General Public License
|
and/or modify it under the terms of the GNU General Public License
|
as published by the Free Software Foundation; either version 2, or
|
as published by the Free Software Foundation; either version 2, or
|
(at your option) any later version.
|
(at your option) any later version.
|
|
|
The GNU Readline Library is distributed in the hope that it will be
|
The GNU Readline Library is distributed in the hope that it will be
|
useful, but WITHOUT ANY WARRANTY; without even the implied warranty
|
useful, but WITHOUT ANY WARRANTY; without even the implied warranty
|
of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
GNU General Public License for more details.
|
GNU General Public License for more details.
|
|
|
The GNU General Public License is often shipped with GNU software, and
|
The GNU General Public License is often shipped with GNU software, and
|
is generally kept in a file called COPYING or LICENSE. If you do not
|
is generally kept in a file called COPYING or LICENSE. If you do not
|
have a copy of the license, write to the Free Software Foundation,
|
have a copy of the license, write to the Free Software Foundation,
|
59 Temple Place, Suite 330, Boston, MA 02111 USA. */
|
59 Temple Place, Suite 330, Boston, MA 02111 USA. */
|
#define READLINE_LIBRARY
|
#define READLINE_LIBRARY
|
|
|
#if defined (HAVE_CONFIG_H)
|
#if defined (HAVE_CONFIG_H)
|
# include <config.h>
|
# include <config.h>
|
#endif
|
#endif
|
|
|
#include <sys/types.h>
|
#include <sys/types.h>
|
|
|
#if defined (HAVE_UNISTD_H)
|
#if defined (HAVE_UNISTD_H)
|
# include <unistd.h>
|
# include <unistd.h>
|
#endif /* HAVE_UNISTD_H */
|
#endif /* HAVE_UNISTD_H */
|
|
|
#if defined (HAVE_STDLIB_H)
|
#if defined (HAVE_STDLIB_H)
|
# include <stdlib.h>
|
# include <stdlib.h>
|
#else
|
#else
|
# include "ansi_stdlib.h"
|
# include "ansi_stdlib.h"
|
#endif /* HAVE_STDLIB_H */
|
#endif /* HAVE_STDLIB_H */
|
|
|
#if defined (HAVE_STRING_H)
|
#if defined (HAVE_STRING_H)
|
# include <string.h>
|
# include <string.h>
|
#else
|
#else
|
# include <strings.h>
|
# include <strings.h>
|
#endif /* !HAVE_STRING_H */
|
#endif /* !HAVE_STRING_H */
|
|
|
#if defined (HAVE_LIMITS_H)
|
#if defined (HAVE_LIMITS_H)
|
# include <limits.h>
|
# include <limits.h>
|
#endif
|
#endif
|
|
|
#if defined (HAVE_FCNTL_H)
|
#if defined (HAVE_FCNTL_H)
|
#include <fcntl.h>
|
#include <fcntl.h>
|
#endif
|
#endif
|
#if defined (HAVE_PWD_H)
|
#if defined (HAVE_PWD_H)
|
#include <pwd.h>
|
#include <pwd.h>
|
#endif
|
#endif
|
|
|
#include <stdio.h>
|
#include <stdio.h>
|
|
|
#include "rlstdc.h"
|
#include "rlstdc.h"
|
#include "rlshell.h"
|
#include "rlshell.h"
|
#include "xmalloc.h"
|
#include "xmalloc.h"
|
|
|
#if defined (HAVE_GETPWUID) && !defined (HAVE_GETPW_DECLS)
|
#if defined (HAVE_GETPWUID) && !defined (HAVE_GETPW_DECLS)
|
extern struct passwd *getpwuid PARAMS((uid_t));
|
extern struct passwd *getpwuid PARAMS((uid_t));
|
#endif /* HAVE_GETPWUID && !HAVE_GETPW_DECLS */
|
#endif /* HAVE_GETPWUID && !HAVE_GETPW_DECLS */
|
|
|
#ifndef NULL
|
#ifndef NULL
|
# define NULL 0
|
# define NULL 0
|
#endif
|
#endif
|
|
|
#ifndef CHAR_BIT
|
#ifndef CHAR_BIT
|
# define CHAR_BIT 8
|
# define CHAR_BIT 8
|
#endif
|
#endif
|
|
|
/* Nonzero if the integer type T is signed. */
|
/* Nonzero if the integer type T is signed. */
|
#define TYPE_SIGNED(t) (! ((t) 0 < (t) -1))
|
#define TYPE_SIGNED(t) (! ((t) 0 < (t) -1))
|
|
|
/* Bound on length of the string representing an integer value of type T.
|
/* Bound on length of the string representing an integer value of type T.
|
Subtract one for the sign bit if T is signed;
|
Subtract one for the sign bit if T is signed;
|
302 / 1000 is log10 (2) rounded up;
|
302 / 1000 is log10 (2) rounded up;
|
add one for integer division truncation;
|
add one for integer division truncation;
|
add one more for a minus sign if t is signed. */
|
add one more for a minus sign if t is signed. */
|
#define INT_STRLEN_BOUND(t) \
|
#define INT_STRLEN_BOUND(t) \
|
((sizeof (t) * CHAR_BIT - TYPE_SIGNED (t)) * 302 / 1000 \
|
((sizeof (t) * CHAR_BIT - TYPE_SIGNED (t)) * 302 / 1000 \
|
+ 1 + TYPE_SIGNED (t))
|
+ 1 + TYPE_SIGNED (t))
|
|
|
/* All of these functions are resolved from bash if we are linking readline
|
/* All of these functions are resolved from bash if we are linking readline
|
as part of bash. */
|
as part of bash. */
|
|
|
/* Does shell-like quoting using single quotes. */
|
/* Does shell-like quoting using single quotes. */
|
char *
|
char *
|
sh_single_quote (string)
|
sh_single_quote (string)
|
char *string;
|
char *string;
|
{
|
{
|
register int c;
|
register int c;
|
char *result, *r, *s;
|
char *result, *r, *s;
|
|
|
result = (char *)xmalloc (3 + (4 * strlen (string)));
|
result = (char *)xmalloc (3 + (4 * strlen (string)));
|
r = result;
|
r = result;
|
*r++ = '\'';
|
*r++ = '\'';
|
|
|
for (s = string; s && (c = *s); s++)
|
for (s = string; s && (c = *s); s++)
|
{
|
{
|
*r++ = c;
|
*r++ = c;
|
|
|
if (c == '\'')
|
if (c == '\'')
|
{
|
{
|
*r++ = '\\'; /* insert escaped single quote */
|
*r++ = '\\'; /* insert escaped single quote */
|
*r++ = '\'';
|
*r++ = '\'';
|
*r++ = '\''; /* start new quoted string */
|
*r++ = '\''; /* start new quoted string */
|
}
|
}
|
}
|
}
|
|
|
*r++ = '\'';
|
*r++ = '\'';
|
*r = '\0';
|
*r = '\0';
|
|
|
return (result);
|
return (result);
|
}
|
}
|
|
|
/* Set the environment variables LINES and COLUMNS to lines and cols,
|
/* Set the environment variables LINES and COLUMNS to lines and cols,
|
respectively. */
|
respectively. */
|
void
|
void
|
sh_set_lines_and_columns (lines, cols)
|
sh_set_lines_and_columns (lines, cols)
|
int lines, cols;
|
int lines, cols;
|
{
|
{
|
char *b;
|
char *b;
|
|
|
#if defined (HAVE_SETENV)
|
#if defined (HAVE_SETENV)
|
b = (char *)xmalloc (INT_STRLEN_BOUND (int) + 1);
|
b = (char *)xmalloc (INT_STRLEN_BOUND (int) + 1);
|
sprintf (b, "%d", lines);
|
sprintf (b, "%d", lines);
|
setenv ("LINES", b, 1);
|
setenv ("LINES", b, 1);
|
free (b);
|
free (b);
|
|
|
b = (char *)xmalloc (INT_STRLEN_BOUND (int) + 1);
|
b = (char *)xmalloc (INT_STRLEN_BOUND (int) + 1);
|
sprintf (b, "%d", cols);
|
sprintf (b, "%d", cols);
|
setenv ("COLUMNS", b, 1);
|
setenv ("COLUMNS", b, 1);
|
free (b);
|
free (b);
|
#else /* !HAVE_SETENV */
|
#else /* !HAVE_SETENV */
|
# if defined (HAVE_PUTENV)
|
# if defined (HAVE_PUTENV)
|
b = (char *)xmalloc (INT_STRLEN_BOUND (int) + sizeof ("LINES=") + 1);
|
b = (char *)xmalloc (INT_STRLEN_BOUND (int) + sizeof ("LINES=") + 1);
|
sprintf (b, "LINES=%d", lines);
|
sprintf (b, "LINES=%d", lines);
|
putenv (b);
|
putenv (b);
|
|
|
b = (char *)xmalloc (INT_STRLEN_BOUND (int) + sizeof ("COLUMNS=") + 1);
|
b = (char *)xmalloc (INT_STRLEN_BOUND (int) + sizeof ("COLUMNS=") + 1);
|
sprintf (b, "COLUMNS=%d", cols);
|
sprintf (b, "COLUMNS=%d", cols);
|
putenv (b);
|
putenv (b);
|
# endif /* HAVE_PUTENV */
|
# endif /* HAVE_PUTENV */
|
#endif /* !HAVE_SETENV */
|
#endif /* !HAVE_SETENV */
|
}
|
}
|
|
|
char *
|
char *
|
sh_get_env_value (varname)
|
sh_get_env_value (varname)
|
const char *varname;
|
const char *varname;
|
{
|
{
|
return ((char *)getenv (varname));
|
return ((char *)getenv (varname));
|
}
|
}
|
|
|
char *
|
char *
|
sh_get_home_dir ()
|
sh_get_home_dir ()
|
{
|
{
|
char *home_dir;
|
char *home_dir;
|
struct passwd *entry;
|
struct passwd *entry;
|
|
|
home_dir = (char *)NULL;
|
home_dir = (char *)NULL;
|
#if defined (HAVE_GETPWUID)
|
#if defined (HAVE_GETPWUID)
|
entry = getpwuid (getuid ());
|
entry = getpwuid (getuid ());
|
if (entry)
|
if (entry)
|
home_dir = entry->pw_dir;
|
home_dir = entry->pw_dir;
|
#endif
|
#endif
|
return (home_dir);
|
return (home_dir);
|
}
|
}
|
|
|
#if !defined (O_NDELAY)
|
#if !defined (O_NDELAY)
|
# if defined (FNDELAY)
|
# if defined (FNDELAY)
|
# define O_NDELAY FNDELAY
|
# define O_NDELAY FNDELAY
|
# endif
|
# endif
|
#endif
|
#endif
|
|
|
int
|
int
|
sh_unset_nodelay_mode (fd)
|
sh_unset_nodelay_mode (fd)
|
int fd;
|
int fd;
|
{
|
{
|
#if defined (HAVE_FCNTL)
|
#if defined (HAVE_FCNTL)
|
int flags, bflags;
|
int flags, bflags;
|
|
|
if ((flags = fcntl (fd, F_GETFL, 0)) < 0)
|
if ((flags = fcntl (fd, F_GETFL, 0)) < 0)
|
return -1;
|
return -1;
|
|
|
bflags = 0;
|
bflags = 0;
|
|
|
#ifdef O_NONBLOCK
|
#ifdef O_NONBLOCK
|
bflags |= O_NONBLOCK;
|
bflags |= O_NONBLOCK;
|
#endif
|
#endif
|
|
|
#ifdef O_NDELAY
|
#ifdef O_NDELAY
|
bflags |= O_NDELAY;
|
bflags |= O_NDELAY;
|
#endif
|
#endif
|
|
|
if (flags & bflags)
|
if (flags & bflags)
|
{
|
{
|
flags &= ~bflags;
|
flags &= ~bflags;
|
return (fcntl (fd, F_SETFL, flags));
|
return (fcntl (fd, F_SETFL, flags));
|
}
|
}
|
#endif
|
#endif
|
|
|
return 0;
|
return 0;
|
}
|
}
|
|
|