protocol.c 2.3 KB

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