ref: 9c70d9eff63775f79cdf91a5cea86de8a169bcbd
parent: d79335d57a7f6f608e8caf800b74155a299a550c
author: Sigrid Solveig Haflínudóttir <sigrid@ftrv.se>
date: Tue Oct 1 22:43:26 EDT 2024
Squashed 'c9/' changes from 3196726..e5e39ad e5e39ad wip revamping aa98d28 C9stat: put qid first, as it has to match with qid[0] git-subtree-dir: c9 git-subtree-split: e5e39adf2d8810e8a0aca2acf4e825ebcb21e3ed
--- a/.gitignore
+++ /dev/null
@@ -1,2 +1,0 @@
-9pex
-9gc
--- a/9gc.c
+++ /dev/null
@@ -1,460 +1,0 @@
-#define _DEFAULT_SOURCE
-#define _FILE_OFFSET_BITS 64
-#include <errno.h>
-#include <netdb.h>
-#include <netinet/in.h>
-#include <stdarg.h>
-#include <stdio.h>
-#include <stdlib.h>
-#include <string.h>
-#include <sys/select.h>
-#include <sys/socket.h>
-#include <sys/types.h>
-#include <termios.h>
-#include <unistd.h>
-#include "c9.h"
-#include "parg.h"
-
-enum {
- Msize = 8192,
-
- Rootfid = 0,
- Indexfid,
- Chatfid,
-
- Error = 1<<0,
- Joined = 1<<1,
- Disconnected = 1<<2,
-};
-
-typedef struct C9aux C9aux;
-typedef struct REntry REntry;
-
-struct C9aux {
- C9ctx c;
- int f;
- int flags;
- uint8_t rdbuf[Msize];
- uint8_t wrbuf[Msize];
- uint32_t wroff;
-};
-
-static const char *nick;
-static int printjoin;
-static uint64_t chatoff, skipuntil;
-//static char *regsrv = "tcp!registry.9p.zone!6675";
-static char *chatsrv = "tcp!chat.9p.zone!9990";
-static const char *channel = "chat";
-
-static uint8_t *
-ctxread(C9ctx *ctx, uint32_t size, int *err)
-{
- uint32_t n;
- int r;
- C9aux *a;
-
- a = ctx->aux;
- *err = 0;
- for (n = 0; n < size; n += r) {
- if ((r = read(a->f, a->rdbuf+n, size-n)) <= 0) {
- if (errno == EINTR)
- continue;
- a->flags |= Disconnected;
- close(a->f);
- return NULL;
- }
- }
-
- return a->rdbuf;
-}
-
-static int
-wrsend(C9aux *a)
-{
- uint32_t n;
- int w;
-
- for (n = 0; n < a->wroff; n += w) {
- if ((w = write(a->f, a->wrbuf+n, a->wroff-n)) <= 0) {
- if (errno == EINTR)
- continue;
- if (errno != EPIPE) /* remote end closed */
- perror("write");
- return -1;
- }
- }
- a->wroff = 0;
-
- return 0;
-}
-
-static uint8_t *
-ctxbegin(C9ctx *ctx, uint32_t size)
-{
- uint8_t *b;
- C9aux *a;
-
- a = ctx->aux;
- if (a->wroff + size > sizeof(a->wrbuf)) {
- if (wrsend(a) != 0 || a->wroff + size > sizeof(a->wrbuf))
- return NULL;
- }
- b = a->wrbuf + a->wroff;
- a->wroff += size;
-
- return b;
-}
-
-static int
-ctxend(C9ctx *ctx)
-{
- (void)ctx;
- return 0;
-}
-
-static int
-dial(char *s)
-{
- struct addrinfo *r, *a, hint = {.ai_flags = AI_ADDRCONFIG, .ai_family = AF_UNSPEC, 0};
- char host[64], *port;
- int e, f;
-
- if (strncmp(s, "udp!", 4) == 0) {
- hint.ai_socktype = SOCK_DGRAM;
- hint.ai_protocol = IPPROTO_UDP;
- } else if (strncmp(s, "tcp!", 4) == 0) {
- hint.ai_socktype = SOCK_STREAM;
- hint.ai_protocol = IPPROTO_TCP;
- } else {
- fprintf(stderr, "invalid dial string: %s\n", s);
- return -1;
- }
- if ((port = strchr(s+4, '!')) == NULL) {
- fprintf(stderr, "invalid dial string: %s\n", s);
- return -1;
- }
- if (snprintf(host, sizeof(host), "%.*s", (int)(port-s-4), s+4) >= (int)sizeof(host)) {
- fprintf(stderr, "host name too large: %s\n", s);
- return -1;
- }
- port++;
- if ((e = getaddrinfo(host, port, &hint, &r)) != 0)
- return -1;
- f = -1;
- for (a = r; a != NULL; a = a->ai_next) {
- if ((f = socket(a->ai_family, a->ai_socktype, a->ai_protocol)) < 0)
- continue;
- if (connect(f, a->ai_addr, a->ai_addrlen) != 0) {
- close(f);
- f = -1;
- continue;
- }
- }
- freeaddrinfo(r);
-
- return f;
-}
-
-static void
-output(uint8_t *d, int sz)
-{
- int i, j;
-
- for (i = j = 0; i < sz; i++) {
- d[j] = d[i];
- if (d[j] > 31 || d[j] == '\t' || d[j] == '\n')
- j++;
- }
- write(1, d, j);
-}
-
-static void
-ctxchatR(C9ctx *ctx, C9r *r)
-{
- C9aux *a;
- C9tag tag;
- const char *path[2];
- char buf[64];
-
- a = ctx->aux;
- switch (r->type) {
- case Rversion:
- c9attach(ctx, &tag, Rootfid, C9nofid, "none", NULL);
- path[0] = channel;
- path[1] = NULL;
- c9walk(ctx, &tag, Rootfid, Chatfid, path);
- c9open(ctx, &tag, Chatfid, C9rdwr);
- break;
-
- case Rread:
- if (chatoff >= skipuntil)
- output(r->read.data, r->read.size);
- chatoff += r->read.size;
- /* fallthrough */
- case Ropen:
- if ((a->flags & Joined) == 0 && printjoin) {
- c9write(ctx, &tag, Chatfid, 0, buf, snprintf(buf, sizeof(buf), "JOIN %s to chat\n", nick));
- a->flags |= Joined;
- }
- c9read(ctx, &tag, Chatfid, chatoff, chatoff < skipuntil ? skipuntil-chatoff : Msize);
- break;
-
- case Rerror:
- fprintf(stderr, "chat error: %s\n", r->error);
- a->flags |= Error;
- break;
-
- default:
- break;
- }
-}
-
-static void
-ctxregistryR(C9ctx *ctx, C9r *r)
-{
- char *s, *b;
- C9aux *a;
- C9tag tag;
- const char *path[2];
-
- a = ctx->aux;
- switch (r->type) {
- case Rversion:
- c9attach(ctx, &tag, Rootfid, C9nofid, "none", NULL);
- path[0] = "index";
- path[1] = NULL;
- c9walk(ctx, &tag, Rootfid, Indexfid, path);
- c9open(ctx, &tag, Indexfid, C9read);
- break;
-
- case Ropen:
- c9read(ctx, &tag, Indexfid, 0, Msize);
- break;
-
- case Rread:
- r->read.data[r->read.size] = 0;
- for (s = (char*)r->read.data;;) {
- if ((s = strstr(s, "chat")) == NULL)
- break;
- for (b = s; b != (char*)r->read.data && *b != '\n'; b--);
- if (*b == '\n')
- b++;
- if ((s = strchr(s, '\n')) == NULL)
- s = (char*)&r->read.data[r->read.size];
- else
- *s++ = 0;
- if (strstr(b, "tlssrv") == NULL && (s = strchr(b, ' ')) != NULL) {
- *s = 0;
-fallback:
- close(a->f);
- if ((a->f = dial(b)) < 0)
- exit(1);
- a->flags = 0;
- ctx->r = ctxchatR;
- a->wroff = 0;
- c9version(ctx, &tag, Msize);
- if (wrsend(a) != 0)
- exit(1);
- return;
- }
- }
- b = chatsrv;
- goto fallback;
-
- case Rerror:
- fprintf(stderr, "registry error: %s\n", r->error);
- a->flags |= Error;
- break;
-
- default:
- break;
- }
-}
-
-__attribute__ ((format (printf, 1, 2)))
-static void
-ctxerror(const char *fmt, ...)
-{
- va_list ap;
-
- va_start(ap, fmt);
- vfprintf(stderr, fmt, ap);
- fprintf(stderr, "\n");
- va_end(ap);
-}
-
-static C9aux *
-srv(char *s)
-{
- C9aux *c;
- int f;
-
- if ((f = dial(s)) < 0)
- return NULL;
- c = calloc(1, sizeof(*c));
- c->f = f;
- c->c.read = ctxread;
- c->c.begin = ctxbegin;
- c->c.end = ctxend;
- c->c.error = ctxerror;
- c->c.aux = c;
-
- return c;
-}
-
-static C9aux *
-registry(void)
-{
- C9aux *a;
- C9tag tag;
- int i;
-
- for (i = 0; i < 10; i++) {
-// if ((a = srv(regsrv)) == NULL) {
- if ((a = srv(chatsrv)) != NULL) {
- a->c.r = ctxchatR;
- c9version(&a->c, &tag, Msize);
- wrsend(a);
- return a;
- }
-// } else {
-// break;
-// }
- sleep(10);
- }
-
- if (a == NULL)
- return NULL;
- a->c.r = ctxregistryR;
- c9version(&a->c, &tag, Msize);
- wrsend(a);
- while (c9proc(&a->c) == 0 && a->c.r == ctxregistryR)
- wrsend(a);
-
- return a;
-}
-
-static int
-chatrw(C9aux *a)
-{
- struct timeval t;
- fd_set r, e;
- int n, sz;
- C9tag tag;
- C9ctx *ctx;
- char *s;
-
- FD_ZERO(&r);
- FD_SET(a->f, &r);
- FD_SET(0, &r);
- FD_ZERO(&e);
- FD_SET(a->f, &e);
- FD_SET(0, &e);
- memset(&t, 0, sizeof(t));
- t.tv_sec = 10;
- for (;;) {
- errno = 0;
- if ((n = select(a->f + 1, &r, NULL, &e, &t)) < 0 || FD_ISSET(a->f, &e) || FD_ISSET(0, &e)) {
- if (errno == EINTR)
- continue;
- return -1;
- }
- break;
- }
-
- ctx = &a->c;
- if (FD_ISSET(a->f, &r)) {
- c9proc(ctx);
- } else if (FD_ISSET(0, &r)) {
- s = (char*)a->rdbuf;
- sz = sprintf(s, "%s → ", nick);
- for (;;) {
- if ((n = read(0, s+sz, sizeof(a->rdbuf)-sz)) > 0)
- sz += n;
- else
- exit(0);
- if (s[sz-1] != '\n'){
- s[sz-1] = '\n';
- }else{
- c9write(ctx, &tag, Chatfid, 0, s, sz);
- break;
- }
- }
- } else {
- const char *path[] = {NULL};
- c9walk(ctx, &tag, Rootfid, Rootfid, path);
- }
-
- return 0;
-}
-
-int
-main(int argc, char **argv)
-{
- struct parg_state ps;
- struct termios t;
- C9aux *a;
- int c, noecho;
-
- parg_init(&ps);
- noecho = 0;
- while ((c = parg_getopt(&ps, argc, argv, "hjedc:")) >= 0) {
- switch (c) {
- case 1:
- if (nick != NULL) {
- fprintf(stderr, "only one nickname can be specified\n");
- return 1;
- }
- nick = ps.optarg;
- break;
- case 'c':
- channel = ps.optarg;
- break;
- case 'j':
- printjoin = 1;
- break;
- case 'e':
- noecho = 1;
- break;
- case 'h':
- fprintf(stderr, "usage: 9gc [-j] [-e] [-c CHANNEL] NICKNAME\n");
- return 0;
- break;
- case '?':
- fprintf(stderr, "unknown option -%c\n", ps.optopt);
- return 1;
- break;
- default:
- fprintf(stderr, "unhandled option -%c\n", c);
- return 1;
- break;
- }
- }
-
- if (nick == NULL) {
- fprintf(stderr, "no nickname specified\n");
- return 1;
- }
-
- if (noecho && tcgetattr(0, &t) == 0) {
- t.c_lflag &= ~ECHO;
- tcsetattr(STDIN_FILENO, TCSANOW, &t);
- }
-
- for (;;) {
- if ((a = registry()) == NULL)
- return 1;
- while (chatrw(a) == 0 && wrsend(a) == 0);
- if (a->flags & (Disconnected|Error)) {
- a->flags &= ~(Disconnected|Error);
- skipuntil = chatoff;
- chatoff = 0;
- free(a);
- } else {
- printf("exiting\n");
- exit(1);
- }
- }
-
- return 0;
-}
--- a/9pex.c
+++ /dev/null
@@ -1,1052 +1,0 @@
-#ifdef __linux__
-#define _GNU_SOURCE
-#else
-#define _DEFAULT_SOURCE
-#endif
-#define _FILE_OFFSET_BITS 64
-#include <dirent.h>
-#include <errno.h>
-#include <fcntl.h>
-#include <grp.h>
-#include <inttypes.h>
-#include <pwd.h>
-#include <signal.h>
-#include <stdarg.h>
-#include <stdint.h>
-#include <stdio.h>
-#include <stdlib.h>
-#include <string.h>
-#include <sys/select.h>
-#include <sys/stat.h>
-#include <sys/time.h>
-#include <unistd.h>
-#include "c9.h"
-#include "parg.h"
-
-#define max(a,b) ((a)>(b)?(a):(b))
-#define used(x) ((void)(x))
-#define nelem(x) (int)(sizeof(x)/sizeof((x)[0]))
-
-uint32_t crc32(const void *data, int len);
-
-enum
-{
- Canrd = 1<<0,
- Canwr = 1<<1,
-};
-
-typedef struct
-{
- char *path; /* full path */
- char *name; /* base name */
-
- DIR *dir; /* set when it's an opened directory */
- uint64_t diroffset; /* to read dirs correctly */
-
- C9qid qid;
- C9fid fid;
- C9mode mode; /* mode in which the file was opened */
- uint32_t iounit;
-
- int fd; /* set when it's an opened file */
-}Fid;
-
-typedef struct
-{
- C9tag tag;
-}Tag;
-
-typedef struct
-{
- char *name;
- uint32_t id;
-}Id;
-
-static char *t2s[] = {
- [Tversion-Tversion] = "Tversion",
- [Tauth-Tversion] = "Tauth",
- [Tattach-Tversion] = "Tattach",
- [Tflush-Tversion] = "Tflush",
- [Twalk-Tversion] = "Twalk",
- [Topen-Tversion] = "Topen",
- [Tcreate-Tversion] = "Tcreate",
- [Tread-Tversion] = "Tread",
- [Twrite-Tversion] = "Twrite",
- [Tclunk-Tversion] = "Tclunk",
- [Tremove-Tversion] = "Tremove",
- [Tstat-Tversion] = "Tstat",
- [Twstat-Tversion] = "Twstat",
-};
-
-static char *modes[] = {
- "read", "write", "rdwr", "exec",
-};
-
-static char *Enoauth = "authentication not required";
-static char *Eunknownfid = "unknown fid";
-static char *Enowstat = "wstat prohibited";
-static char *Eperm = "permission denied";
-static char *Enowrite = "write prohibited";
-static char *Enomem = "out of memory";
-static char *Edupfid = "duplicate fid";
-static char *Ewalknodir = "walk in non-directory";
-static char *Enotfound = "file not found";
-static char *Eduptag = "duplicate tag";
-static char *Ebotch = "9P protocol botch";
-static char *Enocreate = "create prohibited";
-static char *Eisdir = "is a directory";
-static char *Ebadoffset = "bad offset";
-
-static int in, out, eof;
-static C9ctx ctx;
-static int debug, rootescape;
-static Fid **fids;
-static int numfids;
-static Tag **tags;
-static int numtags;
-static char *rootpath;
-static size_t rootlen;
-static C9qid walkqids[C9maxpathel];
-static uint8_t *rdbuf;
-static uint8_t *wrbuf;
-static uint32_t wroff, wrend, wrbufsz;
-static Id *uids, *gids;
-static int numuids, numgids;
-
-__attribute__ ((format (printf, 1, 2)))
-static void
-trace(const char *fmt, ...)
-{
- va_list ap;
-
- if (debug == 0)
- return;
-
- va_start(ap, fmt);
- vfprintf(stderr, fmt, ap);
- va_end(ap);
-}
-
-static int
-canrw(int rdonly, int block)
-{
- struct timeval t;
- fd_set r, w, e;
- int n, fl;
-
- FD_ZERO(&r);
- FD_SET(in, &r);
- FD_ZERO(&w);
- if (rdonly == 0)
- FD_SET(out, &w);
- FD_ZERO(&e);
- FD_SET(in, &e);
- FD_SET(out, &e);
- memset(&t, 0, sizeof(t));
- t.tv_usec = 1000;
- for (;;) {
- errno = 0;
- if ((n = select(max(in, out) + 1, &r, &w, &e, block ? NULL : &t)) < 0 || FD_ISSET(in, &e) || FD_ISSET(out, &e)) {
- if (errno == EINTR)
- continue;
- return -1;
- }
- break;
- }
-
- fl = 0;
- if (FD_ISSET(in, &r))
- fl |= Canrd;
- if (FD_ISSET(out, &w))
- fl |= Canwr;
-
- return fl;
-}
-
-static int
-wrsend(void)
-{
- uint32_t n;
- int w;
-
- if (wrend == 0)
- return 0;
- for (n = 0; n < wrend; n += w) {
- errno = 0;
- if ((w = write(out, wrbuf+n, wrend-n)) <= 0) {
- if (errno == EINTR)
- continue;
- if (errno != EPIPE) /* remote end closed */
- perror("write");
- return -1;
- }
- }
- if (debug >= 2)
- trace("<- %d bytes, %d left\n", wrend, wroff-wrend);
- memmove(wrbuf, wrbuf+wrend, wroff-wrend);
- wroff = wroff - wrend;
-
- return 0;
-}
-
-static int
-hastag(C9tag tag)
-{
- int i;
-
- for (i = 0; i < numtags; i++) {
- if (tags[i] != NULL && tags[i]->tag == tag)
- return 1;
- }
-
- return 0;
-}
-
-static int
-statpath(char *path, struct stat *st, char **err)
-{
- if (stat(path, st) == 0)
- return 0;
-
- if (errno == EACCES)
- *err = Eperm;
- else if (errno == ENOMEM)
- *err = Enomem;
- else if (errno == ENOTDIR)
- *err = Ewalknodir;
- else if (errno == ENOENT)
- *err = Enotfound;
- else
- *err = strerror(errno);
-
- return -1;
-}
-
-static void
-stat2qid(struct stat *st, C9qid *qid, uint32_t *iounit)
-{
- int fmt;
-
- qid->path = st->st_ino;
- qid->version = crc32(&st->st_ctim, sizeof(st->st_ctim));
- qid->type = C9qtfile;
- fmt = st->st_mode & S_IFMT;
- if (fmt == S_IFDIR)
- qid->type |= C9qtdir;
- if ((st->st_mode & 0222) != 0 && (fmt == S_IFCHR || fmt == S_IFCHR || fmt == S_IFSOCK || fmt == S_IFIFO))
- qid->type |= C9qtappend;
- if (iounit != NULL)
- *iounit = st->st_blksize;
-}
-
-static Fid *
-newfid(C9fid fid, char *path, char **err)
-{
- Fid *f, **newfids;
- struct stat st;
- int i;
-
- for (i = 0; i < numfids; i++) {
- if (fids[i] != NULL && fids[i]->fid == fid) {
- *err = Edupfid;
- return NULL;
- }
- }
-
- if (statpath(path, &st, err) != 0)
- return NULL;
-
- if ((f = calloc(1, sizeof(*f))) == NULL) {
- *err = Enomem;
- return NULL;
- }
- f->fd = -1;
- f->path = strdup(path);
- f->name = strrchr(f->path, '/');
- if (f->name == NULL)
- f->name = f->path;
- else
- f->name++;
- if (f->name[0] == 0)
- fprintf(stderr, "%s -> empty file name\n", f->path);
-
- for (i = 0; i < numfids; i++) {
- if (fids[i] == NULL) {
- fids[i] = f;
- break;
- }
- }
- if (i >= numfids) {
- if ((newfids = realloc(fids, (numfids+1)*sizeof(*fids))) == NULL) {
- *err = Enomem;
- return NULL;
- }
- fids = newfids;
- fids[numfids++] = f;
- }
-
- f->fid = fid;
- stat2qid(&st, &f->qid, &f->iounit);
-
- return f;
-}
-
-static Fid *
-findfid(C9fid fid, char **err)
-{
- int i;
-
- for (i = 0; i < numfids; i++) {
- if (fids[i] != NULL && fids[i]->fid == fid) {
- return fids[i];
- }
- }
-
- *err = Eunknownfid;
-
- return NULL;
-}
-
-static int
-delfid(C9fid fid, char **err)
-{
- Fid *f;
- int i;
-
- for (i = 0; i < numfids; i++) {
- f = fids[i];
- if (f != NULL && f->fid == fid) {
- if (f->dir != NULL)
- closedir(f->dir);
- else if (f->fd >= 0)
- close(f->fd);
- free(f->path);
- free(f);
- fids[i] = NULL;
- return 0;
- }
- }
-
- *err = Eunknownfid;
-
- return -1;
-}
-
-static int
-hasperm(struct stat *st, C9mode mode, char **err)
-{
- int m, stm, fmt;
-
- m = mode & 0xf;
- stm = st->st_mode & 0777;
- *err = Eperm;
- if (((stm & 0111) == 0 || (stm & 0444) == 0) && m == C9exec) /* executing needs rx */
- return 0;
- if ((stm & 0222) == 0 && (m == C9write || m == C9rdwr)) /* writing needs w */
- return 0;
- if ((stm & 0444) == 0 && m != C9write) /* reading needs r */
- return 0;
- fmt = st->st_mode & S_IFMT;
- if (fmt == S_IFDIR && ((stm & 0111) == 0 || (stm & 0444) == 0)) /* dirs need rx */
- return 0;
- *err = NULL;
-
- return 1;
-}
-
-static Fid *
-walk(C9fid fid, C9fid nfid, char *el[], C9qid *qids[], char **err)
-{
- Fid *f;
- char *path, *real, *p;
- struct stat st;
- int i, plen, ellen;
-
- if ((f = findfid(fid, err)) == NULL)
- return NULL;
-
- if (el[0] == NULL) { /* nwname = 0 */
- qids[0] = NULL;
- if (fid == nfid)
- return f;
- return newfid(nfid, f->path, err);
- }
-
- if ((f->qid.type & C9qtdir) == 0) { /* has to be a dir */
- *err = Ewalknodir;
- return NULL;
- }
-
- p = strdup(f->path);
- f = NULL;
- for (i = 0; el[i] != NULL; i++) {
- plen = strlen(p);
- ellen = strlen(el[i]);
- path = malloc(plen + 1 + ellen + 1);
- memmove(path, p, plen);
- path[plen] = '/';
- memmove(path+plen+1, el[i], ellen);
- path[plen+1+ellen] = 0;
-
- if (!rootescape) {
- if ((real = realpath(path, NULL)) == NULL)
- break;
- free(path);
- if (strlen(real) < rootlen) { /* don't escape root */
- free(real);
- real = strdup(rootpath);
- }
- } else {
- real = path;
- }
-
- free(p);
- p = real;
-
- if (statpath(p, &st, err) != 0)
- break;
- if (el[i+1] != NULL && !hasperm(&st, C9read, err))
- break;
-
- qids[i] = &walkqids[i];
- stat2qid(&st, qids[i], NULL);
- }
-
- qids[i] = NULL;
- if (el[i] == NULL) { /* could walk all the way */
- f = newfid(nfid, p, err);
- if (f != NULL && f->name[0] == '/' && f->name[1] == 0) /* root */
- f->name = "/";
- } else if (i != 0) { /* didn't fail on the first one */
- *err = NULL;
- }
- free(p);
-
- return f;
-}
-
-static int
-mode2unix(C9mode mode, int qidtype)
-{
- int omode;
-
- omode = O_NOFOLLOW;
- if ((mode & 3) == C9read)
- omode |= O_RDONLY;
- else if ((mode & 3) == C9write)
- omode |= O_WRONLY;
- else if ((mode & 3) == C9rdwr)
- omode |= O_RDWR;
- if (mode & C9trunc)
- omode |= O_TRUNC;
-#ifdef __linux__
- if (mode & C9rclose)
- omode |= O_TMPFILE;
-#endif
- if ((qidtype & C9qtappend) != 0)
- omode |= O_APPEND;
-
- return omode;
-}
-
-static int
-openfid(Fid *f, C9mode mode, char **err)
-{
- struct stat st;
- int omode;
-
- if (stat(f->path, &st) != 0 || !hasperm(&st, mode, err))
- goto error;
- stat2qid(&st, &f->qid, &f->iounit);
-
- if ((f->qid.type & C9qtdir) != 0) {
- if ((f->dir = opendir(f->path)) == NULL) {
- *err = strerror(errno);
- return -1;
- }
- f->fd = dirfd(f->dir);
- } else {
- omode = mode2unix(mode, f->qid.type);
- f->fd = open(f->path, omode);
- }
-
- if (f->fd < 0)
- goto error;
- f->mode = mode;
-
- return 0;
-error:
- if (*err == NULL)
- *err = strerror(errno);
-
- if (f->dir != NULL)
- closedir(f->dir);
- else if (f->fd >= 0)
- close(f->fd);
-
- f->dir = NULL;
- f->fd = -1;
- return -1;
-}
-
-static char *
-uid2str(uid_t uid, char **err)
-{
- struct passwd *p;
- Id *newuids;
- int i;
-
- for (i = 0; i < numuids; i++) {
- if (uids[i].id == uid)
- return uids[i].name;
- }
- if ((p = getpwuid(uid)) == NULL) {
- *err = strerror(errno);
- return NULL;
- }
- if ((newuids = realloc(uids, sizeof(*uids)*(numuids+1))) == NULL) {
- *err = Enomem;
- return NULL;
- }
- uids = newuids;
- uids[numuids].id = uid;
- uids[numuids].name = strdup(p->pw_name);
-
- return uids[numuids++].name;
-}
-
-static char *
-gid2str(gid_t gid, char **err)
-{
- struct group *g;
- Id *newgids;
- int i;
-
- for (i = 0; i < numgids; i++) {
- if (gids[i].id == gid)
- return gids[i].name;
- }
- if ((g = getgrgid(gid)) == NULL) {
- *err = strerror(errno);
- return NULL;
- }
- if ((newgids = realloc(gids, sizeof(*gids)*(numgids+1))) == NULL) {
- *err = Enomem;
- return NULL;
- }
- gids = newgids;
- gids[numgids].id = gid;
- gids[numgids].name = strdup(g->gr_name);
-
- return gids[numgids++].name;
-}
-
-static int
-stat2c9stat(char *name, struct stat *st, C9stat *stout, char **err)
-{
- int fmt;
-
- memset(stout, 0, sizeof(*stout));
- stout->size = st->st_size;
- stat2qid(st, &stout->qid, NULL);
- stout->name = name;
- stout->atime = st->st_atim.tv_sec;
- stout->mtime = st->st_ctim.tv_sec;
-
- fmt = st->st_mode & S_IFMT;
- if (fmt == S_IFDIR)
- stout->mode |= C9stdir;
- if (fmt == S_IFCHR || fmt == S_IFCHR || fmt == S_IFSOCK || fmt == S_IFIFO)
- stout->mode |= C9stappend;
- stout->mode |= st->st_mode & 0x1ff;
- if ((stout->uid = uid2str(st->st_uid, err)) == NULL)
- return -1;
- if ((stout->gid = gid2str(st->st_gid, err)) == NULL)
- return -1;
-
- return 0;
-}
-
-static int
-statfid(Fid *f, C9stat *stout, char **err)
-{
- struct stat st;
- int r;
-
- if (f->fd >= 0)
- r = fstat(f->fd, &st);
- else
- r = stat(f->path, &st);
- if (r != 0) {
- *err = strerror(errno);
- return -1;
- }
-
- return stat2c9stat(f->name, &st, stout, err);
-}
-
-static uint8_t *
-ctxread(C9ctx *c, uint32_t size, int *err)
-{
- uint32_t n;
- int r;
-
- used(c);
- r = 0;
- *err = 0;
- for (n = 0; n < size; n += r) {
- errno = 0;
- if ((r = read(in, rdbuf+n, size-n)) <= 0) {
- if (r == EINTR)
- continue;
- if (r == 0) {
- eof = 1;
- } else {
- *err = 1;
- perror("ctxread");
- }
- return NULL;
- }
- }
-
- return rdbuf;
-}
-
-static uint8_t *
-ctxbegin(C9ctx *c, uint32_t size)
-{
- uint8_t *b;
-
- used(c);
- if (wroff + size > wrbufsz) {
- if (wrsend() != 0 || wroff + size > wrbufsz)
- return NULL;
- }
- b = wrbuf + wroff;
- wroff += size;
-
- return b;
-}
-
-static int
-ctxend(C9ctx *c)
-{
- used(c);
- wrend = wroff;
- return 0;
-}
-
-__attribute__ ((format (printf, 1, 2)))
-static void
-ctxerror(const char *fmt, ...)
-{
- va_list ap;
-
- va_start(ap, fmt);
- vfprintf(stderr, fmt, ap);
- fprintf(stderr, "\n");
- va_end(ap);
-}
-
-static int
-s9do(C9error e, char **err)
-{
- if (e == 0) {
- *err = NULL;
- return 0;
- }
-
- switch (e) {
- case C9Einit: *err = "c9: initialization failed"; break;
- case C9Ever: *err = "c9: protocol version doesn't match"; break;
- case C9Epkt: *err = "c9: incoming packet error"; break;
- case C9Etag: *err = "c9: no free tags or bad tag"; break;
- case C9Ebuf: *err = Enomem; break;
- case C9Epath: *err = "c9: path is too long or just invalid"; break;
- case C9Eflush: *err = "c9: limit of outstanding flushes reached"; break;
- case C9Esize: *err = "c9: can't fit data in one message"; break;
- case C9Estr: *err = "c9: bad string"; break;
- default: *err = "c9: unknown error"; break;
- }
-
- return -1;
-}
-
-static int
-readf(C9ctx *c, C9tag tag, Fid *f, uint64_t offset, uint32_t size, char **err)
-{
- struct stat st;
- void *p;
- struct dirent *e;
- ssize_t n;
- C9stat c9st[16], *c9stp[16];
- long dirpos[16];
- int i, num, res;
-
- if (size > c->msize - 12) /* make sure it fits */
- size = c->msize - 12;
-
- if (f->dir == NULL) { /* a file */
- if ((p = malloc(size)) == NULL) {
- *err = Enomem;
- return -1;
- }
- if ((n = pread(f->fd, p, size, offset)) < 0) {
- *err = strerror(errno);
- free(p);
- return -1;
- }
- res = s9do(s9read(c, tag, p, n), err);
- free(p);
- if (res != 0)
- return -1;
- trace("<- Rread tag=%d count=%zd data=...\n", tag, n);
- return 0;
- }
-
- /* dir */
- if (offset != f->diroffset) {
- if (offset == 0) {
- rewinddir(f->dir);
- f->diroffset = 0;
- } else {
- *err = Ebadoffset;
- return -1;
- }
- }
-
- res = 0;
- num = 0;
- for (i = 0; i < nelem(c9st); i++) {
- dirpos[i] = telldir(f->dir); /* so we can rewind in case another stat doesn't fit */
-
- errno = 0;
- if ((e = readdir(f->dir)) == NULL && errno != 0) {
- *err = strerror(errno);
- res = -1;
- goto done;
- }
- if (e == NULL) /* eof */
- break;
- if (e->d_name[0] == '.' && (e->d_name[1] == 0 || ((e->d_name[1] == '.' && e->d_name[2] == 0))))
- continue;
-
- if (fstatat(f->fd, e->d_name, &st, 0) != 0) {
- *err = strerror(errno);
- if (fstatat(f->fd, e->d_name, &st, AT_SYMLINK_NOFOLLOW) != 0) { /* broken symlink, try to stat the link itself */
- res = -1;
- goto done;
- }
- }
- if (stat2c9stat(e->d_name, &st, &c9st[i], err) != 0) {
- res = -1;
- goto done;
- }
- c9st[i].name = strdup(c9st[i].name);
- c9stp[num++] = &c9st[i];
- }
-
- i = num;
- if (s9do(s9readdir(c, tag, c9stp, &num, &f->diroffset, size), err) != 0) {
- res = -1;
- goto done;
- }
- trace("<- Rread tag=%d count=%"PRIu64" data=...\n", tag, f->diroffset - offset);
- if (i != num)
- seekdir(f->dir, dirpos[num]);
-
-done:
- for (i = 0; i < num; i++)
- free(c9stp[i]->name);
- return res;
-}
-
-static void
-ctxt(C9ctx *c, C9t *t)
-{
- Fid *f;
- C9qid *qids[C9maxpathel+1];
- char *err, *err2;
- C9stat st;
- int i;
-
- trace("-> %s tag=%d", t2s[t->type-Tversion], t->tag);
-
- err = NULL;
- if (hastag(t->tag)) {
- err = Eduptag;
- } else {
- switch (t->type){
- case Tversion:
- trace("\n");
- if (s9do(s9version(c), &err) == 0)
- trace("<- Rversion\n");
- break;
- case Tauth:
- trace(" afid=%d uname=\"%s\" aname=\"%s\"\n", t->auth.afid, t->auth.uname, t->auth.aname);
- err = Enoauth;
- break;
- case Tattach:
- trace(" afid=%d fid=%d uname=\"%s\" aname=\"%s\"\n", t->attach.afid, t->fid, t->attach.uname, t->attach.aname);
- if (t->attach.afid != C9nofid) {
- err = Eunknownfid;
- } else if ((f = newfid(t->fid, rootpath, &err)) != NULL) {
- f->name = "/";
- if (s9do(s9attach(c, t->tag, &f->qid), &err) == 0)
- trace("<- Rattach\n");
- }
- break;
- case Tflush:
- trace(" oldtag=%d\n", t->flush.oldtag);
- /* FIXME flush it for realz */
- if (s9do(s9flush(c, t->tag), &err) == 0)
- trace("<- Rflush tag=%d\n", t->tag);
- break;
- case Twalk:
- trace(" fid=%d newfid=%d", t->fid, t->walk.newfid);
- for (i = 0; t->walk.wname[i] != NULL; i++)
- trace(" \"%s\"", t->walk.wname[i]);
- trace("\n");
- walk(t->fid, t->walk.newfid, t->walk.wname, qids, &err);
- if (err == NULL && s9do(s9walk(c, t->tag, qids), &err) == 0) {
- trace("<- Rwalk tag=%d ", t->tag);
- for (i = 0; qids[i] != NULL; i++)
- trace("qid=[path=%"PRIu64" type=0x%02x version=%"PRIu32"] ", qids[i]->path, qids[i]->type, qids[i]->version);
- trace("\n");
- }
- break;
- case Topen:
- trace(" fid=%d mode=0x%02x\n", t->fid, t->open.mode);
- if ((f = findfid(t->fid, &err)) != NULL) {
- if (f->fd >= 0)
- err = Ebotch;
- else if (t->open.mode != C9read && t->open.mode != C9exec)
- err = Eperm;
- else if (t->open.mode != C9read && (f->qid.type & C9qtdir) != 0)
- err = Eisdir;
- else if (openfid(f, t->open.mode, &err) == 0 && s9do(s9open(c, t->tag, &f->qid, f->iounit), &err) == 0)
- trace("<- Ropen tag=%d qid=[path=%"PRIu64" type=0x%02x version=%"PRIu32"] iounit=%d\n", t->tag, f->qid.path, f->qid.type, f->qid.version, f->iounit);
- }
- break;
- case Tcreate:
- trace("...\n");
- err = Enocreate;
- break;
- case Tread:
- trace(" fid=%d offset=%"PRIu64" count=%"PRIu32"\n", t->fid, t->read.offset, t->read.size);
- if ((f = findfid(t->fid, &err)) != NULL) {
- if ((f->dir == NULL && f->fd < 0) || (f->mode & 0xf) == C9write)
- err = Ebotch;
- else if (readf(c, t->tag, f, t->read.offset, t->read.size, &err) != 0)
- trace("readf failed\n");
- }
- break;
- case Twrite:
- trace("...\n");
- err = Enowrite;
- break;
- case Tclunk:
- trace(" fid=%d\n", t->fid);
- if (delfid(t->fid, &err) == 0 && s9do(s9clunk(c, t->tag), &err) == 0)
- trace("<- Rclunk tag=%d\n", t->tag);
- break;
- case Tremove:
- trace("\n");
- err = Eperm;
- break;
- case Tstat:
- trace(" fid=%d\n", t->fid);
- if ((f = findfid(t->fid, &err)) != NULL && statfid(f, &st, &err) == 0 && s9do(s9stat(c, t->tag, &st), &err) == 0)
- trace("<- Rstat tag=%d ...\n", t->tag);
- break;
- case Twstat:
- trace("...\n");
- err = Enowstat;
- break;
- }
- }
-
- if (err != NULL) {
- if (s9do(s9error(c, t->tag, err), &err2) == 0)
- trace("<- Rerror tag=%d \"%s\"\n", t->tag, err);
- else
- fprintf(stderr, "s9error: %s\n", err2);
- }
-}
-
-static void
-sigdebug(int s)
-{
- Fid *f;
- int i, n;
-
- used(s);
- n = 0;
- for (i = 0; i < numfids; i++) {
- f = fids[i];
- if (f == NULL)
- continue;
-
- fprintf(stderr, "fid %u ", f->fid);
-
- if (f->dir != NULL)
- fprintf(stderr, "open mode=dir ");
- else if (f->fd >= 0)
- fprintf(stderr, "open mode=%s%s%s ", modes[(f->mode & 0xf)], (f->mode & C9trunc) ? ",trunc" : "", (f->mode & C9rclose) ? ",rclose" : "");
-
- fprintf(stderr, "qid=[path=%"PRIu64" type=0x%02x version=%"PRIu32"] iounit=%d ", f->qid.path, f->qid.type, f->qid.version, f->iounit);
- fprintf(stderr, " %s %s\n", f->path, f->name);
- n++;
- }
-
- fprintf(stderr, "fids\t%d\n", n);
- fprintf(stderr, "tags\t%d\n", numtags);
- fprintf(stderr, "uids\t%d\n", numuids);
- fprintf(stderr, "gids\t%d\n", numgids);
-}
-
-int
-main(int argc, char **argv)
-{
- const char *dir;
- char *err;
- Fid *f;
- struct parg_state ps;
- struct sigaction sa;
- int can, i, c, rdonly, block;
-
- parg_init(&ps);
-
- debug = 0;
- dir = NULL;
- while ((c = parg_getopt(&ps, argc, argv, "deh")) >= 0) {
- switch (c) {
- case 1:
- if (dir != NULL) {
- fprintf(stderr, "only one dir can be specified\n");
- return 1;
- }
- dir = ps.optarg;
- break;
- case 'e':
- rootescape++;
- break;
- case 'd':
- debug++;
- break;
- case 'h':
- fprintf(stderr, "usage: 9pex [-e] [-d] DIR\n");
- return 0;
- break;
- case '?':
- fprintf(stderr, "unknown option -%c\n", ps.optopt);
- return 1;
- break;
- default:
- fprintf(stderr, "unhandled option -%c\n", c);
- return 1;
- break;
- }
- }
-
- if (dir == NULL) {
- fprintf(stderr, "no dir specified\n");
- return 1;
- }
-
- if ((rootpath = realpath(dir, NULL)) == NULL) {
- trace("%s: %s\n", dir, strerror(errno));
- return 1;
- }
- rootlen = strlen(rootpath);
-
- in = 0;
- out = 1;
- eof = 0;
- fids = NULL;
- numfids = 0;
- tags = NULL;
- numtags = 0;
- uids = NULL;
- numuids = 0;
- gids = NULL;
- numgids = 0;
-
- memset(&ctx, 0, sizeof(ctx));
- ctx.msize = 64*1024;
- ctx.read = ctxread;
- ctx.begin = ctxbegin;
- ctx.end = ctxend;
- ctx.t = ctxt;
- ctx.error = ctxerror;
-
- rdbuf = calloc(1, ctx.msize);
- wrbufsz = ctx.msize;
- wrbuf = calloc(1, wrbufsz);
- wroff = wrend = 0;
-
- memset(&sa, 0, sizeof(sa));
- sa.sa_handler = SIG_IGN;
- sa.sa_flags = SA_RESTART;
- sigaction(SIGPIPE, &sa, NULL);
-
- memset(&sa, 0, sizeof(sa));
- sa.sa_handler = sigdebug;
- sa.sa_flags = SA_RESTART;
- sigfillset(&sa.sa_mask);
- sigaction(SIGUSR1, &sa, NULL);
-
- err = NULL;
- rdonly = block = 1; /* at first we wait until the client sends in data */
- for (; !eof;) {
- if ((can = canrw(rdonly, block)) < 0)
- break;
- if ((can & Canrd) != 0) { /* if there is data, process it */
- if (s9do(s9proc(&ctx), &err) != 0)
- break;
- /* give it a chance to receive all the data first */
- rdonly = 1;
- block = 0;
- } else if (block == 0) { /* got all the data */
- if (rdonly != 0) { /* wait until we can send OR we get more data */
- rdonly = 0;
- block = 1;
- }
- } else if (rdonly == 0 && (can & Canwr) != 0) { /* can send */
- if (wrsend() != 0) /* send all the data */
- break;
- rdonly = 1; /* and go back to reading */
- block = 1;
- }
- }
-
- if (err != NULL)
- trace("s9proc: %s\n", err);
-
- for (i = 0; i < numfids; i++) {
- if ((f = fids[i]) != NULL) {
- if (f->dir != NULL)
- closedir(f->dir);
- else if (f->fd >= 0)
- close(f->fd);
- free(f->path);
- free(f);
- }
- }
-
- for (i = 0; i < numuids; i++)
- free(uids[i].name);
- free(uids);
- for (i = 0; i < numgids; i++)
- free(gids[i].name);
- free(gids);
-
- memset(wrbuf, 0, ctx.msize);
- free(wrbuf);
- memset(rdbuf, 0, ctx.msize);
- free(rdbuf);
- free(fids);
- free(rootpath);
-
- return 0;
-}
--- a/README.md
+++ b/README.md
@@ -1,25 +1,25 @@
-# 9pro
+# c9
-Plan9-related tools for Unix-like operating systems.
+NOTE (2022-06-27): the library's (yep, it's going to be a library)
+API is getting completely revamped, and examples are being added.
+Use either an older version, or expect API breakage.
- * 9pex - share a directory over stdin/stdout, can be used with socat/inetd
- * 9gc - a very simple stdin/stdout 9gridchan client
+A lightweight library for implmeneting 9p clients and servers in C.
-This is all _WIP_ still. 9pex is working in read-only mode so far but
-lacks proper auth, async IO, some more error control etc.
+This is 9p client and server implementation which aims to be correct,
+small and secure, targetting mainly low-resource MCUs.
-To build static binaries with musl:
+`proto.[ch]`: lowest level of the library, contains nothing but
+(de)serealization of the protocol messages and tag (de)allocation.
-```
-CC=musl-gcc CFLAGS="-Os -s -g0 -static" ./build.sh
-```
+`fs.h`: higher level, blocking file API based on POSIX.
-# Why
+`fs9p.[ch]`: blocking file API implementation for 9p - having a
+context created with `proto.[ch]` one can use POSIX-like file API to
+work with a remote filesystem.
-We can do better.
+Goals:
-# Notes/todo
-
- * 9pex: to allow "escaping the root" by following symlinks you have to pass `-e` option
- * 9pex: chrooting with musl makes `realpath` not work as it requires /proc to be mounted, get rid of `realpath`?
- * 9gc: you can run it in termux on Android like so: `rlwrap -a'XXXXXXwth' -s 0 -t dumb 9gc -e NICKNAME`
+* portability: zero external dependencies, no POSIX, just C
+* no dynamic memory allocation
+* security: no crashes due to incoming data
--- a/build.sh
+++ /dev/null
@@ -1,5 +1,0 @@
-#!/bin/sh
-set -e
-set -x
-${CC:-gcc} -std=c99 -fms-extensions -DC9_NO_SERVER -O0 -g -Wall -Wextra -Wshadow $CFLAGS c9/*.c parg/*.c -Ic9 -Iparg 9gc.c -o 9gc || rm -f 9gc
-${CC:-gcc} -std=c99 -fms-extensions -DC9_NO_CLIENT -O0 -g -Wall -Wextra -Wshadow $CFLAGS c9/*.c parg/*.c -Ic9 -Iparg 9pex.c crc32.c -o 9pex || rm -f 9pex
--- a/c9/LICENSE
+++ /dev/null
@@ -1,1 +1,0 @@
-Public domain.
--- a/c9/README.md
+++ /dev/null
@@ -1,8 +1,0 @@
-# c9
-
-Low level 9p client and server.
-
-## Examples
-
-Until I have time to write a minimal example you could take a look at
-https://git.sr.ht/~ft/9pro/blob/master/9pex.c
--- a/c9/c9.c
+++ /dev/null
@@ -1,1176 +1,0 @@
-/*
- * This is 9p client and server implementation which aims to be
- * correct, small and secure. It's the lowest level implementation.
- * It doesn't have much comments, mostly because it doesn't make
- * any sense to copy-paste protocol documentation, which
- * you can read at http://man.cat-v.org/plan_9/5/, see 'intro'.
- */
-#ifdef __plan9__
-#include <u.h>
-#include <libc.h>
-#else
-#include <stdint.h>
-#include <string.h>
-#endif
-#include "c9.h"
-
-enum
-{
- Svver = 1<<0,
-};
-
-#define safestrlen(s) (s == NULL ? 0 : (uint32_t)strlen(s))
-#define maxread(c) (c->msize-4-4-1-2)
-#define maxwrite(c) maxread(c)
-
-static void
-w08(uint8_t **p, uint8_t x)
-{
- (*p)[0] = x;
- *p += 1;
-}
-
-static void
-w16(uint8_t **p, uint16_t x)
-{
- (*p)[0] = x;
- (*p)[1] = x>>8;
- *p += 2;
-}
-
-static void
-w32(uint8_t **p, uint32_t x)
-{
- (*p)[0] = x;
- (*p)[1] = x>>8;
- (*p)[2] = x>>16;
- (*p)[3] = x>>24;
- *p += 4;
-}
-
-static void
-w64(uint8_t **p, uint64_t x)
-{
- (*p)[0] = x;
- (*p)[1] = x>>8;
- (*p)[2] = x>>16;
- (*p)[3] = x>>24;
- (*p)[4] = x>>32;
- (*p)[5] = x>>40;
- (*p)[6] = x>>48;
- (*p)[7] = x>>56;
- *p += 8;
-}
-
-static void
-wcs(uint8_t **p, const char *s, int len)
-{
- w16(p, len);
- if(s != NULL){
- memmove(*p, s, len);
- *p += len;
- }
-}
-
-static uint8_t
-r08(uint8_t **p)
-{
- *p += 1;
- return (*p)[-1];
-}
-
-static uint16_t
-r16(uint8_t **p)
-{
- *p += 2;
- return (*p)[-2]<<0 | (*p)[-1]<<8;
-}
-
-static uint32_t
-r32(uint8_t **p)
-{
- *p += 4;
- return (*p)[-4]<<0 | (*p)[-3]<<8 | (*p)[-2]<<16 | (*p)[-1]<<24;
-}
-
-static uint64_t
-r64(uint8_t **p)
-{
- uint64_t v;
-
- v = r32(p);
- v |= (uint64_t)r32(p)<<32;
- return v;
-}
-
-#ifndef C9_NO_CLIENT
-
-static C9error
-newtag(C9ctx *c, C9ttype type, C9tag *tag)
-{
- uint32_t i;
-
- if(type == Tversion){
- *tag = 0xffff;
- return 0;
- }
-
- if(c->lowfreetag < C9maxtags){
- uint32_t d = c->lowfreetag / C9tagsbits, m = c->lowfreetag % C9tagsbits;
- if((c->tags[d] & 1<<m) != 0){
- c->tags[d] &= ~(1<<m);
- *tag = c->lowfreetag++;
- return 0;
- }
- }
-
- for(i = 0; i < (int)sizeof(c->tags)/sizeof(c->tags[0]); i++){
- uint32_t x, j;
- if((x = c->tags[i]) == 0)
- continue;
- for(j = 0; j < C9tagsbits; j++){
- if((x & (1<<j)) != 0){
- c->tags[i] &= ~(1<<j);
- *tag = i*C9tagsbits + j;
- c->lowfreetag = *tag + 1;
- return 0;
- }
- }
- }
-
- c->error("newtag: no free tags");
- return C9Etag;
-}
-
-static int
-freetag(C9ctx *c, C9tag tag)
-{
- if(tag != 0xffff){
- uint32_t d = tag / C9tagsbits, m = tag % C9tagsbits;
- if(tag >= C9maxtags){
- c->error("freetag: invalid tag %ud", (uint32_t)tag);
- return -1;
- }
- if((c->tags[d] & 1<<m) != 0){
- c->error("freetag: double free for tag %ud", (uint32_t)tag);
- return -1;
- }
- if(c->lowfreetag > tag)
- c->lowfreetag = tag;
- c->tags[d] |= 1<<m;
- }
- return 0;
-}
-
-static uint8_t *
-T(C9ctx *c, uint32_t size, C9ttype type, C9tag *tag, C9error *err)
-{
- uint8_t *p = NULL;
-
- if(size > c->msize-4-1-2){
- c->error("T: invalid size %ud", size);
- *err = C9Esize;
- }else if((*err = newtag(c, type, tag)) == 0){
- size += 4+1+2;
- if((p = c->begin(c, size)) == NULL){
- c->error("T: no buffer for %ud bytes", size);
- freetag(c, *tag);
- *err = C9Ebuf;
- }else{
- *err = 0;
- w32(&p, size);
- w08(&p, type);
- w16(&p, *tag);
- }
- }
- return p;
-}
-
-C9error
-c9version(C9ctx *c, C9tag *tag, uint32_t msize)
-{
- uint8_t *b;
- C9error err;
-
- if(msize < C9minmsize){
- c->error("c9version: msize too small: %ud", msize);
- return C9Einit;
- }
- memset(c->tags, 0xff, sizeof(c->tags));
- memset(c->flush, 0xff, sizeof(c->flush));
- c->lowfreetag = 0;
- c->msize = msize;
-
- if((b = T(c, 4+2+6, Tversion, tag, &err)) != NULL){
- w32(&b, msize);
- wcs(&b, "9P2000", 6);
- err = c->end(c);
- }
- return err;
-}
-
-C9error
-c9auth(C9ctx *c, C9tag *tag, C9fid afid, const char *uname, const char *aname)
-{
- uint8_t *b;
- uint32_t ulen = safestrlen(uname), alen = safestrlen(aname);
- C9error err;
-
- if(ulen > C9maxstr || alen > C9maxstr){
- c->error("c9auth: string too long: %ud chars", ulen > alen ? ulen : alen);
- return C9Estr;
- }
- if((b = T(c, 4+2+ulen+2+alen, Tauth, tag, &err)) != NULL){
- w32(&b, afid);
- wcs(&b, uname, ulen);
- wcs(&b, aname, alen);
- err = c->end(c);
- }
- return err;
-}
-
-C9error
-c9flush(C9ctx *c, C9tag *tag, C9tag oldtag)
-{
- uint8_t *b;
- C9error err;
- int i;
-
- for(i = 0; i < C9maxflush && c->flush[i] != (uint32_t)~0; i++);
- if(i == C9maxflush){
- c->error("c9flush: no free flush slots");
- return C9Eflush;
- }
- if((b = T(c, 2, Tflush, tag, &err)) != NULL){
- w16(&b, oldtag);
- err = c->end(c);
- if(err == 0)
- c->flush[i] = (uint32_t)oldtag<<16 | *tag;
- }
- return err;
-}
-
-C9error
-c9attach(C9ctx *c, C9tag *tag, C9fid fid, C9fid afid, const char *uname, const char *aname)
-{
- uint32_t ulen = safestrlen(uname), alen = safestrlen(aname);
- uint8_t *b;
- C9error err;
-
- if(ulen > C9maxstr || alen > C9maxstr){
- c->error("c9attach: string too long: %ud chars", ulen > alen ? ulen : alen);
- return C9Estr;
- }
- if((b = T(c, 4+4+2+ulen+2+alen, Tattach, tag, &err)) != NULL){
- w32(&b, fid);
- w32(&b, afid);
- wcs(&b, uname, ulen);
- wcs(&b, aname, alen);
- err = c->end(c);
- }
- return err;
-}
-
-C9error
-c9walk(C9ctx *c, C9tag *tag, C9fid fid, C9fid newfid, const char *path[])
-{
- uint32_t i, j, sz;
- uint32_t len[C9maxpathel];
- uint8_t *b;
- C9error err;
-
- for(sz = i = 0; i < (int)sizeof(len)/sizeof(len[0]) && path[i] != NULL; i++){
- len[i] = safestrlen(path[i]);
- if(len[i] == 0 || len[i] > C9maxstr){
- c->error("c9walk: invalid path element: %ud chars", len[i]);
- return C9Epath;
- }
- sz += 2 + len[i];
- }
- if(path[i] != NULL){
- c->error("c9walk: invalid elements !(0 <= %ud <= %ud)", i, C9maxpathel);
- return C9Epath;
- }
-
- if((b = T(c, 4+4+2+sz, Twalk, tag, &err)) != NULL){
- w32(&b, fid);
- w32(&b, newfid);
- w16(&b, i);
- for(j = 0; j < i; j++)
- wcs(&b, path[j], len[j]);
- err = c->end(c);
- }
- return err;
-}
-
-C9error
-c9open(C9ctx *c, C9tag *tag, C9fid fid, C9mode mode)
-{
- uint8_t *b;
- C9error err;
-
- if((b = T(c, 4+1, Topen, tag, &err)) != NULL){
- w32(&b, fid);
- w08(&b, mode);
- err = c->end(c);
- }
- return err;
-}
-
-C9error
-c9create(C9ctx *c, C9tag *tag, C9fid fid, const char *name, uint32_t perm, C9mode mode)
-{
- uint32_t nlen = strlen(name);
- uint8_t *b;
- C9error err;
-
- if(nlen == 0 || nlen > C9maxstr){
- c->error("c9create: invalid name: %ud chars", nlen);
- return C9Epath;
- }
- if((b = T(c, 4+2+nlen+4+1, Tcreate, tag, &err)) != NULL){
- w32(&b, fid);
- wcs(&b, name, nlen);
- w32(&b, perm);
- w08(&b, mode);
- err = c->end(c);
- }
- return err;
-}
-
-C9error
-c9read(C9ctx *c, C9tag *tag, C9fid fid, uint64_t offset, uint32_t count)
-{
- uint8_t *b;
- C9error err;
-
- if((b = T(c, 4+8+4, Tread, tag, &err)) != NULL){
- w32(&b, fid);
- w64(&b, offset);
- w32(&b, count);
- err = c->end(c);
- }
- return err;
-}
-
-C9error
-c9write(C9ctx *c, C9tag *tag, C9fid fid, uint64_t offset, const void *in, uint32_t count)
-{
- uint8_t *b;
- C9error err;
-
- if((b = T(c, 4+8+4+count, Twrite, tag, &err)) != NULL){
- w32(&b, fid);
- w64(&b, offset);
- w32(&b, count);
- memmove(b, in, count);
- err = c->end(c);
- }
- return err;
-}
-
-C9error
-c9wrstr(C9ctx *c, C9tag *tag, C9fid fid, const char *s)
-{
- return c9write(c, tag, fid, 0, s, strlen(s));
-}
-
-C9error
-c9clunk(C9ctx *c, C9tag *tag, C9fid fid)
-{
- uint8_t *b;
- C9error err;
-
- if((b = T(c, 4, Tclunk, tag, &err)) != NULL){
- w32(&b, fid);
- err = c->end(c);
- }
- return err;
-}
-
-C9error
-c9remove(C9ctx *c, C9tag *tag, C9fid fid)
-{
- uint8_t *b;
- C9error err;
-
- if((b = T(c, 4, Tremove, tag, &err)) != NULL){
- w32(&b, fid);
- err = c->end(c);
- }
- return err;
-}
-
-C9error
-c9stat(C9ctx *c, C9tag *tag, C9fid fid)
-{
- uint8_t *b;
- C9error err;
-
- if((b = T(c, 4, Tstat, tag, &err)) != NULL){
- w32(&b, fid);
- err = c->end(c);
- }
- return err;
-}
-
-C9error
-c9wstat(C9ctx *c, C9tag *tag, C9fid fid, const C9stat *s)
-{
- uint32_t nlen = safestrlen(s->name), ulen = safestrlen(s->uid), glen = safestrlen(s->gid);
- uint32_t unusedsz = 2+4+13, statsz = unusedsz+4+4+4+8+2+nlen+2+ulen+2+glen+2;
- uint8_t *b;
- C9error err;
-
- if(nlen == 0 || nlen > C9maxstr){
- c->error("c9wstat: invalid name: %ud chars", nlen);
- return C9Epath;
- }
- if(ulen > C9maxstr || glen > C9maxstr){
- c->error("c9wstat: string too long: %ud chars", ulen > glen ? ulen : glen);
- return C9Estr;
- }
- if((b = T(c, 4+2+2+statsz, Twstat, tag, &err)) != NULL){
- w32(&b, fid);
- w16(&b, statsz+2);
- w16(&b, statsz);
- memset(b, 0xff, unusedsz); /* leave type(2), dev(4) and qid(13) unchanged */
- b += unusedsz;
- w32(&b, s->mode);
- w32(&b, s->atime);
- w32(&b, s->mtime);
- w64(&b, s->size);
- wcs(&b, s->name, nlen);
- wcs(&b, s->uid, ulen);
- wcs(&b, s->gid, glen);
- wcs(&b, NULL, 0); /* muid unchanged */
- err = c->end(c);
- }
- return err;
-}
-
-C9error
-c9proc(C9ctx *c)
-{
- uint32_t i, sz, cnt, msize;
- uint8_t *b;
- int err;
- C9r r;
-
- err = -1;
- if((b = c->read(c, 4, &err)) == NULL){
- if(err != 0)
- c->error("c9proc: short read");
- return err == 0 ? 0 : C9Epkt;
- }
-
- sz = r32(&b);
- if(sz < 7 || sz > c->msize){
- c->error("c9proc: invalid packet size !(7 <= %ud <= %ud)", sz, c->msize);
- return C9Epkt;
- }
- sz -= 4;
- err = -1;
- if((b = c->read(c, sz, &err)) == NULL){
- if(err != 0)
- c->error("c9proc: short read");
- return err == 0 ? 0 : C9Epkt;
- }
-
- r.type = r08(&b);
- r.tag = r16(&b);
- if(r.type != Rversion){
- if(r.tag >= C9maxtags){
- c->error("c9proc: invalid tag %ud", (uint32_t)r.tag);
- return C9Epkt;
- }
- if(freetag(c, r.tag) != 0)
- return C9Etag;
- }
- sz -= 3;
- r.numqid = 0;
-
- switch(r.type){
- case Rread:
- if(sz < 4 || (cnt = r32(&b)) > sz-4)
- goto error;
- r.read.data = b;
- r.read.size = cnt;
- c->r(c, &r);
- break;
-
- case Rwrite:
- if(sz < 4 || (cnt = r32(&b)) > c->msize)
- goto error;
- r.write.size = cnt;
- c->r(c, &r);
- break;
-
- case Rwalk:
- if(sz < 2 || (cnt = r16(&b))*13 > sz-2)
- goto error;
- if(cnt > C9maxpathel){
- c->error("c9proc: Rwalk !(%ud <= %ud)", cnt, C9maxpathel);
- return C9Epath;
- }
- for(i = 0; i < cnt; i++){
- r.qid[i].type = r08(&b);
- r.qid[i].version = r32(&b);
- r.qid[i].path = r64(&b);
- }
- r.numqid = cnt;
- c->r(c, &r);
- break;
-
- case Rstat:
- b += 2; sz -= 2;
- if((err = c9parsedir(c, &r.stat, &b, &sz)) != 0)
- return err;
- r.numqid = 1;
- c->r(c, &r);
- break;
-
- case Rflush:
- for(i = 0; i < C9maxflush; i++){
- if((c->flush[i] & 0xffff) == r.tag){
- freetag(c, c->flush[i]>>16);
- c->flush[i] = 0xffffffff;
- break;
- }
- }
- /* fallthrough */
- case Rclunk:
- case Rremove:
- case Rwstat:
- c->r(c, &r);
- break;
-
- case Ropen:
- case Rcreate:
- if(sz < 17)
- goto error;
- r.qid[0].type = r08(&b);
- r.qid[0].version = r32(&b);
- r.qid[0].path = r64(&b);
- r.iounit = r32(&b);
- r.numqid = 1;
- c->r(c, &r);
- break;
-
- case Rerror:
- if(sz < 2 || (cnt = r16(&b)) > sz-2)
- goto error;
- r.error = memmove(b-1, b, cnt);
- r.error[cnt] = 0;
- c->r(c, &r);
- break;
-
- case Rauth:
- case Rattach:
- if(sz < 13)
- goto error;
- r.qid[0].type = r08(&b);
- r.qid[0].version = r32(&b);
- r.qid[0].path = r64(&b);
- r.numqid = 1;
- c->r(c, &r);
- break;
-
- case Rversion:
- if(sz < 4+2 || (msize = r32(&b)) < C9minmsize || (cnt = r16(&b)) > sz-4-2)
- goto error;
- if(cnt < 6 || memcmp(b, "9P2000", 6) != 0){
- c->error("invalid version");
- return C9Ever;
- }
- if(msize < c->msize)
- c->msize = msize;
- c->r(c, &r);
- break;
-
- default:
- goto error;
- }
- return 0;
-error:
- c->error("c9proc: invalid packet type %ud", r.type);
- return C9Epkt;
-}
-
-#endif /* C9_NO_CLIENT */
-
-C9error
-c9parsedir(C9ctx *c, C9stat *stat, uint8_t **t, uint32_t *size)
-{
- uint8_t *b;
- uint32_t cnt, sz;
-
- sz = 0;
- if(*size < 49 || (sz = r16(t)) < 47 || *size < 2+sz)
- goto error;
- *size -= 2+sz;
- *t += 6; /* skip type(2) and dev(4) */
- stat->qid.type = r08(t);
- stat->qid.version = r32(t);
- stat->qid.path = r64(t);
- stat->mode = r32(t);
- stat->atime = r32(t);
- stat->mtime = r32(t);
- stat->size = r64(t);
- sz -= 39;
- if((cnt = r16(t)) > sz-2)
- goto error;
- stat->name = (char*)*t; b = *t = *t+cnt; sz -= 2+cnt;
- if(sz < 2 || (cnt = r16(t)) > sz-2)
- goto error;
- stat->uid = (char*)*t; *b = 0; b = *t = *t+cnt; sz -= 2+cnt;
- if(sz < 2 || (cnt = r16(t)) > sz-2)
- goto error;
- stat->gid = (char*)*t; *b = 0; b = *t = *t+cnt; sz -= 2+cnt;
- if(sz < 2 || (cnt = r16(t)) > sz-2)
- goto error;
- stat->muid = memmove(*t-1, *t, cnt); *b = stat->muid[cnt] = 0; *t = *t+cnt; sz -= 2+cnt;
- *t += sz;
- return 0;
-error:
- c->error("c9parsedir: invalid size: size=%ud sz=%ud", *size, sz);
- return C9Epkt;
-}
-
-#ifndef C9_NO_SERVER
-
-static uint8_t *
-R(C9ctx *c, uint32_t size, C9rtype type, C9tag tag, C9error *err)
-{
- uint8_t *p = NULL;
-
- if(size > c->msize-4-1-2){
- c->error("R: invalid size %ud", size);
- *err = C9Esize;
- }else{
- size += 4+1+2;
- if((p = c->begin(c, size)) == NULL){
- c->error("R: no buffer for %ud bytes", size);
- *err = C9Ebuf;
- }else{
- *err = 0;
- w32(&p, size);
- w08(&p, type);
- w16(&p, tag);
- }
- }
- return p;
-}
-
-C9error
-s9version(C9ctx *c)
-{
- uint8_t *b;
- C9error err;
-
- if((b = R(c, 4+2+6, Rversion, 0xffff, &err)) != NULL){
- w32(&b, c->msize);
- wcs(&b, "9P2000", 6);
- err = c->end(c);
- };
- return err;
-}
-
-C9error
-s9auth(C9ctx *c, C9tag tag, const C9qid *aqid)
-{
- uint8_t *b;
- C9error err;
-
- if((b = R(c, 13, Rauth, tag, &err)) != NULL){
- w08(&b, aqid->type);
- w32(&b, aqid->version);
- w64(&b, aqid->path);
- err = c->end(c);
- }
- return err;
-}
-
-C9error
-s9error(C9ctx *c, C9tag tag, const char *ename)
-{
- uint32_t len = safestrlen(ename);
- uint8_t *b;
- C9error err;
-
- if(len > C9maxstr){
- c->error("s9error: invalid ename: %ud chars", len);
- return C9Estr;
- }
- if((b = R(c, 2+len, Rerror, tag, &err)) != NULL){
- wcs(&b, ename, len);
- err = c->end(c);
- }
- return err;
-}
-
-C9error
-s9attach(C9ctx *c, C9tag tag, const C9qid *qid)
-{
- uint8_t *b;
- C9error err;
-
- if((b = R(c, 13, Rattach, tag, &err)) != NULL){
- w08(&b, qid->type);
- w32(&b, qid->version);
- w64(&b, qid->path);
- err = c->end(c);
- }
- return err;
-}
-
-C9error
-s9flush(C9ctx *c, C9tag tag)
-{
- C9error err;
-
- if(R(c, 0, Rflush, tag, &err) != NULL)
- err = c->end(c);
- return err;
-}
-
-C9error
-s9walk(C9ctx *c, C9tag tag, C9qid *qids[])
-{
- uint32_t i, n;
- uint8_t *b;
- C9error err;
-
- for(n = 0; n < C9maxpathel && qids[n] != NULL; n++);
- if(n > C9maxpathel){
- c->error("s9walk: invalid elements !(0 <= %ud <= %ud)", n, C9maxpathel);
- return C9Epath;
- }
-
- if((b = R(c, 2+n*13, Rwalk, tag, &err)) != NULL){
- w16(&b, n);
- for(i = 0; i < n; i++){
- w08(&b, qids[i]->type);
- w32(&b, qids[i]->version);
- w64(&b, qids[i]->path);
- }
- err = c->end(c);
- }
- return err;
-}
-
-C9error
-s9open(C9ctx *c, C9tag tag, const C9qid *qid, uint32_t iounit)
-{
- uint8_t *b;
- C9error err;
-
- if((b = R(c, 13+4, Ropen, tag, &err)) != NULL){
- w08(&b, qid->type);
- w32(&b, qid->version);
- w64(&b, qid->path);
- w32(&b, iounit);
- err = c->end(c);
- }
- return err;
-}
-
-C9error
-s9create(C9ctx *c, C9tag tag, const C9qid *qid, uint32_t iounit)
-{
- uint8_t *b;
- C9error err;
-
- if((b = R(c, 13+4, Rcreate, tag, &err)) != NULL){
- w08(&b, qid->type);
- w32(&b, qid->version);
- w64(&b, qid->path);
- w32(&b, iounit);
- err = c->end(c);
- }
- return err;
-}
-
-C9error
-s9read(C9ctx *c, C9tag tag, const void *data, uint32_t size)
-{
- uint8_t *b;
- C9error err;
-
- if((b = R(c, 4+size, Rread, tag, &err)) != NULL){
- w32(&b, size);
- memmove(b, data, size);
- err = c->end(c);
- }
- return err;
-}
-
-C9error
-s9write(C9ctx *c, C9tag tag, uint32_t size)
-{
- uint8_t *b;
- C9error err;
-
- if((b = R(c, 4, Rwrite, tag, &err)) != NULL){
- w32(&b, size);
- err = c->end(c);
- }
- return err;
-}
-
-C9error
-s9readdir(C9ctx *c, C9tag tag, C9stat *st[], int *num, uint64_t *offset, uint32_t size)
-{
- uint8_t *b;
- const C9stat *s;
- uint32_t nlen, ulen, glen, mulen, m, n;
- C9error err;
- int i;
-
- if(size > c->msize-4-1-2)
- size = c->msize-4-1-2;
-
- m = 0;
- for(i = 0; i < *num; i++){
- s = st[i];
- nlen = safestrlen(s->name);
- ulen = safestrlen(s->uid);
- glen = safestrlen(s->gid);
- mulen = safestrlen(s->muid);
-
- if(nlen == 0 || nlen > C9maxstr){
- c->error("s9readdir: invalid name: %ud chars", nlen);
- return C9Epath;
- }
- if(ulen > C9maxstr || glen > C9maxstr || mulen > C9maxstr){
- ulen = ulen > glen ? ulen : glen;
- ulen = ulen > mulen ? ulen : mulen;
- c->error("s9readdir: string too long: %ud chars", ulen);
- return C9Estr;
- }
-
- n = 2 + 2+4+13+4+4+4+8+2+nlen+2+ulen+2+glen+2+mulen;
- if(4+m+n > size)
- break;
- m += n;
- }
-
- if((b = R(c, 4+m, Rread, tag, &err)) != NULL){
- *num = i;
- w32(&b, m);
- for(i = 0; i < *num; i++){
- s = st[i];
- nlen = safestrlen(s->name);
- ulen = safestrlen(s->uid);
- glen = safestrlen(s->gid);
- mulen = safestrlen(s->muid);
- w16(&b, 2+4+13+4+4+4+8+2+nlen+2+ulen+2+glen+2+mulen);
- w16(&b, 0xffff); /* type */
- w32(&b, 0xffffffff); /* dev */
- w08(&b, s->qid.type);
- w32(&b, s->qid.version);
- w64(&b, s->qid.path);
- w32(&b, s->mode);
- w32(&b, s->atime);
- w32(&b, s->mtime);
- w64(&b, s->size);
- wcs(&b, s->name, nlen);
- wcs(&b, s->uid, ulen);
- wcs(&b, s->gid, glen);
- wcs(&b, s->muid, mulen);
- }
- err = c->end(c);
- if(err == 0)
- *offset += m;
- }
- return err;
-}
-
-C9error
-s9clunk(C9ctx *c, C9tag tag)
-{
- C9error err;
-
- if(R(c, 0, Rclunk, tag, &err) != NULL)
- err = c->end(c);
- return err;
-}
-
-C9error
-s9remove(C9ctx *c, C9tag tag)
-{
- C9error err;
-
- if(R(c, 0, Rremove, tag, &err) != NULL)
- err = c->end(c);
- return err;
-}
-
-C9error
-s9stat(C9ctx *c, C9tag tag, const C9stat *s)
-{
- uint32_t nlen = safestrlen(s->name), ulen = safestrlen(s->uid);
- uint32_t glen = safestrlen(s->gid), mulen = safestrlen(s->name);
- uint32_t statsz = 2+4+13+4+4+4+8+2+nlen+2+ulen+2+glen+2+mulen;
- uint8_t *b;
- C9error err;
-
- if(nlen == 0 || nlen > C9maxstr){
- c->error("s9stat: invalid name: %ud chars", nlen);
- return C9Epath;
- }
- if(ulen > C9maxstr || glen > C9maxstr || mulen > C9maxstr){
- ulen = ulen > glen ? ulen : glen;
- ulen = ulen > mulen ? ulen : mulen;
- c->error("s9stat: string too long: %ud chars", ulen);
- return C9Estr;
- }
-
- if((b = R(c, 2+2+statsz, Rstat, tag, &err)) != NULL){
- w16(&b, statsz+2);
- w16(&b, statsz);
- w16(&b, 0xffff); /* type */
- w32(&b, 0xffffffff); /* dev */
- w08(&b, s->qid.type);
- w32(&b, s->qid.version);
- w64(&b, s->qid.path);
- w32(&b, s->mode);
- w32(&b, s->atime);
- w32(&b, s->mtime);
- w64(&b, s->size);
- wcs(&b, s->name, nlen);
- wcs(&b, s->uid, ulen);
- wcs(&b, s->gid, glen);
- wcs(&b, s->muid, mulen);
- err = c->end(c);
- }
- return err;
-}
-
-C9error
-s9wstat(C9ctx *c, C9tag tag)
-{
- C9error err;
-
- if(R(c, 0, Rwstat, tag, &err) != NULL)
- err = c->end(c);
- return err;
-}
-
-C9error
-s9proc(C9ctx *c)
-{
- uint32_t i, sz, cnt, n, msize;
- int readerr;
- uint8_t *b;
- C9error err;
- C9t t;
-
- readerr = -1;
- if((b = c->read(c, 4, &readerr)) == NULL){
- if(readerr != 0)
- c->error("s9proc: short read");
- return readerr == 0 ? 0 : C9Epkt;
- }
-
- sz = r32(&b);
- if(sz < 7 || sz > c->msize){
- c->error("s9proc: invalid packet size !(7 <= %ud <= %ud)", sz, c->msize);
- return C9Epkt;
- }
- sz -= 4;
- readerr = -1;
- if((b = c->read(c, sz, &readerr)) == NULL){
- if(readerr != 0)
- c->error("s9proc: short read");
- return readerr == 0 ? 0 : C9Epkt;
- }
-
- t.type = r08(&b);
- t.tag = r16(&b);
- sz -= 3;
-
- if((c->svflags & Svver) == 0 && t.type != Tversion){
- c->error("s9proc: expected Tversion, got %ud", t.type);
- return C9Epkt;
- }
-
- switch(t.type){
- case Tread:
- if(sz < 4+8+4)
- goto error;
- t.fid = r32(&b);
- t.read.offset = r64(&b);
- t.read.size = r32(&b);
- if(t.read.size > maxread(c))
- t.read.size = maxread(c);
- c->t(c, &t);
- break;
-
- case Twrite:
- if(sz < 4+8+4)
- goto error;
- t.fid = r32(&b);
- t.write.offset = r64(&b);
- if((t.write.size = r32(&b)) < sz-4-8-4)
- goto error;
- if(t.write.size > maxwrite(c))
- t.write.size = maxwrite(c);
- t.write.data = b;
- c->t(c, &t);
- break;
-
- case Tclunk:
- case Tstat:
- case Tremove:
- if(sz < 4)
- goto error;
- t.fid = r32(&b);
- c->t(c, &t);
- break;
-
- case Twalk:
- if(sz < 4+4+2)
- goto error;
- t.fid = r32(&b);
- t.walk.newfid = r32(&b);
- if((n = r16(&b)) > 16){
- c->error("s9proc: Twalk !(%ud <= 16)", n);
- return C9Epath;
- }
- sz -= 4+4+2;
- if(n > 0){
- for(i = 0; i < n; i++){
- if(sz < 2 || (cnt = r16(&b)) > sz-2)
- goto error;
- if(cnt < 1){
- c->error("s9proc: Twalk invalid element [%ud]", i);
- return C9Epath;
- }
- b[-2] = 0;
- t.walk.wname[i] = (char*)b;
- b += cnt;
- sz -= 2 + cnt;
- }
- memmove(t.walk.wname[i-1]-1, t.walk.wname[i-1], (char*)b - t.walk.wname[i-1]);
- t.walk.wname[i-1]--;
- b[-1] = 0;
- }else
- i = 0;
- t.walk.wname[i] = NULL;
- c->t(c, &t);
- break;
-
- case Topen:
- if(sz < 4+1)
- goto error;
- t.fid = r32(&b);
- t.open.mode = r08(&b);
- c->t(c, &t);
- break;
-
- case Twstat:
- if(sz < 4+2)
- goto error;
- t.fid = r32(&b);
- if((cnt = r16(&b)) > sz-4)
- goto error;
- if((err = c9parsedir(c, &t.wstat, &b, &cnt)) != 0){
- c->error("s9proc");
- return err;
- }
- c->t(c, &t);
- break;
-
- case Tcreate:
- if(sz < 4+2+4+1)
- goto error;
- t.fid = r32(&b);
- if((cnt = r16(&b)) < 1 || cnt > sz-4-2-4-1)
- goto error;
- t.create.name = (char*)b;
- b += cnt;
- t.create.perm = r32(&b);
- t.create.mode = r08(&b);
- t.create.name[cnt] = 0;
- c->t(c, &t);
- break;
-
- case Tflush:
- if(sz < 2)
- goto error;
- t.flush.oldtag = r16(&b);
- c->t(c, &t);
- break;
-
- case Tversion:
- if(sz < 4+2 || (msize = r32(&b)) < C9minmsize || (cnt = r16(&b)) > sz-4-2)
- goto error;
- if(cnt < 6 || memcmp(b, "9P2000", 6) != 0){
- if((b = R(c, 4+2+7, Rversion, 0xffff, &err)) != NULL){
- w32(&b, 0);
- wcs(&b, "unknown", 7);
- err = c->end(c);
- c->error("s9proc: invalid version");
- }
- return C9Ever;
- }
- if(msize < c->msize)
- c->msize = msize;
- c->svflags |= Svver;
- c->t(c, &t);
- break;
-
- case Tattach:
- if(sz < 4+4+2+2)
- goto error;
- t.fid = r32(&b);
- t.attach.afid = r32(&b);
- cnt = r16(&b);
- sz -= 4+4+2;
- if(cnt+2 > sz)
- goto error;
- t.attach.uname = (char*)b;
- b += cnt;
- cnt = r16(&b);
- b[-2] = 0;
- sz -= cnt+2;
- if(cnt > sz)
- goto error;
- memmove(b-1, b, cnt);
- t.attach.aname = (char*)b-1;
- t.attach.aname[cnt] = 0;
- c->t(c, &t);
- break;
-
- case Tauth:
- if(sz < 4+2+2)
- goto error;
- t.auth.afid = r32(&b);
- cnt = r16(&b);
- sz -= 4+2;
- if(cnt+2 > sz)
- goto error;
- t.auth.uname = (char*)b;
- b += cnt;
- cnt = r16(&b);
- b[-2] = 0;
- sz -= cnt+2;
- if(cnt > sz)
- goto error;
- memmove(b-1, b, cnt);
- t.auth.aname = (char*)b-1;
- t.auth.aname[cnt] = 0;
- c->t(c, &t);
- break;
-
- default:
- goto error;
- }
- return 0;
-error:
- c->error("s9proc: invalid packet (type=%ud)", t.type);
- return C9Epkt;
-}
-
-#endif /* C9_NO_SERVER */
--- a/c9/c9.h
+++ /dev/null
@@ -1,365 +1,0 @@
-#ifdef __plan9__
-typedef u64int uint64_t;
-typedef u32int uint32_t;
-typedef u16int uint16_t;
-typedef u8int uint8_t;
-#define __attribute__(...)
-#define NULL nil
-#endif
-
-struct C9aux;
-
-typedef struct C9r C9r;
-typedef struct C9t C9t;
-typedef struct C9stat C9stat;
-typedef struct C9ctx C9ctx;
-typedef struct C9qid C9qid;
-typedef uint32_t C9fid;
-typedef uint16_t C9tag;
-
-/* Stat field is not changed if it's set to this value when calling c9wstat. */
-#define C9nochange (~0)
-
-/* Special fid used with auth/attach to basically avoid authentication. */
-#define C9nofid ((C9fid)~0)
-
-/* C9modes for opening a file. */
-typedef enum
-{
- C9read = 0,
- C9write = 1,
- C9rdwr = 2,
- C9exec = 3,
- C9trunc = 0x10,
- C9rclose = 0x40,
- C9excl = 0x1000,
-}C9mode;
-
-typedef enum
-{
- /* User/owner. */
- C9permur = 1<<8, /* Readable. */
- C9permuw = 1<<7, /* Writable. */
- C9permux = 1<<6, /* Executable. */
-
- /* Group. */
- C9permgr = 1<<5,
- C9permgw = 1<<4,
- C9permgx = 1<<3,
-
- /* Other. */
- C9permor = 1<<2,
- C9permow = 1<<1,
- C9permox = 1<<0,
-}C9perm;
-
-/* Directory. */
-#define C9permdir 0x80000000
-
-/* Bitmask of stat.mode. */
-#define C9stdir 0x80000000
-#define C9stappend 0x40000000
-#define C9stexcl 0x20000000
-#define C9sttmp 0x04000000
-
-/* Limits. */
-enum
-{
- C9maxtags = 64, /* Maximal number of outstanding requests. [1-65535] */
- C9maxflush = 8, /* Maximal number of outstanding flushes. [1-65535] */
- C9maxstr = 0xffff, /* Maximal string length. [1-65535] */
- C9minmsize = 4096, /* Minimal sane msize. [4096-...] */
- C9maxpathel = 16, /* Maximal number of elements in a path. Do not change. */
-};
-
-/* Errors. */
-typedef enum
-{
- C9Einit = -1, /* Initialization failed. */
- C9Ever = -2, /* Protocol version doesn't match. */
- C9Epkt = -3, /* Incoming packet error. */
- C9Etag = -4, /* No free tags or bad tag. */
- C9Ebuf = -5, /* No buffer space enough for a message. */
- C9Epath = -6, /* Path is too long or just invalid. */
- C9Eflush = -7, /* Limit of outstanding flushes reached. */
- C9Esize = -8, /* Can't fit data in one message. */
- C9Estr = -9 /* Bad string. */
-}C9error;
-
-/* Request types. */
-typedef enum
-{
- Tversion = 100,
- Tauth = 102,
- Tattach = 104,
- Tflush = 108,
- Twalk = 110,
- Topen = 112,
- Tcreate = 114,
- Tread = 116,
- Twrite = 118,
- Tclunk = 120,
- Tremove = 122,
- Tstat = 124,
- Twstat = 126
-}C9ttype;
-
-/* Response types. */
-typedef enum
-{
- Rversion = 101,
- Rauth = 103,
- Rattach = 105,
- Rerror = 107,
- Rflush = 109,
- Rwalk = 111,
- Ropen = 113,
- Rcreate = 115,
- Rread = 117,
- Rwrite = 119,
- Rclunk = 121,
- Rremove = 123,
- Rstat = 125,
- Rwstat = 127
-}C9rtype;
-
-/* Unique file id type. */
-typedef enum
-{
- C9qtdir = 1<<7,
- C9qtappend = 1<<6,
- C9qtexcl = 1<<5,
- C9qtauth = 1<<3,
- C9qttmp = 1<<2,
- C9qtfile = 0
-}C9qt;
-
-/* Unique file id. */
-struct C9qid
-{
- uint64_t path;
- uint32_t version;
- C9qt type;
-};
-
-/*
- * File stats. Version and muid are ignored on wstat. Dmdir bit
- * change in mode won't work on wstat. Set any integer field to
- * C9nochange to keep it unchanged on wstat. Set any string to NULL to
- * keep it unchanged. Strings can be empty (""), but never NULL after
- * stat call.
- */
-struct C9stat
-{
- uint64_t size; /* Size of the file (in bytes). */
- char *name; /* Name of the file. */
- char *uid; /* Owner of the file. */
- char *gid; /* Group of the file. */
- char *muid; /* The user who modified the file last. */
- C9qid qid; /* Same as qid[0]. */
- uint32_t mode; /* Permissions. See C9st* and C9perm. */
- uint32_t atime; /* Last access time. */
- uint32_t mtime; /* Last modification time. */
-};
-
-/* Response data. */
-struct C9r
-{
- union
- {
- char *error;
-
- struct
- {
- uint8_t *data;
- uint32_t size;
- }read;
-
- struct
- {
- uint32_t size;
- }write;
-
- /* File stats (only valid if type is Rstat). */
- C9stat stat;
-
- /*
- * Qid(s). qid[0] is valid for auth/attach/create/stat/open.
- * More ids may be a result of a walk, see numqid.
- */
- C9qid qid[C9maxpathel];
- };
- C9rtype type; /* Response type. */
-
- /*
- * If not zero, is the maximum number of bytes that are guaranteed
- * to be read or written atomically, without breaking into multiple
- * messages.
- */
- uint32_t iounit;
-
- int numqid; /* Number of valid unique ids in qid array. */
- C9tag tag; /* Tag number. */
-};
-
-/* Request data. */
-struct C9t
-{
- C9ttype type;
- C9tag tag;
- union
- {
- struct
- {
- char *uname;
- char *aname;
- C9fid afid;
- }attach;
-
- struct
- {
- char *uname;
- char *aname;
- C9fid afid;
- }auth;
-
- struct
- {
- char *name;
- uint32_t perm;
- C9mode mode;
- }create;
-
- struct
- {
- C9tag oldtag;
- }flush;
-
- struct
- {
- C9mode mode;
- }open;
-
- struct
- {
- uint64_t offset;
- uint32_t size;
- }read;
-
- struct
- {
- char *wname[C9maxpathel+1]; /* wname[16] is always NULL */
- C9fid newfid;
- }walk;
-
- struct
- {
- uint64_t offset;
- uint8_t *data;
- uint32_t size;
- }write;
-
- C9stat wstat;
- };
- C9fid fid;
-};
-
-enum
-{
- C9tagsbits = sizeof(uint32_t) * 8,
-};
-
-struct C9ctx
-{
- /*
- * Should return a pointer to the data (exactly 'size' bytes) read.
- * Set 'err' to non-zero and return NULL in case of error.
- * 'err' set to zero (no error) should be used to return from c9process
- * early (timeout on read to do non-blocking operations, for example).
- */
- uint8_t *(*read)(C9ctx *ctx, uint32_t size, int *err) __attribute__((nonnull(1, 3)));
-
- /* Should return a buffer to store 'size' bytes. Nil means no memory. */
- uint8_t *(*begin)(C9ctx *ctx, uint32_t size) __attribute__((nonnull(1)));
-
- /*
- * Marks the end of a message. Callback may decide if any accumulated
- * messages should be sent to the server/client.
- */
- int (*end)(C9ctx *ctx) __attribute__((nonnull(1)));
-
- /* Callback called every time a new R-message is received. */
- void (*r)(C9ctx *ctx, C9r *r) __attribute__((nonnull(1, 2)));
-
- /* Callback called every time a new T-message is received. */
- void (*t)(C9ctx *ctx, C9t *t) __attribute__((nonnull(1, 2)));
-
- /* Callback for error messages. */
- void (*error)(const char *fmt, ...) __attribute__((nonnull(1), format(printf, 1, 2)));
-
- /* Auxiliary data, can be used by any of above callbacks. */
- struct C9aux *aux;
-
- /* private stuff */
- uint32_t msize;
-#ifndef C9_NO_CLIENT
- uint32_t flush[C9maxflush];
- uint32_t tags[C9maxtags/C9tagsbits];
-#endif
- union
- {
- C9tag lowfreetag;
- uint16_t svflags;
- };
-};
-
-/* Parse one directory entry. */
-extern C9error c9parsedir(C9ctx *c, C9stat *stat, uint8_t **data, uint32_t *size) __attribute__((nonnull(1, 2, 3)));
-
-#ifndef C9_NO_CLIENT
-
-extern C9error c9version(C9ctx *c, C9tag *tag, uint32_t msize) __attribute__((nonnull(1, 2)));
-extern C9error c9auth(C9ctx *c, C9tag *tag, C9fid afid, const char *uname, const char *aname) __attribute__((nonnull(1, 2)));
-extern C9error c9flush(C9ctx *c, C9tag *tag, C9tag oldtag) __attribute__((nonnull(1, 2)));
-extern C9error c9attach(C9ctx *c, C9tag *tag, C9fid fid, C9fid afid, const char *uname, const char *aname) __attribute__((nonnull(1, 2)));
-extern C9error c9walk(C9ctx *c, C9tag *tag, C9fid fid, C9fid newfid, const char *path[]) __attribute__((nonnull(1, 2, 5)));
-extern C9error c9open(C9ctx *c, C9tag *tag, C9fid fid, C9mode mode) __attribute__((nonnull(1, 2)));
-extern C9error c9create(C9ctx *c, C9tag *tag, C9fid fid, const char *name, uint32_t perm, C9mode mode) __attribute__((nonnull(1, 2, 4)));
-extern C9error c9read(C9ctx *c, C9tag *tag, C9fid fid, uint64_t offset, uint32_t count) __attribute__((nonnull(1, 2)));
-extern C9error c9write(C9ctx *c, C9tag *tag, C9fid fid, uint64_t offset, const void *in, uint32_t count) __attribute__((nonnull(1, 2, 5)));
-extern C9error c9wrstr(C9ctx *c, C9tag *tag, C9fid fid, const char *s) __attribute__((nonnull(1, 2, 4)));
-extern C9error c9clunk(C9ctx *c, C9tag *tag, C9fid fid) __attribute__((nonnull(1, 2)));
-extern C9error c9remove(C9ctx *c, C9tag *tag, C9fid fid) __attribute__((nonnull(1, 2)));
-extern C9error c9stat(C9ctx *c, C9tag *tag, C9fid fid) __attribute__((nonnull(1, 2)));
-extern C9error c9wstat(C9ctx *c, C9tag *tag, C9fid fid, const C9stat *s) __attribute__((nonnull(1, 2, 4)));
-
-/*
- * Wait until a response comes and process it. If the function returns
- * any error, context must be treated as 'broken' and no subsequent calls
- * should be made without reinitialization (c9version).
- */
-extern C9error c9proc(C9ctx *c) __attribute__((nonnull(1)));
-
-#endif /* C9_NO_CLIENT */
-
-#ifndef C9_NO_SERVER
-
-extern C9error s9version(C9ctx *c) __attribute__((nonnull(1)));
-extern C9error s9auth(C9ctx *c, C9tag tag, const C9qid *aqid) __attribute__((nonnull(1, 3)));
-extern C9error s9error(C9ctx *c, C9tag tag, const char *err) __attribute__((nonnull(1)));
-extern C9error s9attach(C9ctx *c, C9tag tag, const C9qid *qid) __attribute__((nonnull(1, 3)));
-extern C9error s9flush(C9ctx *c, C9tag tag) __attribute__((nonnull(1)));
-extern C9error s9walk(C9ctx *c, C9tag tag, C9qid *qids[]) __attribute__((nonnull(1, 3)));
-extern C9error s9open(C9ctx *c, C9tag tag, const C9qid *qid, uint32_t iounit) __attribute__((nonnull(1, 3)));
-extern C9error s9create(C9ctx *c, C9tag tag, const C9qid *qid, uint32_t iounit) __attribute__((nonnull(1, 3)));
-extern C9error s9read(C9ctx *c, C9tag tag, const void *data, uint32_t size) __attribute__((nonnull(1, 3)));
-extern C9error s9readdir(C9ctx *c, C9tag tag, C9stat *st[], int *num, uint64_t *offset, uint32_t size) __attribute__((nonnull(1, 3, 4)));
-extern C9error s9write(C9ctx *c, C9tag tag, uint32_t size) __attribute__((nonnull(1)));
-extern C9error s9clunk(C9ctx *c, C9tag tag) __attribute__((nonnull(1)));
-extern C9error s9remove(C9ctx *c, C9tag tag) __attribute__((nonnull(1)));
-extern C9error s9stat(C9ctx *c, C9tag tag, const C9stat *s) __attribute__((nonnull(1, 3)));
-extern C9error s9wstat(C9ctx *c, C9tag tag) __attribute__((nonnull(1)));
-
-extern C9error s9proc(C9ctx *c) __attribute__((nonnull(1)));
-
-#endif /* C9_NO_SERVER */
--- a/crc32.c
+++ /dev/null
@@ -1,52 +1,0 @@
-#define _DEFAULT_SOURCE
-#include <stdint.h>
-
-static uint32_t table[256] = {
- 0x00000000, 0x77073096, 0xee0e612c, 0x990951ba, 0x076dc419, 0x706af48f, 0xe963a535, 0x9e6495a3,
- 0x0edb8832, 0x79dcb8a4, 0xe0d5e91e, 0x97d2d988, 0x09b64c2b, 0x7eb17cbd, 0xe7b82d07, 0x90bf1d91,
- 0x1db71064, 0x6ab020f2, 0xf3b97148, 0x84be41de, 0x1adad47d, 0x6ddde4eb, 0xf4d4b551, 0x83d385c7,
- 0x136c9856, 0x646ba8c0, 0xfd62f97a, 0x8a65c9ec, 0x14015c4f, 0x63066cd9, 0xfa0f3d63, 0x8d080df5,
- 0x3b6e20c8, 0x4c69105e, 0xd56041e4, 0xa2677172, 0x3c03e4d1, 0x4b04d447, 0xd20d85fd, 0xa50ab56b,
- 0x35b5a8fa, 0x42b2986c, 0xdbbbc9d6, 0xacbcf940, 0x32d86ce3, 0x45df5c75, 0xdcd60dcf, 0xabd13d59,
- 0x26d930ac, 0x51de003a, 0xc8d75180, 0xbfd06116, 0x21b4f4b5, 0x56b3c423, 0xcfba9599, 0xb8bda50f,
- 0x2802b89e, 0x5f058808, 0xc60cd9b2, 0xb10be924, 0x2f6f7c87, 0x58684c11, 0xc1611dab, 0xb6662d3d,
- 0x76dc4190, 0x01db7106, 0x98d220bc, 0xefd5102a, 0x71b18589, 0x06b6b51f, 0x9fbfe4a5, 0xe8b8d433,
- 0x7807c9a2, 0x0f00f934, 0x9609a88e, 0xe10e9818, 0x7f6a0dbb, 0x086d3d2d, 0x91646c97, 0xe6635c01,
- 0x6b6b51f4, 0x1c6c6162, 0x856530d8, 0xf262004e, 0x6c0695ed, 0x1b01a57b, 0x8208f4c1, 0xf50fc457,
- 0x65b0d9c6, 0x12b7e950, 0x8bbeb8ea, 0xfcb9887c, 0x62dd1ddf, 0x15da2d49, 0x8cd37cf3, 0xfbd44c65,
- 0x4db26158, 0x3ab551ce, 0xa3bc0074, 0xd4bb30e2, 0x4adfa541, 0x3dd895d7, 0xa4d1c46d, 0xd3d6f4fb,
- 0x4369e96a, 0x346ed9fc, 0xad678846, 0xda60b8d0, 0x44042d73, 0x33031de5, 0xaa0a4c5f, 0xdd0d7cc9,
- 0x5005713c, 0x270241aa, 0xbe0b1010, 0xc90c2086, 0x5768b525, 0x206f85b3, 0xb966d409, 0xce61e49f,
- 0x5edef90e, 0x29d9c998, 0xb0d09822, 0xc7d7a8b4, 0x59b33d17, 0x2eb40d81, 0xb7bd5c3b, 0xc0ba6cad,
- 0xedb88320, 0x9abfb3b6, 0x03b6e20c, 0x74b1d29a, 0xead54739, 0x9dd277af, 0x04db2615, 0x73dc1683,
- 0xe3630b12, 0x94643b84, 0x0d6d6a3e, 0x7a6a5aa8, 0xe40ecf0b, 0x9309ff9d, 0x0a00ae27, 0x7d079eb1,
- 0xf00f9344, 0x8708a3d2, 0x1e01f268, 0x6906c2fe, 0xf762575d, 0x806567cb, 0x196c3671, 0x6e6b06e7,
- 0xfed41b76, 0x89d32be0, 0x10da7a5a, 0x67dd4acc, 0xf9b9df6f, 0x8ebeeff9, 0x17b7be43, 0x60b08ed5,
- 0xd6d6a3e8, 0xa1d1937e, 0x38d8c2c4, 0x4fdff252, 0xd1bb67f1, 0xa6bc5767, 0x3fb506dd, 0x48b2364b,
- 0xd80d2bda, 0xaf0a1b4c, 0x36034af6, 0x41047a60, 0xdf60efc3, 0xa867df55, 0x316e8eef, 0x4669be79,
- 0xcb61b38c, 0xbc66831a, 0x256fd2a0, 0x5268e236, 0xcc0c7795, 0xbb0b4703, 0x220216b9, 0x5505262f,
- 0xc5ba3bbe, 0xb2bd0b28, 0x2bb45a92, 0x5cb36a04, 0xc2d7ffa7, 0xb5d0cf31, 0x2cd99e8b, 0x5bdeae1d,
- 0x9b64c2b0, 0xec63f226, 0x756aa39c, 0x026d930a, 0x9c0906a9, 0xeb0e363f, 0x72076785, 0x05005713,
- 0x95bf4a82, 0xe2b87a14, 0x7bb12bae, 0x0cb61b38, 0x92d28e9b, 0xe5d5be0d, 0x7cdcefb7, 0x0bdbdf21,
- 0x86d3d2d4, 0xf1d4e242, 0x68ddb3f8, 0x1fda836e, 0x81be16cd, 0xf6b9265b, 0x6fb077e1, 0x18b74777,
- 0x88085ae6, 0xff0f6a70, 0x66063bca, 0x11010b5c, 0x8f659eff, 0xf862ae69, 0x616bffd3, 0x166ccf45,
- 0xa00ae278, 0xd70dd2ee, 0x4e048354, 0x3903b3c2, 0xa7672661, 0xd06016f7, 0x4969474d, 0x3e6e77db,
- 0xaed16a4a, 0xd9d65adc, 0x40df0b66, 0x37d83bf0, 0xa9bcae53, 0xdebb9ec5, 0x47b2cf7f, 0x30b5ffe9,
- 0xbdbdf21c, 0xcabac28a, 0x53b39330, 0x24b4a3a6, 0xbad03605, 0xcdd70693, 0x54de5729, 0x23d967bf,
- 0xb3667a2e, 0xc4614ab8, 0x5d681b02, 0x2a6f2b94, 0xb40bbe37, 0xc30c8ea1, 0x5a05df1b, 0x2d02ef8d,
-};
-
-uint32_t
-crc32(const void *data, int len)
-{
- const uint8_t *d;
- uint32_t c;
- int i;
-
- d = data;
- c = 0xffffffff;
- for (i = 0; i < len; i++)
- c = table[(c & 0xff) ^ d[i]] ^ (c >> 8);
-
- return ~c;
-}
--- /dev/null
+++ b/examples/dial.c
@@ -1,0 +1,50 @@
+#include <string.h>
+#include <stdio.h>
+#include <sys/types.h>
+#include <sys/socket.h>
+#include <netdb.h>
+#include <unistd.h>
+
+int
+dial(const char *s)
+{
+ struct addrinfo *r, *a, hint = {.ai_flags = AI_ADDRCONFIG, .ai_family = AF_UNSPEC, 0};
+ char host[128], *port;
+ int e, f;
+
+ if(strncmp(s, "udp!", 4) == 0){
+ hint.ai_socktype = SOCK_DGRAM;
+ hint.ai_protocol = IPPROTO_UDP;
+ }else if(strncmp(s, "tcp!", 4) == 0){
+ hint.ai_socktype = SOCK_STREAM;
+ hint.ai_protocol = IPPROTO_TCP;
+ }else{
+ fprintf(stderr, "invalid dial string: %s\n", s);
+ return -1;
+ }
+ if((port = strchr(s+4, '!')) == NULL){
+ fprintf(stderr, "invalid dial string: %s\n", s);
+ return -1;
+ }
+ if(snprintf(host, sizeof(host), "%.*s", (int)(port-s-4), s+4) >= (int)sizeof(host)){
+ fprintf(stderr, "host name too large: %s\n", s);
+ return -1;
+ }
+ port++;
+ if((e = getaddrinfo(host, port, &hint, &r)) != 0){
+ fprintf(stderr, "%s: %s\n", gai_strerror(e), s);
+ return -1;
+ }
+ f = -1;
+ for(a = r; a != NULL; a = a->ai_next){
+ if((f = socket(a->ai_family, a->ai_socktype, a->ai_protocol)) < 0)
+ continue;
+ if(connect(f, a->ai_addr, a->ai_addrlen) == 0)
+ break;
+ close(f);
+ f = -1;
+ }
+ freeaddrinfo(r);
+
+ return f;
+}
--- /dev/null
+++ b/examples/fs9ptest.c
@@ -1,0 +1,52 @@
+#include <stdio.h>
+#include <stdint.h>
+#include <stdarg.h>
+#include "c9/proto.h"
+#include "c9/fs.h"
+#include "c9/fs9p.h"
+
+int dial(const char *s);
+C9ctx *init9p(int f);
+
+static FS9pfd fds[8];
+
+static void
+fserror(FS *fs, const char *fmt, ...)
+{
+ va_list ap;
+
+ (void)fs;
+ va_start(ap, fmt);
+ vfprintf(stderr, fmt, ap);
+ fprintf(stderr, "\n");
+ va_end(ap);
+}
+
+int
+main(int argc, char **argv)
+{
+ C9ctx *ctx;
+ FSstat st;
+ FS fs;
+ FS9p fs9 = {
+ .fds = fds,
+ .numfds = sizeof(fds)/sizeof(fds[0]),
+ };
+ int f;
+
+ (void)argc; (void)argv;
+
+ fs.error = fserror;
+ fs.aux = (void*)&fs9;
+ if((f = dial("tcp!ftrv.se!564")) < 0)
+ return 1;
+ ctx = init9p(f);
+ fs9.ctx = ctx;
+ if(fs9pinit(&fs, 8192) != 0)
+ return 1;
+ if(fs.fstat(&fs, FS9proot, &st) != 0)
+ return 1;
+ printf("%s %d %s %o\n", st.name, (int)st.size, st.uid, st.mode&0777);
+
+ return 0;
+}
--- /dev/null
+++ b/examples/sockets.c
@@ -1,0 +1,127 @@
+#include <stdio.h>
+#include <stdint.h>
+#include <stdarg.h>
+#include <errno.h>
+#include <unistd.h>
+#include <stdlib.h>
+#include "c9/proto.h"
+
+enum {
+ Msize = 8192,
+
+ Disconnected = 1<<0,
+};
+
+typedef struct C9aux C9aux;
+
+struct C9aux {
+ C9ctx c;
+ int f;
+ int flags;
+ uint8_t rdbuf[Msize];
+ uint8_t wrbuf[Msize];
+ uint32_t wroff;
+};
+
+static uint8_t *
+ctxread(C9ctx *ctx, uint32_t size, int *err)
+{
+ uint32_t n;
+ int r;
+ C9aux *a;
+
+ a = ctx->aux;
+ *err = 0;
+ for(n = 0; n < size; n += r){
+ if((r = read(a->f, a->rdbuf+n, size-n)) <= 0){
+ if(errno == EINTR)
+ continue;
+ a->flags |= Disconnected;
+ close(a->f);
+ return NULL;
+ }
+ }
+
+ return a->rdbuf;
+}
+
+static int
+wrsend(C9aux *a)
+{
+ uint32_t n;
+ int w;
+
+ for(n = 0; n < a->wroff; n += w){
+ if((w = write(a->f, a->wrbuf+n, a->wroff-n)) <= 0){
+ if(errno == EINTR)
+ continue;
+ if(errno != EPIPE) /* remote end closed */
+ perror("write");
+ return -1;
+ }
+ }
+ a->wroff = 0;
+
+ return 0;
+}
+
+static uint8_t *
+ctxbegin(C9ctx *ctx, uint32_t size)
+{
+ uint8_t *b;
+ C9aux *a;
+
+ a = ctx->aux;
+ if(a->wroff + size > sizeof(a->wrbuf)){
+ if(wrsend(a) != 0 || a->wroff + size > sizeof(a->wrbuf))
+ return NULL;
+ }
+ b = a->wrbuf + a->wroff;
+ a->wroff += size;
+
+ return b;
+}
+
+static int
+ctxend(C9ctx *ctx)
+{
+ C9aux *a;
+
+ /*
+ * To batch up requests and instead flush them all at once just return 0
+ * here and call wrsend yourself when needed.
+ */
+ a = ctx->aux;
+ return wrsend(a);
+}
+
+static void
+ctxerror(C9ctx *ctx, const char *fmt, ...)
+{
+ va_list ap;
+
+ (void)ctx;
+ va_start(ap, fmt);
+ vfprintf(stderr, fmt, ap);
+ fprintf(stderr, "\n");
+ va_end(ap);
+}
+
+C9aux *
+init9p(int f)
+{
+ C9aux *c;
+
+ if((c = calloc(1, sizeof(*c))) == NULL){
+ close(f);
+ return NULL;
+ }
+ c->f = f;
+ c->c.read = ctxread;
+ c->c.begin = ctxbegin;
+ c->c.end = ctxend;
+ c->c.error = ctxerror;
+ c->c.aux = c;
+
+ return c;
+}
--- /dev/null
+++ b/fs9p.c
@@ -1,0 +1,301 @@
+#include <stdint.h>
+#include <string.h>
+#include "c9/proto.h"
+#include "c9/fs.h"
+#include "c9/fs9p.h"
+
+enum
+{
+ Flinit = 1<<0,
+ Flerror = 1<<1,
+
+ F9init = 0,
+ F9chdir,
+ F9open,
+ F9create,
+ F9read,
+ F9readdir,
+ F9write,
+ F9seek,
+ F9stat,
+ F9fstat,
+ F9close,
+ F9remove,
+ F9rename,
+};
+
+static const FS9pfd freefd =
+{
+ .offset = ~0ULL,
+};
+
+#define isfree(fd) ((fd).offset == ~0ULL)
+
+static const char *f2s[] =
+{
+ [F9init] = "init",
+ [F9chdir] = "chdir",
+ [F9open] = "open",
+ [F9create] = "create",
+ [F9read] = "read",
+ [F9readdir] = "readdir",
+ [F9write] = "write",
+ [F9seek] = "seek",
+ [F9stat] = "stat",
+ [F9fstat] = "fstat",
+ [F9close] = "close",
+ [F9remove] = "remove",
+ [F9rename] = "rename",
+};
+
+static int
+fs9pchdir(FS *fs, const char *path)
+{
+ FS9p *aux = (void*)fs->aux;
+
+ aux->f = F9chdir;
+
+ return -1;
+}
+
+static int
+fs9popen(FS *fs, const char *name, FSmode mode)
+{
+ FS9p *aux = (void*)fs->aux;
+
+ aux->f = F9open;
+
+ return -1;
+}
+
+static int
+fs9pcreate(FS *fs, const char *path, int perm)
+{
+ FS9p *aux = (void*)fs->aux;
+
+ aux->f = F9create;
+
+ return -1;
+}
+
+static int64_t
+fs9pread(FS *fs, int fd, void *buf, uint64_t count)
+{
+ FS9p *aux = (void*)fs->aux;
+
+ aux->f = F9read;
+
+ return -1;
+}
+
+static int64_t
+fs9preaddir(FS *fs, int fd, FSstat *st, int nst)
+{
+ FS9p *aux = (void*)fs->aux;
+
+ aux->f = F9readdir;
+
+ return -1;
+}
+
+static int64_t
+fs9pwrite(FS *fs, int fd, const void *buf, uint64_t count)
+{
+ FS9p *aux = (void*)fs->aux;
+
+ aux->f = F9write;
+
+ return -1;
+}
+
+static int64_t
+fs9pseek(FS *fs, int fd, int64_t offset, int whence)
+{
+ FS9p *aux = (void*)fs->aux;
+
+ aux->f = F9seek;
+
+ return -1;
+}
+
+static int
+fs9pstat(FS *fs, const char *path, FSstat *st)
+{
+ FS9p *aux = (void*)fs->aux;
+
+ aux->f = F9stat;
+
+ return -1;
+}
+
+static int
+fs9pfstat(FS *fs, int fd, FSstat *st)
+{
+ FS9p *aux = (void*)fs->aux;
+ C9tag tag;
+ int e;
+
+ aux->f = F9fstat;
+ aux->p = st;
+ if((e = c9stat(aux->ctx, &tag, fd)) < 0)
+ return e;
+ while(aux->f != -F9fstat)
+ c9proc(aux->ctx);
+
+ return 0;
+}
+
+static int
+fs9pclose(FS *fs, int fd)
+{
+ FS9p *aux = (void*)fs->aux;
+
+ aux->f = F9close;
+
+ return -1;
+}
+
+static int
+fs9premove(FS *fs, const char *name)
+{
+ FS9p *aux = (void*)fs->aux;
+
+ aux->f = F9remove;
+
+ return -1;
+}
+
+static int
+fs9prename(FS *fs, const char *oldpath, const char *newpath)
+{
+ FS9p *aux = (void*)fs->aux;
+
+ aux->f = F9rename;
+
+ return -1;
+}
+
+static void
+fs9pr(C9ctx *ctx, C9r *r)
+{
+ FSstat *st;
+ C9tag tag;
+ FS9p *aux;
+ FS *fs;
+
+ fs = ctx->paux;
+ aux = (void*)fs->aux;
+
+ switch(r->type){
+ case Rversion:
+ c9attach(ctx, &tag, FS9proot, C9nofid, "none", NULL); /* FIXME those need to be configurable */
+ break;
+
+ case Rauth:
+ break;
+
+ case Rattach:
+ aux->flags |= Flinit;
+ break;
+
+ case Rerror:
+ fs->error(fs, "%s: %s", f2s[aux->f], r->error);
+ aux->flags |= Flerror;
+ break;
+
+ case Rflush:
+ break;
+
+ case Rwalk:
+ break;
+
+ case Ropen:
+ break;
+
+ case Rcreate:
+ break;
+
+ case Rread:
+ break;
+
+ case Rwrite:
+ break;
+
+ case Rclunk:
+ break;
+
+ case Rremove:
+ break;
+
+ case Rstat:
+ if(aux->f == F9stat || aux->f == F9fstat){
+ st = aux->p;
+ st->size = r->stat.size;
+ st->name = r->stat.name;
+ st->uid = r->stat.uid;
+ st->gid = r->stat.gid;
+ st->mode = r->stat.mode;
+ st->mtime = r->stat.mtime;
+ aux->f = -aux->f;
+ }
+ break;
+
+ case Rwstat:
+ break;
+ }
+}
+
+int
+fs9pinit(FS *fs, uint32_t msize)
+{
+ FS9p *aux = (void*)fs->aux;
+ C9error err;
+ C9tag tag;
+ int i;
+
+ if(fs->error == NULL)
+ return -1;
+ if(aux == NULL || aux->ctx == NULL || aux->fds == NULL || aux->numfds < 1){
+ fs->error(fs, "fs9pinit: invalid aux");
+ return -1;
+ }
+ if(aux->ctx == NULL){
+ fs->error(fs, "fs9pinit: invalid ctx");
+ return -1;
+ }
+
+ fs->chdir = fs9pchdir;
+ fs->open = fs9popen;
+ fs->create = fs9pcreate;
+ fs->read = fs9pread;
+ fs->readdir = fs9preaddir;
+ fs->write = fs9pwrite;
+ fs->seek = fs9pseek;
+ fs->stat = fs9pstat;
+ fs->fstat = fs9pfstat;
+ fs->close = fs9pclose;
+ fs->remove = fs9premove;
+ fs->rename = fs9prename;
+
+ aux->root.offset = 0;
+ for(i = 0; i < aux->numfds; i++)
+ aux->fds[i] = freefd;
+
+ aux->ctx->r = fs9pr;
+ aux->ctx->paux = fs;
+ aux->f = F9init;
+ aux->flags = 0;
+
+ if((err = c9version(aux->ctx, &tag, msize)) != 0)
+ return err;
+ for(;;){
+ if((err = c9proc(aux->ctx)) != 0)
+ return err;
+ if(aux->flags & Flerror)
+ return -1;
+ if(aux->flags & Flinit)
+ break;
+ }
+
+ return 0;
+}
--- /dev/null
+++ b/include/c9/fs.h
@@ -1,0 +1,48 @@
+typedef struct FS FS;
+typedef struct FSstat FSstat;
+
+typedef enum
+{
+ FS_OREAD,
+ FS_OWRITE,
+ FS_ORDWR,
+ FS_OEXEC,
+ FS_OTRUNC = 0x10,
+ FS_ORCLOSE = 0x40,
+ FS_OEXCL = 0x1000,
+ FS_DIR = 0x80000000U, /* open/create: it's supposed to be a directory. */
+}FSmode;
+
+struct FSstat
+{
+ uint64_t size; /* Size of the file (in bytes). */
+ char *name; /* Name of the file. */
+ char *uid; /* Owner of the file. */
+ char *gid; /* Group of the file. */
+ uint32_t mode; /* Permissions. See C9st* and C9perm. */
+ uint64_t mtime; /* Last modification time. Nanoseconds since epoch. */
+};
+
+struct FSaux;
+
+struct FS
+{
+ /* Callback for error messages. */
+ void (*error)(FS *fs, const char *fmt, ...) __attribute__((nonnull(1), format(printf, 2, 3)));
+
+ /* Do not set these. */
+ int (*chdir)(FS *fs, const char *path) __attribute__((nonnull(1, 2)));
+ int (*open)(FS *fs, const char *name, FSmode mode) __attribute__((nonnull(1, 2)));
+ int (*create)(FS *fs, const char *path, int perm) __attribute__((nonnull(1, 2)));
+ int64_t (*read)(FS *fs, int fd, void *buf, uint64_t count) __attribute__((nonnull(1, 3)));
+ int64_t (*readdir)(FS *fs, int fd, FSstat *st, int nst);
+ int64_t (*write)(FS *fs, int fd, const void *buf, uint64_t count) __attribute__((nonnull(1, 3)));
+ int64_t (*seek)(FS *fs, int fd, int64_t offset, int whence) __attribute__((nonnull(1)));
+ int (*stat)(FS *fs, const char *path, FSstat *st) __attribute__((nonnull(1, 2, 3)));
+ int (*fstat)(FS *fs, int fd, FSstat *st) __attribute__((nonnull(1, 3)));
+ int (*close)(FS *fs, int fd) __attribute__((nonnull(1)));
+ int (*remove)(FS *fs, const char *name) __attribute__((nonnull(1, 2)));
+ int (*rename)(FS *fs, const char *oldpath, const char *newpath) __attribute__((nonnull(1, 2, 3)));
+
+ struct FSaux *aux;
+};
--- /dev/null
+++ b/include/c9/fs9p.h
@@ -1,0 +1,31 @@
+enum
+{
+ FS9proot = 3, /* '/' file descriptor. */
+};
+
+typedef struct FS9p FS9p;
+typedef struct FS9pfd FS9pfd;
+
+struct FS9pfd
+{
+ uint64_t offset;
+};
+
+struct FS9p
+{
+ /* The following three fields need to be set before calling fs9pinit. */
+
+ C9ctx *ctx; /* Set to a full set up context, except "r" and "t" fields. */
+ FS9pfd *fds; /* Point at the allocated fds array. */
+ int numfds; /* Set to the number of entries available in fds array. */
+ void *aux; /* Optional, user-supplied aux value. */
+
+ /* Private, do not touch. */
+ FS9pfd root;
+ FS9pfd cwd;
+ int f;
+ void *p;
+ int flags;
+};
+
+extern int fs9pinit(FS *fs, uint32_t msize) __attribute__((nonnull(1)));
--- /dev/null
+++ b/include/c9/proto.h
@@ -1,0 +1,361 @@
+struct C9aux;
+
+typedef struct C9r C9r;
+typedef struct C9t C9t;
+typedef struct C9stat C9stat;
+typedef struct C9ctx C9ctx;
+typedef struct C9qid C9qid;
+typedef uint32_t C9fid;
+typedef uint16_t C9tag;
+
+/* Stat field is not changed if it's set to this value when calling c9wstat. */
+#define C9nochange (~0)
+
+/* Special fid used with auth/attach to basically avoid authentication. */
+#define C9nofid ((C9fid)~0)
+
+/* C9modes for opening a file. */
+typedef enum
+{
+ C9read = 0,
+ C9write = 1,
+ C9rdwr = 2,
+ C9exec = 3,
+ C9trunc = 0x10,
+ C9rclose = 0x40,
+ C9excl = 0x1000,
+}C9mode;
+
+typedef enum
+{
+ /* User/owner. */
+ C9permur = 1<<8, /* Readable. */
+ C9permuw = 1<<7, /* Writable. */
+ C9permux = 1<<6, /* Executable. */
+
+ /* Group. */
+ C9permgr = 1<<5,
+ C9permgw = 1<<4,
+ C9permgx = 1<<3,
+
+ /* Other. */
+ C9permor = 1<<2,
+ C9permow = 1<<1,
+ C9permox = 1<<0,
+}C9perm;
+
+/* Directory. */
+#define C9permdir 0x80000000
+
+/* Bitmask of stat.mode. */
+#define C9stdir 0x80000000
+#define C9stappend 0x40000000
+#define C9stexcl 0x20000000
+#define C9sttmp 0x04000000
+
+/* Limits. */
+enum
+{
+ C9maxtags = 64, /* Maximal number of outstanding requests. [1-65535] */
+ C9maxflush = 8, /* Maximal number of outstanding flushes. [1-65535] */
+ C9maxstr = 0xffff, /* Maximal string length. [1-65535] */
+ C9minmsize = 4096, /* Minimal sane msize. [4096-...] */
+ C9maxpathel = 16, /* Maximal number of elements in a path. Do not change. */
+};
+
+/* Errors. */
+typedef enum
+{
+ C9Einit = -1, /* Initialization failed. */
+ C9Ever = -2, /* Protocol version doesn't match. */
+ C9Epkt = -3, /* Incoming packet error. */
+ C9Etag = -4, /* No free tags or bad tag. */
+ C9Ebuf = -5, /* No buffer space enough for a message. */
+ C9Epath = -6, /* Path is too long or just invalid. */
+ C9Eflush = -7, /* Limit of outstanding flushes reached. */
+ C9Esize = -8, /* Can't fit data in one message. */
+ C9Estr = -9 /* Bad string. */
+}C9error;
+
+/* Request types. */
+typedef enum
+{
+ Tversion = 100,
+ Tauth = 102,
+ Tattach = 104,
+ Tflush = 108,
+ Twalk = 110,
+ Topen = 112,
+ Tcreate = 114,
+ Tread = 116,
+ Twrite = 118,
+ Tclunk = 120,
+ Tremove = 122,
+ Tstat = 124,
+ Twstat = 126
+}C9ttype;
+
+/* Response types. */
+typedef enum
+{
+ Rversion = 101,
+ Rauth = 103,
+ Rattach = 105,
+ Rerror = 107,
+ Rflush = 109,
+ Rwalk = 111,
+ Ropen = 113,
+ Rcreate = 115,
+ Rread = 117,
+ Rwrite = 119,
+ Rclunk = 121,
+ Rremove = 123,
+ Rstat = 125,
+ Rwstat = 127
+}C9rtype;
+
+/* Unique file id type. */
+typedef enum
+{
+ C9qtdir = 1<<7,
+ C9qtappend = 1<<6,
+ C9qtexcl = 1<<5,
+ C9qtauth = 1<<3,
+ C9qttmp = 1<<2,
+ C9qtfile = 0
+}C9qt;
+
+/* Unique file id. */
+struct C9qid
+{
+ uint64_t path;
+ uint32_t version;
+ C9qt type;
+};
+
+/*
+ * File stats. Version and muid are ignored on wstat. Dmdir bit
+ * change in mode won't work on wstat. Set any integer field to
+ * C9nochange to keep it unchanged on wstat. Set any string to NULL to
+ * keep it unchanged. Strings can be empty (""), but never NULL after
+ * stat call.
+ */
+struct C9stat
+{
+ C9qid qid; /* Same as qid[0]. */
+ uint64_t size; /* Size of the file (in bytes). */
+ char *name; /* Name of the file. */
+ char *uid; /* Owner of the file. */
+ char *gid; /* Group of the file. */
+ char *muid; /* The user who modified the file last. */
+ uint32_t mode; /* Permissions. See C9st* and C9perm. */
+ uint32_t atime; /* Last access time. */
+ uint32_t mtime; /* Last modification time. */
+};
+
+/* Response data. */
+struct C9r
+{
+ union
+ {
+ char *error;
+
+ struct
+ {
+ uint8_t *data;
+ uint32_t size;
+ }read;
+
+ struct
+ {
+ uint32_t size;
+ }write;
+
+ /* File stats (only valid if type is Rstat). */
+ C9stat stat;
+
+ /*
+ * Qid(s). qid[0] is valid for auth/attach/create/stat/open.
+ * More ids may be a result of a walk, see numqid.
+ */
+ C9qid qid[C9maxpathel];
+ };
+ C9rtype type; /* Response type. */
+
+ /*
+ * If not zero, is the maximum number of bytes that are guaranteed
+ * to be read or written atomically, without breaking into multiple
+ * messages.
+ */
+ uint32_t iounit;
+
+ int numqid; /* Number of valid unique ids in qid array. */
+ C9tag tag; /* Tag number. */
+};
+
+/* Request data. */
+struct C9t
+{
+ C9ttype type;
+ C9tag tag;
+ union
+ {
+ struct
+ {
+ char *uname;
+ char *aname;
+ C9fid afid;
+ }attach;
+
+ struct
+ {
+ char *uname;
+ char *aname;
+ C9fid afid;
+ }auth;
+
+ struct
+ {
+ char *name;
+ uint32_t perm;
+ C9mode mode;
+ }create;
+
+ struct
+ {
+ C9tag oldtag;
+ }flush;
+
+ struct
+ {
+ C9mode mode;
+ }open;
+
+ struct
+ {
+ uint64_t offset;
+ uint32_t size;
+ }read;
+
+ struct
+ {
+ char *wname[C9maxpathel+1]; /* wname[16] is always NULL */
+ C9fid newfid;
+ }walk;
+
+ struct
+ {
+ uint64_t offset;
+ uint8_t *data;
+ uint32_t size;
+ }write;
+
+ C9stat wstat;
+ };
+ C9fid fid;
+};
+
+enum
+{
+ C9tagsbits = sizeof(uint32_t) * 8,
+};
+
+struct C9ctx
+{
+ /*
+ * Should return a pointer to the data (exactly 'size' bytes) read.
+ * Set 'err' to non-zero and return NULL in case of error.
+ * 'err' set to zero (no error) should be used to return from c9process
+ * early (timeout on read to do non-blocking operations, for example).
+ */
+ uint8_t *(*read)(C9ctx *ctx, uint32_t size, int *err) __attribute__((nonnull(1, 3)));
+
+ /* Should return a buffer to store 'size' bytes. Nil means no memory. */
+ uint8_t *(*begin)(C9ctx *ctx, uint32_t size) __attribute__((nonnull(1)));
+
+ /*
+ * Marks the end of a message. Callback may decide if any accumulated
+ * messages should be sent to the server/client.
+ */
+ int (*end)(C9ctx *ctx) __attribute__((nonnull(1)));
+
+ /* Callback called every time a new R-message is received. */
+ void (*r)(C9ctx *ctx, C9r *r) __attribute__((nonnull(1, 2)));
+
+ /* Callback called every time a new T-message is received. */
+ void (*t)(C9ctx *ctx, C9t *t) __attribute__((nonnull(1, 2)));
+
+ /* Callback for error messages. */
+ void (*error)(C9ctx *ctx, const char *fmt, ...) __attribute__((nonnull(1), format(printf, 2, 3)));
+
+ /* Auxiliary data, can be used by any of above callbacks. */
+ struct C9aux *aux;
+
+ /* private stuff */
+ void *paux;
+ uint32_t msize;
+#ifndef C9_NO_CLIENT
+ uint32_t flush[C9maxflush];
+ uint32_t tags[C9maxtags/C9tagsbits];
+#endif
+ union
+ {
+#ifndef C9_NO_CLIENT
+ C9tag lowfreetag;
+#endif
+#ifndef C9_NO_SERVER
+ uint16_t svflags;
+#endif
+ };
+};
+
+/* Parse one directory entry. */
+extern C9error c9parsedir(C9ctx *c, C9stat *stat, uint8_t **data, uint32_t *size) __attribute__((nonnull(1, 2, 3)));
+
+#ifndef C9_NO_CLIENT
+
+extern C9error c9version(C9ctx *c, C9tag *tag, uint32_t msize) __attribute__((nonnull(1, 2)));
+extern C9error c9auth(C9ctx *c, C9tag *tag, C9fid afid, const char *uname, const char *aname) __attribute__((nonnull(1, 2)));
+extern C9error c9flush(C9ctx *c, C9tag *tag, C9tag oldtag) __attribute__((nonnull(1, 2)));
+extern C9error c9attach(C9ctx *c, C9tag *tag, C9fid fid, C9fid afid, const char *uname, const char *aname) __attribute__((nonnull(1, 2)));
+extern C9error c9walk(C9ctx *c, C9tag *tag, C9fid fid, C9fid newfid, const char *path[]) __attribute__((nonnull(1, 2, 5)));
+extern C9error c9open(C9ctx *c, C9tag *tag, C9fid fid, C9mode mode) __attribute__((nonnull(1, 2)));
+extern C9error c9create(C9ctx *c, C9tag *tag, C9fid fid, const char *name, uint32_t perm, C9mode mode) __attribute__((nonnull(1, 2, 4)));
+extern C9error c9read(C9ctx *c, C9tag *tag, C9fid fid, uint64_t offset, uint32_t count) __attribute__((nonnull(1, 2)));
+extern C9error c9write(C9ctx *c, C9tag *tag, C9fid fid, uint64_t offset, const void *in, uint32_t count) __attribute__((nonnull(1, 2, 5)));
+extern C9error c9wrstr(C9ctx *c, C9tag *tag, C9fid fid, const char *s) __attribute__((nonnull(1, 2, 4)));
+extern C9error c9clunk(C9ctx *c, C9tag *tag, C9fid fid) __attribute__((nonnull(1, 2)));
+extern C9error c9remove(C9ctx *c, C9tag *tag, C9fid fid) __attribute__((nonnull(1, 2)));
+extern C9error c9stat(C9ctx *c, C9tag *tag, C9fid fid) __attribute__((nonnull(1, 2)));
+extern C9error c9wstat(C9ctx *c, C9tag *tag, C9fid fid, const C9stat *s) __attribute__((nonnull(1, 2, 4)));
+
+/*
+ * Wait until a response comes and process it. If the function returns
+ * any error, context must be treated as 'broken' and no subsequent calls
+ * should be made without reinitialization (c9version).
+ */
+extern C9error c9proc(C9ctx *c) __attribute__((nonnull(1)));
+
+#endif /* C9_NO_CLIENT */
+
+#ifndef C9_NO_SERVER
+
+extern C9error s9version(C9ctx *c) __attribute__((nonnull(1)));
+extern C9error s9auth(C9ctx *c, C9tag tag, const C9qid *aqid) __attribute__((nonnull(1, 3)));
+extern C9error s9error(C9ctx *c, C9tag tag, const char *err) __attribute__((nonnull(1)));
+extern C9error s9attach(C9ctx *c, C9tag tag, const C9qid *qid) __attribute__((nonnull(1, 3)));
+extern C9error s9flush(C9ctx *c, C9tag tag) __attribute__((nonnull(1)));
+extern C9error s9walk(C9ctx *c, C9tag tag, C9qid *qids[]) __attribute__((nonnull(1, 3)));
+extern C9error s9open(C9ctx *c, C9tag tag, const C9qid *qid, uint32_t iounit) __attribute__((nonnull(1, 3)));
+extern C9error s9create(C9ctx *c, C9tag tag, const C9qid *qid, uint32_t iounit) __attribute__((nonnull(1, 3)));
+extern C9error s9read(C9ctx *c, C9tag tag, const void *data, uint32_t size) __attribute__((nonnull(1, 3)));
+extern C9error s9readdir(C9ctx *c, C9tag tag, C9stat *st[], int *num, uint64_t *offset, uint32_t size) __attribute__((nonnull(1, 3, 4)));
+extern C9error s9write(C9ctx *c, C9tag tag, uint32_t size) __attribute__((nonnull(1)));
+extern C9error s9clunk(C9ctx *c, C9tag tag) __attribute__((nonnull(1)));
+extern C9error s9remove(C9ctx *c, C9tag tag) __attribute__((nonnull(1)));
+extern C9error s9stat(C9ctx *c, C9tag tag, const C9stat *s) __attribute__((nonnull(1, 3)));
+extern C9error s9wstat(C9ctx *c, C9tag tag) __attribute__((nonnull(1)));
+
+extern C9error s9proc(C9ctx *c) __attribute__((nonnull(1)));
+
+#endif /* C9_NO_SERVER */
--- a/nixos/README.md
+++ /dev/null
@@ -1,61 +1,0 @@
-# Read-only 9pex in NixOS
-
-```
-sudo mkdir -p /etc/nixos/pkgs
-sudo cp ninepex.nix /etc/nixos/pkgs/
-```
-
-And change `/etc/nixos/configuration.nix`:
-
-```
-{ config, pkgs, ... }:
-
-let
- ninepex = pkgs.callPackage ./pkgs/ninepex.nix {};
-in
-{
-
-.................
-
- services.xinetd = {
- enable = true;
- services = [
- {
- name = "9pfs";
- port = 564;
- user = "none";
- server = "${ninepex}/bin/9pex";
- serverArgs = "/home/9";
- flags = "KEEPALIVE REUSE";
- extraConfig = ''
- instances = UNLIMITED
- '';
- }
- ];
- };
-
-.................
-
- users.users.none = {
- isNormalUser = false;
- isSystemUser = false;
- createHome = true;
- home = "/home/9";
- shell = "${pkgs.shadow}/bin/nologin";
- };
-
-```
-
-Files go to `/home/9`. With `rsync` it can be done as so:
-
-```
-rsync -iCaR \
- --exclude='.git*' \
- --exclude='*.out' \
- --exclude='*.[0125678vqki]' \
- --exclude='[0125678vqki].*' \
- --delete-excluded --delete \
- --chown none:nogroup \
- dir1 dir2 dir3 \
- root@host:/home/9/
-```
--- a/nixos/ninepex.nix
+++ /dev/null
@@ -1,16 +1,0 @@
-{ stdenv }:
-
-stdenv.mkDerivation rec {
- name = "9pex";
- src = fetchGit {
- url = "https://git.sr.ht/~ft/9pro";
- ref = "master";
- rev = "f9ddf46b12b9ee88e3b81f35f80e374722ca6f9d";
- };
-
- installPhase = ''
- ./build.sh
- mkdir -p "$out/bin"
- cp 9pex "$out/bin/"
- '';
-}
--- a/parg/.gitignore
+++ /dev/null
@@ -1,14 +1,0 @@
-*.o
-*.obj
-
-*.a
-*.lib
-
-*.so
-*.so.*
-*.dll
-
-*.exe
-
-/build/
-/doc/
--- a/parg/.travis.yml
+++ /dev/null
@@ -1,22 +1,0 @@
-language: c
-
-compiler: gcc
-
-sudo: false
-
-before_install:
- - pip install --user codecov
-
-before_script:
- - mkdir build
- - cd build
- - cmake -DCMAKE_BUILD_TYPE=Debug -DBUILD_COVERAGE=ON ..
-
-script: cmake --build . && ctest -V .
-
-after_success:
- - gcov CMakeFiles/parg.dir/parg.c.gcno -o CMakeFiles/parg.dir
- - codecov -X gcov
-
-notifications:
- email: false
--- a/parg/CMakeLists.txt
+++ /dev/null
@@ -1,56 +1,0 @@
-cmake_minimum_required(VERSION 2.8.6)
-
-project(parg C)
-
-include(CheckCCompilerFlag)
-include(CTest)
-
-option(BUILD_COVERAGE "Code coverage" OFF)
-
-message(STATUS "CMAKE_BUILD_TYPE: ${CMAKE_BUILD_TYPE}")
-
-function(_add_c_compiler_flags var)
- foreach(flag ${ARGN})
- string(REGEX REPLACE "[^a-zA-Z0-9]+" "_" flag_var "CFLAG_${flag}")
- check_c_compiler_flag("${flag}" ${flag_var})
- if(${flag_var})
- set(${var} "${${var}} ${flag}")
- endif()
- endforeach()
- set(${var} "${${var}}" PARENT_SCOPE)
-endfunction()
-
-if(MSVC)
- _add_c_compiler_flags(CMAKE_C_FLAGS "/analyze")
- add_definitions(/D_CRT_SECURE_NO_WARNINGS)
-elseif(CMAKE_COMPILER_IS_GNUCC)
- _add_c_compiler_flags(CMAKE_C_FLAGS
- -Wall
- -Wextra
- -pedantic
- -Wshadow
- -Wpointer-arith
- -Wcast-qual
- -Wcast-align
- -Wstrict-prototypes
- -Wmissing-prototypes
- )
-elseif(CMAKE_C_COMPILER_ID MATCHES "Clang")
- _add_c_compiler_flags(CMAKE_C_FLAGS "-Weverything;-pedantic")
-endif()
-
-if(BUILD_COVERAGE)
- set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -g -O0 --coverage")
-endif()
-
-include_directories(${PROJECT_SOURCE_DIR})
-
-add_library(parg parg.c parg.h)
-
-if(BUILD_TESTING)
- add_executable(test_parg test/test_parg.c test/greatest.h parg.h)
- target_link_libraries(test_parg parg)
-
- enable_testing()
- add_test(test_parg test_parg)
-endif()
--- a/parg/COPYING
+++ /dev/null
@@ -1,113 +1,0 @@
-CC0 1.0 Universal
-
-Statement of Purpose
-
-The laws of most jurisdictions throughout the world automatically confer
-exclusive Copyright and Related Rights (defined below) upon the creator
-and subsequent owner(s) (each and all, an "owner") of an original work of
-authorship and/or a database (each, a "Work").
-
-Certain owners wish to permanently relinquish those rights to a Work for
-the purpose of contributing to a commons of creative, cultural and
-scientific works ("Commons") that the public can reliably and without fear
-of later claims of infringement build upon, modify, incorporate in other
-works, reuse and redistribute as freely as possible in any form whatsoever
-and for any purposes, including without limitation commercial purposes.
-These owners may contribute to the Commons to promote the ideal of a free
-culture and the further production of creative, cultural and scientific
-works, or to gain reputation or greater distribution for their Work in
-part through the use and efforts of others.
-
-For these and/or other purposes and motivations, and without any
-expectation of additional consideration or compensation, the person
-associating CC0 with a Work (the "Affirmer"), to the extent that he or she
-is an owner of Copyright and Related Rights in the Work, voluntarily
-elects to apply CC0 to the Work and publicly distribute the Work under its
-terms, with knowledge of his or her Copyright and Related Rights in the
-Work and the meaning and intended legal effect of CC0 on those rights.
-
-1. Copyright and Related Rights. A Work made available under CC0 may be
-protected by copyright and related or neighboring rights ("Copyright and
-Related Rights"). Copyright and Related Rights include, but are not
-limited to, the following:
-
- i. the right to reproduce, adapt, distribute, perform, display,
- communicate, and translate a Work;
- ii. moral rights retained by the original author(s) and/or performer(s);
-iii. publicity and privacy rights pertaining to a person's image or
- likeness depicted in a Work;
- iv. rights protecting against unfair competition in regards to a Work,
- subject to the limitations in paragraph 4(a), below;
- v. rights protecting the extraction, dissemination, use and reuse of data
- in a Work;
- vi. database rights (such as those arising under Directive 96/9/EC of the
- European Parliament and of the Council of 11 March 1996 on the legal
- protection of databases, and under any national implementation
- thereof, including any amended or successor version of such
- directive); and
-vii. other similar, equivalent or corresponding rights throughout the
- world based on applicable law or treaty, and any national
- implementations thereof.
-
-2. Waiver. To the greatest extent permitted by, but not in contravention
-of, applicable law, Affirmer hereby overtly, fully, permanently,
-irrevocably and unconditionally waives, abandons, and surrenders all of
-Affirmer's Copyright and Related Rights and associated claims and causes
-of action, whether now known or unknown (including existing as well as
-future claims and causes of action), in the Work (i) in all territories
-worldwide, (ii) for the maximum duration provided by applicable law or
-treaty (including future time extensions), (iii) in any current or future
-medium and for any number of copies, and (iv) for any purpose whatsoever,
-including without limitation commercial, advertising or promotional
-purposes (the "Waiver"). Affirmer makes the Waiver for the benefit of each
-member of the public at large and to the detriment of Affirmer's heirs and
-successors, fully intending that such Waiver shall not be subject to
-revocation, rescission, cancellation, termination, or any other legal or
-equitable action to disrupt the quiet enjoyment of the Work by the public
-as contemplated by Affirmer's express Statement of Purpose.
-
-3. Public License Fallback. Should any part of the Waiver for any reason
-be judged legally invalid or ineffective under applicable law, then the
-Waiver shall be preserved to the maximum extent permitted taking into
-account Affirmer's express Statement of Purpose. In addition, to the
-extent the Waiver is so judged Affirmer hereby grants to each affected
-person a royalty-free, non transferable, non sublicensable, non exclusive,
-irrevocable and unconditional license to exercise Affirmer's Copyright and
-Related Rights in the Work (i) in all territories worldwide, (ii) for the
-maximum duration provided by applicable law or treaty (including future
-time extensions), (iii) in any current or future medium and for any number
-of copies, and (iv) for any purpose whatsoever, including without
-limitation commercial, advertising or promotional purposes (the
-"License"). The License shall be deemed effective as of the date CC0 was
-applied by Affirmer to the Work. Should any part of the License for any
-reason be judged legally invalid or ineffective under applicable law, such
-partial invalidity or ineffectiveness shall not invalidate the remainder
-of the License, and in such case Affirmer hereby affirms that he or she
-will not (i) exercise any of his or her remaining Copyright and Related
-Rights in the Work or (ii) assert any associated claims and causes of
-action with respect to the Work, in either case contrary to Affirmer's
-express Statement of Purpose.
-
-4. Limitations and Disclaimers.
-
- a. No trademark or patent rights held by Affirmer are waived, abandoned,
- surrendered, licensed or otherwise affected by this document.
- b. Affirmer offers the Work as-is and makes no representations or
- warranties of any kind concerning the Work, express, implied,
- statutory or otherwise, including without limitation warranties of
- title, merchantability, fitness for a particular purpose, non
- infringement, or the absence of latent or other defects, accuracy, or
- the present or absence of errors, whether or not discoverable, all to
- the greatest extent permissible under applicable law.
- c. Affirmer disclaims responsibility for clearing rights of other persons
- that may apply to the Work or any use thereof, including without
- limitation any person's Copyright and Related Rights in the Work.
- Further, Affirmer disclaims responsibility for obtaining any necessary
- consents, permissions or other rights required for any use of the
- Work.
- d. Affirmer understands and acknowledges that Creative Commons is not a
- party to this document and has no duty or obligation with respect to
- this CC0 or use of the Work.
-
-For more information, please see
-<http://creativecommons.org/publicdomain/zero/1.0/>
--- a/parg/Doxyfile
+++ /dev/null
@@ -1,10 +1,0 @@
-# Doxyfile 1.8.6
-PROJECT_NAME = "parg"
-OUTPUT_DIRECTORY = doc
-JAVADOC_AUTOBRIEF = YES
-TAB_SIZE = 8
-OPTIMIZE_OUTPUT_FOR_C = YES
-EXTRACT_ALL = YES
-INPUT = parg.h README.md
-GENERATE_LATEX = NO
-USE_MDFILE_AS_MAINPAGE = README.md
--- a/parg/README.md
+++ /dev/null
@@ -1,205 +1,0 @@
-
-[](https://travis-ci.org/jibsen/parg) [](http://codecov.io/github/jibsen/parg?branch=master)
-
-About
------
-
-Most command-line programs have to parse options, so there are a lot of
-different solutions to this problem. Some offer many features, while others
-are more basic.
-
-One of the simpler solutions for C is the [getopt][] function, and its
-extension `getopt_long`. They iterate over the options in `argv`, returning
-them one at a time on successive calls.
-
-One nice thing about them is that they are available on most Unix-like
-operating systems (and usually accompany GCC elsewhere, like Windows).
-Unfortunately, some implementation details vary between platforms.
-
-A potential question is what license the version you get when you include
-them is available under. Some are GPL, others LGPL. There are also ports of
-`getopt` that use more liberal licenses.
-
-`parg` is a parser for `argv` that works similarly to `getopt`, but does not
-aim to be a direct replacement. It attempts to make some choices about how to
-handle the extensions and idiosyncrasies of other `getopt` implementations,
-and document them.
-
-It consists of a single source and include file, written in portable ANSI C.
-It is made available under [CC0][].
-
-[getopt]: https://en.wikipedia.org/wiki/Getopt
-[CC0]: http://creativecommons.org/publicdomain/zero/1.0/
-
-
-Usage
------
-
-The include file `parg.h` contains documentation in the form of [doxygen][]
-comments. A configuration file is included, so you can simply run `doxygen`
-to generate documentation in HTML format.
-
-`parg` uses [CMake][] to generate build systems. To create one for the tools
-on your platform, and build `parg`, use something along the lines of:
-
-~~~sh
-mkdir build
-cd build
-cmake ..
-cmake --build .
-~~~
-
-You can also simply compile the source files and link them into your project.
-CMake just provides an easy way to build and test across various platforms and
-toolsets.
-
-[doxygen]: http://www.doxygen.org/
-[CMake]: http://www.cmake.org/
-
-
-Example
--------
-
-Here is an example that parses command-line options using `parg_getopt()`:
-
-~~~c
-#include <stdio.h>
-#include <stdlib.h>
-
-#include "parg.h"
-
-int main(int argc, char *argv[])
-{
- struct parg_state ps;
- int c;
-
- parg_init(&ps);
-
- while ((c = parg_getopt(&ps, argc, argv, "hs:v")) != -1) {
- switch (c) {
- case 1:
- printf("nonoption '%s'\n", ps.optarg);
- break;
- case 'h':
- printf("Usage: testparg [-h] [-v] [-s STRING]\n");
- return EXIT_SUCCESS;
- break;
- case 's':
- printf("option -s with argument '%s'\n", ps.optarg);
- break;
- case 'v':
- printf("testparg 1.0.0\n");
- return EXIT_SUCCESS;
- break;
- case '?':
- if (ps.optopt == 's') {
- printf("option -s requires an argument\n");
- }
- else {
- printf("unknown option -%c\n", ps.optopt);
- }
- return EXIT_FAILURE;
- break;
- default:
- printf("error: unhandled option -%c\n", c);
- return EXIT_FAILURE;
- break;
- }
- }
-
- for (c = ps.optind; c < argc; ++c) {
- printf("nonoption '%s'\n", argv[c]);
- }
-
- return EXIT_SUCCESS;
-}
-~~~
-
-
-Comparison to `getopt`
-----------------------
-
-### Use of global variables
-
-`getopt` uses global variables to store its state between calls. `parg` uses
-a struct `parg_state`, which you must pass with each call.
-
-### Handling of nonoptions
-
-POSIX and BSD `getopt` return `-1` on the first nonoption argument. GNU
-`getopt` by default reorders `argv` (even though it is passed as const), so
-all options come first.
-
-`parg` does not change `argv`, and returns each nonoption as the option
-argument of an option with value `1` (like GNU `getopt`, if `optstring` were
-prefixed by '`-`').
-
-If you wish to process all options first, and have the nonoptions ordered at
-the end of `argv`, you can use `parg_reorder()`:
-
-~~~c
- optend = parg_reorder(argc, argv, optstring, NULL);
-
- while ((c = parg_getopt(&ps, optend, argv, optstring)) != -1) {
- /* ... */
- }
-
- /* elements of argv[] from optend to argc are nonoptions */
-~~~
-
-### Value of `optind` on error
-
-When there are multiple short options in one argument, `getopt` does not
-increment `optind` until the last one is processed. This makes it harder to
-tell which argument an unknown option came from (if `a` is an unknown option,
-`-a` and `-ab` will return '`?`' with different values in `optind`).
-
-`parg` always increments the `optind` value in it's state so it points to the
-next `argv` element to be processed. So when `parg` returns '`?`' (or '`:`'),
-the element that contains the error is `argv[optind - 1]`.
-
-### Value of `optopt` on error
-
-With `getopt_long`, it varies what the values of `optopt` and `longindex` are
-when an error is found with option arguments of long options. Sometimes these
-values are not documented.
-
-`parg` sets `optopt` to `val` if `flag` is `NULL`, and `0` otherwise (which
-equals the return value on successful match), and `longindex` is set to the
-index of the entry in `longopts` that matched.
-
-### Return value on option argument error
-
-When the first character of `optstring` is '`:`', it varies what `getopt`
-returns on extraneous option arguments.
-
-In this case, `parg` returns '`?`' if no option match is found, and '`:`' if
-a match is found, but is missing a required argument, or has an extraneous
-argument.
-
-
-Alternatives
-------------
-
-Some ports of `getopt`:
-
- - [Free Getopt](http://freegetopt.sourceforge.net/)
- - [ya_getopt](http://github.com/kubo/ya_getopt/)
- - [getopt_port](http://github.com/kimgr/getopt_port/)
-
-Other command-line parsing libraries that support C:
-
- - [Gengetopt](http://www.gnu.org/software/gengetopt/)
- - [Argp](http://www.gnu.org/software/libc/manual/html_node/Argp.html)
- - [popt](http://en.wikipedia.org/wiki/Popt)
- - [argtable](http://argtable.sourceforge.net/)
- - [optlist](http://michael.dipperstein.com/optlist/)
- - [Arg_parser](http://www.nongnu.org/arg-parser/arg_parser.html)
- - [Gopt](http://www.purposeful.co.uk/software/gopt/)
- - [docopt](http://docopt.org/)
-
-A few C++ command-line parsing libraries:
-
- - [TCLAP](http://tclap.sourceforge.net/)
- - [program_options](http://www.boost.org/doc/libs/1_58_0/doc/html/program_options.html)
- - [CommandLine](http://llvm.org/docs/CommandLine.html)
--- a/parg/parg.c
+++ /dev/null
@@ -1,354 +1,0 @@
-/*
- * parg - parse argv
- *
- * Written in 2015-2016 by Joergen Ibsen
- *
- * To the extent possible under law, the author(s) have dedicated all
- * copyright and related and neighboring rights to this software to the
- * public domain worldwide. This software is distributed without any
- * warranty. <http://creativecommons.org/publicdomain/zero/1.0/>
- */
-
-#include "parg.h"
-
-#include <assert.h>
-#include <stdlib.h>
-#include <string.h>
-
-/*
- * Check if state is at end of argv.
- */
-static int
-is_argv_end(const struct parg_state *ps, int argc, char *const argv[])
-{
- return ps->optind >= argc || argv[ps->optind] == NULL;
-}
-
-/*
- * Match nextchar against optstring.
- */
-static int
-match_short(struct parg_state *ps, int argc, char *const argv[],
- const char *optstring)
-{
- const char *p = strchr(optstring, *ps->nextchar);
-
- if (p == NULL) {
- ps->optopt = *ps->nextchar++;
- return '?';
- }
-
- /* If no option argument, return option */
- if (p[1] != ':') {
- return *ps->nextchar++;
- }
-
- /* If more characters, return as option argument */
- if (ps->nextchar[1] != '\0') {
- ps->optarg = &ps->nextchar[1];
- ps->nextchar = NULL;
- return *p;
- }
-
- /* If option argument is optional, return option */
- if (p[2] == ':') {
- return *ps->nextchar++;
- }
-
- /* Option argument required, so return next argv element */
- if (is_argv_end(ps, argc, argv)) {
- ps->optopt = *ps->nextchar++;
- return optstring[0] == ':' ? ':' : '?';
- }
-
- ps->optarg = argv[ps->optind++];
- ps->nextchar = NULL;
- return *p;
-}
-
-/*
- * Match string at nextchar against longopts.
- */
-static int
-match_long(struct parg_state *ps, int argc, char *const argv[],
- const char *optstring,
- const struct parg_option *longopts, int *longindex)
-{
- size_t len;
- int num_match = 0;
- int match = -1;
- int i;
-
- len = strcspn(ps->nextchar, "=");
-
- for (i = 0; longopts[i].name; ++i) {
- if (strncmp(ps->nextchar, longopts[i].name, len) == 0) {
- match = i;
- num_match++;
- /* Take if exact match */
- if (longopts[i].name[len] == '\0') {
- num_match = 1;
- break;
- }
- }
- }
-
- /* Return '?' on no or ambiguous match */
- if (num_match != 1) {
- ps->optopt = 0;
- ps->nextchar = NULL;
- return '?';
- }
-
- assert(match != -1);
-
- if (longindex) {
- *longindex = match;
- }
-
- if (ps->nextchar[len] == '=') {
- /* Option argument present, check if extraneous */
- if (longopts[match].has_arg == PARG_NOARG) {
- ps->optopt = longopts[match].flag ? 0 : longopts[match].val;
- ps->nextchar = NULL;
- return optstring[0] == ':' ? ':' : '?';
- }
- else {
- ps->optarg = &ps->nextchar[len + 1];
- }
- }
- else if (longopts[match].has_arg == PARG_REQARG) {
- /* Option argument required, so return next argv element */
- if (is_argv_end(ps, argc, argv)) {
- ps->optopt = longopts[match].flag ? 0 : longopts[match].val;
- ps->nextchar = NULL;
- return optstring[0] == ':' ? ':' : '?';
- }
-
- ps->optarg = argv[ps->optind++];
- }
-
- ps->nextchar = NULL;
-
- if (longopts[match].flag != NULL) {
- *longopts[match].flag = longopts[match].val;
- return 0;
- }
-
- return longopts[match].val;
-}
-
-void
-parg_init(struct parg_state *ps)
-{
- ps->optarg = NULL;
- ps->optind = 1;
- ps->optopt = '?';
- ps->nextchar = NULL;
-}
-
-int
-parg_getopt(struct parg_state *ps, int argc, char *const argv[],
- const char *optstring)
-{
- return parg_getopt_long(ps, argc, argv, optstring, NULL, NULL);
-}
-
-int
-parg_getopt_long(struct parg_state *ps, int argc, char *const argv[],
- const char *optstring,
- const struct parg_option *longopts, int *longindex)
-{
- assert(ps != NULL);
- assert(argv != NULL);
- assert(optstring != NULL);
-
- ps->optarg = NULL;
-
- if (argc < 2) {
- return -1;
- }
-
- /* Advance to next element if needed */
- if (ps->nextchar == NULL || *ps->nextchar == '\0') {
- if (is_argv_end(ps, argc, argv)) {
- return -1;
- }
-
- ps->nextchar = argv[ps->optind++];
-
- /* Check for nonoption element (including '-') */
- if (ps->nextchar[0] != '-' || ps->nextchar[1] == '\0') {
- ps->optarg = ps->nextchar;
- ps->nextchar = NULL;
- return 1;
- }
-
- /* Check for '--' */
- if (ps->nextchar[1] == '-') {
- if (ps->nextchar[2] == '\0') {
- ps->nextchar = NULL;
- return -1;
- }
-
- if (longopts != NULL) {
- ps->nextchar += 2;
-
- return match_long(ps, argc, argv, optstring,
- longopts, longindex);
- }
- }
-
- ps->nextchar++;
- }
-
- /* Match nextchar */
- return match_short(ps, argc, argv, optstring);
-}
-
-/*
- * Reverse elements of `v` from `i` to `j`.
- */
-static void
-reverse(char *v[], int i, int j)
-{
- while (j - i > 1) {
- char *tmp = v[i];
- v[i] = v[j - 1];
- v[j - 1] = tmp;
- ++i;
- --j;
- }
-}
-
-/*
- * Reorder elements of `argv` with no special cases.
- *
- * This function assumes there is no `--` element, and the last element
- * is not an option missing a required argument.
- *
- * The algorithm is described here:
- * http://hardtoc.com/2016/11/07/reordering-arguments.html
- */
-static int
-parg_reorder_simple(int argc, char *argv[],
- const char *optstring,
- const struct parg_option *longopts)
-{
- struct parg_state ps;
- int change;
- int l = 0;
- int m = 0;
- int r = 0;
-
- if (argc < 2) {
- return argc;
- }
-
- do {
- int nextind;
- int c;
-
- parg_init(&ps);
-
- nextind = ps.optind;
-
- /* Parse until end of argument */
- do {
- c = parg_getopt_long(&ps, argc, argv, optstring, longopts, NULL);
- } while (ps.nextchar != NULL && *ps.nextchar != '\0');
-
- change = 0;
-
- do {
- /* Find next non-option */
- for (l = nextind; c != 1 && c != -1;) {
- l = ps.optind;
-
- do {
- c = parg_getopt_long(&ps, argc, argv, optstring, longopts, NULL);
- } while (ps.nextchar != NULL && *ps.nextchar != '\0');
- }
-
- /* Find next option */
- for (m = l; c == 1;) {
- m = ps.optind;
-
- do {
- c = parg_getopt_long(&ps, argc, argv, optstring, longopts, NULL);
- } while (ps.nextchar != NULL && *ps.nextchar != '\0');
- }
-
- /* Find next non-option */
- for (r = m; c != 1 && c != -1;) {
- r = ps.optind;
-
- do {
- c = parg_getopt_long(&ps, argc, argv, optstring, longopts, NULL);
- } while (ps.nextchar != NULL && *ps.nextchar != '\0');
- }
-
- /* Find next option */
- for (nextind = r; c == 1;) {
- nextind = ps.optind;
-
- do {
- c = parg_getopt_long(&ps, argc, argv, optstring, longopts, NULL);
- } while (ps.nextchar != NULL && *ps.nextchar != '\0');
- }
-
- if (m < r) {
- change = 1;
- reverse(argv, l, m);
- reverse(argv, m, r);
- reverse(argv, l, r);
- }
- } while (c != -1);
- } while (change != 0);
-
- return l + (r - m);
-}
-
-int
-parg_reorder(int argc, char *argv[],
- const char *optstring,
- const struct parg_option *longopts)
-{
- struct parg_state ps;
- int lastind;
- int optend;
- int c;
-
- assert(argv != NULL);
- assert(optstring != NULL);
-
- if (argc < 2) {
- return argc;
- }
-
- parg_init(&ps);
-
- /* Find end of normal arguments */
- do {
- lastind = ps.optind;
-
- c = parg_getopt_long(&ps, argc, argv, optstring, longopts, NULL);
-
- /* Check for trailing option with error */
- if ((c == '?' || c == ':') && is_argv_end(&ps, argc, argv)) {
- lastind = ps.optind - 1;
- break;
- }
- } while (c != -1);
-
- optend = parg_reorder_simple(lastind, argv, optstring, longopts);
-
- /* Rotate `--` or trailing option with error into position */
- if (lastind < argc) {
- reverse(argv, optend, lastind);
- reverse(argv, optend, lastind + 1);
- ++optend;
- }
-
- return optend;
-}
--- a/parg/parg.h
+++ /dev/null
@@ -1,192 +1,0 @@
-/*
- * parg - parse argv
- *
- * Written in 2015-2016 by Joergen Ibsen
- *
- * To the extent possible under law, the author(s) have dedicated all
- * copyright and related and neighboring rights to this software to the
- * public domain worldwide. This software is distributed without any
- * warranty. <http://creativecommons.org/publicdomain/zero/1.0/>
- */
-
-#ifndef PARG_H_INCLUDED
-#define PARG_H_INCLUDED
-
-#ifdef __cplusplus
-extern "C" {
-#endif
-
-#define PARG_VER_MAJOR 1 /**< Major version number */
-#define PARG_VER_MINOR 0 /**< Minor version number */
-#define PARG_VER_PATCH 2 /**< Patch version number */
-#define PARG_VER_STRING "1.0.2" /**< Version number as a string */
-
-/**
- * Structure containing state between calls to parser.
- *
- * @see parg_init
- */
-struct parg_state {
- const char *optarg; /**< Pointer to option argument, if any */
- int optind; /**< Next index in argv to process */
- int optopt; /**< Option value resulting in error, if any */
- const char *nextchar; /**< Next character to process */
-};
-
-/**
- * Structure for supplying long options to `parg_getopt_long()`.
- *
- * @see parg_getopt_long
- */
-struct parg_option {
- const char *name; /**< Name of option */
- int has_arg; /**< Option argument status */
- int *flag; /**< Pointer to flag variable */
- int val; /**< Value of option */
-};
-
-/**
- * Values for `has_arg` flag in `parg_option`.
- *
- * @see parg_option
- */
-typedef enum {
- PARG_NOARG, /**< No argument */
- PARG_REQARG, /**< Required argument */
- PARG_OPTARG /**< Optional argument */
-} parg_arg_num;
-
-/**
- * Initialize `ps`.
- *
- * Must be called before using state with a parser.
- *
- * @see parg_state
- *
- * @param ps pointer to state
- */
-void
-parg_init(struct parg_state *ps);
-
-/**
- * Parse next short option in `argv`.
- *
- * Elements in `argv` that contain short options start with a single dash
- * followed by one or more option characters, and optionally an option
- * argument for the last option character. Examples are '`-d`', '`-ofile`',
- * and '`-dofile`'.
- *
- * Consecutive calls to this function match the command-line arguments in
- * `argv` against the short option characters in `optstring`.
- *
- * If an option character in `optstring` is followed by a colon, '`:`', the
- * option requires an argument. If it is followed by two colons, the option
- * may take an optional argument.
- *
- * If a match is found, `optarg` points to the option argument, if any, and
- * the value of the option character is returned.
- *
- * If a match is found, but is missing a required option argument, `optopt`
- * is set to the option character. If the first character in `optstring` is
- * '`:`', then '`:`' is returned, otherwise '`?`' is returned.
- *
- * If no option character in `optstring` matches a short option, `optopt`
- * is set to the option character, and '`?`' is returned.
- *
- * If an element of argv does not contain options (a nonoption element),
- * `optarg` points to the element, and `1` is returned.
- *
- * An element consisting of a single dash, '`-`', is returned as a nonoption.
- *
- * Parsing stops and `-1` is returned, when the end of `argv` is reached, or
- * if an element contains '`--`'.
- *
- * Works similarly to `getopt`, if `optstring` were prefixed by '`-`'.
- *
- * @param ps pointer to state
- * @param argc number of elements in `argv`
- * @param argv array of pointers to command-line arguments
- * @param optstring string containing option characters
- * @return option value on match, `1` on nonoption element, `-1` on end of
- * arguments, '`?`' on unmatched option, '`?`' or '`:`' on option argument
- * error
- */
-int
-parg_getopt(struct parg_state *ps, int argc, char *const argv[],
- const char *optstring);
-
-/**
- * Parse next long or short option in `argv`.
- *
- * Elements in `argv` that contain a long option start with two dashes
- * followed by a string, and optionally an equal sign and an option argument.
- * Examples are '`--help`' and '`--size=5`'.
- *
- * If no exact match is found, an unambiguous prefix of a long option will
- * match. For example, if '`foo`' and '`foobar`' are valid long options, then
- * '`--fo`' is ambiguous and will not match, '`--foo`' matches exactly, and
- * '`--foob`' is an unambiguous prefix and will match.
- *
- * If a long option match is found, and `flag` is `NULL`, `val` is returned.
- *
- * If a long option match is found, and `flag` is not `NULL`, `val` is stored
- * in the variable `flag` points to, and `0` is returned.
- *
- * If a long option match is found, but is missing a required option argument,
- * or has an option argument even though it takes none, `optopt` is set to
- * `val` if `flag` is `NULL`, and `0` otherwise. If the first character in
- * `optstring` is '`:`', then '`:`' is returned, otherwise '`?`' is returned.
- *
- * If `longindex` is not `NULL`, the index of the entry in `longopts` that
- * matched is stored there.
- *
- * If no long option in `longopts` matches a long option, '`?`' is returned.
- *
- * Handling of nonoptions and short options is like `parg_getopt()`.
- *
- * If no short options are required, an empty string, `""`, should be passed
- * as `optstring`.
- *
- * Works similarly to `getopt_long`, if `optstring` were prefixed by '`-`'.
- *
- * @see parg_getopt
- *
- * @param ps pointer to state
- * @param argc number of elements in `argv`
- * @param argv array of pointers to command-line arguments
- * @param optstring string containing option characters
- * @param longopts array of `parg_option` structures
- * @param longindex pointer to variable to store index of matching option in
- * @return option value on match, `0` for flag option, `1` on nonoption
- * element, `-1` on end of arguments, '`?`' on unmatched or ambiguous option,
- * '`?`' or '`:`' on option argument error
- */
-int
-parg_getopt_long(struct parg_state *ps, int argc, char *const argv[],
- const char *optstring,
- const struct parg_option *longopts, int *longindex);
-
-/**
- * Reorder elements of `argv` so options appear first.
- *
- * If there are no long options, `longopts` may be `NULL`.
- *
- * The return value can be used as `argc` parameter for `parg_getopt()` and
- * `parg_getopt_long()`.
- *
- * @param argc number of elements in `argv`
- * @param argv array of pointers to command-line arguments
- * @param optstring string containing option characters
- * @param longopts array of `parg_option` structures
- * @return index of first nonoption in `argv` on success, `-1` on error
- */
-int
-parg_reorder(int argc, char *argv[],
- const char *optstring,
- const struct parg_option *longopts);
-
-#ifdef __cplusplus
-} /* extern "C" */
-#endif
-
-#endif /* PARG_H_INCLUDED */
--- a/parg/test/greatest.h
+++ /dev/null
@@ -1,1035 +1,0 @@
-/*
- * Copyright (c) 2011-2016 Scott Vokes <vokes.s@gmail.com>
- *
- * Permission to use, copy, modify, and/or distribute this software for any
- * purpose with or without fee is hereby granted, provided that the above
- * copyright notice and this permission notice appear in all copies.
- *
- * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
- * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
- * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
- * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
- * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
- * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
- * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
- */
-
-#ifndef GREATEST_H
-#define GREATEST_H
-
-#ifdef __cplusplus
-extern "C" {
-#endif
-
-/* 1.2.1 */
-#define GREATEST_VERSION_MAJOR 1
-#define GREATEST_VERSION_MINOR 2
-#define GREATEST_VERSION_PATCH 1
-
-/* A unit testing system for C, contained in 1 file.
- * It doesn't use dynamic allocation or depend on anything
- * beyond ANSI C89.
- *
- * An up-to-date version can be found at:
- * https://github.com/silentbicycle/greatest/
- */
-
-
-/*********************************************************************
- * Minimal test runner template
- *********************************************************************/
-#if 0
-
-#include "greatest.h"
-
-TEST foo_should_foo(void) {
- PASS();
-}
-
-static void setup_cb(void *data) {
- printf("setup callback for each test case\n");
-}
-
-static void teardown_cb(void *data) {
- printf("teardown callback for each test case\n");
-}
-
-SUITE(suite) {
- /* Optional setup/teardown callbacks which will be run before/after
- * every test case. If using a test suite, they will be cleared when
- * the suite finishes. */
- SET_SETUP(setup_cb, voidp_to_callback_data);
- SET_TEARDOWN(teardown_cb, voidp_to_callback_data);
-
- RUN_TEST(foo_should_foo);
-}
-
-/* Add definitions that need to be in the test runner's main file. */
-GREATEST_MAIN_DEFS();
-
-/* Set up, run suite(s) of tests, report pass/fail/skip stats. */
-int run_tests(void) {
- GREATEST_INIT(); /* init. greatest internals */
- /* List of suites to run (if any). */
- RUN_SUITE(suite);
-
- /* Tests can also be run directly, without using test suites. */
- RUN_TEST(foo_should_foo);
-
- GREATEST_PRINT_REPORT(); /* display results */
- return greatest_all_passed();
-}
-
-/* main(), for a standalone command-line test runner.
- * This replaces run_tests above, and adds command line option
- * handling and exiting with a pass/fail status. */
-int main(int argc, char **argv) {
- GREATEST_MAIN_BEGIN(); /* init & parse command-line args */
- RUN_SUITE(suite);
- GREATEST_MAIN_END(); /* display results */
-}
-
-#endif
-/*********************************************************************/
-
-
-#include <stdlib.h>
-#include <stdio.h>
-#include <string.h>
-#include <ctype.h>
-
-/***********
- * Options *
- ***********/
-
-/* Default column width for non-verbose output. */
-#ifndef GREATEST_DEFAULT_WIDTH
-#define GREATEST_DEFAULT_WIDTH 72
-#endif
-
-/* FILE *, for test logging. */
-#ifndef GREATEST_STDOUT
-#define GREATEST_STDOUT stdout
-#endif
-
-/* Remove GREATEST_ prefix from most commonly used symbols? */
-#ifndef GREATEST_USE_ABBREVS
-#define GREATEST_USE_ABBREVS 1
-#endif
-
-/* Set to 0 to disable all use of setjmp/longjmp. */
-#ifndef GREATEST_USE_LONGJMP
-#define GREATEST_USE_LONGJMP 1
-#endif
-
-#if GREATEST_USE_LONGJMP
-#include <setjmp.h>
-#endif
-
-/* Set to 0 to disable all use of time.h / clock(). */
-#ifndef GREATEST_USE_TIME
-#define GREATEST_USE_TIME 1
-#endif
-
-#if GREATEST_USE_TIME
-#include <time.h>
-#endif
-
-/* Floating point type, for ASSERT_IN_RANGE. */
-#ifndef GREATEST_FLOAT
-#define GREATEST_FLOAT double
-#define GREATEST_FLOAT_FMT "%g"
-#endif
-
-/*********
- * Types *
- *********/
-
-/* Info for the current running suite. */
-typedef struct greatest_suite_info {
- unsigned int tests_run;
- unsigned int passed;
- unsigned int failed;
- unsigned int skipped;
-
-#if GREATEST_USE_TIME
- /* timers, pre/post running suite and individual tests */
- clock_t pre_suite;
- clock_t post_suite;
- clock_t pre_test;
- clock_t post_test;
-#endif
-} greatest_suite_info;
-
-/* Type for a suite function. */
-typedef void (greatest_suite_cb)(void);
-
-/* Types for setup/teardown callbacks. If non-NULL, these will be run
- * and passed the pointer to their additional data. */
-typedef void (greatest_setup_cb)(void *udata);
-typedef void (greatest_teardown_cb)(void *udata);
-
-/* Type for an equality comparison between two pointers of the same type.
- * Should return non-0 if equal, otherwise 0.
- * UDATA is a closure value, passed through from ASSERT_EQUAL_T[m]. */
-typedef int greatest_equal_cb(const void *exp, const void *got, void *udata);
-
-/* Type for a callback that prints a value pointed to by T.
- * Return value has the same meaning as printf's.
- * UDATA is a closure value, passed through from ASSERT_EQUAL_T[m]. */
-typedef int greatest_printf_cb(const void *t, void *udata);
-
-/* Callbacks for an arbitrary type; needed for type-specific
- * comparisons via GREATEST_ASSERT_EQUAL_T[m].*/
-typedef struct greatest_type_info {
- greatest_equal_cb *equal;
- greatest_printf_cb *print;
-} greatest_type_info;
-
-typedef struct greatest_memory_cmp_env {
- const unsigned char *exp;
- const unsigned char *got;
- size_t size;
-} greatest_memory_cmp_env;
-
-/* Callbacks for string and raw memory types. */
-extern greatest_type_info greatest_type_info_string;
-extern greatest_type_info greatest_type_info_memory;
-
-typedef enum {
- GREATEST_FLAG_FIRST_FAIL = 0x01,
- GREATEST_FLAG_LIST_ONLY = 0x02
-} greatest_flag_t;
-
-/* Struct containing all test runner state. */
-typedef struct greatest_run_info {
- unsigned char flags;
- unsigned char verbosity;
- unsigned int tests_run; /* total test count */
-
- /* overall pass/fail/skip counts */
- unsigned int passed;
- unsigned int failed;
- unsigned int skipped;
- unsigned int assertions;
-
- /* currently running test suite */
- greatest_suite_info suite;
-
- /* info to print about the most recent failure */
- const char *fail_file;
- unsigned int fail_line;
- const char *msg;
-
- /* current setup/teardown hooks and userdata */
- greatest_setup_cb *setup;
- void *setup_udata;
- greatest_teardown_cb *teardown;
- void *teardown_udata;
-
- /* formatting info for ".....s...F"-style output */
- unsigned int col;
- unsigned int width;
-
- /* only run a specific suite or test */
- const char *suite_filter;
- const char *test_filter;
-
-#if GREATEST_USE_TIME
- /* overall timers */
- clock_t begin;
- clock_t end;
-#endif
-
-#if GREATEST_USE_LONGJMP
- jmp_buf jump_dest;
-#endif
-} greatest_run_info;
-
-struct greatest_report_t {
- /* overall pass/fail/skip counts */
- unsigned int passed;
- unsigned int failed;
- unsigned int skipped;
- unsigned int assertions;
-};
-
-/* Global var for the current testing context.
- * Initialized by GREATEST_MAIN_DEFS(). */
-extern greatest_run_info greatest_info;
-
-/* Type for ASSERT_ENUM_EQ's ENUM_STR argument. */
-typedef const char *greatest_enum_str_fun(int value);
-
-/**********************
- * Exported functions *
- **********************/
-
-/* These are used internally by greatest. */
-void greatest_do_pass(const char *name);
-void greatest_do_fail(const char *name);
-void greatest_do_skip(const char *name);
-int greatest_pre_test(const char *name);
-void greatest_post_test(const char *name, int res);
-void greatest_usage(const char *name);
-int greatest_do_assert_equal_t(const void *exp, const void *got,
- greatest_type_info *type_info, void *udata);
-
-/* These are part of the public greatest API. */
-void GREATEST_SET_SETUP_CB(greatest_setup_cb *cb, void *udata);
-void GREATEST_SET_TEARDOWN_CB(greatest_teardown_cb *cb, void *udata);
-int greatest_all_passed(void);
-void greatest_set_test_filter(const char *name);
-void greatest_set_suite_filter(const char *name);
-void greatest_get_report(struct greatest_report_t *report);
-unsigned int greatest_get_verbosity(void);
-void greatest_set_verbosity(unsigned int verbosity);
-void greatest_set_flag(greatest_flag_t flag);
-
-
-/********************
-* Language Support *
-********************/
-
-/* If __VA_ARGS__ (C99) is supported, allow parametric testing
-* without needing to manually manage the argument struct. */
-#if __STDC_VERSION__ >= 19901L || _MSC_VER >= 1800
-#define GREATEST_VA_ARGS
-#endif
-
-
-/**********
- * Macros *
- **********/
-
-/* Define a suite. */
-#define GREATEST_SUITE(NAME) void NAME(void); void NAME(void)
-
-/* Declare a suite, provided by another compilation unit. */
-#define GREATEST_SUITE_EXTERN(NAME) void NAME(void)
-
-/* Start defining a test function.
- * The arguments are not included, to allow parametric testing. */
-#define GREATEST_TEST static enum greatest_test_res
-
-/* PASS/FAIL/SKIP result from a test. Used internally. */
-typedef enum greatest_test_res {
- GREATEST_TEST_RES_PASS = 0,
- GREATEST_TEST_RES_FAIL = -1,
- GREATEST_TEST_RES_SKIP = 1
-} greatest_test_res;
-
-/* Run a suite. */
-#define GREATEST_RUN_SUITE(S_NAME) greatest_run_suite(S_NAME, #S_NAME)
-
-/* Run a test in the current suite. */
-#define GREATEST_RUN_TEST(TEST) \
- do { \
- if (greatest_pre_test(#TEST) == 1) { \
- enum greatest_test_res res = GREATEST_SAVE_CONTEXT(); \
- if (res == GREATEST_TEST_RES_PASS) { \
- res = TEST(); \
- } \
- greatest_post_test(#TEST, res); \
- } else if (GREATEST_LIST_ONLY()) { \
- fprintf(GREATEST_STDOUT, " %s\n", #TEST); \
- } \
- } while (0)
-
-/* Ignore a test, don't warn about it being unused. */
-#define GREATEST_IGNORE_TEST(TEST) (void)TEST
-
-/* Run a test in the current suite with one void * argument,
- * which can be a pointer to a struct with multiple arguments. */
-#define GREATEST_RUN_TEST1(TEST, ENV) \
- do { \
- if (greatest_pre_test(#TEST) == 1) { \
- int res = TEST(ENV); \
- greatest_post_test(#TEST, res); \
- } else if (GREATEST_LIST_ONLY()) { \
- fprintf(GREATEST_STDOUT, " %s\n", #TEST); \
- } \
- } while (0)
-
-#ifdef GREATEST_VA_ARGS
-#define GREATEST_RUN_TESTp(TEST, ...) \
- do { \
- if (greatest_pre_test(#TEST) == 1) { \
- int res = TEST(__VA_ARGS__); \
- greatest_post_test(#TEST, res); \
- } else if (GREATEST_LIST_ONLY()) { \
- fprintf(GREATEST_STDOUT, " %s\n", #TEST); \
- } \
- } while (0)
-#endif
-
-
-/* Check if the test runner is in verbose mode. */
-#define GREATEST_IS_VERBOSE() ((greatest_info.verbosity) > 0)
-#define GREATEST_LIST_ONLY() \
- (greatest_info.flags & GREATEST_FLAG_LIST_ONLY)
-#define GREATEST_FIRST_FAIL() \
- (greatest_info.flags & GREATEST_FLAG_FIRST_FAIL)
-#define GREATEST_FAILURE_ABORT() \
- (greatest_info.suite.failed > 0 && GREATEST_FIRST_FAIL())
-
-/* Message-less forms of tests defined below. */
-#define GREATEST_PASS() GREATEST_PASSm(NULL)
-#define GREATEST_FAIL() GREATEST_FAILm(NULL)
-#define GREATEST_SKIP() GREATEST_SKIPm(NULL)
-#define GREATEST_ASSERT(COND) \
- GREATEST_ASSERTm(#COND, COND)
-#define GREATEST_ASSERT_OR_LONGJMP(COND) \
- GREATEST_ASSERT_OR_LONGJMPm(#COND, COND)
-#define GREATEST_ASSERT_FALSE(COND) \
- GREATEST_ASSERT_FALSEm(#COND, COND)
-#define GREATEST_ASSERT_EQ(EXP, GOT) \
- GREATEST_ASSERT_EQm(#EXP " != " #GOT, EXP, GOT)
-#define GREATEST_ASSERT_EQ_FMT(EXP, GOT, FMT) \
- GREATEST_ASSERT_EQ_FMTm(#EXP " != " #GOT, EXP, GOT, FMT)
-#define GREATEST_ASSERT_IN_RANGE(EXP, GOT, TOL) \
- GREATEST_ASSERT_IN_RANGEm(#EXP " != " #GOT " +/- " #TOL, EXP, GOT, TOL)
-#define GREATEST_ASSERT_EQUAL_T(EXP, GOT, TYPE_INFO, UDATA) \
- GREATEST_ASSERT_EQUAL_Tm(#EXP " != " #GOT, EXP, GOT, TYPE_INFO, UDATA)
-#define GREATEST_ASSERT_STR_EQ(EXP, GOT) \
- GREATEST_ASSERT_STR_EQm(#EXP " != " #GOT, EXP, GOT)
-#define GREATEST_ASSERT_STRN_EQ(EXP, GOT, SIZE) \
- GREATEST_ASSERT_STRN_EQm(#EXP " != " #GOT, EXP, GOT, SIZE)
-#define GREATEST_ASSERT_MEM_EQ(EXP, GOT, SIZE) \
- GREATEST_ASSERT_MEM_EQm(#EXP " != " #GOT, EXP, GOT, SIZE)
-#define GREATEST_ASSERT_ENUM_EQ(EXP, GOT, ENUM_STR) \
- GREATEST_ASSERT_ENUM_EQm(#EXP " != " #GOT, EXP, GOT, ENUM_STR)
-
-/* The following forms take an additional message argument first,
- * to be displayed by the test runner. */
-
-/* Fail if a condition is not true, with message. */
-#define GREATEST_ASSERTm(MSG, COND) \
- do { \
- greatest_info.assertions++; \
- if (!(COND)) { GREATEST_FAILm(MSG); } \
- } while (0)
-
-/* Fail if a condition is not true, longjmping out of test. */
-#define GREATEST_ASSERT_OR_LONGJMPm(MSG, COND) \
- do { \
- greatest_info.assertions++; \
- if (!(COND)) { GREATEST_FAIL_WITH_LONGJMPm(MSG); } \
- } while (0)
-
-/* Fail if a condition is not false, with message. */
-#define GREATEST_ASSERT_FALSEm(MSG, COND) \
- do { \
- greatest_info.assertions++; \
- if ((COND)) { GREATEST_FAILm(MSG); } \
- } while (0)
-
-/* Fail if EXP != GOT (equality comparison by ==). */
-#define GREATEST_ASSERT_EQm(MSG, EXP, GOT) \
- do { \
- greatest_info.assertions++; \
- if ((EXP) != (GOT)) { GREATEST_FAILm(MSG); } \
- } while (0)
-
-/* Fail if EXP != GOT (equality comparison by ==).
- * Warning: EXP and GOT will be evaluated more than once on failure. */
-#define GREATEST_ASSERT_EQ_FMTm(MSG, EXP, GOT, FMT) \
- do { \
- const char *greatest_FMT = ( FMT ); \
- greatest_info.assertions++; \
- if ((EXP) != (GOT)) { \
- fprintf(GREATEST_STDOUT, "\nExpected: "); \
- fprintf(GREATEST_STDOUT, greatest_FMT, EXP); \
- fprintf(GREATEST_STDOUT, "\n Got: "); \
- fprintf(GREATEST_STDOUT, greatest_FMT, GOT); \
- fprintf(GREATEST_STDOUT, "\n"); \
- GREATEST_FAILm(MSG); \
- } \
- } while (0)
-
-/* Fail if EXP is not equal to GOT, printing enum IDs. */
-#define GREATEST_ASSERT_ENUM_EQm(MSG, EXP, GOT, ENUM_STR) \
- do { \
- int greatest_EXP = (int)(EXP); \
- int greatest_GOT = (int)(GOT); \
- greatest_enum_str_fun *greatest_ENUM_STR = ENUM_STR; \
- if (greatest_EXP != greatest_GOT) { \
- fprintf(GREATEST_STDOUT, "\nExpected: %s", \
- greatest_ENUM_STR(greatest_EXP)); \
- fprintf(GREATEST_STDOUT, "\n Got: %s\n", \
- greatest_ENUM_STR(greatest_GOT)); \
- GREATEST_FAILm(MSG); \
- } \
- } while (0) \
-
-/* Fail if GOT not in range of EXP +|- TOL. */
-#define GREATEST_ASSERT_IN_RANGEm(MSG, EXP, GOT, TOL) \
- do { \
- GREATEST_FLOAT greatest_EXP = (EXP); \
- GREATEST_FLOAT greatest_GOT = (GOT); \
- GREATEST_FLOAT greatest_TOL = (TOL); \
- greatest_info.assertions++; \
- if ((greatest_EXP > greatest_GOT && \
- greatest_EXP - greatest_GOT > greatest_TOL) || \
- (greatest_EXP < greatest_GOT && \
- greatest_GOT - greatest_EXP > greatest_TOL)) { \
- fprintf(GREATEST_STDOUT, \
- "\nExpected: " GREATEST_FLOAT_FMT \
- " +/- " GREATEST_FLOAT_FMT \
- "\n Got: " GREATEST_FLOAT_FMT \
- "\n", \
- greatest_EXP, greatest_TOL, greatest_GOT); \
- GREATEST_FAILm(MSG); \
- } \
- } while (0)
-
-/* Fail if EXP is not equal to GOT, according to strcmp. */
-#define GREATEST_ASSERT_STR_EQm(MSG, EXP, GOT) \
- do { \
- GREATEST_ASSERT_EQUAL_Tm(MSG, EXP, GOT, \
- &greatest_type_info_string, NULL); \
- } while (0) \
-
-/* Fail if EXP is not equal to GOT, according to strcmp. */
-#define GREATEST_ASSERT_STRN_EQm(MSG, EXP, GOT, SIZE) \
- do { \
- size_t size = SIZE; \
- GREATEST_ASSERT_EQUAL_Tm(MSG, EXP, GOT, \
- &greatest_type_info_string, &size); \
- } while (0) \
-
-/* Fail if EXP is not equal to GOT, according to memcmp. */
-#define GREATEST_ASSERT_MEM_EQm(MSG, EXP, GOT, SIZE) \
- do { \
- greatest_memory_cmp_env env; \
- env.exp = (const unsigned char *)EXP; \
- env.got = (const unsigned char *)GOT; \
- env.size = SIZE; \
- GREATEST_ASSERT_EQUAL_Tm(MSG, env.exp, env.got, \
- &greatest_type_info_memory, &env); \
- } while (0) \
-
-/* Fail if EXP is not equal to GOT, according to a comparison
- * callback in TYPE_INFO. If they are not equal, optionally use a
- * print callback in TYPE_INFO to print them. */
-#define GREATEST_ASSERT_EQUAL_Tm(MSG, EXP, GOT, TYPE_INFO, UDATA) \
- do { \
- greatest_type_info *type_info = (TYPE_INFO); \
- greatest_info.assertions++; \
- if (!greatest_do_assert_equal_t(EXP, GOT, \
- type_info, UDATA)) { \
- if (type_info == NULL || type_info->equal == NULL) { \
- GREATEST_FAILm("type_info->equal callback missing!"); \
- } else { \
- GREATEST_FAILm(MSG); \
- } \
- } \
- } while (0) \
-
-/* Pass. */
-#define GREATEST_PASSm(MSG) \
- do { \
- greatest_info.msg = MSG; \
- return GREATEST_TEST_RES_PASS; \
- } while (0)
-
-/* Fail. */
-#define GREATEST_FAILm(MSG) \
- do { \
- greatest_info.fail_file = __FILE__; \
- greatest_info.fail_line = __LINE__; \
- greatest_info.msg = MSG; \
- return GREATEST_TEST_RES_FAIL; \
- } while (0)
-
-/* Optional GREATEST_FAILm variant that longjmps. */
-#if GREATEST_USE_LONGJMP
-#define GREATEST_FAIL_WITH_LONGJMP() GREATEST_FAIL_WITH_LONGJMPm(NULL)
-#define GREATEST_FAIL_WITH_LONGJMPm(MSG) \
- do { \
- greatest_info.fail_file = __FILE__; \
- greatest_info.fail_line = __LINE__; \
- greatest_info.msg = MSG; \
- longjmp(greatest_info.jump_dest, GREATEST_TEST_RES_FAIL); \
- } while (0)
-#endif
-
-/* Skip the current test. */
-#define GREATEST_SKIPm(MSG) \
- do { \
- greatest_info.msg = MSG; \
- return GREATEST_TEST_RES_SKIP; \
- } while (0)
-
-/* Check the result of a subfunction using ASSERT, etc. */
-#define GREATEST_CHECK_CALL(RES) \
- do { \
- enum greatest_test_res greatest_RES = RES; \
- if (greatest_RES != GREATEST_TEST_RES_PASS) { \
- return greatest_RES; \
- } \
- } while (0) \
-
-#if GREATEST_USE_TIME
-#define GREATEST_SET_TIME(NAME) \
- NAME = clock(); \
- if (NAME == (clock_t) -1) { \
- fprintf(GREATEST_STDOUT, \
- "clock error: %s\n", #NAME); \
- exit(EXIT_FAILURE); \
- }
-
-#define GREATEST_CLOCK_DIFF(C1, C2) \
- fprintf(GREATEST_STDOUT, " (%lu ticks, %.3f sec)", \
- (long unsigned int) (C2) - (long unsigned int)(C1), \
- (double)((C2) - (C1)) / (1.0 * (double)CLOCKS_PER_SEC))
-#else
-#define GREATEST_SET_TIME(UNUSED)
-#define GREATEST_CLOCK_DIFF(UNUSED1, UNUSED2)
-#endif
-
-#if GREATEST_USE_LONGJMP
-#define GREATEST_SAVE_CONTEXT() \
- /* setjmp returns 0 (GREATEST_TEST_RES_PASS) on first call */ \
- /* so the test runs, then RES_FAIL from FAIL_WITH_LONGJMP. */ \
- ((enum greatest_test_res)(setjmp(greatest_info.jump_dest)))
-#else
-#define GREATEST_SAVE_CONTEXT() \
- /*a no-op, since setjmp/longjmp aren't being used */ \
- GREATEST_TEST_RES_PASS
-#endif
-
-/* Include several function definitions in the main test file. */
-#define GREATEST_MAIN_DEFS() \
- \
-/* Is FILTER a subset of NAME? */ \
-static int greatest_name_match(const char *name, \
- const char *filter) { \
- size_t offset = 0; \
- size_t filter_len = strlen(filter); \
- while (name[offset] != '\0') { \
- if (name[offset] == filter[0]) { \
- if (0 == strncmp(&name[offset], filter, filter_len)) { \
- return 1; \
- } \
- } \
- offset++; \
- } \
- \
- return 0; \
-} \
- \
-int greatest_pre_test(const char *name) { \
- if (!GREATEST_LIST_ONLY() \
- && (!GREATEST_FIRST_FAIL() || greatest_info.suite.failed == 0) \
- && (greatest_info.test_filter == NULL || \
- greatest_name_match(name, greatest_info.test_filter))) { \
- GREATEST_SET_TIME(greatest_info.suite.pre_test); \
- if (greatest_info.setup) { \
- greatest_info.setup(greatest_info.setup_udata); \
- } \
- return 1; /* test should be run */ \
- } else { \
- return 0; /* skipped */ \
- } \
-} \
- \
-void greatest_post_test(const char *name, int res) { \
- GREATEST_SET_TIME(greatest_info.suite.post_test); \
- if (greatest_info.teardown) { \
- void *udata = greatest_info.teardown_udata; \
- greatest_info.teardown(udata); \
- } \
- \
- if (res <= GREATEST_TEST_RES_FAIL) { \
- greatest_do_fail(name); \
- } else if (res >= GREATEST_TEST_RES_SKIP) { \
- greatest_do_skip(name); \
- } else if (res == GREATEST_TEST_RES_PASS) { \
- greatest_do_pass(name); \
- } \
- greatest_info.suite.tests_run++; \
- greatest_info.col++; \
- if (GREATEST_IS_VERBOSE()) { \
- GREATEST_CLOCK_DIFF(greatest_info.suite.pre_test, \
- greatest_info.suite.post_test); \
- fprintf(GREATEST_STDOUT, "\n"); \
- } else if (greatest_info.col % greatest_info.width == 0) { \
- fprintf(GREATEST_STDOUT, "\n"); \
- greatest_info.col = 0; \
- } \
- if (GREATEST_STDOUT == stdout) fflush(stdout); \
-} \
- \
-static void report_suite(void) { \
- if (greatest_info.suite.tests_run > 0) { \
- fprintf(GREATEST_STDOUT, \
- "\n%u test%s - %u passed, %u failed, %u skipped", \
- greatest_info.suite.tests_run, \
- greatest_info.suite.tests_run == 1 ? "" : "s", \
- greatest_info.suite.passed, \
- greatest_info.suite.failed, \
- greatest_info.suite.skipped); \
- GREATEST_CLOCK_DIFF(greatest_info.suite.pre_suite, \
- greatest_info.suite.post_suite); \
- fprintf(GREATEST_STDOUT, "\n"); \
- } \
-} \
- \
-static void update_counts_and_reset_suite(void) { \
- greatest_info.setup = NULL; \
- greatest_info.setup_udata = NULL; \
- greatest_info.teardown = NULL; \
- greatest_info.teardown_udata = NULL; \
- greatest_info.passed += greatest_info.suite.passed; \
- greatest_info.failed += greatest_info.suite.failed; \
- greatest_info.skipped += greatest_info.suite.skipped; \
- greatest_info.tests_run += greatest_info.suite.tests_run; \
- memset(&greatest_info.suite, 0, sizeof(greatest_info.suite)); \
- greatest_info.col = 0; \
-} \
- \
-static void greatest_run_suite(greatest_suite_cb *suite_cb, \
- const char *suite_name) { \
- if (greatest_info.suite_filter && \
- !greatest_name_match(suite_name, greatest_info.suite_filter)) { \
- return; \
- } \
- update_counts_and_reset_suite(); \
- if (GREATEST_FIRST_FAIL() && greatest_info.failed > 0) { return; } \
- fprintf(GREATEST_STDOUT, "\n* Suite %s:\n", suite_name); \
- GREATEST_SET_TIME(greatest_info.suite.pre_suite); \
- suite_cb(); \
- GREATEST_SET_TIME(greatest_info.suite.post_suite); \
- report_suite(); \
-} \
- \
-void greatest_do_pass(const char *name) { \
- if (GREATEST_IS_VERBOSE()) { \
- fprintf(GREATEST_STDOUT, "PASS %s: %s", \
- name, greatest_info.msg ? greatest_info.msg : ""); \
- } else { \
- fprintf(GREATEST_STDOUT, "."); \
- } \
- greatest_info.suite.passed++; \
-} \
- \
-void greatest_do_fail(const char *name) { \
- if (GREATEST_IS_VERBOSE()) { \
- fprintf(GREATEST_STDOUT, \
- "FAIL %s: %s (%s:%u)", \
- name, greatest_info.msg ? greatest_info.msg : "", \
- greatest_info.fail_file, greatest_info.fail_line); \
- } else { \
- fprintf(GREATEST_STDOUT, "F"); \
- greatest_info.col++; \
- /* add linebreak if in line of '.'s */ \
- if (greatest_info.col != 0) { \
- fprintf(GREATEST_STDOUT, "\n"); \
- greatest_info.col = 0; \
- } \
- fprintf(GREATEST_STDOUT, "FAIL %s: %s (%s:%u)\n", \
- name, \
- greatest_info.msg ? greatest_info.msg : "", \
- greatest_info.fail_file, greatest_info.fail_line); \
- } \
- greatest_info.suite.failed++; \
-} \
- \
-void greatest_do_skip(const char *name) { \
- if (GREATEST_IS_VERBOSE()) { \
- fprintf(GREATEST_STDOUT, "SKIP %s: %s", \
- name, \
- greatest_info.msg ? \
- greatest_info.msg : "" ); \
- } else { \
- fprintf(GREATEST_STDOUT, "s"); \
- } \
- greatest_info.suite.skipped++; \
-} \
- \
-int greatest_do_assert_equal_t(const void *exp, const void *got, \
- greatest_type_info *type_info, void *udata) { \
- int eq = 0; \
- if (type_info == NULL || type_info->equal == NULL) { \
- return 0; \
- } \
- eq = type_info->equal(exp, got, udata); \
- if (!eq) { \
- if (type_info->print != NULL) { \
- fprintf(GREATEST_STDOUT, "\nExpected: "); \
- (void)type_info->print(exp, udata); \
- fprintf(GREATEST_STDOUT, "\n Got: "); \
- (void)type_info->print(got, udata); \
- fprintf(GREATEST_STDOUT, "\n"); \
- } else { \
- fprintf(GREATEST_STDOUT, \
- "GREATEST_ASSERT_EQUAL_T failure at %s:%u\n", \
- greatest_info.fail_file, \
- greatest_info.fail_line); \
- } \
- } \
- return eq; \
-} \
- \
-void greatest_usage(const char *name) { \
- fprintf(GREATEST_STDOUT, \
- "Usage: %s [-hlfv] [-s SUITE] [-t TEST]\n" \
- " -h, --help print this Help\n" \
- " -l List suites and their tests, then exit\n" \
- " -f Stop runner after first failure\n" \
- " -v Verbose output\n" \
- " -s SUITE only run suites containing string SUITE\n" \
- " -t TEST only run tests containing string TEST\n", \
- name); \
-} \
- \
-static void greatest_parse_args(int argc, char **argv) { \
- int i = 0; \
- for (i = 1; i < argc; i++) { \
- if (0 == strncmp("-t", argv[i], 2)) { \
- if (argc <= i + 1) { \
- greatest_usage(argv[0]); \
- exit(EXIT_FAILURE); \
- } \
- greatest_info.test_filter = argv[i+1]; \
- i++; \
- } else if (0 == strncmp("-s", argv[i], 2)) { \
- if (argc <= i + 1) { \
- greatest_usage(argv[0]); \
- exit(EXIT_FAILURE); \
- } \
- greatest_info.suite_filter = argv[i+1]; \
- i++; \
- } else if (0 == strncmp("-f", argv[i], 2)) { \
- greatest_info.flags |= GREATEST_FLAG_FIRST_FAIL; \
- } else if (0 == strncmp("-v", argv[i], 2)) { \
- greatest_info.verbosity++; \
- } else if (0 == strncmp("-l", argv[i], 2)) { \
- greatest_info.flags |= GREATEST_FLAG_LIST_ONLY; \
- } else if (0 == strncmp("-h", argv[i], 2) || \
- 0 == strncmp("--help", argv[i], 6)) { \
- greatest_usage(argv[0]); \
- exit(EXIT_SUCCESS); \
- } else if (0 == strncmp("--", argv[i], 2)) { \
- break; \
- } else { \
- fprintf(GREATEST_STDOUT, \
- "Unknown argument '%s'\n", argv[i]); \
- greatest_usage(argv[0]); \
- exit(EXIT_FAILURE); \
- } \
- } \
-} \
- \
-int greatest_all_passed(void) { return (greatest_info.failed == 0); } \
- \
-void greatest_set_test_filter(const char *name) { \
- greatest_info.test_filter = name; \
-} \
- \
-void greatest_set_suite_filter(const char *name) { \
- greatest_info.suite_filter = name; \
-} \
- \
-void greatest_get_report(struct greatest_report_t *report) { \
- if (report) { \
- report->passed = greatest_info.passed; \
- report->failed = greatest_info.failed; \
- report->skipped = greatest_info.skipped; \
- report->assertions = greatest_info.assertions; \
- } \
-} \
- \
-unsigned int greatest_get_verbosity(void) { \
- return greatest_info.verbosity; \
-} \
- \
-void greatest_set_verbosity(unsigned int verbosity) { \
- greatest_info.verbosity = (unsigned char)verbosity; \
-} \
- \
-void greatest_set_flag(greatest_flag_t flag) { \
- greatest_info.flags |= flag; \
-} \
- \
-void GREATEST_SET_SETUP_CB(greatest_setup_cb *cb, void *udata) { \
- greatest_info.setup = cb; \
- greatest_info.setup_udata = udata; \
-} \
- \
-void GREATEST_SET_TEARDOWN_CB(greatest_teardown_cb *cb, \
- void *udata) { \
- greatest_info.teardown = cb; \
- greatest_info.teardown_udata = udata; \
-} \
- \
-static int greatest_string_equal_cb(const void *exp, const void *got, \
- void *udata) { \
- size_t *size = (size_t *)udata; \
- return (size != NULL \
- ? (0 == strncmp((const char *)exp, (const char *)got, *size)) \
- : (0 == strcmp((const char *)exp, (const char *)got))); \
-} \
- \
-static int greatest_string_printf_cb(const void *t, void *udata) { \
- (void)udata; /* note: does not check \0 termination. */ \
- return fprintf(GREATEST_STDOUT, "%s", (const char *)t); \
-} \
- \
-greatest_type_info greatest_type_info_string = { \
- greatest_string_equal_cb, \
- greatest_string_printf_cb, \
-}; \
- \
-static int greatest_memory_equal_cb(const void *exp, const void *got, \
- void *udata) { \
- greatest_memory_cmp_env *env = (greatest_memory_cmp_env *)udata; \
- return (0 == memcmp(exp, got, env->size)); \
-} \
- \
-static int greatest_memory_printf_cb(const void *t, void *udata) { \
- greatest_memory_cmp_env *env = (greatest_memory_cmp_env *)udata; \
- unsigned char *buf = (unsigned char *)t, diff_mark = ' '; \
- FILE *out = GREATEST_STDOUT; \
- size_t i, line_i, line_len = 0; \
- int len = 0; /* format hexdump with differences highlighted */ \
- for (i = 0; i < env->size; i+= line_len) { \
- diff_mark = ' '; \
- line_len = env->size - i; \
- if (line_len > 16) { line_len = 16; } \
- for (line_i = i; line_i < i + line_len; line_i++) { \
- if (env->exp[line_i] != env->got[line_i]) diff_mark = 'X'; \
- } \
- len += fprintf(out, "\n%04x %c ", (unsigned int)i, diff_mark); \
- for (line_i = i; line_i < i + line_len; line_i++) { \
- int m = env->exp[line_i] == env->got[line_i]; /* match? */ \
- len += fprintf(out, "%02x%c", buf[line_i], m ? ' ' : '<'); \
- } \
- for (line_i = 0; line_i < 16 - line_len; line_i++) { \
- len += fprintf(out, " "); \
- } \
- fprintf(out, " "); \
- for (line_i = i; line_i < i + line_len; line_i++) { \
- unsigned char c = buf[line_i]; \
- len += fprintf(out, "%c", isprint(c) ? c : '.'); \
- } \
- } \
- len += fprintf(out, "\n"); \
- return len; \
-} \
- \
-greatest_type_info greatest_type_info_memory = { \
- greatest_memory_equal_cb, \
- greatest_memory_printf_cb, \
-}; \
- \
-greatest_run_info greatest_info
-
-/* Init internals. */
-#define GREATEST_INIT() \
- do { \
- /* Suppress unused function warning if features aren't used */ \
- (void)greatest_run_suite; \
- (void)greatest_parse_args; \
- \
- memset(&greatest_info, 0, sizeof(greatest_info)); \
- greatest_info.width = GREATEST_DEFAULT_WIDTH; \
- GREATEST_SET_TIME(greatest_info.begin); \
- } while (0) \
-
-/* Handle command-line arguments, etc. */
-#define GREATEST_MAIN_BEGIN() \
- do { \
- GREATEST_INIT(); \
- greatest_parse_args(argc, argv); \
- } while (0)
-
-/* Report passes, failures, skipped tests, the number of
- * assertions, and the overall run time. */
-#define GREATEST_PRINT_REPORT() \
- do { \
- if (!GREATEST_LIST_ONLY()) { \
- update_counts_and_reset_suite(); \
- GREATEST_SET_TIME(greatest_info.end); \
- fprintf(GREATEST_STDOUT, \
- "\nTotal: %u test%s", \
- greatest_info.tests_run, \
- greatest_info.tests_run == 1 ? "" : "s"); \
- GREATEST_CLOCK_DIFF(greatest_info.begin, \
- greatest_info.end); \
- fprintf(GREATEST_STDOUT, ", %u assertion%s\n", \
- greatest_info.assertions, \
- greatest_info.assertions == 1 ? "" : "s"); \
- fprintf(GREATEST_STDOUT, \
- "Pass: %u, fail: %u, skip: %u.\n", \
- greatest_info.passed, \
- greatest_info.failed, greatest_info.skipped); \
- } \
- } while (0)
-
-/* Report results, exit with exit status based on results. */
-#define GREATEST_MAIN_END() \
- do { \
- GREATEST_PRINT_REPORT(); \
- return (greatest_all_passed() ? EXIT_SUCCESS : EXIT_FAILURE); \
- } while (0)
-
-/* Make abbreviations without the GREATEST_ prefix for the
- * most commonly used symbols. */
-#if GREATEST_USE_ABBREVS
-#define TEST GREATEST_TEST
-#define SUITE GREATEST_SUITE
-#define SUITE_EXTERN GREATEST_SUITE_EXTERN
-#define RUN_TEST GREATEST_RUN_TEST
-#define RUN_TEST1 GREATEST_RUN_TEST1
-#define RUN_SUITE GREATEST_RUN_SUITE
-#define IGNORE_TEST GREATEST_IGNORE_TEST
-#define ASSERT GREATEST_ASSERT
-#define ASSERTm GREATEST_ASSERTm
-#define ASSERT_FALSE GREATEST_ASSERT_FALSE
-#define ASSERT_EQ GREATEST_ASSERT_EQ
-#define ASSERT_EQ_FMT GREATEST_ASSERT_EQ_FMT
-#define ASSERT_IN_RANGE GREATEST_ASSERT_IN_RANGE
-#define ASSERT_EQUAL_T GREATEST_ASSERT_EQUAL_T
-#define ASSERT_STR_EQ GREATEST_ASSERT_STR_EQ
-#define ASSERT_STRN_EQ GREATEST_ASSERT_STRN_EQ
-#define ASSERT_MEM_EQ GREATEST_ASSERT_MEM_EQ
-#define ASSERT_ENUM_EQ GREATEST_ASSERT_ENUM_EQ
-#define ASSERT_FALSEm GREATEST_ASSERT_FALSEm
-#define ASSERT_EQm GREATEST_ASSERT_EQm
-#define ASSERT_EQ_FMTm GREATEST_ASSERT_EQ_FMTm
-#define ASSERT_IN_RANGEm GREATEST_ASSERT_IN_RANGEm
-#define ASSERT_EQUAL_Tm GREATEST_ASSERT_EQUAL_Tm
-#define ASSERT_STR_EQm GREATEST_ASSERT_STR_EQm
-#define ASSERT_STRN_EQm GREATEST_ASSERT_STRN_EQm
-#define ASSERT_MEM_EQm GREATEST_ASSERT_MEM_EQm
-#define ASSERT_ENUM_EQm GREATEST_ASSERT_ENUM_EQm
-#define PASS GREATEST_PASS
-#define FAIL GREATEST_FAIL
-#define SKIP GREATEST_SKIP
-#define PASSm GREATEST_PASSm
-#define FAILm GREATEST_FAILm
-#define SKIPm GREATEST_SKIPm
-#define SET_SETUP GREATEST_SET_SETUP_CB
-#define SET_TEARDOWN GREATEST_SET_TEARDOWN_CB
-#define CHECK_CALL GREATEST_CHECK_CALL
-
-#ifdef GREATEST_VA_ARGS
-#define RUN_TESTp GREATEST_RUN_TESTp
-#endif
-
-#if GREATEST_USE_LONGJMP
-#define ASSERT_OR_LONGJMP GREATEST_ASSERT_OR_LONGJMP
-#define ASSERT_OR_LONGJMPm GREATEST_ASSERT_OR_LONGJMPm
-#define FAIL_WITH_LONGJMP GREATEST_FAIL_WITH_LONGJMP
-#define FAIL_WITH_LONGJMPm GREATEST_FAIL_WITH_LONGJMPm
-#endif
-
-#endif /* USE_ABBREVS */
-
-#ifdef __cplusplus
-}
-#endif
-
-#endif
--- a/parg/test/test_parg.c
+++ /dev/null
@@ -1,703 +1,0 @@
-/*
- * test_parg - parg unit test
- *
- * Written in 2015-2016 by Joergen Ibsen
- *
- * To the extent possible under law, the author(s) have dedicated all
- * copyright and related and neighboring rights to this software to the
- * public domain worldwide. This software is distributed without any
- * warranty. <http://creativecommons.org/publicdomain/zero/1.0/>
- */
-
-#include "parg.h"
-
-#include <assert.h>
-#include <stdio.h>
-#include <stdlib.h>
-
-#include "greatest.h"
-
-#define ARRAY_SIZE(arr) (sizeof(arr) / sizeof((arr)[0]))
-
-static const char *os_def = ":no::r:";
-
-static const struct parg_option po_def[] = {
- { "noarg", PARG_NOARG, NULL, 'n' },
- { "optarg", PARG_OPTARG, NULL, 'o' },
- { "reqarg", PARG_REQARG, NULL, 'r' },
- { "foo", PARG_NOARG, NULL, 'f' },
- { "foobar", PARG_NOARG, NULL, 'b' },
- { 0, 0, 0, 0 }
-};
-
-/*
- * parg_getopt tests
- */
-
-TEST parg_getopt_app_only(void)
-{
- struct parg_state ps;
- char *argv[] = { "app" };
-
- parg_init(&ps);
-
- ASSERT_EQ(-1, parg_getopt(&ps, ARRAY_SIZE(argv), argv, os_def));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(1, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_dash(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "-" };
-
- parg_init(&ps);
-
- ASSERT_EQ(1, parg_getopt(&ps, ARRAY_SIZE(argv), argv, os_def));
- ASSERT_EQ(argv[1], ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt(&ps, ARRAY_SIZE(argv), argv, os_def));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_double_dash(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "--" };
-
- parg_init(&ps);
-
- ASSERT_EQ(-1, parg_getopt(&ps, ARRAY_SIZE(argv), argv, os_def));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_nonopt(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "foo" };
-
- parg_init(&ps);
-
- ASSERT_EQ(1, parg_getopt(&ps, ARRAY_SIZE(argv), argv, os_def));
- ASSERT_EQ(argv[1], ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt(&ps, ARRAY_SIZE(argv), argv, os_def));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_no_match(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "-u" };
-
- parg_init(&ps);
-
- ASSERT_EQ('?', parg_getopt(&ps, ARRAY_SIZE(argv), argv, os_def));
- ASSERT_EQ('u', ps.optopt);
- ASSERT_EQ(2, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt(&ps, ARRAY_SIZE(argv), argv, os_def));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_noarg(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "-n" };
-
- parg_init(&ps);
-
- ASSERT_EQ('n', parg_getopt(&ps, ARRAY_SIZE(argv), argv, os_def));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt(&ps, ARRAY_SIZE(argv), argv, os_def));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_optarg_missing(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "-o" };
-
- parg_init(&ps);
-
- ASSERT_EQ('o', parg_getopt(&ps, ARRAY_SIZE(argv), argv, os_def));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt(&ps, ARRAY_SIZE(argv), argv, os_def));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_optarg_inline(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "-oarg" };
-
- parg_init(&ps);
-
- ASSERT_EQ('o', parg_getopt(&ps, ARRAY_SIZE(argv), argv, os_def));
- ASSERT_EQ(&argv[1][2], ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt(&ps, ARRAY_SIZE(argv), argv, os_def));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_reqarg_missing(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "-r" };
-
- parg_init(&ps);
-
- ASSERT_EQ(':', parg_getopt(&ps, ARRAY_SIZE(argv), argv, os_def));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ('r', ps.optopt);
- ASSERT_EQ(2, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt(&ps, ARRAY_SIZE(argv), argv, os_def));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_reqarg_inline(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "-rarg" };
-
- parg_init(&ps);
-
- ASSERT_EQ('r', parg_getopt(&ps, ARRAY_SIZE(argv), argv, os_def));
- ASSERT_EQ(&argv[1][2], ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt(&ps, ARRAY_SIZE(argv), argv, os_def));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_reqarg_nextarg(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "-r", "arg" };
-
- parg_init(&ps);
-
- ASSERT_EQ('r', parg_getopt(&ps, ARRAY_SIZE(argv), argv, os_def));
- ASSERT_EQ(argv[2], ps.optarg);
- ASSERT_EQ(3, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt(&ps, ARRAY_SIZE(argv), argv, os_def));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(3, ps.optind);
-
- PASS();
-}
-
-/*
- * parg_getopt_long tests
- */
-
-TEST parg_getopt_long_app_only(void)
-{
- struct parg_state ps;
- char *argv[] = { "app" };
- int li = -1;
-
- parg_init(&ps);
-
- ASSERT_EQ(-1, parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(1, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_long_dash(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "-" };
- int li = -1;
-
- parg_init(&ps);
-
- ASSERT_EQ(1, parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(argv[1], ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_long_double_dash(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "--" };
- int li = -1;
-
- parg_init(&ps);
-
- ASSERT_EQ(-1, parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_long_nonopt(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "foo" };
- int li = -1;
-
- parg_init(&ps);
-
- ASSERT_EQ(1, parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(argv[1], ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_long_no_match(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "--unknown" };
- int li = -1;
-
- parg_init(&ps);
-
- ASSERT_EQ('?', parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(2, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_long_flag(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "--flag" };
- int li = -1;
- int flag = 0;
- struct parg_option po_flag[] = {
- { "flag", PARG_NOARG, 0, 1 },
- { 0, 0, 0, 0 }
- };
-
- po_flag[0].flag = &flag;
-
- parg_init(&ps);
-
- ASSERT_EQ(0, parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_flag, &li));
- ASSERT_EQ(0, li);
- ASSERT_EQ(1, flag);
- ASSERT_EQ(2, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_flag, &li));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_long_noarg(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "--noarg" };
- int li = -1;
-
- parg_init(&ps);
-
- ASSERT_EQ('n', parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(0, li);
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_long_noarg_inline(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "--noarg=arg" };
- int li = -1;
-
- parg_init(&ps);
-
- ASSERT_EQ(':', parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(0, li);
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ('n', ps.optopt);
- ASSERT_EQ(2, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_long_optarg_missing(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "--optarg" };
- int li = -1;
-
- parg_init(&ps);
-
- ASSERT_EQ('o', parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(1, li);
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_long_optarg_inline(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "--optarg=arg" };
- int li = -1;
-
- parg_init(&ps);
-
- ASSERT_EQ('o', parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(1, li);
- ASSERT_EQ(&argv[1][9], ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_long_reqarg_missing(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "--reqarg" };
- int li = -1;
-
- parg_init(&ps);
-
- ASSERT_EQ(':', parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(2, li);
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ('r', ps.optopt);
- ASSERT_EQ(2, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_long_reqarg_inline(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "--reqarg=arg" };
- int li = -1;
-
- parg_init(&ps);
-
- ASSERT_EQ('r', parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(2, li);
- ASSERT_EQ(&argv[1][9], ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_long_reqarg_nextarg(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "--reqarg", "arg" };
- int li = -1;
-
- parg_init(&ps);
-
- ASSERT_EQ('r', parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(2, li);
- ASSERT_EQ(argv[2], ps.optarg);
- ASSERT_EQ(3, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(3, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_long_prefix_ambiguous(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "--fo" };
- int li = -1;
-
- parg_init(&ps);
-
- ASSERT_EQ('?', parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_long_prefix_exact(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "--foo" };
- int li = -1;
-
- parg_init(&ps);
-
- ASSERT_EQ('f', parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(3, li);
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-TEST parg_getopt_long_prefix_unambiguous(void)
-{
- struct parg_state ps;
- char *argv[] = { "app", "--foob" };
- int li = -1;
-
- parg_init(&ps);
-
- ASSERT_EQ('b', parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(4, li);
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- ASSERT_EQ(-1, parg_getopt_long(&ps, ARRAY_SIZE(argv), argv, ":", po_def, &li));
- ASSERT_EQ(NULL, ps.optarg);
- ASSERT_EQ(2, ps.optind);
-
- PASS();
-}
-
-/*
- * parg_reorder tests
- */
-
-TEST parg_reorder_short(void)
-{
- char *argv[] = {
- "app", "first", "-r", "--", "foo", "-u", "bar", "-o", "--",
- "-n", "last"
- };
- char *argv_expected[] = {
- "app", "-r", "--", "-u", "-o", "--", "first", "foo", "bar",
- "-n", "last"
- };
- size_t i;
-
- ASSERT_EQ(6, parg_reorder(ARRAY_SIZE(argv), argv, os_def, NULL));
-
- for (i = 0; i < ARRAY_SIZE(argv); ++i) {
- ASSERT_STR_EQ(argv_expected[i], argv[i]);
- }
-
- PASS();
-}
-
-TEST parg_reorder_long(void)
-{
- char *argv[] = {
- "app", "first", "--reqarg", "--", "foo", "--unknown", "bar",
- "--optarg", "--", "--noarg", "last"
- };
- char *argv_expected[] = {
- "app", "--reqarg", "--", "--unknown", "--optarg", "--",
- "first", "foo", "bar", "--noarg", "last"
- };
- size_t i;
-
- ASSERT_EQ(6, parg_reorder(ARRAY_SIZE(argv), argv, "", po_def));
-
- for (i = 0; i < ARRAY_SIZE(argv); ++i) {
- ASSERT_STR_EQ(argv_expected[i], argv[i]);
- }
-
- PASS();
-}
-
-TEST parg_reorder_app_only(void)
-{
- char *argv[] = { "app" };
-
- ASSERT_EQ(1, parg_reorder(ARRAY_SIZE(argv), argv, os_def, NULL));
-
- PASS();
-}
-
-TEST parg_reorder_double_dash_last(void)
-{
- char *argv[] = { "app", "foo", "-n", "--" };
- char *argv_expected[] = { "app", "-n", "--", "foo" };
- size_t i;
-
- ASSERT_EQ(3, parg_reorder(ARRAY_SIZE(argv), argv, os_def, NULL));
-
- for (i = 0; i < ARRAY_SIZE(argv); ++i) {
- ASSERT_STR_EQ(argv_expected[i], argv[i]);
- }
-
- PASS();
-}
-
-TEST parg_reorder_missing_arg_last(void)
-{
- char *argv[] = { "app", "foo", "-r" };
- char *argv_expected[] = { "app", "-r", "foo" };
- size_t i;
-
- ASSERT_EQ(2, parg_reorder(ARRAY_SIZE(argv), argv, os_def, NULL));
-
- for (i = 0; i < ARRAY_SIZE(argv); ++i) {
- ASSERT_STR_EQ(argv_expected[i], argv[i]);
- }
-
- PASS();
-}
-
-SUITE(parg_getopt_tests)
-{
- RUN_TEST(parg_getopt_app_only);
-
- RUN_TEST(parg_getopt_dash);
-
- RUN_TEST(parg_getopt_double_dash);
-
- RUN_TEST(parg_getopt_nonopt);
-
- RUN_TEST(parg_getopt_no_match);
-
- RUN_TEST(parg_getopt_noarg);
-
- RUN_TEST(parg_getopt_optarg_missing);
- RUN_TEST(parg_getopt_optarg_inline);
-
- RUN_TEST(parg_getopt_reqarg_missing);
- RUN_TEST(parg_getopt_reqarg_inline);
- RUN_TEST(parg_getopt_reqarg_nextarg);
-}
-
-SUITE(parg_getopt_long_tests)
-{
- RUN_TEST(parg_getopt_long_app_only);
-
- RUN_TEST(parg_getopt_long_dash);
-
- RUN_TEST(parg_getopt_long_double_dash);
-
- RUN_TEST(parg_getopt_long_nonopt);
-
- RUN_TEST(parg_getopt_long_no_match);
-
- RUN_TEST(parg_getopt_long_flag);
-
- RUN_TEST(parg_getopt_long_noarg);
- RUN_TEST(parg_getopt_long_noarg_inline);
-
- RUN_TEST(parg_getopt_long_optarg_missing);
- RUN_TEST(parg_getopt_long_optarg_inline);
-
- RUN_TEST(parg_getopt_long_reqarg_missing);
- RUN_TEST(parg_getopt_long_reqarg_inline);
- RUN_TEST(parg_getopt_long_reqarg_nextarg);
-
- RUN_TEST(parg_getopt_long_prefix_ambiguous);
- RUN_TEST(parg_getopt_long_prefix_exact);
- RUN_TEST(parg_getopt_long_prefix_unambiguous);
-}
-
-SUITE(parg_reorder_tests)
-{
- RUN_TEST(parg_reorder_short);
- RUN_TEST(parg_reorder_long);
-
- RUN_TEST(parg_reorder_app_only);
-
- RUN_TEST(parg_reorder_double_dash_last);
-
- RUN_TEST(parg_reorder_missing_arg_last);
-}
-
-GREATEST_MAIN_DEFS();
-
-int main(int argc, char *argv[])
-{
- GREATEST_MAIN_BEGIN();
- RUN_SUITE(parg_getopt_tests);
- RUN_SUITE(parg_getopt_long_tests);
- RUN_SUITE(parg_reorder_tests);
- GREATEST_MAIN_END();
-}
--- /dev/null
+++ b/proto.c
@@ -1,0 +1,1164 @@
+#include <stdint.h>
+#include <string.h>
+#include "c9/proto.h"
+
+enum
+{
+ Svver = 1<<0,
+};
+
+#define safestrlen(s) (s == NULL ? 0 : (uint32_t)strlen(s))
+#define maxread(c) (c->msize-4-4-1-2)
+#define maxwrite(c) maxread(c)
+
+static void
+w08(uint8_t **p, uint8_t x)
+{
+ (*p)[0] = x;
+ *p += 1;
+}
+
+static void
+w16(uint8_t **p, uint16_t x)
+{
+ (*p)[0] = x;
+ (*p)[1] = x>>8;
+ *p += 2;
+}
+
+static void
+w32(uint8_t **p, uint32_t x)
+{
+ (*p)[0] = x;
+ (*p)[1] = x>>8;
+ (*p)[2] = x>>16;
+ (*p)[3] = x>>24;
+ *p += 4;
+}
+
+static void
+w64(uint8_t **p, uint64_t x)
+{
+ (*p)[0] = x;
+ (*p)[1] = x>>8;
+ (*p)[2] = x>>16;
+ (*p)[3] = x>>24;
+ (*p)[4] = x>>32;
+ (*p)[5] = x>>40;
+ (*p)[6] = x>>48;
+ (*p)[7] = x>>56;
+ *p += 8;
+}
+
+static void
+wcs(uint8_t **p, const char *s, int len)
+{
+ w16(p, len);
+ if(s != NULL){
+ memmove(*p, s, len);
+ *p += len;
+ }
+}
+
+static uint8_t
+r08(uint8_t **p)
+{
+ *p += 1;
+ return (*p)[-1];
+}
+
+static uint16_t
+r16(uint8_t **p)
+{
+ *p += 2;
+ return (*p)[-2]<<0 | (*p)[-1]<<8;
+}
+
+static uint32_t
+r32(uint8_t **p)
+{
+ *p += 4;
+ return (*p)[-4]<<0 | (*p)[-3]<<8 | (*p)[-2]<<16 | (*p)[-1]<<24;
+}
+
+static uint64_t
+r64(uint8_t **p)
+{
+ uint64_t v;
+
+ v = r32(p);
+ v |= (uint64_t)r32(p)<<32;
+ return v;
+}
+
+#ifndef C9_NO_CLIENT
+
+static C9error
+newtag(C9ctx *c, C9ttype type, C9tag *tag)
+{
+ uint32_t i;
+
+ if(type == Tversion){
+ *tag = 0xffff;
+ return 0;
+ }
+
+ if(c->lowfreetag < C9maxtags){
+ uint32_t d = c->lowfreetag / C9tagsbits, m = c->lowfreetag % C9tagsbits;
+ if((c->tags[d] & 1<<m) != 0){
+ c->tags[d] &= ~(1<<m);
+ *tag = c->lowfreetag++;
+ return 0;
+ }
+ }
+
+ for(i = 0; i < (int)sizeof(c->tags)/sizeof(c->tags[0]); i++){
+ uint32_t x, j;
+ if((x = c->tags[i]) == 0)
+ continue;
+ for(j = 0; j < C9tagsbits; j++){
+ if((x & (1<<j)) != 0){
+ c->tags[i] &= ~(1<<j);
+ *tag = i*C9tagsbits + j;
+ c->lowfreetag = *tag + 1;
+ return 0;
+ }
+ }
+ }
+
+ c->error(c, "newtag: no free tags");
+ return C9Etag;
+}
+
+static int
+freetag(C9ctx *c, C9tag tag)
+{
+ if(tag != 0xffff){
+ uint32_t d = tag / C9tagsbits, m = tag % C9tagsbits;
+ if(tag >= C9maxtags){
+ c->error(c, "freetag: invalid tag %ud", (uint32_t)tag);
+ return -1;
+ }
+ if((c->tags[d] & 1<<m) != 0){
+ c->error(c, "freetag: double free for tag %ud", (uint32_t)tag);
+ return -1;
+ }
+ if(c->lowfreetag > tag)
+ c->lowfreetag = tag;
+ c->tags[d] |= 1<<m;
+ }
+ return 0;
+}
+
+static uint8_t *
+T(C9ctx *c, uint32_t size, C9ttype type, C9tag *tag, C9error *err)
+{
+ uint8_t *p = NULL;
+
+ if(size > c->msize-4-1-2){
+ c->error(c, "T: invalid size %ud", size);
+ *err = C9Esize;
+ }else if((*err = newtag(c, type, tag)) == 0){
+ size += 4+1+2;
+ if((p = c->begin(c, size)) == NULL){
+ c->error(c, "T: no buffer for %ud bytes", size);
+ freetag(c, *tag);
+ *err = C9Ebuf;
+ }else{
+ *err = 0;
+ w32(&p, size);
+ w08(&p, type);
+ w16(&p, *tag);
+ }
+ }
+ return p;
+}
+
+C9error
+c9version(C9ctx *c, C9tag *tag, uint32_t msize)
+{
+ uint8_t *b;
+ C9error err;
+
+ if(msize < C9minmsize){
+ c->error(c, "c9version: msize too small: %ud", msize);
+ return C9Einit;
+ }
+ memset(c->tags, 0xff, sizeof(c->tags));
+ memset(c->flush, 0xff, sizeof(c->flush));
+ c->lowfreetag = 0;
+ c->msize = msize;
+
+ if((b = T(c, 4+2+6, Tversion, tag, &err)) != NULL){
+ w32(&b, msize);
+ wcs(&b, "9P2000", 6);
+ err = c->end(c);
+ }
+ return err;
+}
+
+C9error
+c9auth(C9ctx *c, C9tag *tag, C9fid afid, const char *uname, const char *aname)
+{
+ uint8_t *b;
+ uint32_t ulen = safestrlen(uname), alen = safestrlen(aname);
+ C9error err;
+
+ if(ulen > C9maxstr || alen > C9maxstr){
+ c->error(c, "c9auth: string too long: %ud chars", ulen > alen ? ulen : alen);
+ return C9Estr;
+ }
+ if((b = T(c, 4+2+ulen+2+alen, Tauth, tag, &err)) != NULL){
+ w32(&b, afid);
+ wcs(&b, uname, ulen);
+ wcs(&b, aname, alen);
+ err = c->end(c);
+ }
+ return err;
+}
+
+C9error
+c9flush(C9ctx *c, C9tag *tag, C9tag oldtag)
+{
+ uint8_t *b;
+ C9error err;
+ int i;
+
+ for(i = 0; i < C9maxflush && c->flush[i] != (uint32_t)~0; i++);
+ if(i == C9maxflush){
+ c->error(c, "c9flush: no free flush slots");
+ return C9Eflush;
+ }
+ if((b = T(c, 2, Tflush, tag, &err)) != NULL){
+ w16(&b, oldtag);
+ err = c->end(c);
+ if(err == 0)
+ c->flush[i] = (uint32_t)oldtag<<16 | *tag;
+ }
+ return err;
+}
+
+C9error
+c9attach(C9ctx *c, C9tag *tag, C9fid fid, C9fid afid, const char *uname, const char *aname)
+{
+ uint32_t ulen = safestrlen(uname), alen = safestrlen(aname);
+ uint8_t *b;
+ C9error err;
+
+ if(ulen > C9maxstr || alen > C9maxstr){
+ c->error(c, "c9attach: string too long: %ud chars", ulen > alen ? ulen : alen);
+ return C9Estr;
+ }
+ if((b = T(c, 4+4+2+ulen+2+alen, Tattach, tag, &err)) != NULL){
+ w32(&b, fid);
+ w32(&b, afid);
+ wcs(&b, uname, ulen);
+ wcs(&b, aname, alen);
+ err = c->end(c);
+ }
+ return err;
+}
+
+C9error
+c9walk(C9ctx *c, C9tag *tag, C9fid fid, C9fid newfid, const char *path[])
+{
+ uint32_t i, j, sz;
+ uint32_t len[C9maxpathel];
+ uint8_t *b;
+ C9error err;
+
+ for(sz = i = 0; i < (int)sizeof(len)/sizeof(len[0]) && path[i] != NULL; i++){
+ len[i] = safestrlen(path[i]);
+ if(len[i] == 0 || len[i] > C9maxstr){
+ c->error(c, "c9walk: invalid path element: %ud chars", len[i]);
+ return C9Epath;
+ }
+ sz += 2 + len[i];
+ }
+ if(path[i] != NULL){
+ c->error(c, "c9walk: invalid elements !(0 <= %ud <= %ud)", i, C9maxpathel);
+ return C9Epath;
+ }
+
+ if((b = T(c, 4+4+2+sz, Twalk, tag, &err)) != NULL){
+ w32(&b, fid);
+ w32(&b, newfid);
+ w16(&b, i);
+ for(j = 0; j < i; j++)
+ wcs(&b, path[j], len[j]);
+ err = c->end(c);
+ }
+ return err;
+}
+
+C9error
+c9open(C9ctx *c, C9tag *tag, C9fid fid, C9mode mode)
+{
+ uint8_t *b;
+ C9error err;
+
+ if((b = T(c, 4+1, Topen, tag, &err)) != NULL){
+ w32(&b, fid);
+ w08(&b, mode);
+ err = c->end(c);
+ }
+ return err;
+}
+
+C9error
+c9create(C9ctx *c, C9tag *tag, C9fid fid, const char *name, uint32_t perm, C9mode mode)
+{
+ uint32_t nlen = strlen(name);
+ uint8_t *b;
+ C9error err;
+
+ if(nlen == 0 || nlen > C9maxstr){
+ c->error(c, "c9create: invalid name: %ud chars", nlen);
+ return C9Epath;
+ }
+ if((b = T(c, 4+2+nlen+4+1, Tcreate, tag, &err)) != NULL){
+ w32(&b, fid);
+ wcs(&b, name, nlen);
+ w32(&b, perm);
+ w08(&b, mode);
+ err = c->end(c);
+ }
+ return err;
+}
+
+C9error
+c9read(C9ctx *c, C9tag *tag, C9fid fid, uint64_t offset, uint32_t count)
+{
+ uint8_t *b;
+ C9error err;
+
+ if((b = T(c, 4+8+4, Tread, tag, &err)) != NULL){
+ w32(&b, fid);
+ w64(&b, offset);
+ w32(&b, count);
+ err = c->end(c);
+ }
+ return err;
+}
+
+C9error
+c9write(C9ctx *c, C9tag *tag, C9fid fid, uint64_t offset, const void *in, uint32_t count)
+{
+ uint8_t *b;
+ C9error err;
+
+ if((b = T(c, 4+8+4+count, Twrite, tag, &err)) != NULL){
+ w32(&b, fid);
+ w64(&b, offset);
+ w32(&b, count);
+ memmove(b, in, count);
+ err = c->end(c);
+ }
+ return err;
+}
+
+C9error
+c9wrstr(C9ctx *c, C9tag *tag, C9fid fid, const char *s)
+{
+ return c9write(c, tag, fid, 0, s, strlen(s));
+}
+
+C9error
+c9clunk(C9ctx *c, C9tag *tag, C9fid fid)
+{
+ uint8_t *b;
+ C9error err;
+
+ if((b = T(c, 4, Tclunk, tag, &err)) != NULL){
+ w32(&b, fid);
+ err = c->end(c);
+ }
+ return err;
+}
+
+C9error
+c9remove(C9ctx *c, C9tag *tag, C9fid fid)
+{
+ uint8_t *b;
+ C9error err;
+
+ if((b = T(c, 4, Tremove, tag, &err)) != NULL){
+ w32(&b, fid);
+ err = c->end(c);
+ }
+ return err;
+}
+
+C9error
+c9stat(C9ctx *c, C9tag *tag, C9fid fid)
+{
+ uint8_t *b;
+ C9error err;
+
+ if((b = T(c, 4, Tstat, tag, &err)) != NULL){
+ w32(&b, fid);
+ err = c->end(c);
+ }
+ return err;
+}
+
+C9error
+c9wstat(C9ctx *c, C9tag *tag, C9fid fid, const C9stat *s)
+{
+ uint32_t nlen = safestrlen(s->name), ulen = safestrlen(s->uid), glen = safestrlen(s->gid);
+ uint32_t unusedsz = 2+4+13, statsz = unusedsz+4+4+4+8+2+nlen+2+ulen+2+glen+2;
+ uint8_t *b;
+ C9error err;
+
+ if(nlen == 0 || nlen > C9maxstr){
+ c->error(c, "c9wstat: invalid name: %ud chars", nlen);
+ return C9Epath;
+ }
+ if(ulen > C9maxstr || glen > C9maxstr){
+ c->error(c, "c9wstat: string too long: %ud chars", ulen > glen ? ulen : glen);
+ return C9Estr;
+ }
+ if((b = T(c, 4+2+2+statsz, Twstat, tag, &err)) != NULL){
+ w32(&b, fid);
+ w16(&b, statsz+2);
+ w16(&b, statsz);
+ memset(b, 0xff, unusedsz); /* leave type(2), dev(4) and qid(13) unchanged */
+ b += unusedsz;
+ w32(&b, s->mode);
+ w32(&b, s->atime);
+ w32(&b, s->mtime);
+ w64(&b, s->size);
+ wcs(&b, s->name, nlen);
+ wcs(&b, s->uid, ulen);
+ wcs(&b, s->gid, glen);
+ wcs(&b, NULL, 0); /* muid unchanged */
+ err = c->end(c);
+ }
+ return err;
+}
+
+C9error
+c9proc(C9ctx *c)
+{
+ uint32_t i, sz, cnt, msize;
+ uint8_t *b;
+ int err;
+ C9r r;
+
+ err = -1;
+ if((b = c->read(c, 4, &err)) == NULL){
+ if(err != 0)
+ c->error(c, "c9proc: short read");
+ return err == 0 ? 0 : C9Epkt;
+ }
+
+ sz = r32(&b);
+ if(sz < 7 || sz > c->msize){
+ c->error(c, "c9proc: invalid packet size !(7 <= %ud <= %ud)", sz, c->msize);
+ return C9Epkt;
+ }
+ sz -= 4;
+ err = -1;
+ if((b = c->read(c, sz, &err)) == NULL){
+ if(err != 0)
+ c->error(c, "c9proc: short read");
+ return err == 0 ? 0 : C9Epkt;
+ }
+
+ r.type = r08(&b);
+ r.tag = r16(&b);
+ if(r.type != Rversion){
+ if(r.tag >= C9maxtags){
+ c->error(c, "c9proc: invalid tag %ud", (uint32_t)r.tag);
+ return C9Epkt;
+ }
+ if(freetag(c, r.tag) != 0)
+ return C9Etag;
+ }
+ sz -= 3;
+ r.numqid = 0;
+
+ switch(r.type){
+ case Rread:
+ if(sz < 4 || (cnt = r32(&b)) > sz-4)
+ goto error;
+ r.read.data = b;
+ r.read.size = cnt;
+ c->r(c, &r);
+ break;
+
+ case Rwrite:
+ if(sz < 4 || (cnt = r32(&b)) > c->msize)
+ goto error;
+ r.write.size = cnt;
+ c->r(c, &r);
+ break;
+
+ case Rwalk:
+ if(sz < 2 || (cnt = r16(&b))*13 > sz-2)
+ goto error;
+ if(cnt > C9maxpathel){
+ c->error(c, "c9proc: Rwalk !(%ud <= %ud)", cnt, C9maxpathel);
+ return C9Epath;
+ }
+ for(i = 0; i < cnt; i++){
+ r.qid[i].type = r08(&b);
+ r.qid[i].version = r32(&b);
+ r.qid[i].path = r64(&b);
+ }
+ r.numqid = cnt;
+ c->r(c, &r);
+ break;
+
+ case Rstat:
+ b += 2; sz -= 2;
+ if((err = c9parsedir(c, &r.stat, &b, &sz)) != 0)
+ return err;
+ r.numqid = 1;
+ c->r(c, &r);
+ break;
+
+ case Rflush:
+ for(i = 0; i < C9maxflush; i++){
+ if((c->flush[i] & 0xffff) == r.tag){
+ freetag(c, c->flush[i]>>16);
+ c->flush[i] = 0xffffffff;
+ break;
+ }
+ }
+ /* fallthrough */
+ case Rclunk:
+ case Rremove:
+ case Rwstat:
+ c->r(c, &r);
+ break;
+
+ case Ropen:
+ case Rcreate:
+ if(sz < 17)
+ goto error;
+ r.qid[0].type = r08(&b);
+ r.qid[0].version = r32(&b);
+ r.qid[0].path = r64(&b);
+ r.iounit = r32(&b);
+ r.numqid = 1;
+ c->r(c, &r);
+ break;
+
+ case Rerror:
+ if(sz < 2 || (cnt = r16(&b)) > sz-2)
+ goto error;
+ r.error = memmove(b-1, b, cnt);
+ r.error[cnt] = 0;
+ c->r(c, &r);
+ break;
+
+ case Rauth:
+ case Rattach:
+ if(sz < 13)
+ goto error;
+ r.qid[0].type = r08(&b);
+ r.qid[0].version = r32(&b);
+ r.qid[0].path = r64(&b);
+ r.numqid = 1;
+ c->r(c, &r);
+ break;
+
+ case Rversion:
+ if(sz < 4+2 || (msize = r32(&b)) < C9minmsize || (cnt = r16(&b)) > sz-4-2)
+ goto error;
+ if(cnt < 6 || memcmp(b, "9P2000", 6) != 0){
+ c->error(c, "invalid version");
+ return C9Ever;
+ }
+ if(msize < c->msize)
+ c->msize = msize;
+ c->r(c, &r);
+ break;
+
+ default:
+ goto error;
+ }
+ return 0;
+error:
+ c->error(c, "c9proc: invalid packet type %ud", r.type);
+ return C9Epkt;
+}
+
+#endif /* C9_NO_CLIENT */
+
+C9error
+c9parsedir(C9ctx *c, C9stat *stat, uint8_t **t, uint32_t *size)
+{
+ uint8_t *b;
+ uint32_t cnt, sz;
+
+ sz = 0;
+ if(*size < 49 || (sz = r16(t)) < 47 || *size < 2+sz)
+ goto error;
+ *size -= 2+sz;
+ *t += 6; /* skip type(2) and dev(4) */
+ stat->qid.type = r08(t);
+ stat->qid.version = r32(t);
+ stat->qid.path = r64(t);
+ stat->mode = r32(t);
+ stat->atime = r32(t);
+ stat->mtime = r32(t);
+ stat->size = r64(t);
+ sz -= 39;
+ if((cnt = r16(t)) > sz-2)
+ goto error;
+ stat->name = (char*)*t; b = *t = *t+cnt; sz -= 2+cnt;
+ if(sz < 2 || (cnt = r16(t)) > sz-2)
+ goto error;
+ stat->uid = (char*)*t; *b = 0; b = *t = *t+cnt; sz -= 2+cnt;
+ if(sz < 2 || (cnt = r16(t)) > sz-2)
+ goto error;
+ stat->gid = (char*)*t; *b = 0; b = *t = *t+cnt; sz -= 2+cnt;
+ if(sz < 2 || (cnt = r16(t)) > sz-2)
+ goto error;
+ stat->muid = memmove(*t-1, *t, cnt); *b = stat->muid[cnt] = 0; *t = *t+cnt; sz -= 2+cnt;
+ *t += sz;
+ return 0;
+error:
+ c->error(c, "c9parsedir: invalid size: size=%ud sz=%ud", *size, sz);
+ return C9Epkt;
+}
+
+#ifndef C9_NO_SERVER
+
+static uint8_t *
+R(C9ctx *c, uint32_t size, C9rtype type, C9tag tag, C9error *err)
+{
+ uint8_t *p = NULL;
+
+ if(size > c->msize-4-1-2){
+ c->error(c, "R: invalid size %ud", size);
+ *err = C9Esize;
+ }else{
+ size += 4+1+2;
+ if((p = c->begin(c, size)) == NULL){
+ c->error(c, "R: no buffer for %ud bytes", size);
+ *err = C9Ebuf;
+ }else{
+ *err = 0;
+ w32(&p, size);
+ w08(&p, type);
+ w16(&p, tag);
+ }
+ }
+ return p;
+}
+
+C9error
+s9version(C9ctx *c)
+{
+ uint8_t *b;
+ C9error err;
+
+ if((b = R(c, 4+2+6, Rversion, 0xffff, &err)) != NULL){
+ w32(&b, c->msize);
+ wcs(&b, "9P2000", 6);
+ err = c->end(c);
+ };
+ return err;
+}
+
+C9error
+s9auth(C9ctx *c, C9tag tag, const C9qid *aqid)
+{
+ uint8_t *b;
+ C9error err;
+
+ if((b = R(c, 13, Rauth, tag, &err)) != NULL){
+ w08(&b, aqid->type);
+ w32(&b, aqid->version);
+ w64(&b, aqid->path);
+ err = c->end(c);
+ }
+ return err;
+}
+
+C9error
+s9error(C9ctx *c, C9tag tag, const char *ename)
+{
+ uint32_t len = safestrlen(ename);
+ uint8_t *b;
+ C9error err;
+
+ if(len > C9maxstr){
+ c->error(c, "s9error: invalid ename: %ud chars", len);
+ return C9Estr;
+ }
+ if((b = R(c, 2+len, Rerror, tag, &err)) != NULL){
+ wcs(&b, ename, len);
+ err = c->end(c);
+ }
+ return err;
+}
+
+C9error
+s9attach(C9ctx *c, C9tag tag, const C9qid *qid)
+{
+ uint8_t *b;
+ C9error err;
+
+ if((b = R(c, 13, Rattach, tag, &err)) != NULL){
+ w08(&b, qid->type);
+ w32(&b, qid->version);
+ w64(&b, qid->path);
+ err = c->end(c);
+ }
+ return err;
+}
+
+C9error
+s9flush(C9ctx *c, C9tag tag)
+{
+ C9error err;
+
+ if(R(c, 0, Rflush, tag, &err) != NULL)
+ err = c->end(c);
+ return err;
+}
+
+C9error
+s9walk(C9ctx *c, C9tag tag, C9qid *qids[])
+{
+ uint32_t i, n;
+ uint8_t *b;
+ C9error err;
+
+ for(n = 0; n < C9maxpathel && qids[n] != NULL; n++);
+ if(n > C9maxpathel){
+ c->error(c, "s9walk: invalid elements !(0 <= %ud <= %ud)", n, C9maxpathel);
+ return C9Epath;
+ }
+
+ if((b = R(c, 2+n*13, Rwalk, tag, &err)) != NULL){
+ w16(&b, n);
+ for(i = 0; i < n; i++){
+ w08(&b, qids[i]->type);
+ w32(&b, qids[i]->version);
+ w64(&b, qids[i]->path);
+ }
+ err = c->end(c);
+ }
+ return err;
+}
+
+C9error
+s9open(C9ctx *c, C9tag tag, const C9qid *qid, uint32_t iounit)
+{
+ uint8_t *b;
+ C9error err;
+
+ if((b = R(c, 13+4, Ropen, tag, &err)) != NULL){
+ w08(&b, qid->type);
+ w32(&b, qid->version);
+ w64(&b, qid->path);
+ w32(&b, iounit);
+ err = c->end(c);
+ }
+ return err;
+}
+
+C9error
+s9create(C9ctx *c, C9tag tag, const C9qid *qid, uint32_t iounit)
+{
+ uint8_t *b;
+ C9error err;
+
+ if((b = R(c, 13+4, Rcreate, tag, &err)) != NULL){
+ w08(&b, qid->type);
+ w32(&b, qid->version);
+ w64(&b, qid->path);
+ w32(&b, iounit);
+ err = c->end(c);
+ }
+ return err;
+}
+
+C9error
+s9read(C9ctx *c, C9tag tag, const void *data, uint32_t size)
+{
+ uint8_t *b;
+ C9error err;
+
+ if((b = R(c, 4+size, Rread, tag, &err)) != NULL){
+ w32(&b, size);
+ memmove(b, data, size);
+ err = c->end(c);
+ }
+ return err;
+}
+
+C9error
+s9write(C9ctx *c, C9tag tag, uint32_t size)
+{
+ uint8_t *b;
+ C9error err;
+
+ if((b = R(c, 4, Rwrite, tag, &err)) != NULL){
+ w32(&b, size);
+ err = c->end(c);
+ }
+ return err;
+}
+
+C9error
+s9readdir(C9ctx *c, C9tag tag, C9stat *st[], int *num, uint64_t *offset, uint32_t size)
+{
+ uint8_t *b;
+ const C9stat *s;
+ uint32_t nlen, ulen, glen, mulen, m, n;
+ C9error err;
+ int i;
+
+ if(size > c->msize-4-1-2)
+ size = c->msize-4-1-2;
+
+ m = 0;
+ for(i = 0; i < *num; i++){
+ s = st[i];
+ nlen = safestrlen(s->name);
+ ulen = safestrlen(s->uid);
+ glen = safestrlen(s->gid);
+ mulen = safestrlen(s->muid);
+
+ if(nlen == 0 || nlen > C9maxstr){
+ c->error(c, "s9readdir: invalid name: %ud chars", nlen);
+ return C9Epath;
+ }
+ if(ulen > C9maxstr || glen > C9maxstr || mulen > C9maxstr){
+ ulen = ulen > glen ? ulen : glen;
+ ulen = ulen > mulen ? ulen : mulen;
+ c->error(c, "s9readdir: string too long: %ud chars", ulen);
+ return C9Estr;
+ }
+
+ n = 2 + 2+4+13+4+4+4+8+2+nlen+2+ulen+2+glen+2+mulen;
+ if(4+m+n > size)
+ break;
+ m += n;
+ }
+
+ if((b = R(c, 4+m, Rread, tag, &err)) != NULL){
+ *num = i;
+ w32(&b, m);
+ for(i = 0; i < *num; i++){
+ s = st[i];
+ nlen = safestrlen(s->name);
+ ulen = safestrlen(s->uid);
+ glen = safestrlen(s->gid);
+ mulen = safestrlen(s->muid);
+ w16(&b, 2+4+13+4+4+4+8+2+nlen+2+ulen+2+glen+2+mulen);
+ w16(&b, 0xffff); /* type */
+ w32(&b, 0xffffffff); /* dev */
+ w08(&b, s->qid.type);
+ w32(&b, s->qid.version);
+ w64(&b, s->qid.path);
+ w32(&b, s->mode);
+ w32(&b, s->atime);
+ w32(&b, s->mtime);
+ w64(&b, s->size);
+ wcs(&b, s->name, nlen);
+ wcs(&b, s->uid, ulen);
+ wcs(&b, s->gid, glen);
+ wcs(&b, s->muid, mulen);
+ }
+ err = c->end(c);
+ if(err == 0)
+ *offset += m;
+ }
+ return err;
+}
+
+C9error
+s9clunk(C9ctx *c, C9tag tag)
+{
+ C9error err;
+
+ if(R(c, 0, Rclunk, tag, &err) != NULL)
+ err = c->end(c);
+ return err;
+}
+
+C9error
+s9remove(C9ctx *c, C9tag tag)
+{
+ C9error err;
+
+ if(R(c, 0, Rremove, tag, &err) != NULL)
+ err = c->end(c);
+ return err;
+}
+
+C9error
+s9stat(C9ctx *c, C9tag tag, const C9stat *s)
+{
+ uint32_t nlen = safestrlen(s->name), ulen = safestrlen(s->uid);
+ uint32_t glen = safestrlen(s->gid), mulen = safestrlen(s->name);
+ uint32_t statsz = 2+4+13+4+4+4+8+2+nlen+2+ulen+2+glen+2+mulen;
+ uint8_t *b;
+ C9error err;
+
+ if(nlen == 0 || nlen > C9maxstr){
+ c->error(c, "s9stat: invalid name: %ud chars", nlen);
+ return C9Epath;
+ }
+ if(ulen > C9maxstr || glen > C9maxstr || mulen > C9maxstr){
+ ulen = ulen > glen ? ulen : glen;
+ ulen = ulen > mulen ? ulen : mulen;
+ c->error(c, "s9stat: string too long: %ud chars", ulen);
+ return C9Estr;
+ }
+
+ if((b = R(c, 2+2+statsz, Rstat, tag, &err)) != NULL){
+ w16(&b, statsz+2);
+ w16(&b, statsz);
+ w16(&b, 0xffff); /* type */
+ w32(&b, 0xffffffff); /* dev */
+ w08(&b, s->qid.type);
+ w32(&b, s->qid.version);
+ w64(&b, s->qid.path);
+ w32(&b, s->mode);
+ w32(&b, s->atime);
+ w32(&b, s->mtime);
+ w64(&b, s->size);
+ wcs(&b, s->name, nlen);
+ wcs(&b, s->uid, ulen);
+ wcs(&b, s->gid, glen);
+ wcs(&b, s->muid, mulen);
+ err = c->end(c);
+ }
+ return err;
+}
+
+C9error
+s9wstat(C9ctx *c, C9tag tag)
+{
+ C9error err;
+
+ if(R(c, 0, Rwstat, tag, &err) != NULL)
+ err = c->end(c);
+ return err;
+}
+
+C9error
+s9proc(C9ctx *c)
+{
+ uint32_t i, sz, cnt, n, msize;
+ int readerr;
+ uint8_t *b;
+ C9error err;
+ C9t t;
+
+ readerr = -1;
+ if((b = c->read(c, 4, &readerr)) == NULL){
+ if(readerr != 0)
+ c->error(c, "s9proc: short read");
+ return readerr == 0 ? 0 : C9Epkt;
+ }
+
+ sz = r32(&b);
+ if(sz < 7 || sz > c->msize){
+ c->error(c, "s9proc: invalid packet size !(7 <= %ud <= %ud)", sz, c->msize);
+ return C9Epkt;
+ }
+ sz -= 4;
+ readerr = -1;
+ if((b = c->read(c, sz, &readerr)) == NULL){
+ if(readerr != 0)
+ c->error(c, "s9proc: short read");
+ return readerr == 0 ? 0 : C9Epkt;
+ }
+
+ t.type = r08(&b);
+ t.tag = r16(&b);
+ sz -= 3;
+
+ if((c->svflags & Svver) == 0 && t.type != Tversion){
+ c->error(c, "s9proc: expected Tversion, got %ud", t.type);
+ return C9Epkt;
+ }
+
+ switch(t.type){
+ case Tread:
+ if(sz < 4+8+4)
+ goto error;
+ t.fid = r32(&b);
+ t.read.offset = r64(&b);
+ t.read.size = r32(&b);
+ if(t.read.size > maxread(c))
+ t.read.size = maxread(c);
+ c->t(c, &t);
+ break;
+
+ case Twrite:
+ if(sz < 4+8+4)
+ goto error;
+ t.fid = r32(&b);
+ t.write.offset = r64(&b);
+ if((t.write.size = r32(&b)) < sz-4-8-4)
+ goto error;
+ if(t.write.size > maxwrite(c))
+ t.write.size = maxwrite(c);
+ t.write.data = b;
+ c->t(c, &t);
+ break;
+
+ case Tclunk:
+ case Tstat:
+ case Tremove:
+ if(sz < 4)
+ goto error;
+ t.fid = r32(&b);
+ c->t(c, &t);
+ break;
+
+ case Twalk:
+ if(sz < 4+4+2)
+ goto error;
+ t.fid = r32(&b);
+ t.walk.newfid = r32(&b);
+ if((n = r16(&b)) > 16){
+ c->error(c, "s9proc: Twalk !(%ud <= 16)", n);
+ return C9Epath;
+ }
+ sz -= 4+4+2;
+ if(n > 0){
+ for(i = 0; i < n; i++){
+ if(sz < 2 || (cnt = r16(&b)) > sz-2)
+ goto error;
+ if(cnt < 1){
+ c->error(c, "s9proc: Twalk invalid element [%ud]", i);
+ return C9Epath;
+ }
+ b[-2] = 0;
+ t.walk.wname[i] = (char*)b;
+ b += cnt;
+ sz -= 2 + cnt;
+ }
+ memmove(t.walk.wname[i-1]-1, t.walk.wname[i-1], (char*)b - t.walk.wname[i-1]);
+ t.walk.wname[i-1]--;
+ b[-1] = 0;
+ }else
+ i = 0;
+ t.walk.wname[i] = NULL;
+ c->t(c, &t);
+ break;
+
+ case Topen:
+ if(sz < 4+1)
+ goto error;
+ t.fid = r32(&b);
+ t.open.mode = r08(&b);
+ c->t(c, &t);
+ break;
+
+ case Twstat:
+ if(sz < 4+2)
+ goto error;
+ t.fid = r32(&b);
+ if((cnt = r16(&b)) > sz-4)
+ goto error;
+ if((err = c9parsedir(c, &t.wstat, &b, &cnt)) != 0){
+ c->error(c, "s9proc");
+ return err;
+ }
+ c->t(c, &t);
+ break;
+
+ case Tcreate:
+ if(sz < 4+2+4+1)
+ goto error;
+ t.fid = r32(&b);
+ if((cnt = r16(&b)) < 1 || cnt > sz-4-2-4-1)
+ goto error;
+ t.create.name = (char*)b;
+ b += cnt;
+ t.create.perm = r32(&b);
+ t.create.mode = r08(&b);
+ t.create.name[cnt] = 0;
+ c->t(c, &t);
+ break;
+
+ case Tflush:
+ if(sz < 2)
+ goto error;
+ t.flush.oldtag = r16(&b);
+ c->t(c, &t);
+ break;
+
+ case Tversion:
+ if(sz < 4+2 || (msize = r32(&b)) < C9minmsize || (cnt = r16(&b)) > sz-4-2)
+ goto error;
+ if(cnt < 6 || memcmp(b, "9P2000", 6) != 0){
+ if((b = R(c, 4+2+7, Rversion, 0xffff, &err)) != NULL){
+ w32(&b, 0);
+ wcs(&b, "unknown", 7);
+ err = c->end(c);
+ c->error(c, "s9proc: invalid version");
+ }
+ return C9Ever;
+ }
+ if(msize < c->msize)
+ c->msize = msize;
+ c->svflags |= Svver;
+ c->t(c, &t);
+ break;
+
+ case Tattach:
+ if(sz < 4+4+2+2)
+ goto error;
+ t.fid = r32(&b);
+ t.attach.afid = r32(&b);
+ cnt = r16(&b);
+ sz -= 4+4+2;
+ if(cnt+2 > sz)
+ goto error;
+ t.attach.uname = (char*)b;
+ b += cnt;
+ cnt = r16(&b);
+ b[-2] = 0;
+ sz -= cnt+2;
+ if(cnt > sz)
+ goto error;
+ memmove(b-1, b, cnt);
+ t.attach.aname = (char*)b-1;
+ t.attach.aname[cnt] = 0;
+ c->t(c, &t);
+ break;
+
+ case Tauth:
+ if(sz < 4+2+2)
+ goto error;
+ t.auth.afid = r32(&b);
+ cnt = r16(&b);
+ sz -= 4+2;
+ if(cnt+2 > sz)
+ goto error;
+ t.auth.uname = (char*)b;
+ b += cnt;
+ cnt = r16(&b);
+ b[-2] = 0;
+ sz -= cnt+2;
+ if(cnt > sz)
+ goto error;
+ memmove(b-1, b, cnt);
+ t.auth.aname = (char*)b-1;
+ t.auth.aname[cnt] = 0;
+ c->t(c, &t);
+ break;
+
+ default:
+ goto error;
+ }
+ return 0;
+error:
+ c->error(c, "s9proc: invalid packet (type=%ud)", t.type);
+ return C9Epkt;
+}
+
+#endif /* C9_NO_SERVER */
--- a/pull_subtree.sh
+++ /dev/null
@@ -1,2 +1,0 @@
-#!/bin/sh
-git subtree pull --prefix c9 git@git.sr.ht:~ft/c9 master --squash