2019-01-22 02:05:50 +08:00
|
|
|
// SPDX-License-Identifier: GPL-2.0+
|
2017-02-05 05:02:44 +08:00
|
|
|
/*
|
|
|
|
* mdio-boardinfo - Collect pre-declarations for MDIO devices
|
|
|
|
*/
|
|
|
|
|
|
|
|
#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
|
2020-07-07 09:49:34 +08:00
|
|
|
* @bus: Bus the board_info belongs to
|
|
|
|
* @cb: Callback to create device on bus
|
2017-02-05 05:02:44 +08:00
|
|
|
* Context: can sleep
|
|
|
|
*/
|
2017-03-29 03:57:09 +08:00
|
|
|
void mdiobus_setup_mdiodev_from_board_info(struct mii_bus *bus,
|
|
|
|
int (*cb)
|
|
|
|
(struct mii_bus *bus,
|
|
|
|
struct mdio_board_info *bi))
|
2017-02-05 05:02:44 +08:00
|
|
|
{
|
|
|
|
struct mdio_board_entry *be;
|
2018-04-19 08:00:47 +08:00
|
|
|
struct mdio_board_entry *tmp;
|
2017-02-05 05:02:44 +08:00
|
|
|
struct mdio_board_info *bi;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
mutex_lock(&mdio_board_lock);
|
2018-04-19 08:00:47 +08:00
|
|
|
list_for_each_entry_safe(be, tmp, &mdio_board_list, list) {
|
2017-02-05 05:02:44 +08:00
|
|
|
bi = &be->board_info;
|
|
|
|
|
|
|
|
if (strcmp(bus->id, bi->bus_id))
|
|
|
|
continue;
|
|
|
|
|
2018-04-19 08:00:47 +08:00
|
|
|
mutex_unlock(&mdio_board_lock);
|
2017-03-29 03:57:09 +08:00
|
|
|
ret = cb(bus, bi);
|
2018-04-19 08:00:47 +08:00
|
|
|
mutex_lock(&mdio_board_lock);
|
2017-03-29 03:57:09 +08:00
|
|
|
if (ret)
|
2017-02-05 05:02:44 +08:00
|
|
|
continue;
|
|
|
|
|
|
|
|
}
|
|
|
|
mutex_unlock(&mdio_board_lock);
|
|
|
|
}
|
2017-03-29 03:57:09 +08:00
|
|
|
EXPORT_SYMBOL(mdiobus_setup_mdiodev_from_board_info);
|
2017-02-05 05:02:44 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* 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;
|
|
|
|
}
|
2017-03-24 01:01:19 +08:00
|
|
|
EXPORT_SYMBOL(mdiobus_register_board_info);
|