]> git.sesse.net Git - ffmpeg/blob - libavformat/aviobuf.c
avio: avio: avio_ prefixes for put_* functions
[ffmpeg] / libavformat / aviobuf.c
1 /*
2  * Buffered I/O for ffmpeg system
3  * Copyright (c) 2000,2001 Fabrice Bellard
4  *
5  * This file is part of FFmpeg.
6  *
7  * FFmpeg is free software; you can redistribute it and/or
8  * modify it under the terms of the GNU Lesser General Public
9  * License as published by the Free Software Foundation; either
10  * version 2.1 of the License, or (at your option) any later version.
11  *
12  * FFmpeg is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15  * Lesser General Public License for more details.
16  *
17  * You should have received a copy of the GNU Lesser General Public
18  * License along with FFmpeg; if not, write to the Free Software
19  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
20  */
21
22 #include "libavutil/crc.h"
23 #include "libavutil/intreadwrite.h"
24 #include "avformat.h"
25 #include "avio.h"
26 #include "avio_internal.h"
27 #include "internal.h"
28 #include <stdarg.h>
29
30 #define IO_BUFFER_SIZE 32768
31
32 /**
33  * Do seeks within this distance ahead of the current buffer by skipping
34  * data instead of calling the protocol seek function, for seekable
35  * protocols.
36  */
37 #define SHORT_SEEK_THRESHOLD 4096
38
39 static void fill_buffer(AVIOContext *s);
40 #if !FF_API_URL_RESETBUF
41 static int url_resetbuf(AVIOContext *s, int flags);
42 #endif
43
44 int ffio_init_context(AVIOContext *s,
45                   unsigned char *buffer,
46                   int buffer_size,
47                   int write_flag,
48                   void *opaque,
49                   int (*read_packet)(void *opaque, uint8_t *buf, int buf_size),
50                   int (*write_packet)(void *opaque, uint8_t *buf, int buf_size),
51                   int64_t (*seek)(void *opaque, int64_t offset, int whence))
52 {
53     s->buffer = buffer;
54     s->buffer_size = buffer_size;
55     s->buf_ptr = buffer;
56     s->opaque = opaque;
57     url_resetbuf(s, write_flag ? URL_WRONLY : URL_RDONLY);
58     s->write_packet = write_packet;
59     s->read_packet = read_packet;
60     s->seek = seek;
61     s->pos = 0;
62     s->must_flush = 0;
63     s->eof_reached = 0;
64     s->error = 0;
65     s->is_streamed = 0;
66     s->max_packet_size = 0;
67     s->update_checksum= NULL;
68     if(!read_packet && !write_flag){
69         s->pos = buffer_size;
70         s->buf_end = s->buffer + buffer_size;
71     }
72     s->read_pause = NULL;
73     s->read_seek  = NULL;
74     return 0;
75 }
76
77 #if FF_API_OLD_AVIO
78 int init_put_byte(AVIOContext *s,
79                   unsigned char *buffer,
80                   int buffer_size,
81                   int write_flag,
82                   void *opaque,
83                   int (*read_packet)(void *opaque, uint8_t *buf, int buf_size),
84                   int (*write_packet)(void *opaque, uint8_t *buf, int buf_size),
85                   int64_t (*seek)(void *opaque, int64_t offset, int whence))
86 {
87     return ffio_init_context(s, buffer, buffer_size, write_flag, opaque,
88                                 read_packet, write_packet, seek);
89 }
90 AVIOContext *av_alloc_put_byte(
91                   unsigned char *buffer,
92                   int buffer_size,
93                   int write_flag,
94                   void *opaque,
95                   int (*read_packet)(void *opaque, uint8_t *buf, int buf_size),
96                   int (*write_packet)(void *opaque, uint8_t *buf, int buf_size),
97                   int64_t (*seek)(void *opaque, int64_t offset, int whence))
98 {
99     return avio_alloc_context(buffer, buffer_size, write_flag, opaque,
100                               read_packet, write_packet, seek);
101 }
102 #endif
103
104 AVIOContext *avio_alloc_context(
105                   unsigned char *buffer,
106                   int buffer_size,
107                   int write_flag,
108                   void *opaque,
109                   int (*read_packet)(void *opaque, uint8_t *buf, int buf_size),
110                   int (*write_packet)(void *opaque, uint8_t *buf, int buf_size),
111                   int64_t (*seek)(void *opaque, int64_t offset, int whence))
112 {
113     AVIOContext *s = av_mallocz(sizeof(AVIOContext));
114     ffio_init_context(s, buffer, buffer_size, write_flag, opaque,
115                   read_packet, write_packet, seek);
116     return s;
117 }
118
119 static void flush_buffer(AVIOContext *s)
120 {
121     if (s->buf_ptr > s->buffer) {
122         if (s->write_packet && !s->error){
123             int ret= s->write_packet(s->opaque, s->buffer, s->buf_ptr - s->buffer);
124             if(ret < 0){
125                 s->error = ret;
126             }
127         }
128         if(s->update_checksum){
129             s->checksum= s->update_checksum(s->checksum, s->checksum_ptr, s->buf_ptr - s->checksum_ptr);
130             s->checksum_ptr= s->buffer;
131         }
132         s->pos += s->buf_ptr - s->buffer;
133     }
134     s->buf_ptr = s->buffer;
135 }
136
137 void avio_w8(AVIOContext *s, int b)
138 {
139     *(s->buf_ptr)++ = b;
140     if (s->buf_ptr >= s->buf_end)
141         flush_buffer(s);
142 }
143
144 void put_nbyte(AVIOContext *s, int b, int count)
145 {
146     while (count > 0) {
147         int len = FFMIN(s->buf_end - s->buf_ptr, count);
148         memset(s->buf_ptr, b, len);
149         s->buf_ptr += len;
150
151         if (s->buf_ptr >= s->buf_end)
152             flush_buffer(s);
153
154         count -= len;
155     }
156 }
157
158 void avio_write(AVIOContext *s, const unsigned char *buf, int size)
159 {
160     while (size > 0) {
161         int len = FFMIN(s->buf_end - s->buf_ptr, size);
162         memcpy(s->buf_ptr, buf, len);
163         s->buf_ptr += len;
164
165         if (s->buf_ptr >= s->buf_end)
166             flush_buffer(s);
167
168         buf += len;
169         size -= len;
170     }
171 }
172
173 void put_flush_packet(AVIOContext *s)
174 {
175     flush_buffer(s);
176     s->must_flush = 0;
177 }
178
179 int64_t url_fseek(AVIOContext *s, int64_t offset, int whence)
180 {
181     int64_t offset1;
182     int64_t pos;
183     int force = whence & AVSEEK_FORCE;
184     whence &= ~AVSEEK_FORCE;
185
186     if(!s)
187         return AVERROR(EINVAL);
188
189     pos = s->pos - (s->write_flag ? 0 : (s->buf_end - s->buffer));
190
191     if (whence != SEEK_CUR && whence != SEEK_SET)
192         return AVERROR(EINVAL);
193
194     if (whence == SEEK_CUR) {
195         offset1 = pos + (s->buf_ptr - s->buffer);
196         if (offset == 0)
197             return offset1;
198         offset += offset1;
199     }
200     offset1 = offset - pos;
201     if (!s->must_flush &&
202         offset1 >= 0 && offset1 <= (s->buf_end - s->buffer)) {
203         /* can do the seek inside the buffer */
204         s->buf_ptr = s->buffer + offset1;
205     } else if ((s->is_streamed ||
206                offset1 <= s->buf_end + SHORT_SEEK_THRESHOLD - s->buffer) &&
207                !s->write_flag && offset1 >= 0 &&
208               (whence != SEEK_END || force)) {
209         while(s->pos < offset && !s->eof_reached)
210             fill_buffer(s);
211         if (s->eof_reached)
212             return AVERROR_EOF;
213         s->buf_ptr = s->buf_end + offset - s->pos;
214     } else {
215         int64_t res;
216
217 #if CONFIG_MUXERS || CONFIG_NETWORK
218         if (s->write_flag) {
219             flush_buffer(s);
220             s->must_flush = 1;
221         }
222 #endif /* CONFIG_MUXERS || CONFIG_NETWORK */
223         if (!s->seek)
224             return AVERROR(EPIPE);
225         if ((res = s->seek(s->opaque, offset, SEEK_SET)) < 0)
226             return res;
227         if (!s->write_flag)
228             s->buf_end = s->buffer;
229         s->buf_ptr = s->buffer;
230         s->pos = offset;
231     }
232     s->eof_reached = 0;
233     return offset;
234 }
235
236 int url_fskip(AVIOContext *s, int64_t offset)
237 {
238     int64_t ret = url_fseek(s, offset, SEEK_CUR);
239     return ret < 0 ? ret : 0;
240 }
241
242 int64_t url_ftell(AVIOContext *s)
243 {
244     return url_fseek(s, 0, SEEK_CUR);
245 }
246
247 int64_t url_fsize(AVIOContext *s)
248 {
249     int64_t size;
250
251     if(!s)
252         return AVERROR(EINVAL);
253
254     if (!s->seek)
255         return AVERROR(ENOSYS);
256     size = s->seek(s->opaque, 0, AVSEEK_SIZE);
257     if(size<0){
258         if ((size = s->seek(s->opaque, -1, SEEK_END)) < 0)
259             return size;
260         size++;
261         s->seek(s->opaque, s->pos, SEEK_SET);
262     }
263     return size;
264 }
265
266 int url_feof(AVIOContext *s)
267 {
268     if(!s)
269         return 0;
270     return s->eof_reached;
271 }
272
273 int url_ferror(AVIOContext *s)
274 {
275     if(!s)
276         return 0;
277     return s->error;
278 }
279
280 void avio_wl32(AVIOContext *s, unsigned int val)
281 {
282     avio_w8(s, val);
283     avio_w8(s, val >> 8);
284     avio_w8(s, val >> 16);
285     avio_w8(s, val >> 24);
286 }
287
288 void avio_wb32(AVIOContext *s, unsigned int val)
289 {
290     avio_w8(s, val >> 24);
291     avio_w8(s, val >> 16);
292     avio_w8(s, val >> 8);
293     avio_w8(s, val);
294 }
295
296 #if FF_API_OLD_AVIO
297 void put_strz(AVIOContext *s, const char *str)
298 {
299     avio_put_str(s, str);
300 }
301
302 #define GET(name, type) \
303     type get_be ##name(AVIOContext *s) \
304 {\
305     return avio_rb ##name(s);\
306 }\
307     type get_le ##name(AVIOContext *s) \
308 {\
309     return avio_rl ##name(s);\
310 }
311
312 GET(16, unsigned int)
313 GET(24, unsigned int)
314 GET(32, unsigned int)
315 GET(64, uint64_t)
316
317 #undef GET
318
319 #define PUT(name, type ) \
320     void put_le ##name(AVIOContext *s, type val)\
321 {\
322         avio_wl ##name(s, val);\
323 }\
324     void put_be ##name(AVIOContext *s, type val)\
325 {\
326         avio_wb ##name(s, val);\
327 }
328
329 PUT(16, unsigned int)
330 PUT(24, unsigned int)
331 PUT(32, unsigned int)
332 PUT(64, uint64_t)
333 #undef PUT
334
335 int get_byte(AVIOContext *s)
336 {
337    return avio_r8(s);
338 }
339 int get_buffer(AVIOContext *s, unsigned char *buf, int size)
340 {
341     return avio_read(s, buf, size);
342 }
343 int get_partial_buffer(AVIOContext *s, unsigned char *buf, int size)
344 {
345     return ffio_read_partial(s, buf, size);
346 }
347 void put_byte(AVIOContext *s, int val)
348 {
349     avio_w8(s, val);
350 }
351 void put_buffer(AVIOContext *s, const unsigned char *buf, int size)
352 {
353     avio_write(s, buf, size);
354 }
355 #endif
356
357 int avio_put_str(AVIOContext *s, const char *str)
358 {
359     int len = 1;
360     if (str) {
361         len += strlen(str);
362         avio_write(s, (const unsigned char *) str, len);
363     } else
364         avio_w8(s, 0);
365     return len;
366 }
367
368 int avio_put_str16le(AVIOContext *s, const char *str)
369 {
370     const uint8_t *q = str;
371     int ret = 0;
372
373     while (*q) {
374         uint32_t ch;
375         uint16_t tmp;
376
377         GET_UTF8(ch, *q++, break;)
378         PUT_UTF16(ch, tmp, avio_wl16(s, tmp);ret += 2;)
379     }
380     avio_wl16(s, 0);
381     ret += 2;
382     return ret;
383 }
384
385 int ff_get_v_length(uint64_t val){
386     int i=1;
387
388     while(val>>=7)
389         i++;
390
391     return i;
392 }
393
394 void ff_put_v(AVIOContext *bc, uint64_t val){
395     int i= ff_get_v_length(val);
396
397     while(--i>0)
398         avio_w8(bc, 128 | (val>>(7*i)));
399
400     avio_w8(bc, val&127);
401 }
402
403 void avio_wl64(AVIOContext *s, uint64_t val)
404 {
405     avio_wl32(s, (uint32_t)(val & 0xffffffff));
406     avio_wl32(s, (uint32_t)(val >> 32));
407 }
408
409 void avio_wb64(AVIOContext *s, uint64_t val)
410 {
411     avio_wb32(s, (uint32_t)(val >> 32));
412     avio_wb32(s, (uint32_t)(val & 0xffffffff));
413 }
414
415 void avio_wl16(AVIOContext *s, unsigned int val)
416 {
417     avio_w8(s, val);
418     avio_w8(s, val >> 8);
419 }
420
421 void avio_wb16(AVIOContext *s, unsigned int val)
422 {
423     avio_w8(s, val >> 8);
424     avio_w8(s, val);
425 }
426
427 void avio_wl24(AVIOContext *s, unsigned int val)
428 {
429     avio_wl16(s, val & 0xffff);
430     avio_w8(s, val >> 16);
431 }
432
433 void avio_wb24(AVIOContext *s, unsigned int val)
434 {
435     avio_wb16(s, val >> 8);
436     avio_w8(s, val);
437 }
438
439 void put_tag(AVIOContext *s, const char *tag)
440 {
441     while (*tag) {
442         avio_w8(s, *tag++);
443     }
444 }
445
446 /* Input stream */
447
448 static void fill_buffer(AVIOContext *s)
449 {
450     uint8_t *dst= !s->max_packet_size && s->buf_end - s->buffer < s->buffer_size ? s->buf_ptr : s->buffer;
451     int len= s->buffer_size - (dst - s->buffer);
452     int max_buffer_size = s->max_packet_size ? s->max_packet_size : IO_BUFFER_SIZE;
453
454     /* no need to do anything if EOF already reached */
455     if (s->eof_reached)
456         return;
457
458     if(s->update_checksum && dst == s->buffer){
459         if(s->buf_end > s->checksum_ptr)
460             s->checksum= s->update_checksum(s->checksum, s->checksum_ptr, s->buf_end - s->checksum_ptr);
461         s->checksum_ptr= s->buffer;
462     }
463
464     /* make buffer smaller in case it ended up large after probing */
465     if (s->buffer_size > max_buffer_size) {
466         url_setbufsize(s, max_buffer_size);
467
468         s->checksum_ptr = dst = s->buffer;
469         len = s->buffer_size;
470     }
471
472     if(s->read_packet)
473         len = s->read_packet(s->opaque, dst, len);
474     else
475         len = 0;
476     if (len <= 0) {
477         /* do not modify buffer if EOF reached so that a seek back can
478            be done without rereading data */
479         s->eof_reached = 1;
480         if(len<0)
481             s->error= len;
482     } else {
483         s->pos += len;
484         s->buf_ptr = dst;
485         s->buf_end = dst + len;
486     }
487 }
488
489 unsigned long ff_crc04C11DB7_update(unsigned long checksum, const uint8_t *buf,
490                                     unsigned int len)
491 {
492     return av_crc(av_crc_get_table(AV_CRC_32_IEEE), checksum, buf, len);
493 }
494
495 unsigned long get_checksum(AVIOContext *s)
496 {
497     s->checksum= s->update_checksum(s->checksum, s->checksum_ptr, s->buf_ptr - s->checksum_ptr);
498     s->update_checksum= NULL;
499     return s->checksum;
500 }
501
502 void init_checksum(AVIOContext *s,
503                    unsigned long (*update_checksum)(unsigned long c, const uint8_t *p, unsigned int len),
504                    unsigned long checksum)
505 {
506     s->update_checksum= update_checksum;
507     if(s->update_checksum){
508         s->checksum= checksum;
509         s->checksum_ptr= s->buf_ptr;
510     }
511 }
512
513 /* XXX: put an inline version */
514 int avio_r8(AVIOContext *s)
515 {
516     if (s->buf_ptr >= s->buf_end)
517         fill_buffer(s);
518     if (s->buf_ptr < s->buf_end)
519         return *s->buf_ptr++;
520     return 0;
521 }
522
523 int url_fgetc(AVIOContext *s)
524 {
525     if (s->buf_ptr >= s->buf_end)
526         fill_buffer(s);
527     if (s->buf_ptr < s->buf_end)
528         return *s->buf_ptr++;
529     return URL_EOF;
530 }
531
532 int avio_read(AVIOContext *s, unsigned char *buf, int size)
533 {
534     int len, size1;
535
536     size1 = size;
537     while (size > 0) {
538         len = s->buf_end - s->buf_ptr;
539         if (len > size)
540             len = size;
541         if (len == 0) {
542             if(size > s->buffer_size && !s->update_checksum){
543                 if(s->read_packet)
544                     len = s->read_packet(s->opaque, buf, size);
545                 if (len <= 0) {
546                     /* do not modify buffer if EOF reached so that a seek back can
547                     be done without rereading data */
548                     s->eof_reached = 1;
549                     if(len<0)
550                         s->error= len;
551                     break;
552                 } else {
553                     s->pos += len;
554                     size -= len;
555                     buf += len;
556                     s->buf_ptr = s->buffer;
557                     s->buf_end = s->buffer/* + len*/;
558                 }
559             }else{
560                 fill_buffer(s);
561                 len = s->buf_end - s->buf_ptr;
562                 if (len == 0)
563                     break;
564             }
565         } else {
566             memcpy(buf, s->buf_ptr, len);
567             buf += len;
568             s->buf_ptr += len;
569             size -= len;
570         }
571     }
572     if (size1 == size) {
573         if (url_ferror(s)) return url_ferror(s);
574         if (url_feof(s))   return AVERROR_EOF;
575     }
576     return size1 - size;
577 }
578
579 int ffio_read_partial(AVIOContext *s, unsigned char *buf, int size)
580 {
581     int len;
582
583     if(size<0)
584         return -1;
585
586     len = s->buf_end - s->buf_ptr;
587     if (len == 0) {
588         fill_buffer(s);
589         len = s->buf_end - s->buf_ptr;
590     }
591     if (len > size)
592         len = size;
593     memcpy(buf, s->buf_ptr, len);
594     s->buf_ptr += len;
595     if (!len) {
596         if (url_ferror(s)) return url_ferror(s);
597         if (url_feof(s))   return AVERROR_EOF;
598     }
599     return len;
600 }
601
602 unsigned int avio_rl16(AVIOContext *s)
603 {
604     unsigned int val;
605     val = avio_r8(s);
606     val |= avio_r8(s) << 8;
607     return val;
608 }
609
610 unsigned int avio_rl24(AVIOContext *s)
611 {
612     unsigned int val;
613     val = avio_rl16(s);
614     val |= avio_r8(s) << 16;
615     return val;
616 }
617
618 unsigned int avio_rl32(AVIOContext *s)
619 {
620     unsigned int val;
621     val = avio_rl16(s);
622     val |= avio_rl16(s) << 16;
623     return val;
624 }
625
626 uint64_t avio_rl64(AVIOContext *s)
627 {
628     uint64_t val;
629     val = (uint64_t)avio_rl32(s);
630     val |= (uint64_t)avio_rl32(s) << 32;
631     return val;
632 }
633
634 unsigned int avio_rb16(AVIOContext *s)
635 {
636     unsigned int val;
637     val = avio_r8(s) << 8;
638     val |= avio_r8(s);
639     return val;
640 }
641
642 unsigned int avio_rb24(AVIOContext *s)
643 {
644     unsigned int val;
645     val = avio_rb16(s) << 8;
646     val |= avio_r8(s);
647     return val;
648 }
649 unsigned int avio_rb32(AVIOContext *s)
650 {
651     unsigned int val;
652     val = avio_rb16(s) << 16;
653     val |= avio_rb16(s);
654     return val;
655 }
656
657 char *get_strz(AVIOContext *s, char *buf, int maxlen)
658 {
659     int i = 0;
660     char c;
661
662     while ((c = avio_r8(s))) {
663         if (i < maxlen-1)
664             buf[i++] = c;
665     }
666
667     buf[i] = 0; /* Ensure null terminated, but may be truncated */
668
669     return buf;
670 }
671
672 int ff_get_line(AVIOContext *s, char *buf, int maxlen)
673 {
674     int i = 0;
675     char c;
676
677     do {
678         c = avio_r8(s);
679         if (c && i < maxlen-1)
680             buf[i++] = c;
681     } while (c != '\n' && c);
682
683     buf[i] = 0;
684     return i;
685 }
686
687 #define GET_STR16(type, read) \
688     int avio_get_str16 ##type(AVIOContext *pb, int maxlen, char *buf, int buflen)\
689 {\
690     char* q = buf;\
691     int ret = 0;\
692     while (ret + 1 < maxlen) {\
693         uint8_t tmp;\
694         uint32_t ch;\
695         GET_UTF16(ch, (ret += 2) <= maxlen ? read(pb) : 0, break;)\
696         if (!ch)\
697             break;\
698         PUT_UTF8(ch, tmp, if (q - buf < buflen - 1) *q++ = tmp;)\
699     }\
700     *q = 0;\
701     return ret;\
702 }\
703
704 GET_STR16(le, avio_rl16)
705 GET_STR16(be, avio_rb16)
706
707 #undef GET_STR16
708
709 uint64_t avio_rb64(AVIOContext *s)
710 {
711     uint64_t val;
712     val = (uint64_t)avio_rb32(s) << 32;
713     val |= (uint64_t)avio_rb32(s);
714     return val;
715 }
716
717 uint64_t ff_get_v(AVIOContext *bc){
718     uint64_t val = 0;
719     int tmp;
720
721     do{
722         tmp = avio_r8(bc);
723         val= (val<<7) + (tmp&127);
724     }while(tmp&128);
725     return val;
726 }
727
728 int url_fdopen(AVIOContext **s, URLContext *h)
729 {
730     uint8_t *buffer;
731     int buffer_size, max_packet_size;
732
733     max_packet_size = url_get_max_packet_size(h);
734     if (max_packet_size) {
735         buffer_size = max_packet_size; /* no need to bufferize more than one packet */
736     } else {
737         buffer_size = IO_BUFFER_SIZE;
738     }
739     buffer = av_malloc(buffer_size);
740     if (!buffer)
741         return AVERROR(ENOMEM);
742
743     *s = av_mallocz(sizeof(AVIOContext));
744     if(!*s) {
745         av_free(buffer);
746         return AVERROR(ENOMEM);
747     }
748
749     if (ffio_init_context(*s, buffer, buffer_size,
750                       (h->flags & URL_WRONLY || h->flags & URL_RDWR), h,
751                       url_read, url_write, url_seek) < 0) {
752         av_free(buffer);
753         av_freep(s);
754         return AVERROR(EIO);
755     }
756     (*s)->is_streamed = h->is_streamed;
757     (*s)->max_packet_size = max_packet_size;
758     if(h->prot) {
759         (*s)->read_pause = (int (*)(void *, int))h->prot->url_read_pause;
760         (*s)->read_seek  = (int64_t (*)(void *, int, int64_t, int))h->prot->url_read_seek;
761     }
762     return 0;
763 }
764
765 int url_setbufsize(AVIOContext *s, int buf_size)
766 {
767     uint8_t *buffer;
768     buffer = av_malloc(buf_size);
769     if (!buffer)
770         return AVERROR(ENOMEM);
771
772     av_free(s->buffer);
773     s->buffer = buffer;
774     s->buffer_size = buf_size;
775     s->buf_ptr = buffer;
776     url_resetbuf(s, s->write_flag ? URL_WRONLY : URL_RDONLY);
777     return 0;
778 }
779
780 #if FF_API_URL_RESETBUF
781 int url_resetbuf(AVIOContext *s, int flags)
782 #else
783 static int url_resetbuf(AVIOContext *s, int flags)
784 #endif
785 {
786 #if FF_API_URL_RESETBUF
787     if (flags & URL_RDWR)
788         return AVERROR(EINVAL);
789 #else
790     assert(flags == URL_WRONLY || flags == URL_RDONLY);
791 #endif
792
793     if (flags & URL_WRONLY) {
794         s->buf_end = s->buffer + s->buffer_size;
795         s->write_flag = 1;
796     } else {
797         s->buf_end = s->buffer;
798         s->write_flag = 0;
799     }
800     return 0;
801 }
802
803 int ff_rewind_with_probe_data(AVIOContext *s, unsigned char *buf, int buf_size)
804 {
805     int64_t buffer_start;
806     int buffer_size;
807     int overlap, new_size, alloc_size;
808
809     if (s->write_flag)
810         return AVERROR(EINVAL);
811
812     buffer_size = s->buf_end - s->buffer;
813
814     /* the buffers must touch or overlap */
815     if ((buffer_start = s->pos - buffer_size) > buf_size)
816         return AVERROR(EINVAL);
817
818     overlap = buf_size - buffer_start;
819     new_size = buf_size + buffer_size - overlap;
820
821     alloc_size = FFMAX(s->buffer_size, new_size);
822     if (alloc_size > buf_size)
823         if (!(buf = av_realloc(buf, alloc_size)))
824             return AVERROR(ENOMEM);
825
826     if (new_size > buf_size) {
827         memcpy(buf + buf_size, s->buffer + overlap, buffer_size - overlap);
828         buf_size = new_size;
829     }
830
831     av_free(s->buffer);
832     s->buf_ptr = s->buffer = buf;
833     s->buffer_size = alloc_size;
834     s->pos = buf_size;
835     s->buf_end = s->buf_ptr + buf_size;
836     s->eof_reached = 0;
837     s->must_flush = 0;
838
839     return 0;
840 }
841
842 int url_fopen(AVIOContext **s, const char *filename, int flags)
843 {
844     URLContext *h;
845     int err;
846
847     err = url_open(&h, filename, flags);
848     if (err < 0)
849         return err;
850     err = url_fdopen(s, h);
851     if (err < 0) {
852         url_close(h);
853         return err;
854     }
855     return 0;
856 }
857
858 int url_fclose(AVIOContext *s)
859 {
860     URLContext *h = s->opaque;
861
862     av_free(s->buffer);
863     av_free(s);
864     return url_close(h);
865 }
866
867 URLContext *url_fileno(AVIOContext *s)
868 {
869     return s->opaque;
870 }
871
872 #if CONFIG_MUXERS
873 int url_fprintf(AVIOContext *s, const char *fmt, ...)
874 {
875     va_list ap;
876     char buf[4096];
877     int ret;
878
879     va_start(ap, fmt);
880     ret = vsnprintf(buf, sizeof(buf), fmt, ap);
881     va_end(ap);
882     avio_write(s, buf, strlen(buf));
883     return ret;
884 }
885 #endif //CONFIG_MUXERS
886
887 char *url_fgets(AVIOContext *s, char *buf, int buf_size)
888 {
889     int c;
890     char *q;
891
892     c = url_fgetc(s);
893     if (c == EOF)
894         return NULL;
895     q = buf;
896     for(;;) {
897         if (c == EOF || c == '\n')
898             break;
899         if ((q - buf) < buf_size - 1)
900             *q++ = c;
901         c = url_fgetc(s);
902     }
903     if (buf_size > 0)
904         *q = '\0';
905     return buf;
906 }
907
908 int url_fget_max_packet_size(AVIOContext *s)
909 {
910     return s->max_packet_size;
911 }
912
913 int av_url_read_fpause(AVIOContext *s, int pause)
914 {
915     if (!s->read_pause)
916         return AVERROR(ENOSYS);
917     return s->read_pause(s->opaque, pause);
918 }
919
920 int64_t av_url_read_fseek(AVIOContext *s, int stream_index,
921                           int64_t timestamp, int flags)
922 {
923     URLContext *h = s->opaque;
924     int64_t ret;
925     if (!s->read_seek)
926         return AVERROR(ENOSYS);
927     ret = s->read_seek(h, stream_index, timestamp, flags);
928     if(ret >= 0) {
929         int64_t pos;
930         s->buf_ptr = s->buf_end; // Flush buffer
931         pos = s->seek(h, 0, SEEK_CUR);
932         if (pos >= 0)
933             s->pos = pos;
934         else if (pos != AVERROR(ENOSYS))
935             ret = pos;
936     }
937     return ret;
938 }
939
940 /* url_open_dyn_buf and url_close_dyn_buf are used in rtp.c to send a response
941  * back to the server even if CONFIG_MUXERS is false. */
942 #if CONFIG_MUXERS || CONFIG_NETWORK
943 /* buffer handling */
944 int url_open_buf(AVIOContext **s, uint8_t *buf, int buf_size, int flags)
945 {
946     int ret;
947     *s = av_mallocz(sizeof(AVIOContext));
948     if(!*s)
949         return AVERROR(ENOMEM);
950     ret = ffio_init_context(*s, buf, buf_size,
951                         (flags & URL_WRONLY || flags & URL_RDWR),
952                         NULL, NULL, NULL, NULL);
953     if(ret != 0)
954         av_freep(s);
955     return ret;
956 }
957
958 int url_close_buf(AVIOContext *s)
959 {
960     put_flush_packet(s);
961     return s->buf_ptr - s->buffer;
962 }
963
964 /* output in a dynamic buffer */
965
966 typedef struct DynBuffer {
967     int pos, size, allocated_size;
968     uint8_t *buffer;
969     int io_buffer_size;
970     uint8_t io_buffer[1];
971 } DynBuffer;
972
973 static int dyn_buf_write(void *opaque, uint8_t *buf, int buf_size)
974 {
975     DynBuffer *d = opaque;
976     unsigned new_size, new_allocated_size;
977
978     /* reallocate buffer if needed */
979     new_size = d->pos + buf_size;
980     new_allocated_size = d->allocated_size;
981     if(new_size < d->pos || new_size > INT_MAX/2)
982         return -1;
983     while (new_size > new_allocated_size) {
984         if (!new_allocated_size)
985             new_allocated_size = new_size;
986         else
987             new_allocated_size += new_allocated_size / 2 + 1;
988     }
989
990     if (new_allocated_size > d->allocated_size) {
991         d->buffer = av_realloc(d->buffer, new_allocated_size);
992         if(d->buffer == NULL)
993              return AVERROR(ENOMEM);
994         d->allocated_size = new_allocated_size;
995     }
996     memcpy(d->buffer + d->pos, buf, buf_size);
997     d->pos = new_size;
998     if (d->pos > d->size)
999         d->size = d->pos;
1000     return buf_size;
1001 }
1002
1003 static int dyn_packet_buf_write(void *opaque, uint8_t *buf, int buf_size)
1004 {
1005     unsigned char buf1[4];
1006     int ret;
1007
1008     /* packetized write: output the header */
1009     AV_WB32(buf1, buf_size);
1010     ret= dyn_buf_write(opaque, buf1, 4);
1011     if(ret < 0)
1012         return ret;
1013
1014     /* then the data */
1015     return dyn_buf_write(opaque, buf, buf_size);
1016 }
1017
1018 static int64_t dyn_buf_seek(void *opaque, int64_t offset, int whence)
1019 {
1020     DynBuffer *d = opaque;
1021
1022     if (whence == SEEK_CUR)
1023         offset += d->pos;
1024     else if (whence == SEEK_END)
1025         offset += d->size;
1026     if (offset < 0 || offset > 0x7fffffffLL)
1027         return -1;
1028     d->pos = offset;
1029     return 0;
1030 }
1031
1032 static int url_open_dyn_buf_internal(AVIOContext **s, int max_packet_size)
1033 {
1034     DynBuffer *d;
1035     int ret;
1036     unsigned io_buffer_size = max_packet_size ? max_packet_size : 1024;
1037
1038     if(sizeof(DynBuffer) + io_buffer_size < io_buffer_size)
1039         return -1;
1040     d = av_mallocz(sizeof(DynBuffer) + io_buffer_size);
1041     if (!d)
1042         return AVERROR(ENOMEM);
1043     *s = av_mallocz(sizeof(AVIOContext));
1044     if(!*s) {
1045         av_free(d);
1046         return AVERROR(ENOMEM);
1047     }
1048     d->io_buffer_size = io_buffer_size;
1049     ret = ffio_init_context(*s, d->io_buffer, io_buffer_size,
1050                         1, d, NULL,
1051                         max_packet_size ? dyn_packet_buf_write : dyn_buf_write,
1052                         max_packet_size ? NULL : dyn_buf_seek);
1053     if (ret == 0) {
1054         (*s)->max_packet_size = max_packet_size;
1055     } else {
1056         av_free(d);
1057         av_freep(s);
1058     }
1059     return ret;
1060 }
1061
1062 int url_open_dyn_buf(AVIOContext **s)
1063 {
1064     return url_open_dyn_buf_internal(s, 0);
1065 }
1066
1067 int url_open_dyn_packet_buf(AVIOContext **s, int max_packet_size)
1068 {
1069     if (max_packet_size <= 0)
1070         return -1;
1071     return url_open_dyn_buf_internal(s, max_packet_size);
1072 }
1073
1074 int url_close_dyn_buf(AVIOContext *s, uint8_t **pbuffer)
1075 {
1076     DynBuffer *d = s->opaque;
1077     int size;
1078     static const char padbuf[FF_INPUT_BUFFER_PADDING_SIZE] = {0};
1079     int padding = 0;
1080
1081     /* don't attempt to pad fixed-size packet buffers */
1082     if (!s->max_packet_size) {
1083         avio_write(s, padbuf, sizeof(padbuf));
1084         padding = FF_INPUT_BUFFER_PADDING_SIZE;
1085     }
1086
1087     put_flush_packet(s);
1088
1089     *pbuffer = d->buffer;
1090     size = d->size;
1091     av_free(d);
1092     av_free(s);
1093     return size - padding;
1094 }
1095 #endif /* CONFIG_MUXERS || CONFIG_NETWORK */