|
@@ -0,0 +1,86 @@
|
|
|
+/*
|
|
|
+ * mdio-boardinfo - Collect pre-declarations for MDIO devices
|
|
|
+ *
|
|
|
+ * This program is free software; you can redistribute it and/or modify it
|
|
|
+ * under the terms of the GNU General Public License as published by the
|
|
|
+ * Free Software Foundation; either version 2 of the License, or (at your
|
|
|
+ * option) any later version.
|
|
|
+ */
|
|
|
+
|
|
|
+#include <linux/kernel.h>
|
|
|
+#include <linux/slab.h>
|
|
|
+#include <linux/export.h>
|
|
|
+#include <linux/mutex.h>
|
|
|
+#include <linux/list.h>
|
|
|
+
|
|
|
+#include "mdio-boardinfo.h"
|
|
|
+
|
|
|
+static LIST_HEAD(mdio_board_list);
|
|
|
+static DEFINE_MUTEX(mdio_board_lock);
|
|
|
+
|
|
|
+/**
|
|
|
+ * mdiobus_setup_mdiodev_from_board_info - create and setup MDIO devices
|
|
|
+ * from pre-collected board specific MDIO information
|
|
|
+ * @mdiodev: MDIO device pointer
|
|
|
+ * Context: can sleep
|
|
|
+ */
|
|
|
+void mdiobus_setup_mdiodev_from_board_info(struct mii_bus *bus)
|
|
|
+{
|
|
|
+ struct mdio_board_entry *be;
|
|
|
+ struct mdio_device *mdiodev;
|
|
|
+ struct mdio_board_info *bi;
|
|
|
+ int ret;
|
|
|
+
|
|
|
+ mutex_lock(&mdio_board_lock);
|
|
|
+ list_for_each_entry(be, &mdio_board_list, list) {
|
|
|
+ bi = &be->board_info;
|
|
|
+
|
|
|
+ if (strcmp(bus->id, bi->bus_id))
|
|
|
+ continue;
|
|
|
+
|
|
|
+ mdiodev = mdio_device_create(bus, bi->mdio_addr);
|
|
|
+ if (IS_ERR(mdiodev))
|
|
|
+ continue;
|
|
|
+
|
|
|
+ strncpy(mdiodev->modalias, bi->modalias,
|
|
|
+ sizeof(mdiodev->modalias));
|
|
|
+ mdiodev->bus_match = mdio_device_bus_match;
|
|
|
+ mdiodev->dev.platform_data = (void *)bi->platform_data;
|
|
|
+
|
|
|
+ ret = mdio_device_register(mdiodev);
|
|
|
+ if (ret) {
|
|
|
+ mdio_device_free(mdiodev);
|
|
|
+ continue;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ mutex_unlock(&mdio_board_lock);
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * mdio_register_board_info - register MDIO devices for a given board
|
|
|
+ * @info: array of devices descriptors
|
|
|
+ * @n: number of descriptors provided
|
|
|
+ * Context: can sleep
|
|
|
+ *
|
|
|
+ * The board info passed can be marked with __initdata but be pointers
|
|
|
+ * such as platform_data etc. are copied as-is
|
|
|
+ */
|
|
|
+int mdiobus_register_board_info(const struct mdio_board_info *info,
|
|
|
+ unsigned int n)
|
|
|
+{
|
|
|
+ struct mdio_board_entry *be;
|
|
|
+ unsigned int i;
|
|
|
+
|
|
|
+ be = kcalloc(n, sizeof(*be), GFP_KERNEL);
|
|
|
+ if (!be)
|
|
|
+ return -ENOMEM;
|
|
|
+
|
|
|
+ for (i = 0; i < n; i++, be++, info++) {
|
|
|
+ memcpy(&be->board_info, info, sizeof(*info));
|
|
|
+ mutex_lock(&mdio_board_lock);
|
|
|
+ list_add_tail(&be->list, &mdio_board_list);
|
|
|
+ mutex_unlock(&mdio_board_lock);
|
|
|
+ }
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|