Coverage Report

Created: 2026-06-25 06:11

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
620k
void get_in6_addr(struct in6_addr *sai, enum In6Addr addr) {
56
620k
  memset(sai, 0, sizeof(*sai));
57
620k
  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.49k
    case IN6_ADDR_LINK_LOCAL: {
67
2.49k
      sai->s6_addr[0] = 0xfe;
68
2.49k
      sai->s6_addr[1] = 0x80;
69
      // TODO(nedwill): set other fields?
70
2.49k
      assert(IN6_IS_ADDR_LINKLOCAL(sai));
71
2.49k
      break;
72
2.49k
    }
73
301k
    case IN6_ADDR_LOOPBACK: {
74
301k
      *sai = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, (uint8_t)addr};
75
301k
      assert(IN6_IS_ADDR_LOOPBACK(sai));
76
301k
      break;
77
301k
    }
78
301k
    case IN6_ADDR_REAL:
79
9.19k
    case MAYBE_LOCALHOST: {
80
9.19k
      *sai = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, (uint8_t)addr};
81
9.19k
      break;
82
8.54k
    }
83
3.26k
    case IN6_ADDR_V4COMPAT: {
84
3.26k
      sai->s6_addr[12] = 1;
85
3.26k
      assert(IN6_IS_ADDR_V4COMPAT(sai));
86
3.26k
      break;
87
3.26k
    }
88
32.1k
    case IN6_ADDR_V4MAPPED: {
89
32.1k
      *(uint32_t *)&sai->s6_addr[8] = 0xffff0000;
90
32.1k
      assert(IN6_IS_ADDR_V4MAPPED(sai));
91
32.1k
      break;
92
32.1k
    }
93
32.1k
    case IN6_ADDR_6TO4: {
94
465
      sai->s6_addr16[0] = ntohs(0x2002);
95
465
      assert(IN6_IS_ADDR_6TO4(sai));
96
465
      break;
97
465
    }
98
3.52k
    case IN6_ADDR_LINKLOCAL: {
99
3.52k
      sai->s6_addr[0] = 0xfe;
100
3.52k
      sai->s6_addr[1] = 0x80;
101
3.52k
      assert(IN6_IS_ADDR_LINKLOCAL(sai));
102
3.52k
      break;
103
3.52k
    }
104
3.52k
    case IN6_ADDR_SITELOCAL: {
105
623
      sai->s6_addr[0] = 0xfe;
106
623
      sai->s6_addr[1] = 0xc0;
107
623
      assert(IN6_IS_ADDR_SITELOCAL(sai));
108
623
      break;
109
623
    }
110
2.13k
    case IN6_ADDR_MULTICAST: {
111
2.13k
      sai->s6_addr[0] = 0xff;
112
2.13k
      assert(IN6_IS_ADDR_MULTICAST(sai));
113
2.13k
      break;
114
2.13k
    }
115
2.13k
    case IN6_ADDR_UNIQUE_LOCAL: {
116
1.25k
      sai->s6_addr[0] = 0xfc;
117
1.25k
      assert(IN6_IS_ADDR_UNIQUE_LOCAL(sai));
118
1.25k
      break;
119
1.25k
    }
120
1.25k
    case IN6_ADDR_MC_NODELOCAL: {
121
1.03k
      sai->s6_addr[0] = 0xff;
122
1.03k
      sai->s6_addr[1] = __IPV6_ADDR_SCOPE_NODELOCAL;
123
1.03k
      assert(IN6_IS_ADDR_MC_NODELOCAL(sai));
124
1.03k
      break;
125
1.03k
    }
126
1.03k
    case IN6_ADDR_MC_INTFACELOCAL: {
127
704
      sai->s6_addr[0] = 0xff;
128
704
      sai->s6_addr[1] = __IPV6_ADDR_SCOPE_INTFACELOCAL;
129
704
      assert(IN6_IS_ADDR_MC_INTFACELOCAL(sai));
130
704
      break;
131
704
    }
132
9.77k
    case IN6_ADDR_MC_LINKLOCAL: {
133
9.77k
      sai->s6_addr[0] = 0xff;
134
9.77k
      sai->s6_addr[1] = __IPV6_ADDR_SCOPE_LINKLOCAL;
135
9.77k
      assert(IN6_IS_ADDR_MC_LINKLOCAL(sai));
136
9.77k
      break;
137
9.77k
    }
138
9.77k
    case IN6_ADDR_MC_SITELOCAL: {
139
2.20k
      sai->s6_addr[0] = 0xff;
140
2.20k
      sai->s6_addr[1] = __IPV6_ADDR_SCOPE_SITELOCAL;
141
2.20k
      assert(IN6_IS_ADDR_MC_SITELOCAL(sai));
142
2.20k
      break;
143
2.20k
    }
