#include "a.h"

Sx *Brdsx1(Biobuf*);

Sx*
Brdsx(Biobuf *b)
{
	Sx **sx, *x;
	int nsx;
	
	nsx = 0;
	sx = nil;
	while((x = Brdsx1(b)) != nil){
		sx = erealloc(sx, (nsx+1)*sizeof sx[0]);
		sx[nsx++] = x;
	}
	x = emalloc(sizeof *x);
	x->sx = sx;
	x->nsx = nsx;
	x->type = SxList;
	return x;
}

int 
sxwalk(Sx *sx)
{
	int i, n;
	
	if(sx == nil)
		return 1;
	switch(sx->type){
	default:
	case SxAtom:
	case SxString:
	case SxNumber:
		return 1;
	case SxList:
		n = 0;
		for(i=0; i<sx->nsx; i++)
			n += sxwalk(sx->sx[i]);
		return n;
	}
}

void
freesx(Sx *sx)
{
	int i;
	
	if(sx == nil)
		return;
	switch(sx->type){
	case SxAtom:
	case SxString:
		free(sx->data);
		break;
	case SxList:
		for(i=0; i<sx->nsx; i++)
			freesx(sx->sx[i]);
		free(sx->sx);
		break;
	}
	free(sx);
}

Sx*
Brdsx1(Biobuf *b)
{
	int c, len, nbr;
	char *s;
	vlong n;
	Sx *x;
	
	c = Bgetc(b);
	if(c == ' ')
		c = Bgetc(b);
	if(c < 0)
		return nil;
	if(c == '\r')
		c = Bgetc(b);
	if(c == '\n')
		return nil;
	if(c == ')'){	/* end of list */
		Bungetc(b);
		return nil;
	}
	if(c == '('){	/* parenthesized list */
		x = Brdsx(b);
		c = Bgetc(b);
		if(c != ')')	/* oops! not good */
			Bungetc(b);
		return x;
	}
	if(c == '{'){	/* length-prefixed string */
		len = 0;
		while((c = Bgetc(b)) >= 0 && isdigit(c))
			len = len*10 + c-'0';
		if(c != '}')	/* oops! not good */
			Bungetc(b);
		c = Bgetc(b);
		if(c != '\r')	/* oops! not good */
			;
		c = Bgetc(b);
		if(c != '\n')	/* oops! not good */
			;
		x = emalloc(sizeof *x);
		x->data = emalloc(len+1);
		if(Bread(b, x->data, len) != len)
			;	/* oops! */
		x->data[len] = 0;
		x->ndata = len;
		x->type = SxString;
		return x;
	}
	if(c == '"'){	/* quoted string */
		s = nil;
		len = 0;
		while((c = Bgetc(b)) >= 0 && c != '"'){
			if(c == '\\')
				c = Bgetc(b);
			s = erealloc(s, len+1);
			s[len++] = c;
		}
		s = erealloc(s, len+1);
		s[len] = 0;
		x = emalloc(sizeof *x);
		x->data = s;
		x->ndata = len;
		x->type = SxString;
		return x;
	}
	if(isdigit(c)){	/* number */
		n = c-'0';;
		while((c = Bgetc(b)) >= 0 && isdigit(c))
			n = n*10 + c-'0';
		Bungetc(b);
		x = emalloc(sizeof *x);
		x->number = n;
		x->type = SxNumber;
		return x;
	}
	/* atom */
	len = 1;
	s = emalloc(1);
	s[0] = c;
	nbr = 0;
	while((c = Bgetc(b)) >= 0 && c > ' ' && !strchr("(){}", c)){ 
		/* allow embedded brackets as in BODY[] */
		if(c == '['){
			if(s[0] == '[')
				break;
			else
				nbr++;
		}
		if(c == ']'){
			if(nbr > 0)
				nbr--;
			else
				break;
		}
		s = erealloc(s, len+1);
		s[len++] = c;
	}
	if(c != ' ')
		Bungetc(b);
	s = erealloc(s, len+1);
	s[len] = 0;
	x = emalloc(sizeof *x);
	x->type = SxAtom;
	x->data = s;
	x->ndata = len;
	return x;		
}

int
sxfmt(Fmt *fmt)
{
	int i, paren;
	Sx *sx;
	
	sx = va_arg(fmt->args, Sx*);
	if(sx == nil)
		return 0;

	switch(sx->type){
	case SxAtom:
	case SxString:
		return fmtprint(fmt, "%q", sx->data);

	case SxNumber:
		return fmtprint(fmt, "%lld", sx->number);

	case SxList:
		paren = !(fmt->flags&FmtSharp);
		if(paren)
			fmtrune(fmt, '(');
		for(i=0; i<sx->nsx; i++){
			if(i)
				fmtrune(fmt, ' ');
			fmtprint(fmt, "%$", sx->sx[i]);
		}
		if(paren)
			return fmtrune(fmt, ')');
		return 0;

	default:
		return fmtstrcpy(fmt, "?");
	}
}

int
oksx(Sx *sx)
{
	return sx->nsx >= 2 
		&& sx->sx[1]->type == SxAtom 
		&& cistrcmp(sx->sx[1]->data, "OK") == 0;
}