xrs700x_i2c.c 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Copyright (C) 2020 NovaTech LLC
  4. * George McCollister <[email protected]>
  5. */
  6. #include <linux/bits.h>
  7. #include <linux/i2c.h>
  8. #include <linux/module.h>
  9. #include "xrs700x.h"
  10. #include "xrs700x_reg.h"
  11. struct xrs700x_i2c_cmd {
  12. __be32 reg;
  13. __be16 val;
  14. } __packed;
  15. static int xrs700x_i2c_reg_read(void *context, unsigned int reg,
  16. unsigned int *val)
  17. {
  18. struct device *dev = context;
  19. struct i2c_client *i2c = to_i2c_client(dev);
  20. struct xrs700x_i2c_cmd cmd;
  21. int ret;
  22. cmd.reg = cpu_to_be32(reg | 1);
  23. ret = i2c_master_send(i2c, (char *)&cmd.reg, sizeof(cmd.reg));
  24. if (ret < 0) {
  25. dev_err(dev, "xrs i2c_master_send returned %d\n", ret);
  26. return ret;
  27. }
  28. ret = i2c_master_recv(i2c, (char *)&cmd.val, sizeof(cmd.val));
  29. if (ret < 0) {
  30. dev_err(dev, "xrs i2c_master_recv returned %d\n", ret);
  31. return ret;
  32. }
  33. *val = be16_to_cpu(cmd.val);
  34. return 0;
  35. }
  36. static int xrs700x_i2c_reg_write(void *context, unsigned int reg,
  37. unsigned int val)
  38. {
  39. struct device *dev = context;
  40. struct i2c_client *i2c = to_i2c_client(dev);
  41. struct xrs700x_i2c_cmd cmd;
  42. int ret;
  43. cmd.reg = cpu_to_be32(reg);
  44. cmd.val = cpu_to_be16(val);
  45. ret = i2c_master_send(i2c, (char *)&cmd, sizeof(cmd));
  46. if (ret < 0) {
  47. dev_err(dev, "xrs i2c_master_send returned %d\n", ret);
  48. return ret;
  49. }
  50. return 0;
  51. }
  52. static const struct regmap_config xrs700x_i2c_regmap_config = {
  53. .val_bits = 16,
  54. .reg_stride = 2,
  55. .reg_bits = 32,
  56. .pad_bits = 0,
  57. .write_flag_mask = 0,
  58. .read_flag_mask = 0,
  59. .reg_read = xrs700x_i2c_reg_read,
  60. .reg_write = xrs700x_i2c_reg_write,
  61. .max_register = 0,
  62. .cache_type = REGCACHE_NONE,
  63. .reg_format_endian = REGMAP_ENDIAN_BIG,
  64. .val_format_endian = REGMAP_ENDIAN_BIG
  65. };
  66. static int xrs700x_i2c_probe(struct i2c_client *i2c,
  67. const struct i2c_device_id *i2c_id)
  68. {
  69. struct xrs700x *priv;
  70. int ret;
  71. priv = xrs700x_switch_alloc(&i2c->dev, i2c);
  72. if (!priv)
  73. return -ENOMEM;
  74. priv->regmap = devm_regmap_init(&i2c->dev, NULL, &i2c->dev,
  75. &xrs700x_i2c_regmap_config);
  76. if (IS_ERR(priv->regmap)) {
  77. ret = PTR_ERR(priv->regmap);
  78. dev_err(&i2c->dev, "Failed to initialize regmap: %d\n", ret);
  79. return ret;
  80. }
  81. i2c_set_clientdata(i2c, priv);
  82. ret = xrs700x_switch_register(priv);
  83. /* Main DSA driver may not be started yet. */
  84. if (ret)
  85. return ret;
  86. return 0;
  87. }
  88. static void xrs700x_i2c_remove(struct i2c_client *i2c)
  89. {
  90. struct xrs700x *priv = i2c_get_clientdata(i2c);
  91. if (!priv)
  92. return;
  93. xrs700x_switch_remove(priv);
  94. }
  95. static void xrs700x_i2c_shutdown(struct i2c_client *i2c)
  96. {
  97. struct xrs700x *priv = i2c_get_clientdata(i2c);
  98. if (!priv)
  99. return;
  100. xrs700x_switch_shutdown(priv);
  101. i2c_set_clientdata(i2c, NULL);
  102. }
  103. static const struct i2c_device_id xrs700x_i2c_id[] = {
  104. { "xrs700x-switch", 0 },
  105. {},
  106. };
  107. MODULE_DEVICE_TABLE(i2c, xrs700x_i2c_id);
  108. static const struct of_device_id __maybe_unused xrs700x_i2c_dt_ids[] = {
  109. { .compatible = "arrow,xrs7003e", .data = &xrs7003e_info },
  110. { .compatible = "arrow,xrs7003f", .data = &xrs7003f_info },
  111. { .compatible = "arrow,xrs7004e", .data = &xrs7004e_info },
  112. { .compatible = "arrow,xrs7004f", .data = &xrs7004f_info },
  113. {},
  114. };
  115. MODULE_DEVICE_TABLE(of, xrs700x_i2c_dt_ids);
  116. static struct i2c_driver xrs700x_i2c_driver = {
  117. .driver = {
  118. .name = "xrs700x-i2c",
  119. .of_match_table = of_match_ptr(xrs700x_i2c_dt_ids),
  120. },
  121. .probe = xrs700x_i2c_probe,
  122. .remove = xrs700x_i2c_remove,
  123. .shutdown = xrs700x_i2c_shutdown,
  124. .id_table = xrs700x_i2c_id,
  125. };
  126. module_i2c_driver(xrs700x_i2c_driver);
  127. MODULE_AUTHOR("George McCollister <[email protected]>");
  128. MODULE_DESCRIPTION("Arrow SpeedChips XRS700x DSA I2C driver");
  129. MODULE_LICENSE("GPL v2");