plugin.c 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119
  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. printf("%u %u\n", model_info->interface_version, PCILIB_EVENT_INTERFACE_VERSION);
  43. pcilib_warning("Plugin %s exposes outdated interface version (%u.%u.%u), pcilib interface version is (%u.%u.%u)", model_info->name,
  44. PCILIB_VERSION_GET_MAJOR(model_info->interface_version),
  45. PCILIB_VERSION_GET_MINOR(model_info->interface_version),
  46. PCILIB_VERSION_GET_MICRO(model_info->interface_version),
  47. PCILIB_VERSION_GET_MAJOR(PCILIB_EVENT_INTERFACE_VERSION),
  48. PCILIB_VERSION_GET_MINOR(PCILIB_EVENT_INTERFACE_VERSION),
  49. PCILIB_VERSION_GET_MICRO(PCILIB_EVENT_INTERFACE_VERSION)
  50. );
  51. return NULL;
  52. }
  53. return model_info;
  54. }
  55. const pcilib_model_description_t *pcilib_find_plugin_model(pcilib_t *pcilib, unsigned short vendor_id, unsigned short device_id, const char *model) {
  56. DIR *dir;
  57. const char *path;
  58. struct dirent *entry;
  59. void *plugin;
  60. const pcilib_model_description_t *model_info = NULL;
  61. path = getenv("PCILIB_PLUGIN_DIR");
  62. if (!path) path = PCILIB_PLUGIN_DIR;
  63. if (model) {
  64. plugin = pcilib_plugin_load(model);
  65. if (plugin) {
  66. model_info = pcilib_get_plugin_model(pcilib, plugin, vendor_id, device_id, model);
  67. if (model_info) {
  68. pcilib->event_plugin = plugin;
  69. return model_info;
  70. }
  71. pcilib_plugin_close(plugin);
  72. }
  73. }
  74. dir = opendir(path);
  75. if (!dir) return NULL;
  76. while ((entry = readdir(dir))) {
  77. const char *suffix = strstr(entry->d_name, ".so");
  78. if ((!suffix)||(strlen(suffix) != 3)) continue;
  79. if ((model)&&(!strcmp(entry->d_name, model)))
  80. continue;
  81. plugin = pcilib_plugin_load(entry->d_name);
  82. if (plugin) {
  83. model_info = pcilib_get_plugin_model(pcilib, plugin, vendor_id, device_id, model);
  84. if (model_info) {
  85. pcilib->event_plugin = plugin;
  86. break;
  87. }
  88. pcilib_plugin_close(plugin);
  89. }
  90. }
  91. closedir(dir);
  92. return model_info;
  93. }