- Timestamp:
- May 7, 2010 3:15:39 PM (14 years ago)
- File:
-
- 1 edited
Legend:
- Unmodified
- Added
- Removed
-
pjproject/trunk/pjmedia/src/pjmedia/stream.c
r3072 r3161 50 50 */ 51 51 #define MAX_PLC_MSEC PJMEDIA_MAX_PLC_DURATION_MSEC 52 53 54 /* Tracing jitter buffer operations in a stream session to a CSV file. 55 * The trace will contain JB operation timestamp, frame info, RTP info, and 56 * the JB state right after the operation. 57 */ 58 #define TRACE_JB 0 /* Enable/disable trace. */ 59 #define TRACE_JB_PATH_PREFIX "" /* Optional path/prefix 60 for the CSV filename. */ 61 #if TRACE_JB 62 # include <pj/file_io.h> 63 # define TRACE_JB_INVALID_FD ((pj_oshandle_t)-1) 64 # define TRACE_JB_OPENED(s) (s->trace_jb_fd != TRACE_JB_INVALID_FD) 65 #endif 66 67 52 68 53 69 /** … … 185 201 packet */ 186 202 #endif 203 204 #if TRACE_JB 205 pj_oshandle_t trace_jb_fd; /**< Jitter tracing file handle.*/ 206 char *trace_jb_buf; /**< Jitter tracing buffer. */ 207 #endif 187 208 }; 188 209 … … 208 229 PJ_LOG(4,(sender, "%s: %s [err:%d]", title, errmsg, status)); 209 230 } 231 232 233 #if TRACE_JB 234 235 PJ_INLINE(int) trace_jb_print_timestamp(char **buf, pj_ssize_t len) 236 { 237 pj_time_val now; 238 pj_parsed_time ptime; 239 char *p = *buf; 240 241 if (len < 14) 242 return -1; 243 244 pj_gettimeofday(&now); 245 pj_time_decode(&now, &ptime); 246 p += pj_utoa_pad(ptime.hour, p, 2, '0'); 247 *p++ = ':'; 248 p += pj_utoa_pad(ptime.min, p, 2, '0'); 249 *p++ = ':'; 250 p += pj_utoa_pad(ptime.sec, p, 2, '0'); 251 *p++ = '.'; 252 p += pj_utoa_pad(ptime.msec, p, 3, '0'); 253 *p++ = ','; 254 255 *buf = p; 256 257 return 0; 258 } 259 260 PJ_INLINE(int) trace_jb_print_state(pjmedia_stream *stream, 261 char **buf, pj_ssize_t len) 262 { 263 char *p = *buf; 264 char *endp = *buf + len; 265 pjmedia_jb_state state; 266 267 pjmedia_jbuf_get_state(stream->jb, &state); 268 269 len = pj_ansi_snprintf(p, endp-p, "%d, %d, %d", 270 state.size, state.burst, state.prefetch); 271 if ((len < 0) || (len >= endp-p)) 272 return -1; 273 274 p += len; 275 *buf = p; 276 return 0; 277 } 278 279 static void trace_jb_get(pjmedia_stream *stream, pjmedia_jb_frame_type ft, 280 pj_size_t fsize) 281 { 282 char *p = stream->trace_jb_buf; 283 char *endp = stream->trace_jb_buf + PJ_LOG_MAX_SIZE; 284 pj_ssize_t len = 0; 285 const char* ft_st; 286 287 if (!TRACE_JB_OPENED(stream)) 288 return; 289 290 /* Print timestamp. */ 291 if (trace_jb_print_timestamp(&p, endp-p)) 292 goto on_insuff_buffer; 293 294 /* Print frame type and size */ 295 switch(ft) { 296 case PJMEDIA_JB_MISSING_FRAME: 297 ft_st = "missing"; 298 break; 299 case PJMEDIA_JB_NORMAL_FRAME: 300 ft_st = "normal"; 301 break; 302 case PJMEDIA_JB_ZERO_PREFETCH_FRAME: 303 ft_st = "prefetch"; 304 break; 305 case PJMEDIA_JB_ZERO_EMPTY_FRAME: 306 ft_st = "empty"; 307 break; 308 default: 309 ft_st = "unknown"; 310 break; 311 } 312 313 /* Print operation, size, frame count, frame type */ 314 len = pj_ansi_snprintf(p, endp-p, "GET,%d,1,%s,,,,", fsize, ft_st); 315 if ((len < 0) || (len >= endp-p)) 316 goto on_insuff_buffer; 317 p += len; 318 319 /* Print JB state */ 320 if (trace_jb_print_state(stream, &p, endp-p)) 321 goto on_insuff_buffer; 322 323 /* Print end of line */ 324 if (endp-p < 2) 325 goto on_insuff_buffer; 326 *p++ = '\n'; 327 328 /* Write and flush */ 329 len = p - stream->trace_jb_buf; 330 pj_file_write(stream->trace_jb_fd, stream->trace_jb_buf, &len); 331 pj_file_flush(stream->trace_jb_fd); 332 return; 333 334 on_insuff_buffer: 335 pj_assert(!"Trace buffer too small, check PJ_LOG_MAX_SIZE!"); 336 } 337 338 static void trace_jb_put(pjmedia_stream *stream, const pjmedia_rtp_hdr *hdr, 339 unsigned payloadlen, unsigned frame_cnt) 340 { 341 char *p = stream->trace_jb_buf; 342 char *endp = stream->trace_jb_buf + PJ_LOG_MAX_SIZE; 343 pj_ssize_t len = 0; 344 345 if (!TRACE_JB_OPENED(stream)) 346 return; 347 348 /* Print timestamp. */ 349 if (trace_jb_print_timestamp(&p, endp-p)) 350 goto on_insuff_buffer; 351 352 /* Print operation, size, frame count, RTP info */ 353 len = pj_ansi_snprintf(p, endp-p, 354 "PUT,%d,%d,,%d,%d,%d,", 355 payloadlen, frame_cnt, 356 pj_ntohs(hdr->seq), pj_ntohl(hdr->ts), hdr->m); 357 if ((len < 0) || (len >= endp-p)) 358 goto on_insuff_buffer; 359 p += len; 360 361 /* Print JB state */ 362 if (trace_jb_print_state(stream, &p, endp-p)) 363 goto on_insuff_buffer; 364 365 /* Print end of line */ 366 if (endp-p < 2) 367 goto on_insuff_buffer; 368 *p++ = '\n'; 369 370 /* Write and flush */ 371 len = p - stream->trace_jb_buf; 372 pj_file_write(stream->trace_jb_fd, stream->trace_jb_buf, &len); 373 pj_file_flush(stream->trace_jb_fd); 374 return; 375 376 on_insuff_buffer: 377 pj_assert(!"Trace buffer too small, check PJ_LOG_MAX_SIZE!"); 378 } 379 380 #endif /* TRACE_JB */ 381 210 382 211 383 #if defined(PJMEDIA_STREAM_ENABLE_KA) && PJMEDIA_STREAM_ENABLE_KA != 0 … … 301 473 pjmedia_jbuf_get_frame2(stream->jb, channel->out_pkt, &frame_size, 302 474 &frame_type, &bit_info); 303 475 476 #if TRACE_JB 477 trace_jb_get(stream, frame_type, frame_size); 478 #endif 479 304 480 if (frame_type == PJMEDIA_JB_MISSING_FRAME) { 305 481 … … 554 730 pjmedia_jbuf_get_frame2(stream->jb, channel->out_pkt, &frame_size, 555 731 &frame_type, &bit_info); 732 733 #if TRACE_JB 734 trace_jb_get(stream, frame_type, frame_size); 735 #endif 556 736 557 737 /* Unlock jitter buffer mutex. */ … … 1569 1749 pkt_discarded = PJ_TRUE; 1570 1750 } 1751 1752 #if TRACE_JB 1753 trace_jb_put(stream, hdr, payloadlen, count); 1754 #endif 1755 1571 1756 } 1572 1757 pj_mutex_unlock( stream->jb_mutex ); … … 2079 2264 #endif 2080 2265 2266 #if TRACE_JB 2267 { 2268 char trace_name[PJ_MAXPATH]; 2269 pj_ssize_t len; 2270 2271 pj_ansi_snprintf(trace_name, sizeof(trace_name), 2272 TRACE_JB_PATH_PREFIX "%s.csv", 2273 stream->port.info.name.ptr); 2274 status = pj_file_open(pool, trace_name, PJ_O_RDWR, &stream->trace_jb_fd); 2275 if (status != PJ_SUCCESS) { 2276 stream->trace_jb_fd = TRACE_JB_INVALID_FD; 2277 PJ_LOG(3,(THIS_FILE, "Failed creating RTP trace file '%s'", 2278 trace_name)); 2279 } else { 2280 stream->trace_jb_buf = (char*)pj_pool_alloc(pool, PJ_LOG_MAX_SIZE); 2281 2282 /* Print column header */ 2283 len = pj_ansi_snprintf(stream->trace_jb_buf, PJ_LOG_MAX_SIZE, 2284 "Time, Operation, Size, Frame Count, " 2285 "Frame type, RTP Seq, RTP TS, RTP M, " 2286 "JB size, JB burst level, JB prefetch\n"); 2287 pj_file_write(stream->trace_jb_fd, stream->trace_jb_buf, &len); 2288 pj_file_flush(stream->trace_jb_fd); 2289 } 2290 } 2291 #endif 2292 2081 2293 /* Success! */ 2082 2294 *p_stream = stream; … … 2181 2393 if (stream->jb) 2182 2394 pjmedia_jbuf_destroy(stream->jb); 2395 2396 #if TRACE_JB 2397 if (TRACE_JB_OPENED(stream)) { 2398 pj_file_close(stream->trace_jb_fd); 2399 stream->trace_jb_fd = TRACE_JB_INVALID_FD; 2400 } 2401 #endif 2183 2402 2184 2403 return PJ_SUCCESS;
Note: See TracChangeset
for help on using the changeset viewer.