Coverage Report

Created: 2025-08-29 06:29

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