You have mail. Read it very closely.
IPFW Primer
trademarks
FreeBSD is a registered trademark of the FreeBSD Foundation.
Git and the Git logo are either registered trademarks or trademarks of Software Freedom Conservancy, Inc., corporate home of the Git Project, in the United States and/or other countries.
Many of the designations used by manufacturers and sellers to distinguish their products are claimed as trademarks. Where those designations appear in this document, and the FreeBSD Project was aware of the trademark claim, the designations have been followed by the “™” or the “®” symbol.
Table of Contents
Abstract
ipfw(8) is a firewall application that comes standard with FreeBSD. This book provides an introduction to IPFW and its capabilities. These capabilities are demonstrated using QEMU virtual machines in real-world scenarios.
Content includes descriptions of the main features of ipfw(8) and lab examples on using IPFW as a firewall or a network gateway.
Like IPFW itself, this book is a work in progress.
Acknowledgments
Thanks to the FreeBSD developers who added and maintain IPFW.
Also, thanks to those who took the time to read early drafts of this document and offered many valuable comments and criticisms.
Attributions
Artwork from the following sources was used in preparing this book:
Stylized nginx logo
Original image: NGINX Logo
Obtained from Wikimedia https://en.wikipedia.org/wiki/Nginx#/media/File:Nginx_logo.svg. Image is listed as in the Public Domain.
The image was converted to .PNG and also to .BMP formats for use in this book and accompanying code. Conversions were accomplished via ImageMagick’s convert program.
Stylized bind 9 logo
Original image: BIND 9’s New Logo
Obtained from https://www.isc.org/blogs/bind-9s-new-logo/
Logo used with permission from Internet Systems Consortium (ISC).
The image was converted to .PNG and also to .BMP formats for use in this book and accompanying code. Conversions were accomplished via ImageMagick’s convert program.
Stylized IPv6 logo
Original image: World IPv6 Launch Logo
Obtained from Wikipedia: https://commons.wikimedia.org/wiki/File:World_IPv6_launch_logo.svg Original CC 3.0 license: https://creativecommons.org/licenses/by/3.0/deed.en Updated CC 4.0 license: https://creativecommons.org/licenses/by/4.0/deed.en
The image was converted to .PNG and also to .BMP formats for use in this book and accompanying code. Conversions were accomplished via ImageMagick’s convert program.
Use of the above artwork does not imply endorsement of any opinions expressed in this book.
All other artwork is original artwork by the author.
Preface
This page describes the conventions used in this book.
Shell Prompts
This table shows the default system prompt and superuser prompt. The examples use these prompts to indicate which type of user is running the example.
User | Prompt |
---|---|
Normal user | % |
| # |
Typographic Conventions
This table shows various typographic conventions used throughout the text.
Meaning | Examples |
---|---|
Data, sysctls, things to note. |
|
The names of files. | Edit .login. |
On-screen computer output. Output highlighting. | |
What the user types, contrasted with on-screen computer output. | # ipfw add 100 check-state 00100 check-state :default |
Manual page references. | Use su(1) to change user identity. |
Emphasis levels | Emphasis. Stronger emphasis. Strongest emphasis. |
Environment variables. |
|
Notes, Tips, Important Information, Warnings, and Examples
Notes, warnings, and examples appear within the text.
Notes are represented like this, and contain information to take note of, as it may affect what the user does. |
Tips are represented like this, and contain information helpful to the user, such as showing an easier way to do something. |
Important information is represented like this. Typically, these show extra steps the user may need to take. |
Warnings are represented like this, and contain information warning about possible damage if the instructions are not followed. This damage may be physical, to the hardware or the user, or it may be non-physical, such as the inadvertent deletion of important files. |
Examples are represented like this, and typically contain examples showing a walkthrough, or the results of a particular action.
Chapter 1. Introduction
This book is about one of the native firewalls included with FreeBSD, ipfw(8) - the Internet Protocol FireWall. ipfw is designed to operate on a FreeBSD host with multiple network interfaces, to filter out unwanted traffic and pass through desired traffic. It does this based on a collection of rules (numbered, text based statements) that are entered into the system from the command line. This usage model is different from many other firewall products that employ Graphical User Interfaces (GUIs), or separate control programs. All ipfw statements are entered into the user shell, typically by a user with root privileges or access to root privilege by means of programs that elevate normal user privileges such as sudo(8) or doas(1).
ipfw reads network traffic from the interfaces it knows about and processes them inside the FreeBSD kernel. ipfw itself is a kernel module that can be either compiled into the kernel or loaded at run time. It includes a number of other kernel modules (ipfw_nat, ipfw_nptv6, etc.) many of which we discuss in this book.
A bird’s-eye view of ipfw operation notes that:
Rules are organized into a sorted list based on a rule number
Packets entering the kernel from a network interface or leaving the kernel via a network interface are checked against the ruleset
Rules are checked one by one and the first rule that matches the packet characteristics wins - that is, it accepts the packet for processing, either allowing transit through the firewall, denying transit, or moving the packet into userspace for specialized processing.
The book makes frequent reference to the ipfw(8) manual page and the reader is advised to become familiar with the manual page alongside this book. There is also a section on ipfw in the FreeBSD Handbook Page on IPFW. The intent with this book is to provide examples and informative material beyond the manual page and handbook to increase understanding and usage of ipfw.
Throughout this book are many examples of using ipfw with virtual machines to simulate actual hardware. These examples were developed with QEMU version 9.0.2. It is, of course, entirely possible to perform all the examples in this book with real hardware. QEMU provides a way to perform the examples without spending any money for hardware. In either case, some setup is required.
Note that QEMU command syntax with some of the examples may have changed slightly by the time this book becomes available. Use the latest QEMU release where possible, and check the QEMU documentation if the examples in this book do not work correctly. |
Also used are a number of scripts that allow easy if_bridge(4) and tap(4) setup, virtual machine setup, and data transfer from external VMs to or through a firewall VM. In the early examples, data transfer is accomplished with the netcat program, specifically the version distributed with the nmap package (www.nmap.org). This version, ncat(1), has the best coverage of features that are used throughout the book. A familiarity with the man page for ncat(1) is helpful, but not required.
All scripts used in this book are found in Appendix B and published under the BSD 3-clause license. The scripts are also available on the GitHub IPFW Primer page.
When copy/pasting examples, be aware that some desktop copy/paste functions add an extra space (or multiple spaces) to the end of a line, messing up the Unix continuation character convention ' … \' at the end of a line. You should ensure that the paste function does not introduce extra spaces at the end of the line.
The lab examples in this book involve passing data between interfaces on the host system. A running firewall on the host such as pf, ipfw, or ipfilter (also known as ipf) may interfere with data transfer, so ensure that any host system firewall is disabled. In addition, take any necessary steps to ensure that this does not compromise the security of the host. |
1.1. Quick Start
This section details the steps to quickly begin using two QEMU virtual machines, one named "firewall" and the other named "external1". These are the first two of several that will be used throughout this book. Additional detail, along with setup instructions for all virtual machines, is provided in Appendix A.
The initial setup is that shown in Figure 1.
1.1.1. QEMU VM Installation Process
Follow the steps below to install and configure the QEMU virtual machines for this example.
On the FreeBSD host, install the necessary packages - qemu(1), sudo(8) (or doas(1)). Sudo, (or doas) is necessary for running the virtual machines as these QEMU configurations open a separate console window through SDL. The examples in this book use sudo.
# pkg install qemu sudo
Configure sudo as desired.
Create a directory layout for virtual machines and scripts, and download an install ISO for FreeBSD.
% mkdir -p ~/ipfw/VM ~/ipfw/SCRIPTS ~/ipfw/ISO % cd ~/ipfw/ISO % fetch https://download.freebsd.org/releases/amd64/amd64/ISO-IMAGES/<latest version>/FreeBSD-<latest-version>-RELEASE-amd64-dvd1.iso
Create the bridge and tap devices for the virtual machines (VMs) to use.
# ifconfig tap0 create # ifconfig tap1 create # sysctl net.link.tap.up_on_open=1 net.link.tap.up_on_open: 0 -> 1 # sysctl net.link.tap.user_open=1 net.link.tap.user_open: 0 -> 1 # ifconfig bridge0 create # ifconfig bridge0 addm tap0 addm tap1 addm hostintf <--- replace hostintf with host network interface (em0, bge0, etc.) # ifconfig bridge0 up
A script for creating and managing bridge and tap devices is introduced in the next section.
Create two new VM image files and install FreeBSD on one.
% cd ~/ipfw/VM % qemu-img create -f qcow2 -o preallocation=full firewall.qcow2 8G % qemu-img create -f qcow2 -o preallocation=full external1.qcow2 8G % cd ~/ipfw/ISO % # Link a shorter name to the ISO image. % ln -s FreeBSD-<latest-version>-RELEASE-amd64-dvd1.iso fbsd.iso % cd ~/ipfw/SCRIPTS Copy the below text into a file (say, firewall.sh) and run: % sudo /bin/sh firewall.sh ------- #!/bin/sh # firewall.sh /usr/local/bin/qemu-system-x86_64 -monitor stdio \ -cpu qemu64 \ -vga std \ -m 4096 \ -smp 4 \ -cdrom ../ISO/fbsd.iso \ -boot order=cd,menu=on \ -blockdev driver=file,aio=threads,node-name=imgright,filename=../VM/firewall.qcow2 \ -blockdev driver=qcow2,node-name=drive0,file=imgright \ -device virtio-blk-pci,drive=drive0,bootindex=1 \ -netdev tap,id=nd0,ifname=tap0,script=no,downscript=no,br=bridge0 \ -device e1000,netdev=nd0,mac=02:69:70:66:77:00 \ -name \"Firewall\" exit -------
The FreeBSD installer should boot. Perform a standard installation of FreeBSD.
During the installation note the following:
Select to use UFS as the filesystem. ZFS does not perform well with small memory sizes.
In this Quick Start, use DHCP for networking. If desired, configure IPv6 if supported by the local LAN.
When adding the default user, ensure they are a member of the wheel group.
Once the installation completes, the virtual machine reboots into the newly installed FreeBSD image.
Login as root and update the system if desired.
Repeat the above step to create another QEMU script file, and perform another installation with these changes:
Copy the below text into a file (say, external1.sh) and run: % sudo /bin/sh external1.sh ------- #!/bin/sh # external1.sh /usr/local/bin/qemu-system-x86_64 -monitor stdio \ -cpu qemu64 \ -vga std \ -m 4096 \ -smp 4 \ -cdrom ../ISO/fbsd.iso \ -boot order=cd,menu=on \ -blockdev driver=file,aio=threads,node-name=imgleft,filename=../VM/external1.qcow2 \ -blockdev driver=qcow2,node-name=drive0,file=imgleft \ -device virtio-blk-pci,drive=drive0,bootindex=1 \ -netdev tap,id=nd0,ifname=tap1,script=no,downscript=no,br=bridge0 \ -device e1000,netdev=nd0,mac=02:20:65:78:74:31 \ -name \"External1\" exit -------
As above, login and update the system if desired.
On both virtual machines (and all later installed VMs) , install the packages listed below The
nmap
package brings in the version of ncat(1) used by scripts on the firewall and external VMs.nginx
,lynx
,cmdwatch
,hping3
, andiperf3
will be used in later chapters.# pkg install nmap nginx lynx cmdwatch hping3 iperf3
Finally, download IPFW_root_bin.tgz file to both VMs. This tar file has a number of scripts needed for the virtual machines.
Move the tarzip file into /root and extract the contents:
# fetch https://raw.githubusercontent.com/jimmyb-gh/ipfw-primer/main/ipfw/SCRIPTS/VM_SCRIPTS/IPFW_root_bin.tgz # # mv IPFW_root_bin.tgz /root # # cd /root # # tar xvzf IPFW_root_bin.tgz ... files are extracted into /root/bin # # chmod +x /root/bin/*.sh
(End installation procedure.)
For this Quick Start, it is Ok to use DHCP
for both VMs.
In later examples there will be multiple external VMs using the 203.0.113.0/24 network and other private networks, all set up the same way and attached via tap(4) interfaces to one or more if_bridge(4) interfaces on the FreeBSD host.
To ensure you have the first two VMs setup correctly, ping the firewall VM from the external1 VM and vice-versa. Communications should be successful. If not, check the above installation details and troubleshoot any network issues. It should be possible ping in both directions, and even ssh(1) from one VM to the other.
If you are having difficulty getting QEMU set up correctly, check the QEMU virtualization section in the FreeBSD Handbook. |
If the mouse is clicked in the QEMU console window, QEMU will “grab” the mouse. If this happens,type, Ctl+Alt+G to release the mouse. |
If suddenly, the QEMU console window is full screen, you have accidentally typed Ctl+Alt+F. If this happens, retype Ctl+Alt+F to restore the desktop screen. |
1.1.2. Disabling Syslog Messages to the Console in the Virtual Machines
You may find it advantageous (even necessary) to stop syslog messages from being sent to the console (either the QEMU console, or the serial port).
To configure syslog to stop logging to the console, configure a file to receive console messages:
# touch /var/log/console.log # # chmod 0600 /var/log/console.log
Then, as root, modify the line in /etc/syslog.conf to read (instead of /dev/console:
*.err;kern.warning;auth.notice;mail.crit /var/log/console.log
And, if necessary:
# service syslogd restart
All messages previously bound for the console, will be directed to /var/log/console.log instead.
Before continuing, we have one more piece to add to each VM - a serial console. We will need a serial console to examine the state of each VM independent of the main console.
1.1.3. Adding and Managing Serial Console Access to the VMs
Adding a serial console to FreeBSD
To add a serial console to each FreeBSD VM, start up the VM and edit the file /boot/loader.conf and add console=“comconsole” to allow use of the serial console. Reboot the VM to begin using the serial console. Note that FreeBSD diverts boot I/O to the serial console, so until the FreeBSD operating system is completely ready, output to the QEMU window will be limited.
Adding a serial device to QEMU
Adding a serial console to QEMU is fairly straightforward. A single configuration line is added to the QEMU configuration to provide a serial device that is actually accessed over a telnet(1) session. The QEMU manual page, qemu(1), describes how the -serial keyword works in detail.
QEMU redirects the serial port I/O to a TCP port on the host system at VM startup, and the QEMU monitor allows a telnet(1) connection on the configured port on the host. Once the FreeBSD system starts booting and recognizes the console directive in /boot/loader.conf it redirects I/O to the serial console. The QEMU monitor detects this and manages the necessary character I/O on that serial port to the TCP port on the host.
It is important to note that the this serial redirect over TCP takes place outside the virtual machine. There is no interaction with any network on the virtual machine and thus it is not subject to any firewall rules. Think of it just like a "dumb terminal" sitting on an RS-232 serial port on a real machine.
Management of serial console windows on the FreeBSD host
However, before we start adding serial devices, we need to plan how to manage them. Each QEMU VM generates a console window, and each serial device also needs its own window, potentially doubling the number of windows used.
Possible solutions are:
Our solution uses the multiplexer approach with the tmux(1) program for window management. Appendix D provides details on using both tmux(1) and screen(1). The following figures and descriptions use tmux(1).
Install tmux with:
# pkg install tmux
and review the file swim.sh in the SCRIPTS directory.
The figure below shows the use of the swim.sh tmux session manager. Run sh swim.sh in the SCRIPTS directory to start up the session manager.
The figure shows five named windows in one session (session [0]) with the tmux status line in green at the bottom:
0:bash - a terminal window of the user running swim.sh
1:firewall - a terminal window to access the firewall VM
2:external1 - a terminal window to access the external1 VM
3:external2 - a terminal window to access the external2 VM
4:external3 - a terminal window to access the external3 VM
The current window is marked with the '*' character on the status bar.
Uncomment additional window lines in swim.sh as you need them. |
Simplified tmux(1) Usage
tmux uses Ctl+b as its control key. To move from window to window use Ctl+b n to move to the next window or Ctl+b p to move to the previous window. Use Ctl+b ? for a list of all key bindings.
Type tmux kill-server in any session window to completely leave tmux.
Consult the tmux manual page tmux(1) for more usage details.
Accessing the QEMU Serial Consoles
To access the VM serial consoles, move to the indicated window and telnet to the port on the local host for that VM:
Move to the external1 window in tmux, then ~/ipfw/SCRIPTS % telnet localhost 4410 Trying ::1... Connected to localhost. Escape character is '^]'. FreeBSD/amd64 (external1) (ttyu0) login:
You can exit out of the telnet session by pressing Ctl+] then pressing q like this: login: (type Ctl+]) telnet> q Connection closed. ~/ipfw/SCRIPTS % |
Edits for the firewall VM to use the serial console are shown below:
#!/bin/sh ## firewall.sh echo echo "NOTE!!! QEMU telnet server running!" echo "To access QEMU, telnet to localhost 4450" echo /usr/local/bin/qemu-system-x86_64 -monitor none \ -serial telnet:localhost:4450,server=on,wait=off \ -cpu qemu64 \ -vga std \ -m 4096 \ -smp 4 \ -cdrom ../ISO/fbsd.iso \ -boot order=cd,menu=on \ -blockdev driver=file,aio=threads,node-name=imgright,filename=../VM/firewall.qcow2 \ -blockdev driver=qcow2,node-name=drive0,file=imgright \ -device virtio-blk-pci,drive=drive0,bootindex=1 \ -netdev tap,id=nd0,ifname=tap0,script=no,downscript=no,br=bridge0 \ -device e1000,netdev=nd0,mac=02:69:70:66:77:00 \ -name \"Firewall\" & exit -------
Edits for the external1 VM to use the serial console:
------- #!/bin/sh ## external1.sh echo echo "NOTE!!! QEMU telnet server running!" echo "To access QEMU, telnet to localhost 4410" echo /usr/local/bin/qemu-system-x86_64 -monitor none \ -serial telnet:localhost:4410,server=on,wait=off \ -cpu qemu64 \ -vga std \ -m 4096 \ -smp 4 \ -cdrom ../ISO/fbsd.iso \ -boot order=cd,menu=on \ -blockdev driver=file,aio=threads,node-name=imgleft,filename=../VM/external1.qcow2 \ -blockdev driver=qcow2,node-name=drive0,file=imgleft \ -device virtio-blk-pci,drive=drive0,bootindex=1 \ -netdev tap,id=nd0,ifname=tap1,script=no,downscript=no,br=bridge0 \ -device e1000,netdev=nd0,mac=02:20:65:78:74:31 \ -name \"External1\" & exit -------
You should now have two QEMU VMs (firewall and external1) started, and have serial console sessions available through the tmux sessions as shown below.
Configure the FreeBSD host, firewall VM, and external1 VM with DHCP addressing as shown in the figure at the beginning on this Quick Start session. You should have full connectivity between your FreeBSD host, the firewall VM and the external1 VM.
Scripts used by the two QEMU VMs are discussed in the next section.
Chapter 2. IPFW Operation
If you have been successful so far, you now have the initial lab setup shown in the figure below except for IP addressing. All the examples in this book use "Special Use Addresses" for both IPv4 and IPv6. Address references used are described here for IPv4 Addresses (RFC 5737) and here for IPv6 Addresses (RFC 3849).
Configure both VMs to use the addresses shown. Once these addresses are in place on the VMs (and the alias address on the FreeBSD host), it is unlikely that the VMs will be able to access sites on the Internet. The 203.0.113.0/24 network is considered non-routable by Internet service providers and major telecom carriers on the Internet. However, this does not really matter, since all the communications for this book are local to the VMs running on the FreeBSD host. Bottom line - if you want your VMs to access the Internet, leave them on DHCP, but for the examples in this book, you’ll want to use manual addressing as shown in the figure below.
Scripts used by the two VMs are discussed in the next section.
2.1. Firewall Server Scripts
To demonstrate the firewall capabilities of ipfw, the firewall VM runs certain scripts that open local UDP and TCP ports using ncat(1) for TCP operation, and nc(1) for UDP operation. While the commands perform similar functions, the nc version on the base system is the one that performs best for receiving UDP traffic as it supports the keep-open option (-k).
The firewall VM runs one of four basic scripts. Recall that these are located in /root/bin:
tserv.sh (userv.sh): Script that opens one TCP (UDP) port and listens for incoming traffic.
tserv3.sh (userv3.sh): Script that opens 3 TCP (UDP) ports and listens for incoming traffic.
These scripts will listen for an incoming connection and print whatever is sent over during the connection. When the connection is closed, the script will continue to listen for the next connection.
These script provide the basic mechanism for receiving a TCP or UDP connection. Once we start the firewall and populate the ipfw ruleset, we will see the effect each rule has on the connection capabilities.
2.2. External VM Scripts
Likewise, there are other basic scripts the external1 (and later, external2 and external3) VMs use for initiating or establishing communications with or through the firewall VM. The TCP and UDP versions perform similarly:
tcon.sh (ucon.sh): Connect via TCP (or UDP). This script takes a single argument, a port number to use for the connection. The external VM host can change the port number at each prompt. If there is no script listening on the port on the firewall, the script will indicate a "connection refused" or timeout error.
tconr.sh (uconr.sh): Connection takes a random port number and a sleep value. The script randomly selects one of three ports for its connection in a loop, controlled by the sleep value. If there is a listener on the firewall active on the port, the connection succeeds - otherwise the connection is refused.
tcont.sh (ucont.sh): Connection takes a port number and sleep value. The communication uses the same port in a loop, controlled by the sleep value.
These are simple scripts, but they allow for independent activity by the external VMs, while the firewall VM admin (you) creates and tests ipfw rulesets. Most of the examples in the first part of this book can be done with just these scripts, so it is a good idea to familiarize yourself with their operation. Later scripts will use hping3(8) and iperf3(1), versatile tools used in network analysis.
By default, the external VMs and firewall VM scripts work on ports 5656
, 5657
, and 5658
.
The randomized communication scripts also utilize port 5659
, but since no services are listening on that port on the firewall VM, the connection fails.
It is important to understanding the underlying network activity. If you are unfamiliar with the basics of Internet protocols, network traffic, traffic monitoring and so forth, there are a number of excellent books, white papers, and tutorials, many free over the Internet. Check Appendix C for a modest selection.
2.3. Loading IPFW
ipfw can be built into the FreeBSD kernel directly, or it can be loaded as a kernel module. We will use the ipfw.ko loadable kernel module for most of the examples in this book. You can load ipfw as root with the command:
# kldload ipfw ipfw2 (+ipv6) initialized, divert loadable, nat loadable, default to deny, logging disabled
Notice the kernel display output - “ipfw2 (+ipv6) initialized, divert loadable, nat loadable, default to deny, logging disabled”. This gives a quick summary of this host’s ipfw capabilities. The most important to note is "default to deny" which indicates that, by default, the firewall has an immutable rule located at the end of the ruleset that denies all Internet Protocol (IP) traffic. This rule depends on how the kernel was configured when it was built. By default it is "default to deny". However, if you are working on a FreeBSD system where you do not know its provenance, use the ipfw list command to make sure:
# ipfw list 65535 deny ip from any to any #
Note that this does not mean it denies all network traffic, only traffic that is based on the Internet Protocol (RFC 791), and all of its derivatives (TCP, UDP, ICMP, etc). If you had the capability to send and receive non-IP based traffic you could possibly still send and receive it. You would need special rules to deny all traffic. We will see an example of that later in this book.
To begin the next section, start with the ipfw firewall unloaded:
# kldunload ipfw IP firewall unloaded #
Unloading (# kldunload ipfw) and loading (# kldload ipfw) the ipfw kernel module is a handy way of completely re-initializing ipfw. This removes all rules, sets, queues, pipes, and other ipfw operations. See kldload(8), kldunload(8), and kldstat(8) for details. |
2.4. Initial Firewall Setup
In this section we will introduce the operation of the scripts described above and demonstrate simple traffic filtering.
In the first example, the firewall host runs tserv.sh
which opens TCP port 5656.
The external1 VM runs tcon.sh
, which repetitively opens a TCP connection and sends data to the firewall VM.
Since there is no firewall in place, all TCP connections succeed.
On the host machine, it is possible to run tcpdump(1) on the bridge0 device to see the traffic in real time. Shown below is one successful transfer, i.e there are no firewall rules preventing the connection.
It follows the basic TCP connection sequence: 3-way handshake setup, send data, and close the connection:
We now load the ipfw firewall on the firewall VM and retry the communication.
# kldload ipfw ipfw2 (+ipv6) initialized, divert loadable, nat loadable, default to deny, logging disabled # # ipfw list 65535 deny ip from any to any #
No communications were successful - the connections time out because the ipfw firewall has denied traffic with the default deny
rule as described above.
The external1 VM host sends SYN packets to start the connection, but they never reach the firewall VM’s TCP service on port 5656.
The TCP 3-way handshake is never completed.
We will use these same techniques to show how communications and data transfer operate. We now turn our attention to firewall rule creation in the next section.
Chapter 3. IPFW Rules
The manual page for ipfw(8) lists the entire command syntax including those for general ruleset construction. We will look first at basic traffic rules and study the more complex capabilities as we progress. Each section will build on the previous sections so that you will be better positioned to understand the advanced material later in the book.
Almost all the examples in this section can be run on the architecture used in the previous chapter, specifically that of Chapter 2, Figure 1. The Quick Start section in the Introduction can provide guidance on setting this back up. Ensure you have the IPv4 addressing set up as shown in Chapter 2, Figure 1.
ipfw rules take the general format of:
# ipfw command [rule_number] [set set_number] [prob match_probability] action [log [logamount number]] [altq queue] [{tag | untag} number] body
Bolded keywords indicate literal option text that is added to a rule. Italicized keywords indicate a block of additional content that is rule dependent.
The rule body has its own syntax format:
[ proto from src to dst ] [ options ]
Here is an example with basic syntax for an entire rule:
The FreeBSD ipfw shell command
The ipfw rule command (add)
The optional rule number (1000)
An optional set keyword and value (set 2)
An optional probability keyword and value (prob 0.5)
The rule action keyword (deny)
An optional log keyword (log)
An optional logamount keyword and value (logamount 50000)
An optional altq keyword and value (altq red)
An optional tag keyword and value (tag 27)
The body of the rule starting with a protocol keyword (tcp)
A source direction keyword (from)
A source address (any internet address)
A destination keyword (to)
A destination address (any interface on the local system)
A destination port number (8081)
A comment. Note that comments are only valid in the match pattern section of a rule.
The example in the above figure would not actually load - there is no altq(9) queue named red
set up yet - but it does show the overall format of how rules are constructed.
This example also shows the use of the Unix line continuation convention using a single backslash at the end of the line (with no following spaces) to continue to the next line.
The essence of ipfw rule processing is found when patterns that are defined in the rule body are matched on incoming or outgoing packets, one at a time and the action keywords are processed in turn.
In other words, ipfw directs traffic flow by first matching each incoming or outgoing packet against patterns supplied within the body of the rule. The patterns include protocols (tcp, udp, igmp, eigrp, etc.), source and destination addresses and ports, and options that apply to the context of the traffic.
3.1. Practical Ruleset Development
In this section, we will concentrate on the basic commands and actions.
The basic command keywords are these:
enable/disable - commands to disable or enable ipfw rule processing. The kernel module remains loaded - the effect is to suspend or resume rule processing. This is important to understand early as these commands function like an 'on/off' switch to firewall operation.
add - adds a rule.
delete - deletes a rule. The rule number must be specified - for example, ipfw delete 1000.
list - lists the contents of the current ruleset. Even if there have been no rules added, the list command should always list out the default rule, by default,
65535 deny ip from any to any
.show - similar to the list command, show includes counters for each rule matched.
flush - delete all the rules in the ruleset except for rules in set 31. (Sets are described later in this Chapter.) Since this is a command with enormous impact, a Yes/No prompt is issued before continuing.
The basic traffic flow action keywords are these:
allow | accept | pass | permit - direct ipfw to allow a packet through this rule should the packet match the rule body.
count - increment a counter applied to a rule. No other processing is applied to the packet.
deny | drop - do not allow a packet to pass through this rule should the packet match the rule body.
check-state [:flowname | :any] - check if a dynamic rule already exists.
reset - resets Network Address Translation tables.
The ipfw(8) man page has the complete list of action keywords, and describes each in detail.
We will also examine these keywords in the rules section:
prob - assign a probability (a value between 0 and 1) to the rule action
set - use a collection of rules
tag and untag - apply an internal tag to a packet affected by the rule
log and logamount - log keywords
reset - send a TCP reset on connection
tee - cause packets to flow in multiple ways
unreach - specify an action if a packet’s destination is unreachable
setdscp - set DiffServe parameters for outbound packets
skipto - jump around a ruleset
divert - pull packets into userspace for programmatic purposes
limit - limit the number of active connections
call and return - another way to jump around a ruleset
lookup tables and the lookup - log keywords
Most action keywords, such as allow
or deny
, determine traffic flow.
It is important to become familiar with these actions as they will be used in almost every rule.
In addition, carefully note what ipfw does after it matches a packet and applies an action - it either terminates its search, or it goes on to the next rule.
Other action keywords perform an activity that does not have any impact on traffic flow. For example, the count action simply updates counters that apply to a rule. It has no effect on traffic flow and ipfw continues processing with the next rule.
Recall that ipfw is a command line program that uses all the words on the command line as parameters. In developing rules, remember that certain constructs such as braces ({,}), brackets ([,]) and even parentheses themselves are all recognized by the shell and must be escaped with a backslash '\'. The ipfw(8) man page has additional caveats on rule syntax.
We start with the smallest possible ruleset that permits the external1 VM to make a TCP connection to the service running on the firewall VM.
Start up the external1 VM, and the firewall VM. Load the ipfw.ko kernel module and start the tserv.sh service on the firewall VM to listen for incoming connections. Then create the following ruleset on the firewall VM:
# ipfw add 100 check-state 00100 check-state :default # ipfw add 1000 allow tcp from any to me 5656 in via em0 setup keep-state 01000 allow tcp from any to me 5656 in via em0 setup keep-state :default
You can test this ruleset immediately by again running sh tserv.sh on the firewall VM and sh tcon.sh 5656 script on the external1 VM as described in the previous chapter. The connection should succeed.
Rule 100 contains the check-state option. It checks to see if a connection is already established and a dynamic rule is in place. If so, any additional packets matching the dynamic rule would be passed. We will discuss "dynamic rules" shortly.
Rule 1000 contains the add command. This command inserts the requested rule into the ipfw ruleset where it can process packets against the specified actions in the rule body. The rule itself contains the allow keyword, which permits traffic to pass.
The rule also uses the setup and keep-state options to create a dynamic rule for the connection.
In a stateful firewall like ipfw, once a connection from an external host to an internal host is established, the firewall creates a dynamic rule permitting continued traffic along this path until the connection is reset.
Note that without the check-state
keyword, no check for a dynamic rule is performed and without the keep-state
, no creation of a dynamic rule is performed.
Consider this ruleset with just a single rule:
# ipfw add 1000 allow tcp from any to me 5656
This rule looks like it should work, but it does not. A TCP packet entering ipfw has no pre-existing dynamic rule. Further, the rule does not create one. And, because there is no corresponding rule for outbound traffic, no TCP 3-way handshake is ever completed. Note that a SYN packet is received by the firewall, but not by the destination service.
If we add the rule:
# ipfw add 2000 allow tcp from me to any
the TCP 3-way handshake is allowed to complete and the data is sent from the external1 VM host to the tserv.sh process running on the firewall VM.
While this method works, it uses two rules instead of one. In this case, the better solution is to use the setup, keep-state, and check-state options early in the ruleset as shown in the original example in this section.
3.2. Dynamic Rules
So, what exactly are "dynamic rules"? The scripts we are using close the TCP connection each time, so the dynamic rules are short lived, and cannot be easily examined.
To see dynamic rules in action, we can manually set up an ncat
listener on the firewall VM and send data with an ncat
sender on the external1 VM:
On the firewall VM, start up the listener service manually:
# ncat -l 203.0.113.50 5656
Then, on the external1 VM, use ncat to connect to the service on the firewall and type a message:
# ncat 203.0.113.50 5656 hello there ^C
The message should appear on the console of the firewall VM. If it does not, ensure that the original rule from the previous section is active.
The above figure shows the connection is open between the external1 and firewall VMs.
While the connection is still open, run the following command on the firewall VM serial console:
You should see output similar to that in the above figure.
The -d
option displays dynamic rules in addition to regular rules.
The -D
option displays just dynamic rules.
3.2.1. Notes on Rule Numbering
Each rule is assigned a rule number, even if you do not specify one.
The details for rule number handing are found in the ipfw(8) man page.
Note that rules are assigned numbers in increments specified by the sysctl net.inet.ip.fw.autoinc_step
.
# sysctl net.inet.ip.fw.autoinc_step net.inet.ip.fw.autoinc_step: 100
We restart with a simple check-state rule and see that ipfw has assigned a number associated with the increment sysctl shown above:
Flush the ipfw ruleset first. # ipfw -q flush # ipfw add check-state 00000 check-state :default # ipfw list 00100 check-state :default 65535 deny ip from any to any00000 check-state :default
ipfw has automatically assigned the rule number 100. While it can be convenient to have ipfw add a rule number automatically, you should always assign rule numbers yourself. This ensures that you consciously put a rule in a specific place within the ruleset. With large rulesets this is critical. A rule automatically assigned by ipfw can be placed where it can have an unexpected effect.
Consider this ruleset:
# ipfw list 00300 deny ip from any to 200.200.200.200 00400 deny ip from any to 200.200.200.201 00500 deny ip from any to 200.200.200.202 00600 deny ip from any to 200.200.200.203 00700 deny ip from any to 200.200.200.204 00800 deny ip from any to 200.200.200.205 00800 deny ip from any to 200.200.200.206 65535 deny ip from any to any
Having forgotten to add the check-state rule you quickly add it:
# ipfw add check-state 00000 check-state :default
But the result may not be what you intended:
# ipfw list 00300 deny ip from any to 200.200.200.200 00400 deny ip from any to 200.200.200.200 00500 deny ip from any to 200.200.200.200 00600 deny ip from any to 200.200.200.200 00700 deny ip from any to 200.200.200.200 00800 deny ip from any to 200.200.200.200 00800 deny ip from any to 200.200.200.200 00900 check-state :default 65535 deny ip from any to any
Also, ipfw allows rules with the same rule number to be added to the ruleset, and it will keep track of the rules in the order they were entered. This is easy to forget when manually entering rules from the command line and using command line editing to change something simple like the last byte of an IP address.
It is important to remember that all such rules are affected by commands that operate on one or more lines, such as the delete command:
# ipfw add 100 check-state 00100 check-state :default # ipfw add 1000 allow tcp from 203.0.113.10 to me 5656 setup keep-state 01000 allow tcp from 203.0.113.10 to me 5656 keep-state :default # ipfw add 1000 allow tcp from 203.0.113.20 to me 5656 setup keep-state 01000 allow tcp from 203.0.113.20 to me 5656 keep-state :default # ipfw add 1000 allow tcp from 203.0.113.30 to me 5656 setup keep-state 01000 allow tcp from 203.0.113.30 to me 5656 keep-state :default # ipfw add 1000 allow tcp from 203.0.113.40 to me 5656 setup keep-state 01000 allow tcp from 203.0.113.40 to me 5656 keep-state :default # # ipfw list 00100 check-state :default 01000 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default 01000 allow tcp from 203.0.113.20 to me 5656 setup keep-state :default 01000 allow tcp from 203.0.113.30 to me 5656 setup keep-state :default 01000 allow tcp from 203.0.113.40 to me 5656 setup keep-state :default 65535 deny ip from any to any # # ipfw delete 1000 # # ipfw list 00100 check-state :default 65535 deny ip from any to any
The delete command can also process both ranges and lists of rules.
Consider the following ruleset:
# ipfw list 00100 check-state :default 01000 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default 01100 allow tcp from 203.0.113.11 to me 5656 setup keep-state :default 01200 allow tcp from 203.0.113.12 to me 5656 setup keep-state :default 01300 allow tcp from 203.0.113.13 to me 5656 setup keep-state :default 02000 allow tcp from 203.0.113.20 to me 5656 setup keep-state :default 02100 allow tcp from 203.0.113.21 to me 5656 setup keep-state :default 02200 allow tcp from 203.0.113.22 to me 5656 setup keep-state :default 02300 allow tcp from 203.0.113.23 to me 5656 setup keep-state :default 03000 allow tcp from 203.0.113.30 to me 5656 setup keep-state :default 03100 allow tcp from 203.0.113.31 to me 5656 setup keep-state :default 03200 allow tcp from 203.0.113.32 to me 5656 setup keep-state :default 03300 allow tcp from 203.0.113.33 to me 5656 setup keep-state :default 04000 allow tcp from 203.0.113.40 to me 5656 setup keep-state :default 04100 allow tcp from 203.0.113.41 to me 5656 setup keep-state :default 04200 allow tcp from 203.0.113.42 to me 5656 setup keep-state :default 04300 allow tcp from 203.0.113.43 to me 5656 setup keep-state :default 04400 allow tcp from 203.0.113.44 to me 5656 setup keep-state :default 04500 allow tcp from 203.0.113.45 to me 5656 setup keep-state :default 04600 allow tcp from 203.0.113.46 to me 5656 setup keep-state :default 04700 allow tcp from 203.0.113.47 to me 5656 setup keep-state :default 04800 allow tcp from 203.0.113.48 to me 5656 setup keep-state :default 04900 allow tcp from 203.0.113.49 to me 5656 setup keep-state :default 65535 deny ip from any to any
A range is specified by two number separated by a dash: for example 5000-7350; whereas a list is a space-separated collection of numbers on the command line.
We wish to delete rules from 1000 to 2999 and certain rules between 4000 and 5000:
# ipfw delete 1000-2999 4100 4300 4500 4700 4900 # # ipfw list 00100 check-state :default 03000 allow tcp from 203.0.113.30 to me 5656 keep-state :default 03100 allow tcp from 203.0.113.31 to me 5656 keep-state :default 03200 allow tcp from 203.0.113.32 to me 5656 keep-state :default 03300 allow tcp from 203.0.113.33 to me 5656 keep-state :default 04000 allow tcp from 203.0.113.40 to me 5656 keep-state :default 04200 allow tcp from 203.0.113.42 to me 5656 keep-state :default 04400 allow tcp from 203.0.113.44 to me 5656 keep-state :default 04600 allow tcp from 203.0.113.46 to me 5656 keep-state :default 04800 allow tcp from 203.0.113.48 to me 5656 keep-state :default 65535 deny ip from any to any
Note that the delete command will operate on comma separated values, but the delete command will only remove the first value in a comma separated list which is usually not what you intend. The command does not throw an error, but it does not delete all the lines requested.
# ipfw delete 3100,3200,3300 # echo $? 0 <--- No error found with the previous command. # # ipfw list 00100 check-state :default 03000 allow tcp from 203.0.113.30 to me 5656 keep-state :default 03200 allow tcp from 203.0.113.32 to me 5656 keep-state :default 03300 allow tcp from 203.0.113.33 to me 5656 keep-state :default 04000 allow tcp from 203.0.113.40 to me 5656 keep-state :default 04200 allow tcp from 203.0.113.42 to me 5656 keep-state :default 04400 allow tcp from 203.0.113.44 to me 5656 keep-state :default 04600 allow tcp from 203.0.113.46 to me 5656 keep-state :default 04800 allow tcp from 203.0.113.48 to me 5656 keep-state :default 65535 deny ip from any to any
The show command is similar to the list command but it also includes a packet count and byte count for each rule.
Stop any existing scripts on the firewall VM and run sh userv3.sh. Then create the following ruleset on the firewall VM:
# ipfw -q flush # # ipfw add 100 check-state 00100 check-state :default # ipfw add 1000 allow udp from 203.0.113.10 to me 5656 01000 allow udp from 203.0.113.10 to me 5656 # ipfw add 2000 allow udp from 203.0.113.10 to me 5657 02000 allow udp from 203.0.113.10 to me 5657 # ipfw add 3000 allow udp from 203.0.113.10 to me 5658 03000 allow udp from 203.0.113.10 to me 5658 # ipfw add 4000 allow udp from 203.0.113.10 to me 5659 04000 allow udp from 203.0.113.10 to me 5659 # # ipfw list 00100 check-state :default 01000 allow udp from 203.0.113.10 to me 5656 02000 allow udp from 203.0.113.10 to me 5657 03000 allow udp from 203.0.113.10 to me 5658 04000 allow udp from 203.0.113.10 to me 5659 65535 deny ip from any to any
Then, on the external1 VM, run sh uconr.sh 5656 1 script to send packets to ports 5656, 5657, and 5658, randomly:
# sh uconr.sh 5656 1 PORT1 = [5656] SLEEPVAL = [1] UDP packet from [203.0.113.10],[5656],[1] UDP packet from [203.0.113.10],[5656],[2] UDP packet from [203.0.113.10],[5658],[3] UDP packet from [203.0.113.10],[5657],[4] UDP packet from [203.0.113.10],[5659],[5] UDP packet from [203.0.113.10],[5659],[6] UDP packet from [203.0.113.10],[5659],[7] UDP packet from [203.0.113.10],[5656],[8] UDP packet from [203.0.113.10],[5658],[9] UDP packet from [203.0.113.10],[5659],[10] UDP packet from [203.0.113.10],[5658],[11] UDP packet from [203.0.113.10],[5656],[12] UDP packet from [203.0.113.10],[5656],[13] UDP packet from [203.0.113.10],[5656],[14] UDP packet from [203.0.113.10],[5659],[15] UDP packet from [203.0.113.10],[5656],[16] UDP packet from [203.0.113.10],[5657],[17] UDP packet from [203.0.113.10],[5659],[18] UDP packet from [203.0.113.10],[5659],[19] UDP packet from [203.0.113.10],[5657],[20] UDP packet from [203.0.113.10],[5659],[21] ^C
Running the ipfw show command outputs:
# ipfw show 00100 0 0 check-state :default 01000 7 494 allow udp from 203.0.113.10 to me 5656 02000 3 212 allow udp from 203.0.113.10 to me 5657 03000 3 211 allow udp from 203.0.113.10 to me 5658 04000 8 565 allow udp from 203.0.113.10 to me 5659 65535 11 897 deny ip from any to any
The output shows the number of packets and the number of bytes processed by each rule, including the default rule which may have processed many more packets.
Note that the -D
command parameter will show counts for dynamic rules similar to the above.
# ipfw -D show Dynamic rules (2 288): 04000 2 100 (1s) STATE tcp 203.0.113.10 31662 <-> 203.0.113.50 5659 :default 03000 8 482 (1s) STATE tcp 203.0.113.10 45732 <-> 203.0.113.50 5658 :default
This is a useful tool for debugging. Paired with the zero command which can clear counters with precise rule selection, it can show what rules are still processing a rule match.
The zero command takes a space separated list of rules (similar to the delete command) to clear counters. However, unlike the delete command, ranges (e.g 2000-3000) are not allowed.
# ipfw zero 2000 3000 # # ipfw show 00100 0 0 check-state :default 01000 7 494 allow udp from 203.0.113.10 to me 5656 02000 0 0 allow udp from 203.0.113.10 to me 5657 03000 0 0 allow udp from 203.0.113.10 to me 5658 04000 8 565 allow udp from 203.0.113.10 to me 5659 65535 11 897 deny ip from any to any
Clearing all rule match counters can be done with ipfw zero with no parameters.
Clearing the default rule match counter can be done with ipfw zero 65535.
Counters are also a feature of rules that specify the log keyword. We will see an example of this when we explore the log and logamount keywords later.
3.3. Keywords
3.3.1. Protocols
Protocols are those defined by IANA - the Internet Assigned Numbers Authority (https://www.iana.org) and are included in Unix systems in /etc/protocols. From this file, we can learn what numbers are assigned to common (and some very obscure) protocols - ip (0), tcp (6), udp (17), icmp (1), and many others.
Source and destination protocols can be the conventional IP or IPv6 addresses. However, the ipfw(8) manual page has this more detailed explanation:
"The first part (proto from src to dst) is for backward compatibility with earlier versions of FreeBSD. In modern FreeBSD any match pattern (including MAC headers, IP protocols, addresses and ports) can be specified in the options section."
The ipfw keywords for common protocols include:
ip4 | ipv4 Matches IPv4 packets.
ip6 | ipv6 Matches IPv6 packets.
ip | all Matches any IP packet.
The logical operator "or" can be use to combine multiple protocols where any one of them applies. The "{" and "}" braces can be used to group "or" conditions (known as "or-blocks"). Only one level of braces can be used. Braces must be escaped with a backslash '\' to prevent them from being interpreted directly by the command line shell:
# ipfw add 1100 deny \{ tcp or udp or eigrp or chaos \} from 1.2.3.4 to 5.6.7.8 01100 deny { tcp or udp or eigrp or chaos } from 1.2.3.4 to 5.6.7.8
Consider this ruleset in a shell script:
#!/bin/sh
ipfw add 5000 deny \{ icmp or ip or igmp or ggp or ipencap or st2 or tcp or cbt or egp or igp or bbn-rcc or nvp or pup or argus or emcon or xnet or chaos or udp or mux or dcn or hmp or prm or xns-idp or trunk-1 or trunk-2 or leaf-1 or leaf-2 or rdp or irtp or iso-tp4 or netblt or mfe-nsp or merit-inp or dccp or 3pc or idpr or xtp or ddp or idpr-cmtp or tp++ or il or ipv6 or sdrp or ipv6-route or ipv6-frag or idrp or rsvp or gre or dsr or bna or esp or ah or i-nlsp or swipe or narp or mobile or tlsp or skip or ipv6-icmp or ipv6-nonxt or ipv6-opts or cftp or sat-expak or kryptolan or rvd or ippc or sat-mon or visa or ipcv or cpnx or cphb or wsn or pvp or br-sat-mon or sun-nd or wb-mon or wb-expak or iso-ip or vmtp or secure-vmtp or vines or ttp or nsfnet-igp or dgp or tcf or eigrp or ospf or sprite-rpc or larp or mtp or ax.25 or ipip or micp or scc-sp or etherip or encap or gmtp or ifmp or pnni or pim or aris or scps or qnx or a/n or ipcomp or snp or compaq-peer or ipx-in-ip or carp or pgm or l2tp or ddx or iatp or stp or srp or uti or smp or sm or ptp or isis or fire or crtp or crudp or sscopmce or iplt or sps or pipe or sctp or fc or rsvp-e2e-ignore or mobility-header or udplite or mpls-in-ip or manet or hip or shim6 or wesp or rohc or pfsync or divert \} from any to me
exit
Note that the above file is shown as one very long line and does not use the Unix line continuation convention.
This command will deny all traffic using all protocols defined in /etc/protocols. The above command will complete successfully. However, due to a bug in the "or-block" parser, you cannot list the "ip" protocol first. If you swap the first two protocols - icmp and ip - the command throws an error.
For example,
# ipfw add 1000 deny \{ igmp or ip or eigrp \} from any to me
works Ok but
# ipfw add 1000 deny \{ ip or igmp or eigrp \} from any to me ipfw: invalid OR block
fails.
The use of the logical "and" operator in a protocol block is an error. A packet can be in only one protocol at a time. However, the use of the logical "not" operator is permitted in front of a protocol identifier:
# ipfw add 1000 deny \{ icmp or not igmp \} from any to me
Careful consideration of all logical conditions is essential to correct operation of a ruleset.
In later versions of FreeBSD, the use of the protocol "or-block" is noted as deprecated in ipfw(8) but the operation may still complete successfully until the feature is removed completely. |
3.3.2. Addresses
Source and destination addresses can be any of the following:
IPv4 or IPv6 addresses
any - matches any IP address. We have already seen many examples of this keyword.
me - matches any IP address configured on an interface in the system.
Note that the interface does not have to have an IP or IPv6 address, nor does it have to be up or even exist at the time the rule is entered. Thus, be aware that a rule with keyword me may affect traffic on interfaces that are configured at a later time. Consider the following system interface list:
# ifconfig -a em0: flags=8863<UP,BROADCAST,RUNNING,SIMPLEX,MULTICAST> metric 0 mtu 1500 options=481209b<RXCSUM,TXCSUM,VLAN_MTU,VLAN_HWTAGGING,VLAN_HWCSUM,WOL_MAGIC,VLAN_HWFILTER,NOMAP> ether 02:49:50:46:57:41 inet 203.0.113.50 netmask 0xffffff00 broadcast 203.0.113.255 media: Ethernet autoselect (1000baseT <full-duplex>) status: active nd6 options=29<PERFORMNUD,IFDISABLED,AUTO_LINKLOCAL> lo0: flags=8049<UP,LOOPBACK,RUNNING,MULTICAST> metric 0 mtu 16384 options=680003<RXCSUM,TXCSUM,LINKSTATE,RXCSUM_IPV6,TXCSUM_IPV6> inet6 ::1 prefixlen 128 inet6 fe80::1%lo0 prefixlen 64 scopeid 0x3 inet 127.0.0.1 netmask 0xff000000 groups: lo
The following attempt to add a rule with a non-existent interface succeeds even though there is no wlan0
interface defined:
# ipfw add 1000 deny tcp from 1.2.4.4 to me via wlan0 01000 deny tcp from 1.2.4.4 to me via wlan0
me6 - matches any IPv6 address similarly to the above me keyword.
table(name[,value]) - matches any IPv4 or IPv6 address for an entry in the named table.
Tables are discussed below in Lookup Tables.
IPv4 and IPv6 addresses follow the usual conventions regarding address and mask or prefix length. Addresses can also be grouped into a list, similar to the capability discussed for protocols above.
# ipfw add 2000 allow tcp from \{ 2.3.4.5/32, 3.4.5.0/24, 10.0.0.0/8 \} to me # ipfw add 2000 allow tcp from \{ 2607:fcc0:0:35::dd/64, 2608:abcd:0:2300::9eac/64 \} to me
However, you cannot mix IPv4 and IPv6 addresses in the same list. Use two different rules instead.
For sparse collections of addresses, consider the alternate form allowed by
addr-set: addr[/masklen]{list}
such as:
# ipfw add 1000 allow tcp from 1.2.3.0/24\{128,9,35-45,7\} to me
In this form, lists and increasing ranges are allowed. ipfw will consolidate overlapping ranges, and will reorder the list in the display to show increasing addresses from left to right. Note that spaces are only allowed after commas between list elements, nowhere else.
This example does not work due to incorrect placement of spaces:
# ipfw add 1000 allow tcp from 1.2.4.0/24\{ 128,9,25-45,7-25 \} to me ipfw: missing ``to''
This example works by correcting where spaces occur on the command line.
# ipfw add 1000 allow tcp from 1.2.4.0/24\{128,9,25-45,7-25\} to me 01000 allow tcp from 1.2.4.0/24{7-45,128} to me
As noted, ipfw will simplify, reorder, and display the list:
# ipfw list 01000 allow tcp from 1.2.4.0/24{7-45,128} to me 65535 deny ip from any to any
Note that ranges must be defined as increasing. Also, as noted in ipfw(8), there is no support for sets of IPv6 addresses.
3.3.3. Ports
Ports may be specified by number or by service name. Service names, also under the provenance of IANA, are found in /etc/services on Unix systems.
Ports can be specified as individual items, lists, or ranges. Typically ports are used to determine a destination service and so apply to the destination address (although specifying source ports is also permitted):
# ipfw add 1000 allow tcp from 1.2.3.4 to me daytime 01000 allow tcp from 1.2.3.4 to me 13 # ipfw add 2000 allow tcp from 2.3.4.5 to me ssh, telnet, smtp 02000 allow tcp from 2.3.4.5 to me 22,23,25 # ipfw add 3000 allow tcp from 3.4.5.6 to me auditd-domain 03000 allow tcp from 3.4.5.6 to me 48-53 # This example uses source and destination ports. # ipfw add 4000 allow tcp from 7.8.9.10 3030 to me ssh 04000 allow tcp from 7.8.9.10 3030 to me 22
General Notes on Port Ranges
A range such as that shown above (
auditd-domain
) may accidentally include ports you do not want. In this case, portstacacs (49)
,re-mail-ck (50)
,la-maint (51)
, andxns-time (52)
would be included. Even if this is really what you want, you should always check actual port numbers when using named ranges for ports.Some service names include the dash character '-' as part of the name, as in the point above. In these cases you will have to escape the dash with a double backslash, one for the shell and one for ipfw:
# ipfw add 4000 allow tcp from 4.5.6.7 to me ftp, ftp\\-data 04000 allow tcp from 4.5.6.7 to me 21,20
Some applications require a range of source and destination ports in both directions. This is easy to accomplish with ranges and a
keep-state
rule:# ipfw add 1000 allow tcp from 203.0.113.10 5200-5205 to me 5656-5658 keep-state 01000 allow tcp from 203.0.113.10 5200-5205 to me 5656-5658 keep-state :default # # ipfw list 01000 allow tcp from 203.0.113.10 5200-5205 to me 5656-5658 keep-state :default 65535 deny ip from any to any
The syntactic sugar provided by the match keywords
dst-port
andsrc-port
are both part of the match section of the rule:# ipfw add 1000 allow tcp from 203.0.113.10 to me src-port 3030 dst-port 1010 01000 allow tcp from 203.0.113.10 3030 to me 1010
To test common ports in both directions we have to manually connect using ncat
so we can set up the source and destination ports as we need them:
# # ipfw list 01000 allow tcp from 203.0.113.10 5200-5205 to me 5656-5658 keep-state :default 65535 deny ip from any to any
Note: the transmission and reception lines have been aligned on each side.
Note that no connection was achieved when the destination port was out of bounds (5659) and when the source port was out of bounds (5206).
3.3.4. Prob
The prob keyword is used to assign a chance, that is a probability (a floating point value between 0 and 1), that an incoming packet will be matched. If the chance is successful, the corresponding rule performs the required action. If the chance is not successful, the rule is not matched and rule processing continues to the next rule.
To test this keyword, we use "sh ucont.sh 5656 1" script on the external1 VM’s side to repeatedly send a UDP packet to the firewall VM, who is listening on UDP port 5656. Using the prob keyword, we set a probability of .5 (a 50% chance) that the packet will be matched. The action is to let the packet pass to the service, which just prints the contents of the packet.
As you can see below, there were 24 out of 50 packets received, very close to 50% for such a small sample:
# ipfw -q flush # # ipfw add 3000 prob 0.5 allow udp from any to me 5656 03000 prob 0.500000 allow udp from any to me 5656 # # ipfw list 03000 prob 0.500000 allow udp from any to me 5656 65535 deny ip from any to any # # sh userv.sh 5656 PORT1 = [5656] Starting UDP listener on [203.0.113.50],[5656] UDP packet from [203.0.113.10],[5656],[5] UDP packet from [203.0.113.10],[5656],[6] UDP packet from [203.0.113.10],[5656],[7] UDP packet from [203.0.113.10],[5656],[10] UDP packet from [203.0.113.10],[5656],[11] UDP packet from [203.0.113.10],[5656],[13] UDP packet from [203.0.113.10],[5656],[14] UDP packet from [203.0.113.10],[5656],[16] UDP packet from [203.0.113.10],[5656],[19] UDP packet from [203.0.113.10],[5656],[20] UDP packet from [203.0.113.10],[5656],[22] UDP packet from [203.0.113.10],[5656],[28] UDP packet from [203.0.113.10],[5656],[29] UDP packet from [203.0.113.10],[5656],[32] UDP packet from [203.0.113.10],[5656],[33] UDP packet from [203.0.113.10],[5656],[34] UDP packet from [203.0.113.10],[5656],[35] UDP packet from [203.0.113.10],[5656],[37] UDP packet from [203.0.113.10],[5656],[38] UDP packet from [203.0.113.10],[5656],[39] UDP packet from [203.0.113.10],[5656],[41] UDP packet from [203.0.113.10],[5656],[43] UDP packet from [203.0.113.10],[5656],[45] UDP packet from [203.0.113.10],[5656],[49] ^C#
3.3.5. Sets
Firewall rules can be grouped into different sets, and you can switch between sets atomically. Why would you want this feature? Consider a datacenter with two sets of identical servers on separate networks. You want to take down one set for maintenance, but first you want to transfer traffic to the other set of servers. Bingo - you have a solution by using sets.
Sets are useful, but they do come with some caveats which we will describe throughout this section.
The default set is set 0. Let’s create rules in set 0 and 1 with slight differences between the two. Note that this example also shows the use of the ipfw comment feature - allowing comments on a per-rule basis.
# ipfw -q flush # # ipfw add 1000 set 0 check-state 01000 check-state :default # # ipfw add 1100 set 0 allow tcp from any to me 5656 setup keep-state // 5656 only 01100 allow tcp from any to me 5656 setup keep-state :default // 5656 only # # ipfw add 2000 set 1 check-state 02000 check-state :default # # ipfw add 2100 set 1 allow tcp from any to me 5657 setup keep-state // 5657 only 02100 allow tcp from any to me 5657 setup keep-state :default // 5657 only # # ipfw set show enable 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30
The above ipfw set show command lists all enabled sets, here showing both 0 and 1 as enabled.
From the external1 VM, commence communications using the tcon.sh script as shown below. The first two communications ([1], [2]) are to port 5656, the next two communications are to port 5657 ([3],[4]). The firewall host shows all four communications received:
Right before communication 5, the firewall host admin disabled set 0, (# ipfw set disable 0) effectively blocking access to port 5656. Disabling set 0, effectively removes the rules for port 5656 and so the communications [5] and [6] fail. The external1 VM goes back to port 5657 for communications [7] and [8], which are successful.
Notice that if you list the firewall ruleset with just ipfw list, only the sets that are enabled actually show up.
If you want to make sure which sets are enabled / disabled, use the -S
flag on the ipfw command as shown below.
# ipfw set enable 0 # # ipfw set show enable 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 # # ipfw -S list 01000 set 0 check-state :default 01100 set 0 allow tcp from any to me 5656 setup keep-state :default // 5656 only 02000 set 1 check-state :default 02100 set 1 allow tcp from any to me 5657 setup keep-state :default // 5657 only 65535 set 31 deny ip from any to any # # ipfw set disable 0 # # ipfw set show disable 0 enable 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 # # ipfw -S list # DISABLED 01000 set 0 check-state :default # DISABLED 01100 set 0 allow tcp from any to me 5656 setup keep-state :default // 5656 only 02000 set 1 check-state :default 02100 set 1 allow tcp from any to me 5657 setup keep-state :default // 5657 only 65535 set 31 deny ip from any to any #
To atomically change from one set to the other use the ipfw set swap command:
# ipfw set swap 0 1
What actually happens is that the rules in the sets get swapped; that is, all the rules in set 0 get put in set 1 and all the rules in set 1 get put in set 0.
# # ipfw -S list # DISABLED 01000 set 0 check-state :default # DISABLED 01100 set 0 allow tcp from any to me 5656 setup keep-state :default // 5656 only 02000 set 1 check-state :default 02100 set 1 allow tcp from any to me 5657 setup keep-state :default // 5657 only 65535 set 31 deny ip from any to any # # ipfw set swap 0 1 # # ipfw -S list 01000 set 1 check-state :default 01100 set 1 allow tcp from any to me 5656 setup keep-state :default // 5656 only # DISABLED 02000 set 0 check-state :default # DISABLED 02100 set 0 allow tcp from any to me 5657 setup keep-state :default // 5657 only 65535 set 31 deny ip from any to any
Note carefully that if you swap sets where one of the sets is disabled, the set number is still disabled after the swap, even though the rules are now different. This can lead to unexpected consequences such as the following:
# ipfw -S list 01000 set 0 check-state :default 01100 set 0 allow tcp from any to me 5656 setup keep-state :default // 5656 only 02000 set 1 check-state :default 02100 set 1 allow tcp from any to me 5660 setup keep-state :default // 5660 only 65535 set 31 deny ip from any to any # # ipfw set disable 0 # # ipfw -S list # DISABLED 01000 set 0 check-state :default # DISABLED 01100 set 0 allow tcp from any to me 5656 setup keep-state :default // 5656 only 02000 set 1 check-state :default 02100 set 1 allow tcp from any to me 5660 setup keep-state :default // 5660 only 65535 set 31 deny ip from any to any # # ipfw set swap 0 1 # # ipfw -S list 01000 set 1 check-state :default 01100 set 1 allow tcp from any to me 5656 setup keep-state :default // 5656 only # DISABLED 02000 set 0 check-state :default # DISABLED 02100 set 0 allow tcp from any to me 5660 setup keep-state :default // 5660 only 65535 set 31 deny ip from any to any
Here set 0 is disabled, and then swapped. After the swap, set 0 is still disabled, though the rules have changed.
Note that all sets are initially enabled. When you disable a set, say set 3, all other sets are still active, even if no rule references them. You can think of sets as the pieces on the back row of a chess board. If you remove a knight or a bishop, that one piece is not able to play, but all the others are able to play.
set 31 cannot be deleted or changed. It can however partially participate in a swap.
# ipfw set swap 1 31
This swap will complete successfully (return code 0), but the effect is not the same as the swaps above. The default rule in set 31 is not swapped, but the set number for the other rules (rules in set 1) are set to 31:
# ipfw -S list 01000 set 1 check-state :default 01100 set 1 allow tcp from 1.2.3.4 to me 8080 01200 set 1 allow tcp from 1.2.3.4 to me 8081 65535 set 31 deny ip from any to any # # ipfw set swap 1 31 # # ipfw -S list 01000 set 31 check-state :default 01100 set 31 allow tcp from 1.2.3.4 to me 8080 01200 set 31 allow tcp from 1.2.3.4 to me 8081 65535 set 31 deny ip from any to any
As noted in the ipfw(8) manual page, rules in set 31 cannot be flushed. We now have 4 rules in set 31:
# ipfw -f flush Flushed all rules. # # ipfw -S list 01000 set 31 check-state :default 01100 set 31 allow tcp from 1.2.3.4 to me 8080 01200 set 31 allow tcp from 1.2.3.4 to me 8081 65535 set 31 deny ip from any to any
We cannot use ipfw flush to clean out non-default rules, but we can use ipfw delete set 31 to clean out all but the default rule, which cannot be changed:
# ipfw delete set 31 # # ipfw -S list 65535 set 31 deny ip from any to any
Further, you should note that any set you disable remains disabled after a flush. Thus, if you disable a set and then flush the ruleset, any rules added back into that set will still be disabled. This includes set 0, the default set.
Consider the following:
# ipfw list 01000 check-state :default 01100 allow tcp from any to me 1111 02000 allow tcp from any to me 2222 03000 allow tcp from any to me 3333 65535 deny ip from any to any # # ipfw set disable 0 # # ipfw -S list # DISABLED 01000 set 0 check-state :default # DISABLED 01100 set 0 allow tcp from any to me 1111 # DISABLED 02000 set 0 allow tcp from any to me 2222 # DISABLED 03000 set 0 allow tcp from any to me 3333 65535 set 31 deny ip from any to any # # ipfw -f flush Flushed all rules. # # ipfw add 100 check-state 00100 check-state :default # ipfw add 200 allow tcp from any to me 5555 00200 allow tcp from any to me 5555 # ipfw add 300 allow tcp from any to me 6666 00300 allow tcp from any to me 6666 # ipfw add 400 allow tcp from any to me 7777 00400 allow tcp from any to me 7777 # # ipfw -S list # DISABLED 00100 set 0 check-state :default # DISABLED 00200 set 0 allow tcp from any to me 5555 # DISABLED 00300 set 0 allow tcp from any to me 6666 # DISABLED 00400 set 0 allow tcp from any to me 7777 65535 set 31 deny ip from any to any #
Because set 0 was disabled before the flush, the flush has no effect on the enable/disable state of that set.
Note that you can even disable sets that do not yet exist:
# kldload ipfw ipfw2 (+ipv6) initialized, divert loadable, nat loadable, default to deny, logging disabled # # ipfw -S list 65535 set 31 deny ip from any to any # # ipfw set show enable 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 # # ipfw set disable 4 5 6 7 8 9 # # ipfw set show disable 4 5 6 7 8 9 enable 0 1 2 3 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 #
Using sets can be very helpful, as long as you understand their properties and limitations.
3.3.6. Tags
Tags allow for marking incoming packets in such a way that later rules can be applied based on the tag.
For a simple example, consider tagging incoming packets from different networks. Later rules determine if the tagged packets are allowed or denied:
# ipfw add 100 check-state 00100 check-state :default # # ipfw add 1000 count tag 10 tcp from 172.16.200.0/24 to me 5656 01000 count tag 10 tcp from 172.16.200.0/24 to me 5656 # # ipfw add 1100 count tag 20 tcp from 172.16.225.0/24 to me 5656 01100 count tag 20 tcp from 172.16.225.0/24 to me 5656 # # ipfw add 1200 count tag 30 tcp from 203.0.113.0/24 to me 5656 01200 count tag 30 tcp from 203.0.113.0/24 to me 5656 # # ipfw add 3000 allow tcp from any to me tagged 30 setup keep-state 03000 allow tcp from any to me tagged 30 setup keep-state :default # # ipfw add 4000 deny tcp from any to me tagged 10,20 04000 deny tcp from any to me tagged 10,20 #
We can test this using the ncat(1) option to set its own source address.
To do this we first setup two alias address on the em0
interface on the external1 VM:
# ifconfig em0 172.16.200.10/24 alias # ifconfig em0 172.16.225.10/24 alias # # ifconfig em0 em0: flags=8843<UP,BROADCAST,RUNNING,SIMPLEX,MULTICAST> metric 0 mtu 1500 options=81209b<RXCSUM,TXCSUM,VLAN_MTU,VLAN_HWTAGGING,VLAN_HWCSUM,WOL_MAGIC,VLAN_HWFILTER> ether 02:49:50:46:57:10 inet 203.0.113.10 netmask 0xffffff00 broadcast 203.0.113.255 inet 172.16.200.10 netmask 0xffffff00 broadcast 172.16.200.255 inet 172.16.225.10 netmask 0xffffff00 broadcast 172.16.225.255 media: Ethernet autoselect (1000baseT <full-duplex>) status: active nd6 options=29<PERFORMNUD,IFDISABLED,AUTO_LINKLOCAL>
Because of the rule tagging in this ruleset, only traffic tagged with value "30" is allowed to pass.
Tags, combined with lookup tables allow for powerful policy based network access.
3.3.7. Logging
ipfw supports two methods of logging:
3.3.7.1. Method 1 – using ipfw0, the IPFW pseudointerface
# kldload ipfw # ifconfig ipfw0 create
Note that you must have the ipfw.ko kernel module loaded before you can create the ipfw0 interface. Also, note that if you unload ipfw.ko, the interface is destroyed and is no longer available.
Why use the ipfw0 interface?
It is possible to read logs in real time with programs such as tcpdump(1), wireshark(1), or other network monitoring programs.
An example is given further below.
3.3.7.2. Method 2 – use syslogd
Setting the sysctl variable net.inet.ip.fw.verbose = 1
will instruct the firewall to log packets to syslogd(8) even when the ipfw0 interface exists.
Syslogd must be configured via /etc/syslog.conf.
ipfw packets will be logged with a LOG_SECURITY facility.
The logging limit is configurable via net.inet.ip.fw.verbose_limit
, which is set to 0 (unlimited) by default.
Why use syslog?
Of the two methods, it is the only one that processes count actions, and is also the only one that prints rule numbers with the log entry.
To test logging, create a rule with the log keyword:
# ipfw add 1000 allow log tcp from any to me 5656 setup keep-state
Counters are also a feature of rules that specify the log keyword. If we replace the above ruleset and add the log keyword, matches for all rules will be included in the log entries with associated counts.
With the ruleset changes to add the log keyword, our ruleset looks like this:
# ipfw list 00100 check-state :default 01000 allow log tcp from 203.0.113.10 to me 5656 keep-state :default 02000 allow log tcp from 203.0.113.10 to me 5657 keep-state :default 03000 allow log tcp from 203.0.113.10 to me 5658 keep-state :default 04000 allow log tcp from 203.0.113.10 to me 5659 keep-state :default 65535 deny ip from any to any
3.3.7.3. Using Method 1
Using Method 1, we capture/view logs with:
# tcpdump -i ipfw0 -X
Note: You can redirect binary data to a 'savefile' with the -w file
option.
If you are just reading tcpdump output, experiment with the -v, -vv, and -vvv options, which gives increasingly more verbose output.
The example below examines traffic with the ucont.sh on the external1 VM and the userv3.sh script on the firewall VM.
Logged traffic from the above communication appears on the log device ipfw0:
# # tcpdump -i ipfw0 -X -vvv tcpdump: WARNING: ipfw0: That device doesn't support promiscuous mode (BIOCPROMISC: Invalid argument) tcpdump: listening on ipfw0, link-type EN10MB (Ethernet), capture size 262144 bytes 20:50:44.259127 IP (tos 0x0, ttl 64, id 61929, offset 0, flags [none], proto UDP (17), length 70) 203.0.113.10.27337 > 203.0.113.50.5656: [udp sum ok] UDP, length 42 0x0000: 4500 0046 f1e9 0000 4011 1c61 ac10 0a0a E..F....@..a.... 0x0010: ac10 0a32 6ac9 1618 0032 00ff 5544 5020 ...2j....2..UDP. 0x0020: 7061 636b 6574 2066 726f 6d20 5b31 3732 packet.from.[172 0x0030: 2e31 362e 3130 2e31 305d 2c5b 3536 3536 .16.10.10],[5656 0x0040: 5d2c 5b31 5d0a ],[1]. 20:50:44.581025 IP (tos 0x0, ttl 64, id 61930, offset 0, flags [none], proto UDP (17), length 70) 203.0.113.10.41914 > 203.0.113.50.5656: [udp sum ok] UDP, length 42 0x0000: 4500 0046 f1ea 0000 4011 1c60 ac10 0a0a E..F....@..`.... 0x0010: ac10 0a32 a3ba 1618 0032 c80c 5544 5020 ...2.....2..UDP. 0x0020: 7061 636b 6574 2066 726f 6d20 5b31 3732 packet.from.[172 0x0030: 2e31 362e 3130 2e31 305d 2c5b 3536 3536 .16.10.10],[5656 0x0040: 5d2c 5b32 5d0a ],[2]. 20:50:45.960845 IP (tos 0x0, ttl 64, id 61931, offset 0, flags [none], proto UDP (17), length 70) 203.0.113.10.33126 > 203.0.113.50.5656: [udp sum ok] UDP, length 42 0x0000: 4500 0046 f1eb 0000 4011 1c5f ac10 0a0a E..F....@.._.... 0x0010: ac10 0a32 8166 1618 0032 ea5f 5544 5020 ...2.f...2._UDP. 0x0020: 7061 636b 6574 2066 726f 6d20 5b31 3732 packet.from.[172 0x0030: 2e31 362e 3130 2e31 305d 2c5b 3536 3536 .16.10.10],[5656 0x0040: 5d2c 5b33 5d0a ],[3]. ^C
3.3.7.4. Using Method 2
Using Method 2, first examine /etc/syslog.conf to see if there is already a facility and level for security listed. In modern versions of FreeBSD it is common to see:
security.* /var/log/security
ipfw creates logs with the LOG_SECURITY facility and sends output to the file /var/log/security in this case.
If this entry exists on your system, you are all set, otherwise read through the below and set up your own entry for an ipfw logfile.
To log to syslogd(8), add the following line to the end of /etc/syslog.conf:
security.info /tmp/ipfw.info
(FreeBSD syslog.conf allows tabs or spaces to be used.)
Create the logfile with
# touch /tmp/ipfw.info
then send a HANGUP
signal to the syslogd
daemon:
# kill -HUP <pid of syslogd>
You can use tail -f to see logs in real time.
Note that ipfw only logs matched rules with this method.
Note that ipfw logs to the syslog .info level and the .debug level.
# tail -f /var/log/security Apr 3 14:50:12 firewall newsyslog[401]: logfile first created Apr 9 21:05:13 firewall kernel: ipfw: 1000 Accept UDP 203.0.113.10:59203 203.0.113.50:5656 in via em0 Apr 9 21:05:15 firewall kernel: ipfw: 1000 Accept UDP 203.0.113.10:12401 203.0.113.50:5656 in via em0 Apr 9 21:05:16 firewall kernel: ipfw: 1000 Accept UDP 203.0.113.10:45319 203.0.113.50:5656 in via em0 ^C
The log entry includes the date, time, host, service, and rule number (1000 above) to make it easy to track which rule is being matched.
General Notes on logging
Consider the following ruleset:
# ipfw list 00100 check-state :default 01000 allow tcp from 203.0.113.100 to me setup keep-state :default 02000 allow icmp from 203.0.113.100 to me 02100 allow icmp from me to 203.0.113.100 03000 allow log udp from 203.0.113.10 to me 5656 04000 allow log logamount 20 udp from 203.0.113.10 to me 5657 05000 allow log logamount 20 udp from 203.0.113.10 to me 5658 65535 deny ip from any to any
Using Method 2 (syslog), on a quiet system, you may notice that the entries do not appear right away if you are reading the security log file in real time (for example, tail -f /var/log/security
).
This is because syslog will buffer identical lines and output a notification only occasionally as in the below example:
Mar 28 22:30:01 firewall kernel: ipfw: 3000 Accept UDP 203.0.113.10:27519 203.0.113.50:5656 in via em0 Mar 28 22:30:03 firewall syslogd: last message repeated 4 times Mar 28 22:32:31 firewall syslogd: last message repeated 31 times
Also, Method 1 (using the ipfw0 interface) and Method 2 (syslog) are mutually exclusive.
You cannot have both active at the same time.
If net.inet.ip.fw.verbose=0
, the output will be to the ipfw0 interface; if the value is 1, the log output will be to syslog.
logamount values in rules only apply to Method 2 - syslog
.
They have no effect on limiting the number of packets sent out the ipfw0 interface.
In Method 2 - syslog
, when the log limit is reached, ipfw will send a notification similar to the following into the designated security logging file (default: /var/log/security):
Mar 28 23:00:10 firewall kernel: ipfw: 5000 Accept UDP 203.0.113.10:63367 203.0.113.50:5658 in via em0 Mar 28 23:00:11 firewall kernel: ipfw: 5000 Accept UDP 203.0.113.10:30909 203.0.113.50:5658 in via em0 Mar 28 23:00:11 firewall kernel: ipfw: limit 20 reached on entry 5000
And, at the same time, it also conveniently sends the same message to /var/log/messages, the standard FreeBSD logfile:
Mar 28 23:00:11 firewall kernel: ipfw: limit 20 reached on entry 5000
After that notification is sent, no more syslog entries will be sent from the matching rule until the log counters are reset with:
# ipfw resetlog <rule number>
When the resetlog command is entered, ipfw will send a reset notification to syslog:
Mar 28 23:04:51 firewall kernel: ipfw: logging count reset.
Unfortunately, as of FreeBSD version 14.1, it does not tell you which rule the count was reset for. Presumably, you should know which rule since you just entered the command, at least if you are the only person configuring the ruleset. In any case, you may have to keep track of that manually when working with many rules that include the log keyword.
If you issue an ipfw resetlog command without specifying a rule number, all counters in all rules are reset and ipfw sends the following notification:
Mar 28 23:08:52 firewall kernel: ipfw: All logging counts reset.
Finally, note that the sysctl variable net.inet.ip.fw.verbose_limit
provides a "default limit" if one is not specified with the logamount keyword in the ruleset.
Consider this scenario:
# sysctl net.inet.ip.fw.verbose_limit net.inet.ip.fw.verbose_limit: 5 <---- The limit is preset to 5 # ipfw list 65535 deny ip from any to any
As new rules are added, ipfw will apply any logamount value it finds in the body of a rule.
If a rule being entered does not have a logamount entry, the value defaults to the current net.inet.ip.fw.verbose_limit
amount.
# ipfw add 100 check-state 00100 check-state :default # # ipfw add 3000 allow log udp from 203.0.113.10 to me 5656 03000 allow log logamount 5 udp from 203.0.113.10 to me 5656 # # ipfw add 4000 allow log logamount 20 udp from 203.0.113.10 to me 5657 04000 allow log logamount 20 udp from 203.0.113.10 to me 5657 # # ipfw list 00100 check-state :default 03000 allow log logamount 5 udp from 203.0.113.10 to me 5656 04000 allow log logamount 20 udp from 203.0.113.10 to me 5657 65535 deny ip from any to any
If the sysctl for net.inet.ip.fw.verbose_limit
is changed after the rule is entered, it has no effect:
# sysctl net.inet.ip.fw.verbose_limit=3 net.inet.ip.fw.verbose_limit: 5 -> 3
and later in /var/log/messages
Mar 29 11:07:02 firewall kernel: ipfw: limit 5 reached on entry 3000 ... Mar 29 11:07:24 firewall kernel: ipfw: limit 20 reached on entry 4000
3.3.8. Reset
The reset keyword sends an immediate TCP reset on a rule match containing that keyword. This immediately shuts down any TCP connection from the source matching the rule.
# ipfw list 00100 check-state :default 01000 allow log tcp from 203.0.113.10 to me 5656 setup keep-state :default 02000 reset log tcp from 203.0.113.10 to me 5657 03000 reset log udp from 203.0.113.10 to me 5658 65535 deny ip from any to any
The syslog view of a TCP reset rule match looks like this:
Apr 9 21:44:49 firewall kernel: ipfw: 1000 Accept TCP 203.0.113.50:5656 203.0.113.10:28218 out via em0 Apr 9 21:44:49 firewall syslogd: last message repeated 1 times Apr 9 21:44:49 firewall kernel: ipfw: 1000 Accept TCP 203.0.113.10:28218 203.0.113.50:5656 in via em0 Apr 9 21:45:01 firewall kernel: ipfw: 2000 Reset TCP 203.0.113.10:12998 203.0.113.50:5657 in via em0 Apr 9 21:45:07 firewall kernel: ipfw: 2000 Reset TCP 203.0.113.10:13782 203.0.113.50:5657 in via em0
When sysctl net.inet.ip.fw.verbose=0
, there is no discernible output on ipfw0 for a reset action.
A TCP SYN packet arrives and that is all that is displayed.
To actually witness the reset, run tcpdump(8) on the specific interface:
# tcpdump -i em0 -X -vvv tcpdump: listening on em0, link-type EN10MB (Ethernet), capture size 262144 bytes 21:53:39.376825 IP (tos 0x0, ttl 64, id 0, offset 0, flags [DF], proto TCP (6), length 60) 203.0.113.10.32945 > 203.0.113.50.5657: Flags [S], cksum 0x68fa (correct), seq 1926269947, win 65535, options [mss 1460,nop,wscale 6,sackOK,TS val 648984165 ecr 0], length 0 0x0000: 4500 003c 0000 4000 4006 ce5f ac10 0a0a E..<..@.@.._.... 0x0010: ac10 0a32 80b1 1619 72d0 8bfb 0000 0000 ...2....r....... 0x0020: a002 ffff 68fa 0000 0204 05b4 0103 0306 ....h........... 0x0030: 0402 080a 26ae b665 0000 0000 ....&..e.... 21:53:39.377143 IP (tos 0x10, ttl 64, id 1965, offset 0, flags [none], proto TCP (6), length 40) 203.0.113.50.5657 > 203.0.113.10.32945: Flags [R.], cksum 0xaddc (correct), seq 0, ack 1926269948, win 0, length 0 0x0000: 4510 0028 07ad 0000 4006 06b7 ac10 0a32 E..(....@......2 0x0010: ac10 0a0a 1619 80b1 0000 0000 72d0 8bfc ............r... 0x0020: 5014 0000 addc 0000 P....... ^C
A UDP rule containing the reset keyword just drops the packet. Nothing is sent back to the source address. If the log keyword is also used on the rule, a log entry is generated for syslog (if enabled):
Apr 9 21:58:49 firewall kernel: ipfw: 3000 Reset UDP 203.0.113.10:56503 203.0.113.50:5658 in via em0
3.3.9. Tee
The tee rule requires a divert(4) socket set up beforehand. Refer to the divert rule covered below for setting up the socket. Once the socket is set up, the tee keyword works like divert except that it is not interested in any packet return. It is simply copying the packet to the socket. Processing continues with the next rule.
In essence, tee allows the packet to be sent to userspace for any purpose desired - monitoring, copying, counting - whatever.
# ipfw add 1000 tee 700 ip from any to me
3.3.10. Unreach
The unreach keyword directs ipfw to respond back to the source when packets arrive with a destination port that is not opened by any service. ipfw sends an ICMP reply with the code set to the keyword parameter. This works for any IP protocol.
Because ipfw sends an ICMP packet back to the source, the ruleset must allow outbound ICMP.
Consider the following ruleset:
# ipfw -a list 00100 0 0 allow icmp from me to any 01000 0 0 unreach 100 log udp from any to me 5656 02000 0 0 unreach 200 log tcp from any to me 5657 03000 0 0 unreach 250 log ip from any to me 5658
The counters are zero when the external1 VM sends its packet, a UDP packet destined for port 5656, for which no service is currently set up.
ipfw matches this with rule 1000 and sends an ICMP unreachable notice with code 100 (an arbitrary value, but see the list in ipfw(8)). The offending packet is encapsulated in the data portion of the ICMP reply:
# tcpdump -i bridge0 -X -vvv tcpdump: listening on bridge0, link-type EN10MB (Ethernet), capture size 262144 bytes 14:55:59.666002 IP (tos 0x0, ttl 64, id 25431, offset 0, flags [none], proto UDP (17), length 70) 203.0.113.10.36146 > 203.0.113.50.5656: [udp sum ok] UDP, length 42 0x0000: 4500 0046 6357 0000 4011 aaf3 ac10 0a0a E..FcW..@....... 0x0010: ac10 0a32 8d32 1618 0032 de95 5544 5020 ...2.2...2..UDP. 0x0020: 6174 7461 636b 2066 726f 6d20 5b31 3732 communication.from.[172 0x0030: 2e31 362e 3130 2e31 305d 2c5b 3536 3536 .16.10.10],[5656 0x0040: 5d2c 5b31 5d0a ],[1].
The results for a TCP unreachable are almost the same. The ICMP packet encapsulates the SYN packet in the data portion of the reply.
Here is a view of an ICMP Reply to unreachable TCP port 5657:
# tcpdump -i bridge0 -X -vvv tcpdump: listening on bridge0, link-type EN10MB (Ethernet), capture size 262144 bytes 15:24:03.663104 IP (tos 0x0, ttl 64, id 0, offset 0, flags [DF], proto TCP (6), length 60) 203.0.113.10.58575 > 203.0.113.50.5657: Flags [S], cksum 0x092c (correct), seq 1062429515, win 65535, options [mss 1460,nop,wscale 6,sackOK,TS val 3566166113 ecr 0], length 0 0x0000: 4500 003c 0000 4000 4006 ce5f ac10 0a0a E..<..@.@..-.... 0x0010: ac10 0a32 e4cf 1619 3f53 634b 0000 0000 ...2....?ScK.... 0x0020: a002 ffff 092c 0000 0204 05b4 0103 0306 .....,.......... 0x0030: 0402 080a d48f 6061 0000 0000 ......`a.... 15:24:03.664168 IP (tos 0x0, ttl 64, id 37717, offset 0, flags [none], proto ICMP (1), length 88) 203.0.113.50 > 203.0.113.10: ICMP # 200 203.0.113.50 unreachable, length 68 IP (tos 0x0, ttl 64, id 0, offset 0, flags [DF], proto TCP (6), length 60) 203.0.113.10.58575 > 203.0.113.50.5657: Flags [S], cksum 0x092c (correct), seq 1062429515, win 65535, options [mss 1460,nop,wscale 6,sackOK,TS val 3566166113 ecr 0], length 0 0x0000: 4500 0058 9355 0000 4001 7af3 ac10 0a32 E..X.U..@.z....2 0x0010: ac10 0a0a 03c8 68c3 0000 0000 4500 003c ......h.....E..< 0x0020: 0000 4000 4006 ce5f ac10 0a0a ac10 0a32 ..@.@.._.......2 0x0030: e4cf 1619 3f53 634b 0000 0000 a002 ffff ....?ScK........ 0x0040: 092c 0000 0204 05b4 0103 0306 0402 080a .,.............. 0x0050: d48f 6061 0000 0000 ..`a....
3.3.11. Setdscp
The setdscp action directs ipfw to set an IP header option on outbound packets. The action has no effect on inbound packets. The header option, formerly known as the "Type of Service" (ToS) option, now defines several classes of differentiated services (DiffServ) per several RFCs - RFC 2474, RFC 3168, and RFC 3260.
These service classes such as "Network Control", "Telephony", "Multimedia Conferencing", "Broadcast Video", "Low-latency Data", etc. all require their packets to receive special handling in the network. This is achieved by inserting "code points" - numerical values in the packet header - that define each class.
Firewalls, routers, switches, and other network devices interpret these values and, in theory, service the packets according to their class. See this Wikipedia article on Differentiated Services: https://en.wikipedia.org/wiki/Differentiated_services.
In practice, support for service classes vary among network operators. Check the man page for a list of code points settable by ipfw.
The example below sets the DSCP value to "af31", a codepoint in the "Multimedia Streaming" class.
# ipfw add 2000 setdscp af31 udp from me to any 5656
Note that rule processing continues to the next rule.
Note also that the DSCP value takes up only a partial byte in the IP header, sharing it with two bits of ECN (Explicit Congestion Notification). The binary value for "af31" is 011010nn, where 'nn' are the two bits for ECN. If no ECN, the value resolves to 0x68 (104 decimal).
An outbound traffic example, generated by ncat -u 203.0.113.10 5656 from the firewall VM is shown as received by the external1 VM:
# tcpdump -i em0 -X -vvv tcpdump: listening on em0, link-type EN10MB (Ethernet), capture size 262144 bytes 10:14:22.173252 IP (tos 0x68, ttl 64, id 30816, offset 0, flags [none], proto UDP (17), length 57) 203.0.113.50.17767 > 203.0.113.10.5656: [udp sum ok] UDP, length 29 0x0000: 4568 0039 7860 0000 4011 958f ac10 0a32 Eh.9x`..@......2 0x0010: ac10 0a0a 4567 1618 0025 0f5b 4772 6565 ....Eg...%.[Gree 0x0020: 7469 6e67 7320 6672 6f6d 2074 6865 2066 tings.from.the.f 0x0030: 6972 6577 616c 6c2e 0a irewall..
Diffserve codepoints can be set on any IP based protocol or restricted to selected protocols and/or ports through suitable rules.
3.3.12. Skipto
The skipto action directs the firewall engine to pass over any rules less than the skipto parameter number. If an early rule can match a packet characteristic such as an address, port, TCP or UDP header option or similar, a skipto rule can jump to a potentially much later section of the firewall ruleset to handle to packet.
Consider the following (contrived) ruleset:
# ipfw add 100 check-state # ipfw add 1000 allow tcp from me to any established keep-state # ipfw add 2000 allow tcp from 203.0.113.10 to me 4500 setup keep-state # ipfw add 3000 allow tcp from 203.0.113.10 to me 4502 setup keep-state # ipfw add 4000 allow tcp from 203.0.113.10 to me 4504 setup keep-state # ipfw add 5000 allow tcp from 203.0.113.10 to me 4506 setup keep-state # ipfw add 6000 allow tcp from 203.0.113.10 to me 4508 setup keep-state # ipfw add 7000 allow tcp from 203.0.113.10 to me 4510 setup keep-state # ipfw add 8000 allow tcp from 203.0.113.10 to me 4512 setup keep-state # ipfw add 9000 allow tcp from 203.0.113.10 to me 4512 setup keep-state # ipfw add 10000 allow tcp from 203.0.113.10 to me 5656 setup keep-state
With the external1 VM using our standard tcon.sh 5656 TCP connection script, ipfw has to traverse the entire firewall ruleset, checking each rule in turn for a match. (When testing this ruleset, ensure that the firewall VM is running the appropriate service script such as tserv3.sh.)
By placing a skipto action rule after the check-state action, we can have ipfw jump directly to the rule we care about:
# ipfw add 100 check-state # ipfw add 500 skipto 10000 tcp from 203.0.113.10 to me 5656 # ipfw add 1000 allow tcp from me to any established keep-state # ipfw add 2000 allow tcp from 203.0.113.10 to me 4500 setup keep-state # ipfw add 3000 allow tcp from 203.0.113.10 to me 4502 setup keep-state # ipfw add 4000 allow tcp from 203.0.113.10 to me 4504 setup keep-state # ipfw add 5000 allow tcp from 203.0.113.10 to me 4506 setup keep-state # ipfw add 6000 allow tcp from 203.0.113.10 to me 4508 setup keep-state # ipfw add 7000 allow tcp from 203.0.113.10 to me 4510 setup keep-state # ipfw add 8000 allow tcp from 203.0.113.10 to me 4512 setup keep-state # ipfw add 9000 allow tcp from 203.0.113.10 to me 4512 setup keep-state # ipfw add 10000 allow tcp from 203.0.113.10 to me 5656 setup keep-state
You can determine if your skipto action is working by using the -a
command line parameter (or use ipfw show):
# ipfw -a list 00100 0 0 check-state :default 00500 1 60 skipto 10000 tcp from 203.0.113.10 to me 5656 01000 0 0 allow tcp from me to any established keep-state :default 02000 0 0 allow tcp from 203.0.113.10 to me 4500 setup keep-state :default 03000 0 0 allow tcp from 203.0.113.10 to me 4502 setup keep-state :default 04000 0 0 allow tcp from 203.0.113.10 to me 4504 setup keep-state :default 05000 0 0 allow tcp from 203.0.113.10 to me 4506 setup keep-state :default 06000 0 0 allow tcp from 203.0.113.10 to me 4508 setup keep-state :default 07000 0 0 allow tcp from 203.0.113.10 to me 4510 setup keep-state :default 08000 0 0 allow tcp from 203.0.113.10 to me 4512 setup keep-state :default 09000 0 0 allow tcp from 203.0.113.10 to me 4514 setup keep-state :default 10000 8 474 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default 65535 0 0 deny ip from any to any
General notes on skipto:
The skipto action does not allow negative numbers or zero as a parameter.
skipto does allow numbers greater than 65536. What happens is the system accepts the input, but the result is the integer remainder of the number modulo 65536. You should ensure the value you enter is the value displayed by the command output. Numbers greater than 65534 should not be used.
You can use the skipto action to skip between sets. However, if the set containing the skipto target is disabled, processing continues with the next rule in any set that is enabled.
For example, if you have three sets - 0, 1, and 2, with a disabled set 1 containing the destination of the skipto action, processing will continue with the next rule. See the below ruleset and counters:
# ipfw -Sa list 00100 0 0 set 0 check-state :default 00101 1 60 set 0 skipto 2000 tcp from 203.0.113.10 to me 00120 0 0 set 0 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default 00150 0 0 set 0 allow tcp from me to any established keep-state :default 01000 0 0 set 0 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default # DISABLED 01200 0 0 set 1 allow tcp from 203.0.113.10 to me 6500 setup keep-state :default # DISABLED 01800 0 0 set 1 allow tcp from 203.0.113.10 to me 6512 setup keep-state :default # DISABLED 01900 0 0 set 1 allow tcp from 203.0.113.10 to me 6514 setup keep-state :default # DISABLED 02000 0 0 set 1 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default 02700 0 0 set 2 allow tcp from 203.0.113.10 to me 7510 setup keep-state :default 02800 0 0 set 2 allow tcp from 203.0.113.10 to me 7512 setup keep-state :default 02900 0 0 set 2 allow tcp from 203.0.113.10 to me 7514 setup keep-state :default 03000 8 474 set 2 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default 65535 0 0 set 31 deny ip from any to any
If you skipto a rule number that has multiple rules, the first matching rule at or after that number is executed:
# ipfw -Sa list 00100 0 0 set 0 check-state :default 00101 1 60 set 0 skipto 2000 tcp from 203.0.113.10 to me 00120 0 0 set 0 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default 00150 0 0 set 0 allow tcp from me to any established keep-state :default 01600 0 0 set 1 allow tcp from 203.0.113.10 to me 6508 setup keep-state :default 01700 0 0 set 1 allow tcp from 203.0.113.10 to me 6510 setup keep-state :default 02000 0 0 set 1 allow tcp from 203.0.113.10 to me 6512 setup keep-state :default 02000 0 0 set 1 allow tcp from 203.0.113.10 to me 6514 setup keep-state :default 02000 0 0 set 2 allow tcp from 203.0.113.10 to me 7500 setup keep-state :default 02000 0 0 set 2 allow tcp from 203.0.113.10 to me 7502 setup keep-state :default 02000 0 0 set 2 allow tcp from 203.0.113.10 to me 7504 setup keep-state :default 02500 0 0 set 2 allow tcp from 203.0.113.10 to me 7506 setup keep-state :default 02600 0 0 set 2 allow tcp from 203.0.113.10 to me 7508 setup keep-state :default 02700 0 0 set 2 allow tcp from 203.0.113.10 to me 7510 setup keep-state :default 02800 0 0 set 2 allow tcp from 203.0.113.10 to me 7512 setup keep-state :default 02900 0 0 set 2 allow tcp from 203.0.113.10 to me 7514 setup keep-state :default 03000 10 567 set 2 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default 65535 0 0 set 31 deny ip from any to any
You can enter a rule with a skipto rule number that is lower than the current rule number, attempting to go backward in the ruleset, but this has no effect, and processing continues with the next rule:
# ipfw -Sa list 00100 0 0 set 0 check-state :default 00101 1 60 set 0 skipto 1000 tcp from 203.0.113.10 to me 00120 0 0 set 0 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default 00150 0 0 set 0 allow tcp from me to any established keep-state :default 00200 0 0 set 0 allow tcp from 203.0.113.10 to me 4500 setup keep-state :default 00300 0 0 set 0 allow tcp from 203.0.113.10 to me 4502 setup keep-state :default 00400 0 0 set 0 allow tcp from 203.0.113.10 to me 4504 setup keep-state :default 00500 0 0 set 0 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default 00600 0 0 set 0 allow tcp from 203.0.113.10 to me 4508 setup keep-state :default 00700 0 0 set 0 allow tcp from 203.0.113.10 to me 4512 setup keep-state :default 00800 0 0 set 0 allow tcp from 203.0.113.10 to me 4514 setup keep-state :default 01000 1 60 set 0 skipto 500 tcp from 203.0.113.10 to me 01100 8 475 set 0 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default 65535 0 0 set 31 deny ip from any to any
You can (but probably should not) skipto a skipto rule:
00100 0 0 set 0 check-state :default 00101 1 60 set 0 skipto 1000 tcp from 203.0.113.10 to me 00120 0 0 set 0 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default 00150 0 0 set 0 allow tcp from me to any established keep-state :default 00200 0 0 set 0 allow tcp from 203.0.113.10 to me 4500 setup keep-state :default 00300 0 0 set 0 allow tcp from 203.0.113.10 to me 4502 setup keep-state :default 00400 0 0 set 0 allow tcp from 203.0.113.10 to me 4504 setup keep-state :default 00500 0 0 set 0 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default 00600 0 0 set 0 allow tcp from 203.0.113.10 to me 4508 setup keep-state :default 00700 0 0 set 0 allow tcp from 203.0.113.10 to me 4510 setup keep-state :default 00800 0 0 set 0 allow tcp from 203.0.113.10 to me 4512 setup keep-state :default 00900 0 0 set 0 allow tcp from 203.0.113.10 to me 4514 setup keep-state :default 01000 1 60 set 0 skipto 1500 tcp from 203.0.113.10 to me 01000 0 0 set 0 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default 01500 1 60 set 0 skipto 2000 tcp from 203.0.113.10 to me 01600 0 0 set 0 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default 02000 1 60 set 0 skipto 2500 tcp from 203.0.113.10 to me 02100 0 0 set 0 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default 02500 1 60 set 0 skipto 3000 tcp from 203.0.113.10 to me 02600 0 0 set 0 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default 03000 8 475 set 0 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default 65535 0 0 set 31 deny ip from any to any
See the section on Lists for additional caveats.
3.3.13. Divert
The divert mechanism in ipfw allows you to pull packets into user space for programmatic purposes.
The divert rule snatches the packet and presents it to a divert(4) socket, a special socket type that can be created from an external program.
See the divert.c program at the end of Appendix B for the sample program used for this book.
Copy the divert.c
program onto the firewall VM and compile it with this command:
# make divert LDFLAGS=-lutil
The code should compile cleanly. If it does not, examine the file closely to ensure it was copied correctly and retry the above command.
Divert sockets can be used as the basis for many specialized applications such as packet examination, in-flight packet modification, experimental routing techniques, etc. The program shown here simply reads from the socket and dumps the contents of the packet in hex and ASCII. It then writes the packet back into the divert socket.
To work with the divert keyword, the divert(4) packet diversion mechanism has to be compiled into the kernel or loaded at runtime:
# kldload ipfw # kldload ipdivert
This loads the ipfw firewall kernel module and the ipdivert kernel module which provides divert(4) functionality.
Once this is done, an application can open a divert(4) socket and process packets.
# ./divert Opening divert on port 700 ... (see below)
In another window, run netstat(1) to see the divert socket:
# netstat -an | more Active Internet connections (including servers) Proto Recv-Q Send-Q Local Address Foreign Address (state) tcp4 0 0 *.22 *.* LISTEN tcp6 0 0 *.22 *.* LISTEN udp4 0 0 *.514 *.* udp6 0 0 *.514 *.* div4 0 0 *.700 *.* Active UNIX domain sockets Address Type Recv-Q Send-Q Inode Conn Refs Nextref Addr fffff80003b83000 stream 0 0 fffff80003cac5a0 0 0 0 /var/run/devd.pipe fffff80003baf800 dgram 0 0 0 fffff80003bafc00 0 fff ...
To examine the divert operation, we first create a suitable ruleset:
# ipfw add 700 divert 700 ip from any to any # # ipfw add 1000 allow udp from 203.0.113.10 to me 01000 allow udp from 203.0.113.10 to me # # ipfw add 1100 allow udp from me to 203.0.113.10 01100 allow udp from me to 203.0.113.10
The syntax is a bit odd in this case. The divert keyword takes a number argument that functions as the divert object. This is similar syntax to pipes, queues, and NAT (network address translation) rules which we will see later.
A common convention, though not required, is to make the divert port the same number as the rule number in the ruleset. Whatever the rule number, when the packet is diverted and processed, and then returned to ipfw, the firewall picks up the packet at the divert rule number, plus one - that is, the next rule.
# ipfw list 00700 divert 700 ip from any to any 01000 allow udp from 203.0.113.10 to me 01100 allow udp from me to 203.0.113.10 65535 deny ip from any to any
Let us look at the ruleset above. The packet is diverted to a divert socket, port 700 at the first rule. When it is returned from the divert.c program, it renters the ruleset at rule 1000. The ruleset allows UDP packets from and to the external1 VM.
If we set up the firewall VM host to run userv3.sh and the external1 VM host to run ucon.sh 5656, we get the following expected output from the divert program:
# ./divert Opening divert on port 700 203.0.113.10:51417 -> 203.0.113.50:5656 |0000 45 00 00 46 a9 0c 00 00 40 11 65 3e ac 10 0a 0a |E..F....@.e>....| |0010 ac 10 0a 32 c8 d9 16 18 00 32 a2 eb 55 44 50 20 |...2.....2..UDP | |0020 70 61 63 6b 65 74 20 66 72 6f 6d 20 5b 31 37 32 |packet from [172| |0030 2e 31 36 2e 31 30 2e 31 30 5d 2c 5b 35 36 35 36 |.16.10.10],[5656| |0040 5d 2c 5b 34 5d 0a |],[4]. |
And we also see the output from userv3.sh on the firewall console:
# sh userv3.sh Starting UDP listeners on [5656],[5657],[5658] UDP communication from [203.0.113.10],[5656],[1]
However, if we shut down the userv3.sh services on the firewall VM, the incoming packets find no open port and are rejected by the firewall VM host. However, they still go through the divert socket:
# ./divert Opening divert on port 700 203.0.113.10:26058 -> 203.0.113.50:5656 |0000 45 00 00 47 a9 12 00 00 40 11 65 37 ac 10 0a 0a |E..G....@.e7....| |0010 ac 10 0a 32 65 ca 16 18 00 33 28 a9 55 44 50 20 |...2e....3(.UDP | |0020 70 61 63 6b 65 74 20 66 72 6f 6d 20 5b 31 37 32 |packet from [172| |0030 2e 31 36 2e 31 30 2e 31 30 5d 2c 5b 35 36 35 36 |.16.10.10],[5656| |0040 5d 2c 5b 31 30 5d 0a |],[10]. | 203.0.113.50:771 -> 203.0.113.10:27038 |0000 45 00 00 63 3a 61 00 00 40 01 65 37 ac 10 0a 32 |E..c:a..@.e7...2| |0010 ac 10 0a 0a 03 03 69 9e 00 00 00 00 45 00 00 47 |......i.....E..G| |0020 a9 12 00 00 40 11 65 37 ac 10 0a 0a ac 10 0a 32 |....@.e7.......2| |0030 65 ca 16 18 00 33 28 a9 55 44 50 20 70 61 63 6b |e....3(.UDP pack| |0040 65 74 20 66 72 6f 6d 20 5b 31 37 32 2e 31 36 2e |et from [172.16.| |0050 31 30 2e 31 30 5d 2c 5b 35 36 35 36 5d 2c 5b 31 |10.10],[5656],[1| |0060 30 5d 0a |0]. | divert: sendto: Permission denied
The last output line, "Permission denied", is because the kernel, faced with a packet and no port to send it to, instead sends an ICMP port unreachable response. The kernel sends the ICMP packet back out the divert port, but there is no ipfw rule for it to re-enter the firewall - thus "Permission denied". The packet is dropped.
To fix, we add a rule for ICMP traffic in either direction:
# ipfw list 00700 divert 700 ip from any to any 00800 allow icmp from any to any 01000 allow udp from 203.0.113.10 to me 01100 allow udp from me to 203.0.113.10 65535 deny ip from any to any
The divert operation now works as expected and the packet re-enters the firewall after rule 700. The next rule (800) permits ICMP in either direction and the packet is sent back to the source host. In the listings below, the ucon.sh script was run for 5 cycles, and after cycle #3, the firewall userv3.sh script was shut down.
The remaining two cycles result in an ICMP message being returned back to the external1 VM:
# ./divert Opening divert on port 700 203.0.113.10:36083 -> 203.0.113.50:5656 |0000 45 00 00 46 a9 20 00 00 40 11 65 2a ac 10 0a 0a |E..F. ..@.e*....| |0010 ac 10 0a 32 8c f3 16 18 00 32 de d4 55 44 50 20 |...2.....2..UDP | |0020 70 61 63 6b 65 74 20 66 72 6f 6d 20 5b 31 37 32 |packet from [172| |0030 2e 31 36 2e 31 30 2e 31 30 5d 2c 5b 35 36 35 36 |.16.10.10],[5656| |0040 5d 2c 5b 31 5d 0a |],[1]. | 203.0.113.10:25662 -> 203.0.113.50:5656 |0000 45 00 00 46 a9 21 00 00 40 11 65 29 ac 10 0a 0a |E..F.!..@.e)....| |0010 ac 10 0a 32 64 3e 16 18 00 32 07 89 55 44 50 20 |...2d>...2..UDP | |0020 70 61 63 6b 65 74 20 66 72 6f 6d 20 5b 31 37 32 |packet from [172| |0030 2e 31 36 2e 31 30 2e 31 30 5d 2c 5b 35 36 35 36 |.16.10.10],[5656| |0040 5d 2c 5b 32 5d 0a |],[2]. | 203.0.113.10:40345 -> 203.0.113.50:5656 |0000 45 00 00 46 a9 22 00 00 40 11 65 28 ac 10 0a 0a |E..F."..@.e(....| |0010 ac 10 0a 32 9d 99 16 18 00 32 ce 2c 55 44 50 20 |...2.....2.,UDP | |0020 70 61 63 6b 65 74 20 66 72 6f 6d 20 5b 31 37 32 |packet from [172| |0030 2e 31 36 2e 31 30 2e 31 30 5d 2c 5b 35 36 35 36 |.16.10.10],[5656| |0040 5d 2c 5b 33 5d 0a |],[3]. | 203.0.113.10:53482 -> 203.0.113.50:5656 |0000 45 00 00 46 a9 23 00 00 40 11 65 27 ac 10 0a 0a |E..F.x..@.e'....| |0010 ac 10 0a 32 d0 ea 16 18 00 32 9a da 55 44 50 20 |...2.....2..UDP | |0020 70 61 63 6b 65 74 20 66 72 6f 6d 20 5b 31 37 32 |packet from [172| |0030 2e 31 36 2e 31 30 2e 31 30 5d 2c 5b 35 36 35 36 |.16.10.10],[5656| |0040 5d 2c 5b 34 5d 0a |],[4]. | 203.0.113.50:771 -> 203.0.113.10:27037 (ICMP packet) |0000 45 00 00 62 3a 68 00 00 40 01 65 27 ac 10 0a 32 |E..b:h..@.e'...2| |0010 ac 10 0a 0a 03 03 69 9d 00 00 00 00 45 00 00 46 |......i.....E..F| |0020 a9 23 00 00 40 11 65 27 ac 10 0a 0a ac 10 0a 32 |.x..@.e'.......2| |0030 d0 ea 16 18 00 32 9a da 55 44 50 20 70 61 63 6b |.....2..UDP pack| |0040 65 74 20 66 72 6f 6d 20 5b 31 37 32 2e 31 36 2e |et from [172.16.| |0050 31 30 2e 31 30 5d 2c 5b 35 36 35 36 5d 2c 5b 34 |10.10],[5656],[4| |0060 5d 0a |]. | 203.0.113.10:35359 -> 203.0.113.50:5656 |0000 45 00 00 46 a9 24 00 00 40 11 65 26 ac 10 0a 0a |E..F.$..@.e&....| |0010 ac 10 0a 32 8a 1f 16 18 00 32 e1 a4 55 44 50 20 |...2.....2..UDP | |0020 70 61 63 6b 65 74 20 66 72 6f 6d 20 5b 31 37 32 |packet from [172| |0030 2e 31 36 2e 31 30 2e 31 30 5d 2c 5b 35 36 35 36 |.16.10.10],[5656| |0040 5d 2c 5b 35 5d 0a |],[5]. | 203.0.113.50:771 -> 203.0.113.10:27037 (ICMP packet) |0000 45 00 00 62 3a 69 00 00 40 01 65 26 ac 10 0a 32 |E..b:i..@.e&...2| |0010 ac 10 0a 0a 03 03 69 9d 00 00 00 00 45 00 00 46 |......i.....E..F| |0020 a9 24 00 00 40 11 65 26 ac 10 0a 0a ac 10 0a 32 |.$..@.e&.......2| |0030 8a 1f 16 18 00 32 e1 a4 55 44 50 20 70 61 63 6b |.....2..UDP pack| |0040 65 74 20 66 72 6f 6d 20 5b 31 37 32 2e 31 36 2e |et from [172.16.| |0050 31 30 2e 31 30 5d 2c 5b 35 36 35 36 5d 2c 5b 35 |10.10],[5656],[5| |0060 5d 0a |]. | ^C
Note the two icmp packets logged by rule 800:
# ipfw show 00700 19 2466 divert 700 ip from any to any 00800 2 196 allow icmp from any to any 01000 5 350 allow udp from 203.0.113.10 to me 01100 0 0 allow udp from me to 203.0.113.10 65535 477 114991 deny ip from any to any
General notes on the divert action:
The ipdivert.ko kernel module must be loaded or compiled into the kernel to create a divert rule, and thus to use divert(4) sockets.
The ipdivert.ko kernel module cannot be unloaded. You must restart the VM to remove the ipdivert.ko kernel module.
You cannot create a rule with a divert port of 0 or 65535. The port number must be between 1 and 65534 (inclusive).
If you create a rule with a divert port on rule 65534, the returning packet will restart firewall rule processing at the default rule, 65535, which cannot be changed.
You can create a divert rule for any protocol in /etc/protocols.
You can use the same divert port for multiple rules.
After returning from a divert rule, if the next rule is in another set, processing will continue with that rule unless the set is disabled. If disabled, it will skip to the next rule in any set that is not disabled.
General notes on creating the divert socket:
Only root can create a divert socket.
Opening a divert socket on port 0 or port 65536 results in a random divert port number.
Opening a divert socket on port 65535 is permitted, but not advised.
Opening a divert port greater than 65536 or less than 0 results in a positive port number modulo 65536.
As with other sockets, you cannot open two divert sockets on the same port number. However, you can open a divert socket on a port already in use for any protocols based on IPv4 or IPv6.
3.3.14. Other Protocols
You can use any protocol in /etc/protocols in a rule.
# ipfw add 1000 allow ospf from any to me # ipfw add 2000 allow chaos from any to me etc.
3.3.15. Limit
ipfw can restrict the number of active connections with the limit option.
This option allows for specifying a parameter in the rule that is regarded as a flow element, that is, one of src-addr
, src-port
, dst-addr
, or dst-port
.
In addition, the limit keyword takes a value, N, that is the maximum number of connections desired:
# ipfw add 1000 allow udp from any to me limit src-addr 5 # ipfw add 1100 count udp from any to me
Concurrent connections via TCP, UDP, ICMP, or any protocol can be limited in this way.
ipfw creates a dynamic rule for each connection allowed by the rule. When the maximum number of connections is reached, additional packets are considered no longer matched and are dropped by the rule after being counted, and the search terminates.
To test, we write a simple script on the external1 VM to flood UDP packets at the firewall VM running sh userv3.sh:
# export NUM=1
# for i in `jot -r 500 5656 5658 1`
> do
> echo "hello [$NUM] to port [$i]" | ncat -u 203.0.113.50 $i
> NUM=expr $NUM + 1
> done
On the firewall VM, ipfw starts creating dynamic rules as soon as the first matching packet is received.
Additional dynamic rules, up to the limit number are created.
Each UDP based dynamic rule has a default 10 second lifetime, controlled by the sysctl node net.inet.ip.fw.dyn_udp_lifetime
.
As they expire under the limit value, space for additional connections is created.
The number of open dynamic rules at any point in time can be viewed with the sysctl node net.inet.ip.fw.dyn_count
.
You can view the dynamic rules with:
# ipfw -SaD list 00500 0 0 check-state :default 01000 10 531 allow udp from 203.0.113.10 to me limit src-addr 3 :default 65535 0 0 deny ip from any to any ## Dynamic rules (4 560): 01000 1 53 (8s) LIMIT udp 203.0.113.10 23755 <-> 203.0.113.50 5657 :default 01000 1 53 (8s) LIMIT udp 203.0.113.10 30144 <-> 203.0.113.50 5656 :default 01000 0 0 (4s) PARENT 3 udp 203.0.113.10 0 <-> 0.0.0.0 0 :default 01000 1 53 (8s) LIMIT udp 203.0.113.10 22722 <-> 203.0.113.50 5658 :default
If you run cmdwatch -n1 ipfw -SaD list on the firewall VM you can see the list of rules grow and shrink in real time. |
It is useful to experiment with the sysctl net.inet.ip.fw.dyn_udp_lifetime
and see its effect on net.inet.ip.fw.dyn_count
.
By adjusting the net.inet.ip.fw.dyn_udp_lifetime
value during a network packet flood (like we created above), you can watch how the ipfw limit rule blocked traffic through the firewall.
Here is the result of a sample run. Note the missing connections due to a limit restriction:
... hello [19] to port [5657] hello [20] to port [5656] hello [21] to port [5656] hello [33] to port [5657] hello [34] to port [5656] hello [35] to port [5657] hello [36] to port [5657] hello [37] to port [5657] hello [48] to port [5657] hello [50] to port [5657] hello [51] to port [5656] hello [52] to port [5657] hello [53] to port [5656] hello [64] to port [5657] ...
3.3.16. Call and Return
The call and return actions allow you to change ruleset processing by jumping to a rule number elsewhere in the ruleset. If the rules at that location contain a return action, processing will jump back to the statement immediately after the original call statement. In practice, this acts like a program function call, or as ipfw(8) notes, like an assembly language subroutine.
Creating a ruleset with call and return actions:
# # ipfw add 500 check-state 00500 check-state :default # # ipfw add 1000 call 20000 udp from 203.0.113.10 to me 5656 01000 call 20000 udp from 203.0.113.10 to me 5656 # # ipfw add 1100 count udp from 203.0.113.10 to me 01100 count udp from 203.0.113.10 to me # # ipfw add 1200 allow udp from 172.168.10.10 to me 5656 01200 allow udp from 203.0.113.10 to me 5656 # # ipfw add 20000 count udp from 203.0.113.10 to me 20000 count udp from 203.0.113.10 to me # # ipfw add 21000 return via any 21000 return # # ipfw -a list 00500 0 0 check-state :default 01000 0 0 call 20000 udp from 203.0.113.10 to me 5656 01100 0 0 count udp from 203.0.113.10 to me 01200 0 0 allow udp from 203.0.113.10 to me 5656 20000 0 0 count udp from 203.0.113.10 to me 21000 0 0 return 65535 0 0 deny ip from any to any
As noted in the man page, you have to put some extra syntactic sugar on the return statement:
# ipfw add 21000 return via any
In this example we have used several count statements to try to trace ruleset processing.
To test, we have the firewall VM host startup 3 listeners with userv3.sh. After the external1 VM uses ucon.sh 5656 to send a udp packet, the count list looks like this:
# ipfw -a list 00500 0 0 check-state :default 01000 1 70 call 20000 udp from 203.0.113.10 to me 5656 01100 1 70 count udp from 203.0.113.10 to me 01200 1 70 allow udp from 203.0.113.10 to me 5656 20000 1 70 count udp from 203.0.113.10 to me 21000 1 70 return via any 65535 0 0 deny ip from any to any
The packet was matched at rule 1000 where it encountered a call action to jump to rule 20000 where it was then matched at 20000. The next rule was a matched return action at rule 21000. Returning to the rule after the call action, it matched a count action at 1100, then matched an allow action at 1200 where it was sent through to the application layer and was received by userv3.sh.
If we remove the return action at rule 21000, the counts look much different:
# ipfw -a list 00500 0 0 check-state :default 01000 1 70 call 20000 udp from 203.0.113.10 to me 5656 01100 0 0 count udp from 203.0.113.10 to me 01200 0 0 allow udp from 203.0.113.10 to me 5656 20000 1 70 count udp from 203.0.113.10 to me 65535 1 70 deny ip from any to any
Without a return at rule 21000, the only rule left is the default deny rule, and there is nothing received by userv3.sh.
Because you can jump both forward and backward with the call action, it is possible to create an endless loop. |
In this example we create an endless loop with in incorrect call rule:
# # ipfw -a list 01000 0 0 count udp from 203.0.113.10 to me 05000 0 0 check-state :default 06000 0 0 call 1000 udp from 203.0.113.10 to me 5656 07000 0 0 count udp from 203.0.113.10 to me 65535 0 0 deny ip from any to any
In this example we are missing a return action between rule 1000, the target of the call action at rule 6000. This creates a loop. ipfw eventually figures out that a loop exists, and breaks out at the next call action with the diagnostic:
# ipfw: call stack error, go to next rule
but it does not currently tell you where the missing return action is or which rule it went to next.
You can pick up a clue by watching the rule counts. Below is the rule count for this errant ruleset after just one packet was received from the external1 VM:
# ipfw -a list 01000 17 1207 count udp from 203.0.113.10 to me 05000 0 0 check-state :default 06000 16 1136 call 1000 udp from 203.0.113.10 to me 5656 07000 1 71 count udp from 203.0.113.10 to me 65535 1 71 deny ip from any to any
This shows that ipfw went around this call loop 16 times before throwing an error.
For TCP connections, call and return operate almost the same. Below is a ruleset with TCP instead of UDP for the desired protocol and including the required setup and keep-state keywords:
# ipfw -a list 00500 0 0 check-state :default 01000 0 0 call 20000 tcp from 203.0.113.10 to me 5656 01100 0 0 count tcp from 203.0.113.10 to me 02000 0 0 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default 20000 0 0 count tcp from 203.0.113.10 to me 21000 0 0 return 65535 0 0 deny ip from any to any
After the connection is successfully made from external1, the observed counts are:
# ipfw -a list 00500 0 0 check-state :default 01000 1 60 call 20000 tcp from 203.0.113.10 to me 5656 01100 1 60 count tcp from 203.0.113.10 to me 02000 8 479 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default 20000 1 60 count tcp from 203.0.113.10 to me 21000 1 60 return 65535 0 0 deny ip from any to any
The difference in rule counts is due to dynamic rules created by the setup and keep-state keywords on rule 2000.
If we manually run the connection we can observe the counts with the -d
command line parameter.
The numbers below are from just the initial 3-way handshake:
# ipfw -ad list 00500 0 0 check-state :default 01000 1 60 call 20000 tcp from 203.0.113.10 to me 5656 01100 1 60 count tcp from 203.0.113.10 to me 02000 5 276 allow tcp from 203.0.113.10 to me 5656 setup keep-state :default 20000 1 60 count tcp from 203.0.113.10 to me 21000 1 60 return 65535 0 0 deny ip from any to any ## Dynamic rules (1 152): 02000 5 276 (296s) STATE tcp 203.0.113.10 19179 <-> 203.0.113.50 5656 :default
The above numbers indicate that the 3-way handshake occurred during the dynamic rule setup.
General notes on call and return:
You can call out to an address either before or after the current rule.
You must have a return for every call.
You can nest call / return pairs up to 16 levels deep. If you add another call rule, ipfw will throw the "call stack error, go to next rule" error and continue with the next rule.
Similar to skip-to, you cannot call to rule 0, 65535, or 65536. If you call to a value greater then 65536, the rule is accepted and the call target rule number is the requested value modulo 65536.
Similar also to sets, if a call is made to a target rule in a set that is disabled, the call will land on the next non-disabled rule in any set and processing continues from there.
If a return is encountered when no call has been made, the return rule is ignored and processing continues with the next rule.
3.3.17. Using uid and gid in rules
An interesting capability of ipfw is its ability to match packets on Unix uid and gid values. Network packets themselves have no inherent ownership, so where does this capability come from? Answer - it comes from the applications that are the source and destination of those packets.
Let us look at a simple example. First, let us examine the actual syntax that allows this matching capability.
Locate or add a user (here 'quarven') if needed: # grep quarven /etc/passwd quarven:*:1002:1002:Quarven:/home/quarven:/bin/sh Copy userv.sh to user quarven home directory # cp ~/bin/userv.sh /home/quarven Now use userid 'quarven' in an ipfw rule: # ipfw add 700 allow udp from 203.0.113.10 to me uid quarven 00700 allow udp from 203.0.113.10 to me uid quarven # # ipfw show 00700 0 0 allow udp from 203.0.113.10 to me uid quarven 65535 0 0 deny ip from any to any
Next, login as user 'quarven' and run the script /root/bin/userv.sh 5656. Then switch to the external1 VM and run echo "hello there" | ncat -u 203.0.113.50 5656. The results should appear on the console running the userv.sh script.
The results are shown below.
quarven@firewall:~ $ /bin/sh userv.sh 5656 PORT1 = [5656] Starting UDP listener on [203.0.113.50],[5656] hello there ^Cquarven@firewall:~ $
This works because the instance of userv.sh is run under the uid quarven as shown below:
Show the user information for the userv.sh instance: root@firewall:~ # ps -o user -xl -U quarven USER UID PID PPID C PRI NI VSZ RSS MWCHAN STAT TT TIME COMMAND quarven 1002 4256 83703 0 61 0 13380 2908 pause I+ v0 0:00.03 sh userv.sh 5656 quarven 1002 6878 4256 0 61 0 13400 2344 select I+ v0 0:00.02 nc -l -k -u 203.0.113.50 5656 quarven 1002 83703 83431 0 20 0 13380 3168 wait I v0 0:00.10 -sh (sh)
ipfw has matched an incoming network packet to a program owned by a userid. If the rule is changed to another userid, even root, the match will not succeed and the packet will be picked up by the default rule.
It is sometimes necessary to immediately shut down all IP traffic to or from a certain user. This capability is ideal for that purpose. Note however, that the deny rule below must come before any check-state rule to catch traffic that may be otherwise allowed by a dynamic rule. # ipfw add 50 deny ip from any to any uid quarven # ipfw add 100 check-state Note also the item about ICMP traffic in the "General Notes" below. |
General Notes on Using uid and gid:
The gid keyword works in an identical fashion to uid described above.
If using a name as the uid or gid, the name must exist in the indicated system file.
Ranges and lists are not allowed. You cannot use "… uid tom,dick,harry" or "… uid 1000-1002".
Outbound traffic works in a similar way, just reversing the source and destination.
Denied traffic will generally have an indication of "<application>: sendto: Permission denied"
ICMP traffic cannot be reliably filtered using uid/gid. This is a known limitation.
As noted in ipfw(8) some contexts, such as initial incoming SYN packets may have no uid/gid associated with them.
Programs using setuid(2) system calls may not behave as expected, though it may be possible to set the uid/gid to the effective id if it can be determined.
Using uid/gid keywords for matching is resource intensive and should be used sparingly if at all.
3.4. Lookup Tables
Lookup tables are a versatile feature of many firewall systems, including ipfw. A lookup table is a virtual container that holds tuples of elements, one of which is a key that functions as a fast lookup feature. Using a key provided by a rule, ipfw can quickly determine if the element is in the table. If it is, that portion of the rule is matched the the value associated with the key is used according to the rule.
Lookup tables are a powerful feature of ipfw and useful in many situations.
ipfw provides five types of lookup tables:
Address tables (addr) - These tables hold addresses that ipfw can rapidly find with an address as a key. If the address is matched the lookup is considered matched and used by the associated rule. This table type takes an additional keyword 'valtype' that can be used to specify IPv4 addresses or IPv6 addresses.
Interface tables (iface) - These tables hold interface names. Each entry is the name of an interface. Note that wildcards such as
em*
are not supported.Number tables (number) - These tables are used for protocols, ports, uids/gids, or jail ids. Entries are 32 bit unsigned integers. Ranges (for example,1234-5678) are not supported.
Flow tables (flow) - These tables contain flow type suboptions that are used in looking up existing traffic flows.
MAC - A MAC address type table holds media access control (MAC) addresses as an address with optional mask length. The mask length defaults to 48 bits if not otherwise specified.
3.4.1. Creating Lookup Tables
All tables must be created before they can be referenced by a rule. Note that the commands to manage tables do not have line numbers - they are independent shell commands that exist outside the ruleset.
# ipfw table foo create
By default, the ipfw table create command creates tables of type addr. Table names share the same namespace and so must be unique even among tables of different types.
To specify other types, add the type keyword:
# ipfw table bar create type iface # # ipfw table baz create type flow # # ipfw table bop create type number
Previously, when creating a table of type number, a bug existed that required an algorithm option such as |
To see all tables, use the list subcommand to show the table and any contents. Shown below, all four table types are created and one entry is added to each table:
# ipfw table all list # # ipfw table foo create type addr # ipfw table foo add 192.168.1.100 33 added: 192.168.1.100/32 33 # # ipfw table bar create type iface # ipfw table bar add em0 33 added: em0 33 # # ipfw table baz create type number # ipfw table baz add 9999 33 added: 9999 33 # # ipfw table bop create type flow # ipfw table bop add 203.0.113.10,192.168.1.100 33 ignored: 33 ipfw: Adding record failed: Invalid argument # # ipfw table bop destroy # # ipfw table bop create type flow:src-ip,dst-ip # ipfw table bop add 203.0.113.10,192.168.1.100 33 added: 203.0.113.10,192.168.1.100 33 # # ipfw table bip create type mac # ipfw table bit add 58:9c:fc:01:02:03 33 added: 58:9c:fc:01:02:03/48 33 # # ipfw table all list --- table(bar), set(0) --- em0 33 --- table(baz), set(0) --- 9999 33 --- table(bip), set(0) --- 58:9c:fc:01:02:03/48 33 --- table(bop), set(0) --- 203.0.113.10,192.168.1.100 33 --- table(foo), set(0) --- 192.168.1.100/32 33
Note the error on the flow table above. Flow tables take an explicit flow specification (discussed below) when they are created. If you try to add an entry to a flow table that does not match the flow specification, ipfw throws an error.
Note that the output list of tables is ordered by table name, not table type. This is worth remembering if you are using many tables of different types.
In all the examples above a key and a value were added to each table. The key was added according to the table type (addr, iface, etc.) The values above, all set to the integer value 33, are just placeholders in the examples. The value for each key is whatever makes sense for the firewall administrator.
"Whatever makes sense" depends on how the table will be used. ipfw(8) identifies some 12 different uses for table values - skipto rule number to skip to, pipe number, fib number, nat number to jump to, dscp value to match or set, tag number to match or set, divert port number to divert traffic to, netgraph hook number to move packet to, limit maximum number of connections, ipf4 ipv4 next hop to forward packets to, ipv6 ipv6 next hop to forward packets to, mark value to match or set.
We have already examined some of these keywords, and will revisit a couple of them to add table operations.
Tables can be removed one at a time with the destroy subcommand, provided the table is not used in any rule:
# ipfw table bar destroy
or removed all at once by specifying the special name all:
# ipfw table all destroy
Note that there is no confirmation with the ipfw table all destroy command as there is with the ipfw flush command, so make sure that is really what you want to do. |
3.4.2. Using Tables in Rules
To begin using tables in rules, it is first necessary to understand the use of the word 'tablearg' which is frequently found in ipfw(8).
3.4.2.1. Understanding the Word tablearg
A tablearg is a value that is the result of a table lookup using a key supplied by a field in a packet. Which field depends on the table type, as discussed above.
The term tablearg is used to show where in the rule the retrieved value will be applied.
For a rule with a tablearg keyword, ipfw
Looks up the key in the identified table. The key is supplied by one or more fields in the packet itself.
Applies the value associated with that key in the table to replace the word tablearg.
Essentially
# ipfw add 50 skipto tablearg ip from 'table(badhosts)' to any
becomes
# ipfw add 50 skipto 65535 ip from 203.0.113.10 to any
if the from address in the packet is matched in the table.
If it is not matched, processing continues with the next rule.
From the ipfw(8) man page - The tablearg argument can be used with the following actions: nat, pipe, queue, divert, tee, netgraph, ngtee, fwd, skipto, setfib; wth action parameters: tag, untag; and with rule options: limit, tagged. |
First Example
Our first example is to use a table with the skipto keyword.
Consider a table of addresses of "bad hosts". If any such host were to try to connect to or through the firewall, they should be denied. Since we already have a deny rule (the immutable rule at 65535), we can load an address table with keys of hosts, and values of the deny rule, 65535:
First, on the firewall VM, create a table called badhosts, and populate it with the addresses of hosts to be denied:
Restart ipfw: # kldunload ipfw # kldload ipfw Create and populate the table: # ipfw table badhosts create type addr # ipfw table badhosts add 203.0.113.10/32 65535 added: 203.0.113.10/32 65535
Then create a rule that uses the table. For this use case, we will want to put the rule before the check-state rule,
# ipfw add 50 skipto tablearg ip from 'table(badhosts)' to any 00050 skipto tablearg ip from table(badhosts) to any # # ipfw add 100 check-state # # ipfw add 1000 allow ip from any to any
Note the single quotes around the table(badhosts)
entry to placate the shell:
To test, start up userv3.sh on the firewall VM:
# sh userv3.sh Starting UDP listeners on [5656],[5657],[5658]
And on the external1 VM (which should have address 203.0.113.50) , start up ucon.sh:
# sh ucon.sh 5656 UDP communicationing [203.0.113.50],[5656],[1]
No communication should be seen on the firewall VM.
If you remove the entry for 203.0.113.10/32
from the badhosts
table, the communications succeed and reach the userv3.sh services listening on the firewall VM:
# ipfw table badhosts delete 203.0.113.10 deleted: 203.0.113.10/32 0
Retrying the ucon.sh communications above will succeed.
It would be tempting to combine the previous two examples into something like:
# ipfw add 1000 allow udp from 'table(badhosts)' to me dst-port 'table(badports)' ipfw: invalid destination port table(badports)
but ipfw does not allow the use of more than one table in a rule.
However, instead of a second table keyword, you could use the lookup keyword for the port:
A contrived example: # ipfw add 25 allow udp from 'table(badhosts)' to me lookup dst-port badports 00025 allow udp from table(badhosts) to me dst-ip lookup dst-port badports
While this does work, the better solution is to use the a flow table:
Unload and reload the ipfw kernel module for the next example, this time for "good hosts".
# ipfw table goodflow create type flow:src-ip,dst-port # # ipfw table goodflow add 203.0.113.10,5656 added: 203.0.113.10,5656 0 # # ipfw table goodflow add 203.0.113.10,5657 added: 203.0.113.10,5657 0 # # ipfw add 500 allow udp from any to me flow 'table(goodflow)' 00500 allow udp from any to me flow table(goodflow)
This gives you much more granular control of exactly what host and what port you want to match together. You can see the results by trying sh ucon.sh 5656, sh ucon.sh 5657, and sh ucon.sh 5658 from the external1 VM. The first two succeed, while the third does not.
Using the valtype keyword for addr tables you can have separate tables for IPv4 and IPv6:
# Create the translation tables. # ipfw table T46 create type addr valtype ipv6 # ipfw table T64 create type addr valtype ipv4
Second Example
Our second example concerns using tables with the limit keyword. Recall that the limit keyword limits the number of active connections at one time.
We can use an address table to keep track of addresses and limits like this:
# Create a table named "limits". # ipfw table limits create type addr # # Assign a value of 23 to the address 203.0.113.10 # ipfw table limits add 203.0.113.10 23 added: 203.0.113.10/32 23 # # Now add a limit rule for this address # ipfw add 1000 allow udp from 'table(limits)' to me limit src-addr tablearg 01000 allow udp from table(limits) to me limit src-addr tablearg :default # # And add a rule to allow the traffic. # ipfw add 1100 allow udp from 'table(limits)' to me #
We now have a table named "limits" with one entry, 203.0.113.10 with value 23. With rule 1000, all packets coming into the firewall will be looked up in the table. If a packet from address 203.0.113.10 arrives, the lookup will succeed and the value of "23" will be applied as the tablearg to the limit option for connections with that address.
Once this is set up, it is possible to change the limit value without changing the rule. The value can either be changed by deleting and re-adding the table entry, or even better, by creating a new table, newlimits, and using the table swap command:
# Create the newlimits table. # ipfw table newlimits create type addr # # Assign a value of 50 to the address 203.0.113.10. # ipfw table newlimits add 203.0.113.10 50 added: 203.0.113.10/32 50 # # Now swap the tables. # ipfw table limits swap newlimits # # ipfw table all list --- table(lims), set(0) --- 203.0.113.10/32 50 --- table(newlims), set(0) --- 203.0.113.10/32 23
With this arrangement, the "newlimits" table can be modified independently from the "limits" table and the rule is not affected.
While this looks like a good solution, there is one major flaw here.
Incoming packets from 203.0.113.10 are processed by rule 1000, which looks like it permits traffic through the firewall, but all it does is perform a table lookup and set a limit value. And then nothing else happens. Processing does not continue to rule 1100. This can be verified by using the ipfw show command to watch the rule counts.
The easiest workaround is to delete rule 1000 after the swap is completed. Other more elegant strategies include preparing a "function" (using the call and return keywords) to set a limit value directly.
3.4.2.2. More on flow tables
A flow table maintains a list of flows. A flow is a designation given to traffic between two endpoints. The designation can be any subset of:
src-ip
src-port
proto
dst-ip
dst-port
that makes sense in source → destination order. Examples include:
# ipfw table zoo create type flow:src-ip # Creates a flow based on soure IP address # ipfw table zar create type flow:proto # Creates a flow based on just a protocol # ipfw table zaz create type flow:dst-ip # Creates a flow based on just the desitination IP # ipfw table zop create type flow:dst-port # Creates a flow based on the destination port
or with extra specificity:
# ipfw table zip create type flow:src-ip,proto,dst-ip,dst-port # Flow based on all four # ipfw table zim create type flow:src-ip,proto # Just source IP and protocol # ipfw table zam create type flow:src-ip,proto,dst-ip # Source IP, destination IP and protocol # ipfw table zap create type flow:src-ip,dst-ip # Just IP address endpoints
Once the table is created for a flow, entries can be placed in the table provided they match the table flow specification.
Matching these additions to the tables created above:
# ipfw table zim add 192.168.200.30,tcp
would succeed, but
# ipfw table zip add 192.168.30.20,172.20.15.20
would fail because the flow specification for table zip is src-ip,proto,dst-ip,dst-port and neither the protocol nor the destination port was given.
A correct flow specification for table zip would be something like:
# ipfw table zip add 192.168.30.20,tcp,172.20.15.20,80
flow tables allow for precise definition of traffic between a source and a destination. Once in the table, rules can be applied to commands allow, deny, divert, queue, etc. for modifying traffic flow.
General notes on all tables:
Table names can be numeric or alphanumeric and can include only hyphen (-), underscore (_), and period (.) as special characters.
Maximum table name length is 63 characters.
Tables names must be unique within a set. Tables can have the same name across different sets, however any rules for tables in sets other than set 0, must include the set number. The sysctl variable
net.inet.ip.fw.tables_sets
controls this behavior.The maximum number of tables across all sets is 65535. Practically however, the number is controlled by the sysctl variable
net.inet.ip.fw.tables_max
. The default is 128.If a table is in use in a rule, it cannot be destroyed. The rule must be removed first, then the table can be destroyed. However, a table can be flushed (ipfw table tablename flush) at anytime.
All table types survive an ipfw flush action, and table contents are not affected.
Make table names as descriptive as possible to avoid confusion when used in rules. The names we are using here (foo, bar, zip, zap, etc.) are just examples.
As noted in the man page, if two tables are used in a rule, the result of the second (destination) is used. Therefore avoid using two tables in a rule, or try using the lookup keyword instead.
Notes on specific table types:
Address tables (addr)
Address tables (addr) support IPv4 and IPv6 address types, and varying mask lengths appropriate for each address type. The default mask length for IPv4 is 32 bits (/32) and the default prefix for IPv6 is 128 bits (/128).
Table lookups will return the most specific entry, so 203.0.113.20/32 is preferred over 203.0.113.0/24.
Interface tables (iface)
Interface tables store interface names as alphanumeric text. The text does not actually have to match a current valid interface.
Special characters in interface names can include any from the set
[-+_?,.!~@#%^&*()=;:/,<>{}|]
Note however, that the shell may recognize some of these characters when adding and during lookup, thus interfering with the table operation, and so special characters in interface names should be avoided.
The maximum key length is 15 characters.
There is no support for interface ranges, for example
em0-4
, even though an interface name "em0-4" can be entered.
Number tables (number)
Number tables support unsigned 32-bit integer types.
Entries can be positive or negative. Negative entries perform unsigned ones complement arithmetic, and positive numbers roll over from 4294967295 to 0.
Any shell element that evaluates to a number can be used: shell variables that resolve to a number (
$MAILCHECK
,$PPID
), backtick operations such asexpr 5 + $UID
,id -u
,date "+%s"
, and special variables like$RANDOM
in bash.As with other table types, ranges are not supported.
Flow tables (flow)
Flow tables describe network traffic based on the desired attributes. The best matches include as much detail as possible:
src-ip
,proto
,dst-ip
,port
. Including less than that may make it difficult to add an element in the table:ipfw table foo create type flow:dst-port # Table based on just the destination port
ipfw table foo add telnet # Fails to add!
ipfw table bar create type flow:dst-ip,dst-port
ipfw table bar add 203.0.113.10,5656
3.5. Stream Control Transport Protocol (SCTP)
A readable introduction to SCTP is found in Wikipedia - https://en.wikipedia.org/wiki/Stream_Control_Transmission_Protocol#RFCs and ever more detail is found in the accompanying RFCs.
3.5.1. SCTP Versions
As of FreeBSD 14.1, there are three different versions of SCTP - a "native" version, a "separate portable" version, and a "userland" version.
FreeBSD ships with the "native" version of the protocol, described in sctp(4). (This is actually the reference implementation of SCTP initially developed on FreeBSD 7.) This version requires loading the kernel module sctp.ko before use. Native SCTP uses the SCTP library on FreeBSD which provides access to the various functions found in <netinet/sctp.h>. You can use the features described in sctp(4) to develop SCTP applications based on the native protocol.
Then, there are a couple of notable packages regarding SCTP:
libusrsctp “Portable SCTP userland stack” – This is a non-kernel implementation of the protocol using “usrsctp_*” functions. It also provides UDP encapsulation as shown in the Wireshark image below:
sctplib “User-space implementation of the SCTP protocol RFC 4960” – This package is a re-implementation of the native SCTP code that can function as a replacement for the default installed code. It does not require the sctp.ko kernel module.
As noted in the BUGS section of sctp(4), the sctp.ko kernel module cannot be unloaded. You will have to restart FreeBSD to remove the module from the kernel. |
Interestingly, the reference implementation for SCTP that was developed on FreeBSD 7 and has been archived on GitHub at https://github.com/cyberroadie/sctp-examples. ipfw played a role in its development and the vestiges of that development remain in the ipfw code base and in the collection of sysctls that support it.
We will study the native SCTP protocol usage and the encapsulated usage with ipfw.
3.5.2. SCTP Protocol Operation
The image below shows a Wireshark view of a typical native SCTP association (connection):
Notice how there are two different interfaces involved with this transfer - 127.0.0.1, and 192.168.1.78. In fact, SCTP supports association setup and data exchange with multiple interfaces to a remote node. The RFC’s explain this capability in detail.
Also in this image, you can see the basic 4-way handshake - INIT, INIT_ACK, COOKIE_ECHO, and COOKIE_ACK. Also shown are the HEARTBEAT, DATA, SACK (Stream Acknowledgement) messages, and the shutdown sequence of SHUTDOWN, SHUTDOWN_ACK, and SHUTDOWN_COMPLETE.
Below is a view of netstat -an
showing the display of a separate SCTP section containing all current associations:
% netstat -an Active Internet connections (including servers) Proto Recv-Q Send-Q Local Address Foreign Address (state) tcp4 0 0 *.22 *.* LISTEN tcp6 0 0 *.22 *.* LISTEN tcp4 0 0 127.0.0.1.631 *.* LISTEN tcp6 0 0 ::1.631 *.* LISTEN udp4 0 0 127.0.0.1.123 *.* udp6 0 0 fe80::1%lo0.123 *.* udp6 0 0 ::1.123 *.* udp4 0 0 192.168.1.78.123 *.* udp6 0 0 2600:1700:3901:4.123 *.* udp6 0 0 fe80::3e97:eff:f.123 *.* udp4 0 0 *.123 *.* udp6 0 0 *.123 *.* Active SCTP associations (including servers) Proto Type Local Address Foreign Address (state) sctp4 1to1 127.0.0.1.5000 127.0.0.1.42227 ESTABLISHED 192.168.1.78.5000 192.168.1.78.42227 sctp4 1to1 127.0.0.1.42227 127.0.0.1.5000 ESTABLISHED 192.168.1.78.42227 192.168.1.78.5000 sctp4 1to1 127.0.0.1.5000 LISTEN 192.168.1.78.5000 Active UNIX domain sockets Address Type Recv-Q Send-Q Inode Conn Refs Nextref Addr fffff8015575c000 stream 0 0 0 fffff80155758c00 0 0 . . .
Internally, the SCTP data packets look different from TCP and/or UDP packets. However, like those two, SCTP has a protocol number, source address, destination address, and port numbers, so using ipfw with SCTP will be fairly straightforward.
Here’s a look at a typical packet:
To study SCTP’s behavior with ipfw we will use the client/server echo program, shown above. We will also look at a streaming application similar to the chargen small server of yore.
3.5.3. Using the TSCTP Testing Tool on FreeBSD
A handy testing tool is the FreeBSD package tsctp
.
Install the package on the external1 and internal VMs.
(You will have to configure the VMs to access the Internet to install the package.
Refer to the Quick Start section for details.)
# pkg install tsctp
This tool uses the native SCTP protocol version. To use, we first load the sctp.ko kernel module:
# kldload sctp.ko
Running the program is discussed below.
To get started, initialize the bridge and tap devices to the same architecture as "Simple NAT", shown in the figure below.
% sudo /bin/sh mkbr.sh reset bridge0 tap1 tap4 bridge1 tap0 tap5 Then start the VMs we need: % /bin/sh runvm.sh firewall external1 internal
Then follow these steps:
Apply the addressing shown in the above figure and ensure connectivity with adjacent interfaces, and with the opposite side interfaces (10.10.10.20 from the external1 VM and 203.0.113.10 from the internal VM).
The default route for the external1 VM should point to 203.0.113.50, and for the internal VM it should point to 10.10.10.50.
Load the sctp.ko kernel module on the external1 and the internal VMs. This enables communication via native SCTP on these VMs.
On the internal VM, run in server mode: tsctp -L 127.0.0.1 -L 10.10.10.20 -p 1234 and on the external1 VM, run in client mode: tsctp -L 203.0.113.10 -p 1234 -n 10 -l 1000 192.168.1.128
Use tscp --help
to get a list of the options and meanings.
The client program sends 10 messages (-n 10
) to the server.
Simple statistics are shown once the program terminates.
With the above procedure, we have established SCTP communications across the firewall. The firewall VM does not need to load the sctp.ko module. To the firewall VM, this is simply normal IP traffic.
Restart the client with -n 0
, for unlimited messages.
While normal SCTP traffic is established from client to server, load ipfw on the firewall VM.
Traffic is immediately halted. Eventually, the client will recognize it has been disconnected. This may take a couple of minutes. The disconnection is shown in the image below:
Creating suitable rules for SCTP traffic is quite similar to other rules we have performed before.
One would think the "one-rule" version using setup and keep-state keywords would work, but it does not. You have to use the "two-rule" version:
root@firewall:~ # ipfw show 00100 0 0 check-state :default 01000 40 4152 allow sctp from 203.0.113.10 to 10.10.10.20 02000 48 4424 allow sctp from 10.10.10.20 to 203.0.113.10 65535 0 0 deny ip from any to any root@firewall:~ #
Also, keep in mind that SCTP is typically used where there are multiple interfaces per association. This example has used only one, but the principles are the same.
3.5.4. Downloading and Building usrsctp Programs
To test SCTP encapsulation with UDP, we must download and build the usrsctp kit. Follow this procedure to download and build the programs for the usrsctp kit. On the external1 VM and the internal VM, the procedure is the same.
For this procedure, you will have to reconfigure the external1 and internal VMs to access the Internet as described in Quick Start.
# pkg install git # pkg install cmake # mkdir /root/src # cd /root/src # git clone https://github.com/sctplab/usrsctp.git # mkdir tmp # cd tmp # cmake ../usrsctp # cmake --build . Once finished, the test programs are located in /root/src/tmp/programs.
Now reconfigure external1 and internal VMs to the architecture in Using the TSCTP Testing Tool.
3.5.5. Encapsulated Echo Server and Client with IPFW
The figure below shows encapsulated usage of SCTP with the chargen_server_upcall program running on the internal VM and the client_upcall program running on the external1 VM. The output is similar to that of the chargen small server program.
All the data exchanged between the two systems was encapsulated in UDP datagrams as shown in the figure below.
Any ipfw rules for this traffic only have to be concerned with UDP, not SCTP.
Chapter 4. IPFW Dummynet and Traffic Shaping
FreeBSD’s dummynet is not a network for dummies. It is a sophisticated network traffic shaping tool for bandwidth usage and scheduling algorithms. In this use of ipfw, the focus is not on ruleset development, although rules are still used to select traffic to pass to dummynet objects. Instead, the focus is on setting up a system to shape traffic flows.
Imagine if you had the ability to model traffic flow across the wild Internet. dummynet allows you to model scheduling, queuing, and similar tasks similar to the real-world Internet.
dummynet works with three main types of objects - a pipe, a queue, and a sched (short for scheduler) which also happen to be the three keywords we now examine.
A pipe (not to be confused with a Unix pipe(2)) is a model of a network link with a configurable bandwidth, and propagation delay.
A queue is an abstraction used to implement packet scheduling using one of several different scheduling algorithms. Packets sent to a queue are first grouped into flows according to a mask on a 5-tuple (protocol, source address, source port, destination address, destination port) specification. Flows are then passed to the scheduler associated with the queue, and each flow uses scheduling parameters (weight, bandwidth, etc.) as configured in the queue itself. A sched (scheduler) in turn is connected to a pipe (an emulated link) and arbitrates the link’s bandwidth among backlogged flows according to weights and to the features of the scheduling algorithm in use.
Network performance testing is a complex subject that can encompass many variables across many different testing strategies. For our purposes, we want to understand the basics behind dummynet, so we will not be diving into the deepest levels of network performance testing - only enough to understand how to use dummynet. Also, for these tests, we will restrict our methodologies to using IP and TCP exclusively.
Setting Up for Traffic Measurement
Most of the examples in this section can be done with the architecture we used in the original lab setup in Chapter 2, copied here for reference:
Where necessary, additional virtual machines can be created and added to the bridge.
4.1. Measuring Default Throughput
The idea behind dummynet is that it lets you model and/or shape network speeds, available bandwidth, and scheduling algorithms. But first you have to know what your current transfer speeds are for the current environment (QEMU virtual machines over a FreeBSD bridge). To find out, we take a short detour to learn iperf3, the network bandwidth testing tool to perform simple transfer and bitrate calculations.
With iperf3, you can determine the effective throughput of data transfer for your system. Sometimes called "goodput", this is the basic speed the user will see for transferring data across the network - the value that is unencumbered by protocol type and overhead.
To use iperf3, ensure that the software is installed on both the firewall VM system, and the external1 VM (and external2 and external3), and that ipfw on the firewall VM is disabled (# kldunload ipfw).
The basic operation of iperf3 is as a client-server architecture, so on the external1 VM system, start the iperf3 software in server mode:
# iperf3 -s <--- run iperf3 in server mode ----------------------------------------------------------- Server listening on 5201 (test #1) ----------------------------------------------------------- . . .
Then, on the firewall VM, run the client:
# iperf3 -c 203.0.113.10 <--- connect to external1 server and send test data Connecting to host 203.0.113.10, port 5201 [ 5] local 203.0.113.50 port 19359 connected to 203.0.113.10 port 5201 [ ID] Interval Transfer Bitrate Retr Cwnd [ 5] 0.00-1.03 sec 12.5 MBytes 102 Mbits/sec 0 1.07 MBytes [ 5] 1.03-2.09 sec 13.8 MBytes 108 Mbits/sec 0 1.07 MBytes [ 5] 2.09-3.07 sec 12.5 MBytes 107 Mbits/sec 0 1.07 MBytes [ 5] 3.07-4.09 sec 12.5 MBytes 103 Mbits/sec 0 1.07 MBytes [ 5] 4.09-5.08 sec 12.5 MBytes 106 Mbits/sec 0 1.07 MBytes [ 5] 5.08-6.09 sec 12.5 MBytes 105 Mbits/sec 0 1.07 MBytes [ 5] 6.09-7.07 sec 12.5 MBytes 107 Mbits/sec 0 1.07 MBytes [ 5] 7.07-8.05 sec 12.5 MBytes 107 Mbits/sec 0 1.07 MBytes [ 5] 8.05-9.04 sec 12.5 MBytes 106 Mbits/sec 0 1.07 MBytes [ 5] 9.04-10.02 sec 12.5 MBytes 107 Mbits/sec 0 1.07 MBytes - - - - - - - - - - - - - - - - - - - - - - - - - [ ID] Interval Transfer Bitrate Retr [ 5] 0.00-10.02 sec 126 MBytes 106 Mbits/sec 0 sender [ 5] 0.00-10.02 sec 126 MBytes 106 Mbits/sec receiver iperf Done. #
A key test for measuring throughput is to send a file of data and measure the transfer speed. To create the file, we can use jot(1) on the firewall VM:
# jot -r -s "" 10000000 0 9 > A.bin
This command creates a file of random ASCII digits exactly 10,000,001 bytes long. (Note that this takes roughly 30 seconds to a minute to create the file on a QEMU virtual machine.)
To transfer the file to the server on the firewall VM we can use this command:
# iperf3 -F A.bin -c 203.0.113.10 -t 10 Connecting to host 203.0.113.10, port 5201 [ 5] local 203.0.113.50 port 51657 connected to 203.0.113.10 port 5201 [ ID] Interval Transfer Bitrate Retr Cwnd [ 5] 0.00-1.04 sec 12.5 MBytes 101 Mbits/sec 0 490 KBytes [ 5] 1.04-1.52 sec 5.81 MBytes 101 Mbits/sec 0 490 KBytes - - - - - - - - - - - - - - - - - - - - - - - - - [ ID] Interval Transfer Bitrate Retr [ 5] 0.00-1.52 sec 18.3 MBytes 101 Mbits/sec 0 sender Sent 18.3 MByte / 18.3 MByte (100%) of A.bin [ 5] 0.00-1.52 sec 18.3 MBytes 101 Mbits/sec receiver iperf Done. #
Running this command several times shows that a consistent average bitrate for throughput on this system is about 101Mbits/second - or about 18.3 MBytes/second. Your values will differ on your local machine.
We now have a baseline TCP-based "goodput" value for testing dummynet traffic shaping commands.
4.2. IPFW Commands for Dummynet
To use dummynet, load the kernel module dummynet.ko on the firewall VM:
# kldload dummynet load_dn_sched dn_sched FIFO loaded load_dn_sched dn_sched QFQ loaded load_dn_sched dn_sched RR loaded load_dn_sched dn_sched WF2Q+ loaded load_dn_sched dn_sched PRIO loaded load_dn_sched dn_sched FQ_CODEL loaded load_dn_sched dn_sched FQ_PIE loaded load_dn_aqm dn_aqm CODEL loaded load_dn_aqm dn_aqm PIE loaded #
dummynet announces the schedulers it is configured to use.
4.2.1. Simple Pipe Configuration
Recall that dummynet uses pipes, queues, and sched (schedulers) to shape traffic.
To see dummynet in action, create a pipe with limited bandwidth, and assign it to a rule matching traffic to the external1 VM:
# Load the ipfw kernel module if needed: # kldload ipfw ipfw2 (+ipv6) initialized, divert loadable, nat loadable, default to deny, logging disabled # # ipfw pipe 1 config bw 300Kbit/s # ipfw pipe 1 show 00001: 300.000 Kbit/s 0 ms burst 0 q131073 50 sl. 0 flows (1 buckets) sched 65537 weight 0 lmax 0 pri 0 droptail sched 65537 type FIFO flags 0x0 0 buckets 0 active #
The above output shows the pipe configration limiting bandwidth (bw) to 300Kbits/sec.
Recent versions of FreeBSD now use the command alias dnctl for configuration of pipes, queues, and schedulers. See dnctl(8) for details. |
Now add ipfw rules to send traffic between the firewall VM and the external1 VM through the pipe:
# ipfw add 100 check-state 00100 check-state :default # # ipfw add 1000 pipe 1 ip from any to any 01000 pipe 1 ip from any to any # # ipfw list 00100 check-state :default 01000 pipe 1 ip from any to any 65535 deny ip from any to any #
By adding the matching phrase "ip from any to any" and assigning it to pipe 1, we have configured the firewall to send all ip-based traffic through the pipe, now configured as a 300K bps link.
If we re-run the basic file transfer command for iperf3 the same as we did earlier, we can see the difference take shape:
# iperf3 -F A.bin -c 203.0.113.10 -t 10 Connecting to host 203.0.113.10, port 5201 [ 5] local 203.0.113.50 port 22303 connected to 203.0.113.10 port 5201 [ ID] Interval Transfer Bitrate Retr Cwnd [ 5] 0.00-1.00 sec 69.1 KBytes 564 Kbits/sec 0 15.6 KBytes [ 5] 1.00-2.01 sec 36.3 KBytes 294 Kbits/sec 0 18.4 KBytes [ 5] 2.01-3.01 sec 33.9 KBytes 278 Kbits/sec 0 21.3 KBytes [ 5] 3.01-4.00 sec 47.6 KBytes 394 Kbits/sec 0 24.1 KBytes [ 5] 4.00-5.01 sec 26.9 KBytes 218 Kbits/sec 0 25.5 KBytes [ 5] 5.01-6.00 sec 37.7 KBytes 312 Kbits/sec 0 27.0 KBytes [ 5] 6.00-7.00 sec 43.8 KBytes 360 Kbits/sec 0 28.4 KBytes [ 5] 7.00-8.01 sec 34.9 KBytes 282 Kbits/sec 0 29.8 KBytes [ 5] 8.01-9.00 sec 29.7 KBytes 246 Kbits/sec 0 31.2 KBytes [ 5] 9.00-10.00 sec 46.2 KBytes 378 Kbits/sec 0 32.7 KBytes - - - - - - - - - - - - - - - - - - - - - - - - - [ ID] Interval Transfer Bitrate Retr [ 5] 0.00-10.00 sec 406 KBytes 332 Kbits/sec 0 sender Sent 406 KByte / 18.3 MByte (2%) of A.bin [ 5] 0.00-10.55 sec 358 KBytes 278 Kbits/sec receiver iperf Done. #
If you get the error "iperf3: error - control socket has closed unexpectedly", simply re-run the command. |
Here, we can see that during iperf3's 10-second run, the ipfw dummynet configuration limited the transfer speed to an average of about 332 Kbits/sec, and only about 2% of the entire 10MB file was transferred.
To see how we can use dummynet to configure different link speeds, we will set up a second pipe:
# ipfw pipe 2 config bw 3Mbit/s # ipfw pipe show 00001: 300.000 Kbit/s 0 ms burst 0 q131073 50 sl. 0 flows (1 buckets) sched 65537 weight 0 lmax 0 pri 0 droptail sched 65537 type FIFO flags 0x0 0 buckets 0 active 00002: 3.000 Mbit/s 0 ms burst 0 q131074 50 sl. 0 flows (1 buckets) sched 65538 weight 0 lmax 0 pri 0 droptail sched 65538 type FIFO flags 0x0 0 buckets 0 active #
This pipe is set up to be 10 times faster (3Mb/sec instead of 300Kb/sec) than pipe 1. To test this pipe, start up the external2 VM and run iperf3 -s. Then reconfigure the ipfw rules to send to the external2 VM through pipe 2:
# ipfw list 00100 check-state :default 01000 pipe 1 ip from any to any 65535 deny ip from any to any # # ipfw delete 1000 # # ipfw add 1000 pipe 1 ip from me to 203.0.113.10 // external1 01000 pipe 1 ip from me to 203.0.113.10 # # ipfw add 1100 pipe 1 ip from 203.0.113.10 to me // external1 01100 pipe 1 ip from 203.0.113.10 to me # # ipfw add 2000 pipe 2 ip from me to 203.0.113.20 // external2 02000 pipe 2 ip from me to 203.0.113.20 # # ipfw add 2100 pipe 2 ip from 203.0.113.20 to me // external2 02100 pipe 2 ip from 203.0.113.20 to me # # ipfw list 00100 check-state :default 01000 pipe 1 ip from me to 203.0.113.10 01100 pipe 1 ip from 203.0.113.10 to me 02000 pipe 2 ip from me to 203.0.113.20 02100 pipe 2 ip from 203.0.113.20 to me 65535 deny ip from any to any #
As expected, pipe 2 is approximately 10 times faster than pipe 1:
# iperf3 -F A.bin -c 203.0.113.20 -t 10 Connecting to host 203.0.113.20, port 5201 [ 5] local 203.0.113.50 port 21569 connected to 203.0.113.20 port 5201 [ ID] Interval Transfer Bitrate Retr Cwnd [ 5] 0.00-1.00 sec 417 KBytes 3.41 Mbits/sec 0 34.1 KBytes [ 5] 1.00-2.00 sec 325 KBytes 2.66 Mbits/sec 0 45.5 KBytes [ 5] 2.00-3.00 sec 373 KBytes 3.06 Mbits/sec 0 55.5 KBytes [ 5] 3.00-4.00 sec 334 KBytes 2.73 Mbits/sec 0 64.0 KBytes [ 5] 4.00-5.00 sec 348 KBytes 2.85 Mbits/sec 0 64.0 KBytes [ 5] 5.00-6.00 sec 337 KBytes 2.76 Mbits/sec 0 64.0 KBytes [ 5] 6.00-7.00 sec 339 KBytes 2.78 Mbits/sec 0 64.0 KBytes [ 5] 7.00-8.00 sec 348 KBytes 2.85 Mbits/sec 0 64.0 KBytes [ 5] 8.00-9.00 sec 351 KBytes 2.87 Mbits/sec 0 64.0 KBytes [ 5] 9.00-10.00 sec 351 KBytes 2.88 Mbits/sec 0 64.0 KBytes - - - - - - - - - - - - - - - - - - - - - - - - - [ ID] Interval Transfer Bitrate Retr [ 5] 0.00-10.00 sec 3.44 MBytes 2.89 Mbits/sec 0 sender Sent 3.44 MByte / 18.3 MByte (18%) of A.bin [ 5] 0.00-10.12 sec 3.42 MBytes 2.83 Mbits/sec receiver iperf Done. #
Next, we change the pipe configuration without changing the ruleset. Below, the pipe 1 bandwidth is changed to the equivalent of a telecommunications T1 line in days of yore:
# ipfw pipe 1 config bw 1544Kbit/s # ipfw pipe show 00001: 1.544 Mbit/s 0 ms burst 0 q131073 50 sl. 0 flows (1 buckets) sched 65537 weight 0 lmax 0 pri 0 droptail sched 65537 type FIFO flags 0x0 0 buckets 0 active 00002: 3.000 Mbit/s 0 ms burst 0 q131074 50 sl. 0 flows (1 buckets) sched 65538 weight 0 lmax 0 pri 0 droptail sched 65538 type FIFO flags 0x0 0 buckets 0 active #
Resending the 10MB file across the T1 configured line shows these results:
# iperf3 -F A.bin -c 203.0.113.10 -t 10 Connecting to host 203.0.113.10, port 5201 [ 5] local 203.0.113.50 port 16696 connected to 203.0.113.10 port 5201 [ ID] Interval Transfer Bitrate Retr Cwnd [ 5] 0.00-1.00 sec 222 KBytes 1.82 Mbits/sec 0 65.0 KBytes [ 5] 1.00-2.00 sec 181 KBytes 1.48 Mbits/sec 0 65.0 KBytes [ 5] 2.00-3.00 sec 181 KBytes 1.48 Mbits/sec 0 65.0 KBytes [ 5] 3.00-4.00 sec 184 KBytes 1.51 Mbits/sec 0 65.0 KBytes [ 5] 4.00-5.00 sec 181 KBytes 1.48 Mbits/sec 0 65.0 KBytes [ 5] 5.00-6.00 sec 181 KBytes 1.48 Mbits/sec 0 65.0 KBytes [ 5] 6.00-7.00 sec 178 KBytes 1.46 Mbits/sec 0 65.0 KBytes [ 5] 7.00-8.00 sec 178 KBytes 1.46 Mbits/sec 0 65.0 KBytes [ 5] 8.00-9.00 sec 181 KBytes 1.48 Mbits/sec 0 65.0 KBytes [ 5] 9.00-10.00 sec 175 KBytes 1.44 Mbits/sec 0 65.0 KBytes - - - - - - - - - - - - - - - - - - - - - - - - - [ ID] Interval Transfer Bitrate Retr [ 5] 0.00-10.00 sec 1.80 MBytes 1.51 Mbits/sec 0 sender Sent 1.80 MByte / 18.3 MByte (9%) of A.bin [ 5] 0.00-10.18 sec 1.78 MBytes 1.46 Mbits/sec receiver iperf Done. #
About half of the 3Mbits/sec speed of pipe 2, again as expected.
So far, we have only been working with the pipe object. By definition, a pipe has just one queue, and it is subject to "First In First Out" (FIFO) operation. All traffic that flows through this pipe shares the same characteristics.
However, creating a pipe also does something else. It creates a default sched (scheduler) that governs the pipe:
Start with no pipes or schedulers # # ipfw pipe list # # ipfw sched list # Create a simple pipe. # ipfw pipe 1 config bw 100KBit/s # # ipfw pipe list 00001: 100.000 Kbit/s 0 ms burst 0 q131073 50 sl. 0 flows (1 buckets) sched 65537 weight 0 lmax 0 pri 0 droptail sched 65537 type FIFO flags 0x0 0 buckets 0 active # Observe the default scheduler for this pipe # ipfw sched list 00001: 100.000 Kbit/s 0 ms burst 0 sched 1 type WF2Q+ flags 0x0 0 buckets 0 active #
The default scheduler for a new pipe is of type WF2Q+
, a version of the Weighted Fair Queueing algorithm for packet transfer.
We now have a single pipe of type FIFO operation that is managed by a WF2Q+
scheduling algorithm.
The ipfw(8) man page makes note of several other scheduling algorithms. These can be selected by using the "type" keyword on the pipe command. The type keyword selects the type of scheduler applied to the pipe - not the type of the pipe itself (the pipe remains FIFO):
# ipfw pipe list # # ipfw sched list # Create a pipe and assign a scheduler of type Round Robin (Deficit Round Robin) # ipfw pipe 1 config bw 100KBit/s type rr # # ipfw pipe list 00001: 100.000 Kbit/s 0 ms burst 0 q131073 50 sl. 0 flows (1 buckets) sched 65537 weight 0 lmax 0 pri 0 droptail sched 65537 type FIFO flags 0x0 0 buckets 0 active # View the new sheduler of type RR (Deficit Round Robin) # ipfw sched list 00001: 100.000 Kbit/s 0 ms burst 0 sched 1 type RR flags 0x0 0 buckets 0 active #
*pipes and *sched*s (schedulers) are tightly bound. In fact, there is no command to delete a scheduler. The scheduler is deleted when the pipe is deleted.
Note however that the scheduler can be configured independently if desired. Below we change the scheduler type from the above type RR to QFQ, a variant of WF2Q+:
# # ipfw sched 1 config type qfq Bump qfq weight to 1 (was 0) Bump qfq maxlen to 1500 (was 0) # # ipfw sched list 00001: 100.000 Kbit/s 0 ms burst 0 sched 1 type QFQ flags 0x0 0 buckets 0 active #
There are other keywords that can be added to a pipe specification: delay, burst, profile, weight, buckets, mask, noerror, plr, queue, red or gred, codel, and pie. These are described in the ipfw(8) man page.
A contrived example might be:
Start fresh # ipfw pipe 1 delete # # ipfw pipe 1 config bw 100kbit/s delay 20 burst 2000 weight 40 buckets 256 mask src-ip 0x000000ff noerror plr 0.01 queue 75 red .3/25/30/.5 type qfq # # ipfw pipe list 00001: 100.000 Kbit/s 20 ms burst 2000 q131073 75 sl.plr 0.010000 0 flows (1 buckets) sched 65537 weight 40 lmax 0 pri 0 RED w_q 0.299988 min_th 25 max_th 30 max_p 0.500000 sched 65537 type FIFO flags 0x1 256 buckets 0 active mask: 0x00 0x000000ff/0x0000 -> 0x00000000/0x0000 # # ipfw sched list 00001: 100.000 Kbit/s 20 ms burst 2000 sched 1 type QFQ flags 0x1 256 buckets 0 active mask: 0x00 0x000000ff/0x0000 -> 0x00000000/0x0000 #
Setting up two separate pipes to send data to the same destination is overkill. It is like setting up two separate network links between the two points. While that may be desirable for redundancy or high-availability, it makes no difference for bandwidth allocation. (Yes, link aggregation is possible, but we are not considering that case here.)
What is usually needed is a way to separate traffic into different "lanes" and assign different "speed limits" to each lane. That is exactly what queues are for.
4.2.2. Simple Pipe and Queue Configuration
Before we go further, it is useful to disambiguate the two meanings of the word "queue".
In a pipe definition, by default, the pipe is assigned a queue where incoming packets are held before processing and transit. The size of this "pipe queue" is by default 50 packets, but can be changed with the queue keyword on the pipe definition:
# ipfw pipe 1 config bw 200Kbit/s # # ipfw pipe list 00001: 200.000 Kbit/s 0 ms burst 0 q131073 50 sl. 0 flows (1 buckets) sched 65537 weight 0 lmax 0 pri 0 droptail sched 65537 type FIFO flags 0x0 0 buckets 0 active # # ipfw pipe 2 config bw 200Kbit/s queue 75 # # ipfw pipe list 00001: 200.000 Kbit/s 0 ms burst 0 q131073 50 sl. 0 flows (1 buckets) sched 65537 weight 0 lmax 0 pri 0 droptail sched 65537 type FIFO flags 0x0 0 buckets 0 active 00002: 200.000 Kbit/s 0 ms burst 0 q131074 75 sl. 0 flows (1 buckets) sched 65538 weight 0 lmax 0 pri 0 droptail sched 65538 type FIFO flags 0x0 0 buckets 0 active #
In contrast, dummynet has the concept of flow queues which are virtual groupings of packets assigned to a flow according to a mask in their own definition with an ipfw queue statements.
Configuring a queue is almost as simple as configuring a pipe.
Start with a clean slate (all objects and rules deleted):
# kldunload dummynet # kldunload ipfw # kldload ipfw ipfw2 (+ipv6) initialized, divert loadable, nat loadable, default to deny, logging disabled # kldload dummynet load_dn_sched dn_sched FIFO loaded load_dn_sched dn_sched QFQ loaded load_dn_sched dn_sched RR loaded load_dn_sched dn_sched WF2Q+ loaded load_dn_sched dn_sched PRIO loaded load_dn_sched dn_sched FQ_CODEL loaded load_dn_sched dn_sched FQ_PIE loaded load_dn_aqm dn_aqm CODEL loaded load_dn_aqm dn_aqm PIE loaded # # ipfw queue 1 config pipe 1 # # ipfw queue show q00001 50 sl. 0 flows (1 buckets) sched 1 weight 0 lmax 0 pri 0 droptail
Here we see that one queue of size 50 packets was created and assigned to pipe 1. Since we did not assign a weight, the default weight is 0 (zero), which is the least weight possible. The queue currently has 0 flows, meaning that this queue has no traffic flowing through it.
Notice however, that we created the queue before we created the pipe. That is why the weight is zero. We have actually done this configuration out of order. To maintain your sanity, (and those reading the configuration after you), it is best to configure the objects in the following order:
pipes (also creates a scheduler, which can be assigned a specific scheduler type)
queues - create queues and assign weights, source and destination masks, delay, and other characteristics to the queue
Assign rules to match traffic using standard 5-tuples or as needed
dummynet also has the ability to separate out different flows within the same pipe to perform different scheduling algorithms.
When transferring a file to the external1 VM and attempting to type interactively on the external1 VM at the same time, the ability to type at speed is dramatically reduced. The file transfer packets, being much larger than interactive typing packets are hogging all the bandwidth. This effect is a well known limitation to anyone who edits documents on a remote site. Since packets are created much faster by a file transfer program than you can type, the outbound queue is almost always full of large packets, leaving your keystrokes to be separated by large amounts of file transfer data in the queue.
You should try this out on the firewall VM by resetting the pipe 1 bandwidth to 300Kbit/sec, and in one session, run iperf3 as iperf3 -c 203.0.113.10 -t 60. Then in another session, add rules for ssh traffic and ssh to external1 VM and try to enter text into a scratch file. The typing delay is almost unbearable. |
To control traffic flow between the firewall VM and any external VM host, you will need to set up individual queues to separate traffic within a pipe. queues can be either static - you define them yourself with the ipfw queue config … - or they can be dynamic. Dynamic queues are created when using the mask keyword. Masks for queues are called flow masks. The mask determines if a packet entering or leaving the firewall is selected to be entered into a queue. Consider the following example:
# ipfw pipe 1 config bw 200Kbit/s mask src-ip 0x000000ff
Each /24 host transferring data through pipe 1 (based on suitable rules) will have its own dynamic queue, all sharing the bandwidth in the pipe according to the configration of the queue.
If a different data transfer that is not related to the pipe, queue, and flow mask is started, it will not have any effect on the data in the pipe and queue. Dummynet keeps such transfers separate from the pipe and queue operations.
If instead, we wish to create separate individual queues with different characteristics such as different weights or delay, we can create static queues and then assign them to individual pipes as desired:
# # ipfw pipe 1 config bw 300kbit/s # # ipfw pipe show 00001: 300.000 Kbit/s 0 ms burst 0 q131073 50 sl. 0 flows (1 buckets) sched 65537 weight 0 lmax 0 pri 0 droptail sched 65537 type FIFO flags 0x0 0 buckets 0 active # # ipfw queue 1 config pipe 1 weight 10 mask dst-ip 0xffffffff dst-port 5201 Bump flowset buckets to 64 (was 0) # # ipfw queue 2 config pipe 1 weight 10 mask dst-ip 0xffffffff dst-port 5202 Bump flowset buckets to 64 (was 0) # # ipfw queue show q00001 50 sl. 0 flows (64 buckets) sched 1 weight 10 lmax 0 pri 0 droptail mask: 0x00 0x00000000/0x0000 -> 0xffffffff/0x1451 q00002 50 sl. 0 flows (64 buckets) sched 1 weight 10 lmax 0 pri 0 droptail mask: 0x00 0x00000000/0x0000 -> 0xffffffff/0x1452 # # ipfw add 10 allow icmp from any to any 00010 allow icmp from any to any # # ipfw add 100 check-state 00100 check-state :default # # ipfw add 1000 queue 1 tcp from me to 203.0.113.10 5201 setup keep-state 01000 queue 1 tcp from me to 203.0.113.10 5201 setup keep-state :default # # ipfw add 1100 queue 2 tcp from me to 203.0.113.20 5202 setup keep-state 01100 queue 2 tcp from me to 203.0.113.20 5202 setup keep-state :default # # ipfw list 00010 allow icmp from any to any 00100 check-state :default 01000 queue 1 tcp from me to 203.0.113.10 5201 setup keep-state :default 01100 queue 2 tcp from me to 203.0.113.20 5202 setup keep-state :default 65535 deny ip from any to any #
Later versions of FreeBSD may not return any output on ipfw queue configuration statements. The configuration is completed successfully, though without any output. |
Running
# iperf3 -c 203.0.113.10 -p 5201 -t 180 -O 30
produces the output below.
The output is the result of using the "omit" flag (-O) on the sender to ignore the first 30 seconds of output. This removes the "slow start" portion of the TCP test, and focuses instead on the "steady state" that occurs after slow start gets up to speed.
This example shows the steady state results of transmitting data through one queue - queue 1. Throughput was consistently about 277Kbits/sec.
Later versions of FreeBSD and iperf3 may differ from the display in the above figure. You can still assess the correctness of the queue setup by examining the transfer summary printed at the end of the iperf3 command output. |
During the transmission, a view of the queue status was:
# ipfw queue show q00001 50 sl. 2 flows (64 buckets) sched 1 weight 10 lmax 0 pri 0 droptail mask: 0x00 0x00000000/0x0000 -> 0xffffffff/0x1451 BKT Prot Source IP/port Dest. IP/port Tot_pkt/bytes Pkt/Byte Drp 136 ip 0.0.0.0/0 203.0.113.10/5201 2293 3425216 42 63000 0 50 ip 0.0.0.0/0 203.0.113.50/1040 752 39104 1 52 0 q00002 50 sl. 0 flows (64 buckets) sched 1 weight 10 lmax 0 pri 0 droptail mask: 0x00 0x00000000/0x0000 -> 0xffffffff/0x1452 #
The queue mask, set to show the full destination address and destination port is highlighted.
Note that port numbers are displayed in hexadecimal. A decimal/hexadecimal calculator may save you some confusion if you are looking at a lot of queue displays. |
The next example shows the result of starting two transmissions, one for each queue.
On the external1 VM, set up the command iperf3 -s -p 5201, and on external2 use the command iperf3 -s -p 5202.
Start the transfer to external1 on the firewall VM with the command:
# iperf3 -c 203.0.113.10 -p 5201 -t 180 -O 30
and start the second transfer from a different session on the firewall VM with the command:
# iperf3 -c 203.0.113.20 -p 5202 -t 180 -O 30
Notice how the queue is adjusted to accommodate the presence of a second queue of equal weight:
Since the queues were equally weighted, the result was that the transmission rate for both ended up at about 139Kbits/sec or roughly half of the previous transmission.
Queue characteristics can be changed at any time, even during an active flow. Consider the case below where, during simultaneous transmission through queues of equal weight, the queue weights were modifed as follows:
queue 1: original weight 10 modified weight 10
queue 2: original weight 10 modified weight 50
This change can be effected by the command:
# ipfw queue 2 config weight 50
The transmission rate for queue 1 dropped from an average of 139 Kbits/sec to an average of 46.3 Kbits/sec; while queue 2, after restarting the transmission with the new queue weight, expanded from an average of 139Kbits/sec to an average of 232 Kbits/sec. As expected, 232 Kbits is about five times the transmission rate of 46.3 Kbits/sec.
Note however, that the above command had a side effect:
# ipfw queue show q00001 50 sl. 0 flows (64 buckets) sched 1 weight 10 lmax 0 pri 0 droptail mask: 0x00 0x00000000/0x0000 -> 0xffffffff/0x1451 q00002 50 sl. 0 flows (1 buckets) sched 1 weight 50 lmax 0 pri 0 droptail #
The flow mask for queue 2 has been deleted. In fact, all settings not explicitly reset will revert to their default settings. Here is a complicated queue setup:
# ipfw queue 1 config pipe 1 weight 40 buckets 256 mask src-ip 0x000000ff dst-ip 0x0000ffff noerror plr 0.01 queue 75 red .3/25/30/.5 # # ipfw queue show q00001 75 sl.plr 0.010000 0 flows (256 buckets) sched 1 weight 40 lmax 0 pri 0 RED w_q 0.299988 min_th 25 max_th 30 max_p 0.500000 mask: 0x00 0x000000ff/0x0000 -> 0x0000ffff/0x0000 #
And if, similar to the previous example, we only change the weight:
# ipfw queue 1 config weight 20 # # ipfw queue show q00001 50 sl. 0 flows (1 buckets) sched 1 weight 20 lmax 0 pri 0 droptail #
All the other parameters of the queue are reset to their defaults. Therefore, it is best to retain the original commands used to construct queues, pipes, and schedulers, so that if you are only changing one parameter, all other parameters can be replicated on the command line. Otherwise you will have to reconstruct the parameters from the output of ipfw queue show which can be quite tedious.
4.2.3. Dynamic Pipes
Here, we note the simplest setup for pipes creates dynamic pipes when needed:
# ipfw pipe 1 config bw 300kbit/s weight 10 mask src-ip 0x0000ffff dst-ip 0xffffffff Bump sched buckets to 64 (was 0) # # ipfw pipe show 00001: 300.000 Kbit/s 0 ms burst 0 q131073 50 sl. 0 flows (1 buckets) sched 65537 weight 10 lmax 0 pri 0 droptail sched 65537 type FIFO flags 0x1 64 buckets 0 active mask: 0x00 0x0000ffff/0x0000 -> 0xffffffff/0x0000 # # ipfw list 00050 allow icmp from any to any 00100 check-state :default 65535 deny ip from any to any # # ipfw add 1000 pipe 1 tcp from me to 203.0.113.0/24 5201-5203 setup keep-state 01000 pipe 1 tcp from me to 203.0.113.0/24 5201-5203 setup keep-state :default # # ipfw list 01000 pipe 1 tcp from me to 203.0.113.0/24 5201-5203 setup keep-state :default 65535 deny ip from any to any #
Sending some data with this configuration:
# ipfw pipe show 00001: 300.000 Kbit/s 0 ms burst 0 q131073 50 sl. 0 flows (1 buckets) sched 65537 weight 10 lmax 0 pri 0 droptail sched 65537 type FIFO flags 0x1 64 buckets 4 active mask: 0x00 0x0000ffff/0x0000 -> 0xffffffff/0x0000 BKT Prot Source IP/port Dest. IP/port Tot_pkt/bytes Pkt/Byte Drp 6 ip 0.0.10.10/0 203.0.113.50/0 236 12272 0 0 0 78 ip 0.0.10.50/0 203.0.113.10/0 1493 2225216 43 64500 0 80 ip 0.0.10.50/0 203.0.113.20/0 1355 2018216 42 63000 0 58 ip 0.0.10.20/0 203.0.113.50/0 366 19032 0 0 0 # # ipfw list 00050 allow icmp from any to any 00100 check-state :default 01000 pipe 1 tcp from me to 203.0.113.0/24 5201-5203 setup keep-state :default 65535 deny ip from any to any #
All three transmissions running together, single pipe:
# ipfw pipe show 00001: 300.000 Kbit/s 0 ms burst 0 q131073 50 sl. 0 flows (1 buckets) sched 65537 weight 10 lmax 0 pri 0 droptail sched 65537 type FIFO flags 0x1 64 buckets 6 active mask: 0x00 0x0000ffff/0x0000 -> 0xffffffff/0x0000 BKT Prot Source IP/port Dest. IP/port Tot_pkt/bytes Pkt/Byte Drp 6 ip 0.0.10.10/0 203.0.113.50/0 588 30576 0 0 0 78 ip 0.0.10.50/0 203.0.113.10/0 1508 2247716 43 64500 0 80 ip 0.0.10.50/0 203.0.113.20/0 1357 2021216 43 64500 0 90 ip 0.0.10.50/0 203.0.113.30/0 1322 1981552 41 61500 0 46 ip 0.0.10.30/0 203.0.113.50/0 34 1768 0 0 0 58 ip 0.0.10.20/0 203.0.113.50/0 702 36504 0 0 0
Because of the ipfw rule:
01000 pipe 1 tcp from me to 203.0.113.0/24 5201-5203 setup keep-state :default
All are getting 290 Kbit/sec from iperf3 and they are all sharing the pipe equally.
If we change iperf3 to send to different ports for each system (5201, 5202, 5203) to external1, external2, and external3 VMs respectively there is no change. It is only with queues, where you can set the individual flow rate, that you can effect change.
Below are examples of different masks and their effect on traffic flow:
* dst-ip 0x0000ffff # ipfw pipe show 00001: 300.000 Kbit/s 0 ms burst 0 q131073 50 sl. 0 flows (1 buckets) sched 65537 weight 0 lmax 0 pri 0 droptail sched 65537 type FIFO flags 0x1 64 buckets 4 active mask: 0x00 0x00000000/0x0000 -> 0x0000ffff/0x0000 BKT Prot Source IP/port Dest. IP/port Tot_pkt/bytes Pkt/Byte Drp 10 ip 0.0.0.0/0 0.0.10.10/0 1183 1760218 43 64500 0 20 ip 0.0.0.0/0 0.0.10.20/0 974 1446718 42 63000 0 30 ip 0.0.0.0/0 0.0.10.30/0 688 1017718 35 52500 0 50 ip 0.0.0.0/0 0.0.10.50/0 1717 89284 0 0 0 * dst-ip 0xffffffff # ipfw pipe show 00001: 300.000 Kbit/s 0 ms burst 0 q131073 50 sl. 0 flows (1 buckets) sched 65537 weight 0 lmax 0 pri 0 droptail sched 65537 type FIFO flags 0x1 64 buckets 4 active mask: 0x00 0x00000000/0x0000 -> 0xffffffff/0x0000 BKT Prot Source IP/port Dest. IP/port Tot_pkt/bytes Pkt/Byte Drp 18 ip 0.0.0.0/0 203.0.113.50/0 402 20888 0 0 0 42 ip 0.0.0.0/0 203.0.113.10/0 144 204722 0 0 0 52 ip 0.0.0.0/0 203.0.113.20/0 359 525971 0 0 0 62 ip 0.0.0.0/0 203.0.113.30/0 562 843000 37 55500 0 * src-ip 0x0000ffff # ipfw pipe show 00001: 300.000 Kbit/s 0 ms burst 0 q131073 50 sl. 0 flows (1 buckets) sched 65537 weight 0 lmax 0 pri 0 droptail sched 65537 type FIFO flags 0x1 64 buckets 4 active mask: 0x00 0x0000ffff/0x0000 -> 0x00000000/0x0000 BKT Prot Source IP/port Dest. IP/port Tot_pkt/bytes Pkt/Byte Drp 20 ip 0.0.10.10/0 0.0.0.0/0 361 19348 0 0 0 100 ip 0.0.10.50/0 0.0.0.0/0 2102 3079974 36 54000 27 40 ip 0.0.10.20/0 0.0.0.0/0 193 10416 0 0 0 60 ip 0.0.10.30/0 0.0.0.0/0 47 2612 0 0 0 * mask src-ip 0x0000ffff dst-ip 0x0000ffff <-only one keyword mask needs to be specified # ipfw pipe show 00001: 300.000 Kbit/s 0 ms burst 0 q131073 50 sl. 0 flows (1 buckets) sched 65537 weight 0 lmax 0 pri 0 droptail sched 65537 type FIFO flags 0x1 64 buckets 6 active mask: 0x00 0x0000ffff/0x0000 -> 0x0000ffff/0x0000 BKT Prot Source IP/port Dest. IP/port Tot_pkt/bytes Pkt/Byte Drp 14 ip 0.0.10.30/0 0.0.10.50/0 253 13156 0 0 0 26 ip 0.0.10.20/0 0.0.10.50/0 61 3172 0 0 0 38 ip 0.0.10.10/0 0.0.10.50/0 771 40094 0 0 0 110 ip 0.0.10.50/0 0.0.10.10/0 853 1265218 40 60000 0 112 ip 0.0.10.50/0 0.0.10.20/0 723 1083052 37 55500 0 122 ip 0.0.10.50/0 0.0.10.30/0 644 951718 34 51000 0 * mask src-ip 0x0000ffff dst-ip 0x0000ffff dst-port 5201 # ipfw pipe show 00001: 300.000 Kbit/s 0 ms burst 0 q131073 50 sl. 0 flows (1 buckets) sched 65537 weight 0 lmax 0 pri 0 droptail sched 65537 type FIFO flags 0x1 64 buckets 6 active mask: 0x00 0x0000ffff/0x0000 -> 0x0000ffff/0x1451 BKT Prot Source IP/port Dest. IP/port ot_pkt/bytes Pkt/Byte Drp 204 ip 0.0.10.50/0 0.0.10.10/5201 2132 3183718 43 64500 0 14 ip 0.0.10.30/0 0.0.10.50/4096 823 42796 0 0 0 210 ip 0.0.10.50/0 0.0.10.20/5201 2001 2987218 43 64500 0 152 ip 0.0.10.20/0 0.0.10.50/4161 663 34476 0 0 0 216 ip 0.0.10.50/0 0.0.10.30/5201 1981 2957218 43 64500 0 164 ip 0.0.10.10/0 0.0.10.50/65 471 24492 0 0 0 * mask src-ip 0xffffffff dst-ip 0xffffffff # ipfw pipe 1 show 00001: 300.000 Kbit/s 0 ms burst 0 q131073 50 sl. 0 flows (1 buckets) sched 65537 weight 0 lmax 0 pri 0 droptail sched 65537 type FIFO flags 0x1 64 buckets 6 active mask: 0x00 0xffffffff/0x0000 -> 0xffffffff/0x0000 BKT Prot Source IP/port Dest. IP/port Tot_pkt/bytes Pkt/Byte Drp 64 ip 203.0.113.50/0 203.0.113.20/0 1215 1808218 43 64500 0 74 ip 203.0.113.50/0 203.0.113.30/0 1023 1533052 43 64500 0 22 ip 203.0.113.10/0 203.0.113.50/0 746 38792 0 0 0 94 ip 203.0.113.50/0 203.0.113.10/0 1863 2780218 42 63000 0 42 ip 203.0.113.20/0 203.0.113.50/0 481 25012 0 0 0 62 ip 203.0.113.30/0 203.0.113.50/0 159 8268 0 0 0
4.2.4. Other Pipe and Queue Commands
To delete pipes and queues use the following syntax:
For queues, specify the queue number on the command line: # ipfw queue delete 1 For pipes, specify the pipe number on the command line: # ipfw pipe delete 1
Note however that:
# ipfw delete pipe 1 <----- does not throw error, and does not delete the pipe.
The same is true for the corresponding queue keyword. You should take care to use the proper syntax.
You can delete a pipe with a pipe statement still in the ruleset. ipfw will not throw an error - but any data transfer matching the pipe statement will not work. |
scheds (schedulers) and pipes are tightly bound. To delete a scheduler, you must first delete the pipe. You can then re-create the pipe if needed. The scheduler for the new pipe is reset to the default scheduler.
To change the scheduler type: # ipfw sched config 1 type wfq2 # or rr or any other sched type
4.3. Adding Additional Virtual Machines
Up to this point, we have been using only two or three virtual machines for exploring ipfw. The later material in this book requires the use of several additional virtual machines.
In the NAT chapter, we will use several more VMs for:
4.3.1. Setting Up The Entire IPFW Lab
A suggested host machine file directory layout for these machines is shown below. All scripts use relative path names, so the directory can be located anywhere.
~/ipfw /SCRIPTS _CreateAllVMs.sh (script to create QEMU disks images) mkbr.sh (script to create bridge and tap devices) vm_envs.sh (script to manage all parameters) dnshost.sh (script for running a BIND 9 DNS server) external1.sh (scripts for running 'external VM host' VMs) external2.sh " external3.sh " firewall.sh (script for running a firewall VM) firewall2.sh (script for running a firewall VM) internal.sh (script for running an internal VM) v6only.sh (script for running an IPv6 only VM) jail1.sh (script for running a jails VM) /ISO fbsd.iso (link to latest FreeBSD install iso) /VM dnshost.qcow2 (QEMU disk image for a BIND 9 DNS server) external1.qcow2 (QEMU disk image for 'external' hosts) external2.qcow2 " external3.qcow2 " firewall.qcow2 (QEMU disk image for the 'firewall') firewall2.qcow2 (QEMU disk image for the 'firewall2' VM) internal.qcow2 (QEMU disk image for an internal VM) v6only.qcow2 (QEMU disk image for an IPv6 only VM) jail1.qcow2 (QEMU disk image for the jails VM) /BMP dns_splash_640x480.bmp (QEMU splash image) external1_splash_640x480.bmp " external2_splash_640x480.bmp " external3_splash_640x480.bmp " firewall_splash_640x480.bmp " firewall2_splash_640x480.bmp " internal_splash_640x480.bmp " v6only_splash_640x480.bmp " jail1_splash_640x480.bmp "
Finish setting up the entire lab by referring to the instructions found in Section Quick Start.
Also, ensure each virtual machine is set up to boot a serial console by adding "console=comconsole" to /boot/loader.conf.
Chapter 5. ipfw NAT
Network Address Translation (NAT) is the process of changing the source or destination address of a packet as it flows through the firewall. This is done chiefly to segregate internal networks and subnets from external networks.
FreeBSD has two capabilities for NAT - natd(8) a daemon process that can perform these translations, and in-kernel NAT with ipfw.
Both of these capabilities use the libalias(3)
library.
This section will focus primarily on in-kernel NAT with ipfw.
5.1. General Procedures for Working NAT Examples
In this section we will use more than two virtual machines (VMs). If you have followed the directions on Setting Up the Entire IPFW Lab, we can begin with the setup for simple NAT.
The examples in this section and later sections grow increasingly complex. Follow this standard procedure for startup with each new example:
On the host, begin by setting up the bridge and tap setup needed for the examples. Use mkbr.sh to configure bridge and tap devices on the host. Examine the figure, and run the script with all bridges and taps accounted for.
Start up the required VMs. Use runvm.sh to start up several VMs at one time.
On each VM, set up the required addressing. Check the diagram in each Section for addressing requirements.
Ensure all VMs have connectivity to their local network peers.
If there are additional scripts to load onto the firewall, external, internal, dnshost, or v6only VMs, load them.
If there are specific DNS entries that are required for an example, load them into the dnshost and test the entries from another VM.
Other VMs in some examples require adding additional routes.
On the firewall VM, unload and reload the firewall: (kldunload ipfw and kldload ipfw).
Check whether any sysctl entries are required for the example.
Follow the procedure given for each section.
Troubleshoot as necessary.
5.2. Setting Up for Simple NAT
Shut down the existing VMs from the previous examples and reload ipfw. To set up the correct network bridge and tap architecture as shown in the figure above, use this command:
# sudo /bin/sh mkbr.sh reset bridge0 tap1 tap4 bridge1 tap0 tap5
Restart the desired VMs with:
# /bin/sh runvm.sh firewall external1 internal
You will have to reconfigure the network addressing. Use the above figure to set up the correct addresses for each VM and ensure you can ping adjacent interfaces.
For routing, the external1 VM should have the default route set to 203.0.113.50.
The internal VM should have its default route set to 10.10.10.50.
The firewall should have its default route set to 203.0.113.10 (external1) since we want all traffic to exit via the firewall em1
interface.
The firewall should already be set up for IP forwarding (sysctl net.inet.ip.forwarding=1), but if not, set the sysctl as indicated. You should be able to ping em0 on external1 VM from the internal VM host and vice-versa. Check all addressing, the host bridge and tap devices, and the sysctl net.inet.ip.forwarding=1 on the firewall if something is not working.
On the firewall VM, restart ipfw with
# kldload ipfw
To use in-kernel NAT, you must first load the ipfw_nat kernel module:
# kldload ipfw_nat
Running kldstat should now show output similar to:
# kldstat Id Refs Address Size Name 1 11 0xffffffff80200000 1f370e8 kernel 2 1 0xffffffff82818000 3220 intpm.ko 3 1 0xffffffff8281c000 2178 smbus.ko 4 2 0xffffffff8281f000 27450 ipfw.ko 5 1 0xffffffff82847000 42d0 ipfw_nat.ko 6 1 0xffffffff8284c000 c962 libalias.ko #
We are now ready to explore ipfw_nat.
Similar to other ipfw entities such as pipes and queues, ipfw_nat works with a NAT object. A NAT object is a single entry in the packet aliasing database.
We first create a NAT object:
# ipfw nat 25 config ip 198.51.100.50 ipfw nat 25 config ip 198.51.100.50 # # ipfw nat show config ipfw nat 25 config ip 198.51.100.50
Note that the NAT object identifier must be numeric, not alphabetic or alphanumeric.
A NAT object identifier such as foo
or 25foo
will be rejected by ipfw.
Next, load two rules that will use that instance:
# ipfw add 1000 nat 25 tcp from any to any # ipfw add 2000 nat 25 icmp from any to any
Listing the ruleset shows the NAT object and the rule body.
# ipfw list 01000 nat 25 tcp from any to any 02000 nat 25 icmp from any to any 65535 deny ip from any to any
We now have an ipfw_nat instance in the packet aliasing database and rules that will engage that instance. This is generally referred to as "static NAT".
The ipfw_nat instance will replace the IP source address of any packet exiting the firewall with 198.51.100.50, provided that packet has reached the ipfw_nat rule and matches its configuration.
To test, start tcpdump(1) on the host system monitoring bridge0. (You should again ensure that the host system is not running a firewall.)
host_system# tcpdump -n -i bridge0 -v
Then, from the firewall VM, telnet(1) to any IP address not used in our lab:
# telnet 172.16.10.10 Trying 172.16.10.10... ^C
All you need is a few seconds to attempt the connection (which will not succeed anyway).
Examining the host tcpdump
output we see the following:
host_system# tcpdump -n -i bridge0 -v tcpdump: listening on bridge0, link-type EN10MB (Ethernet), snapshot length 262144 bytes 19:58:34.099782 IP (tos 0x10, ttl 64, id 0, offset 0, flags [DF], proto TCP (6), length 60) 198.51.100.50.62143 > 172.16.10.10.23: Flags [S], cksum 0x89d4 (correct), seq 3107170690, win 65535, options [mss 1460,nop,wscale 6,sackOK,TS val 384725297 ecr 0], length 0 19:58:38.300043 IP (tos 0x10, ttl 64, id 0, offset 0, flags [DF], proto TCP (6), length 60) 198.51.100.50.62143 > 172.16.10.10.23: Flags [S], cksum 0x796b (correct), seq 3107170690, win 65535, options [mss 1460,nop,wscale 6,sackOK,TS val 384729498 ecr 0], length 0 19:58:46.500217 IP (tos 0x10, ttl 64, id 0, offset 0, flags [DF], proto TCP (6), length 60) 198.51.100.50.62143 > 172.16.10.10.23: Flags [S], cksum 0x5964 (correct), seq 3107170690, win 65535, options [mss 1460,nop,wscale 6,sackOK,TS val 384737697 ecr 0], length 0 ^C
The source address has been changed from 203.0.113.50 to 198.51.100.50 as per our ipfw_nat instance. Note however, that with our configuration binding NAT to an IP address, as opposed to an interface, the NAT aliasing takes place on all configured interfaces, internal and external. You can verify this by repeating the above tcpdump on bridge1, and running a similar command for an existing host on the internal network. This time the destination sends a TCP reset (RST), since the packet reached the destination but the service on the destination was not open.
# telnet 10.10.10.20
host_system# tcpdump -n -i bridge1 -v tcpdump: listening on bridge1, link-type EN10MB (Ethernet), snapshot length 262144 bytes 20:12:13.706505 IP (tos 0x10, ttl 64, id 0, offset 0, flags [DF], proto TCP (6), length 60) 198.51.100.50.32825 > 10.10.10.20.23: Flags [S], cksum 0x6039 (correct), seq 1314409263, win 65535, options [mss 1460,nop,wscale 6,sackOK,TS val 3924141446 ecr 0], length 0 20:12:13.710494 IP (tos 0x0, ttl 64, id 0, offset 0, flags [DF], proto TCP (6), length 40) 10.10.10.20.23 > 198.51.100.50.32825: Flags [R.], cksum 0x5774 (correct), seq 0, ack 1314409264, win 0, length 0 20:12:29.573756 IP (tos 0x10, ttl 64, id 0, offset 0, flags [DF], proto TCP (6), length 60) ^C
To specify that only the outside interface is to be NATed, use the keyword if on the ipfw NAT configuration statement and specify the correct external interface:
# ipfw nat 25 config if em1 # ipfw nat show config ipfw nat 25 config if em1
Note that you cannot use the ip ip_addr and if interf_name options at the same time on the same NAT instance - you must use one or the other.
What happens in this case is that the NAT instance will ensure that the IP address of interface em1 will always be used on traffic exiting through that interface - even if the address changes (because of DHCP or an administrative addressing change):
Traffic destined externally from the internal VM host via:
root@internal:~ # telnet 172.16.10.10 Trying 172.16.10.10... ^C
On the FreeBSD host:
host_system# tcpdump -n -i bridge0 -v tcpdump: listening on bridge0, link-type EN10MB (Ethernet), snapshot length 262144 bytes 20:24:41.147755 IP (tos 0x10, ttl 63, id 0, offset 0, flags [DF], proto TCP (6), length 60) 203.0.113.50.40001 > 172.16.10.10.23: Flags [S], cksum 0x5962 (correct), seq 950423268, win 65535, options [mss 1460,nop,wscale 6,sackOK,TS val 2027118491 ecr 0], length 0 20:24:42.189806 IP (tos 0x10, ttl 63, id 0, offset 0, flags [DF], proto TCP (6), length 60) 203.0.113.50.40001 > 172.16.10.10.23: Flags [S], cksum 0x554b (correct), seq 950423268, win 65535, options [mss 1460,nop,wscale 6,sackOK,TS val 2027119538 ecr 0], length 0 20:24:44.394747 IP (tos 0x10, ttl 63, id 0, offset 0, flags [DF], proto TCP (6), length 60) 203.0.113.50.40001 > 172.16.10.10.23: Flags [S], cksum 0x4caa (correct), seq 950423268, win 65535, options [mss 1460,nop,wscale 6,sackOK,TS val 2027121747 ecr 0], length 0 ^C
Though it does not look like it, ipfw is translating the packets as they exit the firewall.
Consider this exchange where the internal VM host pings the external1 VM:
root@internal:~ # ping 203.0.113.10 PING 203.0.113.10 (203.0.113.10): 56 data bytes 64 bytes from 203.0.113.10: icmp_seq=0 ttl=63 time=2.742 ms 64 bytes from 203.0.113.10: icmp_seq=1 ttl=63 time=2.675 ms ^C
The traffic on the internal bridge (bridge1) shows the packets from the internal1 VM:
host_system# tcpdump -n -i bridge1 -v tcpdump: listening on bridge1, link-type EN10MB (Ethernet), snapshot length 262144 bytes 20:29:27.048162 IP (tos 0x0, ttl 64, id 58916, offset 0, flags [none], proto ICMP (1), length 84) 10.10.10.20 > 203.0.113.10: ICMP echo request, id 15077, seq 0, length 64 20:29:27.052446 IP (tos 0x0, ttl 63, id 36018, offset 0, flags [none], proto ICMP (1), length 84) 203.0.113.10 > 10.10.10.20: ICMP echo reply, id 15077, seq 0, length 64 20:29:28.104133 IP (tos 0x0, ttl 64, id 58917, offset 0, flags [none], proto ICMP (1), length 84) 10.10.10.20 > 203.0.113.10: ICMP echo request, id 15077, seq 1, length 64 20:29:28.105732 IP (tos 0x0, ttl 63, id 36019, offset 0, flags [none], proto ICMP (1), length 84) 203.0.113.10 > 10.10.10.20: ICMP echo reply, id 15077, seq 1, length 64
whereas the traffic on the external bridge (bridge0) shows the correct translation:
host_system# tcpdump -n -i bridge0 -v tcpdump: listening on bridge0, link-type EN10MB (Ethernet), snapshot length 262144 bytes 20:33:19.695939 IP (tos 0x0, ttl 63, id 58919, offset 0, flags [none], proto ICMP (1), length 84) 203.0.113.50 > 203.0.113.10: ICMP echo request, id 58206, seq 0, length 64 20:33:19.696546 IP (tos 0x0, ttl 64, id 36021, offset 0, flags [none], proto ICMP (1), length 84) 203.0.113.10 > 203.0.113.50: ICMP echo reply, id 58206, seq 0, length 64 20:33:20.715148 IP (tos 0x0, ttl 63, id 58920, offset 0, flags [none], proto ICMP (1), length 84) 203.0.113.50 > 203.0.113.10: ICMP echo request, id 58206, seq 1, length 64 20:33:20.715824 IP (tos 0x0, ttl 64, id 36022, offset 0, flags [none], proto ICMP (1), length 84) 203.0.113.10 > 203.0.113.50: ICMP echo reply, id 58206, seq 1, length 64 ^C
The unreg_only and unreg_cgn configuration options allows you to bypass the NAT operation if the source IP of the packet is not one of the RFC 1918 addresses (unreg_only) or the RFC 6598 addresses (unreg_cgn - carrier grade NAT). In these cases, the original source address will be maintained in the packet, even though there is an ipfw_nat instance and a matching rule.
# ipfw nat 25 show config ipfw nat 25 config if em1 # # ipfw nat 25 config if em1 unreg_only ipfw nat 25 config if em1 unreg_only # # ipfw nat 25 show config ipfw nat 25 config if em1 unreg_only #
To try the unreg_only option, on the internal VM, change its IP address on em0
to a registered number, say 140.140.140.140/24, and change the corresponding link on the firewall (em1
) to a compatible address - 140.140.140.1/24.
The internal VM will need a new default route: 140.140.140.1
root@internal:~ # ifconfig em0 140.140.140.140/24 root@internal:~ # route add default 140.140.140.1 add net default: gateway 140.140.140.1 root@internal:~ # and on the firewall root@firewall:~ # ifconfig em0 140.140.140.1/24
From the internal VM, try to ping an external address not in our lab:
# ping 5.5.5.5
and observe on the host system that the ipfw_nat instance did not replace the source address with the configured IP:
host_system# tcpdump -n -i bridge0 -v tcpdump: listening on bridge0, link-type EN10MB (Ethernet), snapshot length 262144 bytes 21:07:18.154319 IP (tos 0x0, ttl 63, id 58943, offset 0, flags [none], proto ICMP (1), length 84) 140.140.140.140 > 5.5.5.5: ICMP echo request, id 38569, seq 0, length 64 21:07:19.180094 IP (tos 0x0, ttl 63, id 58944, offset 0, flags [none], proto ICMP (1), length 84) 140.140.140.140 > 5.5.5.5: ICMP echo request, id 38569, seq 1, length 64 21:07:20.194988 IP (tos 0x0, ttl 63, id 58945, offset 0, flags [none], proto ICMP (1), length 84) 140.140.140.140 > 5.5.5.5: ICMP echo request, id 38569, seq 2, length 64
Not all the options available to ipfw_nat are described in the NAT section of the ipfw(8) man page.
Some of the options usable from natd(8) are available to ipfw_nat. These include:
redirect_port proto targetIP:targetPORT[-targetPORT] [aliasIP:]aliasPORT[-aliasPORT] [remoteIP[:remotePORT[-remotePORT]]] redirect_proto proto localIP [publicIP [remoteIP]] redirect_address localIP publicIP
The below options are used for Load Sharing NAT (LSNAT) as described in RFC 2391.
redirect_port proto targetIP:targetPORT[,targetIP:targetPORT[,...]] [aliasIP:]aliasPORT [remoteIP[:remotePORT]] redirect_address localIP[,localIP[,...]] publicIP
We discuss LSNAT
in the next section.
5.3. Setting Up for LSNAT
For this example, we will use the three VMs external1, external2, and external3 and pretend they are on the inside of the network; and our internal VM is on the outside of the network.
The figure below shows the architecture setup for working with LSNAT.
As before, shutdown all virtual machines and rebuild the network from scratch.
Use this command to set up the network bridge and tap architecture.
# sudo /bin/sh mkbr.sh reset bridge0 tap4 tap5 bridge1 tap0 tap1 tap2 tap3
Note that the host interface is not needed for this example.
Restart the virtual machines with:
# /bin/sh runvm.sh firewall internal external1 external2 external3
or start them up individually.
Configure each virtual machine to ensure its network configuration matches the above figure and test connectivity between adjacent systems with ping(8).
Throughout this section, remember that the "external" VMs are now internal web servers load balancing between .10, .20, .30, and the "internal" server VM is the outside host accessing the internal webservers.
On each inside VM the following commands are necessary to perform the examples in this section:
# route delete default # # route add default 10.10.10.50
On the outside VM perform these commands:
# route delete default # # route add default 198.51.100.50
Also, on each VM, edit the nginx index.html page and insert a line of text that has the VM name or IP address of the VM - something like this:
File: /usr/local/www/nginx/index.html: <h1>Welcome to nginx!</h1> <p>If you see this page, the nginx web server is successfully installed and working. Further configuration is required.</p> <p> This is VM EXTERNAL1</p>
and start nginx on each inside VM:
# service nginx onestart Performing sanity check on nginx configuration: nginx: the configuration file /usr/local/etc/nginx/nginx.conf syntax is ok nginx: configuration file /usr/local/etc/nginx/nginx.conf test is successful Starting nginx. #
With no ipfw loaded on the firewall, you should be able to ping all inside addresses (10.10.10.10, .20, .30) from the outside host (198.51.100.20). You should also be able to access each web server via:
# lynx 203.0.113.10 # (or .20 or .30)
We now start on LSNAT configuration.
5.3.1. Setting up LSNAT- One address (10.10.10.10)
We begin with loading ipfw and ipfw_nat on the firewall VM
# kldload ipfw # # kldload ipfw_nat
The first configuration is similar to static NAT, though from the outside to the inside. The command redirects incoming traffic from the outside VM sent to destination IP 3.3.3.3 to inside VM 10.10.10.10.
# ipfw nat 25 config redirect_addr 10.10.10.10 3.3.3.3 ipfw nat 25 config redirect_addr 10.10.10.10 3.3.3.3
Next create a ruleset that utilizes this NAT instance:
# ipfw add 50 check-state # ipfw add 1000 nat 25 tcp from any to any # # ipfw list 00050 check-state :default 01000 nat 25 tcp from any to any 65535 deny ip from any to any #
Do not use the setup keyword on the ipfw rule referencing LSNAT. The setup keyword causes the final ACK of the TCP 3-way handshake to be never received and the connection is never established. |
From the outside VM, access the web server using:
# lynx 3.3.3.3
brings up the web page on 10.10.10.10.
NAT with one address is working.
5.3.2. Engaging Multiple Hosts With LSNAT
Next, reconfigure the nat 25 instance to utilize all of the inside hosts:
# ipfw nat 25 config redirect_addr 10.10.10.10,10.10.10.20,10.10.10.30 3.3.3.3
(Note that adding a modification to a NAT instance just overwrites the existing instance. It does not create a new instance with the same number.)
On the outside VM, running lynx 3.3.3.3 repeatedly retrieves the home page of each internal server - in round-robin fashion, without regard for any network load, or server utilization.
In the lynx browser, you can reload the current page by pressing Ctl+R. |
# ipfw nat 25 show config ipfw nat 25 config log redirect_addr 10.10.10.10,10.10.10.20,10.10.10.30 3.3.3.3 #
By adding a rule to redirect icmp traffic, both icmp and tcp will be load shared across the firewall.
# ipfw add 2000 nat 25 icmp from any to any # # ipfw list 00050 check-state :default 01000 nat 25 tcp from any to any 02000 nat 25 icmp from any to any 65535 deny ip from any to any
You can test this by running tcpdump -n -i em0 on each inside VM, and running ping -c 1 3.3.3.3 on the outside VM a few times. The incoming ping will hit each inside VM in turn.
However, if you run ping 3.3.3.3, the result is that these pings hit only one internal VM. The reason is that the aliasing engine treats ICMP differently from TCP and UDP. The aliasing engine recognized the ICMP id number, and if this number does not change, it uses the same alias. If the command ping -c 1 3.3.3.3 is used repeatedly, the ICMP id number changes, and this creates a new entry in the aliasing database resulting in redirection to a different VM.
It is common to want to balance the load across servers according to certain characteristics such as system load. This is possible - manually - by reconfiguring the NAT statement. You can add multiple instances of the same host to give that host more traffic. Consider this ruleset created with the Unix line continuation character '\' to close the space between successive IP addresses except for the last one and the alias address:
# ipfw nat 25 config log redirect_addr \ 10.10.10.30,\ 10.10.10.20,10.10.10.20,\ 10.10.10.10,10.10.10.10,10.10.10.10,10.10.10.10 3.3.3.3
This configuration shifts the NAT load heavily toward 10.10.10.10 and moderately toward 10.10.10.20, with 10.10.10.30 having a lot less traffic. Repeat the above single ping example above to see the result. While this works, it is a bit of a hack.
It would be better to have a range assignment feature similar to the sparse address feature already in ipfw, something like:
# ipfw nat 25 config redirect_addr 10.10.10.0/24{10,20-25,30-50} 3.3.3.3 ipfw: unknown host 10.10.10.0/24{10
but this feature does not work with LSNAT.
However, it is possible to use the prob keyword to address load balancing. In a rule with the prob keyword, if the rule matches and the probability is "true", the action of the rule is taken and processing stops for that packet. If the rule matches, and the probability is "not true", the action is not taken, and processing continues with the next rule. You can verify this with a simple test ruleset and the ucont.sh shell rule from an external host.
03000 prob 0.200000 allow udp from any to me 5656 // set probability to 20% chance of matching 04000 count udp from any to me // count how many were not chosen by rule 3000 05000 prob 0.400000 allow udp from any to me 5656 // set probability to 40% chance of matching 06000 count udp from any to me // count how many were not chosen by 3000 and 5000 07000 prob 0.999000 allow udp from any to me 5656 // set probability to 99.9% chance of matching 08000 count udp from any to me // count how many were not chosen by all 3 rules 09000 allow udp from any to me 5656 // unconditional matching 65535 deny ip from any to any // default rule deny
After a run of 200 entries from sh ucont.sh 5656 1 the counts are:
03000 47 3314 prob 0.200000 allow udp from any to me 5656 04000 153 10776 count udp from any to me 05000 64 4505 prob 0.400000 allow udp from any to me 5656 06000 89 6271 count udp from any to me 07000 89 6271 prob 0.999000 allow udp from any to me 5656 08000 0 0 count udp from any to me 09000 0 0 allow udp from any to me 5656 65535 0 0 deny ip from any to any
From the above data, out of 200 packets sent from ucont.sh, 47 were matched by rule 3000, but 153 were not matched (rule 4000). Then, 64 were matched at rule 5000, but 89 were not matched. Finally, 89 where matched at rule 7000.
If you duplicate this example and find some packets hitting the default deny rule (65535), delete the host interface from the bridge and re-run the test. You are then unlikely to have any stray UDP packets hitting the default rule. |
While the above works for UDP, it does not work for TCP. The TCP 3-way handshake is broken because some packets will match, but others will not.
Other NAT Keywords
The other keywords in the NAT section of ipfw(8) are straightforward:
deny_in : deny incoming packets
same_ports : keep the same ports after redirection
reset : clear the aliasing table when the address changes
reverse : reverse the direction of the NAT
proxy_only : packet aliasing is not performed
skip_global
global
tablearg : discussed in Understanding the Word Tablearg
Chapter 6. IPv6 Network Address Translation (IPv6NAT)
ipfw supports both stateful and stateless IPv6 / IPv4 translation.
From the ipfw(8) man page:
Stateful translation
ipfw supports in-kernel IPv6/IPv4 network address and protocol transla-
tion. Stateful NAT64 translation allows IPv6-only clients to contact
IPv4 servers using unicast TCP, UDP or ICMP protocols. One or more IPv4
addresses assigned to a stateful NAT64 translator are shared among sev-
eral IPv6-only clients. When stateful NAT64 is used in conjunction with
DNS64, no changes are usually required in the IPv6 client or the IPv4
server. The kernel module ipfw_nat64 should be loaded or kernel should
have options IPFIREWALL_NAT64
to be able use stateful NAT64 translator.
Stateful translation is suitable for deployment at the client side or at the service provider, allowing IPv6-only client hosts to reach remote IPv4-only nodes.
Stateless translation is appropriate when a NAT64 translator is used in front of IPv4-only servers to allow them to be reached by remote IPv6-only clients.
Specific requirements for these translation services are found in a collection of RFCs:
There are a couple of bugs registered for NAT64. See the following: NAT64 https://bugs.freebsd.org/bugzilla/show_bug.cgi?id=255928 (NAT64 issue on 13.0)
The lab examples for all IPv6 / IPv4 translations will use two new virtual machines:
dnshost - this virtual machine runs a configured copy of BIND 9. Some experience with DNS setup with BIND 9 is helpful but not required.
v6only - this virtual machine only uses IPv6. It is not configured for IPv4 addressing at all.
Readers should have a basic understanding of IPv6 and its addressing characteristics. These resources may be helpful:
6.1. Stateful NAT64 (NAT64LSN) With DNS64
NAT64, described in RFC 6146 is one of a number of transition mechanisms that companies can take as they introduce IPv6 into their environment, or move wholesale into IPv6 locally. The idea with NAT64 is to provide a mechanism to allow an IPv6-only host to make a connection to a remote IPv4-only host. This includes the ability to do a DNS lookup on the remote host, and through the features of DNS64 (a companion transition service described in RFC 6147), translate a received IPv4 address into a special IPv6 address that provides a way to connect using the Network Address Translation variant called NAT64.
A logical view of NAT64 and DNS64 is shown in the figure below:
The process works like this:
An IPv6 only host wants to access a resource from host external1.example.com which only uses IPv4. A DNS lookup for "external1.example.com" is sent to the locally configured DNS64 server. This lookup is for an "AAAA" record for the external1 VM.
The DNS64 server forwards the request to an authoritative server for "example.com".
The authoritative server returns an IPv4 address back to the DNS64 server.
The DNS64 server converts the IPv4 address into an IPv6 address using the transition service described in RFC 6147.
The IPv6 only host, receives the IPv6 address and sends a connection request (SYN) to its local IPV6 router running NAT64.
The NAT64 router converts the IPv6 packet back to IPv4 and forwards the packet to external1.example.com.
The remaining conversions between the IPv6 VM and external1 VM happen in a similar fashion.
In step 4, the DNS64 server converts the IPv4 address into IPv6 by using the "Well Known Prefix" 64:ff9b::
and encapsulating the IPv6 address into the last 4 octets of the address.
In the figure above, "203.0.113.10" has been converted to "cb00:710a" and added as the last four octets of the new address.
Note that this is one instance of a larger selection of translation algorithms to translate an IPv4 address into an IPv6 address. In our implementation, the DNS64 server and the authoritative server are essentially merged together following the description of "Example of 'an IPv6 Network to the IPv4 Internet' Setup with DNS64 in Stub-Resolver Mode" in Section 7.2 of RFC 6147.
6.1.1. Setting Up for NAT64 / DNS64
To exercise the NAT64 capabilities of ipfw, it is first necessary to restart all lab virtual machines and reconfigure the ipfw lab.
The figure below shows the new configuration needed.
On the FreeBSD host system, the appropriate bridge and tap setup is given by this command:
$ sudo /bin/sh mkbr.sh reset bridge0 tap0 tap1 tap7 bridge1 tap4 tap6 tap8
Start up the required virtual machines with:
$ /bin/sh runvm.sh firewall dnshost external1 v6only
As before, configure all interfaces and ensure connectivity of adjacent interfaces. The firewall VM should be set for both IPv4 forwarding and IPv6 forwarding:
# sysctl net.inet.ip.forwarding=1 # sysctl net.inet6.ip6.forwarding=1
The external1 VM’s default route should point to 203.0.113.1 and the v6only host’s default IPv6 route should point to 2001:db8:12::1 as follows:
On external1:
# route add default 203.0.113.50
On v6only:
# route -6 add default 2001:db8:12::50
RFC 5737 describes the use of the 203.0.113.0/24 network for documentation and example purposes. RFC 3849 describes the use of the 2001:db8::/32 network the same purposes. |
6.1.2. Setting Up the dnshost VM
You will have to set up the dnshost VM to provide DNS64 services. ISC’s bind9 (9.18 and above) provides this capability. Setting up bind9, while not trivial, is not impossible. You will have to install the following packages:
bind9 Use the latest supported version. The server running here is using bind 9.18.1
bind-tools Same note as above. The server here uses bind-tools-9.16.27
These packages will install a modest number of dependencies.
If you have downloaded the practice kit, see the tar file dnshost.tar for the bind9 configuration files needed. Use the following command to untar the files:
# tar xvzf dhshost_userlocaletc_namedb.tgz -C /usr/local/etc
This will install all the needed DNS files. Otherwise see the zone files in Appendix E, and try to set up DNS. Note that the files include a stub root zone. This provides a locally complete DNS setup.
Restart the named service with:
# service named restart
There should not be any errors, but if there are, track down and fix.
Test the dnshost configuration with these commands. The first lookup returns the A resource record with an IPv4 address. The second lookup returns the AAAA resource record with the DNS64 configured "Well-known Prefix" 64:ff9b that is used in this section.
root@dnshost:~ # dig @localhost external1.example.com ; <<>> DiG 9.16.27 <<>> @localhost external1.example.com ; (2 servers found) ;; global options: +cmd ;; Got answer: ;; ->>HEADER<<- opcode: QUERY, status: NOERROR, id: 61764 ;; flags: qr aa rd; QUERY: 1, ANSWER: 1, AUTHORITY: 0, ADDITIONAL: 1 ;; WARNING: recursion requested but not available ;; OPT PSEUDOSECTION: ; EDNS: version: 0, flags:; udp: 1232 ; COOKIE: 96f573e1a62ce4380100000062884cacf9b86d6e9f54b542 (good) ;; QUESTION SECTION: ;external1.example.com. IN A ;; ANSWER SECTION: external1.example.com. 3600 IN A 203.0.113.10 ;; Query time: 58 msec ;; SERVER: ::1#53(::1) ;; WHEN: Fri May 20 22:21:32 EDT 2022 ;; MSG SIZE rcvd: 94 # # root@dnshost:~ # dig @localhost external1.example.com aaaa ; <<>> DiG 9.16.27 <<>> @localhost external1.example.com aaaa ; (2 servers found) ;; global options: +cmd ;; Got answer: ;; ->>HEADER<<- opcode: QUERY, status: NOERROR, id: 5865 ;; flags: qr aa rd; QUERY: 1, ANSWER: 1, AUTHORITY: 0, ADDITIONAL: 1 ;; WARNING: recursion requested but not available ;; OPT PSEUDOSECTION: ; EDNS: version: 0, flags:; udp: 1232 ; COOKIE: 0ad6c60faab12f7f0100000062884cb0be6e6bd00f7e482f (good) ;; QUESTION SECTION: ;external1.example.com. IN AAAA ;; ANSWER SECTION: external1.example.com. 3600 IN AAAA 64:ff9b::cb00:710a #
The last test is the most important. The dnshost must return the Well-Known Prefix "64:ff9b::" with the corresponding bits for the embedded IPv4 address. If the test does not return this value, reconfigure the DNS service (named.conf and the primary forward zone "example.com") to fix.
Since the v6only machine will only ever request AAAA lookups, the configuration is complete for this section.
We now have to set up the firewall, the IPV6 only host v6only, and the external VM host, external1. Boot all three VMs and test connectivity without any ipfw running on the firewall host.
We now proceed with the installation of NAT64 on the ipfw firewall.
On the firewall VM:
# kldload ipfw The next line loads the NAT64 module. # kldload ipfw_nat64
Configuring NAT64 is similar to configuring NAT. You must create an instance of the NAT64 translator first.
# ipfw nat64lsn foo create prefix4 203.0.112.0/24 allow_private
The use of the "allow_private" keyword is required. The ipfw(8) manual page notes that the NAT64 translator, by default, will not handle addresses whose destination matches those listed in RFC 1918. The addressing scheme in this lab uses special purpose addresses as noted in RFC 6890 which are also considered "private addresses" by the ipfw NAT64 translator.
Note that the prefix4 address pool (203.0.112.0/24 above) should not be manually configured as an alias on any interface. These addresses are used internally by ipfw. The only requirement is that they be reserved from deployment elsewhere in the local network so they do not cause a routing conflict with ipfw. This allows for 254 simultaneous NAT64 addresses. If more are needed due to high volume, add another prefix4, or increase the existing prefix4 address space.
Continue configuring the NAT64 / DNS64 translator:
# ipfw add allow log ipv6-icmp from any to any icmp6types 135,136 # ipfw add nat64lsn foo log ip from 2001:db8:12::/64 to 64:ff9b::/96 in # ipfw add nat64lsn foo log ip from any to 203.0.112.0/24 in # ipfw add allow log ip from any to any
and the direct_output sysctl must be set to 1 (not zero):
# sysctl net.inet.ip.fw.nat64_direct_output=1
You can also set the nat64_debug sysctl and the firewall verbose sysctl:
# sysctl net.inet.ip.fw.nat64_debug=1 # sysctl net.inet.ip.fw.verbose=1
See /var/log/security for output.
With these prerequisites completed the following tests on the v6only VM should be successful:
root@v6only# ping6 -c 3 64:ff9b::203.0.113.10 PING6(56=40+8+8 bytes) 2001:db8:12::30 --> 64:ff9b::cb00:710a 16 bytes from 64:ff9b::cb00:710a, icmp_seq=0 hlim=63 time=8.401 ms 16 bytes from 64:ff9b::cb00:710a, icmp_seq=1 hlim=63 time=3.429 ms 16 bytes from 64:ff9b::cb00:710a, icmp_seq=2 hlim=63 time=3.398 ms --- 64:ff9b::203.0.113.10 ping6 statistics --- 3 packets transmitted, 3 packets received, 0.0% packet loss round-trip min/avg/max/std-dev = 3.398/5.076/8.401/2.351 ms
and using lynx to grab the external1.example.com home page should be successful:
[root@v6only ~]# lynx external1.example.com
6.1.3. Setting Up for Stateless NAT64 - NAT64STL
The previous ipfw_nat64 examples used "stateful" address translation. ipfw is also capable of performing "stateless" address translation.
Stateless translation is appropriate when a NAT64 translator is used in front of IPv4-only servers to allow them to be reached by remote IPv6-only clients. Stateful translation is suitable for deployment at the client side or at the service provider, allowing IPv6-only client hosts to reach remote IPv4-only nodes.
Stateless configuration of NAT64 is possible with the same architecture as the previous stateful example. Configuration details however, are different. In the stateless case, ipfw uses two tables for translating addresses in either direction: IPv4 → IPv6 and IPv6 → IPv4. A typical configuration is shown below.
Start fresh # kldunload ipfw_nat64 # kldunload ipfw # kldload ipfw # kldload ipfw_nat64 Create the tables used for ipfw_nat64stl # ipfw table T4to6 create type addr valtype ipv6 # ipfw table T6to4 create type addr valtype ipv4 # ipfw table T4to6 add 203.0.112.1 2001:db8:12::6 # ipfw table T6to4 add 2001:db8:12::6 203.0.112.1 # ipfw nat64stl NAT64 create table4 T4to6 table6 T6to4 allow_private Add rules for ipfw_nat64stl # ipfw add allow log icmp6 from any to any icmp6types 135, 136 # ipfw add nat64stl NAT64 log ip from any to 'table(T4to6)' # ipfw add nat64stl NAT64 log ip6 from 'table(T6to4)' to 64:ff9b::/96 # ipfw add allow log ip from any to any Adjust sysctls # sysctl net.inet.ip.fw.verbose=1 # sysctl net.inet.ip.fw.nat64_debug=1 # sysctl net.inet.ip.fw.nat64_direct_output=1
Only the net.inet.ip.fw.nat64_direct_output sysctl is required.
Use the same tests as in the stateful NAT64 example:
[root@v6only ~]# ping6 -c 3 64:ff9b::203.0.113.10 and [root@v6only ~]# lynx external1.example.com
Both tests should be successful.
It may seem limiting to have to use tables to effect communication for stateless NAT64. However, if you look at the architecture involved, the above statements about stateless translation being appropriate when a NAT64 translator is used in front of IPv4-only servers to allow them to be reached by remote IPv6-only clients makes sense. The entire IPv6 cloud can reach a specified server.
This can be accomplished by, for example, changing the T4to6 and T6to4 tables to read:
# ipfw table T4to6 add 203.0.112.0/31 2001:db8:12::30 # ipfw table T6to4 add 2000:0000:0000::/8 203.0.112.0 # ipfw table T6to4 add 2100:0000:0000::/8 203.0.112.1
The T4to6 table allocates two addresses in the address pool: 203.0.112.0 and 203.0.112.1. These are used separately in the T6to4 table to cover vast ranges of IPv6 address space.
Certainly using just one IPv4 pool address is not going to be sufficient to translate such a large range of IPv6 addresses. The point here is that by carefully constructing the translation pool addresses and the T4to6 and T6to4 address tables, ipfw can manage translation to as many IPv6 addresses as needed.
Note that stateless NAT64 shares the same limitations of stateful NAT64.
We move on to the next, and most important IPv6 / IPv4 translation mechanism NAT64 CLAT.
6.2. XLAT464
ipfw supports 464XLAT (RFC 6877) calling it "XLAT464 CLAT". This transition mechanism provides connectivity for IPv4 edge devices across an IPv6 only network. It does this by combining stateful translation in the core and stateless translation at the edge. 464XLAT only supports IPv4 in the client-server model, so it does not support IPv4 peer-to-peer communication or inbound IPv4 connections.
See diagrams and explanation here: https://www.juniper.net/documentation/us/en/software/junos/interfaces-adaptive-services/topics/topic-map/ipv4-connect-ipv6-464xlat.html#id-464xlat-overview The RFC for this mechanism is more enlightening. See RFC 6877 |
The discussion on Wikipedia1 is somewhat sparse:
464XLAT 464XLAT (RFC 6877) allows clients on IPv6-only networks to access IPv4-only Internet services, such as Skype.[13][14] The client uses a SIIT translator to convert packets from IPv4 to IPv6. These are then sent to a NAT64 translator which translates them from IPv6 back into IPv4 and on to an IPv4-only server. The client translator may be implemented on the client itself or on an intermediate device and is known as the CLAT (Customer-side transLATor). The NAT64 translator, or PLAT (Provider-side transLATor), must be able to reach both the server and the client (through the CLAT). The use of NAT64 limits connections to a client-server model using UDP, TCP, and ICMP.
The figure below shows a diagram for implementing 464XLAT in our lab.
As earlier, shutdown all virtual machines and for this example we will reset all the bridge and tap devices to the new architecture.
This example will require two firewalls. The firewall VM and firewall2 VM will both be used as shown in the diagram. In this example, the firewall VM is the CLAT translator (stateless translation) and the firewall2 VM is the PLAT translator (stateful translation).
To start, set up the bridge and tap interfaces with this command on the FreeBSD host:
$ sudo /bin/sh mkbr.sh reset bridge0 tap0 tap1 tap11 bridge1 tap4 tap6 tap7 tap9 \ bridge2 tap2 tap8 tap10
Start up the virtual machines with:
$ /bin/sh runvm.sh external1 firewall firewall2 dnshost v6only external2
As earlier, configure all interfaces according to the diagram and ensure connectivity with adjacent interfaces.
This example is more complex than past examples. There are a number of additional configuration steps needed as follows:
external1.example.com
On external1.example.com: route add default 192.168.1.1 echo "nameserver 192.168.1.53" > /etc/resolv.conf echo "nameserver 203.0.113.53" >> /etc/resolv.conf
firewall.example.com
On firewall.example.com: /bin/sh /root/bin/bsdclat464.sh echo "nameserver 2001:db8:12::53" > /etc/resolv.conf echo "nameserver 192.168.1.53" >> /etc/resolv.conf route -6 add 2001:db8:bbbb::/96 2001:db8:12::1 sysctl net.inet.ip.forwarding=1 sysctl net.inet6.ip6.forwarding=1 sysctl net.inet.ip.fw.nat64_direct_output=1
firewall2.example.com
On firewall2.example.com /bin/sh /root/bin/bsdplat464.sh route -6 add 2001:db8:aaaa::/96 2001:db8:12::2 echo "nameserver 2001:db8:12::53" > /etc/resolv.conf sysctl net.inet.ip.forwarding=1 sysctl net.inet6.ip6.forwarding=1 sysctl net.inet.ip.fw.nat64_direct_output=1
external2.example.com
On external2.example.com route add default 203.0.113.1 echo "nameserver 203.0.113.53" > /etc/resolv.conf service nginx onestart
dnshost.example.com
On dnshost.example.com: echo "nameserver 127.0.0.1" > /etc/resolv.conf echo "nameserver 2001:db8:12::53" >> /etc/resolv.conf service named onestart route add default 203.0.113.1 route add -net 192.168.1.0/24 192.168.1.1 sysctl net.inet.ip.forwarding=0 sysctl net.inet6.ip6.forwarding=0
v6only.example.com
On v6only.example.com: echo "nameserver 2001:db8:12::53" > /etc/resolv.conf route -6 add 2001:db8:bbbb::/96 2001:db8:12::1 route -6 add 2001:db8:aaaa::/96 2001:db8:12::2
Due to the complex nature of these firewall configurations, a script is available for copy and paste. However, do try to understand the configration details.
Once all the above commands are entered on their respective VMs, test the configuration with the following command:
# ping -c 2 external2.example.com
The snippets below show at each step, how the request was transformed.
The examples below are taken from multiple different invocations of the ping command. However, the data transformations are correct. |
On interface em0 on the firewall VM: root@firewall:~/bin # tcpdump -n -i em0 tcpdump: verbose output suppressed, use -v[v]... for full protocol decode listening on em0, link-type EN10MB (Ethernet), snapshot length 262144 bytes 14:38:07.254114 IP 192.168.1.2 > 203.0.113.20: ICMP echo request, id 46395, seq 0, length 64 14:38:07.256893 IP 203.0.113.20 > 192.168.1.2: ICMP echo reply, id 46395, seq 0, length 64 14:38:08.322597 IP 192.168.1.2 > 203.0.113.20: ICMP echo request, id 46395, seq 1, length 64 14:38:08.326715 IP 203.0.113.20 > 192.168.1.2: ICMP echo reply, id 46395, seq 1, length 64 On interface em1 on the firewall VM: root@firewall:~/bin # tcpdump -n -i em1 tcpdump: verbose output suppressed, use -v[v]... for full protocol decode listening on em1, link-type EN10MB (Ethernet), snapshot length 262144 bytes 14:54:45.140746 IP6 2001:db8:aaaa::c0a8:102 > 2001:db8:bbbb::cb00:7114: ICMP6, echo request, id 38233, seq 0, length 64 14:54:45.142995 IP6 2001:db8:bbbb::cb00:7114 > 2001:db8:aaaa::c0a8:102: ICMP6, echo reply, id 38233, seq 0, length 64 14:54:46.171754 IP6 2001:db8:aaaa::c0a8:102 > 2001:db8:bbbb::cb00:7114: ICMP6, echo request, id 38233, seq 1, length 64 14:54:46.173925 IP6 2001:db8:bbbb::cb00:7114 > 2001:db8:aaaa::c0a8:102: ICMP6, echo reply, id 38233, seq 1, length 64 On interface em0 on the firewall2 VM: root@firewall2:~/bin # tcpdump -n -i em0 tcpdump: verbose output suppressed, use -v[v]... for full protocol decode listening on em0, link-type EN10MB (Ethernet), snapshot length 262144 bytes 14:57:32.519334 IP6 2001:db8:aaaa::c0a8:102 > 2001:db8:bbbb::cb00:7114: ICMP6, echo request, id 17270, seq 0, length 64 14:57:32.529066 IP6 2001:db8:bbbb::cb00:7114 > 2001:db8:aaaa::c0a8:102: ICMP6, echo reply, id 17270, seq 0, length 64 14:57:33.560392 IP6 2001:db8:aaaa::c0a8:102 > 2001:db8:bbbb::cb00:7114: ICMP6, echo request, id 17270, seq 1, length 64 14:57:33.561596 IP6 2001:db8:bbbb::cb00:7114 > 2001:db8:aaaa::c0a8:102: ICMP6, echo reply, id 17270, seq 1, length 64 On interface em1 on the firewall2 VM: root@firewall2:~/bin # tcpdump -n -i em1 tcpdump: verbose output suppressed, use -v[v]... for full protocol decode listening on em1, link-type EN10MB (Ethernet), snapshot length 262144 bytes 14:58:37.139612 IP 203.0.112.22 > 203.0.113.20: ICMP echo request, id 1025, seq 0, length 64 14:58:37.141043 IP 203.0.113.20 > 203.0.112.22: ICMP echo reply, id 1025, seq 0, length 64 14:58:38.187477 IP 203.0.112.22 > 203.0.113.20: ICMP echo request, id 1025, seq 1, length 64 14:58:38.188308 IP 203.0.113.20 > 203.0.112.22: ICMP echo reply, id 1025, seq 1, length 64 On interface em0 on the external2 VM: root@external2:~ # tcpdump -n -i em0 tcpdump: verbose output suppressed, use -v[v]... for full protocol decode listening on em0, link-type EN10MB (Ethernet), snapshot length 262144 bytes 15:00:44.171439 IP 203.0.112.22 > 203.0.113.20: ICMP echo request, id 1024, seq 0, length 64 15:00:44.172313 IP 203.0.113.20 > 203.0.112.22: ICMP echo reply, id 1024, seq 0, length 64 15:00:45.200883 IP 203.0.112.22 > 203.0.113.20: ICMP echo request, id 1024, seq 1, length 64 15:00:45.201035 IP 203.0.113.20 > 203.0.112.22: ICMP echo reply, id 1024, seq 1, length 64
The firewall log sysctl was reset to log to syslogd(8) and captured these logs:
Logs on the firewall VM: root@firewall:~/bin # cat /var/log/security Dec 2 15:14:04 firewall kernel: ipfw: 150 Eaction nat64clat ICMP:8.0 192.168.1.2 203.0.113.20 in via em0 Dec 2 15:14:04 firewall kernel: ipfw: 150 Eaction nat64clat ICMPv6:129.0 [2001:db8:bbbb::cb00:7114] [2001:db8:aaaa::c0a8:102] in via em1 Dec 2 15:14:04 firewall kernel: ipfw: 150 Eaction nat64clat ICMP:8.0 192.168.1.2 203.0.113.20 in via em0 Dec 2 15:14:05 firewall kernel: ipfw: 150 Eaction nat64clat ICMPv6:129.0 [2001:db8:bbbb::cb00:7114] [2001:db8:aaaa::c0a8:102] in via em1 Dec 2 15:14:08 firewall kernel: ipfw: 100 Accept ICMPv6:135.0 [2001:db8:12::1] [2001:db8:12::2] in via em1 Dec 2 15:14:08 firewall kernel: ipfw: 100 Accept ICMPv6:136.0 [2001:db8:12::2] [2001:db8:12::1] out via em1 Dec 2 15:14:09 firewall kernel: ipfw: 100 Accept ICMPv6:135.0 [2001:db8:12::2] [2001:db8:12::1] out via em1 Dec 2 15:14:09 firewall kernel: ipfw: 100 Accept ICMPv6:136.0 [2001:db8:12::1] [2001:db8:12::2] in via em1 Logs on the firewall2 VM: root@firewall2:~/bin # cat /var/log/security Dec 2 15:10:25 firewall2 kernel: ipfw: 300 Eaction nat64lsn ICMPv6:128.0 [2001:db8:aaaa::c0a8:102] [2001:db8:bbbb::cb00:7114] in via em0 Dec 2 15:10:25 firewall2 kernel: ipfw: 400 Eaction nat64lsn ICMP:0.0 203.0.113.20 203.0.112.22 in via em1 Dec 2 15:10:26 firewall2 kernel: ipfw: 300 Eaction nat64lsn ICMPv6:128.0 [2001:db8:aaaa::c0a8:102] [2001:db8:bbbb::cb00:7114] in via em0 Dec 2 15:10:26 firewall2 kernel: ipfw: 400 Eaction nat64lsn ICMP:0.0 203.0.113.20 203.0.112.22 in via em1 Dec 2 15:10:29 firewall2 kernel: ipfw: 100 Accept ICMPv6:135.0 [2001:db8:12::1] [2001:db8:12::2] out via em0 Dec 2 15:10:29 firewall2 kernel: ipfw: 100 Accept ICMPv6:136.0 [2001:db8:12::2] [2001:db8:12::1] in via em0 Dec 2 15:10:30 firewall2 kernel: ipfw: 100 Accept ICMPv6:135.0 [2001:db8:12::2] [2001:db8:12::1] in via em0 Dec 2 15:10:30 firewall2 kernel: ipfw: 100 Accept ICMPv6:136.0 [2001:db8:12::1] [2001:db8:12::2] out via em0
Finally, a webpage request was made with:
# lynx external2.example.com
as shown below:
Chapter 7. Other Keywords
This section covers some other lesser used keywords.
7.1. abort / abort6
The abort and abort6 keywords interrupt the data stream between two endpoints. The effect of this keyword, is similar to the reset keyword, but there are important differences.
The above figure shows the effect of inserting the firewall rule:
# ipfw add 50 abort tcp from 203.0.113.30 to me
Unlike the reset keyword, there is no packet sent from the firewall to the source. What happens is that ipfw just starts dropping packets that match the rule. Since there are no more replies coming from the destination (here the firewall itself), the source endpoint issues retransmissions over and over. Eventually the source concludes that the connection is irrevocably broken and it closes the connection.
In the rule above, all TCP connections will be interrupted between the two systems. |
In a TCP connection, ipfw will use dynamic rules if a check-state rule is already in place. If this is the case, issue the abort rule at a rule number before the check-state rule. Otherwise, it will have no effect. |
7.2. mark / setmark
The setmark keyword functions similar to the tag keyword. If the packet matches the rule, ipfw applies a 32-bin identifier to the packet. This identifier (the "mark") is held with the packet internally inside ipfw. It is not sent with the packet on the wire and is not visible to any network monitoring from tools like tcpdump(1) or man:[wireshark].
Like tags, a mark can be used as another filtering device with other ipfw rules to do policy base routing or filtering. Note that only one mark can be applied at a time.
A big advantage of marks over tags are their ability to be matched as a lookup key in a table. Also, a mark can have a bitmask applied to it.
To explore mark and setmark we will use the architecture of Simple NAT shown in Simple NAT. Begin by creating the network with the mkbr.sh script and starting the VMs with the runvm.sh script shown in Simple NAT.
Assign the IP addresses as shown, and ensure all VMs have connectivity with adjacent systems.
On the internal VM, start up the userv.sh script with port number 5656. Then, on the external1 VM, start up the ucont.sh server with the same port and a time value of 1 second.
Before we place a setmark value on a packet, start the data communications scripts and examine the output of the ipfw log by setting the sysctl to log to syslog:
# sysctl net.inet.ip.fw.verbose=1
Now insert the following firewall rules and examine the log file /var/log/security:
# ipfw add 1000 allow log udp from any to 10.10.10.20 dst-port 5656 01000 allow log udp from any to 10.10.10.20 5656 # # tail -f /var/log/security Dec 29 22:32:33 firewall kernel: ipfw: 1000 Accept UDP 203.0.113.10:30463 10.10.10.20:5656 in via em1 Dec 29 22:32:33 firewall kernel: ipfw: 1000 Accept UDP 203.0.113.10:30463 10.10.10.20:5656 out via em0 Dec 29 22:32:36 firewall kernel: ipfw: 1000 Accept UDP 203.0.113.10:24588 10.10.10.20:5656 in via em1 Dec 29 22:32:36 firewall kernel: ipfw: 1000 Accept UDP 203.0.113.10:24588 10.10.10.20:5656 out via em0
Now add the following rule to apply the mark value of 20 (decimal) and observe the change in the logs:
# ipfw add 500 setmark 20 log udp from any to 10.10.10.20 dst-port 5656 00500 setmark 0x14 log udp from any to 10.10.10.20 5656 # # # tail -f /var/log/security Dec 29 22:41:20 firewall kernel: ipfw: 1000 Accept UDP 203.0.113.10:27955 10.10.10.20:5656 in via em1 Dec 29 22:41:20 firewall kernel: ipfw: 1000 Accept UDP 203.0.113.10:27955 10.10.10.20:5656 out via em0 Dec 29 22:41:23 firewall kernel: ipfw: 1000 Accept UDP 203.0.113.10:37423 10.10.10.20:5656 in via em1 Dec 29 22:41:23 firewall kernel: ipfw: 1000 Accept UDP 203.0.113.10:37423 10.10.10.20:5656 out via em0 Dec 29 22:41:25 firewall kernel: ipfw: 500 SetMark 0x14 UDP 203.0.113.10:45176 10.10.10.20:5656 in via em1 Dec 29 22:41:25 firewall kernel: ipfw: 1000 Accept UDP 203.0.113.10:45176 10.10.10.20:5656 mark:0x14 in via em1 Dec 29 22:41:25 firewall kernel: ipfw: 500 SetMark 0x14 UDP 203.0.113.10:45176 10.10.10.20:5656 out via em0 Dec 29 22:41:25 firewall kernel: ipfw: 1000 Accept UDP 203.0.113.10:45176 10.10.10.20:5656 mark:0x14 out via em0 Dec 29 22:41:27 firewall kernel: ipfw: 500 SetMark 0x14 UDP 203.0.113.10:21444 10.10.10.20:5656 in via em1 Dec 29 22:41:27 firewall kernel: ipfw: 1000 Accept UDP 203.0.113.10:21444 10.10.10.20:5656 mark:0x14 in via em1
7.3. NPTv6
IPv6-to-IPv6 Network Prefix Translation (NPTv6) is the process of translating IPv6 header source and destination addresses. Functionally, it is similar to the more well understood Network Address Translation, but without the need to maintain state. It is only the IPv6 source and destination addresses that are translated. The idea here is to allow an edge network to have its own independent addressing scheme while being able to exchange IPv6 traffic with external IPv6 networks through the use of an NPTv6 Translator
RFC 6296 is the definitive document on NPTv6. The example in this section is taken from Sections 2.1 of that document.
The architecture for these examples is based on Simple NAT as in the previous section.
7.3.1. NPTv6 Setup
Use the setup instructions shown in Simple NAT but use the IPv6 addressing as shown below:
At first glance, this appears to be a simple IPv6 forwarding example. As we will see, NPTv6 changes the actual packet source and destination addresses, so no forwarding is needed.
ipfw(8) explains the syntax of the NPTv6 command and options, but there are a number of details that need to be set up correctly. Use the following as a guide:
On the FreeBSD host: $ sudo /bin/sh mkbr.sh reset bridge0 tap1 tap4 bridge1 tap0 tap5 $ /bin/sh runvm.sh external1 firewall internal Ensure all IPv6 addresses on all VMs are set up correctly. On the firewall VM: # kldunload ipfw_nptv6 # kldunload ipfw # kldload ipfw # kldload ipfw_nptv6 # sysctl net.inet.ip.fw.one_pass=0 # sysctl net.inet.ip.fw.verbose=1 # ipfw -q flush # Set up the NPTv6 instance. # ipfw nptv6 foo create int_prefix fd01:0203:0405:: ext_prefix 2001:0db8:0001:: prefixlen 48 # Rule for outbound # ipfw add 2000 nptv6 foo log ip6 from fd01:0203:0405::/48 to any # ipfw add 3000 allow ip6 from any to any
As noted in ipfw(8), the sysctl net.inet6.ip6.forwarding=1
must be applied or NPTv6 will silently stop working.
7.3.2. NPTv6 Testing
Set up a UDP listener on the external1 VM. We could use the userv.sh (and its ucon.sh partner), but that would require editing the scripts to set up an IPv6 address. Try this method instead:
On the external1 VM: # Listen for a UDP packet $ ncat -l -k -u -6 2001:0db8:0001::10 5656 On the internal VM: # Send the desired UDP packet. $ echo "testing123" | ncat -6 -u 2001:0db8:0001::10 5656
In the setup section above, we arranged for logging to syslogd
, so the results can be seen by examining the tail end of /var/log/security:
Dec 31 19:51:44 firewall kernel: ipfw: 2000 Eaction nptv6 UDP [fd01:203:405::20]:52451 [2001:db8:1::10]:5656 in via em0
The output of a tcpdump
on external1 shows:
root@external1:~ # tcpdump -n -i em0 -X "udp" tcpdump: verbose output suppressed, use -v[v]... for full protocol decode listening on em0, link-type EN10MB (Ethernet), snapshot length 262144 bytes 19:51:43.827543 IP6 2001:db8:1:d54f::20.52451 > 2001:db8:1::10.5656: UDP, length 11 0x0000: 600a 145a 0013 113f 2001 0db8 0001 d54f `..Z...?.......O 0x0010: 0000 0000 0000 0020 2001 0db8 0001 0000 ................ 0x0020: 0000 0000 0000 0010 cce3 1618 0013 f72b ...............+ 0x0030: 7465 7374 696e 6731 3233 0a testing123.
The highlighted section shows the effect of the NPTv6 translation. (See RFC 6296, Section 3, for details.)
7.4. ipttl
The ipttl (Time to Live or TTL) keyword identifies packets that have specific TTL characteristics. ipfw(8) notes that the ipttl keyword will accept a single value, a list of values, or a range of values, in the same syntax as that used for the ports keyword. (Recall the discussion of lists and ranges in the Notes on Rule Numbering.)
ipttl is one of a number of ipfw keywords that work on individual fields of packets flowing through the firewall. Similar keywords include ipid, iplen, ipprecedence, etc.
7.4.1. ipttl Setup
Use the setup instructions shown in Simple NAT with IPv4 addressing, not IPv6.
Also, this example will use the hping3 command. (If you have not downloaded the hping3 package, reset the internal VM for access to the Internet, and download the package with pkg install hping3. Remember to reset for Simple NAT for this example.)
Refer to hping3(8) for details.
On the external1 VM: # Listen for a UDP packet $ ncat -l -k -u -6 203.0.113.10 5656 On the internal VM: # Send the desired UDP packet. We deliberate set the initial TTL to 13. $ hping3 --sign "test for ttl 13" --count 1 --udp --ttl 13 --destport 5656 203.00.113.10
Without ipfw in place, you should see something similar to:
01:50:00.765310 IP (tos 0x0, ttl 12, id 27704, offset 0, flags [none], proto UDP (17), length 43) 10.10.10.20.2600 > 203.0.113.10.5656: [udp sum ok] UDP, length 15 0x0000: 4500 002b 6c38 0000 0c11 f261 0a0a 0a14 E..+l8.....a.... 0x0010: cb00 710a 0a28 1618 0017 2f93 7465 7374 ..q..(..../.test 0x0020: 2066 6f72 2074 746c 2031 3300 0000 .for.ttl.13...
The IP Time to Live option was set up to prevent IP packets from bouncing around the Internet forever. RFC 791 initially intended that the value would be considered an actual time value (number of seconds) and that each module processing the packet would subtract processing time from the initial value. This was later changed to an integer identifying a "hop count" where the initial value (now 64) is decremented by every router or gateway or forwarding device, such as a firewall.
In this case the firewall VM, even though it is not running firewall software, is still a 'forwarding device' and decrements the count as it forwards the packet.
7.4.2. ipttl Testing
To examine the ipttl keyword follow this example:
# kldload ipfw # sysctl net.inet.ip.fw.verbose=1 # Count all packets as the flow through # ipfw add 800 count ip from any to any # Count all packets with TTL of exactly 13 as the flow through # ipfw add 900 count ip from any to any ipttl 13 # Allow and log packets with TTL of exactly 13. # ipfw add 1000 allow log udp from any to any ipttl 13 # Count any other ip packets after the ipttl rule # ipfw add 1100 count ip from any to any
Below is a sample run of ncat and hping3 commands to test the above rules:
# echo "UDP with default TTL" | ncat -u 203.0.113.10 5656 # echo "UDP with default TTL" | ncat -u 203.0.113.10 5656 # hping3 --sign "UDP with TTL=13" --count 1 --udp --ttl 13 --destport 5656 203.0.113.10 # hping3 --sign "UDP with TTL=13" --count 1 --udp --ttl 13 --destport 5656 203.0.113.10 The results show the first two packets with default TTL values (64) were not passed by the firewall. The third and fourth packets were passed. # *tcpdump -n -i em0 -X -vvv "udp" tcpdump: listening on em0, link-type EN10MB (Ethernet), snapshot length 262144 bytes 03:26:42.875634 IP (tos 0x0, ttl 12, id 17385, offset 0, flags [none], proto UDP (17), length 43) 10.10.10.20.1648 > 203.0.113.10.5656: [udp sum ok] UDP, length 15 0x0000: 4500 002b 43e9 0000 0c11 1ab1 0a0a 0a14 E..+C........... 0x0010: cb00 710a 0670 1618 0017 1d07 5544 5020 ..q..p......UDP. 0x0020: 7769 7468 2054 544c 3d31 3300 0000 with.TTL=13... 03:26:47.903863 IP (tos 0x0, ttl 12, id 33936, offset 0, flags [none], proto UDP (17), length 43) 10.10.10.20.2539 > 203.0.113.10.5656: [udp sum ok] UDP, length 15 0x0000: 4500 002b 8490 0000 0c11 da09 0a0a 0a14 E..+............ 0x0010: cb00 710a 09eb 1618 0017 198c 5544 5020 ..q.........UDP. 0x0020: 7769 7468 2054 544c 3d31 3300 0000 with.TTL=13...
But it is the ipfw show results that reveal how things really worked:
# ipfw show 00800 6 270 count ip from any to any 00900 2 86 count ip from any to any ipttl 13 01000 2 86 allow log udp from any to any ipttl 13 01100 4 184 count ip from any to any 01200 2 86 allow log udp from any to any ipttl 12 65535 2 98 deny ip from any to any
The count of 6 packets on rule 800 above accounts for the inbound and outbound packets for those that matched later rules. |
7.5. tcpdatalen
The tcpdatalen keyword is one of several related keywords:
tcpack, tcpdatalen, tcpflags, tcpmss, tcpseq, tcpwin, tcpoptions
These keywords are not often used.
However, there is one very important use case. From time to time, an Internet worm - a malicious packet that gets resent to all local and remote hosts matching some criteria - makes its way onto the Internet. Quick thinking network security administrators can sometimes identify a unique characteristic of these malicious packets such as all packets having the same length - akin to tcpdatalen, or a certain set of tcpoptions.
In this example, the firewall VM is running the tserv.sh 5656 script.
The example below configures ipfw to deny all packets having a TCP data length of a certain value range. One of these ranges will cause the malicious packet to be denied. Keep in mind, this is the length of the TCP data payload, not the overall length of the packet.
# ipfw add 10 deny tcp from any to me tcpdatalen 10-19 # ipfw add 20 deny tcp from any to me tcpdatalen 20-29 . . . root@firewall:~/bin # ipfw show 00010 0 0 deny tcp from any to me tcpdatalen 10-19 00020 0 0 deny tcp from any to me tcpdatalen 20-29 00030 0 0 deny tcp from any to me tcpdatalen 30-39 00040 0 0 deny tcp from any to me tcpdatalen 40-49 00050 0 0 deny tcp from any to me tcpdatalen 50-59 00060 0 0 deny tcp from any to me tcpdatalen 60-69 00070 0 0 deny tcp from any to me tcpdatalen 70-79 00080 0 0 deny tcp from any to me tcpdatalen 80-89 00090 0 0 deny tcp from any to me tcpdatalen 90-99 08000 0 0 check-state :default 09000 0 0 allow tcp from any to me setup keep-state :default 65535 0 0 deny ip from any to any
And a test using ncat directly from external3:
# echo "123456789012345678901234567890" | ncat 203.0.113.50 5656
The TCP 3-way handshake completes, but the packet containing the data payload is stopped by rule 30 as shown below:
root@firewall:~/bin # ipfw show 00010 0 0 deny tcp from any to me tcpdatalen 10-19 00020 0 0 deny tcp from any to me tcpdatalen 20-29 00030 13 1079 deny tcp from any to me tcpdatalen 30-39 00040 0 0 deny tcp from any to me tcpdatalen 40-49 00050 0 0 deny tcp from any to me tcpdatalen 50-59 00060 0 0 deny tcp from any to me tcpdatalen 60-69 00070 0 0 deny tcp from any to me tcpdatalen 70-79 00080 0 0 deny tcp from any to me tcpdatalen 80-89 00090 0 0 deny tcp from any to me tcpdatalen 90-99 08000 0 0 check-state :default 09000 8 420 allow tcp from any to me setup keep-state :default 65535 0 0 deny ip from any to any
The reason for the excessive number of packets denied is TCP retransmission trying to account for the dropped packet as shown in the figure below.
Eventually TCP gives up and shuts down the connection.
7.6. verrevpath / versrcreach / antispoof
These keywords all work to determine if an incoming packet is legitimate.
As noted in ipfw(8), verrevpath ("verify reverse path") looks up the incoming packet’s source address in the routing table.
Quoting: "If the interface on which the packet entered the system matches the outgoing interface for the route, the packet matches. If the interfaces do not match up, the packet does not match. All outgoing packets or packets with no incoming interface match."
Consider the figure below:
In this figure, the firewall has interface em0 directly connected to the 10.10.10.0/24
network and the em1 interface directly connected to the 203.0.113.0/24
network.
The firewall VM interfaces and routing table are shown in the text below:
root@firewall:~ # ifconfig em0 em0: flags=1008843<UP,BROADCAST,RUNNING,SIMPLEX,MULTICAST,LOWER_UP> metric 0 mtu 1500 options=48525bb<RXCSUM,TXCSUM,VLAN_MTU,VLAN_HWTAGGING,JUMBO_MTU,VLAN_HWCSUM,TSO4,LRO,WOL_MAGIC,VLAN_HWFILTER,VLAN_HWTSO,HWSTATS,MEXTPG> ether 02:49:50:46:57:41 inet 10.10.10.50 netmask 0xffffff00 broadcast 10.10.10.255 media: Ethernet autoselect (1000baseT <full-duplex>) status: active nd6 options=29<PERFORMNUD,IFDISABLED,AUTO_LINKLOCAL> root@firewall:~ # root@firewall:~ # ifconfig em1 em1: flags=1008843<UP,BROADCAST,RUNNING,SIMPLEX,MULTICAST,LOWER_UP> metric 0 mtu 1500 options=48525bb<RXCSUM,TXCSUM,VLAN_MTU,VLAN_HWTAGGING,JUMBO_MTU,VLAN_HWCSUM,TSO4,LRO,WOL_MAGIC,VLAN_HWFILTER,VLAN_HWTSO,HWSTATS,MEXTPG> ether 02:49:50:46:57:42 inet 203.0.113.50 netmask 0xffffff00 broadcast 203.0.113.255 media: Ethernet autoselect (1000baseT <full-duplex>) status: active nd6 options=29<PERFORMNUD,IFDISABLED,AUTO_LINKLOCAL> root@firewall:~ # root@firewall:~ # netstat -rn Routing tables Internet: Destination Gateway Flags Netif Expire 10.10.10.0/24 link#1 U em0 10.10.10.50 link#3 UHS lo0 127.0.0.1 link#3 UH lo0 203.0.113.0/24 link#2 U em1 203.0.113.50 link#3 UHS lo0 Internet6: Destination Gateway Flags Netif Expire ::/96 link#3 URS lo0 ::1 link#3 UHS lo0 ::ffff:0.0.0.0/96 link#3 URS lo0 fe80::%lo0/10 link#3 URS lo0 fe80::%lo0/64 link#3 U lo0 fe80::1%lo0 link#3 UHS lo0 ff02::/16 link#3 URS lo0 root@firewall:~ #
If a packet came in on the em0 interface with a source address that was not in the 10.10.10.0/24
network, the above quote says the packet should be dropped.
We can test this with our trusty ncat program which has an option to set the source IP.
First, we set up the firewall VM to allow any UDP packets as shown.
Then, we set up the firewall VM to run sh userv.sh 5656, our service to receive UDP packets on the identified port, and send one packet from the internal VM with echo "hello from internal VM" | ncat -u 10.10.10.50 5656.
root@firewall:~/bin # ipfw add 1000 allow udp from any to me verrevpath 01000 allow udp from any to me verrevpath root@firewall:~/bin # root@firewall:~/bin # ipfw show 01000 0 0 allow udp from any to me verrevpath 65535 0 0 deny ip from any to any root@firewall:~/bin # root@firewall:~/bin # sh userv.sh 5656 PORT1 = [5656] Starting UDP listener on [10.10.10.50],[5656] hello from internal VM ^Croot@firewall:~/bin # root@firewall:~/bin # ipfw show 01000 1 51 allow udp from any to me verrevpath 65535 0 0 deny ip from any to any root@firewall:~/bin #
So far, so good. This is expected behavior.
Now we restart the service on the firewall VM and send a similar message from the internal VM, but this time we spoof the source address. To do this, we must add an alias IP address to the interface on the internal VM:
root@internal:~/bin # ifconfig em0 4.4.4.4/32 alias root@internal:~/bin # root@internal:~/bin # echo "hello 2 from internal VM" | ncat -u -s 4.4.4.4 10.10.10.50 5656 root@internal:~/bin #
Now, rule 1000 prevents the matching of the incoming packet with a spoofed source address and no packet is received by the userv.sh service. Instead, the packet is handled by the default deny rule:
root@firewall:~/bin # sh userv.sh 5656 PORT1 = [5656] Starting UDP listener on [10.10.10.50],[5656] ^Croot@firewall:~/bin # root@firewall:~/bin # root@firewall:~/bin # ipfw show 01000 0 0 allow udp from any to me verrevpath 65535 1 53 deny ip from any to any root@firewall:~/bin #
The other keywords in this section, versrcreach and antispoof operate in a similar manner. Check the man page for the slight differences between them.
7.7. jail
Jails are an important component of FreeBSD and have been a part of the base system since FreeBSD 4. ipfw works in tandem with jails to provide networking security. As discussed in the FreeBSD Handbook Section on Jails and Networking, there are three types of jail networking setups. We will discuss the first two:
Host Networking Setup
Virtual Networking (VNET) Setup
7.7.1. Host-based Jail Networking
In this type of networking setup, the jail shares the host networking stack. The jail has the same IP address and interface as the host.
The typical jail configuration file for this set uses the following network configuration:
jailname { . . . # Network ip4 = inherit; interface = em0; . . . }
Here, it is the host that controls the network stack and all ipfw commands (loading, unloading, adding/deleteing rules, etc.) must be done from the host. The jail root user does not have permission to operate ipfw inside the jail.
All ipfw configuration for the jail must be done on the host. ipfw provides the jail keyword. For TCP communications, this keyword applys primarily to outbound packets from the jail. Inbound packets to the jail, follow the normal host rules.
If the jail runs sh tserv.sh 5656, it opens up a TCP socket listening on port 5656 in the jail. The rule for outside access to this jail relies on the host network, and the jail jailname keyword is not needed.
# ipfw add 100 check-state # ipfw add 1000 allow tcp from any to me dst-port 5656 setup keep-state
This rule will allow a connection from an external host to port 5656 in the jail.
For the most part, the ipfw rules that we have used elsewhere in this book are applicable here with the addition of the jail jailname keyword. For outbound communication, the added rule below, using the jail thinjail keyword succeeds.
# ipfw add 100 check-state # ipfw add 1000 allow tcp from any to me dst-port 5656 setup keep-state # ipfw add 2000 allow tcp from me to any setup keep-state jail thinjail
You should always provide the jail name rather than a numeric ID. If the jail is restarted for any reason it may get a new jail ID number and an existing rule with a jail number will be immediately out of date. The rule will have to be re-entered using the jail name. |
When entering a rule with a jail name, ipfw will lookup the name and reply with the number.
So even when listing or showing the ruleset, ipfw will always show the number not the name.
Use the |
It is a good idea to compartmentalize the rules for each jail in a file with the jail name. That way, if a jail is restarted, the specific file can be rerun to update the ipfw rules on the host. |
7.7.2. Virtual Network (VNET) Jail Networking
A more advanced setup is using the VNET networking capabilities of FreeBSD for the jail. There are many good tutorials on setting up VNET jails. This section is focused on the use of ipfw with the vnet network for the jail.
The architecture for this setup is shown in the figure below:
The architecture shows the FreeBSD host, with two QEMU virtual machines, external1 and jail1.
The two VMs are connected by bridge0 and share the 203.0.113.0/24
network.
jail1 has different characteristics than the standard VMs we have been using. It has 8GB memory, and is running ZFS for its filesystem.
The jail1 VM has set up a FreeBSD thin jail inside the VM following the directions in the FreeBSD handbook on Creating a Thin Jail Using OpenZFS Snapshots.
While there are two |
The jail configuration sets up a vnet jail as follows:
# # vnetjail.conf - handbook/jails - setting up a thin jail under ZFS # vnetjail { # Startup / Logging exec.start = "/bin/sh /etc/rc"; exec.stop = "/bin/sh /etc/rc.shutdown"; exec.consolelog = "/var/log/jail_console_${name}.log"; # Permissions allow.raw_sockets; exec.clean; mount.devfs; devfs_ruleset = 5; # Hostname / Path host.hostname = "${name}"; path = "/usr/local/jails/containers/${name}"; # VNET / VIMAGE vnet; vnet.interface = "${epair}b"; # Network $id = "90"; $ip = "203.0.113.${id}/24"; $gateway = "203.0.113.50"; $bridge = "bridge0"; $epair = "epair${id}"; # ADD TO bridge INTERFACE exec.prestart = "/sbin/ifconfig ${bridge} create up"; exec.prestart += "/sbin/ifconfig ${epair} create up"; exec.prestart += "/sbin/ifconfig ${epair}a up descr jail:${name}"; exec.prestart += "/sbin/ifconfig ${bridge} addm ${epair}a up"; exec.prestart += "/sbin/ifconfig ${bridge} addm em0"; exec.start += "/sbin/ifconfig ${epair}b ${ip} up"; exec.start += "/sbin/route add default ${gateway}"; exec.poststop = "/sbin/ifconfig ${bridge} deletem ${epair}a"; exec.poststop += "/sbin/ifconfig ${epair}a destroy"; }
This time, the network stack is completely separate from the host network stack. However, achieving and managing connectivity happens inside the vnetjail jail.
Testing connectivity with the jail can be accomplished by
Ensuring ipfw is not loaded on jail1,
Entering the vnetjail, and
Starting up a listening service using nc(1):
root@jail1:~ # kldunload ipfw IP firewall unloaded root@jail1:~ # root@jail1:~ # jexec -u root vnetjail root@vnetjail:/ # cd root@vnetjail:~ # root@vnetjail:~ # nc -l -k 5656
Connecting from external1 using nc(1):
root@external1:~ # nc 203.0.113.90 5656 hello from external1 ^C #
With no ipfw firewall in place, the test is successful.
To apply ipfw rules for the vnetjail jail, start ipfw in the jail1 VM.
In VNET jails, ipfw is started from outside the jail, but rules are added from inside the jail. ipfw is also stopped from outside the jail. |
Then, from inside the vnetjail jail, start up a listener using nc(1):
root@vnetjail:~ # nc -l -k 5656 root@vnetjail:~ #
Since the vnetjail jail has a separate IP address and network stack from the jail1 VM, we orient ipfw rules around the vnetjail IP address:
root@jail1:~ # kldload ipfw ipfw2 (+ipv6) initialized, divert loadable, nat loadable, default to deny, logging disabled root@jail1:~ # root@jail1:~ # jexec -u root vnetjail root@vnetjail:/ # cd root@vnetjail:~ # root@vnetjail:~ # ipfw show 65535 0 0 deny ip from any to any root@vnetjail:~ # root@vnetjail:~ # ipfw add 100 check-state 00100 check-state :default root@vnetjail:~ # root@vnetjail:~ # ipfw add 1000 allow tcp from any to me dst-port 5656 setup keep-state 01000 allow tcp from any to me 5656 setup keep-state :default root@vnetjail:~ # root@vnetjail:~ # ipfw show 00100 0 0 check-state :default 01000 0 0 allow tcp from any to me 5656 setup keep-state :default 65535 0 0 deny ip from any to any The single rule above is enough to set up a TCP connection. From external1: root@external1:~ # nc 203.0.113.90 5656 Hello from external1 after ipfw rules have been set up. ^C root@external1:~ # After the above: root@vnetjail:~ # root@vnetjail:~ # ipfw show 00100 0 0 check-state :default 01000 18 1012 allow tcp from any to me 5656 setup keep-state :default 65535 0 0 deny ip from any to any root@vnetjail:~ #
Appendix A: Appendix A: QEMU Setup
This appendix contains helpful information for getting QEMU installed on FreeBSD and instructions for common use.
QEMU can be invoked to display a vt(4) style virtual terminal console using SDL.
Because of this the user environment, either X Window or Wayland, must allow for use of the DISPLAY variable in the local environment.
It can also be invoked in -nographics
mode, resulting is the current screen or window immediately becoming the console for the QEMU instance.
The examples in this book utilize the SDL based vt(4) based console and also include a using a FreeBSD serial console (described Adding a Serial Console).
Additional resources for understanding and using QEMU include:
The QEMU Virtualization chapter in the FreeBSD Handbook
The qemu(1) manual page
The QEMU Home Page
QEMU is available as a package or a port. There are a large number of build options on the port, so in most cases it is best to install the package:
# pkg install qemu sudo
Perform the following steps to get a QEMU virtual machine started. The Quick Start section in the Introduction has a suggested layout.
Designate a destination directory for the virtual machine(s). This procedure is an example that does not use the directory layout in this book. We will call this directory QVM.
Create the image files that will become the QEMU virtual machines. Use qemu-img(1) to create the images. Using the
qcow2
format is recommended as the scripts in this book use that format.# cd QVM # qemu-img create -f qcow2 -o preallocation=full MyQemuVM 4G
Download or copy a FreeBSD installation DVD into the QVM directory and rename it to fbsd.iso. You can also just link that name to an existing DVD.
Set up networking You can use QEMU without any networking, with an internal legacy based SLiRP protocol that uses an internal DHCP server, or you can use the FreeBSD tap(4) and if_bridge(4) interfaces to connect to your local network. In this example, we will install without a network interface, and add it later.
Once the image file is complete, you can enter these commands to start up the virtual machine:
/usr/local/bin/qemu-system-x86_64 -monitor stdio \ -cpu qemu64 \ -vga std \ -m 4096 \ -smp 2 \ -cdrom fbsd.iso \ -boot order=cd,menu=on \ -blockdev driver=file,aio=threads,node-name=fbsdimg,filename=MyQemuVM.qcow2 \ -blockdev driver=qcow2,node-name=drive0,file=fbsdimg \ -device virtio-blk-pci,drive=drive0,bootindex=1 \ -name \"FreeBSD\"
The virtual machine should start in a console window and load the FreeBSD installation DVD. Install the system as usual, but select UFS as the filesystem. If you want to use ZFS, increase the amount of memory (
-m parameter
).To add networking, configure a tap(4) device as
tap0
, and an if_bridge(4) device asbridge0
as shown in the Quick Start section. You can add thetap0
andbridge0
devices to the above configuration with these two lines (add above the-name
line).-netdev tap,id=nd0,ifname=tap0,script=no,downscript=no,br=bridge0 \ -device e1000,netdev=nd0,mac=02:55:33:12:34:56 \
If you want the virtual machine to use DHCP on your local network, add your host interface (em0, bge0, etc.) to the bridge. For example, to add an
em0
host interface use:# ifconfig bridge0 addm em0
If you are having trouble, check the other resources noted above. The given examples should work.
Appendix B: Appendix B: Scripts and Code for QEMU Lab
The listing below shows how the scripts are organized on the GitHub ipfw-primer/SCRIPTS site.
. |-- VM_SCRIPTS | |-- IPFW_root_bin.tgz : common scripts for all VMs (see bin list) | |-- Manifest_IPFW_root_bin.txt : Manifest document for IPFW_root_bin.tgz | |-- Manifest_index.txt : Manifest document for index.html files |-- bin | |-- tcon.sh : TCP connection script | |-- tconr.sh : TCP connect with random port script | |-- tcont.sh : TCP continuous connection script | |-- tserv.sh : TCP server script for one port | |-- tserv3.sh : TCP server script for three ports | |-- ucon.sh : UDP connection script | |-- uconr.sh : UDP connect with random port script | |-- ucont.sh : UDP continuous connection script | |-- userv.sh : UDP server script for one port | |-- userv3.sh : UDP server script for three ports | `-- userv5.sh : UDP server script for five ports | |-- dnshost : | | |-- Manifest_namedb.txt : Manifest for dnshost /usr/local/etc/namedb | | |-- dnshost_usrlocaletc_namedb.tgz : Files for the above | | `-- index.html : Nginx index.html file for dnshost VM | |-- external1 : | | `-- index.html : Nginx index.html file for external1 VM | |-- external2 : | | `-- index.html : Nginx index.html file for external2 VM | |-- external3 : | | `-- index.html : Nginx index.html file for external3 VM | |-- firewall : | | |-- bsdclat464.sh : Script for Section 6.2 XLAT464 CLAT | | `-- index.html : Nginx index.html file for firewall VM | |-- firewall2 : | | |-- bsdplat464.sh : Script for Section 6.2 XLAT464 CLAT | | `-- index.html : Nginx index.html file for firewall2 VM | |-- internal : | | `-- index.html : Nginx index.html file for internal VM | |-- v6only : | | `-- index.html : Nginx index.html file for v6only VM | `-- jail1 : | `-- index.html : Nginx index.html file for jail1 VM |-- _CreateAllVMs.sh : Script to create all VMs used in book |-- dnshost.sh : QEMU startup script for dnshost VM |-- external1.sh : QEMU startup script for external1 VM |-- external2.sh : QEMU startup script for external2 VM |-- external3.sh : QEMU startup script for external3 VM |-- firewall.sh : QEMU startup script for firewall VM |-- firewall2.sh : QEMU startup script for firewall2 VM |-- internal.sh : QEMU startup script for internal VM |-- jail1.sh : QEMU startup script for jail1 VM |-- mkbr.sh : Script to make host bridge and tap devices |-- runvm.sh : XFCE4 script to start VMs |-- swim.sh : Script to manage serial terminals on host |-- scim.sh : Script to manage serial terminals on host |-- v6only.sh : QEMU startup script for v6only VM |-- vm_envs.sh : IPFW lab environment variables `-- CODE `-- divert.c : C code for working with divert keyword
All scripts are shown below in lexicographic order:
SCRIPT: VM_SCRIPTS/firewall/bsdclat464.sh #!/bin/sh # IPFW Primer # License: 3-clause BSD # Author: Jim Brown, jpb@jimby.name # Code: https://github.com/jimmyb-gh/ipfw-primer # # FreeBSD 464XLAT CLAT script for firewall VM. # # bsdclat464.sh: FreeBSD IPFW script for 464XLAT CLAT. See Section 6.2 # Usage: # /bin/sh bsdclat464.sh (run script as root) set -x kldunload ipfw_nat64 kldunload ipfw sleep 1 kldload ipfw kldload ipfw_nat64 # Create the nat64clat instance ipfw nat64clat CLAT create clat_prefix 2001:db8:aaaa::/96 plat_prefix 2001:db8:bbbb::/96 allow_private log # Allow neighbor discovery ipfw add 100 allow log icmp6 from any to any icmp6types 135,136 # pass any ip through the nat64clat instance ipfw add 150 nat64clat CLAT log ip from any to any # pass any ip through the nat64plat instance ipfw add 200 nat64clat CLAT log ip from any to 2001:db8:bbbb::/96 # allow ipv6 from any to any ipfw add 300 allow log ip6 from any to any # allow ipv4 from any to any ipfw add 400 allow log ip from any to any # 0=log with ipfwlog0, 1=log with syslog sysctl net.inet.ip.fw.verbose=0 sysctl net.inet.ip.fw.nat64_debug=1 # direct output: 1 enable, 0 disable (packet goes back into ruleset) sysctl net.inet.ip.fw.nat64_direct_output=1 ========================================================== SCRIPT: VM_SCRIPTS/firewall2/bsdplat464.sh #!/bin/sh # IPFW Primer # License: 3-clause BSD # Author: Jim Brown, jpb@jimby.name # Code: https://github.com/jimmyb-gh/ipfw-primer # # FreeBSD 464XLAT PLAT script for firewall2 VM. # # bsdplat464.sh: FreeBSD IPFW script for 464XLAT CLAT. See Section 6.2 # Usage: # /bin/sh bsdplat464.sh (run script as root) set -x kldunload ipfw_nat64 kldunload ipfw sleep 1 kldload ipfw kldload ipfw_nat64 # create the nat64 stateful instance ipfw nat64lsn NAT64 create log prefix4 203.0.112.0/24 prefix6 2001:db8:bbbb::/96 allow_private # Allow neighbor discovery ipfw add allow log icmp6 from any to any icmp6types 135,136 # Allow the nat64 outbound ipfw add nat64lsn NAT64 log ip from 2001:db8:12::/64 to 2001:db8:bbbb::/96 in ipfw add nat64lsn NAT64 log ip from any to 2001:db8:bbbb::/96 in # Allow the nat64 inbound ipfw add nat64lsn NAT64 log ip from any to 203.0.112.0/24 in # Allow ipv4 from any to any ipfw add allow log ip from any to any # Allow ipv6 from any to any ipfw add allow log ip6 from any to any # Logging: 0 interfaces, 1 syslog sysctl net.inet.ip.fw.verbose=0 # Debug nat64 sysctl net.inet.ip.fw.nat64_debug=1 # Direct output: 1 enable, 0 disable (packet goes back into ruleset) sysctl net.inet.ip.fw.nat64_direct_output=1 ========================================================== SCRIPT: _CreateAllVMs.sh #!/bin/sh # IPFW Primer # License: 3-clause BSD # Author: Jim Brown, jpb@jimby.name # Code: https://github.com/jimmyb-gh/ipfw-primer # # _CreateAllVM.sh : Create VMs for the IPFW Primer lab. # Files are created in ../VM/ # echo "Running _CreateAllVMs.sh" echo echo "This script will create 8 virtual machines in ../VM/" echo read -p "DO YOU REALLY WANT TO CREATE NEW QEMU IMAGES OVERWRITING ANY EXISTING IMAGES? Answer YES to continue. " junk echo [${junk}] if [ "X${junk}" != "XYES" ] then echo "Response was [${junk}]" echo "bailing out..." exit 1 fi echo "Response was [${junk}]" echo "Ok, continuing..." #exit for i in dnshost external1 external2 external3 firewall firewall2 internal v6only do echo "Creating ${i} VM" echo qemu-img create -f qcow2 -o preallocation=full ../VM/${i}.qcow2 4G qemu-img create -f qcow2 -o preallocation=full ../VM/${i}.qcow2 4G done echo echo "Done." ========================================================== SCRIPT: tcon.sh #!/bin/sh # # location: external Vms # # sh tcon.sh PORTNUM - start up 1 connection over TCP # usage() { echo "sh tcon.sh PORTNUM" exit 1 } #echo $# if [ $# -ne 1 ] then usage else export PORT1=$1 fi # echo "PORT1 = [$PORT1]" export CONN="203.0.113.50" export COUNT=1 export MYIP=ifconfig em0 | grep inet | grep -v inet6 | awk '{print $2}'
export MYNAME="external1" echo "TCP connection from [$MYIP],[$PORT1],[$COUNT]" echo "TCP connection from [$MYIP],[$PORT1],[$COUNT]"| ncat $CONN $PORT1 export PREVIOUS_PORT=$PORT1 while : do COUNT=expr $COUNT + 1
read -p "ncat [$COUNT] ready. Enter a valid PORTNUM: " PORT1 if [ "X$PORT1" = "X" ] then PORT1=$PREVIOUS_PORT fi echo "TCP connection from [$MYIP],[$PORT1],[$COUNT]" echo "TCP connection from [$MYIP],[$PORT1],[$COUNT]"| ncat $CONN $PORT1 if [ $? -ne 0 ] then echo "TCP connection [$MYIP],[$PORT1],[$COUNT] FAILED" fi PREVIOUS_PORT=$PORT1 done ========================================================== SCRIPT: tconr.sh #!/bin/sh # # location: external Vms # # sh tconr.sh PORTNUM SLEEPVAL (randomized port numbers) - start up 1 connection over TCP # usage() { echo "sh tconr.sh PORT1NUM SLEEPVAL (randomized port numbers)" exit 1 } # echo $# if [ $# -ne 2 ] then usage fi PORT1=$1 SLEEPVAL=$2 echo "PORT1 = [$PORT1]" echo "SLEEPVAL = [$SLEEPVAL]" export CONN="203.0.113.50" export COUNT=1 export MYIP=ifconfig em0 | grep inet | grep -v inet6 | awk '{print $2}'
export MYNAME="external1" echo "TCP connection from [$MYIP],[$PORT1],[$COUNT]" echo "TCP connection from [$MYIP],[$PORT1],[$COUNT]"| ncat $CONN $PORT1 while : do COUNT=expr $COUNT + 1
# use jot(1) to get a random port between 5656 and 5659. # Connection to 5659 has no listener on firewall and will thus fail. PORT1=jot -r 1 5656 5659 $RANDOM
echo "TCP connection from [$MYIP],[$PORT1],[$COUNT]" echo "TCP connection from [$MYIP],[$PORT1],[$COUNT]"| ncat $CONN $PORT1 if [ $? -ne 0 ] then echo "TCP connection [$MYIP],[$PORT1],[$COUNT] FAILED" fi sleep $SLEEPVAL done ========================================================== SCRIPT: tcont.sh #!/bin/sh # # location: external Vms # # sh tcont.sh PORT1NUM SLEEPVAL - keep hammering same TCP port every SLEEPVAL # usage() { echo "sh tcont.sh PORT1NUM SLEEPVAL" exit 1 } #echo $# if [ $# -ne 2 ] then usage fi export PORT1=$1 export SLEEPVAL=$2 echo "PORT1 = [$PORT1]" echo "SLEEPVAL = [$SLEEPVAL]" export CONN="203.0.113.50" export COUNT=1 export MYIP=ifconfig em0 | grep inet | grep -v inet6 | awk '{print $2}'
echo "TCP connection from [$MYIP],[$PORT1],[$COUNT]" echo "TCP connection from [$MYIP],[$PORT1],[$COUNT]"| ncat $CONN $PORT1 while : do COUNT=expr $COUNT + 1
echo "TCP connection from [$MYIP],[$PORT1],[$COUNT]" echo "TCP connection from [$MYIP],[$PORT1],[$COUNT]"| ncat $CONN $PORT1 if [ $? -ne 0 ] then echo "TCP connection [$MYIP],[$PORT1],[$COUNT] FAILED" fi sleep $SLEEPVAL done ========================================================== SCRIPT: tserv.sh #!/bin/sh # # location: firewall VMs # # tserv.sh - start up 1 listener over TCP zapall() { kill -TERM $PID1 } trap zapall SIGINT export MYIP=ifconfig em0 | grep inet | grep -v inet6 | awk '{print $2}'
export PORT1=5656 echo "Starting TCP listener on [$PORT1]" ncat -l -4 -k $MYIP $PORT1 & PID1=$! wait exit ========================================================== SCRIPT: tserv3.sh #!/bin/sh # # location: firewall VMs # # tserv3.sh - start up 3 listeners over TCP zapall() { kill -TERM $PID1 $PID2 $PID3 } trap zapall SIGINT export MYIP=ifconfig em0 | grep inet | grep -v inet6 | awk '{print $2}'
export PORT1=5656 export PORT2=5657 export PORT3=5658 echo "Starting TCP listeners on [$PORT1],[$PORT2],[$PORT3]" ncat -l -4 -k $MYIP $PORT1 & PID1=$! ncat -l -4 -k $MYIP $PORT2 & PID2=$! ncat -l -4 -k $MYIP $PORT3 & PID3=$! wait exit ========================================================== SCRIPT: ucon.sh #!/bin/sh # # location: external Vms # # sh ucon.sh PORTNUM - start up 1 transfer over UDP # usage() { echo "sh ucon.sh PORTNUM" exit 1 } #echo $# if [ $# -ne 1 ] then usage else export PORT1=$1 fi # echo "PORT1 = [$PORT1]" export CONN="203.0.113.50" # export CONN="10.10.10.50" export COUNT=1 export MYIP=ifconfig em0 | grep inet | grep -v inet6 | awk '{print $2}'
export MYNAME="external1" echo "UDP packet from [$MYIP],[$PORT1],[$COUNT]" echo "UDP packet from [$MYIP],[$PORT1],[$COUNT]"| ncat -u $CONN $PORT1 export PREVIOUS_PORT=$PORT1 while : do COUNT=expr $COUNT + 1
read -p "ncat [$COUNT] ready. Enter a valid PORTNUM: " PORT1 if [ "X$PORT1" = "X" ] then PORT1=$PREVIOUS_PORT fi echo "UDP packet from [$MYIP],[$PORT1],[$COUNT]" echo "UDP packet from [$MYIP],[$PORT1],[$COUNT]"| ncat -u $CONN $PORT1 if [ $? -ne 0 ] then echo "UDP packet [$MYIP],[$PORT1],[$COUNT] FAILED" fi PREVIOUS_PORT=$PORT1 done ========================================================== SCRIPT: uconr.sh #!/bin/sh # # location: external Vms # usage() { echo "sh uconr.sh PORT1NUM SLEEPVAL (randomized port numbers)" exit 1 } # echo $# if [ $# -ne 2 ] then usage fi PORT1=$1 SLEEPVAL=$2 echo "PORT1 = [$PORT1]" echo "SLEEPVAL = [$SLEEPVAL]" export CONN="203.0.113.50" export COUNT=1 export MYIP=ifconfig em0 | grep inet | grep -v inet6 | awk '{print $2}'
export MYNAME="external1" echo "UDP packet from [$MYIP],[$PORT1],[$COUNT]" echo "UDP packet from [$MYIP],[$PORT1],[$COUNT]"| ncat -u $CONN $PORT1 while : do COUNT=expr $COUNT + 1
# use jot(1) to get a random port between 5656 and 5659. # Packet on 5659 has no listener on firewall and will thus fail. PORT1=jot -r 1 5656 5659 $RANDOM
echo "UDP packet from [$MYIP],[$PORT1],[$COUNT]" echo "UDP packet from [$MYIP],[$PORT1],[$COUNT]"| ncat -u $CONN $PORT1 if [ $? -ne 0 ] then echo "UDP packet [$MYIP],[$PORT1],[$COUNT] FAILED" fi sleep $SLEEPVAL done ========================================================== SCRIPT: ucont.sh #!/bin/sh # # location: external Vms # # sh ucont.sh PORT1NUM SLEEPVAL - keep hammering same UDP port every SLEEPVAL # usage() { echo "sh ucont.sh PORT1NUM SLEEPVAL" exit 1 } #echo $# if [ $# -ne 2 ] then usage fi export PORT1=$1 export SLEEPVAL=$2 echo "PORT1 = [$PORT1]" echo "SLEEPVAL = [$SLEEPVAL]" # export CONN="10.10.10.50" export CONN="203.0.113.50" export COUNT=1 export MYIP=ifconfig em0 | grep inet | grep -v inet6 | awk '{print $2}'
echo "UDP packet from [$MYIP],[$PORT1],[$COUNT]" echo "UDP packet from [$MYIP],[$PORT1],[$COUNT]"| ncat -u $CONN $PORT1 while : do COUNT=expr $COUNT + 1
# echo "UDP packet from [$MYIP],[$PORT1],[$COUNT]" # echo "UDP packet from [$MYIP],[$PORT1],[$COUNT]"| ncat -u $CONN $PORT1 echo "UDP packet from [$MYIP],[$PORT1],[$COUNT]" echo "UDP packet from [$MYIP],[$PORT1],[$COUNT]"| ncat -u $CONN $PORT1 if [ $? -ne 0 ] then echo "UDP packet [$MYIP],[$PORT1],[$COUNT] FAILED" fi sleep $SLEEPVAL done ========================================================== SCRIPT: userv.sh #!/bin/sh # # location: firewall VMs # # userv.sh PORTNUM - start up 1 listener over UDP # usage() { echo "sh userv.sh PORTNUM" exit 1 } #echo $# if [ $# -ne 1 ] then usage else PORT1=$1 fi echo "PORT1 = [$PORT1]" zapall() { kill -TERM $PID1 } trap zapall SIGINT export MYIP=ifconfig em0 | grep inet | grep -v inet6 | awk '{print $2}'
echo "Starting UDP listener on [$MYIP],[$PORT1]" # echo nc -l -k -u $MYIP $PORT1 nc -l -k -u $MYIP $PORT1 & PID1=$! wait exit ========================================================== SCRIPT: userv3.sh #!/bin/sh # # location: firewall VMs # # userv3.sh - start up 3 listeners over udp zapall() { kill -TERM $PID1 $PID2 $PID3 } trap zapall SIGINT export MYIP=ifconfig em0 | grep inet | grep -v inet6 | awk '{print $2}'
export PORT1=5656 export PORT2=5657 export PORT3=5658 echo "Starting UDP listeners on [$PORT1],[$PORT2],[$PORT3]" nc -l -k -u $MYIP $PORT1 & PID1=$! nc -l -k -u $MYIP $PORT2 & PID2=$! nc -l -k -u $MYIP $PORT3 & PID3=$! wait exit ========================================================== SCRIPT: userv5.sh #!/bin/sh # # location: firewall VMs # # userv5.sh - start up 5 listeners over udp zapall() { kill -TERM $PID1 $PID2 $PID3 $PID4 $PID5 } trap zapall SIGINT export MYIP=ifconfig em0 | grep inet | grep -v inet6 | awk '{print $2}'
export PORT1=5656 export PORT2=5657 export PORT3=5658 export PORT4=5659 export PORT5=5660 echo "Starting UDP listeners on [$PORT1],[$PORT2],[$PORT3],[$PORT4],[$PORT5]" nc -l -k -u $MYIP $PORT1 & PID1=$! nc -l -k -u $MYIP $PORT2 & PID2=$! nc -l -k -u $MYIP $PORT3 &cd PID3=$! nc -l -k -u $MYIP $PORT4 & PID4=$! nc -l -k -u $MYIP $PORT5 & PID5=$! wait exit ========================================================== SCRIPT: dnshost.sh #!/bin/sh # IPFW Primer # License: 3-clause BSD # Author: Jim Brown, jpb@jimby.name # Code: https://github.com/jimmyb-gh/ipfw-primer # # FreeBSD QEMU VM startup script for dnshost VM. # # dnshost.sh: FreeBSD QEMU VM startup script for dnshost VM. # Usage: sudo /bin/sh dnshost.sh # Note: Set up for serial console. Start another session and telnet to the port shown. # #set -x # pick up environment for this run . ./vm_envs.sh echo [ISO=${_DNSHOST_ISO}] echo [mem=${_DNSHOST_mem}] echo [hdsize=${_DNSHOST_hdsize}] echo [img=${_DNSHOST_img}] echo [mac1=${_DNSHOST_mac1}] echo [mac2=${_DNSHOST_mac2}] echo [name=${_DNSHOST_name}] echo [tap7=${_DNSHOST_tap7}] echo [tap8=${_DNSHOST_tap8}] echo [tap11=${_DNSHOST_tap11}] echo [telnetport=${_DNSHOST_telnetport}] #exit # Note - the dnshost has two interfaces - em0 and em1. # em0 is considered the ipv4 interface and # em1 is considered the ipv6 interface. echo echo "NOTE!!! telnet server running! To start QEMU telnet to localhost $_DNSHOST_telnetport" echo /usr/local/bin/qemu-system-x86_64 -monitor none \ -serial telnet:localhost:${_DNSHOST_telnetport},server=on,wait=off \ -cpu qemu64 \ -vga cirrus \ -m ${_DNSHOST_mem} \ -cdrom ${_DNSHOST_ISO} \ -boot order=cd,menu=on,splash=${_DNS_splash},splash-time=3000 \ -drive if=none,id=drive0,cache=none,aio=threads,format=raw,file=${_DNSHOST_img} \ -device virtio-blk,drive=drive0 \ -netdev tap,id=nd0,ifname=${_DNSHOST_tap7},script=no,downscript=no \ -device e1000,netdev=nd0,mac=${_DNSHOST_mac1} \ -netdev tap,id=nd1,ifname=${_DNSHOST_tap8},script=no,downscript=no \ -device e1000,netdev=nd1,mac=${_DNSHOST_mac2} \ -netdev tap,id=nd2,ifname=${_DNSHOST_tap11},script=no,downscript=no \ -device e1000,netdev=nd2,mac=${_DNSHOST_mac3} \ -name \"${_DNSHOST_name}\" & ========================================================== SCRIPT: external1.sh #!/bin/sh # IPFW Primer # License: 3-clause BSD # Author: Jim Brown, jpb@jimby.name # Code: https://github.com/jimmyb-gh/ipfw-primer # # FreeBSD QEMU VM startup script for external1 VM. # # external1.sh: FreeBSD QEMU VM startup script for external1 VM. # Usage: sudo /bin/sh external1.sh # Note: Set up for serial console. Start another session and telnet to the port shown. # # FreeBSD QEMU VM startup script # # external1.sh # #set -x # pick up environment for this run . ./vm_envs.sh echo [$_EXTERNAL1_ISO] echo [$_EXTERNAL1_mem] echo [$_EXTERNAL1_hdsize] echo [$_EXTERNAL1_img] echo [$_EXTERNAL1_mac] echo [$_EXTERNAL1_name] echo [$_EXTERNAL1_tap1] echo [$_EXTERNAL1_telnetport] # #exit # echo echo "NOTE!!! telnet server running! To start QEMU telnet to localhost $_EXTERNAL1_telnetport" echo /usr/local/bin/qemu-system-x86_64 -monitor none \ -serial telnet:localhost:${_EXTERNAL1_telnetport},server=on,wait=off \ -cpu qemu64 \ -vga cirrus \ -m ${_EXTERNAL1_mem} \ -cdrom ${_EXTERNAL1_ISO} \ -boot order=cd,menu=on,splash=${_EX1_splash},splash-time=3000 \ -drive if=none,id=drive0,cache=none,aio=threads,format=raw,file=${_EXTERNAL1_img} \ -device virtio-blk,drive=drive0 \ -netdev tap,id=nd0,ifname=${_EXTERNAL1_tap1},script=no,downscript=no \ -device e1000,netdev=nd0,mac=${_EXTERNAL1_mac} \ -name \"${_EXTERNAL1_name}\" & ========================================================== SCRIPT: external2.sh #!/bin/sh # IPFW Primer # License: 3-clause BSD # Author: Jim Brown, jpb@jimby.name # Code: https://github.com/jimmyb-gh/ipfw-primer # # FreeBSD QEMU VM startup script for external2 VM. # # external2.sh: FreeBSD QEMU VM startup script for external2 VM. # Usage: sudo /bin/sh external2.sh # Note: Set up for serial console. Start another session and telnet to the port shown. # # FreeBSD qemu vm startup script # # external2.sh # #set -x # pick up environment for this run . ./vm_envs.sh echo [ISO=${_EXTERNAL2_ISO}] echo [mem=${_EXTERNAL2_mem}] echo [hdsize=${_EXTERNAL2_hdsize}] echo [img=${_EXTERNAL2_img}] echo [mac=${_EXTERNAL2_mac}] echo [name=${_EXTERNAL2_name}] echo [tap2=${_EXTERNAL2_tap2}] echo [telnetport=${_EXTERNAL2_telnetport}] # #exit # echo echo "NOTE!!! telnet server running! To start QEMU telnet to localhost $_EXTERNAL2_telnetport" echo /usr/local/bin/qemu-system-x86_64 -monitor none \ -serial telnet:localhost:${_EXTERNAL2_telnetport},server=on,wait=off \ -cpu qemu64 \ -vga cirrus \ -m ${_EXTERNAL2_mem} \ -cdrom ${_EXTERNAL2_ISO} \ -boot order=cd,menu=on,splash=${_EX2_splash},splash-time=3000 \ -drive if=none,id=drive0,cache=none,aio=threads,format=raw,file=${_EXTERNAL2_img} \ -device virtio-blk,drive=drive0 \ -netdev tap,id=nd0,ifname=${_EXTERNAL2_tap2},script=no,downscript=no \ -device e1000,netdev=nd0,mac=${_EXTERNAL2_mac} \ -name \"${_EXTERNAL2_name}\" & ========================================================== SCRIPT: external3.sh #!/bin/sh # IPFW Primer # License: 3-clause BSD # Author: Jim Brown, jpb@jimby.name # Code: https://github.com/jimmyb-gh/ipfw-primer # # FreeBSD QEMU VM startup script for external3 VM. # # external3.sh: FreeBSD QEMU VM startup script for external3 VM. # Usage: sudo /bin/sh external3.sh # Note: Set up for serial console. Start another session and telnet to the port shown. # # FreeBSD QEMU VM startup script # # external3.sh # #set -x # pick up environment for this run . ./vm_envs.sh echo [ISO=${_EXTERNAL3_ISO}] echo [mem=${_EXTERNAL3_mem}] echo [hdsize=${_EXTERNAL3_hdsize}] echo [img=${_EXTERNAL3_img}] echo [mac=${_EXTERNAL3_mac}] echo [name=${_EXTERNAL3_name}] echo [tap3=${_EXTERNAL3_tap3}] echo [telnetport=${_EXTERNAL3_telnetport}] # #exit # echo echo "NOTE!!! telnet server running! To start QEMU telnet to localhost $_EXTERNAL3_telnetport" echo /usr/local/bin/qemu-system-x86_64 -monitor none \ -serial telnet:localhost:${_EXTERNAL3_telnetport},server=on,wait=off \ -cpu qemu64 \ -vga cirrus \ -m ${_EXTERNAL3_mem} \ -cdrom ${_EXTERNAL3_ISO} \ -boot order=cd,menu=on,splash=${_EX3_splash},splash-time=3000 \ -drive if=none,id=drive0,cache=none,aio=threads,format=raw,file=${_EXTERNAL3_img} \ -device virtio-blk,drive=drive0 \ -netdev tap,id=nd0,ifname=${_EXTERNAL3_tap3},script=no,downscript=no \ -device e1000,netdev=nd0,mac=${_EXTERNAL3_mac} \ -name \"${_EXTERNAL3_name}\" & ========================================================== SCRIPT: firewall.sh #!/bin/sh # IPFW Primer # License: 3-clause BSD # Author: Jim Brown, jpb@jimby.name # Code: https://github.com/jimmyb-gh/ipfw-primer # # FreeBSD QEMU VM startup script for firewall VM. # # firewall.sh: FreeBSD QEMU VM startup script for firewall VM. # Usage: sudo /bin/sh firewall.sh # Note: Set up for serial console. Start another session and telnet to the port shown. # # FreeBSD QEMU VM startup script # # firewall.sh # #set -x # pick up environment for this run . ./vm_envs.sh echo [$_FIREWALL_ISO] echo [$_FIREWALL_mem] echo [$_FIREWALL_hdsize] echo [$_FIREWALL_img] echo [$_FIREWALL_mac1] echo [$_FIREWALL_mac2] echo [$_FIREWALL_name] echo [$_FIREWALL_tap0] echo [$_FIREWALL_tap4] echo [$_FIREWALL_telnetport] #exit # Note - the firewall has two interfaces - em0 and em1. # em0 is considered the 'external' interface and # em1 is considered the 'internal' interface. echo echo "NOTE!!! telnet server running! To start QEMU telnet to localhost $_FIREWALL_telnetport" echo /usr/local/bin/qemu-system-x86_64 -monitor none \ -serial telnet:localhost:${_FIREWALL_telnetport},server=on,wait=off \ -cpu qemu64 \ -display gtk \ -vga cirrus \ -m ${_FIREWALL_mem} \ -cdrom ${_FIREWALL_ISO} \ -boot order=cd,menu=on,splash=${_FW_splash},splash-time=3000 \ -drive if=none,id=drive0,cache=none,aio=threads,format=raw,file=${_FIREWALL_img} \ -device virtio-blk,drive=drive0 \ -netdev tap,id=nd0,ifname=${_FIREWALL_tap0},script=no,downscript=no \ -device e1000,netdev=nd0,mac=${_FIREWALL_mac1} \ -netdev tap,id=nd1,ifname=${_FIREWALL_tap4},script=no,downscript=no \ -device e1000,netdev=nd1,mac=${_FIREWALL_mac2} \ -name \"${_FIREWALL_name}\" & ========================================================== SCRIPT: firewall2.sh #!/bin/sh # IPFW Primer # License: 3-clause BSD # Author: Jim Brown, jpb@jimby.name # Code: https://github.com/jimmyb-gh/ipfw-primer # # FreeBSD QEMU VM startup script for firewall2 VM. # # firewall2.sh: FreeBSD QEMU VM startup script for firewall2 VM. # Usage: sudo /bin/sh firewall2.sh # Note: Set up for serial console. Start another session and telnet to the port shown. # # FreeBSD QEMU VM startup script # # firewall2.sh # #set -x # pick up environment for this run . ./vm_envs.sh echo [ISO=${_FIREWALL2_ISO}] echo [mem=${_FIREWALL2_mem}] echo [hdsize=${_FIREWALL2_hdsize}] echo [img=${_FIREWALL2_img}] echo [mac1=${_FIREWALL2_mac1}] echo [mac2=${_FIREWALL2_mac2}] echo [name=${_FIREWALL2_name}] echo [tap9=${_FIREWALL2_tap9}] echo [tap10=${_FIREWALL2_tap10}] echo [telnetport=${_FIREWALL2_telnetport}] #exit # Note - the firewall has two interfaces - em0 and em1. # em0 is considered the 'external' interface and # em1 is considered the 'internal' interface. echo echo "NOTE!!! telnet server running! To start QEMU telnet to localhost $_FIREWALL2_telnetport" echo /usr/local/bin/qemu-system-x86_64 -monitor none \ -serial telnet:localhost:${_FIREWALL2_telnetport},server=on,wait=off \ -cpu qemu64 \ -display gtk \ -vga cirrus \ -m ${_FIREWALL2_mem} \ -cdrom ${_FIREWALL_ISO} \ -boot order=cd,menu=on,splash=${_FW2_splash},splash-time=3000 \ -drive if=none,id=drive0,cache=none,aio=threads,format=raw,file=${_FIREWALL2_img} \ -device virtio-blk,drive=drive0 \ -netdev tap,id=nd0,ifname=${_FIREWALL2_tap9},script=no,downscript=no \ -device e1000,netdev=nd0,mac=${_FIREWALL2_mac1} \ -netdev tap,id=nd1,ifname=${_FIREWALL2_tap10},script=no,downscript=no \ -device e1000,netdev=nd1,mac=${_FIREWALL2_mac2} \ -name \"${_FIREWALL2_name}\" & ========================================================== SCRIPT: internal.sh #!/bin/sh # IPFW Primer # License: 3-clause BSD # Author: Jim Brown, jpb@jimby.name # Code: https://github.com/jimmyb-gh/ipfw-primer # # FreeBSD QEMU VM startup script for internal VM. # # internal.sh: FreeBSD QEMU VM startup script for internal VM. # Usage: sudo /bin/sh internal.sh # Note: Set up for serial console. Start another session and telnet to the port shown. # # FreeBSD QEMU VM startup script # # internal.sh # #set -x # pick up environment for this run . ./vm_envs.sh echo [ISO=${_INTERNAL_ISO}] echo [mem=${_INTERNAL_mem}] echo [hdsize=${_INTERNAL_hdsize}] echo [img=${_INTERNAL_img}] echo [mac=${_INTERNAL_mac}] echo [name=${_INTERNAL_name}] echo [tap5=${_INTERNAL_tap5}] echo [telnetport=${_INTERNAL_telnetport}] # #exit echo echo "NOTE!!! telnet server running! To start QEMU telnet to localhost $_INTERNAL_telnetport" echo /usr/local/bin/qemu-system-x86_64 -monitor none \ -serial telnet:localhost:${\_INTERNAL_telnetport},server=on,wait=off \ -cpu qemu64 \ -vga cirrus \ -m ${_INTERNAL_mem} \ -cdrom ${_INTERNAL_ISO} \ -boot order=cd,menu=on,splash=${_INT_splash},splash-time=3000 \ -drive if=none,id=drive0,cache=none,aio=threads,format=raw,file=${_INTERNAL_img} \ -device virtio-blk,drive=drive0 \ -netdev tap,id=nd0,ifname=${_INTERNAL_tap5},script=no,downscript=no \ -device e1000,netdev=nd0,mac=${_INTERNAL_mac} \ -name \"${_INTERNAL_name}\" & ========================================================== #!/bin/sh # FreeBSD qemu vm startup script # # jail1.sh # #set -x # pick up environment for this run . ./vm_envs.sh echo [ISO=${_JAIL1_ISO}] echo [mem=${_JAIL1_mem}] echo [hdsize=${_JAIL1_hdsize}] echo [img=${_JAIL1_img}] echo [mac=${_JAIL1_mac}] echo [name=${_JAIL1_name}] echo [tap2=${_JAIL1_tap2}] echo [telnetport=${_JAIL1_telnetport}] # #exit # # minimal check that environment is sane #if [ "X${_FBSD_ISO}" = "X" -o ! -s ${_FBSD_ISO} ] #then # echo "Parameter or file failure on _FBSD_ISO [${_FBSD_ISO}]" # echo "Check vm_envs.sh" # exit 1 #fi # # #echo #echo "NOTE!!! telnet server running! To start QEMU telnet to localhost $_JAIL1_telnetport" #echo # -serial telnet:localhost:${_JAIL1_telnetport},server=on,wait=on \ /usr/local/bin/qemu-system-x86_64 -monitor none \ -serial telnet:localhost:${_JAIL1_telnetport},server=on,wait=off \ -cpu qemu64 \ -vga cirrus \ -m ${_JAIL1_mem} \ -cdrom ${_JAIL1_ISO} \ -boot order=cd,menu=on,splash=${_JAIL1_splash},splash-time=3000 \ -drive if=none,id=drive0,cache=none,aio=threads,format=raw,file=${_JAIL1_img} \ -device virtio-blk,drive=drive0 \ -netdev tap,id=nd0,ifname=${_JAIL1_tap12},script=no,downscript=no \ -device e1000,netdev=nd0,mac=${\_JAIL1_mac} \ -name \"${_JAIL1_name}\" & ========================================================== SCRIPT: mkbr.sh #!/bin/sh # IPFW Primer # License: 3-clause BSD # Author: Jim Brown, jpb@jimby.name # Code: https://github.com/jimmyb-gh/ipfw-primer # # FreeBSD startup script for bridge and tap devices. # # mkbr.sh: FreeBSD startup script for bridge and tap devices. # EXAMPLE Usage: sudo /bin/sh mkbr.sh reset bridge0 tap0 tap1 bridge1 tap2 bridge2 tap3 tap4 tap5 em0 # # mkbr.sh - manage bridge and tap interfaces for FreeBSD. # # Have fun, but don't blame me if it smokes your machine. # # This script is used to start the bridge and tap interfaces. # # To create one bridge, two tap interfaces, and connect the # local ethernet interace (here em0), run under sudo as follows: # sudo /bin/sh mkbr.sh reset bridge0 tap0 tap1 em0 # # The script can be used to create any number of bridges and taps # for any internal network design: # sudo /bin/sh mkbr.sh reset bridge0 tap0 tap1 bridge1 tap2 tap3 tap4 bridge2 tap5 em0 ... etc. # # To add other taps to existing bridges, do not specify the "reset" parameter. # sudo /bin/sh mkbr.sh bridge0 tap10 tap11 bridge1 tap12 tap13 ... etc. # # To delete all bridge and tap devices: # sudo /bin/sh mkbr.sh reset # # #set -x usage() { echo "Usage: mkbr.sh ["reset"] <bridgeN> <tapA> [[<bringeM>] <tapB> <tapC> ...]" echo "You must be root to run this script." exit 1 } if [ "X0" != "X`id -u`" ] then usage fi if [ $# = 0 ] then usage; fi if [ $1 = "reset" ] then echo echo "Note - if_bridge and/or if_tap may be compiled into the kernel and can't be unloaded. Adjust interfaces manually if necessary." echo echo "unloading..." kldunload if_bridge kldunload if_tap echo echo "Deleting any remaining bridge and tap devices:" for i inifconfig -l
do echo "Interface: ${i}" case ${i} in bridge*) echo " ... destroying bridge ${i}" ifconfig ${i} destroy ;; tap*) echo " ... destroying tap ${i}" ifconfig ${i} destroy ;; esac done sleep 1 echo "loading..." kldload if_bridge kldload if_tap shift RESET="Y" echo "RESET=Y" # Before using the tap devices in QEMU, two sysctls require adjustment: sysctl net.link.tap.user_open=1 sysctl net.link.tap.up_on_open=1 else RESET="N" echo "RESET=N" fi PARAM=$1 while [ "X${PARAM}" != "X" ] do # echo "PARAM=[$PARAM]" case $PARAM in bridge*) BRIDGE=$1 # if [ "$RESET" = "Y" ] # then echo ifconfig $BRIDGE create ifconfig $BRIDGE create echo ifconfig $BRIDGE ifconfig $BRIDGE # fi echo ifconfig $BRIDGE up ifconfig $BRIDGE up ;; tap*) TAP=$1 # if [ "$RESET" = "Y" ] # then echo ifconfig $TAP create ifconfig $TAP create # fi echo "ifconfig $BRIDGE addm $TAP " ifconfig $BRIDGE addm $TAP ;; *) echo "*** Checking to see if $1 is a valid interface" TMPINT=$1 RESULT="IS NOT" for i inifconfig -l
do # echo $i if [ "${i}X" = "${TMPINT}X" ] then echo "Found a valid interface: ${TMPINT} Adding it to the bridge. Check results." echo "ifconfig $BRIDGE addm $TMPINT" ifconfig $BRIDGE addm $TMPINT RESULT="IS" break; else echo -n "." fi done echo "Interface ${TMPINT} $RESULT a valid interface." ;; esac shift PARAM=$1 done exit 0 ========================================================== SCRIPT: runvm.sh #!/bin/sh # IPFW Primer # License: 3-clause BSD # Author: Jim Brown, jpb@jimby.name # Code: https://github.com/jimmyb-gh/ipfw-primer # # FreeBSD QEMU VM startup script for multiple VMs at once. # # runvm.sh: FreeBSD QEMU VM startup script for multiple VMs. # EXAMPLE Usage: /bin/sh runvm.sh firewall external1 external2 internal # # location: FreeBSD Host # # runvm.sh - run virtual machines specified on the command line. # # To use this script, run mkbr.sh first to set up the bridge and # tap configurations for the desired network architecture. # # NOTE: this script works best on XFCE4 desktop as it takes advantage of the # xfce4-terminal and it's ability to use multiple tabs. # # >>>> It is unlikely to work on another desktop. <<<< # # Essentially, this script is a big case statement. It gets the # command line names of the virtual machines and calls a function # that starts the virtual machine. # # pick up environment for this run . ./vm_envs.sh #set -x #WKDIR=$HOME/LAB/SCRIPTS export WKDIR=$HOME/ipfw echo "[${WKDIR}]" usage() { echo "Usage: /bin/sh runvm.sh vmname [vmname ...]" echo "Each virtual machine opens up on xfce4-terminal with two tabs -" echo " one for the qemu virtual machine, and one for the serial" echo " terminal interface." echo "" exit 1 } CURDIR=pwd
if [ "X${CURDIR}" != "X${WKDIR}/SCRIPTS" ] then usage; fi if [ $# = 0 ] then usage; fi # Functions for each VM dnshost_vm () { # DNS host echo "in function: [${_DNSHOST_telnetport}]" xfce4-terminal --window --geometry="80x24+50+50" --zoom="-1" \ -T "${_DNSHOST_name}" -e "bash -c \"cd ${WKDIR}/SCRIPTS && sudo /bin/sh dnshost.sh ; bash\"" \ --tab -T "${_DNSHOST_name}" -e "bash -c \"cd ${WKDIR}/SCRIPTS && sleep 2 && (. ./vm_envs.sh;telnet -4 localhost ${_DNSHOST_telnetport}); bash\"" return } external1_vm () { # external1 echo "in function: [${_EXTERNAL1_telnetport}]" xfce4-terminal --window --geometry="80x24+75+75" --zoom="-1" \ -T "${_EXTERNAL1_name}" -e "bash -c \"cd $WKDIR/SCRIPTS && sudo /bin/sh external1.sh ; bash\"" \ --tab -T "${_EXTERNAL1_name}" -e "bash -c \"cd $WKDIR/SCRIPTS && sleep 2 && (. ./vm_envs.sh;telnet localhost ${_EXTERNAL1_telnetport}); bash\"" return } external2_vm () { # external2 echo "in function: [${_EXTERNAL2_telnetport}]" xfce4-terminal --window --geometry="80x24+100+100" --zoom="-1" \ -T "${_EXTERNAL2_name}" -e "bash -c \"cd $WKDIR/SCRIPTS && sudo /bin/sh external2.sh ; bash\"" \ --tab -T "${_EXTERNAL2_name}" -e "bash -c \"cd $WKDIR/SCRIPTS && sleep 2 && (. ./vm_envs.sh;telnet localhost ${_EXTERNAL2_telnetport}); bash\"" return } external3_vm () { # external3 echo "in function: [${_EXTERNAL3_telnetport}]" xfce4-terminal --window --geometry="80x24+125+125" --zoom="-1" \ -T "${_EXTERNAL3_name}" -e "bash -c \"cd $WKDIR/SCRIPTS && sudo /bin/sh external3.sh ; bash\"" \ --tab -T "${_EXTERNAL3_name}" -e "bash -c \"cd $WKDIR/SCRIPTS && sleep 2 && (. ./vm_envs.sh;telnet localhost ${_EXTERNAL3_telnetport}); bash\"" return } firewall_vm () { # Firewall echo "in function: [${_FIREWALL_telnetport}]" xfce4-terminal --window --geometry="80x24+150+150" --zoom="-1" \ -T "${_FIREWALL_name}" -e "bash -c \"cd $WKDIR/SCRIPTS && sudo /bin/sh firewall.sh ; bash\"" \ --tab -T "${_FIREWALL_name}" -e "bash -c \"cd $WKDIR/SCRIPTS && sleep 2 && (. ./vm_envs.sh; telnet localhost ${_FIREWALL_telnetport}); bash\"" return } firewall2_vm () { # Firewall2 echo "in function: [${_FIREWALL2_telnetport}]" xfce4-terminal --window --geometry="80x24+175+175" --zoom="-1" \ -T "${_FIREWALL2_name}" -e "bash -c \"cd $WKDIR/SCRIPTS && sudo /bin/sh firewall2.sh ; bash\"" \ --tab -T "${_FIREWALL2_name}" -e "bash -c \"cd $WKDIR/SCRIPTS && sleep 2 && (. ./vm_envs.sh;telnet localhost ${_FIREWALL2_telnetport}); bash\"" return } internal_vm () { # internal echo "in function: [${_INTERNAL_telnetport}]" xfce4-terminal --window --geometry="80x24+200+200" --zoom="-1" \ -T "${_INTERNAL_name}" -e "bash -c \"cd $WKDIR/SCRIPTS && sudo /bin/sh internal.sh ; bash\"" \ --tab -T "${_INTERNAL_name}" -e "bash -c \"cd $WKDIR/SCRIPTS && sleep 2 && (. ./vm_envs.sh;telnet localhost ${_INTERNAL_telnetport}); bash\"" return } v6only_vm () { # v6only echo "in function: [${_V6ONLY_telnetport}]" xfce4-terminal --window --geometry="80x24+225+225" --zoom="-1" \ -T "${_V6ONLY_name}" -e "bash -c \"cd $WKDIR/SCRIPTS && sudo /bin/sh v6only.sh ; bash\"" \ --tab -T "${_V6ONLY_name}" -e "bash -c \"cd $WKDIR/SCRIPTS && sleep 2 && (. ./vm_envs.sh;telnet localhost ${_V6ONLY_telnetport}); bash\"" return } # # Startup the requested VMs # PARAM=$1 while [ "X${PARAM}" != "X" ] do echo "PARAM = [${PARAM}]" case ${PARAM} in dnshost) echo "dnshost ..." echo "_DNSHOST_telnetport = [${_DNSHOST_telnetport}]" dnshost_vm ;; external1) echo "external1 ..." echo "_EXTERNAL1_telnetport = [${_EXTERNAL1_telnetport}]" external1_vm ;; external2) echo "external2 ..." echo "_EXTERNAL2_telnetport = [${_EXTERNAL2_telnetport}]" external2_vm ;; external3) echo "external3 ..." echo "_EXTERNAL3_telnetport = [${_EXTERNAL3_telnetport}]" external3_vm ;; firewall) echo "firewall ..." echo "_FIREWALL_telnetport = [${_FIREWALL_telnetport}]" firewall_vm ;; firewall2) echo "firewall2 ..." echo "_FIREWALL2_telnetport = [${_FIREWALL2_telnetport}]" firewall2_vm ;; internal) echo "internal ..." echo "_INTERNAL_telnetport = [${_INTERNAL_telnetport}]" internal_vm ;; v6only) echo "v6only ..." echo "_V6ONLY_telnetport = [${_V6ONLY_telnetport}]" v6only_vm ;; *) echo "" echo "*** ERROR: NO VM NAMED [$PARAM]" echo "" ;; esac shift sleep 3 PARAM=$1 done exit 0 ========================================================== SCRIPT: swim.sh #!/bin/sh # IPFW Primer # License: 3-clause BSD # Author: Jim Brown, jpb@jimby.name # Code: https://github.com/jimmyb-gh/ipfw-primer # # Serial Window Management Script Using tmux. (swim.sh) # # Usage: /bin/sh swim.sh # Note: This program manages multiple serial termainal windows for QEMU # VMs on the host. # Make sure to uncomment the lines below for the windows you want. #set -x # Check for an existing tmux session tmux has-session -t 0 2>/dev/null if [ $? != 0 ]; then tmux new-session -d -s 0 tmux new-window -t 0:1 -n 'firewall' 'echo; echo Use \"telnet localhost 4450\" for firewall ; echo; /bin/sh' tmux new-window -t 0:2 -n 'external1' 'echo; echo Use \"telnet localhost 4410\" for external1; echo; /bin/sh' tmux new-window -t 0:3 -n 'external2' 'echo; echo Use \"telnet localhost 4420\" for external2; echo; /bin/sh' tmux new-window -t 0:4 -n 'external3' 'echo; echo Use \"telnet localhost 4430\" for external3; echo; /bin/sh' # tmux new-window -t 0:5 -n 'internal' 'echo; echo Use \"telnet localhost 44200\" for internal; echo; /bin/sh' # tmux new-window -t 0:6 -n 'firewall2' 'echo; echo Use \"telnet localhost 4250\" for firewall2; echo; /bin/sh' # tmux new-window -t 0:7 -n 'v6only' 'echo; echo Use \"telnet localhost 4460\" for v6only; echo; /bin/sh' # tmux new-window -t 0:8 -n 'dnshost' 'echo; echo Use \"telnet localhost 4453\" for dnshost; echo; /bin/sh' # tmux new-window -t 0:9 -n 'jail1' 'echo; echo Use \"telnet localhost 4470\" for jail1; echo; /bin/sh' # Set the default command shell set-option -g default-command "/bin/sh" fi # Set the focus on window 0:0, your existing shell. tmux select-window -t 0:0 # Attach to the session tmux attach-session -t 0 exit ========================================================== SCRIPT: v6only.sh #!/bin/sh # IPFW Primer # License: 3-clause BSD # Author: Jim Brown, jpb@jimby.name # Code: https://github.com/jimmyb-gh/ipfw-primer # # FreeBSD QEMU VM startup script for v6only VM. # # v6only.sh: FreeBSD QEMU VM startup script for v6only VM. # Usage: sudo /bin/sh v6only.sh # Note: Set up for serial console. Start another session and telnet to the port shown. # # FreeBSD QEMU VM startup script # # v6only.sh # #set -x # pick up environment for this run . ./vm_envs.sh echo [ISO=${_V6ONLY_ISO}] echo [mem=${_V6ONLY_mem}] echo [hdsize=${_V6ONLY_hdsize}] echo [img=${_V6ONLY_img}] echo [mac=${_V6ONLY_mac}] echo [name=${_V6ONLY_name}] echo [tap6=${_V6ONLY_tap6}] echo [telnetport=${_V6ONLY_telnetport}] # #exit echo echo "NOTE!!! telnet server running! To start QEMU telnet to localhost $_V6ONLY_telnetport" echo /usr/local/bin/qemu-system-x86_64 -monitor none \ -serial telnet:localhost:${_V6ONLY_telnetport},server=on,wait=off \ -cpu qemu64 \ -vga cirrus \ -m ${_V6ONLY_mem} \ -cdrom ${_V6ONLY_ISO} \ -boot order=cd,menu=on,splash=${_V6_splash},splash-time=3000 \ -drive if=none,id=drive0,cache=none,aio=threads,format=raw,file=${_V6ONLY_img} \ -device virtio-blk,drive=drive0 \ -netdev tap,id=nd0,ifname=${_V6ONLY_tap6},script=no,downscript=no \ -device e1000,netdev=nd0,mac=${_V6ONLY_mac} \ -name \"${_V6ONLY_name}\" & ========================================================== SCRIPT: vm_envs.sh # IPFW Primer # License: 3-clause BSD # Author: Jim Brown, jpb@jimby.name # Code: https://github.com/jimmyb-gh/ipfw-primer # # FreeBSD QEMU VM environment script. # # vmenv.sh: FreeBSD QEMU VM environment setup script. # Usage: ./bin/sh vmenv.sh # # vm_envs.sh - environment for setting up virtual machines # for the IPFW examples lab. # # Set the environment variables below (or keep the defaults) # Note that the default disk size for each virtual machine is # 4GB - so all five VMs will take up about 32GB if you preallocate # space. # # In brief: # # Install FreeBSD on the host machine and update to latest patch level. # Install desktop software. # Install QEMU (latest) # Install nmap (needed for ncat) # Install sudo # # # The script mkbr.sh should be run before starting # the virtual machines. mkbr.sh sets up the bridge and tap # devices needed by the VMs. # # sudo /bin/sh ./mkbr.sh reset bridge0 tap0 tap1 tap2 tap3 em0 bridge1 tap4 tap5 # # This will set up the devices needed by QEMU. # # #The file directory layout for the examples is: # # ~/ipfw # /SCRIPTS # _CreateAllVMs.sh (create Qemu disks images) # dnshost.sh (run script for dns server VM) # external1.sh (run scripts for external VMs) # external2.sh " # external3.sh " # firewall.sh (run script for firewall VM) # firewall2.sh (run script for firewall2 VM) # internal.sh (script to setup internal host) # jail1.sh (script to setup jail1 host) # v6only.sh (run script for IPv6 only VM) # mkbr.sh (script to create bridge and tap devices) # vm_envs.sh (script to manage all parameters) # runvm.sh (script to manage all virtual machines) # /BMP # dns_splash_640x480.bmp # external1_splash_640x480.bmp # external2_splash_640x480.bmp # external3_splash_640x480.bmp # internal_splash_640x480.bmp # jail1_splash_640x480.bmp # ipfw2_splash_640x480.bmp # ipfw_splash_640x480.bmp # v6only_splash_640x480.bmp # dnshost_splash_640x480.bmp # /ISO # fbsd.iso (latest FreeBSD install iso) # /VM # dnshost.qcow2 (Qemu disk image for dns host) # external1.qcow2 (Qemu disk image for external hosts) # external2.qcow2 " # external3.qcow2 " # firewall.qcow2 (Qemu disk image for firewall) # firewall2.qcow2 (Qemu disk image for firewall2) # internal.qcow2 (Qemu disk image for an internal host) # jail1.qcow2 (Qemu disk image for an jail1 host) # v6only.qcow2 (Qemu disk image for an ipv6only host) # # # Start the VMs and install / test one at a time. # # sudo /bin/sh firewall.sh # sudo /bin/sh firewall2.sh # sudo /bin/sh external1.sh # sudo /bin/sh external2.sh # sudo /bin/sh external3.sh # sudo /bin/sh internal.sh # sudo /bin/sh jail1.sh # sudo /bin/sh v6only.sh # sudo /bin/sh dnshost.sh # # Each install should first utilize DHCP to get a valid IP address # After install, proceed to update FreeBSD with "freebsd-update fetch install" # Install packages: # Use whatever shell you prefer. Bash is listed below. # Firewall - pkg install bash cmdwatch lynx iperf3 nmap hping3 nginx # All others - pkg install bash cmdwatch lynx iperf3 nmap hping3 nginx # DNS host - pkg install bind918 dual-dhclient bash cmdwatch lynx nginx # # Reset all IP addresses for static usage: # # Host interface: add 172.16.10.100/24 alias # Disable any firewall (pf, ipfw, etc.) on the host. # BE SURE this is Ok for your environment. # # Firewall em0 172.16.10.50/24, default gateway 172.16.10.100 # em1 10.10.10.50/24 # # Firewall2 em0 as needed # em1 as needed # # External1: em0 172.16.10.10/24, default gateway 172.16.10.100 # External2: em0 172.16.10.20/24, default gateway 172.16.10.100 # External3: em0 172.16.10.30/24, default gateway 172.16.10.100 # Internal: em0 10.10.10.200/24, default gateway 10.10.10.50 # # v6only as needed # dnshost as needed # jail1 as needed # # export _BASE=/home/jpb/ipfw # Bridge and tap info export _FIREWALL_tap0=tap0 export _EXTERNAL1_tap1=tap1 export _EXTERNAL2_tap2=tap2 export _EXTERNAL3_tap3=tap3 export _FIREWALL_tap4=tap4 export _INTERNAL_tap5=tap5 export _JAIL1_tap12=tap12 export _V6ONLY_tap6=tap6 export _DNSHOST_tap7=tap7 export _DNSHOST_tap8=tap8 export _FIREWALL2_tap9=tap9 export _FIREWALL2_tap10=tap10 export _DNSHOST_tap11=tap11 export _bridge0_=bridge0 export bridge1=bridge1 export bridge2=bridge2 # Disk sizes export _EXTERNAL1_hdsize=4G export _EXTERNAL2_hdsize=4G export _EXTERNAL3_hdsize=4G export _FIREWALL_hdsize=4G export _FIREWALL2_hdsize=4G export _INTERNAL_hdsize=4G export _JAIL1_hdsize=8G # Note larger size disk export _V6ONLY_hdsize=4G export _DNSHOST_hdsize=4G # Is this needed anymore? export _FBSD_ISO=${_BASE}/ISO/fbsd.iso # Boot iso locations export _DNSHOST_ISO=${_BASE}/ISO/fbsd.iso export _EXTERNAL1_ISO=${_BASE}/ISO/fbsd.iso export _EXTERNAL2_ISO=${_BASE}/ISO/fbsd.iso export _EXTERNAL3_ISO=${_BASE}/ISO/fbsd.iso export _FIREWALL_ISO=${_BASE}/ISO/fbsd.iso export _FIREWALL2_ISO=${_BASE}/ISO/fbsd.iso export _INTERNAL_ISO=${_BASE}/ISO/fbsd.iso export _JAIL1_ISO=${_BASE}/ISO/fbsd.iso export _V6ONLY_ISO=${_BASE}/ISO/fbsd.iso # Memory sizes export _DNSHOST_mem=1024 export _EXTERNAL1_mem=1024 # lower all to 512 if necessary export _EXTERNAL2_mem=1024 export _EXTERNAL3_mem=1024 export _FIREWALL_mem=1024 export _FIREWALL2_mem=1024 export _INTERNAL_mem=1024 export _JAIL1_mem=8192 # Note larger size memory for using ZFS export _V6ONLY_mem=1024 # Qemu disk image locations. export _DNSHOST_img=${_BASE}/VM/dnshost.qcow2 export _EXTERNAL1_img=${_BASE}/VM/external1.qcow2 export _EXTERNAL2_img=${_BASE}/VM/external2.qcow2 export _EXTERNAL3_img=${_BASE}/VM/external3.qcow2 export _FIREWALL_img=${_BASE}/VM/firewall.qcow2 export _FIREWALL2_img=${_BASE}/VM/firewall2.qcow2 export _INTERNAL_img=${_BASE}/VM/internal.qcow2 export _JAIL1_img=${_BASE}/VM/jail1.qcow2 export _V6ONLY_img=${_BASE}/VM/v6only.qcow2 # MAC addresses export _DNSHOST_mac1=02:49:53:53:53:53 export _DNSHOST_mac2=02:49:53:53:54:54 export _DNSHOST_mac3=02:49:53:53:55:55 export _EXTERNAL1_mac=02:45:58:54:31:10 export _EXTERNAL2_mac=02:45:58:54:32:20 export _EXTERNAL3_mac=02:45:58:54:33:30 export _FIREWALL_mac1=02:49:50:46:57:41 export _FIREWALL2_mac1=02:49:50:00:22:22 export _FIREWALL_mac2=02:49:50:46:57:42 export _FIREWALL2_mac2=02:49:50:22:22:22 export _INTERNAL_mac=02:49:4E:54:0a:42 export _JAIL1_mac=02:49:ba:ad:ba:be export _V6ONLY_mac=02:49:de:ad:be:ef # VM names export _DNSHOST_name=DNSHOST export _EXTERNAL1_name=EXTERNAL1 export _EXTERNAL2_name=EXTERNAL2 export _EXTERNAL3_name=EXTERNAL3 export _FIREWALL_name=FIREWALL export _FIREWALL2_name=FIREWALL2 export _INTERNAL_name=INTERNAL export _JAIL1_name=JAIL1 export _V6ONLY_name=V6ONLY # Slash images export _DNS_splash=${_BASE}/BMP/dns_splash_640x480.bmp export _EX1_splash=${_BASE}/BMP/external1_splash_640x480.bmp export _EX2_splash=${_BASE}/BMP/external2_splash_640x480.bmp export _EX3_splash=${_BASE}/BMP/external3_splash_640x480.bmp export _FW_splash=${_BASE}/BMP/ipfw_splash_640x480.bmp export _FW2_splash=${_BASE}/BMP/ipfw2_splash_640x480.bmp export _INT_splash=${_BASE}/BMP/internal_splash_640x480.bmp export _JAIL1_splash=${_BASE}/BMP/jail1_splash_640x480.bmp export _V6_splash=${_BASE}/BMP/ipv6_splash_640x480.bmp # # Telnet ports export _DNSHOST_telnetport=4453 export _EXTERNAL1_telnetport=4410 export _EXTERNAL2_telnetport=4420 export _EXTERNAL3_telnetport=4430 export _FIREWALL_telnetport=4450 export _FIREWALL2_telnetport=4250 export _INTERNAL_telnetport=44200 export _V6ONLY_telnetport=4460 export _JAIL1_telnetport=4470 # Bridge and Tap configurations. # # Note: em0 is used for the host interface. # Change as needed. # # Two bridge configuration # Standard examples # # em0 # | # External1(tap1) -----bridge0------(tap0)Firewall # External2(tap2) -----+ | (tap4) # External3(tap3) -------+ | # bridge1 # | # Internal(tap5) -----------------------------+ # # sudo /bin/sh mkbr.sh reset bridge0 tap0 tap1 tap2 tap3 em0 # # # # Two bridge configuration # NAT & LSNAT examples # # # # (firewall does LSNAT load balancing) # External1(tap1) -----bridge0------(tap0)Firewall # External2(tap2) -----+ | (tap4) # External3(tap3) -------+ | # (these function as internal machines) bridge1----em0 # | # Internal(tap5) -----------------------------+ # (this functions as an external machine) # # sudo /bin/sh mkbr.sh reset bridge0 tap0 tap1 tap2 tap3 bridge1 tap4 tap5 em0 # # # # Two bridge configuration # NAT64/DNS64 example # # ipv4 only NAT64 Translator # External1(tap1) ------bridge0-----(tap0)Firewall # (ipv4 only) + (tap4) # (webserver) | + # dnshost(tap7) | # (DNS server) | # (running DNS64) | # (tap8) | # | | # + | # ipv6 only | # v6only(tap6) --------bridge1----------------+ # (v6 only host) # # sudo /bin/sh mkbr.sh reset bridge0 tap0 tap1 tap7 bridge1 tap4 tap6 tap8 # # ========================================================== CODE: divert.c #include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> #include <netinet/ip.h> #include <netinet/tcp.h> #include <arpa/inet.h> #include <stdio.h> #include <string.h> #include <err.h> #include <sys/systm.h> #define DIVERT_PORT 700 void hexdump(void *ptr, int length, const char *hdr, int flags); int main(int argc, char *argv[]) { int fd, s; struct sockaddr_in sin; socklen_t sin_len; printf("Opening divert on port %d\n",DIVERT_PORT); fd = socket(PF_DIVERT, SOCK_RAW, 0); if (fd == -1) err(1, "socket"); memset(&sin, 0, sizeof(sin)); sin.sin_family = AF_INET; sin.sin_port = htons(DIVERT_PORT); sin.sin_addr.s_addr = 0; sin_len = sizeof(struct sockaddr_in); s = bind(fd, (struct sockaddr *) &sin, sin_len); if (s == -1) err(1, "bind"); for (;;) { ssize_t n; char packet[IP_MAXPACKET]; struct ip *ip; struct tcphdr *th; int hlen; char src[64], dst[64], printbuff[12]; memset(src, 0, sizeof(src)); memset(dst, 0, sizeof(dst)); memset(printbuff, 0, sizeof(printbuff)); memset(packet, 0, sizeof(packet)); n = recvfrom(fd, packet, sizeof(packet), 0, (struct sockaddr *) &sin, &sin_len); if (n == -1) { warn("recvfrom"); continue; } if (n < sizeof(struct ip)) { warnx("packet is too short"); continue; } ip = (struct ip *) packet; hlen = ip->ip_hl << 2; if (hlen < sizeof(struct ip) || ntohs(ip->ip_len) < hlen || n < ntohs(ip->ip_len)) { warnx("invalid IPv4 packet"); continue; } th = (struct tcphdr *) (packet + hlen); if (inet_ntop(AF_INET, &ip->ip_src, src, sizeof(src)) == NULL) (void)strlcpy(src, "?", sizeof(src)); if (inet_ntop(AF_INET, &ip->ip_dst, dst, sizeof(dst)) == NULL) (void)strlcpy(dst, "?", sizeof(dst)); printf("%s:%u -> %s:%u\n", src, ntohs(th->th_sport), dst, ntohs(th->th_dport) ); /* * dump the packet in hex and ascii with hexdump(3) */ hexdump((void *)packet, n, "|",0); n = sendto(fd, packet, n, 0, (struct sockaddr *) &sin, sin_len); if (n == -1) warn("sendto"); } return 0; }
Appendix C: Appendix C: Networking References
References for understanding IP based communications and building firewalls.
From /etc/rc.firewall:
Building Internet Firewalls, 2nd Edition Brent Chapman and Elizabeth Zwicky O'Reilly & Associates, Inc ISBN 1-56592-871-7 http://www.ora.com/ http://www.oreilly.com/catalog/fire2/ For a more advanced treatment of Internet Security read: Firewalls and Internet Security: Repelling the Wily Hacker, 2nd Edition William R. Cheswick, Steven M. Bellowin, Aviel D. Rubin Addison-Wesley / Prentice Hall ISBN 0-201-63466-X http://www.pearsonhighered.com/ http://www.pearsonhighered.com/educator/academic/product/0,3110,020163466X,00.html
Additional references:
TCP/IP Illustrated, Volume 1: The Protocols Author: W. Richard Stevens Publisher: Addison-Wesley Professional Publisher Website: Addison-Wesley Professional Date Published: November 1994 ISBN: 978-0201633467 The TCP/IP Guide: A Comprehensive, Illustrated Internet Protocols Reference Author: Charles M. Kozierok Publisher: No Starch Press Publisher Website: No Starch Press Date Published: October 2005 ISBN: 978-1593270476 Internetworking with TCP/IP Volume One: Principles, Protocols, and Architecture Author: Douglas E. Comer Publisher: Pearson Publisher Website: Pearson Date Published: 6th Edition, 2013 ISBN: 978-0136085300 Computer Networks: A Systems Approach Authors: Larry L. Peterson and Bruce S. Davie Publisher: Morgan Kaufmann Publisher Website: Morgan Kaufmann Date Published: 6th Edition, 2021 ISBN: 978-0128182000
Additional resources regarding firewalls:
Network Security, Firewalls, and VPNs Authors: J. Michael Stewart, Denise Kinsey Publisher: Jones & Bartlett Learning Publisher Website: Jones & Bartlett Learning Date Published: October 2020 ISBN: 978-1284183696 Firewall Fundamentals Author: David W. Chapman Jr. Publisher: Cisco Press Publisher Website: Cisco Press Date Published: June 2006 ISBN: 978-1587052213 The Best Damn Firewall Book Period Author: Thomas W. Shinder Publisher: Syngress Publisher Website: Syngress Date Published: January 2008 ISBN: 978-1597492188 Guide to Firewalls and Network Security: Intrusion Detection and VPNs Authors: Michael E. Whitman, Herbert J. Mattord, Richard Austin, Greg Holden Publisher: Cengage Learning Publisher Website: Cengage Learning Date Published: 2008 ISBN: 978-1435420168
Appendix D: Appendix D. Managing Serial Terminals with tmux and screen
D.1. Using tmux(1) for Managing Serial Terminals
Install tmux with:
# pkg install tmux
Run sh swim.sh in the SCRIPTS directory to start up the session manager running tmux.
The figure shows five named windows in one session (session [0]) with the tmux status line in green at the bottom:
Simplified tmux(1) Usage
tmux has a hierarchical organization.
Sessions - you start a session implicitly when you run tmux
Windows - you add a window with the command "tmux add-window" which can take a number of parameters. You can have multiple windows in a session at the same time. The current window is starred ("*") in the status bar at the bottom of the main window.
Panes - you can split a window into one or more panes either horizontally, or vertically.
The swim.sh script shows how to set up one session with several windows. Panes are not used.
tmux is controlled by the user typing in any open window or pane. tmux uses Ctl+b as its control key. You give tmux commands by typing the control key followed by one or more letters. To move from window to window use Ctl+b n to move to the next window or Ctl+b p to move to the previous window. Use Ctl+b ? for a list of all key bindings.
Type tmux kill-server in any session window to completely leave tmux.
Consult the tmux manual page tmux(1) for more usage details.
Accessing the QEMU Serial Consoles
To access the VM serial consoles, move to the indicated window and telnet to the port on the local host for that VM:
Move to the external1 window in tmux, then ~/ipfw/SCRIPTS $ telnet localhost 4410 Trying ::1... Connected to localhost. Escape character is '^]'. FreeBSD/amd64 (external1) (ttyu0) login:
The swim.sh script has the following un-commented lines. Un-comment additional lines as needed
0:bash - a terminal window of the user running swim.sh
1:firewall - a terminal window to access the firewall VM
2:external1 - a terminal window to access the external1 VM
3:external2 - a terminal window to access the external2 VM
4:external3 - a terminal window to access the external3 VM
The current window is marked with the '*' character in the status bar.
Note that the swim.sh script has entries for all windows used in this book. Uncomment the entries you need.
Run sh swim.sh in the SCRIPTS directory to start up the session manager running tmux.
D.2. Using screen(1) for Managing Serial Terminals
Install screen with:
# pkg install screen
screen, like tmux, is a terminal window manager. screen has its own control key - Ctl+a, and like tmux a list of key bindings is available at Ctl+a ?.
By default, it does not use a status line, and once activated, it looks like no manager is active at all. Type screen -ls to determine if there is an active screen session running.
To display a status line in a live session use:
Ctrl+a : hardstatus alwayslastline
Ctrl+a : hardstatus string "%{= bw}%-w%{= rW}[%n %t]%{-}%+w %=%{= kW} %H | %Y-%m-%d %c"
If you want it to always appear, edit the .screenrc file in your $HOME
directory and add:
hardstatus alwayslastline hardstatus string "%{= bw}%-w%{= rW}[%n %t]%{-}%+w %=%{= kW} %H | %Y-%m-%d %c"
To close all screen windows immediately and exit:
% screen -X quit
The scim.sh script has the following un-commented lines. Un-comment additional lines as needed
0:bash - a terminal window of the user running scim.sh
1:firewall - a terminal window to access the firewall VM
2:external1 - a terminal window to access the external1 VM
3:external2 - a terminal window to access the external2 VM
4:external3 - a terminal window to access the external3 VM
The current window is highlighted in the status bar.
Note that the scim.sh script has entries for all windows used in this book. Uncomment the entries you need.
Run sh scim.sh in the SCRIPTS directory to start up the session manager running screen.
Appendix E: Appendix E: DNS Server Configuration
DNS configuration for IPFW Primer book.
Manifest of dnshost scripts and file. File: dnshost_usrlocaletc_namedb.tgz Description: Contains the configuration for the BIND 9 DNS services that run on this machine. Installation: Install bind9 package first, then untar this collection as follows: # cd /usr/local/etc # tar xvzf dnshost_usrlocaletc_namedb.tgz Contents: % tar tvzf dnshost_usrlocaletc_namedb.tgz drwxr-xr-x 0 root wheel 0 Nov 19 12:00 namedb/ -rw-r--r-- 0 bind bind 2403 Nov 19 11:59 namedb/bind.keys drwxr-xr-x 0 bind bind 0 Nov 19 11:59 namedb/dynamic/ -rw-r--r-- 0 bind bind 2618 Dec 2 12:34 namedb/named.conf -rw-r--r-- 0 bind bind 21992 Nov 19 11:59 namedb/named.conf.sample -rw-r--r-- 0 bind bind 927 Nov 19 11:59 namedb/named.root -rw-r--r-- 0 bind bind 3317 Nov 19 11:59 namedb/named.root.SAVE drwxr-xr-x 0 bind bind 0 Dec 2 15:35 namedb/primary/ -rw------- 0 bind bind 100 Nov 19 11:59 namedb/rndc.key drwxr-xr-x 0 bind bind 0 Nov 19 11:59 namedb/secondary/ drwxr-xr-x 0 bind bind 0 Nov 19 11:59 namedb/working/ -rw-r--r-- 0 bind bind 297 Nov 19 11:59 namedb/working/managed-keys.bind -rw-r--r-- 0 bind bind 355 Nov 19 11:59 namedb/primary/ptr_198.51 -rw-r--r-- 0 bind bind 465 Nov 19 11:59 namedb/primary/ptr_203.0 -rw-r--r-- 0 bind bind 693 Dec 1 19:29 namedb/primary/example.com -rw-r--r-- 0 bind bind 148 Nov 19 11:59 namedb/primary/empty -rw-r--r-- 0 bind bind 407 Nov 19 14:12 namedb/primary/ptr_ipv6 -rw-r--r-- 0 bind bind 287 Dec 2 15:35 namedb/primary/managed-keys.bind -rw-r--r-- 0 bind bind 226 Nov 19 11:59 namedb/primary/localhost-reverse -rw-r--r-- 0 bind bind 158 Nov 19 11:59 namedb/primary/localhost-forward -rw-r--r-- 0 bind bind 351 Dec 1 19:30 namedb/primary/ptr_192.168 % ===================================================================================== bind.keys # Copyright (C) Internet Systems Consortium, Inc. ("ISC") # # SPDX-License-Identifier: MPL-2.0 # # This Source Code Form is subject to the terms of the Mozilla Public # License, v. 2.0. If a copy of the MPL was not distributed with this # file, you can obtain one at https://mozilla.org/MPL/2.0/. # # See the COPYRIGHT file distributed with this work for additional # information regarding copyright ownership. # The bind.keys file is used to override the built-in DNSSEC trust anchors # which are included as part of BIND 9. The only trust anchors it contains # are for the DNS root zone ("."). Trust anchors for any other zones MUST # be configured elsewhere; if they are configured here, they will not be # recognized or used by named. # # To use the built-in root key, set "dnssec-validation auto;" in the # named.conf options, or else leave "dnssec-validation" unset. If # "dnssec-validation" is set to "yes", then the keys in this file are # ignored; keys will need to be explicitly configured in named.conf for # validation to work. "auto" is the default setting, unless named is # built with "configure --disable-auto-validation", in which case the # default is "yes". # # This file is NOT expected to be user-configured. # # Servers being set up for the first time can use the contents of this file # as initializing keys; thereafter, the keys in the managed key database # will be trusted and maintained automatically. # # These keys are current as of Mar 2019. If any key fails to initialize # correctly, it may have expired. In that event you should replace this # file with a current version. The latest version of bind.keys can always # be obtained from ISC at https://www.isc.org/bind-keys. # # See https://data.iana.org/root-anchors/root-anchors.xml for current trust # anchor information for the root zone. trust-anchors { # This key (20326) was published in the root zone in 2017. . initial-key 257 3 8 "AwEAAaz/tAm8yTn4Mfeh5eyI96WSVexTBAvkMgJzkKTOiW1vkIbzxeF3 +/4RgWOq7HrxRixHlFlExOLAJr5emLvN7SWXgnLh4+B5xQlNVz8Og8kv ArMtNROxVQuCaSnIDdD5LKyWbRd2n9WGe2R8PzgCmr3EgVLrjyBxWezF 0jLHwVN8efS3rCj/EWgvIWgb9tarpVUDK/b58Da+sqqls3eNbuv7pr+e oZG+SrDK6nWeL3c6H5Apxz7LjVc1uTIdsIXxuOLYA4/ilBmSVIzuDWfd RUfhHdY6+cn8HFRm+2hM8AnXGXws9555KrUB5qihylGa8subX2Nn6UwN R1AkUTV74bU="; }; ===================================================================================== named.conf // Refer to the named.conf(5) and named(8) man pages, and the documentation // in /usr/local/share/doc/bind for more details. acl trusted-queriers { 203.0.113.0/24; 2001:db8:12::/64; 127.0.0.1; ::1; localhost; }; acl v6only-networks { 2001:db8:12::/64; }; options { directory "/usr/local/etc/namedb/primary"; pid-file "/var/run/named/pid"; dump-file "/var/dump/named_dump.db"; statistics-file "/var/stats/named.stats"; listen-on { any; }; listen-on-v6 { any; }; recursion no; allow-transfer { trusted-queriers; }; // NOTE: Remove comments when using DNS64 // dns64 64:FF9B::/96 { // clients { any; }; // exclude { 64:FF9B::/96; ::ffff:0000:0000/96; }; // suffix ::; // }; }; // forward zone zone "example.com" { type primary; file "/usr/local/etc/namedb/primary/example.com"; allow-transfer {trusted-queriers;}; }; // reverse zones for 203.0, 198.51, 192.168, and 2001:0db8 zone "0.203.in-addr.arpa" { type primary; file "/usr/local/etc/namedb/primary/ptr_203.0"; allow-transfer {trusted-queriers;}; }; zone "51.198.in-addr.arpa"{ type primary; file "/usr/local/etc/namedb/primary/ptr_198.51"; allow-transfer {trusted-queriers;}; }; zone "2.1.0.0.8.b.d.0.1.0.0.2.ip6.arpa" { type primary; file "/usr/local/etc/namedb/primary/ptr_ipv6"; allow-transfer {trusted-queriers; }; }; zone "168.192.in-addr.arpa" { type primary; file "/usr/local/etc/namedb/primary/ptr_192.168"; allow-transfer {trusted-queriers;}; }; // Block below added by BIND9 // RFCs 1912, 5735 and 6303 (and BCP 32 for localhost) zone "localhost" { type primary; file "/usr/local/etc/namedb/primary/localhost-forward"; }; zone "127.in-addr.arpa" { type primary; file "/usr/local/etc/namedb/primary/localhost-reverse"; }; zone "255.in-addr.arpa" { type primary; file "/usr/local/etc/namedb/primary/empty"; }; // RFC 1912-style zone for IPv6 localhost address (RFC 6303) zone "0.ip6.arpa" { type primary; file "/usr/local/etc/namedb/primary/localhost-reverse"; }; // "This" Network (RFCs 1912, 5735 and 6303) zone "0.in-addr.arpa" { type primary; file "/usr/local/etc/namedb/primary/empty"; }; // Our own root zone file so we don't leak out to the Internet zone "." { type master; file "/usr/local/etc/namedb/named.root"; allow-transfer {trusted-queriers; }; }; ===================================================================================== named.root ; This file holds the information on root name servers needed to ; initialize cache of Internet domain name servers ; (e.g. reference this file in the "cache . <file>" ; configuration file of BIND domain name servers). ; ; This file is made available by InterNIC ; under anonymous FTP as ; file /domain/named.cache ; on server FTP.INTERNIC.NET ; -OR- RS.INTERNIC.NET ; ; last update: November 16, 2017 ; related version of root zone: 2017111601 ; ; FORMERLY NS.INTERNIC.NET ; $TTL 3600 . 3600 IN SOA dnshost.example.com. jpb.dnshost.example.com ( 100 ; serial 14400 ; refresh 7200 ; retry 28800 ; expire 64000 ) ; min neg cache expire . 3600 NS dnshost.example.com. dnshost.example.com. 3600 A 203.0.113.53 dnshost.example.com. 3600 AAAA 2001:db8:12::53 ===================================================================================== rndc.key key "rndc-key" { algorithm hmac-sha256; secret "wesiGsTgu7OwV44aA6C2P8XmZdW4z/YdPJ4D/vRNPTM="; }; ===================================================================================== empty $TTL 3h @ SOA @ nobody.localhost. 42 1d 12h 1w 3h ; Serial, Refresh, Retry, Expire, Neg. cache TTL @ NS @ ; Silence a BIND warning @ A 127.0.0.1 ===================================================================================== example.com $TTL 3600 @ IN SOA example.com. jpb.example.com. ( 5 ; Serial 3h ; Refresh 1h ; Retry 1w ; Expire 1h ) ; Negative Cache TTL ; ; name servers - NS records @ IN NS dnshost.example.com. ; name servers - A records dnshost IN A 203.0.113.53 ;external1 IN A 203.0.113.10 external1 IN A 192.168.1.2 external2 IN A 203.0.113.20 external3 IN A 203.0.113.30 firewall IN A 203.0.113.50 firewall-em0 IN A 203.0.113.50 firewall-em1 IN A 198.51.100.50 firewall-em1 IN AAAA 2001:db8:12::50 internal IN A 198.51.100.200 ; name servers - AAAA records dnshost IN AAAA 2001:db8:12::53 v6only IN AAAA 2001:db8:12::6 ===================================================================================== localhost-forward $TTL 3h localhost. SOA localhost. nobody.localhost. 42 1d 12h 1w 3h ; Serial, Refresh, Retry, Expire, Neg. cache TTL NS localhost. A 127.0.0.1 AAAA ::1 ===================================================================================== localhost-reverse $TTL 3h @ SOA localhost. nobody.localhost. 42 1d 12h 1w 3h ; Serial, Refresh, Retry, Expire, Neg. cache TTL NS localhost. 1.0.0 PTR localhost. 1.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0 PTR localhost. ===================================================================================== managed-keys.bind $TTL 0 ; 0 seconds . IN SOA . . ( 100 ; serial 0 ; refresh (0 seconds) 0 ; retry (0 seconds) 0 ; expire (0 seconds) 0 ; minimum (0 seconds) ) KEYDATA 20241202213508 19700101000000 19700101000000 0 0 0 ; placeholder ===================================================================================== ptr_192.168 $TTL 3600 @ IN SOA example.com. jpb.example.com. ( 3 ; Serial 3h ; Refresh 1h ; Retry 1w ; Expire 1h ) ; Negative Cache TTL ; ; name servers - NS records IN NS dnshost.example.com. ; PTR Records 53.1 IN PTR dnshost.example.com. 2.1 IN PTR external1.example.com. ===================================================================================== ptr_198.51 $TTL 3600 @ IN SOA example.com. jpb.example.com. ( 3 ; Serial 3h ; Refresh 1h ; Retry 1w ; Expire 1h ) ; Negative Cache TTL ; ; name servers - NS records IN NS dnshost.example.com. ; PTR Records 50.100 IN PTR firewall-em1.example.com. 200.100 IN PTR internal.example.com. ===================================================================================== ptr_203.0 $TTL 3600 @ IN SOA example.com. jpb.example.com. ( 3 ; Serial 3h ; Refresh 1h ; Retry 1w ; Expire 1h ) ; Negative Cache TTL ; ; name servers - NS records IN NS dnshost.example.com. ; PTR Records 53.113 IN PTR dnshost.example.com. 10.113 IN PTR external1.example.com. 20.113 IN PTR external2.example.com. 30.113 IN PTR external3.example.com. 50.113 IN PTR firewall-em0.example.com. ===================================================================================== ptr_ipv6 $TTL 3600 @ IN SOA example.com. jpb.example.com. ( 3 ; Serial 3h ; Refresh 1h ; Retry 1w ; Expire 1h ) ; Negative Cache TTL ; @ IN NS dnshost.example.com. $ORIGIN 0.0.0.0.2.1.0.0.8.b.d.0.1.0.0.2.ip6.arpa. 3.5.0.0.0.0.0.0.0.0.0.0.0.0.0.0 IN PTR dnshost.example.com. 6.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0 IN PTR v6only.example.com. ===================================================================================== managed-keys.bind $ORIGIN . $TTL 0 ; 0 seconds @ IN SOA . . ( 100 ; serial 0 ; refresh (0 seconds) 0 ; retry (0 seconds) 0 ; expire (0 seconds) 0 ; minimum (0 seconds) ) KEYDATA 20220502020339 19700101000000 19700101000000 0 0 0 ; placeholder =====================================================================================