debug.c revision 1.5.6.1 1 1.5.6.1 yamt /* $NetBSD: debug.c,v 1.5.6.1 2012/04/17 00:05:30 yamt 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.5.6.1 yamt #include <stdio.h>
30 1.5.6.1 yamt #include <stdlib.h>
31 1.5.6.1 yamt #include <err.h>
32 1.5.6.1 yamt #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.5.6.1 yamt #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.5.6.1 yamt const char * const 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.5.6.1 yamt perfuse_opname(int opcode)
98 1.1 manu {
99 1.1 manu const struct perfuse_opcode *po;
100 1.1 manu
101 1.1 manu for (po = perfuse_opcode; po->opcode; po++) {
102 1.1 manu if (po->opcode == opcode)
103 1.1 manu return po->opname;
104 1.1 manu }
105 1.1 manu
106 1.1 manu return po->opname; /* "UNKNOWN" */
107 1.1 manu }
108 1.5.6.1 yamt
109 1.5.6.1 yamt char *
110 1.5.6.1 yamt perfuse_opdump_in(struct perfuse_state *ps, perfuse_msg_t *pm)
111 1.5.6.1 yamt {
112 1.5.6.1 yamt struct fuse_in_header *fih;
113 1.5.6.1 yamt static char buf[BUFSIZ] = "";
114 1.5.6.1 yamt
115 1.5.6.1 yamt fih = GET_INHDR(ps, pm);
116 1.5.6.1 yamt
117 1.5.6.1 yamt switch(fih->opcode) {
118 1.5.6.1 yamt case FUSE_LOOKUP:
119 1.5.6.1 yamt (void)snprintf(buf, sizeof(buf), "path = \"%s\"",
120 1.5.6.1 yamt _GET_INPAYLOAD(ps, pm, const char *));
121 1.5.6.1 yamt break;
122 1.5.6.1 yamt default:
123 1.5.6.1 yamt buf[0] = '\0';
124 1.5.6.1 yamt break;
125 1.5.6.1 yamt }
126 1.5.6.1 yamt
127 1.5.6.1 yamt return buf;
128 1.5.6.1 yamt }
129 1.5.6.1 yamt
130 1.5.6.1 yamt struct perfuse_trace *
131 1.5.6.1 yamt perfuse_trace_begin(struct perfuse_state *ps, puffs_cookie_t opc,
132 1.5.6.1 yamt perfuse_msg_t *pm)
133 1.5.6.1 yamt {
134 1.5.6.1 yamt struct perfuse_trace *pt;
135 1.5.6.1 yamt
136 1.5.6.1 yamt if ((pt = malloc(sizeof(*pt))) == NULL)
137 1.5.6.1 yamt DERR(EX_OSERR, "malloc failed");
138 1.5.6.1 yamt
139 1.5.6.1 yamt pt->pt_opcode = ps->ps_get_inhdr(pm)->opcode;
140 1.5.6.1 yamt pt->pt_status = inxchg;
141 1.5.6.1 yamt
142 1.5.6.1 yamt if (clock_gettime(CLOCK_REALTIME, &pt->pt_start) != 0)
143 1.5.6.1 yamt DERR(EX_OSERR, "clock_gettime failed");
144 1.5.6.1 yamt
145 1.5.6.1 yamt if (opc == 0)
146 1.5.6.1 yamt (void)strcpy(pt->pt_path, "");
147 1.5.6.1 yamt else
148 1.5.6.1 yamt (void)strlcpy(pt->pt_path,
149 1.5.6.1 yamt perfuse_node_path(opc),
150 1.5.6.1 yamt sizeof(pt->pt_path));
151 1.5.6.1 yamt
152 1.5.6.1 yamt (void)strlcpy(pt->pt_extra,
153 1.5.6.1 yamt perfuse_opdump_in(ps, pm),
154 1.5.6.1 yamt sizeof(pt->pt_extra));
155 1.5.6.1 yamt
156 1.5.6.1 yamt TAILQ_INSERT_TAIL(&ps->ps_trace, pt, pt_list);
157 1.5.6.1 yamt ps->ps_tracecount++;
158 1.5.6.1 yamt
159 1.5.6.1 yamt return pt;
160 1.5.6.1 yamt }
161 1.5.6.1 yamt
162 1.5.6.1 yamt void
163 1.5.6.1 yamt perfuse_trace_end(struct perfuse_state *ps, struct perfuse_trace *pt, int error)
164 1.5.6.1 yamt {
165 1.5.6.1 yamt if (clock_gettime(CLOCK_REALTIME, &pt->pt_end) != 0)
166 1.5.6.1 yamt DERR(EX_OSERR, "clock_gettime failed");
167 1.5.6.1 yamt
168 1.5.6.1 yamt pt->pt_status = done;
169 1.5.6.1 yamt pt->pt_error = error;
170 1.5.6.1 yamt
171 1.5.6.1 yamt while (ps->ps_tracecount > PERFUSE_TRACECOUNT_MAX) {
172 1.5.6.1 yamt struct perfuse_trace *fpt = TAILQ_FIRST(&ps->ps_trace);
173 1.5.6.1 yamt
174 1.5.6.1 yamt if (fpt == NULL || fpt->pt_status != done)
175 1.5.6.1 yamt break;
176 1.5.6.1 yamt
177 1.5.6.1 yamt TAILQ_REMOVE(&ps->ps_trace, fpt, pt_list);
178 1.5.6.1 yamt free(fpt);
179 1.5.6.1 yamt ps->ps_tracecount--;
180 1.5.6.1 yamt }
181 1.5.6.1 yamt }
182 1.5.6.1 yamt
183 1.5.6.1 yamt void
184 1.5.6.1 yamt perfuse_trace_dump(struct puffs_usermount *pu, FILE *fp)
185 1.5.6.1 yamt {
186 1.5.6.1 yamt struct perfuse_state *ps;
187 1.5.6.1 yamt struct perfuse_trace *pt;
188 1.5.6.1 yamt struct timespec ts_min[FUSE_OPCODE_MAX];
189 1.5.6.1 yamt struct timespec ts_max[FUSE_OPCODE_MAX];
190 1.5.6.1 yamt struct timespec ts_total[FUSE_OPCODE_MAX];
191 1.5.6.1 yamt int count[FUSE_OPCODE_MAX];
192 1.5.6.1 yamt uint64_t avg;
193 1.5.6.1 yamt int i;
194 1.5.6.1 yamt
195 1.5.6.1 yamt if (!(perfuse_diagflags & PDF_TRACE))
196 1.5.6.1 yamt return;
197 1.5.6.1 yamt
198 1.5.6.1 yamt ps = puffs_getspecific(pu);
199 1.5.6.1 yamt
200 1.5.6.1 yamt (void)ftruncate(fileno(fp), 0);
201 1.5.6.1 yamt (void)fseek(fp, 0, SEEK_SET);
202 1.5.6.1 yamt
203 1.5.6.1 yamt (void)memset(&ts_min, 0, sizeof(ts_min));
204 1.5.6.1 yamt (void)memset(&ts_max, 0, sizeof(ts_max));
205 1.5.6.1 yamt (void)memset(&ts_total, 0, sizeof(ts_total));
206 1.5.6.1 yamt (void)memset(&count, 0, sizeof(count));
207 1.5.6.1 yamt
208 1.5.6.1 yamt fprintf(fp, "Last %"PRId64" operations\n", ps->ps_tracecount);
209 1.5.6.1 yamt
210 1.5.6.1 yamt TAILQ_FOREACH(pt, &ps->ps_trace, pt_list) {
211 1.5.6.1 yamt const char *quote = pt->pt_path[0] != '\0' ? "\"" : "";
212 1.5.6.1 yamt
213 1.5.6.1 yamt fprintf(fp, "%lld.%09ld %s %s%s%s %s ",
214 1.5.6.1 yamt (long long)pt->pt_start.tv_sec, pt->pt_start.tv_nsec,
215 1.5.6.1 yamt perfuse_opname(pt->pt_opcode),
216 1.5.6.1 yamt quote, pt->pt_path, quote,
217 1.5.6.1 yamt pt->pt_extra);
218 1.5.6.1 yamt
219 1.5.6.1 yamt if (pt->pt_status == done) {
220 1.5.6.1 yamt struct timespec ts;
221 1.5.6.1 yamt
222 1.5.6.1 yamt ts.tv_sec = 0; /* delint */
223 1.5.6.1 yamt ts.tv_nsec = 0; /* delint */
224 1.5.6.1 yamt timespecsub(&pt->pt_end, &pt->pt_start, &ts);
225 1.5.6.1 yamt
226 1.5.6.1 yamt fprintf(fp, "error = %d elapsed = %lld.%09lu ",
227 1.5.6.1 yamt pt->pt_error, (long long)ts.tv_sec,
228 1.5.6.1 yamt ts.tv_nsec);
229 1.5.6.1 yamt
230 1.5.6.1 yamt count[pt->pt_opcode]++;
231 1.5.6.1 yamt timespecadd(&ts_total[pt->pt_opcode],
232 1.5.6.1 yamt &ts,
233 1.5.6.1 yamt &ts_total[pt->pt_opcode]);
234 1.5.6.1 yamt
235 1.5.6.1 yamt if (timespeccmp(&ts, &ts_min[pt->pt_opcode], <) ||
236 1.5.6.1 yamt (count[pt->pt_opcode] == 1))
237 1.5.6.1 yamt ts_min[pt->pt_opcode] = ts;
238 1.5.6.1 yamt
239 1.5.6.1 yamt if (timespeccmp(&ts, &ts_max[pt->pt_opcode], >))
240 1.5.6.1 yamt ts_max[pt->pt_opcode] = ts;
241 1.5.6.1 yamt } else {
242 1.5.6.1 yamt fprintf(fp, "ongoing ");
243 1.5.6.1 yamt }
244 1.5.6.1 yamt
245 1.5.6.1 yamt fprintf(fp, "\n");
246 1.5.6.1 yamt }
247 1.5.6.1 yamt
248 1.5.6.1 yamt fprintf(fp, "\nStatistics by operation\n");
249 1.5.6.1 yamt fprintf(fp, "operation\tcount\tmin\tavg\tmax\n");
250 1.5.6.1 yamt for (i = 0; i < FUSE_OPCODE_MAX; i++) {
251 1.5.6.1 yamt time_t min;
252 1.5.6.1 yamt
253 1.5.6.1 yamt if (count[i] != 0) {
254 1.5.6.1 yamt avg = timespec2ns(&ts_total[i]) / count[i];
255 1.5.6.1 yamt min = ts_min[i].tv_sec;
256 1.5.6.1 yamt } else {
257 1.5.6.1 yamt avg = 0;
258 1.5.6.1 yamt min = 0;
259 1.5.6.1 yamt }
260 1.5.6.1 yamt
261 1.5.6.1 yamt fprintf(fp, "%s\t%d\t%lld.%09ld\t%lld.%09ld\t%lld.%09ld\t\n",
262 1.5.6.1 yamt perfuse_opname(i), count[i],
263 1.5.6.1 yamt (long long)min, ts_min[i].tv_nsec,
264 1.5.6.1 yamt (long long)(time_t)(avg / 1000000000L),
265 1.5.6.1 yamt (long)(avg % 1000000000L),
266 1.5.6.1 yamt (long long)ts_max[i].tv_sec, ts_max[i].tv_nsec);
267 1.5.6.1 yamt }
268 1.5.6.1 yamt
269 1.5.6.1 yamt (void)fflush(fp);
270 1.5.6.1 yamt return;
271 1.5.6.1 yamt }
272