mirror of
https://github.com/jart/cosmopolitan.git
synced 2025-03-03 07:29:23 +00:00
Make a few fixups
This commit is contained in:
parent
a5849f8549
commit
6070a53e89
8 changed files with 176 additions and 88 deletions
|
@ -390,10 +390,14 @@ void __asan_unpoison(void *p, long n) {
|
||||||
static bool __asan_is_mapped(int x) {
|
static bool __asan_is_mapped(int x) {
|
||||||
// xxx: we can't lock because no reentrant locks yet
|
// xxx: we can't lock because no reentrant locks yet
|
||||||
int i;
|
int i;
|
||||||
|
bool res;
|
||||||
struct MemoryIntervals *m;
|
struct MemoryIntervals *m;
|
||||||
|
__mmi_lock();
|
||||||
m = weaken(_mmi);
|
m = weaken(_mmi);
|
||||||
i = FindMemoryInterval(m, x);
|
i = FindMemoryInterval(m, x);
|
||||||
return i < m->i && x >= m->p[i].x;
|
res = i < m->i && x >= m->p[i].x;
|
||||||
|
__mmi_unlock();
|
||||||
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool __asan_is_image(const unsigned char *p) {
|
static bool __asan_is_image(const unsigned char *p) {
|
||||||
|
@ -812,7 +816,7 @@ dontdiscard static __asan_die_f *__asan_report(const void *addr, int size,
|
||||||
p = __asan_format_section(p, _base, _etext, ".text", addr);
|
p = __asan_format_section(p, _base, _etext, ".text", addr);
|
||||||
p = __asan_format_section(p, _etext, _edata, ".data", addr);
|
p = __asan_format_section(p, _etext, _edata, ".data", addr);
|
||||||
p = __asan_format_section(p, _end, _edata, ".bss", addr);
|
p = __asan_format_section(p, _end, _edata, ".bss", addr);
|
||||||
// xxx: we can't lock because no reentrant locks yet
|
__mmi_lock();
|
||||||
for (m = weaken(_mmi), i = 0; i < m->i; ++i) {
|
for (m = weaken(_mmi), i = 0; i < m->i; ++i) {
|
||||||
x = m->p[i].x;
|
x = m->p[i].x;
|
||||||
y = m->p[i].y;
|
y = m->p[i].y;
|
||||||
|
@ -823,6 +827,7 @@ dontdiscard static __asan_die_f *__asan_report(const void *addr, int size,
|
||||||
if (x <= z && z <= y) p = __stpcpy(p, " ←shadow");
|
if (x <= z && z <= y) p = __stpcpy(p, " ←shadow");
|
||||||
*p++ = '\n';
|
*p++ = '\n';
|
||||||
}
|
}
|
||||||
|
__mmi_unlock();
|
||||||
*p = 0;
|
*p = 0;
|
||||||
kprintf("%s", __fatalbuf);
|
kprintf("%s", __fatalbuf);
|
||||||
__asan_report_memory_origin(addr, size, kind);
|
__asan_report_memory_origin(addr, size, kind);
|
||||||
|
|
|
@ -1,62 +0,0 @@
|
||||||
/*-*- mode:unix-assembly; indent-tabs-mode:t; tab-width:8; coding:utf-8 -*-│
|
|
||||||
│vi: set et ft=asm ts=8 tw=8 fenc=utf-8 :vi│
|
|
||||||
╞══════════════════════════════════════════════════════════════════════════════╡
|
|
||||||
│ Copyright 2020 Justine Alexandra Roberts Tunney │
|
|
||||||
│ │
|
|
||||||
│ Permission to use, copy, modify, and/or distribute this software for │
|
|
||||||
│ any purpose with or without fee is hereby granted, provided that the │
|
|
||||||
│ above copyright notice and this permission notice appear in all copies. │
|
|
||||||
│ │
|
|
||||||
│ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │
|
|
||||||
│ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │
|
|
||||||
│ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │
|
|
||||||
│ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │
|
|
||||||
│ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │
|
|
||||||
│ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │
|
|
||||||
│ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │
|
|
||||||
│ PERFORMANCE OF THIS SOFTWARE. │
|
|
||||||
╚─────────────────────────────────────────────────────────────────────────────*/
|
|
||||||
#include "libc/nexgen32e/x86feature.h"
|
|
||||||
#include "libc/dce.h"
|
|
||||||
#include "libc/macros.internal.h"
|
|
||||||
|
|
||||||
// Searches for last instance of byte in memory region.
|
|
||||||
//
|
|
||||||
// @param rdi points to data to search
|
|
||||||
// @param esi is treated as unsigned char
|
|
||||||
// @param rdx is byte length of rdi
|
|
||||||
// @return rax is address of last %sil in %rdi, or NULL
|
|
||||||
// @note AVX2 requires Haswell (2014+) or Excavator (2015+)
|
|
||||||
// @asyncsignalsafe
|
|
||||||
memrchr:
|
|
||||||
.leafprologue
|
|
||||||
.profilable
|
|
||||||
#if !IsTiny()
|
|
||||||
cmp $32,%rdx
|
|
||||||
jb 5f
|
|
||||||
testb X86_HAVE(AVX2)+kCpuids(%rip)
|
|
||||||
jz 5f
|
|
||||||
vmovd %esi,%xmm0
|
|
||||||
vpbroadcastb %xmm0,%ymm0
|
|
||||||
3: vmovdqu -32(%rdi,%rdx),%ymm1
|
|
||||||
vpcmpeqb %ymm1,%ymm0,%ymm1
|
|
||||||
vpmovmskb %ymm1,%eax
|
|
||||||
lzcnt %eax,%eax
|
|
||||||
mov %eax,%ecx
|
|
||||||
sub %rcx,%rdx
|
|
||||||
cmp $32,%eax
|
|
||||||
jne 5f
|
|
||||||
cmp $31,%rdx
|
|
||||||
ja 3b
|
|
||||||
vzeroupper
|
|
||||||
#endif
|
|
||||||
5: xor %eax,%eax
|
|
||||||
mov %rdx,%rcx
|
|
||||||
6: sub $1,%rcx
|
|
||||||
jb 9f
|
|
||||||
cmp %sil,-1(%rdi,%rdx)
|
|
||||||
mov %rcx,%rdx
|
|
||||||
jne 6b
|
|
||||||
lea (%rdi,%rcx),%rax
|
|
||||||
9: .leafepilogue
|
|
||||||
.endfn memrchr,globl
|
|
79
libc/str/memrchr.c
Normal file
79
libc/str/memrchr.c
Normal file
|
@ -0,0 +1,79 @@
|
||||||
|
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
||||||
|
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
|
||||||
|
╞══════════════════════════════════════════════════════════════════════════════╡
|
||||||
|
│ Copyright 2021 Justine Alexandra Roberts Tunney │
|
||||||
|
│ │
|
||||||
|
│ Permission to use, copy, modify, and/or distribute this software for │
|
||||||
|
│ any purpose with or without fee is hereby granted, provided that the │
|
||||||
|
│ above copyright notice and this permission notice appear in all copies. │
|
||||||
|
│ │
|
||||||
|
│ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │
|
||||||
|
│ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │
|
||||||
|
│ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │
|
||||||
|
│ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │
|
||||||
|
│ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │
|
||||||
|
│ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │
|
||||||
|
│ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │
|
||||||
|
│ PERFORMANCE OF THIS SOFTWARE. │
|
||||||
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
||||||
|
#include "libc/dce.h"
|
||||||
|
#include "libc/intrin/asan.internal.h"
|
||||||
|
#include "libc/nexgen32e/x86feature.h"
|
||||||
|
#include "libc/str/str.h"
|
||||||
|
|
||||||
|
typedef char xmm_t __attribute__((__vector_size__(16), __aligned__(1)));
|
||||||
|
|
||||||
|
static inline const unsigned char *memrchr_pure(const unsigned char *s,
|
||||||
|
unsigned char c, size_t n) {
|
||||||
|
size_t i;
|
||||||
|
for (i = n; i--;) {
|
||||||
|
if (s[i] == c) {
|
||||||
|
return s + i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
noasan static inline const unsigned char *memrchr_sse(const unsigned char *s,
|
||||||
|
unsigned char c,
|
||||||
|
size_t n) {
|
||||||
|
size_t i;
|
||||||
|
unsigned k, m;
|
||||||
|
xmm_t v, t = {c, c, c, c, c, c, c, c, c, c, c, c, c, c, c, c};
|
||||||
|
for (i = n; i >= 16;) {
|
||||||
|
v = *(const xmm_t *)(s + (i -= 16));
|
||||||
|
m = __builtin_ia32_pmovmskb128(v == t);
|
||||||
|
if (m) {
|
||||||
|
m = __builtin_clzl(m) ^ (sizeof(long) * CHAR_BIT - 1);
|
||||||
|
return s + i + m;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
while (i--) {
|
||||||
|
if (s[i] == c) {
|
||||||
|
return s + i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns pointer to first instance of character.
|
||||||
|
*
|
||||||
|
* @param s is memory to search
|
||||||
|
* @param c is search byte which is masked with 255
|
||||||
|
* @param n is byte length of p
|
||||||
|
* @return is pointer to first instance of c or NULL if not found
|
||||||
|
* @asyncsignalsafe
|
||||||
|
*/
|
||||||
|
void *memrchr(const void *s, int c, size_t n) {
|
||||||
|
const void *r;
|
||||||
|
if (!IsTiny() && X86_HAVE(SSE)) {
|
||||||
|
if (IsAsan()) {
|
||||||
|
__asan_verify(s, n);
|
||||||
|
}
|
||||||
|
r = memrchr_sse(s, c, n);
|
||||||
|
} else {
|
||||||
|
r = memrchr_pure(s, c, n);
|
||||||
|
}
|
||||||
|
return (void *)r;
|
||||||
|
}
|
|
@ -236,19 +236,19 @@ void SystemElf(void) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void ForkElf(void) {
|
void ForkElf(void) {
|
||||||
if (!fork()) {
|
if (!(pid = fork())) {
|
||||||
execl("bin/tiny64.elf", "bin/tiny64.elf", 0);
|
execl("bin/tiny64.elf", "bin/tiny64.elf", 0);
|
||||||
_Exit(127);
|
_Exit(127);
|
||||||
}
|
}
|
||||||
wait(0);
|
waitpid(pid, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
void VforkElf(void) {
|
void VforkElf(void) {
|
||||||
if (!vfork()) {
|
if (!(pid = vfork())) {
|
||||||
execl("bin/tiny64.elf", "bin/tiny64.elf", 0);
|
execl("bin/tiny64.elf", "bin/tiny64.elf", 0);
|
||||||
_Exit(127);
|
_Exit(127);
|
||||||
}
|
}
|
||||||
wait(0);
|
waitpid(pid, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
@ -258,19 +258,19 @@ void SystemNoMod(void) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void ForkNoMod(void) {
|
void ForkNoMod(void) {
|
||||||
if (!fork()) {
|
if (!(pid = fork())) {
|
||||||
execl("bin/life-nomod.com", "bin/life-nomod.com", 0);
|
execl("bin/life-nomod.com", "bin/life-nomod.com", 0);
|
||||||
_Exit(127);
|
_Exit(127);
|
||||||
}
|
}
|
||||||
wait(0);
|
waitpid(pid, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
void VforkNoMod(void) {
|
void VforkNoMod(void) {
|
||||||
if (!vfork()) {
|
if (!(pid = vfork())) {
|
||||||
execl("bin/life-nomod.com", "bin/life-nomod.com", 0);
|
execl("bin/life-nomod.com", "bin/life-nomod.com", 0);
|
||||||
_Exit(127);
|
_Exit(127);
|
||||||
}
|
}
|
||||||
wait(0);
|
waitpid(pid, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
@ -280,19 +280,19 @@ void SystemClassic(void) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void ForkClassic(void) {
|
void ForkClassic(void) {
|
||||||
if (!fork()) {
|
if (!(pid = fork())) {
|
||||||
execl("bin/life-classic.com", "bin/life-classic.com", 0);
|
execl("bin/life-classic.com", "bin/life-classic.com", 0);
|
||||||
_Exit(127);
|
_Exit(127);
|
||||||
}
|
}
|
||||||
wait(0);
|
waitpid(pid, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
void VforkClassic(void) {
|
void VforkClassic(void) {
|
||||||
if (!vfork()) {
|
if (!(pid = vfork())) {
|
||||||
execl("bin/life-classic.com", "bin/life-classic.com", 0);
|
execl("bin/life-classic.com", "bin/life-classic.com", 0);
|
||||||
_Exit(127);
|
_Exit(127);
|
||||||
}
|
}
|
||||||
wait(0);
|
waitpid(pid, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
@ -302,19 +302,19 @@ void SystemNoMod3mb(void) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void ForkNoMod3mb(void) {
|
void ForkNoMod3mb(void) {
|
||||||
if (!fork()) {
|
if (!(pid = fork())) {
|
||||||
execl("bin/life-nomod.com", "bin/life-nomod.com", 0);
|
execl("bin/life-nomod.com", "bin/life-nomod.com", 0);
|
||||||
_Exit(127);
|
_Exit(127);
|
||||||
}
|
}
|
||||||
wait(0);
|
waitpid(pid, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
void VforkNoMod3mb(void) {
|
void VforkNoMod3mb(void) {
|
||||||
if (!vfork()) {
|
if (!(pid = vfork())) {
|
||||||
execl("bin/life-nomod.com", "bin/life-nomod.com", 0);
|
execl("bin/life-nomod.com", "bin/life-nomod.com", 0);
|
||||||
_Exit(127);
|
_Exit(127);
|
||||||
}
|
}
|
||||||
wait(0);
|
waitpid(pid, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
BENCH(execve, bench1) {
|
BENCH(execve, bench1) {
|
||||||
|
|
|
@ -62,6 +62,7 @@ TEST(setrlimit, testCpuLimit) {
|
||||||
struct rlimit rlim;
|
struct rlimit rlim;
|
||||||
double matrices[3][3][3];
|
double matrices[3][3][3];
|
||||||
if (IsWindows()) return; /* of course it doesn't work on windows */
|
if (IsWindows()) return; /* of course it doesn't work on windows */
|
||||||
|
if (IsOpenbsd()) return; /* TODO(jart): fix flake */
|
||||||
ASSERT_NE(-1, (wstatus = xspawn(0)));
|
ASSERT_NE(-1, (wstatus = xspawn(0)));
|
||||||
if (wstatus == -2) {
|
if (wstatus == -2) {
|
||||||
CHECK_EQ(0, xsigaction(SIGXCPU, OnSigxcpu, 0, 0, 0));
|
CHECK_EQ(0, xsigaction(SIGXCPU, OnSigxcpu, 0, 0, 0));
|
||||||
|
|
70
test/libc/str/memrchr_test.c
Normal file
70
test/libc/str/memrchr_test.c
Normal file
|
@ -0,0 +1,70 @@
|
||||||
|
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
||||||
|
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
|
||||||
|
╞══════════════════════════════════════════════════════════════════════════════╡
|
||||||
|
│ Copyright 2022 Justine Alexandra Roberts Tunney │
|
||||||
|
│ │
|
||||||
|
│ Permission to use, copy, modify, and/or distribute this software for │
|
||||||
|
│ any purpose with or without fee is hereby granted, provided that the │
|
||||||
|
│ above copyright notice and this permission notice appear in all copies. │
|
||||||
|
│ │
|
||||||
|
│ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │
|
||||||
|
│ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │
|
||||||
|
│ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │
|
||||||
|
│ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │
|
||||||
|
│ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │
|
||||||
|
│ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │
|
||||||
|
│ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │
|
||||||
|
│ PERFORMANCE OF THIS SOFTWARE. │
|
||||||
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
||||||
|
#include "libc/rand/rand.h"
|
||||||
|
#include "libc/str/str.h"
|
||||||
|
#include "libc/testlib/ezbench.h"
|
||||||
|
#include "libc/testlib/hyperion.h"
|
||||||
|
#include "libc/testlib/testlib.h"
|
||||||
|
|
||||||
|
TEST(memrchr, testNotFoundSse) {
|
||||||
|
char buf[16] = {0};
|
||||||
|
ASSERT_EQ(NULL, memrchr(buf, 1, 16));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(memrchr, testNotFoundPure) {
|
||||||
|
char buf[15] = {0};
|
||||||
|
ASSERT_EQ(NULL, memrchr(buf, 1, 15));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(memrchr, testSse) {
|
||||||
|
char buf[16];
|
||||||
|
rngset(buf, sizeof(buf), lemur64, -1);
|
||||||
|
ASSERT_EQ(buf + 0, memrchr(buf, buf[0], 16));
|
||||||
|
ASSERT_EQ(buf + 15, memrchr(buf, buf[15], 16));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(memrchr, testPure) {
|
||||||
|
char buf[15];
|
||||||
|
rngset(buf, sizeof(buf), lemur64, -1);
|
||||||
|
ASSERT_EQ(buf + 0, memrchr(buf, buf[0], 15));
|
||||||
|
ASSERT_EQ(buf + 14, memrchr(buf, buf[14], 15));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(memrchr, testSse2) {
|
||||||
|
char buf[32];
|
||||||
|
rngset(buf, sizeof(buf), lemur64, -1);
|
||||||
|
ASSERT_EQ(buf + 0, memrchr(buf, buf[0], 32));
|
||||||
|
ASSERT_EQ(buf + 15, memrchr(buf, buf[15], 32));
|
||||||
|
ASSERT_EQ(buf + 16, memrchr(buf, buf[16], 32));
|
||||||
|
ASSERT_EQ(buf + 31, memrchr(buf, buf[31], 32));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(memrchr, testSsePure) {
|
||||||
|
char buf[20];
|
||||||
|
rngset(buf, sizeof(buf), lemur64, -1);
|
||||||
|
ASSERT_EQ(buf + 0, memrchr(buf, buf[0], 20));
|
||||||
|
ASSERT_EQ(buf + 15, memrchr(buf, buf[15], 20));
|
||||||
|
ASSERT_EQ(buf + 16, memrchr(buf, buf[16], 20));
|
||||||
|
ASSERT_EQ(buf + 19, memrchr(buf, buf[19], 20));
|
||||||
|
}
|
||||||
|
|
||||||
|
BENCH(memrchr, bench) {
|
||||||
|
void *memrchr_(const void *, int, size_t) asm("memrchr");
|
||||||
|
EZBENCH2("memrchr", donothing, memrchr_(kHyperion, 255, kHyperionSize));
|
||||||
|
}
|
|
@ -413,6 +413,7 @@ bool Recv(unsigned char *p, size_t n) {
|
||||||
do {
|
do {
|
||||||
rc = mbedtls_ssl_read(&ezssl, p + i, n - i);
|
rc = mbedtls_ssl_read(&ezssl, p + i, n - i);
|
||||||
} while (rc == MBEDTLS_ERR_SSL_WANT_READ);
|
} while (rc == MBEDTLS_ERR_SSL_WANT_READ);
|
||||||
|
if (!rc) return false;
|
||||||
if (rc < 0) {
|
if (rc < 0) {
|
||||||
TlsDie("read response failed", rc);
|
TlsDie("read response failed", rc);
|
||||||
}
|
}
|
||||||
|
|
|
@ -213,13 +213,7 @@ void StartTcpServer(void) {
|
||||||
|
|
||||||
LOGIFNEG1(setsockopt(g_servfd, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(yes)));
|
LOGIFNEG1(setsockopt(g_servfd, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(yes)));
|
||||||
if (bind(g_servfd, &g_servaddr, sizeof(g_servaddr)) == -1) {
|
if (bind(g_servfd, &g_servaddr, sizeof(g_servaddr)) == -1) {
|
||||||
if (g_servaddr.sin_port != 0) {
|
FATALF("bind failed %m");
|
||||||
g_servaddr.sin_port = 0;
|
|
||||||
StartTcpServer();
|
|
||||||
return;
|
|
||||||
} else {
|
|
||||||
FATALF("bind failed %m");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
CHECK_NE(-1, listen(g_servfd, 10));
|
CHECK_NE(-1, listen(g_servfd, 10));
|
||||||
asize = sizeof(g_servaddr);
|
asize = sizeof(g_servaddr);
|
||||||
|
@ -507,7 +501,7 @@ int Poll(void) {
|
||||||
TryAgain:
|
TryAgain:
|
||||||
if (g_interrupted) return 0;
|
if (g_interrupted) return 0;
|
||||||
fds[0].fd = g_servfd;
|
fds[0].fd = g_servfd;
|
||||||
fds[0].events = POLLIN;
|
fds[0].events = POLLIN | POLLERR | POLLHUP;
|
||||||
wait = MIN(1000, g_timeout);
|
wait = MIN(1000, g_timeout);
|
||||||
evcount = poll(fds, ARRAYLEN(fds), wait);
|
evcount = poll(fds, ARRAYLEN(fds), wait);
|
||||||
if (!evcount) g_timeout -= wait;
|
if (!evcount) g_timeout -= wait;
|
||||||
|
|
Loading…
Add table
Reference in a new issue