144
2.20k
    case IN6_ADDR_MC_ORGLOCAL: {
145
823
      sai->s6_addr[0] = 0xff;
146
823
      sai->s6_addr[1] = __IPV6_ADDR_SCOPE_ORGLOCAL;
147
823
      assert(IN6_IS_ADDR_MC_ORGLOCAL(sai));
148
823
      break;
149
823
    }
150
823
    case IN6_ADDR_MC_GLOBAL: {
151
737
      sai->s6_addr[0] = 0xff;
152
737
      sai->s6_addr[1] = __IPV6_ADDR_SCOPE_GLOBAL;
153
737
      assert(IN6_IS_ADDR_MC_GLOBAL(sai));
154
737
      break;
155
737
    }
156
8.83k
    case IN6_ADDR_UNSPECIFIED:
157
242k
    case IN6_ADDR_ANY: {
158
242k
      assert(IN6_IS_ADDR_UNSPECIFIED(sai));
159
242k
      break;
160
242k
    }
161
242k
    case IN6_ADDR_LOCAL_ADDRESS: {
162
      // Discovered this address dynamically
163
      // fe80:0001:0000:0000:a8aa:aaaa:aaaa:aaaa
164
1.40k
      sai->s6_addr16[0] = 0xfe80;
165
1.40k
      sai->s6_addr16[1] = 0x0001;
166
1.40k
      sai->s6_addr16[4] = 0xa8aa;
167
1.40k
      sai->s6_addr16[5] = 0xaaaa;
168
1.40k
      sai->s6_addr16[6] = 0xaaaa;
169
1.40k
      sai->s6_addr16[7] = 0xaaaa;
170
1.40k
    }
171
      // *sai = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, (uint8_t)addr};
172
620k
  }
