board.c 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134
  1. /*
  2. * Copyright (C) 2016 Stefan Roese <sr@denx.de>
  3. *
  4. * SPDX-License-Identifier: GPL-2.0+
  5. */
  6. #include <common.h>
  7. #include <i2c.h>
  8. #include <asm/io.h>
  9. #include <asm/arch/cpu.h>
  10. #include <asm/arch/soc.h>
  11. DECLARE_GLOBAL_DATA_PTR;
  12. /* IO expander I2C device */
  13. #define I2C_IO_EXP_ADDR 0x22
  14. #define I2C_IO_CFG_REG_0 0x6
  15. #define I2C_IO_DATA_OUT_REG_0 0x2
  16. #define I2C_IO_REG_0_SATA_OFF 2
  17. #define I2C_IO_REG_0_USB_H_OFF 1
  18. int board_early_init_f(void)
  19. {
  20. /* Nothing to do (yet), perhaps later some pin-muxing etc */
  21. return 0;
  22. }
  23. int board_init(void)
  24. {
  25. /* adress of boot parameters */
  26. gd->bd->bi_boot_params = CONFIG_SYS_SDRAM_BASE + 0x100;
  27. return 0;
  28. }
  29. /* Board specific AHCI / SATA enable code */
  30. int board_ahci_enable(void)
  31. {
  32. struct udevice *dev;
  33. int ret;
  34. u8 buf[8];
  35. /* Configure IO exander PCA9555: 7bit address 0x22 */
  36. ret = i2c_get_chip_for_busnum(0, I2C_IO_EXP_ADDR, 1, &dev);
  37. if (ret) {
  38. printf("Cannot find PCA9555: %d\n", ret);
  39. return 0;
  40. }
  41. ret = dm_i2c_read(dev, I2C_IO_CFG_REG_0, buf, 1);
  42. if (ret) {
  43. printf("Failed to read IO expander value via I2C\n");
  44. return -EIO;
  45. }
  46. /*
  47. * Enable SATA power via IO expander connected via I2C by setting
  48. * the corresponding bit to output mode to enable power for SATA
  49. */
  50. buf[0] &= ~(1 << I2C_IO_REG_0_SATA_OFF);
  51. ret = dm_i2c_write(dev, I2C_IO_CFG_REG_0, buf, 1);
  52. if (ret) {
  53. printf("Failed to set IO expander via I2C\n");
  54. return -EIO;
  55. }
  56. return 0;
  57. }
  58. /* Board specific xHCI enable code */
  59. int board_xhci_enable(void)
  60. {
  61. struct udevice *dev;
  62. int ret;
  63. u8 buf[8];
  64. /* Configure IO exander PCA9555: 7bit address 0x22 */
  65. ret = i2c_get_chip_for_busnum(0, I2C_IO_EXP_ADDR, 1, &dev);
  66. if (ret) {
  67. printf("Cannot find PCA9555: %d\n", ret);
  68. return 0;
  69. }
  70. printf("Enable USB VBUS\n");
  71. /*
  72. * Read configuration (direction) and set VBUS pin as output
  73. * (reset pin = output)
  74. */
  75. ret = dm_i2c_read(dev, I2C_IO_CFG_REG_0, buf, 1);
  76. if (ret) {
  77. printf("Failed to read IO expander value via I2C\n");
  78. return -EIO;
  79. }
  80. buf[0] &= ~(1 << I2C_IO_REG_0_USB_H_OFF);
  81. ret = dm_i2c_write(dev, I2C_IO_CFG_REG_0, buf, 1);
  82. if (ret) {
  83. printf("Failed to set IO expander via I2C\n");
  84. return -EIO;
  85. }
  86. /* Read VBUS output value and disable it */
  87. ret = dm_i2c_read(dev, I2C_IO_DATA_OUT_REG_0, buf, 1);
  88. if (ret) {
  89. printf("Failed to read IO expander value via I2C\n");
  90. return -EIO;
  91. }
  92. buf[0] &= ~(1 << I2C_IO_REG_0_USB_H_OFF);
  93. ret = dm_i2c_write(dev, I2C_IO_DATA_OUT_REG_0, buf, 1);
  94. if (ret) {
  95. printf("Failed to set IO expander via I2C\n");
  96. return -EIO;
  97. }
  98. /*
  99. * Required delay for configuration to settle - must wait for
  100. * power on port is disabled in case VBUS signal was high,
  101. * required 3 seconds delay to let VBUS signal fully settle down
  102. */
  103. mdelay(3000);
  104. /* Enable VBUS power: Set output value of VBUS pin as enabled */
  105. buf[0] |= (1 << I2C_IO_REG_0_USB_H_OFF);
  106. ret = dm_i2c_write(dev, I2C_IO_DATA_OUT_REG_0, buf, 1);
  107. if (ret) {
  108. printf("Failed to set IO expander via I2C\n");
  109. return -EIO;
  110. }
  111. mdelay(500); /* required delay to let output value settle */
  112. return 0;
  113. }