OpenCores
URL https://opencores.org/ocsvn/or1k/or1k/trunk

Subversion Repositories or1k

[/] [or1k/] [trunk/] [insight/] [utils/] [amd-udi/] [mondfe/] [asm.c] - Rev 578

Go to most recent revision | Compare with Previous | Blame | View Log

static char _[] = " @(#)asm.c	5.23 93/10/26 10:17:03, Srini, AMD";
/******************************************************************************
 * Copyright 1991 Advanced Micro Devices, Inc.
 *
 * This software is the property of Advanced Micro Devices, Inc  (AMD)  which
 * specifically  grants the user the right to modify, use and distribute this
 * software provided this notice is not removed or altered.  All other rights
 * are reserved by AMD.
 *
 * AMD MAKES NO WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, WITH REGARD TO THIS
 * SOFTWARE.  IN NO EVENT SHALL AMD BE LIABLE FOR INCIDENTAL OR CONSEQUENTIAL
 * DAMAGES IN CONNECTION WITH OR ARISING FROM THE FURNISHING, PERFORMANCE, OR
 * USE OF THIS SOFTWARE.
 *
 * So that all may benefit from your experience, please report  any  problems
 * or  suggestions about this software to the 29K Technical Support Center at
 * 800-29-29-AMD (800-292-9263) in the USA, or 0800-89-1131  in  the  UK,  or
 * 0031-11-1129 in Japan, toll free.  The direct dial number is 512-462-4118.
 *
 * Advanced Micro Devices, Inc.
 * 29K Support Products
 * Mail Stop 573
 * 5900 E. Ben White Blvd.
 * Austin, TX 78741
 * 800-292-9263
 *****************************************************************************
 *      Engineer: Srini Subramanian.
 *****************************************************************************
 * This module supports the assemble command to assemble 29K instructions
 * in memory.
 *****************************************************************************
 */
 
 
#include <stdio.h>
#include "opcodes.h"
#include "memspcs.h"
#include "main.h"
#include "monitor.h"
#include "macros.h"
#include "miniint.h"
#include "error.h"
 
#ifdef	MSDOS
#include <string.h>
#define	strcasecmp	stricmp
#else
#include <string.h>
#endif
 
 
/*
** There are approximately 23 different instruction formats for the
** Am29000.  Instructions are assembled using one of these formats.
**
** Note:  Opcodes in the "switch" statement are sorted in numerical
**        order.
**
*/
 
 
int  	get_addr_29k_m PARAMS((char *, struct addr_29k_t *, INT32));
int  	addr_29k_ok PARAMS((struct addr_29k_t *));
void 	convert32 PARAMS((BYTE *));
int  	set_data PARAMS((BYTE *, BYTE *, int));
 
int  asm_instr PARAMS((struct instr_t *, char **, int));
 
int  asm_arith_logic PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_load_store PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_vector PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_no_parms PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_one_parms PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_float PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_call_jmp PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_calli_jmpi PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_class PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_clz PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_const PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_consth PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_convert PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_div0 PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_exhws PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_jmp PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_jmpi PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_mfsr PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_mtsr PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_mtsrim PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_mftlb PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_mttlb PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_sqrt PARAMS((struct instr_t *, struct addr_29k_t *, int));
int  asm_emulate PARAMS((struct instr_t *, struct addr_29k_t *, int));
 
extern	void	Mini_poll_kbd PARAMS((char  *cmd_buffer, int size, int mode));
extern	int	Mini_cmdfile_input PARAMS((char  *cmd_buffer, int size));
extern	int   	tokenize_cmd PARAMS((char *, char **));
extern 	void 	lcase_tokens PARAMS((char **, int));
extern	INT32 	do_assemble PARAMS(( struct addr_29k_t	addr_29k,
				     char	*token[],
				     int	token_count));
#ifndef XRAY
 
extern 	char  	cmd_buffer[];
 
#define	MAX_ASM_TOKENS	15
static	char	*asm_token[MAX_ASM_TOKENS];
static	int	asm_token_count;
 
/*
** This function is used to assemble an instruction.  The command
** takes as parameters an array of strings (*token[]) and a
** count (token_count) which gives the number of tokens in the
** array.  These tokens should have the following values:
**
** token[0] - 'a' (the assemble command)
** token[1] - <address> (the address to assemble instruction at)
** token[2] - <opcode>  (the 29K opcode nmemonic)
** token[3] to token[n] - parameters to the assembly instruction.
**
*/
 
