/* * Copyright (c) 2012 The WebRTC project authors. All Rights Reserved. * * Use of this source code is governed by a BSD-style license * that can be found in the LICENSE file in the root of the source * tree. An additional intellectual property rights grant can be found * in the file PATENTS. All contributing project authors may * be found in the AUTHORS file in the root of the source tree. */ //TODO(hlundin): Reformat file to meet style guide. /* header includes */ #include <float.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #ifdef WIN32 #include <winsock2.h> #include <io.h> #endif #ifdef WEBRTC_LINUX #include <netinet/in.h> #endif #include <assert.h> #include "gtest/gtest.h" #include "webrtc/typedefs.h" /*********************/ /* Misc. definitions */ /*********************/ #define FIRSTLINELEN 40 #define CHECK_NOT_NULL(a) if((a)==NULL){fprintf(stderr,"\n %s \n line: %d \nerror at %s\n",__FILE__,__LINE__,#a );return(-1);} struct arr_time { float time; uint32_t ix; }; int filelen(FILE *fid) { fpos_t cur_pos; int len; if (!fid || fgetpos(fid, &cur_pos)) { return(-1); } fseek(fid, 0, SEEK_END); len = ftell(fid); fsetpos(fid, &cur_pos); return (len); } int compare_arr_time(const void *x, const void *y); int main(int argc, char* argv[]) { unsigned int dat_len, rtp_len, Npack, k; arr_time *time_vec; char firstline[FIRSTLINELEN]; unsigned char* rtp_vec = NULL; unsigned char** packet_ptr = NULL; unsigned char* temp_packet = NULL; const unsigned int kRtpDumpHeaderSize = 4 + 4 + 4 + 2 + 2; uint16_t len; uint32_t *offset; /* check number of parameters */ if (argc != 4) { /* print help text and exit */ printf("Apply jitter on RTP stream.\n"); printf("The program reads an RTP stream and packet timing from two files.\n"); printf("The RTP stream is modified to have the same jitter as described in the timing files.\n"); printf("The format of the RTP stream file should be the same as for rtpplay,\n"); printf("and can be obtained e.g., from Ethereal by using\n"); printf("Statistics -> RTP -> Show All Streams -> [select a stream] -> Save As\n\n"); printf("Usage:\n\n"); printf("%s RTP_infile dat_file RTP_outfile\n", argv[0]); printf("where:\n"); printf("RTP_infile : RTP stream input file\n\n"); printf("dat_file : file with packet arrival times in ms\n\n"); printf("RTP_outfile : RTP stream output file\n\n"); return(0); } FILE* in_file=fopen(argv[1],"rb"); CHECK_NOT_NULL(in_file); printf("Input file: %s\n",argv[1]); FILE* dat_file=fopen(argv[2],"rb"); CHECK_NOT_NULL(dat_file); printf("Dat-file: %s\n",argv[2]); FILE* out_file=fopen(argv[3],"wb"); CHECK_NOT_NULL(out_file); printf("Output file: %s\n\n",argv[3]); time_vec = (arr_time *) malloc(sizeof(arr_time)*(filelen(dat_file)/sizeof(float)) + 1000); // add 1000 bytes to avoid (rare) strange error if (time_vec==NULL) { fprintf(stderr, "Error: could not allocate memory for reading dat file\n"); goto closing; } dat_len=0; while(fread(&(time_vec[dat_len].time),sizeof(float),1,dat_file)>0) { time_vec[dat_len].ix=dat_len; dat_len++; } if (dat_len == 0) { fprintf(stderr, "Error: dat_file is empty, no arrival time is given.\n"); goto closing; } qsort(time_vec,dat_len,sizeof(arr_time),compare_arr_time); rtp_vec = (unsigned char *) malloc(sizeof(unsigned char)*filelen(in_file)); if (rtp_vec==NULL) { fprintf(stderr,"Error: could not allocate memory for reading rtp file\n"); goto closing; } // read file header and write directly to output file EXPECT_TRUE(fgets(firstline, FIRSTLINELEN, in_file) != NULL); EXPECT_GT(fputs(firstline, out_file), 0); EXPECT_EQ(kRtpDumpHeaderSize, fread(firstline, 1, kRtpDumpHeaderSize, in_file)); EXPECT_EQ(kRtpDumpHeaderSize, fwrite(firstline, 1, kRtpDumpHeaderSize, out_file)); // read all RTP packets into vector rtp_len=0; Npack=0; len=(uint16_t) fread(&rtp_vec[rtp_len], sizeof(unsigned char), 2, in_file); // read length of first packet while(len==2) { len = ntohs(*((uint16_t *)(rtp_vec + rtp_len))); rtp_len += 2; if(fread(&rtp_vec[rtp_len], sizeof(unsigned char), len-2, in_file)!=(unsigned) (len-2)) { fprintf(stderr,"Error: currupt packet length\n"); goto closing; } rtp_len += len-2; Npack++; len=(uint16_t) fread(&rtp_vec[rtp_len], sizeof(unsigned char), 2, in_file); // read length of next packet } if (Npack == 0) { fprintf(stderr, "Error: No RTP packet found.\n"); goto closing; } packet_ptr = (unsigned char **) malloc(Npack*sizeof(unsigned char*)); packet_ptr[0]=rtp_vec; k=1; while(k<Npack) { len = ntohs(*((uint16_t *) packet_ptr[k-1])); packet_ptr[k]=packet_ptr[k-1]+len; k++; } for(k=0; k<dat_len && k<Npack; k++) { if(time_vec[k].time < FLT_MAX && time_vec[k].ix < Npack){ temp_packet = packet_ptr[time_vec[k].ix]; offset = (uint32_t *) (temp_packet+4); if ( time_vec[k].time >= 0 ) { *offset = htonl((uint32_t) time_vec[k].time); } else { *offset = htonl((uint32_t) 0); fprintf(stderr, "Warning: negative receive time in dat file transformed to 0.\n"); } // write packet to file if (fwrite(temp_packet, sizeof(unsigned char), ntohs(*((uint16_t*) temp_packet)), out_file) != ntohs(*((uint16_t*) temp_packet))) { return -1; } } } closing: free(time_vec); free(rtp_vec); if (packet_ptr != NULL) { free(packet_ptr); } fclose(in_file); fclose(dat_file); fclose(out_file); return(0); } int compare_arr_time(const void *xp, const void *yp) { if(((arr_time *)xp)->time == ((arr_time *)yp)->time) return(0); else if(((arr_time *)xp)->time > ((arr_time *)yp)->time) return(1); return(-1); }