VirtualBox

source: vbox/trunk/src/VBox/Runtime/r0drv/linux/mp-r0drv-linux.c@ 9563

Last change on this file since 9563 was 9429, checked in by vboxsync, 17 years ago

RTMpDoesCpuExist -> RTMpIsCpuPossible. Changed the RTMpGetCount and RTMpGetSet specification to include all possible cpus.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 6.9 KB
Line 
1/* $Id: mp-r0drv-linux.c 9429 2008-06-05 15:22:37Z vboxsync $ */
2/** @file
3 * IPRT - Multiprocessor, Ring-0 Driver, Linux.
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-linux-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(RTCPUID) RTMpCpuId(void)
45{
46 return smp_processor_id();
47}
48
49
50RTDECL(int) RTMpCpuIdToSetIndex(RTCPUID idCpu)
51{
52 return idCpu < NR_CPUS ? idCpu : -1;
53}
54
55
56RTDECL(RTCPUID) RTMpCpuIdFromSetIndex(int iCpu)
57{
58 return iCpu < NR_CPUS ? iCpu : NIL_RTCPUID;
59}
60
61
62RTDECL(RTCPUID) RTMpGetMaxCpuId(void)
63{
64 return NR_CPUS - 1; //???
65}
66
67
68RTDECL(bool) RTMpIsCpuPossible(RTCPUID idCpu)
69{
70#if defined(CONFIG_SMP)
71 if (RT_UNLIKELY(idCpu >= NR_CPUS))
72 return false;
73
74# if defined(cpu_possible)
75 return cpu_possible(idCpu);
76# else /* < 2.5.29 */
77 return idCpu < (RTCPUID)smp_num_cpus;
78# endif
79#else
80 return idCpu == RTMpCpuId();
81#endif
82}
83
84
85RTDECL(PRTCPUSET) RTMpGetSet(PRTCPUSET pSet)
86{
87 RTCPUID idCpu;
88
89 RTCpuSetEmpty(pSet);
90 idCpu = RTMpGetMaxCpuId();
91 do
92 {
93 if (RTMpIsCpuPossible(idCpu))
94 RTCpuSetAdd(pSet, idCpu);
95 } while (idCpu-- > 0);
96 return pSet;
97}
98
99
100RTDECL(RTCPUID) RTMpGetCount(void)
101{
102#ifdef CONFIG_SMP
103# if defined(CONFIG_HOTPLUG_CPU) /* introduced & uses cpu_present */
104 return num_present_cpus();
105# elif defined(num_possible_cpus)
106 return num_possible_cpus();
107# elif LINUX_VERSION_CODE < KERNEL_VERSION(2, 5, 0)
108 return smp_num_cpus;
109# else
110 RTCPUSET Set;
111 RTMpGetSet(&Set);
112 return RTCpuSetCount(&Set);
113# endif
114#else
115 return 1;
116#endif
117}
118
119
120RTDECL(bool) RTMpIsCpuOnline(RTCPUID idCpu)
121{
122#ifdef CONFIG_SMP
123 if (RT_UNLIKELY(idCpu >= NR_CPUS))
124 return false;
125# ifdef cpu_online
126 return cpu_online(idCpu);
127# else /* 2.4: */
128 return cpu_online_map & RT_BIT_64(idCpu);
129# endif
130#else
131 return idCpu == RTMpCpuId();
132#endif
133}
134
135
136RTDECL(PRTCPUSET) RTMpGetOnlineSet(PRTCPUSET pSet)
137{
138#ifdef CONFIG_SMP
139 RTCPUID idCpu;
140
141 RTCpuSetEmpty(pSet);
142 idCpu = RTMpGetMaxCpuId();
143 do
144 {
145 if (RTMpIsCpuOnline(idCpu))
146 RTCpuSetAdd(pSet, idCpu);
147 } while (idCpu-- > 0);
148#else
149 RTCpuSetEmpty(pSet);
150 RTCpuSetAdd(pSet, RTMpCpuId());
151#endif
152 return pSet;
153}
154
155
156RTDECL(RTCPUID) RTMpGetOnlineCount(void)
157{
158#ifdef CONFIG_SMP
159# if defined(num_online_cpus)
160 return num_online_cpus();
161# else
162 RTCPUSET Set;
163 RTMpGetOnlineSet(&Set);
164 return RTCpuSetCount(&Set);
165# endif
166#else
167 return 1;
168#endif
169}
170
171
172/**
173 * Wrapper between the native linux per-cpu callbacks and PFNRTWORKER
174 *
175 * @param pvInfo Pointer to the RTMPARGS package.
176 */
177static void rtmpLinuxWrapper(void *pvInfo)
178{
179 PRTMPARGS pArgs = (PRTMPARGS)pvInfo;
180 ASMAtomicIncU32(&pArgs->cHits);
181 pArgs->pfnWorker(RTMpCpuId(), pArgs->pvUser1, pArgs->pvUser2);
182}
183
184
185RTDECL(int) RTMpOnAll(PFNRTMPWORKER pfnWorker, void *pvUser1, void *pvUser2)
186{
187 int rc;
188 RTMPARGS Args;
189
190 Args.pfnWorker = pfnWorker;
191 Args.pvUser1 = pvUser1;
192 Args.pvUser2 = pvUser2;
193 Args.idCpu = NIL_RTCPUID;
194 Args.cHits = 0;
195
196#if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 0)
197 rc = on_each_cpu(rtmpLinuxWrapper, &Args, 0 /* retry */, 1 /* wait */);
198
199#else /* older kernels */
200
201# ifdef preempt_disable
202 preempt_disable();
203# endif
204 rc = smp_call_function(rtmpLinuxWrapper, &Args, 0 /* retry */, 1 /* wait */);
205 local_irq_disable();
206 rtmpLinuxWrapper(&Args);
207 local_irq_enable();
208# ifdef preempt_enable
209 preempt_enable();
210# endif
211#endif /* older kernels */
212 Assert(rc == 0); NOREF(rc);
213 return VINF_SUCCESS;
214}
215
216
217RTDECL(int) RTMpOnOthers(PFNRTMPWORKER pfnWorker, void *pvUser1, void *pvUser2)
218{
219 int rc;
220 RTMPARGS Args;
221
222 Args.pfnWorker = pfnWorker;
223 Args.pvUser1 = pvUser1;
224 Args.pvUser2 = pvUser2;
225 Args.idCpu = NIL_RTCPUID;
226 Args.cHits = 0;
227
228# ifdef preempt_disable
229 preempt_disable();
230# endif
231 rc = smp_call_function(rtmpLinuxWrapper, &Args, 0 /* retry */, 1 /* wait */);
232# ifdef preempt_enable
233 preempt_enable();
234# endif
235
236 Assert(rc == 0); NOREF(rc);
237 return VINF_SUCCESS;
238}
239
240
241#if LINUX_VERSION_CODE < KERNEL_VERSION(2, 6, 19)
242/**
243 * Wrapper between the native linux per-cpu callbacks and PFNRTWORKER
244 * employed by RTMpOnSpecific on older kernels that lacks smp_call_function_single.
245 *
246 * @param pvInfo Pointer to the RTMPARGS package.
247 */
248static void rtmpOnSpecificLinuxWrapper(void *pvInfo)
249{
250 PRTMPARGS pArgs = (PRTMPARGS)pvInfo;
251 RTCPUID idCpu = RTMpCpuId();
252
253 if (idCpu == pArgs->idCpu)
254 {
255 pArgs->pfnWorker(idCpu, pArgs->pvUser1, pArgs->pvUser2);
256 ASMAtomicIncU32(&pArgs->cHits);
257 }
258}
259#endif
260
261
262RTDECL(int) RTMpOnSpecific(RTCPUID idCpu, PFNRTMPWORKER pfnWorker, void *pvUser1, void *pvUser2)
263{
264 int rc;
265 RTMPARGS Args;
266
267 Args.pfnWorker = pfnWorker;
268 Args.pvUser1 = pvUser1;
269 Args.pvUser2 = pvUser2;
270 Args.idCpu = idCpu;
271 Args.cHits = 0;
272
273 if (!RTMpIsCpuPossible(idCpu))
274 return VERR_CPU_NOT_FOUND;
275
276# ifdef preempt_disable
277 preempt_disable();
278# endif
279 if (idCpu != RTMpCpuId())
280 {
281 if (RTMpIsCpuOnline(idCpu))
282 {
283#if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 19)
284 rc = smp_call_function_single(idCpu, rtmpLinuxWrapper, &Args, 0 /* retry */, 1 /* wait */);
285#else
286 rc = smp_call_function(rtmpOnSpecificLinuxWrapper, &Args, 0 /* retry */, 1 /* wait */);
287#endif
288 Assert(rc == 0);
289 rc = Args.cHits ? VINF_SUCCESS : VERR_CPU_OFFLINE;
290 }
291 else
292 rc = VERR_CPU_OFFLINE;
293 }
294 else
295 {
296 rtmpLinuxWrapper(&Args);
297 rc = VINF_SUCCESS;
298 }
299# ifdef preempt_enable
300 preempt_enable();
301# endif
302
303 NOREF(rc);
304 return rc;
305}
306
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