INT32
asm_cmd(token, token_count)
   char   *token[];
   int     token_count;
   {
   INT32		result;
   struct addr_29k_t 	addr_29k;
   int		asm_done;
 
   /*
   ** Parse parameters
   */
 
   if ((token_count < 2) || (token_count > 9)) {
      return (EMSYNTAX);
   } else if (token_count == 2) {
      /* Get address of assembly */
      result = get_addr_29k_m(token[1], &addr_29k, I_MEM);
      if (result != 0)
         return (result);
      result = addr_29k_ok(&addr_29k);
      if (result != 0)
         return (result);
      asm_done = 0;
      fprintf(stderr, "0x%08lx:\t", addr_29k.address);
      do {
        if (io_config.cmd_file_io == TRUE) {
	     if (Mini_cmdfile_input(cmd_buffer, BUFFER_SIZE) == SUCCESS) {
               fprintf(stderr, "%s", cmd_buffer); 
	     } else {
               Mini_poll_kbd(cmd_buffer, BUFFER_SIZE, BLOCK); /* block */
	     }
	} else {
             Mini_poll_kbd(cmd_buffer, BUFFER_SIZE, BLOCK); /* block */
	}
	if (io_config.log_file)  /* make a log file */
#ifdef	MSDOS
            fprintf(io_config.log_file, "%s\n", cmd_buffer);
#else
            fprintf(io_config.log_file, "%s", cmd_buffer);
#endif
     	if (io_config.echo_mode == (INT32) TRUE)
#ifdef	MSDOS
            fprintf(io_config.echo_file, "%s\n", cmd_buffer);
#else
            fprintf(io_config.echo_file, "%s", cmd_buffer);
#endif
        asm_token_count = tokenize_cmd(cmd_buffer, asm_token);
        lcase_tokens(asm_token, asm_token_count);
	if (strcmp(token[0], ".") == 0)
	  asm_done = 1;
	else {
          result= do_assemble(addr_29k, &asm_token[0], asm_token_count);
	  if (result != SUCCESS)
	    warning (result);
	  else
	    addr_29k.address = addr_29k.address + 4;
	  fprintf(stderr, "0x%08lx:\t", addr_29k.address);
	}
      } while (asm_done != 1);
   } else {
      /* Get address of assembly */
      result = get_addr_29k_m(token[1], &addr_29k, I_MEM);
      if (result != 0)
         return (result);
      result = addr_29k_ok(&addr_29k);
      if (result != 0)
         return (result);
      return (do_assemble(addr_29k, &token[2], (token_count-2)));
   }
   return (SUCCESS);
}
 
 
INT32
do_assemble(addr_29k, token, token_count)
struct addr_29k_t	addr_29k;
char			*token[];
int			token_count;
{
   INT32    result;
   struct instr_t    instr;
 
   INT32	retval;
   BYTE		*write_data;
   INT32	bytes_ret;
   INT32	hostendian;	/* for UDI conformant */
 
   /* Initialize instr */
   instr.op = 0;
   instr.c  = 0;
   instr.a  = 0;
   instr.b  = 0;
 
   /* Assemble instruction */
   result = asm_instr(&instr, &(token[0]), token_count);
 
   if (result != 0)
      return (EMSYNTAX);
 
   /* Will the data overflow the message buffer?  done in TIP */
   write_data = (BYTE *) &instr;
 
   hostendian = FALSE;
   if ((retval = Mini_write_req (addr_29k.memory_space,
				 addr_29k.address, 
				 1, /* count */
				 (INT16) sizeof(INST32),  /* size */
				 &bytes_ret,
				 write_data,
				 hostendian)) != SUCCESS) {
      return(FAILURE);
   }; 
   return (SUCCESS);
}
#endif
 
/*
** This function is used to assemble a single Am29000 instruction.
** The token[] array contains the lower-case tokens for a single
** assembler instruction.  The token_count contains the number of
** tokens in the array.  This number should be at least 1 (as in the
** cases of instructions like IRET) and at most 5 (for instructions
** like LOAD).
*/
 
#ifdef XRAY
  extern struct t_inst_table {
	char  *inst_mnem;
	unsigned char	oprn_fmt;
} inst_table[];
#endif
 
