Coverage Report

Created: 2026-04-09 06:21

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/SockFuzzer/fuzz/net_fuzzer.cc
Line
Count
Source
1
/*
2
 * Copyright 2021 Google LLC
3
 *
4
 * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
5
 *
6
 * This file contains Original Code and/or Modifications of Original Code
7
 * as defined in and that are subject to the Apple Public Source License
8
 * Version 2.0 (the 'License'). You may not use this file except in
9
 * compliance with the License. The rights granted to you under the License
10
 * may not be used to create, or enable the creation or redistribution of,
11
 * unlawful or unlicensed copies of an Apple operating system, or to
12
 * circumvent, violate, or enable the circumvention or violation of, any
13
 * terms of an Apple operating system software license agreement.
14
 *
15
 * Please obtain a copy of the License at
16
 * http://www.opensource.apple.com/apsl/ and read it before using this file.
17
 *
18
 * The Original Code and all software distributed under the License are
19
 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20
 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21
 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22
 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23
 * Please see the License for the specific language governing rights and
24
 * limitations under the License.
25
 *
26
 * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
27
 */
28
29
#include <fuzzer/FuzzedDataProvider.h>
30
#include <stddef.h>
31
#include <stdint.h>
32
#include <string.h>
33
34
#include <iostream>
35
#include <memory>
36
#include <vector>
37
38
#include "net_fuzzer.pb.h"
39
#include "src/libfuzzer/libfuzzer_macro.h"
40
41
extern "C" {
42
#include "api/backend.h"
43
#include "api/syscall_wrappers.h"
44
#include "types.h"
45
}
46
47
/*
48
TODO(nedwill)
49
50
Fuzz vsock domain
51
*/
52
53
// TODO(nedwill): support multiple addresses of each type below,
54
// not just one of each type
55
714k
void get_in6_addr(struct in6_addr *sai, enum In6Addr addr) {
56
714k
  memset(sai, 0, sizeof(*sai));
57
714k
  switch (addr) {
58
5.10k
    case IN6_ADDR_SELF: {
59
5.10k
      sai->__u6_addr.__u6_addr32[0] = 16810238;
60
5.10k
      sai->__u6_addr.__u6_addr32[0] = 0;
61
5.10k
      sai->__u6_addr.__u6_addr32[0] = 0;
62
5.10k
      sai->__u6_addr.__u6_addr32[0] = 16777216;
63
      // assert(IN6_IS_ADDR_SELF(sai));
64
5.10k
      break;
65
0
    }
66
2.74k
    case IN6_ADDR_LINK_LOCAL: {
67
2.74k
      sai->s6_addr[0] = 0xfe;
68
2.74k
      sai->s6_addr[1] = 0x80;
69
      // TODO(nedwill): set other fields?
70
2.74k
      assert(IN6_IS_ADDR_LINKLOCAL(sai));
71
2.74k
      break;
72
2.74k
    }
73
357k
    case IN6_ADDR_LOOPBACK: {
74
357k
      *sai = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, (uint8_t)addr};
75
357k
      assert(IN6_IS_ADDR_LOOPBACK(sai));
76
357k
      break;
77
357k
    }
78
357k
    case IN6_ADDR_REAL:
79
9.95k
    case MAYBE_LOCALHOST: {
80
9.95k
      *sai = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, (uint8_t)addr};
81
9.95k
      break;
82
9.41k
    }
83
3.86k
    case IN6_ADDR_V4COMPAT: {
84
3.86k
      sai->s6_addr[12] = 1;
85
3.86k
      assert(IN6_IS_ADDR_V4COMPAT(sai));
86
3.86k
      break;
87
3.86k
    }
88
31.3k
    case IN6_ADDR_V4MAPPED: {
89
31.3k
      *(uint32_t *)&sai->s6_addr[8] = 0xffff0000;
90
31.3k
      assert(IN6_IS_ADDR_V4MAPPED(sai));
91
31.3k
      break;
92
31.3k
    }
93
31.3k
    case IN6_ADDR_6TO4: {
94
431
      sai->s6_addr16[0] = ntohs(0x2002);
95
431
      assert(IN6_IS_ADDR_6TO4(sai));
96
431
      break;
97
431
    }
98
3.97k
    case IN6_ADDR_LINKLOCAL: {
99
3.97k
      sai->s6_addr[0] = 0xfe;
100
3.97k
      sai->s6_addr[1] = 0x80;
101
3.97k
      assert(IN6_IS_ADDR_LINKLOCAL(sai));
102
3.97k
      break;
103
3.97k
    }
104
3.97k
    case IN6_ADDR_SITELOCAL: {
105
780
      sai->s6_addr[0] = 0xfe;
106
780
      sai->s6_addr[1] = 0xc0;
107
780
      assert(IN6_IS_ADDR_SITELOCAL(sai));
108
780
      break;
109
780
    }
110
2.45k
    case IN6_ADDR_MULTICAST: {
111
2.45k
      sai->s6_addr[0] = 0xff;
112
2.45k
      assert(IN6_IS_ADDR_MULTICAST(sai));
113
2.45k
      break;
114
2.45k
    }
115
2.45k
    case IN6_ADDR_UNIQUE_LOCAL: {
116
1.68k
      sai->s6_addr[0] = 0xfc;
117
1.68k
      assert(IN6_IS_ADDR_UNIQUE_LOCAL(sai));
118
1.68k
      break;
119
1.68k
    }
120
1.68k
    case IN6_ADDR_MC_NODELOCAL: {
121
1.18k
      sai->s6_addr[0] = 0xff;
122
1.18k
      sai->s6_addr[1] = __IPV6_ADDR_SCOPE_NODELOCAL;
123
1.18k
      assert(IN6_IS_ADDR_MC_NODELOCAL(sai));
124
1.18k
      break;
125
1.18k
    }
126
1.18k
    case IN6_ADDR_MC_INTFACELOCAL: {
127
379
      sai->s6_addr[0] = 0xff;
128
379
      sai->s6_addr[1] = __IPV6_ADDR_SCOPE_INTFACELOCAL;
129
379
      assert(IN6_IS_ADDR_MC_INTFACELOCAL(sai));
130
379
      break;
131
379
    }
132
12.1k
    case IN6_ADDR_MC_LINKLOCAL: {
133
12.1k
      sai->s6_addr[0] = 0xff;
134
12.1k
      sai->s6_addr[1] = __IPV6_ADDR_SCOPE_LINKLOCAL;
135
12.1k
      assert(IN6_IS_ADDR_MC_LINKLOCAL(sai));
136
12.1k
      break;
137
12.1k
    }
