README.enetaddr 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110
  1. ---------------------------------
  2. Ethernet Address (MAC) Handling
  3. ---------------------------------
  4. There are a variety of places in U-Boot where the MAC address is used, parsed,
  5. and stored. This document covers proper usage of each location and the moving
  6. of data between them.
  7. -----------
  8. Locations
  9. -----------
  10. Here are the places where MAC addresses might be stored:
  11. - board-specific location (eeprom, dedicated flash, ...)
  12. Note: only used when mandatory due to hardware design etc...
  13. - environment ("ethaddr", "eth1addr", ...) (see CONFIG_ETHADDR)
  14. Note: this is the preferred way to permanently store MAC addresses
  15. - ethernet data (struct eth_device -> enetaddr)
  16. Note: these are temporary copies of the MAC address which exist only
  17. after the respective init steps have run and only to make usage
  18. in other places easier (to avoid constant env lookup/parsing)
  19. - struct bd_info and/or device tree
  20. Note: these are temporary copies of the MAC address only for the
  21. purpose of passing this information to an OS kernel we are about
  22. to boot
  23. Correct flow of setting up the MAC address (summarized):
  24. 1. Read from hardware in initialize() function
  25. 2. Read from environment in net/eth.c after initialize()
  26. 3. Give priority to the value in the environment if a conflict
  27. 4. Program hardware in the device's init() function.
  28. If somebody wants to subvert the design philosophy, this can be done
  29. in the board-specific board_eth_init() function by calling eth_init()
  30. after all the NICs have been registered.
  31. -------
  32. Usage
  33. -------
  34. If the hardware design mandates that the MAC address is stored in some special
  35. place (like EEPROM etc...), then the board specific init code (such as the
  36. board-specific misc_init_r() function) is responsible for locating the MAC
  37. address(es) and initializing the respective environment variable(s) from it.
  38. Note that this shall be done if, and only if, the environment does not already
  39. contain these environment variables, i.e. existing variable definitions must
  40. not be overwritten.
  41. During runtime, the ethernet layer will use the environment variables to sync
  42. the MAC addresses to the ethernet structures. All ethernet driver code should
  43. then only use the enetaddr member of the eth_device structure. This is done
  44. on every network command, so the ethernet copies will stay in sync.
  45. Any other code that wishes to access the MAC address should query the
  46. environment directly. The helper functions documented below should make
  47. working with this storage much smoother.
  48. ---------
  49. Helpers
  50. ---------
  51. To assist in the management of these layers, a few helper functions exist. You
  52. should use these rather than attempt to do any kind of parsing/manipulation
  53. yourself as many common errors have arisen in the past.
  54. * void eth_parse_enetaddr(const char *addr, uchar *enetaddr);
  55. Convert a string representation of a MAC address to the binary version.
  56. char *addr = "00:11:22:33:44:55";
  57. uchar enetaddr[6];
  58. eth_parse_enetaddr(addr, enetaddr);
  59. /* enetaddr now equals { 0x00, 0x11, 0x22, 0x33, 0x44, 0x55 } */
  60. * int eth_getenv_enetaddr(char *name, uchar *enetaddr);
  61. Look up an environment variable and convert the stored address. If the address
  62. is valid, then the function returns 1. Otherwise, the function returns 0. In
  63. all cases, the enetaddr memory is initialized. If the env var is not found,
  64. then it is set to all zeros. The common function is_valid_ether_addr() is used
  65. to determine address validity.
  66. uchar enetaddr[6];
  67. if (!eth_getenv_enetaddr("ethaddr", enetaddr)) {
  68. /* "ethaddr" is not set in the environment */
  69. ... try and setup "ethaddr" in the env ...
  70. }
  71. /* enetaddr is now set to the value stored in the ethaddr env var */
  72. * int eth_setenv_enetaddr(char *name, const uchar *enetaddr);
  73. Store the MAC address into the named environment variable. The return value is
  74. the same as the setenv() function.
  75. uchar enetaddr[6] = { 0x00, 0x11, 0x22, 0x33, 0x44, 0x55 };
  76. eth_setenv_enetaddr("ethaddr", enetaddr);
  77. /* the "ethaddr" env var should now be set to "00:11:22:33:44:55" */
  78. * the %pM format modifier
  79. The %pM format modifier can be used with any standard printf function to format
  80. the binary 6 byte array representation of a MAC address.
  81. uchar enetaddr[6] = { 0x00, 0x11, 0x22, 0x33, 0x44, 0x55 };
  82. printf("The MAC is %pM\n", enetaddr);
  83. char buf[20];
  84. sprintf(buf, "%pM", enetaddr);
  85. /* the buf variable is now set to "00:11:22:33:44:55" */