debug.c revision 1.9 1 1.9 manu /* $NetBSD: debug.c,v 1.9 2012/01/29 06:22:01 manu Exp $ */
2 1.1 manu
3 1.1 manu /*-
4 1.1 manu * Copyright (c) 2010 Emmanuel Dreyfus. All rights reserved.
5 1.1 manu *
6 1.1 manu * Redistribution and use in source and binary forms, with or without
7 1.1 manu * modification, are permitted provided that the following conditions
8 1.1 manu * are met:
9 1.1 manu * 1. Redistributions of source code must retain the above copyright
10 1.1 manu * notice, this list of conditions and the following disclaimer.
11 1.1 manu * 2. Redistributions in binary form must reproduce the above copyright
12 1.1 manu * notice, this list of conditions and the following disclaimer in the
13 1.1 manu * documentation and/or other materials provided with the distribution.
14 1.1 manu *
15 1.1 manu * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
16 1.1 manu * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
17 1.1 manu * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
18 1.1 manu * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
19 1.1 manu * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
20 1.1 manu * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
21 1.1 manu * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
22 1.1 manu * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
23 1.1 manu * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
24 1.1 manu * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
25 1.1 manu * POSSIBILITY OF SUCH DAMAGE.
26 1.1 manu */
27 1.1 manu
28 1.1 manu #include <puffs.h>
29 1.6 manu #include <stdio.h>
30 1.9 manu #include <stdlib.h>
31 1.9 manu #include <err.h>
32 1.9 manu #include <errno.h>
33 1.1 manu #include <sys/types.h>
34 1.1 manu
35 1.1 manu #include "perfuse_if.h"
36 1.6 manu #include "perfuse_priv.h"
37 1.1 manu #include "fuse.h"
38 1.1 manu
39 1.1 manu struct perfuse_opcode {
40 1.1 manu int opcode;
41 1.1 manu const char *opname;
42 1.1 manu };
43 1.1 manu
44 1.1 manu const struct perfuse_opcode perfuse_opcode[] = {
45 1.1 manu { FUSE_LOOKUP, "LOOKUP" },
46 1.1 manu { FUSE_FORGET, "FORGET" },
47 1.1 manu { FUSE_GETATTR, "GETATTR" },
48 1.1 manu { FUSE_SETATTR, "SETATTR" },
49 1.1 manu { FUSE_READLINK, "READLINK" },
50 1.1 manu { FUSE_SYMLINK, "SYMLINK" },
51 1.1 manu { FUSE_MKNOD, "MKNOD" },
52 1.1 manu { FUSE_MKDIR, "MKDIR" },
53 1.1 manu { FUSE_UNLINK, "UNLINK" },
54 1.1 manu { FUSE_RMDIR, "RMDIR" },
55 1.1 manu { FUSE_RENAME, "RENAME" },
56 1.1 manu { FUSE_LINK, "LINK" },
57 1.1 manu { FUSE_OPEN, "OPEN" },
58 1.1 manu { FUSE_READ, "READ" },
59 1.1 manu { FUSE_WRITE, "WRITE" },
60 1.1 manu { FUSE_STATFS, "STATFS" },
61 1.1 manu { FUSE_RELEASE, "RELEASE" },
62 1.1 manu { FUSE_FSYNC, "FSYNC" },
63 1.1 manu { FUSE_SETXATTR, "SETXATTR" },
64 1.1 manu { FUSE_GETXATTR, "GETXATTR" },
65 1.1 manu { FUSE_LISTXATTR, "LISTXATTR" },
66 1.1 manu { FUSE_REMOVEXATTR, "REMOVEXATTR" },
67 1.1 manu { FUSE_FLUSH, "FLUSH" },
68 1.1 manu { FUSE_INIT, "INIT" },
69 1.1 manu { FUSE_OPENDIR, "OPENDIR" },
70 1.1 manu { FUSE_READDIR, "READDIR" },
71 1.1 manu { FUSE_RELEASEDIR, "RELEASEDIR" },
72 1.1 manu { FUSE_FSYNCDIR, "FSYNCDIR" },
73 1.1 manu { FUSE_GETLK, "GETLK" },
74 1.1 manu { FUSE_SETLK, "SETLK" },
75 1.1 manu { FUSE_SETLKW, "SETLKW" },
76 1.1 manu { FUSE_ACCESS, "ACCESS" },
77 1.1 manu { FUSE_CREATE, "CREATE" },
78 1.1 manu { FUSE_INTERRUPT, "INTERRUPT" },
79 1.1 manu { FUSE_BMAP, "BMAP" },
80 1.1 manu { FUSE_DESTROY, "DESTROY" },
81 1.1 manu { FUSE_IOCTL, "IOCTL" },
82 1.1 manu { FUSE_POLL, "POLL" },
83 1.1 manu { FUSE_CUSE_INIT, "CUSE_INIT" },
84 1.1 manu { 0, "UNKNOWN" },
85 1.1 manu };
86 1.1 manu
87 1.3 manu const char *perfuse_qtypestr[] = {
88 1.3 manu "READDIR",
89 1.3 manu "READ",
90 1.3 manu "WRITE",
91 1.3 manu "AFTERWRITE",
92 1.5 manu "OPEN",
93 1.4 manu "AFTERXCHG"
94 1.3 manu };
95 1.2 manu
96 1.1 manu const char *
97 1.1 manu perfuse_opname(opcode)
98 1.1 manu int opcode;
99 1.1 manu {
100 1.1 manu const struct perfuse_opcode *po;
101 1.1 manu
102 1.1 manu for (po = perfuse_opcode; po->opcode; po++) {
103 1.1 manu if (po->opcode == opcode)
104 1.1 manu return po->opname;
105 1.1 manu }
106 1.1 manu
107 1.1 manu return po->opname; /* "UNKNOWN" */
108 1.1 manu }
109 1.6 manu
110 1.6 manu char *
111 1.6 manu perfuse_opdump_in(ps, pm)
112 1.6 manu struct perfuse_state *ps;
113 1.6 manu perfuse_msg_t *pm;
114 1.6 manu {
115 1.6 manu struct fuse_in_header *fih;
116 1.6 manu static char buf[BUFSIZ] = "";
117 1.6 manu
118 1.6 manu fih = GET_INHDR(ps, pm);
119 1.6 manu
120 1.6 manu switch(fih->opcode) {
121 1.6 manu case FUSE_LOOKUP:
122 1.6 manu (void)snprintf(buf, sizeof(buf), "path = \"%s\"",
123 1.6 manu _GET_INPAYLOAD(ps, pm, const char *));
124 1.6 manu break;
125 1.6 manu default:
126 1.6 manu buf[0] = '\0';
127 1.6 manu break;
128 1.6 manu }
129 1.6 manu
130 1.6 manu return buf;
131 1.6 manu }
132 1.6 manu
133 1.9 manu struct perfuse_trace *
134 1.9 manu perfuse_trace_begin(ps, opc, pm)
135 1.9 manu struct perfuse_state *ps;
136 1.9 manu puffs_cookie_t opc;
137 1.9 manu perfuse_msg_t *pm;
138 1.9 manu {
139 1.9 manu struct perfuse_trace *pt;
140 1.9 manu
141 1.9 manu if ((pt = malloc(sizeof(*pt))) == NULL)
142 1.9 manu DERR(EX_OSERR, "malloc failed");
143 1.9 manu
144 1.9 manu pt->pt_opcode = ps->ps_get_inhdr(pm)->opcode;
145 1.9 manu pt->pt_status = inxchg;
146 1.9 manu
147 1.9 manu if (clock_gettime(CLOCK_REALTIME, &pt->pt_start) != 0)
148 1.9 manu DERR(EX_OSERR, "clock_gettime failed");
149 1.9 manu
150 1.9 manu if (opc == 0)
151 1.9 manu (void)strcpy(pt->pt_path, "");
152 1.9 manu else
153 1.9 manu (void)strlcpy(pt->pt_path,
154 1.9 manu perfuse_node_path(opc),
155 1.9 manu sizeof(pt->pt_path));
156 1.9 manu
157 1.9 manu (void)strlcpy(pt->pt_extra,
158 1.9 manu perfuse_opdump_in(ps, pm),
159 1.9 manu sizeof(pt->pt_extra));
160 1.9 manu
161 1.9 manu TAILQ_INSERT_TAIL(&ps->ps_trace, pt, pt_list);
162 1.9 manu ps->ps_tracecount++;
163 1.9 manu
164 1.9 manu return pt;
165 1.9 manu }
166 1.9 manu
167 1.9 manu void
168 1.9 manu perfuse_trace_end(ps, pt, error)
169 1.9 manu struct perfuse_state *ps;
170 1.9 manu struct perfuse_trace *pt;
171 1.9 manu int error;
172 1.9 manu {
173 1.9 manu if (clock_gettime(CLOCK_REALTIME, &pt->pt_end) != 0)
174 1.9 manu DERR(EX_OSERR, "clock_gettime failed");
175 1.9 manu
176 1.9 manu pt->pt_status = done;
177 1.9 manu pt->pt_error = error;
178 1.9 manu
179 1.9 manu while (ps->ps_tracecount > PERFUSE_TRACECOUNT_MAX) {
180 1.9 manu struct perfuse_trace *fpt = TAILQ_FIRST(&ps->ps_trace);
181 1.9 manu
182 1.9 manu if (fpt == NULL || fpt->pt_status != done)
183 1.9 manu break;
184 1.9 manu
185 1.9 manu TAILQ_REMOVE(&ps->ps_trace, fpt, pt_list);
186 1.9 manu free(fpt);
187 1.9 manu ps->ps_tracecount--;
188 1.9 manu }
189 1.9 manu }
190 1.9 manu
191 1.6 manu void
192 1.6 manu perfuse_trace_dump(pu, fp)
193 1.6 manu struct puffs_usermount *pu;
194 1.6 manu FILE *fp;
195 1.6 manu {
196 1.6 manu struct perfuse_state *ps;
197 1.6 manu struct perfuse_trace *pt;
198 1.6 manu struct timespec ts_min[FUSE_OPCODE_MAX];
199 1.6 manu struct timespec ts_max[FUSE_OPCODE_MAX];
200 1.6 manu struct timespec ts_total[FUSE_OPCODE_MAX];
201 1.6 manu int count[FUSE_OPCODE_MAX];
202 1.6 manu uint64_t avg;
203 1.6 manu int i;
204 1.6 manu
205 1.6 manu if (!(perfuse_diagflags & PDF_TRACE))
206 1.6 manu return;
207 1.6 manu
208 1.6 manu ps = puffs_getspecific(pu);
209 1.6 manu
210 1.6 manu (void)ftruncate(fileno(fp), 0);
211 1.6 manu (void)fseek(fp, 0, SEEK_SET);
212 1.6 manu
213 1.6 manu (void)memset(&ts_min, 0, sizeof(ts_min));
214 1.6 manu (void)memset(&ts_max, 0, sizeof(ts_max));
215 1.6 manu (void)memset(&ts_total, 0, sizeof(ts_total));
216 1.6 manu (void)memset(&count, 0, sizeof(count));
217 1.6 manu
218 1.6 manu fprintf(fp, "Last %"PRId64" operations\n", ps->ps_tracecount);
219 1.6 manu
220 1.6 manu TAILQ_FOREACH(pt, &ps->ps_trace, pt_list) {
221 1.6 manu const char *quote = pt->pt_path[0] != '\0' ? "\"" : "";
222 1.6 manu
223 1.9 manu fprintf(fp, "%ld.%09ld %s %s%s%s %s ",
224 1.8 riz pt->pt_start.tv_sec, pt->pt_start.tv_nsec,
225 1.6 manu perfuse_opname(pt->pt_opcode),
226 1.6 manu quote, pt->pt_path, quote,
227 1.6 manu pt->pt_extra);
228 1.6 manu
229 1.6 manu if (pt->pt_status == done) {
230 1.6 manu struct timespec ts;
231 1.6 manu
232 1.6 manu ts.tv_sec = 0; /* delint */
233 1.6 manu ts.tv_nsec = 0; /* delint */
234 1.6 manu timespecsub(&pt->pt_end, &pt->pt_start, &ts);
235 1.6 manu
236 1.9 manu fprintf(fp, "error = %d elapsed = %ld.%09lu ",
237 1.8 riz pt->pt_error, ts.tv_sec, ts.tv_nsec);
238 1.6 manu
239 1.6 manu count[pt->pt_opcode]++;
240 1.6 manu timespecadd(&ts_total[pt->pt_opcode],
241 1.6 manu &ts,
242 1.6 manu &ts_total[pt->pt_opcode]);
243 1.6 manu
244 1.6 manu if (timespeccmp(&ts, &ts_min[pt->pt_opcode], <) ||
245 1.6 manu (count[pt->pt_opcode] == 1))
246 1.6 manu ts_min[pt->pt_opcode] = ts;
247 1.6 manu
248 1.6 manu if (timespeccmp(&ts, &ts_max[pt->pt_opcode], >))
249 1.6 manu ts_max[pt->pt_opcode] = ts;
250 1.6 manu } else {
251 1.6 manu fprintf(fp, "ongoing ");
252 1.6 manu }
253 1.6 manu
254 1.6 manu fprintf(fp, "\n");
255 1.6 manu }
256 1.6 manu
257 1.6 manu fprintf(fp, "\nStatistics by operation\n");
258 1.6 manu fprintf(fp, "operation\tcount\tmin\tavg\tmax\n");
259 1.6 manu for (i = 0; i < FUSE_OPCODE_MAX; i++) {
260 1.6 manu time_t min;
261 1.6 manu
262 1.6 manu if (count[i] != 0) {
263 1.6 manu avg = timespec2ns(&ts_total[i]) / count[i];
264 1.6 manu min = ts_min[i].tv_sec;
265 1.6 manu } else {
266 1.6 manu avg = 0;
267 1.6 manu min = 0;
268 1.6 manu }
269 1.6 manu
270 1.9 manu fprintf(fp, "%s\t%d\t%ld.%09ld\t%ld.%09ld\t%ld.%09ld\t\n",
271 1.6 manu perfuse_opname(i), count[i],
272 1.8 riz min, ts_min[i].tv_nsec,
273 1.8 riz (time_t)(avg / 1000000000L), (long)(avg % 1000000000L),
274 1.8 riz ts_max[i].tv_sec, ts_max[i].tv_nsec);
275 1.6 manu }
276 1.6 manu
277 1.6 manu (void)fflush(fp);
278 1.6 manu return;
279 1.6 manu }
280