|
@@ -30,37 +30,43 @@
|
|
|
static LIST_HEAD(reset_list);
|
|
|
static DEFINE_MUTEX(driver_lock);
|
|
|
|
|
|
-static const struct vfio_platform_reset_combo reset_lookup_table[] = {
|
|
|
- {
|
|
|
- .compat = "calxeda,hb-xgmac",
|
|
|
- .reset_function_name = "vfio_platform_calxedaxgmac_reset",
|
|
|
- .module_name = "vfio-platform-calxedaxgmac",
|
|
|
- },
|
|
|
-};
|
|
|
-
|
|
|
-static void vfio_platform_get_reset(struct vfio_platform_device *vdev,
|
|
|
- struct device *dev)
|
|
|
+static vfio_platform_reset_fn_t vfio_platform_lookup_reset(const char *compat,
|
|
|
+ struct module **module)
|
|
|
{
|
|
|
- int (*reset)(struct vfio_platform_device *);
|
|
|
- int i;
|
|
|
+ struct vfio_platform_reset_node *iter;
|
|
|
+ vfio_platform_reset_fn_t reset_fn = NULL;
|
|
|
|
|
|
- for (i = 0 ; i < ARRAY_SIZE(reset_lookup_table); i++) {
|
|
|
- if (!strcmp(reset_lookup_table[i].compat, vdev->compat)) {
|
|
|
- request_module(reset_lookup_table[i].module_name);
|
|
|
- reset = __symbol_get(
|
|
|
- reset_lookup_table[i].reset_function_name);
|
|
|
- if (reset) {
|
|
|
- vdev->reset = reset;
|
|
|
- return;
|
|
|
- }
|
|
|
+ mutex_lock(&driver_lock);
|
|
|
+ list_for_each_entry(iter, &reset_list, link) {
|
|
|
+ if (!strcmp(iter->compat, compat) &&
|
|
|
+ try_module_get(iter->owner)) {
|
|
|
+ *module = iter->owner;
|
|
|
+ reset_fn = iter->reset;
|
|
|
+ break;
|
|
|
}
|
|
|
}
|
|
|
+ mutex_unlock(&driver_lock);
|
|
|
+ return reset_fn;
|
|
|
+}
|
|
|
+
|
|
|
+static void vfio_platform_get_reset(struct vfio_platform_device *vdev)
|
|
|
+{
|
|
|
+ char modname[256];
|
|
|
+
|
|
|
+ vdev->reset = vfio_platform_lookup_reset(vdev->compat,
|
|
|
+ &vdev->reset_module);
|
|
|
+ if (!vdev->reset) {
|
|
|
+ snprintf(modname, 256, "vfio-reset:%s", vdev->compat);
|
|
|
+ request_module(modname);
|
|
|
+ vdev->reset = vfio_platform_lookup_reset(vdev->compat,
|
|
|
+ &vdev->reset_module);
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
static void vfio_platform_put_reset(struct vfio_platform_device *vdev)
|
|
|
{
|
|
|
if (vdev->reset)
|
|
|
- symbol_put_addr(vdev->reset);
|
|
|
+ module_put(vdev->reset_module);
|
|
|
}
|
|
|
|
|
|
static int vfio_platform_regions_init(struct vfio_platform_device *vdev)
|
|
@@ -557,7 +563,7 @@ int vfio_platform_probe_common(struct vfio_platform_device *vdev,
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
- vfio_platform_get_reset(vdev, dev);
|
|
|
+ vfio_platform_get_reset(vdev);
|
|
|
|
|
|
mutex_init(&vdev->igate);
|
|
|
|