shithub: libstl

ref: 103c98aa1de99f0f91e269416fbced9ed07b20ad
dir: /stl.c/

View raw version
#include <u.h>
#include <libc.h>
#include <bio.h>
#include "stl.h"

#define min(a,b) ((a)<(b)?(a):(b))

typedef struct Line Line;
struct Line
{
	char *file;
	ulong line;
};

static void
error(Line *l, char *fmt, ...)
{
	va_list va;
	char buf[ERRMAX], *bp;

	va_start(va, fmt);
	bp = seprint(buf, buf + sizeof(buf), "line %lud: ", l->line);
	vseprint(bp, buf + sizeof(buf), fmt, va);
	va_end(va);
	werrstr("%s", buf);
}

static int
Bgets(Biobuf *b, u16int *s)
{
	uchar buf[2];

	if(Bread(b, buf, 2) != 2){
		werrstr("could not get 2 bytes");
		return -1;
	}
	*s = buf[0] | buf[1] << 8;
	return 0;
}

static int
Bgetl(Biobuf *b, u32int *l)
{
	uchar buf[4];

	if(Bread(b, buf, 4) != 4){
		werrstr("could not get 4 bytes");
		return -1;
	}
	*l = buf[0] | buf[1] << 8 | buf[2] << 16 | buf[3] << 24;
	return 0;
}

static int
Bgetf(Biobuf *b, float *f)
{
	u32int l;

	if(Bgetl(b, &l) < 0){
		werrstr("Bgetl: %r");
		return -1;
	}
	*f = *(float*)&l;
	return 0;
}

static int bunpack(Biobuf*, char*, ...);

static int
vbunpack(Biobuf *b, char *fmt, va_list a)
{
	u16int s;
	u32int l;
	float f, *v;
	void *p;

	for(;;){
		switch(*fmt++){
		case '\0':
			return 0;
		case 's':
			if(Bgets(b, &s) < 0)
				goto error;
			*va_arg(a, ushort*) = s;
			break;
		case 'l':
			if(Bgetl(b, &l) < 0)
				goto error;
			*va_arg(a, ulong*) = l;
			break;
		case 'f':
			if(Bgetf(b, &f) < 0)
				goto error;
			*va_arg(a, float*) = f;
			break;
		case 'v':
			v = va_arg(a, float*);
			if(bunpack(b, "fff", v+0, v+1, v+2) < 0)
				goto error;
			break;
		case '[':
			p = va_arg(a, void*);
			s = va_arg(a, ushort);
			if(Bread(b, p, s) != s){
				werrstr("Bread: could not read %ud bytes", s);
				goto error;
			}
			break;
		}
	}
error:
	return -1;
}

static int
bunpack(Biobuf *b, char *fmt, ...)
{
	va_list a;
	int n;

	va_start(a, fmt);
	n = vbunpack(b, fmt, a);
	va_end(a);

	return n;
}

static char *
getline(Line *l, Biobuf *b)
{
	char *line;

	line = Brdstr(b, '\n', 1);
	if(line == nil){
		werrstr("could not read a line");
		return nil;
	}
	l->line++;
	return line;
}

