Skip to content
Snippets Groups Projects
asn1fix_dereft.c 1.16 KiB
Newer Older
Lev Walkin's avatar
Lev Walkin committed
#include "asn1fix_internal.h"

int
asn1f_fix_dereference_types(arg_t *arg) {
	asn1p_expr_t *expr = arg->expr;
	asn1p_expr_t *type_expr;
	int r_value = 0;

	if(expr->expr_type != A1TC_REFERENCE
	|| expr->meta_type != AMT_TYPEREF) {
		//assert(expr->reference == 0);
		return 0;	/* Just ignore it */
	}

Lev Walkin's avatar
Lev Walkin committed
	DEBUG("(\"%s\":%x ::= \"%s\") for line %d",
		expr->Identifier, expr->expr_type,
Lev Walkin's avatar
Lev Walkin committed
		asn1f_printable_value(expr->value),
		expr->_lineno);

	assert(TQ_FIRST(&(expr->members)) == 0);
	assert(expr->reference);

	/*
	 * Follow the reference.
	 */
Lev Walkin's avatar
Lev Walkin committed
	type_expr = asn1f_find_terminal_type(arg, expr);
Lev Walkin's avatar
Lev Walkin committed
	if(type_expr == NULL) {
Lev Walkin's avatar
Lev Walkin committed
		const char *type_name;
Lev Walkin's avatar
Lev Walkin committed
		asn1p_expr_t *idexpr;
Lev Walkin's avatar
Lev Walkin committed

		if(errno == EEXIST) {
			/* Ignore missing type
			 * if known to be defined externally:
			 * -fknown-extern-type=<name>
			 */
			return 0;
		}

		type_name = asn1f_printable_reference(expr->reference);
Lev Walkin's avatar
Lev Walkin committed
		/* Avoid NULL in case of unnamed T ::= SEQUENCE OF ... */
		for(idexpr = expr; !idexpr->Identifier && idexpr->parent_expr;
			idexpr = idexpr->parent_expr);
Lev Walkin's avatar
Lev Walkin committed
		FATAL("Unknown type \"%s\" referenced by \"%s\" at line %d",
Lev Walkin's avatar
Lev Walkin committed
			type_name, idexpr->Identifier, expr->_lineno);
Lev Walkin's avatar
Lev Walkin committed
		return -1;
	}

	return r_value;
}