VirtualBox

source: vbox/trunk/src/VBox/Runtime/r0drv/solaris/mpnotification-r0drv-solaris.c

Last change on this file was 98103, checked in by vboxsync, 17 months ago

Copyright year updates by scm.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 4.9 KB
Line 
1/* $Id: mpnotification-r0drv-solaris.c 98103 2023-01-17 14:15:46Z vboxsync $ */
2/** @file
3 * IPRT - Multiprocessor Event Notifications, Ring-0 Driver, Solaris.
4 */
5
6/*
7 * Copyright (C) 2008-2023 Oracle and/or its affiliates.
8 *
9 * This file is part of VirtualBox base platform packages, as
10 * available from https://www.virtualbox.org.
11 *
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU General Public License
14 * as published by the Free Software Foundation, in version 3 of the
15 * License.
16 *
17 * This program is distributed in the hope that it will be useful, but
18 * WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20 * General Public License for more details.
21 *
22 * You should have received a copy of the GNU General Public License
23 * along with this program; if not, see <https://www.gnu.org/licenses>.
24 *
25 * The contents of this file may alternatively be used under the terms
26 * of the Common Development and Distribution License Version 1.0
27 * (CDDL), a copy of it is provided in the "COPYING.CDDL" file included
28 * in the VirtualBox distribution, in which case the provisions of the
29 * CDDL are applicable instead of those of the GPL.
30 *
31 * You may elect to license modified versions of this file under the
32 * terms and conditions of either the GPL or the CDDL or both.
33 *
34 * SPDX-License-Identifier: GPL-3.0-only OR CDDL-1.0
35 */
36
37
38/*********************************************************************************************************************************
39* Header Files *
40*********************************************************************************************************************************/
41#include "the-solaris-kernel.h"
42#include "internal/iprt.h"
43
44#include <iprt/errcore.h>
45#include <iprt/mp.h>
46#include <iprt/cpuset.h>
47#include <iprt/string.h>
48#include <iprt/thread.h>
49#include "r0drv/mp-r0drv.h"
50
51
52/*********************************************************************************************************************************
53* Global Variables *
54*********************************************************************************************************************************/
55/** Whether CPUs are being watched or not. */
56static volatile bool g_fSolCpuWatch = false;
57/** Set of online cpus that is maintained by the MP callback.
58 * This avoids locking issues querying the set from the kernel as well as
59 * eliminating any uncertainty regarding the online status during the
60 * callback. */
61RTCPUSET g_rtMpSolCpuSet;
62
63/**
64 * Internal solaris representation for watching CPUs.
65 */
66typedef struct RTMPSOLWATCHCPUS
67{
68 /** Function pointer to Mp worker. */
69 PFNRTMPWORKER pfnWorker;
70 /** Argument to pass to the Mp worker. */
71 void *pvArg;
72} RTMPSOLWATCHCPUS;
73typedef RTMPSOLWATCHCPUS *PRTMPSOLWATCHCPUS;
74
75
76/**
77 * Solaris callback function for Mp event notification.
78 *
79 * @returns Solaris error code.
80 * @param CpuState The current event/state of the CPU.
81 * @param iCpu Which CPU is this event for.
82 * @param pvArg Ignored.
83 *
84 * @remarks This function assumes index == RTCPUID.
85 * We may -not- be firing on the CPU going online/offline and called
86 * with preemption enabled.
87 */
88static int rtMpNotificationCpuEvent(cpu_setup_t CpuState, int iCpu, void *pvArg)
89{
90 RTMPEVENT enmMpEvent;
91
92 /*
93 * Update our CPU set structures first regardless of whether we've been
94 * scheduled on the right CPU or not, this is just atomic accounting.
95 */
96 if (CpuState == CPU_ON)
97 {
98 enmMpEvent = RTMPEVENT_ONLINE;
99 RTCpuSetAdd(&g_rtMpSolCpuSet, iCpu);
100 }
101 else if (CpuState == CPU_OFF)
102 {
103 enmMpEvent = RTMPEVENT_OFFLINE;
104 RTCpuSetDel(&g_rtMpSolCpuSet, iCpu);
105 }
106 else
107 return 0;
108
109 rtMpNotificationDoCallbacks(enmMpEvent, iCpu);
110 NOREF(pvArg);
111 return 0;
112}
113
114
115DECLHIDDEN(int) rtR0MpNotificationNativeInit(void)
116{
117 if (ASMAtomicReadBool(&g_fSolCpuWatch) == true)
118 return VERR_WRONG_ORDER;
119
120 /*
121 * Register the callback building the online cpu set as we do so.
122 */
123 RTCpuSetEmpty(&g_rtMpSolCpuSet);
124
125 mutex_enter(&cpu_lock);
126 register_cpu_setup_func(rtMpNotificationCpuEvent, NULL /* pvArg */);
127
128 for (int i = 0; i < (int)RTMpGetCount(); ++i)
129 if (cpu_is_online(cpu[i]))
130 rtMpNotificationCpuEvent(CPU_ON, i, NULL /* pvArg */);
131
132 ASMAtomicWriteBool(&g_fSolCpuWatch, true);
133 mutex_exit(&cpu_lock);
134
135 return VINF_SUCCESS;
136}
137
138
139DECLHIDDEN(void) rtR0MpNotificationNativeTerm(void)
140{
141 if (ASMAtomicReadBool(&g_fSolCpuWatch) == true)
142 {
143 mutex_enter(&cpu_lock);
144 unregister_cpu_setup_func(rtMpNotificationCpuEvent, NULL /* pvArg */);
145 ASMAtomicWriteBool(&g_fSolCpuWatch, false);
146 mutex_exit(&cpu_lock);
147 }
148}
149
Note: See TracBrowser for help on using the repository browser.

© 2023 Oracle
ContactPrivacy policyTerms of Use