mirror of
https://github.com/morgan9e/systemd
synced 2026-04-15 00:47:10 +09:00
If the cleanup function returns the appropriate type, use that to reset the variable. For other functions (usually the foreign ones which return void), add an explicit value to reset to. This causes a bit of code churn, but I think it might be worth it. In a following patch static destructors will be called from a fuzzer, and this change allows them to be called multiple times. But I think such a change might help with detecting unitialized code reuse too. We hit various bugs like this, and things are more obvious when a pointer has been set to NULL. I was worried whether this change increases text size, but it doesn't seem to: -Dbuildtype=debug: before "tree-wide: return NULL from freeing functions": -rwxrwxr-x 1 zbyszek zbyszek 4117672 Feb 16 14:36 build/libsystemd.so.0.30.0* -rwxrwxr-x 1 zbyszek zbyszek 4494520 Feb 16 15:06 build/systemd* after "tree-wide: return NULL from freeing functions": -rwxrwxr-x 1 zbyszek zbyszek 4117672 Feb 16 14:36 build/libsystemd.so.0.30.0* -rwxrwxr-x 1 zbyszek zbyszek 4494576 Feb 16 15:10 build/systemd* now: -rwxrwxr-x 1 zbyszek zbyszek 4117672 Feb 16 14:36 build/libsystemd.so.0.30.0* -rwxrwxr-x 1 zbyszek zbyszek 4494640 Feb 16 15:15 build/systemd* -Dbuildtype=release: before "tree-wide: return NULL from freeing functions": -rwxrwxr-x 1 zbyszek zbyszek 5252256 Feb 14 14:47 build-rawhide/libsystemd.so.0.30.0* -rwxrwxr-x 1 zbyszek zbyszek 1834184 Feb 16 15:09 build-rawhide/systemd* after "tree-wide: return NULL from freeing functions": -rwxrwxr-x 1 zbyszek zbyszek 5252256 Feb 14 14:47 build-rawhide/libsystemd.so.0.30.0* -rwxrwxr-x 1 zbyszek zbyszek 1834184 Feb 16 15:10 build-rawhide/systemd* now: -rwxrwxr-x 1 zbyszek zbyszek 5252256 Feb 14 14:47 build-rawhide/libsystemd.so.0.30.0* -rwxrwxr-x 1 zbyszek zbyszek 1834184 Feb 16 15:16 build-rawhide/systemd* I would expect that the compiler would be able to elide the setting of a variable if the variable is never used again. And this seems to be the case: in optimized builds there is no change in size whatsoever. And the change in size in unoptimized build is negligible. Something strange is happening with size of libsystemd: it's bigger in optimized builds. Something to figure out, but unrelated to this patch.
109 lines
2.8 KiB
C
109 lines
2.8 KiB
C
/* SPDX-License-Identifier: LGPL-2.1-or-later */
|
|
#pragma once
|
|
|
|
#include <dirent.h>
|
|
#include <stdbool.h>
|
|
#include <stdio.h>
|
|
#include <sys/socket.h>
|
|
|
|
#include "macro.h"
|
|
|
|
/* Make sure we can distinguish fd 0 and NULL */
|
|
#define FD_TO_PTR(fd) INT_TO_PTR((fd)+1)
|
|
#define PTR_TO_FD(p) (PTR_TO_INT(p)-1)
|
|
|
|
int close_nointr(int fd);
|
|
int safe_close(int fd);
|
|
void safe_close_pair(int p[static 2]);
|
|
|
|
static inline int safe_close_above_stdio(int fd) {
|
|
if (fd < 3) /* Don't close stdin/stdout/stderr, but still invalidate the fd by returning -1 */
|
|
return -1;
|
|
|
|
return safe_close(fd);
|
|
}
|
|
|
|
void close_many(const int fds[], size_t n_fd);
|
|
|
|
int fclose_nointr(FILE *f);
|
|
FILE* safe_fclose(FILE *f);
|
|
DIR* safe_closedir(DIR *f);
|
|
|
|
static inline void closep(int *fd) {
|
|
safe_close(*fd);
|
|
}
|
|
|
|
static inline void close_pairp(int (*p)[2]) {
|
|
safe_close_pair(*p);
|
|
}
|
|
|
|
static inline void fclosep(FILE **f) {
|
|
safe_fclose(*f);
|
|
}
|
|
|
|
DEFINE_TRIVIAL_CLEANUP_FUNC_FULL(FILE*, pclose, NULL);
|
|
DEFINE_TRIVIAL_CLEANUP_FUNC_FULL(DIR*, closedir, NULL);
|
|
|
|
#define _cleanup_close_ _cleanup_(closep)
|
|
#define _cleanup_fclose_ _cleanup_(fclosep)
|
|
#define _cleanup_pclose_ _cleanup_(pclosep)
|
|
#define _cleanup_closedir_ _cleanup_(closedirp)
|
|
#define _cleanup_close_pair_ _cleanup_(close_pairp)
|
|
|
|
int fd_nonblock(int fd, bool nonblock);
|
|
int fd_cloexec(int fd, bool cloexec);
|
|
|
|
int close_all_fds(const int except[], size_t n_except);
|
|
|
|
int same_fd(int a, int b);
|
|
|
|
void cmsg_close_all(struct msghdr *mh);
|
|
|
|
bool fdname_is_valid(const char *s);
|
|
|
|
int fd_get_path(int fd, char **ret);
|
|
|
|
int move_fd(int from, int to, int cloexec);
|
|
|
|
enum {
|
|
ACQUIRE_NO_DEV_NULL = 1 << 0,
|
|
ACQUIRE_NO_MEMFD = 1 << 1,
|
|
ACQUIRE_NO_PIPE = 1 << 2,
|
|
ACQUIRE_NO_TMPFILE = 1 << 3,
|
|
ACQUIRE_NO_REGULAR = 1 << 4,
|
|
};
|
|
|
|
int acquire_data_fd(const void *data, size_t size, unsigned flags);
|
|
|
|
int fd_duplicate_data_fd(int fd);
|
|
|
|
int fd_move_above_stdio(int fd);
|
|
|
|
int rearrange_stdio(int original_input_fd, int original_output_fd, int original_error_fd);
|
|
|
|
static inline int make_null_stdio(void) {
|
|
return rearrange_stdio(-1, -1, -1);
|
|
}
|
|
|
|
/* Like TAKE_PTR() but for file descriptors, resetting them to -1 */
|
|
#define TAKE_FD(fd) \
|
|
({ \
|
|
int _fd_ = (fd); \
|
|
(fd) = -1; \
|
|
_fd_; \
|
|
})
|
|
|
|
/* Like free_and_replace(), but for file descriptors */
|
|
#define CLOSE_AND_REPLACE(a, b) \
|
|
({ \
|
|
int *_fdp_ = &(a); \
|
|
safe_close(*_fdp_); \
|
|
*_fdp_ = TAKE_FD(b); \
|
|
0; \
|
|
})
|
|
|
|
|
|
int fd_reopen(int fd, int flags);
|
|
|
|
int read_nr_open(void);
|