mdio-boardinfo.c 1.9 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * mdio-boardinfo - Collect pre-declarations for MDIO devices
  4. */
  5. #include <linux/kernel.h>
  6. #include <linux/slab.h>
  7. #include <linux/export.h>
  8. #include <linux/mutex.h>
  9. #include <linux/list.h>
  10. #include "mdio-boardinfo.h"
  11. static LIST_HEAD(mdio_board_list);
  12. static DEFINE_MUTEX(mdio_board_lock);
  13. /**
  14. * mdiobus_setup_mdiodev_from_board_info - create and setup MDIO devices
  15. * from pre-collected board specific MDIO information
  16. * @bus: Bus the board_info belongs to
  17. * @cb: Callback to create device on bus
  18. * Context: can sleep
  19. */
  20. void mdiobus_setup_mdiodev_from_board_info(struct mii_bus *bus,
  21. int (*cb)
  22. (struct mii_bus *bus,
  23. struct mdio_board_info *bi))
  24. {
  25. struct mdio_board_entry *be;
  26. struct mdio_board_entry *tmp;
  27. struct mdio_board_info *bi;
  28. int ret;
  29. mutex_lock(&mdio_board_lock);
  30. list_for_each_entry_safe(be, tmp, &mdio_board_list, list) {
  31. bi = &be->board_info;
  32. if (strcmp(bus->id, bi->bus_id))
  33. continue;
  34. mutex_unlock(&mdio_board_lock);
  35. ret = cb(bus, bi);
  36. mutex_lock(&mdio_board_lock);
  37. if (ret)
  38. continue;
  39. }
  40. mutex_unlock(&mdio_board_lock);
  41. }
  42. EXPORT_SYMBOL(mdiobus_setup_mdiodev_from_board_info);
  43. /**
  44. * mdiobus_register_board_info - register MDIO devices for a given board
  45. * @info: array of devices descriptors
  46. * @n: number of descriptors provided
  47. * Context: can sleep
  48. *
  49. * The board info passed can be marked with __initdata but be pointers
  50. * such as platform_data etc. are copied as-is
  51. */
  52. int mdiobus_register_board_info(const struct mdio_board_info *info,
  53. unsigned int n)
  54. {
  55. struct mdio_board_entry *be;
  56. unsigned int i;
  57. be = kcalloc(n, sizeof(*be), GFP_KERNEL);
  58. if (!be)
  59. return -ENOMEM;
  60. for (i = 0; i < n; i++, be++, info++) {
  61. memcpy(&be->board_info, info, sizeof(*info));
  62. mutex_lock(&mdio_board_lock);
  63. list_add_tail(&be->list, &mdio_board_list);
  64. mutex_unlock(&mdio_board_lock);
  65. }
  66. return 0;
  67. }
  68. EXPORT_SYMBOL(mdiobus_register_board_info);