netdev.c 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331
  1. /*!
  2. * Copyright (C) Fraunhofer-Institut for Photonic Microsystems (IPMS)
  3. * Maria-Reiche-Str. 2
  4. * 01109 Dresden
  5. *
  6. * Unauthorized copying of this file, via any medium is strictly prohibited
  7. * Proprietary and confidential
  8. *
  9. * \file netdev.c
  10. * \author zimmerli
  11. * \date 2020-01-16
  12. * \brief Generic Network Device
  13. *
  14. */
  15. #include "netdev.h"
  16. #include <string.h>
  17. #include <kernel/base/kernel.h>
  18. #include <kernel/net/sock.h>
  19. // ----------------------------------------------------------------
  20. static uint32_t _net_dev_cnt = 0;
  21. // TODO implementation might need a list -> currently avoid many malloc
  22. #define GETBYID_MAX 8
  23. static struct netdev_s *_device_ptr[GETBYID_MAX] = { 0 };
  24. const char *NETDEV_CONTEXT = "NETDEV";
  25. //TODO unknown device (name: "unknown", callbacks: unsupported..., ... !?
  26. const struct netdev_s UNKNOWN_DEVICE = { 0 };
  27. // ----------------------------------------------------------------
  28. /**
  29. * \brief Allocate and register network device driver
  30. *
  31. * @return pointer to structure, NULL on error
  32. */
  33. struct netdev_s *netdev_register(void)
  34. {
  35. struct netdev_s *p;
  36. // alloc struct
  37. p = (struct netdev_s *)kallocz(sizeof(struct netdev_s));
  38. if (!p)
  39. return p;
  40. // set fields
  41. memset(p->name, 0, IFNAMSIZ);
  42. p->id = _net_dev_cnt;
  43. _net_dev_cnt++;
  44. // register
  45. if (p->id < GETBYID_MAX) {
  46. _device_ptr[p->id] = p;
  47. }
  48. return p;
  49. }
  50. /**
  51. * \brief Get registered network device driver by interface name
  52. *
  53. * @param name interface name
  54. * @return pointer to device driver, NULL if not found
  55. */
  56. struct netdev_s *netdev_getbyname(char *name)
  57. {
  58. struct netdev_s *ret = NULL;
  59. for (int id = 0; id < GETBYID_MAX; id++) {
  60. ret = _device_ptr[id];
  61. if (ret && strcmp(ret->name, name) == 0)
  62. return ret;
  63. }
  64. return NULL;
  65. }
  66. /**
  67. * \brief Get registered network device driver by id
  68. *
  69. * Maximum supported number of devices -> GETBYID_MAX = 8
  70. *
  71. * @param id id
  72. * @return pointer to device driver, NULL if not found
  73. */
  74. struct netdev_s *netdev_getbyid(uint32_t id)
  75. {
  76. if (id >= GETBYID_MAX)
  77. return NULL;
  78. return _device_ptr[id];
  79. }
  80. struct netdev_s *netdev_port2netdev(uint32_t port)
  81. {
  82. struct netdev_s *ret = NULL;
  83. for (int id = 0; id < GETBYID_MAX; id++) {
  84. ret = _device_ptr[id];
  85. if (ret && ret->tsn_port == port)
  86. return ret;
  87. }
  88. logk(LOG_ALWAYS, NETDEV_CONTEXT, "device not found for tsn port '%d'\n", port);
  89. return NULL;
  90. }
  91. void netdev_set_name(struct netdev_s *netdev, const char *prefix, uint32_t index)
  92. {
  93. char str[5];
  94. _itoa(index, str, 10);
  95. strcpy(netdev->name, prefix);
  96. strcat(netdev->name, str);
  97. }
  98. //---------------------------------------------------------------------------------------------------------------------
  99. //TODO sind das driver function? wo geh�ren die hin
  100. /**
  101. * \brief Network device transmit
  102. *
  103. * @param netdev pointer to device driver
  104. * @param netb pointer to network buffer
  105. * @return NETDEV_RESULT_OK on success
  106. */
  107. int32_t netdev_xmit(struct netdev_s *netdev, struct netb_s *netb)
  108. {
  109. int32_t retval = NETDEV_ERR_UNSUPPORTED;
  110. if (netdev && netdev->netdev_xmit) {
  111. netb->netdev = netdev;
  112. retval = netdev->netdev_xmit(netdev, netb);
  113. } else {
  114. netdev->stats.tx_error++;
  115. }
  116. return retval;
  117. }
  118. /**
  119. * \brief Network device update link speed
  120. *
  121. * @param netdev pointer to device driver
  122. * @param speed speed in mbps (10/100/1000)
  123. */
  124. void netdev_adjust_link(struct netdev_s *netdev, int speed)
  125. {
  126. if (!netdev)
  127. return;
  128. netdev->speed = speed;
  129. if (netdev->netdev_link)
  130. netdev->netdev_link(netdev);
  131. }
  132. /**
  133. * \brief Network device set MAC address, promiscous mode
  134. * @param netdev pointer to device driver
  135. * @param addr pointer to MAC address
  136. * @param rx_promisc promiscous mode flag
  137. */
  138. void netdev_set_addr(struct netdev_s *netdev, uint8_t *addr, uint8_t rx_promisc)
  139. {
  140. uint32_t idx;
  141. if (!netdev)
  142. return;
  143. for (idx = 0; idx < MAC_ADDR_LEN; ++idx)
  144. netdev->dev_addr[idx] = addr[idx];
  145. netdev->promisc = rx_promisc;
  146. if (netdev->netdev_addr)
  147. netdev->netdev_addr(netdev);
  148. }
  149. /**
  150. * \brief Network device, traffic control settings
  151. *
  152. * @param netdev pointer to device driver
  153. * @param access access type (get/set)
  154. * @param type traffic control type (cbs, tas)
  155. * @param data pointer to settings structure
  156. * @return
  157. */
  158. int32_t netdev_tc(struct netdev_s *netdev, enum netdev_access access, enum tc_setup_type type, void *data)
  159. {
  160. if (!netdev)
  161. return NETDEV_ERR_GENERAL;
  162. if (!netdev->netdev_tc)
  163. return NETDEV_ERR_UNSUPPORTED;
  164. return netdev->netdev_tc(netdev, access, type, data);
  165. }
  166. /**
  167. * \brief Network device, frame preemption settings
  168. *
  169. * @param netdev pointer to device driver
  170. * @param access access type (get/set)
  171. * @param data pointer to settings structure
  172. * @return
  173. */
  174. int32_t netdev_fpe(struct netdev_s *netdev, enum netdev_access access, struct netdev_fpe *data)
  175. {
  176. if (!netdev)
  177. return NETDEV_ERR_GENERAL;
  178. if (!netdev->netdev_fpe)
  179. return NETDEV_ERR_UNSUPPORTED;
  180. return netdev->netdev_fpe(netdev, access, data);
  181. }
  182. /**
  183. * \brief Network device, ethtool settings
  184. *
  185. * @param netdev pointer to device driver
  186. * @param access access type (get/set)
  187. * @param data pointer to settings structure
  188. * @return
  189. */
  190. int32_t netdev_ethtool(struct netdev_s *netdev, enum netdev_access access, enum ethtool_command cmd, void *data)
  191. {
  192. if (!netdev || !netdev->netdev_ethtool)
  193. return NETDEV_ERR_GENERAL;
  194. return netdev->netdev_ethtool(netdev, access, cmd, data);
  195. }
  196. /**
  197. * \brief Network device, monitor settings
  198. *
  199. * @param netdev pointer to device driver
  200. * @param access access type (get/set)
  201. * @param data pointer to settings structure
  202. * @return
  203. */
  204. int32_t netdev_monitor(struct netdev_s *netdev, enum netdev_access access, void *data)
  205. {
  206. if (!netdev || !netdev->netdev_monitor)
  207. return NETDEV_ERR_GENERAL;
  208. return netdev->netdev_monitor(netdev, access, data);
  209. }
  210. /**
  211. * \brief Network device, bridge settings
  212. *
  213. * @param netdev pointer to device driver
  214. * @param access access type (get/set)
  215. * @param data pointer to settings structure
  216. * @return
  217. */
  218. int32_t netdev_bridge(struct netdev_s *netdev, enum netdev_access access, enum bridge_command cmd, void *data)
  219. {
  220. if (!netdev || !netdev->netdev_bridge)
  221. return NETDEV_ERR_GENERAL;
  222. return netdev->netdev_bridge(netdev, access, cmd, data);
  223. }
  224. /**
  225. * \brief Network device, fdfifo settings
  226. *
  227. * @param netdev pointer to device driver
  228. * @param access access type (get/set)
  229. * @param data pointer to settings structure
  230. * @return
  231. */
  232. int32_t netdev_fdfifo(struct netdev_s *netdev, enum netdev_access access, enum fdfifo_command cmd, void *data)
  233. {
  234. if (!netdev || !netdev->netdev_fdfifo)
  235. return NETDEV_ERR_GENERAL;
  236. return netdev->netdev_fdfifo(netdev, access, cmd, data);
  237. }
  238. /**
  239. * \brief Network device, general receive funtion
  240. *
  241. * @param netdev pointer to network device
  242. * @param netb pointer to received network buffer
  243. */
  244. void netdev_rx(struct netdev_s *netdev, struct netb_s *netb)
  245. {
  246. socket_rx(netdev, netb);
  247. //TODO see socket_rx()
  248. //sp�ter (+ alte applikationen (lni, linktest, rtping, ...) gehen �ber RAW socket)
  249. //netb --> pbuf
  250. //netb --> netdev --> netif
  251. //netif->input(pbuf, netif)
  252. }
  253. /**
  254. * \brief Network device, register tx-timestamping channel
  255. *
  256. * @param netdev pointer to device driver
  257. * @param qsize size for queue holding timestamps
  258. * @return channel id, 0 otherwise
  259. */
  260. uint8_t netdev_tstamp_register(struct netdev_s *netdev, uint32_t qsize)
  261. {
  262. if (!netdev || !netdev->netdev_tstamp_register)
  263. return 0;
  264. return netdev->netdev_tstamp_register(netdev, qsize);
  265. }
  266. /**
  267. * \brief Network device, unregister tx-timestamping channel
  268. *
  269. * @param netdev pointer to device driver
  270. * @param id channel id, previously registered
  271. */
  272. void netdev_tstamp_unregister(struct netdev_s *netdev, uint8_t id)
  273. {
  274. if (!netdev || !netdev->netdev_tstamp_unregister)
  275. return;
  276. netdev->netdev_tstamp_unregister(netdev, id);
  277. }
  278. /**
  279. * \brief Get tx-timestamp from channel
  280. *
  281. * Wait for reception until timeout. Use timeout=0 for immediate return.
  282. *
  283. * @param netdev pointer to device driver
  284. * @param id channel id, previously registered
  285. * @param timeout timeout in system OS ticks (likely milliseconds)
  286. * @return >0 timestamp; <= 0 on error
  287. */
  288. int64_t netdev_tstamp_get(struct netdev_s *netdev, uint8_t id, uint32_t timeout)
  289. {
  290. if (!netdev || !netdev->netdev_tstamp_get)
  291. return -1;
  292. return netdev->netdev_tstamp_get(netdev, id, timeout);
  293. }