0
0
mirror of https://github.com/netwide-assembler/nasm.git synced 2025-10-10 00:25:06 -04:00
Files
nasm/asm/exprlib.c
H. Peter Anvin 23ce05f906 treewide: replace verbose copyright headers with SPDX tags
SPDX is an international standard for documenting software license
requirements. Remove the existing headers and replace with a brief
SPDX preamble.

See: https://spdx.dev/use/specifications/

The script used to convert the files is added to "tools", and the
file header templates in headers/ are updated.

Signed-off-by: H. Peter Anvin (Intel) <hpa@zytor.com>
2025-10-01 11:45:31 -07:00

181 lines
4.4 KiB
C

/* SPDX-License-Identifier: BSD-2-Clause */
/* Copyright 1996-2017 The NASM Authors - All Rights Reserved */
/*
* exprlib.c
*
* Library routines to manipulate expression data types.
*/
#include "nasm.h"
/*
* Return true if the argument is a simple scalar. (Or a far-
* absolute, which counts.)
*/
bool is_simple(const expr *vect)
{
while (vect->type && !vect->value)
vect++;
if (!vect->type)
return true;
if (vect->type != EXPR_SIMPLE)
return false;
do {
vect++;
} while (vect->type && !vect->value);
if (vect->type && vect->type < EXPR_SEGBASE + SEG_ABS)
return false;
return true;
}
/*
* Return true if the argument is a simple scalar, _NOT_ a far-
* absolute.
*/
bool is_really_simple(const expr *vect)
{
while (vect->type && !vect->value)
vect++;
if (!vect->type)
return true;
if (vect->type != EXPR_SIMPLE)
return false;
do {
vect++;
} while (vect->type && !vect->value);
if (vect->type)
return false;
return true;
}
/*
* Classify an expression based on its components
*/
enum expr_classes expr_class(const expr *vect)
{
enum expr_classes class = EC_ZERO;
for (; vect->type; vect++) {
if (!vect->value) {
/* Value-0 term */
} else if (vect->type < EXPR_UNKNOWN) {
if ((class & EC_REGISTER) || vect->value != 1)
class |= EC_REGEXPR;
else
class |= EC_REGISTER;
} else if (vect->type == EXPR_UNKNOWN) {
class |= EC_UNKNOWN;
} else if (vect->type == EXPR_SIMPLE) {
/* Pure number term */
class |= EC_CONST;
} else if (vect->type == EXPR_WRT) {
class |= EC_WRT;
} else if (vect->type < EXPR_SEGBASE) {
class |= EC_COMPLEX;
} else if (vect->type >= EXPR_SEGBASE + SEG_ABS) {
/* It is an absolute segment */
if (class & (EC_SEG|EC_SEGABS))
class |= EC_COMPLEX;
class |= EC_SEGABS;
} else {
/* It is a segment */
if (vect->value == 1) {
if (class & (EC_SEG|EC_SEGABS))
class |= EC_COMPLEX;
class |= EC_SEG;
} else if (vect->value == -1) {
/* can only subtract current segment, and only once */
if (vect->type != location.segment + EXPR_SEGBASE ||
(class & EC_SELFREL))
class |= EC_COMPLEX;
class |= EC_SELFREL;
} else {
/* Non-simple segment arithmetic */
class |= EC_COMPLEX;
}
}
}
return class;
}
/*
* Return true if the argument is relocatable (i.e. a simple
* scalar, plus at most one segment-base, possibly a subtraction
* of the current segment base, plus possibly a WRT).
*/
bool is_reloc(const expr *vect)
{
return !(expr_class(vect) & ~EC_RELOC);
}
/*
* Return true if the argument contains an `unknown' part.
*/
bool is_unknown(const expr *vect)
{
return !!(expr_class(vect) & EC_UNKNOWN);
}
/*
* Return true if the argument contains nothing but an `unknown'
* part.
*/
bool is_just_unknown(const expr *vect)
{
return expr_class(vect) == EC_UNKNOWN;
}
/*
* Return the scalar part of a relocatable vector. (Including
* simple scalar vectors - those qualify as relocatable.)
*/
int64_t reloc_value(const expr *vect)
{
while (vect->type && !vect->value)
vect++;
if (!vect->type)
return 0;
if (vect->type == EXPR_SIMPLE)
return vect->value;
else
return 0;
}
/*
* Return the segment number of a relocatable vector, or NO_SEG for
* simple scalars.
*/
int32_t reloc_seg(const expr *vect)
{
for (; vect->type; vect++) {
if (vect->type >= EXPR_SEGBASE && vect->value == 1)
return vect->type - EXPR_SEGBASE;
}
return NO_SEG;
}
/*
* Return the WRT segment number of a relocatable vector, or NO_SEG
* if no WRT part is present.
*/
int32_t reloc_wrt(const expr *vect)
{
while (vect->type && vect->type < EXPR_WRT)
vect++;
if (vect->type == EXPR_WRT) {
return vect->value;
} else
return NO_SEG;
}
/*
* Return true if this expression contains a subtraction of the location
*/
bool is_self_relative(const expr *vect)
{
return !!(expr_class(vect) & EC_SELFREL);
}