protocol.c 2.4 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586
  1. /*
  2. * INET An implementation of the TCP/IP protocol suite for the LINUX
  3. * operating system. INET is implemented using the BSD Socket
  4. * interface as the means of communication with the user level.
  5. *
  6. * INET protocol dispatch tables.
  7. *
  8. * Authors: Ross Biro
  9. * Fred N. van Kempen, <waltje@uWalt.NL.Mugnet.ORG>
  10. *
  11. * Fixes:
  12. * Alan Cox : Ahah! udp icmp errors don't work because
  13. * udp_err is never called!
  14. * Alan Cox : Added new fields for init and ready for
  15. * proper fragmentation (_NO_ 4K limits!)
  16. * Richard Colella : Hang on hash collision
  17. * Vince Laviano : Modified inet_del_protocol() to correctly
  18. * maintain copy bit.
  19. *
  20. * This program is free software; you can redistribute it and/or
  21. * modify it under the terms of the GNU General Public License
  22. * as published by the Free Software Foundation; either version
  23. * 2 of the License, or (at your option) any later version.
  24. */
  25. #include <linux/cache.h>
  26. #include <linux/module.h>
  27. #include <linux/netdevice.h>
  28. #include <linux/spinlock.h>
  29. #include <net/protocol.h>
  30. const struct net_protocol __rcu *inet_protos[MAX_INET_PROTOS] __read_mostly;
  31. const struct net_offload __rcu *inet_offloads[MAX_INET_PROTOS] __read_mostly;
  32. /*
  33. * Add a protocol handler to the hash tables
  34. */
  35. int inet_add_protocol(const struct net_protocol *prot, unsigned char protocol)
  36. {
  37. if (!prot->netns_ok) {
  38. pr_err("Protocol %u is not namespace aware, cannot register.\n",
  39. protocol);
  40. return -EINVAL;
  41. }
  42. return !cmpxchg((const struct net_protocol **)&inet_protos[protocol],
  43. NULL, prot) ? 0 : -1;
  44. }
  45. EXPORT_SYMBOL(inet_add_protocol);
  46. int inet_add_offload(const struct net_offload *prot, unsigned char protocol)
  47. {
  48. return !cmpxchg((const struct net_offload **)&inet_offloads[protocol],
  49. NULL, prot) ? 0 : -1;
  50. }
  51. EXPORT_SYMBOL(inet_add_offload);
  52. /*
  53. * Remove a protocol from the hash tables.
  54. */
  55. int inet_del_protocol(const struct net_protocol *prot, unsigned char protocol)
  56. {
  57. int ret;
  58. ret = (cmpxchg((const struct net_protocol **)&inet_protos[protocol],
  59. prot, NULL) == prot) ? 0 : -1;
  60. synchronize_net();
  61. return ret;
  62. }
  63. EXPORT_SYMBOL(inet_del_protocol);
  64. int inet_del_offload(const struct net_offload *prot, unsigned char protocol)
  65. {
  66. int ret;
  67. ret = (cmpxchg((const struct net_offload **)&inet_offloads[protocol],
  68. prot, NULL) == prot) ? 0 : -1;
  69. synchronize_net();
  70. return ret;
  71. }
  72. EXPORT_SYMBOL(inet_del_offload);