138
12.1k
    case IN6_ADDR_MC_SITELOCAL: {
139
1.81k
      sai->s6_addr[0] = 0xff;
140
1.81k
      sai->s6_addr[1] = __IPV6_ADDR_SCOPE_SITELOCAL;
141
1.81k
      assert(IN6_IS_ADDR_MC_SITELOCAL(sai));
142
1.81k
      break;
143
1.81k
    }
144
1.81k
    case IN6_ADDR_MC_ORGLOCAL: {
145
300
      sai->s6_addr[0] = 0xff;
146
300
      sai->s6_addr[1] = __IPV6_ADDR_SCOPE_ORGLOCAL;
147
300
      assert(IN6_IS_ADDR_MC_ORGLOCAL(sai));
148
300
      break;
149
300
    }
150
822
    case IN6_ADDR_MC_GLOBAL: {
151
822
      sai->s6_addr[0] = 0xff;
152
822
      sai->s6_addr[1] = __IPV6_ADDR_SCOPE_GLOBAL;
153
822
      assert(IN6_IS_ADDR_MC_GLOBAL(sai));
154
822
      break;
155
822
    }
156
9.23k
    case IN6_ADDR_UNSPECIFIED:
157
276k
    case IN6_ADDR_ANY: {
158
276k
      assert(IN6_IS_ADDR_UNSPECIFIED(sai));
159
276k
      break;
160
276k
    }
161
276k
    case IN6_ADDR_LOCAL_ADDRESS: {
162
      // Discovered this address dynamically
163
      // fe80:0001:0000:0000:a8aa:aaaa:aaaa:aaaa
164
1.39k
      sai->s6_addr16[0] = 0xfe80;
165
1.39k
      sai->s6_addr16[1] = 0x0001;
166
1.39k
      sai->s6_addr16[4] = 0xa8aa;
167
1.39k
      sai->s6_addr16[5] = 0xaaaa;
168
1.39k
      sai->s6_addr16[6] = 0xaaaa;
169
1.39k
      sai->s6_addr16[7] = 0xaaaa;
170
1.39k
    }
171
      // *sai = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, (uint8_t)addr};
172
714k
  }
