3#include "AudioTools/AudioCodecs/AudioCodecsBase.h"
5#include "AudioTools/AudioCodecs/AudioEncoded.h"
6#include "AudioTools/CoreAudio/AudioBasic/StrView.h"
8#define READ_BUFFER_SIZE 512
9#define MAX_WAV_HEADER_LEN 200
30 bool is_streamed =
true;
31 bool is_valid =
false;
32 uint32_t data_length = 0;
33 uint32_t file_size = 0;
37static const char *wav_mime =
"audio/wav";
51 int write(uint8_t *data,
size_t data_len) {
51 int write(uint8_t *data,
size_t data_len) {
…}
57 LOGI(
"WAVHeader::begin: %u", (
unsigned)buffer.
available());
61 if (!setPos(
"RIFF"))
return false;
62 headerInfo.file_size = read_int32();
63 if (!setPos(
"WAVE"))
return false;
64 if (!setPos(
"fmt "))
return false;
65 int fmt_length = read_int32();
67 headerInfo.channels = read_int16();
68 headerInfo.sample_rate = read_int32();
69 headerInfo.byte_rate = read_int32();
70 headerInfo.block_align = read_int16();
71 headerInfo.bits_per_sample = read_int16();
72 if (!setPos(
"data"))
return false;
73 headerInfo.data_length = read_int32();
74 if (!headerInfo.data_length==0 || headerInfo.data_length >= 0x7fff0000) {
75 headerInfo.is_streamed =
true;
76 headerInfo.data_length = ~0;
87 return pos > 0 && buffer.
available() >= pos;
98 return pos > 0 ? pos + 8 : 0;
111 writeRiffHeader(buffer);
113 writeDataHeader(buffer);
132 for (
int j = 0; j< buffer.
available();j++){
133 char c = (char)buffer.
data()[j];
139 LOGI(
"Header: %s", msg);
144 struct WAVAudioInfo headerInfo;
145 SingleBuffer<uint8_t> buffer{MAX_WAV_HEADER_LEN};
148 bool setPos(
const char*
id){
149 int id_len = strlen(
id);
150 int pos = indexOf(
id);
151 if (pos < 0)
return false;
152 data_pos = pos + id_len;
156 int indexOf(
const char* str){
157 return StrView((
char*)buffer.
data(),MAX_WAV_HEADER_LEN, buffer.
available()).indexOf(str);
160 uint32_t read_tag() {
162 tag = (tag << 8) | getChar();
163 tag = (tag << 8) | getChar();
164 tag = (tag << 8) | getChar();
165 tag = (tag << 8) | getChar();
169 uint32_t getChar32() {
return getChar(); }
171 uint32_t read_int32() {
173 value |= getChar32() << 0;
174 value |= getChar32() << 8;
175 value |= getChar32() << 16;
176 value |= getChar32() << 24;
180 uint16_t read_int16() {
182 value |= getChar() << 0;
183 value |= getChar() << 8;
189 for (i = 0; i < n; i++) getChar();
193 if (data_pos < buffer.size())
194 return buffer.
data()[data_pos++];
199 void seek(
long int offset,
int origin) {
200 if (origin == SEEK_SET) {
202 }
else if (origin == SEEK_CUR) {
207 size_t tell() {
return data_pos; }
209 bool eof() {
return data_pos >= buffer.size() - 1; }
212 LOGI(
"WAVHeader sound_pos: %d",
getDataPos());
213 LOGI(
"WAVHeader channels: %d ", headerInfo.
channels);
215 LOGI(
"WAVHeader sample_rate: %d ", (
int) headerInfo.
sample_rate);
216 LOGI(
"WAVHeader format: %d", (
int)headerInfo.format);
219 void writeRiffHeader(BaseBuffer<uint8_t> &buffer) {
220 buffer.writeArray((uint8_t *)
"RIFF", 4);
221 write32(buffer, headerInfo.file_size - 8);
222 buffer.writeArray((uint8_t *)
"WAVE", 4);
225 void writeFMT(BaseBuffer<uint8_t> &buffer) {
226 uint16_t fmt_len = 16;
227 buffer.writeArray((uint8_t *)
"fmt ", 4);
228 write32(buffer, fmt_len);
229 write16(buffer, (uint16_t)headerInfo.format);
230 write16(buffer, headerInfo.
channels);
232 write32(buffer, headerInfo.byte_rate);
233 write16(buffer, headerInfo.block_align);
237 void write32(BaseBuffer<uint8_t> &buffer, uint64_t value) {
238 buffer.writeArray((uint8_t *)&value, 4);
241 void write16(BaseBuffer<uint8_t> &buffer, uint16_t value) {
242 buffer.writeArray((uint8_t *)&value, 2);
245 void writeDataHeader(BaseBuffer<uint8_t> &buffer) {
246 buffer.writeArray((uint8_t *)
"data", 4);
247 write32(buffer, headerInfo.file_size);
248 int offset = headerInfo.offset;
250 uint8_t empty[offset];
251 memset(empty, 0, offset);
252 buffer.writeArray(empty, offset);
288 decoder_format = fmt;
295 bool begin()
override {
305 void end()
override {
311 const char *mime() {
return wav_mime; }
313 WAVAudioInfo &audioInfoEx() {
return header.
audioInfo(); }
317 virtual size_t write(
const uint8_t *data,
size_t len)
override {
324 if (data_start == 0)
return len;
326 result = data_start + write_out((uint8_t *)data+data_start, len-data_start);
328 }
else if (isValid) {
329 result = write_out((uint8_t *)data, len);
335 virtual operator bool()
override {
return active; }
343 AudioDecoderExt *p_decoder =
nullptr;
344 EncodedAudioOutput dec_out;
345 SingleBuffer<uint8_t> buffer24;
348 return p_decoder==
nullptr ? *p_print : dec_out;
351 virtual size_t write_out(
const uint8_t *in_ptr,
size_t in_size) {
354 if (header.audioInfo().format == AudioFormat::PCM
355 && header.audioInfo().bits_per_sample == 24 &&
sizeof(int24_t)==4){
356 write_out_24(in_ptr, in_size);
359 result = out().write(in_ptr, in_size);
365 size_t write_out_24(
const uint8_t *in_ptr,
size_t in_size) {
367 AudioInfo& info = header.audioInfo();
369 buffer24.resize(info.channels*3);
371 int32_t frame[info.channels];
372 uint8_t val24[3]={0};
375 for (
int j=0;j<in_size;j++){
376 buffer24.write(in_ptr[j]);
378 if (buffer24.availableForWrite()==0){
379 for (
int ch=0;ch<info.channels;ch++){
380 buffer24.readArray((uint8_t*)&val24[0], 3);
381 frame[ch] = interpret24bitAsInt32(val24);
384 assert(buffer24.available()==0);
386 size_t written = out().write((uint8_t*)frame,
sizeof(frame));
387 assert(written==
sizeof(frame));
394 int32_t interpret24bitAsInt32(uint8_t* byteArray) {
396 (
static_cast<int32_t
>(byteArray[2]) << 24)
397 | (
static_cast<int32_t
>(byteArray[1]) << 16)
398 | (
static_cast<int32_t
>(byteArray[0]) << 8)
404 int result = in_size;
406 int written = header.write(in_ptr, in_size);
407 if (!header.isDataComplete()) {
408 LOGW(
"WAV header misses 'data' section in len: %d", (
int) header.available());
413 if (!header.parse()){
414 LOGE(
"WAV header parsing failed");
419 isValid = header.audioInfo().is_valid;
421 LOGI(
"WAV sample_rate: %d", (
int) header.audioInfo().sample_rate);
422 LOGI(
"WAV data_length: %u", (
unsigned)header.audioInfo().data_length);
423 LOGI(
"WAV is_streamed: %d", header.audioInfo().is_streamed);
424 LOGI(
"WAV is_valid: %s",
425 header.audioInfo().is_valid ?
"true" :
"false");
429 isValid = format == decoder_format;
432 if(p_decoder!=
nullptr){
433 int block_size = header.audioInfo().block_align;
434 p_decoder->setBlockSize(block_size);
440 bi.
channels = header.audioInfo().channels;
442 notifyAudioChange(bi);
444 LOGE(
"WAV format not supported: %d", (
int)format);
446 return header.getDataPos();
449 void setupEncodedAudio() {
450 if (p_decoder!=
nullptr){
451 assert(p_print!=
nullptr);
452 dec_out.setOutput(p_print);
453 dec_out.setDecoder(p_decoder);
479 setEncoder(enc, fmt);
484 audioInfo.format = fmt;
495 const char *
mime()
override {
return wav_mime; }
500 info.format = AudioFormat::PCM;
504 info.is_streamed =
true;
505 info.is_valid =
true;
506 info.data_length = 0x7fff0000;
507 info.file_size = info.data_length + 36;
525 LOGI(
"sample_rate: %d", (
int)audioInfo.
sample_rate);
526 LOGI(
"channels: %d", audioInfo.
channels);
529 if (audioInfo.format == AudioFormat::PCM){
532 if (audioInfo.is_streamed || audioInfo.data_length == 0 ||
533 audioInfo.data_length >= 0x7fff0000) {
534 LOGI(
"is_streamed! because length is %u",
535 (
unsigned)audioInfo.data_length);
536 audioInfo.is_streamed =
true;
537 audioInfo.data_length = ~0;
539 size_limit = audioInfo.data_length;
540 LOGI(
"size_limit is %d", (
int)size_limit);
555 header_written =
false;
561 void end()
override { is_open =
false; }
564 virtual size_t write(
const uint8_t *data,
size_t len)
override {
566 LOGE(
"The WAVEncoder is not open - please call begin()");
570 if (p_print ==
nullptr) {
571 LOGE(
"No output stream was provided");
575 if (!header_written) {
576 LOGI(
"Writing Header");
577 header.setAudioInfo(audioInfo);
578 int len = header.writeHeader(p_print);
579 audioInfo.file_size -= len;
580 header_written =
true;
584 Print *p_out = p_encoder==
nullptr ? p_print : &enc_out;;
585 if (audioInfo.is_streamed) {
586 result = p_out->write((uint8_t *)data, len);
587 }
else if (size_limit > 0) {
588 size_t write_size = min((
size_t)len, (
size_t)size_limit);
589 result = p_out->write((uint8_t *)data, write_size);
590 size_limit -= result;
592 if (size_limit <= 0) {
593 LOGI(
"The defined size was written - so we close the WAVEncoder now");
564 virtual size_t write(
const uint8_t *data,
size_t len)
override {
…}
600 operator bool()
override {
return is_open; }
602 bool isOpen() {
return is_open; }
609 Print *p_print =
nullptr;
613 int64_t size_limit = 0;
614 bool header_written =
false;
615 volatile bool is_open =
false;
617 void setupEncodedAudio() {
618 if (p_encoder!=
nullptr){
619 assert(p_print!=
nullptr);
621 enc_out.setEncoder(p_encoder);
625 audioInfo.block_align = p_encoder->blockSize();
AudioFormat
Audio format codes used by Microsoft e.g. in avi or wav files.
Definition AudioFormat.h:19