blob: 72fc611acc91271cfee8382ed264692dabfaec77 [file]
/*
* PDV muxer
*
* Copyright (c) 2026 Priyanshu Thapliyal <priyanshuthapliyal2005@gmail.com>
*
* This file is part of FFmpeg.
*
* FFmpeg is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* FFmpeg is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with FFmpeg; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include "libavutil/mem.h"
#include "libavutil/opt.h"
#include "libavutil/rational.h"
#include "avformat.h"
#include "avio_internal.h"
#include "mux.h"
#define PDV_MAGIC "Playdate VID\x00\x00\x00\x00"
#define PDV_MAX_FRAMES UINT16_MAX
#define PDV_MAX_OFFSET ((1U << 30) - 1)
typedef struct PDVMuxContext {
uint32_t *entries;
int nb_frames;
int max_frames;
uint32_t fps_bits;
int64_t nb_frames_pos;
int64_t table_pos;
int64_t payload_start;
} PDVMuxContext;
static void pdv_deinit(AVFormatContext *s)
{
PDVMuxContext *pdv = s->priv_data;
av_freep(&pdv->entries);
}
static int pdv_get_fps(AVFormatContext *s, AVStream *st, uint32_t *fps_bits)
{
AVRational rate = st->avg_frame_rate;
const AVRational zero = { 0, 1 };
if (!rate.num || !rate.den)
rate = av_inv_q(st->time_base);
if (!rate.num || !rate.den) {
av_log(s, AV_LOG_ERROR, "A valid frame rate is required for PDV output.\n");
return AVERROR(EINVAL);
}
if (av_cmp_q(rate, zero) <= 0) {
av_log(s, AV_LOG_ERROR, "Invalid frame rate for PDV output.\n");
return AVERROR(EINVAL);
}
*fps_bits = av_q2intfloat(rate);
return 0;
}
static int pdv_write_header(AVFormatContext *s)
{
PDVMuxContext *pdv = s->priv_data;
AVStream *st;
int ret;
if (!(s->pb->seekable & AVIO_SEEKABLE_NORMAL)) {
av_log(s, AV_LOG_ERROR, "PDV muxer requires seekable output.\n");
return AVERROR(EINVAL);
}
if (s->nb_streams != 1) {
av_log(s, AV_LOG_ERROR, "PDV muxer supports exactly one stream.\n");
return AVERROR(EINVAL);
}
st = s->streams[0];
if (st->codecpar->width <= 0 || st->codecpar->height <= 0) {
av_log(s, AV_LOG_ERROR, "Invalid output dimensions.\n");
return AVERROR(EINVAL);
}
if (st->codecpar->width > UINT16_MAX || st->codecpar->height > UINT16_MAX) {
av_log(s, AV_LOG_ERROR, "Output dimensions exceed PDV limits.\n");
return AVERROR(EINVAL);
}
ret = pdv_get_fps(s, st, &pdv->fps_bits);
if (ret < 0)
return ret;
if (pdv->max_frames < 1 || pdv->max_frames > PDV_MAX_FRAMES) {
av_log(s, AV_LOG_ERROR,
"The -max_frames option must be set to a value in [1, %u].\n",
PDV_MAX_FRAMES);
return AVERROR(EINVAL);
}
pdv->entries = av_malloc_array(pdv->max_frames + 1, sizeof(*pdv->entries));
if (!pdv->entries)
return AVERROR(ENOMEM);
avio_write(s->pb, PDV_MAGIC, 16);
pdv->nb_frames_pos = avio_tell(s->pb);
avio_wl16(s->pb, 0);
avio_wl16(s->pb, 0);
avio_wl32(s->pb, pdv->fps_bits);
avio_wl16(s->pb, st->codecpar->width);
avio_wl16(s->pb, st->codecpar->height);
pdv->table_pos = avio_tell(s->pb);
ffio_fill(s->pb, 0, 4LL * (pdv->max_frames + 1));
pdv->payload_start = avio_tell(s->pb);
if (pdv->nb_frames_pos < 0 || pdv->table_pos < 0 || pdv->payload_start < 0)
return AVERROR(EIO);
return 0;
}
static int pdv_write_packet(AVFormatContext *s, AVPacket *pkt)
{
PDVMuxContext *pdv = s->priv_data;
int64_t offset = avio_tell(s->pb);
const uint32_t max_table_gap = 4U * pdv->max_frames;
if (offset < 0)
return AVERROR(EIO);
offset -= pdv->payload_start;
if (offset < 0)
return AVERROR(EIO);
if (pkt->size <= 0)
return AVERROR_INVALIDDATA;
if (pdv->nb_frames >= pdv->max_frames) {
av_log(s, AV_LOG_ERROR, "Too many frames for PDV output.\n");
return AVERROR(EINVAL);
}
if (offset > PDV_MAX_OFFSET - max_table_gap ||
pkt->size > PDV_MAX_OFFSET - max_table_gap - offset) {
av_log(s, AV_LOG_ERROR, "PDV payload exceeds container limits.\n");
return AVERROR(EINVAL);
}
pdv->entries[pdv->nb_frames] = ((uint32_t)offset << 2) |
(pkt->flags & AV_PKT_FLAG_KEY ? 1 : 2);
avio_write(s->pb, pkt->data, pkt->size);
pdv->nb_frames++;
return 0;
}
static int pdv_write_trailer(AVFormatContext *s)
{
PDVMuxContext *pdv = s->priv_data;
int64_t payload_size = avio_tell(s->pb);
const uint32_t table_gap = 4U * (pdv->max_frames - pdv->nb_frames);
int ret;
if (payload_size < 0)
return AVERROR(EIO);
payload_size -= pdv->payload_start;
if (payload_size < 0 || payload_size > PDV_MAX_OFFSET - table_gap)
return AVERROR(EINVAL);
pdv->entries[pdv->nb_frames] = (uint32_t)payload_size << 2;
if ((ret = avio_seek(s->pb, pdv->nb_frames_pos, SEEK_SET)) < 0)
return ret;
avio_wl16(s->pb, pdv->nb_frames);
if ((ret = avio_seek(s->pb, pdv->table_pos, SEEK_SET)) < 0)
return ret;
for (int i = 0; i <= pdv->nb_frames; i++) {
const uint32_t frame_off = (pdv->entries[i] >> 2) + table_gap;
if (frame_off > PDV_MAX_OFFSET)
return AVERROR(EINVAL);
avio_wl32(s->pb, frame_off << 2 | (pdv->entries[i] & 3));
}
if ((ret = avio_seek(s->pb, pdv->payload_start + payload_size, SEEK_SET)) < 0)
return ret;
return 0;
}
#define OFFSET(x) offsetof(PDVMuxContext, x)
#define ENC AV_OPT_FLAG_ENCODING_PARAM
static const AVOption options[] = {
{ "max_frames", "maximum number of frames reserved in table (mandatory)",
OFFSET(max_frames), AV_OPT_TYPE_INT, { .i64 = -1 }, -1, PDV_MAX_FRAMES, ENC },
{ NULL },
};
static const AVClass pdv_muxer_class = {
.class_name = "PDV muxer",
.item_name = av_default_item_name,
.option = options,
.version = LIBAVUTIL_VERSION_INT,
.category = AV_CLASS_CATEGORY_MUXER,
};
const FFOutputFormat ff_pdv_muxer = {
.p.name = "pdv",
.p.long_name = NULL_IF_CONFIG_SMALL("PlayDate Video"),
.p.extensions = "pdv",
.p.priv_class = &pdv_muxer_class,
.p.audio_codec = AV_CODEC_ID_NONE,
.p.video_codec = AV_CODEC_ID_PDV,
.p.subtitle_codec = AV_CODEC_ID_NONE,
.priv_data_size = sizeof(PDVMuxContext),
.p.flags = AVFMT_NOTIMESTAMPS,
.flags_internal = FF_OFMT_FLAG_MAX_ONE_OF_EACH |
FF_OFMT_FLAG_ONLY_DEFAULT_CODECS,
.write_header = pdv_write_header,
.write_packet = pdv_write_packet,
.write_trailer = pdv_write_trailer,
.deinit = pdv_deinit,
};