diff options
author | Taru Karttunen <taruti@taruti.net> | 2011-03-30 15:46:40 +0300 |
---|---|---|
committer | Taru Karttunen <taruti@taruti.net> | 2011-03-30 15:46:40 +0300 |
commit | e5888a1ffdae813d7575f5fb02275c6bb07e5199 (patch) | |
tree | d8d51eac403f07814b9e936eed0c9a79195e2450 /sys/src/libcomplete |
Import sources from 2011-03-30 iso image
Diffstat (limited to 'sys/src/libcomplete')
-rwxr-xr-x | sys/src/libcomplete/complete.c | 144 | ||||
-rwxr-xr-x | sys/src/libcomplete/mkfile | 15 |
2 files changed, 159 insertions, 0 deletions
diff --git a/sys/src/libcomplete/complete.c b/sys/src/libcomplete/complete.c new file mode 100755 index 000000000..6c7c7e67a --- /dev/null +++ b/sys/src/libcomplete/complete.c @@ -0,0 +1,144 @@ +#include <u.h> +#include <libc.h> +#include "complete.h" + +static int +longestprefixlength(char *a, char *b, int n) +{ + int i, w; + Rune ra, rb; + + for(i=0; i<n; i+=w){ + w = chartorune(&ra, a); + chartorune(&rb, b); + if(ra != rb) + break; + a += w; + b += w; + } + return i; +} + +void +freecompletion(Completion *c) +{ + if(c){ + free(c->filename); + free(c); + } +} + +static int +strpcmp(const void *va, const void *vb) +{ + char *a, *b; + + a = *(char**)va; + b = *(char**)vb; + return strcmp(a, b); +} + +Completion* +complete(char *dir, char *s) +{ + long i, l, n, nfile, len, nbytes; + int fd, minlen; + Dir *dirp; + char **name, *p; + ulong* mode; + Completion *c; + + if(strchr(s, '/') != nil){ + werrstr("slash character in name argument to complete()"); + return nil; + } + + fd = open(dir, OREAD); + if(fd < 0) + return nil; + + n = dirreadall(fd, &dirp); + if(n <= 0){ + close(fd); + return nil; + } + + /* find longest string, for allocation */ + len = 0; + for(i=0; i<n; i++){ + l = strlen(dirp[i].name) + 1 + 1; /* +1 for / +1 for \0 */ + if(l > len) + len = l; + } + + name = malloc(n*sizeof(char*)); + mode = malloc(n*sizeof(ulong)); + c = malloc(sizeof(Completion) + len); + if(name == nil || mode == nil || c == nil) + goto Return; + memset(c, 0, sizeof(Completion)); + + /* find the matches */ + len = strlen(s); + nfile = 0; + minlen = 1000000; + for(i=0; i<n; i++) + if(strncmp(s, dirp[i].name, len) == 0){ + name[nfile] = dirp[i].name; + mode[nfile] = dirp[i].mode; + if(minlen > strlen(dirp[i].name)) + minlen = strlen(dirp[i].name); + nfile++; + } + + if(nfile > 0) { + /* report interesting results */ + /* trim length back to longest common initial string */ + for(i=1; i<nfile; i++) + minlen = longestprefixlength(name[0], name[i], minlen); + + /* build the answer */ + c->complete = (nfile == 1); + c->advance = c->complete || (minlen > len); + c->string = (char*)(c+1); + memmove(c->string, name[0]+len, minlen-len); + if(c->complete) + c->string[minlen++ - len] = (mode[0]&DMDIR)? '/' : ' '; + c->string[minlen - len] = '\0'; + c->nmatch = nfile; + } else { + /* no match, so return all possible strings */ + for(i=0; i<n; i++){ + name[i] = dirp[i].name; + mode[i] = dirp[i].mode; + } + nfile = n; + c->nmatch = 0; + } + + /* attach list of names */ + nbytes = nfile * sizeof(char*); + for(i=0; i<nfile; i++) + nbytes += strlen(name[i]) + 1 + 1; + c->filename = malloc(nbytes); + if(c->filename == nil) + goto Return; + p = (char*)(c->filename + nfile); + for(i=0; i<nfile; i++){ + c->filename[i] = p; + strcpy(p, name[i]); + p += strlen(p); + if(mode[i] & DMDIR) + *p++ = '/'; + *p++ = '\0'; + } + c->nfile = nfile; + qsort(c->filename, c->nfile, sizeof(c->filename[0]), strpcmp); + + Return: + free(name); + free(mode); + free(dirp); + close(fd); + return c; +} diff --git a/sys/src/libcomplete/mkfile b/sys/src/libcomplete/mkfile new file mode 100755 index 000000000..9ac7ab2a2 --- /dev/null +++ b/sys/src/libcomplete/mkfile @@ -0,0 +1,15 @@ +</$objtype/mkfile + +LIB=/$objtype/lib/libcomplete.a +OFILES=\ + complete.$O\ + +HFILES=/sys/include/complete.h + +UPDATE=\ + mkfile\ + $HFILES\ + ${OFILES:%.$O=%.c}\ + ${LIB:/$objtype/%=/386/%}\ + +</sys/src/cmd/mksyslib |