2012-09-04 13:54:00 +01:00
|
|
|
/*-
|
|
|
|
* BSD LICENSE
|
2014-06-04 00:42:50 +01:00
|
|
|
*
|
2014-02-10 11:46:50 +00:00
|
|
|
* Copyright(c) 2010-2014 Intel Corporation. All rights reserved.
|
2012-09-04 13:54:00 +01:00
|
|
|
* All rights reserved.
|
2014-06-04 00:42:50 +01:00
|
|
|
*
|
2013-09-18 12:00:00 +02:00
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
2012-09-04 13:54:00 +01:00
|
|
|
* are met:
|
2014-06-04 00:42:50 +01:00
|
|
|
*
|
2013-09-18 12:00:00 +02:00
|
|
|
* * Redistributions of source code must retain the above copyright
|
2012-09-04 13:54:00 +01:00
|
|
|
* notice, this list of conditions and the following disclaimer.
|
2013-09-18 12:00:00 +02:00
|
|
|
* * 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
|
2012-09-04 13:54:00 +01:00
|
|
|
* distribution.
|
2013-09-18 12:00:00 +02:00
|
|
|
* * Neither the name of Intel Corporation nor the names of its
|
|
|
|
* contributors may be used to endorse or promote products derived
|
2012-09-04 13:54:00 +01:00
|
|
|
* from this software without specific prior written permission.
|
2014-06-04 00:42:50 +01:00
|
|
|
*
|
2013-09-18 12:00:00 +02:00
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS 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 COPYRIGHT
|
|
|
|
* OWNER 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
|
2012-09-04 13:54:00 +01:00
|
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (c) 2009, Olivier MATZ <zer0@droids-corp.org>
|
|
|
|
* All rights reserved.
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions are met:
|
|
|
|
*
|
|
|
|
* * Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* * 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.
|
|
|
|
* * Neither the name of the University of California, Berkeley 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 AND 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdarg.h>
|
2012-12-20 00:00:00 +01:00
|
|
|
#include <errno.h>
|
2012-09-04 13:54:00 +01:00
|
|
|
#include <ctype.h>
|
|
|
|
|
|
|
|
#include "cmdline_cirbuf.h"
|
|
|
|
#include "cmdline_rdline.h"
|
|
|
|
|
|
|
|
static void rdline_puts(struct rdline *rdl, const char *buf);
|
|
|
|
static void rdline_miniprintf(struct rdline *rdl,
|
|
|
|
const char *buf, unsigned int val);
|
|
|
|
|
|
|
|
static void rdline_remove_old_history_item(struct rdline *rdl);
|
|
|
|
static void rdline_remove_first_history_item(struct rdline *rdl);
|
|
|
|
static unsigned int rdline_get_history_size(struct rdline *rdl);
|
|
|
|
|
|
|
|
|
|
|
|
/* isblank() needs _XOPEN_SOURCE >= 600 || _ISOC99_SOURCE, so use our
|
|
|
|
* own. */
|
|
|
|
static int
|
|
|
|
isblank2(char c)
|
|
|
|
{
|
|
|
|
if (c == ' ' ||
|
|
|
|
c == '\t' )
|
|
|
|
return 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-12-20 00:00:00 +01:00
|
|
|
int
|
2012-09-04 13:54:00 +01:00
|
|
|
rdline_init(struct rdline *rdl,
|
|
|
|
rdline_write_char_t *write_char,
|
|
|
|
rdline_validate_t *validate,
|
|
|
|
rdline_complete_t *complete)
|
|
|
|
{
|
2012-12-20 00:00:00 +01:00
|
|
|
if (!rdl || !write_char || !validate || !complete)
|
|
|
|
return -EINVAL;
|
2012-09-04 13:54:00 +01:00
|
|
|
memset(rdl, 0, sizeof(*rdl));
|
|
|
|
rdl->validate = validate;
|
|
|
|
rdl->complete = complete;
|
|
|
|
rdl->write_char = write_char;
|
|
|
|
rdl->status = RDLINE_INIT;
|
2012-12-20 00:00:00 +01:00
|
|
|
return cirbuf_init(&rdl->history, rdl->history_buf, 0, RDLINE_HISTORY_BUF_SIZE);
|
2012-09-04 13:54:00 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
rdline_newline(struct rdline *rdl, const char *prompt)
|
|
|
|
{
|
|
|
|
unsigned int i;
|
|
|
|
|
2012-12-20 00:00:00 +01:00
|
|
|
if (!rdl || !prompt)
|
|
|
|
return;
|
|
|
|
|
2012-09-04 13:54:00 +01:00
|
|
|
vt100_init(&rdl->vt100);
|
|
|
|
cirbuf_init(&rdl->left, rdl->left_buf, 0, RDLINE_BUF_SIZE);
|
|
|
|
cirbuf_init(&rdl->right, rdl->right_buf, 0, RDLINE_BUF_SIZE);
|
|
|
|
|
2013-03-12 12:03:00 +01:00
|
|
|
rdl->prompt_size = strnlen(prompt, RDLINE_PROMPT_SIZE-1);
|
2012-09-04 13:54:00 +01:00
|
|
|
if (prompt != rdl->prompt)
|
2013-03-12 12:03:00 +01:00
|
|
|
memcpy(rdl->prompt, prompt, rdl->prompt_size);
|
|
|
|
rdl->prompt[RDLINE_PROMPT_SIZE-1] = '\0';
|
2012-09-04 13:54:00 +01:00
|
|
|
|
|
|
|
for (i=0 ; i<rdl->prompt_size ; i++)
|
|
|
|
rdl->write_char(rdl, rdl->prompt[i]);
|
|
|
|
rdl->status = RDLINE_RUNNING;
|
|
|
|
|
|
|
|
rdl->history_cur_line = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
rdline_stop(struct rdline *rdl)
|
|
|
|
{
|
2012-12-20 00:00:00 +01:00
|
|
|
if (!rdl)
|
|
|
|
return;
|
2012-09-04 13:54:00 +01:00
|
|
|
rdl->status = RDLINE_INIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
rdline_quit(struct rdline *rdl)
|
|
|
|
{
|
2012-12-20 00:00:00 +01:00
|
|
|
if (!rdl)
|
|
|
|
return;
|
2012-09-04 13:54:00 +01:00
|
|
|
rdl->status = RDLINE_EXITED;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
rdline_restart(struct rdline *rdl)
|
|
|
|
{
|
2012-12-20 00:00:00 +01:00
|
|
|
if (!rdl)
|
|
|
|
return;
|
2012-09-04 13:54:00 +01:00
|
|
|
rdl->status = RDLINE_RUNNING;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
rdline_reset(struct rdline *rdl)
|
|
|
|
{
|
2012-12-20 00:00:00 +01:00
|
|
|
if (!rdl)
|
|
|
|
return;
|
2012-09-04 13:54:00 +01:00
|
|
|
vt100_init(&rdl->vt100);
|
|
|
|
cirbuf_init(&rdl->left, rdl->left_buf, 0, RDLINE_BUF_SIZE);
|
|
|
|
cirbuf_init(&rdl->right, rdl->right_buf, 0, RDLINE_BUF_SIZE);
|
|
|
|
|
|
|
|
rdl->status = RDLINE_RUNNING;
|
|
|
|
|
|
|
|
rdl->history_cur_line = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *
|
|
|
|
rdline_get_buffer(struct rdline *rdl)
|
|
|
|
{
|
2012-12-20 00:00:00 +01:00
|
|
|
if (!rdl)
|
|
|
|
return NULL;
|
2012-09-04 13:54:00 +01:00
|
|
|
unsigned int len_l, len_r;
|
|
|
|
cirbuf_align_left(&rdl->left);
|
|
|
|
cirbuf_align_left(&rdl->right);
|
|
|
|
|
|
|
|
len_l = CIRBUF_GET_LEN(&rdl->left);
|
|
|
|
len_r = CIRBUF_GET_LEN(&rdl->right);
|
|
|
|
memcpy(rdl->left_buf+len_l, rdl->right_buf, len_r);
|
|
|
|
|
|
|
|
rdl->left_buf[len_l + len_r] = '\n';
|
|
|
|
rdl->left_buf[len_l + len_r + 1] = '\0';
|
|
|
|
return rdl->left_buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
display_right_buffer(struct rdline *rdl, int force)
|
|
|
|
{
|
|
|
|
unsigned int i;
|
|
|
|
char tmp;
|
|
|
|
|
|
|
|
if (!force && CIRBUF_IS_EMPTY(&rdl->right))
|
|
|
|
return;
|
|
|
|
|
|
|
|
rdline_puts(rdl, vt100_clear_right);
|
|
|
|
CIRBUF_FOREACH(&rdl->right, i, tmp) {
|
|
|
|
rdl->write_char(rdl, tmp);
|
|
|
|
}
|
|
|
|
if (!CIRBUF_IS_EMPTY(&rdl->right))
|
|
|
|
rdline_miniprintf(rdl, vt100_multi_left,
|
|
|
|
CIRBUF_GET_LEN(&rdl->right));
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
rdline_redisplay(struct rdline *rdl)
|
|
|
|
{
|
|
|
|
unsigned int i;
|
|
|
|
char tmp;
|
|
|
|
|
2012-12-20 00:00:00 +01:00
|
|
|
if (!rdl)
|
|
|
|
return;
|
|
|
|
|
2012-09-04 13:54:00 +01:00
|
|
|
rdline_puts(rdl, vt100_home);
|
|
|
|
for (i=0 ; i<rdl->prompt_size ; i++)
|
|
|
|
rdl->write_char(rdl, rdl->prompt[i]);
|
|
|
|
CIRBUF_FOREACH(&rdl->left, i, tmp) {
|
|
|
|
rdl->write_char(rdl, tmp);
|
|
|
|
}
|
|
|
|
display_right_buffer(rdl, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
rdline_char_in(struct rdline *rdl, char c)
|
|
|
|
{
|
|
|
|
unsigned int i;
|
|
|
|
int cmd;
|
|
|
|
char tmp;
|
|
|
|
char *buf;
|
2012-12-20 00:00:00 +01:00
|
|
|
|
|
|
|
if (!rdl)
|
|
|
|
return -EINVAL;
|
2012-09-04 13:54:00 +01:00
|
|
|
|
|
|
|
if (rdl->status == RDLINE_EXITED)
|
|
|
|
return RDLINE_RES_EXITED;
|
|
|
|
if (rdl->status != RDLINE_RUNNING)
|
|
|
|
return RDLINE_RES_NOT_RUNNING;
|
|
|
|
|
|
|
|
cmd = vt100_parser(&rdl->vt100, c);
|
|
|
|
if (cmd == -2)
|
|
|
|
return RDLINE_RES_SUCCESS;
|
|
|
|
|
|
|
|
if (cmd >= 0) {
|
|
|
|
switch (cmd) {
|
2012-12-20 00:00:00 +01:00
|
|
|
/* move caret 1 char to the left */
|
2012-09-04 13:54:00 +01:00
|
|
|
case CMDLINE_KEY_CTRL_B:
|
|
|
|
case CMDLINE_KEY_LEFT_ARR:
|
|
|
|
if (CIRBUF_IS_EMPTY(&rdl->left))
|
|
|
|
break;
|
|
|
|
tmp = cirbuf_get_tail(&rdl->left);
|
|
|
|
cirbuf_del_tail(&rdl->left);
|
|
|
|
cirbuf_add_head(&rdl->right, tmp);
|
|
|
|
rdline_puts(rdl, vt100_left_arr);
|
|
|
|
break;
|
|
|
|
|
2012-12-20 00:00:00 +01:00
|
|
|
/* move caret 1 char to the right */
|
2012-09-04 13:54:00 +01:00
|
|
|
case CMDLINE_KEY_CTRL_F:
|
|
|
|
case CMDLINE_KEY_RIGHT_ARR:
|
|
|
|
if (CIRBUF_IS_EMPTY(&rdl->right))
|
|
|
|
break;
|
|
|
|
tmp = cirbuf_get_head(&rdl->right);
|
|
|
|
cirbuf_del_head(&rdl->right);
|
|
|
|
cirbuf_add_tail(&rdl->left, tmp);
|
|
|
|
rdline_puts(rdl, vt100_right_arr);
|
|
|
|
break;
|
|
|
|
|
2012-12-20 00:00:00 +01:00
|
|
|
/* move caret 1 word to the left */
|
|
|
|
/* keyboard equivalent: Alt+B */
|
2012-09-04 13:54:00 +01:00
|
|
|
case CMDLINE_KEY_WLEFT:
|
|
|
|
while (! CIRBUF_IS_EMPTY(&rdl->left) &&
|
|
|
|
(tmp = cirbuf_get_tail(&rdl->left)) &&
|
|
|
|
isblank2(tmp)) {
|
|
|
|
rdline_puts(rdl, vt100_left_arr);
|
|
|
|
cirbuf_del_tail(&rdl->left);
|
|
|
|
cirbuf_add_head(&rdl->right, tmp);
|
|
|
|
}
|
|
|
|
while (! CIRBUF_IS_EMPTY(&rdl->left) &&
|
|
|
|
(tmp = cirbuf_get_tail(&rdl->left)) &&
|
|
|
|
!isblank2(tmp)) {
|
|
|
|
rdline_puts(rdl, vt100_left_arr);
|
|
|
|
cirbuf_del_tail(&rdl->left);
|
|
|
|
cirbuf_add_head(&rdl->right, tmp);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2012-12-20 00:00:00 +01:00
|
|
|
/* move caret 1 word to the right */
|
|
|
|
/* keyboard equivalent: Alt+F */
|
2012-09-04 13:54:00 +01:00
|
|
|
case CMDLINE_KEY_WRIGHT:
|
|
|
|
while (! CIRBUF_IS_EMPTY(&rdl->right) &&
|
|
|
|
(tmp = cirbuf_get_head(&rdl->right)) &&
|
|
|
|
isblank2(tmp)) {
|
|
|
|
rdline_puts(rdl, vt100_right_arr);
|
|
|
|
cirbuf_del_head(&rdl->right);
|
|
|
|
cirbuf_add_tail(&rdl->left, tmp);
|
|
|
|
}
|
|
|
|
while (! CIRBUF_IS_EMPTY(&rdl->right) &&
|
|
|
|
(tmp = cirbuf_get_head(&rdl->right)) &&
|
|
|
|
!isblank2(tmp)) {
|
|
|
|
rdline_puts(rdl, vt100_right_arr);
|
|
|
|
cirbuf_del_head(&rdl->right);
|
|
|
|
cirbuf_add_tail(&rdl->left, tmp);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2012-12-20 00:00:00 +01:00
|
|
|
/* move caret to the left */
|
|
|
|
case CMDLINE_KEY_CTRL_A:
|
|
|
|
if (CIRBUF_IS_EMPTY(&rdl->left))
|
|
|
|
break;
|
|
|
|
rdline_miniprintf(rdl, vt100_multi_left,
|
|
|
|
CIRBUF_GET_LEN(&rdl->left));
|
|
|
|
while (! CIRBUF_IS_EMPTY(&rdl->left)) {
|
|
|
|
tmp = cirbuf_get_tail(&rdl->left);
|
|
|
|
cirbuf_del_tail(&rdl->left);
|
|
|
|
cirbuf_add_head(&rdl->right, tmp);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* move caret to the right */
|
|
|
|
case CMDLINE_KEY_CTRL_E:
|
|
|
|
if (CIRBUF_IS_EMPTY(&rdl->right))
|
|
|
|
break;
|
|
|
|
rdline_miniprintf(rdl, vt100_multi_right,
|
|
|
|
CIRBUF_GET_LEN(&rdl->right));
|
|
|
|
while (! CIRBUF_IS_EMPTY(&rdl->right)) {
|
|
|
|
tmp = cirbuf_get_head(&rdl->right);
|
|
|
|
cirbuf_del_head(&rdl->right);
|
|
|
|
cirbuf_add_tail(&rdl->left, tmp);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* delete 1 char from the left */
|
2012-09-04 13:54:00 +01:00
|
|
|
case CMDLINE_KEY_BKSPACE:
|
|
|
|
if(!cirbuf_del_tail_safe(&rdl->left)) {
|
|
|
|
rdline_puts(rdl, vt100_bs);
|
|
|
|
display_right_buffer(rdl, 1);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2012-12-20 00:00:00 +01:00
|
|
|
/* delete 1 char from the right */
|
|
|
|
case CMDLINE_KEY_SUPPR:
|
|
|
|
case CMDLINE_KEY_CTRL_D:
|
|
|
|
if (cmd == CMDLINE_KEY_CTRL_D &&
|
|
|
|
CIRBUF_IS_EMPTY(&rdl->left) &&
|
|
|
|
CIRBUF_IS_EMPTY(&rdl->right)) {
|
|
|
|
return RDLINE_RES_EOF;
|
|
|
|
}
|
|
|
|
if (!cirbuf_del_head_safe(&rdl->right)) {
|
|
|
|
display_right_buffer(rdl, 1);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* delete 1 word from the left */
|
2012-09-04 13:54:00 +01:00
|
|
|
case CMDLINE_KEY_META_BKSPACE:
|
|
|
|
case CMDLINE_KEY_CTRL_W:
|
|
|
|
while (! CIRBUF_IS_EMPTY(&rdl->left) && isblank2(cirbuf_get_tail(&rdl->left))) {
|
|
|
|
rdline_puts(rdl, vt100_bs);
|
|
|
|
cirbuf_del_tail(&rdl->left);
|
|
|
|
}
|
|
|
|
while (! CIRBUF_IS_EMPTY(&rdl->left) && !isblank2(cirbuf_get_tail(&rdl->left))) {
|
|
|
|
rdline_puts(rdl, vt100_bs);
|
|
|
|
cirbuf_del_tail(&rdl->left);
|
|
|
|
}
|
|
|
|
display_right_buffer(rdl, 1);
|
|
|
|
break;
|
|
|
|
|
2012-12-20 00:00:00 +01:00
|
|
|
/* delete 1 word from the right */
|
2012-09-04 13:54:00 +01:00
|
|
|
case CMDLINE_KEY_META_D:
|
|
|
|
while (! CIRBUF_IS_EMPTY(&rdl->right) && isblank2(cirbuf_get_head(&rdl->right)))
|
|
|
|
cirbuf_del_head(&rdl->right);
|
|
|
|
while (! CIRBUF_IS_EMPTY(&rdl->right) && !isblank2(cirbuf_get_head(&rdl->right)))
|
|
|
|
cirbuf_del_head(&rdl->right);
|
|
|
|
display_right_buffer(rdl, 1);
|
|
|
|
break;
|
|
|
|
|
2012-12-20 00:00:00 +01:00
|
|
|
/* set kill buffer to contents on the right side of caret */
|
2012-09-04 13:54:00 +01:00
|
|
|
case CMDLINE_KEY_CTRL_K:
|
|
|
|
cirbuf_get_buf_head(&rdl->right, rdl->kill_buf, RDLINE_BUF_SIZE);
|
|
|
|
rdl->kill_size = CIRBUF_GET_LEN(&rdl->right);
|
|
|
|
cirbuf_del_buf_head(&rdl->right, rdl->kill_size);
|
|
|
|
rdline_puts(rdl, vt100_clear_right);
|
|
|
|
break;
|
|
|
|
|
2012-12-20 00:00:00 +01:00
|
|
|
/* paste contents of kill buffer to the left side of caret */
|
2012-09-04 13:54:00 +01:00
|
|
|
case CMDLINE_KEY_CTRL_Y:
|
|
|
|
i=0;
|
|
|
|
while(CIRBUF_GET_LEN(&rdl->right) + CIRBUF_GET_LEN(&rdl->left) <
|
|
|
|
RDLINE_BUF_SIZE &&
|
|
|
|
i < rdl->kill_size) {
|
|
|
|
cirbuf_add_tail(&rdl->left, rdl->kill_buf[i]);
|
|
|
|
rdl->write_char(rdl, rdl->kill_buf[i]);
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
display_right_buffer(rdl, 0);
|
|
|
|
break;
|
|
|
|
|
2012-12-20 00:00:00 +01:00
|
|
|
/* clear and newline */
|
2012-09-04 13:54:00 +01:00
|
|
|
case CMDLINE_KEY_CTRL_C:
|
|
|
|
rdline_puts(rdl, "\r\n");
|
|
|
|
rdline_newline(rdl, rdl->prompt);
|
|
|
|
break;
|
|
|
|
|
2012-12-20 00:00:00 +01:00
|
|
|
/* redisplay (helps when prompt is lost in other output) */
|
2012-09-04 13:54:00 +01:00
|
|
|
case CMDLINE_KEY_CTRL_L:
|
|
|
|
rdline_redisplay(rdl);
|
|
|
|
break;
|
|
|
|
|
2012-12-20 00:00:00 +01:00
|
|
|
/* autocomplete */
|
2012-09-04 13:54:00 +01:00
|
|
|
case CMDLINE_KEY_TAB:
|
|
|
|
case CMDLINE_KEY_HELP:
|
|
|
|
cirbuf_align_left(&rdl->left);
|
|
|
|
rdl->left_buf[CIRBUF_GET_LEN(&rdl->left)] = '\0';
|
|
|
|
if (rdl->complete) {
|
|
|
|
char tmp_buf[BUFSIZ];
|
|
|
|
int complete_state;
|
|
|
|
int ret;
|
|
|
|
unsigned int tmp_size;
|
|
|
|
|
|
|
|
if (cmd == CMDLINE_KEY_TAB)
|
|
|
|
complete_state = 0;
|
|
|
|
else
|
|
|
|
complete_state = -1;
|
|
|
|
|
|
|
|
/* see in parse.h for help on complete() */
|
|
|
|
ret = rdl->complete(rdl, rdl->left_buf,
|
|
|
|
tmp_buf, sizeof(tmp_buf),
|
|
|
|
&complete_state);
|
|
|
|
/* no completion or error */
|
|
|
|
if (ret <= 0) {
|
|
|
|
return RDLINE_RES_COMPLETE;
|
|
|
|
}
|
|
|
|
|
|
|
|
tmp_size = strnlen(tmp_buf, sizeof(tmp_buf));
|
|
|
|
/* add chars */
|
|
|
|
if (ret == RDLINE_RES_COMPLETE) {
|
|
|
|
i=0;
|
|
|
|
while(CIRBUF_GET_LEN(&rdl->right) + CIRBUF_GET_LEN(&rdl->left) <
|
|
|
|
RDLINE_BUF_SIZE &&
|
|
|
|
i < tmp_size) {
|
|
|
|
cirbuf_add_tail(&rdl->left, tmp_buf[i]);
|
|
|
|
rdl->write_char(rdl, tmp_buf[i]);
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
display_right_buffer(rdl, 1);
|
|
|
|
return RDLINE_RES_COMPLETE; /* ?? */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* choice */
|
|
|
|
rdline_puts(rdl, "\r\n");
|
|
|
|
while (ret) {
|
|
|
|
rdl->write_char(rdl, ' ');
|
|
|
|
for (i=0 ; tmp_buf[i] ; i++)
|
|
|
|
rdl->write_char(rdl, tmp_buf[i]);
|
|
|
|
rdline_puts(rdl, "\r\n");
|
|
|
|
ret = rdl->complete(rdl, rdl->left_buf,
|
|
|
|
tmp_buf, sizeof(tmp_buf),
|
|
|
|
&complete_state);
|
|
|
|
}
|
|
|
|
|
|
|
|
rdline_redisplay(rdl);
|
|
|
|
}
|
|
|
|
return RDLINE_RES_COMPLETE;
|
|
|
|
|
2012-12-20 00:00:00 +01:00
|
|
|
/* complete buffer */
|
2012-09-04 13:54:00 +01:00
|
|
|
case CMDLINE_KEY_RETURN:
|
|
|
|
case CMDLINE_KEY_RETURN2:
|
|
|
|
rdline_get_buffer(rdl);
|
|
|
|
rdl->status = RDLINE_INIT;
|
|
|
|
rdline_puts(rdl, "\r\n");
|
|
|
|
if (rdl->history_cur_line != -1)
|
|
|
|
rdline_remove_first_history_item(rdl);
|
|
|
|
|
|
|
|
if (rdl->validate)
|
|
|
|
rdl->validate(rdl, rdl->left_buf, CIRBUF_GET_LEN(&rdl->left)+2);
|
|
|
|
/* user may have stopped rdline */
|
|
|
|
if (rdl->status == RDLINE_EXITED)
|
|
|
|
return RDLINE_RES_EXITED;
|
|
|
|
return RDLINE_RES_VALIDATED;
|
|
|
|
|
2012-12-20 00:00:00 +01:00
|
|
|
/* previous element in history */
|
2012-09-04 13:54:00 +01:00
|
|
|
case CMDLINE_KEY_UP_ARR:
|
|
|
|
case CMDLINE_KEY_CTRL_P:
|
|
|
|
if (rdl->history_cur_line == 0) {
|
|
|
|
rdline_remove_first_history_item(rdl);
|
|
|
|
}
|
|
|
|
if (rdl->history_cur_line <= 0) {
|
|
|
|
rdline_add_history(rdl, rdline_get_buffer(rdl));
|
|
|
|
rdl->history_cur_line = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
buf = rdline_get_history_item(rdl, rdl->history_cur_line + 1);
|
|
|
|
if (!buf)
|
|
|
|
break;
|
|
|
|
|
|
|
|
rdl->history_cur_line ++;
|
|
|
|
vt100_init(&rdl->vt100);
|
|
|
|
cirbuf_init(&rdl->left, rdl->left_buf, 0, RDLINE_BUF_SIZE);
|
|
|
|
cirbuf_init(&rdl->right, rdl->right_buf, 0, RDLINE_BUF_SIZE);
|
|
|
|
cirbuf_add_buf_tail(&rdl->left, buf, strnlen(buf, RDLINE_BUF_SIZE));
|
|
|
|
rdline_redisplay(rdl);
|
|
|
|
break;
|
|
|
|
|
2012-12-20 00:00:00 +01:00
|
|
|
/* next element in history */
|
2012-09-04 13:54:00 +01:00
|
|
|
case CMDLINE_KEY_DOWN_ARR:
|
|
|
|
case CMDLINE_KEY_CTRL_N:
|
|
|
|
if (rdl->history_cur_line - 1 < 0)
|
|
|
|
break;
|
|
|
|
|
|
|
|
rdl->history_cur_line --;
|
|
|
|
buf = rdline_get_history_item(rdl, rdl->history_cur_line);
|
|
|
|
if (!buf)
|
|
|
|
break;
|
|
|
|
vt100_init(&rdl->vt100);
|
|
|
|
cirbuf_init(&rdl->left, rdl->left_buf, 0, RDLINE_BUF_SIZE);
|
|
|
|
cirbuf_init(&rdl->right, rdl->right_buf, 0, RDLINE_BUF_SIZE);
|
|
|
|
cirbuf_add_buf_tail(&rdl->left, buf, strnlen(buf, RDLINE_BUF_SIZE));
|
|
|
|
rdline_redisplay(rdl);
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return RDLINE_RES_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isprint((int)c))
|
|
|
|
return RDLINE_RES_SUCCESS;
|
|
|
|
|
|
|
|
/* standard chars */
|
|
|
|
if (CIRBUF_GET_LEN(&rdl->left) + CIRBUF_GET_LEN(&rdl->right) >= RDLINE_BUF_SIZE)
|
|
|
|
return RDLINE_RES_SUCCESS;
|
|
|
|
|
|
|
|
if (cirbuf_add_tail_safe(&rdl->left, c))
|
|
|
|
return RDLINE_RES_SUCCESS;
|
|
|
|
|
|
|
|
rdl->write_char(rdl, c);
|
|
|
|
display_right_buffer(rdl, 0);
|
|
|
|
|
|
|
|
return RDLINE_RES_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* HISTORY */
|
|
|
|
|
|
|
|
static void
|
|
|
|
rdline_remove_old_history_item(struct rdline * rdl)
|
|
|
|
{
|
|
|
|
char tmp;
|
|
|
|
|
|
|
|
while (! CIRBUF_IS_EMPTY(&rdl->history) ) {
|
|
|
|
tmp = cirbuf_get_head(&rdl->history);
|
|
|
|
cirbuf_del_head(&rdl->history);
|
|
|
|
if (!tmp)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
rdline_remove_first_history_item(struct rdline * rdl)
|
|
|
|
{
|
|
|
|
char tmp;
|
|
|
|
|
|
|
|
if ( CIRBUF_IS_EMPTY(&rdl->history) ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
cirbuf_del_tail(&rdl->history);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (! CIRBUF_IS_EMPTY(&rdl->history) ) {
|
|
|
|
tmp = cirbuf_get_tail(&rdl->history);
|
|
|
|
if (!tmp)
|
|
|
|
break;
|
|
|
|
cirbuf_del_tail(&rdl->history);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned int
|
|
|
|
rdline_get_history_size(struct rdline * rdl)
|
|
|
|
{
|
|
|
|
unsigned int i, tmp, ret=0;
|
|
|
|
|
|
|
|
CIRBUF_FOREACH(&rdl->history, i, tmp) {
|
|
|
|
if (tmp == 0)
|
|
|
|
ret ++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *
|
|
|
|
rdline_get_history_item(struct rdline * rdl, unsigned int idx)
|
|
|
|
{
|
|
|
|
unsigned int len, i, tmp;
|
|
|
|
|
2012-12-20 00:00:00 +01:00
|
|
|
if (!rdl)
|
|
|
|
return NULL;
|
|
|
|
|
2012-09-04 13:54:00 +01:00
|
|
|
len = rdline_get_history_size(rdl);
|
|
|
|
if ( idx >= len ) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
cirbuf_align_left(&rdl->history);
|
|
|
|
|
|
|
|
CIRBUF_FOREACH(&rdl->history, i, tmp) {
|
|
|
|
if ( idx == len - 1) {
|
|
|
|
return rdl->history_buf + i;
|
|
|
|
}
|
|
|
|
if (tmp == 0)
|
|
|
|
len --;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
rdline_add_history(struct rdline * rdl, const char * buf)
|
|
|
|
{
|
|
|
|
unsigned int len, i;
|
|
|
|
|
2012-12-20 00:00:00 +01:00
|
|
|
if (!rdl || !buf)
|
|
|
|
return -EINVAL;
|
|
|
|
|
2012-09-04 13:54:00 +01:00
|
|
|
len = strnlen(buf, RDLINE_BUF_SIZE);
|
|
|
|
for (i=0; i<len ; i++) {
|
|
|
|
if (buf[i] == '\n') {
|
|
|
|
len = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( len >= RDLINE_HISTORY_BUF_SIZE )
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
while ( len >= CIRBUF_GET_FREELEN(&rdl->history) ) {
|
|
|
|
rdline_remove_old_history_item(rdl);
|
|
|
|
}
|
|
|
|
|
|
|
|
cirbuf_add_buf_tail(&rdl->history, buf, len);
|
|
|
|
cirbuf_add_tail(&rdl->history, 0);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
rdline_clear_history(struct rdline * rdl)
|
|
|
|
{
|
2012-12-20 00:00:00 +01:00
|
|
|
if (!rdl)
|
|
|
|
return;
|
2012-09-04 13:54:00 +01:00
|
|
|
cirbuf_init(&rdl->history, rdl->history_buf, 0, RDLINE_HISTORY_BUF_SIZE);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* STATIC USEFUL FUNCS */
|
|
|
|
|
|
|
|
static void
|
|
|
|
rdline_puts(struct rdline * rdl, const char * buf)
|
|
|
|
{
|
|
|
|
char c;
|
|
|
|
while ( (c = *(buf++)) != '\0' ) {
|
|
|
|
rdl->write_char(rdl, c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* a very very basic printf with one arg and one format 'u' */
|
|
|
|
static void
|
|
|
|
rdline_miniprintf(struct rdline *rdl, const char * buf, unsigned int val)
|
|
|
|
{
|
|
|
|
char c, started=0, div=100;
|
|
|
|
|
|
|
|
while ( (c=*(buf++)) ) {
|
|
|
|
if (c != '%') {
|
|
|
|
rdl->write_char(rdl, c);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
c = *(buf++);
|
|
|
|
if (c != 'u') {
|
|
|
|
rdl->write_char(rdl, '%');
|
|
|
|
rdl->write_char(rdl, c);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
/* val is never more than 255 */
|
|
|
|
while (div) {
|
|
|
|
c = (char)(val / div);
|
|
|
|
if (c || started) {
|
|
|
|
rdl->write_char(rdl, (char)(c+'0'));
|
|
|
|
started = 1;
|
|
|
|
}
|
|
|
|
val %= div;
|
|
|
|
div /= 10;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|