VirtualBox

source: vbox/trunk/src/VBox/Runtime/testcase/tstR0ThreadPreemption.cpp@ 22277

Last change on this file since 22277 was 21551, checked in by vboxsync, 15 years ago

tstR0ThreadPreemption: Some adjustments to new RTThreadPreemptIsEnabled behavior.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 7.7 KB
Line 
1/* $Id: tstR0ThreadPreemption.cpp 21551 2009-07-13 16:40:43Z vboxsync $ */
2/** @file
3 * IPRT R0 Testcase - Thread Preemption.
4 */
5
6/*
7 * Copyright (C) 2009 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* Header Files *
33*******************************************************************************/
34#include <iprt/thread.h>
35
36#include <iprt/err.h>
37#include <iprt/time.h>
38#include <iprt/string.h>
39#include <VBox/sup.h>
40#include "tstR0ThreadPreemption.h"
41
42
43
44/**
45 * Service request callback function.
46 *
47 * @returns VBox status code.
48 * @param pSession The caller's session.
49 * @param u64Arg 64-bit integer argument.
50 * @param pReqHdr The request header. Input / Output. Optional.
51 */
52DECLEXPORT(int) TSTR0ThreadPreemptionSrvReqHandler(PSUPDRVSESSION pSession, uint32_t uOperation,
53 uint64_t u64Arg, PSUPR0SERVICEREQHDR pReqHdr)
54{
55 if (u64Arg)
56 return VERR_INVALID_PARAMETER;
57 if (!VALID_PTR(pReqHdr))
58 return VERR_INVALID_PARAMETER;
59 char *pszErr = (char *)(pReqHdr + 1);
60 size_t cchErr = pReqHdr->cbReq - sizeof(*pReqHdr);
61 if (cchErr < 32 || cchErr >= 0x10000)
62 return VERR_INVALID_PARAMETER;
63 *pszErr = '\0';
64
65 /*
66 * The big switch.
67 */
68 switch (uOperation)
69 {
70 case TSTR0THREADPREMEPTION_SANITY_OK:
71 break;
72
73 case TSTR0THREADPREMEPTION_SANITY_FAILURE:
74 RTStrPrintf(pszErr, cchErr, "!42failure42%1024s", "");
75 break;
76
77 case TSTR0THREADPREMEPTION_BASIC:
78 {
79 if (!ASMIntAreEnabled())
80 RTStrPrintf(pszErr, cchErr, "!Interrupts disabled");
81 else if (!RTThreadPreemptIsEnabled(NIL_RTTHREAD))
82 RTStrPrintf(pszErr, cchErr, "!RTThreadPreemptIsEnabled returns false by default");
83 else
84 {
85 RTTHREADPREEMPTSTATE State = RTTHREADPREEMPTSTATE_INITIALIZER;
86 RTThreadPreemptDisable(&State);
87 if (RTThreadPreemptIsEnabled(NIL_RTTHREAD))
88 RTStrPrintf(pszErr, cchErr, "!RTThreadPreemptIsEnabled returns true after RTThreadPreemptDisable");
89 else if (!ASMIntAreEnabled())
90 RTStrPrintf(pszErr, cchErr, "!Interrupts disabled");
91 RTThreadPreemptRestore(&State);
92 }
93 break;
94 }
95
96 case TSTR0THREADPREMEPTION_IS_PENDING:
97 {
98 RTTHREADPREEMPTSTATE State = RTTHREADPREEMPTSTATE_INITIALIZER;
99 RTThreadPreemptDisable(&State);
100 if (!RTThreadPreemptIsEnabled(NIL_RTTHREAD))
101 {
102 if (ASMIntAreEnabled())
103 {
104 uint64_t u64StartTS = RTTimeNanoTS();
105 uint64_t u64StartSysTS = RTTimeSystemNanoTS();
106 uint64_t cLoops = 0;
107 uint64_t cNanosSysElapsed;
108 uint64_t cNanosElapsed;
109 bool fPending;
110 do
111 {
112 fPending = RTThreadPreemptIsPending(NIL_RTTHREAD);
113 cNanosElapsed = RTTimeNanoTS() - u64StartTS;
114 cNanosSysElapsed = RTTimeSystemNanoTS() - u64StartSysTS;
115 cLoops++;
116 } while ( !fPending
117 && cNanosElapsed < UINT64_C(2)*1000U*1000U*1000U
118 && cNanosSysElapsed < UINT64_C(2)*1000U*1000U*1000U
119 && cLoops < 100U*_1M);
120 if (!fPending)
121 RTStrPrintf(pszErr, cchErr, "!Preempt not pending after %'llu loops / %'llu ns / %'llu ns (sys)",
122 cLoops, cNanosElapsed, cNanosSysElapsed);
123 else if (cLoops == 1)
124 RTStrPrintf(pszErr, cchErr, "!cLoops=1\n");
125 else
126 RTStrPrintf(pszErr, cchErr, "RTThreadPreemptIsPending returned true after %'llu loops / %'llu ns / %'llu ns (sys)",
127 cLoops, cNanosElapsed, cNanosSysElapsed);
128 }
129 else
130 RTStrPrintf(pszErr, cchErr, "!Interrupts disabled");
131 }
132 else
133 RTStrPrintf(pszErr, cchErr, "!RTThreadPreemptIsEnabled returns true after RTThreadPreemptDisable");
134 RTThreadPreemptRestore(&State);
135 break;
136 }
137
138 case TSTR0THREADPREMEPTION_NESTED:
139 {
140 bool const fDefault = RTThreadPreemptIsEnabled(NIL_RTTHREAD);
141 RTTHREADPREEMPTSTATE State1 = RTTHREADPREEMPTSTATE_INITIALIZER;
142 RTThreadPreemptDisable(&State1);
143 if (!RTThreadPreemptIsEnabled(NIL_RTTHREAD))
144 {
145 RTTHREADPREEMPTSTATE State2 = RTTHREADPREEMPTSTATE_INITIALIZER;
146 RTThreadPreemptDisable(&State2);
147 if (!RTThreadPreemptIsEnabled(NIL_RTTHREAD))
148 {
149 RTTHREADPREEMPTSTATE State3 = RTTHREADPREEMPTSTATE_INITIALIZER;
150 RTThreadPreemptDisable(&State3);
151 if (RTThreadPreemptIsEnabled(NIL_RTTHREAD))
152 RTStrPrintf(pszErr, cchErr, "!RTThreadPreemptIsEnabled returns true after 3rd RTThreadPreemptDisable");
153
154 RTThreadPreemptRestore(&State3);
155 if (RTThreadPreemptIsEnabled(NIL_RTTHREAD) && !*pszErr)
156 RTStrPrintf(pszErr, cchErr, "!RTThreadPreemptIsEnabled returns true after 1st RTThreadPreemptRestore");
157 }
158 else
159 RTStrPrintf(pszErr, cchErr, "!RTThreadPreemptIsEnabled returns true after 2nd RTThreadPreemptDisable");
160
161 RTThreadPreemptRestore(&State2);
162 if (RTThreadPreemptIsEnabled(NIL_RTTHREAD) && !*pszErr)
163 RTStrPrintf(pszErr, cchErr, "!RTThreadPreemptIsEnabled returns true after 2nd RTThreadPreemptRestore");
164 }
165 else
166 RTStrPrintf(pszErr, cchErr, "!RTThreadPreemptIsEnabled returns true after 1st RTThreadPreemptDisable");
167 RTThreadPreemptRestore(&State1);
168 if (RTThreadPreemptIsEnabled(NIL_RTTHREAD) != fDefault && !*pszErr)
169 RTStrPrintf(pszErr, cchErr, "!RTThreadPreemptIsEnabled returns false after 3rd RTThreadPreemptRestore");
170 break;
171 }
172
173 default:
174 RTStrPrintf(pszErr, cchErr, "!Unknown test #%d", uOperation);
175 break;
176 }
177
178 /* The error indicator is the '!' in the message buffer. */
179 return VINF_SUCCESS;
180}
181
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