mirror of
https://github.com/ksherlock/wdc-utils.git
synced 2024-12-12 19:30:38 +00:00
linker updates [WIP]
This commit is contained in:
parent
1deafa3ad1
commit
cef989b17b
249
link.cpp
249
link.cpp
@ -30,6 +30,14 @@
|
|||||||
#define O_BINARY 0
|
#define O_BINARY 0
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
struct {
|
||||||
|
bool _v = false;
|
||||||
|
bool _C = false;
|
||||||
|
bool _X = false;
|
||||||
|
unsigned _errors = 0;
|
||||||
|
} flags;
|
||||||
|
|
||||||
enum class endian {
|
enum class endian {
|
||||||
little = __ORDER_LITTLE_ENDIAN__,
|
little = __ORDER_LITTLE_ENDIAN__,
|
||||||
big = __ORDER_BIG_ENDIAN__,
|
big = __ORDER_BIG_ENDIAN__,
|
||||||
@ -272,7 +280,12 @@ void one_module(const std::vector<uint8_t> &data, const std::vector<uint8_t> &se
|
|||||||
|
|
||||||
// convert local sections to global
|
// convert local sections to global
|
||||||
for (auto &s : local_sections) {
|
for (auto &s : local_sections) {
|
||||||
if (s.number <= SECT_UDATA) continue;
|
if (s.number <= SECT_UDATA) {
|
||||||
|
sections[s.number].size += s.size; // for page0 / udata sections.
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// todo -- should install section name as global symbol?
|
||||||
|
|
||||||
auto iter = section_map.find(s.name);
|
auto iter = section_map.find(s.name);
|
||||||
if (iter == section_map.end()) {
|
if (iter == section_map.end()) {
|
||||||
@ -282,10 +295,13 @@ void one_module(const std::vector<uint8_t> &data, const std::vector<uint8_t> &se
|
|||||||
sections.emplace_back(s);
|
sections.emplace_back(s);
|
||||||
symbol_map.emplace(s.name, virtual_section);
|
symbol_map.emplace(s.name, virtual_section);
|
||||||
} else {
|
} else {
|
||||||
const auto &ss = sections[iter->second];
|
auto &ss = sections[iter->second];
|
||||||
assert(ss.flags == s.flags); // check org????
|
assert(ss.flags == s.flags); // check org????
|
||||||
remap_section[s.number] = iter->second;
|
remap_section[s.number] = iter->second;
|
||||||
s.number = iter->second;
|
s.number = iter->second;
|
||||||
|
|
||||||
|
// update size (for ref-only sections)
|
||||||
|
ss.size += s.size;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -335,7 +351,10 @@ void one_module(const std::vector<uint8_t> &data, const std::vector<uint8_t> &se
|
|||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
// ok if symbols are identical..
|
// ok if symbols are identical..
|
||||||
assert(ss.type == ss.type && ss.flags == s.flags && ss.section == s.section && ss.offset == s.offset);
|
if (ss.type != s.type || ss.flags != s.flags || ss.section != s.section || ss.offset != s.offset) {
|
||||||
|
warnx("Duplicate label %s", s.name.c_str());
|
||||||
|
flags._errors++;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -349,7 +368,6 @@ void one_module(const std::vector<uint8_t> &data, const std::vector<uint8_t> &se
|
|||||||
uint8_t op = read_8(iter);
|
uint8_t op = read_8(iter);
|
||||||
if (op == REC_END) return;
|
if (op == REC_END) return;
|
||||||
|
|
||||||
++iter;
|
|
||||||
if (op < 0xf0) {
|
if (op < 0xf0) {
|
||||||
data_ptr->insert(data_ptr->end(), iter, iter + op);
|
data_ptr->insert(data_ptr->end(), iter, iter + op);
|
||||||
iter += op;
|
iter += op;
|
||||||
@ -475,7 +493,11 @@ void one_module(const std::vector<uint8_t> &data, const std::vector<uint8_t> &se
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* n.b -- UDATA and PAGE0 are ref only, therefore no data is generated.
|
||||||
|
* as a special case for UDATA (but not PAGE0) have a flag so it will be 0-filled and generate data?
|
||||||
|
*
|
||||||
|
*/
|
||||||
void init() {
|
void init() {
|
||||||
|
|
||||||
sections.resize(5);
|
sections.resize(5);
|
||||||
@ -513,9 +535,9 @@ void init() {
|
|||||||
static std::string names[] = {
|
static std::string names[] = {
|
||||||
"_BEG_PAGE0", "_END_PAGE0",
|
"_BEG_PAGE0", "_END_PAGE0",
|
||||||
"_BEG_CODE", "_END_CODE",
|
"_BEG_CODE", "_END_CODE",
|
||||||
"_BEG_KDATA", "_END_KDATA"
|
"_BEG_KDATA", "_END_KDATA",
|
||||||
"_BEG_DATA", "_END_DATA"
|
"_BEG_DATA", "_END_DATA",
|
||||||
"_BEG_UDATA", "_END_UDATA"
|
"_BEG_UDATA", "_END_UDATA",
|
||||||
};
|
};
|
||||||
|
|
||||||
for (int i = 0; i < 5; ++i) {
|
for (int i = 0; i < 5; ++i) {
|
||||||
@ -575,123 +597,123 @@ void merge_data() {
|
|||||||
|
|
||||||
uint32_t total_data_size = 0;
|
uint32_t total_data_size = 0;
|
||||||
uint32_t total_code_size = 0;
|
uint32_t total_code_size = 0;
|
||||||
|
unsigned data_sections = 0;
|
||||||
|
unsigned code_sections = 0;
|
||||||
|
|
||||||
for (const auto &s : sections) {
|
for (const auto &s : sections) {
|
||||||
if (s.flags & SEC_REF_ONLY) continue;
|
if (s.flags & SEC_REF_ONLY) continue;
|
||||||
if (s.flags & SEC_DATA) total_data_size += s.data.size();
|
if (s.flags & SEC_DATA) {
|
||||||
else total_code_size += s.data.size();
|
total_data_size += s.data.size();
|
||||||
|
data_sections++;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
if (s.data.size() > 0xffff) {
|
||||||
|
flags._errors++;
|
||||||
|
warnx("code section %s ($%04x) exceeds bank size.",
|
||||||
|
s.name.c_str(), (uint32_t)s.data.size());
|
||||||
|
}
|
||||||
|
total_code_size += s.data.size();
|
||||||
|
code_sections++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// add in udata...
|
||||||
|
total_data_size += sections[SECT_UDATA].size;
|
||||||
|
|
||||||
|
|
||||||
|
if (sections[SECT_CODE].data.size() + total_data_size < 0xffff) {
|
||||||
|
auto &s = sections[SECT_CODE];
|
||||||
|
new_number = SECT_CODE;
|
||||||
|
new_data = std::move(s.data);
|
||||||
|
new_expr = std::move(s.expressions);
|
||||||
|
s.data.clear();
|
||||||
|
s.expressions.clear();
|
||||||
|
} else {
|
||||||
|
new_number = SECT_DATA;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// also merge code if total size is ok...
|
// new section, offset fudge pair.
|
||||||
uint32_t size = 0;
|
|
||||||
for (int i = 1; i < 5; ++i) {
|
|
||||||
size += sections[i].data.size();
|
|
||||||
}
|
|
||||||
if (size <= 0xffff) {
|
|
||||||
new_data = std::move(sections[SECT_CODE].data);
|
|
||||||
new_expr = std::move(sections[SECT_CODE].expressions);
|
|
||||||
new_number = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector< std::pair<unsigned, uint32_t> > remap;
|
std::vector< std::pair<unsigned, uint32_t> > remap;
|
||||||
remap.reserve(sections.size());
|
remap.reserve(sections.size());
|
||||||
|
|
||||||
for (unsigned i = 0; i < sections.size(); ++i)
|
for (unsigned i = 0; i < sections.size(); ++i)
|
||||||
remap.emplace_back(std::make_pair(i, i));
|
remap.emplace_back(std::make_pair(i, 0));
|
||||||
|
|
||||||
|
|
||||||
uint32_t fudge_kdata = 0;
|
if (total_data_size > 0xffff) {
|
||||||
uint32_t fudge_data = 0;
|
warnx("total data ($%04x) exceeds bank size,", total_data_size);
|
||||||
uint32_t fudge_udata = 0;
|
|
||||||
|
|
||||||
section *s = §ions[SECT_KDATA];
|
|
||||||
fudge_kdata = new_data.size();
|
|
||||||
|
|
||||||
new_data.insert(new_data.end(), s->data.begin(), s->data.end());
|
|
||||||
new_expr.insert(new_expr.end(),
|
|
||||||
std::make_move_iterator(s->expressions.begin()),
|
|
||||||
std::make_move_iterator(s->expressions.end())
|
|
||||||
);
|
|
||||||
*s = section{};
|
|
||||||
|
|
||||||
s = §ions[SECT_DATA];
|
|
||||||
fudge_data = new_data.size();
|
|
||||||
|
|
||||||
new_data.insert(new_data.end(), s->data.begin(), s->data.end());
|
|
||||||
new_expr.insert(new_expr.end(),
|
|
||||||
std::make_move_iterator(s->expressions.begin()),
|
|
||||||
std::make_move_iterator(s->expressions.end())
|
|
||||||
);
|
|
||||||
*s = section{};
|
|
||||||
|
|
||||||
|
|
||||||
s = §ions[SECT_UDATA];
|
|
||||||
fudge_udata = new_data.size();
|
|
||||||
|
|
||||||
new_data.insert(new_data.end(), s->data.begin(), s->data.end());
|
|
||||||
new_expr.insert(new_expr.end(),
|
|
||||||
std::make_move_iterator(s->expressions.begin()),
|
|
||||||
std::make_move_iterator(s->expressions.end())
|
|
||||||
);
|
|
||||||
*s = section{};
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
if (size > 0xffff) {
|
|
||||||
// not merged into code.
|
|
||||||
|
|
||||||
new_number = sections.size();
|
|
||||||
|
|
||||||
section tmp;
|
|
||||||
tmp.name = "data";
|
|
||||||
tmp.flags = SEC_DATA;
|
|
||||||
tmp.data = std::move(new_data);
|
|
||||||
tmp.expressions = std::move(new_expr);
|
|
||||||
sections.emplace_back(tmp);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// update all symbols with the new location / offset.
|
// merge all code sections?
|
||||||
|
if (total_code_size <= 0xffff && code_sections > 0) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
for (auto &s : sections) {
|
||||||
|
if (s.flags & SEC_REF_ONLY) continue;
|
||||||
|
if ((s.flags & SEC_DATA) == 0) continue;
|
||||||
|
|
||||||
|
uint32_t offset = new_data.size();
|
||||||
|
remap[s.number] = std::make_pair(new_number, offset);
|
||||||
|
new_data.insert(new_data.end(), s.data.begin(), s.data.end());
|
||||||
|
new_expr.insert(new_expr.end(),
|
||||||
|
std::make_move_iterator(s.expressions.begin()),
|
||||||
|
std::make_move_iterator(s.expressions.end())
|
||||||
|
);
|
||||||
|
|
||||||
|
// preserve the name and flags.
|
||||||
|
s.data.clear();
|
||||||
|
s.expressions.clear();
|
||||||
|
s.size = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// add in SECT_UDATA
|
||||||
|
{
|
||||||
|
auto &s = sections[SECT_UDATA];
|
||||||
|
uint32_t offset = new_data.size();
|
||||||
|
|
||||||
|
remap[s.number] = std::make_pair(new_number, offset);
|
||||||
|
// reference only -- no expressions or data.
|
||||||
|
new_data.insert(new_data.end(), s.size, 0);
|
||||||
|
|
||||||
|
s.size = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
auto &s = sections[new_number];
|
||||||
|
s.size = new_data.size();
|
||||||
|
s.data = std::move(new_data);
|
||||||
|
s.expressions = std::move(new_expr);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// now remap symbols
|
||||||
for (auto &s : symbols) {
|
for (auto &s : symbols) {
|
||||||
if ((s.type & 0x0f) == S_REL) switch(s.section) {
|
if ((s.type & 0x0f) == S_REL) {
|
||||||
case SECT_DATA:
|
auto x = remap[s.section];
|
||||||
s.section = new_number;
|
s.section = x.first;
|
||||||
s.offset += fudge_data;
|
s.offset += x.second;
|
||||||
break;
|
|
||||||
case SECT_KDATA:
|
|
||||||
s.section = new_number;
|
|
||||||
s.offset += fudge_kdata;
|
|
||||||
break;
|
|
||||||
case SECT_UDATA:
|
|
||||||
s.section = new_number;
|
|
||||||
s.offset += fudge_udata;
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// and expressions...
|
||||||
|
|
||||||
// update expressions with new section / offset.
|
// update expressions with new section / offset.
|
||||||
for (auto &s : sections) {
|
for (auto &s : sections) {
|
||||||
for (auto &e : s.expressions) {
|
for (auto &e : s.expressions) {
|
||||||
bool delta = false;
|
bool delta = false;
|
||||||
for (auto &t : e.stack) {
|
for (auto &t : e.stack) {
|
||||||
if (t.tag == OP_LOC) {
|
if (t.tag == OP_LOC) {
|
||||||
switch(t.section) {
|
auto x = remap[t.section];
|
||||||
case SECT_DATA:
|
if (x.first != t.section || x.second != 0) {
|
||||||
t.section = new_number;
|
|
||||||
t.value += fudge_data;
|
|
||||||
delta = true;
|
delta = true;
|
||||||
break;
|
t.section = x.first;
|
||||||
case SECT_KDATA:
|
t.value += x.second;
|
||||||
t.section = new_number;
|
|
||||||
t.value += fudge_kdata;
|
|
||||||
delta = true;
|
|
||||||
break;
|
|
||||||
case SECT_UDATA:
|
|
||||||
t.section = new_number;
|
|
||||||
t.value += fudge_udata;
|
|
||||||
delta = true;
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -700,7 +722,6 @@ void merge_data() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -810,6 +831,10 @@ bool one_file(const std::string &name) {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (flags._v) {
|
||||||
|
printf("Processing %s:%s\n", name.c_str(), module_name.c_str());
|
||||||
|
}
|
||||||
|
|
||||||
// should probably pass in name and module....
|
// should probably pass in name and module....
|
||||||
one_module(record_data, section_data, symbol_data);
|
one_module(record_data, section_data, symbol_data);
|
||||||
rv = true;
|
rv = true;
|
||||||
@ -839,8 +864,9 @@ int main(int argc, char **argv) {
|
|||||||
|
|
||||||
|
|
||||||
int c;
|
int c;
|
||||||
while ((c = getopt(argc, argv, "CXL:l:o:")) != -1) {
|
while ((c = getopt(argc, argv, "vCXL:l:o:")) != -1) {
|
||||||
switch(c) {
|
switch(c) {
|
||||||
|
case 'v': flags._v = true; break;
|
||||||
case 'X': _X = true; break;
|
case 'X': _X = true; break;
|
||||||
case 'C': _C = true; break;
|
case 'C': _C = true; break;
|
||||||
case 'o': _o = optarg; break;
|
case 'o': _o = optarg; break;
|
||||||
@ -862,13 +888,14 @@ int main(int argc, char **argv) {
|
|||||||
init();
|
init();
|
||||||
|
|
||||||
for (int i = 0 ; i < argc; ++i) {
|
for (int i = 0 ; i < argc; ++i) {
|
||||||
one_file(argv[i]);
|
if (!one_file(argv[i])) flags._errors++;
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
simplify();
|
simplify();
|
||||||
|
|
||||||
// for each undefined, try to find it in a library...
|
// for each undefined, try to find it in a library...
|
||||||
|
// ... except for the _BEG / _END symbols!
|
||||||
|
|
||||||
for (const auto & s : undefined_symbols) {
|
for (const auto & s : undefined_symbols) {
|
||||||
printf("%s\n", s.c_str());
|
printf("%s\n", s.c_str());
|
||||||
@ -876,8 +903,26 @@ int main(int argc, char **argv) {
|
|||||||
|
|
||||||
generate_end();
|
generate_end();
|
||||||
|
|
||||||
|
|
||||||
|
if (flags._v) {
|
||||||
|
for (auto &s : sections) {
|
||||||
|
//if (s.flags & SEC_REF_ONLY) continue;
|
||||||
|
printf("section %3d %-20s $%04x $%04x\n",
|
||||||
|
s.number, s.name.c_str(), (uint32_t)s.data.size(), s.size);
|
||||||
|
}
|
||||||
|
fputs("\n", stdout);
|
||||||
|
}
|
||||||
|
|
||||||
merge_data();
|
merge_data();
|
||||||
|
|
||||||
|
if (flags._v) {
|
||||||
|
for (auto &s : sections) {
|
||||||
|
//if (s.flags & SEC_REF_ONLY) continue;
|
||||||
|
printf("section %3d %-20s $%04x $%04x\n",
|
||||||
|
s.number, s.name.c_str(), (uint32_t)s.data.size(), s.size);
|
||||||
|
}
|
||||||
|
fputs("\n", stdout);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user