Merge tag 'nios2-v4.14-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/lftan...
[sfrench/cifs-2.6.git] / tools / perf / util / data.c
1 #include <linux/compiler.h>
2 #include <linux/kernel.h>
3 #include <sys/types.h>
4 #include <sys/stat.h>
5 #include <errno.h>
6 #include <unistd.h>
7 #include <string.h>
8
9 #include "data.h"
10 #include "util.h"
11 #include "debug.h"
12
13 #ifndef O_CLOEXEC
14 #ifdef __sparc__
15 #define O_CLOEXEC       0x400000
16 #elif defined(__alpha__) || defined(__hppa__)
17 #define O_CLOEXEC       010000000
18 #else
19 #define O_CLOEXEC       02000000
20 #endif
21 #endif
22
23 static bool check_pipe(struct perf_data_file *file)
24 {
25         struct stat st;
26         bool is_pipe = false;
27         int fd = perf_data_file__is_read(file) ?
28                  STDIN_FILENO : STDOUT_FILENO;
29
30         if (!file->path) {
31                 if (!fstat(fd, &st) && S_ISFIFO(st.st_mode))
32                         is_pipe = true;
33         } else {
34                 if (!strcmp(file->path, "-"))
35                         is_pipe = true;
36         }
37
38         if (is_pipe)
39                 file->fd = fd;
40
41         return file->is_pipe = is_pipe;
42 }
43
44 static int check_backup(struct perf_data_file *file)
45 {
46         struct stat st;
47
48         if (!stat(file->path, &st) && st.st_size) {
49                 /* TODO check errors properly */
50                 char oldname[PATH_MAX];
51                 snprintf(oldname, sizeof(oldname), "%s.old",
52                          file->path);
53                 unlink(oldname);
54                 rename(file->path, oldname);
55         }
56
57         return 0;
58 }
59
60 static int open_file_read(struct perf_data_file *file)
61 {
62         struct stat st;
63         int fd;
64         char sbuf[STRERR_BUFSIZE];
65
66         fd = open(file->path, O_RDONLY);
67         if (fd < 0) {
68                 int err = errno;
69
70                 pr_err("failed to open %s: %s", file->path,
71                         str_error_r(err, sbuf, sizeof(sbuf)));
72                 if (err == ENOENT && !strcmp(file->path, "perf.data"))
73                         pr_err("  (try 'perf record' first)");
74                 pr_err("\n");
75                 return -err;
76         }
77
78         if (fstat(fd, &st) < 0)
79                 goto out_close;
80
81         if (!file->force && st.st_uid && (st.st_uid != geteuid())) {
82                 pr_err("File %s not owned by current user or root (use -f to override)\n",
83                        file->path);
84                 goto out_close;
85         }
86
87         if (!st.st_size) {
88                 pr_info("zero-sized file (%s), nothing to do!\n",
89                         file->path);
90                 goto out_close;
91         }
92
93         file->size = st.st_size;
94         return fd;
95
96  out_close:
97         close(fd);
98         return -1;
99 }
100
101 static int open_file_write(struct perf_data_file *file)
102 {
103         int fd;
104         char sbuf[STRERR_BUFSIZE];
105
106         if (check_backup(file))
107                 return -1;
108
109         fd = open(file->path, O_CREAT|O_RDWR|O_TRUNC|O_CLOEXEC,
110                   S_IRUSR|S_IWUSR);
111
112         if (fd < 0)
113                 pr_err("failed to open %s : %s\n", file->path,
114                         str_error_r(errno, sbuf, sizeof(sbuf)));
115
116         return fd;
117 }
118
119 static int open_file(struct perf_data_file *file)
120 {
121         int fd;
122
123         fd = perf_data_file__is_read(file) ?
124              open_file_read(file) : open_file_write(file);
125
126         file->fd = fd;
127         return fd < 0 ? -1 : 0;
128 }
129
130 int perf_data_file__open(struct perf_data_file *file)
131 {
132         if (check_pipe(file))
133                 return 0;
134
135         if (!file->path)
136                 file->path = "perf.data";
137
138         return open_file(file);
139 }
140
141 void perf_data_file__close(struct perf_data_file *file)
142 {
143         close(file->fd);
144 }
145
146 ssize_t perf_data_file__write(struct perf_data_file *file,
147                               void *buf, size_t size)
148 {
149         return writen(file->fd, buf, size);
150 }
151
152 int perf_data_file__switch(struct perf_data_file *file,
153                            const char *postfix,
154                            size_t pos, bool at_exit)
155 {
156         char *new_filepath;
157         int ret;
158
159         if (check_pipe(file))
160                 return -EINVAL;
161         if (perf_data_file__is_read(file))
162                 return -EINVAL;
163
164         if (asprintf(&new_filepath, "%s.%s", file->path, postfix) < 0)
165                 return -ENOMEM;
166
167         /*
168          * Only fire a warning, don't return error, continue fill
169          * original file.
170          */
171         if (rename(file->path, new_filepath))
172                 pr_warning("Failed to rename %s to %s\n", file->path, new_filepath);
173
174         if (!at_exit) {
175                 close(file->fd);
176                 ret = perf_data_file__open(file);
177                 if (ret < 0)
178                         goto out;
179
180                 if (lseek(file->fd, pos, SEEK_SET) == (off_t)-1) {
181                         ret = -errno;
182                         pr_debug("Failed to lseek to %zu: %s",
183                                  pos, strerror(errno));
184                         goto out;
185                 }
186         }
187         ret = file->fd;
188 out:
189         free(new_filepath);
190         return ret;
191 }