-
Notifications
You must be signed in to change notification settings - Fork 265
Expand file tree
/
Copy pathNATNetworkInterface.swift
More file actions
115 lines (103 loc) · 3.27 KB
/
NATNetworkInterface.swift
File metadata and controls
115 lines (103 loc) · 3.27 KB
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
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
//===----------------------------------------------------------------------===//
// Copyright © 2025 Apple Inc. and the Containerization project authors.
// All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// https://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//===----------------------------------------------------------------------===//
#if os(macOS)
import vmnet
import Virtualization
import ContainerizationError
import Foundation
import Synchronization
/// An interface that uses NAT to provide an IP address for a given
/// container/virtual machine.
@available(macOS 16, *)
public final class NATNetworkInterface: Interface, Sendable {
public var address: String {
get { state.withLock { $0.address } }
set { state.withLock { $0.address = newValue } }
}
public var gateway: String {
get { state.withLock { $0.gateway } }
set { state.withLock { $0.gateway = newValue } }
}
public var macAddress: String? {
get { state.withLock { $0.macAddress } }
set { state.withLock { $0.macAddress = newValue } }
}
struct State {
var address: String
var gateway: String
var macAddress: String?
#if !CURRENT_SDK
var reference: vmnet_network_ref
#endif
}
#if !CURRENT_SDK
public var reference: vmnet_network_ref {
state.withLock { $0.reference }
}
#endif
private let state: Mutex<State>
#if !CURRENT_SDK
public init(
address: String,
gateway: String,
reference: sending vmnet_network_ref,
macAddress: String? = nil
) {
self.state = .init(
.init(
address: address,
gateway: gateway,
macAddress: macAddress,
reference: reference
)
)
}
#else
public init(
address: String,
gateway: String,
macAddress: String? = nil
) {
self.state = .init(
.init(
address: address,
gateway: gateway,
macAddress: macAddress
)
)
}
#endif
}
@available(macOS 16, *)
extension NATNetworkInterface: VZInterface {
public func device() throws -> VZVirtioNetworkDeviceConfiguration {
let config = VZVirtioNetworkDeviceConfiguration()
if let macAddress = self.macAddress {
guard let mac = VZMACAddress(string: macAddress) else {
throw ContainerizationError(.invalidArgument, message: "invalid mac address \(macAddress)")
}
config.macAddress = mac
}
#if !CURRENT_SDK
config.attachment = VZVmnetNetworkDeviceAttachment(network: self.reference)
#else
config.attachment = VZNATNetworkDeviceAttachment()
#endif
return config
}
}
#endif