2019-08-30 18:33:05 -07:00
|
|
|
|
/*
|
|
|
|
|
* 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.Generic;
|
|
|
|
|
using System.Diagnostics;
|
|
|
|
|
|
|
|
|
|
namespace SourceGen {
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Given a list of LocalVariableTables, this determines the mapping of values to symbols
|
|
|
|
|
/// at a specific offset.
|
|
|
|
|
/// </summary>
|
2019-11-08 20:44:45 -08:00
|
|
|
|
/// <remarks>
|
|
|
|
|
/// We guarantee that the label will be unique within its scope. This happens at two
|
|
|
|
|
/// different levels:
|
|
|
|
|
/// (1) If the local variable label is present in the main symbol table, we use the
|
|
|
|
|
/// "de-duplication" table to remap it. We try not to let this happen, but it can.
|
Label rework, part 6
Correct handling of local variables. We now correctly uniquify them
with regard to non-unique labels. Because local vars can effectively
have global scope we mostly want to treat them as global, but they're
uniquified relative to other globals very late in the process, so we
can't just throw them in the symbol table and be done. Fortunately
local variables exist in a separate namespace, so we just need to
uniquify the variables relative to the post-localization symbol table.
In other words, we take the symbol table, apply the label map, and
rename any variable that clashes.
This also fixes an older problem where we weren't masking the
leading '_' on variable labels when generating 64tass output.
The code list now makes non-unique labels obvious, but you can't tell
the difference between unique global and unique local. What's more,
the default type value in Edit Label is now adjusted to Global for
unique locals that were auto-generated. To make it a bit easier to
figure out what's what, the Info panel now has a "label type" line
that reports the type.
The 2023-non-unique-labels test had some additional tests added to
exercise conflicts with local variables. The 2019-local-variables
test output changed slightly because the de-duplicated variable
naming convention was simplified.
2019-11-18 13:26:03 -08:00
|
|
|
|
/// The symbol table is latched when the object is constructed.
|
2019-11-08 20:44:45 -08:00
|
|
|
|
/// (2) If the assembler doesn't define a way to re-use variable names, we make them
|
Label rework, part 6
Correct handling of local variables. We now correctly uniquify them
with regard to non-unique labels. Because local vars can effectively
have global scope we mostly want to treat them as global, but they're
uniquified relative to other globals very late in the process, so we
can't just throw them in the symbol table and be done. Fortunately
local variables exist in a separate namespace, so we just need to
uniquify the variables relative to the post-localization symbol table.
In other words, we take the symbol table, apply the label map, and
rename any variable that clashes.
This also fixes an older problem where we weren't masking the
leading '_' on variable labels when generating 64tass output.
The code list now makes non-unique labels obvious, but you can't tell
the difference between unique global and unique local. What's more,
the default type value in Edit Label is now adjusted to Global for
unique locals that were auto-generated. To make it a bit easier to
figure out what's what, the Info panel now has a "label type" line
that reports the type.
The 2023-non-unique-labels test had some additional tests added to
exercise conflicts with local variables. The 2019-local-variables
test output changed slightly because the de-duplicated variable
naming convention was simplified.
2019-11-18 13:26:03 -08:00
|
|
|
|
/// globally unique. [currently not needed]
|
Fix various local variable de-duplication bugs
In 1.5.0-dev1, as part of changes to the way label localization
works, the local variable de-duplicator started checking against a
filtered copy of the symbol table. Unfortunately it never
re-generated the table, so a long-lived LocalVariableLookup (like
the one used by LineListGen) would set up the dup map wrong and
be inconsistent with other parts of the program.
We now regenerate the table on every Reset().
The de-duplication stuff also had problems when opcodes and
operands were double-clicked on. When the opcode is clicked, the
selection should jump to the appropriate variable declaration, but
it wasn't being found because the label generated in the list was
in its original form. Fixed.
When an instruction operand is double-clicked, the instruction operand
editor opens with an "edit variable" shortcut. This was showing
the de-duplicated name, which isn't necessarily a bad thing, but it
was passing that value on to the DefSymbol editor, which thought it
was being asked to create a new entry. Fixed. (Entering the editor
through the LvTable editor works correctly, with nary a de-duplicated
name in sight. You'll be forced to rename it because it'll fail the
uniqueness test.)
References to de-duplicated local variables were getting lost when
the symbol's label was replaced (due largely to a convenient but
flawed shortcut: xrefs are attached to DefSymbol objects). Fixed by
linking the XrefSets.
Given the many issues and their relative subtlety, I decided to make
the modified names more obvious, and went back to the "_DUPn" naming
strategy. (I'm also considering just making it an error and
discarding conflicting entries during analysis... this is much more
complicated than I expected it to be.)
Quick tests can be performed in 2019-local-variables:
- go to +000026, double-click on the opcode, confirm sel change
- go to +000026, double-click on the operand, confirm orig name
shown in shortcut and that shortcut opens editor with orig name
- go to +00001a, down a line, click on PROJ_ZERO_DUP1 and confirm
that it has a single reference (from +000026)
- double-click on var table and confirm editing entry
2020-01-13 17:54:47 -08:00
|
|
|
|
///
|
|
|
|
|
/// De-duplication changes the label in *most* circumstances. We want to show the de-dup
|
|
|
|
|
/// form everywhere except for the LvTable editor and the Lv editor. Using the correct
|
|
|
|
|
/// string at the correct time is necessary for some basic editor operations:
|
|
|
|
|
/// - Double-click on the opcode of LDA [lvar]. The selection should jump to the specific
|
|
|
|
|
/// entry in the LvTable.
|
|
|
|
|
/// - Double-click on the operand of LDA [lvar]. The instruction operand editor should
|
|
|
|
|
/// open and offer to edit the de-dup form. Clicking on the shortcut button should open
|
|
|
|
|
/// the Lv editor, with the original label shown (and perhaps a warning about
|
|
|
|
|
/// non-uniqueness).
|
|
|
|
|
/// - Double-click on the LvTable. The table listing should show the original label.
|
|
|
|
|
/// - Click on a de-duped entry and verify that the correct cross-references exist.
|
|
|
|
|
///
|
|
|
|
|
/// To reduce confusion, the fact that something has been de-duped should be obvious.
|
|
|
|
|
///
|
|
|
|
|
/// A few quick clicks in the 2019-local-variables test should confirm these.
|
2019-11-08 20:44:45 -08:00
|
|
|
|
/// </remarks>
|
2019-08-30 18:33:05 -07:00
|
|
|
|
public class LocalVariableLookup {
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// List of tables. The table's file offset is used as the key.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private SortedList<int, LocalVariableTable> mLvTables;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
Label rework, part 6
Correct handling of local variables. We now correctly uniquify them
with regard to non-unique labels. Because local vars can effectively
have global scope we mostly want to treat them as global, but they're
uniquified relative to other globals very late in the process, so we
can't just throw them in the symbol table and be done. Fortunately
local variables exist in a separate namespace, so we just need to
uniquify the variables relative to the post-localization symbol table.
In other words, we take the symbol table, apply the label map, and
rename any variable that clashes.
This also fixes an older problem where we weren't masking the
leading '_' on variable labels when generating 64tass output.
The code list now makes non-unique labels obvious, but you can't tell
the difference between unique global and unique local. What's more,
the default type value in Edit Label is now adjusted to Global for
unique locals that were auto-generated. To make it a bit easier to
figure out what's what, the Info panel now has a "label type" line
that reports the type.
The 2023-non-unique-labels test had some additional tests added to
exercise conflicts with local variables. The 2019-local-variables
test output changed slightly because the de-duplicated variable
naming convention was simplified.
2019-11-18 13:26:03 -08:00
|
|
|
|
/// Reference to project, so we can query the Anattrib array to identify "hidden" tables.
|
2019-08-30 18:33:05 -07:00
|
|
|
|
/// </summary>
|
Label rework, part 6
Correct handling of local variables. We now correctly uniquify them
with regard to non-unique labels. Because local vars can effectively
have global scope we mostly want to treat them as global, but they're
uniquified relative to other globals very late in the process, so we
can't just throw them in the symbol table and be done. Fortunately
local variables exist in a separate namespace, so we just need to
uniquify the variables relative to the post-localization symbol table.
In other words, we take the symbol table, apply the label map, and
rename any variable that clashes.
This also fixes an older problem where we weren't masking the
leading '_' on variable labels when generating 64tass output.
The code list now makes non-unique labels obvious, but you can't tell
the difference between unique global and unique local. What's more,
the default type value in Edit Label is now adjusted to Global for
unique locals that were auto-generated. To make it a bit easier to
figure out what's what, the Info panel now has a "label type" line
that reports the type.
The 2023-non-unique-labels test had some additional tests added to
exercise conflicts with local variables. The 2019-local-variables
test output changed slightly because the de-duplicated variable
naming convention was simplified.
2019-11-18 13:26:03 -08:00
|
|
|
|
private DisasmProject mProject;
|
2019-08-30 18:33:05 -07:00
|
|
|
|
|
2019-08-31 21:54:20 -07:00
|
|
|
|
/// <summary>
|
Label rework, part 6
Correct handling of local variables. We now correctly uniquify them
with regard to non-unique labels. Because local vars can effectively
have global scope we mostly want to treat them as global, but they're
uniquified relative to other globals very late in the process, so we
can't just throw them in the symbol table and be done. Fortunately
local variables exist in a separate namespace, so we just need to
uniquify the variables relative to the post-localization symbol table.
In other words, we take the symbol table, apply the label map, and
rename any variable that clashes.
This also fixes an older problem where we weren't masking the
leading '_' on variable labels when generating 64tass output.
The code list now makes non-unique labels obvious, but you can't tell
the difference between unique global and unique local. What's more,
the default type value in Edit Label is now adjusted to Global for
unique locals that were auto-generated. To make it a bit easier to
figure out what's what, the Info panel now has a "label type" line
that reports the type.
The 2023-non-unique-labels test had some additional tests added to
exercise conflicts with local variables. The 2019-local-variables
test output changed slightly because the de-duplicated variable
naming convention was simplified.
2019-11-18 13:26:03 -08:00
|
|
|
|
/// Set to true when generating symbols for assemblers like 64tass, which assign a
|
|
|
|
|
/// special meaning to labels with leading underscores.
|
2019-08-31 21:54:20 -07:00
|
|
|
|
/// </summary>
|
Label rework, part 6
Correct handling of local variables. We now correctly uniquify them
with regard to non-unique labels. Because local vars can effectively
have global scope we mostly want to treat them as global, but they're
uniquified relative to other globals very late in the process, so we
can't just throw them in the symbol table and be done. Fortunately
local variables exist in a separate namespace, so we just need to
uniquify the variables relative to the post-localization symbol table.
In other words, we take the symbol table, apply the label map, and
rename any variable that clashes.
This also fixes an older problem where we weren't masking the
leading '_' on variable labels when generating 64tass output.
The code list now makes non-unique labels obvious, but you can't tell
the difference between unique global and unique local. What's more,
the default type value in Edit Label is now adjusted to Global for
unique locals that were auto-generated. To make it a bit easier to
figure out what's what, the Info panel now has a "label type" line
that reports the type.
The 2023-non-unique-labels test had some additional tests added to
exercise conflicts with local variables. The 2019-local-variables
test output changed slightly because the de-duplicated variable
naming convention was simplified.
2019-11-18 13:26:03 -08:00
|
|
|
|
private bool mMaskLeadingUnderscores;
|
2019-08-31 21:54:20 -07:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Set to true if we want all variables to be globally unique (because the assembler
|
|
|
|
|
/// can't redefine them).
|
|
|
|
|
/// </summary>
|
|
|
|
|
private bool mDoUniquify;
|
|
|
|
|
|
Label rework, part 6
Correct handling of local variables. We now correctly uniquify them
with regard to non-unique labels. Because local vars can effectively
have global scope we mostly want to treat them as global, but they're
uniquified relative to other globals very late in the process, so we
can't just throw them in the symbol table and be done. Fortunately
local variables exist in a separate namespace, so we just need to
uniquify the variables relative to the post-localization symbol table.
In other words, we take the symbol table, apply the label map, and
rename any variable that clashes.
This also fixes an older problem where we weren't masking the
leading '_' on variable labels when generating 64tass output.
The code list now makes non-unique labels obvious, but you can't tell
the difference between unique global and unique local. What's more,
the default type value in Edit Label is now adjusted to Global for
unique locals that were auto-generated. To make it a bit easier to
figure out what's what, the Info panel now has a "label type" line
that reports the type.
The 2023-non-unique-labels test had some additional tests added to
exercise conflicts with local variables. The 2019-local-variables
test output changed slightly because the de-duplicated variable
naming convention was simplified.
2019-11-18 13:26:03 -08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// List of all non-variable symbols, for uniquification. This is generated from the
|
|
|
|
|
/// project symbol table. When generating assembly sources, the labels are transformed
|
|
|
|
|
/// through the label map.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private Dictionary<string, Symbol> mAllNvSymbols;
|
|
|
|
|
|
Fix various local variable de-duplication bugs
In 1.5.0-dev1, as part of changes to the way label localization
works, the local variable de-duplicator started checking against a
filtered copy of the symbol table. Unfortunately it never
re-generated the table, so a long-lived LocalVariableLookup (like
the one used by LineListGen) would set up the dup map wrong and
be inconsistent with other parts of the program.
We now regenerate the table on every Reset().
The de-duplication stuff also had problems when opcodes and
operands were double-clicked on. When the opcode is clicked, the
selection should jump to the appropriate variable declaration, but
it wasn't being found because the label generated in the list was
in its original form. Fixed.
When an instruction operand is double-clicked, the instruction operand
editor opens with an "edit variable" shortcut. This was showing
the de-duplicated name, which isn't necessarily a bad thing, but it
was passing that value on to the DefSymbol editor, which thought it
was being asked to create a new entry. Fixed. (Entering the editor
through the LvTable editor works correctly, with nary a de-duplicated
name in sight. You'll be forced to rename it because it'll fail the
uniqueness test.)
References to de-duplicated local variables were getting lost when
the symbol's label was replaced (due largely to a convenient but
flawed shortcut: xrefs are attached to DefSymbol objects). Fixed by
linking the XrefSets.
Given the many issues and their relative subtlety, I decided to make
the modified names more obvious, and went back to the "_DUPn" naming
strategy. (I'm also considering just making it an error and
discarding conflicting entries during analysis... this is much more
complicated than I expected it to be.)
Quick tests can be performed in 2019-local-variables:
- go to +000026, double-click on the opcode, confirm sel change
- go to +000026, double-click on the operand, confirm orig name
shown in shortcut and that shortcut opens editor with orig name
- go to +00001a, down a line, click on PROJ_ZERO_DUP1 and confirm
that it has a single reference (from +000026)
- double-click on var table and confirm editing entry
2020-01-13 17:54:47 -08:00
|
|
|
|
private Dictionary<string, string> mLabelMap;
|
|
|
|
|
|
2019-08-31 14:10:59 -07:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Label uniquification helper.
|
2019-08-31 21:54:20 -07:00
|
|
|
|
///
|
|
|
|
|
/// The BaseLabel does not change, but Label is updated by MakeUnique.
|
2019-08-31 14:10:59 -07:00
|
|
|
|
/// </summary>
|
2019-09-01 10:55:19 -07:00
|
|
|
|
/// <remarks>
|
|
|
|
|
/// LvLookup is run multiple times, and can be restarted in the middle of a run. It's
|
|
|
|
|
/// essential that UniqueLabel behaves deterministically. For this to happen, the
|
|
|
|
|
/// contents of SymbolTable can't change in a way that affects the outcome unless it
|
|
|
|
|
/// also causes us to redo the uniquification. This mostly means that we have to be
|
|
|
|
|
/// very careful about creating duplicate symbols, so that we don't get halfway through
|
|
|
|
|
/// the analysis pass and invalidate our previous work. It's best to leave
|
|
|
|
|
/// uniquification disabled until we're generating assembly source code.
|
|
|
|
|
///
|
|
|
|
|
/// The issues also make it hard to do the uniquification once, rather than every time we
|
|
|
|
|
/// walk the code. Not all symbol changes cause a re-analysis (e.g. renaming a user
|
|
|
|
|
/// label does not), and we don't want to fill the symbol table with the uniquified
|
|
|
|
|
/// names because it could block user labels that would otherwise be valid.
|
|
|
|
|
/// </remarks>
|
2019-08-31 14:10:59 -07:00
|
|
|
|
private class UniqueLabel {
|
|
|
|
|
public string BaseLabel { get; private set; }
|
|
|
|
|
public string Label { get; private set; }
|
|
|
|
|
private int Counter { get; set; }
|
|
|
|
|
|
|
|
|
|
public UniqueLabel(string baseLabel) {
|
|
|
|
|
Label = BaseLabel = baseLabel;
|
|
|
|
|
Counter = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Updates the Label to be unique. Call this when a symbol is defined or
|
|
|
|
|
/// re-defined.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="symbolTable">Symbol table, for uniqueness check.</param>
|
Label rework, part 6
Correct handling of local variables. We now correctly uniquify them
with regard to non-unique labels. Because local vars can effectively
have global scope we mostly want to treat them as global, but they're
uniquified relative to other globals very late in the process, so we
can't just throw them in the symbol table and be done. Fortunately
local variables exist in a separate namespace, so we just need to
uniquify the variables relative to the post-localization symbol table.
In other words, we take the symbol table, apply the label map, and
rename any variable that clashes.
This also fixes an older problem where we weren't masking the
leading '_' on variable labels when generating 64tass output.
The code list now makes non-unique labels obvious, but you can't tell
the difference between unique global and unique local. What's more,
the default type value in Edit Label is now adjusted to Global for
unique locals that were auto-generated. To make it a bit easier to
figure out what's what, the Info panel now has a "label type" line
that reports the type.
The 2023-non-unique-labels test had some additional tests added to
exercise conflicts with local variables. The 2019-local-variables
test output changed slightly because the de-duplicated variable
naming convention was simplified.
2019-11-18 13:26:03 -08:00
|
|
|
|
public void MakeUnique(Dictionary<string, Symbol> allNvSymbols) {
|
2019-08-31 14:10:59 -07:00
|
|
|
|
// The main symbol table might have user-supplied labels like "ptr_2", so we
|
|
|
|
|
// need to keep testing against that. However, it should not be possible for
|
|
|
|
|
// us to clash with other uniquified variables. So we don't need to check
|
|
|
|
|
// for clashes in the UniqueLabel list.
|
|
|
|
|
//
|
|
|
|
|
// It *is* possible to clash with other variable base names, so we can't
|
|
|
|
|
// exclude variables from our SymbolTable lookup.
|
|
|
|
|
string testLabel;
|
|
|
|
|
do {
|
|
|
|
|
Counter++;
|
|
|
|
|
testLabel = BaseLabel + "_" + Counter;
|
Label rework, part 6
Correct handling of local variables. We now correctly uniquify them
with regard to non-unique labels. Because local vars can effectively
have global scope we mostly want to treat them as global, but they're
uniquified relative to other globals very late in the process, so we
can't just throw them in the symbol table and be done. Fortunately
local variables exist in a separate namespace, so we just need to
uniquify the variables relative to the post-localization symbol table.
In other words, we take the symbol table, apply the label map, and
rename any variable that clashes.
This also fixes an older problem where we weren't masking the
leading '_' on variable labels when generating 64tass output.
The code list now makes non-unique labels obvious, but you can't tell
the difference between unique global and unique local. What's more,
the default type value in Edit Label is now adjusted to Global for
unique locals that were auto-generated. To make it a bit easier to
figure out what's what, the Info panel now has a "label type" line
that reports the type.
The 2023-non-unique-labels test had some additional tests added to
exercise conflicts with local variables. The 2019-local-variables
test output changed slightly because the de-duplicated variable
naming convention was simplified.
2019-11-18 13:26:03 -08:00
|
|
|
|
} while (allNvSymbols.TryGetValue(testLabel, out Symbol unused));
|
2019-08-31 14:10:59 -07:00
|
|
|
|
Label = testLabel;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
private Dictionary<string, UniqueLabel> mUniqueLabels;
|
|
|
|
|
|
2019-08-30 18:33:05 -07:00
|
|
|
|
/// <summary>
|
2019-08-31 21:54:20 -07:00
|
|
|
|
/// Duplicate label re-map. This is applied before uniquification.
|
2019-08-30 18:33:05 -07:00
|
|
|
|
/// </summary>
|
2019-08-31 21:54:20 -07:00
|
|
|
|
/// <remarks>
|
|
|
|
|
/// It's hard to do this as part of uniquification because the remapped base name ends
|
2019-09-01 10:55:19 -07:00
|
|
|
|
/// up in the symbol table, and the uniquifier isn't able to tell that the entry in the
|
2019-08-31 21:54:20 -07:00
|
|
|
|
/// symbol table is itself. The logic is simpler if we just rename the label before
|
|
|
|
|
/// the uniquifier ever sees it.
|
2019-09-01 16:40:54 -07:00
|
|
|
|
///
|
|
|
|
|
/// I feel like there has to be a simpler way to do this. This'll do for now.
|
2019-08-31 21:54:20 -07:00
|
|
|
|
/// </remarks>
|
|
|
|
|
private Dictionary<string, string> mDupRemap;
|
2019-08-30 18:33:05 -07:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Most recently processed offset.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private int mRecentOffset;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Symbols defined at mRecentOffset.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private List<DefSymbol> mRecentSymbols;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Cumulative symbols defined at the current offset.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private LocalVariableTable mCurrentTable;
|
|
|
|
|
|
2019-08-31 21:54:20 -07:00
|
|
|
|
// Next point of interest.
|
2019-08-30 18:33:05 -07:00
|
|
|
|
private int mNextLvtIndex;
|
|
|
|
|
private int mNextLvtOffset;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Constructor.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="lvTables">List of tables from the DisasmProject.</param>
|
|
|
|
|
/// <param name="project">Project reference.</param>
|
Label rework, part 6
Correct handling of local variables. We now correctly uniquify them
with regard to non-unique labels. Because local vars can effectively
have global scope we mostly want to treat them as global, but they're
uniquified relative to other globals very late in the process, so we
can't just throw them in the symbol table and be done. Fortunately
local variables exist in a separate namespace, so we just need to
uniquify the variables relative to the post-localization symbol table.
In other words, we take the symbol table, apply the label map, and
rename any variable that clashes.
This also fixes an older problem where we weren't masking the
leading '_' on variable labels when generating 64tass output.
The code list now makes non-unique labels obvious, but you can't tell
the difference between unique global and unique local. What's more,
the default type value in Edit Label is now adjusted to Global for
unique locals that were auto-generated. To make it a bit easier to
figure out what's what, the Info panel now has a "label type" line
that reports the type.
The 2023-non-unique-labels test had some additional tests added to
exercise conflicts with local variables. The 2019-local-variables
test output changed slightly because the de-duplicated variable
naming convention was simplified.
2019-11-18 13:26:03 -08:00
|
|
|
|
/// <param name="labelMap">Label map dictionary, used to rename the labels in
|
|
|
|
|
/// the project symbol table. May be null.</param>
|
|
|
|
|
/// <param name="maskLeadingUnderscores">If true, labels with leading underscores
|
|
|
|
|
/// will be prefixed.</param>
|
2019-08-31 21:54:20 -07:00
|
|
|
|
/// <param name="uniquify">Set to true if variable names cannot be redefined.</param>
|
2019-08-30 18:33:05 -07:00
|
|
|
|
public LocalVariableLookup(SortedList<int, LocalVariableTable> lvTables,
|
Label rework, part 6
Correct handling of local variables. We now correctly uniquify them
with regard to non-unique labels. Because local vars can effectively
have global scope we mostly want to treat them as global, but they're
uniquified relative to other globals very late in the process, so we
can't just throw them in the symbol table and be done. Fortunately
local variables exist in a separate namespace, so we just need to
uniquify the variables relative to the post-localization symbol table.
In other words, we take the symbol table, apply the label map, and
rename any variable that clashes.
This also fixes an older problem where we weren't masking the
leading '_' on variable labels when generating 64tass output.
The code list now makes non-unique labels obvious, but you can't tell
the difference between unique global and unique local. What's more,
the default type value in Edit Label is now adjusted to Global for
unique locals that were auto-generated. To make it a bit easier to
figure out what's what, the Info panel now has a "label type" line
that reports the type.
The 2023-non-unique-labels test had some additional tests added to
exercise conflicts with local variables. The 2019-local-variables
test output changed slightly because the de-duplicated variable
naming convention was simplified.
2019-11-18 13:26:03 -08:00
|
|
|
|
DisasmProject project, Dictionary<string, string> labelMap,
|
|
|
|
|
bool maskLeadingUnderscores, bool uniquify) {
|
2019-08-30 18:33:05 -07:00
|
|
|
|
mLvTables = lvTables;
|
|
|
|
|
mProject = project;
|
Fix various local variable de-duplication bugs
In 1.5.0-dev1, as part of changes to the way label localization
works, the local variable de-duplicator started checking against a
filtered copy of the symbol table. Unfortunately it never
re-generated the table, so a long-lived LocalVariableLookup (like
the one used by LineListGen) would set up the dup map wrong and
be inconsistent with other parts of the program.
We now regenerate the table on every Reset().
The de-duplication stuff also had problems when opcodes and
operands were double-clicked on. When the opcode is clicked, the
selection should jump to the appropriate variable declaration, but
it wasn't being found because the label generated in the list was
in its original form. Fixed.
When an instruction operand is double-clicked, the instruction operand
editor opens with an "edit variable" shortcut. This was showing
the de-duplicated name, which isn't necessarily a bad thing, but it
was passing that value on to the DefSymbol editor, which thought it
was being asked to create a new entry. Fixed. (Entering the editor
through the LvTable editor works correctly, with nary a de-duplicated
name in sight. You'll be forced to rename it because it'll fail the
uniqueness test.)
References to de-duplicated local variables were getting lost when
the symbol's label was replaced (due largely to a convenient but
flawed shortcut: xrefs are attached to DefSymbol objects). Fixed by
linking the XrefSets.
Given the many issues and their relative subtlety, I decided to make
the modified names more obvious, and went back to the "_DUPn" naming
strategy. (I'm also considering just making it an error and
discarding conflicting entries during analysis... this is much more
complicated than I expected it to be.)
Quick tests can be performed in 2019-local-variables:
- go to +000026, double-click on the opcode, confirm sel change
- go to +000026, double-click on the operand, confirm orig name
shown in shortcut and that shortcut opens editor with orig name
- go to +00001a, down a line, click on PROJ_ZERO_DUP1 and confirm
that it has a single reference (from +000026)
- double-click on var table and confirm editing entry
2020-01-13 17:54:47 -08:00
|
|
|
|
mLabelMap = labelMap;
|
Label rework, part 6
Correct handling of local variables. We now correctly uniquify them
with regard to non-unique labels. Because local vars can effectively
have global scope we mostly want to treat them as global, but they're
uniquified relative to other globals very late in the process, so we
can't just throw them in the symbol table and be done. Fortunately
local variables exist in a separate namespace, so we just need to
uniquify the variables relative to the post-localization symbol table.
In other words, we take the symbol table, apply the label map, and
rename any variable that clashes.
This also fixes an older problem where we weren't masking the
leading '_' on variable labels when generating 64tass output.
The code list now makes non-unique labels obvious, but you can't tell
the difference between unique global and unique local. What's more,
the default type value in Edit Label is now adjusted to Global for
unique locals that were auto-generated. To make it a bit easier to
figure out what's what, the Info panel now has a "label type" line
that reports the type.
The 2023-non-unique-labels test had some additional tests added to
exercise conflicts with local variables. The 2019-local-variables
test output changed slightly because the de-duplicated variable
naming convention was simplified.
2019-11-18 13:26:03 -08:00
|
|
|
|
mMaskLeadingUnderscores = maskLeadingUnderscores;
|
2019-08-31 21:54:20 -07:00
|
|
|
|
mDoUniquify = uniquify;
|
2019-08-30 18:33:05 -07:00
|
|
|
|
|
|
|
|
|
mCurrentTable = new LocalVariableTable();
|
2019-08-31 21:54:20 -07:00
|
|
|
|
mDupRemap = new Dictionary<string, string>();
|
|
|
|
|
if (uniquify) {
|
2019-08-31 14:10:59 -07:00
|
|
|
|
mUniqueLabels = new Dictionary<string, UniqueLabel>();
|
|
|
|
|
}
|
2019-08-30 18:33:05 -07:00
|
|
|
|
Reset();
|
|
|
|
|
}
|
|
|
|
|
|
Fix various local variable de-duplication bugs
In 1.5.0-dev1, as part of changes to the way label localization
works, the local variable de-duplicator started checking against a
filtered copy of the symbol table. Unfortunately it never
re-generated the table, so a long-lived LocalVariableLookup (like
the one used by LineListGen) would set up the dup map wrong and
be inconsistent with other parts of the program.
We now regenerate the table on every Reset().
The de-duplication stuff also had problems when opcodes and
operands were double-clicked on. When the opcode is clicked, the
selection should jump to the appropriate variable declaration, but
it wasn't being found because the label generated in the list was
in its original form. Fixed.
When an instruction operand is double-clicked, the instruction operand
editor opens with an "edit variable" shortcut. This was showing
the de-duplicated name, which isn't necessarily a bad thing, but it
was passing that value on to the DefSymbol editor, which thought it
was being asked to create a new entry. Fixed. (Entering the editor
through the LvTable editor works correctly, with nary a de-duplicated
name in sight. You'll be forced to rename it because it'll fail the
uniqueness test.)
References to de-duplicated local variables were getting lost when
the symbol's label was replaced (due largely to a convenient but
flawed shortcut: xrefs are attached to DefSymbol objects). Fixed by
linking the XrefSets.
Given the many issues and their relative subtlety, I decided to make
the modified names more obvious, and went back to the "_DUPn" naming
strategy. (I'm also considering just making it an error and
discarding conflicting entries during analysis... this is much more
complicated than I expected it to be.)
Quick tests can be performed in 2019-local-variables:
- go to +000026, double-click on the opcode, confirm sel change
- go to +000026, double-click on the operand, confirm orig name
shown in shortcut and that shortcut opens editor with orig name
- go to +00001a, down a line, click on PROJ_ZERO_DUP1 and confirm
that it has a single reference (from +000026)
- double-click on var table and confirm editing entry
2020-01-13 17:54:47 -08:00
|
|
|
|
public void Reset(bool rebuildSyms = true) {
|
|
|
|
|
mRecentOffset = -1;
|
|
|
|
|
mRecentSymbols = null;
|
|
|
|
|
mCurrentTable.Clear();
|
|
|
|
|
mUniqueLabels?.Clear();
|
|
|
|
|
mDupRemap.Clear();
|
|
|
|
|
if (mLvTables.Count == 0) {
|
|
|
|
|
mNextLvtIndex = -1;
|
|
|
|
|
mNextLvtOffset = mProject.FileDataLength;
|
|
|
|
|
} else {
|
|
|
|
|
mNextLvtIndex = 0;
|
|
|
|
|
mNextLvtOffset = mLvTables.Keys[0];
|
|
|
|
|
}
|
|
|
|
|
CreateAllSymbolsDict();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void CreateAllSymbolsDict() {
|
|
|
|
|
// TODO(someday): we don't need to regenerate the all-symbols list if the list
|
|
|
|
|
// of symbols hasn't actually changed. Currently no way to tell.
|
|
|
|
|
|
|
|
|
|
Dictionary<string, string> labelMap = mLabelMap;
|
|
|
|
|
|
Label rework, part 6
Correct handling of local variables. We now correctly uniquify them
with regard to non-unique labels. Because local vars can effectively
have global scope we mostly want to treat them as global, but they're
uniquified relative to other globals very late in the process, so we
can't just throw them in the symbol table and be done. Fortunately
local variables exist in a separate namespace, so we just need to
uniquify the variables relative to the post-localization symbol table.
In other words, we take the symbol table, apply the label map, and
rename any variable that clashes.
This also fixes an older problem where we weren't masking the
leading '_' on variable labels when generating 64tass output.
The code list now makes non-unique labels obvious, but you can't tell
the difference between unique global and unique local. What's more,
the default type value in Edit Label is now adjusted to Global for
unique locals that were auto-generated. To make it a bit easier to
figure out what's what, the Info panel now has a "label type" line
that reports the type.
The 2023-non-unique-labels test had some additional tests added to
exercise conflicts with local variables. The 2019-local-variables
test output changed slightly because the de-duplicated variable
naming convention was simplified.
2019-11-18 13:26:03 -08:00
|
|
|
|
SymbolTable symTab = mProject.SymbolTable;
|
|
|
|
|
mAllNvSymbols = new Dictionary<string, Symbol>(symTab.Count);
|
|
|
|
|
foreach (Symbol sym in symTab) {
|
|
|
|
|
if (sym.SymbolSource == Symbol.Source.Variable) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if (labelMap != null && labelMap.TryGetValue(sym.Label, out string newLabel)) {
|
|
|
|
|
// Non-unique labels may map multiple entries to a single entry. That's
|
|
|
|
|
// fine; our goal here is just to avoid duplication. Besides, any symbols
|
|
|
|
|
// being output as locals will have the local prefix character and won't
|
|
|
|
|
// be a match.
|
|
|
|
|
mAllNvSymbols[newLabel] = sym;
|
|
|
|
|
} else {
|
|
|
|
|
mAllNvSymbols[sym.Label] = sym;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-08-30 18:33:05 -07:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Gets the symbol associated with the operand of an instruction.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="offset">Offset of start of instruction.</param>
|
|
|
|
|
/// <param name="operandValue">Operand value.</param>
|
|
|
|
|
/// <param name="type">Operand type. Should be ExternalAddress for DP ops, or
|
|
|
|
|
/// Constant for StackRel ops.</param>
|
|
|
|
|
/// <returns>Symbol, or null if no match found.</returns>
|
|
|
|
|
public DefSymbol GetSymbol(int offset, int operandValue, Symbol.Type type) {
|
|
|
|
|
AdvanceToOffset(offset);
|
|
|
|
|
return mCurrentTable.GetByValueRange(operandValue, 1, type);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
2019-08-31 14:10:59 -07:00
|
|
|
|
/// Gets the symbol associated with a symbol reference. If uniquification is enabled,
|
|
|
|
|
/// the unique-label map for the specified offset will be used to transform the
|
|
|
|
|
/// symbol reference.
|
2019-08-30 18:33:05 -07:00
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="offset">Offset of start of instruction.</param>
|
|
|
|
|
/// <param name="symRef">Reference to symbol.</param>
|
|
|
|
|
/// <returns>Symbol, or null if no match found.</returns>
|
|
|
|
|
public DefSymbol GetSymbol(int offset, WeakSymbolRef symRef) {
|
|
|
|
|
AdvanceToOffset(offset);
|
|
|
|
|
|
2019-08-31 21:54:20 -07:00
|
|
|
|
// The symRef uses the non-uniquified symbol, so we need to get the unique value at
|
|
|
|
|
// the current offset. We may need to do this even when variables can be
|
|
|
|
|
// redefined, because we might have a variable that's a duplicate of a user label
|
|
|
|
|
// or project symbol.
|
Instruction operand editor rework, part 2
Implemented local variable editing. Operands that have a local
variable reference, or are eligible to have one, can now be edited
directly from the instruction operand edit dialog.
Also, updated the code list double-click handler so that, if you
double-click on the opcode of an instruction that uses a local
variable reference, the selection and view will jump to the place
where that variable was defined.
Also, tweaked the way the References window refers to references
to an address that didn't use a symbol at that address. Updated
the explanation in the manual, which was a bit confusing.
Also, fixed some odds and ends in the manual.
Also, fixed a nasty infinite recursion bug (issue #47).
2019-09-07 18:57:22 -07:00
|
|
|
|
|
|
|
|
|
// Start by applying the de-duplication map.
|
2019-08-31 14:10:59 -07:00
|
|
|
|
string label = symRef.Label;
|
2019-08-31 21:54:20 -07:00
|
|
|
|
if (mDupRemap.TryGetValue(symRef.Label, out string remap)) {
|
|
|
|
|
label = remap;
|
|
|
|
|
}
|
|
|
|
|
//Debug.WriteLine("GetSymbol " + symRef.Label + " -> " + label);
|
2019-08-31 14:10:59 -07:00
|
|
|
|
if (mUniqueLabels != null && mUniqueLabels.TryGetValue(label, out UniqueLabel ulab)) {
|
2019-08-31 21:54:20 -07:00
|
|
|
|
//Debug.WriteLine(" Unique var " + symRef.Label + " -> " + ulab.Label);
|
2019-08-31 14:10:59 -07:00
|
|
|
|
label = ulab.Label;
|
|
|
|
|
}
|
|
|
|
|
DefSymbol defSym = mCurrentTable.GetByLabel(label);
|
|
|
|
|
|
|
|
|
|
// In theory this is okay, but in practice the only things asking for symbols are
|
|
|
|
|
// entirely convinced that the symbol exists here. So this is probably a bug.
|
|
|
|
|
Debug.Assert(defSym != null);
|
|
|
|
|
|
|
|
|
|
return defSym;
|
2019-08-30 18:33:05 -07:00
|
|
|
|
}
|
|
|
|
|
|
Instruction operand editor rework, part 2
Implemented local variable editing. Operands that have a local
variable reference, or are eligible to have one, can now be edited
directly from the instruction operand edit dialog.
Also, updated the code list double-click handler so that, if you
double-click on the opcode of an instruction that uses a local
variable reference, the selection and view will jump to the place
where that variable was defined.
Also, tweaked the way the References window refers to references
to an address that didn't use a symbol at that address. Updated
the explanation in the manual, which was a bit confusing.
Also, fixed some odds and ends in the manual.
Also, fixed a nasty infinite recursion bug (issue #47).
2019-09-07 18:57:22 -07:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Identifies the LocalVariableTable that defined the symbol reference.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="offset">Offset at which the symbol was referenced.</param>
|
|
|
|
|
/// <param name="symRef">Reference to symbol.</param>
|
|
|
|
|
/// <returns>Table index, or -1 if not found.</returns>
|
|
|
|
|
public int GetDefiningTableOffset(int offset, WeakSymbolRef symRef) {
|
Fix various local variable de-duplication bugs
In 1.5.0-dev1, as part of changes to the way label localization
works, the local variable de-duplicator started checking against a
filtered copy of the symbol table. Unfortunately it never
re-generated the table, so a long-lived LocalVariableLookup (like
the one used by LineListGen) would set up the dup map wrong and
be inconsistent with other parts of the program.
We now regenerate the table on every Reset().
The de-duplication stuff also had problems when opcodes and
operands were double-clicked on. When the opcode is clicked, the
selection should jump to the appropriate variable declaration, but
it wasn't being found because the label generated in the list was
in its original form. Fixed.
When an instruction operand is double-clicked, the instruction operand
editor opens with an "edit variable" shortcut. This was showing
the de-duplicated name, which isn't necessarily a bad thing, but it
was passing that value on to the DefSymbol editor, which thought it
was being asked to create a new entry. Fixed. (Entering the editor
through the LvTable editor works correctly, with nary a de-duplicated
name in sight. You'll be forced to rename it because it'll fail the
uniqueness test.)
References to de-duplicated local variables were getting lost when
the symbol's label was replaced (due largely to a convenient but
flawed shortcut: xrefs are attached to DefSymbol objects). Fixed by
linking the XrefSets.
Given the many issues and their relative subtlety, I decided to make
the modified names more obvious, and went back to the "_DUPn" naming
strategy. (I'm also considering just making it an error and
discarding conflicting entries during analysis... this is much more
complicated than I expected it to be.)
Quick tests can be performed in 2019-local-variables:
- go to +000026, double-click on the opcode, confirm sel change
- go to +000026, double-click on the operand, confirm orig name
shown in shortcut and that shortcut opens editor with orig name
- go to +00001a, down a line, click on PROJ_ZERO_DUP1 and confirm
that it has a single reference (from +000026)
- double-click on var table and confirm editing entry
2020-01-13 17:54:47 -08:00
|
|
|
|
// Get mDupRemap et. al. into the right state.
|
|
|
|
|
AdvanceToOffset(offset);
|
|
|
|
|
|
|
|
|
|
// symRef is the non-uniquified, de-duplicated symbol that was generated
|
|
|
|
|
// during the analysis pass. We either need to un-de-duplicate the label,
|
|
|
|
|
// or de-duplicate what we pull out of the Lv tables. The former requires
|
|
|
|
|
// a linear string search but will be faster if there are a lot of tables.
|
|
|
|
|
string label = UnDeDuplicate(symRef.Label);
|
|
|
|
|
|
Instruction operand editor rework, part 2
Implemented local variable editing. Operands that have a local
variable reference, or are eligible to have one, can now be edited
directly from the instruction operand edit dialog.
Also, updated the code list double-click handler so that, if you
double-click on the opcode of an instruction that uses a local
variable reference, the selection and view will jump to the place
where that variable was defined.
Also, tweaked the way the References window refers to references
to an address that didn't use a symbol at that address. Updated
the explanation in the manual, which was a bit confusing.
Also, fixed some odds and ends in the manual.
Also, fixed a nasty infinite recursion bug (issue #47).
2019-09-07 18:57:22 -07:00
|
|
|
|
// Walk backward through the list of tables until we find a match.
|
|
|
|
|
IList<int> keys = mLvTables.Keys;
|
|
|
|
|
for (int i = keys.Count - 1; i >= 0; i--) {
|
|
|
|
|
if (keys[i] > offset) {
|
|
|
|
|
// table comes after the point of reference
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
Fix various local variable de-duplication bugs
In 1.5.0-dev1, as part of changes to the way label localization
works, the local variable de-duplicator started checking against a
filtered copy of the symbol table. Unfortunately it never
re-generated the table, so a long-lived LocalVariableLookup (like
the one used by LineListGen) would set up the dup map wrong and
be inconsistent with other parts of the program.
We now regenerate the table on every Reset().
The de-duplication stuff also had problems when opcodes and
operands were double-clicked on. When the opcode is clicked, the
selection should jump to the appropriate variable declaration, but
it wasn't being found because the label generated in the list was
in its original form. Fixed.
When an instruction operand is double-clicked, the instruction operand
editor opens with an "edit variable" shortcut. This was showing
the de-duplicated name, which isn't necessarily a bad thing, but it
was passing that value on to the DefSymbol editor, which thought it
was being asked to create a new entry. Fixed. (Entering the editor
through the LvTable editor works correctly, with nary a de-duplicated
name in sight. You'll be forced to rename it because it'll fail the
uniqueness test.)
References to de-duplicated local variables were getting lost when
the symbol's label was replaced (due largely to a convenient but
flawed shortcut: xrefs are attached to DefSymbol objects). Fixed by
linking the XrefSets.
Given the many issues and their relative subtlety, I decided to make
the modified names more obvious, and went back to the "_DUPn" naming
strategy. (I'm also considering just making it an error and
discarding conflicting entries during analysis... this is much more
complicated than I expected it to be.)
Quick tests can be performed in 2019-local-variables:
- go to +000026, double-click on the opcode, confirm sel change
- go to +000026, double-click on the operand, confirm orig name
shown in shortcut and that shortcut opens editor with orig name
- go to +00001a, down a line, click on PROJ_ZERO_DUP1 and confirm
that it has a single reference (from +000026)
- double-click on var table and confirm editing entry
2020-01-13 17:54:47 -08:00
|
|
|
|
if (mLvTables.Values[i].GetByLabel(label) != null) {
|
Instruction operand editor rework, part 2
Implemented local variable editing. Operands that have a local
variable reference, or are eligible to have one, can now be edited
directly from the instruction operand edit dialog.
Also, updated the code list double-click handler so that, if you
double-click on the opcode of an instruction that uses a local
variable reference, the selection and view will jump to the place
where that variable was defined.
Also, tweaked the way the References window refers to references
to an address that didn't use a symbol at that address. Updated
the explanation in the manual, which was a bit confusing.
Also, fixed some odds and ends in the manual.
Also, fixed a nasty infinite recursion bug (issue #47).
2019-09-07 18:57:22 -07:00
|
|
|
|
// found it
|
|
|
|
|
return keys[i];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// if we didn't find it, it doesn't exist... right?
|
Fix various local variable de-duplication bugs
In 1.5.0-dev1, as part of changes to the way label localization
works, the local variable de-duplicator started checking against a
filtered copy of the symbol table. Unfortunately it never
re-generated the table, so a long-lived LocalVariableLookup (like
the one used by LineListGen) would set up the dup map wrong and
be inconsistent with other parts of the program.
We now regenerate the table on every Reset().
The de-duplication stuff also had problems when opcodes and
operands were double-clicked on. When the opcode is clicked, the
selection should jump to the appropriate variable declaration, but
it wasn't being found because the label generated in the list was
in its original form. Fixed.
When an instruction operand is double-clicked, the instruction operand
editor opens with an "edit variable" shortcut. This was showing
the de-duplicated name, which isn't necessarily a bad thing, but it
was passing that value on to the DefSymbol editor, which thought it
was being asked to create a new entry. Fixed. (Entering the editor
through the LvTable editor works correctly, with nary a de-duplicated
name in sight. You'll be forced to rename it because it'll fail the
uniqueness test.)
References to de-duplicated local variables were getting lost when
the symbol's label was replaced (due largely to a convenient but
flawed shortcut: xrefs are attached to DefSymbol objects). Fixed by
linking the XrefSets.
Given the many issues and their relative subtlety, I decided to make
the modified names more obvious, and went back to the "_DUPn" naming
strategy. (I'm also considering just making it an error and
discarding conflicting entries during analysis... this is much more
complicated than I expected it to be.)
Quick tests can be performed in 2019-local-variables:
- go to +000026, double-click on the opcode, confirm sel change
- go to +000026, double-click on the operand, confirm orig name
shown in shortcut and that shortcut opens editor with orig name
- go to +00001a, down a line, click on PROJ_ZERO_DUP1 and confirm
that it has a single reference (from +000026)
- double-click on var table and confirm editing entry
2020-01-13 17:54:47 -08:00
|
|
|
|
Debug.Assert(mCurrentTable.GetByLabel(label) == null);
|
Instruction operand editor rework, part 2
Implemented local variable editing. Operands that have a local
variable reference, or are eligible to have one, can now be edited
directly from the instruction operand edit dialog.
Also, updated the code list double-click handler so that, if you
double-click on the opcode of an instruction that uses a local
variable reference, the selection and view will jump to the place
where that variable was defined.
Also, tweaked the way the References window refers to references
to an address that didn't use a symbol at that address. Updated
the explanation in the manual, which was a bit confusing.
Also, fixed some odds and ends in the manual.
Also, fixed a nasty infinite recursion bug (issue #47).
2019-09-07 18:57:22 -07:00
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
Fix various local variable de-duplication bugs
In 1.5.0-dev1, as part of changes to the way label localization
works, the local variable de-duplicator started checking against a
filtered copy of the symbol table. Unfortunately it never
re-generated the table, so a long-lived LocalVariableLookup (like
the one used by LineListGen) would set up the dup map wrong and
be inconsistent with other parts of the program.
We now regenerate the table on every Reset().
The de-duplication stuff also had problems when opcodes and
operands were double-clicked on. When the opcode is clicked, the
selection should jump to the appropriate variable declaration, but
it wasn't being found because the label generated in the list was
in its original form. Fixed.
When an instruction operand is double-clicked, the instruction operand
editor opens with an "edit variable" shortcut. This was showing
the de-duplicated name, which isn't necessarily a bad thing, but it
was passing that value on to the DefSymbol editor, which thought it
was being asked to create a new entry. Fixed. (Entering the editor
through the LvTable editor works correctly, with nary a de-duplicated
name in sight. You'll be forced to rename it because it'll fail the
uniqueness test.)
References to de-duplicated local variables were getting lost when
the symbol's label was replaced (due largely to a convenient but
flawed shortcut: xrefs are attached to DefSymbol objects). Fixed by
linking the XrefSets.
Given the many issues and their relative subtlety, I decided to make
the modified names more obvious, and went back to the "_DUPn" naming
strategy. (I'm also considering just making it an error and
discarding conflicting entries during analysis... this is much more
complicated than I expected it to be.)
Quick tests can be performed in 2019-local-variables:
- go to +000026, double-click on the opcode, confirm sel change
- go to +000026, double-click on the operand, confirm orig name
shown in shortcut and that shortcut opens editor with orig name
- go to +00001a, down a line, click on PROJ_ZERO_DUP1 and confirm
that it has a single reference (from +000026)
- double-click on var table and confirm editing entry
2020-01-13 17:54:47 -08:00
|
|
|
|
private string UnDeDuplicate(string label) {
|
|
|
|
|
foreach (KeyValuePair<string, string> kvp in mDupRemap) {
|
|
|
|
|
if (kvp.Value == label) {
|
|
|
|
|
return kvp.Key;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return label;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Restores a de-duplicated symbol to original form.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <remarks>
|
|
|
|
|
/// Another kluge on the de-duplication system. This is used by the instruction
|
|
|
|
|
/// operand editor's "edit variable" shortcut mechanism, because trying to edit the
|
|
|
|
|
/// DefSymbol with the de-duplicated name ends badly.
|
|
|
|
|
/// </remarks>
|
|
|
|
|
/// <param name="sym">Symbol to un-de-duplicate.</param>
|
|
|
|
|
/// <returns>Original or un-de-duplicated symbol.</returns>
|
|
|
|
|
public DefSymbol GetOriginalForm(DefSymbol sym) {
|
|
|
|
|
string orig = UnDeDuplicate(sym.Label);
|
|
|
|
|
if (orig == sym.Label) {
|
|
|
|
|
return sym;
|
|
|
|
|
}
|
|
|
|
|
return new DefSymbol(sym, orig);
|
|
|
|
|
}
|
|
|
|
|
|
2019-08-30 18:33:05 -07:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Gets a LocalVariableTable that is the result of merging all tables up to this point.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="offset">Target offset.</param>
|
|
|
|
|
/// <returns>Combined table.</returns>
|
|
|
|
|
public LocalVariableTable GetMergedTableAtOffset(int offset) {
|
|
|
|
|
AdvanceToOffset(offset);
|
|
|
|
|
return mCurrentTable;
|
|
|
|
|
}
|
|
|
|
|
|
Instruction operand editor rework, part 2
Implemented local variable editing. Operands that have a local
variable reference, or are eligible to have one, can now be edited
directly from the instruction operand edit dialog.
Also, updated the code list double-click handler so that, if you
double-click on the opcode of an instruction that uses a local
variable reference, the selection and view will jump to the place
where that variable was defined.
Also, tweaked the way the References window refers to references
to an address that didn't use a symbol at that address. Updated
the explanation in the manual, which was a bit confusing.
Also, fixed some odds and ends in the manual.
Also, fixed a nasty infinite recursion bug (issue #47).
2019-09-07 18:57:22 -07:00
|
|
|
|
/// <summary>
|
2019-10-28 16:54:01 -07:00
|
|
|
|
/// Finds the closest table that is defined at or before the specified offset. Will
|
|
|
|
|
/// attempt to only return un-hidden tables, but will return a hidden table if no
|
|
|
|
|
/// others are available.
|
Instruction operand editor rework, part 2
Implemented local variable editing. Operands that have a local
variable reference, or are eligible to have one, can now be edited
directly from the instruction operand edit dialog.
Also, updated the code list double-click handler so that, if you
double-click on the opcode of an instruction that uses a local
variable reference, the selection and view will jump to the place
where that variable was defined.
Also, tweaked the way the References window refers to references
to an address that didn't use a symbol at that address. Updated
the explanation in the manual, which was a bit confusing.
Also, fixed some odds and ends in the manual.
Also, fixed a nasty infinite recursion bug (issue #47).
2019-09-07 18:57:22 -07:00
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="offset">Target offset.</param>
|
|
|
|
|
/// <returns>The table's definition offset, or -1 if no tables were defined before this
|
|
|
|
|
/// point.</returns>
|
|
|
|
|
public int GetNearestTableOffset(int offset) {
|
|
|
|
|
int nearest = -1;
|
2019-10-28 16:54:01 -07:00
|
|
|
|
int nearestUnhidden = -1;
|
Instruction operand editor rework, part 2
Implemented local variable editing. Operands that have a local
variable reference, or are eligible to have one, can now be edited
directly from the instruction operand edit dialog.
Also, updated the code list double-click handler so that, if you
double-click on the opcode of an instruction that uses a local
variable reference, the selection and view will jump to the place
where that variable was defined.
Also, tweaked the way the References window refers to references
to an address that didn't use a symbol at that address. Updated
the explanation in the manual, which was a bit confusing.
Also, fixed some odds and ends in the manual.
Also, fixed a nasty infinite recursion bug (issue #47).
2019-09-07 18:57:22 -07:00
|
|
|
|
|
|
|
|
|
// Could do a smarter search, but I'm expecting the set to be small.
|
|
|
|
|
foreach (KeyValuePair<int, LocalVariableTable> kvp in mLvTables) {
|
|
|
|
|
if (kvp.Key > offset) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
nearest = kvp.Key;
|
2019-10-28 16:54:01 -07:00
|
|
|
|
if (mProject.GetAnattrib(nearest).IsStart) {
|
|
|
|
|
nearestUnhidden = nearest;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (nearestUnhidden >= 0) {
|
|
|
|
|
return nearestUnhidden;
|
|
|
|
|
} else {
|
|
|
|
|
return nearest;
|
Instruction operand editor rework, part 2
Implemented local variable editing. Operands that have a local
variable reference, or are eligible to have one, can now be edited
directly from the instruction operand edit dialog.
Also, updated the code list double-click handler so that, if you
double-click on the opcode of an instruction that uses a local
variable reference, the selection and view will jump to the place
where that variable was defined.
Also, tweaked the way the References window refers to references
to an address that didn't use a symbol at that address. Updated
the explanation in the manual, which was a bit confusing.
Also, fixed some odds and ends in the manual.
Also, fixed a nasty infinite recursion bug (issue #47).
2019-09-07 18:57:22 -07:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-08-30 18:33:05 -07:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Generates a list of variables defined at the specified offset, if a table is
|
|
|
|
|
/// associated with that offset.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="offset">File data offset.</param>
|
|
|
|
|
/// <returns>List of symbols, uniquified if desired, or null if no LocalVariableTable
|
|
|
|
|
/// exists at the specified offset.</returns>
|
|
|
|
|
public List<DefSymbol> GetVariablesDefinedAtOffset(int offset) {
|
|
|
|
|
AdvanceToOffset(offset);
|
|
|
|
|
|
|
|
|
|
if (mRecentOffset == offset) {
|
|
|
|
|
return mRecentSymbols;
|
|
|
|
|
}
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Updates internal state to reflect the state of the world at the specified offset.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <remarks>
|
|
|
|
|
/// When the offset is greater than or equal to its value on a previous call, we can
|
|
|
|
|
/// do an incremental update. If the offset moves backward, we have to reset and walk
|
|
|
|
|
/// forward again.
|
|
|
|
|
/// </remarks>
|
2019-08-31 14:10:59 -07:00
|
|
|
|
/// <param name="targetOffset">Target offset.</param>
|
|
|
|
|
private void AdvanceToOffset(int targetOffset) {
|
2019-08-30 18:33:05 -07:00
|
|
|
|
if (mNextLvtIndex < 0) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2019-08-31 14:10:59 -07:00
|
|
|
|
if (targetOffset < mRecentOffset) {
|
2019-08-30 18:33:05 -07:00
|
|
|
|
// We went backwards.
|
Fix various local variable de-duplication bugs
In 1.5.0-dev1, as part of changes to the way label localization
works, the local variable de-duplicator started checking against a
filtered copy of the symbol table. Unfortunately it never
re-generated the table, so a long-lived LocalVariableLookup (like
the one used by LineListGen) would set up the dup map wrong and
be inconsistent with other parts of the program.
We now regenerate the table on every Reset().
The de-duplication stuff also had problems when opcodes and
operands were double-clicked on. When the opcode is clicked, the
selection should jump to the appropriate variable declaration, but
it wasn't being found because the label generated in the list was
in its original form. Fixed.
When an instruction operand is double-clicked, the instruction operand
editor opens with an "edit variable" shortcut. This was showing
the de-duplicated name, which isn't necessarily a bad thing, but it
was passing that value on to the DefSymbol editor, which thought it
was being asked to create a new entry. Fixed. (Entering the editor
through the LvTable editor works correctly, with nary a de-duplicated
name in sight. You'll be forced to rename it because it'll fail the
uniqueness test.)
References to de-duplicated local variables were getting lost when
the symbol's label was replaced (due largely to a convenient but
flawed shortcut: xrefs are attached to DefSymbol objects). Fixed by
linking the XrefSets.
Given the many issues and their relative subtlety, I decided to make
the modified names more obvious, and went back to the "_DUPn" naming
strategy. (I'm also considering just making it an error and
discarding conflicting entries during analysis... this is much more
complicated than I expected it to be.)
Quick tests can be performed in 2019-local-variables:
- go to +000026, double-click on the opcode, confirm sel change
- go to +000026, double-click on the operand, confirm orig name
shown in shortcut and that shortcut opens editor with orig name
- go to +00001a, down a line, click on PROJ_ZERO_DUP1 and confirm
that it has a single reference (from +000026)
- double-click on var table and confirm editing entry
2020-01-13 17:54:47 -08:00
|
|
|
|
Reset(false);
|
2019-08-30 18:33:05 -07:00
|
|
|
|
}
|
2019-08-31 14:10:59 -07:00
|
|
|
|
while (mNextLvtOffset <= targetOffset) {
|
2019-08-30 18:33:05 -07:00
|
|
|
|
if (!mProject.GetAnattrib(mNextLvtOffset).IsStart) {
|
|
|
|
|
// Hidden table, ignore it.
|
|
|
|
|
Debug.WriteLine("Ignoring LvTable at +" + mNextLvtOffset.ToString("x6"));
|
|
|
|
|
} else {
|
|
|
|
|
// Process this table.
|
|
|
|
|
LocalVariableTable lvt = mLvTables.Values[mNextLvtIndex];
|
|
|
|
|
if (lvt.ClearPrevious) {
|
|
|
|
|
mCurrentTable.Clear();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Create a list for GetVariablesDefinedAtOffset
|
|
|
|
|
mRecentSymbols = new List<DefSymbol>();
|
|
|
|
|
mRecentOffset = mNextLvtOffset;
|
|
|
|
|
|
|
|
|
|
// Merge the new entries into the work table. This automatically
|
2019-08-31 14:10:59 -07:00
|
|
|
|
// discards entries that clash by name or value.
|
2019-08-30 18:33:05 -07:00
|
|
|
|
for (int i = 0; i < lvt.Count; i++) {
|
2019-08-31 14:10:59 -07:00
|
|
|
|
DefSymbol defSym = lvt[i];
|
Label rework, part 6
Correct handling of local variables. We now correctly uniquify them
with regard to non-unique labels. Because local vars can effectively
have global scope we mostly want to treat them as global, but they're
uniquified relative to other globals very late in the process, so we
can't just throw them in the symbol table and be done. Fortunately
local variables exist in a separate namespace, so we just need to
uniquify the variables relative to the post-localization symbol table.
In other words, we take the symbol table, apply the label map, and
rename any variable that clashes.
This also fixes an older problem where we weren't masking the
leading '_' on variable labels when generating 64tass output.
The code list now makes non-unique labels obvious, but you can't tell
the difference between unique global and unique local. What's more,
the default type value in Edit Label is now adjusted to Global for
unique locals that were auto-generated. To make it a bit easier to
figure out what's what, the Info panel now has a "label type" line
that reports the type.
The 2023-non-unique-labels test had some additional tests added to
exercise conflicts with local variables. The 2019-local-variables
test output changed slightly because the de-duplicated variable
naming convention was simplified.
2019-11-18 13:26:03 -08:00
|
|
|
|
string newLabel = defSym.Label;
|
|
|
|
|
|
|
|
|
|
if (mMaskLeadingUnderscores && newLabel[0] == '_') {
|
|
|
|
|
newLabel = AsmGen.LabelLocalizer.NO_UNDER_PFX + newLabel;
|
|
|
|
|
}
|
2019-08-31 21:54:20 -07:00
|
|
|
|
|
|
|
|
|
// Look for non-variable symbols with the same label. Ordinarily the
|
|
|
|
|
// editor prevents this from happening, but there are ways to trick
|
Label rework, part 6
Correct handling of local variables. We now correctly uniquify them
with regard to non-unique labels. Because local vars can effectively
have global scope we mostly want to treat them as global, but they're
uniquified relative to other globals very late in the process, so we
can't just throw them in the symbol table and be done. Fortunately
local variables exist in a separate namespace, so we just need to
uniquify the variables relative to the post-localization symbol table.
In other words, we take the symbol table, apply the label map, and
rename any variable that clashes.
This also fixes an older problem where we weren't masking the
leading '_' on variable labels when generating 64tass output.
The code list now makes non-unique labels obvious, but you can't tell
the difference between unique global and unique local. What's more,
the default type value in Edit Label is now adjusted to Global for
unique locals that were auto-generated. To make it a bit easier to
figure out what's what, the Info panel now has a "label type" line
that reports the type.
The 2023-non-unique-labels test had some additional tests added to
exercise conflicts with local variables. The 2019-local-variables
test output changed slightly because the de-duplicated variable
naming convention was simplified.
2019-11-18 13:26:03 -08:00
|
|
|
|
// the system (e.g. add a symbol while the LvTable is hidden, or have
|
|
|
|
|
// a non-unique local promoted to global). We deal with it here.
|
|
|
|
|
//
|
|
|
|
|
// TODO(someday): this is not necessary for assemblers like Merlin 32
|
|
|
|
|
// that put variables in a separate namespace.
|
|
|
|
|
if (mAllNvSymbols.TryGetValue(newLabel, out Symbol unused)) {
|
|
|
|
|
Debug.WriteLine("Detected duplicate non-var label " + newLabel +
|
2019-08-31 21:54:20 -07:00
|
|
|
|
" at +" + mNextLvtOffset.ToString("x6"));
|
Fix various local variable de-duplication bugs
In 1.5.0-dev1, as part of changes to the way label localization
works, the local variable de-duplicator started checking against a
filtered copy of the symbol table. Unfortunately it never
re-generated the table, so a long-lived LocalVariableLookup (like
the one used by LineListGen) would set up the dup map wrong and
be inconsistent with other parts of the program.
We now regenerate the table on every Reset().
The de-duplication stuff also had problems when opcodes and
operands were double-clicked on. When the opcode is clicked, the
selection should jump to the appropriate variable declaration, but
it wasn't being found because the label generated in the list was
in its original form. Fixed.
When an instruction operand is double-clicked, the instruction operand
editor opens with an "edit variable" shortcut. This was showing
the de-duplicated name, which isn't necessarily a bad thing, but it
was passing that value on to the DefSymbol editor, which thought it
was being asked to create a new entry. Fixed. (Entering the editor
through the LvTable editor works correctly, with nary a de-duplicated
name in sight. You'll be forced to rename it because it'll fail the
uniqueness test.)
References to de-duplicated local variables were getting lost when
the symbol's label was replaced (due largely to a convenient but
flawed shortcut: xrefs are attached to DefSymbol objects). Fixed by
linking the XrefSets.
Given the many issues and their relative subtlety, I decided to make
the modified names more obvious, and went back to the "_DUPn" naming
strategy. (I'm also considering just making it an error and
discarding conflicting entries during analysis... this is much more
complicated than I expected it to be.)
Quick tests can be performed in 2019-local-variables:
- go to +000026, double-click on the opcode, confirm sel change
- go to +000026, double-click on the operand, confirm orig name
shown in shortcut and that shortcut opens editor with orig name
- go to +00001a, down a line, click on PROJ_ZERO_DUP1 and confirm
that it has a single reference (from +000026)
- double-click on var table and confirm editing entry
2020-01-13 17:54:47 -08:00
|
|
|
|
newLabel = GenerateDeDupLabel(newLabel);
|
Label rework, part 6
Correct handling of local variables. We now correctly uniquify them
with regard to non-unique labels. Because local vars can effectively
have global scope we mostly want to treat them as global, but they're
uniquified relative to other globals very late in the process, so we
can't just throw them in the symbol table and be done. Fortunately
local variables exist in a separate namespace, so we just need to
uniquify the variables relative to the post-localization symbol table.
In other words, we take the symbol table, apply the label map, and
rename any variable that clashes.
This also fixes an older problem where we weren't masking the
leading '_' on variable labels when generating 64tass output.
The code list now makes non-unique labels obvious, but you can't tell
the difference between unique global and unique local. What's more,
the default type value in Edit Label is now adjusted to Global for
unique locals that were auto-generated. To make it a bit easier to
figure out what's what, the Info panel now has a "label type" line
that reports the type.
The 2023-non-unique-labels test had some additional tests added to
exercise conflicts with local variables. The 2019-local-variables
test output changed slightly because the de-duplicated variable
naming convention was simplified.
2019-11-18 13:26:03 -08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (newLabel != defSym.Label) {
|
2019-08-31 21:54:20 -07:00
|
|
|
|
mDupRemap[defSym.Label] = newLabel;
|
|
|
|
|
defSym = new DefSymbol(defSym, newLabel);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (mDoUniquify) {
|
2019-08-31 14:10:59 -07:00
|
|
|
|
if (mUniqueLabels.TryGetValue(defSym.Label, out UniqueLabel ulab)) {
|
2019-08-31 21:54:20 -07:00
|
|
|
|
// We've seen this label before; generate a unique version by
|
|
|
|
|
// increasing the appended number.
|
Label rework, part 6
Correct handling of local variables. We now correctly uniquify them
with regard to non-unique labels. Because local vars can effectively
have global scope we mostly want to treat them as global, but they're
uniquified relative to other globals very late in the process, so we
can't just throw them in the symbol table and be done. Fortunately
local variables exist in a separate namespace, so we just need to
uniquify the variables relative to the post-localization symbol table.
In other words, we take the symbol table, apply the label map, and
rename any variable that clashes.
This also fixes an older problem where we weren't masking the
leading '_' on variable labels when generating 64tass output.
The code list now makes non-unique labels obvious, but you can't tell
the difference between unique global and unique local. What's more,
the default type value in Edit Label is now adjusted to Global for
unique locals that were auto-generated. To make it a bit easier to
figure out what's what, the Info panel now has a "label type" line
that reports the type.
The 2023-non-unique-labels test had some additional tests added to
exercise conflicts with local variables. The 2019-local-variables
test output changed slightly because the de-duplicated variable
naming convention was simplified.
2019-11-18 13:26:03 -08:00
|
|
|
|
ulab.MakeUnique(mAllNvSymbols);
|
2019-08-31 14:10:59 -07:00
|
|
|
|
defSym = new DefSymbol(defSym, ulab.Label);
|
|
|
|
|
} else {
|
|
|
|
|
// Haven't seen this before. Add it to the unique-labels table.
|
|
|
|
|
mUniqueLabels.Add(defSym.Label, new UniqueLabel(defSym.Label));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
mCurrentTable.AddOrReplace(defSym);
|
2019-08-30 18:33:05 -07:00
|
|
|
|
|
2019-08-31 14:10:59 -07:00
|
|
|
|
mRecentSymbols.Add(defSym);
|
2019-08-30 18:33:05 -07:00
|
|
|
|
}
|
|
|
|
|
|
2019-08-31 20:32:43 -07:00
|
|
|
|
//mCurrentTable.DebugDump(mNextLvtOffset);
|
2019-08-30 18:33:05 -07:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Update state to look for next table.
|
|
|
|
|
mNextLvtIndex++;
|
|
|
|
|
if (mNextLvtIndex < mLvTables.Keys.Count) {
|
|
|
|
|
mNextLvtOffset = mLvTables.Keys[mNextLvtIndex];
|
|
|
|
|
} else {
|
|
|
|
|
mNextLvtOffset = mProject.FileDataLength; // never reached
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-08-31 21:54:20 -07:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Generates a unique label for the duplicate remap table.
|
|
|
|
|
/// </summary>
|
Fix various local variable de-duplication bugs
In 1.5.0-dev1, as part of changes to the way label localization
works, the local variable de-duplicator started checking against a
filtered copy of the symbol table. Unfortunately it never
re-generated the table, so a long-lived LocalVariableLookup (like
the one used by LineListGen) would set up the dup map wrong and
be inconsistent with other parts of the program.
We now regenerate the table on every Reset().
The de-duplication stuff also had problems when opcodes and
operands were double-clicked on. When the opcode is clicked, the
selection should jump to the appropriate variable declaration, but
it wasn't being found because the label generated in the list was
in its original form. Fixed.
When an instruction operand is double-clicked, the instruction operand
editor opens with an "edit variable" shortcut. This was showing
the de-duplicated name, which isn't necessarily a bad thing, but it
was passing that value on to the DefSymbol editor, which thought it
was being asked to create a new entry. Fixed. (Entering the editor
through the LvTable editor works correctly, with nary a de-duplicated
name in sight. You'll be forced to rename it because it'll fail the
uniqueness test.)
References to de-duplicated local variables were getting lost when
the symbol's label was replaced (due largely to a convenient but
flawed shortcut: xrefs are attached to DefSymbol objects). Fixed by
linking the XrefSets.
Given the many issues and their relative subtlety, I decided to make
the modified names more obvious, and went back to the "_DUPn" naming
strategy. (I'm also considering just making it an error and
discarding conflicting entries during analysis... this is much more
complicated than I expected it to be.)
Quick tests can be performed in 2019-local-variables:
- go to +000026, double-click on the opcode, confirm sel change
- go to +000026, double-click on the operand, confirm orig name
shown in shortcut and that shortcut opens editor with orig name
- go to +00001a, down a line, click on PROJ_ZERO_DUP1 and confirm
that it has a single reference (from +000026)
- double-click on var table and confirm editing entry
2020-01-13 17:54:47 -08:00
|
|
|
|
/// <remarks>
|
|
|
|
|
/// We need to worry about clashes with the main symbol table, but we don't have to
|
|
|
|
|
/// worry about other entries in the remap table because we know their baseLabels
|
|
|
|
|
/// are different.
|
|
|
|
|
/// </remarks>
|
|
|
|
|
private string GenerateDeDupLabel(string baseLabel) {
|
2019-08-31 21:54:20 -07:00
|
|
|
|
string testLabel;
|
|
|
|
|
int counter = 0;
|
|
|
|
|
do {
|
|
|
|
|
counter++;
|
Fix various local variable de-duplication bugs
In 1.5.0-dev1, as part of changes to the way label localization
works, the local variable de-duplicator started checking against a
filtered copy of the symbol table. Unfortunately it never
re-generated the table, so a long-lived LocalVariableLookup (like
the one used by LineListGen) would set up the dup map wrong and
be inconsistent with other parts of the program.
We now regenerate the table on every Reset().
The de-duplication stuff also had problems when opcodes and
operands were double-clicked on. When the opcode is clicked, the
selection should jump to the appropriate variable declaration, but
it wasn't being found because the label generated in the list was
in its original form. Fixed.
When an instruction operand is double-clicked, the instruction operand
editor opens with an "edit variable" shortcut. This was showing
the de-duplicated name, which isn't necessarily a bad thing, but it
was passing that value on to the DefSymbol editor, which thought it
was being asked to create a new entry. Fixed. (Entering the editor
through the LvTable editor works correctly, with nary a de-duplicated
name in sight. You'll be forced to rename it because it'll fail the
uniqueness test.)
References to de-duplicated local variables were getting lost when
the symbol's label was replaced (due largely to a convenient but
flawed shortcut: xrefs are attached to DefSymbol objects). Fixed by
linking the XrefSets.
Given the many issues and their relative subtlety, I decided to make
the modified names more obvious, and went back to the "_DUPn" naming
strategy. (I'm also considering just making it an error and
discarding conflicting entries during analysis... this is much more
complicated than I expected it to be.)
Quick tests can be performed in 2019-local-variables:
- go to +000026, double-click on the opcode, confirm sel change
- go to +000026, double-click on the operand, confirm orig name
shown in shortcut and that shortcut opens editor with orig name
- go to +00001a, down a line, click on PROJ_ZERO_DUP1 and confirm
that it has a single reference (from +000026)
- double-click on var table and confirm editing entry
2020-01-13 17:54:47 -08:00
|
|
|
|
testLabel = baseLabel + "_DUP" + counter;
|
Label rework, part 6
Correct handling of local variables. We now correctly uniquify them
with regard to non-unique labels. Because local vars can effectively
have global scope we mostly want to treat them as global, but they're
uniquified relative to other globals very late in the process, so we
can't just throw them in the symbol table and be done. Fortunately
local variables exist in a separate namespace, so we just need to
uniquify the variables relative to the post-localization symbol table.
In other words, we take the symbol table, apply the label map, and
rename any variable that clashes.
This also fixes an older problem where we weren't masking the
leading '_' on variable labels when generating 64tass output.
The code list now makes non-unique labels obvious, but you can't tell
the difference between unique global and unique local. What's more,
the default type value in Edit Label is now adjusted to Global for
unique locals that were auto-generated. To make it a bit easier to
figure out what's what, the Info panel now has a "label type" line
that reports the type.
The 2023-non-unique-labels test had some additional tests added to
exercise conflicts with local variables. The 2019-local-variables
test output changed slightly because the de-duplicated variable
naming convention was simplified.
2019-11-18 13:26:03 -08:00
|
|
|
|
} while (mAllNvSymbols.TryGetValue(testLabel, out Symbol unused));
|
2019-08-31 21:54:20 -07:00
|
|
|
|
return testLabel;
|
|
|
|
|
}
|
2019-10-28 16:54:01 -07:00
|
|
|
|
|
|
|
|
|
public static bool IsTableHidden(int offset, DisasmProject project) {
|
|
|
|
|
return !project.GetAnattrib(offset).IsStart;
|
|
|
|
|
}
|
2019-08-30 18:33:05 -07:00
|
|
|
|
}
|
|
|
|
|
}
|