p2wi.c 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Sunxi A31 Power Management Unit
  4. *
  5. * (C) Copyright 2013 Oliver Schinagl <oliver@schinagl.nl>
  6. * http://linux-sunxi.org
  7. *
  8. * Based on sun6i sources and earlier U-Boot Allwiner A10 SPL work
  9. *
  10. * (C) Copyright 2006-2013
  11. * Allwinner Technology Co., Ltd. <www.allwinnertech.com>
  12. * Berg Xing <bergxing@allwinnertech.com>
  13. * Tom Cubie <tangliang@allwinnertech.com>
  14. */
  15. #include <common.h>
  16. #include <errno.h>
  17. #include <asm/io.h>
  18. #include <asm/arch/cpu.h>
  19. #include <asm/arch/gpio.h>
  20. #include <asm/arch/p2wi.h>
  21. #include <asm/arch/prcm.h>
  22. #include <asm/arch/clock.h>
  23. #include <asm/arch/sys_proto.h>
  24. void p2wi_init(void)
  25. {
  26. struct sunxi_p2wi_reg *p2wi = (struct sunxi_p2wi_reg *)SUN6I_P2WI_BASE;
  27. /* Enable p2wi and PIO clk, and de-assert their resets */
  28. prcm_apb0_enable(PRCM_APB0_GATE_PIO | PRCM_APB0_GATE_P2WI);
  29. sunxi_gpio_set_cfgpin(SUNXI_GPL(0), SUN6I_GPL0_R_P2WI_SCK);
  30. sunxi_gpio_set_cfgpin(SUNXI_GPL(1), SUN6I_GPL1_R_P2WI_SDA);
  31. /* Reset p2wi controller and set clock to CLKIN(12)/8 = 1.5 MHz */
  32. writel(P2WI_CTRL_RESET, &p2wi->ctrl);
  33. sdelay(0x100);
  34. writel(P2WI_CC_SDA_OUT_DELAY(1) | P2WI_CC_CLK_DIV(8),
  35. &p2wi->cc);
  36. }
  37. int p2wi_change_to_p2wi_mode(u8 slave_addr, u8 ctrl_reg, u8 init_data)
  38. {
  39. struct sunxi_p2wi_reg *p2wi = (struct sunxi_p2wi_reg *)SUN6I_P2WI_BASE;
  40. unsigned long tmo = timer_get_us() + 1000000;
  41. writel(P2WI_PM_DEV_ADDR(slave_addr) |
  42. P2WI_PM_CTRL_ADDR(ctrl_reg) |
  43. P2WI_PM_INIT_DATA(init_data) |
  44. P2WI_PM_INIT_SEND,
  45. &p2wi->pm);
  46. while ((readl(&p2wi->pm) & P2WI_PM_INIT_SEND)) {
  47. if (timer_get_us() > tmo)
  48. return -ETIME;
  49. }
  50. return 0;
  51. }
  52. static int p2wi_await_trans(void)
  53. {
  54. struct sunxi_p2wi_reg *p2wi = (struct sunxi_p2wi_reg *)SUN6I_P2WI_BASE;
  55. unsigned long tmo = timer_get_us() + 1000000;
  56. int ret;
  57. u8 reg;
  58. while (1) {
  59. reg = readl(&p2wi->status);
  60. if (reg & P2WI_STAT_TRANS_ERR) {
  61. ret = -EIO;
  62. break;
  63. }
  64. if (reg & P2WI_STAT_TRANS_DONE) {
  65. ret = 0;
  66. break;
  67. }
  68. if (timer_get_us() > tmo) {
  69. ret = -ETIME;
  70. break;
  71. }
  72. }
  73. writel(reg, &p2wi->status); /* Clear status bits */
  74. return ret;
  75. }
  76. int p2wi_read(const u8 addr, u8 *data)
  77. {
  78. struct sunxi_p2wi_reg *p2wi = (struct sunxi_p2wi_reg *)SUN6I_P2WI_BASE;
  79. int ret;
  80. writel(P2WI_DATADDR_BYTE_1(addr), &p2wi->dataddr0);
  81. writel(P2WI_DATA_NUM_BYTES(1) |
  82. P2WI_DATA_NUM_BYTES_READ, &p2wi->numbytes);
  83. writel(P2WI_STAT_TRANS_DONE, &p2wi->status);
  84. writel(P2WI_CTRL_TRANS_START, &p2wi->ctrl);
  85. ret = p2wi_await_trans();
  86. *data = readl(&p2wi->data0) & P2WI_DATA_BYTE_1_MASK;
  87. return ret;
  88. }
  89. int p2wi_write(const u8 addr, u8 data)
  90. {
  91. struct sunxi_p2wi_reg *p2wi = (struct sunxi_p2wi_reg *)SUN6I_P2WI_BASE;
  92. writel(P2WI_DATADDR_BYTE_1(addr), &p2wi->dataddr0);
  93. writel(P2WI_DATA_BYTE_1(data), &p2wi->data0);
  94. writel(P2WI_DATA_NUM_BYTES(1), &p2wi->numbytes);
  95. writel(P2WI_STAT_TRANS_DONE, &p2wi->status);
  96. writel(P2WI_CTRL_TRANS_START, &p2wi->ctrl);
  97. return p2wi_await_trans();
  98. }