plugin.c 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <dlfcn.h>
  4. #include <dirent.h>
  5. #include <unistd.h>
  6. #include <string.h>
  7. #include <sys/types.h>
  8. #include "model.h"
  9. #include "error.h"
  10. #include "pci.h"
  11. #include "config.h"
  12. void *pcilib_plugin_load(const char *name) {
  13. void *plug;
  14. char *fullname;
  15. const char *path;
  16. path = getenv("PCILIB_PLUGIN_DIR");
  17. if (!path) path = PCILIB_PLUGIN_DIR;
  18. fullname = malloc(strlen(path) + strlen(name) + 2);
  19. if (!fullname) return NULL;
  20. sprintf(fullname, "%s/%s", path, name);
  21. plug = dlopen(fullname, RTLD_LAZY|RTLD_LOCAL);
  22. free(fullname);
  23. return plug;
  24. }
  25. void pcilib_plugin_close(void *plug) {
  26. if (plug)
  27. dlclose(plug);
  28. }
  29. void *pcilib_plugin_get_symbol(void *plug, const char *symbol) {
  30. if ((!plug)||(!symbol)) return NULL;
  31. return dlsym(plug, symbol);
  32. }
  33. const pcilib_model_description_t *pcilib_get_plugin_model(pcilib_t *pcilib, void *plug, unsigned short vendor_id, unsigned short device_id, const char *model) {
  34. void *get_model;
  35. const pcilib_model_description_t *model_info;
  36. if (!plug) return NULL;
  37. get_model = dlsym(plug, "pcilib_get_event_model");
  38. if (!get_model) return NULL;
  39. model_info = ((const pcilib_model_description_t *(*)(pcilib_t *pcilib, unsigned short vendor_id, unsigned short device_id, const char *model))get_model)(pcilib, vendor_id, device_id, model);
  40. if (!model_info) return model_info;
  41. if ((PCILIB_VERSION_GET_MAJOR(model_info->interface_version) != PCILIB_VERSION_MAJOR)||(PCILIB_VERSION_GET_MINOR(model_info->interface_version) != PCILIB_VERSION_MINOR)) {
  42. pcilib_warning("Plugin %s exposes outdated interface version (%lu), pcitool supports (%lu)", model_info->name, model_info->interface_version, PCILIB_EVENT_INTERFACE_VERSION);
  43. return NULL;
  44. }
  45. return model_info;
  46. }
  47. const pcilib_model_description_t *pcilib_find_plugin_model(pcilib_t *pcilib, unsigned short vendor_id, unsigned short device_id, const char *model) {
  48. DIR *dir;
  49. const char *path;
  50. struct dirent *entry;
  51. void *plugin;
  52. const pcilib_model_description_t *model_info = NULL;
  53. path = getenv("PCILIB_PLUGIN_DIR");
  54. if (!path) path = PCILIB_PLUGIN_DIR;
  55. if (model) {
  56. plugin = pcilib_plugin_load(model);
  57. if (plugin) {
  58. model_info = pcilib_get_plugin_model(pcilib, plugin, vendor_id, device_id, model);
  59. if (model_info) {
  60. pcilib->event_plugin = plugin;
  61. return model_info;
  62. }
  63. pcilib_plugin_close(plugin);
  64. }
  65. }
  66. dir = opendir(path);
  67. if (!dir) return NULL;
  68. while ((entry = readdir(dir))) {
  69. const char *suffix = strstr(entry->d_name, ".so");
  70. if ((!suffix)||(strlen(suffix) != 3)) continue;
  71. if ((model)&&(!strcmp(entry->d_name, model)))
  72. continue;
  73. plugin = pcilib_plugin_load(entry->d_name);
  74. if (plugin) {
  75. model_info = pcilib_get_plugin_model(pcilib, plugin, vendor_id, device_id, model);
  76. if (model_info) {
  77. pcilib->event_plugin = plugin;
  78. break;
  79. }
  80. pcilib_plugin_close(plugin);
  81. }
  82. }
  83. closedir(dir);
  84. return model_info;
  85. }