Coverage Report

Created: 2025-08-25 06:58

/src/WasmEdge/lib/host/wasi/inode-linux.cpp
Line
Count
Source (jump to first uncovered line)
1
// SPDX-License-Identifier: Apache-2.0
2
// SPDX-FileCopyrightText: 2019-2024 Second State INC
3
4
#include "common/defines.h"
5
#if WASMEDGE_OS_LINUX
6
7
#include "common/errcode.h"
8
#include "common/variant.h"
9
#include "host/wasi/environ.h"
10
#include "host/wasi/inode.h"
11
#include "host/wasi/vfs.h"
12
#include "linux.h"
13
#include <algorithm>
14
#include <cstddef>
15
#include <cstdint>
16
#include <cstring>
17
#include <new>
18
#include <string>
19
#include <string_view>
20
#include <vector>
21
22
using namespace std::literals;
23
24
namespace WasmEdge {
25
namespace Host {
26
namespace WASI {
27
28
namespace {
29
30
0
inline constexpr bool isSpecialFd(int Fd) noexcept {
31
0
  switch (Fd) {
32
0
  case STDIN_FILENO:
33
0
  case STDOUT_FILENO:
34
0
  case STDERR_FILENO:
35
0
    return true;
36
0
  default:
37
0
    return false;
38
0
  }
39
0
}
40
41
inline constexpr __wasi_size_t
42
0
calculateAddrinfoLinkedListSize(struct addrinfo *const Addrinfo) {
43
0
  __wasi_size_t Length = 0;
44
0
  for (struct addrinfo *TmpPointer = Addrinfo; TmpPointer != nullptr;
45
0
       TmpPointer = TmpPointer->ai_next) {
46
0
    Length++;
47
0
  }
48
0
  return Length;
49
0
};
50
51
constexpr int openFlags(__wasi_oflags_t OpenFlags, __wasi_fdflags_t FdFlags,
52
0
                        VFS::Flags VFSFlags) noexcept {
53
0
  int Flags = O_NOFOLLOW;
54
0
#ifdef O_CLOEXEC
55
0
  Flags |= O_CLOEXEC;
56
0
#endif
57
58
0
  if (VFSFlags & VFS::Read) {
59
0
    if (VFSFlags & VFS::Write) {
60
0
      Flags |= O_RDWR;
61
0
    } else {
62
0
      Flags |= O_RDONLY;
63
0
    }
64
0
  } else if (VFSFlags & VFS::Write) {
65
0
    Flags |= O_WRONLY;
66
0
  } else {
67
0
#ifdef O_PATH
68
0
    if (OpenFlags == __WASI_OFLAGS_DIRECTORY) {
69
0
      Flags |= O_PATH;
70
0
    } else {
71
0
      Flags |= O_RDONLY;
72
0
    }
73
#else
74
    Flags |= O_RDONLY;
75
#endif
76
0
  }
77
78
0
  if (OpenFlags & __WASI_OFLAGS_CREAT) {
79
0
    Flags |= O_CREAT;
80
0
  }
81
0
  if (OpenFlags & __WASI_OFLAGS_DIRECTORY) {
82
0
    Flags |= O_DIRECTORY;
83
0
  }
84
0
  if (OpenFlags & __WASI_OFLAGS_EXCL) {
85
0
    Flags |= O_EXCL;
86
0
  }
87
0
  if ((OpenFlags & __WASI_OFLAGS_TRUNC) && (VFSFlags & VFS::Write)) {
88
0
    Flags |= O_TRUNC;
89
0
  }
90
91
  // Convert file descriptor flags.
92
0
  if ((FdFlags & __WASI_FDFLAGS_DSYNC) != 0) {
93
0
#ifdef O_DSYNC
94
0
    Flags |= O_DSYNC;
95
#else
96
    Flags |= O_SYNC;
97
#endif
98
0
  }
99
0
  if ((FdFlags & __WASI_FDFLAGS_NONBLOCK) != 0) {
100
0
    Flags |= O_NONBLOCK;
101
0
  }
102
0
  if ((FdFlags & __WASI_FDFLAGS_RSYNC) != 0) {
103
0
#ifdef O_RSYNC
104
0
    Flags |= O_RSYNC;
105
#else
106
    Flags |= O_SYNC;
107
#endif
108
0
  }
109
0
  if ((FdFlags & __WASI_FDFLAGS_SYNC) != 0) {
110
0
    Flags |= O_SYNC;
111
0
  }
112
113
0
  return Flags;
114
0
}
115
116
std::pair<const char *, std::unique_ptr<char[]>>
117
0
createNullTerminatedString(std::string_view View) noexcept {
118
0
  const char *CStr = nullptr;
119
0
  std::unique_ptr<char[]> Buffer;
120
0
  if (!View.empty()) {
121
0
    if (const auto Pos = View.find_first_of('\0');
122
0
        Pos != std::string_view::npos) {
123
0
      CStr = View.data();
124
0
    } else {
125
0
      Buffer = std::make_unique<char[]>(View.size() + 1);
126
0
      std::copy(View.begin(), View.end(), Buffer.get());
127
0
      CStr = Buffer.get();
128
0
    }
129
0
  }
130
0
  return {CStr, std::move(Buffer)};
131
0
}
132
133
} // namespace
134
135
0
void FdHolder::reset() noexcept {
136
0
  if (likely(ok())) {
137
0
    if (likely(!isSpecialFd(Fd))) {
138
0
      ::close(Fd);
139
0
    }
140
0
    Fd = -1;
141
0
  }
142
0
}
143
144
0
void TimerHolder::reset() noexcept {
145
0
  if (likely(Id.has_value())) {
146
0
    timer_delete(*Id);
147
0
    Id.reset();
148
0
  }
149
0
}
150
151
0
void DirHolder::reset() noexcept {
152
0
  if (likely(Dir != nullptr)) {
153
0
    closedir(Dir);
154
0
    Dir = nullptr;
155
0
    Cookie = 0;
156
0
  }
157
0
}
158
159
namespace {
160
0
WasiExpect<INode> createStdNode(int32_t Fd) {
161
0
  if (Fd < 0) {
162
0
    spdlog::error("    Invalid file descriptor: {}"sv, Fd);
163
0
    return WasiUnexpect(__WASI_ERRNO_BADF);
164
0
  }
165
166
0
  if (fcntl(Fd, F_GETFD) == -1) {
167
0
    spdlog::error("    Invalid file descriptor: {}"sv, Fd);
168
0
    return WasiUnexpect(__WASI_ERRNO_BADF);
169
0
  }
170
171
0
  return INode(Fd, false, false);
172
0
}
173
} // namespace
174
175
0
INode INode::stdIn() noexcept { return INode(STDIN_FILENO); }
176
177
0
INode INode::stdOut() noexcept { return INode(STDOUT_FILENO); }
178
179
0
INode INode::stdErr() noexcept { return INode(STDERR_FILENO); }
180
181
0
WasiExpect<INode> INode::fromFd(int32_t Fd) { return createStdNode(Fd); }
182
183
WasiExpect<INode> INode::open(std::string Path, __wasi_oflags_t OpenFlags,
184
                              __wasi_fdflags_t FdFlags,
185
0
                              VFS::Flags VFSFlags) noexcept {
186
0
  const int Flags = openFlags(OpenFlags, FdFlags, VFSFlags);
187
188
0
  if (auto NewFd = ::open(Path.c_str(), Flags, 0644); unlikely(NewFd < 0)) {
189
0
    return WasiUnexpect(fromErrNo(errno));
190
0
  } else {
191
0
    INode New(NewFd, true, FdFlags & __WASI_FDFLAGS_APPEND);
192
193
#ifndef O_CLOEXEC
194
    if (auto Res = ::fcntl(New.Fd, F_SETFD, FD_CLOEXEC); unlikely(Res != 0)) {
195
      return WasiUnexpect(fromErrNo(errno));
196
    }
197
#endif
198
0
    return New;
199
0
  }
200
0
}
201
202
WasiExpect<void> INode::fdAdvise(__wasi_filesize_t Offset,
203
                                 __wasi_filesize_t Len,
204
0
                                 __wasi_advice_t Advice) const noexcept {
205
0
  if (auto Res = ::posix_fadvise(Fd, Offset, Len, toAdvice(Advice));
206
0
      unlikely(Res != 0)) {
207
0
    return WasiUnexpect(fromErrNo(errno));
208
0
  }
209
210
0
  return {};
211
0
}
212
213
WasiExpect<void> INode::fdAllocate(__wasi_filesize_t Offset,
214
0
                                   __wasi_filesize_t Len) const noexcept {
215
0
  if (auto Res = ::posix_fallocate(Fd, Offset, Len); unlikely(Res != 0)) {
216
    // https://man7.org/linux/man-pages/man3/posix_fallocate.3.html
217
    // posix_fallocate will not set errno, use return the value directly.
218
0
    return WasiUnexpect(fromErrNo(Res));
219
0
  }
220
221
0
  return {};
222
0
}
223
224
0
WasiExpect<void> INode::fdDatasync() const noexcept {
225
0
  if (auto Res = ::fdatasync(Fd); unlikely(Res != 0)) {
226
0
    return WasiUnexpect(fromErrNo(errno));
227
0
  }
228
229
0
  return {};
230
0
}
231
232
0
WasiExpect<void> INode::fdFdstatGet(__wasi_fdstat_t &FdStat) const noexcept {
233
0
  EXPECTED_TRY(updateStat());
234
235
0
  if (int FdFlags = ::fcntl(Fd, F_GETFL); unlikely(FdFlags < 0)) {
236
0
    return WasiUnexpect(fromErrNo(errno));
237
0
  } else {
238
0
    FdStat.fs_filetype = unsafeFiletype();
239
240
0
    FdStat.fs_flags = static_cast<__wasi_fdflags_t>(0);
241
0
    if (Append) {
242
0
      FdStat.fs_flags |= __WASI_FDFLAGS_APPEND;
243
0
    }
244
0
    if (FdFlags & O_DSYNC) {
245
0
      FdStat.fs_flags |= __WASI_FDFLAGS_DSYNC;
246
0
    }
247
0
    if (FdFlags & O_NONBLOCK) {
248
0
      FdStat.fs_flags |= __WASI_FDFLAGS_NONBLOCK;
249
0
    }
250
0
    if (FdFlags & O_SYNC) {
251
0
      FdStat.fs_flags |= __WASI_FDFLAGS_RSYNC | __WASI_FDFLAGS_SYNC;
252
0
    }
253
0
  }
254
255
0
  return {};
256
0
}
257
258
WasiExpect<void>
259
0
INode::fdFdstatSetFlags(__wasi_fdflags_t FdFlags) const noexcept {
260
0
  int SysFlag = 0;
261
0
  if (FdFlags & __WASI_FDFLAGS_NONBLOCK) {
262
0
    SysFlag |= O_NONBLOCK;
263
0
  }
264
0
  if (FdFlags & __WASI_FDFLAGS_DSYNC) {
265
0
    SysFlag |= O_DSYNC;
266
0
  }
267
0
  if (FdFlags & __WASI_FDFLAGS_RSYNC) {
268
0
    SysFlag |= O_RSYNC;
269
0
  }
270
0
  if (FdFlags & __WASI_FDFLAGS_SYNC) {
271
0
    SysFlag |= O_SYNC;
272
0
  }
273
274
0
  if (auto Res = ::fcntl(Fd, F_SETFL, SysFlag); unlikely(Res != 0)) {
275
0
    return WasiUnexpect(fromErrNo(errno));
276
0
  }
277
278
0
  Append = FdFlags & __WASI_FDFLAGS_APPEND;
279
0
  return {};
280
0
}
281
282
WasiExpect<void>
283
0
INode::fdFilestatGet(__wasi_filestat_t &Filestat) const noexcept {
284
0
  EXPECTED_TRY(updateStat());
285
286
  // Zeroing out these values to prevent leaking information about the host
287
  // environment from special fd such as stdin, stdout and stderr.
288
0
  Filestat.dev = isSpecialFd(Fd) ? 0 : Stat->st_dev;
289
0
  Filestat.ino = isSpecialFd(Fd) ? 0 : Stat->st_ino;
290
0
  Filestat.filetype = unsafeFiletype();
291
0
  Filestat.nlink = isSpecialFd(Fd) ? 0 : Stat->st_nlink;
292
0
  Filestat.size = isSpecialFd(Fd) ? 0 : Stat->st_size;
293
0
  Filestat.atim = isSpecialFd(Fd) ? 0 : fromTimespec(Stat->st_atim);
294
0
  Filestat.mtim = isSpecialFd(Fd) ? 0 : fromTimespec(Stat->st_mtim);
295
0
  Filestat.ctim = isSpecialFd(Fd) ? 0 : fromTimespec(Stat->st_ctim);
296
297
0
  return {};
298
0
}
299
300
WasiExpect<void>
301
0
INode::fdFilestatSetSize(__wasi_filesize_t Size) const noexcept {
302
0
  if (auto Res = ::ftruncate(Fd, Size); unlikely(Res == -1)) {
303
0
    return WasiUnexpect(fromErrNo(errno));
304
0
  }
305
306
0
  return {};
307
0
}
308
309
WasiExpect<void>
310
INode::fdFilestatSetTimes(__wasi_timestamp_t ATim, __wasi_timestamp_t MTim,
311
0
                          __wasi_fstflags_t FstFlags) const noexcept {
312
0
#if __GLIBC_PREREQ(2, 6) || __BIONIC__
313
0
  timespec SysTimespec[2];
314
0
  if (FstFlags & __WASI_FSTFLAGS_ATIM) {
315
0
    SysTimespec[0] = toTimespec(ATim);
316
0
  } else if (FstFlags & __WASI_FSTFLAGS_ATIM_NOW) {
317
0
    SysTimespec[0].tv_nsec = UTIME_NOW;
318
0
  } else {
319
0
    SysTimespec[0].tv_nsec = UTIME_OMIT;
320
0
  }
321
0
  if (FstFlags & __WASI_FSTFLAGS_MTIM) {
322
0
    SysTimespec[1] = toTimespec(MTim);
323
0
  } else if (FstFlags & __WASI_FSTFLAGS_MTIM_NOW) {
324
0
    SysTimespec[1].tv_nsec = UTIME_NOW;
325
0
  } else {
326
0
    SysTimespec[1].tv_nsec = UTIME_OMIT;
327
0
  }
328
329
0
  if (auto Res = ::futimens(Fd, SysTimespec); unlikely(Res != 0)) {
330
0
    return WasiUnexpect(fromErrNo(errno));
331
0
  }
332
#else
333
  bool NeedNow = false;
334
  bool NeedFile = false;
335
  if (FstFlags & __WASI_FSTFLAGS_ATIM) {
336
    // Nothing to do.
337
  } else if (FstFlags & __WASI_FSTFLAGS_ATIM_NOW) {
338
    NeedNow = true;
339
  } else {
340
    NeedFile = true;
341
  }
342
  if (FstFlags & __WASI_FSTFLAGS_MTIM) {
343
    // Nothing to do.
344
  } else if (FstFlags & __WASI_FSTFLAGS_MTIM_NOW) {
345
    NeedNow = true;
346
  } else {
347
    NeedFile = true;
348
  }
349
350
  if (NeedFile) {
351
    EXPECTED_TRY(updateStat());
352
  }
353
354
  timespec Now;
355
  if (NeedNow) {
356
    if (auto Res = ::clock_gettime(CLOCK_REALTIME, &Now); unlikely(Res != 0)) {
357
      return WasiUnexpect(fromErrNo(errno));
358
    }
359
  }
360
361
  timeval SysTimeval[2];
362
  if (FstFlags & __WASI_FSTFLAGS_ATIM) {
363
    SysTimeval[0] = toTimeval(ATim);
364
  } else if (FstFlags & __WASI_FSTFLAGS_ATIM_NOW) {
365
    SysTimeval[0] = toTimeval(Now);
366
  } else {
367
    SysTimeval[0] = toTimeval(Stat->st_atim);
368
  }
369
  if (FstFlags & __WASI_FSTFLAGS_MTIM) {
370
    SysTimeval[1] = toTimeval(MTim);
371
  } else if (FstFlags & __WASI_FSTFLAGS_MTIM_NOW) {
372
    SysTimeval[1] = toTimeval(Now);
373
  } else {
374
    SysTimeval[1] = toTimeval(Stat->st_mtim);
375
  }
376
377
  if (auto Res = ::futimes(Fd, SysTimeval); unlikely(Res != 0)) {
378
    return WasiUnexpect(fromErrNo(errno));
379
  }
380
#endif
381
382
0
  return {};
383
0
}
384
385
WasiExpect<void> INode::fdPread(Span<Span<uint8_t>> IOVs,
386
                                __wasi_filesize_t Offset,
387
0
                                __wasi_size_t &NRead) const noexcept {
388
0
  iovec SysIOVs[kIOVMax];
389
0
  size_t SysIOVsSize = 0;
390
0
  for (auto &IOV : IOVs) {
391
0
    SysIOVs[SysIOVsSize].iov_base = IOV.data();
392
0
    SysIOVs[SysIOVsSize].iov_len = IOV.size();
393
0
    ++SysIOVsSize;
394
0
  }
395
396
0
#if __GLIBC_PREREQ(2, 10)
397
  // Store read bytes length.
398
0
  if (auto Res = ::preadv(Fd, SysIOVs, SysIOVsSize, Offset);
399
0
      unlikely(Res < 0)) {
400
0
    return WasiUnexpect(fromErrNo(errno));
401
0
  } else {
402
0
    NRead = Res;
403
0
  }
404
#else
405
  const auto OldOffset = ::lseek(Fd, 0, SEEK_CUR);
406
  if (OldOffset < 0) {
407
    return WasiUnexpect(fromErrNo(errno));
408
  }
409
  if (::lseek(Fd, Offset, SEEK_SET) < 0 ||
410
      ::lseek(Fd, OldOffset, SEEK_SET) < 0) {
411
    return WasiUnexpect(fromErrNo(errno));
412
  }
413
  if (auto Res = ::readv(Fd, SysIOVs, SysIOVsSize); unlikely(Res < 0)) {
414
    ::lseek(Fd, OldOffset, SEEK_SET);
415
    return WasiUnexpect(fromErrNo(errno));
416
  } else {
417
    if (::lseek(Fd, OldOffset, SEEK_SET) < 0) {
418
      return WasiUnexpect(fromErrNo(errno));
419
    }
420
    NRead = Res;
421
  }
422
#endif
423
424
0
  return {};
425
0
}
426
427
WasiExpect<void> INode::fdPwrite(Span<Span<const uint8_t>> IOVs,
428
                                 __wasi_filesize_t Offset,
429
0
                                 __wasi_size_t &NWritten) const noexcept {
430
0
  iovec SysIOVs[kIOVMax];
431
0
  size_t SysIOVsSize = 0;
432
0
  for (auto &IOV : IOVs) {
433
0
    SysIOVs[SysIOVsSize].iov_base = const_cast<uint8_t *>(IOV.data());
434
0
    SysIOVs[SysIOVsSize].iov_len = IOV.size();
435
0
    ++SysIOVsSize;
436
0
  }
437
438
0
#if __GLIBC_PREREQ(2, 10)
439
0
  if (auto Res = ::pwritev(Fd, SysIOVs, SysIOVsSize, Offset);
440
0
      unlikely(Res < 0)) {
441
0
    return WasiUnexpect(fromErrNo(errno));
442
0
  } else {
443
0
    NWritten = Res;
444
0
  }
445
#else
446
  const auto OldOffset = ::lseek(Fd, 0, SEEK_CUR);
447
  if (OldOffset < 0) {
448
    return WasiUnexpect(fromErrNo(errno));
449
  }
450
  if (::lseek(Fd, Offset, SEEK_SET) < 0 ||
451
      ::lseek(Fd, OldOffset, SEEK_SET) < 0) {
452
    return WasiUnexpect(fromErrNo(errno));
453
  }
454
  if (auto Res = ::writev(Fd, SysIOVs, SysIOVsSize); unlikely(Res < 0)) {
455
    ::lseek(Fd, OldOffset, SEEK_SET);
456
    return WasiUnexpect(fromErrNo(errno));
457
  } else {
458
    if (::lseek(Fd, OldOffset, SEEK_SET) < 0) {
459
      return WasiUnexpect(fromErrNo(errno));
460
    }
461
    NWritten = Res;
462
  }
463
#endif
464
465
0
  return {};
466
0
}
467
468
WasiExpect<void> INode::fdRead(Span<Span<uint8_t>> IOVs,
469
0
                               __wasi_size_t &NRead) const noexcept {
470
0
  iovec SysIOVs[kIOVMax];
471
0
  size_t SysIOVsSize = 0;
472
0
  for (auto &IOV : IOVs) {
473
0
    SysIOVs[SysIOVsSize].iov_base = IOV.data();
474
0
    SysIOVs[SysIOVsSize].iov_len = IOV.size();
475
0
    ++SysIOVsSize;
476
0
  }
477
478
0
  if (auto Res = ::readv(Fd, SysIOVs, SysIOVsSize); unlikely(Res < 0)) {
479
0
    return WasiUnexpect(fromErrNo(errno));
480
0
  } else {
481
0
    NRead = Res;
482
0
  }
483
484
0
  return {};
485
0
}
486
487
// Due to the unfortunate design of wasi::fd_readdir, It's nearly impossible to
488
// provide a correct implementation. The below implementation is just a
489
// workaround for most usages and may not be correct in some edge cases. The
490
// readdir entry API is going to be updated to use a stream type, so we don't
491
// have to deal with it right now.
492
WasiExpect<void> INode::fdReaddir(Span<uint8_t> Buffer,
493
                                  __wasi_dircookie_t Cookie,
494
0
                                  __wasi_size_t &Size) noexcept {
495
0
  if (unlikely(!Dir.ok())) {
496
0
    if (FdHolder NewFd(::dup(Fd)); unlikely(!NewFd.ok())) {
497
0
      return WasiUnexpect(fromErrNo(errno));
498
0
    } else if (DIR *D = ::fdopendir(NewFd.Fd); unlikely(!D)) {
499
0
      return WasiUnexpect(fromErrNo(errno));
500
0
    } else {
501
0
      NewFd.release();
502
0
      Dir.emplace(D);
503
0
    }
504
0
  }
505
506
0
  if (Cookie == 0) {
507
0
    ::rewinddir(Dir.Dir);
508
0
  } else if (unlikely(Cookie != Dir.Cookie)) {
509
0
    ::seekdir(Dir.Dir, Cookie);
510
0
  }
511
512
0
  Size = 0;
513
0
  do {
514
0
    if (!Dir.Buffer.empty()) {
515
0
      const auto NewDataSize =
516
0
          std::min<uint32_t>(Buffer.size(), Dir.Buffer.size());
517
0
      std::copy(Dir.Buffer.begin(), Dir.Buffer.begin() + NewDataSize,
518
0
                Buffer.begin());
519
0
      Buffer = Buffer.subspan(NewDataSize);
520
0
      Size += NewDataSize;
521
0
      Dir.Buffer.clear();
522
0
      if (unlikely(Buffer.empty())) {
523
0
        break;
524
0
      }
525
0
    }
526
0
    errno = 0;
527
0
    dirent *SysDirent = ::readdir(Dir.Dir);
528
0
    if (SysDirent == nullptr) {
529
0
      if (errno != 0) {
530
0
        return WasiUnexpect(fromErrNo(errno));
531
0
      }
532
      // End of entries
533
0
      break;
534
0
    }
535
0
    Dir.Cookie = SysDirent->d_off;
536
0
    std::string_view Name = SysDirent->d_name;
537
538
0
    Dir.Buffer.resize(sizeof(__wasi_dirent_t) + Name.size());
539
540
0
    __wasi_dirent_t *const Dirent =
541
0
        reinterpret_cast<__wasi_dirent_t *>(Dir.Buffer.data());
542
0
    Dirent->d_next = Dir.Cookie;
543
0
    Dirent->d_ino = SysDirent->d_ino;
544
0
    Dirent->d_type = fromFileType(SysDirent->d_type);
545
0
    Dirent->d_namlen = Name.size();
546
0
    std::copy(Name.cbegin(), Name.cend(),
547
0
              Dir.Buffer.begin() + sizeof(__wasi_dirent_t));
548
0
  } while (!Buffer.empty());
549
550
0
  return {};
551
0
}
552
553
WasiExpect<void> INode::fdSeek(__wasi_filedelta_t Offset,
554
                               __wasi_whence_t Whence,
555
0
                               __wasi_filesize_t &Size) const noexcept {
556
0
  if (auto Res = ::lseek(Fd, Offset, toWhence(Whence)); unlikely(Res < 0)) {
557
0
    return WasiUnexpect(fromErrNo(errno));
558
0
  } else {
559
0
    Size = Res;
560
0
  }
561
562
0
  return {};
563
0
}
564
565
0
WasiExpect<void> INode::fdSync() const noexcept {
566
0
  if (auto Res = ::fsync(Fd); unlikely(Res != 0)) {
567
0
    return WasiUnexpect(fromErrNo(errno));
568
0
  }
569
570
0
  return {};
571
0
}
572
573
0
WasiExpect<void> INode::fdTell(__wasi_filesize_t &Size) const noexcept {
574
0
  if (auto Res = ::lseek(Fd, 0, SEEK_CUR); unlikely(Res < 0)) {
575
0
    return WasiUnexpect(fromErrNo(errno));
576
0
  } else {
577
0
    Size = Res;
578
0
  }
579
580
0
  return {};
581
0
}
582
583
WasiExpect<void> INode::fdWrite(Span<Span<const uint8_t>> IOVs,
584
0
                                __wasi_size_t &NWritten) const noexcept {
585
0
  iovec SysIOVs[kIOVMax];
586
0
  size_t SysIOVsSize = 0;
587
0
  for (auto &IOV : IOVs) {
588
0
    SysIOVs[SysIOVsSize].iov_base = const_cast<uint8_t *>(IOV.data());
589
0
    SysIOVs[SysIOVsSize].iov_len = IOV.size();
590
0
    ++SysIOVsSize;
591
0
  }
592
593
0
  if (Append) {
594
0
    ::lseek(Fd, 0, SEEK_END);
595
0
  }
596
597
0
  if (auto Res = ::writev(Fd, SysIOVs, SysIOVsSize); unlikely(Res < 0)) {
598
0
    return WasiUnexpect(fromErrNo(errno));
599
0
  } else {
600
0
    NWritten = Res;
601
0
  }
602
603
0
  return {};
604
0
}
605
606
0
WasiExpect<uint64_t> INode::getNativeHandler() const noexcept {
607
0
  return static_cast<uint64_t>(Fd);
608
0
}
609
610
0
WasiExpect<void> INode::pathCreateDirectory(std::string Path) const noexcept {
611
0
  if (auto Res = ::mkdirat(Fd, Path.c_str(), 0755); unlikely(Res != 0)) {
612
0
    return WasiUnexpect(fromErrNo(errno));
613
0
  }
614
615
0
  return {};
616
0
}
617
618
WasiExpect<void>
619
INode::pathFilestatGet(std::string Path,
620
0
                       __wasi_filestat_t &Filestat) const noexcept {
621
0
  struct stat SysFStat;
622
0
  if (int Res = ::fstatat(Fd, Path.c_str(), &SysFStat, AT_SYMLINK_NOFOLLOW);
623
0
      unlikely(Res != 0)) {
624
0
    return WasiUnexpect(fromErrNo(errno));
625
0
  }
626
627
0
  Filestat.dev = SysFStat.st_dev;
628
0
  Filestat.ino = SysFStat.st_ino;
629
0
  Filestat.filetype = fromFileType(static_cast<mode_t>(SysFStat.st_mode));
630
0
  Filestat.nlink = SysFStat.st_nlink;
631
0
  Filestat.size = SysFStat.st_size;
632
0
  Filestat.atim = fromTimespec(SysFStat.st_atim);
633
0
  Filestat.mtim = fromTimespec(SysFStat.st_mtim);
634
0
  Filestat.ctim = fromTimespec(SysFStat.st_ctim);
635
636
0
  return {};
637
0
}
638
639
WasiExpect<void>
640
INode::pathFilestatSetTimes(std::string Path, __wasi_timestamp_t ATim,
641
                            __wasi_timestamp_t MTim,
642
0
                            __wasi_fstflags_t FstFlags) const noexcept {
643
0
#if __GLIBC_PREREQ(2, 6) || __BIONIC__
644
0
  timespec SysTimespec[2];
645
0
  if (FstFlags & __WASI_FSTFLAGS_ATIM) {
646
0
    SysTimespec[0] = toTimespec(ATim);
647
0
  } else if (FstFlags & __WASI_FSTFLAGS_ATIM_NOW) {
648
0
    SysTimespec[0].tv_nsec = UTIME_NOW;
649
0
  } else {
650
0
    SysTimespec[0].tv_nsec = UTIME_OMIT;
651
0
  }
652
0
  if (FstFlags & __WASI_FSTFLAGS_MTIM) {
653
0
    SysTimespec[1] = toTimespec(MTim);
654
0
  } else if (FstFlags & __WASI_FSTFLAGS_MTIM_NOW) {
655
0
    SysTimespec[1].tv_nsec = UTIME_NOW;
656
0
  } else {
657
0
    SysTimespec[1].tv_nsec = UTIME_OMIT;
658
0
  }
659
660
0
  if (auto Res =
661
0
          ::utimensat(Fd, Path.c_str(), SysTimespec, AT_SYMLINK_NOFOLLOW);
662
0
      unlikely(Res != 0)) {
663
0
    return WasiUnexpect(fromErrNo(errno));
664
0
  }
665
#else
666
  bool NeedNow = false;
667
  bool NeedFile = false;
668
  if (FstFlags & __WASI_FSTFLAGS_ATIM) {
669
    // Nothing to do.
670
  } else if (FstFlags & __WASI_FSTFLAGS_ATIM_NOW) {
671
    NeedNow = true;
672
  } else {
673
    NeedFile = true;
674
  }
675
  if (FstFlags & __WASI_FSTFLAGS_MTIM) {
676
    // Nothing to do.
677
  } else if (FstFlags & __WASI_FSTFLAGS_MTIM_NOW) {
678
    NeedNow = true;
679
  } else {
680
    NeedFile = true;
681
  }
682
683
#ifdef O_PATH
684
  const int OFlags = O_PATH | O_SYMLINK;
685
#else
686
  const int OFlags = O_RDONLY | O_SYMLINK;
687
#endif
688
689
  FdHolder Target(::openat(Fd, Path.c_str(), OFlags));
690
  if (unlikely(!Target.ok())) {
691
    return WasiUnexpect(fromErrNo(errno));
692
  }
693
694
  struct stat SysStat;
695
  if (NeedFile) {
696
    if (auto Res = ::fstat(Target.Fd, &SysStat); unlikely(Res != 0)) {
697
      return WasiUnexpect(fromErrNo(errno));
698
    }
699
  }
700
701
  timespec Now;
702
  if (NeedNow) {
703
    if (auto Res = ::clock_gettime(CLOCK_REALTIME, &Now); unlikely(Res != 0)) {
704
      return WasiUnexpect(fromErrNo(errno));
705
    }
706
  }
707
708
  timeval SysTimeval[2];
709
  if (FstFlags & __WASI_FSTFLAGS_ATIM) {
710
    SysTimeval[0] = toTimeval(ATim);
711
  } else if (FstFlags & __WASI_FSTFLAGS_ATIM_NOW) {
712
    SysTimeval[0] = toTimeval(Now);
713
  } else {
714
    SysTimeval[0] = toTimeval(SysStat.st_atim);
715
  }
716
  if (FstFlags & __WASI_FSTFLAGS_MTIM) {
717
    SysTimeval[1] = toTimeval(MTim);
718
  } else if (FstFlags & __WASI_FSTFLAGS_MTIM_NOW) {
719
    SysTimeval[1] = toTimeval(Now);
720
  } else {
721
    SysTimeval[1] = toTimeval(SysStat.st_mtim);
722
  }
723
724
  if (auto Res = ::futimes(Target.Fd, SysTimeval); unlikely(Res != 0)) {
725
    return WasiUnexpect(fromErrNo(errno));
726
  }
727
#endif
728
729
0
  return {};
730
0
}
731
732
WasiExpect<void> INode::pathLink(const INode &Old, std::string OldPath,
733
                                 const INode &New,
734
0
                                 std::string NewPath) noexcept {
735
0
  if (auto Res = ::linkat(Old.Fd, OldPath.c_str(), New.Fd, NewPath.c_str(), 0);
736
0
      unlikely(Res != 0)) {
737
0
    return WasiUnexpect(fromErrNo(errno));
738
0
  }
739
740
0
  return {};
741
0
}
742
743
WasiExpect<INode> INode::pathOpen(std::string Path, __wasi_oflags_t OpenFlags,
744
                                  __wasi_fdflags_t FdFlags,
745
0
                                  VFS::Flags VFSFlags) const noexcept {
746
0
  const int Flags = openFlags(OpenFlags, FdFlags, VFSFlags);
747
748
0
  if (auto NewFd = ::openat(Fd, Path.c_str(), Flags, 0644);
749
0
      unlikely(NewFd < 0)) {
750
0
    return WasiUnexpect(fromErrNo(errno));
751
0
  } else {
752
0
    INode New(NewFd, true, FdFlags & __WASI_FDFLAGS_APPEND);
753
754
#ifndef O_CLOEXEC
755
    if (auto Res = ::fcntl(New.Fd, F_SETFD, FD_CLOEXEC); unlikely(Res != 0)) {
756
      return WasiUnexpect(fromErrNo(errno));
757
    }
758
#endif
759
0
    return New;
760
0
  }
761
0
}
762
763
WasiExpect<void> INode::pathReadlink(std::string Path, Span<char> Buffer,
764
0
                                     __wasi_size_t &NRead) const noexcept {
765
0
  if (auto Res = ::readlinkat(Fd, Path.c_str(), Buffer.data(), Buffer.size());
766
0
      unlikely(Res < 0)) {
767
0
    return WasiUnexpect(fromErrNo(errno));
768
0
  } else {
769
0
    NRead = Res;
770
0
  }
771
772
0
  return {};
773
0
}
774
775
0
WasiExpect<void> INode::pathRemoveDirectory(std::string Path) const noexcept {
776
0
  if (auto Res = ::unlinkat(Fd, Path.c_str(), AT_REMOVEDIR);
777
0
      unlikely(Res < 0)) {
778
0
    return WasiUnexpect(fromErrNo(errno));
779
0
  }
780
781
0
  return {};
782
0
}
783
784
WasiExpect<void> INode::pathRename(const INode &Old, std::string OldPath,
785
                                   const INode &New,
786
0
                                   std::string NewPath) noexcept {
787
0
  if (auto Res = ::renameat(Old.Fd, OldPath.c_str(), New.Fd, NewPath.c_str());
788
0
      unlikely(Res != 0)) {
789
0
    return WasiUnexpect(fromErrNo(errno));
790
0
  }
791
792
0
  return {};
793
0
}
794
795
WasiExpect<void> INode::pathSymlink(std::string OldPath,
796
0
                                    std::string NewPath) const noexcept {
797
0
  if (auto Res = ::symlinkat(OldPath.c_str(), Fd, NewPath.c_str());
798
0
      unlikely(Res != 0)) {
799
0
    return WasiUnexpect(fromErrNo(errno));
800
0
  }
801
802
0
  return {};
803
0
}
804
805
0
WasiExpect<void> INode::pathUnlinkFile(std::string Path) const noexcept {
806
0
  if (auto Res = ::unlinkat(Fd, Path.c_str(), 0); unlikely(Res < 0)) {
807
0
    return WasiUnexpect(fromErrNo(errno));
808
0
  }
809
810
0
  return {};
811
0
}
812
813
WasiExpect<void> INode::getAddrinfo(std::string_view Node,
814
                                    std::string_view Service,
815
                                    const __wasi_addrinfo_t &Hint,
816
                                    uint32_t MaxResLength,
817
                                    Span<__wasi_addrinfo_t *> WasiAddrinfoArray,
818
                                    Span<__wasi_sockaddr_t *> WasiSockaddrArray,
819
                                    Span<char *> AiAddrSaDataArray,
820
                                    Span<char *> AiCanonnameArray,
821
0
                                    /*Out*/ __wasi_size_t &ResLength) noexcept {
822
0
  const auto [NodeCStr, NodeBuf] = createNullTerminatedString(Node);
823
0
  const auto [ServiceCStr, ServiceBuf] = createNullTerminatedString(Service);
824
825
0
  struct addrinfo SysHint;
826
0
  SysHint.ai_flags = toAIFlags(Hint.ai_flags);
827
0
  SysHint.ai_family = toAddressFamily(Hint.ai_family);
828
0
  SysHint.ai_socktype = toSockType(Hint.ai_socktype);
829
0
  SysHint.ai_protocol = toProtocol(Hint.ai_protocol);
830
0
  SysHint.ai_addrlen = Hint.ai_addrlen;
831
0
  SysHint.ai_addr = nullptr;
832
0
  SysHint.ai_canonname = nullptr;
833
0
  SysHint.ai_next = nullptr;
834
835
0
  struct addrinfo *SysResPtr = nullptr;
836
0
  if (auto Res = ::getaddrinfo(NodeCStr, ServiceCStr, &SysHint, &SysResPtr);
837
0
      unlikely(Res < 0)) {
838
0
    return WasiUnexpect(fromEAIErrNo(Res));
839
0
  }
840
  // calculate ResLength
841
0
  if (ResLength = calculateAddrinfoLinkedListSize(SysResPtr);
842
0
      ResLength > MaxResLength) {
843
0
    ResLength = MaxResLength;
844
0
  }
845
846
0
  struct addrinfo *SysResItem = SysResPtr;
847
0
  for (uint32_t Idx = 0; Idx < ResLength; Idx++) {
848
0
    auto &CurAddrinfo = WasiAddrinfoArray[Idx];
849
0
    CurAddrinfo->ai_flags = fromAIFlags(SysResItem->ai_flags);
850
0
    CurAddrinfo->ai_socktype = fromSockType(SysResItem->ai_socktype);
851
0
    CurAddrinfo->ai_protocol = fromProtocol(SysResItem->ai_protocol);
852
0
    CurAddrinfo->ai_family = fromAddressFamily(SysResItem->ai_family);
853
0
    CurAddrinfo->ai_addrlen = SysResItem->ai_addrlen;
854
855
    // process ai_canonname in addrinfo
856
0
    if (SysResItem->ai_canonname != nullptr) {
857
0
      CurAddrinfo->ai_canonname_len = std::strlen(SysResItem->ai_canonname);
858
0
      auto &CurAiCanonname = AiCanonnameArray[Idx];
859
0
      std::memcpy(CurAiCanonname, SysResItem->ai_canonname,
860
0
                  CurAddrinfo->ai_canonname_len + 1);
861
0
    } else {
862
0
      CurAddrinfo->ai_canonname_len = 0;
863
0
    }
864
865
    // process socket address
866
0
    if (SysResItem->ai_addrlen > 0) {
867
0
      auto &CurSockaddr = WasiSockaddrArray[Idx];
868
0
      CurSockaddr->sa_family =
869
0
          fromAddressFamily(SysResItem->ai_addr->sa_family);
870
871
      // process sa_data in socket address
872
0
      size_t SaSize = 0;
873
0
      switch (CurSockaddr->sa_family) {
874
0
      case __WASI_ADDRESS_FAMILY_INET4:
875
0
        SaSize = sizeof(sockaddr_in) - offsetof(sockaddr_in, sin_port);
876
0
        break;
877
0
      case __WASI_ADDRESS_FAMILY_INET6:
878
0
        SaSize = sizeof(sockaddr_in6) - offsetof(sockaddr_in6, sin6_port);
879
0
        break;
880
0
      default:
881
0
        assumingUnreachable();
882
0
      }
883
0
      std::memcpy(AiAddrSaDataArray[Idx], SysResItem->ai_addr->sa_data, SaSize);
884
0
      CurSockaddr->sa_data_len = __wasi_size_t(SaSize);
885
0
    }
886
    // process ai_next in addrinfo
887
0
    SysResItem = SysResItem->ai_next;
888
0
  }
889
0
  ::freeaddrinfo(SysResPtr);
890
891
0
  return {};
892
0
}
893
894
WasiExpect<INode> INode::sockOpen(__wasi_address_family_t AddressFamily,
895
0
                                  __wasi_sock_type_t SockType) noexcept {
896
0
  int SysProtocol = IPPROTO_IP;
897
0
  int SysDomain = 0;
898
0
  int SysType = 0;
899
900
0
  switch (AddressFamily) {
901
0
  case __WASI_ADDRESS_FAMILY_INET4:
902
0
    SysDomain = AF_INET;
903
0
    break;
904
0
  case __WASI_ADDRESS_FAMILY_INET6:
905
0
    SysDomain = AF_INET6;
906
0
    break;
907
0
  case __WASI_ADDRESS_FAMILY_AF_UNIX:
908
0
    SysDomain = AF_UNIX;
909
0
    break;
910
0
  default:
911
0
    return WasiUnexpect(__WASI_ERRNO_INVAL);
912
0
  }
913
914
0
  switch (SockType) {
915
0
  case __WASI_SOCK_TYPE_SOCK_DGRAM:
916
0
    SysType = SOCK_DGRAM;
917
0
    break;
918
0
  case __WASI_SOCK_TYPE_SOCK_STREAM:
919
0
    SysType = SOCK_STREAM;
920
0
    break;
921
0
  default:
922
0
    return WasiUnexpect(__WASI_ERRNO_INVAL);
923
0
  }
924
925
0
  if (auto NewFd = ::socket(SysDomain, SysType, SysProtocol);
926
0
      unlikely(NewFd < 0)) {
927
0
    return WasiUnexpect(fromErrNo(errno));
928
0
  } else {
929
0
    INode New(NewFd);
930
0
    return New;
931
0
  }
932
0
}
933
934
struct SockEmptyAddr {};
935
using VarAddrT = std::variant<SockEmptyAddr, sockaddr_storage, sockaddr,
936
                              sockaddr_in, sockaddr_in6, sockaddr_un>;
937
938
struct VarAddrBuf {
939
0
  template <typename T> sockaddr *operator()(T &V) {
940
0
    return reinterpret_cast<struct sockaddr *>(&V);
941
0
  }
Unexecuted instantiation: sockaddr* WasmEdge::Host::WASI::VarAddrBuf::operator()<sockaddr_storage>(sockaddr_storage&)
Unexecuted instantiation: sockaddr* WasmEdge::Host::WASI::VarAddrBuf::operator()<sockaddr>(sockaddr&)
Unexecuted instantiation: sockaddr* WasmEdge::Host::WASI::VarAddrBuf::operator()<sockaddr_in>(sockaddr_in&)
Unexecuted instantiation: sockaddr* WasmEdge::Host::WASI::VarAddrBuf::operator()<sockaddr_in6>(sockaddr_in6&)
Unexecuted instantiation: sockaddr* WasmEdge::Host::WASI::VarAddrBuf::operator()<sockaddr_un>(sockaddr_un&)
942
0
  sockaddr *operator()(SockEmptyAddr &) { return nullptr; }
943
};
944
945
struct VarAddrSize {
946
0
  template <typename T> int operator()(const T &) { return sizeof(T); }
Unexecuted instantiation: int WasmEdge::Host::WASI::VarAddrSize::operator()<sockaddr_storage>(sockaddr_storage const&)
Unexecuted instantiation: int WasmEdge::Host::WASI::VarAddrSize::operator()<sockaddr>(sockaddr const&)
Unexecuted instantiation: int WasmEdge::Host::WASI::VarAddrSize::operator()<sockaddr_in>(sockaddr_in const&)
Unexecuted instantiation: int WasmEdge::Host::WASI::VarAddrSize::operator()<sockaddr_in6>(sockaddr_in6 const&)
Unexecuted instantiation: int WasmEdge::Host::WASI::VarAddrSize::operator()<sockaddr_un>(sockaddr_un const&)
947
0
  int operator()(const SockEmptyAddr &) { return 0; }
948
};
949
950
static VarAddrT sockAddressAssignHelper(__wasi_address_family_t AddrFamily,
951
                                        const Span<const uint8_t> &Address,
952
0
                                        uint16_t Port) {
953
0
  VarAddrT Addr;
954
0
  if (Address.size() == 0) {
955
0
    Addr.emplace<SockEmptyAddr>();
956
0
  } else if (AddrFamily == __WASI_ADDRESS_FAMILY_INET4) {
957
0
    auto &ServerAddr4 = Addr.emplace<sockaddr_in>();
958
959
0
    ServerAddr4.sin_family = AF_INET;
960
0
    ServerAddr4.sin_port = htons(Port);
961
0
    assuming(Address.size() >= sizeof(in_addr));
962
0
    std::memcpy(&ServerAddr4.sin_addr, Address.data(), sizeof(in_addr));
963
0
  } else if (AddrFamily == __WASI_ADDRESS_FAMILY_INET6) {
964
0
    auto &ServerAddr6 = Addr.emplace<sockaddr_in6>();
965
966
0
    ServerAddr6.sin6_family = AF_INET6;
967
0
    ServerAddr6.sin6_port = htons(Port);
968
0
    ServerAddr6.sin6_flowinfo = 0;
969
0
    assuming(Address.size() >= sizeof(in6_addr));
970
0
    std::memcpy(&ServerAddr6.sin6_addr, Address.data(), sizeof(in6_addr));
971
0
  } else if (AddrFamily == __WASI_ADDRESS_FAMILY_AF_UNIX) {
972
0
    auto &ServerAddrUN = Addr.emplace<sockaddr_un>();
973
974
0
    ServerAddrUN.sun_family = AF_UNIX;
975
    // The length of sockaddr_un::sun_path is depend on cruuent system
976
    // We should always check the size of it.
977
0
    assuming(Address.size() >= sizeof(sockaddr_un::sun_path));
978
0
    std::memcpy(&ServerAddrUN.sun_path, Address.data(),
979
0
                sizeof(sockaddr_un::sun_path));
980
0
  } else {
981
0
    assumingUnreachable();
982
0
  }
983
984
0
  return Addr;
985
0
}
986
987
WasiExpect<void> INode::sockBind(__wasi_address_family_t AddressFamily,
988
                                 Span<const uint8_t> Address,
989
0
                                 uint16_t Port) noexcept {
990
0
  auto AddressBuffer = sockAddressAssignHelper(AddressFamily, Address, Port);
991
992
0
  auto ServerAddr = std::visit(VarAddrBuf(), AddressBuffer);
993
0
  int Size = std::visit(VarAddrSize(), AddressBuffer);
994
995
0
  if (auto Res = ::bind(Fd, ServerAddr, Size); unlikely(Res < 0)) {
996
0
    return WasiUnexpect(fromErrNo(errno));
997
0
  }
998
0
  return {};
999
0
}
1000
1001
0
WasiExpect<void> INode::sockListen(int32_t Backlog) noexcept {
1002
0
  if (auto Res = ::listen(Fd, Backlog); unlikely(Res < 0)) {
1003
0
    return WasiUnexpect(fromErrNo(errno));
1004
0
  }
1005
0
  return {};
1006
0
}
1007
1008
0
WasiExpect<INode> INode::sockAccept(__wasi_fdflags_t FdFlags) noexcept {
1009
0
  int NewFd;
1010
0
  if (NewFd = ::accept(Fd, nullptr, nullptr); unlikely(NewFd < 0)) {
1011
0
    return WasiUnexpect(fromErrNo(errno));
1012
0
  }
1013
1014
0
  INode New(NewFd);
1015
1016
0
  if (FdFlags & __WASI_FDFLAGS_NONBLOCK) {
1017
0
    int SysFlag = fcntl(NewFd, F_GETFL, 0);
1018
0
    SysFlag |= O_NONBLOCK;
1019
0
    if (auto Res = ::fcntl(Fd, F_SETFL, SysFlag); unlikely(Res != 0)) {
1020
0
      return WasiUnexpect(fromErrNo(errno));
1021
0
    }
1022
0
  }
1023
1024
0
  return New;
1025
0
}
1026
1027
WasiExpect<void> INode::sockConnect(__wasi_address_family_t AddressFamily,
1028
                                    Span<const uint8_t> Address,
1029
0
                                    uint16_t Port) noexcept {
1030
0
  auto AddressBuffer = sockAddressAssignHelper(AddressFamily, Address, Port);
1031
1032
0
  auto ClientAddr = std::visit(VarAddrBuf(), AddressBuffer);
1033
0
  int Size = std::visit(VarAddrSize(), AddressBuffer);
1034
1035
0
  if (auto Res = ::connect(Fd, ClientAddr, Size); unlikely(Res < 0)) {
1036
0
    return WasiUnexpect(fromErrNo(errno));
1037
0
  }
1038
1039
0
  return {};
1040
0
}
1041
1042
WasiExpect<void> INode::sockRecv(Span<Span<uint8_t>> RiData,
1043
                                 __wasi_riflags_t RiFlags, __wasi_size_t &NRead,
1044
0
                                 __wasi_roflags_t &RoFlags) const noexcept {
1045
0
  return sockRecvFrom(RiData, RiFlags, nullptr, {}, nullptr, NRead, RoFlags);
1046
0
}
1047
1048
WasiExpect<void> INode::sockRecvFrom(Span<Span<uint8_t>> RiData,
1049
                                     __wasi_riflags_t RiFlags,
1050
                                     __wasi_address_family_t *AddressFamilyPtr,
1051
                                     Span<uint8_t> Address, uint16_t *PortPtr,
1052
                                     __wasi_size_t &NRead,
1053
0
                                     __wasi_roflags_t &RoFlags) const noexcept {
1054
0
  int SysRiFlags = 0;
1055
0
  if (RiFlags & __WASI_RIFLAGS_RECV_PEEK) {
1056
0
    SysRiFlags |= MSG_PEEK;
1057
0
  }
1058
0
  if (RiFlags & __WASI_RIFLAGS_RECV_WAITALL) {
1059
0
    SysRiFlags |= MSG_WAITALL;
1060
0
  }
1061
1062
0
  iovec SysIOVs[kIOVMax];
1063
0
  size_t SysIOVsSize = 0;
1064
0
  for (auto &IOV : RiData) {
1065
0
    SysIOVs[SysIOVsSize].iov_base = IOV.data();
1066
0
    SysIOVs[SysIOVsSize].iov_len = IOV.size();
1067
0
    ++SysIOVsSize;
1068
0
  }
1069
1070
0
  const bool NeedAddress =
1071
0
      AddressFamilyPtr != nullptr || !Address.empty() || PortPtr != nullptr;
1072
0
  sockaddr_storage SockAddr = {};
1073
0
  msghdr SysMsgHdr;
1074
0
  if (NeedAddress) {
1075
0
    SysMsgHdr.msg_name = &SockAddr;
1076
0
    SysMsgHdr.msg_namelen = sizeof(SockAddr);
1077
0
  } else {
1078
0
    SysMsgHdr.msg_name = nullptr;
1079
0
    SysMsgHdr.msg_namelen = 0;
1080
0
  }
1081
0
  SysMsgHdr.msg_iov = SysIOVs;
1082
0
  SysMsgHdr.msg_iovlen = SysIOVsSize;
1083
0
  SysMsgHdr.msg_control = nullptr;
1084
0
  SysMsgHdr.msg_controllen = 0;
1085
0
  SysMsgHdr.msg_flags = 0;
1086
1087
  // Store recv bytes length and flags.
1088
0
  if (auto Res = ::recvmsg(Fd, &SysMsgHdr, SysRiFlags); unlikely(Res < 0)) {
1089
0
    return WasiUnexpect(fromErrNo(errno));
1090
0
  } else {
1091
0
    NRead = Res;
1092
0
  }
1093
1094
0
  if (NeedAddress) {
1095
0
    switch (SockAddr.ss_family) {
1096
0
    case AF_UNSPEC: {
1097
0
      spdlog::warn("remote address unavailable"sv);
1098
      // if ss_family is AF_UNSPEC, the access of the other members are
1099
      // undefined.
1100
0
      break;
1101
0
    }
1102
0
    case AF_INET: {
1103
0
      const auto &SockAddr4 = reinterpret_cast<sockaddr_in &>(SockAddr);
1104
0
      if (AddressFamilyPtr) {
1105
0
        *AddressFamilyPtr = __WASI_ADDRESS_FAMILY_INET4;
1106
0
      }
1107
0
      if (Address.size() >= sizeof(in_addr)) {
1108
0
        std::memcpy(Address.data(), &SockAddr4.sin_addr, sizeof(in_addr));
1109
0
      }
1110
0
      if (PortPtr != nullptr) {
1111
0
        *PortPtr = SockAddr4.sin_port;
1112
0
      }
1113
0
      break;
1114
0
    }
1115
0
    case AF_INET6: {
1116
0
      const auto &SockAddr6 = reinterpret_cast<sockaddr_in6 &>(SockAddr);
1117
0
      if (AddressFamilyPtr) {
1118
0
        *AddressFamilyPtr = __WASI_ADDRESS_FAMILY_INET6;
1119
0
      }
1120
0
      if (Address.size() >= sizeof(in6_addr)) {
1121
0
        std::memcpy(Address.data(), &SockAddr6.sin6_addr, sizeof(in6_addr));
1122
0
      }
1123
0
      if (PortPtr != nullptr) {
1124
0
        *PortPtr = SockAddr6.sin6_port;
1125
0
      }
1126
0
      break;
1127
0
    }
1128
0
    case AF_UNIX: {
1129
0
      const auto &SockAddrUN = reinterpret_cast<sockaddr_un &>(SockAddr);
1130
0
      if (AddressFamilyPtr) {
1131
0
        *AddressFamilyPtr = __WASI_ADDRESS_FAMILY_AF_UNIX;
1132
0
      }
1133
0
      if (Address.size() >= sizeof(sockaddr_un::sun_path)) {
1134
0
        std::memcpy(Address.data(), &SockAddrUN.sun_path,
1135
0
                    sizeof(sockaddr_un::sun_path));
1136
0
      } else {
1137
0
        return WasiUnexpect(__WASI_ERRNO_INVAL);
1138
0
      }
1139
0
      break;
1140
0
    }
1141
0
    default:
1142
0
      return WasiUnexpect(__WASI_ERRNO_NOSYS);
1143
0
    }
1144
0
  }
1145
1146
0
  RoFlags = static_cast<__wasi_roflags_t>(0);
1147
0
  if (SysMsgHdr.msg_flags & MSG_TRUNC) {
1148
0
    RoFlags |= __WASI_ROFLAGS_RECV_DATA_TRUNCATED;
1149
0
  }
1150
1151
0
  return {};
1152
0
}
1153
1154
WasiExpect<void> INode::sockSend(Span<Span<const uint8_t>> SiData,
1155
                                 __wasi_siflags_t SiFlags,
1156
0
                                 __wasi_size_t &NWritten) const noexcept {
1157
0
  return sockSendTo(SiData, SiFlags, __WASI_ADDRESS_FAMILY_UNSPEC, {}, 0,
1158
0
                    NWritten);
1159
0
}
1160
1161
WasiExpect<void> INode::sockSendTo(Span<Span<const uint8_t>> SiData,
1162
                                   __wasi_siflags_t,
1163
                                   __wasi_address_family_t AddressFamily,
1164
                                   Span<const uint8_t> Address, uint16_t Port,
1165
0
                                   __wasi_size_t &NWritten) const noexcept {
1166
0
  int SysSiFlags = MSG_NOSIGNAL;
1167
0
  sockaddr *ClientAddr = nullptr;
1168
0
  socklen_t MsgNameLen = 0;
1169
0
  VarAddrT AddressBuffer;
1170
1171
0
  if (Address.size()) {
1172
0
    AddressBuffer = sockAddressAssignHelper(AddressFamily, Address, Port);
1173
0
    ClientAddr = std::visit(VarAddrBuf(), AddressBuffer);
1174
0
    MsgNameLen = std::visit(VarAddrSize(), AddressBuffer);
1175
0
  }
1176
1177
0
  iovec SysIOVs[kIOVMax];
1178
0
  size_t SysIOVsSize = 0;
1179
0
  for (auto &IOV : SiData) {
1180
0
    SysIOVs[SysIOVsSize].iov_base = const_cast<uint8_t *>(IOV.data());
1181
0
    SysIOVs[SysIOVsSize].iov_len = IOV.size();
1182
0
    ++SysIOVsSize;
1183
0
  }
1184
1185
0
  msghdr SysMsgHdr;
1186
0
  SysMsgHdr.msg_name = MsgNameLen == 0 ? nullptr : ClientAddr;
1187
0
  SysMsgHdr.msg_namelen = MsgNameLen;
1188
0
  SysMsgHdr.msg_iov = SysIOVs;
1189
0
  SysMsgHdr.msg_iovlen = SysIOVsSize;
1190
0
  SysMsgHdr.msg_control = nullptr;
1191
0
  SysMsgHdr.msg_controllen = 0;
1192
1193
  // Store recv bytes length and flags.
1194
0
  if (auto Res = ::sendmsg(Fd, &SysMsgHdr, SysSiFlags); unlikely(Res < 0)) {
1195
0
    return WasiUnexpect(fromErrNo(errno));
1196
0
  } else {
1197
0
    NWritten = Res;
1198
0
  }
1199
1200
0
  return {};
1201
0
}
1202
1203
0
WasiExpect<void> INode::sockShutdown(__wasi_sdflags_t SdFlags) const noexcept {
1204
0
  int SysFlags = 0;
1205
0
  if (SdFlags == __WASI_SDFLAGS_RD) {
1206
0
    SysFlags = SHUT_RD;
1207
0
  } else if (SdFlags == __WASI_SDFLAGS_WR) {
1208
0
    SysFlags = SHUT_WR;
1209
0
  } else if (SdFlags == (__WASI_SDFLAGS_RD | __WASI_SDFLAGS_WR)) {
1210
0
    SysFlags = SHUT_RDWR;
1211
0
  }
1212
1213
0
  if (auto Res = ::shutdown(Fd, SysFlags); unlikely(Res < 0)) {
1214
0
    return WasiUnexpect(fromErrNo(errno));
1215
0
  }
1216
1217
0
  return {};
1218
0
}
1219
1220
WasiExpect<void> INode::sockGetOpt(__wasi_sock_opt_level_t SockOptLevel,
1221
                                   __wasi_sock_opt_so_t SockOptName,
1222
0
                                   Span<uint8_t> &Flag) const noexcept {
1223
0
  auto SysSockOptLevel = toSockOptLevel(SockOptLevel);
1224
0
  auto SysSockOptName = toSockOptSoName(SockOptName);
1225
0
  socklen_t Size = static_cast<socklen_t>(Flag.size());
1226
0
  if (auto Res =
1227
0
          ::getsockopt(Fd, SysSockOptLevel, SysSockOptName, Flag.data(), &Size);
1228
0
      unlikely(Res < 0)) {
1229
0
    return WasiUnexpect(fromErrNo(errno));
1230
0
  }
1231
1232
0
  switch (SockOptName) {
1233
0
  case __WASI_SOCK_OPT_SO_ERROR: {
1234
0
    assuming(Size == sizeof(int32_t));
1235
0
    Flag = Flag.first(static_cast<size_t>(Size));
1236
0
    auto *Error = reinterpret_cast<int32_t *>(Flag.data());
1237
0
    *Error = static_cast<int32_t>(fromErrNo(*Error));
1238
0
    break;
1239
0
  }
1240
0
  case __WASI_SOCK_OPT_SO_TYPE: {
1241
0
    assuming(Size == sizeof(int32_t));
1242
0
    Flag = Flag.first(static_cast<size_t>(Size));
1243
0
    assuming(Flag.size() == sizeof(int32_t));
1244
0
    auto &SockType = *reinterpret_cast<int32_t *>(Flag.data());
1245
0
    SockType = static_cast<int32_t>(fromSockType(SockType));
1246
0
    break;
1247
0
  }
1248
0
  default:
1249
0
    Flag = Flag.first(static_cast<size_t>(Size));
1250
0
  }
1251
1252
0
  return {};
1253
0
}
1254
1255
WasiExpect<void> INode::sockSetOpt(__wasi_sock_opt_level_t SockOptLevel,
1256
                                   __wasi_sock_opt_so_t SockOptName,
1257
0
                                   Span<const uint8_t> Flag) const noexcept {
1258
0
  auto SysSockOptLevel = toSockOptLevel(SockOptLevel);
1259
0
  auto SysSockOptName = toSockOptSoName(SockOptName);
1260
1261
0
  if (auto Res = ::setsockopt(Fd, SysSockOptLevel, SysSockOptName, Flag.data(),
1262
0
                              Flag.size());
1263
0
      unlikely(Res < 0)) {
1264
0
    return WasiUnexpect(fromErrNo(errno));
1265
0
  }
1266
1267
0
  return {};
1268
0
}
1269
1270
WasiExpect<void>
1271
INode::sockGetLocalAddr(__wasi_address_family_t *AddressFamilyPtr,
1272
                        Span<uint8_t> Address,
1273
0
                        uint16_t *PortPtr) const noexcept {
1274
0
  sockaddr_storage SocketAddr = {};
1275
0
  socklen_t Slen = sizeof(SocketAddr);
1276
1277
0
  if (auto Res =
1278
0
          ::getsockname(Fd, reinterpret_cast<sockaddr *>(&SocketAddr), &Slen);
1279
0
      unlikely(Res < 0)) {
1280
0
    return WasiUnexpect(fromErrNo(errno));
1281
0
  }
1282
1283
0
  switch (SocketAddr.ss_family) {
1284
0
  case AF_INET: {
1285
0
    if (Address.size() < sizeof(in_addr)) {
1286
0
      return WasiUnexpect(__WASI_ERRNO_NOMEM);
1287
0
    }
1288
0
    const auto &SocketAddr4 = reinterpret_cast<sockaddr_in &>(SocketAddr);
1289
0
    if (AddressFamilyPtr) {
1290
0
      *AddressFamilyPtr = __WASI_ADDRESS_FAMILY_INET4;
1291
0
    }
1292
0
    if (PortPtr) {
1293
0
      *PortPtr = ntohs(SocketAddr4.sin_port);
1294
0
    }
1295
0
    std::memcpy(Address.data(), &SocketAddr4.sin_addr, sizeof(in_addr));
1296
0
    return {};
1297
0
  }
1298
0
  case AF_INET6: {
1299
0
    if (Address.size() < sizeof(in6_addr)) {
1300
0
      return WasiUnexpect(__WASI_ERRNO_NOMEM);
1301
0
    }
1302
0
    const auto &SocketAddr6 = reinterpret_cast<sockaddr_in6 &>(SocketAddr);
1303
0
    if (AddressFamilyPtr) {
1304
0
      *AddressFamilyPtr = __WASI_ADDRESS_FAMILY_INET6;
1305
0
    }
1306
0
    if (PortPtr) {
1307
0
      *PortPtr = ntohs(SocketAddr6.sin6_port);
1308
0
    }
1309
0
    std::memcpy(Address.data(), &SocketAddr6.sin6_addr, sizeof(in6_addr));
1310
0
    return {};
1311
0
  }
1312
0
  case AF_UNIX: {
1313
0
    if (Address.size() < sizeof(sockaddr_un::sun_path)) {
1314
0
      return WasiUnexpect(__WASI_ERRNO_NOMEM);
1315
0
    }
1316
0
    const auto &SocketAddrUN = reinterpret_cast<sockaddr_un &>(SocketAddr);
1317
0
    if (AddressFamilyPtr) {
1318
0
      *AddressFamilyPtr = __WASI_ADDRESS_FAMILY_AF_UNIX;
1319
0
    }
1320
1321
0
    std::memcpy(Address.data(), &SocketAddrUN.sun_path,
1322
0
                sizeof(sockaddr_un::sun_path));
1323
0
    return {};
1324
0
  }
1325
0
  default:
1326
0
    return WasiUnexpect(__WASI_ERRNO_NOSYS);
1327
0
  }
1328
0
}
1329
1330
WasiExpect<void>
1331
INode::sockGetPeerAddr(__wasi_address_family_t *AddressFamilyPtr,
1332
                       Span<uint8_t> Address,
1333
0
                       uint16_t *PortPtr) const noexcept {
1334
0
  sockaddr_storage SocketAddr = {};
1335
0
  socklen_t Slen = sizeof(SocketAddr);
1336
1337
0
  if (auto Res =
1338
0
          ::getpeername(Fd, reinterpret_cast<sockaddr *>(&SocketAddr), &Slen);
1339
0
      unlikely(Res < 0)) {
1340
0
    return WasiUnexpect(fromErrNo(errno));
1341
0
  }
1342
1343
0
  switch (SocketAddr.ss_family) {
1344
0
  case AF_INET: {
1345
0
    if (Address.size() < sizeof(in_addr)) {
1346
0
      return WasiUnexpect(__WASI_ERRNO_NOMEM);
1347
0
    }
1348
0
    const auto &SocketAddr4 = reinterpret_cast<sockaddr_in &>(SocketAddr);
1349
0
    if (AddressFamilyPtr) {
1350
0
      *AddressFamilyPtr = __WASI_ADDRESS_FAMILY_INET4;
1351
0
    }
1352
0
    if (PortPtr) {
1353
0
      *PortPtr = ntohs(SocketAddr4.sin_port);
1354
0
    }
1355
0
    std::memcpy(Address.data(), &SocketAddr4.sin_addr, sizeof(in_addr));
1356
0
    return {};
1357
0
  }
1358
0
  case AF_INET6: {
1359
0
    if (Address.size() < sizeof(in6_addr)) {
1360
0
      return WasiUnexpect(__WASI_ERRNO_NOMEM);
1361
0
    }
1362
0
    const auto &SocketAddr6 = reinterpret_cast<sockaddr_in6 &>(SocketAddr);
1363
0
    if (AddressFamilyPtr) {
1364
0
      *AddressFamilyPtr = __WASI_ADDRESS_FAMILY_INET6;
1365
0
    }
1366
0
    if (PortPtr) {
1367
0
      *PortPtr = ntohs(SocketAddr6.sin6_port);
1368
0
    }
1369
0
    std::memcpy(Address.data(), &SocketAddr6.sin6_addr, sizeof(in6_addr));
1370
0
    return {};
1371
0
  }
1372
0
  case AF_UNIX: {
1373
0
    if (Address.size() < sizeof(sockaddr_un::sun_path)) {
1374
0
      return WasiUnexpect(__WASI_ERRNO_NOMEM);
1375
0
    }
1376
0
    const auto &SocketAddrUN = reinterpret_cast<sockaddr_un &>(SocketAddr);
1377
0
    if (AddressFamilyPtr) {
1378
0
      *AddressFamilyPtr = __WASI_ADDRESS_FAMILY_AF_UNIX;
1379
0
    }
1380
1381
0
    std::memcpy(Address.data(), &SocketAddrUN.sun_path,
1382
0
                sizeof(sockaddr_un::sun_path));
1383
0
    return {};
1384
0
  }
1385
0
  default:
1386
0
    return WasiUnexpect(__WASI_ERRNO_NOSYS);
1387
0
  }
1388
0
}
1389
1390
0
__wasi_filetype_t INode::unsafeFiletype() const noexcept {
1391
0
  return fromFileType(static_cast<mode_t>(Stat->st_mode));
1392
0
}
1393
1394
0
WasiExpect<__wasi_filetype_t> INode::filetype() const noexcept {
1395
0
  if (!Stat) {
1396
0
    EXPECTED_TRY(updateStat());
1397
0
  }
1398
0
  return unsafeFiletype();
1399
0
}
1400
1401
0
bool INode::isDirectory() const noexcept {
1402
0
  if (!Stat) {
1403
0
    if (!updateStat()) {
1404
0
      return false;
1405
0
    }
1406
0
  }
1407
0
  return (Stat->st_mode & S_IFMT) == S_IFDIR;
1408
0
}
1409
1410
0
bool INode::isSymlink() const noexcept {
1411
0
  if (!Stat) {
1412
0
    if (!updateStat()) {
1413
0
      return false;
1414
0
    }
1415
0
  }
1416
0
  return (Stat->st_mode & S_IFMT) == S_IFLNK;
1417
0
}
1418
1419
0
WasiExpect<__wasi_filesize_t> INode::filesize() const noexcept {
1420
0
  if (!Stat) {
1421
0
    EXPECTED_TRY(updateStat());
1422
0
  }
1423
0
  return Stat->st_size;
1424
0
}
1425
1426
0
bool INode::canBrowse() const noexcept {
1427
0
  return ::faccessat(Fd, ".", X_OK, 0) == 0;
1428
0
}
1429
1430
0
WasiExpect<void> INode::updateStat() const noexcept {
1431
0
  Stat.emplace();
1432
0
  if (unlikely(::fstat(Fd, &*Stat) != 0)) {
1433
0
    return WasiUnexpect(fromErrNo(errno));
1434
0
  }
1435
0
  return {};
1436
0
}
1437
1438
WasiExpect<Poller::Timer>
1439
0
PollerContext::acquireTimer(__wasi_clockid_t Clock) noexcept {
1440
0
  std::unique_lock Lock(TimerMutex);
1441
0
  if (auto &Timers = TimerPool.try_emplace(Clock).first->second;
1442
0
      Timers.empty()) {
1443
0
    Poller::Timer Result(Clock);
1444
0
    if (auto Res = Result.create(); unlikely(!Res)) {
1445
0
      return WasiUnexpect(fromErrNo(errno));
1446
0
    }
1447
0
    return Result;
1448
0
  } else {
1449
0
    Poller::Timer Result(std::move(Timers.back()));
1450
0
    Timers.pop_back();
1451
0
    return Result;
1452
0
  }
1453
0
}
1454
1455
0
void PollerContext::releaseTimer(Poller::Timer &&Timer) noexcept {
1456
0
  std::unique_lock Lock(TimerMutex);
1457
0
  const auto Clock = Timer.Clock;
1458
0
  try {
1459
0
    TimerPool.try_emplace(Clock).first->second.push_back(std::move(Timer));
1460
0
  } catch (std::bad_alloc &) {
1461
    // giving up caching timer
1462
0
  }
1463
0
}
1464
1465
#if __GLIBC_PREREQ(2, 8)
1466
0
WasiExpect<void> Poller::Timer::create() noexcept {
1467
0
  if (const auto NewFd =
1468
0
          ::timerfd_create(toClockId(Clock), TFD_NONBLOCK | TFD_CLOEXEC);
1469
0
      unlikely(NewFd < 0)) {
1470
0
    return WasiUnexpect(fromErrNo(errno));
1471
0
  } else {
1472
0
    FdHolder::emplace(NewFd);
1473
0
  }
1474
0
  return {};
1475
0
}
1476
1477
WasiExpect<void> Poller::Timer::setTime(__wasi_timestamp_t Timeout,
1478
                                        __wasi_timestamp_t,
1479
0
                                        __wasi_subclockflags_t Flags) noexcept {
1480
  // disarm timer
1481
0
  {
1482
0
    itimerspec Spec{toTimespec(0), toTimespec(0)};
1483
0
    if (auto Res = ::timerfd_settime(Fd, 0, &Spec, nullptr);
1484
0
        unlikely(Res < 0)) {
1485
0
      errno = 0;
1486
0
    }
1487
0
  }
1488
1489
0
  int SysFlags = 0;
1490
0
  if (Flags & __WASI_SUBCLOCKFLAGS_SUBSCRIPTION_CLOCK_ABSTIME) {
1491
0
    SysFlags |= TFD_TIMER_ABSTIME;
1492
0
  }
1493
  // Zero timeout has a special meaning. When the itimerspec is set to 0, then
1494
  // it will disarm timer.
1495
0
  Timeout = std::max<__wasi_timestamp_t>(Timeout, 1U);
1496
0
  itimerspec Spec{toTimespec(0), toTimespec(Timeout)};
1497
0
  if (auto Res = ::timerfd_settime(Fd, SysFlags, &Spec, nullptr);
1498
0
      unlikely(Res < 0)) {
1499
0
    return WasiUnexpect(fromErrNo(errno));
1500
0
  }
1501
1502
0
  return {};
1503
0
}
1504
#else
1505
namespace {
1506
static void sigevCallback(union sigval Value) noexcept {
1507
  const uint64_t One = 1;
1508
  ::write(Value.sival_int, &One, sizeof(One));
1509
}
1510
} // namespace
1511
1512
WasiExpect<void> Poller::Timer::create() noexcept {
1513
  {
1514
    int PipeFd[2] = {-1, -1};
1515
1516
    if (auto Res = ::pipe(PipeFd); unlikely(Res != 0)) {
1517
      return WasiUnexpect(fromErrNo(errno));
1518
    }
1519
    FdHolder::emplace(PipeFd[0]);
1520
    Notify.emplace(PipeFd[1]);
1521
  }
1522
1523
  timer_t TId;
1524
  {
1525
    sigevent Event;
1526
    Event.sigev_notify = SIGEV_THREAD;
1527
    Event.sigev_notify_function = &sigevCallback;
1528
    Event.sigev_value.sival_int = Notify.Fd;
1529
    Event.sigev_notify_attributes = nullptr;
1530
1531
    if (unlikely(::fcntl(Fd, F_SETFD, O_NONBLOCK | FD_CLOEXEC) != 0 ||
1532
                 ::fcntl(Notify.Fd, F_SETFD, O_NONBLOCK | FD_CLOEXEC) != 0 ||
1533
                 ::timer_create(toClockId(Clock), &Event, &TId) < 0)) {
1534
      return WasiUnexpect(fromErrNo(errno));
1535
    }
1536
  }
1537
  TimerId.emplace(TId);
1538
  return {};
1539
}
1540
1541
WasiExpect<void> Poller::Timer::setTime(__wasi_timestamp_t Timeout,
1542
                                        __wasi_timestamp_t,
1543
                                        __wasi_subclockflags_t Flags) noexcept {
1544
  if (unlikely(!TimerId.Id)) {
1545
    return WasiUnexpect(__WASI_ERRNO_INVAL);
1546
  }
1547
1548
  // disarm timer
1549
  {
1550
    itimerspec Spec{toTimespec(0), toTimespec(0)};
1551
    if (auto Res = ::timer_settime(*TimerId.Id, 0, &Spec, nullptr);
1552
        unlikely(Res < 0)) {
1553
      errno = 0;
1554
    }
1555
  }
1556
  // reset pipe
1557
  {
1558
    uint64_t Buffer[16];
1559
    while (true) {
1560
      if (auto Res = ::read(Fd, &Buffer, sizeof(Buffer)); Res <= 0) {
1561
        break;
1562
      }
1563
    }
1564
  }
1565
1566
  int SysFlags = 0;
1567
  if (Flags & __WASI_SUBCLOCKFLAGS_SUBSCRIPTION_CLOCK_ABSTIME) {
1568
    SysFlags |= TIMER_ABSTIME;
1569
  }
1570
  // Zero timeout has a special meaning. When the itimerspec is set to 0, then
1571
  // it will disarm timer.
1572
  Timeout = std::max<__wasi_timestamp_t>(Timeout, 1U);
1573
  itimerspec Spec{toTimespec(0), toTimespec(Timeout)};
1574
  if (auto Res = ::timer_settime(*TimerId.Id, SysFlags, &Spec, nullptr);
1575
      unlikely(Res < 0)) {
1576
    return WasiUnexpect(fromErrNo(errno));
1577
  }
1578
  return {};
1579
}
1580
#endif
1581
1582
Poller::Poller(PollerContext &C) noexcept
1583
0
    : FdHolder(
1584
0
#if __GLIBC_PREREQ(2, 9)
1585
0
          ::epoll_create1(EPOLL_CLOEXEC)
1586
#else
1587
          // Guessing a number that might be sufficient for linux before 2.6.8
1588
          ::epoll_create(32)
1589
#endif
1590
0
              ),
1591
0
      Ctx(C) {
1592
#if !__GLIBC_PREREQ(2, 9)
1593
  if (auto Res = ::fcntl(Fd, F_SETFD, FD_CLOEXEC); unlikely(Res != 0)) {
1594
    FdHolder::reset();
1595
    return;
1596
  }
1597
#endif
1598
0
}
1599
1600
0
WasiExpect<void> Poller::prepare(Span<__wasi_event_t> E) noexcept {
1601
0
  WasiEvents = E;
1602
0
  try {
1603
0
    Events.reserve(E.size());
1604
0
    Timers.reserve(E.size());
1605
0
    EPollEvents.reserve(E.size());
1606
0
  } catch (std::bad_alloc &) {
1607
0
    return WasiUnexpect(__WASI_ERRNO_NOMEM);
1608
0
  }
1609
1610
0
  return {};
1611
0
}
1612
1613
void Poller::clock(__wasi_clockid_t Clock, __wasi_timestamp_t Timeout,
1614
                   __wasi_timestamp_t Precision, __wasi_subclockflags_t Flags,
1615
0
                   __wasi_userdata_t UserData) noexcept {
1616
0
  assuming(Events.size() < WasiEvents.size());
1617
0
  auto &Event = Events.emplace_back();
1618
0
  Event.Valid = false;
1619
0
  Event.userdata = UserData;
1620
0
  Event.type = __WASI_EVENTTYPE_CLOCK;
1621
1622
0
  if (auto Res = Ctx.get().acquireTimer(Clock); unlikely(!Res)) {
1623
0
    Event.Valid = true;
1624
0
    Event.error = Res.error();
1625
0
    return;
1626
0
  } else {
1627
0
    Timers.emplace_back(std::move(*Res));
1628
0
  }
1629
1630
0
  auto &Timer = Timers.back();
1631
0
  if (auto Res = Timer.setTime(Timeout, Precision, Flags); unlikely(!Res)) {
1632
0
    Ctx.get().releaseTimer(std::move(Timer));
1633
0
    Timers.pop_back();
1634
0
    Event.Valid = true;
1635
0
    Event.error = Res.error();
1636
0
    return;
1637
0
  }
1638
1639
0
  assuming(Timer.Fd != Fd);
1640
0
  try {
1641
0
    auto [Iter, Added] = FdDatas.try_emplace(Timer.Fd);
1642
1643
0
    Iter->second.ReadEvent = &Event;
1644
0
    assuming(Added);
1645
1646
0
    epoll_event EPollEvent;
1647
0
    EPollEvent.events = EPOLLIN;
1648
0
#if defined(EPOLLRDHUP)
1649
0
    EPollEvent.events |= EPOLLRDHUP;
1650
0
#endif
1651
0
    EPollEvent.data.fd = Timer.Fd;
1652
1653
0
    if (auto Res = ::epoll_ctl(Fd, EPOLL_CTL_ADD, Timer.Fd, &EPollEvent);
1654
0
        unlikely(Res < 0)) {
1655
0
      FdDatas.erase(Iter);
1656
0
      Ctx.get().releaseTimer(std::move(Timer));
1657
0
      Timers.pop_back();
1658
0
      Event.Valid = true;
1659
0
      Event.error = fromErrNo(errno);
1660
0
      return;
1661
0
    }
1662
1663
0
    return;
1664
0
  } catch (std::bad_alloc &) {
1665
0
    Ctx.get().releaseTimer(std::move(Timer));
1666
0
    Timers.pop_back();
1667
0
    Event.Valid = true;
1668
0
    Event.error = __WASI_ERRNO_NOMEM;
1669
0
    return;
1670
0
  }
1671
0
}
1672
1673
0
void Poller::close(const INode &Node) noexcept {
1674
0
  FdDatas.erase(Node.Fd);
1675
0
  OldFdDatas.erase(Node.Fd);
1676
0
}
1677
1678
void Poller::read(const INode &Node, TriggerType Trigger,
1679
0
                  __wasi_userdata_t UserData) noexcept {
1680
0
  assuming(Events.size() < WasiEvents.size());
1681
0
  auto &Event = Events.emplace_back();
1682
0
  Event.Valid = false;
1683
0
  Event.userdata = UserData;
1684
0
  Event.type = __WASI_EVENTTYPE_FD_READ;
1685
1686
0
  assuming(Node.Fd != Fd);
1687
0
  try {
1688
0
    auto [Iter, Added] = FdDatas.try_emplace(Node.Fd);
1689
0
    const bool New = OldFdDatas.find(Node.Fd) == OldFdDatas.end();
1690
1691
0
    if (unlikely(!Added && Iter->second.ReadEvent != nullptr)) {
1692
0
      Event.Valid = true;
1693
0
      Event.error = __WASI_ERRNO_EXIST;
1694
0
      return;
1695
0
    }
1696
0
    Iter->second.ReadEvent = &Event;
1697
1698
0
    epoll_event EPollEvent;
1699
0
    EPollEvent.events = EPOLLIN;
1700
0
    if (!Added) {
1701
0
      assuming(Iter->second.WriteEvent != nullptr);
1702
0
      EPollEvent.events |= EPOLLOUT;
1703
0
    }
1704
0
    if (Trigger == TriggerType::Edge) {
1705
0
      EPollEvent.events |= EPOLLET;
1706
0
    }
1707
0
#if defined(EPOLLRDHUP)
1708
0
    EPollEvent.events |= EPOLLRDHUP;
1709
0
#endif
1710
0
    EPollEvent.data.fd = Node.Fd;
1711
1712
0
    if (likely(Added) && New) {
1713
0
      if (auto Res = ::epoll_ctl(Fd, EPOLL_CTL_ADD, Node.Fd, &EPollEvent);
1714
0
          unlikely(Res < 0)) {
1715
0
        FdDatas.erase(Iter);
1716
0
        Event.Valid = true;
1717
0
        Event.error = fromErrNo(errno);
1718
0
        return;
1719
0
      }
1720
0
    } else {
1721
0
      if (auto Res = ::epoll_ctl(Fd, EPOLL_CTL_MOD, Node.Fd, &EPollEvent);
1722
0
          unlikely(Res < 0)) {
1723
0
        Event.Valid = true;
1724
0
        Event.error = fromErrNo(errno);
1725
0
        return;
1726
0
      }
1727
0
    }
1728
0
  } catch (std::bad_alloc &) {
1729
0
    Event.Valid = true;
1730
0
    Event.error = __WASI_ERRNO_NOMEM;
1731
0
    return;
1732
0
  }
1733
0
}
1734
1735
void Poller::write(const INode &Node, TriggerType Trigger,
1736
0
                   __wasi_userdata_t UserData) noexcept {
1737
0
  assuming(Events.size() < WasiEvents.size());
1738
0
  auto &Event = Events.emplace_back();
1739
0
  Event.Valid = false;
1740
0
  Event.userdata = UserData;
1741
0
  Event.type = __WASI_EVENTTYPE_FD_WRITE;
1742
1743
0
  assuming(Node.Fd != Fd);
1744
0
  try {
1745
0
    auto [Iter, Added] = FdDatas.try_emplace(Node.Fd);
1746
0
    const bool New = OldFdDatas.find(Node.Fd) == OldFdDatas.end();
1747
1748
0
    if (unlikely(!Added && Iter->second.WriteEvent != nullptr)) {
1749
0
      Event.Valid = true;
1750
0
      Event.error = __WASI_ERRNO_EXIST;
1751
0
      return;
1752
0
    }
1753
0
    Iter->second.WriteEvent = &Event;
1754
1755
0
    epoll_event EPollEvent;
1756
0
    EPollEvent.events = EPOLLOUT;
1757
0
    if (!Added) {
1758
0
      assuming(Iter->second.ReadEvent != nullptr);
1759
0
      EPollEvent.events |= EPOLLIN;
1760
0
    }
1761
0
    if (Trigger == TriggerType::Edge) {
1762
0
      EPollEvent.events |= EPOLLET;
1763
0
    }
1764
0
#if defined(EPOLLRDHUP)
1765
0
    EPollEvent.events |= EPOLLRDHUP;
1766
0
#endif
1767
0
    EPollEvent.data.fd = Node.Fd;
1768
1769
0
    if (likely(Added) && New) {
1770
0
      if (auto Res = ::epoll_ctl(Fd, EPOLL_CTL_ADD, Node.Fd, &EPollEvent);
1771
0
          unlikely(Res < 0)) {
1772
0
        FdDatas.erase(Iter);
1773
0
        Event.Valid = true;
1774
0
        Event.error = fromErrNo(errno);
1775
0
        return;
1776
0
      }
1777
0
    } else {
1778
0
      if (auto Res = ::epoll_ctl(Fd, EPOLL_CTL_MOD, Node.Fd, &EPollEvent);
1779
0
          unlikely(Res < 0)) {
1780
0
        Event.Valid = true;
1781
0
        Event.error = fromErrNo(errno);
1782
0
        return;
1783
0
      }
1784
0
    }
1785
0
  } catch (std::bad_alloc &) {
1786
0
    Event.Valid = true;
1787
0
    Event.error = __WASI_ERRNO_NOMEM;
1788
0
    return;
1789
0
  }
1790
0
}
1791
1792
0
void Poller::wait() noexcept {
1793
0
  for (const auto &[NodeFd, FdData] : OldFdDatas) {
1794
0
    if (auto Iter = FdDatas.find(NodeFd); Iter == FdDatas.end()) {
1795
      // Remove unused event, ignore failed.
1796
      // In kernel before 2.6.9, EPOLL_CTL_DEL required a non-null pointer. Use
1797
      // `this` as the dummy parameter.
1798
0
      ::epoll_ctl(Fd, EPOLL_CTL_DEL, NodeFd,
1799
0
                  reinterpret_cast<struct epoll_event *>(this));
1800
0
    }
1801
0
  }
1802
1803
0
  EPollEvents.resize(Events.size());
1804
0
  const int Count =
1805
0
      ::epoll_wait(Fd, EPollEvents.data(), EPollEvents.size(), -1);
1806
0
  if (unlikely(Count < 0)) {
1807
0
    const auto Error = fromErrNo(errno);
1808
0
    for (auto &Event : Events) {
1809
0
      Event.Valid = true;
1810
0
      Event.error = Error;
1811
0
    }
1812
0
    return;
1813
0
  }
1814
1815
0
  auto ProcessEvent = [](const struct epoll_event &EPollEvent,
1816
0
                         OptionalEvent &Event) noexcept {
1817
0
    Event.Valid = true;
1818
0
    Event.error = __WASI_ERRNO_SUCCESS;
1819
0
    switch (Event.type) {
1820
0
    case __WASI_EVENTTYPE_CLOCK:
1821
0
      break;
1822
0
    case __WASI_EVENTTYPE_FD_READ: {
1823
0
      Event.fd_readwrite.flags = static_cast<__wasi_eventrwflags_t>(0);
1824
0
      if (EPollEvent.events & EPOLLHUP) {
1825
0
        Event.fd_readwrite.flags |= __WASI_EVENTRWFLAGS_FD_READWRITE_HANGUP;
1826
0
      }
1827
0
      bool UnknownNBytes = false;
1828
0
      int ReadBufUsed = 0;
1829
0
      if (auto Res = ::ioctl(EPollEvent.data.fd, FIONREAD, &ReadBufUsed);
1830
0
          unlikely(Res == 0)) {
1831
0
        UnknownNBytes = true;
1832
0
      }
1833
0
      if (UnknownNBytes) {
1834
0
        Event.fd_readwrite.nbytes = 1;
1835
0
      } else {
1836
0
        Event.fd_readwrite.nbytes = ReadBufUsed;
1837
0
      }
1838
0
      break;
1839
0
    }
1840
0
    case __WASI_EVENTTYPE_FD_WRITE: {
1841
0
      Event.fd_readwrite.flags = static_cast<__wasi_eventrwflags_t>(0);
1842
0
      if (EPollEvent.events & EPOLLHUP) {
1843
0
        Event.fd_readwrite.flags |= __WASI_EVENTRWFLAGS_FD_READWRITE_HANGUP;
1844
0
      }
1845
0
      bool UnknownNBytes = false;
1846
0
      int WriteBufSize = 0;
1847
0
      socklen_t IntSize = sizeof(WriteBufSize);
1848
0
      if (auto Res = ::getsockopt(EPollEvent.data.fd, SOL_SOCKET, SO_SNDBUF,
1849
0
                                  &WriteBufSize, &IntSize);
1850
0
          unlikely(Res != 0)) {
1851
0
        UnknownNBytes = true;
1852
0
      }
1853
0
      int WriteBufUsed = 0;
1854
0
      if (auto Res = ::ioctl(EPollEvent.data.fd, TIOCOUTQ, &WriteBufUsed);
1855
0
          unlikely(Res != 0)) {
1856
0
        UnknownNBytes = true;
1857
0
      }
1858
0
      if (UnknownNBytes) {
1859
0
        Event.fd_readwrite.nbytes = 1;
1860
0
      } else {
1861
0
        Event.fd_readwrite.nbytes = WriteBufSize - WriteBufUsed;
1862
0
      }
1863
0
      break;
1864
0
    }
1865
0
    }
1866
0
  };
1867
1868
0
  for (int I = 0; I < Count; ++I) {
1869
0
    const auto &EPollEvent = EPollEvents[I];
1870
0
    const auto Iter = FdDatas.find(EPollEvent.data.fd);
1871
0
    assuming(Iter != FdDatas.end());
1872
1873
0
    const bool NoInOut = !(EPollEvent.events & (EPOLLIN | EPOLLOUT));
1874
0
    if ((EPollEvent.events & EPOLLIN) ||
1875
0
        (NoInOut && EPollEvent.events & EPOLLHUP && Iter->second.ReadEvent)) {
1876
0
      assuming(Iter->second.ReadEvent);
1877
0
      assuming(Iter->second.ReadEvent->type == __WASI_EVENTTYPE_CLOCK ||
1878
0
               Iter->second.ReadEvent->type == __WASI_EVENTTYPE_FD_READ);
1879
0
      ProcessEvent(EPollEvent, *Iter->second.ReadEvent);
1880
0
    }
1881
0
    if (EPollEvent.events & EPOLLOUT ||
1882
0
        (NoInOut && EPollEvent.events & EPOLLHUP && Iter->second.WriteEvent)) {
1883
0
      assuming(Iter->second.WriteEvent);
1884
0
      assuming(Iter->second.WriteEvent->type == __WASI_EVENTTYPE_FD_WRITE);
1885
0
      ProcessEvent(EPollEvent, *Iter->second.WriteEvent);
1886
0
    }
1887
0
  }
1888
0
  for (auto &Timer : Timers) {
1889
    // Remove unused timer event, ignore failed.
1890
    // In kernel before 2.6.9, EPOLL_CTL_DEL required a non-null pointer. Use
1891
    // `this` as the dummy parameter.
1892
0
    ::epoll_ctl(Fd, EPOLL_CTL_DEL, Timer.Fd,
1893
0
                reinterpret_cast<struct epoll_event *>(this));
1894
0
    Ctx.get().releaseTimer(std::move(Timer));
1895
0
  }
1896
1897
0
  std::swap(FdDatas, OldFdDatas);
1898
0
  FdDatas.clear();
1899
0
  Timers.clear();
1900
0
  EPollEvents.clear();
1901
0
}
1902
1903
0
void Poller::reset() noexcept {
1904
0
  WasiEvents = {};
1905
0
  Events.clear();
1906
0
}
1907
1908
0
bool Poller::ok() noexcept { return FdHolder::ok(); }
1909
1910
} // namespace WASI
1911
} // namespace Host
1912
} // namespace WasmEdge
1913
1914
#endif