173
620k
}
174
175
88.7k
void get_sockaddr6(struct sockaddr_in6 *sai, const SockAddr6 &sa6) {
176
88.7k
  sai->sin6_len = sizeof(struct sockaddr_in6);
177
88.7k
  sai->sin6_family = (sa_family_t)AF_INET6;  // sa6.family();
178
88.7k
  sai->sin6_port = (in_port_t)sa6.port();
179
88.7k
  sai->sin6_flowinfo = sa6.flow_info();
180
88.7k
  get_in6_addr(&sai->sin6_addr, sa6.sin6_addr());
181
88.7k
  sai->sin6_scope_id = sa6.sin6_scope_id();
182
88.7k
}
183
184
1.17M
std::string get_sockaddr(const SockAddr &sockaddr) {
185
1.17M
  std::string dat;
186
1.17M
  switch (sockaddr.sockaddr_case()) {
187
91.6k
    case SockAddr::kSockaddrGeneric: {
188
91.6k
      const SockAddrGeneric &sag = sockaddr.sockaddr_generic();
189
      // data size + sizeof(sa_len) + sizeof(sa_family)
190
91.6k
      struct sockaddr_generic sag_s = {
191
91.6k
          .sa_len = (uint8_t)(sizeof(sockaddr_generic) + sag.sa_data().size()),
192
91.6k
          .sa_family = (uint8_t)sag.sa_family(),
193
91.6k
      };
194
195
91.6k
      dat = std::string((char *)&sag_s, (char *)&sag_s + sizeof(sag_s));
196
91.6k
      dat += sag.sa_data();
197
91.6k
      break;
198
0
    }
199
60.9k
    case SockAddr::kSockaddr4: {
200
60.9k
      struct sockaddr_in sai = {
201
60.9k
          .sin_len = sizeof(struct sockaddr_in),
202
60.9k
          .sin_family =
203
60.9k
              AF_INET,  // (unsigned char)sockaddr.sockaddr4().sin_family(),
204
60.9k
          .sin_port = (unsigned short)sockaddr.sockaddr4().sin_port(),
205
60.9k
          .sin_addr = {(unsigned int)sockaddr.sockaddr4().sin_addr()},
206
60.9k
          .sin_zero = {},
207
60.9k
      };
208
60.9k
      dat = std::string((char *)&sai, (char *)&sai + sizeof(sai));
209
60.9k
      break;
210
0
    }
211
39.8k
    case SockAddr::kSockaddr6: {
212
39.8k
      struct sockaddr_in6 sai = {};
213
39.8k
      get_sockaddr6(&sai, sockaddr.sockaddr6());
214
39.8k
      dat = std::string((char *)&sai, (char *)&sai + sizeof(sai));
215
39.8k
      break;
216
0
    }
217
    // case SockAddr::kRawBytes: {
218
    //   dat = sockaddr.raw_bytes();
219
    //   break;
220
    // }
221
982k
    case SockAddr::SOCKADDR_NOT_SET: {
222
982k
      break;
223
0
    }
224
1.17M
  }
225
1.17M
  return dat;
226
1.17M
}
227
228
265k
std::string get_ip6_hdr(const Ip6Hdr &hdr, uint16_t expected_size) {
229
265k
  struct ip6_hdr ip6_hdr;
230
265k
  memset(&ip6_hdr, 0, sizeof(ip6_hdr));
231
265k
  get_in6_addr(&ip6_hdr.ip6_src, hdr.ip6_src());
232
265k
  get_in6_addr(&ip6_hdr.ip6_dst, hdr.ip6_src());
233
265k
  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
265k
  ip6_hdr.ip6_ctlun.ip6_un1.ip6_un1_plen =
237
265k
      __builtin_bswap16(expected_size);  // hdr.ip6_hdrctl().ip6_un1_plen();
238
265k
  ip6_hdr.ip6_ctlun.ip6_un1.ip6_un1_nxt = hdr.ip6_hdrctl().ip6_un1_nxt();
239
265k
  ip6_hdr.ip6_ctlun.ip6_un1.ip6_un1_hlim = hdr.ip6_hdrctl().ip6_un1_hlim();
240
265k
  std::string dat((char *)&ip6_hdr, (char *)&ip6_hdr + sizeof(ip6_hdr));
241
265k
  return dat;
242
265k
}
243
244
185k
std::string get_ip_hdr(const IpHdr &hdr, size_t expected_size) {
245
185k
  struct in_addr ip_src = {.s_addr = (unsigned int)hdr.ip_src()};
246
185k
  struct in_addr ip_dst = {.s_addr = (unsigned int)hdr.ip_dst()};
247
185k
  struct ip ip_hdr = {
248
185k
      .ip_hl = 5,    // TODO(nedwill): support options // hdr.ip_hl(),
249
185k
      .ip_v = IPV4,  // hdr.ip_v(),
250
185k
      .ip_tos = (u_char)hdr.ip_tos(),
251
185k
      .ip_len = (u_short)__builtin_bswap16(expected_size),
252
185k
      .ip_id = (u_short)hdr.ip_id(),
253
185k
      .ip_off = (u_short)hdr.ip_off(),
254
185k
      .ip_ttl = (u_char)hdr.ip_ttl(),
255
185k
      .ip_p = (u_char)hdr.ip_p(),
256
185k
      .ip_sum = 0,
257
185k
      .ip_src = ip_src,
258
185k
      .ip_dst = ip_dst,
259
185k
  };
260
185k
  std::string dat((char *)&ip_hdr, (char *)&ip_hdr + sizeof(ip_hdr));
261
185k
  return dat;
262
185k
}
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
59.1k
std::string get_tcp_hdr(const TcpHdr &hdr) {
277
59.1k
  struct tcphdr tcphdr = {
278
59.1k
      .th_sport = (unsigned short)hdr.th_sport(),
279
59.1k
      .th_dport = (unsigned short)hdr.th_dport(),
280
59.1k
      .th_seq = __builtin_bswap32(hdr.th_seq()),
281
59.1k
      .th_ack = __builtin_bswap32(hdr.th_ack()),
282
59.1k
      .th_off = hdr.th_off(),
283
59.1k
      .th_flags = 0,
284
59.1k
      .th_win = (unsigned short)hdr.th_win(),
285
59.1k
      .th_sum = 0,
286
59.1k
      .th_urp = (unsigned short)hdr.th_urp(),
287
59.1k
  };
288
289
59.1k
  for (const int flag : hdr.th_flags()) {
290
16.8k
    tcphdr.th_flags ^= flag;
291
16.8k
  }
292
293
  // Prefer pure syn
294
59.1k
  if (hdr.is_pure_syn()) {
295
3.15k
    tcphdr.th_flags &= ~(TH_RST | TH_ACK);
296
3.15k
    tcphdr.th_flags |= TH_SYN;
297
56.0k
  } else if (hdr.is_pure_ack()) {
298
3.21k
    tcphdr.th_flags &= ~(TH_RST | TH_SYN);
299
3.21k
    tcphdr.th_flags |= TH_ACK;
300
3.21k
  }
301
302
59.1k
  std::string dat((char *)&tcphdr, (char *)&tcphdr + sizeof(tcphdr));
303
59.1k
  return dat;
304
59.1k
}
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
986k
void get_fuzzed_bytes(void *addr, size_t bytes) {
400
  // If we didn't initialize the fdp just clear the bytes.
401
986k
  if (!fdp) {
402
2
    memset(addr, 0, bytes);
403
2
    return;
404
2
  }
405
986k
  memset(addr, 0, bytes);
406
986k
  std::vector<uint8_t> dat = fdp->ConsumeBytes<uint8_t>(bytes);
407
986k
  memcpy(addr, dat.data(), dat.size());
408
986k
}
409
410
651k
bool get_fuzzed_bool(void) {
411
  // If we didn't initialize the fdp just return false.
412
651k
  if (!fdp) {
413
0
    return false;
414
0
  }
415
651k
  return fdp->ConsumeBool();
416
651k
}
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
16.2k
                             const In6AddrLifetime_64 &msg) {
442
16.2k
  sai->ia6t_expire = msg.ia6t_expire();
443
16.2k
  sai->ia6t_preferred = msg.ia6t_preferred();
444
16.2k
  sai->ia6t_vltime = msg.ia6t_vltime();
445
16.2k
  sai->ia6t_pltime = msg.ia6t_pltime();
446
16.2k
}
447
448
67.0k
void get_ifr_name(void *dest, const IfrName name) {
449
67.0k
  switch (name) {
450
67.0k
    case LO0: {
451
67.0k
      memcpy(dest, "lo0", sizeof("lo0"));
452
67.0k
      break;
453
0
    }
454
0
    case STF0: {
455
0
      memcpy(dest, "stf0", sizeof("stf0"));
456
0
      break;
457
0
    }
458
67.0k
  }
459
67.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
25.2k
void DoTcpInput(const TcpPacket &tcp_packet) {
550
25.2k
  std::string packet_s;
551
552
25.2k
  size_t expected_size =
553
25.2k
      sizeof(struct ip) + sizeof(struct tcphdr) + tcp_packet.data().size();
554
25.2k
  packet_s += get_ip_hdr(tcp_packet.ip_hdr(), expected_size);
555
25.2k
  packet_s += get_tcp_hdr(tcp_packet.tcp_hdr());
556
25.2k
  packet_s += tcp_packet.data();
557
25.2k
  assert(expected_size == packet_s.size());
558
559
25.2k
  if (packet_s.empty()) {
560
0
    return;
561
0
  }
562
563
  // TODO(nedwill): fuzz structure of mbuf itself
564
25.2k
  void *mbuf_data = get_mbuf_data(packet_s.data(), packet_s.size(), PKTF_LOOP);
565
25.2k
  if (!mbuf_data) {
566
0
    return;
567
0
  }
568
569
25.2k
  ip_input_wrapper(mbuf_data);
570
25.2k
}
571
572
33.9k
void DoTcp6Input(const Tcp6Packet &tcp6_packet) {
573
33.9k
  std::string packet_s;
574
575
  // TODO(nedwill): support hop-by-hop and other options
576
33.9k
  size_t expected_size = sizeof(struct tcphdr) + tcp6_packet.data().size();
577
33.9k
  packet_s += get_ip6_hdr(tcp6_packet.ip6_hdr(), expected_size);
578
33.9k
  packet_s += get_tcp_hdr(tcp6_packet.tcp_hdr());
579
33.9k
  packet_s += tcp6_packet.data();
580
581
33.9k
  if (packet_s.empty()) {
582
0
    return;
583
0
  }
584
585
33.9k
  void *mbuf_data = get_mbuf_data(packet_s.data(), packet_s.size(), PKTF_LOOP);
586
33.9k
  if (!mbuf_data) {
587
0
    return;
588
0
  }
589
590
33.9k
  ip6_input_wrapper(mbuf_data);
591
33.9k
}
592
593
160k
void DoIp4Packet(const Ip4Packet &packet) {
594
160k
  size_t expected_size = sizeof(struct ip) + packet.data().size();
595
160k
  std::string packet_s = get_ip_hdr(packet.ip_hdr(), expected_size);
596
160k
  packet_s += packet.data();
597
598
160k
  void *mbuf_data = get_mbuf_data(packet_s.data(), packet_s.size(), PKTF_LOOP);
599
160k
  if (!mbuf_data) {
600
0
    return;
601
0
  }
602
603
160k
  ip_input_wrapper(mbuf_data);
604
160k
}
605
606
231k
void DoIp6Packet(const Ip6Packet &packet) {
607
231k
  size_t expected_size = packet.data().size();
608
231k
  std::string packet_s = get_ip6_hdr(packet.ip6_hdr(), expected_size);
609
231k
  packet_s += packet.data();
610
611
231k
  void *mbuf_data = get_mbuf_data(packet_s.data(), packet_s.size(), PKTF_LOOP);
612
231k
  if (!mbuf_data) {
613
0
    return;
614
0
  }
615
616
231k
  ip6_input_wrapper(mbuf_data);
617
231k
}
618
619
1.21M
void DoIpInput(const Packet &packet) {
620
1.21M
  switch (packet.packet_case()) {
621
25.2k
    case Packet::kTcpPacket: {
622
25.2k
      DoTcpInput(packet.tcp_packet());
623
25.2k
      break;
624
0
    }
625
33.9k
    case Packet::kTcp6Packet: {
626
33.9k
      DoTcp6Input(packet.tcp6_packet());
627
33.9k
      break;
628
0
    }
629
160k
    case Packet::kIp4Packet: {
630
160k
      DoIp4Packet(packet.ip4_packet());
631
160k
      break;
632
0
    }
633
231k
    case Packet::kIp6Packet: {
634
231k
      DoIp6Packet(packet.ip6_packet());
635
231k
      break;
636
0
    }
637
163k
    case Packet::kRawIp4: {
638
163k
      void *mbuf_data = get_mbuf_data(packet.raw_ip4().data(),
639
163k
                                      packet.raw_ip4().size(), PKTF_LOOP);
640
163k
      if (!mbuf_data) {
641
0
        return;
642
0
      }
643
644
163k
      ip_input_wrapper(mbuf_data);
645
163k
      break;
646
163k
    }
647
90.4k
    case Packet::kRawIp6: {
648
90.4k
      void *mbuf_data = get_mbuf_data(packet.raw_ip6().data(),
649
90.4k
                                      packet.raw_ip6().size(), PKTF_LOOP);
650
90.4k
      if (!mbuf_data) {
651
0
        return;
652
0
      }
653
654
90.4k
      ip6_input_wrapper(mbuf_data);
655
90.4k
      break;
656
90.4k
    }
657
513k
    case Packet::PACKET_NOT_SET: {
658
513k
      break;
659
90.4k
    }
660
1.21M
  }
661
1.21M
}
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
6.51k
DEFINE_BINARY_PROTO_FUZZER(const Session &session) {
666
  // std::string str;
667
  // google::protobuf::TextFormat::PrintToString(session, &str);
668
  // std::cout << str;
669
6.51k
  if (!ready) {
670
1
    initialize_network();
671
1
    init_proc();
672
1
    ready = true;
673
1
  }
674
675
6.51k
  FuzzedDataProvider dp((const uint8_t *)session.data_provider().data(),
676
6.51k
                        session.data_provider().size());
677
6.51k
  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
6.51k
  std::vector<uint32_t> cids;
682
6.51k
  std::set<int> open_fds;
683
684
13.6M
  for (const Command &command : session.commands()) {
685
13.6M
    int retval = 0;
686
13.6M
    switch (command.command_case()) {
687
64.7k
      case Command::kSocket: {
688
64.7k
        int fd = 0;
689
64.7k
        int err = socket_wrapper(command.socket().domain(),
690
64.7k
                                 command.socket().so_type(),
691
64.7k
                                 command.socket().protocol(), &fd);
692
64.7k
        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
64.7k
        break;
701
64.7k
      }
702
128k
      case Command::kClose: {
703
128k
        open_fds.erase(command.close().fd());
704
128k
        close_wrapper(command.close().fd(), nullptr);
705
128k
        break;
706
64.7k
      }
707
140k
      case Command::kSetSockOpt: {
708
140k
        int s = command.set_sock_opt().fd();
709
140k
        int level = command.set_sock_opt().level();
710
140k
        int name = command.set_sock_opt().name();
711
140k
        size_t size = command.set_sock_opt().val().size();
712
140k
        std::unique_ptr<char[]> val(new char[size]);
713
140k
        memcpy(val.get(), command.set_sock_opt().val().data(), size);
714
140k
        setsockopt_wrapper(s, level, name, val.get(), size, nullptr);
715
140k
        break;
716
64.7k
      }
717
288k
      case Command::kGetSockOpt: {
718
288k
        int s = command.get_sock_opt().fd();
719
288k
        int level = command.get_sock_opt().level();
720
288k
        int name = command.get_sock_opt().name();
721
288k
        socklen_t size = command.get_sock_opt().size();
722
288k
        if (size < 0 || size > 4096) {
723
5.76k
          break;
724
5.76k
        }
725
283k
        std::unique_ptr<char[]> val(new char[size]);
726
283k
        getsockopt_wrapper(s, level, name, val.get(), &size, nullptr);
727
283k
        break;
728
288k
      }
729
77.9k
      case Command::kBind: {
730
77.9k
        std::string sockaddr_s = get_sockaddr(command.bind().sockaddr());
731
77.9k
        bind_wrapper(command.bind().fd(), (caddr_t)sockaddr_s.data(),
732
77.9k
                     sockaddr_s.size(), nullptr);
733
77.9k
        break;
734
288k
      }
735
510k
      case Command::kIoctl: {
736
        // TODO: pick these values more efficiently
737
        // XXX: these mutate global state
738
510k
        uint32_t fd = command.ioctl().fd();
739
510k
        uint32_t com = ioctls[command.ioctl().ioctl_idx() - 1];
740
510k
        real_copyout = false;
741
510k
        ioctl_wrapper(fd, com, /*data=*/(caddr_t)1, nullptr);
742
510k
        real_copyout = true;
743
510k
        break;
744
288k
      }
745
130k
      case Command::kAccept: {
746
130k
        std::string sockaddr_s = get_sockaddr(command.accept().sockaddr());
747
130k
        socklen_t size = sockaddr_s.size();
748
130k
        int retval = 0;
749
130k
        accept_wrapper(command.accept().fd(), (caddr_t)sockaddr_s.data(), &size,
750
130k
                       &retval);
751
130k
        break;
752
288k
      }
753
1.21M
      case Command::kIpInput: {
754
1.21M
        DoIpInput(command.ip_input());
755
1.21M
        break;
756
288k
      }
757
154k
      case Command::kIoctlReal: {
758
154k
        switch (command.ioctl_real().ioctl_case()) {
759
16.2k
          case IoctlReal::kSiocaifaddrIn664: {
760
16.2k
            const In6_AliasReq_64 &req =
761
16.2k
                command.ioctl_real().siocaifaddr_in6_64();
762
16.2k
            struct in6_aliasreq_64 alias = {};
763
16.2k
            memcpy(alias.ifra_name, req.ifra_name().data(),
764
16.2k
                   std::min(req.ifra_name().size(), sizeof(alias.ifra_name)));
765
16.2k
            get_sockaddr6(&alias.ifra_addr, req.ifra_addr());
766
16.2k
            get_sockaddr6(&alias.ifra_dstaddr, req.ifra_dstaddr());
767
16.2k
            get_sockaddr6(&alias.ifra_prefixmask, req.ifra_prefixmask());
768
16.2k
            for (int flag : req.ifra_flags()) {
769
              // make mutations that dupe a flag more useful by xoring
770
4.41k
              alias.ifra_flags ^= flag;
771
4.41k
            }
772
16.2k
            get_in6_addrlifetime_64(&alias.ifra_lifetime, req.ifra_lifetime());
773
16.2k
            ioctl_wrapper(command.ioctl_real().fd(), siocaifaddr_in6_64,
774
16.2k
                          (caddr_t)&alias, nullptr);
775
16.2k
          }
776
67.0k
          case IoctlReal::kSiocsifflags: {
777
67.0k
            struct ifreq ifreq = {};
778
67.0k
            for (int flag : command.ioctl_real().siocsifflags().flags()) {
779
58.9k
              ifreq.ifr_flags |= flag;
780
58.9k
            }
781
67.0k
            get_ifr_name(ifreq.ifr_name, LO0);
782
67.0k
            ioctl_wrapper(command.ioctl_real().fd(), siocsifflags,
783
67.0k
                          (caddr_t)&ifreq, nullptr);
784
67.0k
            break;
785
16.2k
          }
786
87.6k
          case IoctlReal::IOCTL_NOT_SET: {
787
87.6k
            break;
788
16.2k
          }
789
154k
        }
790
154k
      }
791
257k
      case Command::kConnectx: {
792
257k
        bool has_srcaddr = command.connectx().endpoints().has_sae_srcaddr();
793
794
257k
        std::string srcaddr_s;
795
257k
        if (has_srcaddr) {
796
23.0k
          srcaddr_s =
797
23.0k
              get_sockaddr(command.connectx().endpoints().sae_srcaddr());
798
23.0k
        }
799
800
257k
        std::string dstaddr_s =
801
257k
            get_sockaddr(command.connectx().endpoints().sae_dstaddr());
802
803
257k
        void *srcaddr = (void *)srcaddr_s.data();
804
257k
        uint32_t srcsize = srcaddr_s.size();
805
257k
        if (!has_srcaddr) {
806
234k
          srcaddr = nullptr;
807
234k
          assert(!srcsize);
808
234k
        }
809
810
257k
        void *dstaddr = (void *)dstaddr_s.data();
811
257k
        uint32_t dstsize = dstaddr_s.size();
812
813
        // We ignore failure here since it's ok to try to send things regardless
814
257k
        uint32_t connectx_flags = 0;
815
257k
        for (const int flag : command.connectx().flags()) {
816
14.0k
          connectx_flags |= flag;
817
14.0k
        }
818
257k
        uint32_t cid = 0;
819
820
257k
        struct user64_sa_endpoints endpoints = {
821
257k
            .sae_srcif = static_cast<unsigned int>(command.connectx().endpoints().sae_srcif()),
822
257k
            .sae_srcaddr = (user64_addr_t)srcaddr,
823
257k
            .sae_srcaddrlen = srcsize,
824
257k
            .sae_dstaddr = (user64_addr_t)dstaddr,
825
257k
            .sae_dstaddrlen = dstsize};
826
827
        // TODO(nedwill): is this a return value?
828
257k
        size_t len = 0;
829
        // TODO(nedwill): add IOV mocking
830
257k
        connectx_wrapper(command.connectx().socket(), &endpoints,
831
257k
                         command.connectx().associd(), connectx_flags, nullptr,
832
257k
                         0, &len, &cid, nullptr);
833
257k
        cids.push_back(cid);
834
257k
        break;
835
257k
      }
836
80.5k
      case Command::kConnect: {
837
80.5k
        std::string sockaddr_s = get_sockaddr(command.connect().sockaddr());
838
80.5k
        connect_wrapper(command.connect().fd(), (caddr_t)sockaddr_s.data(),
839
80.5k
                        sockaddr_s.size(), nullptr);
840
80.5k
        break;
841
257k
      }
842
90.2k
      case Command::kListen: {
843
90.2k
        listen_wrapper(command.listen().socket(), command.listen().backlog(),
844
90.2k
                       nullptr);
845
90.2k
        break;
846
257k
      }
847
58.4k
      case Command::kDisconnectx: {
848
58.4k
        uint32_t cid = 0;
849
58.4k
        if (!cids.empty()) {
850
57.0k
          cid = cids[command.disconnectx().cid() % cids.size()];
851
57.0k
        } else {
852
1.38k
          cid = command.disconnectx().cid();
853
1.38k
        }
854
58.4k
        disconnectx_wrapper(command.disconnectx().fd(),
855
58.4k
                            command.disconnectx().associd(), cid, nullptr);
856
58.4k
        break;
857
257k
      }
858
315k
      case Command::kClearAll: {
859
        // TODO(nedwill): finer-grained calling of clear_all functions here
860
315k
        clear_all();
861
315k
        break;
862
257k
      }
863
51.1k
      case Command::kNecpMatchPolicy: {
864
51.1k
        std::unique_ptr<uint8_t[]> parameters_u8(
865
51.1k
            new uint8_t[command.necp_match_policy().parameters().size()]);
866
51.1k
        memcpy(parameters_u8.get(),
867
51.1k
               command.necp_match_policy().parameters().data(),
868
51.1k
               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
51.1k
        break;
874
257k
      }
875
129k
      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
129k
        break;
890
257k
      }
891
82.3k
      case Command::kNecpClientAction: {
892
        // DoNecpClientAction(command.necp_client_action());
893
82.3k
        break;
894
257k
      }
895
109k
      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
109k
        break;
907
257k
      }
908
107k
      case Command::kNecpSessionAction: {
909
107k
        size_t out_buffer_size =
910
107k
            command.necp_session_action().out_buffer_size() % 4096;
911
107k
        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
107k
        break;
920
257k
      }
921
81.4k
      case Command::kAcceptNocancel: {
922
81.4k
        std::string sockaddr_s = get_sockaddr(command.accept_nocancel().name());
923
81.4k
        socklen_t size = sockaddr_s.size();
924
81.4k
        accept_nocancel_wrapper(command.accept_nocancel().s(),
925
81.4k
                                (caddr_t)sockaddr_s.data(), &size, &retval);
926
81.4k
        break;
927
257k
      }
928
99.9k
      case Command::kConnectNocancel: {
929
99.9k
        std::string sockaddr_s =
930
99.9k
            get_sockaddr(command.connect_nocancel().name());
931
99.9k
        socklen_t size = sockaddr_s.size();
932
99.9k
        connect_nocancel_wrapper(command.connect_nocancel().s(),
933
99.9k
                                 (caddr_t)sockaddr_s.data(), size, &retval);
934
99.9k
        break;
935
257k
      }
936
88.3k
      case Command::kGetpeername: {
937
88.3k
        std::string sockaddr_s = get_sockaddr(command.getpeername().asa());
938
88.3k
        socklen_t size = sockaddr_s.size();
939
88.3k
        getpeername_wrapper(command.getpeername().fdes(),
940
88.3k
                            (caddr_t)sockaddr_s.data(), &size, &retval);
941
88.3k
        break;
942
257k
      }
943
68.5k
      case Command::kGetsockname: {
944
68.5k
        std::string sockaddr_s = get_sockaddr(command.getsockname().asa());
945
68.5k
        socklen_t size = sockaddr_s.size();
946
68.5k
        getsockname_wrapper(command.getsockname().fdes(),
947
68.5k
                            (caddr_t)sockaddr_s.data(), &size, &retval);
948
68.5k
        break;
949
257k
      }
950
82.4k
      case Command::kPeeloff: {
951
82.4k
        peeloff_wrapper(command.peeloff().s(), command.peeloff().aid(),
952
82.4k
                        &retval);
953
82.4k
        break;
954
257k
      }
955
62.1k
      case Command::kRecvfrom: {
956
62.1k
        std::string sockaddr_s = get_sockaddr(command.recvfrom().from());
957
62.1k
        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
62.1k
        recvfrom_wrapper(
961
62.1k
            command.recvfrom().s(), (caddr_t)command.recvfrom().buf().data(),
962
62.1k
            command.recvfrom().buf().size(), command.recvfrom().flags(),
963
62.1k
            (struct sockaddr *)sockaddr_s.data(), &size, &retval);
964
62.1k
        break;
965
257k
      }
966
67.8k
      case Command::kRecvfromNocancel: {
967
67.8k
        std::string sockaddr_s =
968
67.8k
            get_sockaddr(command.recvfrom_nocancel().from());
969
67.8k
        int size = sockaddr_s.size();
970
67.8k
        recvfrom_nocancel_wrapper(
971
67.8k
            command.recvfrom_nocancel().s(),
972
67.8k
            (caddr_t)command.recvfrom_nocancel().buf().data(),
973
67.8k
            command.recvfrom_nocancel().buf().size(),
974
67.8k
            command.recvfrom_nocancel().flags(),
975
67.8k
            (struct sockaddr *)sockaddr_s.data(), &size, &retval);
976
67.8k
        break;
977
257k
      }
978
79.2k
      case Command::kRecvmsg: {
979
        // TODO(nedwill): fuzz this msg field
980
79.2k
        user64_msghdr msg = {};
981
79.2k
        recvmsg_wrapper(command.recvmsg().s(), (struct msghdr *)&msg,
982
79.2k
                        command.recvmsg().flags(), &retval);
983
79.2k
        break;
984
257k
      }
985
136k
      case Command::kSendto: {
986
136k
        std::string sockaddr_s = get_sockaddr(command.sendto().to());
987
136k
        socklen_t size = sockaddr_s.size();
988
136k
        sendto_wrapper(command.sendto().s(),
989
136k
                       (caddr_t)command.sendto().buf().data(),
990
136k
                       command.sendto().buf().size(), command.sendto().flags(),
991
136k
                       (caddr_t)sockaddr_s.data(), size, &retval);
992
136k
        break;
993
257k
      }
994
1.55M
      case Command::kSocketpair: {
995
1.55M
        int rsv[2] = {};
996
1.55M
        int ret = socketpair_wrapper(
997
1.55M
            command.socketpair().domain(), command.socketpair().type(),
998
1.55M
            command.socketpair().protocol(), rsv, &retval);
999
1.55M
        if (!ret) {
1000
3
          if (open_fds.find(rsv[0]) != open_fds.end()) {
1001
0
            assert(false);
1002
0
          }
1003
3
          open_fds.insert(rsv[0]);
1004
3
          if (open_fds.find(rsv[1]) != open_fds.end()) {
1005
0
            assert(false);
1006
0
          }
1007
3
          open_fds.insert(rsv[1]);
1008
3
        }
1009
1.55M
        break;
1010
1.55M
      }
1011
1.55M
      case Command::kPipe: {
1012
152k
        int rsv[2] = {};
1013
152k
        int ret = pipe_wrapper(rsv);
1014
152k
        if (!ret) {
1015
1
          if (open_fds.find(rsv[0]) != open_fds.end()) {
1016
0
            assert(false);
1017
0
          }
1018
1
          open_fds.insert(rsv[0]);
1019
1
          if (open_fds.find(rsv[1]) != open_fds.end()) {
1020
0
            assert(false);
1021
0
          }
1022
1
          open_fds.insert(rsv[1]);
1023
1
        }
1024
152k
        break;
1025
152k
      }
1026
152k
      case Command::kShutdown: {
1027
105k
        shutdown_wrapper(command.shutdown().s(), command.shutdown().how(),
1028
105k
                         &retval);
1029
105k
        break;
1030
152k
      }
1031
7.20M
      case Command::COMMAND_NOT_SET: {
1032
7.20M
        break;
1033
152k
      }
1034
13.6M
    }
1035
13.6M
  }
1036
1037
6.51k
  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
6.51k
  clear_all();
1045
6.51k
}
1046
}