e05c129dde
Our libedit has been diverging from the mainstream version maintained in NetBSD. As a consequence it has been difficult to do an appropriate MFV and we have been bringing only partial updates. Here we update most of the files to at least match the version available in NetBSD's snapshot of 20091228. This version was chosen because it still doesn't include wide character support (UTF-8), which involves many changes and new files. From NetBSD's logs: Dec 15 22:13:33 2006 - editline.3 el.c el.h histedit.h add EL_GETFP, and EL_SETFP. Apr 5 15:53:28 2008 - editline.3 el.c histedit.h readline.c add EL_REFRESH for the benefit of readline Sep 10 15:45:37 2008 - common.c el.c read.c refresh.c sig.c term.c term.h tty.c Allow a single process to control multiple ttys (for pthreads using _REENTRANT) using multiple EditLine objects. Jan 18 12:17:24 2009 - el.c read.c readline.c fix -Wsign-compare issues Feb 6 14:40:32 2009 - history.c Plug memory leak, from MySQL. Feb 5 19:15:44 2009 - histedit.h read.c match documentation in el_push Feb 6 13:14:37 2009 - vi.c Portability fix. Feb 12 13:39:49 2009 - readline.c term.c More fixes for existing portability stuff. Feb 15 21:24:13 2009 - el.h read.c don't restart on EINTR, instead return NULL immediately. From Anon Ymous Feb 15 21:25:01 2009 - sig.c sig.h in order for read() to return EINTR we need to use sigaction, not signal, otherwise SA_RESTART is set. Feb 15 21:55:23 2009 - chared.c chared.h common.c emacs.c filecomplete.c filecomplete.h key.c key.h read.c readline.c refresh.c search.c term.c tokenizer.c tty.c vi.c pass lint on _LP64. Feb 17 21:34:26 2009 - el.c histedit.h prompt.c prompt.h allow for a prompt argument. Feb 18 15:04:40 2009 - sig.c SA_RESTART for all signals but SIGINT. From Anon Ymous. Feb 19 15:20:22 2009 - read.c sig.c sig.h reset and redraw on sigcont. From Anon Ymous. Feb 21 23:31:56 2009 - key.c key.h readline.c vi.c more size_t stuff. Mar 10 20:46:15 2009 - editline.3 read.c make el_gets set the count to -1 on error to distinguish between EOF and error. Mar 31 17:38:27 2009 - editline.3 el.c histedit.h prompt.c prompt.h refresh.c term.c term.h Implement literal prompt sequences. Now someone can implement RL_PROMPT_START_LITERAL/RL_PROMPT_END_LITERAL :-) Mar 31 21:33:17 2009 - term.c cast to size_t to avoid sign / unsigned comparison warning. Apr 23 02:03 2009 - term.c Apply patch (requested by msaitoh in ticket #2007): Coverity CID 1668: Plug memory leak when malloc() failed.:55 2009 May 11 18:33:30 2009 - editline.3 el.c histedit.h restore binary compatibility by providing new prompt functions that take an extra literal character. May 19 21:45:14 2009 - refresh.c always scroll when we advance past bottom. From Caleb Welton cwelton at greenplum dot com. Jul 17 12:27:57 2009 - term.c - off by one in the term.h case. - make code more similar to tcsh (if we want to handle wide chars, this is needed; for now it is a no-op) Jul 22 15:56:29 2009 - el.c Move filename to the scope it is being used. From Michael Cook mcook at bbn dot com Jul 22 15:57:00 2009 - read.c Always initialize nread since it is an out param. From Michael Cook mcook at bbn dot com Jul 22 18:25:26 2009 - el.c Only need path if we have issetugid... From Anon Ymous Jul 25 21:19:23 2009 - el.c Ignore comment lines in .editrc from Jess Thrysoee Sep 7 21:24:33 2009 histedit.h history.c readline.c apply apple patches from: http://opensource.apple.com/source/libedit/libedit-11/patches/ Dec 28 21:52:43 2009 - refresh.c Fix bug where tab completion on the second or > line that caused listing ended up corrupting the display by an extra space in the beginning. Reported by Mac Chan. Dec 28 22:15:36 2009 - refresh.c term.c reduce diff with tcsh Obtained from: NetBSD Tested by: bapt, jilles and current@ MFC after: 1 week
707 lines
16 KiB
C
707 lines
16 KiB
C
/*-
|
|
* Copyright (c) 1992, 1993
|
|
* The Regents of the University of California. All rights reserved.
|
|
*
|
|
* This code is derived from software contributed to Berkeley by
|
|
* Christos Zoulas of Cornell University.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* 3. Neither the name of the University nor the names of its contributors
|
|
* may be used to endorse or promote products derived from this software
|
|
* without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
* SUCH DAMAGE.
|
|
*
|
|
* $NetBSD: key.c,v 1.20 2009/02/15 21:55:23 christos Exp $
|
|
*/
|
|
|
|
#if !defined(lint) && !defined(SCCSID)
|
|
static char sccsid[] = "@(#)key.c 8.1 (Berkeley) 6/4/93";
|
|
#endif /* not lint && not SCCSID */
|
|
#include <sys/cdefs.h>
|
|
__FBSDID("$FreeBSD$");
|
|
|
|
/*
|
|
* key.c: This module contains the procedures for maintaining
|
|
* the extended-key map.
|
|
*
|
|
* An extended-key (key) is a sequence of keystrokes introduced
|
|
* with a sequence introducer and consisting of an arbitrary
|
|
* number of characters. This module maintains a map (the el->el_key.map)
|
|
* to convert these extended-key sequences into input strs
|
|
* (XK_STR), editor functions (XK_CMD), or unix commands (XK_EXE).
|
|
*
|
|
* Warning:
|
|
* If key is a substr of some other keys, then the longer
|
|
* keys are lost!! That is, if the keys "abcd" and "abcef"
|
|
* are in el->el_key.map, adding the key "abc" will cause the first two
|
|
* definitions to be lost.
|
|
*
|
|
* Restrictions:
|
|
* -------------
|
|
* 1) It is not possible to have one key that is a
|
|
* substr of another.
|
|
*/
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
|
|
#include "el.h"
|
|
|
|
/*
|
|
* The Nodes of the el->el_key.map. The el->el_key.map is a linked list
|
|
* of these node elements
|
|
*/
|
|
struct key_node_t {
|
|
char ch; /* single character of key */
|
|
int type; /* node type */
|
|
key_value_t val; /* command code or pointer to str, */
|
|
/* if this is a leaf */
|
|
struct key_node_t *next; /* ptr to next char of this key */
|
|
struct key_node_t *sibling; /* ptr to another key with same prefix*/
|
|
};
|
|
|
|
private int node_trav(EditLine *, key_node_t *, char *,
|
|
key_value_t *);
|
|
private int node__try(EditLine *, key_node_t *, const char *,
|
|
key_value_t *, int);
|
|
private key_node_t *node__get(int);
|
|
private void node__free(key_node_t *);
|
|
private void node__put(EditLine *, key_node_t *);
|
|
private int node__delete(EditLine *, key_node_t **, const char *);
|
|
private int node_lookup(EditLine *, const char *, key_node_t *,
|
|
size_t);
|
|
private int node_enum(EditLine *, key_node_t *, size_t);
|
|
|
|
#define KEY_BUFSIZ EL_BUFSIZ
|
|
|
|
|
|
/* key_init():
|
|
* Initialize the key maps
|
|
*/
|
|
protected int
|
|
key_init(EditLine *el)
|
|
{
|
|
|
|
el->el_key.buf = (char *) el_malloc(KEY_BUFSIZ);
|
|
if (el->el_key.buf == NULL)
|
|
return (-1);
|
|
el->el_key.map = NULL;
|
|
key_reset(el);
|
|
return (0);
|
|
}
|
|
|
|
/* key_end():
|
|
* Free the key maps
|
|
*/
|
|
protected void
|
|
key_end(EditLine *el)
|
|
{
|
|
|
|
el_free((ptr_t) el->el_key.buf);
|
|
el->el_key.buf = NULL;
|
|
node__free(el->el_key.map);
|
|
}
|
|
|
|
|
|
/* key_map_cmd():
|
|
* Associate cmd with a key value
|
|
*/
|
|
protected key_value_t *
|
|
key_map_cmd(EditLine *el, int cmd)
|
|
{
|
|
|
|
el->el_key.val.cmd = (el_action_t) cmd;
|
|
return (&el->el_key.val);
|
|
}
|
|
|
|
|
|
/* key_map_str():
|
|
* Associate str with a key value
|
|
*/
|
|
protected key_value_t *
|
|
key_map_str(EditLine *el, char *str)
|
|
{
|
|
|
|
el->el_key.val.str = str;
|
|
return (&el->el_key.val);
|
|
}
|
|
|
|
|
|
/* key_reset():
|
|
* Takes all nodes on el->el_key.map and puts them on free list. Then
|
|
* initializes el->el_key.map with arrow keys
|
|
* [Always bind the ansi arrow keys?]
|
|
*/
|
|
protected void
|
|
key_reset(EditLine *el)
|
|
{
|
|
|
|
node__put(el, el->el_key.map);
|
|
el->el_key.map = NULL;
|
|
return;
|
|
}
|
|
|
|
|
|
/* key_get():
|
|
* Calls the recursive function with entry point el->el_key.map
|
|
* Looks up *ch in map and then reads characters until a
|
|
* complete match is found or a mismatch occurs. Returns the
|
|
* type of the match found (XK_STR, XK_CMD, or XK_EXE).
|
|
* Returns NULL in val.str and XK_STR for no match.
|
|
* The last character read is returned in *ch.
|
|
*/
|
|
protected int
|
|
key_get(EditLine *el, char *ch, key_value_t *val)
|
|
{
|
|
|
|
return (node_trav(el, el->el_key.map, ch, val));
|
|
}
|
|
|
|
|
|
/* key_add():
|
|
* Adds key to the el->el_key.map and associates the value in val with it.
|
|
* If key is already is in el->el_key.map, the new code is applied to the
|
|
* existing key. Ntype specifies if code is a command, an
|
|
* out str or a unix command.
|
|
*/
|
|
protected void
|
|
key_add(EditLine *el, const char *key, key_value_t *val, int ntype)
|
|
{
|
|
|
|
if (key[0] == '\0') {
|
|
(void) fprintf(el->el_errfile,
|
|
"key_add: Null extended-key not allowed.\n");
|
|
return;
|
|
}
|
|
if (ntype == XK_CMD && val->cmd == ED_SEQUENCE_LEAD_IN) {
|
|
(void) fprintf(el->el_errfile,
|
|
"key_add: sequence-lead-in command not allowed\n");
|
|
return;
|
|
}
|
|
if (el->el_key.map == NULL)
|
|
/* tree is initially empty. Set up new node to match key[0] */
|
|
el->el_key.map = node__get(key[0]);
|
|
/* it is properly initialized */
|
|
|
|
/* Now recurse through el->el_key.map */
|
|
(void) node__try(el, el->el_key.map, key, val, ntype);
|
|
return;
|
|
}
|
|
|
|
|
|
/* key_clear():
|
|
*
|
|
*/
|
|
protected void
|
|
key_clear(EditLine *el, el_action_t *map, const char *in)
|
|
{
|
|
|
|
if ((map[(unsigned char)*in] == ED_SEQUENCE_LEAD_IN) &&
|
|
((map == el->el_map.key &&
|
|
el->el_map.alt[(unsigned char)*in] != ED_SEQUENCE_LEAD_IN) ||
|
|
(map == el->el_map.alt &&
|
|
el->el_map.key[(unsigned char)*in] != ED_SEQUENCE_LEAD_IN)))
|
|
(void) key_delete(el, in);
|
|
}
|
|
|
|
|
|
/* key_delete():
|
|
* Delete the key and all longer keys staring with key, if
|
|
* they exists.
|
|
*/
|
|
protected int
|
|
key_delete(EditLine *el, const char *key)
|
|
{
|
|
|
|
if (key[0] == '\0') {
|
|
(void) fprintf(el->el_errfile,
|
|
"key_delete: Null extended-key not allowed.\n");
|
|
return (-1);
|
|
}
|
|
if (el->el_key.map == NULL)
|
|
return (0);
|
|
|
|
(void) node__delete(el, &el->el_key.map, key);
|
|
return (0);
|
|
}
|
|
|
|
|
|
/* key_print():
|
|
* Print the binding associated with key key.
|
|
* Print entire el->el_key.map if null
|
|
*/
|
|
protected void
|
|
key_print(EditLine *el, const char *key)
|
|
{
|
|
|
|
/* do nothing if el->el_key.map is empty and null key specified */
|
|
if (el->el_key.map == NULL && *key == 0)
|
|
return;
|
|
|
|
el->el_key.buf[0] = '"';
|
|
if (node_lookup(el, key, el->el_key.map, 1) <= -1)
|
|
/* key is not bound */
|
|
(void) fprintf(el->el_errfile, "Unbound extended key \"%s\"\n",
|
|
key);
|
|
return;
|
|
}
|
|
|
|
|
|
/* node_trav():
|
|
* recursively traverses node in tree until match or mismatch is
|
|
* found. May read in more characters.
|
|
*/
|
|
private int
|
|
node_trav(EditLine *el, key_node_t *ptr, char *ch, key_value_t *val)
|
|
{
|
|
|
|
if (ptr->ch == *ch) {
|
|
/* match found */
|
|
if (ptr->next) {
|
|
/* key not complete so get next char */
|
|
if (el_getc(el, ch) != 1) { /* if EOF or error */
|
|
val->cmd = ED_END_OF_FILE;
|
|
return (XK_CMD);
|
|
/* PWP: Pretend we just read an end-of-file */
|
|
}
|
|
return (node_trav(el, ptr->next, ch, val));
|
|
} else {
|
|
*val = ptr->val;
|
|
if (ptr->type != XK_CMD)
|
|
*ch = '\0';
|
|
return (ptr->type);
|
|
}
|
|
} else {
|
|
/* no match found here */
|
|
if (ptr->sibling) {
|
|
/* try next sibling */
|
|
return (node_trav(el, ptr->sibling, ch, val));
|
|
} else {
|
|
/* no next sibling -- mismatch */
|
|
val->str = NULL;
|
|
return (XK_STR);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/* node__try():
|
|
* Find a node that matches *str or allocate a new one
|
|
*/
|
|
private int
|
|
node__try(EditLine *el, key_node_t *ptr, const char *str, key_value_t *val, int ntype)
|
|
{
|
|
|
|
if (ptr->ch != *str) {
|
|
key_node_t *xm;
|
|
|
|
for (xm = ptr; xm->sibling != NULL; xm = xm->sibling)
|
|
if (xm->sibling->ch == *str)
|
|
break;
|
|
if (xm->sibling == NULL)
|
|
xm->sibling = node__get(*str); /* setup new node */
|
|
ptr = xm->sibling;
|
|
}
|
|
if (*++str == '\0') {
|
|
/* we're there */
|
|
if (ptr->next != NULL) {
|
|
node__put(el, ptr->next);
|
|
/* lose longer keys with this prefix */
|
|
ptr->next = NULL;
|
|
}
|
|
switch (ptr->type) {
|
|
case XK_CMD:
|
|
case XK_NOD:
|
|
break;
|
|
case XK_STR:
|
|
case XK_EXE:
|
|
if (ptr->val.str)
|
|
el_free((ptr_t) ptr->val.str);
|
|
break;
|
|
default:
|
|
EL_ABORT((el->el_errfile, "Bad XK_ type %d\n",
|
|
ptr->type));
|
|
break;
|
|
}
|
|
|
|
switch (ptr->type = ntype) {
|
|
case XK_CMD:
|
|
ptr->val = *val;
|
|
break;
|
|
case XK_STR:
|
|
case XK_EXE:
|
|
if ((ptr->val.str = el_strdup(val->str)) == NULL)
|
|
return -1;
|
|
break;
|
|
default:
|
|
EL_ABORT((el->el_errfile, "Bad XK_ type %d\n", ntype));
|
|
break;
|
|
}
|
|
} else {
|
|
/* still more chars to go */
|
|
if (ptr->next == NULL)
|
|
ptr->next = node__get(*str); /* setup new node */
|
|
(void) node__try(el, ptr->next, str, val, ntype);
|
|
}
|
|
return (0);
|
|
}
|
|
|
|
|
|
/* node__delete():
|
|
* Delete node that matches str
|
|
*/
|
|
private int
|
|
node__delete(EditLine *el, key_node_t **inptr, const char *str)
|
|
{
|
|
key_node_t *ptr;
|
|
key_node_t *prev_ptr = NULL;
|
|
|
|
ptr = *inptr;
|
|
|
|
if (ptr->ch != *str) {
|
|
key_node_t *xm;
|
|
|
|
for (xm = ptr; xm->sibling != NULL; xm = xm->sibling)
|
|
if (xm->sibling->ch == *str)
|
|
break;
|
|
if (xm->sibling == NULL)
|
|
return (0);
|
|
prev_ptr = xm;
|
|
ptr = xm->sibling;
|
|
}
|
|
if (*++str == '\0') {
|
|
/* we're there */
|
|
if (prev_ptr == NULL)
|
|
*inptr = ptr->sibling;
|
|
else
|
|
prev_ptr->sibling = ptr->sibling;
|
|
ptr->sibling = NULL;
|
|
node__put(el, ptr);
|
|
return (1);
|
|
} else if (ptr->next != NULL &&
|
|
node__delete(el, &ptr->next, str) == 1) {
|
|
if (ptr->next != NULL)
|
|
return (0);
|
|
if (prev_ptr == NULL)
|
|
*inptr = ptr->sibling;
|
|
else
|
|
prev_ptr->sibling = ptr->sibling;
|
|
ptr->sibling = NULL;
|
|
node__put(el, ptr);
|
|
return (1);
|
|
} else {
|
|
return (0);
|
|
}
|
|
}
|
|
|
|
|
|
/* node__put():
|
|
* Puts a tree of nodes onto free list using free(3).
|
|
*/
|
|
private void
|
|
node__put(EditLine *el, key_node_t *ptr)
|
|
{
|
|
if (ptr == NULL)
|
|
return;
|
|
|
|
if (ptr->next != NULL) {
|
|
node__put(el, ptr->next);
|
|
ptr->next = NULL;
|
|
}
|
|
node__put(el, ptr->sibling);
|
|
|
|
switch (ptr->type) {
|
|
case XK_CMD:
|
|
case XK_NOD:
|
|
break;
|
|
case XK_EXE:
|
|
case XK_STR:
|
|
if (ptr->val.str != NULL)
|
|
el_free((ptr_t) ptr->val.str);
|
|
break;
|
|
default:
|
|
EL_ABORT((el->el_errfile, "Bad XK_ type %d\n", ptr->type));
|
|
break;
|
|
}
|
|
el_free((ptr_t) ptr);
|
|
}
|
|
|
|
|
|
/* node__get():
|
|
* Returns pointer to a key_node_t for ch.
|
|
*/
|
|
private key_node_t *
|
|
node__get(int ch)
|
|
{
|
|
key_node_t *ptr;
|
|
|
|
ptr = (key_node_t *) el_malloc((size_t) sizeof(key_node_t));
|
|
if (ptr == NULL)
|
|
return NULL;
|
|
ptr->ch = ch;
|
|
ptr->type = XK_NOD;
|
|
ptr->val.str = NULL;
|
|
ptr->next = NULL;
|
|
ptr->sibling = NULL;
|
|
return (ptr);
|
|
}
|
|
|
|
private void
|
|
node__free(key_node_t *k)
|
|
{
|
|
if (k == NULL)
|
|
return;
|
|
node__free(k->sibling);
|
|
node__free(k->next);
|
|
el_free((ptr_t) k);
|
|
}
|
|
|
|
/* node_lookup():
|
|
* look for the str starting at node ptr.
|
|
* Print if last node
|
|
*/
|
|
private int
|
|
node_lookup(EditLine *el, const char *str, key_node_t *ptr, size_t cnt)
|
|
{
|
|
size_t ncnt;
|
|
|
|
if (ptr == NULL)
|
|
return (-1); /* cannot have null ptr */
|
|
|
|
if (*str == 0) {
|
|
/* no more chars in str. node_enum from here. */
|
|
(void) node_enum(el, ptr, cnt);
|
|
return (0);
|
|
} else {
|
|
/* If match put this char into el->el_key.buf. Recurse */
|
|
if (ptr->ch == *str) {
|
|
/* match found */
|
|
ncnt = key__decode_char(el->el_key.buf,
|
|
(size_t)KEY_BUFSIZ, cnt,
|
|
(unsigned char) ptr->ch);
|
|
if (ptr->next != NULL)
|
|
/* not yet at leaf */
|
|
return (node_lookup(el, str + 1, ptr->next,
|
|
ncnt + 1));
|
|
else {
|
|
/* next node is null so key should be complete */
|
|
if (str[1] == 0) {
|
|
el->el_key.buf[ncnt + 1] = '"';
|
|
el->el_key.buf[ncnt + 2] = '\0';
|
|
key_kprint(el, el->el_key.buf,
|
|
&ptr->val, ptr->type);
|
|
return (0);
|
|
} else
|
|
return (-1);
|
|
/* mismatch -- str still has chars */
|
|
}
|
|
} else {
|
|
/* no match found try sibling */
|
|
if (ptr->sibling)
|
|
return (node_lookup(el, str, ptr->sibling,
|
|
cnt));
|
|
else
|
|
return (-1);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/* node_enum():
|
|
* Traverse the node printing the characters it is bound in buffer
|
|
*/
|
|
private int
|
|
node_enum(EditLine *el, key_node_t *ptr, size_t cnt)
|
|
{
|
|
size_t ncnt;
|
|
|
|
if (cnt >= KEY_BUFSIZ - 5) { /* buffer too small */
|
|
el->el_key.buf[++cnt] = '"';
|
|
el->el_key.buf[++cnt] = '\0';
|
|
(void) fprintf(el->el_errfile,
|
|
"Some extended keys too long for internal print buffer");
|
|
(void) fprintf(el->el_errfile, " \"%s...\"\n", el->el_key.buf);
|
|
return (0);
|
|
}
|
|
if (ptr == NULL) {
|
|
#ifdef DEBUG_EDIT
|
|
(void) fprintf(el->el_errfile,
|
|
"node_enum: BUG!! Null ptr passed\n!");
|
|
#endif
|
|
return (-1);
|
|
}
|
|
/* put this char at end of str */
|
|
ncnt = key__decode_char(el->el_key.buf, (size_t)KEY_BUFSIZ, cnt,
|
|
(unsigned char)ptr->ch);
|
|
if (ptr->next == NULL) {
|
|
/* print this key and function */
|
|
el->el_key.buf[ncnt + 1] = '"';
|
|
el->el_key.buf[ncnt + 2] = '\0';
|
|
key_kprint(el, el->el_key.buf, &ptr->val, ptr->type);
|
|
} else
|
|
(void) node_enum(el, ptr->next, ncnt + 1);
|
|
|
|
/* go to sibling if there is one */
|
|
if (ptr->sibling)
|
|
(void) node_enum(el, ptr->sibling, cnt);
|
|
return (0);
|
|
}
|
|
|
|
|
|
/* key_kprint():
|
|
* Print the specified key and its associated
|
|
* function specified by val
|
|
*/
|
|
protected void
|
|
key_kprint(EditLine *el, const char *key, key_value_t *val, int ntype)
|
|
{
|
|
el_bindings_t *fp;
|
|
char unparsbuf[EL_BUFSIZ];
|
|
static const char fmt[] = "%-15s-> %s\n";
|
|
|
|
if (val != NULL)
|
|
switch (ntype) {
|
|
case XK_STR:
|
|
case XK_EXE:
|
|
(void) key__decode_str(val->str, unparsbuf,
|
|
sizeof(unparsbuf),
|
|
ntype == XK_STR ? "\"\"" : "[]");
|
|
(void) fprintf(el->el_outfile, fmt, key, unparsbuf);
|
|
break;
|
|
case XK_CMD:
|
|
for (fp = el->el_map.help; fp->name; fp++)
|
|
if (val->cmd == fp->func) {
|
|
(void) fprintf(el->el_outfile, fmt,
|
|
key, fp->name);
|
|
break;
|
|
}
|
|
#ifdef DEBUG_KEY
|
|
if (fp->name == NULL)
|
|
(void) fprintf(el->el_outfile,
|
|
"BUG! Command not found.\n");
|
|
#endif
|
|
|
|
break;
|
|
default:
|
|
EL_ABORT((el->el_errfile, "Bad XK_ type %d\n", ntype));
|
|
break;
|
|
}
|
|
else
|
|
(void) fprintf(el->el_outfile, fmt, key, "no input");
|
|
}
|
|
|
|
|
|
#define ADDC(c) \
|
|
if (b < eb) \
|
|
*b++ = c; \
|
|
else \
|
|
b++
|
|
/* key__decode_char():
|
|
* Put a printable form of char in buf.
|
|
*/
|
|
protected size_t
|
|
key__decode_char(char *buf, size_t cnt, size_t off, int ch)
|
|
{
|
|
char *sb = buf + off;
|
|
char *eb = buf + cnt;
|
|
char *b = sb;
|
|
|
|
ch = (unsigned char)ch;
|
|
if (ch == 0) {
|
|
ADDC('^');
|
|
ADDC('@');
|
|
return (int)(b - sb);
|
|
}
|
|
if (iscntrl(ch)) {
|
|
ADDC('^');
|
|
if (ch == '\177')
|
|
ADDC('?');
|
|
else
|
|
ADDC(toascii(ch) | 0100);
|
|
} else if (ch == '^') {
|
|
ADDC('\\');
|
|
ADDC('^');
|
|
} else if (ch == '\\') {
|
|
ADDC('\\');
|
|
ADDC('\\');
|
|
} else if (ch == ' ' || (isprint(ch) && !isspace(ch))) {
|
|
ADDC(ch);
|
|
} else {
|
|
ADDC('\\');
|
|
ADDC((((unsigned int) ch >> 6) & 7) + '0');
|
|
ADDC((((unsigned int) ch >> 3) & 7) + '0');
|
|
ADDC((ch & 7) + '0');
|
|
}
|
|
return (size_t)(b - sb);
|
|
}
|
|
|
|
|
|
/* key__decode_str():
|
|
* Make a printable version of the ey
|
|
*/
|
|
protected size_t
|
|
key__decode_str(const char *str, char *buf, size_t len, const char *sep)
|
|
{
|
|
char *b = buf, *eb = b + len;
|
|
const char *p;
|
|
|
|
b = buf;
|
|
if (sep[0] != '\0') {
|
|
ADDC(sep[0]);
|
|
}
|
|
if (*str == '\0') {
|
|
ADDC('^');
|
|
ADDC('@');
|
|
if (sep[0] != '\0' && sep[1] != '\0') {
|
|
ADDC(sep[1]);
|
|
}
|
|
goto done;
|
|
}
|
|
for (p = str; *p != 0; p++) {
|
|
if (iscntrl((unsigned char) *p)) {
|
|
ADDC('^');
|
|
if (*p == '\177') {
|
|
ADDC('?');
|
|
} else {
|
|
ADDC(toascii(*p) | 0100);
|
|
}
|
|
} else if (*p == '^' || *p == '\\') {
|
|
ADDC('\\');
|
|
ADDC(*p);
|
|
} else if (*p == ' ' || (isprint((unsigned char) *p) &&
|
|
!isspace((unsigned char) *p))) {
|
|
ADDC(*p);
|
|
} else {
|
|
ADDC('\\');
|
|
ADDC((((unsigned int) *p >> 6) & 7) + '0');
|
|
ADDC((((unsigned int) *p >> 3) & 7) + '0');
|
|
ADDC((*p & 7) + '0');
|
|
}
|
|
}
|
|
if (sep[0] != '\0' && sep[1] != '\0') {
|
|
ADDC(sep[1]);
|
|
}
|
|
done:
|
|
ADDC('\0');
|
|
if ((size_t)(b - buf) >= len)
|
|
buf[len - 1] = '\0';
|
|
return (size_t)(b - buf);
|
|
}
|