From aba6a49f74c766fe36cc83d5f515bc935eed1820 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 17 Aug 2024 05:32:32 +0200 Subject: [PATCH 1/4] Add Pixmap Recorder module --- dub.json | 12 ++ pixmaprecorder.d | 411 +++++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 423 insertions(+) create mode 100644 pixmaprecorder.d diff --git a/dub.json b/dub.json index b8d057a..66b4572 100644 --- a/dub.json +++ b/dub.json @@ -694,6 +694,18 @@ "dflags-ldc": ["--mv=arsd.pixmappaint=$PACKAGE_DIR/pixmappaint.d"], "dflags-gdc": ["-fmodule-file=arsd.pixmappaint=$PACKAGE_DIR/pixmappaint.d"] }, + { + "name": "pixmaprecorder", + "description": "Video rendering extension for Pixmap Paint. Fancy wrapper for piping frame data to FFmpeg.", + "targetType": "library", + "sourceFiles": ["pixmaprecorder.d"], + "dependencies": { + "arsd-official:pixmappaint":"*" + }, + "dflags-dmd": ["-mv=arsd.pixmaprecorder=$PACKAGE_DIR/pixmaprecorder.d"], + "dflags-ldc": ["--mv=arsd.pixmaprecorder=$PACKAGE_DIR/pixmaprecorder.d"], + "dflags-gdc": ["-fmodule-file=arsd.pixmaprecorder=$PACKAGE_DIR/pixmaprecorder.d"] + }, { "name": "pixmappresenter", "description": "High-level display library. Designed to blit fully-rendered frames to the screen.", diff --git a/pixmaprecorder.d b/pixmaprecorder.d new file mode 100644 index 0000000..89b39d5 --- /dev/null +++ b/pixmaprecorder.d @@ -0,0 +1,411 @@ +/+ + == pixmaprecorder == + Copyright Elias Batek (0xEAB) 2024. + Distributed under the Boost Software License, Version 1.0. + +/ +/++ + $(B Pixmap Recorder) is a helper library for rendering video files from + [arsd.pixmappaint.Pixmap|Pixmap] frames, by piping them to + [FFmpeg](https://ffmpeg.org/about.html). + + $(SIDEBAR + Piping frame data to an independent copy of FFmpeg enables this library + to be used with a wide range of verions of said third-party program + and (hopefully) helps to reduce the chances of breaking changes. + + It also allows end-users to upgrade their possibilities by swapping the + accompanying copy FFmpeg. + This could be useful in cases where software distributors can only + provide limited functionality in their bundled binaries because of + legal requirements like patent licenses. + ) + + $(TIP + The value of the `outputFormat` parameter of the constructor overloads + is passed to FFmpeg via the `-f` option. + + Run `ffmpeg -formats` to get a list of available formats. + ) + + $(TIP + To pass additional options to FFmpeg, use the + [PixmapRecorder.advancedFFmpegAdditionalOutputArgs|additional-output-args property]. + ) + + --- + import arsd.pixmaprecorder; + import arsd.pixmappaint; + + /++ + This demo renders a 1280×720 video at 30 FPS + fading from white (#FFF) to blue (#00F). + +/ + int main() { + // Instantiate a recorder. + auto recorder = new PixmapRecorder( + 30, // Video framerate [=FPS] + "out.mkv", // Output path to write the video file to. + ); + + // We will use this framebuffer later on to provide image data + // to the encoder. + auto frame = Pixmap(1280, 720); + + for (int light = 0xFF; light >= 0; --light) { + auto color = Color(light, light, 0xFF); + frame.clear(color); + + // Record the current frame. + // The video resolution to use is derived from the first frame. + recorder.put(frame); + } + + // End and finalize the recording process. + return recorder.stopRecording(); + } + --- + +/ +module arsd.pixmaprecorder; + +import arsd.pixmappaint; + +import std.format; +import std.path : buildPath; +import std.process; +import std.sumtype; +import std.stdio : File; + +private @safe { + + auto stderrFauxSafe() @trusted { + import std.stdio : stderr; + + return stderr; + } + + auto stdoutFauxSafe() @trusted { + import std.stdio : stderr; + + return stderr; + } + + auto stderr() { + return stderrFauxSafe; + } + + auto stdout() { + return stderrFauxSafe; + } + + alias RecorderOutput = SumType!(string, File); +} + +final class PixmapRecorder { + +@safe: + + private { + string _ffmpegExecutablePath; + double _frameRate; + string _outputFormat; + RecorderOutput _output; + File _log; + string[] _outputAdditionalArgs; + + Pid _pid; + Pipe _input; + Size _resolution; + bool _outputIsOurs = false; + } + + private this( + string ffmpegExecutablePath, + double frameRate, + string outputFormat, + RecorderOutput output, + File log, + ) { + _ffmpegExecutablePath = ffmpegExecutablePath; + _frameRate = frameRate; + _outputFormat = outputFormat; + _output = output; + _log = log; + } + + /++ + Prepares a recorder for encoding video frames + into the specified file pipe. + + $(WARNING + Certain formats cannot be produced in pipes by FFmpeg. + Look out for error message like such: + + ($BLOCKQUOTE + `[mp4 @ 0xdead1337beef] muxer does not support non seekable output` + ) + + This is not a limitation of this library (but rather one of FFmpeg). + Let FFmpeg output the video to file path instead; + check out the other overloads of this constructor. + ) + + Params: + frameRate = Framerate of the video output; in frames per second. + output = File handle to write the video output to. + outputFormat = Video (container) format to output. + This is value passed to FFmpeg via the `-f` option. + log = Target file for the stderr log output of FFmpeg. + This is where error messages are written to. + ffmpegExecutablePath = Path to the FFmpeg executable + (e.g. `ffmpeg`, `ffmpeg.exe` or `/usr/bin/ffmpeg`). + /* Keep this table in sync with the ones of other overloads. */ + +/ + public this( + double frameRate, + File output, + string outputFormat, + File log = stderr, + string ffmpegExecutablePath = "ffmpeg", + ) + in (frameRate > 0) + in (output.isOpen) + in (outputFormat != "") + in (log.isOpen) + in (ffmpegExecutablePath != "") { + this( + ffmpegExecutablePath, + frameRate, + outputFormat, + RecorderOutput(output), + log, + ); + } + + /++ + Prepares a recorder for encoding video frames + into a video file saved to the specified path. + + Params: + frameRate = Framerate of the video output; in frames per second. + outputPath = File path to write the video output to. + Existing files will be overwritten. + FFmpeg will use this to autodetect the format + when no `outputFormat` is provided. + log = Target file for the stderr log output of FFmpeg. + This is where error messages are written to. + outputFormat = Video (container) format to output. + This is value passed to FFmpeg via the `-f` option. + If `null`, the format is not provided and FFmpeg + will try to autodetect the format from the filename + of the `outputPath`. + ffmpegExecutablePath = Path to the FFmpeg executable + (e.g. `ffmpeg`, `ffmpeg.exe` or `/usr/bin/ffmpeg`). + /* Keep this table in sync with the ones of other overloads. */ + +/ + public this( + double frameRate, + string outputPath, + File log = stderr, + string outputFormat = null, + string ffmpegExecutablePath = "ffmpeg", + ) + in (frameRate > 0) + in ((outputPath != "") && (outputPath != "-")) + in (log.isOpen) + in ((outputFormat is null) || outputFormat != "") + in (ffmpegExecutablePath != "") { + + // Sanitize output path + // if it would get confused with a command-line arg. + // Otherwise a relative path like `-my.mkv` would make FFmpeg complain + // about an “Unrecognized option 'out.mkv'”. + if (outputPath[0] == '-') { + outputPath = buildPath(".", outputPath); + } + + this( + ffmpegExecutablePath, + frameRate, + null, + RecorderOutput(outputPath), + log, + ); + } + + /++ + $(I Advanced users only:) + Additional command-line arguments passed to FFmpeg. + + $(WARNING + The values provided through this property function are not + validated and passed verbatim to FFmpeg. + ) + + $(PITFAL + If code makes use of this and FFmpeg errors, + check the arguments provided here this first. + ) + +/ + void advancedFFmpegAdditionalOutputArgs(string[] args) { + _outputAdditionalArgs = args; + } + + /++ + Determines whether the recorder is active + (which implies that an output file is open). + +/ + bool isOpen() { + return _input.writeEnd.isOpen; + } + + private string[] buildFFmpegCommand() pure { + // Build resolution as understood by FFmpeg. + const string resolutionString = format!"%sx%s"( + _resolution.width, + _resolution.height, + ); + + // Convert framerate to string. + const string frameRateString = format!"%s"(_frameRate); + + // Build command-line argument list. + auto cmd = [ + _ffmpegExecutablePath, + "-y", + "-r", + frameRateString, + "-f", + "rawvideo", + "-pix_fmt", + "rgba", + "-s", + resolutionString, + "-i", + "-", + ]; + + if (_outputFormat !is null) { + cmd ~= "-f"; + cmd ~= _outputFormat; + } + + if (_outputAdditionalArgs.length > 0) { + cmd = cmd ~ _outputAdditionalArgs; + } + + cmd ~= _output.match!( + (string filePath) => filePath, + (ref File file) => "-", + ); + + return cmd; + } + + /++ + Starts the video encoding process. + Launches FFmpeg. + + This function sets the video resolution for the encoding process. + All frames to record must match it. + + $(SIDEBAR + Variable/dynamic resolution is neither supported by this library + nor most real-world applications. + ) + + $(NOTE + This function is called by [put|put()] automatically. + There’s usually no need to call this manually. + ) + +/ + void open(Size resolution) + in (!this.isOpen) { + // Save resolution for sanity checks. + _resolution = resolution; + + const string[] cmd = buildFFmpegCommand(); + + // Prepare arsd → FFmpeg I/O pipe. + _input = pipe(); + + // Launch FFmpeg. + const processConfig = ( + Config.suppressConsole + | Config.newEnv + ); + + // dfmt off + _pid = _output.match!( + delegate(string filePath) { + auto stdout = pipe(); + stdout.readEnd.close(); + return spawnProcess( + cmd, + _input.readEnd, + stdout.writeEnd, + _log, + null, + processConfig, + ); + }, + delegate(File file) { + auto stdout = pipe(); + stdout.readEnd.close(); + return spawnProcess( + cmd, + _input.readEnd, + file, + _log, + null, + processConfig, + ); + } + ); + // dfmt on + } + + /// ditto + alias startRecording = close; + + /++ + Provides the next video frame to encode. + + $(TIP + This function automatically calls [open|open()] if necessary. + ) + +/ + void put(Pixmap frame) { + if (!this.isOpen) { + this.open(frame.size); + } else { + assert(frame.size == _resolution, "Variable resolutions are not supported."); + } + + _input.writeEnd.rawWrite(frame.data); + } + + /++ + Ends the recording process. + + $(NOTE + Waits for the FFmpeg process to exit in a blocking way. + ) + + Returns: + The status code provided by the FFmpeg program. + +/ + int close() + in (this.isOpen) { + + _input.writeEnd.flush(); + _input.writeEnd.close(); + scope (exit) { + _input.close(); + } + + return wait(_pid); + } + + /// ditto + alias stopRecording = close; +} From f1b69132aff7ac90d34eacb2d76c7932fc288491 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 17 Aug 2024 05:35:16 +0200 Subject: [PATCH 2/4] Improve description of Pixmap Recorder --- pixmaprecorder.d | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/pixmaprecorder.d b/pixmaprecorder.d index 89b39d5..5b51245 100644 --- a/pixmaprecorder.d +++ b/pixmaprecorder.d @@ -4,8 +4,8 @@ Distributed under the Boost Software License, Version 1.0. +/ /++ - $(B Pixmap Recorder) is a helper library for rendering video files from - [arsd.pixmappaint.Pixmap|Pixmap] frames, by piping them to + $(B Pixmap Recorder) is an auxiliary library for rendering video files from + [arsd.pixmappaint.Pixmap|Pixmap] frames by piping them to [FFmpeg](https://ffmpeg.org/about.html). $(SIDEBAR From a49e7c16e52b6691fab067539df8ba034f2837f4 Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 17 Aug 2024 05:39:22 +0200 Subject: [PATCH 3/4] Add Pixmap Recorder to the changelog --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index 73f740c..cdf5714 100644 --- a/README.md +++ b/README.md @@ -28,7 +28,7 @@ Future release, likely May 2024 or later. Nothing is planned for it at this time. -arsd.pixmappresenter and arsd.pixmappaint were added. +arsd.pixmappresenter, arsd.pixmappaint and arsd.pixmaprecorder were added. ## 11.0 From d20992343385ae4a3d84452e5cd52c2707dc44af Mon Sep 17 00:00:00 2001 From: Elias Batek Date: Sat, 17 Aug 2024 06:10:16 +0200 Subject: [PATCH 4/4] Allow PixmapRecorder.close() to be called on inactive recorders --- pixmaprecorder.d | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/pixmaprecorder.d b/pixmaprecorder.d index 5b51245..79f8329 100644 --- a/pixmaprecorder.d +++ b/pixmaprecorder.d @@ -394,8 +394,10 @@ final class PixmapRecorder { Returns: The status code provided by the FFmpeg program. +/ - int close() - in (this.isOpen) { + int close() { + if (!this.isOpen) { + return; + } _input.writeEnd.flush(); _input.writeEnd.close();