eth_common.c 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166
  1. /*
  2. * (C) Copyright 2001-2015
  3. * Wolfgang Denk, DENX Software Engineering, wd@denx.de.
  4. * Joe Hershberger, National Instruments
  5. *
  6. * SPDX-License-Identifier: GPL-2.0+
  7. */
  8. #include <common.h>
  9. #include <dm.h>
  10. #include <miiphy.h>
  11. #include <net.h>
  12. #include "eth_internal.h"
  13. void eth_parse_enetaddr(const char *addr, uchar *enetaddr)
  14. {
  15. char *end;
  16. int i;
  17. for (i = 0; i < 6; ++i) {
  18. enetaddr[i] = addr ? simple_strtoul(addr, &end, 16) : 0;
  19. if (addr)
  20. addr = (*end) ? end + 1 : end;
  21. }
  22. }
  23. int eth_getenv_enetaddr(const char *name, uchar *enetaddr)
  24. {
  25. eth_parse_enetaddr(getenv(name), enetaddr);
  26. return is_valid_ethaddr(enetaddr);
  27. }
  28. int eth_setenv_enetaddr(const char *name, const uchar *enetaddr)
  29. {
  30. char buf[20];
  31. sprintf(buf, "%pM", enetaddr);
  32. return setenv(name, buf);
  33. }
  34. int eth_getenv_enetaddr_by_index(const char *base_name, int index,
  35. uchar *enetaddr)
  36. {
  37. char enetvar[32];
  38. sprintf(enetvar, index ? "%s%daddr" : "%saddr", base_name, index);
  39. return eth_getenv_enetaddr(enetvar, enetaddr);
  40. }
  41. int eth_setenv_enetaddr_by_index(const char *base_name, int index,
  42. uchar *enetaddr)
  43. {
  44. char enetvar[32];
  45. sprintf(enetvar, index ? "%s%daddr" : "%saddr", base_name, index);
  46. return eth_setenv_enetaddr(enetvar, enetaddr);
  47. }
  48. void eth_common_init(void)
  49. {
  50. bootstage_mark(BOOTSTAGE_ID_NET_ETH_START);
  51. #if defined(CONFIG_MII) || defined(CONFIG_CMD_MII) || defined(CONFIG_PHYLIB)
  52. miiphy_init();
  53. #endif
  54. #ifdef CONFIG_PHYLIB
  55. phy_init();
  56. #endif
  57. }
  58. int eth_mac_skip(int index)
  59. {
  60. char enetvar[15];
  61. char *skip_state;
  62. sprintf(enetvar, index ? "eth%dmacskip" : "ethmacskip", index);
  63. skip_state = getenv(enetvar);
  64. return skip_state != NULL;
  65. }
  66. void eth_current_changed(void)
  67. {
  68. char *act = getenv("ethact");
  69. char *ethrotate;
  70. /*
  71. * The call to eth_get_dev() below has a side effect of rotating
  72. * ethernet device if uc_priv->current == NULL. This is not what
  73. * we want when 'ethrotate' variable is 'no'.
  74. */
  75. ethrotate = getenv("ethrotate");
  76. if ((ethrotate != NULL) && (strcmp(ethrotate, "no") == 0))
  77. return;
  78. /* update current ethernet name */
  79. if (eth_get_dev()) {
  80. if (act == NULL || strcmp(act, eth_get_name()) != 0)
  81. setenv("ethact", eth_get_name());
  82. }
  83. /*
  84. * remove the variable completely if there is no active
  85. * interface
  86. */
  87. else if (act != NULL)
  88. setenv("ethact", NULL);
  89. }
  90. void eth_try_another(int first_restart)
  91. {
  92. static void *first_failed;
  93. char *ethrotate;
  94. /*
  95. * Do not rotate between network interfaces when
  96. * 'ethrotate' variable is set to 'no'.
  97. */
  98. ethrotate = getenv("ethrotate");
  99. if ((ethrotate != NULL) && (strcmp(ethrotate, "no") == 0))
  100. return;
  101. if (!eth_get_dev())
  102. return;
  103. if (first_restart)
  104. first_failed = eth_get_dev();
  105. eth_set_current_to_next();
  106. eth_current_changed();
  107. if (first_failed == eth_get_dev())
  108. net_restart_wrap = 1;
  109. }
  110. void eth_set_current(void)
  111. {
  112. static char *act;
  113. static int env_changed_id;
  114. int env_id;
  115. env_id = get_env_id();
  116. if ((act == NULL) || (env_changed_id != env_id)) {
  117. act = getenv("ethact");
  118. env_changed_id = env_id;
  119. }
  120. if (act == NULL) {
  121. char *ethprime = getenv("ethprime");
  122. void *dev = NULL;
  123. if (ethprime)
  124. dev = eth_get_dev_by_name(ethprime);
  125. if (dev)
  126. eth_set_dev(dev);
  127. else
  128. eth_set_dev(NULL);
  129. } else {
  130. eth_set_dev(eth_get_dev_by_name(act));
  131. }
  132. eth_current_changed();
  133. }
  134. const char *eth_get_name(void)
  135. {
  136. return eth_get_dev() ? eth_get_dev()->name : "unknown";
  137. }