2018-09-28 17:05:11 +00:00
|
|
|
|
/*
|
|
|
|
|
* Copyright 2018 faddenSoft
|
|
|
|
|
*
|
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
|
*
|
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
*
|
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
|
* limitations under the License.
|
|
|
|
|
*/
|
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.ComponentModel;
|
|
|
|
|
using System.Diagnostics;
|
2018-10-17 20:11:26 +00:00
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Text;
|
2018-09-28 17:05:11 +00:00
|
|
|
|
|
2018-10-17 20:11:26 +00:00
|
|
|
|
using Asm65;
|
2018-09-28 17:05:11 +00:00
|
|
|
|
using CommonUtil;
|
|
|
|
|
|
|
|
|
|
namespace SourceGen.AsmGen {
|
2018-10-17 20:11:26 +00:00
|
|
|
|
#region IGenerator
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Generate source code compatible with Brutal Deluxe's Merlin 32 assembler
|
|
|
|
|
/// (https://www.brutaldeluxe.fr/products/crossdevtools/merlin/).
|
|
|
|
|
/// </summary>
|
|
|
|
|
public class GenMerlin32 : IGenerator {
|
2018-10-23 23:06:29 +00:00
|
|
|
|
private const string ASM_FILE_SUFFIX = "_Merlin32.S"; // must start with underscore
|
2018-10-17 20:11:26 +00:00
|
|
|
|
private const int MAX_OPERAND_LEN = 64;
|
|
|
|
|
|
|
|
|
|
// IGenerator
|
|
|
|
|
public DisasmProject Project { get; private set; }
|
|
|
|
|
|
|
|
|
|
// IGenerator
|
|
|
|
|
public Formatter SourceFormatter { get; private set; }
|
|
|
|
|
|
|
|
|
|
// IGenerator
|
|
|
|
|
public AppSettings Settings { get; private set; }
|
|
|
|
|
|
|
|
|
|
// IGenerator
|
|
|
|
|
public AssemblerQuirks Quirks { get; private set; }
|
|
|
|
|
|
|
|
|
|
// IGenerator
|
|
|
|
|
public LabelLocalizer Localizer { get { return mLocalizer; } }
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Working directory, i.e. where we write our output file(s).
|
|
|
|
|
/// </summary>
|
|
|
|
|
private string mWorkDirectory;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// If set, long labels get their own line.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private bool mLongLabelNewLine;
|
|
|
|
|
|
2018-10-21 04:24:28 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Output column widths.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private int[] mColumnWidths;
|
|
|
|
|
|
2018-10-17 20:11:26 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Base filename. Typically the project file name without the ".dis65" extension.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private string mFileNameBase;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// StringBuilder to use when composing a line. Held here to reduce allocations.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private StringBuilder mLineBuilder = new StringBuilder(100);
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Label localization helper.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private LabelLocalizer mLocalizer;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Stream to send the output to.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private StreamWriter mOutStream;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Holds detected version of configured assembler.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private CommonUtil.Version mAsmVersion = CommonUtil.Version.NO_VERSION;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Semi-convenient way to hold all the interesting string constants in one place.
|
|
|
|
|
// Note the actual usage of the pseudo-op may not match what the main app does,
|
|
|
|
|
// e.g. RegWidthDirective behaves differently from "mx". I'm just trying to avoid
|
|
|
|
|
// having string constants scattered all over.
|
|
|
|
|
private static PseudoOp.PseudoOpNames sDataOpNames = new PseudoOp.PseudoOpNames() {
|
|
|
|
|
EquDirective = "equ",
|
|
|
|
|
OrgDirective = "org",
|
|
|
|
|
RegWidthDirective = "mx",
|
|
|
|
|
DefineData1 = "dfb",
|
|
|
|
|
DefineData2 = "dw",
|
|
|
|
|
DefineData3 = "adr",
|
|
|
|
|
DefineData4 = "adrl",
|
|
|
|
|
DefineBigData2 = "ddb",
|
|
|
|
|
//DefineBigData3
|
|
|
|
|
//DefineBigData4
|
|
|
|
|
Fill = "ds",
|
|
|
|
|
Dense = "hex",
|
|
|
|
|
StrGeneric = "asc",
|
|
|
|
|
StrGenericHi = "asc",
|
|
|
|
|
StrReverse = "rev",
|
|
|
|
|
StrReverseHi = "rev",
|
|
|
|
|
//StrNullTerm
|
|
|
|
|
StrLen8 = "str",
|
|
|
|
|
StrLen8Hi = "str",
|
|
|
|
|
StrLen16 = "strl",
|
|
|
|
|
StrLen16Hi = "strl",
|
|
|
|
|
StrDci = "dci",
|
|
|
|
|
StrDciHi = "dci",
|
|
|
|
|
//StrDciReverse
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2018-10-21 23:36:48 +00:00
|
|
|
|
// IGenerator
|
|
|
|
|
public void GetDefaultDisplayFormat(out PseudoOp.PseudoOpNames pseudoOps,
|
|
|
|
|
out Formatter.FormatConfig formatConfig) {
|
|
|
|
|
// This is not intended to match up with the Merlin generator, which uses
|
|
|
|
|
// the same pseudo-op for low/high ASCII but different string delimiters. We
|
|
|
|
|
// don't change the delimiters for the display list, so instead we tweak the
|
|
|
|
|
// opcode slightly.
|
|
|
|
|
char hiAscii = '\u2191';
|
|
|
|
|
pseudoOps = new PseudoOp.PseudoOpNames() {
|
|
|
|
|
EquDirective = "equ",
|
|
|
|
|
OrgDirective = "org",
|
|
|
|
|
DefineData1 = "dfb",
|
|
|
|
|
DefineData2 = "dw",
|
|
|
|
|
DefineData3 = "adr",
|
|
|
|
|
DefineData4 = "adrl",
|
|
|
|
|
DefineBigData2 = "ddb",
|
|
|
|
|
Fill = "ds",
|
|
|
|
|
Dense = "hex",
|
|
|
|
|
StrGeneric = "asc",
|
|
|
|
|
StrGenericHi = "asc" + hiAscii,
|
|
|
|
|
StrReverse = "rev",
|
|
|
|
|
StrReverseHi = "rev" + hiAscii,
|
|
|
|
|
StrLen8 = "str",
|
|
|
|
|
StrLen8Hi = "str" + hiAscii,
|
|
|
|
|
StrLen16 = "strl",
|
|
|
|
|
StrLen16Hi = "strl" + hiAscii,
|
|
|
|
|
StrDci = "dci",
|
|
|
|
|
StrDciHi = "dci" + hiAscii,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
formatConfig = new Formatter.FormatConfig();
|
|
|
|
|
SetFormatConfigValues(ref formatConfig);
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-17 20:11:26 +00:00
|
|
|
|
// IGenerator
|
|
|
|
|
public void Configure(DisasmProject project, string workDirectory, string fileNameBase,
|
|
|
|
|
AssemblerVersion asmVersion, AppSettings settings) {
|
|
|
|
|
Debug.Assert(project != null);
|
|
|
|
|
Debug.Assert(!string.IsNullOrEmpty(workDirectory));
|
|
|
|
|
Debug.Assert(!string.IsNullOrEmpty(fileNameBase));
|
|
|
|
|
|
|
|
|
|
Project = project;
|
|
|
|
|
Quirks = new AssemblerQuirks();
|
|
|
|
|
Quirks.TracksSepRepNotEmu = true;
|
|
|
|
|
Quirks.NoPcRelBankWrap = true;
|
|
|
|
|
|
|
|
|
|
mWorkDirectory = workDirectory;
|
|
|
|
|
mFileNameBase = fileNameBase;
|
|
|
|
|
Settings = settings;
|
|
|
|
|
|
|
|
|
|
mLongLabelNewLine = Settings.GetBool(AppSettings.SRCGEN_LONG_LABEL_NEW_LINE, false);
|
2018-10-21 04:24:28 +00:00
|
|
|
|
|
|
|
|
|
AssemblerConfig config = AssemblerConfig.GetConfig(settings,
|
|
|
|
|
AssemblerInfo.Id.Merlin32);
|
|
|
|
|
mColumnWidths = (int[])config.ColumnWidths.Clone();
|
2018-10-17 20:11:26 +00:00
|
|
|
|
}
|
|
|
|
|
|
2018-10-21 23:36:48 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Configures the assembler-specific format items.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private void SetFormatConfigValues(ref Formatter.FormatConfig config) {
|
|
|
|
|
config.mForceAbsOpcodeSuffix = ":";
|
|
|
|
|
config.mForceLongOpcodeSuffix = "l";
|
2018-11-02 22:32:54 +00:00
|
|
|
|
config.mForceDirectOperandPrefix = string.Empty;
|
2018-10-21 23:36:48 +00:00
|
|
|
|
config.mForceAbsOperandPrefix = string.Empty;
|
|
|
|
|
config.mForceLongOperandPrefix = string.Empty;
|
|
|
|
|
config.mEndOfLineCommentDelimiter = ";";
|
|
|
|
|
config.mFullLineCommentDelimiterBase = ";";
|
|
|
|
|
config.mBoxLineCommentDelimiter = string.Empty;
|
|
|
|
|
config.mAllowHighAsciiCharConst = true;
|
|
|
|
|
config.mExpressionMode = Formatter.FormatConfig.ExpressionMode.Merlin;
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-17 20:11:26 +00:00
|
|
|
|
// IGenerator; executes on background thread
|
|
|
|
|
public List<string> GenerateSource(BackgroundWorker worker) {
|
|
|
|
|
List<string> pathNames = new List<string>(1);
|
|
|
|
|
|
|
|
|
|
string fileName = mFileNameBase + ASM_FILE_SUFFIX;
|
|
|
|
|
string pathName = Path.Combine(mWorkDirectory, fileName);
|
|
|
|
|
pathNames.Add(pathName);
|
|
|
|
|
|
|
|
|
|
Formatter.FormatConfig config = new Formatter.FormatConfig();
|
|
|
|
|
GenCommon.ConfigureFormatterFromSettings(Settings, ref config);
|
2018-10-21 23:36:48 +00:00
|
|
|
|
SetFormatConfigValues(ref config);
|
2018-10-17 20:11:26 +00:00
|
|
|
|
SourceFormatter = new Formatter(config);
|
|
|
|
|
|
|
|
|
|
string msg = string.Format(Properties.Resources.PROGRESS_GENERATING_FMT, pathName);
|
|
|
|
|
worker.ReportProgress(0, msg);
|
|
|
|
|
|
|
|
|
|
mLocalizer = new LabelLocalizer(Project);
|
|
|
|
|
if (!Settings.GetBool(AppSettings.SRCGEN_DISABLE_LABEL_LOCALIZATION, false)) {
|
|
|
|
|
mLocalizer.LocalPrefix = ":";
|
|
|
|
|
mLocalizer.Analyze();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Use UTF-8 encoding, without a byte-order mark.
|
|
|
|
|
using (StreamWriter sw = new StreamWriter(pathName, false, new UTF8Encoding(false))) {
|
|
|
|
|
mOutStream = sw;
|
|
|
|
|
|
|
|
|
|
if (Settings.GetBool(AppSettings.SRCGEN_ADD_IDENT_COMMENT, false)) {
|
|
|
|
|
// No version-specific stuff yet. We're generating code for v1.0.
|
|
|
|
|
OutputLine(SourceFormatter.FullLineCommentDelimiter +
|
|
|
|
|
string.Format(Properties.Resources.GENERATED_FOR_VERSION,
|
|
|
|
|
"Merlin 32", new CommonUtil.Version(1, 0)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
GenCommon.Generate(this, sw, worker);
|
|
|
|
|
}
|
|
|
|
|
mOutStream = null;
|
|
|
|
|
|
|
|
|
|
return pathNames;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// IGenerator
|
|
|
|
|
public void OutputDataOp(int offset) {
|
|
|
|
|
Formatter formatter = SourceFormatter;
|
|
|
|
|
byte[] data = Project.FileData;
|
|
|
|
|
Anattrib attr = Project.GetAnattrib(offset);
|
|
|
|
|
|
|
|
|
|
string labelStr = string.Empty;
|
|
|
|
|
if (attr.Symbol != null) {
|
|
|
|
|
labelStr = mLocalizer.ConvLabel(attr.Symbol.Label);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
string commentStr = SourceFormatter.FormatEolComment(Project.Comments[offset]);
|
|
|
|
|
string opcodeStr, operandStr;
|
|
|
|
|
|
|
|
|
|
FormatDescriptor dfd = attr.DataDescriptor;
|
|
|
|
|
Debug.Assert(dfd != null);
|
|
|
|
|
int length = dfd.Length;
|
|
|
|
|
Debug.Assert(length > 0);
|
|
|
|
|
|
|
|
|
|
bool multiLine = false;
|
|
|
|
|
switch (dfd.FormatType) {
|
|
|
|
|
case FormatDescriptor.Type.Default:
|
|
|
|
|
if (length != 1) {
|
|
|
|
|
Debug.Assert(false);
|
|
|
|
|
length = 1;
|
|
|
|
|
}
|
|
|
|
|
opcodeStr = sDataOpNames.DefineData1;
|
|
|
|
|
int operand = RawData.GetWord(data, offset, length, false);
|
|
|
|
|
operandStr = formatter.FormatHexValue(operand, length * 2);
|
|
|
|
|
break;
|
|
|
|
|
case FormatDescriptor.Type.NumericLE:
|
|
|
|
|
opcodeStr = sDataOpNames.GetDefineData(length);
|
|
|
|
|
operand = RawData.GetWord(data, offset, length, false);
|
|
|
|
|
operandStr = PseudoOp.FormatNumericOperand(formatter, Project.SymbolTable,
|
2018-10-26 22:45:39 +00:00
|
|
|
|
mLocalizer.LabelMap, dfd, operand, length,
|
|
|
|
|
PseudoOp.FormatNumericOpFlags.None);
|
2018-10-17 20:11:26 +00:00
|
|
|
|
break;
|
|
|
|
|
case FormatDescriptor.Type.NumericBE:
|
|
|
|
|
opcodeStr = sDataOpNames.GetDefineBigData(length);
|
|
|
|
|
if (opcodeStr == null) {
|
|
|
|
|
// Nothing defined, output as comma-separated single-byte values.
|
|
|
|
|
GenerateShortSequence(offset, length, out opcodeStr, out operandStr);
|
|
|
|
|
} else {
|
|
|
|
|
operand = RawData.GetWord(data, offset, length, true);
|
|
|
|
|
operandStr = PseudoOp.FormatNumericOperand(formatter, Project.SymbolTable,
|
2018-10-26 22:45:39 +00:00
|
|
|
|
mLocalizer.LabelMap, dfd, operand, length,
|
|
|
|
|
PseudoOp.FormatNumericOpFlags.None);
|
2018-10-17 20:11:26 +00:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case FormatDescriptor.Type.Fill:
|
|
|
|
|
opcodeStr = sDataOpNames.Fill;
|
|
|
|
|
operandStr = length + "," + formatter.FormatHexValue(data[offset], 2);
|
|
|
|
|
break;
|
|
|
|
|
case FormatDescriptor.Type.Dense:
|
|
|
|
|
multiLine = true;
|
|
|
|
|
opcodeStr = operandStr = null;
|
|
|
|
|
OutputDenseHex(offset, length, labelStr, commentStr);
|
|
|
|
|
break;
|
|
|
|
|
case FormatDescriptor.Type.String:
|
|
|
|
|
multiLine = true;
|
|
|
|
|
opcodeStr = operandStr = null;
|
|
|
|
|
OutputString(offset, labelStr, commentStr);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
opcodeStr = "???";
|
|
|
|
|
operandStr = "***";
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!multiLine) {
|
|
|
|
|
opcodeStr = formatter.FormatPseudoOp(opcodeStr);
|
|
|
|
|
OutputLine(labelStr, opcodeStr, operandStr, commentStr);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void OutputDenseHex(int offset, int length, string labelStr, string commentStr) {
|
|
|
|
|
Formatter formatter = SourceFormatter;
|
|
|
|
|
byte[] data = Project.FileData;
|
|
|
|
|
int maxPerLine = MAX_OPERAND_LEN / 2;
|
|
|
|
|
|
|
|
|
|
string opcodeStr = formatter.FormatPseudoOp(sDataOpNames.Dense);
|
|
|
|
|
for (int i = 0; i < length; i += maxPerLine) {
|
|
|
|
|
int subLen = length - i;
|
|
|
|
|
if (subLen > maxPerLine) {
|
|
|
|
|
subLen = maxPerLine;
|
|
|
|
|
}
|
|
|
|
|
string operandStr = formatter.FormatDenseHex(data, offset + i, subLen);
|
|
|
|
|
|
|
|
|
|
OutputLine(labelStr, opcodeStr, operandStr, commentStr);
|
|
|
|
|
labelStr = commentStr = string.Empty;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// IGenerator
|
2018-11-02 20:49:27 +00:00
|
|
|
|
public string ModifyOpcode(int offset, OpDef op) {
|
2018-10-17 20:11:26 +00:00
|
|
|
|
if (op.IsUndocumented) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
2018-11-02 20:49:27 +00:00
|
|
|
|
|
|
|
|
|
// The assembler works correctly if the symbol is defined as a two-digit hex
|
|
|
|
|
// value (e.g. "foo equ $80") but fails if it's four (e.g. "foo equ $0080"). We
|
|
|
|
|
// output symbols with minimal digits, but we have no control over labels when
|
|
|
|
|
// the code has a zero-page EQU. So if the operand is a reference to a user
|
|
|
|
|
// label, we need to output the instruction as hex.
|
|
|
|
|
if (op == OpDef.OpPEI_StackDPInd ||
|
|
|
|
|
op == OpDef.OpSTY_DPIndexX ||
|
|
|
|
|
op == OpDef.OpSTX_DPIndexY ||
|
|
|
|
|
op.AddrMode == OpDef.AddressMode.DPIndLong ||
|
|
|
|
|
op.AddrMode == OpDef.AddressMode.DPInd ||
|
|
|
|
|
op.AddrMode == OpDef.AddressMode.DPIndexXInd) {
|
|
|
|
|
FormatDescriptor dfd = Project.GetAnattrib(offset).DataDescriptor;
|
|
|
|
|
if (dfd != null && dfd.HasSymbol) {
|
|
|
|
|
// It has a symbol. See if the symbol target is a label (auto or user).
|
|
|
|
|
if (Project.SymbolTable.TryGetValue(dfd.SymbolRef.Label, out Symbol sym)) {
|
|
|
|
|
if (sym.IsInternalLabel) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return string.Empty;
|
2018-10-17 20:11:26 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// IGenerator
|
|
|
|
|
public void GenerateShortSequence(int offset, int length, out string opcode,
|
|
|
|
|
out string operand) {
|
|
|
|
|
Debug.Assert(length >= 1 && length <= 4);
|
|
|
|
|
|
|
|
|
|
// Use a comma-separated list of individual hex bytes.
|
|
|
|
|
opcode = sDataOpNames.DefineData1;
|
|
|
|
|
|
|
|
|
|
StringBuilder sb = new StringBuilder(length * 4);
|
|
|
|
|
for (int i = 0; i < length; i++) {
|
|
|
|
|
if (i != 0) {
|
|
|
|
|
sb.Append(',');
|
|
|
|
|
}
|
|
|
|
|
sb.Append(SourceFormatter.FormatHexValue(Project.FileData[offset + i], 2));
|
|
|
|
|
}
|
|
|
|
|
operand = sb.ToString();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// IGenerator
|
|
|
|
|
public void OutputAsmConfig() {
|
|
|
|
|
// nothing to do
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// IGenerator
|
|
|
|
|
public void OutputEquDirective(string name, string valueStr, string comment) {
|
|
|
|
|
OutputLine(name, SourceFormatter.FormatPseudoOp(sDataOpNames.EquDirective),
|
|
|
|
|
valueStr, SourceFormatter.FormatEolComment(comment));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// IGenerator
|
2018-10-23 23:06:29 +00:00
|
|
|
|
public void OutputOrgDirective(int offset, int address) {
|
2018-10-17 20:11:26 +00:00
|
|
|
|
OutputLine(string.Empty, SourceFormatter.FormatPseudoOp(sDataOpNames.OrgDirective),
|
|
|
|
|
SourceFormatter.FormatHexValue(address, 4), string.Empty);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// IGenerator
|
|
|
|
|
public void OutputRegWidthDirective(int offset, int prevM, int prevX, int newM, int newX) {
|
|
|
|
|
// prevM/prevX may be ambiguous for offset 0, but otherwise everything
|
|
|
|
|
// should be either 0 or 1.
|
|
|
|
|
Debug.Assert(newM == 0 || newM == 1);
|
|
|
|
|
Debug.Assert(newX == 0 || newX == 1);
|
|
|
|
|
|
|
|
|
|
if (offset == 0 && newM == 1 && newX == 1) {
|
|
|
|
|
// Assembler defaults to short regs, so we can skip this.
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
OutputLine(string.Empty,
|
|
|
|
|
SourceFormatter.FormatPseudoOp(sDataOpNames.RegWidthDirective),
|
|
|
|
|
"%" + newM + newX, string.Empty);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// IGenerator
|
|
|
|
|
public void OutputLine(string fullLine) {
|
|
|
|
|
mOutStream.WriteLine(fullLine);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// IGenerator
|
|
|
|
|
public void OutputLine(string label, string opcode, string operand, string comment) {
|
|
|
|
|
// Split long label, but not on EQU directives (confuses the assembler).
|
2018-10-21 04:24:28 +00:00
|
|
|
|
if (mLongLabelNewLine && label.Length >= mColumnWidths[0] &&
|
2018-10-21 23:36:48 +00:00
|
|
|
|
!string.Equals(opcode, sDataOpNames.EquDirective,
|
2018-10-17 20:11:26 +00:00
|
|
|
|
StringComparison.InvariantCultureIgnoreCase)) {
|
|
|
|
|
mOutStream.WriteLine(label);
|
|
|
|
|
label = string.Empty;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mLineBuilder.Clear();
|
2018-10-21 04:24:28 +00:00
|
|
|
|
TextUtil.AppendPaddedString(mLineBuilder, label, mColumnWidths[0]);
|
|
|
|
|
TextUtil.AppendPaddedString(mLineBuilder, opcode, mColumnWidths[0] + mColumnWidths[1]);
|
|
|
|
|
TextUtil.AppendPaddedString(mLineBuilder, operand,
|
|
|
|
|
mColumnWidths[0] + mColumnWidths[1] + mColumnWidths[2]);
|
2018-10-17 20:11:26 +00:00
|
|
|
|
if (string.IsNullOrEmpty(comment)) {
|
|
|
|
|
// Trim trailing spaces off of opcode or operand. If they want trailing
|
|
|
|
|
// spaces at the end of a comment, that's fine.
|
|
|
|
|
CommonUtil.TextUtil.TrimEnd(mLineBuilder);
|
|
|
|
|
} else {
|
|
|
|
|
mLineBuilder.Append(comment);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mOutStream.WriteLine(mLineBuilder.ToString());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private enum RevMode { Forward, Reverse, BlockReverse };
|
|
|
|
|
|
|
|
|
|
private void OutputString(int offset, string labelStr, string commentStr) {
|
|
|
|
|
// This gets complicated.
|
|
|
|
|
//
|
|
|
|
|
// For Dci, L8String, and L16String, the entire string needs to fit in the
|
|
|
|
|
// operand of one line. If it can't, we need to separate the length byte/word
|
|
|
|
|
// or inverted character out, and just dump the rest as ASCII. Computing the
|
|
|
|
|
// line length requires factoring delimiter character escapes. (NOTE: contrary
|
|
|
|
|
// to the documentation, STR and STRL do include trailing hex characters in the
|
|
|
|
|
// length calculation, so it's possible to escape delimiters.)
|
|
|
|
|
//
|
|
|
|
|
// For Reverse, we can span lines, but only if we emit the lines in
|
|
|
|
|
// backward order. Also, Merlin doesn't allow hex to be embedded in a REV
|
|
|
|
|
// operation, so we can't use REV if the string contains a delimiter.
|
|
|
|
|
//
|
|
|
|
|
// DciReverse is deprecated, but we can handle it as a Reverse string with a
|
|
|
|
|
// trailing byte on a following line.
|
|
|
|
|
//
|
|
|
|
|
// For aesthetic purposes, zero-length CString, L8String, and L16String
|
|
|
|
|
// should be output as DFB/DW zeroes rather than an empty string -- makes
|
|
|
|
|
// it easier to read.
|
|
|
|
|
|
|
|
|
|
Formatter formatter = SourceFormatter;
|
|
|
|
|
byte[] data = Project.FileData;
|
|
|
|
|
Anattrib attr = Project.GetAnattrib(offset);
|
|
|
|
|
FormatDescriptor dfd = attr.DataDescriptor;
|
|
|
|
|
Debug.Assert(dfd != null);
|
|
|
|
|
Debug.Assert(dfd.FormatType == FormatDescriptor.Type.String);
|
|
|
|
|
Debug.Assert(dfd.Length > 0);
|
|
|
|
|
|
|
|
|
|
bool highAscii = false;
|
|
|
|
|
int showZeroes = 0;
|
|
|
|
|
int leadingBytes = 0;
|
|
|
|
|
int trailingBytes = 0;
|
|
|
|
|
bool showLeading = false;
|
|
|
|
|
bool showTrailing = false;
|
|
|
|
|
RevMode revMode = RevMode.Forward;
|
|
|
|
|
|
|
|
|
|
switch (dfd.FormatSubType) {
|
|
|
|
|
case FormatDescriptor.SubType.None:
|
|
|
|
|
highAscii = (data[offset] & 0x80) != 0;
|
|
|
|
|
break;
|
|
|
|
|
case FormatDescriptor.SubType.Dci:
|
|
|
|
|
highAscii = (data[offset] & 0x80) != 0;
|
|
|
|
|
break;
|
|
|
|
|
case FormatDescriptor.SubType.Reverse:
|
|
|
|
|
highAscii = (data[offset] & 0x80) != 0;
|
|
|
|
|
revMode = RevMode.Reverse;
|
|
|
|
|
break;
|
|
|
|
|
case FormatDescriptor.SubType.DciReverse:
|
|
|
|
|
highAscii = (data[offset + dfd.Length - 1] & 0x80) != 0;
|
|
|
|
|
revMode = RevMode.Reverse;
|
|
|
|
|
break;
|
|
|
|
|
case FormatDescriptor.SubType.CString:
|
|
|
|
|
highAscii = (data[offset] & 0x80) != 0;
|
|
|
|
|
if (dfd.Length == 1) {
|
|
|
|
|
showZeroes = 1; // empty null-terminated string
|
|
|
|
|
}
|
|
|
|
|
trailingBytes = 1;
|
|
|
|
|
showTrailing = true;
|
|
|
|
|
break;
|
|
|
|
|
case FormatDescriptor.SubType.L8String:
|
|
|
|
|
if (dfd.Length > 1) {
|
|
|
|
|
highAscii = (data[offset + 1] & 0x80) != 0;
|
|
|
|
|
} else {
|
|
|
|
|
//showZeroes = 1;
|
|
|
|
|
}
|
|
|
|
|
leadingBytes = 1;
|
|
|
|
|
break;
|
|
|
|
|
case FormatDescriptor.SubType.L16String:
|
|
|
|
|
if (dfd.Length > 2) {
|
|
|
|
|
highAscii = (data[offset + 2] & 0x80) != 0;
|
|
|
|
|
} else {
|
|
|
|
|
//showZeroes = 2;
|
|
|
|
|
}
|
|
|
|
|
leadingBytes = 2;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
Debug.Assert(false);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (showZeroes != 0) {
|
|
|
|
|
// Empty string. Just output the length byte(s) or null terminator.
|
|
|
|
|
GenerateShortSequence(offset, showZeroes, out string opcode, out string operand);
|
|
|
|
|
OutputLine(labelStr, opcode, operand, commentStr);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Merlin 32 uses single-quote for low ASCII, double-quote for high ASCII. When
|
|
|
|
|
// quoting the delimiter we use a hexadecimal value. We need to bear in mind that
|
|
|
|
|
// we're forcing the characters to low ASCII, but the actual character being
|
|
|
|
|
// escaped might be in high ASCII. Hence delim vs. delimReplace.
|
|
|
|
|
char delim = highAscii ? '"' : '\'';
|
|
|
|
|
char delimReplace = highAscii ? ((char)(delim | 0x80)) : delim;
|
|
|
|
|
StringGather gath = null;
|
|
|
|
|
|
|
|
|
|
// Run the string through so we can see if it'll fit on one line. As a minor
|
|
|
|
|
// optimization, we skip this step for "generic" strings, which are probably
|
|
|
|
|
// the most common thing.
|
|
|
|
|
if (dfd.FormatSubType != FormatDescriptor.SubType.None) {
|
|
|
|
|
gath = new StringGather(this, labelStr, "???", commentStr, delim,
|
|
|
|
|
delimReplace, StringGather.ByteStyle.DenseHex, MAX_OPERAND_LEN, true);
|
|
|
|
|
FeedGath(gath, data, offset, dfd.Length, revMode, leadingBytes, showLeading,
|
|
|
|
|
trailingBytes, showTrailing);
|
|
|
|
|
Debug.Assert(gath.NumLinesOutput > 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
string opcodeStr;
|
|
|
|
|
|
|
|
|
|
switch (dfd.FormatSubType) {
|
|
|
|
|
case FormatDescriptor.SubType.None:
|
|
|
|
|
opcodeStr = highAscii ? sDataOpNames.StrGenericHi : sDataOpNames.StrGeneric;
|
|
|
|
|
break;
|
|
|
|
|
case FormatDescriptor.SubType.Dci:
|
|
|
|
|
if (gath.NumLinesOutput == 1) {
|
|
|
|
|
opcodeStr = highAscii ? sDataOpNames.StrDciHi : sDataOpNames.StrDci;
|
|
|
|
|
} else {
|
|
|
|
|
opcodeStr = highAscii ? sDataOpNames.StrGenericHi : sDataOpNames.StrGeneric;
|
|
|
|
|
trailingBytes = 1;
|
|
|
|
|
showTrailing = true;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case FormatDescriptor.SubType.Reverse:
|
|
|
|
|
if (gath.HasDelimiter) {
|
|
|
|
|
// can't include escaped delimiters in REV
|
|
|
|
|
opcodeStr = highAscii ? sDataOpNames.StrGenericHi : sDataOpNames.StrGeneric;
|
|
|
|
|
revMode = RevMode.Forward;
|
|
|
|
|
} else if (gath.NumLinesOutput > 1) {
|
|
|
|
|
opcodeStr = highAscii ? sDataOpNames.StrReverseHi : sDataOpNames.StrReverse;
|
|
|
|
|
revMode = RevMode.BlockReverse;
|
|
|
|
|
} else {
|
|
|
|
|
opcodeStr = highAscii ? sDataOpNames.StrReverseHi : sDataOpNames.StrReverse;
|
|
|
|
|
Debug.Assert(revMode == RevMode.Reverse);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case FormatDescriptor.SubType.DciReverse:
|
|
|
|
|
// Mostly punt -- output as ASCII with special handling for first byte.
|
|
|
|
|
opcodeStr = highAscii ? sDataOpNames.StrGenericHi : sDataOpNames.StrGeneric;
|
|
|
|
|
revMode = RevMode.Forward;
|
|
|
|
|
leadingBytes = 1;
|
|
|
|
|
showLeading = true;
|
|
|
|
|
break;
|
|
|
|
|
case FormatDescriptor.SubType.CString:
|
|
|
|
|
//opcodeStr = sDataOpNames.StrNullTerm[highAscii ? 1 : 0];
|
|
|
|
|
opcodeStr = highAscii ? sDataOpNames.StrGenericHi : sDataOpNames.StrGeneric;
|
|
|
|
|
break;
|
|
|
|
|
case FormatDescriptor.SubType.L8String:
|
|
|
|
|
if (gath.NumLinesOutput == 1) {
|
|
|
|
|
opcodeStr = highAscii ? sDataOpNames.StrLen8Hi : sDataOpNames.StrLen8;
|
|
|
|
|
} else {
|
|
|
|
|
opcodeStr = highAscii ? sDataOpNames.StrGenericHi : sDataOpNames.StrGeneric;
|
|
|
|
|
leadingBytes = 1;
|
|
|
|
|
showLeading = true;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case FormatDescriptor.SubType.L16String:
|
|
|
|
|
if (gath.NumLinesOutput == 1) {
|
|
|
|
|
opcodeStr = highAscii ? sDataOpNames.StrLen16Hi : sDataOpNames.StrLen16;
|
|
|
|
|
} else {
|
|
|
|
|
opcodeStr = highAscii ? sDataOpNames.StrGenericHi : sDataOpNames.StrGeneric;
|
|
|
|
|
leadingBytes = 2;
|
|
|
|
|
showLeading = true;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
Debug.Assert(false);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
opcodeStr = formatter.FormatPseudoOp(opcodeStr);
|
|
|
|
|
|
|
|
|
|
// Create a new StringGather, with the final opcode choice.
|
|
|
|
|
gath = new StringGather(this, labelStr, opcodeStr, commentStr, delim,
|
|
|
|
|
delimReplace, StringGather.ByteStyle.DenseHex, MAX_OPERAND_LEN, false);
|
|
|
|
|
FeedGath(gath, data, offset, dfd.Length, revMode, leadingBytes, showLeading,
|
|
|
|
|
trailingBytes, showTrailing);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Feeds the bytes into the StringGather.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private void FeedGath(StringGather gath, byte[] data, int offset, int length,
|
|
|
|
|
RevMode revMode, int leadingBytes, bool showLeading,
|
|
|
|
|
int trailingBytes, bool showTrailing) {
|
|
|
|
|
int startOffset = offset;
|
|
|
|
|
int strEndOffset = offset + length - trailingBytes;
|
|
|
|
|
|
|
|
|
|
if (showLeading) {
|
|
|
|
|
while (leadingBytes-- > 0) {
|
|
|
|
|
gath.WriteByte(data[offset++]);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
offset += leadingBytes;
|
|
|
|
|
}
|
|
|
|
|
if (revMode == RevMode.BlockReverse) {
|
|
|
|
|
const int maxPerLine = MAX_OPERAND_LEN - 2;
|
|
|
|
|
int numBlockLines = (length + maxPerLine - 1) / maxPerLine;
|
|
|
|
|
|
|
|
|
|
for (int chunk = 0; chunk < numBlockLines; chunk++) {
|
|
|
|
|
int chunkOffset = startOffset + chunk * maxPerLine;
|
|
|
|
|
int endOffset = chunkOffset + maxPerLine;
|
|
|
|
|
if (endOffset > strEndOffset) {
|
|
|
|
|
endOffset = strEndOffset;
|
|
|
|
|
}
|
|
|
|
|
for (int off = endOffset - 1; off >= chunkOffset; off--) {
|
|
|
|
|
gath.WriteChar((char)(data[off] & 0x7f));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
for (; offset < strEndOffset; offset++) {
|
|
|
|
|
if (revMode == RevMode.Forward) {
|
|
|
|
|
gath.WriteChar((char)(data[offset] & 0x7f));
|
|
|
|
|
} else if (revMode == RevMode.Reverse) {
|
|
|
|
|
int posn = startOffset + (strEndOffset - offset) - 1;
|
|
|
|
|
gath.WriteChar((char)(data[posn] & 0x7f));
|
|
|
|
|
} else {
|
|
|
|
|
Debug.Assert(false);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
while (showTrailing && trailingBytes-- > 0) {
|
|
|
|
|
gath.WriteByte(data[offset++]);
|
|
|
|
|
}
|
|
|
|
|
gath.Finish();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endregion IGenerator
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#region IAssembler
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Cross-assembler execution interface.
|
|
|
|
|
/// </summary>
|
2018-09-28 17:05:11 +00:00
|
|
|
|
public class AsmMerlin32 : IAssembler {
|
2018-10-21 04:24:28 +00:00
|
|
|
|
// Paths from generator.
|
|
|
|
|
private List<string> mPathNames;
|
2018-09-28 17:05:11 +00:00
|
|
|
|
|
2018-10-21 04:24:28 +00:00
|
|
|
|
// Directory to make current before executing assembler.
|
|
|
|
|
private string mWorkDirectory;
|
2018-09-28 17:05:11 +00:00
|
|
|
|
|
|
|
|
|
|
2018-10-18 00:12:37 +00:00
|
|
|
|
// IAssembler
|
|
|
|
|
public void GetExeIdentifiers(out string humanName, out string exeName) {
|
|
|
|
|
humanName = "Merlin Assembler";
|
|
|
|
|
exeName = "Merlin32";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// IAssembler
|
|
|
|
|
public AssemblerConfig GetDefaultConfig() {
|
|
|
|
|
return new AssemblerConfig(string.Empty, new int[] { 9, 6, 11, 74 });
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-28 17:05:11 +00:00
|
|
|
|
// IAssembler
|
|
|
|
|
public AssemblerVersion QueryVersion() {
|
2018-10-18 00:12:37 +00:00
|
|
|
|
AssemblerConfig config =
|
|
|
|
|
AssemblerConfig.GetConfig(AppSettings.Global, AssemblerInfo.Id.Merlin32);
|
|
|
|
|
if (config == null || string.IsNullOrEmpty(config.ExecutablePath)) {
|
2018-09-28 17:05:11 +00:00
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-18 00:12:37 +00:00
|
|
|
|
ShellCommand cmd = new ShellCommand(config.ExecutablePath, string.Empty,
|
|
|
|
|
Directory.GetCurrentDirectory(), null);
|
2018-09-28 17:05:11 +00:00
|
|
|
|
cmd.Execute();
|
|
|
|
|
if (string.IsNullOrEmpty(cmd.Stdout)) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Stdout: "C:\Src\WorkBench\Merlin32.exe v 1.0, (c) Brutal Deluxe ..."
|
|
|
|
|
// Other platforms may not have the ".exe". Find first occurrence of " v ".
|
|
|
|
|
|
|
|
|
|
const string PREFIX = " v "; // not expecting this to appear in the path
|
|
|
|
|
string str = cmd.Stdout;
|
|
|
|
|
int start = str.IndexOf(PREFIX);
|
|
|
|
|
int end = (start < 0) ? -1 : str.IndexOf(',', start);
|
|
|
|
|
|
|
|
|
|
if (start < 0 || end < 0 || start + PREFIX.Length >= end) {
|
|
|
|
|
Debug.WriteLine("Couldn't find version in " + str);
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
start += PREFIX.Length;
|
|
|
|
|
string versionStr = str.Substring(start, end - start);
|
|
|
|
|
CommonUtil.Version version = CommonUtil.Version.Parse(versionStr);
|
|
|
|
|
if (!version.IsValid) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
return new AssemblerVersion(versionStr, version);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// IAssembler
|
|
|
|
|
public void Configure(List<string> pathNames, string workDirectory) {
|
|
|
|
|
// Clone pathNames, in case the caller decides to modify the original.
|
2018-10-21 04:24:28 +00:00
|
|
|
|
mPathNames = new List<string>(pathNames.Count);
|
2018-09-28 17:05:11 +00:00
|
|
|
|
foreach (string str in pathNames) {
|
2018-10-21 04:24:28 +00:00
|
|
|
|
mPathNames.Add(str);
|
2018-09-28 17:05:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
2018-10-21 04:24:28 +00:00
|
|
|
|
mWorkDirectory = workDirectory;
|
2018-09-28 17:05:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// IAssembler
|
|
|
|
|
public AssemblerResults RunAssembler(BackgroundWorker worker) {
|
|
|
|
|
// Reduce input file to a partial path if possible. This is really just to make
|
|
|
|
|
// what we display to the user a little easier to read.
|
2018-10-21 04:24:28 +00:00
|
|
|
|
string pathName = mPathNames[0];
|
|
|
|
|
if (pathName.StartsWith(mWorkDirectory)) {
|
|
|
|
|
pathName = pathName.Remove(0, mWorkDirectory.Length + 1);
|
2018-09-28 17:05:11 +00:00
|
|
|
|
} else {
|
|
|
|
|
// Unexpected, but shouldn't be a problem.
|
|
|
|
|
Debug.WriteLine("NOTE: source file is not in work directory");
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-18 00:12:37 +00:00
|
|
|
|
AssemblerConfig config =
|
|
|
|
|
AssemblerConfig.GetConfig(AppSettings.Global, AssemblerInfo.Id.Merlin32);
|
|
|
|
|
if (string.IsNullOrEmpty(config.ExecutablePath)) {
|
2018-09-28 17:05:11 +00:00
|
|
|
|
Debug.WriteLine("Assembler not configured");
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-30 23:41:56 +00:00
|
|
|
|
worker.ReportProgress(0, Properties.Resources.PROGRESS_ASSEMBLING);
|
|
|
|
|
|
2018-09-28 17:05:11 +00:00
|
|
|
|
// Wrap pathname in quotes in case it has spaces.
|
|
|
|
|
// (Do we need to shell-escape quotes in the pathName?)
|
2018-10-18 00:12:37 +00:00
|
|
|
|
ShellCommand cmd = new ShellCommand(config.ExecutablePath, ". \"" + pathName + "\"",
|
2018-10-21 04:24:28 +00:00
|
|
|
|
mWorkDirectory, null);
|
2018-09-28 17:05:11 +00:00
|
|
|
|
cmd.Execute();
|
|
|
|
|
|
|
|
|
|
// Can't really do anything with a "cancel" request.
|
|
|
|
|
|
|
|
|
|
// Output filename is the input filename without the ".S". Since the filename
|
|
|
|
|
// was generated by us we can be confident in the format.
|
2018-10-21 04:24:28 +00:00
|
|
|
|
string outputFile = mPathNames[0].Substring(0, mPathNames[0].Length - 2);
|
2018-09-28 17:05:11 +00:00
|
|
|
|
|
|
|
|
|
return new AssemblerResults(cmd.FullCommandLine, cmd.ExitCode, cmd.Stdout,
|
|
|
|
|
cmd.Stderr, outputFile);
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-10-17 20:11:26 +00:00
|
|
|
|
|
|
|
|
|
#endregion IAssembler
|
2018-09-28 17:05:11 +00:00
|
|
|
|
}
|