forked from amberisvibin/chibi-pc09
		
	
		
			
				
	
	
		
			256 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			256 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  *	Z80 - Assembler
 | |
|  *	Copyright (C) 1987-2006 by Udo Munk
 | |
|  *
 | |
|  *	History:
 | |
|  *	17-SEP-1987 Development under Digital Research CP/M 2.2
 | |
|  *	28-JUN-1988 Switched to Unix System V.3
 | |
|  *	22-OCT-2006 changed to ANSI C for modern POSIX OS's
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  *	module with table oprations on opcode and symbol tables
 | |
|  */
 | |
| 
 | |
| #include <stdlib.h>
 | |
| #include <stdio.h>
 | |
| #include <string.h>
 | |
| #include "z80a.h"
 | |
| #include "z80aglb.h"
 | |
| 
 | |
| extern void fatal(int, char *);
 | |
| extern void asmerr(int);
 | |
| 
 | |
| /*
 | |
|  *	binary search in sorted table opctab
 | |
|  *
 | |
|  *	Input: pointer to string with opcode
 | |
|  *
 | |
|  *	Output: pointer to table element, or NULL if not found
 | |
|  */
 | |
| struct opc *search_op(char *op_name)
 | |
| {
 | |
| 	register int cond;
 | |
| 	register struct opc *low, *high, *mid;
 | |
| 
 | |
| 	low = &opctab[0];
 | |
| 	high = &opctab[no_opcodes - 1];
 | |
| 	while (low <= high) {
 | |
| 		mid = low + (high - low) / 2;
 | |
| 		if ((cond = strcmp(op_name, mid->op_name)) < 0)
 | |
| 			high = mid - 1;
 | |
| 		else if (cond > 0)
 | |
| 			low = mid + 1;
 | |
| 		else
 | |
| 			return(mid);
 | |
| 	}
 | |
| 	return(NULL);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  *	binary search on sorted table opetab
 | |
|  *
 | |
|  *	Input: pointer to string with operand
 | |
|  *
 | |
|  *	Output: symbol for operand, NOOPERA if empty operand,
 | |
|  *		NOREG if operand not found
 | |
|  */
 | |
| int get_reg(char *s)
 | |
| {
 | |
| 	register int cond;
 | |
| 	register struct ope *low, *high, *mid;
 | |
| 
 | |
| 	if (s == NULL || *s == '\0')
 | |
| 		return(NOOPERA);
 | |
| 	low = &opetab[0];
 | |
| 	high = &opetab[no_operands - 1];
 | |
| 	while (low <= high) {
 | |
| 		mid = low + (high - low) / 2;
 | |
| 		if ((cond = strcmp(s, mid->ope_name)) < 0)
 | |
| 			high = mid - 1;
 | |
| 		else if (cond > 0)
 | |
| 			low = mid + 1;
 | |
| 		else
 | |
| 			return(mid->ope_sym);
 | |
| 	}
 | |
| 	return(NOREG);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  *	hash search on symbol table symtab
 | |
|  *
 | |
|  *	Input: pointer to string with symbol
 | |
|  *
 | |
|  *	Output: pointer to table element, or NULL if not found
 | |
|  */
 | |
| struct sym *get_sym(char *sym_name)
 | |
| {
 | |
| 	register struct sym *np;
 | |
| 
 | |
| 	for (np = symtab[hash(sym_name)]; np != NULL; np = np->sym_next)
 | |
| 		if (strcmp(sym_name, np->sym_name) == 0)
 | |
| 			return(np);
 | |
| 	return(NULL);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  *	add symbol to symbol table symtab, or modify existing symbol
 | |
|  *
 | |
|  *	Input: sym_name pointer to string with symbol name
 | |
|  *	       sym_wert value of symbol
 | |
|  *
 | |
|  *	Output: 0 symbol added/modified
 | |
|  *		1 out of memory
 | |
|  */
 | |
| int put_sym(char *sym_name, int sym_wert)
 | |
| {
 | |
| 	struct sym *get_sym();
 | |
| 	register int hashval;
 | |
| 	register struct sym *np;
 | |
| 
 | |
| 	char *strsave(char *);
 | |
| 
 | |
| 	if (!gencode)
 | |
| 		return(0);
 | |
| 	if ((np = get_sym(sym_name)) == NULL) {
 | |
| 		np = (struct sym *) malloc(sizeof (struct sym));
 | |
| 		if (np == NULL)
 | |
| 			return(1);
 | |
| 		if ((np->sym_name = strsave(sym_name)) == NULL)
 | |
| 			return(1);
 | |
| 		hashval = hash(sym_name);
 | |
| 		np->sym_next = symtab[hashval];
 | |
| 		symtab[hashval] = np;
 | |
| 	}
 | |
| 	np->sym_wert = sym_wert;
 | |
| 	return(0);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  *	add label to symbol table, error if symbol already exists
 | |
|  */
 | |
| void put_label(void)
 | |
| {
 | |
| 	struct sym *get_sym(char *);
 | |
| 
 | |
| 	if (get_sym(label) == NULL) {
 | |
| 		if (put_sym(label, pc))
 | |
| 			fatal(F_OUTMEM, "symbols");
 | |
| 	} else
 | |
| 		asmerr(E_MULSYM);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  *	hash algorithm
 | |
|  *
 | |
|  *	Input: pointer to string with name
 | |
|  *
 | |
|  *	Output: hash value
 | |
|  */
 | |
| int hash(char *name)
 | |
| {
 | |
| 	register int hashval;
 | |
| 
 | |
| 	for (hashval = 0; *name;)
 | |
| 		hashval += *name++;
 | |
| 	return(hashval % HASHSIZE);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  *	save string into allocated memory
 | |
|  *
 | |
|  *	Input: pointer to string
 | |
|  *
 | |
|  *	Output: pointer to allocated memory with string
 | |
|  */
 | |
| char *strsave(char *s)
 | |
| {
 | |
| 	register char *p;
 | |
| 
 | |
| 	if ((p = malloc((unsigned) strlen(s)+1)) != NULL)
 | |
| 		strcpy(p, s);
 | |
| 	return(p);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  *	copy whole symbol hast table into allocated pointer array
 | |
|  *	used for sorting the symbol table later
 | |
|  */
 | |
| int copy_sym(void)
 | |
| {
 | |
| 	register int i, j;
 | |
| 	register struct sym *np;
 | |
| 
 | |
| 	symarray = (struct sym **) malloc(SYMINC * sizeof(struct sym *));
 | |
| 	if (symarray == NULL)
 | |
| 		fatal(F_OUTMEM, "sorting symbol table");
 | |
| 	symsize = SYMINC;
 | |
| 	for (i = 0, j = 0; i < HASHSIZE; i++) {
 | |
| 		if (symtab[i] != NULL) {
 | |
| 			for (np = symtab[i]; np != NULL; np = np->sym_next) {
 | |
| 				symarray[j++] = np;
 | |
| 				if (j == symsize) {
 | |
| 					symarray = (struct sym **) realloc((char *) symarray, symsize * sizeof(struct sym *) + SYMINC * sizeof(struct sym *));
 | |
| 					if (symarray == NULL)
 | |
| 						fatal(F_OUTMEM, "sorting symbol table");
 | |
| 					symsize += SYMINC;
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	return(j);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  *	sort symbol table by name
 | |
|  */
 | |
| void n_sort_sym(int len)
 | |
| {
 | |
| 	register int gap, i, j;
 | |
| 	register struct sym *temp;
 | |
| 
 | |
| 	for (gap = len/2; gap > 0; gap /= 2)
 | |
| 		for (i = gap; i < len; i++)
 | |
| 			for (j = i-gap; j >= 0; j -= gap) {
 | |
| 				if (strcmp(symarray[j]->sym_name,
 | |
| 				    symarray[j+gap]->sym_name) <= 0)
 | |
| 					break;
 | |
| 				temp = symarray[j];
 | |
| 				symarray[j] = symarray[j+gap];
 | |
| 				symarray[j+gap] = temp;
 | |
| 			}
 | |
| }
 | |
| 
 | |
| /*
 | |
|  *	sort symbol table by address
 | |
|  */
 | |
| void a_sort_sym(int len)
 | |
| {
 | |
| 	register int gap, i, j;
 | |
| 	register struct sym *temp;
 | |
| 
 | |
| 	for (gap = len/2; gap > 0; gap /= 2)
 | |
| 		for (i = gap; i < len; i++)
 | |
| 			for (j = i-gap; j >= 0; j -= gap) {
 | |
| 				if (numcmp(symarray[j]->sym_wert,
 | |
| 				    symarray[j+gap]->sym_wert) <= 0)
 | |
| 					break;
 | |
| 				temp = symarray[j];
 | |
| 				symarray[j] = symarray[j+gap];
 | |
| 				symarray[j+gap] = temp;
 | |
| 			}
 | |
| }
 | |
| 
 | |
| /*
 | |
|  *	compares two 16bit values, result like strcmp()
 | |
|  */
 | |
| int numcmp(int n1, int n2)
 | |
| {
 | |
| 	if ((unsigned) (n1 & 0xffff) < (unsigned) (n2 & 0xffff))
 | |
| 		return(-1);
 | |
| 	else if ((unsigned) (n1 & 0xffff) > (unsigned) (n2 & 0xffff))
 | |
| 		return(1);
 | |
| 	else
 | |
| 		return(0);
 | |
| }
 | 
