5 * This code is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License version 2 only, as
7 * published by the Free Software Foundation.
8 *
9 * This code is distributed in the hope that it will be useful, but WITHOUT
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12 * version 2 for more details (a copy is included in the LICENSE file that
13 * accompanied this code).
14 *
15 * You should have received a copy of the GNU General Public License version
16 * 2 along with this work; if not, write to the Free Software Foundation,
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18 *
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20 * or visit www.oracle.com if you need additional information or have any
21 * questions.
22 *
23 */
24
25 #include "runtime/arguments.hpp"
26 #include "libperfstat_aix.hpp"
27
28 // For dlopen and friends
29 #include <fcntl.h>
30
31 // handle to the libperfstat
32 static void* g_libhandle = NULL;
33
34 // whether initialization worked
35 static bool g_initialized = false;
36
37
38 typedef int (*fun_perfstat_cpu_total_t) (perfstat_id_t *name, perfstat_cpu_total_t* userbuff,
39 int sizeof_userbuff, int desired_number);
40
41 typedef int (*fun_perfstat_memory_total_t) (perfstat_id_t *name, perfstat_memory_total_t* userbuff,
42 int sizeof_userbuff, int desired_number);
43
44 typedef void (*fun_perfstat_reset_t) ();
45
46 static fun_perfstat_cpu_total_t g_fun_perfstat_cpu_total = NULL;
47 static fun_perfstat_memory_total_t g_fun_perfstat_memory_total = NULL;
48 static fun_perfstat_reset_t g_fun_perfstat_reset = NULL;
49
50 bool libperfstat::init() {
51
52 if (g_initialized) {
53 return true;
54 }
55
56 g_initialized = false;
57
58 // dynamically load the libperfstat porting library.
59 g_libhandle = dlopen("/usr/lib/libperfstat.a(shr_64.o)", RTLD_MEMBER | RTLD_NOW);
60 if (!g_libhandle) {
61 if (Verbose) {
62 fprintf(stderr, "Cannot load libperfstat.a (dlerror: %s)", dlerror());
63 }
64 return false;
65 }
66
67 // resolve function pointers
68
69 #define RESOLVE_FUN_NO_ERROR(name) \
70 g_fun_##name = (fun_##name##_t) dlsym(g_libhandle, #name);
71
72 #define RESOLVE_FUN(name) \
73 RESOLVE_FUN_NO_ERROR(name) \
74 if (!g_fun_##name) { \
75 if (Verbose) { \
76 fprintf(stderr, "Cannot resolve " #name "() from libperfstat.a\n" \
77 " (dlerror: %s)", dlerror()); \
78 } \
79 return false; \
80 }
81
82 RESOLVE_FUN(perfstat_cpu_total);
83 RESOLVE_FUN(perfstat_memory_total);
84 RESOLVE_FUN(perfstat_reset);
85
86 g_initialized = true;
87
88 return true;
89 }
90
91 void libperfstat::cleanup() {
92
93 g_initialized = false;
94
95 if (g_libhandle) {
96 dlclose(g_libhandle);
97 g_libhandle = NULL;
98 }
99
100 g_fun_perfstat_cpu_total = NULL;
101 g_fun_perfstat_memory_total = NULL;
102 g_fun_perfstat_reset = NULL;
103 }
104
105 int libperfstat::perfstat_memory_total(perfstat_id_t *name,
106 perfstat_memory_total_t* userbuff,
107 int sizeof_userbuff, int desired_number) {
108 assert(g_initialized, "libperfstat not initialized");
109 assert(g_fun_perfstat_memory_total, "");
110 return g_fun_perfstat_memory_total(name, userbuff, sizeof_userbuff, desired_number);
111 }
112
113 int libperfstat::perfstat_cpu_total(perfstat_id_t *name, perfstat_cpu_total_t* userbuff,
114 int sizeof_userbuff, int desired_number) {
115 assert(g_initialized, "libperfstat not initialized");
116 assert(g_fun_perfstat_cpu_total, "");
117 return g_fun_perfstat_cpu_total(name, userbuff, sizeof_userbuff, desired_number);
118 }
119
120 void libperfstat::perfstat_reset() {
121 assert(g_initialized, "libperfstat not initialized");
122 assert(g_fun_perfstat_reset, "");
123 g_fun_perfstat_reset();
124 }
|
5 * This code is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License version 2 only, as
7 * published by the Free Software Foundation.
8 *
9 * This code is distributed in the hope that it will be useful, but WITHOUT
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12 * version 2 for more details (a copy is included in the LICENSE file that
13 * accompanied this code).
14 *
15 * You should have received a copy of the GNU General Public License version
16 * 2 along with this work; if not, write to the Free Software Foundation,
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18 *
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20 * or visit www.oracle.com if you need additional information or have any
21 * questions.
22 *
23 */
24
25 #include "libperfstat_aix.hpp"
26 #include "misc_aix.hpp"
27
28 #include <dlfcn.h>
29 #include <sys/systemcfg.h>
30
31 // Handle to the libperfstat.
32 static void* g_libhandle = NULL;
33
34 typedef int (*fun_perfstat_cpu_total_t) (perfstat_id_t *name, PERFSTAT_CPU_TOTAL_T_LATEST* userbuff,
35 int sizeof_userbuff, int desired_number);
36
37 typedef int (*fun_perfstat_memory_total_t) (perfstat_id_t *name, perfstat_memory_total_t* userbuff,
38 int sizeof_userbuff, int desired_number);
39
40 typedef int (*fun_perfstat_partition_total_t) (perfstat_id_t *name,
41 PERFSTAT_PARTITON_TOTAL_T_LATEST* userbuff, int sizeof_userbuff,
42 int desired_number);
43
44 typedef int (*fun_perfstat_wpar_total_t) (perfstat_id_wpar_t *name,
45 PERFSTAT_WPAR_TOTAL_T_LATEST* userbuff, int sizeof_userbuff,
46 int desired_number);
47
48 typedef void (*fun_perfstat_reset_t) ();
49
50 typedef cid_t (*fun_wpar_getcid_t) ();
51
52 static fun_perfstat_cpu_total_t g_fun_perfstat_cpu_total = NULL;
53 static fun_perfstat_memory_total_t g_fun_perfstat_memory_total = NULL;
54 static fun_perfstat_partition_total_t g_fun_perfstat_partition_total = NULL;
55 static fun_perfstat_wpar_total_t g_fun_perfstat_wpar_total = NULL;
56 static fun_perfstat_reset_t g_fun_perfstat_reset = NULL;
57 static fun_wpar_getcid_t g_fun_wpar_getcid = NULL;
58
59 bool libperfstat::init() {
60
61 // Dynamically load the libperfstat porting library.
62 g_libhandle = dlopen("/usr/lib/libperfstat.a(shr_64.o)", RTLD_MEMBER | RTLD_NOW);
63 if (!g_libhandle) {
64 trcVerbose("Cannot load libperfstat.a (dlerror: %s)", dlerror());
65 return false;
66 }
67
68 // Resolve function pointers
69
70 #define RESOLVE_FUN_NO_ERROR(name) \
71 g_fun_##name = (fun_##name##_t) dlsym(g_libhandle, #name);
72
73 #define RESOLVE_FUN(name) \
74 RESOLVE_FUN_NO_ERROR(name) \
75 if (!g_fun_##name) { \
76 trcVerbose("Cannot resolve " #name "() from libperfstat.a\n" \
77 " (dlerror: %s)", dlerror()); \
78 return false; \
79 }
80
81 // These functions may or may not be there depending on the OS release.
82 RESOLVE_FUN_NO_ERROR(perfstat_partition_total);
83 RESOLVE_FUN_NO_ERROR(perfstat_wpar_total);
84 RESOLVE_FUN_NO_ERROR(wpar_getcid);
85
86 // These functions are required for every release.
87 RESOLVE_FUN(perfstat_cpu_total);
88 RESOLVE_FUN(perfstat_memory_total);
89 RESOLVE_FUN(perfstat_reset);
90
91 trcVerbose("libperfstat loaded.");
92
93 return true;
94 }
95
96 void libperfstat::cleanup() {
97
98 if (g_libhandle) {
99 dlclose(g_libhandle);
100 g_libhandle = NULL;
101 }
102
103 g_fun_perfstat_cpu_total = NULL;
104 g_fun_perfstat_memory_total = NULL;
105 g_fun_perfstat_partition_total = NULL;
106 g_fun_perfstat_wpar_total = NULL;
107 g_fun_perfstat_reset = NULL;
108 g_fun_wpar_getcid = NULL;
109
110 }
111
112 int libperfstat::perfstat_memory_total(perfstat_id_t *name,
113 perfstat_memory_total_t* userbuff,
114 int sizeof_userbuff, int desired_number) {
115 if (g_fun_perfstat_memory_total == NULL) {
116 return -1;
117 }
118 return g_fun_perfstat_memory_total(name, userbuff, sizeof_userbuff, desired_number);
119 }
120
121 int libperfstat::perfstat_cpu_total(perfstat_id_t *name, PERFSTAT_CPU_TOTAL_T_LATEST* userbuff,
122 int sizeof_userbuff, int desired_number) {
123 if (g_fun_perfstat_cpu_total == NULL) {
124 return -1;
125 }
126 return g_fun_perfstat_cpu_total(name, userbuff, sizeof_userbuff, desired_number);
127 }
128
129 int libperfstat::perfstat_partition_total(perfstat_id_t *name, PERFSTAT_PARTITON_TOTAL_T_LATEST* userbuff,
130 int sizeof_userbuff, int desired_number)
131 {
132 if (g_fun_perfstat_partition_total == NULL) {
133 return -1;
134 }
135 return g_fun_perfstat_partition_total(name, userbuff, sizeof_userbuff, desired_number);
136 }
137
138 int libperfstat::perfstat_wpar_total(perfstat_id_wpar_t *name, PERFSTAT_WPAR_TOTAL_T_LATEST* userbuff,
139 int sizeof_userbuff, int desired_number)
140 {
141 if (g_fun_perfstat_wpar_total == NULL) {
142 return -1;
143 }
144 return g_fun_perfstat_wpar_total(name, userbuff, sizeof_userbuff, desired_number);
145 }
146
147 void libperfstat::perfstat_reset() {
148 if (g_fun_perfstat_reset != NULL) {
149 g_fun_perfstat_reset();
150 }
151 }
152
153 cid_t libperfstat::wpar_getcid() {
154 if (g_fun_wpar_getcid == NULL) {
155 return (cid_t) -1;
156 }
157 return g_fun_wpar_getcid();
158 }
159
160
161 //////////////////// convenience functions, release-independent /////////////////////////////
162
163 // Excerpts from systemcfg.h definitions newer than AIX 5.2 and XLC 8
164 // SAPJVM MBS 2012-08-14 : on AIX 5.3/ xlC10 some of the definitions are available but
165 // e.g. PV_/ stuff is still missing so keep the definitions
166 //
167 // XLC 8 build is using the systemcfg.h from /usr/vacpp/include/sys instead of /usr/include/sys
168 // Missing values for the version field
169 #define PV_5_2 0x0F0001 /* Power PC 5 */
170 #define PV_5_3 0x0F0002 /* Power PC 5 */
171 #define PV_6 0x100000 /* Power PC 6 */
172 #define PV_6_1 0x100001 /* Power PC 6 DD1.x */
173 #define PV_7 0x200000 /* Power PC 7 */
174 #define PV_5_Compat 0x0F8000 /* Power PC 5 */
175 #define PV_6_Compat 0x108000 /* Power PC 6 */
176 #define PV_7_Compat 0x208000 /* Power PC 7 */
177 #define PV_8 0x300000 /* Power PC 8 */
178 #define PV_8_Compat 0x308000 /* Power PC 8 */
179
180
181 // Retrieve global cpu information.
182 bool libperfstat::get_cpuinfo(cpuinfo_t* pci) {
183
184 assert(pci, "get_cpuinfo: invalid parameter");
185 memset(pci, 0, sizeof(cpuinfo_t));
186
187 PERFSTAT_CPU_TOTAL_T_LATEST psct;
188 memset (&psct, '\0', sizeof(psct));
189
190 if (-1 == libperfstat::perfstat_cpu_total(NULL, &psct, sizeof(PERFSTAT_CPU_TOTAL_T_LATEST), 1)) {
191 if (-1 == libperfstat::perfstat_cpu_total(NULL, &psct, sizeof(perfstat_cpu_total_t_61), 1)) {
192 if (-1 == libperfstat::perfstat_cpu_total(NULL, &psct, sizeof(perfstat_cpu_total_t_53), 1)) {
193 if (-1 == libperfstat::perfstat_cpu_total(NULL, &psct, sizeof(perfstat_cpu_total_t_52), 1)) {
194 trcVerbose("perfstat_cpu_total() failed (errno=%d)", errno);
195 return false;
196 }
197 }
198 }
199 }
200
201 // Global cpu information.
202 strcpy (pci->description, psct.description);
203 pci->processorHZ = psct.processorHZ;
204 pci->ncpus = psct.ncpus;
205 for (int i = 0; i < 3; i++) {
206 pci->loadavg[i] = (double) psct.loadavg[i] / (1 << SBITS);
207 }
208
209 pci->user_clock_ticks = psct.user;
210 pci->sys_clock_ticks = psct.sys;
211 pci->idle_clock_ticks = psct.idle;
212 pci->wait_clock_ticks = psct.wait;
213
214 // Get the processor version from _system_configuration.
215 switch (_system_configuration.version) {
216 case PV_8:
217 strcpy(pci->version, "Power PC 8");
218 break;
219 case PV_7:
220 strcpy(pci->version, "Power PC 7");
221 break;
222 case PV_6_1:
223 strcpy(pci->version, "Power PC 6 DD1.x");
224 break;
225 case PV_6:
226 strcpy(pci->version, "Power PC 6");
227 break;
228 case PV_5:
229 strcpy(pci->version, "Power PC 5");
230 break;
231 case PV_5_2:
232 strcpy(pci->version, "Power PC 5_2");
233 break;
234 case PV_5_3:
235 strcpy(pci->version, "Power PC 5_3");
236 break;
237 case PV_5_Compat:
238 strcpy(pci->version, "PV_5_Compat");
239 break;
240 case PV_6_Compat:
241 strcpy(pci->version, "PV_6_Compat");
242 break;
243 case PV_7_Compat:
244 strcpy(pci->version, "PV_7_Compat");
245 break;
246 case PV_8_Compat:
247 strcpy(pci->version, "PV_8_Compat");
248 break;
249 default:
250 strcpy(pci->version, "unknown");
251 }
252
253 return true;
254
255 }
256
257 // Retrieve partition information.
258 bool libperfstat::get_partitioninfo(partitioninfo_t* ppi) {
259
260 assert(ppi, "get_partitioninfo: invalid parameter");
261 memset(ppi, 0, sizeof(partitioninfo_t));
262
263 PERFSTAT_PARTITON_TOTAL_T_LATEST pspt;
264 memset(&pspt, '\0', sizeof(pspt));
265
266 bool ame_details = true;
267
268 if (-1 == libperfstat::perfstat_partition_total(NULL, &pspt, sizeof(PERFSTAT_PARTITON_TOTAL_T_LATEST), 1)) {
269 if (-1 == libperfstat::perfstat_partition_total(NULL, &pspt, sizeof(perfstat_partition_total_t_71), 1)) {
270 ame_details = false;
271 if (-1 == libperfstat::perfstat_partition_total(NULL, &pspt, sizeof(perfstat_partition_total_t_61), 1)) {
272 if (-1 == libperfstat::perfstat_partition_total(NULL, &pspt, sizeof(perfstat_partition_total_t_53), 1)) {
273 if (-1 == libperfstat::perfstat_partition_total(NULL, &pspt, sizeof(perfstat_partition_total_t_53_5), 1)) {
274 trcVerbose("perfstat_partition_total() failed (errno=%d)", errno);
275 return false;
276 }
277 }
278 }
279 }
280 }
281
282 // partition type info
283 ppi->shared_enabled = pspt.type.b.shared_enabled;
284 ppi->smt_capable = pspt.type.b.smt_capable;
285 ppi->smt_enabled = pspt.type.b.smt_enabled;
286 ppi->lpar_capable = pspt.type.b.lpar_capable;
287 ppi->lpar_enabled = pspt.type.b.lpar_enabled;
288 ppi->dlpar_capable = pspt.type.b.dlpar_capable;
289 ppi->capped = pspt.type.b.capped;
290 ppi->kernel_is_64 = pspt.type.b.kernel_is_64;
291 ppi->pool_util_authority = pspt.type.b.pool_util_authority;
292 ppi->donate_capable = pspt.type.b.donate_capable;
293 ppi->donate_enabled = pspt.type.b.donate_enabled;
294 ppi->ams_capable = pspt.type.b.ams_capable;
295 ppi->ams_enabled = pspt.type.b.ams_enabled;
296 ppi->power_save = pspt.type.b.power_save;
297 ppi->ame_enabled = pspt.type.b.ame_enabled;
298
299 // Partition total info.
300 ppi->online_cpus = pspt.online_cpus;
301 ppi->entitled_proc_capacity = pspt.entitled_proc_capacity;
302 ppi->var_proc_capacity_weight = pspt.var_proc_capacity_weight;
303 ppi->phys_cpus_pool = pspt.phys_cpus_pool;
304 ppi->pool_id = pspt.pool_id;
305 ppi->entitled_pool_capacity = pspt.entitled_pool_capacity;
306 strcpy(ppi->name, pspt.name);
307
308 // Added values to ppi that we need for later computation of cpu utilization
309 // ( pool authorization needed for pool_idle_time ??? )
310 ppi->timebase_last = pspt.timebase_last;
311 ppi->pool_idle_time = pspt.pool_idle_time;
312 ppi->pcpu_tics_user = pspt.puser;
313 ppi->pcpu_tics_sys = pspt.psys;
314 ppi->pcpu_tics_idle = pspt.pidle;
315 ppi->pcpu_tics_wait = pspt.pwait;
316
317 // Additional AME information.
318 if(ame_details) {
319 ppi->true_memory = pspt.true_memory * 4096;
320 ppi->expanded_memory = pspt.expanded_memory * 4096;
321 ppi->target_memexp_factr = pspt.target_memexp_factr;
322 ppi->current_memexp_factr = pspt.current_memexp_factr;
323 ppi->cmcs_total_time = pspt.cmcs_total_time;
324 }
325
326 return true;
327
328 }
329
330 // Retrieve wpar information.
331 bool libperfstat::get_wparinfo(wparinfo_t* pwi) {
332
333 assert(pwi, "get_wparinfo: invalid parameter");
334 memset(pwi, 0, sizeof(wparinfo_t));
335
336 if (libperfstat::wpar_getcid() <= 0) {
337 return false;
338 }
339
340 PERFSTAT_WPAR_TOTAL_T_LATEST pswt;
341 memset (&pswt, '\0', sizeof(pswt));
342
343 if (-1 == libperfstat::perfstat_wpar_total(NULL, &pswt, sizeof(PERFSTAT_WPAR_TOTAL_T_LATEST), 1)) {
344 if (-1 == libperfstat::perfstat_wpar_total(NULL, &pswt, sizeof(perfstat_wpar_total_t_61), 1)) {
345 trcVerbose("perfstat_wpar_total() failed (errno=%d)", errno);
346 return false;
347 }
348 }
349
350 // WPAR type info.
351 pwi->app_wpar = pswt.type.b.app_wpar;
352 pwi->cpu_rset = pswt.type.b.cpu_rset;
353 pwi->cpu_xrset = pswt.type.b.cpu_xrset;
354 pwi->cpu_limits = pswt.type.b.cpu_limits;
355 pwi->mem_limits = pswt.type.b.mem_limits;
356 // WPAR total info.
357 strcpy(pwi->name, pswt.name);
358 pwi->wpar_id = pswt.wpar_id;
359 pwi->cpu_limit = pswt.cpu_limit;
360 pwi->mem_limit = pswt.mem_limit;
361
362 return true;
363
364 }
365
366
367
368
|