mirror of
https://github.com/fadden/6502bench.git
synced 2024-11-09 00:06:05 +00:00
327ad4fbbc
A "cooked" form of the relocation data is added to the project, for use during data analysis. Also, changed the data grids in the segment viewer to allow multi- select, so users can copy & paste the contents.
171 lines
6.4 KiB
C#
171 lines
6.4 KiB
C#
/*
|
|
* Copyright 2020 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.Runtime.CompilerServices;
|
|
using System.Windows;
|
|
|
|
using Asm65;
|
|
|
|
namespace SourceGen.Tools.Omf.WpfGui {
|
|
/// <summary>
|
|
/// Apple IIgs OMF segment viewer.
|
|
/// </summary>
|
|
public partial class OmfSegmentViewer : Window, INotifyPropertyChanged {
|
|
//private OmfFile mOmfFile;
|
|
private OmfSegment mOmfSeg;
|
|
private Formatter mFormatter;
|
|
|
|
// INotifyPropertyChanged implementation
|
|
public event PropertyChangedEventHandler PropertyChanged;
|
|
private void OnPropertyChanged([CallerMemberName] string propertyName = "") {
|
|
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
|
|
}
|
|
|
|
private string mFileOffsetLenStr;
|
|
public string FileOffsetLenStr {
|
|
get { return mFileOffsetLenStr; }
|
|
set { mFileOffsetLenStr = value; OnPropertyChanged(); }
|
|
}
|
|
|
|
private string mRecordHeaderStr;
|
|
public string RecordHeaderStr {
|
|
get { return mRecordHeaderStr; }
|
|
set { mRecordHeaderStr = value; OnPropertyChanged(); }
|
|
}
|
|
|
|
private string mRelocHeaderStr;
|
|
public string RelocHeaderStr {
|
|
get { return mRelocHeaderStr; }
|
|
set { mRelocHeaderStr = value; OnPropertyChanged(); }
|
|
}
|
|
|
|
public class HeaderItem {
|
|
public string Name { get; private set; }
|
|
public string Value { get; private set; }
|
|
public string Note { get; private set; }
|
|
|
|
public HeaderItem(string name, string value, string note) {
|
|
Name = name;
|
|
Value = value;
|
|
Note = note;
|
|
}
|
|
}
|
|
public List<HeaderItem> HeaderItems { get; private set; } = new List<HeaderItem>();
|
|
|
|
public List<OmfRecord> RecordItems { get; private set; }
|
|
|
|
/// <summary>
|
|
/// Formatted version of OmfReloc object.
|
|
/// </summary>
|
|
public class RelocItem {
|
|
public string Width { get; private set; }
|
|
public string Shift { get; private set; }
|
|
public string Offset { get; private set; }
|
|
public string RelOffset { get; private set; }
|
|
public string FileNum { get; private set; }
|
|
public string SegNum { get; private set; }
|
|
public string SuperType { get; private set; }
|
|
|
|
public OmfReloc RelocRef { get; private set; }
|
|
|
|
public RelocItem(OmfReloc reloc, Formatter formatter) {
|
|
RelocRef = reloc;
|
|
|
|
Width = reloc.Width.ToString();
|
|
if (reloc.Shift == 0) {
|
|
Shift = string.Empty;
|
|
} else if (reloc.Shift < 0) {
|
|
Shift = ">> " + -reloc.Shift;
|
|
} else {
|
|
Shift = "<< " + reloc.Shift;
|
|
}
|
|
Offset = formatter.FormatHexValue(reloc.Offset, 4);
|
|
RelOffset = formatter.FormatHexValue(reloc.RelOffset, 4);
|
|
FileNum = reloc.FileNum < 0 ? string.Empty : reloc.FileNum.ToString();
|
|
SegNum = reloc.SegNum < 0 ? string.Empty : reloc.SegNum.ToString();
|
|
if (reloc.SuperType < 0) {
|
|
SuperType = string.Empty;
|
|
} else if (reloc.SuperType <= 1) {
|
|
SuperType = "RELOC" + (reloc.SuperType + 2);
|
|
} else {
|
|
SuperType = "INTERSEG" + (reloc.SuperType - 1);
|
|
}
|
|
}
|
|
}
|
|
public List<RelocItem> RelocItems { get; private set; }
|
|
public bool HasRelocs { get { return RelocItems.Count != 0; } }
|
|
|
|
|
|
/// <summary>
|
|
/// Constructor.
|
|
/// </summary>
|
|
/// <param name="owner">Parent window.</param>
|
|
/// <param name="omfFile">OMF file object.</param>
|
|
/// <param name="omfSeg">Segment to view. Must be part of omfFile.</param>
|
|
/// <param name="formatter">Text formatter.</param>
|
|
public OmfSegmentViewer(Window owner, OmfFile omfFile, OmfSegment omfSeg,
|
|
Formatter formatter) {
|
|
InitializeComponent();
|
|
Owner = owner;
|
|
DataContext = this;
|
|
|
|
//mOmfFile = omfFile;
|
|
mOmfSeg = omfSeg;
|
|
mFormatter = formatter;
|
|
|
|
string fmt = (string)FindResource("str_FileOffsetLenFmt");
|
|
FileOffsetLenStr = string.Format(fmt,
|
|
mFormatter.FormatOffset24(omfSeg.FileOffset),
|
|
omfSeg.FileLength,
|
|
mFormatter.FormatHexValue(omfSeg.FileLength, 4));
|
|
|
|
GenerateHeaderItems();
|
|
|
|
RecordItems = omfSeg.Records;
|
|
RelocItems = new List<RelocItem>(omfSeg.Relocs.Count);
|
|
foreach (OmfReloc omfRel in omfSeg.Relocs) {
|
|
RelocItems.Add(new RelocItem(omfRel, formatter));
|
|
}
|
|
RelocItems.Sort(delegate (RelocItem a, RelocItem b) {
|
|
return a.RelocRef.Offset - b.RelocRef.Offset;
|
|
});
|
|
|
|
fmt = (string)FindResource("str_RecordHeaderFmt");
|
|
RecordHeaderStr = string.Format(fmt, RecordItems.Count);
|
|
fmt = (string)FindResource("str_RelocHeaderFmt");
|
|
RelocHeaderStr = string.Format(fmt, RelocItems.Count);
|
|
}
|
|
|
|
private void GenerateHeaderItems() {
|
|
foreach (OmfSegment.NameValueNote nvn in mOmfSeg.RawValues) {
|
|
string value;
|
|
if (nvn.Value is int) {
|
|
int byteWidth = nvn.Width;
|
|
if (byteWidth > 3) {
|
|
byteWidth = 3;
|
|
}
|
|
value = mFormatter.FormatHexValue((int)nvn.Value, byteWidth * 2);
|
|
} else {
|
|
value = nvn.Value.ToString();
|
|
}
|
|
HeaderItems.Add(new HeaderItem(nvn.Name, value, nvn.Note));
|
|
}
|
|
}
|
|
}
|
|
}
|