/*
* Copyright 2019 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;
using System.Collections.Generic;
using System.Diagnostics;
namespace SourceGen {
///
/// Tracks a set of offsets that reference a single address or label.
///
/// This is used internally, when refactoring labels, as well as for the "references"
/// UI panel and label localizer.
///
public class XrefSet : IEnumerable {
///
/// Reference type. This is mostly useful for display to the user.
///
public enum XrefType {
Unknown = 0,
SubCallOp, // subroutine call
BranchOp, // branch instruction
RefFromData, // reference in data area, e.g. ".dd2 "
MemAccessOp, // instruction that accesses memory, or refers to an address
// TODO(someday): track 16-bit vs. 24-bit addressing, so we can show whether
// something is a "far" reference (and maybe carry this into auto-label annotation)
}
///
/// Cross-reference descriptor. Instances are immutable.
///
public class Xref {
///
/// Offset of start of instruction or data that refers to the target offset.
///
public int Offset { get; private set; }
///
/// True if this reference is by name.
///
///
/// The time this is of use is when determining the set of project/platform symbols
/// that are actually used. If we have FOO1=$101 and FOO2=$102, and we LDA FOO1
/// and LDA FOO1+1, we only want to output an equate for FOO1 even though FOO2's
/// address was referenced.
///
public bool IsByName { get; private set; }
///
/// Type of reference.
///
public XrefType Type { get; private set; }
///
/// For Type==MemAccessOp, what type of memory access is performed.
///
public Asm65.OpDef.MemoryEffect AccType { get; private set; }
///
/// For Type==MemAccessOp, true if the instruction applies an index offset, to
/// the operand, meaning the referenced address might not actually be accessed.
///
public bool IsIndexedAccess { get; private set; }
///
/// Adjustment to symbol. For example, "LDA label+2" adds an xref entry to
/// "label", with an adjustment of +2.
///
public int Adjustment { get; private set; }
public Xref(int offset, bool isByName, XrefType type,
Asm65.OpDef.MemoryEffect accType, bool isIndexedAccess, int adjustment) {
Offset = offset;
IsByName = isByName;
Type = type;
AccType = accType;
IsIndexedAccess = isIndexedAccess;
Adjustment = adjustment;
}
public override string ToString() {
return "Xref off=+" + Offset.ToString("x6") + " sym=" + IsByName +
" type=" + Type + " accType= " + AccType + " idx=" + IsIndexedAccess +
" adj=" + Adjustment;
}
}
///
/// Internal storage for xrefs.
///
private List mRefs = new List();
///
/// Constructs an empty set.
///
public XrefSet() { }
///
/// Returns the number of cross-references in the set.
///
public int Count { get { return mRefs.Count; } }
///
/// Removes all entries from the set.
///
public void Clear() {
mRefs.Clear();
}
///
/// Returns the Nth entry in the set.
///
public Xref this[int index] {
get {
return mRefs[index];
}
}
///
/// Adds an xref to the set.
///
public void Add(Xref xref) {
// TODO(someday): not currently enforcing set behavior; start by adding .equals to
// Xref, then check Contains before allowing Add. (Should probably complain
// loudly if item already exists, since we're not expecting that.)
mRefs.Add(xref);
}
// IEnumerable
public IEnumerator GetEnumerator() {
return ((IEnumerable)mRefs).GetEnumerator();
}
// IEnumerable, generic
IEnumerator IEnumerable.GetEnumerator() {
return ((IEnumerable)mRefs).GetEnumerator();
}
public override string ToString() {
return "[XrefSet count=" + Count + "]";
}
}
}