Get started - it's free!
Login
© 2025

Site-to-site networking

You can use Tailscale subnet routers to connect two or more subnets (such as different physical locations or cloud environments) within a single, secure network mesh. This type of configuration is called site-to-site (L3) networking.

Requirements and limitations

The subnets and subnet router devices within a Tailscale network (known as a tailnet) must meet the following requirements for site-to-site networking to work:

  • The subnets must not have overlapping CIDR ranges or use 4via6 subnet routing.
  • Both subnet routers must use a Linux-based operating system.

Create a site-to-site connection

To connect two or more subnets within your tailnet (creating a site-to-site connection), you must do the following for each subnet:

  1. Make sure each subnet meets the minimum requirements.
  2. Select a device within the subnet to act as the subnet router.
  3. Configure the device as a subnet router.
  4. Approve the subnet router.
  5. Update the tailnet access control policies to allow communication between the subnets.
  6. Configure the other devices in the subnet.

Select a subnet router

The first step is to select a device within the subnet to act as the designated subnet router. This device must use a Linux-based operating system.

Configure the subnet router

After selecting a device in the subnet to function as the subnet router, you must configure it to act as a subnet router. Configuring the subnet router involves installing the Tailscale client, enabling IP address forwarding, starting the Tailscale client with the correct configuration options, and configuring the device's iptables settings.

IP address forwarding

If your Linux system has a /etc/sysctl.d directory, use:

echo 'net.ipv4.ip_forward = 1' | sudo tee -a /etc/sysctl.d/99-tailscale.conf
echo 'net.ipv6.conf.all.forwarding = 1' | sudo tee -a /etc/sysctl.d/99-tailscale.conf
sudo sysctl -p /etc/sysctl.d/99-tailscale.conf

Otherwise, use:

echo 'net.ipv4.ip_forward = 1' | sudo tee -a /etc/sysctl.conf
echo 'net.ipv6.conf.all.forwarding = 1' | sudo tee -a /etc/sysctl.conf
sudo sysctl -p /etc/sysctl.conf

Subnet router configuration options

Configuring a device as a subnet router involves specifying the subnet routes to advertise, disabling source network address translation (SNAT), and enabling the device to accept routes. You can use the tailscale up or tailscale set CLI commands to set these configuration options.

Configuration optionCLI flagDescription
Advertise subnet routes--advertise-routes=<CIDR>The --advertise-routes flag specifies a CIDR range of the IP addresses to be exposed to the tailnet. Access to those addresses is controlled by Tailscale access control policies.
Disable source network address translation (SNAT)--snat-subnet-routes=false (Linux only)The --snat-subnet-routes=false flag disables source NAT (SNAT). By default, a device behind a subnet router sees traffic as originating from the subnet router. This simplifies routing but prevents traversing multiple networks. By disabling source NAT, the end device sees the IP address of the originating device as the source, which might be a Tailscale IP address or an address behind another subnet router.
Accept routes--accept-routesThe --accept-routes flag accepts the advertised routes of all other subnet routers in the tailnet.

Clamp the MSS to the MTU

You should also clamp the maximum segment size (MSS) to the maximum transmission unit (MTU). You can do this using iptables. The following command updates the Tailscale network interface (tailscale0) to clamp MSS to MTU.

iptables -t mangle -A FORWARD -o tailscale0 -p tcp -m tcp \
--tcp-flags SYN,RST SYN -j TCPMSS --clamp-mss-to-pmtu

Approve the subnet router

After selecting and configuring the subnet router, you must approve it from the Machines page of the admin console. If you used an auto approver policy in the tailnet policy file for the device, you don’t need to approve the subnet router manually.

  1. Open the Machines page of the admin console.
  2. Locate the subnet router devices by locating the Subnets badge or using the property:subnet filter.
  3. For each subnet router:
    1. Select the ellipsis icon menu > Edit route settings.
    2. Approve the device.

You might prefer to disable key expiry on your subnet routers to avoid having to periodically reauthenticate. If you are using tags, key expiry is disabled by default.

Update tailnet access control policies

You must update the access control policies for your tailnet to allow communication between the subnets. You can do this by updating the tailnet policy file with the appropriate ACLs or grants.

For example, the following tailnet policy file allows all traffic between the subnets using ACLs:

