lsdc_i2c.c 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Copyright (C) 2023 Loongson Technology Corporation Limited
  4. */
  5. #include <drm/drm_managed.h>
  6. #include <drm/drm_print.h>
  7. #include "lsdc_drv.h"
  8. #include "lsdc_output.h"
  9. /*
  10. * __lsdc_gpio_i2c_set - set the state of a gpio pin indicated by mask
  11. * @mask: gpio pin mask
  12. * @state: "0" for low, "1" for high
  13. */
  14. static void __lsdc_gpio_i2c_set(struct lsdc_i2c * const li2c, int mask, int state)
  15. {
  16. struct lsdc_device *ldev = to_lsdc(li2c->ddev);
  17. unsigned long flags;
  18. u8 val;
  19. spin_lock_irqsave(&ldev->reglock, flags);
  20. if (state) {
  21. /*
  22. * Setting this pin as input directly, write 1 for input.
  23. * The external pull-up resistor will pull the level up
  24. */
  25. val = readb(li2c->dir_reg);
  26. val |= mask;
  27. writeb(val, li2c->dir_reg);
  28. } else {
  29. /* First set this pin as output, write 0 for output */
  30. val = readb(li2c->dir_reg);
  31. val &= ~mask;
  32. writeb(val, li2c->dir_reg);
  33. /* Then, make this pin output 0 */
  34. val = readb(li2c->dat_reg);
  35. val &= ~mask;
  36. writeb(val, li2c->dat_reg);
  37. }
  38. spin_unlock_irqrestore(&ldev->reglock, flags);
  39. }
  40. /*
  41. * __lsdc_gpio_i2c_get - read value back from the gpio pin indicated by mask
  42. * @mask: gpio pin mask
  43. * return "0" for low, "1" for high
  44. */
  45. static int __lsdc_gpio_i2c_get(struct lsdc_i2c * const li2c, int mask)
  46. {
  47. struct lsdc_device *ldev = to_lsdc(li2c->ddev);
  48. unsigned long flags;
  49. u8 val;
  50. spin_lock_irqsave(&ldev->reglock, flags);
  51. /* First set this pin as input */
  52. val = readb(li2c->dir_reg);
  53. val |= mask;
  54. writeb(val, li2c->dir_reg);
  55. /* Then get level state from this pin */
  56. val = readb(li2c->dat_reg);
  57. spin_unlock_irqrestore(&ldev->reglock, flags);
  58. return (val & mask) ? 1 : 0;
  59. }
  60. static void lsdc_gpio_i2c_set_sda(void *i2c, int state)
  61. {
  62. struct lsdc_i2c * const li2c = (struct lsdc_i2c *)i2c;
  63. /* set state on the li2c->sda pin */
  64. return __lsdc_gpio_i2c_set(li2c, li2c->sda, state);
  65. }
  66. static void lsdc_gpio_i2c_set_scl(void *i2c, int state)
  67. {
  68. struct lsdc_i2c * const li2c = (struct lsdc_i2c *)i2c;
  69. /* set state on the li2c->scl pin */
  70. return __lsdc_gpio_i2c_set(li2c, li2c->scl, state);
  71. }
  72. static int lsdc_gpio_i2c_get_sda(void *i2c)
  73. {
  74. struct lsdc_i2c * const li2c = (struct lsdc_i2c *)i2c;
  75. /* read value from the li2c->sda pin */
  76. return __lsdc_gpio_i2c_get(li2c, li2c->sda);
  77. }
  78. static int lsdc_gpio_i2c_get_scl(void *i2c)
  79. {
  80. struct lsdc_i2c * const li2c = (struct lsdc_i2c *)i2c;
  81. /* read the value from the li2c->scl pin */
  82. return __lsdc_gpio_i2c_get(li2c, li2c->scl);
  83. }
  84. static void lsdc_destroy_i2c(struct drm_device *ddev, void *data)
  85. {
  86. struct lsdc_i2c *li2c = (struct lsdc_i2c *)data;
  87. if (li2c) {
  88. i2c_del_adapter(&li2c->adapter);
  89. kfree(li2c);
  90. }
  91. }
  92. /*
  93. * The DC in ls7a1000/ls7a2000/ls2k2000 has builtin gpio hardware
  94. *
  95. * @reg_base: gpio reg base
  96. * @index: output channel index, 0 for PIPE0, 1 for PIPE1
  97. */
  98. int lsdc_create_i2c_chan(struct drm_device *ddev,
  99. struct lsdc_display_pipe *dispipe,
  100. unsigned int index)
  101. {
  102. struct lsdc_device *ldev = to_lsdc(ddev);
  103. struct i2c_adapter *adapter;
  104. struct lsdc_i2c *li2c;
  105. int ret;
  106. li2c = kzalloc_obj(*li2c);
  107. if (!li2c)
  108. return -ENOMEM;
  109. dispipe->li2c = li2c;
  110. if (index == 0) {
  111. li2c->sda = 0x01; /* pin 0 */
  112. li2c->scl = 0x02; /* pin 1 */
  113. } else if (index == 1) {
  114. li2c->sda = 0x04; /* pin 2 */
  115. li2c->scl = 0x08; /* pin 3 */
  116. } else {
  117. return -ENOENT;
  118. }
  119. li2c->ddev = ddev;
  120. li2c->dir_reg = ldev->reg_base + LS7A_DC_GPIO_DIR_REG;
  121. li2c->dat_reg = ldev->reg_base + LS7A_DC_GPIO_DAT_REG;
  122. li2c->bit.setsda = lsdc_gpio_i2c_set_sda;
  123. li2c->bit.setscl = lsdc_gpio_i2c_set_scl;
  124. li2c->bit.getsda = lsdc_gpio_i2c_get_sda;
  125. li2c->bit.getscl = lsdc_gpio_i2c_get_scl;
  126. li2c->bit.udelay = 5;
  127. li2c->bit.timeout = usecs_to_jiffies(2200);
  128. li2c->bit.data = li2c;
  129. adapter = &li2c->adapter;
  130. adapter->algo_data = &li2c->bit;
  131. adapter->owner = THIS_MODULE;
  132. adapter->dev.parent = ddev->dev;
  133. adapter->nr = -1;
  134. snprintf(adapter->name, sizeof(adapter->name), "lsdc-i2c%u", index);
  135. i2c_set_adapdata(adapter, li2c);
  136. ret = i2c_bit_add_bus(adapter);
  137. if (ret) {
  138. kfree(li2c);
  139. return ret;
  140. }
  141. ret = drmm_add_action_or_reset(ddev, lsdc_destroy_i2c, li2c);
  142. if (ret)
  143. return ret;
  144. drm_info(ddev, "%s(sda pin mask=%u, scl pin mask=%u) created\n",
  145. adapter->name, li2c->sda, li2c->scl);
  146. return 0;
  147. }