diff --git a/libc/calls/readwrite-nt.c b/libc/calls/readwrite-nt.c
index e1c05922380..1c983ca80bd 100644
--- a/libc/calls/readwrite-nt.c
+++ b/libc/calls/readwrite-nt.c
@@ -16,7 +16,6 @@
 │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR             │
 │ PERFORMANCE OF THIS SOFTWARE.                                                │
 ╚─────────────────────────────────────────────────────────────────────────────*/
-#include "libc/assert.h"
 #include "libc/calls/createfileflags.internal.h"
 #include "libc/calls/internal.h"
 #include "libc/calls/sig.internal.h"
@@ -170,14 +169,11 @@ sys_readwrite_nt(int fd, void *data, size_t size, ssize_t offset,
     }
 
     // the i/o operation was successfully canceled
-    if (got_eagain) {
-      unassert(!got_sig);
+    if (got_eagain)
       return eagain();
-    }
 
     // it's now reasonable to report semaphore creation error
     if (other_error) {
-      unassert(!got_sig);
       errno = __dos2errno(other_error);
       return -1;
     }
diff --git a/libc/calls/sig.c b/libc/calls/sig.c
index 196de339791..8db280e0ce9 100644
--- a/libc/calls/sig.c
+++ b/libc/calls/sig.c
@@ -302,10 +302,10 @@ static textwindows int __sig_killer(struct PosixThread *pt, int sig, int sic) {
     return 0;
   }
 
