stream-deploy/ZLM/3rdpart/media-server/librtsp/test/rtsp-client-input-test.cpp

203 lines
5.6 KiB
C++
Raw Normal View History

#if defined(_DEBUG) || defined(DEBUG)
#include "sockutil.h"
#include "rtsp-client.h"
#include <assert.h>
#include <stdlib.h>
#include "sockpair.h"
#include "cstringext.h"
#include "sys/system.h"
#include "cpm/unuse.h"
#include "sdp.h"
//#define UDP_MULTICAST_ADDR "239.0.0.2"
extern "C" void rtp_receiver_tcp_input(uint8_t channel, const void* data, uint16_t bytes);
extern "C" void rtp_receiver_test(socket_t rtp[2], const char* peer, int peerport[2], int payload, const char* encoding);
extern "C" void* rtp_receiver_tcp_test(uint8_t interleave1, uint8_t interleave2, int payload, const char* encoding);
extern "C" int rtsp_addr_is_multicast(const char* ip);
struct rtsp_client_input_test_t
{
rtsp_client_t* rtsp;
int transport;
socket_t rtp[5][2];
unsigned short port[5][2];
};
static int rtsp_client_send(void* param, const char* uri, const void* req, size_t bytes)
{
//TODO: check uri and make socket
//1. uri != rtsp describe uri(user input)
//2. multi-uri if media_count > 1
struct rtsp_client_input_test_t* ctx = (struct rtsp_client_input_test_t*)param;
return bytes;
}
static int rtpport(void* param, int media, const char* source, unsigned short rtp[2], char* ip, int len)
{
struct rtsp_client_input_test_t* ctx = (struct rtsp_client_input_test_t*)param;
int m = rtsp_client_get_media_type(ctx->rtsp, media);
if (SDP_M_MEDIA_AUDIO != m && SDP_M_MEDIA_VIDEO != m)
return 0; // ignore
switch (ctx->transport)
{
case RTSP_TRANSPORT_RTP_UDP:
// TODO: ipv6
assert(0 == sockpair_create("127.0.0.1", ctx->rtp[media], ctx->port[media]));
rtp[0] = ctx->port[media][0];
rtp[1] = ctx->port[media][1];
if (rtsp_addr_is_multicast(ip))
{
if (0 != socket_udp_multicast(ctx->rtp[media][0], ip, source, 16) || 0 != socket_udp_multicast(ctx->rtp[media][1], ip, source, 16))
return -1;
}
#if defined(UDP_MULTICAST_ADDR)
else
{
if (0 != socket_udp_multicast(ctx->rtp[media][0], UDP_MULTICAST_ADDR, source, 16) || 0 != socket_udp_multicast(ctx->rtp[media][1], UDP_MULTICAST_ADDR, source, 16))
return -1;
snprintf(ip, len, "%s", UDP_MULTICAST_ADDR);
}
#endif
break;
case RTSP_TRANSPORT_RTP_TCP:
rtp[0] = 2 * media;
rtp[1] = 2 * media + 1;
break;
default:
assert(0);
return -1;
}
return ctx->transport;
}
int rtsp_client_options(rtsp_client_t* rtsp, const char* commands);
static void onrtp(void* param, uint8_t channel, const void* data, uint16_t bytes)
{
static int keepalive = 0;
struct rtsp_client_input_test_t* ctx = (struct rtsp_client_input_test_t*)param;
rtp_receiver_tcp_input(channel, data, bytes);
if (++keepalive % 1000 == 0)
{
rtsp_client_play(ctx->rtsp, NULL, NULL);
}
}
static int ondescribe(void* param, const char* sdp, int len)
{
struct rtsp_client_input_test_t* ctx = (struct rtsp_client_input_test_t*)param;
return rtsp_client_setup(ctx->rtsp, sdp, len);
}
static int onsetup(void* param, int timeout, int64_t duration)
{
int i;
uint64_t npt = 0;
char ip[65];
u_short rtspport;
struct rtsp_client_input_test_t* ctx = (struct rtsp_client_input_test_t*)param;
assert(0 == rtsp_client_play(ctx->rtsp, &npt, NULL));
for (i = 0; i < rtsp_client_media_count(ctx->rtsp); i++)
{
int payload, port[2];
const char* encoding;
const struct rtsp_header_transport_t* transport;
transport = rtsp_client_get_media_transport(ctx->rtsp, i);
encoding = rtsp_client_get_media_encoding(ctx->rtsp, i);
payload = rtsp_client_get_media_payload(ctx->rtsp, i);
if (RTSP_TRANSPORT_RTP_UDP == transport->transport)
{
//assert(RTSP_TRANSPORT_RTP_UDP == transport->transport); // udp only
assert(0 == transport->multicast); // unicast only
assert(transport->rtp.u.client_port1 == ctx->port[i][0]);
assert(transport->rtp.u.client_port2 == ctx->port[i][1]);
port[0] = transport->rtp.u.server_port1;
port[1] = transport->rtp.u.server_port2;
if (*transport->source)
{
rtp_receiver_test(ctx->rtp[i], transport->source, port, payload, encoding);
}
else
{
assert(0);
//socket_getpeername(ctx->socket, ip, &rtspport);
rtp_receiver_test(ctx->rtp[i], ip, port, payload, encoding);
}
}
else if (RTSP_TRANSPORT_RTP_TCP == transport->transport)
{
//assert(transport->rtp.u.client_port1 == transport->interleaved1);
//assert(transport->rtp.u.client_port2 == transport->interleaved2);
rtp_receiver_tcp_test(transport->interleaved1, transport->interleaved2, payload, encoding);
}
else
{
assert(0); // TODO
}
}
return 0;
}
static int onteardown(void* param)
{
return 0;
}
static int onplay(void* param, int media, const uint64_t* nptbegin, const uint64_t* nptend, const double* scale, const struct rtsp_rtp_info_t* rtpinfo, int count)
{
return 0;
}
static int onpause(void* param)
{
return 0;
}
void rtsp_client_input_test(const char* file)
{
int r;
struct rtsp_client_input_test_t ctx;
struct rtsp_client_handler_t handler;
static char packet[2 * 1024 * 1024];
memset(&ctx, 0, sizeof(ctx));
handler.send = rtsp_client_send;
handler.rtpport = rtpport;
handler.ondescribe = ondescribe;
handler.onsetup = onsetup;
handler.onplay = onplay;
handler.onpause = onpause;
handler.onteardown = onteardown;
handler.onrtp = onrtp;
ctx.transport = RTSP_TRANSPORT_RTP_TCP;
snprintf(packet, sizeof(packet), "rtsp://%s/%s", "127.0.0.1", file); // url
//ctx.rtsp = rtsp_client_create(NULL, NULL, &handler, &ctx);
ctx.rtsp = rtsp_client_create(packet, "username1", "password1", &handler, &ctx);
assert(ctx.rtsp);
assert(0 == rtsp_client_describe(ctx.rtsp));
FILE* fp = fopen(file, "rb");
while ( (r = fread(packet, 1, sizeof(packet), fp)) > 0)
{
assert(0 == rtsp_client_input(ctx.rtsp, packet, r));
}
assert(0 == rtsp_client_teardown(ctx.rtsp));
rtsp_client_destroy(ctx.rtsp);
fclose(fp);
}
#endif