ref: 65121e6d5d2b9cb15283a0d5c46addeb67d20a7b
dir: /src/asm/charmap.c/
/*
 * This file is part of RGBDS.
 *
 * Copyright (c) 2013-2018, stag019 and RGBDS contributors.
 *
 * SPDX-License-Identifier: MIT
 */
#include <errno.h>
#include <stdbool.h>
#include <stdint.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include "asm/asm.h"
#include "asm/charmap.h"
#include "asm/main.h"
#include "asm/output.h"
#include "asm/util.h"
#include "asm/warning.h"
#include "hashmap.h"
#define CHARMAP_HASH_SIZE (1 << 9)
struct CharmapStackEntry {
	struct Charmap *charmap;
	struct CharmapStackEntry *next;
};
static HashMap charmaps;
static struct Charmap *mainCharmap;
static struct Charmap *currentCharmap;
struct CharmapStackEntry *charmapStack;
static inline struct Charmap *charmap_Get(const char *name)
{
	return hash_GetElement(charmaps, name);
}
static void CopyNode(struct Charmap *dest,
		     const struct Charmap *src,
		     int nodeIdx)
{
	dest->nodes[nodeIdx].code = src->nodes[nodeIdx].code;
	dest->nodes[nodeIdx].isCode = src->nodes[nodeIdx].isCode;
	for (int i = 0; i < 256; i++)
		if (src->nodes[nodeIdx].next[i])
			dest->nodes[nodeIdx].next[i] = dest->nodes +
				(src->nodes[nodeIdx].next[i] - src->nodes);
}
struct Charmap *charmap_New(const char *name, const char *baseName)
{
	struct Charmap *base = NULL;
	if (baseName != NULL) {
		base = charmap_Get(baseName);
		if (base == NULL)
			yyerror("Base charmap '%s' doesn't exist", baseName);
	}
	struct Charmap *charmap = charmap_Get(name);
	if (charmap != NULL) {
		yyerror("Charmap '%s' already exists", name);
		return NULL;
	}
	charmap = malloc(sizeof(*charmap));
	if (charmap == NULL)
		fatalerror("Failed to create charmap: %s", strerror(errno));
	/* Init the new charmap's fields */
	snprintf(charmap->name, sizeof(charmap->name), "%s", name);
	if (base != NULL) {
		charmap->charCount = base->charCount;
		charmap->nodeCount = base->nodeCount;
		for (int i = 0; i < MAXCHARNODES; i++)
			CopyNode(charmap, base, i);
	} else {
		charmap->charCount = 0;
		charmap->nodeCount = 0;
		memset(charmap->nodes, 0, sizeof(charmap->nodes));
	}
	hash_AddElement(charmaps, charmap->name, charmap);
	currentCharmap = charmap;
	return charmap;
}
void charmap_Set(const char *name)
{
	struct Charmap *charmap = charmap_Get(name);
	if (charmap == NULL)
		yyerror("Charmap '%s' doesn't exist", name);
	else
		currentCharmap = charmap;
}
void charmap_Push(void)
{
	struct CharmapStackEntry *stackEntry;
	stackEntry = malloc(sizeof(struct CharmapStackEntry));
	if (stackEntry == NULL)
		fatalerror("No memory for charmap stack");
	stackEntry->charmap = currentCharmap;
	stackEntry->next = charmapStack;
	charmapStack = stackEntry;
}
void charmap_Pop(void)
{
	if (charmapStack == NULL)
		fatalerror("No entries in the charmap stack");
	struct CharmapStackEntry *top = charmapStack;
	currentCharmap = top->charmap;
	charmapStack = top->next;
	free(top);
}
void charmap_InitMain(void)
{
	mainCharmap = charmap_New("main", NULL);
}
int32_t charmap_Add(char *input, uint8_t output)
{
	int32_t i;
	uint8_t v;
	struct Charmap  *charmap = currentCharmap;
	struct Charnode *curr_node, *temp_node;
	if (charmap->charCount >= MAXCHARMAPS || strlen(input) > CHARMAPLENGTH)
		return -1;
	curr_node = &charmap->nodes[0];
	for (i = 0; (v = (uint8_t)input[i]); i++) {
		if (curr_node->next[v]) {
			curr_node = curr_node->next[v];
		} else {
			temp_node = &charmap->nodes[charmap->nodeCount + 1];
			curr_node->next[v] = temp_node;
			curr_node = temp_node;
			++charmap->nodeCount;
		}
	}
	/* prevent duplicated keys by accepting only first key-value pair.  */
	if (curr_node->isCode)
		return charmap->charCount;
	curr_node->code = output;
	curr_node->isCode = 1;
	return ++charmap->charCount;
}
int32_t charmap_Convert(char **input)
{
	struct Charmap  *charmap = currentCharmap;
	struct Charnode *charnode;
	char *output;
	char outchar[8];
	int32_t i, match, length;
	uint8_t v, foundCode;
	output = malloc(strlen(*input));
	if (output == NULL)
		fatalerror("Not enough memory for buffer");
	length = 0;
	while (**input) {
		charnode = &charmap->nodes[0];
		/*
		 * Find the longest valid match which has been registered in
		 * charmap, possibly yielding multiple or no matches.
		 * The longest match is taken, meaning partial matches shorter
		 * than the longest one are ignored.
		 */
		for (i = match = 0; (v = (*input)[i]);) {
			if (!charnode->next[v])
				break;
			charnode = charnode->next[v];
			i++;
			if (charnode->isCode) {
				match = i;
				foundCode = charnode->code;
			}
		}
		if (match) {
			output[length] = foundCode;
			length++;
		} else {
			/*
			 * put a utf-8 character
			 * if failed to find a match.
			 */
			match = readUTF8Char(outchar, *input);
			if (match) {
				memcpy(output + length, *input, match);
			} else {
				output[length] = 0;
				match = 1;
			}
			length += match;
		}
		*input += match;
	}
	*input = output;
	return length;
}