udp_tunnels.c 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. // Copyright (c) 2020 Facebook Inc.
  3. #include <linux/debugfs.h>
  4. #include <linux/netdevice.h>
  5. #include <linux/slab.h>
  6. #include <net/udp_tunnel.h>
  7. #include "netdevsim.h"
  8. static int
  9. nsim_udp_tunnel_set_port(struct net_device *dev, unsigned int table,
  10. unsigned int entry, struct udp_tunnel_info *ti)
  11. {
  12. struct netdevsim *ns = netdev_priv(dev);
  13. int ret;
  14. ret = -ns->udp_ports.inject_error;
  15. ns->udp_ports.inject_error = 0;
  16. if (ns->udp_ports.sleep)
  17. msleep(ns->udp_ports.sleep);
  18. if (!ret) {
  19. if (ns->udp_ports.ports[table][entry]) {
  20. WARN(1, "entry already in use\n");
  21. ret = -EBUSY;
  22. } else {
  23. ns->udp_ports.ports[table][entry] =
  24. be16_to_cpu(ti->port) << 16 | ti->type;
  25. }
  26. }
  27. netdev_info(dev, "set [%d, %d] type %d family %d port %d - %d\n",
  28. table, entry, ti->type, ti->sa_family, ntohs(ti->port),
  29. ret);
  30. return ret;
  31. }
  32. static int
  33. nsim_udp_tunnel_unset_port(struct net_device *dev, unsigned int table,
  34. unsigned int entry, struct udp_tunnel_info *ti)
  35. {
  36. struct netdevsim *ns = netdev_priv(dev);
  37. int ret;
  38. ret = -ns->udp_ports.inject_error;
  39. ns->udp_ports.inject_error = 0;
  40. if (ns->udp_ports.sleep)
  41. msleep(ns->udp_ports.sleep);
  42. if (!ret) {
  43. u32 val = be16_to_cpu(ti->port) << 16 | ti->type;
  44. if (val == ns->udp_ports.ports[table][entry]) {
  45. ns->udp_ports.ports[table][entry] = 0;
  46. } else {
  47. WARN(1, "entry not installed %x vs %x\n",
  48. val, ns->udp_ports.ports[table][entry]);
  49. ret = -ENOENT;
  50. }
  51. }
  52. netdev_info(dev, "unset [%d, %d] type %d family %d port %d - %d\n",
  53. table, entry, ti->type, ti->sa_family, ntohs(ti->port),
  54. ret);
  55. return ret;
  56. }
  57. static int
  58. nsim_udp_tunnel_sync_table(struct net_device *dev, unsigned int table)
  59. {
  60. struct netdevsim *ns = netdev_priv(dev);
  61. struct udp_tunnel_info ti;
  62. unsigned int i;
  63. int ret;
  64. ret = -ns->udp_ports.inject_error;
  65. ns->udp_ports.inject_error = 0;
  66. for (i = 0; i < NSIM_UDP_TUNNEL_N_PORTS; i++) {
  67. udp_tunnel_nic_get_port(dev, table, i, &ti);
  68. ns->udp_ports.ports[table][i] =
  69. be16_to_cpu(ti.port) << 16 | ti.type;
  70. }
  71. return ret;
  72. }
  73. static const struct udp_tunnel_nic_info nsim_udp_tunnel_info = {
  74. .set_port = nsim_udp_tunnel_set_port,
  75. .unset_port = nsim_udp_tunnel_unset_port,
  76. .sync_table = nsim_udp_tunnel_sync_table,
  77. .tables = {
  78. {
  79. .n_entries = NSIM_UDP_TUNNEL_N_PORTS,
  80. .tunnel_types = UDP_TUNNEL_TYPE_VXLAN,
  81. },
  82. {
  83. .n_entries = NSIM_UDP_TUNNEL_N_PORTS,
  84. .tunnel_types = UDP_TUNNEL_TYPE_GENEVE |
  85. UDP_TUNNEL_TYPE_VXLAN_GPE,
  86. },
  87. },
  88. };
  89. static ssize_t
  90. nsim_udp_tunnels_info_reset_write(struct file *file, const char __user *data,
  91. size_t count, loff_t *ppos)
  92. {
  93. struct net_device *dev = file->private_data;
  94. struct netdevsim *ns = netdev_priv(dev);
  95. memset(ns->udp_ports.ports, 0, sizeof(ns->udp_ports.__ports));
  96. rtnl_lock();
  97. udp_tunnel_nic_reset_ntf(dev);
  98. rtnl_unlock();
  99. return count;
  100. }
  101. static const struct file_operations nsim_udp_tunnels_info_reset_fops = {
  102. .open = simple_open,
  103. .write = nsim_udp_tunnels_info_reset_write,
  104. .llseek = generic_file_llseek,
  105. .owner = THIS_MODULE,
  106. };
  107. int nsim_udp_tunnels_info_create(struct nsim_dev *nsim_dev,
  108. struct net_device *dev)
  109. {
  110. struct netdevsim *ns = netdev_priv(dev);
  111. struct udp_tunnel_nic_info *info;
  112. if (nsim_dev->udp_ports.shared && nsim_dev->udp_ports.open_only) {
  113. dev_err(&nsim_dev->nsim_bus_dev->dev,
  114. "shared can't be used in conjunction with open_only\n");
  115. return -EINVAL;
  116. }
  117. if (!nsim_dev->udp_ports.shared)
  118. ns->udp_ports.ports = ns->udp_ports.__ports;
  119. else
  120. ns->udp_ports.ports = nsim_dev->udp_ports.__ports;
  121. debugfs_create_u32("udp_ports_inject_error", 0600,
  122. ns->nsim_dev_port->ddir,
  123. &ns->udp_ports.inject_error);
  124. ns->udp_ports.dfs_ports[0].array = ns->udp_ports.ports[0];
  125. ns->udp_ports.dfs_ports[0].n_elements = NSIM_UDP_TUNNEL_N_PORTS;
  126. debugfs_create_u32_array("udp_ports_table0", 0400,
  127. ns->nsim_dev_port->ddir,
  128. &ns->udp_ports.dfs_ports[0]);
  129. ns->udp_ports.dfs_ports[1].array = ns->udp_ports.ports[1];
  130. ns->udp_ports.dfs_ports[1].n_elements = NSIM_UDP_TUNNEL_N_PORTS;
  131. debugfs_create_u32_array("udp_ports_table1", 0400,
  132. ns->nsim_dev_port->ddir,
  133. &ns->udp_ports.dfs_ports[1]);
  134. debugfs_create_file("udp_ports_reset", 0200, ns->nsim_dev_port->ddir,
  135. dev, &nsim_udp_tunnels_info_reset_fops);
  136. /* Note: it's not normal to allocate the info struct like this!
  137. * Drivers are expected to use a static const one, here we're testing.
  138. */
  139. info = kmemdup(&nsim_udp_tunnel_info, sizeof(nsim_udp_tunnel_info),
  140. GFP_KERNEL);
  141. if (!info)
  142. return -ENOMEM;
  143. ns->udp_ports.sleep = nsim_dev->udp_ports.sleep;
  144. if (nsim_dev->udp_ports.sync_all) {
  145. info->set_port = NULL;
  146. info->unset_port = NULL;
  147. } else {
  148. info->sync_table = NULL;
  149. }
  150. if (ns->udp_ports.sleep)
  151. info->flags |= UDP_TUNNEL_NIC_INFO_MAY_SLEEP;
  152. if (nsim_dev->udp_ports.open_only)
  153. info->flags |= UDP_TUNNEL_NIC_INFO_OPEN_ONLY;
  154. if (nsim_dev->udp_ports.ipv4_only)
  155. info->flags |= UDP_TUNNEL_NIC_INFO_IPV4_ONLY;
  156. if (nsim_dev->udp_ports.shared)
  157. info->shared = &nsim_dev->udp_ports.utn_shared;
  158. if (nsim_dev->udp_ports.static_iana_vxlan)
  159. info->flags |= UDP_TUNNEL_NIC_INFO_STATIC_IANA_VXLAN;
  160. dev->udp_tunnel_nic_info = info;
  161. return 0;
  162. }
  163. void nsim_udp_tunnels_info_destroy(struct net_device *dev)
  164. {
  165. kfree(dev->udp_tunnel_nic_info);
  166. dev->udp_tunnel_nic_info = NULL;
  167. }
  168. void nsim_udp_tunnels_debugfs_create(struct nsim_dev *nsim_dev)
  169. {
  170. debugfs_create_bool("udp_ports_sync_all", 0600, nsim_dev->ddir,
  171. &nsim_dev->udp_ports.sync_all);
  172. debugfs_create_bool("udp_ports_open_only", 0600, nsim_dev->ddir,
  173. &nsim_dev->udp_ports.open_only);
  174. debugfs_create_bool("udp_ports_ipv4_only", 0600, nsim_dev->ddir,
  175. &nsim_dev->udp_ports.ipv4_only);
  176. debugfs_create_bool("udp_ports_shared", 0600, nsim_dev->ddir,
  177. &nsim_dev->udp_ports.shared);
  178. debugfs_create_bool("udp_ports_static_iana_vxlan", 0600, nsim_dev->ddir,
  179. &nsim_dev->udp_ports.static_iana_vxlan);
  180. debugfs_create_u32("udp_ports_sleep", 0600, nsim_dev->ddir,
  181. &nsim_dev->udp_ports.sleep);
  182. }