VirtualBox

source: vbox/trunk/src/VBox/Runtime/r0drv/linux/threadctxhooks-r0drv-linux.c@ 47199

Last change on this file since 47199 was 47199, checked in by vboxsync, 11 years ago

IPRT: Kernel thread-context hooks, linux implementation. Extended tstR0ThreadPreemption testcase to test the thread-context hooks.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 8.2 KB
Line 
1/* $Id: threadctxhooks-r0drv-linux.c 47199 2013-07-16 15:45:42Z vboxsync $ */
2/** @file
3 * IPRT - Thread-Context Hook, Ring-0 Driver, Linux.
4 */
5
6/*
7 * Copyright (C) 2013 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#include "the-linux-kernel.h"
32#include "internal/iprt.h"
33
34#include <iprt/mem.h>
35#include <iprt/assert.h>
36#include <iprt/thread.h>
37#include <iprt/err.h>
38#include <iprt/asm.h>
39#include "internal/thread.h"
40
41#if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 23) && defined(CONFIG_PREEMPT_NOTIFIERS)
42
43/*******************************************************************************
44* Structures and Typedefs *
45*******************************************************************************/
46/**
47 * The internal thread-context event.
48 */
49typedef struct RTTHREADCTXINT
50{
51 /** Magic value (RTTHREADCTXINT_MAGIC). */
52 uint32_t volatile u32Magic;
53 /** The thread handle (owner) for which the context-hooks are registered. */
54 RTTHREAD hOwner;
55 /** The preemption notifier object. */
56 struct preempt_notifier hPreemptNotifier;
57 /** Whether this handle has any hooks registered or not. */
58 bool fRegistered;
59 /** Pointer to the registered thread-context hook. */
60 PFNRTTHREADCTXHOOK pfnThreadCtxHook;
61 /** User argument passed to the thread-context hook. */
62 void *pvUser;
63 /** The thread-context operations. */
64 struct preempt_ops hPreemptOps;
65} RTTHREADCTXINT, *PRTTHREADCTXINT;
66
67
68/**
69 * Hook function for the thread-preempting event.
70 *
71 * @param pPreemptNotifier Pointer to the preempt_notifier struct.
72 * @param pNext Pointer to the task that is preempting the
73 * current thread.
74 *
75 * @remarks Called with the rq (runqueue) lock held and with preemption
76 * disabled!
77 */
78static void rtThreadCtxHooksLnxSchedOut(struct preempt_notifier *pPreemptNotifier, struct task_struct *pNext)
79{
80 PRTTHREADCTXINT pThis = RT_FROM_MEMBER(pPreemptNotifier, RTTHREADCTXINT, hPreemptNotifier);
81 AssertPtr(pThis);
82 AssertPtr(pThis->pfnThreadCtxHook);
83 Assert(!RTThreadPreemptIsEnabled(NIL_RTTHREAD));
84
85 pThis->pfnThreadCtxHook(RTTHREADCTXEVENT_PREEMPTING, pThis->pvUser);
86}
87
88
89/**
90 * Hook function for the thread-resumed event.
91 *
92 * @param pPreemptNotifier Pointer to the preempt_notifier struct.
93 * @param iCpu The CPU this thread is scheduled on.
94 *
95 * @remarks Called without holding the rq (runqueue) lock and with preemption
96 * enabled!
97 */
98static void rtThreadCtxHooksLnxSchedIn(struct preempt_notifier *pPreemptNotifier, int iCpu)
99{
100 PRTTHREADCTXINT pThis = RT_FROM_MEMBER(pPreemptNotifier, RTTHREADCTXINT, hPreemptNotifier);
101 AssertPtr(pThis);
102 AssertPtr(pThis->pfnThreadCtxHook);
103
104 pThis->pfnThreadCtxHook(RTTHREADCTXEVENT_RESUMED, pThis->pvUser);
105}
106
107
108/**
109 * Worker function for RTThreadCtxHooks(Deregister|Destroy)().
110 *
111 * @param pThis Pointer to the internal thread-context struct.
112 */
113DECLINLINE(void) rtThreadCtxHooksDeregister(PRTTHREADCTXINT pThis)
114{
115 preempt_notifier_unregister(&pThis->hPreemptNotifier);
116 pThis->hPreemptOps.sched_out = NULL;
117 pThis->hPreemptOps.sched_in = NULL;
118 pThis->fRegistered = false;
119}
120
121
122RTDECL(int) RTThreadCtxHooksCreate(PRTTHREADCTX phThreadCtx)
123{
124 PRTTHREADCTXINT pThis;
125 Assert(RTThreadPreemptIsEnabled(NIL_RTTHREAD));
126
127 pThis = (PRTTHREADCTXINT)RTMemAllocZ(sizeof(*pThis));
128 if (RT_UNLIKELY(!pThis))
129 return VERR_NO_MEMORY;
130 pThis->u32Magic = RTTHREADCTXINT_MAGIC;
131 pThis->hOwner = RTThreadSelf();
132 pThis->fRegistered = false;
133 preempt_notifier_init(&pThis->hPreemptNotifier, &pThis->hPreemptOps);
134
135 *phThreadCtx = pThis;
136 return VINF_SUCCESS;
137}
138RT_EXPORT_SYMBOL(RTThreadCtxHooksCreate);
139
140
141RTDECL(void) RTThreadCtxHooksDestroy(RTTHREADCTX hThreadCtx)
142{
143 /*
144 * Validate input.
145 */
146 PRTTHREADCTXINT pThis = hThreadCtx;
147 if (pThis == NIL_RTTHREADCTX)
148 return;
149 AssertPtr(pThis);
150 AssertMsgReturnVoid(pThis->u32Magic == RTTHREADCTXINT_MAGIC, ("pThis->u32Magic=%RX32 pThis=%p\n", pThis->u32Magic, pThis));
151 Assert(pThis->hOwner == RTThreadSelf());
152
153 /*
154 * If there's still a registered thread-context hook, deregister it now before destroying the object.
155 */
156 if (pThis->fRegistered)
157 rtThreadCtxHooksDeregister(pThis);
158
159 /*
160 * Paranoia... but since these are ring-0 threads we can't be too careful.
161 */
162 Assert(!pThis->fRegistered);
163 Assert(!pThis->hPreemptOps.sched_out);
164 Assert(!pThis->hPreemptOps.sched_in);
165
166 /*
167 * Destroy the object.
168 */
169 ASMAtomicWriteU32(&pThis->u32Magic, ~RTTHREADCTXINT_MAGIC);
170 RTMemFree(pThis);
171}
172RT_EXPORT_SYMBOL(RTThreadCtxHooksDestroy);
173
174
175RTDECL(int) RTThreadCtxHooksRegister(RTTHREADCTX hThreadCtx, PFNRTTHREADCTXHOOK pfnThreadCtxHook, void *pvUser)
176{
177 /*
178 * Validate input.
179 */
180 PRTTHREADCTXINT pThis = hThreadCtx;
181 if (pThis == NIL_RTTHREADCTX)
182 return VINF_SUCCESS;
183 AssertPtr(pThis);
184 AssertMsgReturn(pThis->u32Magic == RTTHREADCTXINT_MAGIC, ("pThis->u32Magic=%RX32 pThis=%p\n", pThis->u32Magic, pThis),
185 VERR_INVALID_HANDLE);
186 Assert(pThis->hOwner == RTThreadSelf());
187 Assert(!pThis->hPreemptOps.sched_out);
188 Assert(!pThis->hPreemptOps.sched_in);
189
190 /*
191 * Register the callback.
192 */
193 pThis->hPreemptOps.sched_out = rtThreadCtxHooksLnxSchedOut;
194 pThis->hPreemptOps.sched_in = rtThreadCtxHooksLnxSchedIn;
195 pThis->pvUser = pvUser;
196 pThis->pfnThreadCtxHook = pfnThreadCtxHook;
197 pThis->fRegistered = true;
198 preempt_notifier_register(&pThis->hPreemptNotifier);
199
200 return VINF_SUCCESS;
201}
202RT_EXPORT_SYMBOL(RTThreadCtxHooksRegister);
203
204
205RTDECL(int) RTThreadCtxHooksDeregister(RTTHREADCTX hThreadCtx)
206{
207 /*
208 * Validate input.
209 */
210 PRTTHREADCTXINT pThis = hThreadCtx;
211 if (pThis == NIL_RTTHREADCTX)
212 return VINF_SUCCESS;
213 AssertPtr(pThis);
214 AssertMsgReturn(pThis->u32Magic == RTTHREADCTXINT_MAGIC, ("pThis->u32Magic=%RX32 pThis=%p\n", pThis->u32Magic, pThis),
215 VERR_INVALID_HANDLE);
216 Assert(pThis->hOwner == RTThreadSelf());
217 Assert(pThis->fRegistered);
218
219 /*
220 * Deregister the callback.
221 */
222 rtThreadCtxHooksDeregister(pThis);
223 return VINF_SUCCESS;
224}
225RT_EXPORT_SYMBOL(RTThreadCtxHooksDeregister);
226
227#else /* Not supported / Not needed */
228
229RTDECL(int) RTThreadCtxHooksCreate(PRTTHREADCTX phThreadCtx)
230{
231 NOREF(phThreadCtx);
232 return VERR_NOT_SUPPORTED;
233}
234RT_EXPORT_SYMBOL(RTThreadCtxHooksCreate);
235
236
237RTDECL(void) RTThreadCtxHooksDestroy(RTTHREADCTX hThreadCtx)
238{
239 NOREF(hThreadCtx);
240}
241RT_EXPORT_SYMBOL(RTThreadCtxHooksDestroy);
242
243
244RTDECL(int) RTThreadCtxHooksRegister(RTTHREADCTX hThreadCtx, PFNRTTHREADCTXHOOK pfnThreadCtxHook, void *pvUser)
245{
246 NOREF(hThreadCtx);
247 NOREF(pfnThreadCtxHook);
248 NOREF(pvUser);
249 return VERR_NOT_SUPPORTED;
250}
251RT_EXPORT_SYMBOL(RTThreadCtxHooksRegister);
252
253
254RTDECL(int) RTThreadCtxHooksDeregister(RTTHREADCTX hThreadCtx)
255{
256 NOREF(hThreadCtx);
257 return VERR_NOT_SUPPORTED;
258}
259RT_EXPORT_SYMBOL(RTThreadCtxHooksDeregister);
260
261#endif /* Not supported / Not needed */
262
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