diff lwasm/pass4.c @ 365:6a98cc90c14f

Added resolve passes
author lost@starbug
date Wed, 14 Apr 2010 20:49:04 -0600
parents
children 84dc6d2ec6ba
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/lwasm/pass4.c	Wed Apr 14 20:49:04 2010 -0600
@@ -0,0 +1,117 @@
+/*
+pass4.c
+
+Copyright © 2010 William Astle
+
+This file is part of LWTOOLS.
+
+LWTOOLS is free software: you can redistribute it and/or modify it under the
+terms of the GNU General Public License as published by the Free Software
+Foundation, either version 3 of the License, or (at your option) any later
+version.
+
+This program is distributed in the hope that it will be useful, but WITHOUT
+ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
+FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
+more details.
+
+You should have received a copy of the GNU General Public License along with
+this program. If not, see <http://www.gnu.org/licenses/>.
+*/
+
+#include <config.h>
+
+#include <stdio.h>
+#include <string.h>
+
+#include <lw_alloc.h>
+#include <lw_string.h>
+
+#include "lwasm.h"
+#include "instab.h"
+
+/*
+Resolve2 Pass
+
+Force resolution of instruction sizes.
+
+*/
+void do_pass4(asmstate_t *as)
+{
+	int rc;
+	int cnt;
+	line_t *cl, *sl;
+	struct line_expr_s *le;
+
+	// first, count the number of unresolved instructions
+	for (cnt = 0, cl = as -> line_head; cl; cl = cl -> next)
+	{
+		if (cl -> len == -1)
+			cnt++;
+	}
+
+	sl = as -> line_head;
+	while (cnt > 0)
+	{
+		// find an unresolved instruction
+		for ( ; sl && sl -> len != -1; sl = sl -> next)
+			/* do nothing */ ;
+		
+		// simplify address
+		as -> cl = sl;
+		lwasm_reduce_expr(as, sl -> addr);
+		
+		// simplify each expression
+		for (le = sl -> exprs; le; le = le -> next)
+			lwasm_reduce_expr(as, le -> expr);
+
+
+		if (sl -> insn >= 0 && instab[sl -> insn].resolve)
+		{
+			(instab[cl -> insn].resolve)(as, sl, 1);
+			if (sl -> len == -1)
+			{
+				lwasm_register_error(as, sl, "Instruction failed to resolve.");
+				return;
+			}
+		}
+		cnt--;
+		if (cnt == 0)
+			return;
+
+		do
+		{
+			rc = 0;
+			for (cl = sl; cl; cl = cl -> next)
+			{
+				as -> cl = cl;
+			
+				// simplify address
+				lwasm_reduce_expr(as, cl -> addr);
+		
+				// simplify each expression
+				for (le = cl -> exprs; le; le = le -> next)
+					lwasm_reduce_expr(as, le -> expr);
+			
+				if (cl -> len == -1)
+				{
+					// try resolving the instruction length
+					// but don't force resolution
+					if (cl -> insn >= 0 && instab[cl -> insn].resolve)
+					{
+						(instab[cl -> insn].resolve)(as, cl, 0);
+						if (cl -> len != -1)
+						{
+							rc++;
+							cnt--;
+							if (cnt == 0)
+								return;
+						}
+					}
+				}
+			}
+			if (as -> errorcount > 0)
+				return;
+		} while (rc > 0);
+	}
+}