This is an automated email from the git hooks/post-receive script.

Git pushed a commit to branch master
in repository ffmpeg.

commit 44065ea8787fad4074b8e6b942475dfa63626c8d
Author:     Marvin Scholz <[email protected]>
AuthorDate: Tue Dec 9 16:02:03 2025 +0100
Commit:     Marvin Scholz <[email protected]>
CommitDate: Thu Feb 19 17:18:12 2026 +0100

    avformat/rtspdec: Add SET_PARAMETER command support
    
    Add SET_PARAMETER support, this allows sending SET_PARAMETER requests to
    the server using the API.
---
 libavformat/avformat.h | 40 ++++++++++++++++++++++++
 libavformat/rtspdec.c  | 84 ++++++++++++++++++++++++++++++++++++++++++++++++++
 2 files changed, 124 insertions(+)

diff --git a/libavformat/avformat.h b/libavformat/avformat.h
index 21b0e3103b..76c251ac02 100644
--- a/libavformat/avformat.h
+++ b/libavformat/avformat.h
@@ -2389,6 +2389,46 @@ enum AVFormatCommandID {
     AVFORMAT_COMMAND_RTSP_SET_PARAMETER,
 };
 
+typedef struct AVRTSPCommandRequest {
+    /**
+     * Headers sent in the request to the server
+     */
+    AVDictionary *headers;
+
+    /**
+     * Body payload size
+     */
+    size_t body_len;
+
+    /**
+     * Body payload
+     */
+    char *body;
+} AVRTSPCommandRequest;
+
+typedef struct AVRTSPResponse {
+    /**
+     * Response status code from server
+     */
+    int status_code;
+
+    /**
+     * Reason phrase from the server, describing the
+     * status in a human-readable way.
+     */
+    char *reason;
+
+    /**
+     * Body payload size
+     */
+    size_t body_len;
+
+    /**
+     * Body payload
+     */
+    unsigned char *body;
+} AVRTSPResponse;
+
 /**
  * Send a command to the demuxer
  *
diff --git a/libavformat/rtspdec.c b/libavformat/rtspdec.c
index 3681b57ee7..0d39daa39d 100644
--- a/libavformat/rtspdec.c
+++ b/libavformat/rtspdec.c
@@ -22,6 +22,7 @@
 #include "config_components.h"
 
 #include "libavutil/avstring.h"
+#include "libavutil/avassert.h"
 #include "libavutil/intreadwrite.h"
 #include "libavutil/mathematics.h"
 #include "libavutil/mem.h"
@@ -624,6 +625,88 @@ static int rtsp_read_set_state(AVFormatContext *s,
     }
 }
 
+static char *dict_to_headers(AVDictionary *headers)
+{
+    char *buf;
+    AVBPrint bprint;
+    av_bprint_init(&bprint, 0, AV_BPRINT_SIZE_UNLIMITED);
+
+    const AVDictionaryEntry *header = NULL;
+    while ((header = av_dict_iterate(headers, header))) {
+        av_bprintf(&bprint, "%s: %s\r\n", header->key, header->value);
+    }
+
+    av_bprint_finalize(&bprint, &buf);
+    return buf;
+}
+
+static int rtsp_submit_command(struct AVFormatContext *s, enum 
AVFormatCommandID id, void *data)
+{
+    if (id != AVFORMAT_COMMAND_RTSP_SET_PARAMETER)
+        return AVERROR(ENOTSUP);
+    if (!data)
+        return AVERROR(EINVAL);
+
+    RTSPState *rt = s->priv_data;
+    AVRTSPCommandRequest *req = data;
+
+    av_log(s, AV_LOG_DEBUG, "Sending SET_PARAMETER command to %s\n", 
rt->control_uri);
+    char *headers = dict_to_headers(req->headers);
+
+    int ret = ff_rtsp_send_cmd_with_content_async(s, "SET_PARAMETER", 
rt->control_uri,
+        headers, req->body, req->body_len);
+    av_free(headers);
+
+    if (ret != 0)
+        av_log(s, AV_LOG_ERROR, "Failure sending SET_PARAMETER command: %s\n", 
av_err2str(ret));
+
+    return ret;
+}
+
+static int rtsp_read_command_reply(AVFormatContext *s, enum AVFormatCommandID 
id, void **data_out)
+{
+    if (id != AVFORMAT_COMMAND_RTSP_SET_PARAMETER)
+        return AVERROR(ENOTSUP);
+
+    if (!data_out)
+        return AVERROR(EINVAL);
+
+    AVRTSPResponse *res = av_malloc(sizeof(*res));
+    if (!res)
+        return AVERROR(ENOMEM);
+
+    RTSPMessageHeader reply;
+    int ret = ff_rtsp_read_reply(s, &reply, &res->body, 0, "SET_PARAMETER");
+    if (ret < 0)
+        return ret;
+
+    res->status_code = reply.status_code;
+    res->body_len = reply.content_length;
+
+    res->reason = av_strdup(reply.reason);
+    if (!res->reason) {
+        av_free(res);
+        return AVERROR(ENOMEM);
+    }
+
+    *data_out = res;
+    return 0;
+}
+
+static int rtsp_handle_command(struct AVFormatContext *s,
+    enum FFInputFormatCommandOption opt,
+    enum AVFormatCommandID id, void *data)
+{
+    switch (opt) {
+        case FF_INFMT_COMMAND_SUBMIT:
+        return rtsp_submit_command(s, id, data);
+        case FF_INFMT_COMMAND_GET_REPLY:
+        return rtsp_read_command_reply(s, id, data);
+        default:
+        av_unreachable("Invalid command option");
+    }
+}
+
 int ff_rtsp_setup_input_streams(AVFormatContext *s, RTSPMessageHeader *reply)
 {
     RTSPState *rt = s->priv_data;
@@ -1019,4 +1102,5 @@ const FFInputFormat ff_rtsp_demuxer = {
     .read_close     = rtsp_read_close,
     .read_seek      = rtsp_read_seek,
     .read_set_state = rtsp_read_set_state,
+    .handle_command = rtsp_handle_command,
 };

_______________________________________________
ffmpeg-cvslog mailing list -- [email protected]
To unsubscribe send an email to [email protected]

Reply via email to