-  // we can't preempt threads that masked sig or are blocked. we aso
-  // need to ensure we don't the target thread's stack if many signals
-  // need to be delivered at once. we also need to make sure two threads
-  // can't deadlock by killing each other at the same time.
+  // we can't preempt threads that masked sigs or are blocked. we also
+  // need to ensure we don't overflow the target thread's stack if many
+  // signals need to be delivered at once. we also need to make sure two
+  // threads can't deadlock by killing each other at the same time.
   if ((atomic_load_explicit(&pt->tib->tib_sigmask, memory_order_acquire) &
        (1ull << (sig - 1))) ||
       atomic_exchange_explicit(&pt->pt_intoff, 1, memory_order_acquire)) {
diff --git a/test/posix/pipe_write_eagain_test.c b/test/posix/pipe_write_eagain_test.c
new file mode 100644
index 00000000000..66a1c42eaa1
--- /dev/null
+++ b/test/posix/pipe_write_eagain_test.c
@@ -0,0 +1,107 @@
+// Copyright 2024 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 <errno.h>
+#include <poll.h>
+#include <pthread.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <unistd.h>
+
+/**
+ * @fileoverview Tests that EAGAIN won't corrupt pipe.
+ *
+ * This is a real bug when using CancelIoEx() on winsock writes, so we
+ * need to make sure it doesn't happen on pipes too.
+ */
+
+#define ITERATIONS 100000
+#define ASYMMETRY  3
+
+int fds[2];
+int got_read_eagains;
+int got_write_eagains;
+
+void *worker(void *arg) {
+  for (int expect = 0; expect < ITERATIONS;) {
+    int number;
+    ssize_t rc = read(fds[0], &number, sizeof(number));
+    if (rc == -1) {
+      if (errno == EAGAIN) {
+        ++got_read_eagains;
+        if (poll(&(struct pollfd){fds[0], POLLIN}, 1, -1) == -1)
+          exit(11);
+        continue;
+      }
+      perror("read");
+      exit(8);
+    }
+    size_t got = rc;
+    if (got != sizeof(int))
+      exit(9);
+    if (expect != number)
+      exit(10);
+    ++expect;
+  }
+  return 0;
+}
+
+int main(int argc, char *argv[]) {
+
+  if (pipe2(fds, O_NONBLOCK))
+    return 1;
+
+  pthread_t th;
+  if (pthread_create(&th, 0, worker, 0))
+    return 2;
+
+  int number = 0;
+  for (;;) {
+    int chunk = 0;
+    int numbers[ASYMMETRY];
+    for (;;) {
+      numbers[chunk] = number + chunk;
+      if (++chunk == ASYMMETRY)
+        break;
+      if (number + chunk == ITERATIONS)
+        break;
+    }
+    for (;;) {
+      ssize_t rc = write(fds[1], numbers, chunk * sizeof(int));
+      if (rc == -1) {
+        if (errno == EAGAIN) {
+          ++got_write_eagains;
+          if (poll(&(struct pollfd){fds[1], POLLOUT}, 1, -1) == -1)
+            return 10;
+          continue;
+        }
+        return 3;
+      }
+      if (rc % sizeof(int))
+        return 4;
+      chunk = rc / sizeof(int);
+      number += chunk;
+      break;
+    }
+    if (number == ITERATIONS)
+      break;
+  }
+
+  if (pthread_join(th, 0))
+    return 5;
+
+  if (!got_read_eagains && !got_write_eagains)
+    return 7;
+}
diff --git a/test/posix/sa_resethand_test.c b/test/posix/sa_resethand_test.c
index c582d90fe19..eed501bf6ea 100644
--- a/test/posix/sa_resethand_test.c
+++ b/test/posix/sa_resethand_test.c
@@ -1,22 +1,21 @@
-/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
-│ vi: set et ft=c ts=2 sts=2 sw=2 fenc=utf-8                               :vi │
-╞══════════════════════════════════════════════════════════════════════════════╡
-│ Copyright 2023 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.                                                │
-╚─────────────────────────────────────────────────────────────────────────────*/
+// Copyright 2024 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 <errno.h>
 #include <signal.h>
+#include <stdlib.h>
 
 volatile int gotsig;
 
@@ -24,23 +23,68 @@ void OnSig(int sig) {
   gotsig = sig;
 }
 
-int main() {
+void test_sa_resethand_raise(void) {
   struct sigaction sa;
   sa.sa_handler = OnSig;
   sa.sa_flags = SA_RESETHAND;
   sigemptyset(&sa.sa_mask);
   if (sigaction(SIGUSR1, &sa, 0))
-    return 1;
+    exit(1);
   if (sigaction(SIGUSR1, 0, &sa))
-    return 2;
+    exit(2);
   if (sa.sa_handler != OnSig)
-    return 3;
+    exit(3);
   if (raise(SIGUSR1))
-    return 4;
+    exit(4);
   if (gotsig != SIGUSR1)
-    return 5;
+    exit(5);
   if (sigaction(SIGUSR1, 0, &sa))
-    return 6;
+    exit(6);
+  if (sa.sa_handler != SIG_DFL)
+    exit(7);
+}
+
+void test_sa_resethand_pause(void) {
+  struct sigaction sa;
+  sa.sa_handler = OnSig;
+  sa.sa_flags = SA_RESETHAND;
+  sigemptyset(&sa.sa_mask);
+  if (sigaction(SIGALRM, &sa, 0))
+    exit(10);
+  ualarm(10000, 0);
+  if (pause() != -1 || errno != EINTR)
+    exit(11);
+  if (gotsig != SIGALRM)
+    exit(12);
+  if (sigaction(SIGALRM, 0, &sa))
+    exit(13);
   if (sa.sa_handler != SIG_DFL)
-    return 7;
+    exit(14);
+}
+
+void test_sa_resethand_read(void) {
+  struct sigaction sa;
+  sa.sa_handler = OnSig;
+  sa.sa_flags = SA_RESETHAND;
+  sigemptyset(&sa.sa_mask);
+  if (sigaction(SIGALRM, &sa, 0))
+    exit(20);
+  int fds[2];
+  if (pipe(fds))
+    exit(21);
+  ualarm(10000, 0);
+  if (read(fds[0], (char[]){0}, 1) != -1 || errno != EINTR)
+    exit(22);
+  if (gotsig != SIGALRM)
+    exit(23);
+  if (sigaction(SIGALRM, 0, &sa))
+    exit(24);
+  if (sa.sa_handler != SIG_DFL)
+    exit(25);
+}
+
+int main() {
+  test_sa_resethand_raise();
+  test_sa_resethand_pause();
+  test_sa_resethand_read();
 }
diff --git a/test/posix/signal_latency_async_test.c b/test/posix/signal_latency_async_test.c
new file mode 100644
index 00000000000..438f5214cb2
--- /dev/null
+++ b/test/posix/signal_latency_async_test.c
@@ -0,0 +1,148 @@
+// Copyright 2024 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 <pthread.h>
+#include <signal.h>
+#include <stdatomic.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <time.h>
+#include <unistd.h>
+
+#define ITERATIONS 10000
+
+pthread_t sender_thread;
+pthread_t receiver_thread;
+struct timespec send_time;
+atomic_int sender_got_signal;
+double latencies[ITERATIONS];
+
+void sender_signal_handler(int signo) {
+  sender_got_signal = 1;
+}
+
+void receiver_signal_handler(int signo) {
+  struct timespec receive_time;
+  clock_gettime(CLOCK_MONOTONIC, &receive_time);
+
+  long sec_diff = receive_time.tv_sec - send_time.tv_sec;
+  long nsec_diff = receive_time.tv_nsec - send_time.tv_nsec;
+  double latency_ns = sec_diff * 1e9 + nsec_diff;
+
+  static int iteration = 0;
+  if (iteration < ITERATIONS)
+    latencies[iteration++] = latency_ns;
+
+  // Pong sender
+  if (pthread_kill(sender_thread, SIGUSR2))
+    exit(2);
+
+  // Exit if done
+  if (iteration >= ITERATIONS)
+    pthread_exit(0);
+}
+
+void *sender_func(void *arg) {
+
+  for (int i = 0; i < ITERATIONS; i++) {
+
+    // Wait a bit sometimes
+    if (rand() % 2 == 1) {
+      volatile unsigned v = 0;
+      for (;;)
+        if (++v == 4000)
+          break;
+    }
+
+    // Ping receiver
+    clock_gettime(CLOCK_MONOTONIC, &send_time);
+    if (pthread_kill(receiver_thread, SIGUSR1))
+      exit(6);
+
+    // Wait for pong
+    for (;;)
+      if (atomic_load_explicit(&sender_got_signal, memory_order_relaxed))
+        break;
+    sender_got_signal = 0;
+  }
+
+  return 0;
+}
+
+void *receiver_func(void *arg) {
+
+  // Wait for asynchronous signals
+  volatile unsigned v = 0;
+  for (;;)
+    ++v;
+
+  return 0;
+}
+
+int compare(const void *a, const void *b) {
+  const double *x = a, *y = b;
+  if (*x < *y)
+    return -1;
+  else if (*x > *y)
+    return 1;
+  else
+    return 0;
+}
+
+int main() {
+
+  // Install signal handlers
+  struct sigaction sa;
+  sa.sa_handler = receiver_signal_handler;
+  sa.sa_flags = 0;
+  sigemptyset(&sa.sa_mask);
+  sigaction(SIGUSR1, &sa, 0);
+  sa.sa_handler = sender_signal_handler;
+  sigaction(SIGUSR2, &sa, 0);
+
+  // Create receiver thread first
+  if (pthread_create(&receiver_thread, 0, receiver_func, 0))
+    exit(11);
+
+  // Create sender thread
+  if (pthread_create(&sender_thread, 0, sender_func, 0))
+    exit(12);
+
+  // Wait for threads to finish
+  if (pthread_join(sender_thread, 0))
+    exit(13);
+  if (pthread_join(receiver_thread, 0))
+    exit(14);
+
+  // Compute mean latency
+  double total_latency = 0;
+  for (int i = 0; i < ITERATIONS; i++)
+    total_latency += latencies[i];
+  double mean_latency = total_latency / ITERATIONS;
+
+  // Sort latencies to compute percentiles
+  qsort(latencies, ITERATIONS, sizeof(double), compare);
+
+  double p50 = latencies[(int)(0.50 * ITERATIONS)];
+  double p90 = latencies[(int)(0.90 * ITERATIONS)];
+  double p95 = latencies[(int)(0.95 * ITERATIONS)];
+  double p99 = latencies[(int)(0.99 * ITERATIONS)];
+
+  printf("Mean latency: %.2f ns\n", mean_latency);
+  printf("50th percentile latency: %.2f ns\n", p50);
+  printf("90th percentile latency: %.2f ns\n", p90);
+  printf("95th percentile latency: %.2f ns\n", p95);
+  printf("99th percentile latency: %.2f ns\n", p99);
+}
diff --git a/test/posix/signal_latency_test.c b/test/posix/signal_latency_test.c
index 02929aa8c00..9f599f43809 100644
--- a/test/posix/signal_latency_test.c
+++ b/test/posix/signal_latency_test.c
@@ -22,7 +22,6 @@
 #include <stdlib.h>
 #include <time.h>
 #include <unistd.h>
-#include "libc/thread/posixthread.internal.h"
 
 #define ITERATIONS 10000
 
diff --git a/tool/net/redbean.c b/tool/net/redbean.c
index eeb2e311646..1000ffc2c56 100644
--- a/tool/net/redbean.c
+++ b/tool/net/redbean.c
@@ -2276,7 +2276,7 @@ static struct Asset *GetAssetZip(const char *path, size_t pathlen) {
   hash = Hash(path, pathlen);
   for (step = 0;; ++step) {
     i = (hash + ((step * (step + 1)) >> 1)) & (assets.n - 1);
-    if (!assets.p[i].hash)
+    if (i >= assets.n || !assets.p || !assets.p[i].hash)
       return NULL;
     if (hash == assets.p[i].hash &&
         pathlen == ZIP_CFILE_NAMESIZE(zmap + assets.p[i].cf) &&