/* * 2-channel UHJ Encoder * * Copyright (c) Chris Robinson * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ #include "config.h" #include #include #include #include #include #include #include #include #include #include #include "alnumbers.h" #include "alspan.h" #include "fmt/core.h" #include "phase_shifter.h" #include "vector.h" #include "sndfile.h" #include "win_main_utf8.h" namespace { using namespace std::string_view_literals; struct SndFileDeleter { void operator()(SNDFILE *sndfile) { sf_close(sndfile); } }; using SndFilePtr = std::unique_ptr; using uint = unsigned int; constexpr uint BufferLineSize{1024}; using FloatBufferLine = std::array; using FloatBufferSpan = al::span; struct UhjEncoder { constexpr static size_t sFilterDelay{1024}; /* Delays and processing storage for the unfiltered signal. */ alignas(16) std::array mW{}; alignas(16) std::array mX{}; alignas(16) std::array mY{}; alignas(16) std::array mZ{}; alignas(16) std::array mS{}; alignas(16) std::array mD{}; alignas(16) std::array mT{}; /* History for the FIR filter. */ alignas(16) std::array mWXHistory1{}; alignas(16) std::array mWXHistory2{}; alignas(16) std::array mTemp{}; void encode(const al::span OutSamples, const al::span InSamples, const size_t SamplesToDo); }; const PhaseShifterT PShift{}; /* Encoding UHJ from B-Format is done as: * * S = 0.9396926*W + 0.1855740*X * D = j(-0.3420201*W + 0.5098604*X) + 0.6554516*Y * * Left = (S + D)/2.0 * Right = (S - D)/2.0 * T = j(-0.1432*W + 0.6512*X) - 0.7071068*Y * Q = 0.9772*Z * * where j is a wide-band +90 degree phase shift. T is excluded from 2-channel * output, and Q is excluded from 2- and 3-channel output. */ void UhjEncoder::encode(const al::span OutSamples, const al::span InSamples, const size_t SamplesToDo) { const auto winput = al::span{InSamples[0]}.first(SamplesToDo); const auto xinput = al::span{InSamples[1]}.first(SamplesToDo); const auto yinput = al::span{InSamples[2]}.first(SamplesToDo); const auto zinput = al::span{InSamples[3]}.first(SamplesToDo); /* Combine the previously delayed input signal with the new input. */ std::copy(winput.begin(), winput.end(), mW.begin()+sFilterDelay); std::copy(xinput.begin(), xinput.end(), mX.begin()+sFilterDelay); std::copy(yinput.begin(), yinput.end(), mY.begin()+sFilterDelay); std::copy(zinput.begin(), zinput.end(), mZ.begin()+sFilterDelay); /* S = 0.9396926*W + 0.1855740*X */ for(size_t i{0};i < SamplesToDo;++i) mS[i] = 0.9396926f*mW[i] + 0.1855740f*mX[i]; /* Precompute j(-0.3420201*W + 0.5098604*X) and store in mD. */ auto tmpiter = std::copy(mWXHistory1.cbegin(), mWXHistory1.cend(), mTemp.begin()); std::transform(winput.begin(), winput.end(), xinput.begin(), tmpiter, [](const float w, const float x) noexcept -> float { return -0.3420201f*w + 0.5098604f*x; }); std::copy_n(mTemp.cbegin()+SamplesToDo, mWXHistory1.size(), mWXHistory1.begin()); PShift.process(al::span{mD}.first(SamplesToDo), mTemp); /* D = j(-0.3420201*W + 0.5098604*X) + 0.6554516*Y */ for(size_t i{0};i < SamplesToDo;++i) mD[i] = mD[i] + 0.6554516f*mY[i]; /* Left = (S + D)/2.0 */ auto left = al::span{OutSamples[0]}; for(size_t i{0};i < SamplesToDo;i++) left[i] = (mS[i] + mD[i]) * 0.5f; /* Right = (S - D)/2.0 */ auto right = al::span{OutSamples[1]}; for(size_t i{0};i < SamplesToDo;i++) right[i] = (mS[i] - mD[i]) * 0.5f; if(OutSamples.size() > 2) { /* Precompute j(-0.1432*W + 0.6512*X) and store in mT. */ tmpiter = std::copy(mWXHistory2.cbegin(), mWXHistory2.cend(), mTemp.begin()); std::transform(winput.begin(), winput.end(), xinput.begin(), tmpiter, [](const float w, const float x) noexcept -> float { return -0.1432f*w + 0.6512f*x; }); std::copy_n(mTemp.cbegin()+SamplesToDo, mWXHistory2.size(), mWXHistory2.begin()); PShift.process(al::span{mT}.first(SamplesToDo), mTemp); /* T = j(-0.1432*W + 0.6512*X) - 0.7071068*Y */ auto t = al::span{OutSamples[2]}; for(size_t i{0};i < SamplesToDo;i++) t[i] = mT[i] - 0.7071068f*mY[i]; } if(OutSamples.size() > 3) { /* Q = 0.9772*Z */ auto q = al::span{OutSamples[3]}; for(size_t i{0};i < SamplesToDo;i++) q[i] = 0.9772f*mZ[i]; } /* Copy the future samples to the front for next time. */ std::copy(mW.cbegin()+SamplesToDo, mW.cbegin()+SamplesToDo+sFilterDelay, mW.begin()); std::copy(mX.cbegin()+SamplesToDo, mX.cbegin()+SamplesToDo+sFilterDelay, mX.begin()); std::copy(mY.cbegin()+SamplesToDo, mY.cbegin()+SamplesToDo+sFilterDelay, mY.begin()); std::copy(mZ.cbegin()+SamplesToDo, mZ.cbegin()+SamplesToDo+sFilterDelay, mZ.begin()); } struct SpeakerPos { int mChannelID; float mAzimuth; float mElevation; }; /* Azimuth is counter-clockwise. */ constexpr std::array MonoMap{ SpeakerPos{SF_CHANNEL_MAP_CENTER, 0.0f, 0.0f}, }; constexpr std::array StereoMap{ SpeakerPos{SF_CHANNEL_MAP_LEFT, 30.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_RIGHT, -30.0f, 0.0f}, }; constexpr std::array QuadMap{ SpeakerPos{SF_CHANNEL_MAP_LEFT, 45.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_RIGHT, -45.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_REAR_LEFT, 135.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_REAR_RIGHT, -135.0f, 0.0f}, }; constexpr std::array X51Map{ SpeakerPos{SF_CHANNEL_MAP_LEFT, 30.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_RIGHT, -30.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_CENTER, 0.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_LFE, 0.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_SIDE_LEFT, 110.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_SIDE_RIGHT, -110.0f, 0.0f}, }; constexpr std::array X51RearMap{ SpeakerPos{SF_CHANNEL_MAP_LEFT, 30.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_RIGHT, -30.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_CENTER, 0.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_LFE, 0.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_REAR_LEFT, 110.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_REAR_RIGHT, -110.0f, 0.0f}, }; constexpr std::array X71Map{ SpeakerPos{SF_CHANNEL_MAP_LEFT, 30.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_RIGHT, -30.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_CENTER, 0.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_LFE, 0.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_REAR_LEFT, 150.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_REAR_RIGHT, -150.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_SIDE_LEFT, 90.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_SIDE_RIGHT, -90.0f, 0.0f}, }; constexpr std::array X714Map{ SpeakerPos{SF_CHANNEL_MAP_LEFT, 30.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_RIGHT, -30.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_CENTER, 0.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_LFE, 0.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_REAR_LEFT, 150.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_REAR_RIGHT, -150.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_SIDE_LEFT, 90.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_SIDE_RIGHT, -90.0f, 0.0f}, SpeakerPos{SF_CHANNEL_MAP_TOP_FRONT_LEFT, 45.0f, 35.0f}, SpeakerPos{SF_CHANNEL_MAP_TOP_FRONT_RIGHT, -45.0f, 35.0f}, SpeakerPos{SF_CHANNEL_MAP_TOP_REAR_LEFT, 135.0f, 35.0f}, SpeakerPos{SF_CHANNEL_MAP_TOP_REAR_RIGHT, -135.0f, 35.0f}, }; constexpr auto GenCoeffs(double x /*+front*/, double y /*+left*/, double z /*+up*/) noexcept { /* Coefficients are +3dB of FuMa. */ return std::array{{ 1.0f, static_cast(al::numbers::sqrt2 * x), static_cast(al::numbers::sqrt2 * y), static_cast(al::numbers::sqrt2 * z) }}; } int main(al::span args) { if(args.size() < 2 || args[1] == "-h" || args[1] == "--help") { fmt::println("Usage: {} <[options] infile...>\n\n" " Options:\n" " -bhj Encode 2-channel UHJ, aka \"BJH\" (default).\n" " -thj Encode 3-channel UHJ, aka \"TJH\".\n" " -phj Encode 4-channel UHJ, aka \"PJH\".\n" "\n" "3-channel UHJ supplements 2-channel UHJ with an extra channel that allows full\n" "reconstruction of first-order 2D ambisonics. 4-channel UHJ supplements 3-channel\n" "UHJ with an extra channel carrying height information, providing for full\n" "reconstruction of first-order 3D ambisonics.\n" "\n" "Note: The third and fourth channels should be ignored if they're not being\n" "decoded. Unlike the first two channels, they are not designed for undecoded\n" "playback, so the resulting files will not play correctly if this isn't handled.", args[0]); return 1; } args = args.subspan(1); uint uhjchans{2}; size_t num_files{0}, num_encoded{0}; auto process_arg = [&uhjchans,&num_files,&num_encoded](std::string_view arg) -> void { if(arg == "-bhj"sv) { uhjchans = 2; return; } if(arg == "-thj"sv) { uhjchans = 3; return; } if(arg == "-phj"sv) { uhjchans = 4; return; } ++num_files; auto outname = std::string{arg}; const auto lastslash = outname.rfind('/'); if(lastslash != std::string::npos) outname.erase(0, lastslash+1); const auto extpos = outname.rfind('.'); if(extpos != std::string::npos) outname.resize(extpos); outname += ".uhj.flac"; SF_INFO ininfo{}; SndFilePtr infile{sf_open(std::string{arg}.c_str(), SFM_READ, &ininfo)}; if(!infile) { fmt::println(stderr, "Failed to open {}", arg); return; } fmt::println("Converting {} to {}...", arg, outname); /* Work out the channel map, preferably using the actual channel map * from the file/format, but falling back to assuming WFX order. */ al::span spkrs; auto chanmap = std::vector(static_cast(ininfo.channels), SF_CHANNEL_MAP_INVALID); if(sf_command(infile.get(), SFC_GET_CHANNEL_MAP_INFO, chanmap.data(), ininfo.channels*int{sizeof(int)}) == SF_TRUE) { static const std::array monomap{{SF_CHANNEL_MAP_CENTER}}; static const std::array stereomap{{SF_CHANNEL_MAP_LEFT, SF_CHANNEL_MAP_RIGHT}}; static const std::array quadmap{{SF_CHANNEL_MAP_LEFT, SF_CHANNEL_MAP_RIGHT, SF_CHANNEL_MAP_REAR_LEFT, SF_CHANNEL_MAP_REAR_RIGHT}}; static const std::array x51map{{SF_CHANNEL_MAP_LEFT, SF_CHANNEL_MAP_RIGHT, SF_CHANNEL_MAP_CENTER, SF_CHANNEL_MAP_LFE, SF_CHANNEL_MAP_SIDE_LEFT, SF_CHANNEL_MAP_SIDE_RIGHT}}; static const std::array x51rearmap{{SF_CHANNEL_MAP_LEFT, SF_CHANNEL_MAP_RIGHT, SF_CHANNEL_MAP_CENTER, SF_CHANNEL_MAP_LFE, SF_CHANNEL_MAP_REAR_LEFT, SF_CHANNEL_MAP_REAR_RIGHT}}; static const std::array x71map{{SF_CHANNEL_MAP_LEFT, SF_CHANNEL_MAP_RIGHT, SF_CHANNEL_MAP_CENTER, SF_CHANNEL_MAP_LFE, SF_CHANNEL_MAP_REAR_LEFT, SF_CHANNEL_MAP_REAR_RIGHT, SF_CHANNEL_MAP_SIDE_LEFT, SF_CHANNEL_MAP_SIDE_RIGHT}}; static const std::array x714map{{SF_CHANNEL_MAP_LEFT, SF_CHANNEL_MAP_RIGHT, SF_CHANNEL_MAP_CENTER, SF_CHANNEL_MAP_LFE, SF_CHANNEL_MAP_REAR_LEFT, SF_CHANNEL_MAP_REAR_RIGHT, SF_CHANNEL_MAP_SIDE_LEFT, SF_CHANNEL_MAP_SIDE_RIGHT, SF_CHANNEL_MAP_TOP_FRONT_LEFT, SF_CHANNEL_MAP_TOP_FRONT_RIGHT, SF_CHANNEL_MAP_TOP_REAR_LEFT, SF_CHANNEL_MAP_TOP_REAR_RIGHT}}; static const std::array ambi2dmap{{SF_CHANNEL_MAP_AMBISONIC_B_W, SF_CHANNEL_MAP_AMBISONIC_B_X, SF_CHANNEL_MAP_AMBISONIC_B_Y}}; static const std::array ambi3dmap{{SF_CHANNEL_MAP_AMBISONIC_B_W, SF_CHANNEL_MAP_AMBISONIC_B_X, SF_CHANNEL_MAP_AMBISONIC_B_Y, SF_CHANNEL_MAP_AMBISONIC_B_Z}}; auto match_chanmap = [](const al::span a, const al::span b) -> bool { if(a.size() != b.size()) return false; auto find_channel = [b](const int id) -> bool { return std::find(b.begin(), b.end(), id) != b.end(); }; return std::all_of(a.cbegin(), a.cend(), find_channel); }; if(match_chanmap(chanmap, monomap)) spkrs = MonoMap; else if(match_chanmap(chanmap, stereomap)) spkrs = StereoMap; else if(match_chanmap(chanmap, quadmap)) spkrs = QuadMap; else if(match_chanmap(chanmap, x51map)) spkrs = X51Map; else if(match_chanmap(chanmap, x51rearmap)) spkrs = X51RearMap; else if(match_chanmap(chanmap, x71map)) spkrs = X71Map; else if(match_chanmap(chanmap, x714map)) spkrs = X714Map; else if(match_chanmap(chanmap, ambi2dmap) || match_chanmap(chanmap, ambi3dmap)) { /* Do nothing. */ } else { std::string mapstr; if(!chanmap.empty()) { mapstr = std::to_string(chanmap[0]); for(int idx : al::span{chanmap}.subspan<1>()) { mapstr += ','; mapstr += std::to_string(idx); } } fmt::println(stderr, " ... {} channels not supported (map: {})", chanmap.size(), mapstr); return; } } else if(sf_command(infile.get(), SFC_WAVEX_GET_AMBISONIC, nullptr, 0) == SF_AMBISONIC_B_FORMAT) { if(ininfo.channels == 4) { fmt::println(stderr, " ... detected FuMa 3D B-Format"); chanmap[0] = SF_CHANNEL_MAP_AMBISONIC_B_W; chanmap[1] = SF_CHANNEL_MAP_AMBISONIC_B_X; chanmap[2] = SF_CHANNEL_MAP_AMBISONIC_B_Y; chanmap[3] = SF_CHANNEL_MAP_AMBISONIC_B_Z; } else if(ininfo.channels == 3) { fmt::println(stderr, " ... detected FuMa 2D B-Format"); chanmap[0] = SF_CHANNEL_MAP_AMBISONIC_B_W; chanmap[1] = SF_CHANNEL_MAP_AMBISONIC_B_X; chanmap[2] = SF_CHANNEL_MAP_AMBISONIC_B_Y; } else { fmt::println(stderr, " ... unhandled {}-channel B-Format", ininfo.channels); return; } } else if(ininfo.channels == 1) { fmt::println(stderr, " ... assuming front-center"); spkrs = MonoMap; chanmap[0] = SF_CHANNEL_MAP_CENTER; } else if(ininfo.channels == 2) { fmt::println(stderr, " ... assuming WFX order stereo"); spkrs = StereoMap; chanmap[0] = SF_CHANNEL_MAP_LEFT; chanmap[1] = SF_CHANNEL_MAP_RIGHT; } else if(ininfo.channels == 6) { fmt::println(stderr, " ... assuming WFX order 5.1"); spkrs = X51Map; chanmap[0] = SF_CHANNEL_MAP_LEFT; chanmap[1] = SF_CHANNEL_MAP_RIGHT; chanmap[2] = SF_CHANNEL_MAP_CENTER; chanmap[3] = SF_CHANNEL_MAP_LFE; chanmap[4] = SF_CHANNEL_MAP_SIDE_LEFT; chanmap[5] = SF_CHANNEL_MAP_SIDE_RIGHT; } else if(ininfo.channels == 8) { fmt::println(stderr, " ... assuming WFX order 7.1"); spkrs = X71Map; chanmap[0] = SF_CHANNEL_MAP_LEFT; chanmap[1] = SF_CHANNEL_MAP_RIGHT; chanmap[2] = SF_CHANNEL_MAP_CENTER; chanmap[3] = SF_CHANNEL_MAP_LFE; chanmap[4] = SF_CHANNEL_MAP_REAR_LEFT; chanmap[5] = SF_CHANNEL_MAP_REAR_RIGHT; chanmap[6] = SF_CHANNEL_MAP_SIDE_LEFT; chanmap[7] = SF_CHANNEL_MAP_SIDE_RIGHT; } else { fmt::println(stderr, " ... unmapped {}-channel audio not supported", ininfo.channels); return; } SF_INFO outinfo{}; outinfo.frames = ininfo.frames; outinfo.samplerate = ininfo.samplerate; outinfo.channels = static_cast(uhjchans); outinfo.format = SF_FORMAT_PCM_24 | SF_FORMAT_FLAC; SndFilePtr outfile{sf_open(outname.c_str(), SFM_WRITE, &outinfo)}; if(!outfile) { fmt::println(stderr, " ... failed to create {}", outname); return; } auto encoder = std::make_unique(); auto splbuf = al::vector(9); auto ambmem = al::span{splbuf}.subspan<0,4>(); auto encmem = al::span{splbuf}.subspan<4,4>(); auto srcmem = al::span{splbuf[8]}; auto membuf = al::vector((static_cast(ininfo.channels)+size_t{uhjchans}) * BufferLineSize); auto outmem = al::span{membuf}.first(size_t{BufferLineSize}*uhjchans); auto inmem = al::span{membuf}.last(size_t{BufferLineSize} * static_cast(ininfo.channels)); /* A number of initial samples need to be skipped to cut the lead-in * from the all-pass filter delay. The same number of samples need to * be fed through the encoder after reaching the end of the input file * to ensure none of the original input is lost. */ size_t total_wrote{0}; size_t LeadIn{UhjEncoder::sFilterDelay}; sf_count_t LeadOut{UhjEncoder::sFilterDelay}; while(LeadIn > 0 || LeadOut > 0) { auto sgot = sf_readf_float(infile.get(), inmem.data(), BufferLineSize); sgot = std::max(sgot, 0); if(sgot < BufferLineSize) { const sf_count_t remaining{std::min(BufferLineSize - sgot, LeadOut)}; std::fill_n(inmem.begin() + sgot*ininfo.channels, remaining*ininfo.channels, 0.0f); sgot += remaining; LeadOut -= remaining; } for(auto&& buf : ambmem) buf.fill(0.0f); auto got = static_cast(sgot); if(spkrs.empty()) { /* B-Format is already in the correct order. It just needs a * +3dB boost. */ static constexpr float scale{al::numbers::sqrt2_v}; const size_t chans{std::min(static_cast(ininfo.channels), 4u)}; for(size_t c{0};c < chans;++c) { for(size_t i{0};i < got;++i) ambmem[c][i] = inmem[i*static_cast(ininfo.channels) + c] * scale; } } else for(size_t idx{0};idx < chanmap.size();++idx) { const int chanid{chanmap[idx]}; /* Skip LFE. Or mix directly into W? Or W+X? */ if(chanid == SF_CHANNEL_MAP_LFE) continue; const auto spkr = std::find_if(spkrs.cbegin(), spkrs.cend(), [chanid](const SpeakerPos pos){return pos.mChannelID == chanid;}); if(spkr == spkrs.cend()) { fmt::println(stderr, " ... failed to find channel ID {}", chanid); continue; } for(size_t i{0};i < got;++i) srcmem[i] = inmem[i*static_cast(ininfo.channels) + idx]; static constexpr auto Deg2Rad = al::numbers::pi / 180.0; const auto coeffs = GenCoeffs( std::cos(spkr->mAzimuth*Deg2Rad) * std::cos(spkr->mElevation*Deg2Rad), std::sin(spkr->mAzimuth*Deg2Rad) * std::cos(spkr->mElevation*Deg2Rad), std::sin(spkr->mElevation*Deg2Rad)); for(size_t c{0};c < 4;++c) { for(size_t i{0};i < got;++i) ambmem[c][i] += srcmem[i] * coeffs[c]; } } encoder->encode(encmem.subspan(0, uhjchans), ambmem, got); if(LeadIn >= got) { LeadIn -= got; continue; } got -= LeadIn; for(size_t c{0};c < uhjchans;++c) { static constexpr float max_val{8388607.0f / 8388608.0f}; for(size_t i{0};i < got;++i) outmem[i*uhjchans + c] = std::clamp(encmem[c][LeadIn+i], -1.0f, max_val); } LeadIn = 0; sf_count_t wrote{sf_writef_float(outfile.get(), outmem.data(), static_cast(got))}; if(wrote < 0) fmt::println(stderr, " ... failed to write samples: {}", sf_error(outfile.get())); else total_wrote += static_cast(wrote); } fmt::println(" ... wrote {} samples ({}).", total_wrote, ininfo.frames); ++num_encoded; }; std::for_each(args.begin(), args.end(), process_arg); if(num_encoded == 0) fmt::println(stderr, "Failed to encode any input files"); else if(num_encoded < num_files) fmt::println(stderr, "Encoded {} of {} files", num_encoded, num_files); else fmt::println("Encoded {}{} file{}", (num_encoded > 1) ? "all " : "", num_encoded, (num_encoded == 1) ? "" : "s"); return 0; } } /* namespace */ int main(int argc, char **argv) { assert(argc >= 0); auto args = std::vector(static_cast(argc)); std::copy_n(argv, args.size(), args.begin()); return main(al::span{args}); }