1cabdff1aSopenharmony_ci/*
2cabdff1aSopenharmony_ci * This file is part of FFmpeg.
3cabdff1aSopenharmony_ci *
4cabdff1aSopenharmony_ci * FFmpeg is free software; you can redistribute it and/or
5cabdff1aSopenharmony_ci * modify it under the terms of the GNU Lesser General Public
6cabdff1aSopenharmony_ci * License as published by the Free Software Foundation; either
7cabdff1aSopenharmony_ci * version 2.1 of the License, or (at your option) any later version.
8cabdff1aSopenharmony_ci *
9cabdff1aSopenharmony_ci * FFmpeg is distributed in the hope that it will be useful,
10cabdff1aSopenharmony_ci * but WITHOUT ANY WARRANTY; without even the implied warranty of
11cabdff1aSopenharmony_ci * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12cabdff1aSopenharmony_ci * Lesser General Public License for more details.
13cabdff1aSopenharmony_ci *
14cabdff1aSopenharmony_ci * You should have received a copy of the GNU Lesser General Public
15cabdff1aSopenharmony_ci * License along with FFmpeg; if not, write to the Free Software
16cabdff1aSopenharmony_ci * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
17cabdff1aSopenharmony_ci */
18cabdff1aSopenharmony_ci
19cabdff1aSopenharmony_ci/**
20cabdff1aSopenharmony_ci * @file
21cabdff1aSopenharmony_ci * Reliable Internet Streaming Transport protocol
22cabdff1aSopenharmony_ci */
23cabdff1aSopenharmony_ci
24cabdff1aSopenharmony_ci#include "libavutil/avassert.h"
25cabdff1aSopenharmony_ci#include "libavutil/avstring.h"
26cabdff1aSopenharmony_ci#include "libavutil/opt.h"
27cabdff1aSopenharmony_ci#include "libavutil/parseutils.h"
28cabdff1aSopenharmony_ci#include "libavutil/time.h"
29cabdff1aSopenharmony_ci
30cabdff1aSopenharmony_ci#include "avformat.h"
31cabdff1aSopenharmony_ci#include "internal.h"
32cabdff1aSopenharmony_ci#include "network.h"
33cabdff1aSopenharmony_ci#include "os_support.h"
34cabdff1aSopenharmony_ci#include "url.h"
35cabdff1aSopenharmony_ci
36cabdff1aSopenharmony_ci#include <librist/librist.h>
37cabdff1aSopenharmony_ci#include <librist/version.h>
38cabdff1aSopenharmony_ci
39cabdff1aSopenharmony_ci// RIST_MAX_PACKET_SIZE - 28 minimum protocol overhead
40cabdff1aSopenharmony_ci#define MAX_PAYLOAD_SIZE (10000-28)
41cabdff1aSopenharmony_ci#define FIFO_SIZE_DEFAULT 8192
42cabdff1aSopenharmony_ci
43cabdff1aSopenharmony_citypedef struct RISTContext {
44cabdff1aSopenharmony_ci    const AVClass *class;
45cabdff1aSopenharmony_ci
46cabdff1aSopenharmony_ci    int profile;
47cabdff1aSopenharmony_ci    int buffer_size;
48cabdff1aSopenharmony_ci    int packet_size;
49cabdff1aSopenharmony_ci    int log_level;
50cabdff1aSopenharmony_ci    int encryption;
51cabdff1aSopenharmony_ci    int fifo_size;
52cabdff1aSopenharmony_ci    int overrun_nonfatal;
53cabdff1aSopenharmony_ci    char *secret;
54cabdff1aSopenharmony_ci
55cabdff1aSopenharmony_ci    struct rist_logging_settings logging_settings;
56cabdff1aSopenharmony_ci    struct rist_peer_config peer_config;
57cabdff1aSopenharmony_ci
58cabdff1aSopenharmony_ci    struct rist_peer *peer;
59cabdff1aSopenharmony_ci    struct rist_ctx *ctx;
60cabdff1aSopenharmony_ci} RISTContext;
61cabdff1aSopenharmony_ci
62cabdff1aSopenharmony_ci#define D AV_OPT_FLAG_DECODING_PARAM
63cabdff1aSopenharmony_ci#define E AV_OPT_FLAG_ENCODING_PARAM
64cabdff1aSopenharmony_ci#define OFFSET(x) offsetof(RISTContext, x)
65cabdff1aSopenharmony_cistatic const AVOption librist_options[] = {
66cabdff1aSopenharmony_ci    { "rist_profile","set profile",     OFFSET(profile),     AV_OPT_TYPE_INT,   {.i64=RIST_PROFILE_MAIN},     0, 2, .flags = D|E, "profile" },
67cabdff1aSopenharmony_ci    { "simple",      NULL,              0,                   AV_OPT_TYPE_CONST, {.i64=RIST_PROFILE_SIMPLE},   0, 0, .flags = D|E, "profile" },
68cabdff1aSopenharmony_ci    { "main",        NULL,              0,                   AV_OPT_TYPE_CONST, {.i64=RIST_PROFILE_MAIN},     0, 0, .flags = D|E, "profile" },
69cabdff1aSopenharmony_ci    { "advanced",    NULL,              0,                   AV_OPT_TYPE_CONST, {.i64=RIST_PROFILE_ADVANCED}, 0, 0, .flags = D|E, "profile" },
70cabdff1aSopenharmony_ci    { "buffer_size", "set buffer_size in ms", OFFSET(buffer_size), AV_OPT_TYPE_INT, {.i64=0},                 0, 30000, .flags = D|E },
71cabdff1aSopenharmony_ci    { "fifo_size",   "set fifo buffer size, must be a power of 2", OFFSET(fifo_size), AV_OPT_TYPE_INT, {.i64=FIFO_SIZE_DEFAULT}, 32, 262144, .flags = D|E },
72cabdff1aSopenharmony_ci    { "overrun_nonfatal", "survive in case of receiving fifo buffer overrun", OFFSET(overrun_nonfatal), AV_OPT_TYPE_BOOL, {.i64 = 0}, 0, 1,    D },
73cabdff1aSopenharmony_ci    { "pkt_size",    "set packet size", OFFSET(packet_size), AV_OPT_TYPE_INT,   {.i64=1316},                  1, MAX_PAYLOAD_SIZE,    .flags = D|E },
74cabdff1aSopenharmony_ci    { "log_level",   "set loglevel",    OFFSET(log_level),   AV_OPT_TYPE_INT,   {.i64=RIST_LOG_INFO},        -1, INT_MAX, .flags = D|E },
75cabdff1aSopenharmony_ci    { "secret", "set encryption secret",OFFSET(secret),      AV_OPT_TYPE_STRING,{.str=NULL},                  0, 0,       .flags = D|E },
76cabdff1aSopenharmony_ci    { "encryption","set encryption type",OFFSET(encryption), AV_OPT_TYPE_INT   ,{.i64=0},                     0, INT_MAX, .flags = D|E },
77cabdff1aSopenharmony_ci    { NULL }
78cabdff1aSopenharmony_ci};
79cabdff1aSopenharmony_ci
80cabdff1aSopenharmony_cistatic int risterr2ret(int err)
81cabdff1aSopenharmony_ci{
82cabdff1aSopenharmony_ci    switch (err) {
83cabdff1aSopenharmony_ci    case RIST_ERR_MALLOC:
84cabdff1aSopenharmony_ci        return AVERROR(ENOMEM);
85cabdff1aSopenharmony_ci    default:
86cabdff1aSopenharmony_ci        return AVERROR_EXTERNAL;
87cabdff1aSopenharmony_ci    }
88cabdff1aSopenharmony_ci}
89cabdff1aSopenharmony_ci
90cabdff1aSopenharmony_cistatic int log_cb(void *arg, enum rist_log_level log_level, const char *msg)
91cabdff1aSopenharmony_ci{
92cabdff1aSopenharmony_ci    int level;
93cabdff1aSopenharmony_ci
94cabdff1aSopenharmony_ci    switch (log_level) {
95cabdff1aSopenharmony_ci    case RIST_LOG_ERROR:    level = AV_LOG_ERROR;   break;
96cabdff1aSopenharmony_ci    case RIST_LOG_WARN:     level = AV_LOG_WARNING; break;
97cabdff1aSopenharmony_ci    case RIST_LOG_NOTICE:   level = AV_LOG_INFO;    break;
98cabdff1aSopenharmony_ci    case RIST_LOG_INFO:     level = AV_LOG_VERBOSE; break;
99cabdff1aSopenharmony_ci    case RIST_LOG_DEBUG:    level = AV_LOG_DEBUG;   break;
100cabdff1aSopenharmony_ci    case RIST_LOG_DISABLE:  level = AV_LOG_QUIET;   break;
101cabdff1aSopenharmony_ci    default: level = AV_LOG_WARNING;
102cabdff1aSopenharmony_ci    }
103cabdff1aSopenharmony_ci
104cabdff1aSopenharmony_ci    av_log(arg, level, "%s", msg);
105cabdff1aSopenharmony_ci
106cabdff1aSopenharmony_ci    return 0;
107cabdff1aSopenharmony_ci}
108cabdff1aSopenharmony_ci
109cabdff1aSopenharmony_cistatic int librist_close(URLContext *h)
110cabdff1aSopenharmony_ci{
111cabdff1aSopenharmony_ci    RISTContext *s = h->priv_data;
112cabdff1aSopenharmony_ci    int ret = 0;
113cabdff1aSopenharmony_ci
114cabdff1aSopenharmony_ci    s->peer = NULL;
115cabdff1aSopenharmony_ci
116cabdff1aSopenharmony_ci    if (s->ctx)
117cabdff1aSopenharmony_ci        ret = rist_destroy(s->ctx);
118cabdff1aSopenharmony_ci    s->ctx = NULL;
119cabdff1aSopenharmony_ci
120cabdff1aSopenharmony_ci    return risterr2ret(ret);
121cabdff1aSopenharmony_ci}
122cabdff1aSopenharmony_ci
123cabdff1aSopenharmony_cistatic int librist_open(URLContext *h, const char *uri, int flags)
124cabdff1aSopenharmony_ci{
125cabdff1aSopenharmony_ci    RISTContext *s = h->priv_data;
126cabdff1aSopenharmony_ci    struct rist_logging_settings *logging_settings = &s->logging_settings;
127cabdff1aSopenharmony_ci    struct rist_peer_config *peer_config = &s->peer_config;
128cabdff1aSopenharmony_ci    int ret;
129cabdff1aSopenharmony_ci
130cabdff1aSopenharmony_ci    if ((flags & AVIO_FLAG_READ_WRITE) == AVIO_FLAG_READ_WRITE)
131cabdff1aSopenharmony_ci        return AVERROR(EINVAL);
132cabdff1aSopenharmony_ci
133cabdff1aSopenharmony_ci    s->logging_settings = (struct rist_logging_settings)LOGGING_SETTINGS_INITIALIZER;
134cabdff1aSopenharmony_ci    ret = rist_logging_set(&logging_settings, s->log_level, log_cb, h, NULL, NULL);
135cabdff1aSopenharmony_ci    if (ret < 0)
136cabdff1aSopenharmony_ci        return risterr2ret(ret);
137cabdff1aSopenharmony_ci
138cabdff1aSopenharmony_ci    if (flags & AVIO_FLAG_WRITE) {
139cabdff1aSopenharmony_ci        h->max_packet_size = s->packet_size;
140cabdff1aSopenharmony_ci        ret = rist_sender_create(&s->ctx, s->profile, 0, logging_settings);
141cabdff1aSopenharmony_ci    }
142cabdff1aSopenharmony_ci    if (ret < 0)
143cabdff1aSopenharmony_ci        goto err;
144cabdff1aSopenharmony_ci
145cabdff1aSopenharmony_ci    if (flags & AVIO_FLAG_READ) {
146cabdff1aSopenharmony_ci        h->max_packet_size = MAX_PAYLOAD_SIZE;
147cabdff1aSopenharmony_ci        ret = rist_receiver_create(&s->ctx, s->profile, logging_settings);
148cabdff1aSopenharmony_ci    }
149cabdff1aSopenharmony_ci    if (ret < 0)
150cabdff1aSopenharmony_ci        goto err;
151cabdff1aSopenharmony_ci
152cabdff1aSopenharmony_ci    ret = rist_peer_config_defaults_set(peer_config);
153cabdff1aSopenharmony_ci    if (ret < 0)
154cabdff1aSopenharmony_ci        goto err;
155cabdff1aSopenharmony_ci
156cabdff1aSopenharmony_ci    ret = rist_parse_address2(uri, &peer_config);
157cabdff1aSopenharmony_ci    if (ret < 0)
158cabdff1aSopenharmony_ci        goto err;
159cabdff1aSopenharmony_ci
160cabdff1aSopenharmony_ci    if (flags & AVIO_FLAG_READ) {
161cabdff1aSopenharmony_ci        ret = rist_receiver_set_output_fifo_size(s->ctx, s->fifo_size);
162cabdff1aSopenharmony_ci        if (ret != 0)
163cabdff1aSopenharmony_ci            goto err;
164cabdff1aSopenharmony_ci    }
165cabdff1aSopenharmony_ci
166cabdff1aSopenharmony_ci    if (((s->encryption == 128 || s->encryption == 256) && !s->secret) ||
167cabdff1aSopenharmony_ci        ((peer_config->key_size == 128 || peer_config->key_size == 256) && !peer_config->secret[0])) {
168cabdff1aSopenharmony_ci        av_log(h, AV_LOG_ERROR, "secret is mandatory if encryption is enabled\n");
169cabdff1aSopenharmony_ci        librist_close(h);
170cabdff1aSopenharmony_ci        return AVERROR(EINVAL);
171cabdff1aSopenharmony_ci    }
172cabdff1aSopenharmony_ci
173cabdff1aSopenharmony_ci    if (s->secret && peer_config->secret[0] == 0)
174cabdff1aSopenharmony_ci        av_strlcpy(peer_config->secret, s->secret, RIST_MAX_STRING_SHORT);
175cabdff1aSopenharmony_ci
176cabdff1aSopenharmony_ci    if (s->secret && (s->encryption == 128 || s->encryption == 256))
177cabdff1aSopenharmony_ci        peer_config->key_size = s->encryption;
178cabdff1aSopenharmony_ci
179cabdff1aSopenharmony_ci    if (s->buffer_size) {
180cabdff1aSopenharmony_ci        peer_config->recovery_length_min = s->buffer_size;
181cabdff1aSopenharmony_ci        peer_config->recovery_length_max = s->buffer_size;
182cabdff1aSopenharmony_ci    }
183cabdff1aSopenharmony_ci
184cabdff1aSopenharmony_ci    ret = rist_peer_create(s->ctx, &s->peer, &s->peer_config);
185cabdff1aSopenharmony_ci    if (ret < 0)
186cabdff1aSopenharmony_ci        goto err;
187cabdff1aSopenharmony_ci
188cabdff1aSopenharmony_ci    ret = rist_start(s->ctx);
189cabdff1aSopenharmony_ci    if (ret < 0)
190cabdff1aSopenharmony_ci        goto err;
191cabdff1aSopenharmony_ci
192cabdff1aSopenharmony_ci    return 0;
193cabdff1aSopenharmony_ci
194cabdff1aSopenharmony_cierr:
195cabdff1aSopenharmony_ci    librist_close(h);
196cabdff1aSopenharmony_ci
197cabdff1aSopenharmony_ci    return risterr2ret(ret);
198cabdff1aSopenharmony_ci}
199cabdff1aSopenharmony_ci
200cabdff1aSopenharmony_cistatic int librist_read(URLContext *h, uint8_t *buf, int size)
201cabdff1aSopenharmony_ci{
202cabdff1aSopenharmony_ci    RISTContext *s = h->priv_data;
203cabdff1aSopenharmony_ci    int ret;
204cabdff1aSopenharmony_ci
205cabdff1aSopenharmony_ci    struct rist_data_block *data_block;
206cabdff1aSopenharmony_ci    ret = rist_receiver_data_read2(s->ctx, &data_block, POLLING_TIME);
207cabdff1aSopenharmony_ci
208cabdff1aSopenharmony_ci    if (ret < 0)
209cabdff1aSopenharmony_ci        return risterr2ret(ret);
210cabdff1aSopenharmony_ci
211cabdff1aSopenharmony_ci    if (ret == 0)
212cabdff1aSopenharmony_ci        return AVERROR(EAGAIN);
213cabdff1aSopenharmony_ci
214cabdff1aSopenharmony_ci    if (data_block->payload_len > MAX_PAYLOAD_SIZE) {
215cabdff1aSopenharmony_ci        rist_receiver_data_block_free2(&data_block);
216cabdff1aSopenharmony_ci        return AVERROR_EXTERNAL;
217cabdff1aSopenharmony_ci    }
218cabdff1aSopenharmony_ci
219cabdff1aSopenharmony_ci    if (data_block->flags & RIST_DATA_FLAGS_OVERFLOW) {
220cabdff1aSopenharmony_ci        if (!s->overrun_nonfatal) {
221cabdff1aSopenharmony_ci            av_log(h, AV_LOG_ERROR, "Fifo buffer overrun. "
222cabdff1aSopenharmony_ci                    "To avoid, increase fifo_size option. "
223cabdff1aSopenharmony_ci                    "To survive in such case, use overrun_nonfatal option\n");
224cabdff1aSopenharmony_ci            size = AVERROR(EIO);
225cabdff1aSopenharmony_ci            goto out_free;
226cabdff1aSopenharmony_ci        }
227cabdff1aSopenharmony_ci    }
228cabdff1aSopenharmony_ci
229cabdff1aSopenharmony_ci    size = data_block->payload_len;
230cabdff1aSopenharmony_ci    memcpy(buf, data_block->payload, size);
231cabdff1aSopenharmony_ciout_free:
232cabdff1aSopenharmony_ci    rist_receiver_data_block_free2(&data_block);
233cabdff1aSopenharmony_ci    return size;
234cabdff1aSopenharmony_ci}
235cabdff1aSopenharmony_ci
236cabdff1aSopenharmony_cistatic int librist_write(URLContext *h, const uint8_t *buf, int size)
237cabdff1aSopenharmony_ci{
238cabdff1aSopenharmony_ci    RISTContext *s = h->priv_data;
239cabdff1aSopenharmony_ci    struct rist_data_block data_block = { 0 };
240cabdff1aSopenharmony_ci    int ret;
241cabdff1aSopenharmony_ci
242cabdff1aSopenharmony_ci    data_block.ts_ntp = 0;
243cabdff1aSopenharmony_ci    data_block.payload = buf;
244cabdff1aSopenharmony_ci    data_block.payload_len = size;
245cabdff1aSopenharmony_ci
246cabdff1aSopenharmony_ci    ret = rist_sender_data_write(s->ctx, &data_block);
247cabdff1aSopenharmony_ci    if (ret < 0)
248cabdff1aSopenharmony_ci        return risterr2ret(ret);
249cabdff1aSopenharmony_ci
250cabdff1aSopenharmony_ci    return ret;
251cabdff1aSopenharmony_ci}
252cabdff1aSopenharmony_ci
253cabdff1aSopenharmony_cistatic const AVClass librist_class = {
254cabdff1aSopenharmony_ci    .class_name = "librist",
255cabdff1aSopenharmony_ci    .item_name  = av_default_item_name,
256cabdff1aSopenharmony_ci    .option     = librist_options,
257cabdff1aSopenharmony_ci    .version    = LIBAVUTIL_VERSION_INT,
258cabdff1aSopenharmony_ci};
259cabdff1aSopenharmony_ci
260cabdff1aSopenharmony_ciconst URLProtocol ff_librist_protocol = {
261cabdff1aSopenharmony_ci    .name                = "rist",
262cabdff1aSopenharmony_ci    .url_open            = librist_open,
263cabdff1aSopenharmony_ci    .url_read            = librist_read,
264cabdff1aSopenharmony_ci    .url_write           = librist_write,
265cabdff1aSopenharmony_ci    .url_close           = librist_close,
266cabdff1aSopenharmony_ci    .priv_data_size      = sizeof(RISTContext),
267cabdff1aSopenharmony_ci    .flags               = URL_PROTOCOL_FLAG_NETWORK,
268cabdff1aSopenharmony_ci    .priv_data_class     = &librist_class,
269cabdff1aSopenharmony_ci};
270