How would one go about linking (some) symbols to specific fixed addresses using GNU ld so that the binary could still be executed as normal in Linux (x86)? There will not be any accesses to those symbols, but their addresses are important.
For example, I’d have the following structure:
struct FooBar { Register32 field_1; Register32 field_2; //... }; struct FooBar foobar;
I’d like to link foobar to address 0x76543210, but link the standard libraries and the rest of the application normally. The application will then make use of the address of foobar, but will not reference the (possibly non-existent) memory behind it.
The rationale for this request is that this same source can be used on two platforms: On the native platform, Register32 can simply be a volatile uint32_t, but on Linux Register32 is a C++ object with the same size as a uint32_t that defines e.g. operator=, which will then use the address of the object and sends a request to a communication framework with that address (and the data) to perform the actual access on remote hardware. The linker would thus ensure the Register32 fields of the struct refer to the correct ‘addresses’.
The suggestion by litb to use
--defsym symbol=addressdoes work, but is a bit cumbersome when you have a few dozen such instances to map. However,--just-symbols=symbolfiledoes just the trick. It took me a while to find out the syntax of thesymbolfile, which isThe spaces seem to be required, as otherwise
ldreportsfile format not recognized; treating as linker script.