Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
W
wine-fonts
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Registry
Registry
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Aleksandr Isakov
wine-fonts
Commits
979108df
Commit
979108df
authored
Mar 22, 2015
by
Mark Harmstone
Committed by
Vitaly Lipatov
Jul 30, 2022
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
dsound: Implement EAX early reflections.
parent
cf0b1b2c
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
121 additions
and
0 deletions
+121
-0
dsound_eax.h
dlls/dsound/dsound_eax.h
+7
-0
eax.c
dlls/dsound/eax.c
+114
-0
No files found.
dlls/dsound/dsound_eax.h
View file @
979108df
...
@@ -144,6 +144,13 @@ typedef struct {
...
@@ -144,6 +144,13 @@ typedef struct {
DelayLine
Delay
;
DelayLine
Delay
;
unsigned
int
DelayTap
[
2
];
unsigned
int
DelayTap
[
2
];
struct
{
float
Gain
;
float
Coeff
[
4
];
DelayLine
Delay
[
4
];
unsigned
int
Offset
[
4
];
}
Early
;
unsigned
int
Offset
;
unsigned
int
Offset
;
}
eax_buffer_info
;
}
eax_buffer_info
;
...
...
dlls/dsound/eax.c
View file @
979108df
...
@@ -92,6 +92,11 @@ static const EFXEAXREVERBPROPERTIES efx_presets[] = {
...
@@ -92,6 +92,11 @@ static const EFXEAXREVERBPROPERTIES efx_presets[] = {
{
0
.
0625
f
,
0
.
5000
f
,
0
.
3162
f
,
0
.
8404
f
,
1
.
0000
f
,
7
.
5600
f
,
0
.
9100
f
,
1
.
0000
f
,
0
.
4864
f
,
0
.
0200
f
,
{
0
.
0000
f
,
0
.
0000
f
,
0
.
0000
f
},
2
.
4378
f
,
0
.
0300
f
,
{
0
.
0000
f
,
0
.
0000
f
,
0
.
0000
f
},
0
.
2500
f
,
0
.
0000
f
,
4
.
0000
f
,
1
.
0000
f
,
0
.
9943
f
,
5000
.
0000
f
,
250
.
0000
f
,
0
.
0000
f
,
0x0
}
/* psychotic */
{
0
.
0625
f
,
0
.
5000
f
,
0
.
3162
f
,
0
.
8404
f
,
1
.
0000
f
,
7
.
5600
f
,
0
.
9100
f
,
1
.
0000
f
,
0
.
4864
f
,
0
.
0200
f
,
{
0
.
0000
f
,
0
.
0000
f
,
0
.
0000
f
},
2
.
4378
f
,
0
.
0300
f
,
{
0
.
0000
f
,
0
.
0000
f
,
0
.
0000
f
},
0
.
2500
f
,
0
.
0000
f
,
4
.
0000
f
,
1
.
0000
f
,
0
.
9943
f
,
5000
.
0000
f
,
250
.
0000
f
,
0
.
0000
f
,
0x0
}
/* psychotic */
};
};
static
const
float
EARLY_LINE_LENGTH
[
4
]
=
{
0
.
0015
f
,
0
.
0045
f
,
0
.
0135
f
,
0
.
0405
f
};
static
float
lpFilter2P
(
FILTER
*
iir
,
unsigned
int
offset
,
float
input
)
static
float
lpFilter2P
(
FILTER
*
iir
,
unsigned
int
offset
,
float
input
)
{
{
float
*
history
=
&
iir
->
history
[
offset
*
2
];
float
*
history
=
&
iir
->
history
[
offset
*
2
];
...
@@ -116,6 +121,62 @@ static float DelayLineOut(DelayLine *Delay, unsigned int offset)
...
@@ -116,6 +121,62 @@ static float DelayLineOut(DelayLine *Delay, unsigned int offset)
return
Delay
->
Line
[
offset
&
Delay
->
Mask
];
return
Delay
->
Line
[
offset
&
Delay
->
Mask
];
}
}
static
float
AttenuatedDelayLineOut
(
DelayLine
*
Delay
,
unsigned
int
offset
,
float
coeff
)
{
return
coeff
*
Delay
->
Line
[
offset
&
Delay
->
Mask
];
}
static
float
EarlyDelayLineOut
(
IDirectSoundBufferImpl
*
dsb
,
unsigned
int
index
)
{
return
AttenuatedDelayLineOut
(
&
dsb
->
eax
.
Early
.
Delay
[
index
],
dsb
->
eax
.
Offset
-
dsb
->
eax
.
Early
.
Offset
[
index
],
dsb
->
eax
.
Early
.
Coeff
[
index
]);
}
static
void
EarlyReflection
(
IDirectSoundBufferImpl
*
dsb
,
float
in
,
float
*
out
)
{
float
d
[
4
],
v
,
f
[
4
];
/* Obtain the decayed results of each early delay line. */
d
[
0
]
=
EarlyDelayLineOut
(
dsb
,
0
);
d
[
1
]
=
EarlyDelayLineOut
(
dsb
,
1
);
d
[
2
]
=
EarlyDelayLineOut
(
dsb
,
2
);
d
[
3
]
=
EarlyDelayLineOut
(
dsb
,
3
);
/* The following uses a lossless scattering junction from waveguide
* theory. It actually amounts to a householder mixing matrix, which
* will produce a maximally diffuse response, and means this can probably
* be considered a simple feed-back delay network (FDN).
* N
* ---
* \
* v = 2/N / d_i
* ---
* i=1
*/
v
=
(
d
[
0
]
+
d
[
1
]
+
d
[
2
]
+
d
[
3
])
*
0
.
5
f
;
/* The junction is loaded with the input here. */
v
+=
in
;
/* Calculate the feed values for the delay lines. */
f
[
0
]
=
v
-
d
[
0
];
f
[
1
]
=
v
-
d
[
1
];
f
[
2
]
=
v
-
d
[
2
];
f
[
3
]
=
v
-
d
[
3
];
/* Re-feed the delay lines. */
DelayLineIn
(
&
dsb
->
eax
.
Early
.
Delay
[
0
],
dsb
->
eax
.
Offset
,
f
[
0
]);
DelayLineIn
(
&
dsb
->
eax
.
Early
.
Delay
[
1
],
dsb
->
eax
.
Offset
,
f
[
1
]);
DelayLineIn
(
&
dsb
->
eax
.
Early
.
Delay
[
2
],
dsb
->
eax
.
Offset
,
f
[
2
]);
DelayLineIn
(
&
dsb
->
eax
.
Early
.
Delay
[
3
],
dsb
->
eax
.
Offset
,
f
[
3
]);
/* Output the results of the junction for all four channels. */
out
[
0
]
=
dsb
->
eax
.
Early
.
Gain
*
f
[
0
];
out
[
1
]
=
dsb
->
eax
.
Early
.
Gain
*
f
[
1
];
out
[
2
]
=
dsb
->
eax
.
Early
.
Gain
*
f
[
2
];
out
[
3
]
=
dsb
->
eax
.
Early
.
Gain
*
f
[
3
];
}
static
void
VerbPass
(
IDirectSoundBufferImpl
*
dsb
,
float
in
,
float
*
out
)
static
void
VerbPass
(
IDirectSoundBufferImpl
*
dsb
,
float
in
,
float
*
out
)
{
{
/* Low-pass filter the incoming sample. */
/* Low-pass filter the incoming sample. */
...
@@ -124,7 +185,9 @@ static void VerbPass(IDirectSoundBufferImpl* dsb, float in, float* out)
...
@@ -124,7 +185,9 @@ static void VerbPass(IDirectSoundBufferImpl* dsb, float in, float* out)
/* Feed the initial delay line. */
/* Feed the initial delay line. */
DelayLineIn
(
&
dsb
->
eax
.
Delay
,
dsb
->
eax
.
Offset
,
in
);
DelayLineIn
(
&
dsb
->
eax
.
Delay
,
dsb
->
eax
.
Offset
,
in
);
/* Calculate the early reflection from the first delay tap. */
in
=
DelayLineOut
(
&
dsb
->
eax
.
Delay
,
dsb
->
eax
.
Offset
-
dsb
->
eax
.
DelayTap
[
0
]);
in
=
DelayLineOut
(
&
dsb
->
eax
.
Delay
,
dsb
->
eax
.
Offset
-
dsb
->
eax
.
DelayTap
[
0
]);
EarlyReflection
(
dsb
,
in
,
out
);
/* Step all delays forward one sample. */
/* Step all delays forward one sample. */
dsb
->
eax
.
Offset
++
;
dsb
->
eax
.
Offset
++
;
...
@@ -173,6 +236,27 @@ static void UpdateDelayLine(float earlyDelay, float lateDelay, unsigned int freq
...
@@ -173,6 +236,27 @@ static void UpdateDelayLine(float earlyDelay, float lateDelay, unsigned int freq
State
->
DelayTap
[
1
]
=
fastf2u
((
earlyDelay
+
lateDelay
)
*
frequency
);
State
->
DelayTap
[
1
]
=
fastf2u
((
earlyDelay
+
lateDelay
)
*
frequency
);
}
}
static
float
CalcDecayCoeff
(
float
length
,
float
decayTime
)
{
return
powf
(
0
.
001
f
/*-60 dB*/
,
length
/
decayTime
);
}
static
void
UpdateEarlyLines
(
float
reverbGain
,
float
earlyGain
,
float
lateDelay
,
eax_buffer_info
*
State
)
{
unsigned
int
index
;
/* Calculate the early reflections gain (from the master effect gain, and
* reflections gain parameters) with a constant attenuation of 0.5. */
State
->
Early
.
Gain
=
0
.
5
f
*
reverbGain
*
earlyGain
;
/* Calculate the gain (coefficient) for each early delay line using the
* late delay time. This expands the early reflections to the start of
* the late reverb. */
for
(
index
=
0
;
index
<
4
;
index
++
)
State
->
Early
.
Coeff
[
index
]
=
CalcDecayCoeff
(
EARLY_LINE_LENGTH
[
index
],
lateDelay
);
}
static
float
lpCoeffCalc
(
float
g
,
float
cw
)
static
float
lpCoeffCalc
(
float
g
,
float
cw
)
{
{
float
a
=
0
.
0
f
;
float
a
=
0
.
0
f
;
...
@@ -244,6 +328,11 @@ static BOOL AllocLines(unsigned int frequency, IDirectSoundBufferImpl *dsb)
...
@@ -244,6 +328,11 @@ static BOOL AllocLines(unsigned int frequency, IDirectSoundBufferImpl *dsb)
totalSamples
+=
CalcLineLength
(
length
,
totalSamples
,
frequency
,
totalSamples
+=
CalcLineLength
(
length
,
totalSamples
,
frequency
,
&
dsb
->
eax
.
Delay
);
&
dsb
->
eax
.
Delay
);
/* The early reflection lines. */
for
(
index
=
0
;
index
<
4
;
index
++
)
totalSamples
+=
CalcLineLength
(
EARLY_LINE_LENGTH
[
index
],
totalSamples
,
frequency
,
&
dsb
->
eax
.
Early
.
Delay
[
index
]);
if
(
totalSamples
!=
dsb
->
eax
.
TotalSamples
)
if
(
totalSamples
!=
dsb
->
eax
.
TotalSamples
)
{
{
TRACE
(
"New reverb buffer length: %u samples (%f sec)
\n
"
,
totalSamples
,
totalSamples
/
(
float
)
frequency
);
TRACE
(
"New reverb buffer length: %u samples (%f sec)
\n
"
,
totalSamples
,
totalSamples
/
(
float
)
frequency
);
...
@@ -261,6 +350,10 @@ static BOOL AllocLines(unsigned int frequency, IDirectSoundBufferImpl *dsb)
...
@@ -261,6 +350,10 @@ static BOOL AllocLines(unsigned int frequency, IDirectSoundBufferImpl *dsb)
/* Update all delays to reflect the new sample buffer. */
/* Update all delays to reflect the new sample buffer. */
RealizeLineOffset
(
dsb
->
eax
.
SampleBuffer
,
&
dsb
->
eax
.
Delay
);
RealizeLineOffset
(
dsb
->
eax
.
SampleBuffer
,
&
dsb
->
eax
.
Delay
);
for
(
index
=
0
;
index
<
4
;
index
++
)
{
RealizeLineOffset
(
dsb
->
eax
.
SampleBuffer
,
&
dsb
->
eax
.
Early
.
Delay
[
index
]);
}
/* Clear the sample buffer. */
/* Clear the sample buffer. */
for
(
index
=
0
;
index
<
dsb
->
eax
.
TotalSamples
;
index
++
)
for
(
index
=
0
;
index
<
dsb
->
eax
.
TotalSamples
;
index
++
)
...
@@ -271,6 +364,7 @@ static BOOL AllocLines(unsigned int frequency, IDirectSoundBufferImpl *dsb)
...
@@ -271,6 +364,7 @@ static BOOL AllocLines(unsigned int frequency, IDirectSoundBufferImpl *dsb)
static
void
ReverbUpdate
(
IDirectSoundBufferImpl
*
dsb
)
static
void
ReverbUpdate
(
IDirectSoundBufferImpl
*
dsb
)
{
{
unsigned
int
index
;
float
cw
;
float
cw
;
/* avoid segfaults in mixing thread when we recalculate the line offsets */
/* avoid segfaults in mixing thread when we recalculate the line offsets */
...
@@ -280,6 +374,11 @@ static void ReverbUpdate(IDirectSoundBufferImpl *dsb)
...
@@ -280,6 +374,11 @@ static void ReverbUpdate(IDirectSoundBufferImpl *dsb)
LeaveCriticalSection
(
&
dsb
->
device
->
mixlock
);
LeaveCriticalSection
(
&
dsb
->
device
->
mixlock
);
for
(
index
=
0
;
index
<
4
;
index
++
)
{
dsb
->
eax
.
Early
.
Offset
[
index
]
=
fastf2u
(
EARLY_LINE_LENGTH
[
index
]
*
dsb
->
device
->
pwfx
->
nSamplesPerSec
);
}
cw
=
CalcI3DL2HFreq
(
dsb
->
device
->
eax
.
eax_props
.
flHFReference
,
dsb
->
device
->
pwfx
->
nSamplesPerSec
);
cw
=
CalcI3DL2HFreq
(
dsb
->
device
->
eax
.
eax_props
.
flHFReference
,
dsb
->
device
->
pwfx
->
nSamplesPerSec
);
dsb
->
eax
.
LpFilter
.
coeff
=
lpCoeffCalc
(
dsb
->
device
->
eax
.
eax_props
.
flGainHF
,
cw
);
dsb
->
eax
.
LpFilter
.
coeff
=
lpCoeffCalc
(
dsb
->
device
->
eax
.
eax_props
.
flGainHF
,
cw
);
...
@@ -287,6 +386,10 @@ static void ReverbUpdate(IDirectSoundBufferImpl *dsb)
...
@@ -287,6 +386,10 @@ static void ReverbUpdate(IDirectSoundBufferImpl *dsb)
UpdateDelayLine
(
dsb
->
device
->
eax
.
eax_props
.
flReflectionsDelay
,
UpdateDelayLine
(
dsb
->
device
->
eax
.
eax_props
.
flReflectionsDelay
,
dsb
->
device
->
eax
.
eax_props
.
flLateReverbDelay
,
dsb
->
device
->
eax
.
eax_props
.
flLateReverbDelay
,
dsb
->
device
->
pwfx
->
nSamplesPerSec
,
&
dsb
->
eax
);
dsb
->
device
->
pwfx
->
nSamplesPerSec
,
&
dsb
->
eax
);
UpdateEarlyLines
(
dsb
->
device
->
eax
.
eax_props
.
flGain
,
dsb
->
device
->
eax
.
eax_props
.
flReflectionsGain
,
dsb
->
device
->
eax
.
eax_props
.
flLateReverbDelay
,
&
dsb
->
eax
);
}
}
static
BOOL
ReverbDeviceUpdate
(
DirectSoundDevice
*
dev
)
static
BOOL
ReverbDeviceUpdate
(
DirectSoundDevice
*
dev
)
...
@@ -302,6 +405,8 @@ static BOOL ReverbDeviceUpdate(DirectSoundDevice *dev)
...
@@ -302,6 +405,8 @@ static BOOL ReverbDeviceUpdate(DirectSoundDevice *dev)
void
init_eax_buffer
(
IDirectSoundBufferImpl
*
dsb
)
void
init_eax_buffer
(
IDirectSoundBufferImpl
*
dsb
)
{
{
unsigned
int
index
;
dsb
->
eax
.
TotalSamples
=
0
;
dsb
->
eax
.
TotalSamples
=
0
;
dsb
->
eax
.
SampleBuffer
=
NULL
;
dsb
->
eax
.
SampleBuffer
=
NULL
;
...
@@ -314,6 +419,15 @@ void init_eax_buffer(IDirectSoundBufferImpl *dsb)
...
@@ -314,6 +419,15 @@ void init_eax_buffer(IDirectSoundBufferImpl *dsb)
dsb
->
eax
.
DelayTap
[
0
]
=
0
;
dsb
->
eax
.
DelayTap
[
0
]
=
0
;
dsb
->
eax
.
DelayTap
[
1
]
=
0
;
dsb
->
eax
.
DelayTap
[
1
]
=
0
;
dsb
->
eax
.
Early
.
Gain
=
0
.
0
f
;
for
(
index
=
0
;
index
<
4
;
index
++
)
{
dsb
->
eax
.
Early
.
Coeff
[
index
]
=
0
.
0
f
;
dsb
->
eax
.
Early
.
Delay
[
index
].
Mask
=
0
;
dsb
->
eax
.
Early
.
Delay
[
index
].
Line
=
NULL
;
dsb
->
eax
.
Early
.
Offset
[
index
]
=
0
;
}
dsb
->
eax
.
Offset
=
0
;
dsb
->
eax
.
Offset
=
0
;
ReverbUpdate
(
dsb
);
ReverbUpdate
(
dsb
);
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment