protocol.c 1.8 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162
  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. /*
  32. * Add a protocol handler to the hash tables
  33. */
  34. int inet_add_protocol(const struct net_protocol *prot, unsigned char protocol)
  35. {
  36. int hash = protocol & (MAX_INET_PROTOS - 1);
  37. return !cmpxchg((const struct net_protocol **)&inet_protos[hash],
  38. NULL, prot) ? 0 : -1;
  39. }
  40. EXPORT_SYMBOL(inet_add_protocol);
  41. /*
  42. * Remove a protocol from the hash tables.
  43. */
  44. int inet_del_protocol(const struct net_protocol *prot, unsigned char protocol)
  45. {
  46. int ret, hash = protocol & (MAX_INET_PROTOS - 1);
  47. ret = (cmpxchg((const struct net_protocol **)&inet_protos[hash],
  48. prot, NULL) == prot) ? 0 : -1;
  49. synchronize_net();
  50. return ret;
  51. }
  52. EXPORT_SYMBOL(inet_del_protocol);