protocol.c 2.3 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980
  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. 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. EXPORT_SYMBOL(inet_offloads);
  33. int inet_add_protocol(const struct net_protocol *prot, unsigned char protocol)
  34. {
  35. if (!prot->netns_ok) {
  36. pr_err("Protocol %u is not namespace aware, cannot register.\n",
  37. protocol);
  38. return -EINVAL;
  39. }
  40. return !cmpxchg((const struct net_protocol **)&inet_protos[protocol],
  41. NULL, prot) ? 0 : -1;
  42. }
  43. EXPORT_SYMBOL(inet_add_protocol);
  44. int inet_add_offload(const struct net_offload *prot, unsigned char protocol)
  45. {
  46. return !cmpxchg((const struct net_offload **)&inet_offloads[protocol],
  47. NULL, prot) ? 0 : -1;
  48. }
  49. EXPORT_SYMBOL(inet_add_offload);
  50. int inet_del_protocol(const struct net_protocol *prot, unsigned char protocol)
  51. {
  52. int ret;
  53. ret = (cmpxchg((const struct net_protocol **)&inet_protos[protocol],
  54. prot, NULL) == prot) ? 0 : -1;
  55. synchronize_net();
  56. return ret;
  57. }
  58. EXPORT_SYMBOL(inet_del_protocol);
  59. int inet_del_offload(const struct net_offload *prot, unsigned char protocol)
  60. {
  61. int ret;
  62. ret = (cmpxchg((const struct net_offload **)&inet_offloads[protocol],
  63. prot, NULL) == prot) ? 0 : -1;
  64. synchronize_net();
  65. return ret;
  66. }
  67. EXPORT_SYMBOL(inet_del_offload);