mirror of
https://github.com/netwide-assembler/nasm.git
synced 2025-04-06 18:30:21 +08:00
Diagnostics: make debug more dynamic, note -> info, add listmsg level
Make debug messages more dynamic by making it easy to conditionalize the messages. Change ERR_NOTE to ERR_INFO which reflects the usage better. Other compilers use note: for additional information. Don't unwind the macro stack with ERR_HERE; it is only going to give confusing results as it will unwind the wrong macro stack. Add ERR_LISTMSG level which is *always* suppressed, but will still appear in the list file. Signed-off-by: H. Peter Anvin (Intel) <hpa@zytor.com>
This commit is contained in:
parent
524918394d
commit
d66927a677
@ -70,8 +70,9 @@ _type nasm_ ## _name (const char *fmt, ...) \
|
||||
abort(); \
|
||||
}
|
||||
|
||||
nasm_err_helpers(void, listmsg, ERR_LISTMSG)
|
||||
nasm_err_helpers(void, debug, ERR_DEBUG)
|
||||
nasm_err_helpers(void, note, ERR_NOTE)
|
||||
nasm_err_helpers(void, info, ERR_INFO)
|
||||
nasm_err_helpers(void, nonfatal, ERR_NONFATAL)
|
||||
nasm_err_helpers(fatal_func, fatal, ERR_FATAL)
|
||||
nasm_err_helpers(fatal_func, panic, ERR_PANIC)
|
||||
|
58
asm/nasm.c
58
asm/nasm.c
@ -87,6 +87,8 @@ static const struct error_format errfmt_msvc = { "(", ")", " : " };
|
||||
static const struct error_format *errfmt = &errfmt_gnu;
|
||||
static struct strlist *warn_list;
|
||||
|
||||
unsigned int debug_nasm; /* Debugging messages? */
|
||||
|
||||
static bool using_debug_info, opt_verbose_info;
|
||||
static const char *debug_format;
|
||||
|
||||
@ -832,7 +834,8 @@ enum text_options {
|
||||
OPT_BEFORE,
|
||||
OPT_LIMIT,
|
||||
OPT_KEEP_ALL,
|
||||
OPT_NO_LINE
|
||||
OPT_NO_LINE,
|
||||
OPT_DEBUG
|
||||
};
|
||||
struct textargs {
|
||||
const char *label;
|
||||
@ -857,6 +860,7 @@ static const struct textargs textopts[] = {
|
||||
{"limit-", OPT_LIMIT, true, 0},
|
||||
{"keep-all", OPT_KEEP_ALL, false, 0},
|
||||
{"no-line", OPT_NO_LINE, false, 0},
|
||||
{"debug", OPT_DEBUG, false, 0},
|
||||
{NULL, OPT_BOGUS, false, 0}
|
||||
};
|
||||
|
||||
@ -1197,6 +1201,9 @@ static bool process_arg(char *p, char *q, int pass)
|
||||
case OPT_NO_LINE:
|
||||
pp_noline = true;
|
||||
break;
|
||||
case OPT_DEBUG:
|
||||
debug_nasm = param ? strtoul(param, NULL, 10) : 1;
|
||||
break;
|
||||
case OPT_HELP:
|
||||
help(0);
|
||||
exit(0);
|
||||
@ -1592,7 +1599,7 @@ static void assemble_file(const char *fname, struct strlist *depend_list)
|
||||
"after %"PRId64" passes; "
|
||||
"stalled for %"PRId64", giving up.",
|
||||
pass_count(), stall_count);
|
||||
nasm_notef(ERR_UNDEAD,
|
||||
nasm_nonfatalf(ERR_UNDEAD,
|
||||
"Possible causes: recursive EQUs, macro abuse.");
|
||||
}
|
||||
break;
|
||||
@ -1625,8 +1632,7 @@ static void assemble_file(const char *fname, struct strlist *depend_list)
|
||||
|
||||
if (opt_verbose_info && pass_final()) {
|
||||
/* -On and -Ov switches */
|
||||
nasm_note("info: assembly required 1+%"PRId64"+2 passes\n",
|
||||
pass_count()-3);
|
||||
nasm_info("assembly required 1+%"PRId64"+2 passes\n", pass_count()-3);
|
||||
}
|
||||
|
||||
strlist_free(&warn_list);
|
||||
@ -1654,14 +1660,24 @@ static size_t warn_index(errflags severity)
|
||||
|
||||
static bool skip_this_pass(errflags severity)
|
||||
{
|
||||
errflags type = severity & ERR_MASK;
|
||||
|
||||
/*
|
||||
* See if it's a pass-specific error or warning which should be skipped.
|
||||
* We can never skip fatal errors as by definition they cannot be
|
||||
* resumed from.
|
||||
*/
|
||||
if ((severity & ERR_MASK) >= ERR_FATAL)
|
||||
if (type >= ERR_FATAL)
|
||||
return false;
|
||||
|
||||
/*
|
||||
* ERR_LISTMSG messages are always skipped; the list file
|
||||
* receives them anyway as this function is not consulted
|
||||
* for sending to the list file.
|
||||
*/
|
||||
if (type == ERR_LISTMSG)
|
||||
return true;
|
||||
|
||||
/* This message not applicable unless pass_final */
|
||||
return (severity & ERR_PASS2) && !pass_final();
|
||||
}
|
||||
@ -1726,13 +1742,12 @@ static void nasm_verror_asm(errflags severity, const char *fmt, va_list args)
|
||||
char warnsuf[64];
|
||||
char linestr[64];
|
||||
const char *pfx;
|
||||
errflags spec_type = severity & ERR_MASK; /* type originally specified */
|
||||
errflags true_type = true_error_type(severity);
|
||||
const char *currentfile = NULL;
|
||||
int32_t lineno = 0;
|
||||
static const char * const pfx_table[ERR_MASK+1] = {
|
||||
"debug: ", "note: ", "warning: ", "error: ",
|
||||
"", "", "fatal: ", "panic: "
|
||||
";;; ", "debug: ", "info: ", "warning: ",
|
||||
"error: ", "", "fatal: ", "panic: "
|
||||
};
|
||||
|
||||
if (is_suppressed(severity))
|
||||
@ -1749,13 +1764,6 @@ static void nasm_verror_asm(errflags severity, const char *fmt, va_list args)
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* For a debug/warning/note event, if ERR_HERE is set don't
|
||||
* output anything if there is no current filename available
|
||||
*/
|
||||
if (!currentfile && (severity & ERR_HERE) && true_type <= ERR_WARNING)
|
||||
return;
|
||||
|
||||
if (severity & ERR_NO_SEVERITY)
|
||||
pfx = "";
|
||||
else
|
||||
@ -1763,7 +1771,11 @@ static void nasm_verror_asm(errflags severity, const char *fmt, va_list args)
|
||||
|
||||
vsnprintf(msg, sizeof msg, fmt, args);
|
||||
*warnsuf = 0;
|
||||
if (spec_type == ERR_WARNING) {
|
||||
if ((severity & (ERR_MASK|ERR_HERE|ERR_PP_LISTMACRO)) == ERR_WARNING) {
|
||||
/*
|
||||
* It's a warning without ERR_HERE defined, and we are not already
|
||||
* unwinding the macros that led us here.
|
||||
*/
|
||||
snprintf(warnsuf, sizeof warnsuf, " [-w+%s%s]",
|
||||
(true_type >= ERR_NONFATAL) ? "error=" : "",
|
||||
warning_name[warn_index(severity)]);
|
||||
@ -1777,7 +1789,10 @@ static void nasm_verror_asm(errflags severity, const char *fmt, va_list args)
|
||||
|
||||
if (!skip_this_pass(severity)) {
|
||||
const char *file = currentfile ? currentfile : "nasm";
|
||||
const char *here = (severity & ERR_HERE) ? " here" : "";
|
||||
const char *here = "";
|
||||
|
||||
if (severity & ERR_HERE)
|
||||
here = currentfile ? " here" : " in an unknown location";
|
||||
|
||||
if (warn_list && true_type < ERR_NONFATAL &&
|
||||
!(pass_first() && (severity & ERR_PASS1)))
|
||||
@ -1822,7 +1837,7 @@ static void nasm_verror_asm(errflags severity, const char *fmt, va_list args)
|
||||
lfmt->error(severity, "%s%s in file %s%s",
|
||||
pfx, msg, currentfile, warnsuf);
|
||||
else
|
||||
lfmt->error(severity, "%s%s in unknown location%s",
|
||||
lfmt->error(severity, "%s%s in an unknown location%s",
|
||||
pfx, msg, warnsuf);
|
||||
} else {
|
||||
lfmt->error(severity, "%s%s%s", pfx, msg, warnsuf);
|
||||
@ -1834,11 +1849,14 @@ static void nasm_verror_asm(errflags severity, const char *fmt, va_list args)
|
||||
if (severity & ERR_USAGE)
|
||||
want_usage = true;
|
||||
|
||||
preproc->error_list_macros(severity);
|
||||
/* error_list_macros can for obvious reasons not work with ERR_HERE */
|
||||
if (!(severity & ERR_HERE))
|
||||
preproc->error_list_macros(severity);
|
||||
|
||||
switch (true_type) {
|
||||
case ERR_NOTE:
|
||||
case ERR_LISTMSG:
|
||||
case ERR_DEBUG:
|
||||
case ERR_INFO:
|
||||
case ERR_WARNING:
|
||||
/* no further action, by definition */
|
||||
break;
|
||||
|
@ -1,6 +1,6 @@
|
||||
/* ----------------------------------------------------------------------- *
|
||||
*
|
||||
* Copyright 1996-2018 The NASM Authors - All Rights Reserved
|
||||
* Copyright 1996-2019 The NASM Authors - All Rights Reserved
|
||||
* See the file AUTHORS included with the NASM distribution for
|
||||
* the specific copyright holders.
|
||||
*
|
||||
@ -54,10 +54,12 @@ typedef uint32_t errflags;
|
||||
* An error reporting function should look like this.
|
||||
*/
|
||||
void printf_func(2, 3) nasm_error(errflags severity, const char *fmt, ...);
|
||||
void printf_func(1, 2) nasm_listmsg(const char *fmt, ...);
|
||||
void printf_func(2, 3) nasm_listmsgf(errflags flags, const char *fmt, ...);
|
||||
void printf_func(1, 2) nasm_debug(const char *fmt, ...);
|
||||
void printf_func(2, 3) nasm_debugf(errflags flags, const char *fmt, ...);
|
||||
void printf_func(1, 2) nasm_note(const char *fmt, ...);
|
||||
void printf_func(2, 3) nasm_notef(errflags flags, const char *fmt, ...);
|
||||
void printf_func(1, 2) nasm_info(const char *fmt, ...);
|
||||
void printf_func(2, 3) nasm_intof(errflags flags, const char *fmt, ...);
|
||||
void printf_func(2, 3) nasm_warn(errflags flags, const char *fmt, ...);
|
||||
void printf_func(1, 2) nasm_nonfatal(const char *fmt, ...);
|
||||
void printf_func(2, 3) nasm_nonfatalf(errflags flags, const char *fmt, ...);
|
||||
@ -82,10 +84,11 @@ static inline vefunc nasm_set_verror(vefunc ve)
|
||||
* These are the error severity codes which get passed as the first
|
||||
* argument to an efunc.
|
||||
*/
|
||||
#define ERR_DEBUG 0x00000000 /* put out debugging message */
|
||||
#define ERR_NOTE 0x00000001 /* additional error information */
|
||||
#define ERR_WARNING 0x00000002 /* warn only: no further action */
|
||||
#define ERR_NONFATAL 0x00000003 /* terminate assembly after phase */
|
||||
#define ERR_LISTMSG 0x00000000 /* for the listing file only */
|
||||
#define ERR_DEBUG 0x00000001 /* debugging message */
|
||||
#define ERR_INFO 0x00000002 /* information for the list file */
|
||||
#define ERR_WARNING 0x00000003 /* warn only: no further action */
|
||||
#define ERR_NONFATAL 0x00000004 /* terminate assembly after phase */
|
||||
#define ERR_FATAL 0x00000006 /* instantly fatal: exit with error */
|
||||
#define ERR_PANIC 0x00000007 /* internal error: panic instantly
|
||||
* and dump core for reference */
|
||||
@ -131,4 +134,18 @@ void reset_warnings(void);
|
||||
/* Should be included from within error.h only */
|
||||
#include "warnings.h"
|
||||
|
||||
/* By defining MAX_DEBUG, we can compile out messages entirely */
|
||||
#ifndef MAX_DEBUG
|
||||
# define MAX_DEBUG (~0U)
|
||||
#endif
|
||||
|
||||
/* Debug level checks */
|
||||
static inline bool debug_level(unsigned int level)
|
||||
{
|
||||
extern unsigned int debug_nasm;
|
||||
if (is_constant(level) && level > MAX_DEBUG)
|
||||
return false;
|
||||
return unlikely(level <= debug_nasm);
|
||||
}
|
||||
|
||||
#endif /* NASM_ERROR_H */
|
||||
|
@ -228,11 +228,11 @@ static void bin_cleanup(void)
|
||||
uint64_t pend;
|
||||
int h;
|
||||
|
||||
#ifdef DEBUG
|
||||
nasm_debug("bin_cleanup: Sections were initially referenced in this order:\n");
|
||||
for (h = 0, s = sections; s; h++, s = s->next)
|
||||
fprintf(stdout, "%i. %s\n", h, s->name);
|
||||
#endif
|
||||
if (debug_level(1)) {
|
||||
nasm_debug("bin_cleanup: Sections were initially referenced in this order:\n");
|
||||
for (h = 0, s = sections; s; h++, s = s->next)
|
||||
nasm_debug("%i. %s\n", h, s->name);
|
||||
}
|
||||
|
||||
/* Assembly has completed, so now we need to generate the output file.
|
||||
* Step 1: Separate progbits and nobits sections into separate lists.
|
||||
@ -511,12 +511,12 @@ static void bin_cleanup(void)
|
||||
if (h)
|
||||
nasm_fatal("circular vfollows path detected");
|
||||
|
||||
#ifdef DEBUG
|
||||
nasm_debug("bin_cleanup: Confirm final section order for output file:\n");
|
||||
for (h = 0, s = sections; s && (s->flags & TYPE_PROGBITS);
|
||||
h++, s = s->next)
|
||||
fprintf(stdout, "%i. %s\n", h, s->name);
|
||||
#endif
|
||||
if (debug_level(1)) {
|
||||
nasm_debug("bin_cleanup: Confirm final section order for output file:\n");
|
||||
for (h = 0, s = sections; s && (s->flags & TYPE_PROGBITS);
|
||||
h++, s = s->next)
|
||||
nasm_debug("%i. %s\n", h, s->name);
|
||||
}
|
||||
|
||||
/* Step 5: Apply relocations. */
|
||||
|
||||
|
@ -734,10 +734,11 @@ static void elf_deflabel(char *name, int32_t segment, int64_t offset,
|
||||
struct elf_symbol *sym;
|
||||
bool special_used = false;
|
||||
|
||||
#if defined(DEBUG) && DEBUG>2
|
||||
nasm_debug(" elf_deflabel: %s, seg=%"PRIx32", off=%"PRIx64", is_global=%d, %s\n",
|
||||
name, segment, offset, is_global, special);
|
||||
#endif
|
||||
if (debug_level(2)) {
|
||||
nasm_debug(" elf_deflabel: %s, seg=%"PRIx32", off=%"PRIx64", is_global=%d, %s\n",
|
||||
name, segment, offset, is_global, special);
|
||||
}
|
||||
|
||||
if (name[0] == '.' && name[1] == '.' && name[2] != '@') {
|
||||
/*
|
||||
* This is a NASM special symbol. We never allow it into
|
||||
|
@ -771,10 +771,9 @@ static void obj_deflabel(char *name, int32_t segment,
|
||||
int i;
|
||||
bool used_special = false; /* have we used the special text? */
|
||||
|
||||
#if defined(DEBUG) && DEBUG>2
|
||||
nasm_debug(" obj_deflabel: %s, seg=%"PRIx32", off=%"PRIx64", is_global=%d, %s\n",
|
||||
name, segment, offset, is_global, special);
|
||||
#endif
|
||||
if (debug_level(2))
|
||||
nasm_debug(" obj_deflabel: %s, seg=%"PRIx32", off=%"PRIx64", is_global=%d, %s\n",
|
||||
name, segment, offset, is_global, special);
|
||||
|
||||
/*
|
||||
* If it's a special-retry from pass two, discard it.
|
||||
@ -1330,10 +1329,9 @@ static int32_t obj_segment(char *name, int *bits)
|
||||
* using the pointer it gets passed. That way we save memory,
|
||||
* by sponging off the label manager.
|
||||
*/
|
||||
#if defined(DEBUG) && DEBUG>=3
|
||||
nasm_debug(" obj_segment: < %s >, pass=%d, *bits=%d\n",
|
||||
name, pass, *bits);
|
||||
#endif
|
||||
if (debug_level(3))
|
||||
nasm_debug(" obj_segment: < %s >, *bits=%d\n", name, *bits);
|
||||
|
||||
if (!name) {
|
||||
*bits = 16;
|
||||
current_seg = NULL;
|
||||
|
Loading…
x
Reference in New Issue
Block a user