0
0
mirror of https://github.com/netwide-assembler/nasm.git synced 2025-09-22 10:43:39 -04:00
Files
nasm/asm/pragma.c
H. Peter Anvin 98578071b9 Cleanup of label renaming infrastructure, add subsection support
In order to support Mach-O better, add support for subsections, as
used by Mach-O "subsections_via_symbols". We also want to add
infrastructure to support this by downcalling to the backend to
indicate if a new subsection is needed.

Currently this supports a maximum of 2^14 subsections per section for
Mach-O; this can be addressed by adding a level of indirection (or
cleaning up the handling of sections so we have an actual data
structure.)

Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
2018-06-01 18:06:25 -07:00

257 lines
8.0 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.
*
* ----------------------------------------------------------------------- */
/*
* Parse and handle [pragma] directives. The preprocessor handles
* %pragma preproc directives separately, all other namespaces are
* simply converted to [pragma].
*/
#include "compiler.h"
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <limits.h>
#include "nasm.h"
#include "nasmlib.h"
#include "assemble.h"
#include "error.h"
static enum directive_result asm_pragma(const struct pragma *pragma);
/*
* Handle [pragma] directives. [pragma] is generally produced by
* the %pragma preprocessor directive, which simply passes on any
* string that it finds *except* %pragma preproc. The idea is
* that pragmas are of the form:
*
* %pragma <facility> <opname> [<options>...]
*
* ... where "facility" can be either a generic facility or a backend
* name.
*
* The following names are currently reserved for global facilities;
* so far none of these have any defined pragmas at all:
*
* preproc - preprocessor
* asm - assembler
* list - listing generator
* file - generic file handling
* input - input file handling
* output - backend-independent output handling
* debug - backend-independent debug handling
* ignore - dummy pragma (can be used to "comment out")
*
* This function should generally not error out if it doesn't understand
* what a pragma is for, for unknown arguments, etc; the whole point of
* a pragma is that future releases might add new ones that should be
* ignored rather than be an error. Erroring out is acceptable for
* known pragmas suffering from parsing errors and so on.
*
* Adding default-suppressed warnings would, however, be a good idea
* at some point.
*/
static struct pragma_facility global_pragmas[] =
{
{ "asm", asm_pragma },
{ "list", NULL },
{ "file", NULL },
{ "input", NULL },
/* None of these should actually happen... */
{ "preproc", NULL }, /* This shouldn't happen... */
{ "output", NULL },
{ "debug", NULL },
{ "ignore", NULL },
{ NULL, NULL }
};
/*
* Search a pragma list for a known pragma facility and if so, invoke
* the handler. Return true if processing is complete.
* The "default name", if set, matches the final NULL entry (used
* for backends, so multiple backends can share the same list under
* some circumstances.)
*/
static bool search_pragma_list(const struct pragma_facility *list,
const char *default_name,
struct pragma *pragma)
{
const struct pragma_facility *pf;
enum directive_result rv;
if (!list)
return false;
for (pf = list; pf->name; pf++) {
if (!nasm_stricmp(pragma->facility_name, pf->name))
goto found_it;
}
if (default_name && !nasm_stricmp(pragma->facility_name, default_name))
goto found_it;
return false;
found_it:
pragma->facility = pf;
/* If the handler is NULL all pragmas are unknown... */
if (pf->handler)
rv = pf->handler(pragma);
else
rv = DIRR_UNKNOWN;
switch (rv) {
case DIRR_UNKNOWN:
switch (pragma->opcode) {
case D_none:
nasm_error(ERR_WARNING|ERR_PASS2|ERR_WARN_BAD_PRAGMA,
"empty %%pragma %s", pragma->facility_name);
break;
default:
nasm_error(ERR_WARNING|ERR_PASS2|ERR_WARN_UNKNOWN_PRAGMA,
"unknown %%pragma %s %s",
pragma->facility_name, pragma->opname);
break;
}
break;
case DIRR_OK:
case DIRR_ERROR:
break; /* Nothing to do */
case DIRR_BADPARAM:
/*
* This one is an error. Don't use it if forward compatibility
* would be compromised, as opposed to an inherent error.
*/
nasm_error(ERR_NONFATAL, "bad argument to %%pragma %s %s",
pragma->facility_name, pragma->opname);
break;
default:
panic();
}
return true;
}
void process_pragma(char *str)
{
struct pragma pragma;
char *p;
nasm_zero(pragma);
pragma.facility_name = nasm_get_word(str, &p);
if (!pragma.facility_name) {
nasm_error(ERR_WARNING|ERR_PASS2|ERR_WARN_BAD_PRAGMA,
"empty pragma directive");
return; /* Empty pragma */
}
/*
* The facility "ignore" means just that; don't even complain of
* the absence of an operation.
*/
if (!nasm_stricmp(pragma.facility_name, "ignore"))
return;
/*
* The "output" and "debug" facilities are aliases for the
* current output and debug formats, respectively.
*/
if (!nasm_stricmp(pragma.facility_name, "output"))
pragma.facility_name = ofmt->shortname;
if (!nasm_stricmp(pragma.facility_name, "debug"))
pragma.facility_name = dfmt->shortname;
pragma.opname = nasm_get_word(p, &p);
if (!pragma.opname)
pragma.opcode = D_none;
else
pragma.opcode = directive_find(pragma.opname);
pragma.tail = nasm_trim_spaces(p);
/* Look for a global pragma namespace */
if (search_pragma_list(global_pragmas, NULL, &pragma))
return;
/* Look to see if it is an output backend pragma */
if (search_pragma_list(ofmt->pragmas, ofmt->shortname, &pragma))
return;
/* Look to see if it is a debug format pragma */
if (search_pragma_list(dfmt->pragmas, dfmt->shortname, &pragma))
return;
/*
* Note: it would be nice to warn for an unknown namespace,
* but in order to do so we need to walk *ALL* the backends
* in order to make sure we aren't dealing with a pragma that
* is for another backend. On the other hand, that could
* also be a warning with a separate warning flag.
*
* Leave this for the future, however, the warning classes are
* already defined for future compatibility.
*/
}
/*
* Pragmas for the assembler proper
*/
static enum directive_result asm_pragma(const struct pragma *pragma)
{
switch (pragma->opcode) {
case D_PREFIX:
case D_GPREFIX:
set_label_mangle(LM_GPREFIX, pragma->tail);
return DIRR_OK;
case D_SUFFIX:
case D_GSUFFIX:
set_label_mangle(LM_GSUFFIX, pragma->tail);
return DIRR_OK;
case D_LPREFIX:
set_label_mangle(LM_LPREFIX, pragma->tail);
return DIRR_OK;
case D_LSUFFIX:
set_label_mangle(LM_LSUFFIX, pragma->tail);
return DIRR_OK;
default:
return DIRR_UNKNOWN;
}
}