Hai due opzioni, da cui è possibile scegliere:
Opzione 1: esportare tutti i simboli dal vostro eseguibile. Questa è un'opzione semplice, solo quando si costruisce un file eseguibile, aggiungere un flag -Wl,--export-dynamic
. Ciò renderebbe tutte le funzioni disponibili per le chiamate in biblioteca.
Opzione 2: creare un file di simboli di esportazione con un elenco di funzioni e utilizzare -Wl,--dynamic-list=exported.txt
. Ciò richiede una manutenzione, ma più accurata.
Per dimostrare: semplice eseguibile e libreria caricata dinamicamente.
#include <stdio.h>
#include <dlfcn.h>
void exported_callback() /*< Function we want to export */
{
printf("Hello from callback!\n");
}
viud unexported_callback() /*< Function we don't want to export */
{
printf("Hello from unexported callback!\n");
}
typedef void (*lib_func)();
int call_library()
{
void *handle = NULL;
lib_func func = NULL;
handle = dlopen("./libprog.so", RTLD_NOW | RTLD_GLOBAL);
if (handle == NULL)
{
fprintf(stderr, "Unable to open lib: %s\n", dlerror());
return -1;
}
func = dlsym(handle, "library_function");
if (func == NULL) {
fprintf(stderr, "Unable to get symbol\n");
return -1;
}
func();
return 0;
}
int main(int argc, const char *argv[])
{
printf("Hello from main!\n");
call_library();
return 0;
}
Codice libreria (lib.c):
#include <stdio.h>
int exported_callback();
int library_function()
{
printf("Hello from library!\n");
exported_callback();
/* unexported_callback(); */ /*< This one will not be exported in the second case */
return 0;
}
Quindi, prima costruire la biblioteca (questo passaggio non differisce):
gcc -shared -fPIC lib.c -o libprog.so
Ora costruire eseguibile con tutti i simboli esportati:
gcc -Wl,--export-dynamic main.c -o prog.exe -ldl
Run esempio:
$ ./prog.exe
Hello from main!
Hello from library!
Hello from callback!
Simboli expo ari:
$ objdump -e prog.exe -T | grep callback
00000000004009f4 g DF .text 0000000000000015 Base exported_callback
0000000000400a09 g DF .text 0000000000000015 Base unexported_callback
Ora, con l'elenco esportato (exported.txt
):
{
extern "C"
{
exported_callback;
};
};
Corporatura & controllo simboli visibili:
$ gcc -Wl,--dynamic-list=./exported.txt main.c -o prog.exe -ldl
$ objdump -e prog.exe -T | grep callback
0000000000400774 g DF .text 0000000000000015 Base exported_callback
Preferisco fare 'if (callback) {callback(); ritorno 0; } else {return -1; } 'per indicare un errore ed evitare di chiamare' NULL' (che sarebbe fatale). – glglgl
A destra, rimosso il pugno. – user746527