26 using id_t = uint32_t;
31 using length_t = uint32_t;
36 constexpr size_t align = 2;
47 dat << static_cast<uint32_t>(
data.type);
55 if (!dat.ok()) _Unlikely_
goto error1;
103 dat << static_cast<uint16_t>(
data.compression);
104 dat <<
data.num_channels;
105 dat <<
data.sample_rate;
106 dat <<
data.bytes_per_second;
107 dat <<
data.block_align;
108 dat <<
data.bits_per_channel;
110 if (
auto size =
data.extra.size(); size) {
111 if (size > UINT16_MAX) _Unlikely_
112 throw std::invalid_argument(
"extra data too big");
113 dat << static_cast<uint16_t>(size);
115 dat.write(
data.extra.data(),
static_cast<size_t>(size));
126 if (!dat.ok()) _Unlikely_
goto error1;
128 dat >>
data.num_channels;
129 dat >>
data.sample_rate;
130 dat >>
data.bytes_per_second;
131 dat >>
data.block_align;
132 dat >>
data.bits_per_channel;
134 if (!dat.ok() || !tmp16)
goto error7;
136 data.extra.write_stream(dat, tmp16);
137 data.extra.truncate();
143 data.num_channels = 0;
144 data.sample_rate = 0;
145 data.bytes_per_second = 0;
146 data.block_align = 0;
147 data.bits_per_channel = 0;
150 data.extra.truncate();
166 if (!dat.ok()) _Unlikely_
return dat;
191 dat <<
data.num_samples;
197 dat >>
data.num_samples;
217 inline int compare_by_id(_In_
const cue& a, _In_
const cue& b)
219 if (a.id < b.id)
return -1;
220 if (a.id > b.id)
return 1;
224 inline int compare_by_pos(_In_
const cue& a, _In_
const cue& b)
226 if (a.position < b.position)
return -1;
227 if (a.position > b.position)
return 1;
234 using cue_vector = std::vector<cue>;
239 size_t num_cues = data.size();
240 if (num_cues > UINT32_MAX) _Unlikely_
241 throw std::invalid_argument(
"too many cues");
242 dat << static_cast<uint32_t>(num_cues);
244 dat.write_array(data.data(),
sizeof(cue), num_cues);
252 if (!dat.ok()) _Unlikely_
goto error1;
253 data.resize(num_cues);
254 data.resize(dat.read_array(data.data(),
sizeof(cue), num_cues));
279 dat <<
data.duration;
280 dat <<
data.purpose_id;
282 dat <<
data.language;
285 if (
size_t num_chars =
data.description.size(); num_chars && dat.ok()) {
286 dat.write_array(
data.description.data(),
sizeof(
char), num_chars);
295 dat >>
data.duration;
296 dat >>
data.purpose_id;
298 dat >>
data.language;
302 auto tmp = dat.read_remainder();
303 data.description.assign(
304 reinterpret_cast<const char*
>(tmp.data()),
305 stdex::strnlen(
reinterpret_cast<const char*
>(tmp.data()), tmp.size()));
308 data.description.clear();
326 if (
size_t num_chars =
data.title.size(); num_chars && dat.ok()) {
327 dat.write_array(
data.title.data(),
sizeof(
char), num_chars);
337 auto tmp = dat.read_remainder();
339 reinterpret_cast<const char*
>(tmp.data()),
340 stdex::strnlen(
reinterpret_cast<const char*
>(tmp.data()), tmp.size()));
361 if (
size_t num_chars =
data.note.size(); num_chars && dat.ok()) {
362 dat.write_array(
data.note.data(),
sizeof(
char), num_chars);
372 auto tmp = dat.read_remainder();
374 reinterpret_cast<const char*
>(tmp.data()),
375 stdex::strnlen(
reinterpret_cast<const char*
>(tmp.data()), tmp.size()));
412 using cue_ex_vector = std::vector<cue_ex>;
417 size_t num_cues =
data.size();
418 if (num_cues > UINT32_MAX) _Unlikely_
419 throw std::invalid_argument(
"too many cues");
420 dat << static_cast<uint32_t>(num_cues);
421 for (
size_t i = 0; i < num_cues && dat.ok(); ++i)
422 dat.write_array(
static_cast<const cue*
>(&
data[i]),
sizeof(
cue), 1);
423 stdex::idrec::close<id_t, length_t, align>(dat, start);
426 dat << *reinterpret_cast<const id_t*>(
"adtl");
427 for (
size_t i = 0; i < num_cues && dat.ok(); ++i) {
439 if (!c.
title.empty()) {
442 title.title = c.
title;
445 if (!c.
note.empty()) {
452 stdex::idrec::close<id_t, length_t, align>(dat, start);
458 _Success_(
return!=0) bool find_first(
459 _In_ stdex::stream::basic_file& dat,
461 _In_ stdex::stream::fpos_t block_end = stdex::stream::fpos_max,
462 _Out_opt_ stdex::stream::fpos_t* found_block_end =
nullptr)
464 while (dat.tell() < block_end) {
465 if (!stdex::idrec::record<T, id_t, T::record::id(), length_t, align>::find(dat, block_end))
470 if (!dat.ok()) _Unlikely_
472 stdex::stream::fpos_t end = dat.tell() + size;
475 if (!dat.ok()) _Unlikely_
478 if (found_block_end) *found_block_end = end;
483 end += (align - end) % align;
490 _Success_(
return != 0) bool read_first(
491 _In_ stdex::stream::basic_file& dat,
493 _In_ stdex::stream::fpos_t block_end = stdex::stream::fpos_max)
501 inline _Success_(
return != 0) bool find_content(
502 _In_ stdex::stream::basic_file& dat,
503 _In_ stdex::stream::fpos_t block_end = stdex::stream::fpos_max,
504 _Out_opt_ stdex::stream::fpos_t* found_block_end =
nullptr)
506 return find_first<header>(dat, wave::record::id(), block_end, found_block_end);
509 inline _Success_(
return != 0) bool read_cues(
510 _In_ stdex::stream::basic_file& dat,
511 _Inout_ cue_ex_vector& cues, stdex::stream::fpos_t block_end = stdex::stream::fpos_max)
513 static auto cue_less = [](_In_
const cue& a, _In_
const cue& b) {
return compare_by_id(a, b) < 0; };
514 static int (__cdecl* cue_cmp)(
void const*,
void const*) = [](_In_
void const* a, _In_
void const* b) {
return compare_by_id(*
reinterpret_cast<const cue*
>(a), *
reinterpret_cast<const cue*
>(b)); };
516 stdex::stream::fpos_t start = dat.tell();
517 while (dat.tell() < block_end) {
518 if (
stdex::idrec::record<cue_vector, id_t, cue_vector_record::id(), length_t, align>::find(dat, block_end)) {
521 if (!dat.ok()) _Unlikely_
return false;
525 if (!_dat.ok()) _Unlikely_
return false;
526 cues.resize(num_cues);
529 for (i = 0; i < num_cues; i++) {
530 _dat.read_array(
static_cast<cue*
>(&cues[i]),
sizeof(cue), 1);
531 if (!_dat.ok()) _Unlikely_
533 if (i && cue_less(cues[i], cues[i - 1]))
538 std::sort(cues.begin(), cues.end(), cue_less);
544 while (dat.tell() < block_end) {
545 stdex::stream::fpos_t found_block_end;
546 if (find_first<list>(dat, *(
const id_t*)
"adtl", block_end, &found_block_end)) {
547 while (dat.tell() < found_block_end) {
550 if (!dat.ok())
break;
552 if (
id == ltxt::record::id()) {
554 dat >> ltxt::record(ltxt);
557 cue_ex* c =
reinterpret_cast<cue_ex*
>(std::bsearch(&tmp, cues.data(), cues.size(),
sizeof(cues[0]), cue_cmp));
559 c->duration = ltxt.duration;
560 c->purpose_id = ltxt.purpose_id;
561 c->country = ltxt.country;
562 c->language = ltxt.language;
563 c->dialect = ltxt.dialect;
564 c->charset = ltxt.charset;
565 c->description = ltxt.description;
568 else if (
id == label::record::id()) {
570 dat >> label::record(title);
573 cue_ex* c =
reinterpret_cast<cue_ex*
>(std::bsearch(&tmp, cues.data(), cues.size(),
sizeof(cues[0]), cue_cmp));
575 c->title = title.title;
577 else if (
id == note::record::id()) {
579 dat >> note::record(note);
582 cue_ex* c =
reinterpret_cast<cue_ex*
>(std::bsearch(&tmp, cues.data(), cues.size(),
sizeof(cues[0]), cue_cmp));
586 else if (!stdex::idrec::ignore<length_t, align>(dat)) _Unlikely_
596 inline _Success_(
return != 0) bool remove_cues(
597 _Inout_ stdex::stream::basic_file& input,
598 _Inout_ stdex::stream::basic_file& output,
599 _In_ stdex::stream::fpos_t block_end = stdex::stream::fpos_max)
601 static id_t removable[] = {
602 cue_vector_record::id(),
608 while (input.tell() < block_end) {
611 if (!input.ok()) break;
612 for (size_t i = 0; i < _countof(removable); i++)
613 if (id == removable[i])
615 if (!stdex::idrec::ignore<length_t, align>(input)) _Unlikely_
622 if (!input.ok()) _Unlikely_ return false;
623 auto start = stdex::idrec::open<id_t, length_t>(output, id);
625 stdex::strupr(reinterpret_cast<char*>(&id2), sizeof(id_t) / sizeof(char));
626 if (id != id2 || id == *reinterpret_cast<const id_t*>(
"ICRD")) {
628 if (!output.ok()) _Unlikely_ return false;
629 output.write_stream(input, size);
630 if (!input.ok()) _Unlikely_ return false;
631 stdex::idrec::close<id_t, length_t, align>(output, start);
632 size = static_cast<length_t>(align - size) % align;
640 if (!input.ok()) _Unlikely_ return false;
642 if (!output.ok()) _Unlikely_ return false;
643 stdex::stream::fpos_t end = output.tell();
644 if (!remove_cues(input, output, input.tell() + size - sizeof(id_t))) _Unlikely_
646 if (end != output.tell())
647 stdex::idrec::close<id_t, length_t, align>(output, start);
649 output.seekbeg(start);
Helper class for read/write of records to/from memory.
Definition idrec.hpp:291
static constexpr T_id id()
Returns record id.
Definition idrec.hpp:310
Basic seekable stream operations.
Definition stream.hpp:815
Basic stream operations.
Definition stream.hpp:85
Limits file reading/writing to a predefined window.
Definition stream.hpp:1692
In-memory file.
Definition stream.hpp:3219
virtual fsize_t size() const
Returns file size Should the file size cannot be determined, the method returns fsize_max and it does...
Definition stream.hpp:3921
size_t write_stream(basic &stream, size_t amount=SIZE_MAX)
Writes content of another stream.
Definition stream.hpp:3836
virtual fpos_t seek(foff_t offset, seek_t how=seek_t::beg)
Seeks to specified relative file position.
Definition stream.hpp:3900
virtual void truncate()
Sets file size - truncates the remainder of file content from the current file position to the end of...
Definition stream.hpp:3926
const void * data() const
Returns pointer to data.
Definition stream.hpp:3549
Extended cue.
Definition wav.hpp:397
uint32_t duration
How many samples from the cue point the region or section spans.
Definition wav.hpp:398
std::string note
Note text.
Definition wav.hpp:406
std::string description
Description text.
Definition wav.hpp:404
std::string title
Title text.
Definition wav.hpp:405
uint16_t country
Country code used by texts.
Definition wav.hpp:400
uint16_t charset
Charset used by texts.
Definition wav.hpp:403
id_t purpose_id
What the text is used for. For example a value of "scrp" means script text, and "capt" means close-ca...
Definition wav.hpp:399
uint16_t language
Language code used by texts.
Definition wav.hpp:401
uint16_t dialect
Dialect code used by texts.
Definition wav.hpp:402
Cue point.
Definition wav.hpp:208
uint32_t chunk_id
The four byte ID used by the chunk containing the sample that corresponds to this cue point....
Definition wav.hpp:211
uint32_t id
Each cue point has a unique identification value used to associate cue points with information in oth...
Definition wav.hpp:209
uint32_t block_start
The byte offset into the "data" or "slnt" Chunk to the start of the block containing the sample....
Definition wav.hpp:213
uint32_t position
The sample offset associated with the cue point in terms of the sample's position in the final stream...
Definition wav.hpp:210
uint32_t chunk_offset
The byte offset into the Wave List Chunk of the chunk containing the sample that corresponds to this ...
Definition wav.hpp:212
uint32_t block_offset
An offset into the block (specified by Block Start) for the sample that corresponds to the cue point....
Definition wav.hpp:214
Encoded waveform content.
Definition wav.hpp:161
stdex::stream::memory_file content
Encoded waveform.
Definition wav.hpp:162
Label.
Definition wav.hpp:319
std::string title
Title text.
Definition wav.hpp:321
Associated data list.
Definition wav.hpp:389
Labeled text.
Definition wav.hpp:266
uint16_t charset
Charset used by text.
Definition wav.hpp:273
uint32_t id
The starting sample point that corresponds to this text label by providing the ID of a Cue Point defi...
Definition wav.hpp:267
uint16_t language
Language code used by text.
Definition wav.hpp:271
uint16_t country
Country code used by text.
Definition wav.hpp:270
std::string description
Description text.
Definition wav.hpp:274
id_t purpose_id
What the text is used for. For example a value of "scrp" means script text, and "capt" means close-ca...
Definition wav.hpp:269
uint32_t duration
How many samples from the cue point the region or section spans.
Definition wav.hpp:268
uint16_t dialect
Dialect code used by text.
Definition wav.hpp:272
Note.
Definition wav.hpp:354
std::string note
Note text.
Definition wav.hpp:356
uint32_t id
The sample point that corresponds to this text comment by providing the ID of a Cue Point defined in ...
Definition wav.hpp:355
Silence.
Definition wav.hpp:186
uint32_t num_samples
The number of silent samples that appear in the waveform at this point in the wave list chunk.
Definition wav.hpp:187
Waveform block.
Definition wav.hpp:71