VirtualBox

source: vbox/trunk/src/VBox/Runtime/r0drv/solaris/vbi/mp-r0drv-solaris.c@ 21094

Last change on this file since 21094 was 19391, checked in by vboxsync, 16 years ago

IPRT: Explained RTMpOthers preemption req to the caller in mp.h, but play safe. Comment in RTMpOnSpecific isn't relevant.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 6.1 KB
Line 
1/* $Id: mp-r0drv-solaris.c 19391 2009-05-05 17:26:18Z vboxsync $ */
2/** @file
3 * IPRT - Multiprocessor, Ring-0 Driver, Solaris.
4 */
5
6/*
7 * Copyright (C) 2008 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
27 * Clara, CA 95054 USA or visit http://www.sun.com if you need
28 * additional information or have any questions.
29 */
30
31
32/*******************************************************************************
33* Header Files *
34*******************************************************************************/
35#include "the-solaris-kernel.h"
36
37#include <iprt/mp.h>
38#include <iprt/cpuset.h>
39#include <iprt/err.h>
40#include <iprt/asm.h>
41#include "r0drv/mp-r0drv.h"
42
43
44RTDECL(bool) RTMpIsCpuWorkPending(void)
45{
46 return false;
47}
48
49
50RTDECL(RTCPUID) RTMpCpuId(void)
51{
52 return vbi_cpu_id();
53}
54
55
56RTDECL(int) RTMpCpuIdToSetIndex(RTCPUID idCpu)
57{
58 return idCpu < vbi_cpu_maxcount() ? idCpu : -1;
59}
60
61
62RTDECL(RTCPUID) RTMpCpuIdFromSetIndex(int iCpu)
63{
64 return (unsigned)iCpu < vbi_cpu_maxcount() ? iCpu : NIL_RTCPUID;
65}
66
67
68RTDECL(RTCPUID) RTMpGetMaxCpuId(void)
69{
70 return vbi_max_cpu_id();
71}
72
73
74RTDECL(bool) RTMpIsCpuOnline(RTCPUID idCpu)
75{
76 return idCpu < vbi_cpu_count() && vbi_cpu_online(idCpu);
77}
78
79
80RTDECL(bool) RTMpIsCpuPossible(RTCPUID idCpu)
81{
82 return idCpu < vbi_cpu_count();
83}
84
85
86RTDECL(PRTCPUSET) RTMpGetSet(PRTCPUSET pSet)
87{
88 RTCPUID idCpu;
89
90 RTCpuSetEmpty(pSet);
91 idCpu = RTMpGetMaxCpuId(); /* it's inclusive */
92 do
93 {
94 if (RTMpIsCpuPossible(idCpu))
95 RTCpuSetAdd(pSet, idCpu);
96 } while (idCpu-- > 0);
97
98 return pSet;
99}
100
101
102RTDECL(RTCPUID) RTMpGetCount(void)
103{
104 return vbi_cpu_count();
105}
106
107
108RTDECL(PRTCPUSET) RTMpGetOnlineSet(PRTCPUSET pSet)
109{
110 RTCPUID idCpu;
111
112 RTCpuSetEmpty(pSet);
113 idCpu = RTMpGetMaxCpuId(); /* it's inclusive */
114 do
115 {
116 if (RTMpIsCpuOnline(idCpu))
117 RTCpuSetAdd(pSet, idCpu);
118 } while (idCpu-- > 0);
119
120 return pSet;
121}
122
123
124RTDECL(RTCPUID) RTMpGetOnlineCount(void)
125{
126 int c;
127 int cnt = 0;
128
129 for (c = 0; c < vbi_cpu_count(); ++c)
130 {
131 if (vbi_cpu_online(c))
132 ++cnt;
133 }
134 return cnt;
135}
136
137
138
139/**
140 * Wrapper between the native solaris per-cpu callback and PFNRTWORKER
141 * for the RTMpOnAll API.
142 *
143 * @param uArgs Pointer to the RTMPARGS package.
144 * @param uIgnored1 Ignored.
145 * @param uIgnored2 Ignored.
146 */
147static int rtmpOnAllSolarisWrapper(void *uArg, void *uIgnored1, void *uIgnored2)
148{
149 PRTMPARGS pArgs = (PRTMPARGS)(uArg);
150
151 pArgs->pfnWorker(RTMpCpuId(), pArgs->pvUser1, pArgs->pvUser2);
152
153 NOREF(uIgnored1);
154 NOREF(uIgnored2);
155 return 0;
156}
157
158
159RTDECL(int) RTMpOnAll(PFNRTMPWORKER pfnWorker, void *pvUser1, void *pvUser2)
160{
161 RTMPARGS Args;
162
163 Args.pfnWorker = pfnWorker;
164 Args.pvUser1 = pvUser1;
165 Args.pvUser2 = pvUser2;
166 Args.idCpu = NIL_RTCPUID;
167 Args.cHits = 0;
168
169 vbi_preempt_disable();
170
171 vbi_execute_on_all(rtmpOnAllSolarisWrapper, &Args);
172
173 vbi_preempt_enable();
174
175 return VINF_SUCCESS;
176}
177
178
179/**
180 * Wrapper between the native solaris per-cpu callback and PFNRTWORKER
181 * for the RTMpOnOthers API.
182 *
183 * @param uArgs Pointer to the RTMPARGS package.
184 * @param uIgnored1 Ignored.
185 * @param uIgnored2 Ignored.
186 */
187static int rtmpOnOthersSolarisWrapper(void *uArg, void *uIgnored1, void *uIgnored2)
188{
189 PRTMPARGS pArgs = (PRTMPARGS)(uArg);
190 RTCPUID idCpu = RTMpCpuId();
191
192 Assert(idCpu != pArgs->idCpu);
193 pArgs->pfnWorker(idCpu, pArgs->pvUser1, pArgs->pvUser2);
194
195 NOREF(uIgnored1);
196 NOREF(uIgnored2);
197 return 0;
198}
199
200
201RTDECL(int) RTMpOnOthers(PFNRTMPWORKER pfnWorker, void *pvUser1, void *pvUser2)
202{
203 int rc;
204 RTMPARGS Args;
205
206 /* The caller is supposed to have disabled preemption, but take no chances. */
207 vbi_preempt_disable();
208
209 Args.pfnWorker = pfnWorker;
210 Args.pvUser1 = pvUser1;
211 Args.pvUser2 = pvUser2;
212 Args.idCpu = RTMpCpuId();
213 Args.cHits = 0;
214
215 vbi_execute_on_others(rtmpOnOthersSolarisWrapper, &Args);
216
217 vbi_preempt_enable();
218
219 return VINF_SUCCESS;
220}
221
222
223/**
224 * Wrapper between the native solaris per-cpu callback and PFNRTWORKER
225 * for the RTMpOnSpecific API.
226 *
227 *
228 * @param uArgs Pointer to the RTMPARGS package.
229 * @param uIgnored1 Ignored.
230 * @param uIgnored2 Ignored.
231 */
232static int rtmpOnSpecificSolarisWrapper(void *uArg, void *uIgnored1, void *uIgnored2)
233{
234 PRTMPARGS pArgs = (PRTMPARGS)(uArg);
235 RTCPUID idCpu = RTMpCpuId();
236
237 Assert(idCpu == pArgs->idCpu);
238 pArgs->pfnWorker(idCpu, pArgs->pvUser1, pArgs->pvUser2);
239 ASMAtomicIncU32(&pArgs->cHits);
240
241 NOREF(uIgnored1);
242 NOREF(uIgnored2);
243 return 0;
244}
245
246
247RTDECL(int) RTMpOnSpecific(RTCPUID idCpu, PFNRTMPWORKER pfnWorker, void *pvUser1, void *pvUser2)
248{
249 int rc;
250 RTMPARGS Args;
251
252 if (idCpu >= vbi_cpu_count())
253 return VERR_CPU_NOT_FOUND;
254
255 Args.pfnWorker = pfnWorker;
256 Args.pvUser1 = pvUser1;
257 Args.pvUser2 = pvUser2;
258 Args.idCpu = idCpu;
259 Args.cHits = 0;
260
261 vbi_preempt_disable();
262
263 vbi_execute_on_one(rtmpOnSpecificSolarisWrapper, &Args, idCpu);
264
265 vbi_preempt_enable();
266
267 Assert(ASMAtomicUoReadU32(&Args.cHits) <= 1);
268
269 return ASMAtomicUoReadU32(&Args.cHits) == 1
270 ? VINF_SUCCESS
271 : VERR_CPU_NOT_FOUND;
272}
273
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