173
714k
}
174
175
100k
void get_sockaddr6(struct sockaddr_in6 *sai, const SockAddr6 &sa6) {
176
100k
  sai->sin6_len = sizeof(struct sockaddr_in6);
177
100k
  sai->sin6_family = (sa_family_t)AF_INET6;  // sa6.family();
178
100k
  sai->sin6_port = (in_port_t)sa6.port();
179
100k
  sai->sin6_flowinfo = sa6.flow_info();
180
100k
  get_in6_addr(&sai->sin6_addr, sa6.sin6_addr());
181
100k
  sai->sin6_scope_id = sa6.sin6_scope_id();
182
100k
}
183
184
1.29M
std::string get_sockaddr(const SockAddr &sockaddr) {
185
1.29M
  std::string dat;
186
1.29M
  switch (sockaddr.sockaddr_case()) {
187
103k
    case SockAddr::kSockaddrGeneric: {
188
103k
      const SockAddrGeneric &sag = sockaddr.sockaddr_generic();
189
      // data size + sizeof(sa_len) + sizeof(sa_family)
190
103k
      struct sockaddr_generic sag_s = {
191
103k
          .sa_len = (uint8_t)(sizeof(sockaddr_generic) + sag.sa_data().size()),
192
103k
          .sa_family = (uint8_t)sag.sa_family(),
193
103k
      };
194
195
103k
      dat = std::string((char *)&sag_s, (char *)&sag_s + sizeof(sag_s));
196
103k
      dat += sag.sa_data();
197
103k
      break;
198
0
    }
199
62.1k
    case SockAddr::kSockaddr4: {
200
62.1k
      struct sockaddr_in sai = {
201
62.1k
          .sin_len = sizeof(struct sockaddr_in),
202
62.1k
          .sin_family =
203
62.1k
              AF_INET,  // (unsigned char)sockaddr.sockaddr4().sin_family(),
204
62.1k
          .sin_port = (unsigned short)sockaddr.sockaddr4().sin_port(),
205
62.1k
          .sin_addr = {(unsigned int)sockaddr.sockaddr4().sin_addr()},
206
62.1k
          .sin_zero = {},
207
62.1k
      };
208
62.1k
      dat = std::string((char *)&sai, (char *)&sai + sizeof(sai));
209
62.1k
      break;
210
0
    }
211
41.4k
    case SockAddr::kSockaddr6: {
212
41.4k
      struct sockaddr_in6 sai = {};
213
41.4k
      get_sockaddr6(&sai, sockaddr.sockaddr6());
214
41.4k
      dat = std::string((char *)&sai, (char *)&sai + sizeof(sai));
215
41.4k
      break;
216
0
    }
217
    // case SockAddr::kRawBytes: {
218
    //   dat = sockaddr.raw_bytes();
219
    //   break;
220
    // }
221
1.08M
    case SockAddr::SOCKADDR_NOT_SET: {
222
1.08M
      break;
223
0
    }
224
1.29M
  }
225
1.29M
  return dat;
226
1.29M
}
227
228
306k
std::string get_ip6_hdr(const Ip6Hdr &hdr, uint16_t expected_size) {
229
306k
  struct ip6_hdr ip6_hdr;
230
306k
  memset(&ip6_hdr, 0, sizeof(ip6_hdr));
231
306k
  get_in6_addr(&ip6_hdr.ip6_src, hdr.ip6_src());
232
306k
  get_in6_addr(&ip6_hdr.ip6_dst, hdr.ip6_src());
233
306k
  ip6_hdr.ip6_ctlun.ip6_un2_vfc = IPV6_VERSION;
234
  // How TF does flow work?
235
  // ip6_hdr.ip6_ctlun.ip6_un1.ip6_un1_flow = hdr.ip6_hdrctl().ip6_un1_flow();
236
306k
  ip6_hdr.ip6_ctlun.ip6_un1.ip6_un1_plen =
237
306k
      __builtin_bswap16(expected_size);  // hdr.ip6_hdrctl().ip6_un1_plen();
238
306k
  ip6_hdr.ip6_ctlun.ip6_un1.ip6_un1_nxt = hdr.ip6_hdrctl().ip6_un1_nxt();
239
306k
  ip6_hdr.ip6_ctlun.ip6_un1.ip6_un1_hlim = hdr.ip6_hdrctl().ip6_un1_hlim();
240
306k
  std::string dat((char *)&ip6_hdr, (char *)&ip6_hdr + sizeof(ip6_hdr));
241
306k
  return dat;
242
306k
}
243
244
202k
std::string get_ip_hdr(const IpHdr &hdr, size_t expected_size) {
245
202k
  struct in_addr ip_src = {.s_addr = (unsigned int)hdr.ip_src()};
246
202k
  struct in_addr ip_dst = {.s_addr = (unsigned int)hdr.ip_dst()};
247
202k
  struct ip ip_hdr = {
248
202k
      .ip_hl = 5,    // TODO(nedwill): support options // hdr.ip_hl(),
249
202k
      .ip_v = IPV4,  // hdr.ip_v(),
250
202k
      .ip_tos = (u_char)hdr.ip_tos(),
251
202k
      .ip_len = (u_short)__builtin_bswap16(expected_size),
252
202k
      .ip_id = (u_short)hdr.ip_id(),
253
202k
      .ip_off = (u_short)hdr.ip_off(),
254
202k
      .ip_ttl = (u_char)hdr.ip_ttl(),
255
202k
      .ip_p = (u_char)hdr.ip_p(),
256
202k
      .ip_sum = 0,
257
202k
      .ip_src = ip_src,
258
202k
      .ip_dst = ip_dst,
259
202k
  };
260
202k
  std::string dat((char *)&ip_hdr, (char *)&ip_hdr + sizeof(ip_hdr));
261
202k
  return dat;
262
202k
}
263
264
// message TcpHdr {
265
//   required Port th_sport = 1;
266
//   required Port th_dport = 2;
267
//   required uint32 th_seq = 3;
268
//   required uint32 th_ack = 4;
269
//   required uint32 th_off = 5;
270
//   repeated TcpFlag th_flags = 6;
271
//   required uint32 th_win = 7;
272
//   required uint32 th_sum = 8;
273
//   required uint32 th_urp = 9;
274
// }
275
276
64.0k
std::string get_tcp_hdr(const TcpHdr &hdr) {
277
64.0k
  struct tcphdr tcphdr = {
278
64.0k
      .th_sport = (unsigned short)hdr.th_sport(),
279
64.0k
      .th_dport = (unsigned short)hdr.th_dport(),
280
64.0k
      .th_seq = __builtin_bswap32(hdr.th_seq()),
281
64.0k
      .th_ack = __builtin_bswap32(hdr.th_ack()),
282
64.0k
      .th_off = hdr.th_off(),
283
64.0k
      .th_flags = 0,
284
64.0k
      .th_win = (unsigned short)hdr.th_win(),
285
64.0k
      .th_sum = 0,
286
64.0k
      .th_urp = (unsigned short)hdr.th_urp(),
287
64.0k
  };
288
289
64.0k
  for (const int flag : hdr.th_flags()) {
290
21.3k
    tcphdr.th_flags ^= flag;
291
21.3k
  }
292
293
  // Prefer pure syn
294
64.0k
  if (hdr.is_pure_syn()) {
295
3.37k
    tcphdr.th_flags &= ~(TH_RST | TH_ACK);
296
3.37k
    tcphdr.th_flags |= TH_SYN;
297
60.6k
  } else if (hdr.is_pure_ack()) {
298
3.42k
    tcphdr.th_flags &= ~(TH_RST | TH_SYN);
299
3.42k
    tcphdr.th_flags |= TH_ACK;
300
3.42k
  }
301
302
64.0k
  std::string dat((char *)&tcphdr, (char *)&tcphdr + sizeof(tcphdr));
303
64.0k
  return dat;
304
64.0k
}
305
306
0
std::string get_icmp6_hdr(const Icmp6Hdr &hdr) {
307
0
  struct icmp6_hdr icmp6_hdr = {
308
0
      .icmp6_type = (uint8_t)hdr.icmp6_type(),
309
0
      .icmp6_code = (uint8_t)hdr.icmp6_code(),
310
0
      .icmp6_cksum = 0,
311
0
  };
312
0
  icmp6_hdr.icmp6_dataun.icmp6_un_data32[0] = hdr.icmp6_dataun();
313
314
0
  std::string dat((char *)&icmp6_hdr, (char *)&icmp6_hdr + sizeof(icmp6_hdr));
315
0
  return dat;
316
0
}
317
318
0
std::string get_ip6_route_hdr(const Ip6RtHdr &hdr) {
319
0
  struct ip6_rthdr ip6_rthdr = {
320
0
      .ip6r_nxt = (uint8_t)hdr.ip6r_nxt(),
321
0
      .ip6r_len = (uint8_t)hdr.ip6r_len(),
322
0
      .ip6r_type = (uint8_t)hdr.ip6r_type(),
323
0
      .ip6r_segleft = (uint8_t)hdr.ip6r_segleft(),
324
0
  };
325
326
0
  std::string dat((char *)&ip6_rthdr, (char *)&ip6_rthdr + sizeof(ip6_rthdr));
327
0
  return dat;
328
0
}
329
330
0
std::string get_ip6_route0_hdr(const Ip6Rt0Hdr &hdr) {
331
0
  struct ip6_rthdr0 ip6_rthdr0 = {};
332
0
  ip6_rthdr0.ip6r0_nxt = hdr.ip6r0_nxt();
333
0
  ip6_rthdr0.ip6r0_len = hdr.ip6r0_len();
334
0
  ip6_rthdr0.ip6r0_type = hdr.ip6r0_type();
335
0
  ip6_rthdr0.ip6r0_segleft = hdr.ip6r0_segleft();
336
0
  ip6_rthdr0.ip6r0_reserved = hdr.ip6r0_reserved();
337
0
  *(uint32_t *)&ip6_rthdr0.ip6r0_slmap[0] = hdr.ip6r0_slmap();
338
339
0
  int i = 0;
340
0
  for (int in6addr : hdr.ip6r0_addr()) {
341
0
    if (i >= 23) {
342
0
      break;
343
0
    }
344
345
0
    get_in6_addr(&ip6_rthdr0.ip6r0_addr[i], (In6Addr)in6addr);
346
347
0
    i++;
348
0
  }
349
350
0
  std::string dat((char *)&ip6_rthdr0,
351
0
                  (char *)&ip6_rthdr0 + sizeof(ip6_rthdr0));
352
0
  return dat;
353
0
}
354
355
0
std::string get_ip6_frag_hdr(const Ip6FragHdr &hdr) {
356
0
  struct ip6_frag ip6_frag = {
357
0
      .ip6f_nxt = (uint8_t)hdr.ip6f_nxt(),
358
0
      .ip6f_reserved = (uint8_t)hdr.ip6f_reserved(),
359
0
      .ip6f_offlg = (uint16_t)hdr.ip6f_offlg(),
360
0
      .ip6f_ident = hdr.ip6f_ident(),
361
0
  };
362
363
0
  std::string dat((char *)&ip6_frag, (char *)&ip6_frag + sizeof(ip6_frag));
364
0
  return dat;
365
0
}
366
367
0
std::string get_ip6_ext(const Ip6Ext &hdr) {
368
0
  struct ip6_ext ip6_ext = {
369
0
      .ip6e_nxt = (uint8_t)hdr.ip6e_nxt(),
370
0
      .ip6e_len = (uint8_t)hdr.ip6e_len(),
371
0
  };
372
373
0
  std::string dat((char *)&ip6_ext, (char *)&ip6_ext + sizeof(ip6_ext));
374
0
  return dat;
375
0
}
376
377
0
std::string GetNecpClient(const NecpClientId &necp_client_id) {
378
0
  switch (necp_client_id) {
379
0
    case CLIENT_0: {
380
0
      return "0000000000000000";
381
0
    }
382
0
    case CLIENT_1: {
383
0
      return "1111111111111111";
384
0
    }
385
0
    case CLIENT_2: {
386
0
      return "2222222222222222";
387
0
    }
388
0
  }
389
0
  assert(false);
390
0
  return "";
391
0
}
392
393
extern "C" {
394
395
static FuzzedDataProvider *fdp = nullptr;
396
397
// These are callbacks to let the C-based backend access the fuzzed input
398
// stream.
399
1.27M
void get_fuzzed_bytes(void *addr, size_t bytes) {
400
  // If we didn't initialize the fdp just clear the bytes.
401
1.27M
  if (!fdp) {
402
2
    memset(addr, 0, bytes);
403
2
    return;
404
2
  }
405
1.27M
  memset(addr, 0, bytes);
406
1.27M
  std::vector<uint8_t> dat = fdp->ConsumeBytes<uint8_t>(bytes);
407
1.27M
  memcpy(addr, dat.data(), dat.size());
408
1.27M
}
409
410
663k
bool get_fuzzed_bool(void) {
411
  // If we didn't initialize the fdp just return false.
412
663k
  if (!fdp) {
413
0
    return false;
414
0
  }
415
663k
  return fdp->ConsumeBool();
416
663k
}
417
418
0
int get_fuzzed_int32(int low, int high) {
419
0
  return fdp->ConsumeIntegralInRange<int>(low, high);
420
0
}
421
422
0
unsigned int get_fuzzed_uint32(unsigned int low, unsigned int high) {
423
0
  return fdp->ConsumeIntegralInRange<unsigned int>(low, high);
424
0
}
425
426
0
unsigned int get_remaining_bytes() { return fdp->remaining_bytes(); }
427
428
static bool ready = false;
429
430
bool initialize_network(void);
431
432
extern unsigned long ioctls[];
433
extern int num_ioctls;
434
extern const unsigned long siocaifaddr_in6_64;
435
extern const unsigned long siocsifflags;
436
437
// Enable this when copyout should work.
438
extern bool real_copyout;
439
440
void get_in6_addrlifetime_64(struct in6_addrlifetime_64 *sai,
441
19.7k
                             const In6AddrLifetime_64 &msg) {
442
19.7k
  sai->ia6t_expire = msg.ia6t_expire();
443
19.7k
  sai->ia6t_preferred = msg.ia6t_preferred();
444
19.7k
  sai->ia6t_vltime = msg.ia6t_vltime();
445
19.7k
  sai->ia6t_pltime = msg.ia6t_pltime();
446
19.7k
}
447
448
79.0k
void get_ifr_name(void *dest, const IfrName name) {
449
79.0k
  switch (name) {
450
79.0k
    case LO0: {
451
79.0k
      memcpy(dest, "lo0", sizeof("lo0"));
452
79.0k
      break;
453
0
    }
454
0
    case STF0: {
455
0
      memcpy(dest, "stf0", sizeof("stf0"));
456
0
      break;
457
0
    }
458
79.0k
  }
459
79.0k
}
460
461
// NECP client wrappers
462
// TODO(nedwill): move these to their own file
463
void necp_client_add(int fd, NecpClientId client_id, unsigned char *data,
464
0
                     size_t size) {
465
0
  std::string client_id_s = GetNecpClient(client_id);
466
0
  int retval = 0;
467
0
  necp_client_action_wrapper(fd, NECP_CLIENT_ACTION_ADD,
468
                             // parameters
469
0
                             (unsigned char *)client_id_s.data(),
470
0
                             client_id_s.size(), data, size, &retval);
471
0
}
472
473
// TODO(nedwill): support flow_ifnet_stats
474
0
void necp_client_remove(int fd, NecpClientId client_id) {
475
0
  std::string client_id_s = GetNecpClient(client_id);
476
0
  int retval = 0;
477
0
  necp_client_action_wrapper(fd, NECP_CLIENT_ACTION_REMOVE,
478
0
                             (unsigned char *)client_id_s.data(),
479
0
                             client_id_s.size(), nullptr, 0, &retval);
480
0
}
481
482
void necp_client_copy_parameters(int fd, NecpClientId client_id,
483
0
                                 uint32_t copyout_size) {
484
0
  std::string client_id_s = GetNecpClient(client_id);
485
0
  copyout_size %= 4096;
486
0
  std::unique_ptr<uint8_t[]> copyout_buffer(new uint8_t[copyout_size]);
487
0
  int retval = 0;
488
0
  necp_client_action_wrapper(fd, NECP_CLIENT_ACTION_COPY_PARAMETERS,
489
0
                             (unsigned char *)client_id_s.data(),
490
0
                             client_id_s.size(), copyout_buffer.get(),
491
0
                             copyout_size, &retval);
492
0
}
493
494
void necp_client_agent(
495
    int fd, NecpClientId client_id,
496
0
    const ::google::protobuf::RepeatedPtrField<::NecpTlv> &necp_tlv) {
497
0
  std::string client_id_s = GetNecpClient(client_id);
498
0
  std::string parameters;
499
0
  for (const NecpTlv &tlv : necp_tlv) {
500
    // std::string dat((char *)&icmp6_hdr, (char *)&icmp6_hdr +
501
    // sizeof(icmp6_hdr));
502
0
    struct necp_tlv_header header = {
503
0
        .type = (uint8_t)tlv.necp_type(),
504
0
        .length = (uint32_t)tlv.data().size(),
505
0
    };
506
0
    std::string tlv_s((char *)&header, (char *)&header + sizeof(header));
507
0
    tlv_s += tlv.data();
508
0
    parameters += tlv_s;
509
0
  }
510
0
  int retval = 0;
511
0
  necp_client_action_wrapper(fd, NECP_CLIENT_ACTION_AGENT,
512
0
                             (unsigned char *)client_id_s.data(),
513
0
                             client_id_s.size(), (uint8_t *)parameters.data(),
514
0
                             parameters.size(), &retval);
515
0
}
516
517
0
void DoNecpClientAction(const NecpClientAction &necp_client_action) {
518
0
  switch (necp_client_action.action_case()) {
519
0
    case NecpClientAction::kAdd: {
520
0
      necp_client_add(necp_client_action.necp_fd(),
521
0
                      necp_client_action.client_id(),
522
0
                      (unsigned char *)necp_client_action.add().buffer().data(),
523
0
                      necp_client_action.add().buffer().size());
524
0
      break;
525
0
    }
526
0
    case NecpClientAction::kRemove: {
527
0
      necp_client_remove(necp_client_action.necp_fd(),
528
0
                         necp_client_action.client_id());
529
0
      break;
530
0
    }
531
0
    case NecpClientAction::kCopyParameters: {
532
0
      necp_client_copy_parameters(
533
0
          necp_client_action.necp_fd(), necp_client_action.client_id(),
534
0
          necp_client_action.copy_parameters().copyout_size());
535
0
      break;
536
0
    }
537
0
    case NecpClientAction::kAgent: {
538
0
      necp_client_agent(necp_client_action.necp_fd(),
539
0
                        necp_client_action.client_id(),
540
0
                        necp_client_action.agent().necp_tlv());
541
0
      break;
542
0
    }
543
0
    case NecpClientAction::ACTION_NOT_SET: {
544
0
      break;
545
0
    }
546
0
  }
547
0
}
548
549
29.1k
void DoTcpInput(const TcpPacket &tcp_packet) {
550
29.1k
  std::string packet_s;
551
552
29.1k
  size_t expected_size =
553
29.1k
      sizeof(struct ip) + sizeof(struct tcphdr) + tcp_packet.data().size();
554
29.1k
  packet_s += get_ip_hdr(tcp_packet.ip_hdr(), expected_size);
555
29.1k
  packet_s += get_tcp_hdr(tcp_packet.tcp_hdr());
556
29.1k
  packet_s += tcp_packet.data();
557
29.1k
  assert(expected_size == packet_s.size());
558
559
29.1k
  if (packet_s.empty()) {
560
0
    return;
561
0
  }
562
563
  // TODO(nedwill): fuzz structure of mbuf itself
564
29.1k
  void *mbuf_data = get_mbuf_data(packet_s.data(), packet_s.size(), PKTF_LOOP);
565
29.1k
  if (!mbuf_data) {
566
0
    return;
567
0
  }
568
569
29.1k
  ip_input_wrapper(mbuf_data);
570
29.1k
}
571
572
34.9k
void DoTcp6Input(const Tcp6Packet &tcp6_packet) {
573
34.9k
  std::string packet_s;
574
575
  // TODO(nedwill): support hop-by-hop and other options
576
34.9k
  size_t expected_size = sizeof(struct tcphdr) + tcp6_packet.data().size();
577
34.9k
  packet_s += get_ip6_hdr(tcp6_packet.ip6_hdr(), expected_size);
578
34.9k
  packet_s += get_tcp_hdr(tcp6_packet.tcp_hdr());
579
34.9k
  packet_s += tcp6_packet.data();
580
581
34.9k
  if (packet_s.empty()) {
582
0
    return;
583
0
  }
584
585
34.9k
  void *mbuf_data = get_mbuf_data(packet_s.data(), packet_s.size(), PKTF_LOOP);
586
34.9k
  if (!mbuf_data) {
587
0
    return;
588
0
  }
589
590
34.9k
  ip6_input_wrapper(mbuf_data);
591
34.9k
}
592
593
173k
void DoIp4Packet(const Ip4Packet &packet) {
594
173k
  size_t expected_size = sizeof(struct ip) + packet.data().size();
595
173k
  std::string packet_s = get_ip_hdr(packet.ip_hdr(), expected_size);
596
173k
  packet_s += packet.data();
597
598
173k
  void *mbuf_data = get_mbuf_data(packet_s.data(), packet_s.size(), PKTF_LOOP);
599
173k
  if (!mbuf_data) {
600
0
    return;
601
0
  }
602
603
173k
  ip_input_wrapper(mbuf_data);
604
173k
}
605
606
271k
void DoIp6Packet(const Ip6Packet &packet) {
607
271k
  size_t expected_size = packet.data().size();
608
271k
  std::string packet_s = get_ip6_hdr(packet.ip6_hdr(), expected_size);
609
271k
  packet_s += packet.data();
610
611
271k
  void *mbuf_data = get_mbuf_data(packet_s.data(), packet_s.size(), PKTF_LOOP);
612
271k
  if (!mbuf_data) {
613
0
    return;
614
0
  }
615
616
271k
  ip6_input_wrapper(mbuf_data);
617
271k
}
618
619
1.38M
void DoIpInput(const Packet &packet) {
620
1.38M
  switch (packet.packet_case()) {
621
29.1k
    case Packet::kTcpPacket: {
622
29.1k
      DoTcpInput(packet.tcp_packet());
623
29.1k
      break;
624
0
    }
625
34.9k
    case Packet::kTcp6Packet: {
626
34.9k
      DoTcp6Input(packet.tcp6_packet());
627
34.9k
      break;
628
0
    }
629
173k
    case Packet::kIp4Packet: {
630
173k
      DoIp4Packet(packet.ip4_packet());
631
173k
      break;
632
0
    }
633
271k
    case Packet::kIp6Packet: {
634
271k
      DoIp6Packet(packet.ip6_packet());
635
271k
      break;
636
0
    }
637
192k
    case Packet::kRawIp4: {
638
192k
      void *mbuf_data = get_mbuf_data(packet.raw_ip4().data(),
639
192k
                                      packet.raw_ip4().size(), PKTF_LOOP);
640
192k
      if (!mbuf_data) {
641
0
        return;
642
0
      }
643
644
192k
      ip_input_wrapper(mbuf_data);
645
192k
      break;
646
192k
    }
647
105k
    case Packet::kRawIp6: {
648
105k
      void *mbuf_data = get_mbuf_data(packet.raw_ip6().data(),
649
105k
                                      packet.raw_ip6().size(), PKTF_LOOP);
650
105k
      if (!mbuf_data) {
651
0
        return;
652
0
      }
653
654
105k
      ip6_input_wrapper(mbuf_data);
655
105k
      break;
656
105k
    }
657
577k
    case Packet::PACKET_NOT_SET: {
658
577k
      break;
659
105k
    }
660
1.38M
  }
661
1.38M
}
662
663
// TODO(nedwill): make alternative build of this function that runs the
664
// the testcases against the real XNU syscalls, also KCOV-enabled
665
5.97k
DEFINE_BINARY_PROTO_FUZZER(const Session &session) {
666
  // std::string str;
667
  // google::protobuf::TextFormat::PrintToString(session, &str);
668
  // std::cout << str;
669
5.97k
  if (!ready) {
670
1
    initialize_network();
671
1
    init_proc();
672
1
    ready = true;
673
1
  }
674
675
5.97k
  FuzzedDataProvider dp((const uint8_t *)session.data_provider().data(),
676
5.97k
                        session.data_provider().size());
677
5.97k
  fdp = &dp;
678
679
  // TODO(nedwill): Make these containers own the references to their
680
  // objects so that we can use RAII to avoid leaks.
681
5.97k
  std::vector<uint32_t> cids;
682
5.97k
  std::set<int> open_fds;
683
684
15.0M
  for (const Command &command : session.commands()) {
685
15.0M
    int retval = 0;
686
15.0M
    switch (command.command_case()) {
687
69.1k
      case Command::kSocket: {
688
69.1k
        int fd = 0;
689
69.1k
        int err = socket_wrapper(command.socket().domain(),
690
69.1k
                                 command.socket().so_type(),
691
69.1k
                                 command.socket().protocol(), &fd);
692
69.1k
        if (err == 0) {
693
          // Make sure we're tracking fds properly.
694
2
          if (open_fds.find(fd) != open_fds.end()) {
695
0
            printf("Found existing fd %d\n", fd);
696
0
            assert(false);
697
0
          }
698
2
          open_fds.insert(fd);
699
2
        }
700
69.1k
        break;
701
69.1k
      }
702
143k
      case Command::kClose: {
703
143k
        open_fds.erase(command.close().fd());
704
143k
        close_wrapper(command.close().fd(), nullptr);
705
143k
        break;
706
69.1k
      }
707
150k
      case Command::kSetSockOpt: {
708
150k
        int s = command.set_sock_opt().fd();
709
150k
        int level = command.set_sock_opt().level();
710
150k
        int name = command.set_sock_opt().name();
711
150k
        size_t size = command.set_sock_opt().val().size();
712
150k
        std::unique_ptr<char[]> val(new char[size]);
713
150k
        memcpy(val.get(), command.set_sock_opt().val().data(), size);
714
150k
        setsockopt_wrapper(s, level, name, val.get(), size, nullptr);
715
150k
        break;
716
69.1k
      }
717
324k
      case Command::kGetSockOpt: {
718
324k
        int s = command.get_sock_opt().fd();
719
324k
        int level = command.get_sock_opt().level();
720
324k
        int name = command.get_sock_opt().name();
721
324k
        socklen_t size = command.get_sock_opt().size();
722
324k
        if (size < 0 || size > 4096) {
723
6.06k
          break;
724
6.06k
        }
725
318k
        std::unique_ptr<char[]> val(new char[size]);
726
318k
        getsockopt_wrapper(s, level, name, val.get(), &size, nullptr);
727
318k
        break;
728
324k
      }
729
82.1k
      case Command::kBind: {
730
82.1k
        std::string sockaddr_s = get_sockaddr(command.bind().sockaddr());
731
82.1k
        bind_wrapper(command.bind().fd(), (caddr_t)sockaddr_s.data(),
732
82.1k
                     sockaddr_s.size(), nullptr);
733
82.1k
        break;
734
324k
      }
735
556k
      case Command::kIoctl: {
736
        // TODO: pick these values more efficiently
737
        // XXX: these mutate global state
738
556k
        uint32_t fd = command.ioctl().fd();
739
556k
        uint32_t com = ioctls[command.ioctl().ioctl_idx() - 1];
740
556k
        real_copyout = false;
741
556k
        ioctl_wrapper(fd, com, /*data=*/(caddr_t)1, nullptr);
742
556k
        real_copyout = true;
743
556k
        break;
744
324k
      }
745
149k
      case Command::kAccept: {
746
149k
        std::string sockaddr_s = get_sockaddr(command.accept().sockaddr());
747
149k
        socklen_t size = sockaddr_s.size();
748
149k
        int retval = 0;
749
149k
        accept_wrapper(command.accept().fd(), (caddr_t)sockaddr_s.data(), &size,
750
149k
                       &retval);
751
149k
        break;
752
324k
      }
753
1.38M
      case Command::kIpInput: {
754
1.38M
        DoIpInput(command.ip_input());
755
1.38M
        break;
756
324k
      }
757
177k
      case Command::kIoctlReal: {
758
177k
        switch (command.ioctl_real().ioctl_case()) {
759
19.7k
          case IoctlReal::kSiocaifaddrIn664: {
760
19.7k
            const In6_AliasReq_64 &req =
761
19.7k
                command.ioctl_real().siocaifaddr_in6_64();
762
19.7k
            struct in6_aliasreq_64 alias = {};
763
19.7k
            memcpy(alias.ifra_name, req.ifra_name().data(),
764
19.7k
                   std::min(req.ifra_name().size(), sizeof(alias.ifra_name)));
765
19.7k
            get_sockaddr6(&alias.ifra_addr, req.ifra_addr());
766
19.7k
            get_sockaddr6(&alias.ifra_dstaddr, req.ifra_dstaddr());
767
19.7k
            get_sockaddr6(&alias.ifra_prefixmask, req.ifra_prefixmask());
768
19.7k
            for (int flag : req.ifra_flags()) {
769
              // make mutations that dupe a flag more useful by xoring
770
4.75k
              alias.ifra_flags ^= flag;
771
4.75k
            }
772
19.7k
            get_in6_addrlifetime_64(&alias.ifra_lifetime, req.ifra_lifetime());
773
19.7k
            ioctl_wrapper(command.ioctl_real().fd(), siocaifaddr_in6_64,
774
19.7k
                          (caddr_t)&alias, nullptr);
775
19.7k
          }
776
79.0k
          case IoctlReal::kSiocsifflags: {
777
79.0k
            struct ifreq ifreq = {};
778
79.0k
            for (int flag : command.ioctl_real().siocsifflags().flags()) {
779
70.7k
              ifreq.ifr_flags |= flag;
780
70.7k
            }
781
79.0k
            get_ifr_name(ifreq.ifr_name, LO0);
782
79.0k
            ioctl_wrapper(command.ioctl_real().fd(), siocsifflags,
783
79.0k
                          (caddr_t)&ifreq, nullptr);
784
79.0k
            break;
785
19.7k
          }
786
98.2k
          case IoctlReal::IOCTL_NOT_SET: {
787
98.2k
            break;
788
19.7k
          }
789
177k
        }
790
177k
      }
791
290k
      case Command::kConnectx: {
792
290k
        bool has_srcaddr = command.connectx().endpoints().has_sae_srcaddr();
793
794
290k
        std::string srcaddr_s;
795
290k
        if (has_srcaddr) {
796
22.2k
          srcaddr_s =
797
22.2k
              get_sockaddr(command.connectx().endpoints().sae_srcaddr());
798
22.2k
        }
799
800
290k
        std::string dstaddr_s =
801
290k
            get_sockaddr(command.connectx().endpoints().sae_dstaddr());
802
803
290k
        void *srcaddr = (void *)srcaddr_s.data();
804
290k
        uint32_t srcsize = srcaddr_s.size();
805
290k
        if (!has_srcaddr) {
806
268k
          srcaddr = nullptr;
807
268k
          assert(!srcsize);
808
268k
        }
809
810
290k
        void *dstaddr = (void *)dstaddr_s.data();
811
290k
        uint32_t dstsize = dstaddr_s.size();
812
813
        // We ignore failure here since it's ok to try to send things regardless
814
290k
        uint32_t connectx_flags = 0;
815
290k
        for (const int flag : command.connectx().flags()) {
816
14.0k
          connectx_flags |= flag;
817
14.0k
        }
818
290k
        uint32_t cid = 0;
819
820
290k
        struct user64_sa_endpoints endpoints = {
821
290k
            .sae_srcif = static_cast<unsigned int>(command.connectx().endpoints().sae_srcif()),
822
290k
            .sae_srcaddr = (user64_addr_t)srcaddr,
823
290k
            .sae_srcaddrlen = srcsize,
824
290k
            .sae_dstaddr = (user64_addr_t)dstaddr,
825
290k
            .sae_dstaddrlen = dstsize};
826
827
        // TODO(nedwill): is this a return value?
828
290k
        size_t len = 0;
829
        // TODO(nedwill): add IOV mocking
830
290k
        connectx_wrapper(command.connectx().socket(), &endpoints,
831
290k
                         command.connectx().associd(), connectx_flags, nullptr,
832
290k
                         0, &len, &cid, nullptr);
833
290k
        cids.push_back(cid);
834
290k
        break;
835
290k
      }
836
84.7k
      case Command::kConnect: {
837
84.7k
        std::string sockaddr_s = get_sockaddr(command.connect().sockaddr());
838
84.7k
        connect_wrapper(command.connect().fd(), (caddr_t)sockaddr_s.data(),
839
84.7k
                        sockaddr_s.size(), nullptr);
840
84.7k
        break;
841
290k
      }
842
99.5k
      case Command::kListen: {
843
99.5k
        listen_wrapper(command.listen().socket(), command.listen().backlog(),
844
99.5k
                       nullptr);
845
99.5k
        break;
846
290k
      }
847
62.8k
      case Command::kDisconnectx: {
848
62.8k
        uint32_t cid = 0;
849
62.8k
        if (!cids.empty()) {
850
61.8k
          cid = cids[command.disconnectx().cid() % cids.size()];
851
61.8k
        } else {
852
954
          cid = command.disconnectx().cid();
853
954
        }
854
62.8k
        disconnectx_wrapper(command.disconnectx().fd(),
855
62.8k
                            command.disconnectx().associd(), cid, nullptr);
856
62.8k
        break;
857
290k
      }
858
352k
      case Command::kClearAll: {
859
        // TODO(nedwill): finer-grained calling of clear_all functions here
860
352k
        clear_all();
861
352k
        break;
862
290k
      }
863
54.5k
      case Command::kNecpMatchPolicy: {
864
54.5k
        std::unique_ptr<uint8_t[]> parameters_u8(
865
54.5k
            new uint8_t[command.necp_match_policy().parameters().size()]);
866
54.5k
        memcpy(parameters_u8.get(),
867
54.5k
               command.necp_match_policy().parameters().data(),
868
54.5k
               command.necp_match_policy().parameters().size());
869
        // necp_match_policy_wrapper(
870
        //     parameters_u8.get(),
871
        //     command.necp_match_policy().parameters().size(),
872
        //     /*returned_result*/ nullptr, nullptr);
873
54.5k
        break;
874
290k
      }
875
143k
      case Command::kNecpOpen: {
876
        // int flags = 0;
877
        // for (int flag : command.necp_open().flags()) {
878
        //   flags |= flag;
879
        // }
880
        // int fd = 0;
881
        // int err = necp_open_wrapper(flags, &fd);
882
        // if (err == 0) {
883
        //   if (open_fds.find(fd) != open_fds.end()) {
884
        //     printf("fd %d already in open sockets\n", fd);
885
        //     assert(false);
886
        //   }
887
        //   open_fds.insert(fd);
888
        // }
889
143k
        break;
890
290k
      }
891
89.2k
      case Command::kNecpClientAction: {
892
        // DoNecpClientAction(command.necp_client_action());
893
89.2k
        break;
894
290k
      }
895
121k
      case Command::kNecpSessionOpen: {
896
        // int flags = 0;
897
        // int fd = 0;
898
        // int err = necp_session_open_wrapper(flags, &fd);
899
        // if (err == 0) {
900
        //   if (open_fds.find(fd) != open_fds.end()) {
901
        //     printf("fd %d already in open sockets\n", fd);
902
        //     assert(false);
903
        //   }
904
        //   open_fds.insert(fd);
905
        // }
906
121k
        break;
907
290k
      }
908
121k
      case Command::kNecpSessionAction: {
909
121k
        size_t out_buffer_size =
910
121k
            command.necp_session_action().out_buffer_size() % 4096;
911
121k
        std::unique_ptr<uint8_t[]> out_buffer(new uint8_t[out_buffer_size]);
912
        // necp_session_action_wrapper(
913
        //     command.necp_session_action().necp_fd(),
914
        //     command.necp_session_action().action(),
915
        //     // TODO(nedwill): fix const cast
916
        //     (uint8_t *)command.necp_session_action().in_buffer().data(),
917
        //     command.necp_session_action().in_buffer().size(), out_buffer.get(),
918
        //     out_buffer_size, &retval);
919
121k
        break;
920
290k
      }
921
88.8k
      case Command::kAcceptNocancel: {
922
88.8k
        std::string sockaddr_s = get_sockaddr(command.accept_nocancel().name());
923
88.8k
        socklen_t size = sockaddr_s.size();
924
88.8k
        accept_nocancel_wrapper(command.accept_nocancel().s(),
925
88.8k
                                (caddr_t)sockaddr_s.data(), &size, &retval);
926
88.8k
        break;
927
290k
      }
928
110k
      case Command::kConnectNocancel: {
929
110k
        std::string sockaddr_s =
930
110k
            get_sockaddr(command.connect_nocancel().name());
931
110k
        socklen_t size = sockaddr_s.size();
932
110k
        connect_nocancel_wrapper(command.connect_nocancel().s(),
933
110k
                                 (caddr_t)sockaddr_s.data(), size, &retval);
934
110k
        break;
935
290k
      }
936
100k
      case Command::kGetpeername: {
937
100k
        std::string sockaddr_s = get_sockaddr(command.getpeername().asa());
938
100k
        socklen_t size = sockaddr_s.size();
939
100k
        getpeername_wrapper(command.getpeername().fdes(),
940
100k
                            (caddr_t)sockaddr_s.data(), &size, &retval);
941
100k
        break;
942
290k
      }
943
77.1k
      case Command::kGetsockname: {
944
77.1k
        std::string sockaddr_s = get_sockaddr(command.getsockname().asa());
945
77.1k
        socklen_t size = sockaddr_s.size();
946
77.1k
        getsockname_wrapper(command.getsockname().fdes(),
947
77.1k
                            (caddr_t)sockaddr_s.data(), &size, &retval);
948
77.1k
        break;
949
290k
      }
950
92.8k
      case Command::kPeeloff: {
951
92.8k
        peeloff_wrapper(command.peeloff().s(), command.peeloff().aid(),
952
92.8k
                        &retval);
953
92.8k
        break;
954
290k
      }
955
66.9k
      case Command::kRecvfrom: {
956
66.9k
        std::string sockaddr_s = get_sockaddr(command.recvfrom().from());
957
66.9k
        int size = sockaddr_s.size();
958
        // TODO(nedwill): just specify the size of the buffer, don't store bytes
959
        // in the proto message just for them to be overwritten
960
66.9k
        recvfrom_wrapper(
961
66.9k
            command.recvfrom().s(), (caddr_t)command.recvfrom().buf().data(),
962
66.9k
            command.recvfrom().buf().size(), command.recvfrom().flags(),
963
66.9k
            (struct sockaddr *)sockaddr_s.data(), &size, &retval);
964
66.9k
        break;
965
290k
      }
966
66.9k
      case Command::kRecvfromNocancel: {
967
66.9k
        std::string sockaddr_s =
968
66.9k
            get_sockaddr(command.recvfrom_nocancel().from());
969
66.9k
        int size = sockaddr_s.size();
970
66.9k
        recvfrom_nocancel_wrapper(
971
66.9k
            command.recvfrom_nocancel().s(),
972
66.9k
            (caddr_t)command.recvfrom_nocancel().buf().data(),
973
66.9k
            command.recvfrom_nocancel().buf().size(),
974
66.9k
            command.recvfrom_nocancel().flags(),
975
66.9k
            (struct sockaddr *)sockaddr_s.data(), &size, &retval);
976
66.9k
        break;
977
290k
      }
978
86.9k
      case Command::kRecvmsg: {
979
        // TODO(nedwill): fuzz this msg field
980
86.9k
        user64_msghdr msg = {};
981
86.9k
        recvmsg_wrapper(command.recvmsg().s(), (struct msghdr *)&msg,
982
86.9k
                        command.recvmsg().flags(), &retval);
983
86.9k
        break;
984
290k
      }
985
154k
      case Command::kSendto: {
986
154k
        std::string sockaddr_s = get_sockaddr(command.sendto().to());
987
154k
        socklen_t size = sockaddr_s.size();
988
154k
        sendto_wrapper(command.sendto().s(),
989
154k
                       (caddr_t)command.sendto().buf().data(),
990
154k
                       command.sendto().buf().size(), command.sendto().flags(),
991
154k
                       (caddr_t)sockaddr_s.data(), size, &retval);
992
154k
        break;
993
290k
      }
994
1.68M
      case Command::kSocketpair: {
995
1.68M
        int rsv[2] = {};
996
1.68M
        int ret = socketpair_wrapper(
997
1.68M
            command.socketpair().domain(), command.socketpair().type(),
998
1.68M
            command.socketpair().protocol(), rsv, &retval);
999
1.68M
        if (!ret) {
1000
2
          if (open_fds.find(rsv[0]) != open_fds.end()) {
1001
0
            assert(false);
1002
0
          }
1003
2
          open_fds.insert(rsv[0]);
1004
2
          if (open_fds.find(rsv[1]) != open_fds.end()) {
1005
0
            assert(false);
1006
0
          }
1007
2
          open_fds.insert(rsv[1]);
1008
2
        }
1009
1.68M
        break;
1010
1.68M
      }
1011
1.68M
      case Command::kPipe: {
1012
167k
        int rsv[2] = {};
1013
167k
        int ret = pipe_wrapper(rsv);
1014
167k
        if (!ret) {
1015
2
          if (open_fds.find(rsv[0]) != open_fds.end()) {
1016
0
            assert(false);
1017
0
          }
1018
2
          open_fds.insert(rsv[0]);
1019
2
          if (open_fds.find(rsv[1]) != open_fds.end()) {
1020
0
            assert(false);
1021
0
          }
1022
2
          open_fds.insert(rsv[1]);
1023
2
        }
1024
167k
        break;
1025
167k
      }
1026
167k
      case Command::kShutdown: {
1027
113k
        shutdown_wrapper(command.shutdown().s(), command.shutdown().how(),
1028
113k
                         &retval);
1029
113k
        break;
1030
167k
      }
1031
7.98M
      case Command::COMMAND_NOT_SET: {
1032
7.98M
        break;
1033
167k
      }
1034
15.0M
    }
1035
15.0M
  }
1036
1037
5.97k
  for (int fd : open_fds) {
1038
10
    int err = close_wrapper(fd, nullptr);
1039
10
    assert(err != EBADF);
1040
10
  }
1041
1042
  // TODO(nedwill): split up these worker threads and run them as
1043
  // part of main loop
1044
5.97k
  clear_all();
1045
5.97k
}
1046
}