WavWriter.cpp 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106
  1. /*
  2. * Copyright (c) 2020, William McPherson <willmcpherson2@gmail.com>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <LibAudio/WavWriter.h>
  7. namespace Audio {
  8. WavWriter::WavWriter(const StringView& path, int sample_rate, int num_channels, int bits_per_sample)
  9. : m_sample_rate(sample_rate)
  10. , m_num_channels(num_channels)
  11. , m_bits_per_sample(bits_per_sample)
  12. {
  13. set_file(path);
  14. }
  15. WavWriter::WavWriter(int sample_rate, int num_channels, int bits_per_sample)
  16. : m_sample_rate(sample_rate)
  17. , m_num_channels(num_channels)
  18. , m_bits_per_sample(bits_per_sample)
  19. {
  20. }
  21. WavWriter::~WavWriter()
  22. {
  23. if (!m_finalized)
  24. finalize();
  25. }
  26. void WavWriter::set_file(const StringView& path)
  27. {
  28. m_file = Core::File::construct(path);
  29. if (!m_file->open(Core::OpenMode::ReadWrite)) {
  30. m_error_string = String::formatted("Can't open file: {}", m_file->error_string());
  31. return;
  32. }
  33. m_file->seek(44);
  34. m_finalized = false;
  35. }
  36. void WavWriter::write_samples(const u8* samples, size_t size)
  37. {
  38. m_data_sz += size;
  39. m_file->write(samples, size);
  40. }
  41. void WavWriter::finalize()
  42. {
  43. VERIFY(!m_finalized);
  44. m_finalized = true;
  45. if (m_file) {
  46. m_file->seek(0);
  47. write_header();
  48. m_file->close();
  49. }
  50. m_data_sz = 0;
  51. }
  52. void WavWriter::write_header()
  53. {
  54. // "RIFF"
  55. static u32 riff = 0x46464952;
  56. m_file->write(reinterpret_cast<u8*>(&riff), sizeof(riff));
  57. // Size of data + (size of header - previous field - this field)
  58. u32 sz = m_data_sz + (44 - 4 - 4);
  59. m_file->write(reinterpret_cast<u8*>(&sz), sizeof(sz));
  60. // "WAVE"
  61. static u32 wave = 0x45564157;
  62. m_file->write(reinterpret_cast<u8*>(&wave), sizeof(wave));
  63. // "fmt "
  64. static u32 fmt_id = 0x20746D66;
  65. m_file->write(reinterpret_cast<u8*>(&fmt_id), sizeof(fmt_id));
  66. // Size of the next 6 fields
  67. static u32 fmt_size = 16;
  68. m_file->write(reinterpret_cast<u8*>(&fmt_size), sizeof(fmt_size));
  69. // 1 for PCM
  70. static u16 audio_format = 1;
  71. m_file->write(reinterpret_cast<u8*>(&audio_format), sizeof(audio_format));
  72. m_file->write(reinterpret_cast<u8*>(&m_num_channels), sizeof(m_num_channels));
  73. m_file->write(reinterpret_cast<u8*>(&m_sample_rate), sizeof(m_sample_rate));
  74. u32 byte_rate = m_sample_rate * m_num_channels * (m_bits_per_sample / 8);
  75. m_file->write(reinterpret_cast<u8*>(&byte_rate), sizeof(byte_rate));
  76. u16 block_align = m_num_channels * (m_bits_per_sample / 8);
  77. m_file->write(reinterpret_cast<u8*>(&block_align), sizeof(block_align));
  78. m_file->write(reinterpret_cast<u8*>(&m_bits_per_sample), sizeof(m_bits_per_sample));
  79. // "data"
  80. static u32 chunk_id = 0x61746164;
  81. m_file->write(reinterpret_cast<u8*>(&chunk_id), sizeof(chunk_id));
  82. m_file->write(reinterpret_cast<u8*>(&m_data_sz), sizeof(m_data_sz));
  83. }
  84. }