Limit number of passes to 1000
[nasm/perl-rewrite.git] / nasm.c
blob3e4b78e68585db4a0fa72157fb9f74dd14f0366a
1 /* The Netwide Assembler main program module
3 * The Netwide Assembler is copyright (C) 1996 Simon Tatham and
4 * Julian Hall. All rights reserved. The software is
5 * redistributable under the license given in the file "LICENSE"
6 * distributed in the NASM archive.
7 */
9 #include "compiler.h"
11 #include <stdio.h>
12 #include <stdarg.h>
13 #include <stdlib.h>
14 #include <string.h>
15 #include <ctype.h>
16 #include <inttypes.h>
17 #include <limits.h>
18 #include <time.h>
20 #include "nasm.h"
21 #include "nasmlib.h"
22 #include "saa.h"
23 #include "raa.h"
24 #include "float.h"
25 #include "stdscan.h"
26 #include "insns.h"
27 #include "preproc.h"
28 #include "parser.h"
29 #include "eval.h"
30 #include "assemble.h"
31 #include "labels.h"
32 #include "outform.h"
33 #include "listing.h"
35 struct forwrefinfo { /* info held on forward refs. */
36 int lineno;
37 int operand;
40 static int get_bits(char *value);
41 static uint32_t get_cpu(char *cpu_str);
42 static void parse_cmdline(int, char **);
43 static void assemble_file(char *, StrList **);
44 static void register_output_formats(void);
45 static void report_error_gnu(int severity, const char *fmt, ...);
46 static void report_error_vc(int severity, const char *fmt, ...);
47 static void report_error_common(int severity, const char *fmt,
48 va_list args);
49 static bool is_suppressed_warning(int severity);
50 static void usage(void);
51 static efunc report_error;
53 static int using_debug_info, opt_verbose_info;
54 bool tasm_compatible_mode = false;
55 int pass0, passn;
56 int maxbits = 0;
57 int globalrel = 0;
59 time_t official_compile_time;
61 static char inname[FILENAME_MAX];
62 static char outname[FILENAME_MAX];
63 static char listname[FILENAME_MAX];
64 static char errname[FILENAME_MAX];
65 static int globallineno; /* for forward-reference tracking */
66 /* static int pass = 0; */
67 static struct ofmt *ofmt = NULL;
69 static FILE *error_file; /* Where to write error messages */
71 static FILE *ofile = NULL;
72 int optimizing = -1; /* number of optimization passes to take */
73 static int sb, cmd_sb = 16; /* by default */
74 static uint32_t cmd_cpu = IF_PLEVEL; /* highest level by default */
75 static uint32_t cpu = IF_PLEVEL; /* passed to insn_size & assemble.c */
76 bool global_offset_changed; /* referenced in labels.c */
78 static struct location location;
79 int in_abs_seg; /* Flag we are in ABSOLUTE seg */
80 int32_t abs_seg; /* ABSOLUTE segment basis */
81 int32_t abs_offset; /* ABSOLUTE offset */
83 static struct RAA *offsets;
85 static struct SAA *forwrefs; /* keep track of forward references */
86 static const struct forwrefinfo *forwref;
88 static Preproc *preproc;
89 enum op_type {
90 op_normal, /* Preprocess and assemble */
91 op_preprocess, /* Preprocess only */
92 op_depend, /* Generate dependencies */
94 static enum op_type operating_mode;
95 /* Dependency flags */
96 static bool depend_emit_phony = false;
97 static bool depend_missing_ok = false;
98 static const char *depend_target = NULL;
99 static const char *depend_file = NULL;
102 * Which of the suppressible warnings are suppressed. Entry zero
103 * isn't an actual warning, but it used for -w+error/-Werror.
105 static bool suppressed[ERR_WARN_MAX+1];
107 static bool suppressed_global[ERR_WARN_MAX+1] = {
108 true, false, true, false, false, false, true, false, true, true, false
111 * The option names for the suppressible warnings. As before, entry
112 * zero does nothing.
114 static const char *suppressed_names[ERR_WARN_MAX+1] = {
115 "error", "macro-params", "macro-selfref", "macro-defaults",
116 "orphan-labels", "number-overflow", "gnu-elf-extensions",
117 "float-overflow", "float-denorm", "float-underflow", "float-toolong"
121 * The explanations for the suppressible warnings. As before, entry
122 * zero does nothing.
124 static const char *suppressed_what[ERR_WARN_MAX+1] = {
125 "treat warnings as errors",
126 "macro calls with wrong parameter count",
127 "cyclic macro references",
128 "macros with more default than optional parameters",
129 "labels alone on lines without trailing `:'",
130 "numeric constants does not fit in 64 bits",
131 "using 8- or 16-bit relocation in ELF32, a GNU extension",
132 "floating point overflow",
133 "floating point denormal",
134 "floating point underflow",
135 "too many digits in floating-point number"
139 * This is a null preprocessor which just copies lines from input
140 * to output. It's used when someone explicitly requests that NASM
141 * not preprocess their source file.
144 static void no_pp_reset(char *, int, efunc, evalfunc, ListGen *, StrList **);
145 static char *no_pp_getline(void);
146 static void no_pp_cleanup(int);
147 static Preproc no_pp = {
148 no_pp_reset,
149 no_pp_getline,
150 no_pp_cleanup
154 * get/set current offset...
156 #define GET_CURR_OFFS (in_abs_seg?abs_offset:\
157 raa_read(offsets,location.segment))
158 #define SET_CURR_OFFS(x) (in_abs_seg?(void)(abs_offset=(x)):\
159 (void)(offsets=raa_write(offsets,location.segment,(x))))
161 static int want_usage;
162 static int terminate_after_phase;
163 int user_nolist = 0; /* fbk 9/2/00 */
165 static void nasm_fputs(const char *line, FILE * outfile)
167 if (outfile) {
168 fputs(line, outfile);
169 putc('\n', outfile);
170 } else
171 puts(line);
174 /* Convert a struct tm to a POSIX-style time constant */
175 static int64_t posix_mktime(struct tm *tm)
177 int64_t t;
178 int64_t y = tm->tm_year;
180 /* See IEEE 1003.1:2004, section 4.14 */
182 t = (y-70)*365 + (y-69)/4 - (y-1)/100 + (y+299)/400;
183 t += tm->tm_yday;
184 t *= 24;
185 t += tm->tm_hour;
186 t *= 60;
187 t += tm->tm_min;
188 t *= 60;
189 t += tm->tm_sec;
191 return t;
194 static void define_macros_early(void)
196 char temp[128];
197 struct tm lt, *lt_p, gm, *gm_p;
198 int64_t posix_time;
200 lt_p = localtime(&official_compile_time);
201 if (lt_p) {
202 lt = *lt_p;
204 strftime(temp, sizeof temp, "__DATE__=\"%Y-%m-%d\"", &lt);
205 pp_pre_define(temp);
206 strftime(temp, sizeof temp, "__DATE_NUM__=%Y%m%d", &lt);
207 pp_pre_define(temp);
208 strftime(temp, sizeof temp, "__TIME__=\"%H:%M:%S\"", &lt);
209 pp_pre_define(temp);
210 strftime(temp, sizeof temp, "__TIME_NUM__=%H%M%S", &lt);
211 pp_pre_define(temp);
214 gm_p = gmtime(&official_compile_time);
215 if (gm_p) {
216 gm = *gm_p;
218 strftime(temp, sizeof temp, "__UTC_DATE__=\"%Y-%m-%d\"", &gm);
219 pp_pre_define(temp);
220 strftime(temp, sizeof temp, "__UTC_DATE_NUM__=%Y%m%d", &gm);
221 pp_pre_define(temp);
222 strftime(temp, sizeof temp, "__UTC_TIME__=\"%H:%M:%S\"", &gm);
223 pp_pre_define(temp);
224 strftime(temp, sizeof temp, "__UTC_TIME_NUM__=%H%M%S", &gm);
225 pp_pre_define(temp);
228 if (gm_p)
229 posix_time = posix_mktime(&gm);
230 else if (lt_p)
231 posix_time = posix_mktime(&lt);
232 else
233 posix_time = 0;
235 if (posix_time) {
236 snprintf(temp, sizeof temp, "__POSIX_TIME__=%"PRId64, posix_time);
237 pp_pre_define(temp);
241 static void define_macros_late(void)
243 char temp[128];
245 snprintf(temp, sizeof(temp), "__OUTPUT_FORMAT__=%s\n",
246 ofmt->shortname);
247 pp_pre_define(temp);
250 static void emit_dependencies(StrList *list)
252 FILE *deps;
253 int linepos, len;
254 StrList *l, *nl;
256 if (depend_file && strcmp(depend_file, "-")) {
257 deps = fopen(depend_file, "w");
258 if (!deps) {
259 report_error(ERR_NONFATAL|ERR_NOFILE|ERR_USAGE,
260 "unable to write dependency file `%s'", depend_file);
261 return;
263 } else {
264 deps = stdout;
267 linepos = fprintf(deps, "%s:", depend_target);
268 for (l = list; l; l = l->next) {
269 len = strlen(l->str);
270 if (linepos + len > 62) {
271 fprintf(deps, " \\\n ");
272 linepos = 1;
274 fprintf(deps, " %s", l->str);
275 linepos += len+1;
277 fprintf(deps, "\n\n");
279 for (l = list; l; l = nl) {
280 if (depend_emit_phony)
281 fprintf(deps, "%s:\n\n", l->str);
283 nl = l->next;
284 nasm_free(l);
287 if (deps != stdout)
288 fclose(deps);
291 int main(int argc, char **argv)
293 StrList *depend_list = NULL, **depend_ptr;
295 time(&official_compile_time);
297 pass0 = 0;
298 want_usage = terminate_after_phase = false;
299 report_error = report_error_gnu;
301 error_file = stderr;
303 tolower_init();
305 nasm_set_malloc_error(report_error);
306 offsets = raa_init();
307 forwrefs = saa_init((int32_t)sizeof(struct forwrefinfo));
309 preproc = &nasmpp;
310 operating_mode = op_normal;
312 seg_init();
314 register_output_formats();
316 /* Define some macros dependent on the runtime, but not
317 on the command line. */
318 define_macros_early();
320 parse_cmdline(argc, argv);
322 if (terminate_after_phase) {
323 if (want_usage)
324 usage();
325 return 1;
328 /* If debugging info is disabled, suppress any debug calls */
329 if (!using_debug_info)
330 ofmt->current_dfmt = &null_debug_form;
332 if (ofmt->stdmac)
333 pp_extra_stdmac(ofmt->stdmac);
334 parser_global_info(ofmt, &location);
335 eval_global_info(ofmt, lookup_label, &location);
337 /* define some macros dependent of command-line */
338 define_macros_late();
340 depend_ptr = (depend_file || (operating_mode == op_depend))
341 ? &depend_list : NULL;
342 if (!depend_target)
343 depend_target = outname;
345 switch (operating_mode) {
346 case op_depend:
348 char *line;
350 if (depend_missing_ok)
351 pp_include_path(NULL); /* "assume generated" */
353 preproc->reset(inname, 0, report_error, evaluate, &nasmlist,
354 depend_ptr);
355 if (outname[0] == '\0')
356 ofmt->filename(inname, outname, report_error);
357 ofile = NULL;
358 while ((line = preproc->getline()))
359 nasm_free(line);
360 preproc->cleanup(0);
362 break;
364 case op_preprocess:
366 char *line;
367 char *file_name = NULL;
368 int32_t prior_linnum = 0;
369 int lineinc = 0;
371 if (*outname) {
372 ofile = fopen(outname, "w");
373 if (!ofile)
374 report_error(ERR_FATAL | ERR_NOFILE,
375 "unable to open output file `%s'",
376 outname);
377 } else
378 ofile = NULL;
380 location.known = false;
382 /* pass = 1; */
383 preproc->reset(inname, 2, report_error, evaluate, &nasmlist,
384 depend_ptr);
386 while ((line = preproc->getline())) {
388 * We generate %line directives if needed for later programs
390 int32_t linnum = prior_linnum += lineinc;
391 int altline = src_get(&linnum, &file_name);
392 if (altline) {
393 if (altline == 1 && lineinc == 1)
394 nasm_fputs("", ofile);
395 else {
396 lineinc = (altline != -1 || lineinc != 1);
397 fprintf(ofile ? ofile : stdout,
398 "%%line %"PRId32"+%d %s\n", linnum, lineinc,
399 file_name);
401 prior_linnum = linnum;
403 nasm_fputs(line, ofile);
404 nasm_free(line);
406 nasm_free(file_name);
407 preproc->cleanup(0);
408 if (ofile)
409 fclose(ofile);
410 if (ofile && terminate_after_phase)
411 remove(outname);
413 break;
415 case op_normal:
418 * We must call ofmt->filename _anyway_, even if the user
419 * has specified their own output file, because some
420 * formats (eg OBJ and COFF) use ofmt->filename to find out
421 * the name of the input file and then put that inside the
422 * file.
424 ofmt->filename(inname, outname, report_error);
426 ofile = fopen(outname, "wb");
427 if (!ofile) {
428 report_error(ERR_FATAL | ERR_NOFILE,
429 "unable to open output file `%s'", outname);
433 * We must call init_labels() before ofmt->init() since
434 * some object formats will want to define labels in their
435 * init routines. (eg OS/2 defines the FLAT group)
437 init_labels();
439 ofmt->init(ofile, report_error, define_label, evaluate);
441 assemble_file(inname, depend_ptr);
443 if (!terminate_after_phase) {
444 ofmt->cleanup(using_debug_info);
445 cleanup_labels();
446 } else {
448 * Despite earlier comments, we need this fclose.
449 * The object output drivers only fclose on cleanup,
450 * and we just skipped that.
452 fclose (ofile);
454 remove(outname);
455 if (listname[0])
456 remove(listname);
459 break;
462 if (depend_list)
463 emit_dependencies(depend_list);
465 if (want_usage)
466 usage();
468 raa_free(offsets);
469 saa_free(forwrefs);
470 eval_cleanup();
471 stdscan_cleanup();
473 if (terminate_after_phase)
474 return 1;
475 else
476 return 0;
480 * Get a parameter for a command line option.
481 * First arg must be in the form of e.g. -f...
483 static char *get_param(char *p, char *q, bool *advance)
485 *advance = false;
486 if (p[2]) { /* the parameter's in the option */
487 p += 2;
488 while (nasm_isspace(*p))
489 p++;
490 return p;
492 if (q && q[0]) {
493 *advance = true;
494 return q;
496 report_error(ERR_NONFATAL | ERR_NOFILE | ERR_USAGE,
497 "option `-%c' requires an argument", p[1]);
498 return NULL;
502 * Copy a filename
504 static void copy_filename(char *dst, const char *src)
506 size_t len = strlen(src);
508 if (len >= (size_t)FILENAME_MAX) {
509 report_error(ERR_FATAL | ERR_NOFILE, "file name too long");
510 return;
512 strncpy(dst, src, FILENAME_MAX);
516 * Convert a string to Make-safe form
518 static char *quote_for_make(const char *str)
520 const char *p;
521 char *os, *q;
523 size_t n = 1; /* Terminating zero */
524 size_t nbs = 0;
526 if (!str)
527 return NULL;
529 for (p = str; *p; p++) {
530 switch (*p) {
531 case ' ':
532 case '\t':
533 /* Convert N backslashes + ws -> 2N+1 backslashes + ws */
534 n += nbs + 2;
535 nbs = 0;
536 break;
537 case '$':
538 case '#':
539 nbs = 0;
540 n += 2;
541 break;
542 case '\\':
543 nbs++;
544 n++;
545 break;
546 default:
547 nbs = 0;
548 n++;
549 break;
553 /* Convert N backslashes at the end of filename to 2N backslashes */
554 if (nbs)
555 n += nbs;
557 os = q = nasm_malloc(n);
559 nbs = 0;
560 for (p = str; *p; p++) {
561 switch (*p) {
562 case ' ':
563 case '\t':
564 while (nbs--)
565 *q++ = '\\';
566 *q++ = '\\';
567 *q++ = *p;
568 break;
569 case '$':
570 *q++ = *p;
571 *q++ = *p;
572 nbs = 0;
573 break;
574 case '#':
575 *q++ = '\\';
576 *q++ = *p;
577 nbs = 0;
578 break;
579 case '\\':
580 *q++ = *p;
581 nbs++;
582 break;
583 default:
584 *q++ = *p;
585 nbs = 0;
586 break;
589 while (nbs--)
590 *q++ = '\\';
592 *q = '\0';
594 return os;
597 struct textargs {
598 const char *label;
599 int value;
602 #define OPT_PREFIX 0
603 #define OPT_POSTFIX 1
604 struct textargs textopts[] = {
605 {"prefix", OPT_PREFIX},
606 {"postfix", OPT_POSTFIX},
607 {NULL, 0}
610 static bool stopoptions = false;
611 static bool process_arg(char *p, char *q)
613 char *param;
614 int i;
615 bool advance = false;
616 bool suppress;
618 if (!p || !p[0])
619 return false;
621 if (p[0] == '-' && !stopoptions) {
622 if (strchr("oOfpPdDiIlFXuUZwW", p[1])) {
623 /* These parameters take values */
624 if (!(param = get_param(p, q, &advance)))
625 return advance;
628 switch (p[1]) {
629 case 's':
630 error_file = stdout;
631 break;
633 case 'o': /* output file */
634 copy_filename(outname, param);
635 break;
637 case 'f': /* output format */
638 ofmt = ofmt_find(param);
639 if (!ofmt) {
640 report_error(ERR_FATAL | ERR_NOFILE | ERR_USAGE,
641 "unrecognised output format `%s' - "
642 "use -hf for a list", param);
643 } else {
644 ofmt->current_dfmt = ofmt->debug_formats[0];
646 break;
648 case 'O': /* Optimization level */
650 int opt;
652 if (!*param) {
653 /* Naked -O == -Ox */
654 optimizing = INT_MAX >> 1; /* Almost unlimited */
655 } else {
656 while (*param) {
657 switch (*param) {
658 case '0': case '1': case '2': case '3': case '4':
659 case '5': case '6': case '7': case '8': case '9':
660 opt = strtoul(param, &param, 10);
662 /* -O0 -> optimizing == -1, 0.98 behaviour */
663 /* -O1 -> optimizing == 0, 0.98.09 behaviour */
664 if (opt < 2)
665 optimizing = opt - 1;
666 else
667 optimizing = opt;
668 break;
670 case 'v':
671 case '+':
672 param++;
673 opt_verbose_info = true;
674 break;
676 case 'x':
677 param++;
678 optimizing = INT_MAX >> 1; /* Almost unlimited */
679 break;
681 default:
682 report_error(ERR_FATAL,
683 "unknown optimization option -O%c\n",
684 *param);
685 break;
689 break;
692 case 'p': /* pre-include */
693 case 'P':
694 pp_pre_include(param);
695 break;
697 case 'd': /* pre-define */
698 case 'D':
699 pp_pre_define(param);
700 break;
702 case 'u': /* un-define */
703 case 'U':
704 pp_pre_undefine(param);
705 break;
707 case 'i': /* include search path */
708 case 'I':
709 pp_include_path(param);
710 break;
712 case 'l': /* listing file */
713 copy_filename(listname, param);
714 break;
716 case 'Z': /* error messages file */
717 strcpy(errname, param);
718 break;
720 case 'F': /* specify debug format */
721 ofmt->current_dfmt = dfmt_find(ofmt, param);
722 if (!ofmt->current_dfmt) {
723 report_error(ERR_FATAL | ERR_NOFILE | ERR_USAGE,
724 "unrecognized debug format `%s' for"
725 " output format `%s'",
726 param, ofmt->shortname);
728 using_debug_info = true;
729 break;
731 case 'X': /* specify error reporting format */
732 if (nasm_stricmp("vc", param) == 0)
733 report_error = report_error_vc;
734 else if (nasm_stricmp("gnu", param) == 0)
735 report_error = report_error_gnu;
736 else
737 report_error(ERR_FATAL | ERR_NOFILE | ERR_USAGE,
738 "unrecognized error reporting format `%s'",
739 param);
740 break;
742 case 'g':
743 using_debug_info = true;
744 break;
746 case 'h':
747 printf
748 ("usage: nasm [-@ response file] [-o outfile] [-f format] "
749 "[-l listfile]\n"
750 " [options...] [--] filename\n"
751 " or nasm -v for version info\n\n"
752 " -t assemble in SciTech TASM compatible mode\n"
753 " -g generate debug information in selected format.\n");
754 printf
755 (" -E (or -e) preprocess only (writes output to stdout by default)\n"
756 " -a don't preprocess (assemble only)\n"
757 " -M generate Makefile dependencies on stdout\n"
758 " -MG d:o, missing files assumed generated\n\n"
759 " -Z<file> redirect error messages to file\n"
760 " -s redirect error messages to stdout\n\n"
761 " -F format select a debugging format\n\n"
762 " -I<path> adds a pathname to the include file path\n");
763 printf
764 (" -O<digit> optimize branch offsets (-O0 disables, default)\n"
765 " -P<file> pre-includes a file\n"
766 " -D<macro>[=<value>] pre-defines a macro\n"
767 " -U<macro> undefines a macro\n"
768 " -X<format> specifies error reporting format (gnu or vc)\n"
769 " -w+foo enables warning foo (equiv. -Wfoo)\n"
770 " -w-foo disable warning foo (equiv. -Wno-foo)\n"
771 "Warnings:\n");
772 for (i = 0; i <= ERR_WARN_MAX; i++)
773 printf(" %-23s %s (default %s)\n",
774 suppressed_names[i], suppressed_what[i],
775 suppressed_global[i] ? "off" : "on");
776 printf
777 ("\nresponse files should contain command line parameters"
778 ", one per line.\n");
779 if (p[2] == 'f') {
780 printf("\nvalid output formats for -f are"
781 " (`*' denotes default):\n");
782 ofmt_list(ofmt, stdout);
783 } else {
784 printf("\nFor a list of valid output formats, use -hf.\n");
785 printf("For a list of debug formats, use -f <form> -y.\n");
787 exit(0); /* never need usage message here */
788 break;
790 case 'y':
791 printf("\nvalid debug formats for '%s' output format are"
792 " ('*' denotes default):\n", ofmt->shortname);
793 dfmt_list(ofmt, stdout);
794 exit(0);
795 break;
797 case 't':
798 tasm_compatible_mode = true;
799 break;
801 case 'v':
803 const char *nasm_version_string =
804 "NASM version " NASM_VER " compiled on " __DATE__
805 #ifdef DEBUG
806 " with -DDEBUG"
807 #endif
809 puts(nasm_version_string);
810 exit(0); /* never need usage message here */
812 break;
814 case 'e': /* preprocess only */
815 case 'E':
816 operating_mode = op_preprocess;
817 break;
819 case 'a': /* assemble only - don't preprocess */
820 preproc = &no_pp;
821 break;
823 case 'W':
824 if (param[0] == 'n' && param[1] == 'o' && param[2] == '-') {
825 suppress = true;
826 param += 3;
827 } else {
828 suppress = false;
830 goto set_warning;
832 case 'w':
833 if (param[0] != '+' && param[0] != '-') {
834 report_error(ERR_NONFATAL | ERR_NOFILE | ERR_USAGE,
835 "invalid option to `-w'");
836 break;
838 suppress = (param[0] == '-');
839 param++;
840 goto set_warning;
841 set_warning:
842 for (i = 0; i <= ERR_WARN_MAX; i++)
843 if (!nasm_stricmp(param, suppressed_names[i]))
844 break;
845 if (i <= ERR_WARN_MAX)
846 suppressed_global[i] = suppress;
847 else if (!nasm_stricmp(param, "all"))
848 for (i = 1; i <= ERR_WARN_MAX; i++)
849 suppressed_global[i] = suppress;
850 else if (!nasm_stricmp(param, "none"))
851 for (i = 1; i <= ERR_WARN_MAX; i++)
852 suppressed_global[i] = !suppress;
853 else
854 report_error(ERR_NONFATAL | ERR_NOFILE | ERR_USAGE,
855 "invalid warning `%s'", param);
856 break;
858 case 'M':
859 switch (p[2]) {
860 case 0:
861 operating_mode = op_depend;
862 break;
863 case 'G':
864 operating_mode = op_depend;
865 depend_missing_ok = true;
866 break;
867 case 'P':
868 depend_emit_phony = true;
869 break;
870 case 'D':
871 depend_file = q;
872 advance = true;
873 break;
874 case 'T':
875 depend_target = q;
876 advance = true;
877 break;
878 case 'Q':
879 depend_target = quote_for_make(q);
880 advance = true;
881 break;
882 default:
883 report_error(ERR_NONFATAL|ERR_NOFILE|ERR_USAGE,
884 "unknown dependency option `-M%c'", p[2]);
885 break;
887 if (advance && (!q || !q[0])) {
888 report_error(ERR_NONFATAL|ERR_NOFILE|ERR_USAGE,
889 "option `-M%c' requires a parameter", p[2]);
890 break;
892 break;
894 case '-':
896 int s;
898 if (p[2] == 0) { /* -- => stop processing options */
899 stopoptions = 1;
900 break;
902 for (s = 0; textopts[s].label; s++) {
903 if (!nasm_stricmp(p + 2, textopts[s].label)) {
904 break;
908 switch (s) {
910 case OPT_PREFIX:
911 case OPT_POSTFIX:
913 if (!q) {
914 report_error(ERR_NONFATAL | ERR_NOFILE |
915 ERR_USAGE,
916 "option `--%s' requires an argument",
917 p + 2);
918 break;
919 } else {
920 advance = 1, param = q;
923 if (s == OPT_PREFIX) {
924 strncpy(lprefix, param, PREFIX_MAX - 1);
925 lprefix[PREFIX_MAX - 1] = 0;
926 break;
928 if (s == OPT_POSTFIX) {
929 strncpy(lpostfix, param, POSTFIX_MAX - 1);
930 lpostfix[POSTFIX_MAX - 1] = 0;
931 break;
933 break;
935 default:
937 report_error(ERR_NONFATAL | ERR_NOFILE | ERR_USAGE,
938 "unrecognised option `--%s'", p + 2);
939 break;
942 break;
945 default:
946 if (!ofmt->setinfo(GI_SWITCH, &p))
947 report_error(ERR_NONFATAL | ERR_NOFILE | ERR_USAGE,
948 "unrecognised option `-%c'", p[1]);
949 break;
951 } else {
952 if (*inname) {
953 report_error(ERR_NONFATAL | ERR_NOFILE | ERR_USAGE,
954 "more than one input file specified");
955 } else {
956 copy_filename(inname, p);
960 return advance;
963 #define ARG_BUF_DELTA 128
965 static void process_respfile(FILE * rfile)
967 char *buffer, *p, *q, *prevarg;
968 int bufsize, prevargsize;
970 bufsize = prevargsize = ARG_BUF_DELTA;
971 buffer = nasm_malloc(ARG_BUF_DELTA);
972 prevarg = nasm_malloc(ARG_BUF_DELTA);
973 prevarg[0] = '\0';
975 while (1) { /* Loop to handle all lines in file */
976 p = buffer;
977 while (1) { /* Loop to handle long lines */
978 q = fgets(p, bufsize - (p - buffer), rfile);
979 if (!q)
980 break;
981 p += strlen(p);
982 if (p > buffer && p[-1] == '\n')
983 break;
984 if (p - buffer > bufsize - 10) {
985 int offset;
986 offset = p - buffer;
987 bufsize += ARG_BUF_DELTA;
988 buffer = nasm_realloc(buffer, bufsize);
989 p = buffer + offset;
993 if (!q && p == buffer) {
994 if (prevarg[0])
995 process_arg(prevarg, NULL);
996 nasm_free(buffer);
997 nasm_free(prevarg);
998 return;
1002 * Play safe: remove CRs, LFs and any spurious ^Zs, if any of
1003 * them are present at the end of the line.
1005 *(p = &buffer[strcspn(buffer, "\r\n\032")]) = '\0';
1007 while (p > buffer && nasm_isspace(p[-1]))
1008 *--p = '\0';
1010 p = buffer;
1011 while (nasm_isspace(*p))
1012 p++;
1014 if (process_arg(prevarg, p))
1015 *p = '\0';
1017 if ((int) strlen(p) > prevargsize - 10) {
1018 prevargsize += ARG_BUF_DELTA;
1019 prevarg = nasm_realloc(prevarg, prevargsize);
1021 strncpy(prevarg, p, prevargsize);
1025 /* Function to process args from a string of args, rather than the
1026 * argv array. Used by the environment variable and response file
1027 * processing.
1029 static void process_args(char *args)
1031 char *p, *q, *arg, *prevarg;
1032 char separator = ' ';
1034 p = args;
1035 if (*p && *p != '-')
1036 separator = *p++;
1037 arg = NULL;
1038 while (*p) {
1039 q = p;
1040 while (*p && *p != separator)
1041 p++;
1042 while (*p == separator)
1043 *p++ = '\0';
1044 prevarg = arg;
1045 arg = q;
1046 if (process_arg(prevarg, arg))
1047 arg = NULL;
1049 if (arg)
1050 process_arg(arg, NULL);
1053 static void process_response_file(const char *file)
1055 char str[2048];
1056 FILE *f = fopen(file, "r");
1057 if (!f) {
1058 perror(file);
1059 exit(-1);
1061 while (fgets(str, sizeof str, f)) {
1062 process_args(str);
1064 fclose(f);
1067 static void parse_cmdline(int argc, char **argv)
1069 FILE *rfile;
1070 char *envreal, *envcopy = NULL, *p, *arg;
1072 *inname = *outname = *listname = *errname = '\0';
1075 * First, process the NASMENV environment variable.
1077 envreal = getenv("NASMENV");
1078 arg = NULL;
1079 if (envreal) {
1080 envcopy = nasm_strdup(envreal);
1081 process_args(envcopy);
1082 nasm_free(envcopy);
1086 * Now process the actual command line.
1088 while (--argc) {
1089 bool advance;
1090 argv++;
1091 if (argv[0][0] == '@') {
1092 /* We have a response file, so process this as a set of
1093 * arguments like the environment variable. This allows us
1094 * to have multiple arguments on a single line, which is
1095 * different to the -@resp file processing below for regular
1096 * NASM.
1098 process_response_file(argv[0]+1);
1099 argc--;
1100 argv++;
1102 if (!stopoptions && argv[0][0] == '-' && argv[0][1] == '@') {
1103 p = get_param(argv[0], argc > 1 ? argv[1] : NULL, &advance);
1104 if (p) {
1105 rfile = fopen(p, "r");
1106 if (rfile) {
1107 process_respfile(rfile);
1108 fclose(rfile);
1109 } else
1110 report_error(ERR_NONFATAL | ERR_NOFILE | ERR_USAGE,
1111 "unable to open response file `%s'", p);
1113 } else
1114 advance = process_arg(argv[0], argc > 1 ? argv[1] : NULL);
1115 argv += advance, argc -= advance;
1118 /* Look for basic command line typos. This definitely doesn't
1119 catch all errors, but it might help cases of fumbled fingers. */
1120 if (!*inname)
1121 report_error(ERR_NONFATAL | ERR_NOFILE | ERR_USAGE,
1122 "no input file specified");
1123 else if (!strcmp(inname, errname) ||
1124 !strcmp(inname, outname) ||
1125 !strcmp(inname, listname) ||
1126 (depend_file && !strcmp(inname, depend_file)))
1127 report_error(ERR_FATAL | ERR_NOFILE | ERR_USAGE,
1128 "file `%s' is both input and output file",
1129 inname);
1131 if (*errname) {
1132 error_file = fopen(errname, "w");
1133 if (!error_file) {
1134 error_file = stderr; /* Revert to default! */
1135 report_error(ERR_FATAL | ERR_NOFILE | ERR_USAGE,
1136 "cannot open file `%s' for error messages",
1137 errname);
1142 /* List of directives */
1143 enum directives {
1144 D_NONE, D_ABSOLUTE, D_BITS, D_COMMON, D_CPU, D_DEBUG, D_DEFAULT,
1145 D_EXTERN, D_FLOAT, D_GLOBAL, D_LIST, D_SECTION, D_SEGMENT, D_WARNING
1147 static const char *directives[] = {
1148 "", "absolute", "bits", "common", "cpu", "debug", "default",
1149 "extern", "float", "global", "list", "section", "segment", "warning"
1151 static enum directives getkw(char **directive, char **value);
1153 static void assemble_file(char *fname, StrList **depend_ptr)
1155 char *directive, *value, *p, *q, *special, *line, debugid[80];
1156 insn output_ins;
1157 int i, validid;
1158 bool rn_error;
1159 int32_t seg;
1160 int64_t offs;
1161 struct tokenval tokval;
1162 expr *e;
1163 int pass_max;
1165 if (cmd_sb == 32 && cmd_cpu < IF_386)
1166 report_error(ERR_FATAL, "command line: "
1167 "32-bit segment size requires a higher cpu");
1169 pass_max = 1000; /* Always terminate in a reasonable time */
1170 /* No real program should need this many passes */
1172 for (passn = 1; pass0 <= 2; passn++) {
1173 int pass1, pass2;
1174 ldfunc def_label;
1176 pass1 = pass0 == 2 ? 2 : 1; /* 1, 1, 1, ..., 1, 2 */
1177 pass2 = passn > 1 ? 2 : 1; /* 1, 2, 2, ..., 2, 2 */
1178 /* pass0 0, 0, 0, ..., 1, 2 */
1180 def_label = passn > 1 ? redefine_label : define_label;
1182 globalbits = sb = cmd_sb; /* set 'bits' to command line default */
1183 cpu = cmd_cpu;
1184 if (pass0 == 2) {
1185 if (*listname)
1186 nasmlist.init(listname, report_error);
1188 in_abs_seg = false;
1189 global_offset_changed = false; /* set by redefine_label */
1190 location.segment = ofmt->section(NULL, pass2, &sb);
1191 globalbits = sb;
1192 if (passn > 1) {
1193 saa_rewind(forwrefs);
1194 forwref = saa_rstruct(forwrefs);
1195 raa_free(offsets);
1196 offsets = raa_init();
1198 preproc->reset(fname, pass1, report_error, evaluate, &nasmlist,
1199 pass1 == 2 ? depend_ptr : NULL);
1200 memcpy(suppressed, suppressed_global, (ERR_WARN_MAX+1) * sizeof(bool));
1202 globallineno = 0;
1203 if (passn == 1)
1204 location.known = true;
1205 location.offset = offs = GET_CURR_OFFS;
1207 while ((line = preproc->getline())) {
1208 enum directives d;
1209 globallineno++;
1211 /* here we parse our directives; this is not handled by the 'real'
1212 * parser. */
1213 directive = line;
1214 d = getkw(&directive, &value);
1215 if (d) {
1216 int err = 0;
1218 switch (d) {
1219 case D_SEGMENT: /* [SEGMENT n] */
1220 case D_SECTION:
1221 seg = ofmt->section(value, pass2, &sb);
1222 if (seg == NO_SEG) {
1223 report_error(pass1 == 1 ? ERR_NONFATAL : ERR_PANIC,
1224 "segment name `%s' not recognized",
1225 value);
1226 } else {
1227 in_abs_seg = false;
1228 location.segment = seg;
1230 break;
1231 case D_EXTERN: /* [EXTERN label:special] */
1232 if (*value == '$')
1233 value++; /* skip initial $ if present */
1234 if (pass0 == 2) {
1235 q = value;
1236 while (*q && *q != ':')
1237 q++;
1238 if (*q == ':') {
1239 *q++ = '\0';
1240 ofmt->symdef(value, 0L, 0L, 3, q);
1242 } else if (passn == 1) {
1243 q = value;
1244 validid = true;
1245 if (!isidstart(*q))
1246 validid = false;
1247 while (*q && *q != ':') {
1248 if (!isidchar(*q))
1249 validid = false;
1250 q++;
1252 if (!validid) {
1253 report_error(ERR_NONFATAL,
1254 "identifier expected after EXTERN");
1255 break;
1257 if (*q == ':') {
1258 *q++ = '\0';
1259 special = q;
1260 } else
1261 special = NULL;
1262 if (!is_extern(value)) { /* allow re-EXTERN to be ignored */
1263 int temp = pass0;
1264 pass0 = 1; /* fake pass 1 in labels.c */
1265 declare_as_global(value, special,
1266 report_error);
1267 define_label(value, seg_alloc(), 0L, NULL,
1268 false, true, ofmt, report_error);
1269 pass0 = temp;
1271 } /* else pass0 == 1 */
1272 break;
1273 case D_BITS: /* [BITS bits] */
1274 globalbits = sb = get_bits(value);
1275 break;
1276 case D_GLOBAL: /* [GLOBAL symbol:special] */
1277 if (*value == '$')
1278 value++; /* skip initial $ if present */
1279 if (pass0 == 2) { /* pass 2 */
1280 q = value;
1281 while (*q && *q != ':')
1282 q++;
1283 if (*q == ':') {
1284 *q++ = '\0';
1285 ofmt->symdef(value, 0L, 0L, 3, q);
1287 } else if (pass2 == 1) { /* pass == 1 */
1288 q = value;
1289 validid = true;
1290 if (!isidstart(*q))
1291 validid = false;
1292 while (*q && *q != ':') {
1293 if (!isidchar(*q))
1294 validid = false;
1295 q++;
1297 if (!validid) {
1298 report_error(ERR_NONFATAL,
1299 "identifier expected after GLOBAL");
1300 break;
1302 if (*q == ':') {
1303 *q++ = '\0';
1304 special = q;
1305 } else
1306 special = NULL;
1307 declare_as_global(value, special, report_error);
1308 } /* pass == 1 */
1309 break;
1310 case D_COMMON: /* [COMMON symbol size:special] */
1311 if (*value == '$')
1312 value++; /* skip initial $ if present */
1313 if (pass0 == 1) {
1314 p = value;
1315 validid = true;
1316 if (!isidstart(*p))
1317 validid = false;
1318 while (*p && !nasm_isspace(*p)) {
1319 if (!isidchar(*p))
1320 validid = false;
1321 p++;
1323 if (!validid) {
1324 report_error(ERR_NONFATAL,
1325 "identifier expected after COMMON");
1326 break;
1328 if (*p) {
1329 int64_t size;
1331 while (*p && nasm_isspace(*p))
1332 *p++ = '\0';
1333 q = p;
1334 while (*q && *q != ':')
1335 q++;
1336 if (*q == ':') {
1337 *q++ = '\0';
1338 special = q;
1339 } else
1340 special = NULL;
1341 size = readnum(p, &rn_error);
1342 if (rn_error)
1343 report_error(ERR_NONFATAL,
1344 "invalid size specified"
1345 " in COMMON declaration");
1346 else
1347 define_common(value, seg_alloc(), size,
1348 special, ofmt, report_error);
1349 } else
1350 report_error(ERR_NONFATAL,
1351 "no size specified in"
1352 " COMMON declaration");
1353 } else if (pass0 == 2) { /* pass == 2 */
1354 q = value;
1355 while (*q && *q != ':') {
1356 if (nasm_isspace(*q))
1357 *q = '\0';
1358 q++;
1360 if (*q == ':') {
1361 *q++ = '\0';
1362 ofmt->symdef(value, 0L, 0L, 3, q);
1365 break;
1366 case D_ABSOLUTE: /* [ABSOLUTE address] */
1367 stdscan_reset();
1368 stdscan_bufptr = value;
1369 tokval.t_type = TOKEN_INVALID;
1370 e = evaluate(stdscan, NULL, &tokval, NULL, pass2,
1371 report_error, NULL);
1372 if (e) {
1373 if (!is_reloc(e))
1374 report_error(pass0 ==
1375 1 ? ERR_NONFATAL : ERR_PANIC,
1376 "cannot use non-relocatable expression as "
1377 "ABSOLUTE address");
1378 else {
1379 abs_seg = reloc_seg(e);
1380 abs_offset = reloc_value(e);
1382 } else if (passn == 1)
1383 abs_offset = 0x100; /* don't go near zero in case of / */
1384 else
1385 report_error(ERR_PANIC, "invalid ABSOLUTE address "
1386 "in pass two");
1387 in_abs_seg = true;
1388 location.segment = NO_SEG;
1389 break;
1390 case D_DEBUG: /* [DEBUG] */
1391 p = value;
1392 q = debugid;
1393 validid = true;
1394 if (!isidstart(*p))
1395 validid = false;
1396 while (*p && !nasm_isspace(*p)) {
1397 if (!isidchar(*p))
1398 validid = false;
1399 *q++ = *p++;
1401 *q++ = 0;
1402 if (!validid) {
1403 report_error(passn == 1 ? ERR_NONFATAL : ERR_PANIC,
1404 "identifier expected after DEBUG");
1405 break;
1407 while (*p && nasm_isspace(*p))
1408 p++;
1409 if (pass0 == 2)
1410 ofmt->current_dfmt->debug_directive(debugid, p);
1411 break;
1412 case D_WARNING: /* [WARNING {+|-|*}warn-name] */
1413 if (pass1 == 1) {
1414 while (*value && nasm_isspace(*value))
1415 value++;
1417 switch(*value) {
1418 case '-': validid = 0; value++; break;
1419 case '+': validid = 1; value++; break;
1420 case '*': validid = 2; value++; break;
1421 default: /*
1422 * Should this error out?
1423 * I'll keep it so nothing breaks.
1425 validid = 1; break;
1428 for (i = 1; i <= ERR_WARN_MAX; i++)
1429 if (!nasm_stricmp(value, suppressed_names[i]))
1430 break;
1431 if (i <= ERR_WARN_MAX) {
1432 switch(validid) {
1433 case 0: suppressed[i] = true; break;
1434 case 1: suppressed[i] = false; break;
1435 case 2: suppressed[i] = suppressed_global[i];
1436 break;
1439 else
1440 report_error(ERR_NONFATAL,
1441 "invalid warning id in WARNING directive");
1443 break;
1444 case D_CPU: /* [CPU] */
1445 cpu = get_cpu(value);
1446 break;
1447 case D_LIST: /* [LIST {+|-}] */
1448 while (*value && nasm_isspace(*value))
1449 value++;
1451 if (*value == '+') {
1452 user_nolist = 0;
1453 } else {
1454 if (*value == '-') {
1455 user_nolist = 1;
1456 } else {
1457 err = 1;
1460 break;
1461 case D_DEFAULT: /* [DEFAULT] */
1462 stdscan_reset();
1463 stdscan_bufptr = value;
1464 tokval.t_type = TOKEN_INVALID;
1465 if (stdscan(NULL, &tokval) == TOKEN_SPECIAL) {
1466 switch ((int)tokval.t_integer) {
1467 case S_REL:
1468 globalrel = 1;
1469 break;
1470 case S_ABS:
1471 globalrel = 0;
1472 break;
1473 default:
1474 err = 1;
1475 break;
1477 } else {
1478 err = 1;
1480 break;
1481 case D_FLOAT:
1482 if (float_option(value)) {
1483 report_error(pass1 == 1 ? ERR_NONFATAL : ERR_PANIC,
1484 "unknown 'float' directive: %s",
1485 value);
1487 break;
1488 default:
1489 if (!ofmt->directive(directive, value, pass2))
1490 report_error(pass1 == 1 ? ERR_NONFATAL : ERR_PANIC,
1491 "unrecognised directive [%s]",
1492 directive);
1494 if (err) {
1495 report_error(ERR_NONFATAL,
1496 "invalid parameter to [%s] directive",
1497 directive);
1499 } else { /* it isn't a directive */
1501 parse_line(pass1, line, &output_ins,
1502 report_error, evaluate, def_label);
1504 if (optimizing > 0) {
1505 if (forwref != NULL && globallineno == forwref->lineno) {
1506 output_ins.forw_ref = true;
1507 do {
1508 output_ins.oprs[forwref->operand].opflags |=
1509 OPFLAG_FORWARD;
1510 forwref = saa_rstruct(forwrefs);
1511 } while (forwref != NULL
1512 && forwref->lineno == globallineno);
1513 } else
1514 output_ins.forw_ref = false;
1517 if (optimizing > 0) {
1518 if (passn == 1) {
1519 for (i = 0; i < output_ins.operands; i++) {
1520 if (output_ins.oprs[i].
1521 opflags & OPFLAG_FORWARD) {
1522 struct forwrefinfo *fwinf =
1523 (struct forwrefinfo *)
1524 saa_wstruct(forwrefs);
1525 fwinf->lineno = globallineno;
1526 fwinf->operand = i;
1532 /* forw_ref */
1533 if (output_ins.opcode == I_EQU) {
1534 if (pass1 == 1) {
1536 * Special `..' EQUs get processed in pass two,
1537 * except `..@' macro-processor EQUs which are done
1538 * in the normal place.
1540 if (!output_ins.label)
1541 report_error(ERR_NONFATAL,
1542 "EQU not preceded by label");
1544 else if (output_ins.label[0] != '.' ||
1545 output_ins.label[1] != '.' ||
1546 output_ins.label[2] == '@') {
1547 if (output_ins.operands == 1 &&
1548 (output_ins.oprs[0].type & IMMEDIATE) &&
1549 output_ins.oprs[0].wrt == NO_SEG) {
1550 int isext =
1551 output_ins.oprs[0].
1552 opflags & OPFLAG_EXTERN;
1553 def_label(output_ins.label,
1554 output_ins.oprs[0].segment,
1555 output_ins.oprs[0].offset, NULL,
1556 false, isext, ofmt,
1557 report_error);
1558 } else if (output_ins.operands == 2
1559 && (output_ins.oprs[0].
1560 type & IMMEDIATE)
1561 && (output_ins.oprs[0].type & COLON)
1562 && output_ins.oprs[0].segment ==
1563 NO_SEG
1564 && output_ins.oprs[0].wrt == NO_SEG
1565 && (output_ins.oprs[1].
1566 type & IMMEDIATE)
1567 && output_ins.oprs[1].segment ==
1568 NO_SEG
1569 && output_ins.oprs[1].wrt ==
1570 NO_SEG) {
1571 def_label(output_ins.label,
1572 output_ins.oprs[0].
1573 offset | SEG_ABS,
1574 output_ins.oprs[1].offset, NULL,
1575 false, false, ofmt,
1576 report_error);
1577 } else
1578 report_error(ERR_NONFATAL,
1579 "bad syntax for EQU");
1581 } else {
1583 * Special `..' EQUs get processed here, except
1584 * `..@' macro processor EQUs which are done above.
1586 if (output_ins.label[0] == '.' &&
1587 output_ins.label[1] == '.' &&
1588 output_ins.label[2] != '@') {
1589 if (output_ins.operands == 1 &&
1590 (output_ins.oprs[0].type & IMMEDIATE)) {
1591 define_label(output_ins.label,
1592 output_ins.oprs[0].segment,
1593 output_ins.oprs[0].offset,
1594 NULL, false, false, ofmt,
1595 report_error);
1596 } else if (output_ins.operands == 2
1597 && (output_ins.oprs[0].
1598 type & IMMEDIATE)
1599 && (output_ins.oprs[0].type & COLON)
1600 && output_ins.oprs[0].segment ==
1601 NO_SEG
1602 && (output_ins.oprs[1].
1603 type & IMMEDIATE)
1604 && output_ins.oprs[1].segment ==
1605 NO_SEG) {
1606 define_label(output_ins.label,
1607 output_ins.oprs[0].
1608 offset | SEG_ABS,
1609 output_ins.oprs[1].offset,
1610 NULL, false, false, ofmt,
1611 report_error);
1612 } else
1613 report_error(ERR_NONFATAL,
1614 "bad syntax for EQU");
1617 } else { /* instruction isn't an EQU */
1619 if (pass1 == 1) {
1621 int64_t l = insn_size(location.segment, offs, sb, cpu,
1622 &output_ins, report_error);
1624 /* if (using_debug_info) && output_ins.opcode != -1) */
1625 if (using_debug_info)
1626 { /* fbk 03/25/01 */
1627 /* this is done here so we can do debug type info */
1628 int32_t typeinfo =
1629 TYS_ELEMENTS(output_ins.operands);
1630 switch (output_ins.opcode) {
1631 case I_RESB:
1632 typeinfo =
1633 TYS_ELEMENTS(output_ins.oprs[0].
1634 offset) | TY_BYTE;
1635 break;
1636 case I_RESW:
1637 typeinfo =
1638 TYS_ELEMENTS(output_ins.oprs[0].
1639 offset) | TY_WORD;
1640 break;
1641 case I_RESD:
1642 typeinfo =
1643 TYS_ELEMENTS(output_ins.oprs[0].
1644 offset) | TY_DWORD;
1645 break;
1646 case I_RESQ:
1647 typeinfo =
1648 TYS_ELEMENTS(output_ins.oprs[0].
1649 offset) | TY_QWORD;
1650 break;
1651 case I_REST:
1652 typeinfo =
1653 TYS_ELEMENTS(output_ins.oprs[0].
1654 offset) | TY_TBYTE;
1655 break;
1656 case I_RESO:
1657 typeinfo =
1658 TYS_ELEMENTS(output_ins.oprs[0].
1659 offset) | TY_OWORD;
1660 break;
1661 case I_RESY:
1662 typeinfo =
1663 TYS_ELEMENTS(output_ins.oprs[0].
1664 offset) | TY_YWORD;
1665 break;
1666 case I_DB:
1667 typeinfo |= TY_BYTE;
1668 break;
1669 case I_DW:
1670 typeinfo |= TY_WORD;
1671 break;
1672 case I_DD:
1673 if (output_ins.eops_float)
1674 typeinfo |= TY_FLOAT;
1675 else
1676 typeinfo |= TY_DWORD;
1677 break;
1678 case I_DQ:
1679 typeinfo |= TY_QWORD;
1680 break;
1681 case I_DT:
1682 typeinfo |= TY_TBYTE;
1683 break;
1684 case I_DO:
1685 typeinfo |= TY_OWORD;
1686 break;
1687 case I_DY:
1688 typeinfo |= TY_YWORD;
1689 break;
1690 default:
1691 typeinfo = TY_LABEL;
1695 ofmt->current_dfmt->debug_typevalue(typeinfo);
1698 if (l != -1) {
1699 offs += l;
1700 SET_CURR_OFFS(offs);
1703 * else l == -1 => invalid instruction, which will be
1704 * flagged as an error on pass 2
1707 } else {
1708 offs += assemble(location.segment, offs, sb, cpu,
1709 &output_ins, ofmt, report_error,
1710 &nasmlist);
1711 SET_CURR_OFFS(offs);
1714 } /* not an EQU */
1715 cleanup_insn(&output_ins);
1717 nasm_free(line);
1718 location.offset = offs = GET_CURR_OFFS;
1719 } /* end while (line = preproc->getline... */
1721 if (pass1 == 2 && global_offset_changed)
1722 report_error(ERR_NONFATAL,
1723 "phase error detected at end of assembly.");
1725 if (pass1 == 1)
1726 preproc->cleanup(1);
1728 if (pass1 == 1 && terminate_after_phase) {
1729 fclose(ofile);
1730 remove(outname);
1731 if (want_usage)
1732 usage();
1733 exit(1);
1735 if (passn > 1 && !global_offset_changed)
1736 pass0++;
1738 if(passn >= pass_max)
1739 /* We get here if the labels don't converge
1740 * Example: FOO equ FOO + 1
1742 report_error(ERR_NONFATAL,
1743 "Can't find valid values for all labels "
1744 "after %d passes, giving up. "
1745 "Possible cause: recursive equ's.", passn);
1748 preproc->cleanup(0);
1749 nasmlist.cleanup();
1750 #if 1
1751 if (opt_verbose_info) /* -On and -Ov switches */
1752 fprintf(stdout,
1753 "info:: assembly required 1+%d+1 passes\n", passn-3);
1754 #endif
1755 } /* exit from assemble_file (...) */
1757 static enum directives getkw(char **directive, char **value)
1759 char *p, *q, *buf;
1761 buf = *directive;
1763 /* allow leading spaces or tabs */
1764 while (*buf == ' ' || *buf == '\t')
1765 buf++;
1767 if (*buf != '[')
1768 return 0;
1770 p = buf;
1772 while (*p && *p != ']')
1773 p++;
1775 if (!*p)
1776 return 0;
1778 q = p++;
1780 while (*p && *p != ';') {
1781 if (!nasm_isspace(*p))
1782 return 0;
1783 p++;
1785 q[1] = '\0';
1787 *directive = p = buf + 1;
1788 while (*buf && *buf != ' ' && *buf != ']' && *buf != '\t')
1789 buf++;
1790 if (*buf == ']') {
1791 *buf = '\0';
1792 *value = buf;
1793 } else {
1794 *buf++ = '\0';
1795 while (nasm_isspace(*buf))
1796 buf++; /* beppu - skip leading whitespace */
1797 *value = buf;
1798 while (*buf != ']')
1799 buf++;
1800 *buf++ = '\0';
1803 return bsii(*directive, directives, elements(directives));
1807 * gnu style error reporting
1808 * This function prints an error message to error_file in the
1809 * style used by GNU. An example would be:
1810 * file.asm:50: error: blah blah blah
1811 * where file.asm is the name of the file, 50 is the line number on
1812 * which the error occurs (or is detected) and "error:" is one of
1813 * the possible optional diagnostics -- it can be "error" or "warning"
1814 * or something else. Finally the line terminates with the actual
1815 * error message.
1817 * @param severity the severity of the warning or error
1818 * @param fmt the printf style format string
1820 static void report_error_gnu(int severity, const char *fmt, ...)
1822 va_list ap;
1824 if (is_suppressed_warning(severity))
1825 return;
1827 if (severity & ERR_NOFILE)
1828 fputs("nasm: ", error_file);
1829 else {
1830 char *currentfile = NULL;
1831 int32_t lineno = 0;
1832 src_get(&lineno, &currentfile);
1833 fprintf(error_file, "%s:%"PRId32": ", currentfile, lineno);
1834 nasm_free(currentfile);
1836 va_start(ap, fmt);
1837 report_error_common(severity, fmt, ap);
1838 va_end(ap);
1842 * MS style error reporting
1843 * This function prints an error message to error_file in the
1844 * style used by Visual C and some other Microsoft tools. An example
1845 * would be:
1846 * file.asm(50) : error: blah blah blah
1847 * where file.asm is the name of the file, 50 is the line number on
1848 * which the error occurs (or is detected) and "error:" is one of
1849 * the possible optional diagnostics -- it can be "error" or "warning"
1850 * or something else. Finally the line terminates with the actual
1851 * error message.
1853 * @param severity the severity of the warning or error
1854 * @param fmt the printf style format string
1856 static void report_error_vc(int severity, const char *fmt, ...)
1858 va_list ap;
1860 if (is_suppressed_warning(severity))
1861 return;
1863 if (severity & ERR_NOFILE)
1864 fputs("nasm: ", error_file);
1865 else {
1866 char *currentfile = NULL;
1867 int32_t lineno = 0;
1868 src_get(&lineno, &currentfile);
1869 fprintf(error_file, "%s(%"PRId32") : ", currentfile, lineno);
1870 nasm_free(currentfile);
1872 va_start(ap, fmt);
1873 report_error_common(severity, fmt, ap);
1874 va_end(ap);
1878 * check for supressed warning
1879 * checks for suppressed warning or pass one only warning and we're
1880 * not in pass 1
1882 * @param severity the severity of the warning or error
1883 * @return true if we should abort error/warning printing
1885 static bool is_suppressed_warning(int severity)
1888 * See if it's a suppressed warning.
1890 return (severity & ERR_MASK) == ERR_WARNING &&
1891 (((severity & ERR_WARN_MASK) != 0 &&
1892 suppressed[(severity & ERR_WARN_MASK) >> ERR_WARN_SHR]) ||
1893 /* See if it's a pass-one only warning and we're not in pass one. */
1894 ((severity & ERR_PASS1) && pass0 != 1));
1898 * common error reporting
1899 * This is the common back end of the error reporting schemes currently
1900 * implemented. It prints the nature of the warning and then the
1901 * specific error message to error_file and may or may not return. It
1902 * doesn't return if the error severity is a "panic" or "debug" type.
1904 * @param severity the severity of the warning or error
1905 * @param fmt the printf style format string
1907 static void report_error_common(int severity, const char *fmt,
1908 va_list args)
1910 switch (severity & (ERR_MASK|ERR_NO_SEVERITY)) {
1911 case ERR_WARNING:
1912 fputs("warning: ", error_file);
1913 break;
1914 case ERR_NONFATAL:
1915 fputs("error: ", error_file);
1916 break;
1917 case ERR_FATAL:
1918 fputs("fatal: ", error_file);
1919 break;
1920 case ERR_PANIC:
1921 fputs("panic: ", error_file);
1922 break;
1923 case ERR_DEBUG:
1924 fputs("debug: ", error_file);
1925 break;
1926 default:
1927 break;
1930 vfprintf(error_file, fmt, args);
1931 putc('\n', error_file);
1933 if (severity & ERR_USAGE)
1934 want_usage = true;
1936 switch (severity & ERR_MASK) {
1937 case ERR_DEBUG:
1938 /* no further action, by definition */
1939 break;
1940 case ERR_WARNING:
1941 if (!suppressed[0]) /* Treat warnings as errors */
1942 terminate_after_phase = true;
1943 break;
1944 case ERR_NONFATAL:
1945 terminate_after_phase = true;
1946 break;
1947 case ERR_FATAL:
1948 if (ofile) {
1949 fclose(ofile);
1950 remove(outname);
1952 if (want_usage)
1953 usage();
1954 exit(1); /* instantly die */
1955 break; /* placate silly compilers */
1956 case ERR_PANIC:
1957 fflush(NULL);
1958 /* abort(); *//* halt, catch fire, and dump core */
1959 exit(3);
1960 break;
1964 static void usage(void)
1966 fputs("type `nasm -h' for help\n", error_file);
1969 static void register_output_formats(void)
1971 ofmt = ofmt_register(report_error);
1974 #define BUF_DELTA 512
1976 static FILE *no_pp_fp;
1977 static efunc no_pp_err;
1978 static ListGen *no_pp_list;
1979 static int32_t no_pp_lineinc;
1981 static void no_pp_reset(char *file, int pass, efunc error, evalfunc eval,
1982 ListGen * listgen, StrList **deplist)
1984 src_set_fname(nasm_strdup(file));
1985 src_set_linnum(0);
1986 no_pp_lineinc = 1;
1987 no_pp_err = error;
1988 no_pp_fp = fopen(file, "r");
1989 if (!no_pp_fp)
1990 no_pp_err(ERR_FATAL | ERR_NOFILE,
1991 "unable to open input file `%s'", file);
1992 no_pp_list = listgen;
1993 (void)pass; /* placate compilers */
1994 (void)eval; /* placate compilers */
1996 if (deplist) {
1997 StrList *sl = nasm_malloc(strlen(file)+1+sizeof sl->next);
1998 sl->next = NULL;
1999 strcpy(sl->str, file);
2000 *deplist = sl;
2004 static char *no_pp_getline(void)
2006 char *buffer, *p, *q;
2007 int bufsize;
2009 bufsize = BUF_DELTA;
2010 buffer = nasm_malloc(BUF_DELTA);
2011 src_set_linnum(src_get_linnum() + no_pp_lineinc);
2013 while (1) { /* Loop to handle %line */
2015 p = buffer;
2016 while (1) { /* Loop to handle long lines */
2017 q = fgets(p, bufsize - (p - buffer), no_pp_fp);
2018 if (!q)
2019 break;
2020 p += strlen(p);
2021 if (p > buffer && p[-1] == '\n')
2022 break;
2023 if (p - buffer > bufsize - 10) {
2024 int offset;
2025 offset = p - buffer;
2026 bufsize += BUF_DELTA;
2027 buffer = nasm_realloc(buffer, bufsize);
2028 p = buffer + offset;
2032 if (!q && p == buffer) {
2033 nasm_free(buffer);
2034 return NULL;
2038 * Play safe: remove CRs, LFs and any spurious ^Zs, if any of
2039 * them are present at the end of the line.
2041 buffer[strcspn(buffer, "\r\n\032")] = '\0';
2043 if (!nasm_strnicmp(buffer, "%line", 5)) {
2044 int32_t ln;
2045 int li;
2046 char *nm = nasm_malloc(strlen(buffer));
2047 if (sscanf(buffer + 5, "%"PRId32"+%d %s", &ln, &li, nm) == 3) {
2048 nasm_free(src_set_fname(nm));
2049 src_set_linnum(ln);
2050 no_pp_lineinc = li;
2051 continue;
2053 nasm_free(nm);
2055 break;
2058 no_pp_list->line(LIST_READ, buffer);
2060 return buffer;
2063 static void no_pp_cleanup(int pass)
2065 (void)pass; /* placate GCC */
2066 fclose(no_pp_fp);
2069 static uint32_t get_cpu(char *value)
2071 if (!strcmp(value, "8086"))
2072 return IF_8086;
2073 if (!strcmp(value, "186"))
2074 return IF_186;
2075 if (!strcmp(value, "286"))
2076 return IF_286;
2077 if (!strcmp(value, "386"))
2078 return IF_386;
2079 if (!strcmp(value, "486"))
2080 return IF_486;
2081 if (!strcmp(value, "586") || !nasm_stricmp(value, "pentium"))
2082 return IF_PENT;
2083 if (!strcmp(value, "686") ||
2084 !nasm_stricmp(value, "ppro") ||
2085 !nasm_stricmp(value, "pentiumpro") || !nasm_stricmp(value, "p2"))
2086 return IF_P6;
2087 if (!nasm_stricmp(value, "p3") || !nasm_stricmp(value, "katmai"))
2088 return IF_KATMAI;
2089 if (!nasm_stricmp(value, "p4") || /* is this right? -- jrc */
2090 !nasm_stricmp(value, "willamette"))
2091 return IF_WILLAMETTE;
2092 if (!nasm_stricmp(value, "prescott"))
2093 return IF_PRESCOTT;
2094 if (!nasm_stricmp(value, "x64") ||
2095 !nasm_stricmp(value, "x86-64"))
2096 return IF_X86_64;
2097 if (!nasm_stricmp(value, "ia64") ||
2098 !nasm_stricmp(value, "ia-64") ||
2099 !nasm_stricmp(value, "itanium") ||
2100 !nasm_stricmp(value, "itanic") || !nasm_stricmp(value, "merced"))
2101 return IF_IA64;
2103 report_error(pass0 < 2 ? ERR_NONFATAL : ERR_FATAL,
2104 "unknown 'cpu' type");
2106 return IF_PLEVEL; /* the maximum level */
2109 static int get_bits(char *value)
2111 int i;
2113 if ((i = atoi(value)) == 16)
2114 return i; /* set for a 16-bit segment */
2115 else if (i == 32) {
2116 if (cpu < IF_386) {
2117 report_error(ERR_NONFATAL,
2118 "cannot specify 32-bit segment on processor below a 386");
2119 i = 16;
2121 } else if (i == 64) {
2122 if (cpu < IF_X86_64) {
2123 report_error(ERR_NONFATAL,
2124 "cannot specify 64-bit segment on processor below an x86-64");
2125 i = 16;
2127 if (i != maxbits) {
2128 report_error(ERR_NONFATAL,
2129 "%s output format does not support 64-bit code",
2130 ofmt->shortname);
2131 i = 16;
2133 } else {
2134 report_error(pass0 < 2 ? ERR_NONFATAL : ERR_FATAL,
2135 "`%s' is not a valid segment size; must be 16, 32 or 64",
2136 value);
2137 i = 16;
2139 return i;
2142 /* end of nasm.c */