mirror of
https://github.com/cc65/cc65.git
synced 2025-01-16 13:31:16 +00:00
9aa17b4c53
- A HTML generator isn't really the core of a C/asm toolchain. - A Perl script isn't actually as portable as one might think.
1221 lines
38 KiB
Perl
1221 lines
38 KiB
Perl
#!/usr/bin/perl
|
|
###############################################################################
|
|
# #
|
|
# ca65html #
|
|
# #
|
|
# Convert a ca65 source into HTML #
|
|
# #
|
|
# #
|
|
# #
|
|
# (C) 2000-2007 Ullrich von Bassewitz #
|
|
# Roemerstrasse 52 #
|
|
# D-70794 Filderstadt #
|
|
# EMail: uz@cc65.org #
|
|
# #
|
|
# #
|
|
# This software is provided 'as-is', without any expressed or implied #
|
|
# warranty. In no event will the authors be held liable for any damages #
|
|
# arising from the use of this software. #
|
|
# #
|
|
# Permission is granted to anyone to use this software for any purpose, #
|
|
# including commercial applications, and to alter it and redistribute it #
|
|
# freely, subject to the following restrictions: #
|
|
# #
|
|
# 1. The origin of this software must not be misrepresented; you must not #
|
|
# claim that you wrote the original software. If you use this software #
|
|
# in a product, an acknowledgment in the product documentation would be #
|
|
# appreciated but is not required. #
|
|
# 2. Altered source versions must be plainly marked as such, and must not #
|
|
# be misrepresented as being the original software. #
|
|
# 3. This notice may not be removed or altered from any source #
|
|
# distribution. #
|
|
# #
|
|
###############################################################################
|
|
|
|
|
|
|
|
# Things currently missing:
|
|
#
|
|
# - Scoping with .proc/.endproc, .scope/.endscope, .enum/.endenum,
|
|
# .struct/.endstruct, .union/endunion, .repeat/.endrep, .local
|
|
# - .global is ignored
|
|
# - .case is ignored, labels are always case-sensitive
|
|
# - .include handling (difficult)
|
|
# - The global namespace operator ::
|
|
#
|
|
|
|
|
|
|
|
use strict 'vars';
|
|
use warnings;
|
|
|
|
# Modules
|
|
use Getopt::Long;
|
|
|
|
|
|
|
|
#-----------------------------------------------------------------------------#
|
|
# Variables #
|
|
# ----------------------------------------------------------------------------#
|
|
|
|
|
|
|
|
# Global variables
|
|
my %Files = (); # List of all files.
|
|
my $FileCount = 0; # Number of input files
|
|
my %Exports = (); # List of exported symbols.
|
|
my %Imports = (); # List of imported symbols.
|
|
my %Labels = (); # List of all labels
|
|
my $LabelNum = 0; # Counter to generate unique labels
|
|
|
|
# Command line options
|
|
my $BGColor = "#FFFFFF"; # Background color
|
|
my $Colorize = 0; # Colorize the output
|
|
my $CommentColor = "#B22222"; # Color for comments
|
|
my $CRefs = 0; # Add references to the C file
|
|
my $CtrlColor = "#228B22"; # Color for control directives
|
|
my $CvtTabs = 0; # Convert tabs to spaces
|
|
my $TabSize = 8; # This is how god created them
|
|
my $Debug = 0; # No debugging
|
|
my $Help = 0; # Help flag
|
|
my $HTMLDir = ""; # Directory in which to create the files
|
|
my $IndexCols = 6; # Columns in the file listing
|
|
my $IndexTitle = "Index"; # Title of index page
|
|
my $IndexName = "index.html"; # Name of index page
|
|
my $IndexPage = 0; # Create an index page
|
|
my $KeywordColor = "#A020F0"; # Color for keywords
|
|
my $LineLabels = 0; # Add a HTML label to each line
|
|
my $LineNumbers = 0; # Add line numbers to the output
|
|
my $LinkStyle = 0; # Default link style
|
|
my $ReplaceExt = 0; # Replace extension instead of appending
|
|
my $StringColor = "#6169C1"; # Color for strings
|
|
my $TextColor = "#000000"; # Text color
|
|
my $Verbose = 0; # Be quiet
|
|
|
|
# Table used to convert the label number into names
|
|
my @NameTab = ('A' .. 'Z', '0' .. '9');
|
|
|
|
|
|
|
|
#-----------------------------------------------------------------------------#
|
|
# Helper functions #
|
|
# ----------------------------------------------------------------------------#
|
|
|
|
|
|
|
|
# Terminate with an error
|
|
sub Abort {
|
|
print STDERR "ca65html: @_\n";
|
|
exit 1;
|
|
}
|
|
|
|
# Print a message if verbose is true
|
|
sub Gabble {
|
|
if ($Verbose) {
|
|
print "ca65html: @_\n";
|
|
}
|
|
}
|
|
|
|
# Generate a label and return it
|
|
sub GenLabel {
|
|
|
|
my $I;
|
|
my $L = "";;
|
|
my $Num = $LabelNum++;
|
|
|
|
# Generate the label
|
|
for ($I = 0; $I < 4; $I++) {
|
|
$L = $NameTab[$Num % 36] . $L;
|
|
$Num /= 36;
|
|
}
|
|
return $L;
|
|
}
|
|
|
|
# Make an output file name from an input file name
|
|
sub GetOutName {
|
|
|
|
# Input name is parameter
|
|
my $InName = $_[0];
|
|
|
|
# Create the output file name from the input file name
|
|
if ($ReplaceExt && $InName =~ /^(.+)\.([^\.\/]*)$/) {
|
|
return "$1.html";
|
|
} else {
|
|
return "$InName.html";
|
|
}
|
|
}
|
|
|
|
# Translate some HTML characters into harmless names.
|
|
sub Cleanup {
|
|
my $S = shift (@_);
|
|
$S =~ s/&/&/g;
|
|
$S =~ s/</</g;
|
|
$S =~ s/>/>/g;
|
|
$S =~ s/\"/"/g;
|
|
return $S;
|
|
}
|
|
|
|
# Strip a path from a filename and return just the name
|
|
sub StripPath {
|
|
|
|
# Filename is argument
|
|
my $FileName = $_[0];
|
|
|
|
# Remove a path name if we have one
|
|
$FileName =~ /^(.*?)([^\/]*)$/;
|
|
return $2;
|
|
}
|
|
|
|
|
|
|
|
#-----------------------------------------------------------------------------#
|
|
# Document header and footer #
|
|
# ----------------------------------------------------------------------------#
|
|
|
|
|
|
|
|
# Print the document header
|
|
sub DocHeader {
|
|
my $OUT = shift (@_);
|
|
my $Asm = shift (@_);
|
|
print $OUT "<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Strict//EN\" \"http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd\">\n";
|
|
print $OUT <<"EOF";
|
|
<html>
|
|
<head>
|
|
<meta http-equiv="Content-Type" content="text/html; charset=iso-8859-1">
|
|
<meta name="GENERATOR" content="ca65html">
|
|
<title>$Asm</title>
|
|
<style type=\"text/css\">
|
|
body {
|
|
background-color: $BGColor;
|
|
color: $TextColor;
|
|
}
|
|
h1 {
|
|
text-align: center;
|
|
}
|
|
#top {
|
|
margin: 2em 0 3em 0;
|
|
border-bottom: 1px solid grey;
|
|
}
|
|
#bottom {
|
|
margin: 3em 0 1em 0;
|
|
padding-top: 1em;
|
|
border-top: 1px solid grey;
|
|
}
|
|
img {
|
|
border: 0;
|
|
margin: 0;
|
|
float: right;
|
|
}
|
|
.ctrl {
|
|
color: $CtrlColor;
|
|
}
|
|
.keyword {
|
|
color: $KeywordColor;
|
|
}
|
|
.string {
|
|
color: $StringColor;
|
|
}
|
|
.comment {
|
|
color: $CommentColor;
|
|
}
|
|
a:link {
|
|
color: #0000d0;
|
|
}
|
|
a:visited {
|
|
color: #000060;
|
|
}
|
|
a:active {
|
|
color: #00d0d0;
|
|
}
|
|
</style>
|
|
</head>
|
|
<body>
|
|
<div id=\"top\"><h1>$Asm</h1></div>
|
|
EOF
|
|
}
|
|
|
|
# Print the document footer
|
|
sub DocFooter {
|
|
my $OUT = shift (@_);
|
|
my $Name = shift (@_);
|
|
|
|
# Get the current date and time
|
|
my $Today = localtime;
|
|
|
|
# Print
|
|
print $OUT "<div id=\"bottom\"><address>\n";
|
|
print $OUT "<a href=\"http://validator.w3.org/check?uri=referer\">\n";
|
|
print $OUT "<img src=\"http://www.w3.org/Icons/valid-xhtml10-blue\" alt=\"Valid XHTML 1.0 Strict\" height=\"31\" width=\"88\" /></a><br>\n";
|
|
print $OUT "$Name; generated on $Today by ca65html<br>\n";
|
|
print $OUT "<a href=\"mailto:uz@cc65.org\">uz@cc65.org</a>\n";
|
|
print $OUT "</address></div>\n";
|
|
print $OUT "</body></html>\n";
|
|
}
|
|
|
|
|
|
|
|
#-----------------------------------------------------------------------------#
|
|
# Colorization #
|
|
#-----------------------------------------------------------------------------#
|
|
|
|
|
|
|
|
sub ColorizeComment {
|
|
if ($Colorize && $_[0] ne "") {
|
|
return "<span class=\"comment\">$_[0]</span>";
|
|
} else {
|
|
return $_[0];
|
|
}
|
|
}
|
|
|
|
|
|
|
|
sub ColorizeCtrl {
|
|
if ($Colorize) {
|
|
return "<span class=\"ctrl\">$_[0]</span>";
|
|
} else {
|
|
return $_[0];
|
|
}
|
|
}
|
|
|
|
|
|
|
|
sub ColorizeKeyword {
|
|
if ($Colorize) {
|
|
return "<span class=\"keyword\">$_[0]</span>";
|
|
} else {
|
|
return $_[0];
|
|
}
|
|
}
|
|
|
|
|
|
|
|
sub ColorizeString {
|
|
if ($Colorize) {
|
|
return "<span class=\"string\">$_[0]</span>";
|
|
} else {
|
|
return $_[0];
|
|
}
|
|
}
|
|
|
|
|
|
|
|
#-----------------------------------------------------------------------------#
|
|
# File list management #
|
|
#-----------------------------------------------------------------------------#
|
|
|
|
|
|
|
|
sub AddFile {
|
|
|
|
# Argument is file to add
|
|
my $FileName = $_[0];
|
|
|
|
# Get just the name (remove a path if there is one)
|
|
my $Name = StripPath ($FileName);
|
|
|
|
# Check if we have the file already
|
|
if (exists ($Files{$Name})) {
|
|
Gabble ("File \"$FileName\" already known");
|
|
return;
|
|
}
|
|
|
|
# Check with the full pathname. If we don't find it, search in the current
|
|
# directory
|
|
if (-f $FileName && -r _) {
|
|
$Files{$Name} = $FileName;
|
|
$FileCount++;
|
|
} elsif (-f $Name && -r _) {
|
|
$Files{$Name} = $Name;
|
|
$FileCount++;
|
|
} else {
|
|
Abort ("$FileName not found or not readable");
|
|
}
|
|
}
|
|
|
|
|
|
|
|
#-----------------------------------------------------------------------------#
|
|
# Referencing and defining labels #
|
|
#-----------------------------------------------------------------------------#
|
|
|
|
|
|
|
|
# Get a label reference
|
|
sub RefLabel {
|
|
|
|
# Arguments are: Filename, identifier, item that should be tagged
|
|
my $FileName = $_[0];
|
|
my $Id = $_[1];
|
|
my $Item = $_[2];
|
|
|
|
# Search for the identifier in the list of labels
|
|
if (exists ($Labels{$FileName}{$Id})) {
|
|
# It is a label (in this file)
|
|
return sprintf ("<a href=\"#%s\">%s</a>", $Labels{$FileName}{$Id}, $Item);
|
|
} elsif (exists ($Imports{$FileName}{$Id})) {
|
|
# It is an import. If LinkStyle is 1, or if the file exporting the
|
|
# identifier is not visible, we link to the .import statement in the
|
|
# current file. Otherwise we link directly to the referenced symbol
|
|
# in the file that exports it.
|
|
if ($LinkStyle == 1 or not exists ($Exports{$Id})) {
|
|
return sprintf ("<a href=\"#%s\">%s</a>", $Imports{$FileName}{$Id}, $Item);
|
|
} else {
|
|
# Get the filename from the export
|
|
my $Label;
|
|
($FileName, $Label) = split (/#/, $Exports{$Id});
|
|
if (not defined ($Labels{$FileName}{$Id})) {
|
|
# This may currently happen because we don't see .include
|
|
# statements, so we may have an export but no definition.
|
|
# Link to the .export statement instead
|
|
$Label = $Exports{$Id};
|
|
} else {
|
|
# Link to the definition in the file
|
|
$Label = sprintf ("%s#%s", $FileName, $Labels{$FileName}{$Id});
|
|
}
|
|
return sprintf ("<a href=\"%s\">%s</a>", $Label, $Item);
|
|
}
|
|
} else {
|
|
# The symbol is unknown, return as is
|
|
return $Item;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
#-----------------------------------------------------------------------------#
|
|
# Pass 1 #
|
|
# ----------------------------------------------------------------------------#
|
|
|
|
|
|
|
|
# Process1: Read one file for the first time.
|
|
sub Process1 {
|
|
|
|
# Variables
|
|
my $Line;
|
|
my $Id;
|
|
|
|
# Filename is parameter
|
|
my $InName = shift(@_);
|
|
|
|
# Create the output file name from the input file name
|
|
my $OutName = GetOutName ($InName);
|
|
|
|
# Current cheap local label prefix is empty
|
|
my $CheapPrefix = "";
|
|
|
|
# Open a the input file
|
|
my $FileName = $Files{$InName}; # Includes path if needed
|
|
open (INPUT, "<$FileName") or Abort ("Cannot open $FileName: $!");
|
|
|
|
# Keep the user happy
|
|
Gabble ("$FileName => $OutName");
|
|
|
|
# Read and process all lines from the file
|
|
while ($Line = <INPUT>) {
|
|
|
|
# Remove the newline
|
|
chomp ($Line);
|
|
|
|
# Check for a label
|
|
if ($Line =~ /^\s*(([\@?]?)[_a-zA-Z]\w*)\s*(?::=?|=)/) {
|
|
|
|
# Is this a local label?
|
|
if ($2 ne "") {
|
|
# Use the prefix
|
|
$Id = "$CheapPrefix$1";
|
|
} else {
|
|
# Use as is
|
|
$Id = $1;
|
|
# Remember the id as new cheap local prefix
|
|
$CheapPrefix = $Id;
|
|
}
|
|
|
|
# Remember the label
|
|
$Labels{$OutName}{$Id} = GenLabel();
|
|
|
|
# Check for an import statement
|
|
} elsif ($Line =~ /^\s*\.(?:(?:force)?import|importzp)\s+(.*?)\s*(?:;.*)?$/i) {
|
|
|
|
# Split into a list of identifiers
|
|
my @Ids = split (/\s*(?::\s*[A-Za-z]+\s*)?,\s*/, $1);
|
|
|
|
# Remove an address-size specifier, from the last identifier,
|
|
# if there is one.
|
|
$Ids[$#Ids] =~ s/\s*:\s*[A-Za-z]+//;
|
|
|
|
for $Id (@Ids) {
|
|
$Imports{$OutName}{$Id} = GenLabel();
|
|
}
|
|
|
|
# Check for an export statement
|
|
} elsif ($Line =~ /^\s*\.export(?:zp)?\s+(.*?)\s*(?:;.*)?$/i) {
|
|
|
|
# Split into a list of identifiers
|
|
my @Ids = split (/\s*(?::\s*[A-Za-z]+\s*)?,\s*/, $1);
|
|
|
|
# Remove an address-size specifier, from the last identifier,
|
|
# if there is one.
|
|
$Ids[$#Ids] =~ s/\s*:\s*[A-Za-z]+//;
|
|
|
|
for $Id (@Ids) {
|
|
$Exports{$Id} = sprintf ("%s#%s", $OutName, GenLabel());
|
|
}
|
|
|
|
# Check for an actor statement.
|
|
} elsif ($Line =~ /^\s*\.(?:(?:(?:con|de)struc|interrup)tor|condes)\s+([_a-z]\w*)/i) {
|
|
$Exports{$1} = sprintf ("%s#%s", $OutName, GenLabel());
|
|
|
|
# Check for a .proc statement
|
|
} elsif ($Line =~ /^\s*\.proc\s+([_a-z]\w*)/i) {
|
|
|
|
# Remember the ID as the new cheap-local prefix.
|
|
$CheapPrefix = $1;
|
|
$Labels{$OutName}{$1} = GenLabel();
|
|
}
|
|
}
|
|
|
|
# Close the input file
|
|
close (INPUT);
|
|
}
|
|
|
|
|
|
|
|
# Pass1: Read all files for the first time.
|
|
sub Pass1 () {
|
|
|
|
# Keep the user happy
|
|
Gabble ("Pass 1");
|
|
|
|
# Walk over the files
|
|
for my $InName (keys (%Files)) {
|
|
# Process one file
|
|
Process1 ($InName);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
#-----------------------------------------------------------------------------#
|
|
# Pass 2 #
|
|
# ----------------------------------------------------------------------------#
|
|
|
|
|
|
|
|
# Process2: Read one file the second time.
|
|
sub Process2 {
|
|
|
|
# Variables
|
|
my $Base;
|
|
my $Ext;
|
|
my $Line;
|
|
my $OutLine;
|
|
my $Id;
|
|
my $Label;
|
|
my $Comment;
|
|
my $Trailer;
|
|
|
|
# Input file is parameter
|
|
my $InName = shift(@_);
|
|
|
|
# Create the output file name from the input file name
|
|
my $OutName = GetOutName ($InName);
|
|
|
|
# Current cheap local label prefix is empty
|
|
my $CheapPrefix = "";
|
|
|
|
# Open a the input file
|
|
my $FileName = $Files{$InName}; # Includes path if needed
|
|
open (INPUT, "<$FileName") or Abort ("Cannot open $FileName: $!");
|
|
|
|
# Open the output file and print the HTML header
|
|
open (OUTPUT, ">$HTMLDir$OutName") or Abort ("Cannot open $OutName: $!");
|
|
DocHeader (OUTPUT, $InName);
|
|
print OUTPUT "<pre>\n";
|
|
|
|
# Keep the user happy
|
|
Gabble ("$FileName => $OutName");
|
|
|
|
# The instructions that will have hyperlinks if a label is used.
|
|
# And, they will be highlighted when color is used.
|
|
my $LabelIns = "adc|add|and|asl|bb[rs][0-7]|b[cv][cs]|beq|bge|bit|blt|".
|
|
"bmi|bne|bpl|br[akl]|bsr|cmp|cop|cp[axy]|dec|eor|inc|jml|".
|
|
"jmp|jsl|jsr|ld[axy]|lsr|mvn|mvp|ora|pe[air]|rep|".
|
|
"[rs]mb[0-7]|rol|ror|sbc|sep|st[012axyz]|sub|tai|tam|tdd|".
|
|
"ti[ain]|tma|trb|tsb|tst";
|
|
|
|
# Instructions that have only the implied-addressing mode -- therefore,
|
|
# no hyperlinking. They will be highlighted only, when color is used.
|
|
my $OtherIns = "cl[acdivxy]|csh|csl|de[axy]|in[axy]|nop|ph[abdkpxy]|".
|
|
"pl[abdpxy]|rt[ils]|sax|say|se[cdit]|stp|swa|sxy|ta[dsxy]|".
|
|
"tam[0-7]|tcd|tcs|tda|tdc|tma[0-7]|ts[acx]|tx[asy]|tya|tyx|".
|
|
"wai|xba|xce";
|
|
|
|
# Read the input file, replacing references with hyperlinks; and, mark
|
|
# labels as link targets.
|
|
my $LineNo = 0;
|
|
LINE: while ($Line = <INPUT>) {
|
|
|
|
# Count input lines
|
|
$LineNo++;
|
|
|
|
# Remove the newline at the end of line. Don't use chomp to be able to
|
|
# read dos/windows sources on unices.
|
|
$Line =~ s/[\r\n]*$//;
|
|
|
|
# If requested, convert tabs to spaces
|
|
if ($CvtTabs) {
|
|
# Don't ask me - this is from the perl manual page
|
|
1 while ($Line =~ s/\t+/' ' x (length($&) * $TabSize - length($`) % $TabSize)/e) ;
|
|
}
|
|
|
|
# Clear the output line
|
|
$OutLine = "";
|
|
|
|
# If requested, add a html label to each line with a name "linexxx",
|
|
# so it can be referenced from the outside (this is the same convention
|
|
# that is used by c2html). If we have line numbers enabled, add them.
|
|
if ($LineLabels && $LineNumbers) {
|
|
$OutLine .= sprintf ("<a name=\"line%d\">%6d</a>: ", $LineNo, $LineNo);
|
|
} elsif ($LineLabels) {
|
|
$OutLine .= sprintf ("<a name=\"line%d\"></a>", $LineNo);
|
|
} elsif ($LineNumbers) {
|
|
$OutLine .= sprintf ("%6d: ", $LineNo);
|
|
}
|
|
|
|
# Cut off a comment from the input line. Beware: We have to check for
|
|
# strings, since these may contain a semicolon that is no comment
|
|
# start.
|
|
($Line, $Comment) = $Line =~ /^((?:[^"';]+|".*?"|'.*?')*)(.*)$/;
|
|
if ($Comment =~ /^["']/) {
|
|
# Line with invalid syntax - there's a string start but
|
|
# no string end.
|
|
Abort (sprintf ("Invalid input at %s(%d)", $FileName, $LineNo));
|
|
}
|
|
|
|
# Remove trailing whitespace and move it together with the comment
|
|
# into the $Trailer variable.
|
|
$Line =~ s/\s*$//;
|
|
$Trailer = $& . ColorizeComment (Cleanup ($Comment));
|
|
|
|
# Check for a label at the start of the line. If we have one, process
|
|
# it, and remove it from the line.
|
|
if ($Line =~ s/^\s*?(([\@?]?)[_a-zA-Z]\w*)(\s*(?::=?|=))//) {
|
|
|
|
# Is this a local label?
|
|
if ($2 ne "") {
|
|
# Use the prefix
|
|
$Id = "$CheapPrefix$1";
|
|
} else {
|
|
# Use as is
|
|
$Id = $1;
|
|
# Remember the id as new cheap local prefix
|
|
$CheapPrefix = $Id;
|
|
}
|
|
|
|
# Get the label for the id
|
|
$Label = $Labels{$OutName}{$Id};
|
|
|
|
# Print the label with a tag
|
|
$OutLine .= "<a name=\"$Label\">$1</a>$3";
|
|
|
|
# Is the name explicitly assigned a value?
|
|
if ($3 =~ /=$/) {
|
|
# Print all identifiers if there are any.
|
|
while ($Line =~ s/^([^_a-zA-Z]*?)(([\@?]?)[_a-zA-Z]\w*)//) {
|
|
# Add the non-label stuff.
|
|
$OutLine .= Cleanup ($1);
|
|
|
|
# Use the prefix if the label is local.
|
|
# Get the reference to that label if we find it.
|
|
$OutLine .= RefLabel ($OutName, ($3 ne "") ? "$CheapPrefix$2" : $2, $2);
|
|
}
|
|
|
|
# Add a remainder if there is one.
|
|
$OutLine .= Cleanup ($Line);
|
|
|
|
# The line is complete; print it.
|
|
next LINE;
|
|
}
|
|
}
|
|
|
|
# Print any leading whitespace and remove it, so we don't have to
|
|
# care about whitespace below.
|
|
if ($Line =~ s/^\s+//) {
|
|
$OutLine .= $&;
|
|
}
|
|
|
|
# Handle the import statements
|
|
if ($Line =~ s/^\.(?:(?:force)?import|importzp)\s+//i) {
|
|
|
|
# Print any fixed stuff from the line and remove it
|
|
$OutLine .= $&;
|
|
|
|
# Print all identifiers if there are any
|
|
while ($Line =~ s/^[_a-zA-Z]\w*//) {
|
|
|
|
# Remember the identifier
|
|
my $Id = $&;
|
|
|
|
# Variable to assemble HTML representation
|
|
my $Contents = "";
|
|
|
|
# Make this import a link target
|
|
if (exists ($Imports{$OutName}{$Id})) {
|
|
$Label = $Imports{$OutName}{$Id};
|
|
$Contents .= sprintf (" name=\"%s\"", $Label);
|
|
}
|
|
|
|
# If we have an export for this import, add a link to this
|
|
# export definition
|
|
if (exists ($Exports{$Id})) {
|
|
$Label = $Exports{$Id};
|
|
$Contents .= sprintf (" href=\"%s\"", $Label);
|
|
}
|
|
|
|
# Add the HTML stuff to the output line
|
|
if ($Contents ne "") {
|
|
$OutLine .= sprintf ("<a%s>%s</a>", $Contents, $Id);
|
|
} else {
|
|
$OutLine .= $Id;
|
|
}
|
|
|
|
# Check if another identifier follows
|
|
if ($Line =~ s/^\s*(?::\s*[A-Za-z]+\s*)?,\s*//) {
|
|
$OutLine .= $&;
|
|
} else {
|
|
last;
|
|
}
|
|
}
|
|
|
|
# Add an remainder if there is one
|
|
$OutLine .= Cleanup ($Line);
|
|
|
|
# Handle export statements
|
|
} elsif ($Line =~ s/^\.export(?:zp)?\s+//i) {
|
|
|
|
# Print the command and the whitespace.
|
|
$OutLine .= $&;
|
|
|
|
# Print all identifiers if there are any
|
|
while ($Line =~ s/^[_a-zA-Z]\w*//) {
|
|
|
|
# Remember the identifier
|
|
my $Id = $&;
|
|
|
|
# Variable to assemble HTML representation
|
|
my $Contents = "";
|
|
|
|
# If we have a definition for this export in this file, add
|
|
# a link to the definition.
|
|
if (exists ($Labels{$OutName}{$Id})) {
|
|
$Label = $Labels{$OutName}{$Id};
|
|
$Contents = sprintf (" href=\"#%s\"", $Label);
|
|
}
|
|
|
|
# If we have this identifier in the list of exports, add a
|
|
# jump target for the export.
|
|
if (exists ($Exports{$Id})) {
|
|
$Label = $Exports{$Id};
|
|
# Be sure to use only the label part
|
|
$Label =~ s/^.*#//;
|
|
$Contents .= sprintf (" name=\"%s\"", $Label);
|
|
}
|
|
|
|
# Add the HTML stuff to the output line
|
|
if ($Contents ne "") {
|
|
$OutLine .= sprintf ("<a%s>%s</a>", $Contents, $Id);
|
|
} else {
|
|
$OutLine .= $Id;
|
|
}
|
|
|
|
# Check if another identifier follows
|
|
if ($Line =~ s/^\s*(?::\s*[A-Za-z]+\s*)?,\s*//) {
|
|
$OutLine .= $&;
|
|
} else {
|
|
last;
|
|
}
|
|
}
|
|
|
|
# Add an remainder if there is one
|
|
$OutLine .= Cleanup ($Line);
|
|
|
|
# Handle actor statements.
|
|
} elsif ($Line =~ s/^(\.(?:(?:(?:con|de)struc|interrup)tor|condes)\s+)([_a-z]\w*)//i) {
|
|
|
|
# Print the command and the whitespace.
|
|
$OutLine .= $1;
|
|
|
|
# Remember the identifier.
|
|
$Id = $2;
|
|
|
|
# Variable to assemble HTML representation
|
|
my $Contents = "";
|
|
|
|
# If we have a definition for this actor, in this file,
|
|
# then add a link to that definition.
|
|
if (exists ($Labels{$OutName}{$Id})) {
|
|
$Contents = sprintf (" href=\"#%s\"", $Labels{$OutName}{$Id});
|
|
}
|
|
|
|
# Get the target, for linking from imports in other files.
|
|
$Label = $Exports{$Id};
|
|
# Be sure to use only the label part.
|
|
$Label =~ s/^.*#//;
|
|
|
|
# Add the HTML stuff and the remainder of the actor
|
|
# to the output line.
|
|
$OutLine .= sprintf ("<a name=\"%s\"%s>%s</a>%s", $Label,
|
|
$Contents, $Id, Cleanup ($Line));
|
|
|
|
# Check for .faraddr, .addr, .dword, .word, .dbyt, .byt, .byte, .res,
|
|
# .elseif, .if, .align, and .org.
|
|
} elsif ($Line =~ s/^\.(?:(?:far)?addr|d?word|d?byte?|res|(?:else)?if|align|org)\s+//i) {
|
|
|
|
# Print the command and the white space
|
|
$OutLine .= $&;
|
|
|
|
# Print all identifiers if there are any
|
|
while ($Line =~ s/^([^_a-zA-Z]*?)(([\@?]?)[_a-zA-Z]\w*)//) {
|
|
# Add the non label stuff
|
|
$OutLine .= Cleanup ($1);
|
|
|
|
# Use the prefix if the label is local.
|
|
# Get the reference to that label if we find it.
|
|
$OutLine .= RefLabel ($OutName, ($3 ne "") ? "$CheapPrefix$2" : $2, $2);
|
|
}
|
|
|
|
# Add an remainder if there is one
|
|
$OutLine .= Cleanup ($Line);
|
|
|
|
# Handle .proc
|
|
} elsif ($Line =~ /^(\.proc)(\s+)([_a-z]\w*)?(.*)$/i) {
|
|
|
|
# Do we have an identifier?
|
|
if ($3 ne "") {
|
|
# Remember the ID as the new cheap-local prefix.
|
|
$CheapPrefix = $3;
|
|
|
|
# Get the label for the id
|
|
$Label = $Labels{$OutName}{$3};
|
|
|
|
# Print the label with a tag
|
|
$OutLine .= "$1$2<a name=\"$Label\">$3</a>";
|
|
|
|
} else {
|
|
|
|
# Print a line that has invalid syntax (its operand isn't
|
|
# a correctly formed name).
|
|
$OutLine .= "$1$2";
|
|
}
|
|
|
|
# Add the remainder
|
|
$OutLine .= Cleanup ($4);
|
|
|
|
# Handle .include
|
|
} elsif ($Line =~ /^(\.include)(\s*)\"((?:[^\"]+?|\\\")+)(\".*)$/i) {
|
|
|
|
# Add the fixed stuff to the output line
|
|
$OutLine .= "$1$2"";
|
|
|
|
# Get the filename into a named variable
|
|
my $FileName = Cleanup ($3);
|
|
|
|
# Get the name without a path
|
|
my $Name = StripPath ($3);
|
|
|
|
# If the include file is among the list of our files, add a link,
|
|
# otherwise just add the name as is.
|
|
if (exists ($Files{$Name})) {
|
|
$OutLine .= sprintf ("<a href=\"%s\">%s</a>", GetOutName ($Name), $FileName);
|
|
} else {
|
|
$OutLine .= $FileName;
|
|
}
|
|
|
|
# Add the remainder
|
|
$OutLine .= Cleanup ($4);
|
|
|
|
# Handle .dbg line
|
|
} elsif ($CRefs && $Line =~ s/^\.dbg\s+//) {
|
|
|
|
# Add the fixed stuff to the output line
|
|
$OutLine .= $&;
|
|
|
|
# Check for the type of the .dbg directive
|
|
if ($Line =~ /^(line,\s*)\"((?:[^\"]+?|\\\")+)\"(,\s*)(\d+)(.*)$/) {
|
|
|
|
# Add the fixed stuff to the output line
|
|
$OutLine .= "$1"";
|
|
|
|
# Get the filename and line number into named variables
|
|
my $DbgFile = $2;
|
|
my $DbgLine = $4;
|
|
|
|
# Remember the remainder
|
|
$Line = "\"$3$4$5";
|
|
|
|
# Get the name without a path
|
|
my $Name = StripPath ($DbgFile);
|
|
|
|
# We don't need FileName any longer as is, so clean it up
|
|
$DbgFile = Cleanup ($DbgFile);
|
|
|
|
# Add a link to the source file
|
|
$OutLine .= sprintf ("<a href=\"%s.html#line%d\">%s</a>", $Name, $DbgLine, $DbgFile);
|
|
|
|
# Add the remainder
|
|
$OutLine .= Cleanup ($Line);
|
|
|
|
} elsif ($Line =~ /^(file,\s*)\"((?:[^\"]+?|\\\")+)\"(.*)$/) { #pf FIXME: doesn't handle \" correctly!
|
|
|
|
# Get the filename into a named variables
|
|
my $DbgFile = Cleanup ($2);
|
|
|
|
# Get the name without a path
|
|
my $Name = Cleanup (StripPath ($2));
|
|
|
|
# Add the fixed stuff to the output line
|
|
$OutLine .= sprintf ("%s\"<a href=\"%s.html\">%s</a>\"%s",
|
|
$1, $Name, $DbgFile, $3);
|
|
|
|
} else {
|
|
|
|
# Add the remainder
|
|
$OutLine .= Cleanup ($Line);
|
|
|
|
}
|
|
|
|
} elsif ($CRefs && $Line =~ /^(\.dbg)(\s+line,\s*)\"((?:[^\"]+?|\\\")+)\"(,\s*)(\d+)(.*$)/) {
|
|
|
|
# Add the fixed stuff to the output line
|
|
$OutLine .= "$1$2"";
|
|
|
|
# Get the filename and line number into named variables
|
|
my $FileName = $3;
|
|
my $LineNo = $5;
|
|
|
|
# Remember the remainder
|
|
$Line = "\"$4$5$6";
|
|
|
|
# Get the name without a path
|
|
my $Name = StripPath ($FileName);
|
|
|
|
# We don't need FileName any longer as is, so clean it up
|
|
$FileName = Cleanup ($FileName);
|
|
|
|
# Add a link to the source file
|
|
$OutLine .= sprintf ("<a href=\"%s.html#line%d\">%s</a>", $Name, $LineNo, $FileName);
|
|
|
|
# Add the remainder
|
|
$OutLine .= Cleanup ($Line);
|
|
|
|
# Check for .ifdef, .ifndef, .ifref, and .ifnref.
|
|
} elsif ($Line =~ s/^(\.ifn?[dr]ef\s+)(([\@?]?)[_a-z]\w*)?//i) {
|
|
|
|
# Print the command and the whitespace.
|
|
$OutLine .= $1;
|
|
|
|
if ($2 ne "") {
|
|
# Use the prefix if the label is local.
|
|
# Get the reference to that label if we find it.
|
|
$OutLine .= RefLabel ($OutName, ($3 ne "") ? "$CheapPrefix$2" : $2, $2);
|
|
}
|
|
|
|
# Add a remainder if there is one.
|
|
$OutLine .= Cleanup ($Line);
|
|
|
|
# Check for assertions.
|
|
} elsif ($Line =~ s/^(\.assert\s+)(.+?)(,\s*(?:error|warning)\s*(?:,.*)?)$/$2/i) {
|
|
|
|
# Print the command and the whitespace.
|
|
$OutLine .= $1;
|
|
|
|
$Comment = $3;
|
|
|
|
# Print all identifiers if there are any.
|
|
while ($Line =~ s/^([^_a-zA-Z]*?)(([\@?]?)[_a-zA-Z]\w*)//) {
|
|
# Add the non-label stuff.
|
|
$OutLine .= Cleanup ($1);
|
|
|
|
# Use the prefix if the label is local.
|
|
# Get the reference to that label if we find it.
|
|
$OutLine .= RefLabel ($OutName, ($3 ne "") ? "$CheapPrefix$2" : $2, $2);
|
|
}
|
|
|
|
# Add a remainder if there is one.
|
|
$OutLine .= Cleanup ($Line . $Comment);
|
|
|
|
# Check for instructions with labels
|
|
} elsif ($Line =~ s/^($LabelIns)\b(\s*)//io) {
|
|
|
|
# Print the instruction and white space
|
|
$OutLine .= ColorizeKeyword ($1) . $2;
|
|
|
|
# Print all identifiers if there are any.
|
|
while ($Line =~ s/^([^_a-zA-Z]*?)(([\@?]?)[_a-zA-Z]\w*)//) {
|
|
|
|
# Add the non-label stuff.
|
|
$OutLine .= Cleanup ($1);
|
|
|
|
# Is this a local label?
|
|
if ($3 ne "") {
|
|
# Use the prefix
|
|
$Id = "$CheapPrefix$2";
|
|
} else {
|
|
# Use as is
|
|
$Id = $2;
|
|
}
|
|
|
|
# Get the reference to this label if we find it
|
|
$OutLine .= RefLabel ($OutName, $Id, $2);
|
|
}
|
|
|
|
# Reassemble and print the line
|
|
$OutLine .= Cleanup ($Line);
|
|
|
|
# Check for all other instructions
|
|
} elsif ($Line =~ /^($OtherIns)\b(.*)$/io) {
|
|
|
|
# Colorize and print
|
|
$OutLine .= ColorizeKeyword ($1) . Cleanup ($2);
|
|
|
|
} else {
|
|
|
|
# Nothing known - print the line
|
|
$OutLine .= Cleanup ($Line);
|
|
|
|
}
|
|
|
|
} continue {
|
|
# Colorize all keywords
|
|
$OutLine =~ s/(?<![\w;])\.[_a-zA-Z]\w*/ColorizeCtrl ($&)/ge;
|
|
|
|
# Print the result with the trailer.
|
|
print OUTPUT "$OutLine$Trailer\n";
|
|
}
|
|
|
|
# Print the HTML footer
|
|
print OUTPUT "</pre>\n";
|
|
DocFooter (OUTPUT, $OutName);
|
|
|
|
# Close the files
|
|
close (INPUT);
|
|
close (OUTPUT);
|
|
}
|
|
|
|
|
|
|
|
# Pass2: Read all files the second time.
|
|
sub Pass2 () {
|
|
|
|
# Keep the user happy
|
|
Gabble ("Pass 2");
|
|
|
|
# Walk over the files
|
|
for my $InName (keys (%Files)) {
|
|
# Process one file
|
|
Process2 ($InName);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
#-----------------------------------------------------------------------------#
|
|
# Create an index page #
|
|
# ----------------------------------------------------------------------------#
|
|
|
|
|
|
|
|
# Print a list of all files
|
|
sub FileIndex {
|
|
|
|
# File is argument
|
|
my $INDEX = $_[0];
|
|
|
|
# Print the file list in a table
|
|
print $INDEX "<h2>Files</h2><p>\n";
|
|
print $INDEX "<table border=\"0\" width=\"100%\">\n";
|
|
my $Count = 0;
|
|
for my $File (sort (keys (%Files))) {
|
|
|
|
#
|
|
if (($Count % $IndexCols) == 0) {
|
|
print $INDEX "<tr>\n";
|
|
}
|
|
printf $INDEX "<td><a href=\"%s\">%s</a></td>\n", GetOutName ($File), $File;
|
|
if (($Count % $IndexCols) == $IndexCols-1) {
|
|
print $INDEX "</tr>\n";
|
|
}
|
|
$Count++;
|
|
}
|
|
if (($Count % $IndexCols) != 0) {
|
|
print $INDEX "</tr>\n";
|
|
}
|
|
print $INDEX "</table><p><br><p>\n";
|
|
}
|
|
|
|
|
|
|
|
# Print a list of all exports
|
|
sub ExportIndex {
|
|
|
|
# File is argument
|
|
my $INDEX = $_[0];
|
|
|
|
# Print the file list in a table
|
|
print $INDEX "<h2>Exports</h2><p>\n";
|
|
print $INDEX "<table border=\"0\" width=\"100%\">\n";
|
|
my $Count = 0;
|
|
for my $Export (sort (keys (%Exports))) {
|
|
|
|
# Get the export
|
|
my $File;
|
|
my $Label;
|
|
($File, $Label) = split (/#/, $Exports{$Export});
|
|
|
|
# The label is the label of the export statement. If we can find the
|
|
# actual label, use this instead.
|
|
if (exists ($Labels{$File}{$Export})) {
|
|
$Label = $Labels{$File}{$Export};
|
|
}
|
|
|
|
#
|
|
if (($Count % $IndexCols) == 0) {
|
|
print $INDEX "<tr>\n";
|
|
}
|
|
printf $INDEX "<td><a href=\"%s#%s\">%s</a></td>\n", $File, $Label, $Export;
|
|
if (($Count % $IndexCols) == $IndexCols-1) {
|
|
print $INDEX "</tr>\n";
|
|
}
|
|
$Count++;
|
|
}
|
|
if (($Count % $IndexCols) != 0) {
|
|
print $INDEX "</tr>\n";
|
|
}
|
|
print $INDEX "</table><p><br><p>\n";
|
|
}
|
|
|
|
|
|
|
|
sub CreateIndex {
|
|
|
|
# Open the index page file
|
|
open (INDEX, ">$HTMLDir$IndexName") or Abort ("Cannot open $IndexName: $!");
|
|
|
|
# Print the header
|
|
DocHeader (INDEX, $IndexTitle, 0);
|
|
|
|
# Print the file list in a table
|
|
FileIndex (INDEX);
|
|
ExportIndex (INDEX);
|
|
|
|
# Print the document footer
|
|
DocFooter (INDEX, $IndexName);
|
|
|
|
# Close the index file
|
|
close (INDEX);
|
|
}
|
|
|
|
|
|
|
|
#-----------------------------------------------------------------------------#
|
|
# Print usage information #
|
|
# ----------------------------------------------------------------------------#
|
|
|
|
|
|
|
|
sub Usage {
|
|
print "Usage: ca65html [options] file ...\n";
|
|
print "Options:\n";
|
|
print " --bgcolor c Use background color c instead of $BGColor\n";
|
|
print " --colorize Add color highlights to the output\n";
|
|
print " --commentcolor c Use color c for comments instead of $CommentColor\n";
|
|
print " --crefs Generate references to the C source file(s)\n";
|
|
print " --ctrlcolor c Use color c for directives instead of $CtrlColor\n";
|
|
print " --cvttabs Convert tabs to spaces in the output\n";
|
|
print " --help This text\n";
|
|
print " --htmldir dir Specify directory for HTML files\n";
|
|
print " --indexcols n Use n columns on index page (default $IndexCols)\n";
|
|
print " --indexname file Use file for the index file instead of $IndexName\n";
|
|
print " --indexpage Create an index page\n";
|
|
print " --indextitle title Use title as the index title instead of $IndexTitle\n";
|
|
print " --keywordcolor c Use color c for keywords instead of $KeywordColor\n";
|
|
print " --linelabels Generate a linexxx HTML label for each line\n";
|
|
print " --linenumbers Add line numbers to the output\n";
|
|
print " --linkstyle style Use the given link style\n";
|
|
print " --replaceext Replace source extension instead of appending .html\n";
|
|
print " --tabsize n Use n spaces when replacing tabs (default $TabSize)\n";
|
|
print " --textcolor c Use text color c instead of $TextColor\n";
|
|
print " --verbose Be more verbose\n";
|
|
}
|
|
|
|
|
|
|
|
#-----------------------------------------------------------------------------#
|
|
# Main #
|
|
# ----------------------------------------------------------------------------#
|
|
|
|
|
|
|
|
# Get program options
|
|
GetOptions ("bgcolor=s" => \$BGColor,
|
|
"colorize" => \$Colorize,
|
|
"commentcolor=s" => \$CommentColor,
|
|
"crefs" => \$CRefs,
|
|
"ctrlcolor=s" => \$CtrlColor,
|
|
"cvttabs" => \$CvtTabs,
|
|
"debug!" => \$Debug,
|
|
"help" => \$Help,
|
|
"htmldir=s" => \$HTMLDir,
|
|
"indexcols=i" => \$IndexCols,
|
|
"indexname=s" => \$IndexName,
|
|
"indexpage" => \$IndexPage,
|
|
"indextitle=s" => \$IndexTitle,
|
|
"keywordcolor=s" => \$KeywordColor,
|
|
"linelabels" => \$LineLabels,
|
|
"linenumbers" => \$LineNumbers,
|
|
"linkstyle=i" => \$LinkStyle,
|
|
"replaceext" => \$ReplaceExt,
|
|
"tabsize=i" => \$TabSize,
|
|
"textcolor=s" => \$TextColor,
|
|
"verbose!" => \$Verbose,
|
|
"<>" => \&AddFile);
|
|
|
|
# Check some arguments
|
|
if ($IndexCols <= 0 || $IndexCols >= 20) {
|
|
Abort ("Invalid value for --indexcols option");
|
|
}
|
|
if ($TabSize < 1 || $TabSize > 16) {
|
|
Abort ("Invalid value for --tabsize option");
|
|
}
|
|
if ($HTMLDir ne "" && $HTMLDir =~ /[^\/]$/) {
|
|
# Add a trailing path separator
|
|
$HTMLDir .= "/";
|
|
}
|
|
|
|
|
|
|
|
# Print help if requested
|
|
if ($Help) {
|
|
Usage ();
|
|
}
|
|
|
|
# Check if we have input files given
|
|
if ($FileCount == 0) {
|
|
Abort ("No input files");
|
|
}
|
|
|
|
# Convert the documents
|
|
Pass1 ();
|
|
Pass2 ();
|
|
|
|
# Generate an index page if requested
|
|
if ($IndexPage) {
|
|
CreateIndex ();
|
|
}
|
|
|
|
# Done
|
|
exit 0;
|