-
-
Notifications
You must be signed in to change notification settings - Fork 22k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add support for AIFF files and other WAV formats #96545
base: master
Are you sure you want to change the base?
Conversation
@@ -35,6 +35,15 @@ | |||
#include "core/io/resource_saver.h" | |||
#include "scene/resources/audio_stream_wav.h" | |||
|
|||
#define DRWAV_IMPLEMENTATION | |||
#define DR_WAV_NO_STDIO | |||
#define DR_WAV_LIBSNDFILE_COMPAT |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Despite not using libsndfile
, the way Godot converts streams from/to f32 follows similar calculations, therefore not defining this leads to some test errors (save/reimport datas being different).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Horray
Added a check to detect if the container is RF64 or W64. Even if they could be imported in theory, this is the best non-compatibility breaking way to deny getting a stream that is too big for |
Modified the check again, this time it actually checks the amount of samples (maximum of 2147483647). Normally, the importer uses an |
Does the dr mp4 problem affect dr_wav? Not sure of the exact problem. |
The problem with Doesn't apply to WAV. |
0925491
to
ce27e00
Compare
Added |
else if (loop.type == drwav_smpl_loop_type_backward) | ||
loop_mode = AudioStreamWAV::LOOP_BACKWARD; | ||
loop_begin = loop.firstSampleByteOffset; | ||
loop_end = loop.lastSampleByteOffset; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The "Byte" in the names here is misleading, it's actually the offsets of loop begin/end in frames, not bytes.
I have tested this considering the old read code actually copied a sample value directly, not a byte offset.
Works the same way as the previous implementation just fine.
ffbc68e
to
723e117
Compare
Updated ResourceImporterWAV documentation to point out it also supports AIFF. |
c133869
to
67fe879
Compare
Rebased. I took the opportunity to redo how the data is read. dr_wav offers 3 ways: From a file through stdio, from memory and from custom read/seek functions. Before the last push, it was loading the entire file data into the memory, then reading from memory, which resulted into a massive increase in memory usage during import if the file was too large. Now it uses custom functions (lambdas) that tell FileAccess to react accordingly, effectively being an integration between FileAccess and dr_wav. I don't know if I violated any Godot style guideline, so I'd like to ask for another review. The patch in dr_wav could be applied directly into a define, so it was removed. |
Addressing changes from #93831, which moved most of the WAV loading code from ResourceImporterWAV to AudioStreamWAV. AudioStreamWAV documentation updated to explain that the load functions also work with AIFF files and data. Since the load now requires loading into memory anyway, the old init_memory read has been restored. |
For some reason, when the tests call This is odd since I didn't even touch that function. Edit: it seems like getting the data pointer directly instead of using |
Any probability of this being approved for 4.4? |
Rebased on top of |
The best I can do is review for 4.5 and queue it for merge. Test plan
|
Since the WAV import procedure was moved from the editor into the templates during the development of this PR (which was relying on it being editor only), I decided to make some builds for measurement. On Linux builds, I don't see a solution to this other than making |
This PR only changes function to read non-imported WAV files, not the playback (in fact, this should have gotten a Additionally, modifying AudioStreamWAV to use dr_wav's save procedure (not done in this PR) increases binary size by an additional 4KB. Unit tests already cover this by generating a PCM stream, saving it, loading and comparing if the loaded stream is 1:1 to the generated one before saving. It's the reason |
d4f9d37
to
ab6fbd6
Compare
Rebased once more. A modification from #93831 bothered me. It changed the file read procedure to load the entire file into memory instead of using FileAccess to get values from disk. Thanks to dr_wav, we can initialize a |
2d04924
to
a13f446
Compare
I was doing an experiment with On the other hand, I decided to wait for a patch in |
AIFF is also the default format for Garage Band, so it comes in very handy to have this support. |
Changes
In
AudioStreamWAV::load_from_buffer()
, replaces the entire data read step with one that usesdr_wav
.Formats already supported by Godot and loop info detection should remain supported without regressions.
Extra WAV encoding formats like A-law, μ-law, MS ADPCM and IMA ADPCM can now be imported.
AIFF support
dr_wav
is also able to read AIFF files, soResourceImporterWAV
was enabled to recognize it too.load_from_buffer
andload_from_file
should recognize AIFF data/files as well.The importer was renamed from
Microsoft WAV
toMicrosoft WAV/Apple AIFF
to reflect that.When it comes to sound effects, freesound.org provides much more AIFF than Ogg Vorbis files, so I thought adding support was reasonable:

wave
,aif
,aiff
andaifc
file extensions have been added as recognized extensions alongsidewav
in the importer.Actually load from file instead of a file buffer
#93831 changed the load function to copy the entire file data into memory from using
FileAccess
to get values.You can tell dr_wav to use specific read/seek callbacks, so I wrote callbacks to make
load_from_file
read using FileAccess instead of a memory copy.Caveats
I noticed a 36 KiB size penalty on production template_release builds (71991600 -> 72028464). This PR was opened when the load procedure was editor-only, and during its development was moved into
AudioStreamWAV
to serve as a load function. I don't know if 36 KiB is too much for this, but as a side effect reduces the probability of errors.