pmic.c 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210
  1. /*
  2. * Copyright (C) 2014-2015 Samsung Electronics
  3. * Przemyslaw Marczak <p.marczak@samsung.com>
  4. *
  5. * SPDX-License-Identifier: GPL-2.0+
  6. */
  7. #include <common.h>
  8. #include <errno.h>
  9. #include <dm.h>
  10. #include <dm/uclass-internal.h>
  11. #include <power/pmic.h>
  12. #define LIMIT_DEV 32
  13. #define LIMIT_PARENT 20
  14. static struct udevice *currdev;
  15. static int failure(int ret)
  16. {
  17. printf("Error: %d (%s)\n", ret, errno_str(ret));
  18. return CMD_RET_FAILURE;
  19. }
  20. static int do_dev(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
  21. {
  22. char *name;
  23. int ret = -ENODEV;
  24. switch (argc) {
  25. case 2:
  26. name = argv[1];
  27. ret = pmic_get(name, &currdev);
  28. if (ret) {
  29. printf("Can't get PMIC: %s!\n", name);
  30. return failure(ret);
  31. }
  32. case 1:
  33. if (!currdev) {
  34. printf("PMIC device is not set!\n\n");
  35. return CMD_RET_USAGE;
  36. }
  37. printf("dev: %d @ %s\n", currdev->seq, currdev->name);
  38. }
  39. return CMD_RET_SUCCESS;
  40. }
  41. static int do_list(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
  42. {
  43. struct udevice *dev;
  44. int ret;
  45. printf("| %-*.*s| %-*.*s| %s @ %s\n",
  46. LIMIT_DEV, LIMIT_DEV, "Name",
  47. LIMIT_PARENT, LIMIT_PARENT, "Parent name",
  48. "Parent uclass", "seq");
  49. for (ret = uclass_first_device(UCLASS_PMIC, &dev); dev;
  50. ret = uclass_next_device(&dev)) {
  51. if (ret)
  52. continue;
  53. printf("| %-*.*s| %-*.*s| %s @ %d\n",
  54. LIMIT_DEV, LIMIT_DEV, dev->name,
  55. LIMIT_PARENT, LIMIT_PARENT, dev->parent->name,
  56. dev_get_uclass_name(dev->parent), dev->parent->seq);
  57. }
  58. if (ret)
  59. return CMD_RET_FAILURE;
  60. return CMD_RET_SUCCESS;
  61. }
  62. static int do_dump(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
  63. {
  64. struct udevice *dev;
  65. uint8_t value;
  66. uint reg;
  67. int ret;
  68. if (!currdev) {
  69. printf("First, set the PMIC device!\n");
  70. return CMD_RET_USAGE;
  71. }
  72. dev = currdev;
  73. printf("Dump pmic: %s registers\n", dev->name);
  74. for (reg = 0; reg < pmic_reg_count(dev); reg++) {
  75. ret = pmic_read(dev, reg, &value, 1);
  76. if (ret) {
  77. printf("Can't read register: %d\n", reg);
  78. return failure(ret);
  79. }
  80. if (!(reg % 16))
  81. printf("\n0x%02x: ", reg);
  82. printf("%2.2x ", value);
  83. }
  84. printf("\n");
  85. return CMD_RET_SUCCESS;
  86. }
  87. static int do_read(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
  88. {
  89. struct udevice *dev;
  90. int regs, ret;
  91. uint8_t value;
  92. uint reg;
  93. if (!currdev) {
  94. printf("First, set the PMIC device!\n");
  95. return CMD_RET_USAGE;
  96. }
  97. dev = currdev;
  98. if (argc != 2)
  99. return CMD_RET_USAGE;
  100. reg = simple_strtoul(argv[1], NULL, 0);
  101. regs = pmic_reg_count(dev);
  102. if (reg > regs) {
  103. printf("PMIC max reg: %d\n", regs);
  104. return failure(-EFAULT);
  105. }
  106. ret = pmic_read(dev, reg, &value, 1);
  107. if (ret) {
  108. printf("Can't read PMIC register: %d!\n", reg);
  109. return failure(ret);
  110. }
  111. printf("0x%02x: 0x%2.2x\n", reg, value);
  112. return CMD_RET_SUCCESS;
  113. }
  114. static int do_write(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
  115. {
  116. struct udevice *dev;
  117. int regs, ret;
  118. uint8_t value;
  119. uint reg;
  120. if (!currdev) {
  121. printf("First, set the PMIC device!\n");
  122. return CMD_RET_USAGE;
  123. }
  124. dev = currdev;
  125. if (argc != 3)
  126. return CMD_RET_USAGE;
  127. reg = simple_strtoul(argv[1], NULL, 0);
  128. regs = pmic_reg_count(dev);
  129. if (reg > regs) {
  130. printf("PMIC max reg: %d\n", regs);
  131. return failure(-EFAULT);
  132. }
  133. value = simple_strtoul(argv[2], NULL, 0);
  134. ret = pmic_write(dev, reg, &value, 1);
  135. if (ret) {
  136. printf("Can't write PMIC register: %d!\n", reg);
  137. return failure(ret);
  138. }
  139. return CMD_RET_SUCCESS;
  140. }
  141. static cmd_tbl_t subcmd[] = {
  142. U_BOOT_CMD_MKENT(dev, 2, 1, do_dev, "", ""),
  143. U_BOOT_CMD_MKENT(list, 1, 1, do_list, "", ""),
  144. U_BOOT_CMD_MKENT(dump, 1, 1, do_dump, "", ""),
  145. U_BOOT_CMD_MKENT(read, 2, 1, do_read, "", ""),
  146. U_BOOT_CMD_MKENT(write, 3, 1, do_write, "", ""),
  147. };
  148. static int do_pmic(cmd_tbl_t *cmdtp, int flag, int argc,
  149. char * const argv[])
  150. {
  151. cmd_tbl_t *cmd;
  152. argc--;
  153. argv++;
  154. cmd = find_cmd_tbl(argv[0], subcmd, ARRAY_SIZE(subcmd));
  155. if (cmd == NULL || argc > cmd->maxargs)
  156. return CMD_RET_USAGE;
  157. return cmd->cmd(cmdtp, flag, argc, argv);
  158. }
  159. U_BOOT_CMD(pmic, CONFIG_SYS_MAXARGS, 1, do_pmic,
  160. " operations",
  161. "list - list pmic devices\n"
  162. "pmic dev [name] - show or [set] operating PMIC device\n"
  163. "pmic dump - dump registers\n"
  164. "pmic read address - read byte of register at address\n"
  165. "pmic write address - write byte to register at address\n"
  166. );