shithub: scc

ref: ed874de117ff9d2eea35b69cd6a2fcb61f521c1a
dir: /as/emit.c/

View raw version

#include <stdio.h>
#include <stdint.h>
#include <string.h>

#include "../inc/scc.h"
#include "as.h"

#define HASHSIZ 64

static Section abss = {
	.name = "abs",
	.flags = SREAD|SWRITE
};

static Section bss = {
	.name = "bss",
	.flags = SRELOC|SREAD|SWRITE,
	.next = &abss
};

static Section data = {
	.name = "data",
	.next = &bss,
	.flags = SRELOC|SREAD|SWRITE|SFILE
};

static Section text = {
	.name = "text",
	.next = &data,
	.flags = SRELOC|SFILE
};

Section *cursec = &text, *headp = &text;

int pass;

static Symbol *hashtbl[HASHSIZ];

Symbol *
lookup(char *name)
{
	unsigned h;
	Symbol *sym;
	int c;
	char *t, *s;

	s = name;
	for (h = 0; c = *s; ++s)
		h = h*33 ^ c;
	h &= HASHSIZ-1;

	c = *name;
	for (sym = hashtbl[h]; sym; sym = sym->next) {
		t = sym->name;
		if (c == *t && !strcmp(t, name))
			return sym;
	}

	sym = xmalloc(sizeof(sym));
	sym->name = xstrdup(name);
	sym->type = FUNDEF;
	sym->desc = 0;
	sym->value = 0;
	sym->next = hashtbl[h];
	hashtbl[h] = sym;

	return sym;
}

char *
pack(TUINT v, int n, int inc)
{
	static char buf[sizeof(TUINT)];
	int idx;

	idx = (inc < 0) ? n-1 : 0;
	while (n--) {
		buf[idx] = v;
		idx += inc;
		v >>= 8;
	}

	if (v)
		error("overflow in immediate value");
	return buf;
}

static void
isect(Section *sec)
{
	TUINT siz;

	sec->curpc = sec->pc = sec->base;
	if (pass == 1 || !(sec->flags & SFILE))
		return;

	siz = sec->max - sec->base;
	if (siz > SIZE_MAX)
		die("out of memory");
	sec->mem = xmalloc(sec->max - sec->base);
}

Section *
section(char *name)
{
	Section *sec;

	for (sec = headp; sec; sec = sec->next) {
		if (!strcmp(sec->name, name))
			break;
	}
	if (!sec) {
		sec = xmalloc(sizeof(*sec));
		sec->name = xstrdup(name);
		sec->base = sec->max = sec->pc = sec->curpc = 0;
		sec->next = headp;
		sec->flags = SRELOC|SREAD|SWRITE|SFILE;
	}
	cursec = sec;
}

void
isections(void)
{
	Section *sec;

	for (sec = headp; sec; sec = sec->next)
		isect(sec);
}

void
emit(Section *sec, char *bytes, int n)
{
	if (sec->mem)
		memcpy(&sec->mem[sec->pc - sec->base], bytes, n);
	incpc(n);
}

void
writeout(char *name)
{
	FILE *fp;
	Section *secp;

	if ((fp = fopen(name, "wb")) == NULL)
		die("error opening output file '%s'\n", name);

	for (secp = headp; secp; secp = secp->next) {
		if (!secp->mem)
			continue;
		fwrite(secp->mem, secp->max - secp->base, 1, fp);
	}

	if (fclose(fp))
		die("error writing the output file");
}