2016-05-25 20:42:47 +08:00
|
|
|
/* ----------------------------------------------------------------------- *
|
2017-03-08 17:26:40 +08:00
|
|
|
*
|
2018-06-13 04:50:37 +08:00
|
|
|
* Copyright 1996-2018 The NASM Authors - All Rights Reserved
|
2016-05-25 20:42:47 +08:00
|
|
|
* 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.
|
2017-03-08 17:26:40 +08:00
|
|
|
*
|
2016-05-25 20:42:47 +08:00
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* ----------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
/*
|
2017-03-08 11:23:03 +08:00
|
|
|
* error.c - error message handling routines for the assembler
|
2016-05-25 20:42:47 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "compiler.h"
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include "nasmlib.h"
|
2017-03-08 11:23:03 +08:00
|
|
|
#include "error.h"
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Description of the suppressible warnings for the command line and
|
2017-03-08 17:26:40 +08:00
|
|
|
* the [warning] directive.
|
2017-03-08 11:23:03 +08:00
|
|
|
*/
|
2018-12-13 08:11:08 +08:00
|
|
|
#define on (WARN_ST_ENABLED)
|
|
|
|
#define off 0
|
|
|
|
#define err (WARN_ST_ENABLED|WARN_ST_ERROR)
|
|
|
|
|
2018-12-13 09:48:38 +08:00
|
|
|
const struct warning warnings[WARN_ALL+1] = {
|
2018-12-13 08:11:08 +08:00
|
|
|
{NULL, NULL, on}, /* must be on - used for unconditional enable */
|
|
|
|
{"macro-params", "macro calls with wrong parameter count", on},
|
|
|
|
{"macro-selfref", "cyclic macro references", off},
|
|
|
|
{"macro-defaults", "macros with more default than optional parameters", on},
|
|
|
|
{"orphan-labels", "labels alone on lines without trailing `:'", on},
|
|
|
|
{"number-overflow", "numeric constant does not fit", on},
|
|
|
|
{"gnu-elf-extensions", "using 8- or 16-bit relocation in ELF32, a GNU extension", off},
|
|
|
|
{"float-overflow", "floating point overflow", on},
|
|
|
|
{"float-denorm", "floating point denormal", off},
|
|
|
|
{"float-underflow", "floating point underflow", off},
|
|
|
|
{"float-toolong", "too many digits in floating-point number", on},
|
|
|
|
{"user", "%warning directives", on},
|
|
|
|
{"lock", "lock prefix on unlockable instructions", on},
|
|
|
|
{"hle", "invalid hle prefixes", on},
|
|
|
|
{"bnd", "invalid bnd prefixes", on},
|
|
|
|
{"zext-reloc", "relocation zero-extended to match output format", on},
|
|
|
|
{"ptr", "non-NASM keyword used in other assemblers", on},
|
|
|
|
{"bad-pragma", "empty or malformed %pragma", off},
|
|
|
|
{"unknown-pragma", "unknown %pragma facility or directive", off},
|
|
|
|
{"not-my-pragma", "%pragma not applicable to this compilation", off},
|
|
|
|
{"unknown-warning", "unknown warning in -W/-w or warning directive", off},
|
|
|
|
{"negative-rep", "regative %rep count", on},
|
|
|
|
{"phase", "phase error during stabilization", off},
|
2018-12-13 08:49:07 +08:00
|
|
|
{"label-redef", "label redefined to an identical value", off},
|
|
|
|
{"label-redef-late", "label (re)defined during code generation", err},
|
2017-03-08 17:26:40 +08:00
|
|
|
|
2018-12-13 07:58:32 +08:00
|
|
|
/* THESE ENTRIES SHOULD COME LAST */
|
2018-12-13 08:12:36 +08:00
|
|
|
{"other", "any warning not specifially mentioned above", on},
|
2018-12-13 08:11:08 +08:00
|
|
|
{"all", "all possible warnings", off}
|
2017-03-08 11:23:03 +08:00
|
|
|
};
|
2017-03-08 17:26:40 +08:00
|
|
|
|
2018-12-13 06:38:50 +08:00
|
|
|
uint8_t warning_state[WARN_ALL];/* Current state */
|
|
|
|
uint8_t warning_state_init[WARN_ALL]; /* Command-line state, for reset */
|
2016-05-25 20:42:47 +08:00
|
|
|
|
2018-11-24 23:58:11 +08:00
|
|
|
/* Global error handling function */
|
|
|
|
vefunc nasm_verror;
|
2016-05-25 20:42:47 +08:00
|
|
|
|
2018-12-11 13:53:54 +08:00
|
|
|
/* Common function body */
|
|
|
|
#define nasm_do_error(s) \
|
|
|
|
va_list ap; \
|
|
|
|
va_start(ap, fmt); \
|
|
|
|
nasm_verror((s), fmt, ap); \
|
2018-11-24 23:58:11 +08:00
|
|
|
va_end(ap);
|
2018-11-25 03:17:47 +08:00
|
|
|
|
2018-12-11 13:53:54 +08:00
|
|
|
void nasm_error(int severity, const char *fmt, ...)
|
2016-05-25 20:42:47 +08:00
|
|
|
{
|
2018-12-11 13:53:54 +08:00
|
|
|
nasm_do_error(severity);
|
2018-11-25 03:17:47 +08:00
|
|
|
}
|
2016-05-25 20:42:47 +08:00
|
|
|
|
2018-12-11 13:53:54 +08:00
|
|
|
#define nasm_err_helpers(_type, _name, _sev) \
|
|
|
|
_type nasm_ ## _name ## f (int flags, const char *fmt, ...) \
|
|
|
|
{ \
|
|
|
|
nasm_do_error((_sev)|flags); \
|
|
|
|
if (_sev >= ERR_FATAL) \
|
|
|
|
abort(); \
|
|
|
|
} \
|
|
|
|
_type nasm_ ## _name (const char *fmt, ...) \
|
|
|
|
{ \
|
|
|
|
nasm_do_error(_sev); \
|
|
|
|
if (_sev >= ERR_FATAL) \
|
|
|
|
abort(); \
|
2018-11-25 03:17:47 +08:00
|
|
|
}
|
|
|
|
|
2018-12-11 13:53:54 +08:00
|
|
|
nasm_err_helpers(void, debug, ERR_DEBUG)
|
|
|
|
nasm_err_helpers(void, note, ERR_NOTE)
|
|
|
|
nasm_err_helpers(void, warn, ERR_WARNING)
|
|
|
|
nasm_err_helpers(void, nonfatal, ERR_NONFATAL)
|
|
|
|
nasm_err_helpers(fatal_func, fatal, ERR_FATAL)
|
|
|
|
nasm_err_helpers(fatal_func, panic, ERR_PANIC)
|
2016-05-25 20:42:47 +08:00
|
|
|
|
2018-02-23 06:52:50 +08:00
|
|
|
fatal_func nasm_panic_from_macro(const char *file, int line)
|
2016-05-25 20:42:47 +08:00
|
|
|
{
|
2018-11-24 23:58:11 +08:00
|
|
|
nasm_panic("internal error at %s:%d\n", file, line);
|
2016-05-25 20:42:47 +08:00
|
|
|
}
|
|
|
|
|
2018-02-23 06:52:50 +08:00
|
|
|
fatal_func nasm_assert_failed(const char *file, int line, const char *msg)
|
2016-05-25 20:42:47 +08:00
|
|
|
{
|
2018-11-24 23:58:11 +08:00
|
|
|
nasm_panic("assertion %s failed at %s:%d", msg, file, line);
|
2017-03-08 17:26:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This is called when processing a -w or -W option, or a warning directive.
|
2018-12-13 08:11:08 +08:00
|
|
|
* Returns on if if the action was successful.
|
2017-03-08 17:26:40 +08:00
|
|
|
*/
|
|
|
|
bool set_warning_status(const char *value)
|
|
|
|
{
|
2018-11-24 23:58:11 +08:00
|
|
|
enum warn_action { WID_OFF, WID_ON, WID_RESET };
|
|
|
|
enum warn_action action;
|
|
|
|
bool ok = false;
|
|
|
|
uint8_t mask;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
value = nasm_skip_spaces(value);
|
|
|
|
switch (*value) {
|
|
|
|
case '-':
|
|
|
|
action = WID_OFF;
|
|
|
|
value++;
|
|
|
|
break;
|
|
|
|
case '+':
|
|
|
|
action = WID_ON;
|
|
|
|
value++;
|
|
|
|
break;
|
|
|
|
case '*':
|
|
|
|
action = WID_RESET;
|
|
|
|
value++;
|
|
|
|
break;
|
|
|
|
case 'N':
|
|
|
|
case 'n':
|
|
|
|
if (!nasm_strnicmp(value, "no-", 3)) {
|
|
|
|
action = WID_OFF;
|
|
|
|
value += 3;
|
|
|
|
break;
|
|
|
|
} else if (!nasm_stricmp(value, "none")) {
|
|
|
|
action = WID_OFF;
|
|
|
|
value = NULL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/* else fall through */
|
|
|
|
default:
|
|
|
|
action = WID_ON;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
mask = WARN_ST_ENABLED;
|
|
|
|
|
|
|
|
if (value && !nasm_strnicmp(value, "error", 5)) {
|
|
|
|
switch (value[5]) {
|
|
|
|
case '=':
|
|
|
|
mask = WARN_ST_ERROR;
|
|
|
|
value += 6;
|
|
|
|
break;
|
|
|
|
case '\0':
|
|
|
|
mask = WARN_ST_ERROR;
|
|
|
|
value = NULL;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* Just an accidental prefix? */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (value && !nasm_stricmp(value, "all"))
|
|
|
|
value = NULL;
|
|
|
|
|
|
|
|
/* This is inefficient, but it shouldn't matter... */
|
2018-12-13 09:48:38 +08:00
|
|
|
for (i = 0; i < WARN_ALL; i++) {
|
2018-11-24 23:58:11 +08:00
|
|
|
if (!value || !nasm_stricmp(value, warnings[i].name)) {
|
|
|
|
ok = true; /* At least one action taken */
|
|
|
|
switch (action) {
|
|
|
|
case WID_OFF:
|
|
|
|
warning_state[i] &= ~mask;
|
|
|
|
break;
|
|
|
|
case WID_ON:
|
|
|
|
warning_state[i] |= mask;
|
|
|
|
break;
|
|
|
|
case WID_RESET:
|
|
|
|
warning_state[i] &= ~mask;
|
|
|
|
warning_state[i] |= warning_state_init[i] & mask;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ok;
|
2016-05-25 20:42:47 +08:00
|
|
|
}
|