VirtualBox

Changeset 48199 in vbox


Ignore:
Timestamp:
Aug 30, 2013 3:05:23 PM (11 years ago)
Author:
vboxsync
Message:

PDMAllCritSectRw.cpp: do the same preemption/interrupt-enabled checks as the standard pdm critsect code or we might end up with deadlocks when preemption is enabled. (untested changes)

File:
1 edited

Legend:

Unmodified
Added
Removed
  • trunk/src/VBox/VMM/VMMAll/PDMAllCritSectRw.cpp

    r45310 r48199  
    110110    AssertPtrReturn(pThis, RTLOCKVAL_SUB_CLASS_INVALID);
    111111    AssertReturn(pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC, RTLOCKVAL_SUB_CLASS_INVALID);
    112 #if defined(PDMCRITSECTRW_STRICT) && defined(IN_RING3)
     112# if defined(PDMCRITSECTRW_STRICT) && defined(IN_RING3)
    113113    AssertReturn(!(pThis->s.Core.fFlags & RTCRITSECT_FLAGS_NOP), RTLOCKVAL_SUB_CLASS_INVALID);
    114114
     
    228228            }
    229229
    230 #if defined(IN_RING3)
    231             /*
    232              * Add ourselves to the queue and wait for the direction to change.
    233              */
    234             uint64_t c = (u64State & RTCSRW_CNT_RD_MASK) >> RTCSRW_CNT_RD_SHIFT;
    235             c++;
    236             Assert(c < RTCSRW_CNT_MASK / 2);
    237 
    238             uint64_t cWait = (u64State & RTCSRW_WAIT_CNT_RD_MASK) >> RTCSRW_WAIT_CNT_RD_SHIFT;
    239             cWait++;
    240             Assert(cWait <= c);
    241             Assert(cWait < RTCSRW_CNT_MASK / 2);
    242 
    243             u64State &= ~(RTCSRW_CNT_RD_MASK | RTCSRW_WAIT_CNT_RD_MASK);
    244             u64State |= (c << RTCSRW_CNT_RD_SHIFT) | (cWait << RTCSRW_WAIT_CNT_RD_SHIFT);
    245 
    246             if (ASMAtomicCmpXchgU64(&pThis->s.Core.u64State, u64State, u64OldState))
     230#if defined(IN_RING3) || defined(IN_RING0)
     231# ifdef IN_RING0
     232            if (   RTThreadPreemptIsEnabled(NIL_RTTHREAD)
     233                && ASMIntAreEnabled())
     234# endif
    247235            {
    248                 for (uint32_t iLoop = 0; ; iLoop++)
     236                /*
     237                 * Add ourselves to the queue and wait for the direction to change.
     238                 */
     239                uint64_t c = (u64State & RTCSRW_CNT_RD_MASK) >> RTCSRW_CNT_RD_SHIFT;
     240                c++;
     241                Assert(c < RTCSRW_CNT_MASK / 2);
     242
     243                uint64_t cWait = (u64State & RTCSRW_WAIT_CNT_RD_MASK) >> RTCSRW_WAIT_CNT_RD_SHIFT;
     244                cWait++;
     245                Assert(cWait <= c);
     246                Assert(cWait < RTCSRW_CNT_MASK / 2);
     247
     248                u64State &= ~(RTCSRW_CNT_RD_MASK | RTCSRW_WAIT_CNT_RD_MASK);
     249                u64State |= (c << RTCSRW_CNT_RD_SHIFT) | (cWait << RTCSRW_WAIT_CNT_RD_SHIFT);
     250
     251                if (ASMAtomicCmpXchgU64(&pThis->s.Core.u64State, u64State, u64OldState))
    249252                {
    250                     int rc;
     253                    for (uint32_t iLoop = 0; ; iLoop++)
     254                    {
     255                        int rc;
     256# ifdef IN_RING3
     257#  if defined(PDMCRITSECTRW_STRICT) && defined(IN_RING3)
     258                        rc = RTLockValidatorRecSharedCheckBlocking(pThis->s.Core.pValidatorRead, hThreadSelf, pSrcPos, true,
     259                                                                   RT_INDEFINITE_WAIT, RTTHREADSTATE_RW_READ, false);
     260                        if (RT_SUCCESS(rc))
     261#  else
     262                        RTTHREAD hThreadSelf = RTThreadSelf();
     263                        RTThreadBlocking(hThreadSelf, RTTHREADSTATE_RW_READ, false);
     264#  endif
     265# endif
     266                        {
     267                            do
     268                                rc = SUPSemEventMultiWaitNoResume(pThis->s.CTX_SUFF(pVM)->pSession,
     269                                                                  (SUPSEMEVENTMULTI)pThis->s.Core.hEvtRead,
     270                                                                  RT_INDEFINITE_WAIT);
     271                            while (rc == VERR_INTERRUPTED && pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC);
     272# ifdef IN_RING3
     273                            RTThreadUnblocked(hThreadSelf, RTTHREADSTATE_RW_READ);
     274# endif
     275                            if (pThis->s.Core.u32Magic != RTCRITSECTRW_MAGIC)
     276                                return VERR_SEM_DESTROYED;
     277                        }
     278                        if (RT_FAILURE(rc))
     279                        {
     280                            /* Decrement the counts and return the error. */
     281                            for (;;)
     282                            {
     283                                u64OldState = u64State = ASMAtomicReadU64(&pThis->s.Core.u64State);
     284                                c = (u64State & RTCSRW_CNT_RD_MASK) >> RTCSRW_CNT_RD_SHIFT; Assert(c > 0);
     285                                c--;
     286                                cWait = (u64State & RTCSRW_WAIT_CNT_RD_MASK) >> RTCSRW_WAIT_CNT_RD_SHIFT; Assert(cWait > 0);
     287                                cWait--;
     288                                u64State &= ~(RTCSRW_CNT_RD_MASK | RTCSRW_WAIT_CNT_RD_MASK);
     289                                u64State |= (c << RTCSRW_CNT_RD_SHIFT) | (cWait << RTCSRW_WAIT_CNT_RD_SHIFT);
     290                                if (ASMAtomicCmpXchgU64(&pThis->s.Core.u64State, u64State, u64OldState))
     291                                    break;
     292                            }
     293                            return rc;
     294                        }
     295
     296                        Assert(pThis->s.Core.fNeedReset);
     297                        u64State = ASMAtomicReadU64(&pThis->s.Core.u64State);
     298                        if ((u64State & RTCSRW_DIR_MASK) == (RTCSRW_DIR_READ << RTCSRW_DIR_SHIFT))
     299                            break;
     300                        AssertMsg(iLoop < 1, ("%u\n", iLoop));
     301                    }
     302
     303                    /* Decrement the wait count and maybe reset the semaphore (if we're last). */
     304                    for (;;)
     305                    {
     306                        u64OldState = u64State;
     307
     308                        cWait = (u64State & RTCSRW_WAIT_CNT_RD_MASK) >> RTCSRW_WAIT_CNT_RD_SHIFT;
     309                        Assert(cWait > 0);
     310                        cWait--;
     311                        u64State &= ~RTCSRW_WAIT_CNT_RD_MASK;
     312                        u64State |= cWait << RTCSRW_WAIT_CNT_RD_SHIFT;
     313
     314                        if (ASMAtomicCmpXchgU64(&pThis->s.Core.u64State, u64State, u64OldState))
     315                        {
     316                            if (cWait == 0)
     317                            {
     318                                if (ASMAtomicXchgBool(&pThis->s.Core.fNeedReset, false))
     319                                {
     320                                    int rc = SUPSemEventMultiReset(pThis->s.CTX_SUFF(pVM)->pSession,
     321                                                                   (SUPSEMEVENTMULTI)pThis->s.Core.hEvtRead);
     322                                    AssertRCReturn(rc, rc);
     323                                }
     324                            }
     325                            break;
     326                        }
     327                        u64State = ASMAtomicReadU64(&pThis->s.Core.u64State);
     328                    }
     329
    251330# if defined(PDMCRITSECTRW_STRICT) && defined(IN_RING3)
    252                     rc = RTLockValidatorRecSharedCheckBlocking(pThis->s.Core.pValidatorRead, hThreadSelf, pSrcPos, true,
    253                                                                RT_INDEFINITE_WAIT, RTTHREADSTATE_RW_READ, false);
    254                     if (RT_SUCCESS(rc))
    255 # else
    256                     RTTHREAD hThreadSelf = RTThreadSelf();
    257                     RTThreadBlocking(hThreadSelf, RTTHREADSTATE_RW_READ, false);
     331                    if (!fNoVal)
     332                        RTLockValidatorRecSharedAddOwner(pThis->s.Core.pValidatorRead, hThreadSelf, pSrcPos);
    258333# endif
    259                     {
    260                         do
    261                             rc = SUPSemEventMultiWaitNoResume(pThis->s.CTX_SUFF(pVM)->pSession,
    262                                                               (SUPSEMEVENTMULTI)pThis->s.Core.hEvtRead,
    263                                                               RT_INDEFINITE_WAIT);
    264                         while (rc == VERR_INTERRUPTED && pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC);
    265                         RTThreadUnblocked(hThreadSelf, RTTHREADSTATE_RW_READ);
    266                         if (pThis->s.Core.u32Magic != RTCRITSECTRW_MAGIC)
    267                             return VERR_SEM_DESTROYED;
    268                     }
    269                     if (RT_FAILURE(rc))
    270                     {
    271                         /* Decrement the counts and return the error. */
    272                         for (;;)
    273                         {
    274                             u64OldState = u64State = ASMAtomicReadU64(&pThis->s.Core.u64State);
    275                             c = (u64State & RTCSRW_CNT_RD_MASK) >> RTCSRW_CNT_RD_SHIFT; Assert(c > 0);
    276                             c--;
    277                             cWait = (u64State & RTCSRW_WAIT_CNT_RD_MASK) >> RTCSRW_WAIT_CNT_RD_SHIFT; Assert(cWait > 0);
    278                             cWait--;
    279                             u64State &= ~(RTCSRW_CNT_RD_MASK | RTCSRW_WAIT_CNT_RD_MASK);
    280                             u64State |= (c << RTCSRW_CNT_RD_SHIFT) | (cWait << RTCSRW_WAIT_CNT_RD_SHIFT);
    281                             if (ASMAtomicCmpXchgU64(&pThis->s.Core.u64State, u64State, u64OldState))
    282                                 break;
    283                         }
    284                         return rc;
    285                     }
    286 
    287                     Assert(pThis->s.Core.fNeedReset);
    288                     u64State = ASMAtomicReadU64(&pThis->s.Core.u64State);
    289                     if ((u64State & RTCSRW_DIR_MASK) == (RTCSRW_DIR_READ << RTCSRW_DIR_SHIFT))
    290                         break;
    291                     AssertMsg(iLoop < 1, ("%u\n", iLoop));
     334                    break;
    292335                }
    293 
    294                 /* Decrement the wait count and maybe reset the semaphore (if we're last). */
    295                 for (;;)
     336            }
     337#endif /* IN_RING3 || IN_RING3 */
     338#ifndef IN_RING3
     339# ifdef IN_RING0
     340            else
     341# endif
     342            {
     343                /*
     344                 * We cannot call SUPSemEventMultiWaitNoResume in this context. Go
     345                 * back to ring-3 and do it there or return rcBusy.
     346                 */
     347                STAM_REL_COUNTER_INC(&pThis->s.CTX_MID_Z(StatContention,EnterShared));
     348                if (rcBusy == VINF_SUCCESS)
    296349                {
    297                     u64OldState = u64State;
    298 
    299                     cWait = (u64State & RTCSRW_WAIT_CNT_RD_MASK) >> RTCSRW_WAIT_CNT_RD_SHIFT;
    300                     Assert(cWait > 0);
    301                     cWait--;
    302                     u64State &= ~RTCSRW_WAIT_CNT_RD_MASK;
    303                     u64State |= cWait << RTCSRW_WAIT_CNT_RD_SHIFT;
    304 
    305                     if (ASMAtomicCmpXchgU64(&pThis->s.Core.u64State, u64State, u64OldState))
    306                     {
    307                         if (cWait == 0)
    308                         {
    309                             if (ASMAtomicXchgBool(&pThis->s.Core.fNeedReset, false))
    310                             {
    311                                 int rc = SUPSemEventMultiReset(pThis->s.CTX_SUFF(pVM)->pSession,
    312                                                                (SUPSEMEVENTMULTI)pThis->s.Core.hEvtRead);
    313                                 AssertRCReturn(rc, rc);
    314                             }
    315                         }
    316                         break;
    317                     }
    318                     u64State = ASMAtomicReadU64(&pThis->s.Core.u64State);
     350                    PVM     pVM   = pThis->s.CTX_SUFF(pVM);     AssertPtr(pVM);
     351                    PVMCPU  pVCpu = VMMGetCpu(pVM);             AssertPtr(pVCpu);
     352                    /** @todo Should actually do this in via VMMR0.cpp instead of going all the way
     353                     *        back to ring-3. Goes for both kind of crit sects. */
     354                    return VMMRZCallRing3(pVM, pVCpu, VMMCALLRING3_PDM_CRIT_SECT_RW_ENTER_SHARED, MMHyperCCToR3(pVM, pThis));
    319355                }
    320 
    321 # if defined(PDMCRITSECTRW_STRICT) && defined(IN_RING3)
    322                 if (!fNoVal)
    323                     RTLockValidatorRecSharedAddOwner(pThis->s.Core.pValidatorRead, hThreadSelf, pSrcPos);
    324 # endif
    325                 break;
     356                return rcBusy;
    326357            }
    327 
    328 #else
    329             /*
    330              * We cannot call SUPSemEventMultiWaitNoResume in this context. Go
    331              * back to ring-3 and do it there or return rcBusy.
    332              */
    333             STAM_REL_COUNTER_INC(&pThis->s.CTX_MID_Z(StatContention,EnterShared));
    334             if (rcBusy == VINF_SUCCESS)
    335             {
    336                 PVM     pVM   = pThis->s.CTX_SUFF(pVM);     AssertPtr(pVM);
    337                 PVMCPU  pVCpu = VMMGetCpu(pVM);             AssertPtr(pVCpu);
    338                 /** @todo Should actually do this in via VMMR0.cpp instead of going all the way
    339                  *        back to ring-3. Goes for both kind of crit sects. */
    340                 return VMMRZCallRing3(pVM, pVCpu, VMMCALLRING3_PDM_CRIT_SECT_RW_ENTER_SHARED, MMHyperCCToR3(pVM, pThis));
    341             }
    342             return rcBusy;
    343 #endif
     358#endif /* !IN_RING3 */
    344359        }
    345360
     
    558573            else
    559574            {
    560 #if defined(IN_RING3)
    561                 /* Reverse the direction and signal the writer threads. */
    562                 u64State &= ~(RTCSRW_CNT_RD_MASK | RTCSRW_DIR_MASK);
    563                 u64State |= RTCSRW_DIR_WRITE << RTCSRW_DIR_SHIFT;
    564                 if (ASMAtomicCmpXchgU64(&pThis->s.Core.u64State, u64State, u64OldState))
     575#if defined(IN_RING3) || defined(IN_RING0)
     576# ifdef IN_RING0
     577                if (   RTThreadPreemptIsEnabled(NIL_RTTHREAD)
     578                    && ASMIntAreEnabled())
     579# endif
    565580                {
    566                     int rc = SUPSemEventSignal(pThis->s.CTX_SUFF(pVM)->pSession, (SUPSEMEVENT)pThis->s.Core.hEvtWrite);
    567                     AssertRC(rc);
     581                    /* Reverse the direction and signal the writer threads. */
     582                    u64State &= ~(RTCSRW_CNT_RD_MASK | RTCSRW_DIR_MASK);
     583                    u64State |= RTCSRW_DIR_WRITE << RTCSRW_DIR_SHIFT;
     584                    if (ASMAtomicCmpXchgU64(&pThis->s.Core.u64State, u64State, u64OldState))
     585                    {
     586                        int rc = SUPSemEventSignal(pThis->s.CTX_SUFF(pVM)->pSession, (SUPSEMEVENT)pThis->s.Core.hEvtWrite);
     587                        AssertRC(rc);
     588                        break;
     589                    }
     590                }
     591#endif /* IN_RING3 || IN_RING0 */
     592#ifndef IN_RING3
     593# ifdef IN_RING0
     594                else
     595# endif
     596                {
     597                    /* Queue the exit request (ring-3). */
     598                    PVM         pVM   = pThis->s.CTX_SUFF(pVM);         AssertPtr(pVM);
     599                    PVMCPU      pVCpu = VMMGetCpu(pVM);                 AssertPtr(pVCpu);
     600                    uint32_t    i     = pVCpu->pdm.s.cQueuedCritSectRwShrdLeaves++;
     601                    LogFlow(("PDMCritSectRwLeaveShared: [%d]=%p => R3 c=%d (%#llx)\n", i, pThis, c, u64State));
     602                    AssertFatal(i < RT_ELEMENTS(pVCpu->pdm.s.apQueuedCritSectRwShrdLeaves));
     603                    pVCpu->pdm.s.apQueuedCritSectRwShrdLeaves[i] = MMHyperCCToR3(pVM, pThis);
     604                    VMCPU_FF_SET(pVCpu, VMCPU_FF_PDM_CRITSECT);
     605                    VMCPU_FF_SET(pVCpu, VMCPU_FF_TO_R3);
     606                    STAM_REL_COUNTER_INC(&pVM->pdm.s.StatQueuedCritSectLeaves);
     607                    STAM_REL_COUNTER_INC(&pThis->s.StatContentionRZLeaveShared);
    568608                    break;
    569609                }
    570 #else
    571                 /* Queue the exit request (ring-3). */
    572                 PVM         pVM   = pThis->s.CTX_SUFF(pVM);         AssertPtr(pVM);
    573                 PVMCPU      pVCpu = VMMGetCpu(pVM);                 AssertPtr(pVCpu);
    574                 uint32_t    i     = pVCpu->pdm.s.cQueuedCritSectRwShrdLeaves++;
    575                 LogFlow(("PDMCritSectRwLeaveShared: [%d]=%p => R3 c=%d (%#llx)\n", i, pThis, c, u64State));
    576                 AssertFatal(i < RT_ELEMENTS(pVCpu->pdm.s.apQueuedCritSectRwShrdLeaves));
    577                 pVCpu->pdm.s.apQueuedCritSectRwShrdLeaves[i] = MMHyperCCToR3(pVM, pThis);
    578                 VMCPU_FF_SET(pVCpu, VMCPU_FF_PDM_CRITSECT);
    579                 VMCPU_FF_SET(pVCpu, VMCPU_FF_TO_R3);
    580                 STAM_REL_COUNTER_INC(&pVM->pdm.s.StatQueuedCritSectLeaves);
    581                 STAM_REL_COUNTER_INC(&pThis->s.StatContentionRZLeaveShared);
    582                 break;
    583610#endif
    584611            }
     
    762789        STAM_REL_COUNTER_INC(&pThis->s.CTX_MID_Z(StatContention,EnterExcl));
    763790
    764 #if defined(IN_RING3)
    765         if (!fTryOnly)
    766         {
     791#if defined(IN_RING3) || defined(IN_RING0)
     792        if (   !fTryOnly
     793# ifdef IN_RING0
     794            && RTThreadPreemptIsEnabled(NIL_RTTHREAD)
     795            && ASMIntAreEnabled()
     796# endif
     797           )
     798        {
     799
    767800            /*
    768801             * Wait for our turn.
     
    771804            {
    772805                int rc;
    773 # if defined(PDMCRITSECTRW_STRICT) && defined(IN_RING3)
     806# ifdef IN_RING3
     807#  ifdef PDMCRITSECTRW_STRICT
    774808                if (hThreadSelf == NIL_RTTHREAD)
    775809                    hThreadSelf = RTThreadSelfAutoAdopt();
     
    777811                                                         RT_INDEFINITE_WAIT, RTTHREADSTATE_RW_WRITE, false);
    778812                if (RT_SUCCESS(rc))
    779 # else
     813#  else
    780814                RTTHREAD hThreadSelf = RTThreadSelf();
    781815                RTThreadBlocking(hThreadSelf, RTTHREADSTATE_RW_WRITE, false);
     816#  endif
    782817# endif
    783818                {
     
    787822                                                     RT_INDEFINITE_WAIT);
    788823                    while (rc == VERR_INTERRUPTED && pThis->s.Core.u32Magic == RTCRITSECTRW_MAGIC);
     824# ifdef IN_RING3
    789825                    RTThreadUnblocked(hThreadSelf, RTTHREADSTATE_RW_WRITE);
     826# endif
    790827                    if (pThis->s.Core.u32Magic != RTCRITSECTRW_MAGIC)
    791828                        return VERR_SEM_DESTROYED;
     
    819856        }
    820857        else
    821 #endif /* IN_RING3 */
     858#endif /* IN_RING3 || IN_RING0 */
    822859        {
    823860#ifdef IN_RING3
     
    844881            if (rcBusy == VINF_SUCCESS)
    845882            {
     883                Assert(!fTryOnly);
    846884                PVM     pVM   = pThis->s.CTX_SUFF(pVM);     AssertPtr(pVM);
    847885                PVMCPU  pVCpu = VMMGetCpu(pVM);             AssertPtr(pVCpu);
     
    10521090         * Update the state.
    10531091         */
    1054 #if defined(IN_RING3)
    1055         ASMAtomicWriteU32(&pThis->s.Core.cWriteRecursions, 0);
    1056         STAM_PROFILE_ADV_STOP(&pThis->s.StatWriteLocked, swl);
    1057         ASMAtomicWriteHandle(&pThis->s.Core.hNativeWriter, NIL_RTNATIVETHREAD);
    1058 
    1059         for (;;)
    1060         {
    1061             uint64_t u64State    = ASMAtomicReadU64(&pThis->s.Core.u64State);
    1062             uint64_t u64OldState = u64State;
    1063 
    1064             uint64_t c = (u64State & RTCSRW_CNT_WR_MASK) >> RTCSRW_CNT_WR_SHIFT;
    1065             Assert(c > 0);
    1066             c--;
    1067 
    1068             if (   c > 0
    1069                 || (u64State & RTCSRW_CNT_RD_MASK) == 0)
     1092#if defined(IN_RING3) || defined(IN_RING0)
     1093# ifdef IN_RING0
     1094        if (   RTThreadPreemptIsEnabled(NIL_RTTHREAD)
     1095            && ASMIntAreEnabled())
     1096# endif
     1097        {
     1098            ASMAtomicWriteU32(&pThis->s.Core.cWriteRecursions, 0);
     1099            STAM_PROFILE_ADV_STOP(&pThis->s.StatWriteLocked, swl);
     1100            ASMAtomicWriteHandle(&pThis->s.Core.hNativeWriter, NIL_RTNATIVETHREAD);
     1101
     1102            for (;;)
    10701103            {
    1071                 /* Don't change the direction, wake up the next writer if any. */
    1072                 u64State &= ~RTCSRW_CNT_WR_MASK;
    1073                 u64State |= c << RTCSRW_CNT_WR_SHIFT;
    1074                 if (ASMAtomicCmpXchgU64(&pThis->s.Core.u64State, u64State, u64OldState))
     1104                uint64_t u64State    = ASMAtomicReadU64(&pThis->s.Core.u64State);
     1105                uint64_t u64OldState = u64State;
     1106
     1107                uint64_t c = (u64State & RTCSRW_CNT_WR_MASK) >> RTCSRW_CNT_WR_SHIFT;
     1108                Assert(c > 0);
     1109                c--;
     1110
     1111                if (   c > 0
     1112                    || (u64State & RTCSRW_CNT_RD_MASK) == 0)
    10751113                {
    1076                     if (c > 0)
     1114                    /* Don't change the direction, wake up the next writer if any. */
     1115                    u64State &= ~RTCSRW_CNT_WR_MASK;
     1116                    u64State |= c << RTCSRW_CNT_WR_SHIFT;
     1117                    if (ASMAtomicCmpXchgU64(&pThis->s.Core.u64State, u64State, u64OldState))
    10771118                    {
    1078                         int rc = SUPSemEventSignal(pThis->s.CTX_SUFF(pVM)->pSession, (SUPSEMEVENT)pThis->s.Core.hEvtWrite);
     1119                        if (c > 0)
     1120                        {
     1121                            int rc = SUPSemEventSignal(pThis->s.CTX_SUFF(pVM)->pSession, (SUPSEMEVENT)pThis->s.Core.hEvtWrite);
     1122                            AssertRC(rc);
     1123                        }
     1124                        break;
     1125                    }
     1126                }
     1127                else
     1128                {
     1129                    /* Reverse the direction and signal the reader threads. */
     1130                    u64State &= ~(RTCSRW_CNT_WR_MASK | RTCSRW_DIR_MASK);
     1131                    u64State |= RTCSRW_DIR_READ << RTCSRW_DIR_SHIFT;
     1132                    if (ASMAtomicCmpXchgU64(&pThis->s.Core.u64State, u64State, u64OldState))
     1133                    {
     1134                        Assert(!pThis->s.Core.fNeedReset);
     1135                        ASMAtomicWriteBool(&pThis->s.Core.fNeedReset, true);
     1136                        int rc = SUPSemEventMultiSignal(pThis->s.CTX_SUFF(pVM)->pSession, (SUPSEMEVENTMULTI)pThis->s.Core.hEvtRead);
    10791137                        AssertRC(rc);
     1138                        break;
    10801139                    }
    1081                     break;
    10821140                }
     1141
     1142                ASMNopPause();
     1143                if (pThis->s.Core.u32Magic != RTCRITSECTRW_MAGIC)
     1144                    return VERR_SEM_DESTROYED;
    10831145            }
    1084             else
    1085             {
    1086                 /* Reverse the direction and signal the reader threads. */
    1087                 u64State &= ~(RTCSRW_CNT_WR_MASK | RTCSRW_DIR_MASK);
    1088                 u64State |= RTCSRW_DIR_READ << RTCSRW_DIR_SHIFT;
    1089                 if (ASMAtomicCmpXchgU64(&pThis->s.Core.u64State, u64State, u64OldState))
    1090                 {
    1091                     Assert(!pThis->s.Core.fNeedReset);
    1092                     ASMAtomicWriteBool(&pThis->s.Core.fNeedReset, true);
    1093                     int rc = SUPSemEventMultiSignal(pThis->s.CTX_SUFF(pVM)->pSession, (SUPSEMEVENTMULTI)pThis->s.Core.hEvtRead);
    1094                     AssertRC(rc);
    1095                     break;
    1096                 }
    1097             }
    1098 
    1099             ASMNopPause();
    1100             if (pThis->s.Core.u32Magic != RTCRITSECTRW_MAGIC)
    1101                 return VERR_SEM_DESTROYED;
    1102         }
    1103 #else
    1104         /*
    1105          * We cannot call neither SUPSemEventSignal nor SUPSemEventMultiSignal,
    1106          * so queue the exit request (ring-3).
    1107          */
    1108         PVM         pVM   = pThis->s.CTX_SUFF(pVM);         AssertPtr(pVM);
    1109         PVMCPU      pVCpu = VMMGetCpu(pVM);                 AssertPtr(pVCpu);
    1110         uint32_t    i     = pVCpu->pdm.s.cQueuedCritSectRwExclLeaves++;
    1111         LogFlow(("PDMCritSectRwLeaveShared: [%d]=%p => R3\n", i, pThis));
    1112         AssertFatal(i < RT_ELEMENTS(pVCpu->pdm.s.apQueuedCritSectLeaves));
    1113         pVCpu->pdm.s.apQueuedCritSectRwExclLeaves[i] = MMHyperCCToR3(pVM, pThis);
    1114         VMCPU_FF_SET(pVCpu, VMCPU_FF_PDM_CRITSECT);
    1115         VMCPU_FF_SET(pVCpu, VMCPU_FF_TO_R3);
    1116         STAM_REL_COUNTER_INC(&pVM->pdm.s.StatQueuedCritSectLeaves);
    1117         STAM_REL_COUNTER_INC(&pThis->s.StatContentionRZLeaveExcl);
     1146        }
     1147#endif /* IN_RING3 || IN_RING0 */
     1148#ifndef IN_RING3
     1149# ifdef IN_RING0
     1150        else
     1151# endif
     1152        {
     1153            /*
     1154             * We cannot call neither SUPSemEventSignal nor SUPSemEventMultiSignal,
     1155             * so queue the exit request (ring-3).
     1156             */
     1157            PVM         pVM   = pThis->s.CTX_SUFF(pVM);         AssertPtr(pVM);
     1158            PVMCPU      pVCpu = VMMGetCpu(pVM);                 AssertPtr(pVCpu);
     1159            uint32_t    i     = pVCpu->pdm.s.cQueuedCritSectRwExclLeaves++;
     1160            LogFlow(("PDMCritSectRwLeaveShared: [%d]=%p => R3\n", i, pThis));
     1161            AssertFatal(i < RT_ELEMENTS(pVCpu->pdm.s.apQueuedCritSectLeaves));
     1162            pVCpu->pdm.s.apQueuedCritSectRwExclLeaves[i] = MMHyperCCToR3(pVM, pThis);
     1163            VMCPU_FF_SET(pVCpu, VMCPU_FF_PDM_CRITSECT);
     1164            VMCPU_FF_SET(pVCpu, VMCPU_FF_TO_R3);
     1165            STAM_REL_COUNTER_INC(&pVM->pdm.s.StatQueuedCritSectLeaves);
     1166            STAM_REL_COUNTER_INC(&pThis->s.StatContentionRZLeaveExcl);
     1167        }
    11181168#endif
    11191169    }
Note: See TracChangeset for help on using the changeset viewer.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette