#include #include #include #include #include #include #include #include #include #include #include #include // trim from start // Taken from https://stackoverflow.com/a/217605 static inline void ltrim(std::string &s) { s.erase(s.begin(), std::find_if(s.begin(), s.end(), [](int ch) { return !std::isspace(ch); })); } // trim from end (in place) static inline void rtrim(std::string &s) { s.erase(std::find_if(s.rbegin(), s.rend(), [](int ch) { return !std::isspace(ch); }).base(), s.end()); } // trim from both ends static inline void trim(std::string &s) { ltrim(s); rtrim(s); } uint32_t byte_swap(uint32_t v) { return ((v >> 24) & 0xff) | ((v << 8) & 0xff0000) | ((v >> 8) & 0xff00) | ((v << 24) & 0xff000000); } enum class ReadMode { INVALID, TEXT, DATA }; void writeExports(std::ofstream &out, const std::string &moduleName, bool isData, const std::vector &exports) { // Align module name up to 8 bytes auto moduleNameSize = (moduleName.length() + 1 + 7) & ~7; // Calculate the data block size auto exportSecSize = exports.size() * 8; if (exportSecSize < moduleNameSize) { exportSecSize = moduleNameSize; } // Calculate export hash uint32_t exportsHash = crc32(0, Z_NULL, 0); for (auto &exp : exports) { exportsHash = crc32(exportsHash, reinterpret_cast(exp.data()), exp.size() + 1); } std::array extraHashBytes; extraHashBytes.fill(0); exportsHash = crc32(exportsHash, extraHashBytes.data(), extraHashBytes.size()); // Setup section data std::vector secData; secData.resize(exportSecSize / 4, 0); memcpy(secData.data(), moduleName.c_str(), moduleName.length()); out << std::endl; if (isData) { out << ".section .dimport_"; } else { out << ".section .fimport_"; } out << moduleName << ", \"ax\", @0x80000002" << std::endl; out << ".align 4" << std::endl; out << std::endl; out << ".long " << exports.size() << std::endl; out << ".long 0x" << std::hex << exportsHash << std::endl; out << std::endl; const char *type = isData ? "@object" : "@function"; for (auto i = 0; i < exportSecSize / 8; ++i) { if (i < exports.size()) { out << ".global " << exports[i] << std::endl; out << ".type " << exports[i] << ", " << type << std::endl; out << exports[i] << ":" << std::endl; } out << ".long 0x" << std::hex << byte_swap(secData[i * 2 + 0]) << std::endl; out << ".long 0x" << std::hex << byte_swap(secData[i * 2 + 1]) << std::endl; out << std::endl; } } int main(int argc, char **argv) { std::string moduleName; std::vector funcExports, dataExports; ReadMode readMode = ReadMode::INVALID; if (argc < 3) { std::cout << argv[0] << " " << std::endl; return 0; } { std::ifstream in; in.open(argv[1]); if (!in.is_open()) { std::cout << "Could not open file " << argv[1] << " for reading" << std::endl; return -1; } std::string line; while (std::getline(in, line)) { // Trim comments std::size_t commentOffset = line.find("//"); if (commentOffset != std::string::npos) { line = line.substr(0, commentOffset); } // Trim whitespace trim(line); // Skip blank lines if (line.length() == 0) { continue; } // Look for section headers if (line[0] == ':') { if (line.substr(1) == "TEXT") { readMode = ReadMode::TEXT; } else if (line.substr(1) == "DATA") { readMode = ReadMode::DATA; } else if (line.substr(1, 4) == "NAME") { moduleName = line.substr(6); } else { std::cout << "Unexpected section type" << std::endl; return -1; } continue; } if (readMode == ReadMode::TEXT) { funcExports.push_back(line); } else if (readMode == ReadMode::DATA) { dataExports.push_back(line); } else { std::cout << "Unexpected section data" << std::endl; return -1; } } } { std::ofstream out; out.open(argv[2]); if (!out.is_open()) { std::cout << "Could not open file " << argv[2] << " for writing" << std::endl; return -1; } if (funcExports.size() > 0) { writeExports(out, moduleName, false, funcExports); } if (dataExports.size() > 0) { writeExports(out, moduleName, true, dataExports); } } return 0; }