2010-01-30 23:26:47 +01:00
|
|
|
/*
|
|
|
|
* This file is part of MPlayer.
|
|
|
|
*
|
|
|
|
* MPlayer is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* MPlayer is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along
|
|
|
|
* with MPlayer; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*/
|
|
|
|
|
2008-02-22 10:09:46 +01:00
|
|
|
#ifndef MPLAYER_STREAM_H
|
|
|
|
#define MPLAYER_STREAM_H
|
2001-04-22 18:56:20 +02:00
|
|
|
|
2013-12-17 02:39:45 +01:00
|
|
|
#include "common/msg.h"
|
2012-11-17 18:12:13 +01:00
|
|
|
#include <stdbool.h>
|
2010-11-02 02:17:41 +01:00
|
|
|
#include <stdio.h>
|
2002-09-16 00:38:01 +02:00
|
|
|
#include <string.h>
|
2002-01-07 10:22:01 +01:00
|
|
|
#include <inttypes.h>
|
2002-03-23 22:52:13 +01:00
|
|
|
#include <sys/types.h>
|
2010-03-06 08:24:41 +01:00
|
|
|
#include <fcntl.h>
|
|
|
|
|
2013-12-17 02:39:45 +01:00
|
|
|
#include "bstr/bstr.h"
|
2011-02-25 17:10:00 +01:00
|
|
|
|
2013-07-12 22:05:43 +02:00
|
|
|
enum streamtype {
|
|
|
|
STREAMTYPE_GENERIC = 0,
|
|
|
|
STREAMTYPE_FILE,
|
|
|
|
STREAMTYPE_RADIO,
|
|
|
|
STREAMTYPE_DVB,
|
|
|
|
STREAMTYPE_DVD,
|
|
|
|
STREAMTYPE_PVR,
|
|
|
|
STREAMTYPE_TV,
|
|
|
|
STREAMTYPE_MF,
|
2013-11-19 22:26:35 +01:00
|
|
|
STREAMTYPE_EDL,
|
2013-07-12 22:05:43 +02:00
|
|
|
STREAMTYPE_AVDEVICE,
|
|
|
|
};
|
2001-04-22 18:56:20 +02:00
|
|
|
|
2002-09-16 00:38:01 +02:00
|
|
|
#define STREAM_BUFFER_SIZE 2048
|
2013-01-24 17:43:07 +01:00
|
|
|
#define STREAM_MAX_SECTOR_SIZE (8 * 1024)
|
2002-09-16 00:38:01 +02:00
|
|
|
|
stream: add stream_unread_buffer()
demux_lavf probes up to 2 MB of data in the worst case. When the ffmpeg
demuxer is actually opened, the stream is seeked back to 0, and the
previously read data is thrown away.
This wasn't a problem for playback of local files, but it's less than
ideal for playing from slow media (like web streams), and breaks
completely if the media is not seekable (pipes, some web streams).
This new function is intended to allow fixing this. demux_lavf will use
it to put the read probe data back into the buffer.
The simplest way of implementing this function is by making it
transparently extend the normal stream buffer. This makes sure no
existing code is broken by new weird special cases. For simplicity
and to avoid possible performance loss due to extra dereferencing
when accessing the buffer, we just extend the static buffer from
8 KB to 2 MB. Normally, most of these 2 MB will stay uncommitted, so
there's no associated waste of memory. If demux_lavf really reads all
2 MB, the memory will be committed and stay unused, though.
2013-05-24 23:20:09 +02:00
|
|
|
// Max buffer for initial probe.
|
|
|
|
#define STREAM_MAX_BUFFER_SIZE (2 * 1024 * 1024)
|
|
|
|
|
2013-06-06 20:39:53 +02:00
|
|
|
|
|
|
|
// stream->mode
|
2003-04-02 18:25:07 +02:00
|
|
|
#define STREAM_READ 0
|
|
|
|
#define STREAM_WRITE 1
|
2013-06-06 20:39:53 +02:00
|
|
|
|
2013-08-25 22:50:16 +02:00
|
|
|
// flags for stream_open_ext (this includes STREAM_READ and STREAM_WRITE)
|
|
|
|
#define STREAM_NO_FILTERS 2
|
|
|
|
|
2013-06-06 20:39:53 +02:00
|
|
|
// stream->flags
|
2013-08-22 18:21:32 +02:00
|
|
|
#define MP_STREAM_FAST_SKIPPING 1 // allow forward seeks by skipping
|
2009-11-22 16:18:21 +01:00
|
|
|
#define MP_STREAM_SEEK_BW 2
|
|
|
|
#define MP_STREAM_SEEK_FW 4
|
2013-01-24 17:43:07 +01:00
|
|
|
#define MP_STREAM_SEEK (MP_STREAM_SEEK_BW | MP_STREAM_SEEK_FW)
|
2003-04-02 18:25:07 +02:00
|
|
|
|
2013-08-25 22:50:16 +02:00
|
|
|
#define STREAM_NO_MATCH -2
|
2007-08-29 00:38:45 +02:00
|
|
|
#define STREAM_UNSUPPORTED -1
|
2003-04-02 18:25:07 +02:00
|
|
|
#define STREAM_ERROR 0
|
|
|
|
#define STREAM_OK 1
|
|
|
|
|
2013-10-02 21:19:16 +02:00
|
|
|
enum stream_ctrl {
|
|
|
|
STREAM_CTRL_GET_TIME_LENGTH = 1,
|
|
|
|
STREAM_CTRL_SEEK_TO_CHAPTER,
|
|
|
|
STREAM_CTRL_GET_CURRENT_CHAPTER,
|
|
|
|
STREAM_CTRL_GET_NUM_CHAPTERS,
|
|
|
|
STREAM_CTRL_GET_CURRENT_TIME,
|
|
|
|
STREAM_CTRL_SEEK_TO_TIME,
|
|
|
|
STREAM_CTRL_GET_SIZE,
|
|
|
|
STREAM_CTRL_GET_ASPECT_RATIO,
|
|
|
|
STREAM_CTRL_GET_NUM_ANGLES,
|
|
|
|
STREAM_CTRL_GET_ANGLE,
|
|
|
|
STREAM_CTRL_SET_ANGLE,
|
|
|
|
STREAM_CTRL_GET_NUM_TITLES,
|
|
|
|
STREAM_CTRL_GET_LANG,
|
|
|
|
STREAM_CTRL_GET_CURRENT_TITLE,
|
Add prelimimary (basic, possibly broken) dvdnav support
This readds a more or less completely new dvdnav implementation, though
it's based on the code from before commit 41fbcee. Note that this is
rather basic, and might be broken or not quite usable in many cases.
Most importantly, navigation highlights are not correctly implemented.
This would require changes in the FFmpeg dvdsub decoder (to apply a
different internal CLUT), so supporting it is not really possible right
now. And in fact, I don't think I ever want to support it, because it's
a very small gain for a lot of work. Instead, mpv will display fake
highlights, which are an approximate bounding box around the real
highlights.
Some things like mouse input or switching audio/subtitles stream using
the dvdnav VM are not supported.
Might be quite fragile on transitions: if dvdnav initiates a transition,
and doesn't give us enough mpeg data to initialize video playback, the
player will just quit.
This is added only because some users seem to want it. I don't intend to
make mpv a good DVD player, so the very basic minimum will have to do.
How about you just convert your DVD to proper video files?
2013-12-12 01:44:28 +01:00
|
|
|
STREAM_CTRL_SET_CURRENT_TITLE,
|
2013-10-02 21:19:16 +02:00
|
|
|
STREAM_CTRL_GET_CACHE_SIZE,
|
|
|
|
STREAM_CTRL_GET_CACHE_FILL,
|
|
|
|
STREAM_CTRL_GET_CACHE_IDLE,
|
2013-12-14 00:59:26 +01:00
|
|
|
STREAM_CTRL_RESUME_CACHE,
|
2013-10-02 21:19:16 +02:00
|
|
|
STREAM_CTRL_RECONNECT,
|
|
|
|
// DVD/Bluray, signal general support for GET_CURRENT_TIME etc.
|
|
|
|
STREAM_CTRL_MANAGES_TIMELINE,
|
|
|
|
STREAM_CTRL_GET_START_TIME,
|
|
|
|
STREAM_CTRL_GET_CHAPTER_TIME,
|
|
|
|
STREAM_CTRL_GET_DVD_INFO,
|
|
|
|
STREAM_CTRL_SET_CONTENTS,
|
|
|
|
STREAM_CTRL_GET_METADATA,
|
2013-12-06 23:00:19 +01:00
|
|
|
STREAM_CTRL_GET_BASE_FILENAME,
|
Add prelimimary (basic, possibly broken) dvdnav support
This readds a more or less completely new dvdnav implementation, though
it's based on the code from before commit 41fbcee. Note that this is
rather basic, and might be broken or not quite usable in many cases.
Most importantly, navigation highlights are not correctly implemented.
This would require changes in the FFmpeg dvdsub decoder (to apply a
different internal CLUT), so supporting it is not really possible right
now. And in fact, I don't think I ever want to support it, because it's
a very small gain for a lot of work. Instead, mpv will display fake
highlights, which are an approximate bounding box around the real
highlights.
Some things like mouse input or switching audio/subtitles stream using
the dvdnav VM are not supported.
Might be quite fragile on transitions: if dvdnav initiates a transition,
and doesn't give us enough mpeg data to initialize video playback, the
player will just quit.
This is added only because some users seem to want it. I don't intend to
make mpv a good DVD player, so the very basic minimum will have to do.
How about you just convert your DVD to proper video files?
2013-12-12 01:44:28 +01:00
|
|
|
STREAM_CTRL_GET_NAV_EVENT, // struct mp_nav_event**
|
|
|
|
STREAM_CTRL_NAV_CMD, // struct mp_nav_cmd*
|
2013-10-02 21:19:16 +02:00
|
|
|
};
|
2003-04-02 18:25:07 +02:00
|
|
|
|
2012-02-19 14:15:41 +01:00
|
|
|
struct stream_lang_req {
|
2013-01-24 17:43:07 +01:00
|
|
|
int type; // STREAM_AUDIO, STREAM_SUB
|
|
|
|
int id;
|
|
|
|
char name[50];
|
2012-02-19 14:15:41 +01:00
|
|
|
};
|
2008-10-16 20:28:38 +02:00
|
|
|
|
2013-06-05 01:59:04 +02:00
|
|
|
struct stream_dvd_info_req {
|
|
|
|
unsigned int palette[16];
|
|
|
|
int num_subs;
|
|
|
|
};
|
|
|
|
|
2008-04-24 04:49:44 +02:00
|
|
|
struct stream;
|
2003-04-02 18:25:07 +02:00
|
|
|
typedef struct stream_info_st {
|
2013-01-24 17:43:07 +01:00
|
|
|
const char *name;
|
2013-06-06 20:39:53 +02:00
|
|
|
// opts is set from ->opts
|
2013-08-02 17:02:34 +02:00
|
|
|
int (*open)(struct stream *st, int mode);
|
stream: fix url_options field, make protocols field not fixed length
The way the url_options field was handled was not entirely sane: it's
actually a flexible array member, so it points to garbage for streams
which do not initialize this member (it just points to the data right
after the struct, which is garbage in theory and practice). This was
not actually a problem, since the field is only used if priv_size is
set (due to how this stuff is used). But it doesn't allow setting
priv_size only, which might be useful in some cases.
Also, make the protocols array not a fixed size array. Most stream
implementations have only 1 protocol prefix, but stream_lavf.c has
over 10 (whitelists ffmpeg protocols). The high size of the fixed
size protocol array wastes space, and it is _still_ annoying to
add new prefixes to stream_lavf (have to bump the maximum length),
so make it arbitrary length.
The two changes (plus some more cosmetic changes) arte conflated into
one, because it was annoying going over all the stream implementations.
2013-08-25 22:49:27 +02:00
|
|
|
const char **protocols;
|
2013-08-02 17:02:34 +02:00
|
|
|
int priv_size;
|
|
|
|
const void *priv_defaults;
|
|
|
|
const struct m_option *options;
|
stream: fix url_options field, make protocols field not fixed length
The way the url_options field was handled was not entirely sane: it's
actually a flexible array member, so it points to garbage for streams
which do not initialize this member (it just points to the data right
after the struct, which is garbage in theory and practice). This was
not actually a problem, since the field is only used if priv_size is
set (due to how this stuff is used). But it doesn't allow setting
priv_size only, which might be useful in some cases.
Also, make the protocols array not a fixed size array. Most stream
implementations have only 1 protocol prefix, but stream_lavf.c has
over 10 (whitelists ffmpeg protocols). The high size of the fixed
size protocol array wastes space, and it is _still_ annoying to
add new prefixes to stream_lavf (have to bump the maximum length),
so make it arbitrary length.
The two changes (plus some more cosmetic changes) arte conflated into
one, because it was annoying going over all the stream implementations.
2013-08-25 22:49:27 +02:00
|
|
|
const char **url_options;
|
2013-08-25 22:50:16 +02:00
|
|
|
bool stream_filter;
|
2003-04-02 18:25:07 +02:00
|
|
|
} stream_info_t;
|
|
|
|
|
2008-04-24 04:49:44 +02:00
|
|
|
typedef struct stream {
|
2013-08-02 17:02:34 +02:00
|
|
|
const struct stream_info_st *info;
|
|
|
|
|
2013-01-24 17:43:07 +01:00
|
|
|
// Read
|
|
|
|
int (*fill_buffer)(struct stream *s, char *buffer, int max_len);
|
|
|
|
// Write
|
|
|
|
int (*write_buffer)(struct stream *s, char *buffer, int len);
|
|
|
|
// Seek
|
|
|
|
int (*seek)(struct stream *s, int64_t pos);
|
|
|
|
// Control
|
|
|
|
// Will be later used to let streams like dvd and cdda report
|
|
|
|
// their structure (ie tracks, chapters, etc)
|
|
|
|
int (*control)(struct stream *s, int cmd, void *arg);
|
|
|
|
// Close
|
|
|
|
void (*close)(struct stream *s);
|
|
|
|
|
2013-07-12 22:05:43 +02:00
|
|
|
enum streamtype type; // see STREAMTYPE_*
|
|
|
|
enum streamtype uncached_type; // if stream is cache, type of wrapped str.
|
cache: make the stream cache a proper stream that wraps other streams
Before this commit, the cache was franken-hacked on top of the stream
API. You had to use special functions (like cache_stream_fill_buffer()
instead of stream_fill_buffer()), which would access the stream in a
cached manner.
The whole idea about the previous design was that the cache runs in a
thread or in a forked process, while the cache awa functions made sure
the stream instance looked consistent to the user. If you used the
normal functions instead of the special ones while the cache was
running, you were out of luck.
Make it a bit more reasonable by turning the cache into a stream on its
own. This makes it behave exactly like a normal stream. The stream
callbacks call into the original (uncached) stream to do work. No
special cache functions or redirections are needed. The only different
thing about cache streams is that they are created by special functions,
instead of being part of the auto_open_streams[] array.
To make things simpler, remove the threading implementation, which was
messed into the code. The threading code could perhaps be kept, but I
don't really want to have to worry about this special case. A proper
threaded implementation will be added later.
Remove the cache enabling code from stream_radio.c. Since enabling the
cache involves replacing the old stream with a new one, the code as-is
can't be kept. It would be easily possible to enable the cache by
requesting a cache size (which is also much simpler). But nobody uses
stream_radio.c and I can't even test this thing, and the cache is
probably not really important for it either.
2013-05-24 18:49:09 +02:00
|
|
|
int flags; // MP_STREAM_SEEK_* or'ed flags
|
2013-01-24 17:43:07 +01:00
|
|
|
int sector_size; // sector size (seek will be aligned on this size if non 0)
|
2013-05-25 15:03:30 +02:00
|
|
|
int read_chunk; // maximum amount of data to read at once to limit latency
|
2013-01-24 17:43:07 +01:00
|
|
|
unsigned int buf_pos, buf_len;
|
|
|
|
int64_t pos, start_pos, end_pos;
|
|
|
|
int eof;
|
|
|
|
int mode; //STREAM_READ or STREAM_WRITE
|
|
|
|
bool streaming; // known to be a network stream if true
|
|
|
|
void *priv; // used for DVD, TV, RTSP etc
|
2013-08-02 17:02:34 +02:00
|
|
|
char *url; // filename/url (possibly including protocol prefix)
|
|
|
|
char *path; // filename (url without protocol prefix)
|
2013-01-24 17:43:07 +01:00
|
|
|
char *mime_type; // when HTTP streaming is used
|
2013-07-11 21:10:42 +02:00
|
|
|
char *demuxer; // request demuxer to be used
|
2013-01-24 17:43:07 +01:00
|
|
|
char *lavf_type; // name of expected demuxer type for lavf
|
2013-08-25 20:40:21 +02:00
|
|
|
bool safe_origin; // used for playlists that can be opened safely
|
2013-12-14 01:21:06 +01:00
|
|
|
bool allow_caching; // stream cache makes sense
|
2013-01-24 17:43:07 +01:00
|
|
|
struct MPOpts *opts;
|
2013-05-11 22:19:33 +02:00
|
|
|
|
|
|
|
FILE *capture_file;
|
|
|
|
char *capture_filename;
|
cache: make the stream cache a proper stream that wraps other streams
Before this commit, the cache was franken-hacked on top of the stream
API. You had to use special functions (like cache_stream_fill_buffer()
instead of stream_fill_buffer()), which would access the stream in a
cached manner.
The whole idea about the previous design was that the cache runs in a
thread or in a forked process, while the cache awa functions made sure
the stream instance looked consistent to the user. If you used the
normal functions instead of the special ones while the cache was
running, you were out of luck.
Make it a bit more reasonable by turning the cache into a stream on its
own. This makes it behave exactly like a normal stream. The stream
callbacks call into the original (uncached) stream to do work. No
special cache functions or redirections are needed. The only different
thing about cache streams is that they are created by special functions,
instead of being part of the auto_open_streams[] array.
To make things simpler, remove the threading implementation, which was
messed into the code. The threading code could perhaps be kept, but I
don't really want to have to worry about this special case. A proper
threaded implementation will be added later.
Remove the cache enabling code from stream_radio.c. Since enabling the
cache involves replacing the old stream with a new one, the code as-is
can't be kept. It would be easily possible to enable the cache by
requesting a cache size (which is also much simpler). But nobody uses
stream_radio.c and I can't even test this thing, and the cache is
probably not really important for it either.
2013-05-24 18:49:09 +02:00
|
|
|
|
2013-08-25 22:50:16 +02:00
|
|
|
struct stream *uncached_stream; // underlying stream for cache wrapper
|
|
|
|
struct stream *source;
|
stream: add stream_unread_buffer()
demux_lavf probes up to 2 MB of data in the worst case. When the ffmpeg
demuxer is actually opened, the stream is seeked back to 0, and the
previously read data is thrown away.
This wasn't a problem for playback of local files, but it's less than
ideal for playing from slow media (like web streams), and breaks
completely if the media is not seekable (pipes, some web streams).
This new function is intended to allow fixing this. demux_lavf will use
it to put the read probe data back into the buffer.
The simplest way of implementing this function is by making it
transparently extend the normal stream buffer. This makes sure no
existing code is broken by new weird special cases. For simplicity
and to avoid possible performance loss due to extra dereferencing
when accessing the buffer, we just extend the static buffer from
8 KB to 2 MB. Normally, most of these 2 MB will stay uncommitted, so
there's no associated waste of memory. If demux_lavf really reads all
2 MB, the memory will be committed and stay unused, though.
2013-05-24 23:20:09 +02:00
|
|
|
|
|
|
|
// Includes additional padding in case sizes get rounded up by sector size.
|
|
|
|
unsigned char buffer[];
|
2001-04-22 18:56:20 +02:00
|
|
|
} stream_t;
|
|
|
|
|
2008-08-12 11:49:37 +02:00
|
|
|
int stream_fill_buffer(stream_t *s);
|
2010-02-22 00:30:34 +01:00
|
|
|
|
2013-05-11 22:19:33 +02:00
|
|
|
void stream_set_capture_file(stream_t *s, const char *filename);
|
|
|
|
|
cache: make the stream cache a proper stream that wraps other streams
Before this commit, the cache was franken-hacked on top of the stream
API. You had to use special functions (like cache_stream_fill_buffer()
instead of stream_fill_buffer()), which would access the stream in a
cached manner.
The whole idea about the previous design was that the cache runs in a
thread or in a forked process, while the cache awa functions made sure
the stream instance looked consistent to the user. If you used the
normal functions instead of the special ones while the cache was
running, you were out of luck.
Make it a bit more reasonable by turning the cache into a stream on its
own. This makes it behave exactly like a normal stream. The stream
callbacks call into the original (uncached) stream to do work. No
special cache functions or redirections are needed. The only different
thing about cache streams is that they are created by special functions,
instead of being part of the auto_open_streams[] array.
To make things simpler, remove the threading implementation, which was
messed into the code. The threading code could perhaps be kept, but I
don't really want to have to worry about this special case. A proper
threaded implementation will be added later.
Remove the cache enabling code from stream_radio.c. Since enabling the
cache involves replacing the old stream with a new one, the code as-is
can't be kept. It would be easily possible to enable the cache by
requesting a cache size (which is also much simpler). But nobody uses
stream_radio.c and I can't even test this thing, and the cache is
probably not really important for it either.
2013-05-24 18:49:09 +02:00
|
|
|
int stream_enable_cache_percent(stream_t **stream, int64_t stream_cache_size,
|
2013-07-10 15:03:54 +02:00
|
|
|
int64_t stream_cache_def_size,
|
2013-01-24 17:43:07 +01:00
|
|
|
float stream_cache_min_percent,
|
|
|
|
float stream_cache_seek_min_percent);
|
cache: make the stream cache a proper stream that wraps other streams
Before this commit, the cache was franken-hacked on top of the stream
API. You had to use special functions (like cache_stream_fill_buffer()
instead of stream_fill_buffer()), which would access the stream in a
cached manner.
The whole idea about the previous design was that the cache runs in a
thread or in a forked process, while the cache awa functions made sure
the stream instance looked consistent to the user. If you used the
normal functions instead of the special ones while the cache was
running, you were out of luck.
Make it a bit more reasonable by turning the cache into a stream on its
own. This makes it behave exactly like a normal stream. The stream
callbacks call into the original (uncached) stream to do work. No
special cache functions or redirections are needed. The only different
thing about cache streams is that they are created by special functions,
instead of being part of the auto_open_streams[] array.
To make things simpler, remove the threading implementation, which was
messed into the code. The threading code could perhaps be kept, but I
don't really want to have to worry about this special case. A proper
threaded implementation will be added later.
Remove the cache enabling code from stream_radio.c. Since enabling the
cache involves replacing the old stream with a new one, the code as-is
can't be kept. It would be easily possible to enable the cache by
requesting a cache size (which is also much simpler). But nobody uses
stream_radio.c and I can't even test this thing, and the cache is
probably not really important for it either.
2013-05-24 18:49:09 +02:00
|
|
|
|
|
|
|
// Internal
|
|
|
|
int stream_cache_init(stream_t *cache, stream_t *stream, int64_t size,
|
|
|
|
int64_t min, int64_t seek_limit);
|
|
|
|
|
2006-12-18 21:50:31 +01:00
|
|
|
int stream_write_buffer(stream_t *s, unsigned char *buf, int len);
|
2001-08-01 01:18:16 +02:00
|
|
|
|
2013-01-24 17:43:07 +01:00
|
|
|
inline static int stream_read_char(stream_t *s)
|
|
|
|
{
|
|
|
|
return (s->buf_pos < s->buf_len) ? s->buffer[s->buf_pos++] :
|
cache: make the stream cache a proper stream that wraps other streams
Before this commit, the cache was franken-hacked on top of the stream
API. You had to use special functions (like cache_stream_fill_buffer()
instead of stream_fill_buffer()), which would access the stream in a
cached manner.
The whole idea about the previous design was that the cache runs in a
thread or in a forked process, while the cache awa functions made sure
the stream instance looked consistent to the user. If you used the
normal functions instead of the special ones while the cache was
running, you were out of luck.
Make it a bit more reasonable by turning the cache into a stream on its
own. This makes it behave exactly like a normal stream. The stream
callbacks call into the original (uncached) stream to do work. No
special cache functions or redirections are needed. The only different
thing about cache streams is that they are created by special functions,
instead of being part of the auto_open_streams[] array.
To make things simpler, remove the threading implementation, which was
messed into the code. The threading code could perhaps be kept, but I
don't really want to have to worry about this special case. A proper
threaded implementation will be added later.
Remove the cache enabling code from stream_radio.c. Since enabling the
cache involves replacing the old stream with a new one, the code as-is
can't be kept. It would be easily possible to enable the cache by
requesting a cache size (which is also much simpler). But nobody uses
stream_radio.c and I can't even test this thing, and the cache is
probably not really important for it either.
2013-05-24 18:49:09 +02:00
|
|
|
(stream_fill_buffer(s) ? s->buffer[s->buf_pos++] : -256);
|
2001-04-22 18:56:20 +02:00
|
|
|
}
|
|
|
|
|
2013-01-24 17:43:07 +01:00
|
|
|
inline static unsigned int stream_read_dword(stream_t *s)
|
|
|
|
{
|
|
|
|
unsigned int y;
|
|
|
|
y = stream_read_char(s);
|
|
|
|
y = (y << 8) | stream_read_char(s);
|
|
|
|
y = (y << 8) | stream_read_char(s);
|
|
|
|
y = (y << 8) | stream_read_char(s);
|
|
|
|
return y;
|
2001-04-22 18:56:20 +02:00
|
|
|
}
|
|
|
|
|
2013-01-24 17:43:07 +01:00
|
|
|
inline static uint64_t stream_read_qword(stream_t *s)
|
|
|
|
{
|
|
|
|
uint64_t y;
|
|
|
|
y = stream_read_char(s);
|
|
|
|
y = (y << 8) | stream_read_char(s);
|
|
|
|
y = (y << 8) | stream_read_char(s);
|
|
|
|
y = (y << 8) | stream_read_char(s);
|
|
|
|
y = (y << 8) | stream_read_char(s);
|
|
|
|
y = (y << 8) | stream_read_char(s);
|
|
|
|
y = (y << 8) | stream_read_char(s);
|
|
|
|
y = (y << 8) | stream_read_char(s);
|
|
|
|
return y;
|
2002-01-05 20:21:06 +01:00
|
|
|
}
|
|
|
|
|
2013-01-24 17:43:07 +01:00
|
|
|
unsigned char *stream_read_line(stream_t *s, unsigned char *mem, int max,
|
|
|
|
int utf16);
|
2013-08-25 20:40:21 +02:00
|
|
|
int stream_skip_bom(struct stream *s);
|
2006-03-16 15:42:51 +01:00
|
|
|
|
2013-01-24 17:43:07 +01:00
|
|
|
inline static int stream_eof(stream_t *s)
|
|
|
|
{
|
|
|
|
return s->eof;
|
2001-04-22 18:56:20 +02:00
|
|
|
}
|
|
|
|
|
2013-01-24 17:43:07 +01:00
|
|
|
inline static int64_t stream_tell(stream_t *s)
|
|
|
|
{
|
|
|
|
return s->pos + s->buf_pos - s->buf_len;
|
2001-04-22 18:56:20 +02:00
|
|
|
}
|
|
|
|
|
2013-05-24 11:56:49 +02:00
|
|
|
int stream_skip(stream_t *s, int64_t len);
|
|
|
|
int stream_seek(stream_t *s, int64_t pos);
|
|
|
|
int stream_read(stream_t *s, char *mem, int total);
|
2013-05-27 21:53:40 +02:00
|
|
|
int stream_read_partial(stream_t *s, char *buf, int buf_size);
|
2013-06-21 21:06:36 +02:00
|
|
|
struct bstr stream_peek(stream_t *s, int len);
|
2013-12-14 00:51:00 +01:00
|
|
|
void stream_drop_buffers(stream_t *s);
|
2001-04-22 18:56:20 +02:00
|
|
|
|
2008-04-23 05:35:36 +02:00
|
|
|
struct MPOpts;
|
2013-06-11 12:16:42 +02:00
|
|
|
|
2011-02-25 17:10:00 +01:00
|
|
|
struct bstr stream_read_complete(struct stream *s, void *talloc_ctx,
|
2013-06-11 12:16:42 +02:00
|
|
|
int max_size);
|
2006-03-01 22:56:30 +01:00
|
|
|
int stream_control(stream_t *s, int cmd, void *arg);
|
2012-11-18 21:23:17 +01:00
|
|
|
void stream_update_size(stream_t *s);
|
2001-04-22 18:56:20 +02:00
|
|
|
void free_stream(stream_t *s);
|
2013-08-25 22:50:16 +02:00
|
|
|
struct stream *stream_create(const char *url, int flags, struct MPOpts *options);
|
2013-07-11 21:10:42 +02:00
|
|
|
struct stream *stream_open(const char *filename, struct MPOpts *options);
|
2010-03-10 00:46:46 +01:00
|
|
|
stream_t *open_output_stream(const char *filename, struct MPOpts *options);
|
2013-06-21 00:47:58 +02:00
|
|
|
stream_t *open_memory_stream(void *data, int len);
|
2011-02-10 11:15:21 +01:00
|
|
|
struct demux_stream;
|
|
|
|
|
2008-04-09 02:36:28 +02:00
|
|
|
/// Set the callback to be used by libstream to check for user
|
|
|
|
/// interruption during long blocking operations (cache filling, etc).
|
2008-04-30 06:15:52 +02:00
|
|
|
struct input_ctx;
|
2013-01-24 17:43:07 +01:00
|
|
|
void stream_set_interrupt_callback(int (*cb)(struct input_ctx *, int),
|
2008-04-30 06:15:52 +02:00
|
|
|
struct input_ctx *ctx);
|
2010-05-28 20:45:25 +02:00
|
|
|
/// Call the interrupt checking callback if there is one and
|
|
|
|
/// wait for time milliseconds
|
2008-04-09 02:36:28 +02:00
|
|
|
int stream_check_interrupt(int time);
|
cache: make the stream cache a proper stream that wraps other streams
Before this commit, the cache was franken-hacked on top of the stream
API. You had to use special functions (like cache_stream_fill_buffer()
instead of stream_fill_buffer()), which would access the stream in a
cached manner.
The whole idea about the previous design was that the cache runs in a
thread or in a forked process, while the cache awa functions made sure
the stream instance looked consistent to the user. If you used the
normal functions instead of the special ones while the cache was
running, you were out of luck.
Make it a bit more reasonable by turning the cache into a stream on its
own. This makes it behave exactly like a normal stream. The stream
callbacks call into the original (uncached) stream to do work. No
special cache functions or redirections are needed. The only different
thing about cache streams is that they are created by special functions,
instead of being part of the auto_open_streams[] array.
To make things simpler, remove the threading implementation, which was
messed into the code. The threading code could perhaps be kept, but I
don't really want to have to worry about this special case. A proper
threaded implementation will be added later.
Remove the cache enabling code from stream_radio.c. Since enabling the
cache involves replacing the old stream with a new one, the code as-is
can't be kept. It would be easily possible to enable the cache by
requesting a cache size (which is also much simpler). But nobody uses
stream_radio.c and I can't even test this thing, and the cache is
probably not really important for it either.
2013-05-24 18:49:09 +02:00
|
|
|
|
2013-05-03 19:52:28 +02:00
|
|
|
bool stream_manages_timeline(stream_t *s);
|
|
|
|
|
2013-09-22 03:04:57 +02:00
|
|
|
/* stream/stream_dvd.c */
|
2001-10-30 18:03:11 +01:00
|
|
|
extern int dvd_title;
|
|
|
|
extern int dvd_angle;
|
2013-09-22 03:04:57 +02:00
|
|
|
extern int dvd_speed;
|
|
|
|
extern char *dvd_device, *cdrom_device;
|
2001-06-04 19:51:17 +02:00
|
|
|
|
2013-04-26 21:50:19 +02:00
|
|
|
extern int bluray_angle;
|
2010-07-05 19:04:46 +02:00
|
|
|
extern char *bluray_device;
|
2002-07-03 00:31:50 +02:00
|
|
|
|
2001-11-16 23:26:57 +01:00
|
|
|
typedef struct {
|
2013-01-24 17:43:07 +01:00
|
|
|
int id; // 0 - 31 mpeg; 128 - 159 ac3; 160 - 191 pcm
|
|
|
|
int language;
|
|
|
|
int type;
|
|
|
|
int channels;
|
2001-12-26 00:31:37 +01:00
|
|
|
} stream_language_t;
|
2001-11-21 11:30:59 +01:00
|
|
|
|
2013-08-02 17:03:30 +02:00
|
|
|
void mp_url_unescape_inplace(char *buf);
|
2013-08-25 22:58:29 +02:00
|
|
|
char *mp_url_escape(void *talloc_ctx, const char *s, const char *ok);
|
2013-08-02 17:03:30 +02:00
|
|
|
|
2008-02-22 10:09:46 +01:00
|
|
|
#endif /* MPLAYER_STREAM_H */
|