]> git.sesse.net Git - ffmpeg/blob - libavformat/cache.c
Merge remote-tracking branch 'qatar/master'
[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     char *buffername;
55     Context *c= h->priv_data;
56
57     av_strstart(arg, "cache:", &arg);
58
59     c->fd = av_tempfile("ffcache", &buffername, 0, h);
60     if (c->fd < 0){
61         av_log(h, AV_LOG_ERROR, "Failed to create tempfile\n");
62         return c->fd;
63     }
64
65     unlink(buffername);
66     av_freep(&buffername);
67
68     return ffurl_open(&c->inner, arg, flags, &h->interrupt_callback, NULL);
69 }
70
71 static int cache_read(URLContext *h, unsigned char *buf, int size)
72 {
73     Context *c= h->priv_data;
74     int r;
75
76     if(c->pos<c->end){
77         r = read(c->fd, buf, FFMIN(size, c->end - c->pos));
78         if(r>0)
79             c->pos += r;
80         return (-1 == r)?AVERROR(errno):r;
81     }else{
82         r = ffurl_read(c->inner, buf, size);
83         if(r > 0){
84             int r2= write(c->fd, buf, r);
85             av_assert0(r2==r); // FIXME handle cache failure
86             c->pos += r;
87             c->end += r;
88         }
89         return r;
90     }
91 }
92
93 static int64_t cache_seek(URLContext *h, int64_t pos, int whence)
94 {
95     Context *c= h->priv_data;
96
97     if (whence == AVSEEK_SIZE) {
98         pos= ffurl_seek(c->inner, pos, whence);
99         if(pos <= 0){
100             pos= ffurl_seek(c->inner, -1, SEEK_END);
101             ffurl_seek(c->inner, c->end, SEEK_SET);
102             if(pos <= 0)
103                 return c->end;
104         }
105         return pos;
106     }
107
108     pos= lseek(c->fd, pos, whence);
109     if(pos<0){
110         return pos;
111     }else if(pos <= c->end){
112         c->pos= pos;
113         return pos;
114     }else{
115         lseek(c->fd, c->pos, SEEK_SET);
116         return AVERROR(EPIPE);
117     }
118 }
119
120 static int cache_close(URLContext *h)
121 {
122     Context *c= h->priv_data;
123     close(c->fd);
124     ffurl_close(c->inner);
125
126     return 0;
127 }
128
129 URLProtocol ff_cache_protocol = {
130     .name                = "cache",
131     .url_open            = cache_open,
132     .url_read            = cache_read,
133     .url_seek            = cache_seek,
134     .url_close           = cache_close,
135     .priv_data_size      = sizeof(Context),
136 };