VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/solaris/mp-solaris.cpp@ 74014

Last change on this file since 74014 was 69111, checked in by vboxsync, 7 years ago

(C) year

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 13.7 KB
Line 
1/* $Id: mp-solaris.cpp 69111 2017-10-17 14:26:02Z vboxsync $ */
2/** @file
3 * IPRT - Multiprocessor, Solaris.
4 */
5
6/*
7 * Copyright (C) 2008-2017 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#define LOG_GROUP RTLOGGROUP_DEFAULT
32#include <unistd.h>
33#include <stdio.h>
34#include <errno.h>
35#include <kstat.h>
36#include <sys/processor.h>
37
38#include <iprt/mp.h>
39#include <iprt/cpuset.h>
40#include <iprt/assert.h>
41#include <iprt/string.h>
42#include <iprt/alloc.h>
43#include <iprt/log.h>
44#include <iprt/once.h>
45#include <iprt/critsect.h>
46
47
48/*********************************************************************************************************************************
49* Global Variables *
50*********************************************************************************************************************************/
51/** Initialization serializing (rtMpSolarisOnce). */
52static RTONCE g_MpSolarisOnce = RTONCE_INITIALIZER;
53/** Critical section serializing access to kstat. */
54static RTCRITSECT g_MpSolarisCritSect;
55/** The kstat handle. */
56static kstat_ctl_t *g_pKsCtl;
57/** Array pointing to the cpu_info instances. */
58static kstat_t **g_papCpuInfo;
59/** The number of entries in g_papCpuInfo */
60static RTCPUID g_capCpuInfo;
61/** Array of core ids. */
62static uint64_t *g_pu64CoreIds;
63/** Number of entries in g_pu64CoreIds. */
64static size_t g_cu64CoreIds;
65/** Number of cores in the system. */
66static size_t g_cCores;
67
68
69/**
70 * Helper for getting the core ID for a given CPU/strand/hyperthread.
71 *
72 * @returns The core ID.
73 * @param idCpu The CPU ID instance.
74 */
75static inline uint64_t rtMpSolarisGetCoreId(RTCPUID idCpu)
76{
77 kstat_named_t *pStat = (kstat_named_t *)kstat_data_lookup(g_papCpuInfo[idCpu], (char *)"core_id");
78 Assert(pStat->data_type == KSTAT_DATA_LONG);
79 Assert(pStat->value.l >= 0);
80 AssertCompile(sizeof(uint64_t) >= sizeof(long)); /* Paranoia. */
81 return (uint64_t)pStat->value.l;
82}
83
84
85/**
86 * Populates 'g_pu64CoreIds' array with unique core identifiers in the system.
87 *
88 * @returns VBox status code.
89 */
90static int rtMpSolarisGetCoreIds(void)
91{
92 for (RTCPUID idCpu = 0; idCpu < g_capCpuInfo; idCpu++)
93 {
94 /*
95 * It is possible that the number of cores don't match the maximum number
96 * of cores possible on the system. Hence check if we have a valid cpu_info
97 * object. We don't want to break out if it's NULL, the array may be sparse
98 * in theory, see @bugref{8469}.
99 */
100 if (g_papCpuInfo[idCpu])
101 {
102 if (kstat_read(g_pKsCtl, g_papCpuInfo[idCpu], 0) != -1)
103 {
104 /* Strands/Hyperthreads share the same core ID. */
105 uint64_t u64CoreId = rtMpSolarisGetCoreId(idCpu);
106 bool fAddedCore = false;
107 for (RTCPUID i = 0; i < g_cCores; i++)
108 {
109 if (g_pu64CoreIds[i] == u64CoreId)
110 {
111 fAddedCore = true;
112 break;
113 }
114 }
115
116 if (!fAddedCore)
117 {
118 g_pu64CoreIds[g_cCores] = u64CoreId;
119 ++g_cCores;
120 }
121 }
122 else
123 return VERR_INTERNAL_ERROR_2;
124 }
125 }
126
127 return VINF_SUCCESS;
128}
129
130
131/**
132 * Run once function that initializes the kstats we need here.
133 *
134 * @returns IPRT status code.
135 * @param pvUser Unused.
136 */
137static DECLCALLBACK(int) rtMpSolarisOnce(void *pvUser)
138{
139 int rc = VINF_SUCCESS;
140 NOREF(pvUser);
141
142 /*
143 * Open kstat and find the cpu_info entries for each of the CPUs.
144 */
145 g_pKsCtl = kstat_open();
146 if (g_pKsCtl)
147 {
148 g_capCpuInfo = RTMpGetCount();
149 if (RT_LIKELY(g_capCpuInfo > 0))
150 {
151 g_papCpuInfo = (kstat_t **)RTMemAllocZ(g_capCpuInfo * sizeof(kstat_t *));
152 if (g_papCpuInfo)
153 {
154 g_cu64CoreIds = g_capCpuInfo;
155 g_pu64CoreIds = (uint64_t *)RTMemAllocZ(g_cu64CoreIds * sizeof(uint64_t));
156 if (g_pu64CoreIds)
157 {
158 rc = RTCritSectInit(&g_MpSolarisCritSect);
159 if (RT_SUCCESS(rc))
160 {
161 RTCPUID i = 0;
162 for (kstat_t *pKsp = g_pKsCtl->kc_chain; pKsp != NULL; pKsp = pKsp->ks_next)
163 {
164 if (!RTStrCmp(pKsp->ks_module, "cpu_info"))
165 {
166 AssertBreak(i < g_capCpuInfo);
167 g_papCpuInfo[i++] = pKsp;
168 /** @todo ks_instance == cpu_id (/usr/src/uts/common/os/cpu.c)? Check this and fix it ASAP. */
169 }
170 }
171
172 rc = rtMpSolarisGetCoreIds();
173 if (RT_SUCCESS(rc))
174 return VINF_SUCCESS;
175 else
176 Log(("rtMpSolarisGetCoreIds failed. rc=%Rrc\n", rc));
177 }
178
179 RTMemFree(g_pu64CoreIds);
180 g_pu64CoreIds = NULL;
181 }
182 else
183 rc = VERR_NO_MEMORY;
184
185 /* bail out, we failed. */
186 RTMemFree(g_papCpuInfo);
187 g_papCpuInfo = NULL;
188 }
189 else
190 rc = VERR_NO_MEMORY;
191 }
192 else
193 rc = VERR_CPU_IPE_1;
194 kstat_close(g_pKsCtl);
195 g_pKsCtl = NULL;
196 }
197 else
198 {
199 rc = RTErrConvertFromErrno(errno);
200 if (RT_SUCCESS(rc))
201 rc = VERR_INTERNAL_ERROR;
202 Log(("kstat_open() -> %d (%Rrc)\n", errno, rc));
203 }
204
205 return rc;
206}
207
208
209/**
210 * RTOnceEx() cleanup function.
211 *
212 * @param pvUser Unused.
213 * @param fLazyCleanUpOk Whether lazy cleanup is okay or not.
214 */
215static DECLCALLBACK(void) rtMpSolarisCleanUp(void *pvUser, bool fLazyCleanUpOk)
216{
217 if (g_pKsCtl)
218 kstat_close(g_pKsCtl);
219 RTMemFree(g_pu64CoreIds);
220 RTMemFree(g_papCpuInfo);
221}
222
223
224/**
225 * Worker for RTMpGetCurFrequency and RTMpGetMaxFrequency.
226 *
227 * @returns The desired frequency on success, 0 on failure.
228 *
229 * @param idCpu The CPU ID.
230 * @param pszStatName The cpu_info stat name.
231 */
232static uint64_t rtMpSolarisGetFrequency(RTCPUID idCpu, const char *pszStatName)
233{
234 uint64_t u64 = 0;
235 int rc = RTOnceEx(&g_MpSolarisOnce, rtMpSolarisOnce, rtMpSolarisCleanUp, NULL /* pvUser */);
236 if (RT_SUCCESS(rc))
237 {
238 if ( idCpu < g_capCpuInfo
239 && g_papCpuInfo[idCpu])
240 {
241 rc = RTCritSectEnter(&g_MpSolarisCritSect);
242 AssertRC(rc);
243 if (RT_SUCCESS(rc))
244 {
245 if (kstat_read(g_pKsCtl, g_papCpuInfo[idCpu], 0) != -1)
246 {
247 /* Solaris really need to fix their APIs. Explicitly cast for now. */
248 kstat_named_t *pStat = (kstat_named_t *)kstat_data_lookup(g_papCpuInfo[idCpu], (char*)pszStatName);
249 if (pStat)
250 {
251 Assert(pStat->data_type == KSTAT_DATA_UINT64 || pStat->data_type == KSTAT_DATA_LONG);
252 switch (pStat->data_type)
253 {
254 case KSTAT_DATA_UINT64: u64 = pStat->value.ui64; break; /* current_clock_Hz */
255 case KSTAT_DATA_INT32: u64 = pStat->value.i32; break; /* clock_MHz */
256
257 /* just in case... */
258 case KSTAT_DATA_UINT32: u64 = pStat->value.ui32; break;
259 case KSTAT_DATA_INT64: u64 = pStat->value.i64; break;
260 default:
261 AssertMsgFailed(("%d\n", pStat->data_type));
262 break;
263 }
264 }
265 else
266 Log(("kstat_data_lookup(%s) -> %d\n", pszStatName, errno));
267 }
268 else
269 Log(("kstat_read() -> %d\n", errno));
270 RTCritSectLeave(&g_MpSolarisCritSect);
271 }
272 }
273 else
274 Log(("invalid idCpu: %d (g_capCpuInfo=%d)\n", (int)idCpu, (int)g_capCpuInfo));
275 }
276
277 return u64;
278}
279
280
281RTDECL(uint32_t) RTMpGetCurFrequency(RTCPUID idCpu)
282{
283 return rtMpSolarisGetFrequency(idCpu, "current_clock_Hz") / 1000000;
284}
285
286
287RTDECL(uint32_t) RTMpGetMaxFrequency(RTCPUID idCpu)
288{
289 return rtMpSolarisGetFrequency(idCpu, "clock_MHz");
290}
291
292
293#if defined(RT_ARCH_SPARC) || defined(RT_ARCH_SPARC64)
294RTDECL(RTCPUID) RTMpCpuId(void)
295{
296 /** @todo implement RTMpCpuId on solaris/r3! */
297 return NIL_RTCPUID;
298}
299#endif
300
301
302RTDECL(int) RTMpCpuIdToSetIndex(RTCPUID idCpu)
303{
304 return idCpu < RTCPUSET_MAX_CPUS ? (int)idCpu : -1;
305}
306
307
308RTDECL(RTCPUID) RTMpCpuIdFromSetIndex(int iCpu)
309{
310 return (unsigned)iCpu < RTCPUSET_MAX_CPUS ? iCpu : NIL_RTCPUID;
311}
312
313
314RTDECL(RTCPUID) RTMpGetMaxCpuId(void)
315{
316 return RTMpGetCount() - 1;
317}
318
319
320RTDECL(bool) RTMpIsCpuPossible(RTCPUID idCpu)
321{
322 return idCpu != NIL_RTCPUID
323 && idCpu < (RTCPUID)RTMpGetCount();
324}
325
326
327RTDECL(bool) RTMpIsCpuOnline(RTCPUID idCpu)
328{
329 int iStatus = p_online(idCpu, P_STATUS);
330 return iStatus == P_ONLINE
331 || iStatus == P_NOINTR;
332}
333
334
335RTDECL(bool) RTMpIsCpuPresent(RTCPUID idCpu)
336{
337 int iStatus = p_online(idCpu, P_STATUS);
338 return iStatus != -1;
339}
340
341
342RTDECL(RTCPUID) RTMpGetCount(void)
343{
344 /*
345 * Solaris has sysconf.
346 */
347 int cCpus = sysconf(_SC_NPROCESSORS_MAX);
348 if (cCpus < 0)
349 cCpus = sysconf(_SC_NPROCESSORS_CONF);
350 return cCpus;
351}
352
353
354RTDECL(PRTCPUSET) RTMpGetSet(PRTCPUSET pSet)
355{
356 RTCpuSetEmpty(pSet);
357 int idCpu = RTMpGetCount();
358 while (idCpu-- > 0)
359 RTCpuSetAdd(pSet, idCpu);
360 return pSet;
361}
362
363
364RTDECL(RTCPUID) RTMpGetOnlineCount(void)
365{
366 /*
367 * Solaris has sysconf.
368 */
369 return sysconf(_SC_NPROCESSORS_ONLN);
370}
371
372
373RTDECL(PRTCPUSET) RTMpGetOnlineSet(PRTCPUSET pSet)
374{
375 RTCpuSetEmpty(pSet);
376 RTCPUID cCpus = RTMpGetCount();
377 for (RTCPUID idCpu = 0; idCpu < cCpus; idCpu++)
378 if (RTMpIsCpuOnline(idCpu))
379 RTCpuSetAdd(pSet, idCpu);
380 return pSet;
381}
382
383
384RTDECL(PRTCPUSET) RTMpGetPresentSet(PRTCPUSET pSet)
385{
386#ifdef RT_STRICT
387 RTCPUID cCpusPresent = 0;
388#endif
389 RTCpuSetEmpty(pSet);
390 RTCPUID cCpus = RTMpGetCount();
391 for (RTCPUID idCpu = 0; idCpu < cCpus; idCpu++)
392 if (RTMpIsCpuPresent(idCpu))
393 {
394 RTCpuSetAdd(pSet, idCpu);
395#ifdef RT_STRICT
396 cCpusPresent++;
397#endif
398 }
399 Assert(cCpusPresent == RTMpGetPresentCount());
400 return pSet;
401}
402
403
404RTDECL(RTCPUID) RTMpGetPresentCount(void)
405{
406 /*
407 * Solaris has sysconf.
408 */
409 return sysconf(_SC_NPROCESSORS_CONF);
410}
411
412
413RTDECL(RTCPUID) RTMpGetPresentCoreCount(void)
414{
415 return RTMpGetCoreCount();
416}
417
418
419RTDECL(RTCPUID) RTMpGetCoreCount(void)
420{
421 int rc = RTOnceEx(&g_MpSolarisOnce, rtMpSolarisOnce, rtMpSolarisCleanUp, NULL /* pvUser */);
422 if (RT_SUCCESS(rc))
423 return g_cCores;
424
425 return 0;
426}
427
428
429RTDECL(RTCPUID) RTMpGetOnlineCoreCount(void)
430{
431 RTCPUID uOnlineCores = 0;
432 int rc = RTOnceEx(&g_MpSolarisOnce, rtMpSolarisOnce, rtMpSolarisCleanUp, NULL /* pvUser */);
433 if (RT_SUCCESS(rc))
434 {
435 rc = RTCritSectEnter(&g_MpSolarisCritSect);
436 AssertRC(rc);
437
438 /*
439 * For each core in the system, count how many are currently online.
440 */
441 for (RTCPUID j = 0; j < g_cCores; j++)
442 {
443 uint64_t u64CoreId = g_pu64CoreIds[j];
444 for (RTCPUID idCpu = 0; idCpu < g_capCpuInfo; idCpu++)
445 {
446 rc = kstat_read(g_pKsCtl, g_papCpuInfo[idCpu], 0);
447 AssertReturn(rc != -1, 0 /* rc */);
448 uint64_t u64ThreadCoreId = rtMpSolarisGetCoreId(idCpu);
449 if (u64ThreadCoreId == u64CoreId)
450 {
451 kstat_named_t *pStat = (kstat_named_t *)kstat_data_lookup(g_papCpuInfo[idCpu], (char *)"state");
452 Assert(pStat->data_type == KSTAT_DATA_CHAR);
453 if( !RTStrNCmp(pStat->value.c, PS_ONLINE, sizeof(PS_ONLINE) - 1)
454 || !RTStrNCmp(pStat->value.c, PS_NOINTR, sizeof(PS_NOINTR) - 1))
455 {
456 uOnlineCores++;
457 break; /* Move to the next core. We have at least 1 hyperthread online in the current core. */
458 }
459 }
460 }
461 }
462
463 RTCritSectLeave(&g_MpSolarisCritSect);
464 }
465
466 return uOnlineCores;
467}
468
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette