Skip to content

Commit 067af53

Browse files
committed
Add documentation on Rust side
1 parent 956c32a commit 067af53

File tree

15 files changed

+200
-85
lines changed

15 files changed

+200
-85
lines changed

ios/MullvadREST/ApiHandlers/MullvadApiRequestFactory.swift

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ enum MullvadApiRequest {
1616
struct MullvadApiRequestFactory {
1717
let apiContext: MullvadApiContext
1818

19-
func makeRequest(_ request: MullvadApiRequest) -> REST.MullvadApiRequestHandler {
19+
func makeRequest(_ request: MullvadApiRequest) -> REST.MullvadApiRequestHandler {
2020
{ completion in
2121
let pointerClass = MullvadApiCompletion { apiResponse in
2222
try? completion?(apiResponse)

ios/MullvadREST/ApiHandlers/RESTRustNetworkOperation.swift

+2-2
Original file line numberDiff line numberDiff line change
@@ -75,7 +75,7 @@ extension REST {
7575
networkTask = requestHandler { [weak self] response in
7676
guard let self else { return }
7777

78-
if let error = try response.restError() {
78+
if let error = response.restError() {
7979
if response.shouldRetry {
8080
retryRequest(with: error)
8181
} else {
@@ -146,7 +146,7 @@ extension REST {
146146
}
147147

148148
extension MullvadApiResponse {
149-
public func restError() throws -> REST.Error? {
149+
public func restError() -> REST.Error? {
150150
guard !success else {
151151
return nil
152152
}

ios/MullvadRustRuntime/MullvadApiCompletion.swift

+1-1
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@ func mullvadApiCompletionFinish(
1313
) {
1414
let completionBridge = Unmanaged<MullvadApiCompletion>
1515
.fromOpaque(completionCookie)
16-
.takeUnretainedValue()
16+
.takeRetainedValue()
1717
let apiResponse = MullvadApiResponse(response: response)
1818

1919
completionBridge.completion(apiResponse)

ios/MullvadRustRuntime/include/mullvad_rust_runtime.h

+100-34
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,29 @@ typedef struct ExchangeCancelToken ExchangeCancelToken;
2626

2727
typedef struct RequestCancelHandle RequestCancelHandle;
2828

29+
typedef struct SwiftApiContext {
30+
const struct ApiContext *_0;
31+
} SwiftApiContext;
32+
33+
typedef struct SwiftCancelHandle {
34+
struct RequestCancelHandle *ptr;
35+
} SwiftCancelHandle;
36+
37+
typedef struct SwiftMullvadApiResponse {
38+
uint8_t *body;
39+
uintptr_t body_size;
40+
uint16_t status_code;
41+
uint8_t *error_description;
42+
uint8_t *server_response_code;
43+
bool success;
44+
bool should_retry;
45+
uint64_t retry_after;
46+
} SwiftMullvadApiResponse;
47+
48+
typedef struct CompletionCookie {
49+
void *_0;
50+
} CompletionCookie;
51+
2952
typedef struct ProxyHandle {
3053
void *context;
3154
uint16_t port;
@@ -51,30 +74,87 @@ typedef struct EphemeralPeerParameters {
5174
struct WgTcpConnectionFunctions funcs;
5275
} EphemeralPeerParameters;
5376

54-
typedef struct SwiftApiContext {
55-
const struct ApiContext *_0;
56-
} SwiftApiContext;
77+
extern const uint16_t CONFIG_SERVICE_PORT;
5778

58-
typedef struct SwiftCancelHandle {
59-
struct RequestCancelHandle *ptr;
60-
} SwiftCancelHandle;
79+
/**
80+
* # Safety
81+
*
82+
* `host` must be a pointer to a null terminated string representing a hostname for Mullvad API host.
83+
* This hostname will be used for TLS validation but not used for domain name resolution.
84+
*
85+
* `address` must be a pointer to a null terminated string representing a socket address through which
86+
* the Mullvad API can be reached directly.
87+
*
88+
* If a context cannot be constructed this function will panic since the call site would not be able
89+
* to proceed in a meaningful way anyway.
90+
*
91+
* This function is safe.
92+
*/
93+
struct SwiftApiContext mullvad_api_init_new(const uint8_t *host,
94+
const uint8_t *address);
6195

62-
typedef struct SwiftMullvadApiResponse {
63-
uint8_t *body;
64-
uintptr_t body_size;
65-
uint16_t status_code;
66-
uint8_t *error_description;
67-
uint8_t *server_response_code;
68-
bool success;
69-
bool should_retry;
70-
uint64_t retry_after;
71-
} SwiftMullvadApiResponse;
96+
/**
97+
* # Safety
98+
*
99+
* `api_context` must be pointing to a valid instance of `SwiftApiContext`. A `SwiftApiContext` is created
100+
* by calling `mullvad_api_init_new`.
101+
*
102+
* `completion_cookie` must be pointing to a valid instance of `CompletionCookie`. `CompletionCookie` is
103+
* safe because the pointer in `MullvadApiCompletion` is valid for the lifetime of the process where this
104+
* type is intended to be used.
105+
*
106+
* This function is not safe to call multiple times with the same `CompletionCookie`.
107+
*/
108+
struct SwiftCancelHandle mullvad_api_get_addresses(struct SwiftApiContext api_context,
109+
void *completion_cookie);
72110

73-
typedef struct CompletionCookie {
74-
void *_0;
75-
} CompletionCookie;
111+
/**
112+
* Called by the Swift side to signal that a Mullvad API call should be cancelled.
113+
* After this call, the cancel token is no longer valid.
114+
*
115+
* # Safety
116+
*
117+
* `handle_ptr` must be pointing to a valid instance of `SwiftCancelHandle`. This function
118+
* is not safe to call multiple times with the same `SwiftCancelHandle`.
119+
*/
120+
void mullvad_api_cancel_task(struct SwiftCancelHandle handle_ptr);
76121

77-
extern const uint16_t CONFIG_SERVICE_PORT;
122+
/**
123+
* Called by the Swift side to signal that the Rust `SwiftCancelHandle` can be safely
124+
* dropped from memory.
125+
*
126+
* # Safety
127+
*
128+
* `handle_ptr` must be pointing to a valid instance of `SwiftCancelHandle`. This function
129+
* is not safe to call multiple times with the same `SwiftCancelHandle`.
130+
*/
131+
void mullvad_api_cancel_task_drop(struct SwiftCancelHandle handle_ptr);
132+
133+
/**
134+
* Maps to `mullvadApiCompletionFinish` on Swift side to facilitate callback based completion flow when doing
135+
* network calls through Mullvad API on Rust side.
136+
*
137+
* # Safety
138+
*
139+
* `response` must be pointing to a valid instance of `SwiftMullvadApiResponse`.
140+
*
141+
* `completion_cookie` must be pointing to a valid instance of `CompletionCookie`. `CompletionCookie` is safe
142+
* because the pointer in `MullvadApiCompletion` is valid for the lifetime of the process where this type is
143+
* intended to be used.
144+
*/
145+
extern void mullvad_api_completion_finish(struct SwiftMullvadApiResponse response,
146+
struct CompletionCookie completion_cookie);
147+
148+
/**
149+
* Called by the Swift side to signal that the Rust `SwiftMullvadApiResponse` can be safely
150+
* dropped from memory.
151+
*
152+
* # Safety
153+
*
154+
* `response` must be pointing to a valid instance of `SwiftMullvadApiResponse`. This function
155+
* is not safe to call multiple times with the same `SwiftMullvadApiResponse`.
156+
*/
157+
void mullvad_response_drop(struct SwiftMullvadApiResponse response);
78158

79159
/**
80160
* Initializes a valid pointer to an instance of `EncryptedDnsProxyState`.
@@ -200,20 +280,6 @@ int32_t start_shadowsocks_proxy(const uint8_t *forward_address,
200280
*/
201281
int32_t stop_shadowsocks_proxy(struct ProxyHandle *proxy_config);
202282

203-
struct SwiftApiContext mullvad_api_init_new(const uint8_t *host, const uint8_t *address);
204-
205-
struct SwiftCancelHandle mullvad_api_get_addresses(struct SwiftApiContext api_context,
206-
void *completion_cookie);
207-
208-
void mullvad_api_cancel_task(struct SwiftCancelHandle handle_ptr);
209-
210-
void mullvad_api_cancel_task_drop(struct SwiftCancelHandle handle_ptr);
211-
212-
extern void mullvad_api_completion_finish(struct SwiftMullvadApiResponse response,
213-
struct CompletionCookie completion_cookie);
214-
215-
void mullvad_response_drop(struct SwiftMullvadApiResponse response);
216-
217283
int32_t start_tunnel_obfuscator_proxy(const uint8_t *peer_address,
218284
uintptr_t peer_address_len,
219285
uint16_t peer_port,

ios/MullvadTypes/AsyncExample.swift

Whitespace-only changes.

ios/MullvadVPN.xcodeproj/project.pbxproj

+8-4
Original file line numberDiff line numberDiff line change
@@ -583,6 +583,8 @@
583583
7A8A19242CF4C9BF000BCB5B /* MultihopPage.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7A8A19232CF4C9B8000BCB5B /* MultihopPage.swift */; };
584584
7A8A19262CF4D37B000BCB5B /* DAITAPage.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7A8A19252CF4D373000BCB5B /* DAITAPage.swift */; };
585585
7A8A19282CF603EB000BCB5B /* SettingsViewControllerFactory.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7A8A19272CF603E3000BCB5B /* SettingsViewControllerFactory.swift */; };
586+
7A95B6792D5F729300687524 /* DAITASettingsCoordinator.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7A95B6782D5F729300687524 /* DAITASettingsCoordinator.swift */; };
587+
7A95B67B2D5F758300687524 /* relays.json in Resources */ = {isa = PBXBuildFile; fileRef = 7A95B67A2D5F758300687524 /* relays.json */; };
586588
7A99D36F2D56070400891FF7 /* MullvadApiRequestFactory.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7A99D36E2D5606F900891FF7 /* MullvadApiRequestFactory.swift */; };
587589
7A99D3712D56222000891FF7 /* MullvadApiCancellable.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7A99D3702D56220E00891FF7 /* MullvadApiCancellable.swift */; };
588590
7A9BE5A22B8F88C500E2A7D0 /* LocationNodeTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7A9BE5A12B8F88C500E2A7D0 /* LocationNodeTests.swift */; };
@@ -1992,6 +1994,8 @@
19921994
7A8A19232CF4C9B8000BCB5B /* MultihopPage.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MultihopPage.swift; sourceTree = "<group>"; };
19931995
7A8A19252CF4D373000BCB5B /* DAITAPage.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DAITAPage.swift; sourceTree = "<group>"; };
19941996
7A8A19272CF603E3000BCB5B /* SettingsViewControllerFactory.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SettingsViewControllerFactory.swift; sourceTree = "<group>"; };
1997+
7A95B6782D5F729300687524 /* DAITASettingsCoordinator.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DAITASettingsCoordinator.swift; sourceTree = "<group>"; };
1998+
7A95B67A2D5F758300687524 /* relays.json */ = {isa = PBXFileReference; lastKnownFileType = text.json; path = relays.json; sourceTree = "<group>"; };
19951999
7A99D36E2D5606F900891FF7 /* MullvadApiRequestFactory.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MullvadApiRequestFactory.swift; sourceTree = "<group>"; };
19962000
7A99D3702D56220E00891FF7 /* MullvadApiCancellable.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MullvadApiCancellable.swift; sourceTree = "<group>"; };
19972001
7A9BE5A12B8F88C500E2A7D0 /* LocationNodeTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LocationNodeTests.swift; sourceTree = "<group>"; };
@@ -2532,7 +2536,7 @@
25322536
isa = PBXGroup;
25332537
children = (
25342538
06799AB428F98CE700ACD94E /* le_root_cert.cer */,
2535-
7A99D36C2D54FCC400891FF7 /* relays.json */,
2539+
7A95B67A2D5F758300687524 /* relays.json */,
25362540
);
25372541
path = Assets;
25382542
sourceTree = "<group>";
@@ -4121,7 +4125,7 @@
41214125
7A8A19082CE5FFD7000BCB5B /* DAITA */ = {
41224126
isa = PBXGroup;
41234127
children = (
4124-
7A3215732D3E5A7B005DF395 /* DAITASettingsCoordinator.swift */,
4128+
7A95B6782D5F729300687524 /* DAITASettingsCoordinator.swift */,
41254129
F041BE4E2C983C2B0083EC28 /* DAITASettingsPromptItem.swift */,
41264130
7A8A19132CEF2527000BCB5B /* DAITATunnelSettingsViewModel.swift */,
41274131
7A8A19092CE5FFDF000BCB5B /* SettingsDAITAView.swift */,
@@ -5273,7 +5277,7 @@
52735277
buildActionMask = 2147483647;
52745278
files = (
52755279
062B45A328FD4CA700746E77 /* le_root_cert.cer in Resources */,
5276-
7A99D36D2D54FCC400891FF7 /* relays.json in Resources */,
5280+
7A95B67B2D5F758300687524 /* relays.json in Resources */,
52775281
);
52785282
runOnlyForDeploymentPostprocessing = 0;
52795283
};
@@ -5940,6 +5944,7 @@
59405944
44075DFB2CDA4F7400F61139 /* UDPOverTCPObfuscationSettingsViewModel.swift in Sources */,
59415945
7A6389DC2B7E3BD6008E77E1 /* CustomListViewModel.swift in Sources */,
59425946
4422C0712CCFF6790001A385 /* UDPOverTCPObfuscationSettingsView.swift in Sources */,
5947+
7A95B6792D5F729300687524 /* DAITASettingsCoordinator.swift in Sources */,
59435948
7A9CCCC42A96302800DD6A34 /* TunnelCoordinator.swift in Sources */,
59445949
5827B0A42B0F38FD00CCBBA1 /* EditAccessMethodInteractorProtocol.swift in Sources */,
59455950
586C0D852B03D31E00E7CDD7 /* SocksSectionHandler.swift in Sources */,
@@ -6042,7 +6047,6 @@
60426047
588D7EDC2AF3A55E005DF40A /* ListAccessMethodInteractorProtocol.swift in Sources */,
60436048
588D7ED62AF3903F005DF40A /* ListAccessMethodViewController.swift in Sources */,
60446049
7A8A190E2CEB77C1000BCB5B /* SettingsRowViewFooter.swift in Sources */,
6045-
7A3215742D3E5A85005DF395 /* DAITASettingsCoordinator.swift in Sources */,
60466050
7A6000FC2B628DF6001CF0D9 /* ListCellContentConfiguration.swift in Sources */,
60476051
582BB1B1229569620055B6EF /* UINavigationBar+Appearance.swift in Sources */,
60486052
7A9FA1442A2E3FE5000B728D /* CheckableSettingsCell.swift in Sources */,

ios/MullvadVPN/AddressCacheTracker/AddressCacheTracker.swift

+1-2
Original file line numberDiff line numberDiff line change
@@ -118,8 +118,7 @@ final class AddressCacheTracker: @unchecked Sendable {
118118
nslock.lock()
119119
defer { nslock.unlock() }
120120

121-
return Date()
122-
// return _nextScheduleDate()
121+
return _nextScheduleDate()
123122
}
124123

125124
private func setEndpoints(from result: Result<[AnyIPEndpoint], Error>) {

ios/build-rust-library.sh

+1-5
Original file line numberDiff line numberDiff line change
@@ -53,15 +53,11 @@ for arch in $ARCHS; do
5353
case "$arch" in
5454
arm64)
5555
if [ $IS_SIMULATOR -eq 0 ]; then
56-
# Hardware iOS targets
57-
rustup target add aarch64-apple-ios
58-
56+
# Hardware iOS targets
5957
"$HOME"/.cargo/bin/cargo build -p "$FFI_TARGET" --lib $RELFLAG --target aarch64-apple-ios ${FEATURE_FLAGS:+--features "$FEATURE_FLAGS"}
6058
"$HOME"/.cargo/bin/cargo build -p "$FFI_TARGET" --lib --target aarch64-apple-ios ${FEATURE_FLAGS:+--features "$FEATURE_FLAGS"}
6159
else
6260
# iOS Simulator targets for arm64
63-
rustup target add aarch64-apple-ios-sim
64-
6561
"$HOME"/.cargo/bin/cargo build -p "$FFI_TARGET" --lib $RELFLAG --target aarch64-apple-ios-sim ${FEATURE_FLAGS:+--features "$FEATURE_FLAGS"}
6662
"$HOME"/.cargo/bin/cargo build -p "$FFI_TARGET" --lib --target aarch64-apple-ios-sim ${FEATURE_FLAGS:+--features "$FEATURE_FLAGS"}
6763
fi

mullvad-ios/Cargo.toml

+4
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,10 @@ rust-version.workspace = true
1010
[lints]
1111
workspace = true
1212

13+
[features]
14+
# Allow the API server to be used
15+
api-override = ["mullvad-api/api-override"]
16+
1317
[target.'cfg(target_os = "ios")'.dependencies]
1418
libc = "0.2"
1519
log = { workspace = true }

mullvad-ios/src/api/api.rs mullvad-ios/src/api_client/api.rs

+18-4
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,25 @@
1-
use mullvad_api::{rest::{self, MullvadRestHandle}, ApiProxy};
1+
use mullvad_api::{
2+
rest::{self, MullvadRestHandle},
3+
ApiProxy,
4+
};
25

36
use super::{
47
cancellation::{RequestCancelHandle, SwiftCancelHandle},
58
completion::{CompletionCookie, SwiftCompletionHandler},
6-
response::SwiftMullvadApiResponse, SwiftApiContext,
9+
response::SwiftMullvadApiResponse,
10+
SwiftApiContext,
711
};
812

13+
/// # Safety
14+
///
15+
/// `api_context` must be pointing to a valid instance of `SwiftApiContext`. A `SwiftApiContext` is created
16+
/// by calling `mullvad_api_init_new`.
17+
///
18+
/// `completion_cookie` must be pointing to a valid instance of `CompletionCookie`. `CompletionCookie` is
19+
/// safe because the pointer in `MullvadApiCompletion` is valid for the lifetime of the process where this
20+
/// type is intended to be used.
21+
///
22+
/// This function is not safe to call multiple times with the same `CompletionCookie`.
923
#[no_mangle]
1024
pub unsafe extern "C" fn mullvad_api_get_addresses(
1125
api_context: SwiftApiContext,
@@ -18,7 +32,7 @@ pub unsafe extern "C" fn mullvad_api_get_addresses(
1832
return SwiftCancelHandle::empty();
1933
};
2034

21-
let api_context = api_context.to_rust_context();
35+
let api_context = api_context.into_rust_context();
2236

2337
let completion = completion_handler.clone();
2438
let task = tokio_handle.clone().spawn(async move {
@@ -31,7 +45,7 @@ pub unsafe extern "C" fn mullvad_api_get_addresses(
3145
}
3246
});
3347

34-
RequestCancelHandle::new(task, completion_handler.clone()).to_swift()
48+
RequestCancelHandle::new(task, completion_handler.clone()).into_swift()
3549
}
3650

3751
async fn mullvad_api_get_addresses_inner(

mullvad-ios/src/api/cancellation.rs mullvad-ios/src/api_client/cancellation.rs

+18-4
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ impl SwiftCancelHandle {
1616

1717
/// This consumes and nulls out the pointer. The caller is responsible for the pointer being valid
1818
/// when calling `to_handle`.
19-
unsafe fn to_handle(mut self) -> RequestCancelHandle {
19+
unsafe fn into_handle(mut self) -> RequestCancelHandle {
2020
// # Safety
2121
// This call is safe as long as the pointer is only ever used from a single thread and the
2222
// instance of `SwiftCancelHandle` was created with a valid pointer to
@@ -38,7 +38,7 @@ impl RequestCancelHandle {
3838
Self { task, completion }
3939
}
4040

41-
pub fn to_swift(self) -> SwiftCancelHandle {
41+
pub fn into_swift(self) -> SwiftCancelHandle {
4242
SwiftCancelHandle {
4343
ptr: Box::into_raw(Box::new(self)),
4444
}
@@ -54,21 +54,35 @@ impl RequestCancelHandle {
5454
}
5555
}
5656

57+
/// Called by the Swift side to signal that a Mullvad API call should be cancelled.
58+
/// After this call, the cancel token is no longer valid.
59+
///
60+
/// # Safety
61+
///
62+
/// `handle_ptr` must be pointing to a valid instance of `SwiftCancelHandle`. This function
63+
/// is not safe to call multiple times with the same `SwiftCancelHandle`.
5764
#[no_mangle]
5865
extern "C" fn mullvad_api_cancel_task(handle_ptr: SwiftCancelHandle) {
5966
if handle_ptr.ptr.is_null() {
6067
return;
6168
}
6269

63-
let handle = unsafe { handle_ptr.to_handle() };
70+
let handle = unsafe { handle_ptr.into_handle() };
6471
handle.cancel()
6572
}
6673

74+
/// Called by the Swift side to signal that the Rust `SwiftCancelHandle` can be safely
75+
/// dropped from memory.
76+
///
77+
/// # Safety
78+
///
79+
/// `handle_ptr` must be pointing to a valid instance of `SwiftCancelHandle`. This function
80+
/// is not safe to call multiple times with the same `SwiftCancelHandle`.
6781
#[no_mangle]
6882
extern "C" fn mullvad_api_cancel_task_drop(handle_ptr: SwiftCancelHandle) {
6983
if handle_ptr.ptr.is_null() {
7084
return;
7185
}
7286

73-
let _handle = unsafe { handle_ptr.to_handle() };
87+
let _handle = unsafe { handle_ptr.into_handle() };
7488
}

0 commit comments

Comments
 (0)