{
   "acls": [
      {
            "action": "accept",
            "src": [ <CIDR-range-of-Subnet-A> ],
            "dst": [ <CIDR-range-of-Subnet-B> ]
      },
      {
            "action": "accept",
            "src": [ <CIDR-range-of-Subnet-B> ],
            "dst": [ <CIDR-range-of-Subnet-A> ]
      }
   ]
}

Configure the other subnet devices

After configuring and approving the subnet router, make sure the devices in the subnet use the designated subnet router for the correct routes. The best way to configure the devices in the subnet depends on your infrastructure. For example, if the subnet devices already use the subnet router as the default gateway, you don't need to configure them because they will automatically use the subnet router for all traffic (unless a more specific route applies). If the subnet devices don't use the subnet router as the default gateway, you must configure them to use the subnet router for the correct routes.

For example, on Linux devices, you can use the ip route command. You don't need to configure the subnet devices if the subnet router you selected for the subnet is already the default gateway for the subnet.

ip route add <first-subnet-CIDR> via <first-subnet-router-IP-address>
ip route add <second-subnet-CIDR> via <second-subnet-router-IP-address>

The ip route commands do not persist after rebooting. You must run them again after each reboot. Depending on your setup, you can make the route settings persistent by adding them to your network manager or netplan configuration. Alternatively, you can manage route settings with a DHCP server on your network.

If the subnet is in a cloud environment, such as AWS, you can usually update the cloud provider's routing tables instead of configuring each device directly.

Example scenario

The following sections demonstrate using two subnet routers to connect two subnets within a tailnet. The example subnets and subnet routers are documented in the table below.

Subnet nameSubnet CIDR rangeSubnet router IP address
Subnet A192.0.2.0/24192.0.2.2 (subnet router A)
Subnet B172.16.100.0/24172.16.100.2 (subnet router A)

To create a site-to-site connection between Subnet A and Subnet B, follow these steps:

  1. Configure both subnet routers (Subnet A at 192.0.2.2 and Subnet B at 172.16.100.2):

    1. Install the Tailscale client on each subnet router.

    2. Enable IP forwarding.

    3. Configure the Tailscale client.

      Make sure to replace <CIDR> with the correct subnet routes. For the 192.0.2.2 subnet router, use 192.0.2.0/24. For the 172.16.100.2 subnet router, use 172.16.100.0/24.

      tailscale up --advertise-routes=<CIDR> --snat-subnet-routes=false --accept-routes
      

      The --snat-subnet-routes flag only works if the operating system is Linux.

    4. Configure iptables to clamp the MSS to the MTU.

  2. Approve the subnet routes.

  3. Configure the subnet devices by running the following commands on each device in the subnet (except the subnet router):

    1. For each device in the 192.0.2.0/24 subnet (except the subnet router), run the following commands:

      ip route add 100.64.0.0/10 via 192.0.2.2
      ip route add 172.16.100.0/24 via 192.0.2.2
      
    2. For each device in the 172.16.100.0/24 subnet (except the subnet router), run the following commands:

      ip route add 100.64.0.0/10 via 172.16.100.2
      ip route add 192.0.2.0/24 via 172.16.100.2
      
  4. Update the tailnet access control policies to allow communication between the subnets. In the following example, the tailnet policy file allows all traffic between the subnets using ACLs:

    {
       "acls": [
          {
                "action": "accept",
                "src": [ 100.64.0.0/10 ], // CIDR range of Subnet A
                "dst": [ 192.0.2.0/24:* ], // CIDR range of Subnet B
          },
          {
                "action": "accept",
                "src": [ 192.0.2.0/24 ], // CIDR range of Subnet B
                "dst": [ 100.64.0.0/10:* ], // CIDR range of Subnet A
          }
       ]
    }
    
  5. Test the connectivity between Subnet A and Subnet B:

    Now a device in Subnet A can connect to a device in Subnet B (and vice versa) without either needing to install the Tailscale client. You can test the connection by running the ping command from a Subnet A device to a Subnet B device. For example, ping 172.16.100.3 from 198.0.2.3:

    ping 172.16.100.3
    
    PING 172.16.100.3 (172.16.100.3) 56(84) bytes of data.
    64 bytes from 172.16.100.3: icmp_seq=1 ttl=64 time=9.34 ms
    64 bytes from 172.16.100.3: icmp_seq=2 ttl=64 time=3.85 ms
    

Last updated Jan 10, 2025