diff options
author | Tavian Barnes <tavianator@tavianator.com> | 2021-04-15 14:00:11 -0400 |
---|---|---|
committer | Tavian Barnes <tavianator@tavianator.com> | 2021-04-15 14:00:11 -0400 |
commit | b7111f2f01dce9516aea5250e765124300f2fa3a (patch) | |
tree | 25a723baf00892f61d103029a983fec6b60bf550 /util.c | |
parent | 1d8bbdc1a59fd5246ec60bc3db1ece055ef83639 (diff) | |
parent | 3dad2125b9048fdc3790d3e7c4770f7174be889c (diff) | |
download | bfs-b7111f2f01dce9516aea5250e765124300f2fa3a.tar.xz |
Merge pull request #73 from markus-oberhumer/safe-read-write
Diffstat (limited to 'util.c')
-rw-r--r-- | util.c | 55 |
1 files changed, 55 insertions, 0 deletions
@@ -368,3 +368,58 @@ int bfs_minor(dev_t dev) { return dev & 0xFF; #endif } + +ssize_t safe_read(int fd, void *buf, size_t nbytes) { + for (;;) { + ssize_t ret = read(fd, buf, nbytes); + if (ret < 0 && errno == EINTR) { + continue; + } + return ret; + } +} + +ssize_t safe_read_all(int fd, void *buf, size_t nbytes) { + size_t count = 0; + for (;;) { + ssize_t ret = read(fd, (char *)buf + count, nbytes - count); + if (ret < 0 && errno == EINTR) { + continue; + } + if (ret < 0) { + return ret; // always return error < 0 + } + count += ret; + if (ret == 0 || count == nbytes) { // EOF or success + return count; + } + } +} + +ssize_t safe_write(int fd, const void *buf, size_t nbytes) { + for (;;) { + ssize_t ret = write(fd, buf, nbytes); + if (ret < 0 && errno == EINTR) { + continue; + } + return ret; + } +} + +ssize_t safe_write_all(int fd, const void *buf, size_t nbytes) +{ + size_t count = 0; + for (;;) { + ssize_t ret = write(fd, (const char *)buf + count, nbytes - count); + if (ret < 0 && errno == EINTR) { + continue; + } + if (ret < 0) { + return ret; // always return error < 0 + } + count += ret; + if (ret == 0 || count == nbytes) { // EOF (should never happen with write) or success + return count; + } + } +} |