1 /*******************************************************
3 * Citadel Dynamic Loading Module
4 * Written by Brian Costello
7 ******************************************************/
13 #include <sys/types.h>
18 #include "dynloader.h"
21 #include "sysdep_decls.h"
23 symtab *global_symtab;
25 struct FunctionHook *HookTable = NULL;
27 int DLoader_Exec_Cmd(char *cmdbuf)
32 void (*cmd_ptr)(void *);
34 for (t_sym = global_symtab; ((t_sym) && (strncmp(t_sym->server_cmd, cmdbuf, strlen(t_sym->server_cmd))) ); t_sym=t_sym->next)
39 if (!(fcn_handle = dlopen(t_sym->module_path, RTLD_NOW)))
42 syslog(LOG_NOTICE, "WARNING: module %s failed to load", t_sym->module_path);
46 cmd_ptr = dlsym(fcn_handle, t_sym->fcn_name);
47 if ((dl_error = dlerror()) != NULL)
49 syslog(LOG_NOTICE, "dlsym error: %s - %s", dl_error, t_sym->module_path);
52 (*cmd_ptr)(&cmdbuf[5]);
54 if ((dl_error = dlerror()) != NULL)
56 syslog(LOG_NOTICE, "dlclose error: %s", dl_error);
60 } /* If symbol found */
65 void add_symbol(char *fcn_name, char *server_cmd, char *info_msg, symtab **first_symtab)
67 symtab *new_symtab, *t_sym, *last_sym;
69 if (!(new_symtab = malloc(sizeof(symtab))))
71 perror("Malloc new symtab");
75 new_symtab->fcn_name = strdup(fcn_name);
76 new_symtab->server_cmd = strdup(server_cmd);
77 new_symtab->info_msg = strdup(info_msg);
78 new_symtab->next = NULL;
81 (*first_symtab) = new_symtab;
85 for (t_sym = (*first_symtab); (t_sym); t_sym = t_sym->next)
87 last_sym->next = new_symtab;
91 void DLoader_Init(char *pathname, symtab **my_symtab)
94 void (*h_init_fcn)(struct DLModule_Info *);
95 void (*h_get_symtab)(symtab **);
102 struct DLModule_Info dl_info;
106 global_symtab = NULL; /* Global symbol table */
108 if ((dir = opendir(pathname))==NULL)
114 while ((dptr=readdir(dir))!= NULL)
116 if (dptr->d_name[0] == '.')
119 filename = strdup(dptr->d_name);
120 snprintf(pathbuf, 512, "%s/%s", pathname, filename);
121 if (!(fcn_handle = dlopen(pathbuf, RTLD_NOW)))
123 dl_error = dlerror();
124 fprintf(stderr, "DLoader_Init dlopen failed (%s)", dl_error);
128 h_init_fcn = dlsym(fcn_handle, "Dynamic_Module_Init");
129 if ((dl_error = dlerror()) != NULL)
131 fprintf(stderr,"DLoader_Init dlsym failed (%s)", dl_error);
135 (*h_init_fcn)(&dl_info);
137 printf("Loaded module %s v%d.%d\nBy %s (%s)\n", dl_info.module_name,
138 dl_info.major_version,
139 dl_info.minor_version,
140 dl_info.module_author,
141 dl_info.module_author_email);
143 h_get_symtab = dlsym(fcn_handle, "Get_Symtab");
144 if ((dl_error = dlerror()) != NULL)
146 fprintf(stderr,"DLoader_Init dlsym failed for Get_Symtab (%s) on module %s", dl_error, dl_info.module_name);
150 /* Get the symbol table for the current module and link it on */
152 (*h_get_symtab)(&stab);
155 (*my_symtab) = global_symtab = stab;
159 for (t_sym = (*my_symtab) ; t_sym->next; t_sym=t_sym->next);
163 for (t_sym = stab; t_sym; t_sym=t_sym->next)
164 t_sym->module_path = (char *)strdup(pathbuf);
166 if ((dl_error = dlerror()) != NULL)
168 fprintf(stderr,"DLoader_Init dlclose failed (%s)", dl_error);
178 void CtdlRegisterHook(void *fcn_ptr, int fcn_type) {
180 struct FunctionHook *newfcn;
182 newfcn = (struct FunctionHook *) malloc(sizeof(struct FunctionHook));
183 newfcn->next = HookTable;
184 newfcn->h_function_pointer = fcn_ptr;
185 newfcn->h_type = fcn_type;
189 lprintf(5, "Registered a new function (type %d)\n", fcn_type);