391 lines
7.1 KiB
C
391 lines
7.1 KiB
C
/*
|
|
* complete.c - functions for TAB completion
|
|
*
|
|
* Copyright (c) 2008 Sascha Hauer <s.hauer@pengutronix.de>, Pengutronix
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License version 2
|
|
* as published by the Free Software Foundation.
|
|
*
|
|
* This program 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 this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#include <common.h>
|
|
#include <complete.h>
|
|
#include <xfuncs.h>
|
|
#include <fs.h>
|
|
#include <linux/stat.h>
|
|
#include <libgen.h>
|
|
#include <command.h>
|
|
#include <environment.h>
|
|
|
|
static int file_complete(struct string_list *sl, char *instr, int exec)
|
|
{
|
|
char *path = strdup(instr);
|
|
struct stat s;
|
|
DIR *dir;
|
|
struct dirent *d;
|
|
char tmp[PATH_MAX];
|
|
char *base, *dirn;
|
|
|
|
base = basename(instr);
|
|
dirn = dirname(path);
|
|
|
|
dir = opendir(dirn);
|
|
if (!dir)
|
|
goto out;
|
|
|
|
while ((d = readdir(dir))) {
|
|
if (!strcmp(d->d_name, ".") || !strcmp(d->d_name, ".."))
|
|
continue;
|
|
|
|
if (strncmp(base, d->d_name, strlen(base)))
|
|
continue;
|
|
|
|
strcpy(tmp, instr);
|
|
strcat(tmp, d->d_name + strlen(base));
|
|
if (!stat(tmp, &s) && S_ISDIR(s.st_mode)) {
|
|
strcat(tmp, "/");
|
|
} else {
|
|
if (exec && !S_ISREG(s.st_mode))
|
|
continue;
|
|
strcat(tmp, " ");
|
|
}
|
|
|
|
string_list_add_sorted(sl, tmp);
|
|
}
|
|
|
|
closedir(dir);
|
|
|
|
out:
|
|
free(path);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int path_command_complete(struct string_list *sl, char *instr)
|
|
{
|
|
struct stat s;
|
|
DIR *dir;
|
|
struct dirent *d;
|
|
char tmp[PATH_MAX];
|
|
char *path, *p, *n;
|
|
|
|
p = path = strdup(getenv("PATH"));
|
|
|
|
if (!path)
|
|
return -1;
|
|
|
|
while (p) {
|
|
n = strchr(p, ':');
|
|
if (n)
|
|
*n++ = '\0';
|
|
if (*p == '\0') {
|
|
p = n;
|
|
continue;
|
|
}
|
|
dir = opendir(p);
|
|
|
|
/* We need to check all PATH dirs, so if one failed,
|
|
* try next */
|
|
if (!dir) {
|
|
p = n;
|
|
continue;
|
|
}
|
|
|
|
while ((d = readdir(dir))) {
|
|
if (!strcmp(d->d_name, ".") ||
|
|
!strcmp(d->d_name, ".."))
|
|
continue;
|
|
|
|
if (!strncmp(instr, d->d_name, strlen(instr))) {
|
|
strcpy(tmp, d->d_name);
|
|
if (!stat(tmp, &s) &&
|
|
S_ISDIR(s.st_mode))
|
|
continue;
|
|
else
|
|
strcat(tmp, " ");
|
|
|
|
/* This function is called
|
|
* after command_complete,
|
|
* so we check if a double
|
|
* entry exist */
|
|
if (string_list_contains
|
|
(sl, tmp) == 0) {
|
|
string_list_add_sorted(sl, tmp);
|
|
}
|
|
}
|
|
}
|
|
|
|
closedir(dir);
|
|
p = n;
|
|
}
|
|
|
|
free(path);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int command_complete(struct string_list *sl, char *instr)
|
|
{
|
|
struct command *cmdtp;
|
|
|
|
if (!instr)
|
|
instr = "";
|
|
|
|
for_each_command(cmdtp) {
|
|
if (strncmp(instr, cmdtp->name, strlen(instr)))
|
|
continue;
|
|
|
|
string_list_add_asprintf(sl, "%s ", cmdtp->name);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int device_complete(struct string_list *sl, char *instr)
|
|
{
|
|
struct device_d *dev;
|
|
int len;
|
|
|
|
if (!instr)
|
|
instr = "";
|
|
|
|
len = strlen(instr);
|
|
|
|
for_each_device(dev) {
|
|
if (strncmp(instr, dev_name(dev), len))
|
|
continue;
|
|
|
|
string_list_add_asprintf(sl, "%s ", dev_name(dev));
|
|
}
|
|
|
|
return COMPLETE_CONTINUE;
|
|
}
|
|
|
|
static int device_param_complete(char *begin, struct device_d *dev,
|
|
struct string_list *sl, char *instr)
|
|
{
|
|
struct param_d *param;
|
|
int len;
|
|
|
|
if (!instr)
|
|
instr = "";
|
|
|
|
len = strlen(instr);
|
|
|
|
list_for_each_entry(param, &dev->parameters, list) {
|
|
if (strncmp(instr, param->name, len))
|
|
continue;
|
|
|
|
string_list_add_asprintf(sl, "%s%s.%s%c",
|
|
begin ? begin : "", dev_name(dev), param->name,
|
|
begin ? ' ' : '=');
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int empty_complete(struct string_list *sl, char *instr)
|
|
{
|
|
return COMPLETE_END;
|
|
}
|
|
|
|
int cammand_var_complete(struct string_list *sl, char *instr)
|
|
{
|
|
return COMPLETE_CONTINUE;
|
|
}
|
|
|
|
static int env_param_complete(struct string_list *sl, char *instr, int eval)
|
|
{
|
|
struct device_d *dev;
|
|
struct variable_d *var;
|
|
struct env_context *c, *current_c;
|
|
char *instr_param;
|
|
int len;
|
|
char end = '=';
|
|
char *begin = "";
|
|
|
|
if (!instr)
|
|
instr = "";
|
|
|
|
if (eval) {
|
|
begin = "$";
|
|
end = ' ';
|
|
}
|
|
|
|
instr_param = strrchr(instr, '.');
|
|
len = strlen(instr);
|
|
|
|
current_c = get_current_context();
|
|
for(var = current_c->local->next; var; var = var->next) {
|
|
if (strncmp(instr, var_name(var), len))
|
|
continue;
|
|
string_list_add_asprintf(sl, "%s%s%c",
|
|
begin, var_name(var), end);
|
|
}
|
|
|
|
for (c = get_current_context(); c; c = c->parent) {
|
|
for (var = c->global->next; var; var = var->next) {
|
|
if (strncmp(instr, var_name(var), len))
|
|
continue;
|
|
string_list_add_asprintf(sl, "%s%s%c",
|
|
begin, var_name(var), end);
|
|
}
|
|
}
|
|
|
|
if (instr_param) {
|
|
len = (instr_param - instr);
|
|
instr_param++;
|
|
} else {
|
|
len = strlen(instr);
|
|
instr_param = "";
|
|
}
|
|
|
|
for_each_device(dev) {
|
|
if (!strncmp(instr, dev_name(dev), len)) {
|
|
if (eval)
|
|
device_param_complete("$", dev, sl, instr_param);
|
|
else
|
|
device_param_complete(NULL, dev, sl, instr_param);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int tab_pressed = 0;
|
|
|
|
void complete_reset(void)
|
|
{
|
|
tab_pressed = 0;
|
|
}
|
|
|
|
static char* cmd_complete_lookup(struct string_list *sl, char *instr)
|
|
{
|
|
struct command *cmdtp;
|
|
int len;
|
|
int ret = COMPLETE_END;
|
|
char *res = NULL;
|
|
|
|
for_each_command(cmdtp) {
|
|
len = strlen(cmdtp->name);
|
|
if (!strncmp(instr, cmdtp->name, len) && instr[len] == ' ') {
|
|
instr += len + 1;
|
|
if (cmdtp->complete) {
|
|
ret = cmdtp->complete(sl, instr);
|
|
res = instr;
|
|
}
|
|
goto end;
|
|
}
|
|
}
|
|
|
|
end:
|
|
if (ret == COMPLETE_CONTINUE && *instr == '$')
|
|
env_param_complete(sl, instr + 1, 1);
|
|
|
|
return res;
|
|
}
|
|
|
|
int complete(char *instr, char **outstr)
|
|
{
|
|
struct string_list sl, *entry, *first_entry;
|
|
int pos;
|
|
char ch;
|
|
int changed;
|
|
static char out[256];
|
|
int outpos = 0;
|
|
int reprint = 0;
|
|
char *t;
|
|
|
|
string_list_init(&sl);
|
|
|
|
/* advance to the last command */
|
|
t = strrchr(instr, ';');
|
|
if (!t)
|
|
t = instr;
|
|
else
|
|
t++;
|
|
|
|
while (*t == ' ')
|
|
t++;
|
|
|
|
/* get the completion possibilities */
|
|
instr = cmd_complete_lookup(&sl, t);
|
|
if (!instr) {
|
|
instr = t;
|
|
if (t && (t[0] == '/' || !strncmp(t, "./", 2))) {
|
|
file_complete(&sl, t, 1);
|
|
instr = t;
|
|
} else if ((t = strrchr(t, ' '))) {
|
|
t++;
|
|
file_complete(&sl, t, 0);
|
|
instr = t;
|
|
} else {
|
|
command_complete(&sl, instr);
|
|
path_command_complete(&sl, instr);
|
|
env_param_complete(&sl, instr, 0);
|
|
}
|
|
if (*instr == '$')
|
|
env_param_complete(&sl, instr + 1, 1);
|
|
}
|
|
|
|
pos = strlen(instr);
|
|
|
|
*outstr = "";
|
|
if (list_empty(&sl.list))
|
|
return reprint;
|
|
|
|
out[0] = 0;
|
|
|
|
first_entry = list_first_entry(&sl.list, struct string_list, list);
|
|
|
|
while (1) {
|
|
entry = first_entry;
|
|
ch = entry->str[pos];
|
|
if (!ch)
|
|
break;
|
|
|
|
changed = 0;
|
|
list_for_each_entry(entry, &sl.list, list) {
|
|
if (!entry->str[pos])
|
|
break;
|
|
if (ch != entry->str[pos]) {
|
|
changed = 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (changed)
|
|
break;
|
|
out[outpos++] = ch;
|
|
pos++;
|
|
}
|
|
|
|
if (!list_is_last(&first_entry->list, &sl.list) && !outpos && tab_pressed) {
|
|
printf("\n");
|
|
string_list_print_by_column(&sl);
|
|
reprint = 1;
|
|
}
|
|
|
|
out[outpos++] = 0;
|
|
*outstr = out;
|
|
|
|
if (*out == 0)
|
|
tab_pressed = 1;
|
|
else
|
|
tab_pressed = 0;
|
|
|
|
string_list_free(&sl);
|
|
|
|
return reprint;
|
|
}
|
|
|