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. u32 tmp;
  45. if (!slave) {
  46. slave = pmic_spi_probe();
  47. if (!slave)
  48. return -1;
  49. }
  50. if (reg > 63 || write > 1) {
  51. printf("<reg num> = %d is invalid. Should be less then 63\n",
  52. reg);
  53. return -1;
  54. }
  55. if (spi_claim_bus(slave))
  56. return -1;
  57. pmic_tx = (write << 31) | (reg << 25) | (val & 0x00FFFFFF);
  58. tmp = cpu_to_be32(pmic_tx);
  59. if (spi_xfer(slave, 4 << 3, &tmp, &pmic_rx,
  60. SPI_XFER_BEGIN | SPI_XFER_END)) {
  61. spi_release_bus(slave);
  62. return -1;
  63. }
  64. if (write) {
  65. pmic_tx &= ~(1 << 31);
  66. tmp = cpu_to_be32(pmic_tx);
  67. if (spi_xfer(slave, 4 << 3, &tmp, &pmic_rx,
  68. SPI_XFER_BEGIN | SPI_XFER_END)) {
  69. spi_release_bus(slave);
  70. return -1;
  71. }
  72. }
  73. spi_release_bus(slave);
  74. return cpu_to_be32(pmic_rx);
  75. }
  76. void pmic_reg_write(u32 reg, u32 value)
  77. {
  78. pmic_reg(reg, value, 1);
  79. }
  80. u32 pmic_reg_read(u32 reg)
  81. {
  82. return pmic_reg(reg, 0, 0);
  83. }
  84. void pmic_show_pmic_info(void)
  85. {
  86. u32 rev_id;
  87. rev_id = pmic_reg_read(REG_IDENTIFICATION);
  88. printf("PMIC ID: 0x%08x [Rev: ", rev_id);
  89. switch (rev_id & 0x1F) {
  90. case 0x1:
  91. puts("1.0");
  92. break;
  93. case 0x9:
  94. puts("1.1");
  95. break;
  96. case 0xA:
  97. puts("1.2");
  98. break;
  99. case 0x10:
  100. puts("2.0");
  101. break;
  102. case 0x11:
  103. puts("2.1");
  104. break;
  105. case 0x18:
  106. puts("3.0");
  107. break;
  108. case 0x19:
  109. puts("3.1");
  110. break;
  111. case 0x1A:
  112. puts("3.2");
  113. break;
  114. case 0x2:
  115. puts("3.2A");
  116. break;
  117. case 0x1B:
  118. puts("3.3");
  119. break;
  120. case 0x1D:
  121. puts("3.5");
  122. break;
  123. default:
  124. puts("unknown");
  125. break;
  126. }
  127. puts("]\n");
  128. }
  129. static void pmic_dump(int numregs)
  130. {
  131. u32 val;
  132. int i;
  133. pmic_show_pmic_info();
  134. for (i = 0; i < numregs; i++) {
  135. val = pmic_reg_read(i);
  136. if (!(i % 8))
  137. printf ("\n0x%02x: ", i);
  138. printf("%08x ", val);
  139. }
  140. puts("\n");
  141. }
  142. int do_pmic(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
  143. {
  144. char *cmd;
  145. int nregs;
  146. u32 val;
  147. /* at least two arguments please */
  148. if (argc < 2)
  149. return cmd_usage(cmdtp);
  150. cmd = argv[1];
  151. if (strcmp(cmd, "dump") == 0) {
  152. if (argc < 3)
  153. return cmd_usage(cmdtp);
  154. nregs = simple_strtoul(argv[2], NULL, 16);
  155. pmic_dump(nregs);
  156. return 0;
  157. }
  158. if (strcmp(cmd, "write") == 0) {
  159. if (argc < 4)
  160. return cmd_usage(cmdtp);
  161. nregs = simple_strtoul(argv[2], NULL, 16);
  162. val = simple_strtoul(argv[3], NULL, 16);
  163. pmic_reg_write(nregs, val);
  164. return 0;
  165. }
  166. /* No subcommand found */
  167. return 1;
  168. }
  169. U_BOOT_CMD(
  170. pmic, CONFIG_SYS_MAXARGS, 1, do_pmic,
  171. "Freescale PMIC (Atlas)",
  172. "dump [numregs] dump registers\n"
  173. "pmic write <reg> <value> - write register"
  174. );