tps65912-core.c 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. /*
  2. * tps65912-core.c -- TI TPS65912x
  3. *
  4. * Copyright 2011 Texas Instruments Inc.
  5. *
  6. * Author: Margarita Olaya Cabrera <magi@slimlogic.co.uk>
  7. *
  8. * This program is free software; you can redistribute it and/or modify it
  9. * under the terms of the GNU General Public License as published by the
  10. * Free Software Foundation; either version 2 of the License, or (at your
  11. * option) any later version.
  12. *
  13. * This driver is based on wm8350 implementation.
  14. */
  15. #include <linux/module.h>
  16. #include <linux/moduleparam.h>
  17. #include <linux/slab.h>
  18. #include <linux/gpio.h>
  19. #include <linux/mfd/core.h>
  20. #include <linux/mfd/tps65912.h>
  21. static const struct mfd_cell tps65912s[] = {
  22. {
  23. .name = "tps65912-pmic",
  24. },
  25. };
  26. int tps65912_set_bits(struct tps65912 *tps65912, u8 reg, u8 mask)
  27. {
  28. u8 data;
  29. int err;
  30. mutex_lock(&tps65912->io_mutex);
  31. err = tps65912->read(tps65912, reg, 1, &data);
  32. if (err) {
  33. dev_err(tps65912->dev, "Read from reg 0x%x failed\n", reg);
  34. goto out;
  35. }
  36. data |= mask;
  37. err = tps65912->write(tps65912, reg, 1, &data);
  38. if (err)
  39. dev_err(tps65912->dev, "Write to reg 0x%x failed\n", reg);
  40. out:
  41. mutex_unlock(&tps65912->io_mutex);
  42. return err;
  43. }
  44. EXPORT_SYMBOL_GPL(tps65912_set_bits);
  45. int tps65912_clear_bits(struct tps65912 *tps65912, u8 reg, u8 mask)
  46. {
  47. u8 data;
  48. int err;
  49. mutex_lock(&tps65912->io_mutex);
  50. err = tps65912->read(tps65912, reg, 1, &data);
  51. if (err) {
  52. dev_err(tps65912->dev, "Read from reg 0x%x failed\n", reg);
  53. goto out;
  54. }
  55. data &= ~mask;
  56. err = tps65912->write(tps65912, reg, 1, &data);
  57. if (err)
  58. dev_err(tps65912->dev, "Write to reg 0x%x failed\n", reg);
  59. out:
  60. mutex_unlock(&tps65912->io_mutex);
  61. return err;
  62. }
  63. EXPORT_SYMBOL_GPL(tps65912_clear_bits);
  64. static inline int tps65912_read(struct tps65912 *tps65912, u8 reg)
  65. {
  66. u8 val;
  67. int err;
  68. err = tps65912->read(tps65912, reg, 1, &val);
  69. if (err < 0)
  70. return err;
  71. return val;
  72. }
  73. static inline int tps65912_write(struct tps65912 *tps65912, u8 reg, u8 val)
  74. {
  75. return tps65912->write(tps65912, reg, 1, &val);
  76. }
  77. int tps65912_reg_read(struct tps65912 *tps65912, u8 reg)
  78. {
  79. int data;
  80. mutex_lock(&tps65912->io_mutex);
  81. data = tps65912_read(tps65912, reg);
  82. if (data < 0)
  83. dev_err(tps65912->dev, "Read from reg 0x%x failed\n", reg);
  84. mutex_unlock(&tps65912->io_mutex);
  85. return data;
  86. }
  87. EXPORT_SYMBOL_GPL(tps65912_reg_read);
  88. int tps65912_reg_write(struct tps65912 *tps65912, u8 reg, u8 val)
  89. {
  90. int err;
  91. mutex_lock(&tps65912->io_mutex);
  92. err = tps65912_write(tps65912, reg, val);
  93. if (err < 0)
  94. dev_err(tps65912->dev, "Write for reg 0x%x failed\n", reg);
  95. mutex_unlock(&tps65912->io_mutex);
  96. return err;
  97. }
  98. EXPORT_SYMBOL_GPL(tps65912_reg_write);
  99. int tps65912_device_init(struct tps65912 *tps65912)
  100. {
  101. struct tps65912_board *pmic_plat_data = dev_get_platdata(tps65912->dev);
  102. struct tps65912_platform_data *init_data;
  103. int ret, dcdc_avs, value;
  104. init_data = kzalloc(sizeof(struct tps65912_platform_data), GFP_KERNEL);
  105. if (init_data == NULL)
  106. return -ENOMEM;
  107. mutex_init(&tps65912->io_mutex);
  108. dev_set_drvdata(tps65912->dev, tps65912);
  109. dcdc_avs = (pmic_plat_data->is_dcdc1_avs << 0 |
  110. pmic_plat_data->is_dcdc2_avs << 1 |
  111. pmic_plat_data->is_dcdc3_avs << 2 |
  112. pmic_plat_data->is_dcdc4_avs << 3);
  113. if (dcdc_avs) {
  114. tps65912->read(tps65912, TPS65912_I2C_SPI_CFG, 1, &value);
  115. dcdc_avs |= value;
  116. tps65912->write(tps65912, TPS65912_I2C_SPI_CFG, 1, &dcdc_avs);
  117. }
  118. ret = mfd_add_devices(tps65912->dev, -1,
  119. tps65912s, ARRAY_SIZE(tps65912s),
  120. NULL, 0, NULL);
  121. if (ret < 0)
  122. goto err;
  123. init_data->irq = pmic_plat_data->irq;
  124. init_data->irq_base = pmic_plat_data->irq_base;
  125. ret = tps65912_irq_init(tps65912, init_data->irq, init_data);
  126. if (ret < 0)
  127. goto err;
  128. kfree(init_data);
  129. return ret;
  130. err:
  131. kfree(init_data);
  132. mfd_remove_devices(tps65912->dev);
  133. return ret;
  134. }
  135. void tps65912_device_exit(struct tps65912 *tps65912)
  136. {
  137. mfd_remove_devices(tps65912->dev);
  138. tps65912_irq_exit(tps65912);
  139. }
  140. MODULE_AUTHOR("Margarita Olaya <magi@slimlogic.co.uk>");
  141. MODULE_DESCRIPTION("TPS65912x chip family multi-function driver");
  142. MODULE_LICENSE("GPL");