/*****************************************************************************/
/* */
-/* main.c */
+/* main.c */
/* */
-/* Main program for the da65 disassembler */
+/* Main program for the da65 disassembler */
/* */
/* */
/* */
-/* (C) 1998-2000 Ullrich von Bassewitz */
-/* Wacholderweg 14 */
-/* D-70597 Stuttgart */
-/* EMail: uz@musoftware.de */
+/* (C) 1998-2014, Ullrich von Bassewitz */
+/* Roemerstrasse 52 */
+/* D-70794 Filderstadt */
+/* EMail: uz@cc65.org */
/* */
/* */
/* This software is provided 'as-is', without any expressed or implied */
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
+#include <limits.h>
#include <time.h>
/* common */
#include "abend.h"
#include "cmdline.h"
+#include "cpu.h"
#include "fname.h"
#include "print.h"
#include "version.h"
/* da65 */
#include "attrtab.h"
#include "code.h"
-#include "config.h"
-#include "cpu.h"
+#include "comments.h"
#include "data.h"
#include "error.h"
#include "global.h"
+#include "infofile.h"
+#include "labels.h"
#include "opctable.h"
#include "output.h"
#include "scanner.h"
+#include "segment.h"
/*****************************************************************************/
-/* Code */
+/* Code */
/*****************************************************************************/
static void Usage (void)
/* Print usage information and exit */
{
- fprintf (stderr,
- "Usage: %s [options] file\n"
- "Short options:\n"
- " -g\t\t\tAdd debug info to object file\n"
- " -h\t\t\tHelp (this text)\n"
- " -o name\t\tName the output file\n"
- " -v\t\t\tIncrease verbosity\n"
- " -F\t\t\tAdd formfeeds to the output\n"
- " -S addr\t\tSet the start/load address\n"
- " -V\t\t\tPrint the disassembler version\n"
- "\n"
- "Long options:\n"
- " --cpu type\t\tSet cpu type\n"
- " --formfeeds\t\tAdd formfeeds to the output\n"
- " --help\t\tHelp (this text)\n"
- " --pagelength n\tSet the page length for the listing\n"
- " --start-addr addr\tSet the start/load address\n"
- " --verbose\t\tIncrease verbosity\n"
- " --version\t\tPrint the disassembler version\n",
- ProgName);
+ printf ("Usage: %s [options] [inputfile]\n"
+ "Short options:\n"
+ " -g\t\t\tAdd debug info to object file\n"
+ " -h\t\t\tHelp (this text)\n"
+ " -i name\t\tSpecify an info file\n"
+ " -o name\t\tName the output file\n"
+ " -v\t\t\tIncrease verbosity\n"
+ " -F\t\t\tAdd formfeeds to the output\n"
+ " -S addr\t\tSet the start/load address\n"
+ " -V\t\t\tPrint the disassembler version\n"
+ "\n"
+ "Long options:\n"
+ " --argument-column n\tSpecify argument start column\n"
+ " --comment-column n\tSpecify comment start column\n"
+ " --comments n\t\tSet the comment level for the output\n"
+ " --cpu type\t\tSet cpu type\n"
+ " --debug-info\t\tAdd debug info to object file\n"
+ " --formfeeds\t\tAdd formfeeds to the output\n"
+ " --help\t\tHelp (this text)\n"
+ " --hexoffs\t\tUse hexadecimal label offsets\n"
+ " --info name\t\tSpecify an info file\n"
+ " --label-break n\tAdd newline if label exceeds length n\n"
+ " --mnemonic-column n\tSpecify mnemonic start column\n"
+ " --pagelength n\tSet the page length for the listing\n"
+ " --start-addr addr\tSet the start/load address\n"
+ " --text-column n\tSpecify text start column\n"
+ " --verbose\t\tIncrease verbosity\n"
+ " --version\t\tPrint the disassembler version\n",
+ ProgName);
+}
+
+
+
+static void RangeCheck (const char* Opt, unsigned long Val,
+ unsigned long Min, unsigned long Max)
+/* Do a range check for the given option and abort if there's a range
+** error.
+*/
+{
+ if (Val < Min || Val > Max) {
+ Error ("Argument for %s outside valid range (%ld-%ld)", Opt, Min, Max);
+ }
}
static unsigned long CvtNumber (const char* Arg, const char* Number)
/* Convert a number from a string. Allow '$' and '0x' prefixes for hex
- * numbers.
- */
+** numbers.
+*/
{
unsigned long Val;
- int Converted;
+ int Converted;
+ char BoundsCheck;
/* Convert */
if (*Number == '$') {
- ++Number;
- Converted = sscanf (Number, "%lx", &Val);
+ ++Number;
+ Converted = sscanf (Number, "%lx%c", &Val, &BoundsCheck);
} else {
- Converted = sscanf (Number, "%li", (long*)&Val);
+ Converted = sscanf (Number, "%li%c", (long*)&Val, &BoundsCheck);
}
/* Check if we do really have a number */
if (Converted != 1) {
- Error ("Invalid number given in argument: %s\n", Arg);
+ Error ("Invalid number given in argument: %s\n", Arg);
}
/* Return the result */
-static void OptCPU (const char* Opt, const char* Arg)
+static void OptArgumentColumn (const char* Opt, const char* Arg)
+/* Handle the --argument-column option */
+{
+ /* Convert the argument to a number */
+ unsigned long Val = CvtNumber (Opt, Arg);
+
+ /* Check for a valid range */
+ RangeCheck (Opt, Val, MIN_ACOL, MAX_ACOL);
+
+ /* Use the value */
+ ACol = (unsigned char) Val;
+}
+
+
+
+static void OptBytesPerLine (const char* Opt, const char* Arg)
+/* Handle the --bytes-per-line option */
+{
+ /* Convert the argument to a number */
+ unsigned long Val = CvtNumber (Opt, Arg);
+
+ /* Check for a valid range */
+ RangeCheck (Opt, Val, MIN_BYTESPERLINE, MAX_BYTESPERLINE);
+
+ /* Use the value */
+ BytesPerLine = (unsigned char) Val;
+}
+
+
+
+static void OptCommentColumn (const char* Opt, const char* Arg)
+/* Handle the --comment-column option */
+{
+ /* Convert the argument to a number */
+ unsigned long Val = CvtNumber (Opt, Arg);
+
+ /* Check for a valid range */
+ RangeCheck (Opt, Val, MIN_CCOL, MAX_CCOL);
+
+ /* Use the value */
+ CCol = (unsigned char) Val;
+}
+
+
+
+static void OptComments (const char* Opt, const char* Arg)
+/* Handle the --comments option */
+{
+ /* Convert the argument to a number */
+ unsigned long Val = CvtNumber (Opt, Arg);
+
+ /* Check for a valid range */
+ RangeCheck (Opt, Val, MIN_COMMENTS, MAX_COMMENTS);
+
+ /* Use the value */
+ Comments = (unsigned char) Val;
+}
+
+
+
+static void OptCPU (const char* Opt attribute ((unused)), const char* Arg)
/* Handle the --cpu option */
{
- if (Arg == 0) {
- NeedArg (Opt);
- }
- if (strcmp (Arg, "6502") == 0) {
- SetCPU (CPU_6502);
- } else if (strcmp (Arg, "65C02") == 0) {
- SetCPU (CPU_65C02);
- } else if (strcmp (Arg, "65816") == 0) {
- SetCPU (CPU_65816);
-#ifdef SUNPLUS
- } else if (strcmp (Arg, "sunplus") == 0) {
- SetCPU (CPU_SUNPLUS);
-#endif
- } else {
- AbEnd ("Invalid CPU: `%s'", Arg);
- }
+ /* Find the CPU from the given name */
+ CPU = FindCPU (Arg);
+ SetOpcTable (CPU);
+}
+
+
+
+static void OptDebugInfo (const char* Opt attribute ((unused)),
+ const char* Arg attribute ((unused)))
+/* Add debug info to the object file */
+{
+ DebugInfo = 1;
}
-static void OptFormFeeds (const char* Opt, const char* Arg)
+static void OptFormFeeds (const char* Opt attribute ((unused)),
+ const char* Arg attribute ((unused)))
/* Add form feeds to the output */
{
FormFeeds = 1;
-static void OptHelp (const char* Opt, const char* Arg)
+static void OptHelp (const char* Opt attribute ((unused)),
+ const char* Arg attribute ((unused)))
/* Print usage information and exit */
{
Usage ();
-static void OptPageLength (const char* Opt, const char* Arg)
+static void OptHexOffs (const char* Opt attribute ((unused)),
+ const char* Arg attribute ((unused)))
+/* Handle the --hexoffs option */
+{
+ UseHexOffs = 1;
+}
+
+
+
+static void OptInfo (const char* Opt attribute ((unused)), const char* Arg)
+/* Handle the --info option */
+{
+ InfoSetName (Arg);
+}
+
+
+
+static void OptLabelBreak (const char* Opt, const char* Arg)
+/* Handle the --label-break option */
+{
+ /* Convert the argument to a number */
+ unsigned long Val = CvtNumber (Opt, Arg);
+
+ /* Check for a valid range */
+ RangeCheck (Opt, Val, MIN_LABELBREAK, MAX_LABELBREAK);
+
+ /* Use the value */
+ LBreak = (unsigned char) Val;
+}
+
+
+
+static void OptMnemonicColumn (const char* Opt, const char* Arg)
+/* Handle the --mnemonic-column option */
+{
+ /* Convert the argument to a number */
+ unsigned long Val = CvtNumber (Opt, Arg);
+
+ /* Check for a valid range */
+ RangeCheck (Opt, Val, MIN_MCOL, MAX_MCOL);
+
+ /* Use the value */
+ MCol = (unsigned char) Val;
+}
+
+
+
+static void OptPageLength (const char* Opt attribute ((unused)), const char* Arg)
/* Handle the --pagelength option */
{
- int Len;
- if (Arg == 0) {
- NeedArg (Opt);
- }
- Len = atoi (Arg);
- if (Len != 0 && (Len < MIN_PAGE_LEN || Len > MAX_PAGE_LEN)) {
- AbEnd ("Invalid page length: %d", Len);
+ int Len = atoi (Arg);
+ if (Len != 0) {
+ RangeCheck (Opt, Len, MIN_PAGE_LEN, MAX_PAGE_LEN);
}
PageLength = Len;
}
-static void OptVerbose (const char* Opt, const char* Arg)
+static void OptTextColumn (const char* Opt, const char* Arg)
+/* Handle the --text-column option */
+{
+ /* Convert the argument to a number */
+ unsigned long Val = CvtNumber (Opt, Arg);
+
+ /* Check for a valid range */
+ RangeCheck (Opt, Val, MIN_TCOL, MAX_TCOL);
+
+ /* Use the value */
+ TCol = (unsigned char) Val;
+}
+
+
+
+static void OptVerbose (const char* Opt attribute ((unused)),
+ const char* Arg attribute ((unused)))
/* Increase verbosity */
{
++Verbosity;
-static void OptVersion (const char* Opt, const char* Arg)
+static void OptVersion (const char* Opt attribute ((unused)),
+ const char* Arg attribute ((unused)))
/* Print the disassembler version */
{
- fprintf (stderr,
- "da65 V%u.%u.%u - (C) Copyright 2000 Ullrich von Bassewitz\n",
- VER_MAJOR, VER_MINOR, VER_PATCH);
+ fprintf (stderr, "%s V%s\n", ProgName, GetVersionAsString ());
+ exit(EXIT_SUCCESS);
}
static void OneOpcode (unsigned RemainingBytes)
/* Disassemble one opcode */
{
+ unsigned I;
+
/* Get the opcode from the current address */
unsigned char OPC = GetCodeByte (PC);
/* Get the opcode description for the opcode byte */
const OpcDesc* D = &OpcTable[OPC];
- /* If we have a label at this address, output the label */
- if (MustDefLabel (PC)) {
- DefLabel (GetLabel (PC));
+ /* Get the output style for the current PC */
+ attr_t Style = GetStyleAttr (PC);
+
+ /* If a segment begins here, then name that segment.
+ ** Note that the segment is named even if its code is being skipped,
+ ** because some of its later code might not be skipped.
+ */
+ if (IsSegmentStart (PC)) {
+ StartSegment (GetSegmentStartName (PC), GetSegmentAddrSize (PC));
+ }
+
+ /* If we have a label at this address, output the label and an attached
+ ** comment, provided that we aren't in a skip area.
+ */
+ if (Style != atSkip && MustDefLabel (PC)) {
+ const char* Comment = GetComment (PC);
+ if (Comment) {
+ UserComment (Comment);
+ }
+ DefLabel (GetLabelName (PC));
}
/* Check...
- * - ...if we have enough bytes remaining for the code at this address.
- * - ...if the current instruction is valid for the given CPU.
- * - ...if there is no label somewhere between the instruction bytes.
- * If any of these conditions is true, switch to data mode.
- */
- if (GetStyleAttr (PC) == atDefault) {
- if (D->Size > RemainingBytes) {
- MarkAddr (PC, atIllegal);
- } else if ((D->CPU & CPU) != CPU) {
- MarkAddr (PC, atIllegal);
- } else {
- unsigned I;
- for (I = 1; I < D->Size; ++I) {
- if (HaveLabel (PC+I)) {
- MarkAddr (PC, atIllegal);
- break;
- }
- }
- }
+ ** - ...if we have enough bytes remaining for the code at this address.
+ ** - ...if the current instruction is valid for the given CPU.
+ ** - ...if there is no label somewhere between the instruction bytes.
+ ** - ...if there is no segment change between the instruction bytes.
+ ** If any one of those conditions is false, switch to data mode.
+ */
+ if (Style == atDefault) {
+ if (D->Size > RemainingBytes) {
+ Style = atIllegal;
+ MarkAddr (PC, Style);
+ } else if (D->Flags & flIllegal) {
+ Style = atIllegal;
+ MarkAddr (PC, Style);
+ } else {
+ for (I = PC + D->Size; --I > PC; ) {
+ if (HaveLabel (I) || IsSegmentStart (I)) {
+ Style = atIllegal;
+ MarkAddr (PC, Style);
+ break;
+ }
+ }
+ for (I = 0; I < D->Size - 1u; ++I) {
+ if (IsSegmentEnd (PC + I)) {
+ Style = atIllegal;
+ MarkAddr (PC, Style);
+ break;
+ }
+ }
+ }
}
/* Disassemble the line */
- switch (GetStyleAttr (PC)) {
-
- case atDefault:
- case atCode:
- D->Handler (D);
- PC += D->Size;
- break;
-
- case atByteTab:
- ByteTable ();
- break;
-
- case atWordTab:
- WordTable ();
- break;
-
- case atDWordTab:
- DWordTable ();
- break;
-
- case atAddrTab:
- AddrTable ();
- break;
-
- case atRtsTab:
- RtsTable ();
- break;
-
- case atTextTab:
- TextTable ();
- break;
-
- default:
- DataByteLine (1);
- ++PC;
- break;
+ switch (Style) {
+
+ case atDefault:
+ D->Handler (D);
+ PC += D->Size;
+ break;
+
+ case atCode:
+ /* Beware: If we don't have enough bytes left to disassemble the
+ ** following insn, fall through to byte mode.
+ */
+ if (D->Size <= RemainingBytes) {
+ /* Output labels within the next insn */
+ for (I = 1; I < D->Size; ++I) {
+ ForwardLabel (I);
+ }
+ /* Output the insn */
+ D->Handler (D);
+ PC += D->Size;
+ break;
+ }
+ /* FALLTHROUGH */
+
+ case atByteTab:
+ ByteTable ();
+ break;
+
+ case atDByteTab:
+ DByteTable ();
+ break;
+
+ case atWordTab:
+ WordTable ();
+ break;
+
+ case atDWordTab:
+ DWordTable ();
+ break;
+
+ case atAddrTab:
+ AddrTable ();
+ break;
+
+ case atRtsTab:
+ RtsTable ();
+ break;
+
+ case atTextTab:
+ TextTable ();
+ break;
+
+ case atSkip:
+ ++PC;
+ break;
+
+ default:
+ DataByteLine (1);
+ ++PC;
+ break;
+ }
+ /* Change back to the default CODE segment if
+ ** a named segment stops at the current address.
+ */
+ for (I = D->Size; I >= 1; --I) {
+ if (IsSegmentEnd (PC - I)) {
+ EndSegment ();
+ break;
+ }
}
}
/* Disassemble until nothing left */
while ((Count = GetRemainingBytes()) > 0) {
- OneOpcode (Count);
+ OneOpcode (Count);
}
}
/* Pass 2 */
Pass = 2;
ResetCode ();
+ OutputSettings ();
DefOutOfRangeLabels ();
OnePass ();
}
{
/* Program long options */
static const LongOpt OptTab[] = {
- { "--cpu", 1, OptCPU },
- { "--formfeeds", 0, OptFormFeeds },
- { "--help", 0, OptHelp },
- { "--pagelength", 1, OptPageLength },
- { "--start-addr", 1, OptStartAddr },
- { "--verbose", 0, OptVerbose },
- { "--version", 0, OptVersion },
+ { "--argument-column", 1, OptArgumentColumn },
+ { "--bytes-per-line", 1, OptBytesPerLine },
+ { "--comment-column", 1, OptCommentColumn },
+ { "--comments", 1, OptComments },
+ { "--cpu", 1, OptCPU },
+ { "--debug-info", 0, OptDebugInfo },
+ { "--formfeeds", 0, OptFormFeeds },
+ { "--help", 0, OptHelp },
+ { "--hexoffs", 0, OptHexOffs },
+ { "--info", 1, OptInfo },
+ { "--label-break", 1, OptLabelBreak },
+ { "--mnemonic-column", 1, OptMnemonicColumn },
+ { "--pagelength", 1, OptPageLength },
+ { "--start-addr", 1, OptStartAddr },
+ { "--text-column", 1, OptTextColumn },
+ { "--verbose", 0, OptVerbose },
+ { "--version", 0, OptVersion },
};
unsigned I;
+ time_t T;
/* Initialize the cmdline module */
InitCmdLine (&argc, &argv, "da65");
I = 1;
while (I < ArgCount) {
- /* Get the argument */
- const char* Arg = ArgVec[I];
-
- /* Check for an option */
- if (Arg [0] == '-') {
- switch (Arg [1]) {
-
- case '-':
- LongOption (&I, OptTab, sizeof(OptTab)/sizeof(OptTab[0]));
- break;
-
- case 'h':
- OptHelp (Arg, 0);
- break;
-
- case 'o':
- OutFile = GetArg (&I, 2);
- break;
-
- case 'v':
- OptVerbose (Arg, 0);
- break;
-
- case 'S':
- OptStartAddr (Arg, GetArg (&I, 2));
- break;
-
- case 'V':
- OptVersion (Arg, 0);
- break;
-
- default:
- UnknownOption (Arg);
- break;
-
- }
- } else {
- /* Filename. Check if we already had one */
- if (InFile) {
- fprintf (stderr, "%s: Don't know what to do with `%s'\n",
- ProgName, Arg);
- exit (EXIT_FAILURE);
- } else {
- InFile = Arg;
- }
- }
-
- /* Next argument */
- ++I;
+ /* Get the argument */
+ const char* Arg = ArgVec[I];
+
+ /* Check for an option */
+ if (Arg [0] == '-') {
+ switch (Arg [1]) {
+
+ case '-':
+ LongOption (&I, OptTab, sizeof(OptTab)/sizeof(OptTab[0]));
+ break;
+
+ case 'g':
+ OptDebugInfo (Arg, 0);
+ break;
+
+ case 'h':
+ OptHelp (Arg, 0);
+ break;
+
+ case 'i':
+ OptInfo (Arg, GetArg (&I, 2));
+ break;
+
+ case 'o':
+ OutFile = GetArg (&I, 2);
+ break;
+
+ case 'v':
+ OptVerbose (Arg, 0);
+ break;
+
+ case 'S':
+ OptStartAddr (Arg, GetArg (&I, 2));
+ break;
+
+ case 'V':
+ OptVersion (Arg, 0);
+ break;
+
+ default:
+ UnknownOption (Arg);
+ break;
+
+ }
+ } else {
+ /* Filename. Check if we already had one */
+ if (InFile) {
+ fprintf (stderr, "%s: Don't know what to do with `%s'\n",
+ ProgName, Arg);
+ exit (EXIT_FAILURE);
+ } else {
+ InFile = Arg;
+ }
+ }
+
+ /* Next argument */
+ ++I;
}
+ /* Try to read the info file */
+ ReadInfoFile ();
+
/* Must have an input file */
if (InFile == 0) {
- AbEnd ("No input file");
+ AbEnd ("No input file");
}
- /* Make the config file name from the input file if none was given */
- if (!CfgAvail ()) {
- CfgSetName (MakeFilename (InFile, CfgExt));
+ /* Check the formatting options for reasonable values. Note: We will not
+ ** really check that they make sense, just that they aren't complete
+ ** garbage.
+ */
+ if (MCol >= ACol) {
+ AbEnd ("mnemonic-column value must be smaller than argument-column value");
+ }
+ if (ACol >= CCol) {
+ AbEnd ("argument-column value must be smaller than comment-column value");
+ }
+ if (CCol >= TCol) {
+ AbEnd ("comment-column value must be smaller than text-column value");
}
- /* Try to read the configuration file */
- CfgRead ();
-
- /* Make the output file name from the input file name if none was given */
- if (OutFile == 0) {
- OutFile = MakeFilename (InFile, OutExt);
+ /* If no CPU given, use the default CPU */
+ if (CPU == CPU_UNKNOWN) {
+ CPU = CPU_6502;
}
+ /* Get the current time and convert it to string so it can be used in
+ ** the output page headers.
+ */
+ T = time (0);
+ strftime (Now, sizeof (Now), "%Y-%m-%d %H:%M:%S", localtime (&T));
+
/* Load the input file */
- LoadCode (InFile, StartAddr);
+ LoadCode ();
/* Open the output file */
OpenOutput (OutFile);
/* Done */
return EXIT_SUCCESS;
}
-
-
-