fsl_pmic.c 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  1. /*
  2. * (C) Copyright 2008-2009 Freescale Semiconductor, Inc.
  3. *
  4. * See file CREDITS for list of people who contributed to this
  5. * project.
  6. *
  7. * This program is free software; you can redistribute it and/or
  8. * modify it under the terms of the GNU General Public License as
  9. * published by the Free Software Foundation; either version 2 of
  10. * the License, or (at your option) any later version.
  11. *
  12. * This program is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. * GNU General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU General Public License
  18. * along with this program; if not, write to the Free Software
  19. * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
  20. * MA 02111-1307 USA
  21. */
  22. #include <config.h>
  23. #include <common.h>
  24. #include <spi.h>
  25. #include <asm/errno.h>
  26. #include <linux/types.h>
  27. #include <fsl_pmic.h>
  28. static struct spi_slave *slave;
  29. struct spi_slave *pmic_spi_probe(void)
  30. {
  31. return spi_setup_slave(CONFIG_FSL_PMIC_BUS,
  32. CONFIG_FSL_PMIC_CS,
  33. CONFIG_FSL_PMIC_CLK,
  34. CONFIG_FSL_PMIC_MODE);
  35. }
  36. void pmic_spi_free(struct spi_slave *slave)
  37. {
  38. if (slave)
  39. spi_free_slave(slave);
  40. }
  41. u32 pmic_reg(u32 reg, u32 val, u32 write)
  42. {
  43. u32 pmic_tx, pmic_rx;
  44. if (!slave) {
  45. slave = pmic_spi_probe();
  46. if (!slave)
  47. return -1;
  48. }
  49. if (reg > 63 || write > 1) {
  50. printf("<reg num> = %d is invalid. Should be less then 63\n",
  51. reg);
  52. return -1;
  53. }
  54. if (spi_claim_bus(slave))
  55. return -1;
  56. pmic_tx = (write << 31) | (reg << 25) | (val & 0x00FFFFFF);
  57. if (spi_xfer(slave, 4 << 3, &pmic_tx, &pmic_rx,
  58. SPI_XFER_BEGIN | SPI_XFER_END)) {
  59. spi_release_bus(slave);
  60. return -1;
  61. }
  62. if (write) {
  63. pmic_tx &= ~(1 << 31);
  64. if (spi_xfer(slave, 4 << 3, &pmic_tx, &pmic_rx,
  65. SPI_XFER_BEGIN | SPI_XFER_END)) {
  66. spi_release_bus(slave);
  67. return -1;
  68. }
  69. }
  70. spi_release_bus(slave);
  71. return pmic_rx;
  72. }
  73. void pmic_reg_write(u32 reg, u32 value)
  74. {
  75. pmic_reg(reg, value, 1);
  76. }
  77. u32 pmic_reg_read(u32 reg)
  78. {
  79. return pmic_reg(reg, 0, 0);
  80. }
  81. void pmic_show_pmic_info(void)
  82. {
  83. u32 rev_id;
  84. rev_id = pmic_reg_read(REG_IDENTIFICATION);
  85. printf("PMIC ID: 0x%08x [Rev: ", rev_id);
  86. switch (rev_id & 0x1F) {
  87. case 0x1:
  88. puts("1.0");
  89. break;
  90. case 0x9:
  91. puts("1.1");
  92. break;
  93. case 0xA:
  94. puts("1.2");
  95. break;
  96. case 0x10:
  97. puts("2.0");
  98. break;
  99. case 0x11:
  100. puts("2.1");
  101. break;
  102. case 0x18:
  103. puts("3.0");
  104. break;
  105. case 0x19:
  106. puts("3.1");
  107. break;
  108. case 0x1A:
  109. puts("3.2");
  110. break;
  111. case 0x2:
  112. puts("3.2A");
  113. break;
  114. case 0x1B:
  115. puts("3.3");
  116. break;
  117. case 0x1D:
  118. puts("3.5");
  119. break;
  120. default:
  121. puts("unknown");
  122. break;
  123. }
  124. puts("]\n");
  125. }
  126. static void pmic_dump(int numregs)
  127. {
  128. u32 val;
  129. int i;
  130. pmic_show_pmic_info();
  131. for (i = 0; i < numregs; i++) {
  132. val = pmic_reg_read(i);
  133. if (!(i % 8))
  134. printf ("\n0x%02x: ", i);
  135. printf("%08x ", val);
  136. }
  137. puts("\n");
  138. }
  139. int do_pmic(cmd_tbl_t *cmdtp, int flag, int argc, char *argv[])
  140. {
  141. char *cmd;
  142. int nregs;
  143. u32 val;
  144. /* at least two arguments please */
  145. if (argc < 2) {
  146. cmd_usage(cmdtp);
  147. return 1;
  148. }
  149. cmd = argv[1];
  150. if (strcmp(cmd, "dump") == 0) {
  151. if (argc < 3) {
  152. cmd_usage(cmdtp);
  153. return 1;
  154. }
  155. nregs = simple_strtoul(argv[2], NULL, 16);
  156. pmic_dump(nregs);
  157. return 0;
  158. }
  159. if (strcmp(cmd, "write") == 0) {
  160. if (argc < 4) {
  161. cmd_usage(cmdtp);
  162. return 1;
  163. }
  164. nregs = simple_strtoul(argv[2], NULL, 16);
  165. val = simple_strtoul(argv[3], NULL, 16);
  166. pmic_reg_write(nregs, val);
  167. return 0;
  168. }
  169. /* No subcommand found */
  170. return 1;
  171. }
  172. U_BOOT_CMD(
  173. pmic, CONFIG_SYS_MAXARGS, 1, do_pmic,
  174. "Freescale PMIC (Atlas)",
  175. "dump [numregs] dump registers\n"
  176. "pmic write <reg> <value> - write register"
  177. );