ksz9477_i2c.c 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Microchip KSZ9477 series register access through I2C
  4. *
  5. * Copyright (C) 2018-2019 Microchip Technology Inc.
  6. */
  7. #include <linux/i2c.h>
  8. #include <linux/kernel.h>
  9. #include <linux/module.h>
  10. #include <linux/regmap.h>
  11. #include "ksz_common.h"
  12. KSZ_REGMAP_TABLE(ksz9477, not_used, 16, 0, 0);
  13. static int ksz9477_i2c_probe(struct i2c_client *i2c,
  14. const struct i2c_device_id *i2c_id)
  15. {
  16. struct regmap_config rc;
  17. struct ksz_device *dev;
  18. int i, ret;
  19. dev = ksz_switch_alloc(&i2c->dev, i2c);
  20. if (!dev)
  21. return -ENOMEM;
  22. for (i = 0; i < ARRAY_SIZE(ksz9477_regmap_config); i++) {
  23. rc = ksz9477_regmap_config[i];
  24. rc.lock_arg = &dev->regmap_mutex;
  25. dev->regmap[i] = devm_regmap_init_i2c(i2c, &rc);
  26. if (IS_ERR(dev->regmap[i])) {
  27. ret = PTR_ERR(dev->regmap[i]);
  28. dev_err(&i2c->dev,
  29. "Failed to initialize regmap%i: %d\n",
  30. ksz9477_regmap_config[i].val_bits, ret);
  31. return ret;
  32. }
  33. }
  34. if (i2c->dev.platform_data)
  35. dev->pdata = i2c->dev.platform_data;
  36. ret = ksz_switch_register(dev);
  37. /* Main DSA driver may not be started yet. */
  38. if (ret)
  39. return ret;
  40. i2c_set_clientdata(i2c, dev);
  41. return 0;
  42. }
  43. static void ksz9477_i2c_remove(struct i2c_client *i2c)
  44. {
  45. struct ksz_device *dev = i2c_get_clientdata(i2c);
  46. if (dev)
  47. ksz_switch_remove(dev);
  48. }
  49. static void ksz9477_i2c_shutdown(struct i2c_client *i2c)
  50. {
  51. struct ksz_device *dev = i2c_get_clientdata(i2c);
  52. if (!dev)
  53. return;
  54. if (dev->dev_ops->reset)
  55. dev->dev_ops->reset(dev);
  56. dsa_switch_shutdown(dev->ds);
  57. i2c_set_clientdata(i2c, NULL);
  58. }
  59. static const struct i2c_device_id ksz9477_i2c_id[] = {
  60. { "ksz9477-switch", 0 },
  61. {},
  62. };
  63. MODULE_DEVICE_TABLE(i2c, ksz9477_i2c_id);
  64. static const struct of_device_id ksz9477_dt_ids[] = {
  65. {
  66. .compatible = "microchip,ksz9477",
  67. .data = &ksz_switch_chips[KSZ9477]
  68. },
  69. {
  70. .compatible = "microchip,ksz9896",
  71. .data = &ksz_switch_chips[KSZ9896]
  72. },
  73. {
  74. .compatible = "microchip,ksz9897",
  75. .data = &ksz_switch_chips[KSZ9897]
  76. },
  77. {
  78. .compatible = "microchip,ksz9893",
  79. .data = &ksz_switch_chips[KSZ9893]
  80. },
  81. {
  82. .compatible = "microchip,ksz9563",
  83. .data = &ksz_switch_chips[KSZ9893]
  84. },
  85. {
  86. .compatible = "microchip,ksz8563",
  87. .data = &ksz_switch_chips[KSZ8563]
  88. },
  89. {
  90. .compatible = "microchip,ksz9567",
  91. .data = &ksz_switch_chips[KSZ9567]
  92. },
  93. {},
  94. };
  95. MODULE_DEVICE_TABLE(of, ksz9477_dt_ids);
  96. static struct i2c_driver ksz9477_i2c_driver = {
  97. .driver = {
  98. .name = "ksz9477-switch",
  99. .of_match_table = of_match_ptr(ksz9477_dt_ids),
  100. },
  101. .probe = ksz9477_i2c_probe,
  102. .remove = ksz9477_i2c_remove,
  103. .shutdown = ksz9477_i2c_shutdown,
  104. .id_table = ksz9477_i2c_id,
  105. };
  106. module_i2c_driver(ksz9477_i2c_driver);
  107. MODULE_AUTHOR("Tristram Ha <[email protected]>");
  108. MODULE_DESCRIPTION("Microchip KSZ9477 Series Switch I2C access Driver");
  109. MODULE_LICENSE("GPL v2");