int
asm_instr(instr, token, token_count)
   struct   instr_t *instr;
   char    *token[];
   int      token_count;
   {
   int    i;
   int    result;
   struct addr_29k_t parm[10];
   char   temp_opcode[20];
   char  *temp_ptr;
   int    opcode_found;
   static char  *str_0x40="0x40";
   static char  *str_gr1="gr1";
 
 
   /* Is token_count valid, and is the first token a string? */
   if ((token_count < 1) ||
       (token_count > 7) ||
       (strcmp(token[0], "") == 0))
      return (EMSYNTAX);
 
   /* Get opcode */
 
   /*
   ** Note:  Since the opcode_name[] string used in the disassembler
   ** uses padded strings, we cannot do a strcmp().  We canot do a
   ** strncmp() of the length of token[0] either, since "and" will
   ** match up (for instance) with "andn".  So we copy the string,
   ** null terminate at the first pad character (space), and then
   ** compare.  This is inefficient, but necessary.
   */
 
   i=0;
   opcode_found = FALSE;
   while ((i<256) && (opcode_found != TRUE)) {
#ifdef XRAY
      result = strcasecmp(token[0], inst_table[i].inst_mnem);
#else
      temp_ptr = strcpy(temp_opcode, opcode_name[i]);
 
      if (strcmp(temp_ptr, "") != 0)
         temp_ptr = strtok(temp_opcode, " ");
      result = strcmp(token[0], temp_ptr);
#endif
 
      if (result == 0) {
         opcode_found = TRUE;
         instr->op = (BYTE) i;
         }
      i = i + 1;
      }  /* end while */
 
   /* Check for a NOP */
   if ((opcode_found == FALSE) &&
       (strcasecmp(token[0], "nop") == 0)) {
      opcode_found = TRUE;
      instr->op = ASEQ0;
      /* Fake up tokens to give "aseq 0x40,gr1,gr1" */
      token_count = 4;
      token[1] = str_0x40;
      token[2] = str_gr1;
      token[3] = str_gr1;
      }
 
   if (opcode_found == FALSE)
      return (EMSYNTAX);
 
   if ((strcasecmp(token[0], "iretinv") == 0) ||
       (strcasecmp(token[0], "inv") == 0) ) {
       /* iretinv and inv instructions */
      for (i=1; i<token_count; i=i+1) {
         result = get_addr_29k_m(token[i], &(parm[i-1]), GENERIC_SPACE);
         if (result != 0)
            return (result);
      }
   } else {
   /* Make the other tokens into addr_29k */
   for (i=1; i<token_count; i=i+1) {
      result = get_addr_29k_m(token[i], &(parm[i-1]), I_MEM);
      if (result != 0)
         return (result);
      /* Check if register values are legal */
      if (ISREG(parm[i-1].memory_space)) {
         result = addr_29k_ok(&(parm[i-1]));
         if (result != 0)
            return (EMBADREG);
         }
      /* Set bit 7 if a LOCAL_REG */
      if (parm[i-1].memory_space == LOCAL_REG)
         parm[i-1].address = (parm[i-1].address | 0x80);
      }
   }
 
 
   switch (instr->op) {   
 
      /* Opcodes 0x00 to 0x0F */
      case ILLEGAL_00:  result = EMSYNTAX;
                        break;
      case CONSTN:      result = asm_const(instr, parm, 2);
                        break;
      case CONSTH:      result = asm_consth(instr, parm, 2);
                        break;
      case CONST:       result = asm_const(instr, parm, 2);
                        break;
      case MTSRIM:      result = asm_mtsrim(instr, parm, 2);
                        break;
      case CONSTHZ:     result = asm_const(instr, parm, 2);
                        break;
      case LOADL0:      result = asm_load_store(instr, parm, 4);
                        break;
      case LOADL1:      result = asm_load_store(instr, parm, 4);
                        break;
      case CLZ0:        result = asm_clz(instr, parm, 2);
                        break;
      case CLZ1:        result = asm_clz(instr, parm, 2);
                        break;
      case EXBYTE0:     result = asm_arith_logic(instr, parm, 3);
                        break;
      case EXBYTE1:     result = asm_arith_logic(instr, parm, 3);
                        break;
      case INBYTE0:     result = asm_arith_logic(instr, parm, 3);
                        break;
      case INBYTE1:     result = asm_arith_logic(instr, parm, 3);
                        break;
      case STOREL0:     result = asm_load_store(instr, parm, 4);
                        break;
      case STOREL1:     result = asm_load_store(instr, parm, 4);
                        break;
 
      /* Opcodes 0x10 to 0x1F */
      case ADDS0:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case ADDS1:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case ADDU0:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case ADDU1:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case ADD0:        result = asm_arith_logic(instr, parm, 3);
                        break;
      case ADD1:        result = asm_arith_logic(instr, parm, 3);
                        break;
      case LOAD0:       result = asm_load_store(instr, parm, 4);
                        break;
      case LOAD1:       result = asm_load_store(instr, parm, 4);
                        break;
      case ADDCS0:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case ADDCS1:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case ADDCU0:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case ADDCU1:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case ADDC0:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case ADDC1:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case STORE0:      result = asm_load_store(instr, parm, 4);
                        break;
      case STORE1:      result = asm_load_store(instr, parm, 4);
                        break;
 
      /* Opcodes 0x20 to 0x2F */
      case SUBS0:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case SUBS1:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case SUBU0:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case SUBU1:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case SUB0:        result = asm_arith_logic(instr, parm, 3);
                        break;
      case SUB1:        result = asm_arith_logic(instr, parm, 3);
                        break;
      case LOADSET0:    result = asm_load_store(instr, parm, 4);
                        break;
      case LOADSET1:    result = asm_load_store(instr, parm, 4);
                        break;
      case SUBCS0:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case SUBCS1:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case SUBCU0:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case SUBCU1:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case SUBC0:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case SUBC1:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case CPBYTE0:     result = asm_arith_logic(instr, parm, 3);
                        break;
      case CPBYTE1:     result = asm_arith_logic(instr, parm, 3);
                        break;
 
 
      /* Opcodes 0x30 to 0x3F */
      case SUBRS0:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case SUBRS1:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case SUBRU0:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case SUBRU1:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case SUBR0:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case SUBR1:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case LOADM0:      result = asm_load_store(instr, parm, 4);
                        break;
      case LOADM1:      result = asm_load_store(instr, parm, 4);
                        break;
      case SUBRCS0:     result = asm_arith_logic(instr, parm, 3);
                        break;
      case SUBRCS1:     result = asm_arith_logic(instr, parm, 3);
                        break;
      case SUBRCU0:     result = asm_arith_logic(instr, parm, 3);
                        break;
      case SUBRCU1:     result = asm_arith_logic(instr, parm, 3);
                        break;
      case SUBRC0:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case SUBRC1:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case STOREM0:     result = asm_load_store(instr, parm, 4);
                        break;
      case STOREM1:     result = asm_load_store(instr, parm, 4);
                        break;
 
      /* Opcodes 0x40 to 0x4F */
      case CPLT0:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case CPLT1:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case CPLTU0:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case CPLTU1:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case CPLE0:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case CPLE1:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case CPLEU0:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case CPLEU1:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case CPGT0:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case CPGT1:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case CPGTU0:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case CPGTU1:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case CPGE0:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case CPGE1:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case CPGEU0:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case CPGEU1:      result = asm_arith_logic(instr, parm, 3);
                        break;
 
      /* Opcodes 0x50 to 0x5F */
      case ASLT0:       result = asm_vector(instr, parm, 3);
                        break;
      case ASLT1:       result = asm_vector(instr, parm, 3);
                        break;
      case ASLTU0:      result = asm_vector(instr, parm, 3);
                        break;
      case ASLTU1:      result = asm_vector(instr, parm, 3);
                        break;
      case ASLE0:       result = asm_vector(instr, parm, 3);
                        break;
      case ASLE1:       result = asm_vector(instr, parm, 3);
                        break;
      case ASLEU0:      result = asm_vector(instr, parm, 3);
                        break;
      case ASLEU1:      result = asm_vector(instr, parm, 3);
                        break;
      case ASGT0:       result = asm_vector(instr, parm, 3);
                        break;
      case ASGT1:       result = asm_vector(instr, parm, 3);
                        break;
      case ASGTU0:      result = asm_vector(instr, parm, 3);
                        break;
      case ASGTU1:      result = asm_vector(instr, parm, 3);
                        break;
      case ASGE0:       result = asm_vector(instr, parm, 3);
                        break;
      case ASGE1:       result = asm_vector(instr, parm, 3);
                        break;
      case ASGEU0:      result = asm_vector(instr, parm, 3);
                        break;
      case ASGEU1:      result = asm_vector(instr, parm, 3);
                        break;
 
      /* Opcodes 0x60 to 0x6F */
      case CPEQ0:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case CPEQ1:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case CPNEQ0:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case CPNEQ1:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case MUL0:        result = asm_arith_logic(instr, parm, 3);
                        break;
      case MUL1:        result = asm_arith_logic(instr, parm, 3);
                        break;
      case MULL0:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case MULL1:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case DIV0_OP0:    result = asm_div0(instr, parm, 2);
                        break;
      case DIV0_OP1:    result = asm_div0(instr, parm, 2);
                        break;
      case DIV_OP0:     result = asm_arith_logic(instr, parm, 3);
                        break;
      case DIV_OP1:     result = asm_arith_logic(instr, parm, 3);
                        break;
      case DIVL0:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case DIVL1:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case DIVREM0:     result = asm_arith_logic(instr, parm, 3);
                        break;
      case DIVREM1:     result = asm_arith_logic(instr, parm, 3);
                        break;
 
      /* Opcodes 0x70 to 0x7F */
      case ASEQ0:       result = asm_vector(instr, parm, 3);
                        break;
      case ASEQ1:       result = asm_vector(instr, parm, 3);
                        break;
      case ASNEQ0:      result = asm_vector(instr, parm, 3);
                        break;
      case ASNEQ1:      result = asm_vector(instr, parm, 3);
                        break;
      case MULU0:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case MULU1:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case ILLEGAL_76:  result = EMSYNTAX;
                        break;
      case ILLEGAL_77:  result = EMSYNTAX;
                        break;
      case INHW0:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case INHW1:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case EXTRACT0:    result = asm_arith_logic(instr, parm, 3);
                        break;
      case EXTRACT1:    result = asm_arith_logic(instr, parm, 3);
                        break;
      case EXHW0:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case EXHW1:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case EXHWS:       result = asm_exhws(instr, parm, 2);
                        break;
      case ILLEGAL_7F:  result = EMSYNTAX;
                        break;
 
      /* Opcodes 0x80 to 0x8F */
      case SLL0:        result = asm_arith_logic(instr, parm, 3);
                        break;
      case SLL1:        result = asm_arith_logic(instr, parm, 3);
                        break;
      case SRL0:        result = asm_arith_logic(instr, parm, 3);
                        break;
      case SRL1:        result = asm_arith_logic(instr, parm, 3);
                        break;
      case ILLEGAL_84:  result = EMSYNTAX;
                        break;
      case ILLEGAL_85:  result = EMSYNTAX;
                        break;
      case SRA0:        result = asm_arith_logic(instr, parm, 3);
                        break;
      case SRA1:        result = asm_arith_logic(instr, parm, 3);
                        break;
      case IRET:        
			result = asm_no_parms(instr, parm, 0);
                        break;
      case HALT_OP:     result = asm_no_parms(instr, parm, 0);
                        break;
      case ILLEGAL_8A:  result = EMSYNTAX;
                        break;
      case ILLEGAL_8B:  result = EMSYNTAX;
                        break;
      case IRETINV:     
			if (token_count > 1)
			    result = asm_one_parms(instr, parm, 1);
		        else
			    result = asm_no_parms(instr, parm, 0);
                        break;
      case ILLEGAL_8D:  result = EMSYNTAX;
                        break;
      case ILLEGAL_8E:  result = EMSYNTAX;
                        break;
      case ILLEGAL_8F:  result = EMSYNTAX;
                        break;
 
      /* Opcodes 0x90 to 0x9F */
      case AND_OP0:     result = asm_arith_logic(instr, parm, 3);
                        break;
      case AND_OP1:     result = asm_arith_logic(instr, parm, 3);
                        break;
      case OR_OP0:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case OR_OP1:      result = asm_arith_logic(instr, parm, 3);
                        break;
      case XOR_OP0:     result = asm_arith_logic(instr, parm, 3);
                        break;
      case XOR_OP1:     result = asm_arith_logic(instr, parm, 3);
                        break;
      case XNOR0:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case XNOR1:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case NOR0:        result = asm_arith_logic(instr, parm, 3);
                        break;
      case NOR1:        result = asm_arith_logic(instr, parm, 3);
                        break;
      case NAND0:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case NAND1:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case ANDN0:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case ANDN1:       result = asm_arith_logic(instr, parm, 3);
                        break;
      case SETIP:       result = asm_float(instr, parm, 3);
                        break;
      case INV:         
			  if (token_count > 1)
			    result = asm_one_parms(instr, parm, 1);
			  else
			    result = asm_no_parms(instr, parm, 0);
                        break;
 
      /* Opcodes 0xA0 to 0xAF */
      case JMP0:        result = asm_jmp(instr, parm, 1);
                        break;
      case JMP1:        result = asm_jmp(instr, parm, 1);
                        break;
      case ILLEGAL_A2:  result = EMSYNTAX;
                        break;
      case ILLEGAL_A3:  result = EMSYNTAX;
                        break;
      case JMPF0:       result = asm_call_jmp(instr, parm, 2);
                        break;
      case JMPF1:       result = asm_call_jmp(instr, parm, 2);
                        break;
      case ILLEGAL_A6:  result = EMSYNTAX;
                        break;
      case ILLEGAL_A7:  result = EMSYNTAX;
                        break;
      case CALL0:       result = asm_call_jmp(instr, parm, 2);
                        break;
      case CALL1:       result = asm_call_jmp(instr, parm, 2);
                        break;
      case ORN_OP0:  	result = EMSYNTAX;
                        break;
      case ORN_OP1:  	result = EMSYNTAX;
                        break;
      case JMPT0:       result = asm_call_jmp(instr, parm, 2);
                        break;
      case JMPT1:       result = asm_call_jmp(instr, parm, 2);
                        break;
      case ILLEGAL_AE:  result = EMSYNTAX;
                        break;
      case ILLEGAL_AF:  result = EMSYNTAX;
                        break;
 
      /* Opcodes 0xB0 to 0xBF */
      case ILLEGAL_B0:  result = EMSYNTAX;
                        break;
      case ILLEGAL_B1:  result = EMSYNTAX;
                        break;
      case ILLEGAL_B2:  result = EMSYNTAX;
                        break;
      case ILLEGAL_B3:  result = EMSYNTAX;
                        break;
      case JMPFDEC0:    result = asm_call_jmp(instr, parm, 2);
                        break;
      case JMPFDEC1:    result = asm_call_jmp(instr, parm, 2);
                        break;
      case MFTLB:       result = asm_mftlb(instr, parm, 2);
                        break;
      case ILLEGAL_B7:  result = EMSYNTAX;
                        break;
      case ILLEGAL_B8:  result = EMSYNTAX;
                        break;
      case ILLEGAL_B9:  result = EMSYNTAX;
                        break;
      case ILLEGAL_BA:  result = EMSYNTAX;
                        break;
      case ILLEGAL_BB:  result = EMSYNTAX;
                        break;
      case ILLEGAL_BC:  result = EMSYNTAX;
                        break;
      case ILLEGAL_BD:  result = EMSYNTAX;
                        break;
      case MTTLB:       result = asm_mttlb(instr, parm, 2);
                        break;
      case ILLEGAL_BF:  result = EMSYNTAX;
                        break;
 
      /* Opcodes 0xC0 to 0xCF */
      case JMPI:        result = asm_jmpi(instr, parm, 1);
                        break;
      case ILLEGAL_C1:  result = EMSYNTAX;
                        break;
      case ILLEGAL_C2:  result = EMSYNTAX;
                        break;
      case ILLEGAL_C3:  result = EMSYNTAX;
                        break;
      case JMPFI:       result = asm_calli_jmpi(instr, parm, 2);
                        break;
      case ILLEGAL_C5:  result = EMSYNTAX;
                        break;
      case MFSR:        result = asm_mfsr(instr, parm, 2);
                        break;
      case ILLEGAL_C7:  result = EMSYNTAX;
                        break;
      case CALLI:       result = asm_calli_jmpi(instr, parm, 2);
                        break;
      case ILLEGAL_C9:  result = EMSYNTAX;
                        break;
      case ILLEGAL_CA:  result = EMSYNTAX;
                        break;
      case ILLEGAL_CB:  result = EMSYNTAX;
                        break;
      case JMPTI:       result = asm_calli_jmpi(instr, parm, 2);
                        break;
      case ILLEGAL_CD:  result = EMSYNTAX;
                        break;
      case MTSR:        result = asm_mtsr(instr, parm, 2);
                        break;
      case ILLEGAL_CF:  result = EMSYNTAX;
                        break;
 
      /* Opcodes 0xD0 to 0xDF */
      case ILLEGAL_D0:  result = EMSYNTAX;
                        break;
      case ILLEGAL_D1:  result = EMSYNTAX;
                        break;
      case ILLEGAL_D2:  result = EMSYNTAX;
                        break;
      case ILLEGAL_D3:  result = EMSYNTAX;
                        break;
      case ILLEGAL_D4:  result = EMSYNTAX;
                        break;
      case ILLEGAL_D5:  result = EMSYNTAX;
                        break;
      case ILLEGAL_D6:  result = EMSYNTAX;
                        break;
      case EMULATE:     result = asm_emulate(instr, parm, 3);
                        break;
      case ILLEGAL_D8:  result = EMSYNTAX;
                        break;
      case ILLEGAL_D9:  result = EMSYNTAX;
                        break;
      case ILLEGAL_DA:  result = EMSYNTAX;
                        break;
      case ILLEGAL_DB:  result = EMSYNTAX;
                        break;
      case ILLEGAL_DC:  result = EMSYNTAX;
                        break;
      case ILLEGAL_DD:  result = EMSYNTAX;
                        break;
      case MULTM:       result = asm_float(instr, parm, 3);
                        break;
      case MULTMU:      result = asm_float(instr, parm, 3);
                        break;
 
      /* Opcodes 0xE0 to 0xEF */
      case MULTIPLY:    result = asm_float(instr, parm, 3);
                        break;
      case DIVIDE:      result = asm_float(instr, parm, 3);
                        break;
      case MULTIPLU:    result = asm_float(instr, parm, 3);
                        break;
      case DIVIDU:      result = asm_float(instr, parm, 3);
                        break;
      case CONVERT:     result = asm_convert(instr, parm, 6);
                        break;
      case SQRT:        result = asm_sqrt(instr, parm, 3);
                        break;
      case CLASS:       result = asm_class(instr, parm, 3);
                        break;
      case ILLEGAL_E7:  result = EMSYNTAX;
                        break;
      case ILLEGAL_E8:  result = EMSYNTAX;
                        break;
      case ILLEGAL_E9:  result = EMSYNTAX;
                        break;
      case FEQ:         result = asm_float(instr, parm, 3);
                        break;
      case DEQ:         result = asm_float(instr, parm, 3);
                        break;
      case FGT:         result = asm_float(instr, parm, 3);
                        break;
      case DGT:         result = asm_float(instr, parm, 3);
                        break;
      case FGE:         result = asm_float(instr, parm, 3);
                        break;
      case DGE:         result = asm_float(instr, parm, 3);
                        break;
 
      /* Opcodes 0xF0 to 0xFF */
      case FADD:        result = asm_float(instr, parm, 3);
                        break;
      case DADD:        result = asm_float(instr, parm, 3);
                        break;
      case FSUB:        result = asm_float(instr, parm, 3);
                        break;
      case DSUB:        result = asm_float(instr, parm, 3);
                        break;
      case FMUL:        result = asm_float(instr, parm, 3);
                        break;
      case DMUL:        result = asm_float(instr, parm, 3);
                        break;
      case FDIV:        result = asm_float(instr, parm, 3);
                        break;
      case DDIV:        result = asm_float(instr, parm, 3);
                        break;
      case ILLEGAL_F8:  result = EMSYNTAX;
                        break;
      case FDMUL:       result = asm_float(instr, parm, 3);
                        break;
      case ILLEGAL_FA:  result = EMSYNTAX;
                        break;
      case ILLEGAL_FB:  result = EMSYNTAX;
                        break;
      case ILLEGAL_FC:  result = EMSYNTAX;
                        break;
      case ILLEGAL_FD:  result = EMSYNTAX;
                        break;
      case ILLEGAL_FE:  result = EMSYNTAX;
                        break;
      case ILLEGAL_FF:  result = EMSYNTAX;
                        break;
      }  /* end switch */
 
   return (result);
 
   }  /* End asm_instr() */
 
 
 
 
