Store a reference to the effect buffer as an active property
This commit is contained in:
parent
1a9fbc1b2f
commit
a6bd53c4e1
@ -702,6 +702,8 @@ ALeffectslot::~ALeffectslot()
|
||||
|
||||
if(Params.mEffectState)
|
||||
Params.mEffectState->release();
|
||||
if(Params.mEffectBuffer)
|
||||
Params.mEffectBuffer->release();
|
||||
}
|
||||
|
||||
ALenum ALeffectslot::init()
|
||||
@ -764,6 +766,7 @@ ALenum ALeffectslot::initEffect(ALeffect *effect, ALCcontext *context)
|
||||
while(props)
|
||||
{
|
||||
props->State = nullptr;
|
||||
props->Buffer = nullptr;
|
||||
props = props->next.load(std::memory_order_relaxed);
|
||||
}
|
||||
|
||||
@ -792,10 +795,8 @@ void ALeffectslot::updateProps(ALCcontext *context)
|
||||
|
||||
props->Type = Effect.Type;
|
||||
props->Props = Effect.Props;
|
||||
/* Swap out any stale effect state object there may be in the container, to
|
||||
* delete it.
|
||||
*/
|
||||
props->State = Effect.State;
|
||||
props->Buffer = Effect.Buffer;
|
||||
|
||||
/* Set the new container for updating internal parameters. */
|
||||
props = Params.Update.exchange(props, std::memory_order_acq_rel);
|
||||
@ -805,6 +806,7 @@ void ALeffectslot::updateProps(ALCcontext *context)
|
||||
* freelist.
|
||||
*/
|
||||
props->State = nullptr;
|
||||
props->Buffer = nullptr;
|
||||
AtomicReplaceHead(context->mFreeEffectslotProps, props);
|
||||
}
|
||||
}
|
||||
|
@ -32,6 +32,7 @@ struct ALeffectslotProps {
|
||||
EffectProps Props;
|
||||
|
||||
al::intrusive_ptr<EffectState> State;
|
||||
al::intrusive_ptr<EffectBufferBase> Buffer;
|
||||
|
||||
std::atomic<ALeffectslotProps*> next;
|
||||
|
||||
@ -67,6 +68,7 @@ struct ALeffectslot {
|
||||
ALenum EffectType{AL_EFFECT_NULL};
|
||||
EffectProps mEffectProps{};
|
||||
EffectState *mEffectState{nullptr};
|
||||
EffectBufferBase *mEffectBuffer{nullptr};
|
||||
|
||||
float RoomRolloff{0.0f}; /* Added to the source's room rolloff, not multiplied. */
|
||||
float DecayTime{0.0f};
|
||||
|
@ -60,6 +60,11 @@ static int EventThread(ALCcontext *context)
|
||||
evt.u.mEffectState->release();
|
||||
continue;
|
||||
}
|
||||
if(evt.EnumType == EventType_ReleaseEffectBuffer)
|
||||
{
|
||||
evt.u.mEffectBuffer->release();
|
||||
continue;
|
||||
}
|
||||
|
||||
ALbitfieldSOFT enabledevts{context->mEnabledEvts.load(std::memory_order_acquire)};
|
||||
if(!context->mEventCb) continue;
|
||||
|
@ -6,6 +6,7 @@
|
||||
|
||||
#include "almalloc.h"
|
||||
|
||||
struct EffectBufferBase;
|
||||
struct EffectState;
|
||||
|
||||
|
||||
@ -23,6 +24,7 @@ enum {
|
||||
|
||||
/* Internal events. */
|
||||
EventType_ReleaseEffectState = 65536,
|
||||
EventType_ReleaseEffectBuffer,
|
||||
};
|
||||
|
||||
struct AsyncEvent {
|
||||
@ -44,6 +46,7 @@ struct AsyncEvent {
|
||||
ALchar msg[232];
|
||||
} user;
|
||||
EffectState *mEffectState;
|
||||
EffectBufferBase *mEffectBuffer;
|
||||
} u{};
|
||||
|
||||
AsyncEvent() noexcept = default;
|
||||
|
18
alc/alu.cpp
18
alc/alu.cpp
@ -505,6 +505,24 @@ bool CalcEffectSlotParams(ALeffectslot *slot, ALeffectslot **sorted_slots, ALCco
|
||||
}
|
||||
}
|
||||
|
||||
EffectBufferBase *buffer{props->Buffer.release()};
|
||||
EffectBufferBase *oldbuffer{slot->Params.mEffectBuffer};
|
||||
slot->Params.mEffectBuffer = buffer;
|
||||
|
||||
if(oldbuffer && !oldbuffer->releaseIfNoDelete())
|
||||
{
|
||||
RingBuffer *ring{context->mAsyncEvents.get()};
|
||||
auto evt_vec = ring->getWriteVector();
|
||||
if LIKELY(evt_vec.first.len > 0)
|
||||
{
|
||||
AsyncEvent *evt{::new(evt_vec.first.buf) AsyncEvent{EventType_ReleaseEffectBuffer}};
|
||||
evt->u.mEffectBuffer = oldbuffer;
|
||||
ring->writeAdvance(1);
|
||||
}
|
||||
else
|
||||
props->Buffer.reset(oldbuffer);
|
||||
}
|
||||
|
||||
AtomicReplaceHead(context->mFreeEffectslotProps, props);
|
||||
|
||||
EffectTarget output;
|
||||
|
Loading…
x
Reference in New Issue
Block a user