1a2443fd1SAndrew Lunn // SPDX-License-Identifier: GPL-2.0+
2648ea013SFlorian Fainelli /*
3648ea013SFlorian Fainelli * mdio-boardinfo - Collect pre-declarations for MDIO devices
4648ea013SFlorian Fainelli */
5648ea013SFlorian Fainelli
6648ea013SFlorian Fainelli #include <linux/kernel.h>
7648ea013SFlorian Fainelli #include <linux/slab.h>
8648ea013SFlorian Fainelli #include <linux/export.h>
9648ea013SFlorian Fainelli #include <linux/mutex.h>
10648ea013SFlorian Fainelli #include <linux/list.h>
11648ea013SFlorian Fainelli
12648ea013SFlorian Fainelli #include "mdio-boardinfo.h"
13648ea013SFlorian Fainelli
14648ea013SFlorian Fainelli static LIST_HEAD(mdio_board_list);
15648ea013SFlorian Fainelli static DEFINE_MUTEX(mdio_board_lock);
16648ea013SFlorian Fainelli
17648ea013SFlorian Fainelli /**
18648ea013SFlorian Fainelli * mdiobus_setup_mdiodev_from_board_info - create and setup MDIO devices
19648ea013SFlorian Fainelli * from pre-collected board specific MDIO information
2019c5a5feSAndrew Lunn * @bus: Bus the board_info belongs to
2119c5a5feSAndrew Lunn * @cb: Callback to create device on bus
22648ea013SFlorian Fainelli * Context: can sleep
23648ea013SFlorian Fainelli */
mdiobus_setup_mdiodev_from_board_info(struct mii_bus * bus,int (* cb)(struct mii_bus * bus,struct mdio_board_info * bi))24d0281a56SFlorian Fainelli void mdiobus_setup_mdiodev_from_board_info(struct mii_bus *bus,
25d0281a56SFlorian Fainelli int (*cb)
26d0281a56SFlorian Fainelli (struct mii_bus *bus,
27d0281a56SFlorian Fainelli struct mdio_board_info *bi))
28648ea013SFlorian Fainelli {
29648ea013SFlorian Fainelli struct mdio_board_entry *be;
300263ea5cSAndrew Lunn struct mdio_board_entry *tmp;
31648ea013SFlorian Fainelli struct mdio_board_info *bi;
32648ea013SFlorian Fainelli int ret;
33648ea013SFlorian Fainelli
34648ea013SFlorian Fainelli mutex_lock(&mdio_board_lock);
350263ea5cSAndrew Lunn list_for_each_entry_safe(be, tmp, &mdio_board_list, list) {
36648ea013SFlorian Fainelli bi = &be->board_info;
37648ea013SFlorian Fainelli
38648ea013SFlorian Fainelli if (strcmp(bus->id, bi->bus_id))
39648ea013SFlorian Fainelli continue;
40648ea013SFlorian Fainelli
410263ea5cSAndrew Lunn mutex_unlock(&mdio_board_lock);
42d0281a56SFlorian Fainelli ret = cb(bus, bi);
430263ea5cSAndrew Lunn mutex_lock(&mdio_board_lock);
44d0281a56SFlorian Fainelli if (ret)
45648ea013SFlorian Fainelli continue;
46648ea013SFlorian Fainelli
47648ea013SFlorian Fainelli }
48648ea013SFlorian Fainelli mutex_unlock(&mdio_board_lock);
49648ea013SFlorian Fainelli }
50d0281a56SFlorian Fainelli EXPORT_SYMBOL(mdiobus_setup_mdiodev_from_board_info);
51648ea013SFlorian Fainelli
52648ea013SFlorian Fainelli /**
53*4db0964aSYang Yingliang * mdiobus_register_board_info - register MDIO devices for a given board
54648ea013SFlorian Fainelli * @info: array of devices descriptors
55648ea013SFlorian Fainelli * @n: number of descriptors provided
56648ea013SFlorian Fainelli * Context: can sleep
57648ea013SFlorian Fainelli *
58648ea013SFlorian Fainelli * The board info passed can be marked with __initdata but be pointers
59648ea013SFlorian Fainelli * such as platform_data etc. are copied as-is
60648ea013SFlorian Fainelli */
mdiobus_register_board_info(const struct mdio_board_info * info,unsigned int n)61648ea013SFlorian Fainelli int mdiobus_register_board_info(const struct mdio_board_info *info,
62648ea013SFlorian Fainelli unsigned int n)
63648ea013SFlorian Fainelli {
64648ea013SFlorian Fainelli struct mdio_board_entry *be;
65648ea013SFlorian Fainelli unsigned int i;
66648ea013SFlorian Fainelli
67648ea013SFlorian Fainelli be = kcalloc(n, sizeof(*be), GFP_KERNEL);
68648ea013SFlorian Fainelli if (!be)
69648ea013SFlorian Fainelli return -ENOMEM;
70648ea013SFlorian Fainelli
71648ea013SFlorian Fainelli for (i = 0; i < n; i++, be++, info++) {
72648ea013SFlorian Fainelli memcpy(&be->board_info, info, sizeof(*info));
73648ea013SFlorian Fainelli mutex_lock(&mdio_board_lock);
74648ea013SFlorian Fainelli list_add_tail(&be->list, &mdio_board_list);
75648ea013SFlorian Fainelli mutex_unlock(&mdio_board_lock);
76648ea013SFlorian Fainelli }
77648ea013SFlorian Fainelli
78648ea013SFlorian Fainelli return 0;
79648ea013SFlorian Fainelli }
8090eff909SFlorian Fainelli EXPORT_SYMBOL(mdiobus_register_board_info);
81