serial_dm.c 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. /*
  2. * Copyright (c) 2006-2022, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2022-11-16 GuEe-GUI first version
  9. */
  10. #include <rtatomic.h>
  11. #include <drivers/serial_dm.h>
  12. static int uid_min = -1;
  13. static volatile rt_atomic_t uid = 0;
  14. int serial_dev_set_name(struct rt_serial_device *sdev)
  15. {
  16. int id = -1;
  17. RT_ASSERT(sdev != RT_NULL);
  18. #ifdef RT_USING_OFW
  19. if (sdev->parent.ofw_node)
  20. {
  21. id = rt_ofw_get_alias_id(sdev->parent.ofw_node, "serial");
  22. if (id < 0)
  23. {
  24. id = rt_ofw_get_alias_id(sdev->parent.ofw_node, "uart");
  25. }
  26. }
  27. #endif
  28. if (id < 0)
  29. {
  30. id = (int)rt_hw_atomic_add(&uid, 1);
  31. }
  32. return rt_dm_dev_set_name(&sdev->parent, "uart%u", id);
  33. }
  34. static int serial_dm_naming_framework_init(void)
  35. {
  36. #ifdef RT_USING_OFW
  37. uid_min = rt_ofw_get_alias_last_id("serial");
  38. if (uid_min < 0)
  39. {
  40. uid_min = rt_ofw_get_alias_last_id("uart");
  41. }
  42. uid_min = uid_min < 0 ? 0 : (uid_min + 1);
  43. rt_hw_atomic_store(&uid, uid_min);
  44. #endif
  45. return 0;
  46. }
  47. INIT_PLATFORM_EXPORT(serial_dm_naming_framework_init);
  48. void *serial_base_from_args(char *str)
  49. {
  50. rt_ubase_t base = 0;
  51. while (*str && !(*str == 'x' || *str == 'X'))
  52. {
  53. ++str;
  54. }
  55. ++str;
  56. /* The str may get from bootargs that we need check it */
  57. while (*str)
  58. {
  59. if ((*str >= 'a' && *str <= 'f') || (*str >= 'A' && *str <= 'F'))
  60. {
  61. base = (base << 4) | (((*str | ' ') - 'a') + 10);
  62. }
  63. else if (*str >= '0' && *str <= '9')
  64. {
  65. base = (base << 4) | (*str - '0');
  66. }
  67. else break;
  68. ++str;
  69. }
  70. return (void *)base;
  71. }
  72. struct serial_configure serial_cfg_from_args(char *_str)
  73. {
  74. char *str = _str;
  75. struct serial_configure cfg = RT_SERIAL_CONFIG_DEFAULT;
  76. /* Format baudrate/parity/bits/flow (BBBBPNF), Default is 115200n8 */
  77. if (str && *str)
  78. {
  79. rt_uint32_t baudrate = 0;
  80. /* BBBB is the speed */
  81. while (*str && (*str >= '0' && *str <= '9'))
  82. {
  83. baudrate *= 10;
  84. baudrate += *str - '0';
  85. ++str;
  86. }
  87. if (baudrate)
  88. {
  89. cfg.baud_rate = baudrate;
  90. }
  91. /* P is parity (n/o/e) */
  92. switch (*str)
  93. {
  94. case 'n':
  95. cfg.parity = PARITY_NONE;
  96. break;
  97. case 'o':
  98. cfg.parity = PARITY_ODD;
  99. break;
  100. case 'e':
  101. cfg.parity = PARITY_EVEN;
  102. break;
  103. default:
  104. --str;
  105. break;
  106. }
  107. ++str;
  108. /* N is number of bits */
  109. if (*str && (*str >= '0' && *str <= '9'))
  110. {
  111. cfg.data_bits = *str - '0';
  112. ++str;
  113. }
  114. /* F is flow ontrol ('r' for RTS) */
  115. if (*str)
  116. {
  117. cfg.flowcontrol = (*str == 'r' ? RT_SERIAL_FLOWCONTROL_CTSRTS : RT_SERIAL_FLOWCONTROL_NONE);
  118. ++str;
  119. }
  120. #ifdef RT_USING_OFW
  121. if (*str == '\0')
  122. {
  123. const char earlycon_magic[] = { 'O', 'F', 'W', '\0' };
  124. if (!rt_strcmp(++str, earlycon_magic))
  125. {
  126. /* Is OFW earlycon, we should ACK it */
  127. rt_memset(str, 0, RT_ARRAY_SIZE(earlycon_magic));
  128. }
  129. }
  130. #endif
  131. }
  132. return cfg;
  133. }