/*
** The following functions are used to convert instruction
** parameters as an arrays of addr_29k_t memory space / address
** pairs into a 32 bit Am29000 binary instruction.
** All of the Am29000 instruction formats are supported below.
*/
 
 
/*
** Formats:   <nmemonic>, RC, RA, (RB or I)
** Examples:  ADD, OR, SLL, all arithmetic and
**            logic instructions
**
*/
 
int
asm_arith_logic(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 3)
      return (EMSYNTAX);
 
   if (ISGENERAL(parm[0].memory_space) &&
       ISGENERAL(parm[1].memory_space) &&
       ISGENERAL(parm[2].memory_space)) {
      /* Make sure M flag is cleared */
      instr->op = (BYTE) (instr->op & 0xfe);
      instr->c = (BYTE) (parm[0].address & 0xff);
      instr->a = (BYTE) (parm[1].address & 0xff);
      instr->b = (BYTE) (parm[2].address & 0xff);
      }
   else
   if (ISGENERAL(parm[0].memory_space) &&
       ISGENERAL(parm[1].memory_space) &&
       ISMEM(parm[2].memory_space)) {
      /* Make sure M flag is set */
      instr->op = (BYTE) (instr->op | 0x01);
      instr->c = (BYTE) (parm[0].address & 0xff);
      instr->a = (BYTE) (parm[1].address & 0xff);
      instr->b = (BYTE) (parm[2].address & 0xff);
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_arith_logic() */
 
 
 
/*
** Formats:   <nmemonic>, VN, RA, (RB or I)
** Examples:  ASSEQ, ASLE, ASLT, all trap assertion
**            instructions
**
*/
 
int
asm_vector(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 3)
      return (EMSYNTAX);
 
   if (ISMEM(parm[0].memory_space) &&
       ISGENERAL(parm[1].memory_space) &&
       ISGENERAL(parm[2].memory_space)) {
      /* Make sure M flag is cleared */
      instr->op = (BYTE) (instr->op & 0xfe);
      instr->c = (BYTE) (parm[0].address & 0xff);
      instr->a = (BYTE) (parm[1].address & 0xff);
      instr->b = (BYTE) (parm[2].address & 0xff);
      }
   else
   if (ISMEM(parm[0].memory_space) &&
       ISGENERAL(parm[1].memory_space) &&
       ISMEM(parm[2].memory_space)) {
      /* Make sure M flag is set */
      instr->op = (BYTE) (instr->op | 0x01);
      instr->c = (BYTE) (parm[0].address & 0xff);
      instr->a = (BYTE) (parm[1].address & 0xff);
      instr->b = (BYTE) (parm[2].address & 0xff);
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_vector() */
 
 
 
/*
** Formats:   <nmemonic>, CE, CNTL, RA, (RB or I)
** Examples:  LOAD, LOADM, STORE, all load and store
**            instructions
**
*/
 
int
asm_load_store(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   int  ce;
   int  cntl;
 
   if (parm_count != 4)
      return (EMSYNTAX);
 
   if (ISMEM(parm[0].memory_space) &&
       ISMEM(parm[1].memory_space) &&
       ISGENERAL(parm[2].memory_space) &&
       ISGENERAL(parm[3].memory_space)) {
      /* Make sure M flag is cleared */
      instr->op = (BYTE) (instr->op & 0xfe);
      if (parm[0].address > 1)
         return (EMSYNTAX);
      if (parm[1].address > 0x7f)
         return (EMSYNTAX);
      ce =   (int) ((parm[0].address << 7) & 0x80);
      cntl = (int)  (parm[1].address & 0x7f);
      instr->c = (BYTE) (ce | cntl);
      instr->a = (BYTE) (parm[2].address & 0xff);
      instr->b = (BYTE) (parm[3].address & 0xff);
      }
   else
   if (ISMEM(parm[0].memory_space) &&
       ISMEM(parm[1].memory_space) &&
       ISGENERAL(parm[2].memory_space) &&
       ISMEM(parm[3].memory_space)) {
      /* Make sure M flag is set */
      instr->op = (BYTE) (instr->op | 0x01);
      if (parm[0].address > 1)
         return (EMSYNTAX);
      if (parm[1].address > 0x7f)
         return (EMSYNTAX);
      if (parm[3].address > 0xff)
         return (EMSYNTAX);
      ce =   (int) ((parm[0].address << 7) & 0x80);
      cntl = (int)  (parm[1].address & 0x7f);
      instr->c = (BYTE) (ce | cntl);
      instr->a = (BYTE) (parm[2].address & 0xff);
      instr->b = (BYTE) (parm[3].address & 0xff);
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_load_store() */
 
 
 
/*
** Formats:   <nmemonic>
** Examples:  HALT, INV, IRET
*/
/*ARGSUSED*/
 
int
asm_no_parms(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 0)
      return (EMSYNTAX);
 
   /* Put zeros in the "reserved" fields */
   instr->c = 0;
   instr->a = 0;
   instr->b = 0;
 
   return (0);
   }  /* end asm_no_parms() */
 
 
int
asm_one_parms(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 1)
      return (EMSYNTAX);
 
   instr->c = (BYTE) (parm[0].address & 0x3);
 
   /* Put zeros in the "reserved" fields */
   instr->a = 0;
   instr->b = 0;
 
   return (0);
   } /* end asm_one_parms */
 
 
/*
** Formats:   <nmemonic>, RC, RA, RB
** Examples:  DADD, FADD, all floating point
**            instructions
**
*/
 
int
asm_float(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 3)
      return (EMSYNTAX);
 
   if (ISGENERAL(parm[0].memory_space) &&
       ISGENERAL(parm[1].memory_space) &&
       ISGENERAL(parm[2].memory_space)) {
      instr->c = (BYTE) (parm[0].address & 0xff);
      instr->a = (BYTE) (parm[1].address & 0xff);
      instr->b = (BYTE) (parm[2].address & 0xff);
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_float() */
 
 
 
/*
** Formats:   <nmemonic> RA, <target>
** Examples:  CALL, JMPF, JMPFDEC, JMPT
**
** Note:  This function is used only with the CALL,
**        JMPF, JMPFDEC and JMPT operations.
*/
 
int
asm_call_jmp(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 2)
      return (EMSYNTAX);
 
   if (ISGENERAL(parm[0].memory_space) &&
       ISMEM(parm[1].memory_space)) {
      /* Make sure M flag is set */
      if (parm[1].memory_space != PC_RELATIVE)
         instr->op = (BYTE) (instr->op | 0x01);
      else
         instr->op = (BYTE) instr->op ;
      instr->c = (BYTE) ((parm[1].address >> 10) & 0xff);
      instr->a = (BYTE) (parm[0].address & 0xff);
      instr->b = (BYTE) ((parm[1].address >> 2) & 0xff);
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_call_jmp() */
 
 
 
 
/*
** Formats:   <nmemonic> RA, RB
** Examples:  CALLI, JMPFI, JMPTI
**
** Note:  This function is used only with the CALLI,
**        JMPFI and JMPTI (but not JMPI) operations.
*/
 
int
asm_calli_jmpi(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 2)
      return (EMSYNTAX);
 
   if (ISGENERAL(parm[0].memory_space) &&
       ISREG(parm[1].memory_space)) {
      instr->c = 0;
      instr->a = (BYTE) (parm[0].address & 0xff);
      instr->b = (BYTE) (parm[1].address & 0xff);
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_calli_jmpi() */
 
 
 
/*
** Formats:   <nmemonic> RC, RB, FS
** Examples:  CLASS
**
** Note:  This function is used only with the CLASS
**        operation.
*/
 
int
asm_class(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 3)
      return (EMSYNTAX);
 
   if (ISGENERAL(parm[0].memory_space) &&
       ISGENERAL(parm[1].memory_space) &&
       ISMEM(parm[2].memory_space)) {
      if (parm[2].address > 0x03)
         return (EMSYNTAX);
      instr->c = (BYTE) (parm[0].address & 0xff);
      instr->a = (BYTE) (parm[1].address & 0xff);
      instr->b = (BYTE) (parm[2].address & 0x03);
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_class() */
 
 
/*
** Formats:   <nmemonic> RC, (RB or I)
** Examples:  CLZ
**
** Note:  This function is used only with the CLZ
**        operation.
*/
 
int
asm_clz(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 2)
      return (EMSYNTAX);
 
   if (ISGENERAL(parm[0].memory_space) &&
       ISGENERAL(parm[1].memory_space)) {
      /* Make sure M flag is cleared */
      instr->op = (BYTE) (instr->op & 0xfe);
      instr->c = (BYTE) (parm[0].address & 0xff);
      instr->a = 0;
      instr->b = (BYTE) (parm[1].address & 0xff);
      }
   else
   if (ISGENERAL(parm[0].memory_space) &&
       ISMEM(parm[1].memory_space)) {
      /* Check param1 */
      if ((parm[1].address) > 0xff)
         return(EMSYNTAX);
      /* Make sure M flag is set */
      instr->op = (BYTE) (instr->op | 0x01);
      instr->c = (BYTE) (parm[0].address & 0xff);
      instr->a = 0;
      instr->b = (BYTE) (parm[1].address & 0xff);
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_clz() */
 
 
 
/*
** Formats:   <nmemonic> RA, <const16>
** Examples:  CONST, CONSTN
**
*/
 
int
asm_const(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 2)
      return (EMSYNTAX);
 
   if (ISGENERAL(parm[0].memory_space) &&
       ISMEM(parm[1].memory_space)) {
      instr->c = (BYTE) ((parm[1].address >> 8) & 0xff);
      instr->a = (BYTE) (parm[0].address & 0xff);
      instr->b = (BYTE) (parm[1].address & 0xff);
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_const() */
 
 
 
/*
** Formats:   <nmemonic> RA, <const16>
** Examples:  CONSTH
**
*/
 
int
asm_consth(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 2)
      return (EMSYNTAX);
 
   if (ISGENERAL(parm[0].memory_space) &&
       ISMEM(parm[1].memory_space)) {
      instr->c = (BYTE) ((parm[1].address >> 24) & 0xff);
      instr->a = (BYTE) (parm[0].address & 0xff);
      instr->b = (BYTE) ((parm[1].address >> 16) & 0xff);
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_consth() */
 
 
 
/*
** Formats:   <nmemonic> RC, RA, UI, RND, FD, FS
** Examples:  CONVERT
**
** Note:  This function is used only with the CONVERT
**        operation.
**
** Note:  Some assembler examples show this operation with
**        only five parameters.  It should have six.
*/
 
int
asm_convert(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   BYTE  ui;
   BYTE  rnd;
   BYTE  fd;
   BYTE  fs;
 
   if (parm_count != 6)
      return (EMSYNTAX);
 
   if (ISGENERAL(parm[0].memory_space) &&
       ISGENERAL(parm[1].memory_space) &&
       ISMEM(parm[2].memory_space) &&
       ISMEM(parm[3].memory_space) &&
       ISMEM(parm[4].memory_space) &&
       ISMEM(parm[5].memory_space)) {
      if (parm[2].address > 1)
         return (EMSYNTAX);
      if (parm[3].address > 0x07)
         return (EMSYNTAX);
      if (parm[4].address > 0x03)
         return (EMSYNTAX);
      if (parm[5].address > 0x03)
         return (EMSYNTAX);
      ui =  (BYTE) ((parm[2].address << 7) & 0x80);
      rnd = (BYTE) ((parm[3].address << 4) & 0x70);
      fd =  (BYTE) ((parm[4].address << 2) & 0x0c);
      fs =  (BYTE) (parm[5].address & 0x03);
      instr->c = (BYTE) (parm[0].address & 0xff);
      instr->a = (BYTE) (parm[1].address & 0xff);
      instr->b = (ui | rnd | fd | fs);
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_convert() */
 
 
 
/*
** Formats:   <nmemonic> RC, RA
** Examples:  DIV0
**
** Note:  This function is used only with the DIV0
**        operation.
*/
 
int
asm_div0(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 2)
      return (EMSYNTAX);
 
   if (ISGENERAL(parm[0].memory_space) &&
       ISGENERAL(parm[1].memory_space)) {
      /* Make sure M flag is cleared */
      instr->op = (BYTE) (instr->op & 0xfe);
      instr->c = (BYTE) (parm[0].address & 0xff);
      instr->a = 0;
      instr->b = (BYTE) (parm[1].address & 0xff);
      }
   else
   if (ISGENERAL(parm[0].memory_space) &&
       ISMEM(parm[1].memory_space)) {
      /* Check immediate value */
      if (parm[1].address > 0xff)
         return (EMSYNTAX);
      /* Make sure M flag is set */
      instr->op = (BYTE) (instr->op | 0x01);
      instr->c = (BYTE) (parm[0].address & 0xff);
      instr->a = 0;
      instr->b = (BYTE) (parm[1].address & 0xff);
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_div0() */
 
 
/*
** Formats:   <nmemonic> RC, RA
** Examples:  EXHWS
**
** Note:  This function is used only with the EXHWS
**        operation.
*/
 
int
asm_exhws(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 2)
      return (EMSYNTAX);
 
   if (ISGENERAL(parm[0].memory_space) &&
       ISGENERAL(parm[1].memory_space)){
      instr->c = (BYTE) (parm[0].address & 0xff);
      instr->a = (BYTE) (parm[1].address & 0xff);
      instr->b = 0;
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_exhws() */
 
 
 
 
/*
** Formats:   <nmemonic>  <target>
** Examples:  JMP
**
** Note:  This function is used only with the JMP
**        operation.
**
** Note:  This function will only do absolute jumps.
*/
 
int
asm_jmp(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 1)
      return (EMSYNTAX);
 
   if (ISMEM(parm[0].memory_space)) {
      /* Make sure M flag is set */
      if (parm[0].memory_space != PC_RELATIVE)
        instr->op = (BYTE) (instr->op | 0x01);
      else
        instr->op = (BYTE) instr->op ;
      instr->c = (BYTE) ((parm[0].address >> 10) & 0xff);
      instr->a = 0;
      instr->b = (BYTE) ((parm[0].address >> 2) & 0xff);
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_jmp() */
 
 
 
/*
** Formats:   <nmemonic> RB
** Examples:  JMPI
**
** Note:  This function is used only with the JMPI
**        operation.
*/
 
int
asm_jmpi(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 1)
      return (EMSYNTAX);
 
   if (ISGENERAL(parm[0].memory_space)) {
      instr->c = 0;
      instr->a = 0;
      instr->b = (BYTE) (parm[0].address & 0xff);
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_jmpi() */
 
 
 
/*
** Formats:   <nmemonic> RC, SA
** Examples:  MFSR
**
** Note:  This function is used only with the MFSR
**        operation.
*/
 
int
asm_mfsr(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 2)
      return (EMSYNTAX);
 
   if (ISGENERAL(parm[0].memory_space) &&
       ISSPECIAL(parm[1].memory_space)) {
      instr->c = (BYTE) (parm[0].address & 0xff);
      instr->a = (BYTE) (parm[1].address & 0xff);
      instr->b = 0;
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_mfsr() */
 
 
 
/*
** Formats:   <nmemonic> SA, RB
** Examples:  MTSR
**
** Note:  This function is used only with the MTSR
**        operation.
*/
 
int
asm_mtsr(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 2)
      return (EMSYNTAX);
 
   if (ISSPECIAL(parm[0].memory_space) &&
       ISGENERAL(parm[1].memory_space)) {
      instr->c = 0;
      instr->a = (BYTE) (parm[0].address & 0xff);
      instr->b = (BYTE) (parm[1].address & 0xff);
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_mtsr() */
 
 
 
 
/*
** Formats:   <nmemonic> SA, <const16>
** Examples:  MTSRIM
**
** Note:  This function is used only with the MTSRIM
**        operation.
*/
 
int
asm_mtsrim(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 2)
      return (EMSYNTAX);
 
   if (ISSPECIAL(parm[0].memory_space) &&
       ISMEM(parm[1].memory_space)) {
      instr->c = (BYTE) ((parm[1].address >> 8) & 0xff);
      instr->a = (BYTE) (parm[0].address & 0xff);
      instr->b = (BYTE) (parm[1].address & 0xff);
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_mtsrim() */
 
 
 
 
/*
** Formats:   <nmemonic> RC, RA
** Examples:  MFTLB
**
** Note:  This function is used only with the MFTLB
**        operation.
*/
 
int
asm_mftlb(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 2)
      return (EMSYNTAX);
 
   if (ISGENERAL(parm[0].memory_space) &&
       ISGENERAL(parm[1].memory_space)) {
      instr->c = (BYTE) (parm[0].address & 0xff);
      instr->a = (BYTE) (parm[1].address & 0xff);
      instr->b = 0;
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_mftlb() */
 
 
/*
** Formats:   <nmemonic> RA, RB
** Examples:  MTTLB
**
** Note:  This function is used only with the MTTLB
**        operation.
*/
 
int
asm_mttlb(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 2)
      return (EMSYNTAX);
 
   if (ISGENERAL(parm[0].memory_space) &&
       ISGENERAL(parm[1].memory_space)) {
      instr->c = 0;
      instr->a = (BYTE) (parm[0].address & 0xff);
      instr->b = (BYTE) (parm[1].address & 0xff);
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_mttlb() */
 
 
 
 
/*
** Formats:   <nmemonic> RC, RA, FS
** Examples:  SQRT
**
** Note:  This function is used only with the SQRT
**        operation.
*/
 
int
asm_sqrt(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 3)
      return (EMSYNTAX);
 
   if (ISGENERAL(parm[0].memory_space) &&
       ISGENERAL(parm[1].memory_space) &&
       ISMEM(parm[2].memory_space)) {
      instr->c = (BYTE) (parm[0].address & 0xff);
      instr->a = (BYTE) (parm[1].address & 0xff);
      instr->b = (BYTE) (parm[2].address & 0x03);
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_sqrt() */
 
 
 
/*
** Formats:   <nmemonic>, VN, RA, RB
** Examples:  EMULATE
**
** Note:  This function is used only with the EMULATE
**        operation.
**
*/
 
int
asm_emulate(instr, parm, parm_count)
   struct   instr_t    *instr;
   struct   addr_29k_t *parm;
   int      parm_count;
   {
   if (parm_count != 3)
      return (EMSYNTAX);
 
   if (ISMEM(parm[0].memory_space) &&
       ISGENERAL(parm[1].memory_space) &&
       ISGENERAL(parm[2].memory_space)) {
      instr->c = (BYTE) (parm[0].address & 0xff);
      instr->a = (BYTE) (parm[1].address & 0xff);
      instr->b = (BYTE) (parm[2].address & 0xff);
      }
   else
      return(EMSYNTAX);
 
   return (0);
   }  /* end asm_emulate() */
 
 
 
 
 
 

Go to most recent revision | Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2025 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.