]> git.sesse.net Git - ffmpeg/blob - libavformat/cache.c
avidec: Fix infinite loop caused by rounding of timestamps in non interleaved avis.
[ffmpeg] / libavformat / cache.c
1 /*
2  * Input cache protocol.
3  * Copyright (c) 2011 Michael Niedermayer
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  * Based on file.c by Fabrice Bellard
22  */
23
24 /**
25  * @TODO
26  *      support non continuous caching
27  *      support keeping files
28  *      support filling with a background thread
29  */
30
31 #include "libavutil/avassert.h"
32 #include "libavutil/avstring.h"
33 #include "libavutil/file.h"
34 #include "avformat.h"
35 #include <fcntl.h>
36 #if HAVE_SETMODE
37 #include <io.h>
38 #endif
39 #include <unistd.h>
40 #include <sys/stat.h>
41 #include <stdlib.h>
42 #include "os_support.h"
43 #include "url.h"
44
45 typedef struct Context {
46     int fd;
47     int64_t end;
48     int64_t pos;
49     URLContext *inner;
50 } Context;
51
52 static int cache_open(URLContext *h, const char *arg, int flags)
53 {
54     int access;
55     const char *buffername;
56     Context *c= h->priv_data;
57
58     av_strstart(arg, "cache:", &arg);
59
60     c->fd = av_tempfile("ffcache", &buffername, 0, h);
61     if (c->fd < 0){
62         av_log(h, AV_LOG_ERROR, "Failed to create tempfile\n");
63         return c->fd;
64     }
65
66     unlink(buffername);
67     av_free(buffername);
68
69     return ffurl_open(&c->inner, arg, flags, &h->interrupt_callback, NULL);
70 }
71
72 static int cache_read(URLContext *h, unsigned char *buf, int size)
73 {
74     Context *c= h->priv_data;
75     int r;
76
77     if(c->pos<c->end){
78         r = read(c->fd, buf, FFMIN(size, c->end - c->pos));
79         if(r>0)
80             c->pos += r;
81         return (-1 == r)?AVERROR(errno):r;
82     }else{
83         r = ffurl_read(c->inner, buf, size);
84         if(r > 0){
85             int r2= write(c->fd, buf, r);
86             av_assert0(r2==r); // FIXME handle cache failure
87             c->pos += r;
88             c->end += r;
89         }
90         return r;
91     }
92 }
93
94 static int64_t cache_seek(URLContext *h, int64_t pos, int whence)
95 {
96     Context *c= h->priv_data;
97
98     if (whence == AVSEEK_SIZE) {
99         pos= ffurl_seek(c->inner, pos, whence);
100         if(pos <= 0){
101             pos= ffurl_seek(c->inner, -1, SEEK_END);
102             ffurl_seek(c->inner, c->end, SEEK_SET);
103             if(pos <= 0)
104                 return c->end;
105         }
106         return pos;
107     }
108
109     pos= lseek(c->fd, pos, whence);
110     if(pos<0){
111         return pos;
112     }else if(pos <= c->end){
113         c->pos= pos;
114         return pos;
115     }else{
116         lseek(c->fd, c->pos, SEEK_SET);
117         return AVERROR(EPIPE);
118     }
119 }
120
121 static int cache_close(URLContext *h)
122 {
123     Context *c= h->priv_data;
124     close(c->fd);
125     ffurl_close(c->inner);
126
127     return 0;
128 }
129
130 URLProtocol ff_cache_protocol = {
131     .name                = "cache",
132     .url_open            = cache_open,
133     .url_read            = cache_read,
134     .url_seek            = cache_seek,
135     .url_close           = cache_close,
136     .priv_data_size      = sizeof(Context),
137 };