cpld.c 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * ULCB board CPLD access support
  4. *
  5. * Copyright (C) 2017 Renesas Electronics Corporation
  6. * Copyright (C) 2017 Cogent Embedded, Inc.
  7. */
  8. #include <common.h>
  9. #include <asm/gpio.h>
  10. #include <asm/io.h>
  11. #include <dm.h>
  12. #include <errno.h>
  13. #include <linux/err.h>
  14. #include <sysreset.h>
  15. #define CPLD_ADDR_MODE 0x00 /* RW */
  16. #define CPLD_ADDR_MUX 0x02 /* RW */
  17. #define CPLD_ADDR_DIPSW6 0x08 /* R */
  18. #define CPLD_ADDR_RESET 0x80 /* RW */
  19. #define CPLD_ADDR_VERSION 0xFF /* R */
  20. struct renesas_ulcb_sysreset_priv {
  21. struct gpio_desc miso;
  22. struct gpio_desc mosi;
  23. struct gpio_desc sck;
  24. struct gpio_desc sstbz;
  25. };
  26. static u32 cpld_read(struct udevice *dev, u8 addr)
  27. {
  28. struct renesas_ulcb_sysreset_priv *priv = dev_get_priv(dev);
  29. u32 data = 0;
  30. int i;
  31. for (i = 0; i < 8; i++) {
  32. dm_gpio_set_value(&priv->mosi, !!(addr & 0x80)); /* MSB first */
  33. dm_gpio_set_value(&priv->sck, 1);
  34. addr <<= 1;
  35. dm_gpio_set_value(&priv->sck, 0);
  36. }
  37. dm_gpio_set_value(&priv->mosi, 0); /* READ */
  38. dm_gpio_set_value(&priv->sstbz, 0);
  39. dm_gpio_set_value(&priv->sck, 1);
  40. dm_gpio_set_value(&priv->sck, 0);
  41. dm_gpio_set_value(&priv->sstbz, 1);
  42. for (i = 0; i < 32; i++) {
  43. dm_gpio_set_value(&priv->sck, 1);
  44. data <<= 1;
  45. data |= dm_gpio_get_value(&priv->miso); /* MSB first */
  46. dm_gpio_set_value(&priv->sck, 0);
  47. }
  48. return data;
  49. }
  50. static void cpld_write(struct udevice *dev, u8 addr, u32 data)
  51. {
  52. struct renesas_ulcb_sysreset_priv *priv = dev_get_priv(dev);
  53. int i;
  54. for (i = 0; i < 32; i++) {
  55. dm_gpio_set_value(&priv->mosi, data & (1 << 31)); /* MSB first */
  56. dm_gpio_set_value(&priv->sck, 1);
  57. data <<= 1;
  58. dm_gpio_set_value(&priv->sck, 0);
  59. }
  60. for (i = 0; i < 8; i++) {
  61. dm_gpio_set_value(&priv->mosi, addr & 0x80); /* MSB first */
  62. dm_gpio_set_value(&priv->sck, 1);
  63. addr <<= 1;
  64. dm_gpio_set_value(&priv->sck, 0);
  65. }
  66. dm_gpio_set_value(&priv->mosi, 1); /* WRITE */
  67. dm_gpio_set_value(&priv->sstbz, 0);
  68. dm_gpio_set_value(&priv->sck, 1);
  69. dm_gpio_set_value(&priv->sck, 0);
  70. dm_gpio_set_value(&priv->sstbz, 1);
  71. }
  72. static int do_cpld(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
  73. {
  74. struct udevice *dev;
  75. u32 addr, val;
  76. int ret;
  77. ret = uclass_get_device_by_driver(UCLASS_SYSRESET,
  78. DM_GET_DRIVER(sysreset_renesas_ulcb),
  79. &dev);
  80. if (ret)
  81. return ret;
  82. if (argc == 2 && strcmp(argv[1], "info") == 0) {
  83. printf("CPLD version:\t\t\t0x%08x\n",
  84. cpld_read(dev, CPLD_ADDR_VERSION));
  85. printf("H3 Mode setting (MD0..28):\t0x%08x\n",
  86. cpld_read(dev, CPLD_ADDR_MODE));
  87. printf("Multiplexer settings:\t\t0x%08x\n",
  88. cpld_read(dev, CPLD_ADDR_MUX));
  89. printf("DIPSW (SW6):\t\t\t0x%08x\n",
  90. cpld_read(dev, CPLD_ADDR_DIPSW6));
  91. return 0;
  92. }
  93. if (argc < 3)
  94. return CMD_RET_USAGE;
  95. addr = simple_strtoul(argv[2], NULL, 16);
  96. if (!(addr == CPLD_ADDR_VERSION || addr == CPLD_ADDR_MODE ||
  97. addr == CPLD_ADDR_MUX || addr == CPLD_ADDR_DIPSW6 ||
  98. addr == CPLD_ADDR_RESET)) {
  99. printf("Invalid CPLD register address\n");
  100. return CMD_RET_USAGE;
  101. }
  102. if (argc == 3 && strcmp(argv[1], "read") == 0) {
  103. printf("0x%x\n", cpld_read(dev, addr));
  104. } else if (argc == 4 && strcmp(argv[1], "write") == 0) {
  105. val = simple_strtoul(argv[3], NULL, 16);
  106. cpld_write(dev, addr, val);
  107. }
  108. return 0;
  109. }
  110. U_BOOT_CMD(
  111. cpld, 4, 1, do_cpld,
  112. "CPLD access",
  113. "info\n"
  114. "cpld read addr\n"
  115. "cpld write addr val\n"
  116. );
  117. static int renesas_ulcb_sysreset_request(struct udevice *dev, enum sysreset_t type)
  118. {
  119. cpld_write(dev, CPLD_ADDR_RESET, 1);
  120. return -EINPROGRESS;
  121. }
  122. static int renesas_ulcb_sysreset_probe(struct udevice *dev)
  123. {
  124. struct renesas_ulcb_sysreset_priv *priv = dev_get_priv(dev);
  125. if (gpio_request_by_name(dev, "gpio-miso", 0, &priv->miso,
  126. GPIOD_IS_IN))
  127. return -EINVAL;
  128. if (gpio_request_by_name(dev, "gpio-sck", 0, &priv->sck,
  129. GPIOD_IS_OUT))
  130. return -EINVAL;
  131. if (gpio_request_by_name(dev, "gpio-sstbz", 0, &priv->sstbz,
  132. GPIOD_IS_OUT | GPIOD_IS_OUT_ACTIVE))
  133. return -EINVAL;
  134. if (gpio_request_by_name(dev, "gpio-mosi", 0, &priv->mosi,
  135. GPIOD_IS_OUT))
  136. return -EINVAL;
  137. /* PULL-UP on MISO line */
  138. setbits_le32(PFC_PUEN5, PUEN_SSI_SDATA4);
  139. /* Dummy read */
  140. cpld_read(dev, CPLD_ADDR_VERSION);
  141. return 0;
  142. }
  143. static struct sysreset_ops renesas_ulcb_sysreset = {
  144. .request = renesas_ulcb_sysreset_request,
  145. };
  146. static const struct udevice_id renesas_ulcb_sysreset_ids[] = {
  147. { .compatible = "renesas,ulcb-cpld" },
  148. { }
  149. };
  150. U_BOOT_DRIVER(sysreset_renesas_ulcb) = {
  151. .name = "renesas_ulcb_sysreset",
  152. .id = UCLASS_SYSRESET,
  153. .ops = &renesas_ulcb_sysreset,
  154. .probe = renesas_ulcb_sysreset_probe,
  155. .of_match = renesas_ulcb_sysreset_ids,
  156. .priv_auto_alloc_size = sizeof(struct renesas_ulcb_sysreset_priv),
  157. };