| // SPDX-License-Identifier: GPL-2.0 | 
 | #include <linux/compiler.h> | 
 | #include <linux/kernel.h> | 
 | #include <sys/types.h> | 
 | #include <sys/stat.h> | 
 | #include <errno.h> | 
 | #include <unistd.h> | 
 | #include <string.h> | 
 |  | 
 | #include "data.h" | 
 | #include "util.h" | 
 | #include "debug.h" | 
 |  | 
 | #ifndef O_CLOEXEC | 
 | #ifdef __sparc__ | 
 | #define O_CLOEXEC	0x400000 | 
 | #elif defined(__alpha__) || defined(__hppa__) | 
 | #define O_CLOEXEC	010000000 | 
 | #else | 
 | #define O_CLOEXEC	02000000 | 
 | #endif | 
 | #endif | 
 |  | 
 | static bool check_pipe(struct perf_data_file *file) | 
 | { | 
 | 	struct stat st; | 
 | 	bool is_pipe = false; | 
 | 	int fd = perf_data_file__is_read(file) ? | 
 | 		 STDIN_FILENO : STDOUT_FILENO; | 
 |  | 
 | 	if (!file->path) { | 
 | 		if (!fstat(fd, &st) && S_ISFIFO(st.st_mode)) | 
 | 			is_pipe = true; | 
 | 	} else { | 
 | 		if (!strcmp(file->path, "-")) | 
 | 			is_pipe = true; | 
 | 	} | 
 |  | 
 | 	if (is_pipe) | 
 | 		file->fd = fd; | 
 |  | 
 | 	return file->is_pipe = is_pipe; | 
 | } | 
 |  | 
 | static int check_backup(struct perf_data_file *file) | 
 | { | 
 | 	struct stat st; | 
 |  | 
 | 	if (!stat(file->path, &st) && st.st_size) { | 
 | 		/* TODO check errors properly */ | 
 | 		char oldname[PATH_MAX]; | 
 | 		snprintf(oldname, sizeof(oldname), "%s.old", | 
 | 			 file->path); | 
 | 		unlink(oldname); | 
 | 		rename(file->path, oldname); | 
 | 	} | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int open_file_read(struct perf_data_file *file) | 
 | { | 
 | 	struct stat st; | 
 | 	int fd; | 
 | 	char sbuf[STRERR_BUFSIZE]; | 
 |  | 
 | 	fd = open(file->path, O_RDONLY); | 
 | 	if (fd < 0) { | 
 | 		int err = errno; | 
 |  | 
 | 		pr_err("failed to open %s: %s", file->path, | 
 | 			str_error_r(err, sbuf, sizeof(sbuf))); | 
 | 		if (err == ENOENT && !strcmp(file->path, "perf.data")) | 
 | 			pr_err("  (try 'perf record' first)"); | 
 | 		pr_err("\n"); | 
 | 		return -err; | 
 | 	} | 
 |  | 
 | 	if (fstat(fd, &st) < 0) | 
 | 		goto out_close; | 
 |  | 
 | 	if (!file->force && st.st_uid && (st.st_uid != geteuid())) { | 
 | 		pr_err("File %s not owned by current user or root (use -f to override)\n", | 
 | 		       file->path); | 
 | 		goto out_close; | 
 | 	} | 
 |  | 
 | 	if (!st.st_size) { | 
 | 		pr_info("zero-sized file (%s), nothing to do!\n", | 
 | 			file->path); | 
 | 		goto out_close; | 
 | 	} | 
 |  | 
 | 	file->size = st.st_size; | 
 | 	return fd; | 
 |  | 
 |  out_close: | 
 | 	close(fd); | 
 | 	return -1; | 
 | } | 
 |  | 
 | static int open_file_write(struct perf_data_file *file) | 
 | { | 
 | 	int fd; | 
 | 	char sbuf[STRERR_BUFSIZE]; | 
 |  | 
 | 	if (check_backup(file)) | 
 | 		return -1; | 
 |  | 
 | 	fd = open(file->path, O_CREAT|O_RDWR|O_TRUNC|O_CLOEXEC, | 
 | 		  S_IRUSR|S_IWUSR); | 
 |  | 
 | 	if (fd < 0) | 
 | 		pr_err("failed to open %s : %s\n", file->path, | 
 | 			str_error_r(errno, sbuf, sizeof(sbuf))); | 
 |  | 
 | 	return fd; | 
 | } | 
 |  | 
 | static int open_file(struct perf_data_file *file) | 
 | { | 
 | 	int fd; | 
 |  | 
 | 	fd = perf_data_file__is_read(file) ? | 
 | 	     open_file_read(file) : open_file_write(file); | 
 |  | 
 | 	file->fd = fd; | 
 | 	return fd < 0 ? -1 : 0; | 
 | } | 
 |  | 
 | int perf_data_file__open(struct perf_data_file *file) | 
 | { | 
 | 	if (check_pipe(file)) | 
 | 		return 0; | 
 |  | 
 | 	if (!file->path) | 
 | 		file->path = "perf.data"; | 
 |  | 
 | 	return open_file(file); | 
 | } | 
 |  | 
 | void perf_data_file__close(struct perf_data_file *file) | 
 | { | 
 | 	close(file->fd); | 
 | } | 
 |  | 
 | ssize_t perf_data_file__write(struct perf_data_file *file, | 
 | 			      void *buf, size_t size) | 
 | { | 
 | 	return writen(file->fd, buf, size); | 
 | } | 
 |  | 
 | int perf_data_file__switch(struct perf_data_file *file, | 
 | 			   const char *postfix, | 
 | 			   size_t pos, bool at_exit) | 
 | { | 
 | 	char *new_filepath; | 
 | 	int ret; | 
 |  | 
 | 	if (check_pipe(file)) | 
 | 		return -EINVAL; | 
 | 	if (perf_data_file__is_read(file)) | 
 | 		return -EINVAL; | 
 |  | 
 | 	if (asprintf(&new_filepath, "%s.%s", file->path, postfix) < 0) | 
 | 		return -ENOMEM; | 
 |  | 
 | 	/* | 
 | 	 * Only fire a warning, don't return error, continue fill | 
 | 	 * original file. | 
 | 	 */ | 
 | 	if (rename(file->path, new_filepath)) | 
 | 		pr_warning("Failed to rename %s to %s\n", file->path, new_filepath); | 
 |  | 
 | 	if (!at_exit) { | 
 | 		close(file->fd); | 
 | 		ret = perf_data_file__open(file); | 
 | 		if (ret < 0) | 
 | 			goto out; | 
 |  | 
 | 		if (lseek(file->fd, pos, SEEK_SET) == (off_t)-1) { | 
 | 			ret = -errno; | 
 | 			pr_debug("Failed to lseek to %zu: %s", | 
 | 				 pos, strerror(errno)); | 
 | 			goto out; | 
 | 		} | 
 | 	} | 
 | 	ret = file->fd; | 
 | out: | 
 | 	free(new_filepath); | 
 | 	return ret; | 
 | } |