VirtualBox

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

Last change on this file since 7554 was 7352, checked in by vboxsync, 17 years ago

smp_call_function_single only exists starting 2.6.19. Added missing idCpu validation in RTMpOnSpecific.

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