0
0
mirror of https://github.com/netwide-assembler/nasm.git synced 2025-07-24 10:25:42 -04:00
nasm/asm/exprlib.c
H. Peter Anvin 164d24677a Support self-relative expressions in offsets and immediates
Handle, hopefully correctly, self-relative expressions (that is,
expressions of the form X - Y where Y is a symbol in the current
segment, possibly $ or $$) used as offsets or immediates, as opposed
to arguments to Dx statements (which have already been supported for a
while.)

Signed-off-by: H. Peter Anvin <hpa@zytor.com>
2017-02-20 02:39:56 -08:00

243 lines
6.6 KiB
C

/* ----------------------------------------------------------------------- *
*
* Copyright 1996-2017 The NASM Authors - All Rights Reserved
* See the file AUTHORS included with the NASM distribution for
* the specific copyright holders.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following
* conditions are met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following
* disclaimer in the documentation and/or other materials provided
* with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
* CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
* INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
* ----------------------------------------------------------------------- */
/*
* 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;
}
/*
* 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)
{
bool has_rel = false; /* Has a self-segment-subtract */
bool has_seg = false; /* Has a segment base */
for (; vect->type; vect++) {
if (!vect->value) {
/* skip value-0 terms */
continue;
} else if (vect->type < EXPR_SIMPLE) {
/* false if a register is present */
return false;
} else if (vect->type == EXPR_SIMPLE) {
/* skip over a pure number term... */
continue;
} else if (vect->type == EXPR_WRT) {
/* skip over a WRT term... */
continue;
} else if (vect->type < EXPR_SEGBASE) {
/* other special type -> problem */
return false;
} else if (vect->value == 1) {
if (has_seg)
return false; /* only one segbase allowed */
has_seg = true;
} else if (vect->value == -1) {
if (vect->type != location.segment + EXPR_SEGBASE)
return false; /* can only subtract current segment */
if (has_rel)
return false; /* already is relative */
has_rel = true;
}
}
return true;
}
/*
* Return true if the argument contains an `unknown' part.
*/
bool is_unknown(const expr *vect)
{
while (vect->type && vect->type < EXPR_UNKNOWN)
vect++;
return (vect->type == EXPR_UNKNOWN);
}
/*
* Return true if the argument contains nothing but an `unknown'
* part.
*/
bool is_just_unknown(const expr *vect)
{
while (vect->type && !vect->value)
vect++;
return (vect->type == EXPR_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)
{
for (; vect->type; vect++) {
if (vect->type == location.segment + EXPR_SEGBASE && vect->value == -1)
return true;
}
return false;
}
/*
* Debug support: dump a description of an expression vector to stdout
*/
static const char *expr_type(int32_t type)
{
static char seg_str[64];
switch (type) {
case 0:
return "null";
case EXPR_UNKNOWN:
return "unknown";
case EXPR_SIMPLE:
return "simple";
case EXPR_WRT:
return "wrt";
case EXPR_RDSAE:
return "sae";
default:
break;
}
if (type >= EXPR_REG_START && type <= EXPR_REG_END) {
return nasm_reg_names[type - EXPR_REG_START];
} else if (type >= EXPR_SEGBASE) {
snprintf(seg_str, sizeof seg_str, "%sseg %d",
(type - EXPR_SEGBASE) == location.segment ? "this " : "",
type - EXPR_SEGBASE);
return seg_str;
} else {
return "ERR";
}
}
void dump_expr(const expr *e)
{
printf("[");
for (; e->type; e++)
printf("<%s(%d),%ld>", expr_type(e->type), e->type, e->value);
printf("]\n");
}