/* z80adr.c */

/*
 * (C) Copyright 1989,1990
 * All Rights Reserved
 *
 * Alan R. Baldwin
 * 721 Berkeley St.
 * Kent, Ohio  44240
 */

#include <stdio.h>
#include <setjmp.h>
#include "asm.h"
#include "z80.h"

/*
 * Read an address specifier. Pack the
 * address information into the supplied
 * `expr' structure. Return the mode of
 * the address.
 *
 * This addr(esp) routine performs the following addressing decoding:
 *
 *	address		mode		flag		addr		base
 *	#n			S_IMMED		0			n			NULL
 *	label		s_type		----		s_addr		s_area
 *	[REG]		S_IND+icode	0			0			NULL
 *	[label]		S_INDM		----		s_addr		s_area
 *	offset[REG]	S_IND+icode	----		offset		----
 */
int addr(register struct expr *esp)
{
	register int c, mode, indx;

	if ((c = getnb()) == '#') {
		expr(esp, 0);
		esp->e_mode = S_IMMED;
	} else
	if (c == LFIND) {
		if ((indx = admode(R8)) != 0) {
			mode = S_INDB;
		} else
		if ((indx = admode(R16)) != 0) {
			mode = S_INDR;
		} else	
		if ((indx = admode(R8X)) != 0) {
			mode = S_R8X;
			aerr();
		} else
		if ((indx = admode(R16X)) != 0) {
			mode = S_R16X;
			aerr();
		} else {
			expr(esp, 0);
			esp->e_mode = S_INDM;
		}
		if (indx) {
			esp->e_flag = 0;
			esp->e_addr = 0;
			esp->e_mode = mode + (indx&0xFF);
			esp->e_base.e_ap = NULL;
		}
		if ((c = getnb()) != RTIND)
			qerr();
	} else {
		unget(c);
		if ((indx = admode(R8)) != 0) {
			mode = S_R8;
		} else
		if ((indx = admode(R16)) != 0) {
			mode = S_R16;
		} else	
		if ((indx = admode(R8X)) != 0) {
			mode = S_R8X;
		} else
		if ((indx = admode(R16X)) != 0) {
			mode = S_R16X;
		} else {
			expr(esp, 0);
			esp->e_mode = S_USER;
		}
		if (indx) {
			esp->e_flag = 0;
			esp->e_addr = indx&0xFF;
			esp->e_mode = mode;
			esp->e_base.e_ap = NULL;
		}
		if ((c = getnb()) == LFIND) {
			if ((indx=admode(R16))!=0 && ((indx&0xFF)==IX ||
				(indx&0xFF)==IY || (indx&0xFF)==SP)) {
				esp->e_mode = S_INDR + (indx&0xFF);
			} else {
				aerr();
			}
			if ((c = getnb()) != RTIND)
				qerr();
		} else {
			unget(c);
		}
	}
	return (esp->e_mode);
}

/*
 * Enter admode() to search a specific addressing mode table
 * for a match. Return the addressing value on a match or
 * zero for no match.
 */
int admode(register struct adsym *sp)
{
	register char *ptr;
	register int i;
	unget(getnb());
	i = 0;
	while ( *(ptr = (char *) &sp[i]) ) {
		if (srch(ptr)) {
			return(sp[i].a_val);
		}
		i++;
	}
	return(0);
}

/*
 *      srch --- does string match ?
 */
int srch(register char *str)
{
	register char *ptr;
	ptr = ip;

	if(casesensitive) {
		while (*ptr && *str) {
			if (*ptr != *str)
				break;
			ptr++;
			str++;
		}
		if (*ptr == *str) {
			ip = ptr;
			return(1);
		}
	} else {
		while (*ptr && *str) {
			if (ccase[(int)*ptr] != ccase[(int)*str])
				break;
			ptr++;
			str++;
		}
		if (ccase[(int)*ptr] == ccase[(int)*str]) {
			ip = ptr;
			return(1);
		}
	}

	if (!*str)
		if (any(*ptr," \t\n,);")) {
			ip = ptr;
			return(1);
		}
	return(0);
}

/*
 *      any --- does str contain c?
 */
int any(char c, char *str)
{
	while (*str)
		if(*str++ == c)
			return(1);
	return(0);
}

/*
 * Registers
 */

struct	adsym	R8[] = {
	{ "b",	B|0400 },
	{ "c",	C|0400 },
	{ "d",	D|0400 },
	{ "e",	E|0400 },
	{ "h",	H|0400 },
	{ "l",	L|0400 },
	{ "a",	A|0400 },
	{ "",	0000 }
};

struct	adsym	R8X[] = {
	{ "i",	I|0400 },
	{ "r",	R|0400 },
	{ "",	0000 }
};

struct	adsym	R16[] = {
	{ "bc",	BC|0400 },
	{ "de",	DE|0400 },
	{ "hl-",	HLD|0400 },
	{ "hl+",	HLI|0400 },
	{ "hl",	HL|0400 },
	{ "sp",	SP|0400 },
	{ "ix",	IX|0400 },
	{ "iy",	IY|0400 },
	{ "",	0000 }
};

struct	adsym	R16X[] = {
	{ "af",	AF|0400 },
	{ "af'",	AF|0400 },
	{ "",	0000 }
};

/*
 * Conditional definitions
 */

struct	adsym	CND[] = {
	{ "NZ",	NZ|0400 },
	{ "Z",	Z |0400 },
	{ "NC",	NC|0400 },
	{ "C",	CS|0400 },
	{ "PO",	PO|0400 },
	{ "PE",	PE|0400 },
	{ "P",	P |0400 },
	{ "M",	M |0400 },
	{ "",	0000 }
};
