Move some functions around to improve overall structure
[lttv.git] / lttv / lttv / sync / graph_functions.c
1 /* This file is part of the Linux Trace Toolkit viewer
2 * Copyright (C) 2009 Benjamin Poirier <benjamin.poirier@polymtl.ca>
3 *
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License Version 2 as
6 * published by the Free Software Foundation;
7 *
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
12 *
13 * You should have received a copy of the GNU General Public License
14 * along with this program; if not, write to the Free Software
15 * Foundation, Inc., 59 Temple Place - Suite 330, Boston,
16 * MA 02111-1307, USA.
17 */
18
19 #ifdef HAVE_CONFIG_H
20 #include <config.h>
21 #endif
22
23 #include <errno.h>
24 #include <fcntl.h>
25 #include <stdlib.h>
26 #include <sys/stat.h>
27 #include <sys/types.h>
28 #include <unistd.h>
29
30 #include "sync_chain.h"
31 #include "graph_functions.h"
32
33
34 /*
35 * Create the directory used to hold graphs and the header of the gnuplot
36 * script.
37 *
38 * Args:
39 * graphsDir: name of directory
40 *
41 * Returns:
42 * The current working directory before the execution of the function. The
43 * string must be free'd by the caller.
44 */
45 FILE* createGraphsDir(const char* const graphsDir)
46 {
47 char* cwd;
48 int graphsFp;
49 FILE* result;
50 int retval;
51
52 cwd= changeToGraphsDir(graphsDir);
53
54 if ((graphsFp= open("graphs.gnu", O_WRONLY | O_CREAT | O_TRUNC, S_IRUSR |
55 S_IWUSR | S_IXUSR | S_IRGRP | S_IWGRP | S_IXGRP | S_IROTH
56 | S_IWOTH | S_IXOTH)) == -1)
57 {
58 g_error(strerror(errno));
59 }
60 if ((result= fdopen(graphsFp, "w")) == NULL)
61 {
62 g_error(strerror(errno));
63 }
64
65 fprintf(result,
66 "#!/usr/bin/gnuplot\n\n"
67 "set terminal postscript eps color size 8in,6in\n");
68
69 retval= chdir(cwd);
70 if (retval == -1)
71 {
72 g_error(strerror(errno));
73 }
74 free(cwd);
75
76 return result;
77 }
78
79
80 /*
81 * Change to the directory used to hold graphs. Create it if necessary.
82 *
83 * Args:
84 * graphsDir: name of directory
85 *
86 * Returns:
87 * The current working directory before the execution of the function. The
88 * string must be free'd by the caller.
89 */
90 char* changeToGraphsDir(const char* const graphsDir)
91 {
92 int retval;
93 char* cwd;
94
95 cwd= getcwd(NULL, 0);
96 if (cwd == NULL)
97 {
98 g_error(strerror(errno));
99 }
100 while ((retval= chdir(graphsDir)) != 0)
101 {
102 if (errno == ENOENT)
103 {
104 retval= mkdir(graphsDir, S_IRUSR | S_IWUSR | S_IXUSR | S_IRGRP |
105 S_IWGRP | S_IXGRP | S_IROTH | S_IWOTH | S_IXOTH);
106 if (retval != 0)
107 {
108 g_error(strerror(errno));
109 }
110 }
111 else
112 {
113 g_error(strerror(errno));
114 }
115 }
116
117 return cwd;
118 }
119
120
121 /*
122 * Call each graph variable, option and plot line function of each module to
123 * produce a gnuplot script.
124 *
125 * Args:
126 * syncState: container for synchronization data
127 */
128 void writeGraphsScript(SyncState* const syncState)
129 {
130 unsigned int i, j, k, l, m;
131 long pos1, pos2;
132 const GraphFunctions* moduleGraphFunctions[]= {
133 &syncState->processingModule->graphFunctions,
134 &syncState->matchingModule->graphFunctions,
135 &syncState->analysisModule->graphFunctions,
136 };
137 const struct {
138 char* name;
139 size_t plotsOffsets[2];
140 size_t optionsOffset;
141 } graphTypes[]= {
142 {
143 "TraceTrace",
144 {
145 offsetof(GraphFunctions, writeTraceTraceBackPlots),
146 offsetof(GraphFunctions, writeTraceTraceForePlots),
147 },
148 offsetof(GraphFunctions, writeTraceTraceOptions),
149 },
150 {
151 "TraceTime",
152 {
153 offsetof(GraphFunctions, writeTraceTimeBackPlots),
154 offsetof(GraphFunctions, writeTraceTimeForePlots),
155 },
156 offsetof(GraphFunctions, writeTraceTimeOptions),
157 },
158 };
159
160 fprintf(syncState->graphsStream, "\n");
161
162 // Write variables
163 pos1= ftell(syncState->graphsStream);
164 for (i= 0; i < syncState->traceNb; i++)
165 {
166 for (k= 0; k < sizeof(moduleGraphFunctions) /
167 sizeof(*moduleGraphFunctions); k++)
168 {
169 GraphVariableFunction** writeVariables= (void*)
170 moduleGraphFunctions[k] + offsetof(GraphFunctions,
171 writeVariables);
172
173 if (*writeVariables)
174 {
175 (*writeVariables)(syncState, i);
176 }
177 }
178 }
179 fflush(syncState->graphsStream);
180 pos2= ftell(syncState->graphsStream);
181 if (pos1 != pos2)
182 {
183 fprintf(syncState->graphsStream, "\n");
184 }
185
186 // Write plots and options
187 for (l= 0; l < sizeof(graphTypes) / sizeof(*graphTypes); l++)
188 {
189 // Cover the upper triangular matrix, i is the reference node.
190 for (i= 0; i < syncState->traceNb; i++)
191 {
192 for (j= i + 1; j < syncState->traceNb; j++)
193 {
194 long trunc;
195
196 fprintf(syncState->graphsStream,
197 "reset\n"
198 "set output \"%03d-%03d-%s.eps\"\n"
199 "plot \\\n", i, j, graphTypes[l].name);
200
201 pos1= ftell(syncState->graphsStream);
202
203 for (m= 0; m < sizeof(graphTypes[l].plotsOffsets) /
204 sizeof(*graphTypes[l].plotsOffsets); m++)
205 {
206 for (k= 0; k < sizeof(moduleGraphFunctions) /
207 sizeof(*moduleGraphFunctions); k++)
208 {
209 GraphFunction** writePlots= (void*)
210 moduleGraphFunctions[k] +
211 graphTypes[l].plotsOffsets[m];
212
213 if (*writePlots)
214 {
215 (*writePlots)(syncState, i, j);
216 }
217 }
218 }
219
220 fflush(syncState->graphsStream);
221 pos2= ftell(syncState->graphsStream);
222 if (pos1 != pos2)
223 {
224 // Remove the ", \\\n" from the last graph plot line
225 trunc= pos2 - 4;
226 }
227 else
228 {
229 // Remove the "plot \\\n" line to avoid creating an invalid
230 // gnuplot script
231 trunc= pos2 - 7;
232 }
233
234 if (ftruncate(fileno(syncState->graphsStream), trunc) == -1)
235 {
236 g_error(strerror(errno));
237 }
238 if (fseek(syncState->graphsStream, 0, SEEK_END) == -1)
239 {
240 g_error(strerror(errno));
241 }
242
243 fprintf(syncState->graphsStream,
244 "\nset output \"%03d-%03d-%s.eps\"\n"
245 "set title \"\"\n", i, j, graphTypes[l].name);
246
247 for (k= 0; k < sizeof(moduleGraphFunctions) /
248 sizeof(*moduleGraphFunctions); k++)
249 {
250 GraphFunction** writeOptions= (void*)
251 moduleGraphFunctions[k] + graphTypes[l].optionsOffset;
252
253 if (*writeOptions)
254 {
255 (*writeOptions)(syncState, i, j);
256 }
257 }
258
259 if (pos1 != pos2)
260 {
261 fprintf(syncState->graphsStream, "replot\n\n");
262 }
263 else
264 {
265 fprintf(syncState->graphsStream, "\n");
266 }
267 }
268 }
269 }
270 }
This page took 0.034179 seconds and 4 git commands to generate.