static Stl *
parsetxt(Biobuf *bin)
{
	Line ctx;
	Stltri *tri;
	Stl *stl;
	char *line, *f[5];
	int nf, i;

	memset(&ctx, 0, sizeof(Line));

	/* header */
	if((line = getline(&ctx, bin)) == nil){
		error(&ctx, "getline: %r");
		return nil;
	}

	stl = mallocz(sizeof(Stl), 1);
	if(stl == nil){
		error(&ctx, "mallocz0: %r");
		free(line);
		return nil;
	}
	memmove(stl->hdr, line, min(sizeof(stl->hdr), Blinelen(bin)));

	for(;;){
		/* facet normal i j k */
		if((line = getline(&ctx, bin)) == nil){
badline0:
			error(&ctx, "getline: %r");
			goto error;
		}
		nf = tokenize(line, f, nelem(f));
repeat:
		if(nf != 5){
notenough0:
			error(&ctx, "not enough fields %d", nf);
cleanup0:
			free(line);
			goto error;
		}
		if(strcmp(f[0], "facet") == 0 && strcmp(f[1], "normal") == 0){
			tri = mallocz(sizeof(Stltri), 1);
			if(tri == nil){
				werrstr("mallocz1: %r");
				goto cleanup0;
			}
			tri->n[0] = strtod(f[2], nil);
			tri->n[1] = strtod(f[3], nil);
			tri->n[2] = strtod(f[4], nil);
		}else{
			error(&ctx, "expected \"facet normal n₀ n₁ n₂\"");
			goto cleanup0;
		}
		free(line);

		/* outer loop */
		if((line = getline(&ctx, bin)) == nil){
badline1:
			free(tri);
			goto badline0;
		}
		nf = tokenize(line, f, nelem(f));
		if(nf != 2){
notenough1:
			free(tri);
			goto notenough0;
		}
		if(strcmp(f[0], "outer") != 0 && strcmp(f[1], "loop") != 0){
			error(&ctx, "expected \"outer loop\"");
cleanup1:
			free(tri);
			goto cleanup0;
		}
		free(line);

		/* [3]vertex x y z */
		for(i = 0; i < 3; i++){
			if((line = getline(&ctx, bin)) == nil)
				goto badline1;
			nf = tokenize(line, f, nelem(f));
			if(nf != 4)
				goto notenough1;
			if(strcmp(f[0], "vertex") == 0){
				tri->v[i][0] = strtod(f[1], nil);
				tri->v[i][1] = strtod(f[2], nil);
				tri->v[i][2] = strtod(f[3], nil);
			}else{
				error(&ctx, "expected \"vertex x y z\"");
				goto cleanup1;
			}
			free(line);
		}

		/* endloop */
		if((line = getline(&ctx, bin)) == nil){
			goto badline1;
		}
		nf = tokenize(line, f, nelem(f));
		if(nf != 1){
			goto notenough1;
		}
		if(strcmp(f[0], "endloop") != 0){
			error(&ctx, "expected \"endloop\"");
			goto cleanup1;
		}
		free(line);

		/* endfacet */
		if((line = getline(&ctx, bin)) == nil){
			goto badline1;
		}
		nf = tokenize(line, f, nelem(f));
		if(nf != 1){
			goto notenough1;
		}
		if(strcmp(f[0], "endfacet") == 0){
			stl->tris = realloc(stl->tris, (stl->ntris+1)*sizeof(*stl->tris));
			if(stl->tris == nil){
				error(&ctx, "realloc1: %r");
				goto cleanup1;
			}
			stl->tris[stl->ntris++] = tri;
		}else{
			error(&ctx, "expected \"endfacet\"");
			goto cleanup1;
		}
		free(line);

		/* endsolid? */
		if((line = getline(&ctx, bin)) == nil){
			goto badline0;
		}
		nf = tokenize(line, f, nelem(f));
		if(nf < 2){
			goto notenough0;
		}
		if(nf == 2 && strcmp(f[0], "endsolid") == 0){
			free(line);
			break;
		}else
			goto repeat;
	}

	return stl;
error:
	freestl(stl);
	return nil;
}

static Stl *
parsebin(Biobuf *bin, char *magic)
{
	Stltri *tri;
	Stl *stl;
	u32int i, ntris;

	stl = mallocz(sizeof(Stl), 1);
	if(stl == nil){
		werrstr("mallocz0: %r");
		return nil;
	}
	memmove(stl->hdr, magic, 5);
	if(bunpack(bin, "[l", stl->hdr+5, sizeof(stl->hdr)-5, &ntris) < 0){
		werrstr("hdr bunpack: %r");
		goto error;
	}

	for(i = 0; i < ntris; i++){
		tri = mallocz(sizeof(Stltri), 1);
		if(tri == nil){
			werrstr("mallocz1: %r");
			goto error;
		}
		if(bunpack(bin, "vvvvs", tri->n, tri->v[0], tri->v[1], tri->v[2], &tri->attrlen) < 0){
			free(tri);
			werrstr("tri bunpack0: %r");
			goto error;
		}
		tri = realloc(tri, sizeof(Stltri) + tri->attrlen);
		if(tri == nil){
			werrstr("realloc0: %r");
			goto error;
		}
		if(bunpack(bin, "[", tri->attrs, tri->attrlen) < 0){
			free(tri);
			werrstr("tri bunpack1: %r");
			goto error;
		}

		stl->tris = realloc(stl->tris, (stl->ntris+1)*sizeof(*stl->tris));
		if(stl->tris == nil){
			free(tri);
			werrstr("realloc1: %r");
			goto error;
		}
		stl->tris[stl->ntris++] = tri;
	}

	return stl;
error:
	freestl(stl);
	return nil;
}

Stl *
readstl(int fd)
{
	Biobuf *bin;
	Stl *stl;
	char magic[5+1];

	stl = nil;

	bin = Bfdopen(fd, OREAD);
	if(bin == nil){
		werrstr("Bfdopen: %r");
		return nil;
	}

	if(Bread(bin, magic, 5) != 5){
		werrstr("Bread: could not read 5 bytes");
		goto out;
	}
	magic[5] = 0;

	if(strcmp(magic, "solid") == 0){
		if((stl = parsetxt(bin)) == nil)
			werrstr("parsetxt: %r");
	}else
		if((stl = parsebin(bin, magic)) == nil)
			werrstr("parsebin: %r");

out:
	Bterm(bin);
	return stl;
}

void
freestl(Stl *stl)
{
	int i;

	if(stl == nil)
		return;

	for(i = 0; i < stl->ntris; i++)
		free(stl->tris[i]);
	free(stl->tris);
	free(stl);
}