Mercurial > libavformat.hg
comparison rtmppkt.c @ 5432:f282f7758d6e libavformat
Dump RTMP packet contents in debug mode
author | kostya |
---|---|
date | Fri, 11 Dec 2009 17:13:35 +0000 |
parents | 584c1ba93552 |
children | c00ff770b4fc |
comparison
equal
deleted
inserted
replaced
5431:b0eb87793222 | 5432:f282f7758d6e |
---|---|
320 return -1; | 320 return -1; |
321 data += len; | 321 data += len; |
322 } | 322 } |
323 return -1; | 323 return -1; |
324 } | 324 } |
325 | |
326 static const char* rtmp_packet_type(int type) | |
327 { | |
328 switch (type) { | |
329 case RTMP_PT_CHUNK_SIZE: return "chunk size"; | |
330 case RTMP_PT_BYTES_READ: return "bytes read"; | |
331 case RTMP_PT_PING: return "ping"; | |
332 case RTMP_PT_SERVER_BW: return "server bandwidth"; | |
333 case RTMP_PT_CLIENT_BW: return "client bandwidth"; | |
334 case RTMP_PT_AUDIO: return "audio packet"; | |
335 case RTMP_PT_VIDEO: return "video packet"; | |
336 case RTMP_PT_FLEX_STREAM: return "Flex shared stream"; | |
337 case RTMP_PT_FLEX_OBJECT: return "Flex shared object"; | |
338 case RTMP_PT_FLEX_MESSAGE: return "Flex shared message"; | |
339 case RTMP_PT_NOTIFY: return "notification"; | |
340 case RTMP_PT_SHARED_OBJ: return "shared object"; | |
341 case RTMP_PT_INVOKE: return "invoke"; | |
342 case RTMP_PT_METADATA: return "metadata"; | |
343 default: return "unknown"; | |
344 } | |
345 } | |
346 | |
347 static void ff_amf_tag_contents(void *ctx, const uint8_t *data, const uint8_t *data_end) | |
348 { | |
349 const uint8_t *base = data; | |
350 int i, size; | |
351 char buf[1024]; | |
352 | |
353 if (data >= data_end) | |
354 return; | |
355 switch (*data++) { | |
356 case AMF_DATA_TYPE_NUMBER: | |
357 av_log(ctx, AV_LOG_DEBUG, " number %g\n", av_int2dbl(AV_RB64(data))); | |
358 return; | |
359 case AMF_DATA_TYPE_BOOL: | |
360 av_log(ctx, AV_LOG_DEBUG, " bool %d\n", *data); | |
361 return; | |
362 case AMF_DATA_TYPE_STRING: | |
363 case AMF_DATA_TYPE_LONG_STRING: | |
364 if (data[-1] == AMF_DATA_TYPE_STRING) { | |
365 size = bytestream_get_be16(&data); | |
366 } else { | |
367 size = bytestream_get_be32(data); | |
368 } | |
369 size = FFMIN(size, 1023); | |
370 memcpy(buf, data, size); | |
371 buf[size] = 0; | |
372 av_log(ctx, AV_LOG_DEBUG, " string '%s'\n", buf); | |
373 return; | |
374 case AMF_DATA_TYPE_NULL: | |
375 av_log(ctx, AV_LOG_DEBUG, " NULL\n"); | |
376 return; | |
377 case AMF_DATA_TYPE_ARRAY: | |
378 data += 4; | |
379 case AMF_DATA_TYPE_OBJECT: | |
380 av_log(ctx, AV_LOG_DEBUG, " {\n"); | |
381 for (;;) { | |
382 int size = bytestream_get_be16(&data); | |
383 int t; | |
384 memcpy(buf, data, size); | |
385 buf[size] = 0; | |
386 if (!size) { | |
387 av_log(ctx, AV_LOG_DEBUG, " }\n"); | |
388 data++; | |
389 break; | |
390 } | |
391 if (data + size >= data_end || data + size < data) | |
392 return; | |
393 data += size; | |
394 av_log(ctx, AV_LOG_DEBUG, " %s: ", buf); | |
395 ff_amf_tag_contents(ctx, data, data_end); | |
396 t = ff_amf_tag_size(data, data_end); | |
397 if (t < 0 || data + t >= data_end) | |
398 return; | |
399 data += t; | |
400 } | |
401 return; | |
402 case AMF_DATA_TYPE_OBJECT_END: | |
403 av_log(ctx, AV_LOG_DEBUG, " }\n"); | |
404 return; | |
405 default: | |
406 return; | |
407 } | |
408 } | |
409 | |
410 void ff_rtmp_packet_dump(void *ctx, RTMPPacket *p) | |
411 { | |
412 av_log(ctx, AV_LOG_DEBUG, "RTMP packet type '%s'(%d) for channel %d, timestamp %d, extra field %d size %d\n", | |
413 rtmp_packet_type(p->type), p->type, p->channel_id, p->timestamp, p->extra, p->data_size); | |
414 if (p->type == RTMP_PT_INVOKE || p->type == RTMP_PT_NOTIFY) { | |
415 uint8_t *src = p->data, *src_end = p->data + p->data_size; | |
416 while (src < src_end) { | |
417 int sz; | |
418 ff_amf_tag_contents(ctx, src, src_end); | |
419 sz = ff_amf_tag_size(src, src_end); | |
420 if (sz < 0) | |
421 break; | |
422 src += sz; | |
423 } | |
424 } else if (p->type == RTMP_PT_SERVER_BW){ | |
425 av_log(ctx, AV_LOG_DEBUG, "Server BW = %d\n", AV_RB32(p->data)); | |
426 } else if (p->type == RTMP_PT_CLIENT_BW){ | |
427 av_log(ctx, AV_LOG_DEBUG, "Client BW = %d\n", AV_RB32(p->data)); | |
428 } else if (p->type != RTMP_PT_AUDIO && p->type != RTMP_PT_VIDEO && p->type != RTMP_PT_METADATA) { | |
429 int i; | |
430 for (i = 0; i < p->data_size; i++) | |
431 av_log(ctx, AV_LOG_DEBUG, " %02X", p->data[i]); | |
432 av_log(ctx, AV_LOG_DEBUG, "\n"); | |
433 } | |
434 } |