2 * Copyright (C) 2008 Ramiro Polla
4 * This file is part of Libav.
6 * Libav is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * Libav is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with Libav; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
21 #include "libavcodec/bytestream.h"
25 #define HEADER_SIZE 24
29 * uint16_t ss; // struct size
30 * uint16_t width; // frame width
31 * uint16_t height; // frame height
32 * uint16_t ff; // keyframe + some other info(???)
33 * uint32_t size; // size of data
34 * uint32_t fourcc; // ML20
36 * uint32_t ts; // time
39 static int msnwc_tcp_probe(AVProbeData
*p
)
43 for(i
= 0 ; i
+ HEADER_SIZE
<= p
->buf_size
; i
++) {
44 uint16_t width
, height
;
46 const uint8_t *bytestream
= p
->buf
+i
;
48 if(bytestream_get_le16(&bytestream
) != HEADER_SIZE
)
50 width
= bytestream_get_le16(&bytestream
);
51 height
= bytestream_get_le16(&bytestream
);
52 if(!(width
==320 && height
==240) && !(width
==160 && height
==120))
54 bytestream
+= 2; // keyframe
55 bytestream
+= 4; // size
56 fourcc
= bytestream_get_le32(&bytestream
);
57 if(fourcc
!= MKTAG('M', 'L', '2', '0'))
61 if(i
< 14) /* starts with SwitchBoard connection info */
62 return AVPROBE_SCORE_MAX
/ 2;
63 else /* starts in the middle of stream */
64 return AVPROBE_SCORE_MAX
/ 3;
66 return AVPROBE_SCORE_MAX
;
73 static int msnwc_tcp_read_header(AVFormatContext
*ctx
)
75 AVIOContext
*pb
= ctx
->pb
;
76 AVCodecContext
*codec
;
79 st
= avformat_new_stream(ctx
, NULL
);
81 return AVERROR(ENOMEM
);
84 codec
->codec_type
= AVMEDIA_TYPE_VIDEO
;
85 codec
->codec_id
= AV_CODEC_ID_MIMIC
;
86 codec
->codec_tag
= MKTAG('M', 'L', '2', '0');
88 avpriv_set_pts_info(st
, 32, 1, 1000);
90 /* Some files start with "connected\r\n\r\n".
91 * So skip until we find the first byte of struct size */
92 while(avio_r8(pb
) != HEADER_SIZE
&& !pb
->eof_reached
);
95 av_log(ctx
, AV_LOG_ERROR
, "Could not find valid start.");
102 static int msnwc_tcp_read_packet(AVFormatContext
*ctx
, AVPacket
*pkt
)
104 AVIOContext
*pb
= ctx
->pb
;
106 uint32_t size
, timestamp
;
108 avio_skip(pb
, 1); /* one byte has been read ahead */
111 keyframe
= avio_rl16(pb
);
112 size
= avio_rl32(pb
);
115 timestamp
= avio_rl32(pb
);
117 if(!size
|| av_get_packet(pb
, pkt
, size
) != size
)
120 avio_skip(pb
, 1); /* Read ahead one byte of struct size like read_header */
122 pkt
->pts
= timestamp
;
123 pkt
->dts
= timestamp
;
124 pkt
->stream_index
= 0;
126 /* Some aMsn generated videos (or was it Mercury Messenger?) don't set
127 * this bit and rely on the codec to get keyframe information */
129 pkt
->flags
|= AV_PKT_FLAG_KEY
;
131 return HEADER_SIZE
+ size
;
134 AVInputFormat ff_msnwc_tcp_demuxer
= {
136 .long_name
= NULL_IF_CONFIG_SMALL("MSN TCP Webcam stream"),
137 .read_probe
= msnwc_tcp_probe
,
138 .read_header
= msnwc_tcp_read_header
,
139 .read_packet
= msnwc_tcp_read_packet
,