mirror of
https://github.com/Control-D-Inc/ctrld.git
synced 2026-04-20 00:36:37 +02:00
207 lines
4.7 KiB
Markdown
207 lines
4.7 KiB
Markdown
# Netstack - Full Packet Capture for Mobile VPN
|
|
|
|
Complete TCP/UDP/DNS packet capture implementation using gVisor netstack for Android and iOS.
|
|
|
|
## Overview
|
|
|
|
Provides full packet capture for mobile VPN applications:
|
|
- **DNS filtering** through ControlD proxy
|
|
- **TCP forwarding** for all TCP traffic
|
|
- **UDP forwarding** with session tracking
|
|
- **Socket protection** to prevent routing loops
|
|
- **QUIC blocking** for better content filtering
|
|
|
|
## Architecture
|
|
|
|
```
|
|
Mobile Apps → VPN TUN Interface → PacketHandler → gVisor Netstack
|
|
↓
|
|
├─→ DNS Filter (Port 53)
|
|
│ └─→ ControlD DNS Proxy
|
|
├─→ TCP Forwarder
|
|
│ └─→ net.Dial("tcp") + protect(fd)
|
|
└─→ UDP Forwarder
|
|
└─→ net.Dial("udp") + protect(fd)
|
|
↓
|
|
Real Network (Protected Sockets)
|
|
```
|
|
|
|
## Components
|
|
|
|
### DNS Filter (`dns_filter.go`)
|
|
Detects DNS packets on port 53 and routes to ControlD proxy.
|
|
|
|
### DNS Bridge (`dns_bridge.go`)
|
|
Tracks DNS queries by transaction ID with 5-second timeout.
|
|
|
|
### TCP Forwarder (`tcp_forwarder.go`)
|
|
Forwards TCP connections using gVisor's `tcp.NewForwarder()`.
|
|
|
|
### UDP Forwarder (`udp_forwarder.go`)
|
|
Forwards UDP packets with session tracking and 60-second idle timeout.
|
|
|
|
### Packet Handler (`packet_handler.go`)
|
|
Interface for TUN I/O and socket protection.
|
|
|
|
### Netstack Controller (`netstack.go`)
|
|
Manages gVisor stack and coordinates all components.
|
|
|
|
## Socket Protection
|
|
|
|
Critical for preventing routing loops:
|
|
|
|
```go
|
|
// Protection happens BEFORE connect()
|
|
dialer.Control = func(network, address string, c syscall.RawConn) error {
|
|
return c.Control(func(fd uintptr) {
|
|
protectSocket(int(fd))
|
|
})
|
|
}
|
|
```
|
|
|
|
**All protected sockets:**
|
|
- TCP/UDP forwarder sockets (user traffic)
|
|
- ControlD API sockets (api.controld.com)
|
|
- DoH upstream sockets (freedns.controld.com)
|
|
|
|
## Platform Configuration
|
|
|
|
### Android
|
|
|
|
```kotlin
|
|
Builder()
|
|
.addAddress("10.0.0.2", 24)
|
|
.addRoute("0.0.0.0", 0)
|
|
.addDnsServer("10.0.0.1")
|
|
.setMtu(1500)
|
|
|
|
override fun protectSocket(fd: Long) {
|
|
protect(fd.toInt()) // VpnService.protect()
|
|
}
|
|
|
|
// DNS Proxy: 0.0.0.0:5354
|
|
```
|
|
|
|
### iOS
|
|
|
|
```swift
|
|
NEIPv4Settings(addresses: ["10.0.0.2"], ...)
|
|
NEDNSSettings(servers: ["10.0.0.1"])
|
|
includedRoutes = [NEIPv4Route.default()]
|
|
|
|
func protectSocket(_ fd: Int) throws {
|
|
// No action needed - iOS Network Extension sockets
|
|
// automatically bypass VPN tunnel
|
|
}
|
|
|
|
// DNS Proxy: 127.0.0.1:53
|
|
```
|
|
|
|
**Note:** iOS Network Extensions run in separate process - sockets automatically bypass VPN.
|
|
|
|
## Protocol Support
|
|
|
|
| Protocol | Support |
|
|
|----------|---------|
|
|
| DNS (UDP/TCP port 53) | ✅ Full |
|
|
| TCP (all ports) | ✅ Full |
|
|
| UDP (except 53, 80, 443) | ✅ Full |
|
|
| QUIC (UDP/443, UDP/80) | 🚫 Blocked |
|
|
| ICMP | ⚠️ Partial |
|
|
| IPv4 | ✅ Full |
|
|
| IPv6 | ✅ Full |
|
|
|
|
## QUIC Blocking
|
|
|
|
Drops UDP packets on ports 443 and 80 to force TCP fallback:
|
|
|
|
**Why:**
|
|
- Better DNS filtering (QUIC bypasses DNS)
|
|
- Visible SNI for content filtering
|
|
- Consistent ControlD policy enforcement
|
|
|
|
**Result:**
|
|
- Apps automatically fallback to TCP/TLS
|
|
- No user-visible errors
|
|
- Slightly slower initial connection, then normal
|
|
|
|
## Usage (Android)
|
|
|
|
```kotlin
|
|
// Create callback
|
|
val callback = object : PacketAppCallback {
|
|
override fun readPacket(): ByteArray { ... }
|
|
override fun writePacket(packet: ByteArray) { ... }
|
|
override fun protectSocket(fd: Long) {
|
|
protect(fd.toInt())
|
|
}
|
|
override fun closePacketIO() { ... }
|
|
override fun exit(s: String) { ... }
|
|
override fun hostname(): String = "android-device"
|
|
override fun lanIp(): String = "10.0.0.2"
|
|
override fun macAddress(): String = "00:00:00:00:00:00"
|
|
}
|
|
|
|
// Create controller
|
|
val controller = Ctrld_library.newPacketCaptureController(callback)
|
|
|
|
// Start
|
|
controller.startWithPacketCapture(
|
|
callback,
|
|
cdUID,
|
|
"", "",
|
|
filesDir.absolutePath,
|
|
"doh",
|
|
2,
|
|
"$filesDir/ctrld.log"
|
|
)
|
|
|
|
// Stop
|
|
controller.stop(false, 0)
|
|
```
|
|
|
|
## Usage (iOS)
|
|
|
|
```swift
|
|
// DNS-only mode
|
|
let proxy = LocalProxy()
|
|
proxy.start(
|
|
cUID: cdUID,
|
|
deviceName: UIDevice.current.name,
|
|
upstreamProto: "doh",
|
|
logLevel: 3,
|
|
provisionID: ""
|
|
)
|
|
|
|
// Firewall mode (full capture)
|
|
let proxy = LocalProxy()
|
|
proxy.startFirewall(
|
|
cUID: cdUID,
|
|
deviceName: UIDevice.current.name,
|
|
upstreamProto: "doh",
|
|
logLevel: 3,
|
|
provisionID: "",
|
|
packetFlow: packetFlow
|
|
)
|
|
```
|
|
|
|
## Requirements
|
|
|
|
- **Android**: API 24+ (Android 7.0+)
|
|
- **iOS**: iOS 12.0+
|
|
- **Go**: 1.23+
|
|
- **gVisor**: v0.0.0-20240722211153-64c016c92987
|
|
|
|
## Files
|
|
|
|
- `packet_handler.go` - TUN I/O interface
|
|
- `netstack.go` - gVisor controller
|
|
- `dns_filter.go` - DNS packet detection
|
|
- `dns_bridge.go` - Transaction tracking
|
|
- `tcp_forwarder.go` - TCP forwarding
|
|
- `udp_forwarder.go` - UDP forwarding
|
|
|
|
## License
|
|
|
|
Same as parent ctrld project.
|