Welcome to "Computer Networking". In this lab, we will build a network stack from scratch, starting from Layer 2 to Layer 4. We will see and learn how the network protocol operates, in particular how the TCP protocol works. We expect this lab takes you between 30 to 50 hours to complete.
What I can not create, I do not understand — Richard Feynman.
All the assignments must be your own work. You must write each line of code, except for the code we provide as a part of the assignments. Do not copy any code from other students. To prevent plagiarism, we will use a code structure-similarity detection tool to check all submissions. An exception is that code snippets that do not relate to the network protocol (e.g., unlikely macros and unique pointer) are allowed to reuse.
Discussion with others. You can discuss high-level implementation ideas with other students. But again, DO NOT copy or look at anyone else's code.
We aim to make our implementation RFC-compliant (that is, following the formal requirements for the protocols in the TCP/IP stack as specified in a number of RFCs). This lab handout document is intended to be a friendly guidebook rather than a formal specification. Therefore, we encourage you to read the following materials and strictly follow them to your best.
- Ethernet II frame format, 802.3-2018 - IEEE Standard. We recommend a quick overview from Wikipedia.
- Internet Protocol, RFC791, in particular section 3.1.
- Transmission Control Protocol, RFC793, in particular section 3.9.
- Git: distributed version control system.
- CMake and Makefile: build automation, testing, packaging, and installation.
- Wireshark: the world's foremost and widely-used network protocol analyzer.
- vnetUtils: a utility for creating and interacting with the virtual network on your machine. See
3rd/vnetUtils/README.md
for details.
# Download from Github. You may create a fork and download it.
git clone https://github.com/SOAR-PKU/lab-netstack.git
cd lab-netstack
# Download submodules
git submodule init --recursive
git submodule update --recursive
# Build the targets. All built objects are in the `build` directory.
cmake -B build -DCMAKE_BUILD_TYPE=Debug
cmake --build build -j$(nproc)
Run all unit tests.
(cd build && ./runUnitTests)
Use a filter to run only the tests we specified.
(cd build && ./runUnitTests '--gtest_filter=SocketSessionTest.*')
This project uses clang-format-14 to format the source code. Follow https://apt.llvm.org/ to install. As an example for Ubuntu 20.04, use the following commands to install:
sudo add-apt-repository 'deb http://apt.llvm.org/focal/ llvm-toolchain-focal-14 main'
sudo apt update
sudo apt install -y clang-format-14
Use the clang-format
CMake target to format the code in /src
directory. Warning: this command permanently changes the code.
cmake --build build --target clang-format
This project uses Backward-cpp to print the stack trace. Install the following library to read the debug info:
sudo apt install -y binutils-dev
Our code will be done in C++11. The provided code might be different from what you have written and read before. Here are some hints that we hope can help you understand our code as well as write better C++ style code.
This protocol stack uses a single-thread non-blocking I/O framework. We use epoll to enable non-blocking I/O. We strongly suggest you to read files in src/base/
and make sure you understand them, especially EpollAlarm
and EpollServer
. src/base/
also includes some utilities you will use in the following sections.
We use Google Logging framework to log and debug.
// Log
LOG(INFO) << "Someting" << 42;
LOG(ERROR) << "Someting" << 42;
// If a line of code is executed many times, it may be useful to only log a
// message at certain intervals. This kind of logging is most useful for
// informational messages.
LOG_EVERY_N(INFO, 10) << "Got the " << google::COUNTER << "th packet";
// Log and kill the program
LOG(FATAL) << "Someting" << 42;
// Only output when compiled in debug mode (-DCMAKE_BUILD_TYPE=Debug)
DLOG(INFO) << "Someting" << 42;
// Sanity check in debug mode.
DCHECK(1 + 1 == 2);
This protocol stack uses Google Test for unit testing. See utils/
for testing mocks and ip_address_test.cpp
as an example. We encourage you to write your test suits but do not modify test suits that are parts of checkpoints.
C makes it easy to shoot yourself in the foot; C++ makes it harder, but when you do it blows your whole leg off. -- Bjarne Stroustrup
C leaves the memory allocation completely up to the developer; C++ develops constructors and destructors, RAII, and smart pointers to help take care of the memory allocation. When talking about memory safety, people use the concept of "ownership" to denote who is responsible for freeing the memory that was allocated. Check this article to learn more about memory ownership. In this lab, we use unique_ptr
s to mark the ownership and raw pointers to mark normal references.
In this lab, our goal is to complete the following tasks, each of which is marked as any one of PT (Programming Task), WT (Writing Task), or CP (Checkpoint).
- For the programming tasks, you should submit your implementation (additional documentation files are not necessary);
- For the writing tasks, you should submit your answer to each task;
- For each checkpoint, you should attach images, videos, or typescript files, along with a brief explanation.
Finish Exercise 1.2 of the guidebook. This gives you an intuitive overview of Ethernet II frames.
Insert your code to make the link layer work!
Show that your implementation can capture frames from a device and inject frames to a device using libpcap.
Finish Exercise 2.1 in the guidebook. This gives you an intuitive overview of IP packets.
Insert your code to make the network layer work!a
Run unit tests and show that your implementation passes test suit RoutingTableTest
.
Finish Exercises 2.2 and 2.3 in the guidebook. Run router
in each node and show the results.
Finish Exercise 3.1 of the guidebook. This gives you an intuitive overview of TCP segments.
Insert your code to let the transport layer work!
Run unit tests and show that your implementation passes test suit SocketSessionTest
Create a virtual network with the following topology and run echo_server
at ns4 and echo_client
at ns1. Paste the output of them.
ns1 --- ns2 --- ns3 --- ns4
These labs are worth 25 points (pts). Your gained points will be your final grade of lab 1, 2, 3 in this course. See points for each task above.
You should submit your project in an archive named lab[1|2|3]-[your name]-[your student ID].[tar|tar.gz|zip]
. In addition to the provided code, you should add:
- a file named
solution.pdf
containing your writing task solutions; - a file named
not-implemented.pdf
listing tasks/features that are required but not implemented. You also need to explicitly give the reasons why you are not implementing them. Submit an empty file if you finished all the tasks. You may lose credits by adding items to this list, but being dishonest on this will make you lose more; - a directory named
checkpoint/
containing your texts, images, videos, or typescript files for each checkpoint.
For each submission, please send an email with the title lab[1|2|3] Name StudentID
to liuxym@pku.edu.cn. Missing the deadlines incurs a penalty.