mirror of
https://github.com/ksherlock/x65.git
synced 2025-04-06 12:38:13 +00:00
Fixed Merlin LNK directive and streamlined the LINK directive
- No longer merging sections but assigning sequential addresses to sections - Required separate linking logic for Merlin.
This commit is contained in:
parent
90fc74a6fb
commit
25eab76c95
@ -921,11 +921,12 @@ FindFirstSpace
|
||||
Fish food! Assembler has all important features and switching to make a 6502 project for more testing. Currently the assembler is in a limited release. Primarily tested with personal archive of sources written for Kick assmebler, DASM, TASM, XASM, etc. and passing most of Apple II Prince of Persia and Pinball Construction set.
|
||||
|
||||
**TODO**
|
||||
* Rather than keeping track of merged section for fixed address linking, just append the user linked section to the current section and empty the old section.
|
||||
* OMF export for Apple II GS/OS executables
|
||||
* irp (indefinite repeat)
|
||||
|
||||
**FIXED**
|
||||
* Removed the concept of linking by merging sections and instead keeping the sections separate and individually assigned memory addresses so they won't overlap.
|
||||
* Fixed up Merlin LNK directive to work with new linker
|
||||
* Fixed linker merged section reloc confusion.
|
||||
* -org command line argument to override the built-in assumption of org $1000, to avoid ever having to use the ORG directive inlined in code.
|
||||
* dump_x65 now shows the code offset of each section into the .x65 file which can be copied and pasted into the disassembler in case the object file assembler output needs to be inspected.
|
||||
|
206
x65.cpp
206
x65.cpp
@ -1202,7 +1202,7 @@ typedef struct Section {
|
||||
|
||||
void Cleanup() { if (output) free(output); reset(); }
|
||||
bool empty() const { return merged_offset<0 && curr==output; }
|
||||
bool unused() const { return address == start_address; }
|
||||
bool unused() const { return !address_assigned && address == start_address; }
|
||||
|
||||
int DataOffset() const { return int(curr - output); }
|
||||
size_t size() const { return curr - output; }
|
||||
@ -1213,7 +1213,7 @@ typedef struct Section {
|
||||
void SetLoadAddress(int addr) { load_address = addr; }
|
||||
int GetLoadAddress() const { return load_address; }
|
||||
|
||||
void SetDummySection(bool enable) { dummySection = enable; }
|
||||
void SetDummySection(bool enable) { dummySection = enable; type = ST_BSS; }
|
||||
bool IsDummySection() const { return dummySection; }
|
||||
bool IsRelativeSection() const { return address_assigned == false; }
|
||||
bool IsMergedSection() const { return merged_offset >= 0; }
|
||||
@ -1456,7 +1456,6 @@ public:
|
||||
bool error_encountered; // if any error encountered, don't export binary
|
||||
bool list_assembly; // generate assembler listing
|
||||
bool end_macro_directive; // whether to use { } or macro / endmacro for macro scope
|
||||
bool link_all_section; // link all known relative sections to this section at end
|
||||
|
||||
// Convert source to binary
|
||||
void Assemble(strref source, strref filename, bool obj_target);
|
||||
@ -1479,13 +1478,12 @@ public:
|
||||
void LinkLabelsToAddress(int section_id, int section_new, int section_address);
|
||||
StatusCode LinkRelocs(int section_id, int section_new, int section_address);
|
||||
StatusCode AssignAddressToSection(int section_id, int address);
|
||||
StatusCode AppendSection(Section &relSection, Section &trgSection);
|
||||
StatusCode LinkSections(strref name); // link relative address sections with this name here
|
||||
void DummySection(int address); // non-data section (fixed)
|
||||
void DummySection(); // non-data section (relative)
|
||||
void EndSection(); // pop current section
|
||||
void LinkAllToSection(); // link all loaded relative sections to this section at end
|
||||
Section& CurrSection() { return *current_section; }
|
||||
void AssignAddressToGroup(); // Merlin LNK support
|
||||
unsigned char* BuildExport(strref append, int &file_size, int &addr);
|
||||
int GetExportNames(strref *aNames, int maxNames);
|
||||
StatusCode LinkZP();
|
||||
@ -1497,7 +1495,7 @@ public:
|
||||
|
||||
// Object file handling
|
||||
StatusCode WriteObjectFile(strref filename); // write x65 object file
|
||||
StatusCode ReadObjectFile(strref filename); // read x65 object file
|
||||
StatusCode ReadObjectFile(strref filename, int link_to_section = -1); // read x65 object file
|
||||
|
||||
// Scope management
|
||||
StatusCode EnterScope();
|
||||
@ -1616,7 +1614,6 @@ void Asm::Cleanup() {
|
||||
for (std::vector<ExtLabels>::iterator exti = externals.begin(); exti !=externals.end(); ++exti)
|
||||
exti->labels.clear();
|
||||
externals.clear();
|
||||
link_all_section = false;
|
||||
// this section is relocatable but is assigned address $1000 if exporting without directives
|
||||
SetSection(strref("default"));
|
||||
current_section = &allSections[0];
|
||||
@ -1774,8 +1771,6 @@ void Asm::SetSection(strref name, int address)
|
||||
}
|
||||
}
|
||||
}
|
||||
if (link_all_section)
|
||||
LinkAllToSection();
|
||||
if (allSections.size()==allSections.capacity())
|
||||
allSections.reserve(allSections.size() + 16);
|
||||
Section newSection(name, address);
|
||||
@ -1787,9 +1782,7 @@ void Asm::SetSection(strref name, int address)
|
||||
|
||||
void Asm::SetSection(strref line)
|
||||
{
|
||||
if (link_all_section)
|
||||
LinkAllToSection();
|
||||
else if (allSections.size() && CurrSection().unused())
|
||||
if (allSections.size() && CurrSection().unused())
|
||||
allSections.erase(allSections.begin() + SectionId());
|
||||
if (allSections.size() == allSections.capacity())
|
||||
allSections.reserve(allSections.size() + 16);
|
||||
@ -1837,30 +1830,8 @@ void Asm::SetSection(strref line)
|
||||
current_section = &allSections[allSections.size()-1];
|
||||
}
|
||||
|
||||
// Merlin linking includes one file at a time ignoring the section naming
|
||||
// so just wait until the end of the current section to link all unlinked
|
||||
// sections.
|
||||
void Asm::LinkAllToSection() {
|
||||
if (CurrSection().IsDummySection())
|
||||
return;
|
||||
bool gotRelSect = true;
|
||||
while (gotRelSect) {
|
||||
gotRelSect = false;
|
||||
for (std::vector<Section>::iterator s = allSections.begin(); s!=allSections.end(); ++s) {
|
||||
if (s->IsRelativeSection()) {
|
||||
LinkSections(s->name);
|
||||
gotRelSect = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
link_all_section = false;
|
||||
}
|
||||
|
||||
// Fixed address dummy section
|
||||
void Asm::DummySection(int address) {
|
||||
if (link_all_section)
|
||||
LinkAllToSection();
|
||||
if (allSections.size()==allSections.capacity())
|
||||
allSections.reserve(allSections.size() + 16);
|
||||
Section newSection(strref(), address);
|
||||
@ -1875,13 +1846,77 @@ void Asm::DummySection() {
|
||||
}
|
||||
|
||||
void Asm::EndSection() {
|
||||
if (link_all_section)
|
||||
LinkAllToSection();
|
||||
int section = (int)(current_section - &allSections[0]);
|
||||
if (section)
|
||||
current_section = &allSections[section-1];
|
||||
}
|
||||
|
||||
// Iterate through the current group of sections and assign addresses if this section is fixed
|
||||
// This is to handle the special linking of Merlin where sections are brought together pre-export
|
||||
void Asm::AssignAddressToGroup()
|
||||
{
|
||||
Section &curr = CurrSection();
|
||||
if (!curr.address_assigned)
|
||||
return;
|
||||
|
||||
// Put in all the sections cared about into either the fixed sections or the relative sections
|
||||
std::vector<Section*> FixedExport;
|
||||
std::vector<Section*> RelativeExport;
|
||||
int seg = SectionId();
|
||||
while (seg>=0) {
|
||||
Section &s = allSections[seg];
|
||||
if (s.address_assigned && s.type != ST_ZEROPAGE && s.start_address >= curr.start_address) {
|
||||
bool inserted = false;
|
||||
for (std::vector<Section*>::iterator i = FixedExport.begin(); i!=FixedExport.end(); ++i) {
|
||||
if (s.start_address < (*i)->start_address) {
|
||||
FixedExport.insert(i, &s);
|
||||
inserted = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (!inserted)
|
||||
FixedExport.push_back(&s);
|
||||
} else if (!s.address_assigned && s.type != ST_ZEROPAGE) {
|
||||
RelativeExport.push_back(&s);
|
||||
s.export_append = curr.export_append;
|
||||
}
|
||||
seg = allSections[seg].next_group;
|
||||
}
|
||||
|
||||
// in this case each block should be added individually in order of code / data / bss
|
||||
for (int type = ST_CODE; type <= ST_BSS; type++) {
|
||||
std::vector<Section*>::iterator i = RelativeExport.begin();
|
||||
while (i!=RelativeExport.end()) {
|
||||
Section *pSec = *i;
|
||||
if (pSec->type == type) {
|
||||
int bytes = pSec->address - pSec->start_address;
|
||||
size_t insert_after = FixedExport.size()-1;
|
||||
for (size_t p = 0; p<insert_after; p++) {
|
||||
int end_prev = FixedExport[p]->address;
|
||||
int start_next = FixedExport[p+1]->start_address;
|
||||
int avail = start_next - end_prev;
|
||||
if (avail >= bytes) {
|
||||
int addr = end_prev;
|
||||
addr += pSec->align_address <= 1 ? 0 :
|
||||
(pSec->align_address - (addr % pSec->align_address)) % pSec->align_address;
|
||||
if ((addr + bytes) <= start_next) {
|
||||
insert_after = p;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
int address = FixedExport[insert_after]->address;
|
||||
address += pSec->align_address <= 1 ? 0 :
|
||||
(pSec->align_address - (address % pSec->align_address)) % pSec->align_address;
|
||||
AssignAddressToSection((int)(pSec - &allSections[0]), address);
|
||||
FixedExport.insert((FixedExport.begin() + insert_after + 1), pSec);
|
||||
i = RelativeExport.erase(i);
|
||||
} else
|
||||
++i;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// list all export append names
|
||||
// for each valid export append name build a binary fixed address code
|
||||
// - find lowest and highest address
|
||||
@ -1906,7 +1941,7 @@ unsigned char* Asm::BuildExport(strref append, int &file_size, int &addr)
|
||||
if (((!append && !i->export_append) || append.same_str_case(i->export_append)) && i->type != ST_ZEROPAGE) {
|
||||
if (!i->IsMergedSection()) {
|
||||
if (i->IsRelativeSection()) {
|
||||
// priritize code over data, local code over included code for initial binary segment
|
||||
// prioritize code over data, local code over included code for initial binary segment
|
||||
if ((i->type == ST_CODE || i->type == ST_DATA) && i->first_group < 0 &&
|
||||
(first_link_section < 0 || (i->type == ST_CODE &&
|
||||
(allSections[first_link_section].type == ST_DATA ||
|
||||
@ -1959,7 +1994,7 @@ unsigned char* Asm::BuildExport(strref append, int &file_size, int &addr)
|
||||
if (sectype == i->type && ((!append && !i->export_append) || append.same_str_case(i->export_append))) {
|
||||
int id = (int)(&*i - &allSections[0]);
|
||||
if (i->IsRelativeSection() && i->first_group < 0) {
|
||||
// try to fit this section inbetween existing sections if possible
|
||||
// try to fit this section in between existing sections if possible
|
||||
|
||||
int insert_after = (int)FixedExport.size()-1;
|
||||
for (int f = 0; f < insert_after; f++) {
|
||||
@ -2219,70 +2254,6 @@ StatusCode Asm::AssignAddressToSection(int section_id, int address)
|
||||
return LinkRelocs(section_id, -1, s.start_address);
|
||||
}
|
||||
|
||||
// Append one section to the end of another
|
||||
StatusCode Asm::AppendSection(Section &s, Section &curr)
|
||||
{
|
||||
int section_id = int(&s - &allSections[0]);
|
||||
int section_new = (int)(&curr - &allSections[0]);
|
||||
if (s.IsRelativeSection() && !s.IsMergedSection()) {
|
||||
int section_size = (int)s.size();
|
||||
|
||||
int section_address = curr.GetPC();
|
||||
|
||||
// calculate space for alignment
|
||||
int align_size = s.align_address <= 1 ? 0 :
|
||||
(s.align_address-(section_address % s.align_address)) % s.align_address;
|
||||
|
||||
// Get base addresses
|
||||
curr.CheckOutputCapacity(section_size + align_size);
|
||||
|
||||
// add 0's
|
||||
for (int a = 0; a<align_size; a++)
|
||||
curr.AddByte(0);
|
||||
|
||||
section_address += align_size;
|
||||
|
||||
curr.CheckOutputCapacity((int)s.size());
|
||||
unsigned char *section_out = curr.curr;
|
||||
if (s.output)
|
||||
memcpy(section_out, s.output, s.size());
|
||||
curr.address += (int)s.size();
|
||||
curr.curr += s.size();
|
||||
free(s.output);
|
||||
s.output = 0;
|
||||
s.curr = 0;
|
||||
s.output_capacity = 0;
|
||||
|
||||
// Update address range and mark section as merged
|
||||
s.start_address = section_address;
|
||||
s.address += section_address;
|
||||
s.address_assigned = curr.address_assigned;
|
||||
s.merged_section = section_new;
|
||||
s.merged_offset = (int)(section_out - CurrSection().output);
|
||||
|
||||
// Merge in the listing at this point
|
||||
if (s.pListing) {
|
||||
if (!curr.pListing)
|
||||
curr.pListing = new Listing;
|
||||
if ((curr.pListing->size() + s.pListing->size()) > curr.pListing->capacity())
|
||||
curr.pListing->reserve(curr.pListing->size() + s.pListing->size() + 256);
|
||||
for (Listing::iterator si = s.pListing->begin(); si != s.pListing->end(); ++si) {
|
||||
struct ListLine lst = *si;
|
||||
lst.address += s.merged_offset;
|
||||
curr.pListing->push_back(lst);
|
||||
}
|
||||
delete s.pListing;
|
||||
s.pListing = nullptr;
|
||||
}
|
||||
|
||||
// All labels in this section can now be assigned
|
||||
LinkLabelsToAddress(section_id, curr.address_assigned ? -1 : section_new, section_address);
|
||||
|
||||
return LinkRelocs(section_id, section_new, section_address);
|
||||
}
|
||||
return ERROR_CANT_APPEND_SECTION_TO_TARGET;
|
||||
}
|
||||
|
||||
// Link sections with a specific name at this point
|
||||
// Relative sections will just be appeneded to a grouping list
|
||||
// Fixed address sections will be merged together
|
||||
@ -2301,18 +2272,15 @@ StatusCode Asm::LinkSections(strref name) {
|
||||
continue;
|
||||
// Zero page sections can only be linked with zero page sections
|
||||
if (i->type != ST_ZEROPAGE || CurrSection().type == ST_ZEROPAGE) {
|
||||
if (CurrSection().IsRelativeSection()) {
|
||||
i->export_append = CurrSection().export_append;
|
||||
if (!i->address_assigned) {
|
||||
if (i->first_group < 0) {
|
||||
int prev = last_section_group >= 0 ? last_section_group : SectionId();
|
||||
int curr = (int)(&*i - &allSections[0]);
|
||||
allSections[prev].next_group = curr;
|
||||
i->first_group = SectionId();
|
||||
i->first_group = CurrSection().first_group ? CurrSection().first_group : SectionId();
|
||||
last_section_group = curr;
|
||||
}
|
||||
} else {
|
||||
StatusCode status = AppendSection(*i, CurrSection());
|
||||
if (status != STATUS_OK)
|
||||
return status;
|
||||
}
|
||||
} else
|
||||
return ERROR_CANT_LINK_ZP_AND_NON_ZP;
|
||||
@ -4204,7 +4172,7 @@ StatusCode Asm::Directive_ORG(strref line)
|
||||
ERROR_TARGET_ADDRESS_MUST_EVALUATE_IMMEDIATELY : error;
|
||||
|
||||
// Section immediately followed by ORG reassigns that section to be fixed
|
||||
if (CurrSection().size()==0 && !CurrSection().IsDummySection()) {
|
||||
if (CurrSection().size()==0 && !CurrSection().IsDummySection() && !CurrSection().address_assigned) {
|
||||
if (CurrSection().type == ST_ZEROPAGE && addr >= 0x100)
|
||||
return ERROR_ZEROPAGE_SECTION_OUT_OF_RANGE;
|
||||
AssignAddressToSection(SectionId(), addr);
|
||||
@ -4237,9 +4205,10 @@ StatusCode Asm::Directive_LNK(strref line)
|
||||
strref file = line.between('"', '"');
|
||||
if (!file) // MERLIN: No quotes around include filenames
|
||||
file = line.split_range(filename_end_char_range);
|
||||
StatusCode error = ReadObjectFile(file);
|
||||
if (!error && !CurrSection().IsRelativeSection())
|
||||
link_all_section = true;
|
||||
int section_id = SectionId();
|
||||
StatusCode error = ReadObjectFile(file, SectionId());
|
||||
// restore current section
|
||||
current_section = &allSections[section_id];
|
||||
return error;
|
||||
}
|
||||
|
||||
@ -4491,6 +4460,7 @@ StatusCode Asm::ApplyDirective(AssemblerDirective dir, strref line, strref sourc
|
||||
line.skip(export_base_name.get_len());
|
||||
if (line)
|
||||
CurrSection().export_append = line.split_label();
|
||||
AssignAddressToGroup();
|
||||
break;
|
||||
|
||||
case AD_XC: // XC: MERLIN version of setting CPU
|
||||
@ -5537,8 +5507,6 @@ void Asm::Assemble(strref source, strref filename, bool obj_target)
|
||||
} else
|
||||
contextStack.curr().restart();
|
||||
}
|
||||
if (link_all_section)
|
||||
LinkAllToSection();
|
||||
if (error == STATUS_OK) {
|
||||
if (!obj_target)
|
||||
LinkZP();
|
||||
@ -5859,7 +5827,7 @@ StatusCode Asm::WriteObjectFile(strref filename)
|
||||
return STATUS_OK;
|
||||
}
|
||||
|
||||
StatusCode Asm::ReadObjectFile(strref filename)
|
||||
StatusCode Asm::ReadObjectFile(strref filename, int link_to_section)
|
||||
{
|
||||
size_t size;
|
||||
strown<512> file;
|
||||
@ -5889,6 +5857,9 @@ StatusCode Asm::ReadObjectFile(strref filename)
|
||||
int prevSection = SectionId();
|
||||
|
||||
short *aSctRmp = (short*)malloc(hdr.sections * sizeof(short));
|
||||
int last_linked_section = link_to_section;
|
||||
while (last_linked_section>=0 && allSections[last_linked_section].next_group >= 0)
|
||||
last_linked_section = allSections[last_linked_section].next_group;
|
||||
|
||||
// for now just append to existing assembler data
|
||||
|
||||
@ -5924,6 +5895,11 @@ StatusCode Asm::ReadObjectFile(strref filename)
|
||||
|
||||
bin_data += aSect[si].output_size;
|
||||
}
|
||||
if (last_linked_section>=0) {
|
||||
allSections[last_linked_section].next_group = SectionId();
|
||||
s.first_group = allSections[last_linked_section].first_group >=0 ? allSections[last_linked_section].first_group : last_linked_section;
|
||||
last_linked_section = SectionId();
|
||||
}
|
||||
}
|
||||
aSctRmp[si] = (short)allSections.size()-1;
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user