2
* This file is part of MPlayer.
4
* MPlayer is free software; you can redistribute it and/or modify
5
* it under the terms of the GNU General Public License as published by
6
* the Free Software Foundation; either version 2 of the License, or
7
* (at your option) any later version.
9
* MPlayer is distributed in the hope that it will be useful,
10
* but WITHOUT ANY WARRANTY; without even the implied warranty of
11
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12
* GNU General Public License for more details.
14
* You should have received a copy of the GNU General Public License along
15
* with MPlayer; if not, write to the Free Software Foundation, Inc.,
16
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
21
#if defined(__MINGW32__) || defined(__CYGWIN__)
25
#include "osdep/osdep.h"
35
#if !defined(__MINGW32__) && !defined(__CYGWIN__)
36
#include <sys/ioctl.h>
40
#if defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || defined(__DragonFly__) || defined(__NetBSD__) || defined(__OpenBSD__)
41
#include "vcd_read_fbsd.h"
42
#elif defined(__APPLE__)
43
#include "vcd_read_darwin.h"
44
#elif defined(__MINGW32__) || defined(__CYGWIN__)
45
#include "vcd_read_win32.h"
46
#elif defined(__OS2__)
47
#include "vcd_read_os2.h"
52
#include "libmpdemux/demuxer.h"
54
extern char *cdrom_device;
56
static struct stream_priv_s {
59
} stream_priv_dflts = {
64
#define ST_OFF(f) M_ST_OFF(struct stream_priv_s,f)
66
static const m_option_t stream_opts_fields[] = {
67
{ "track", ST_OFF(track), CONF_TYPE_INT, M_OPT_MIN, 1, 0, NULL },
68
{ "device", ST_OFF(device), CONF_TYPE_STRING, 0, 0 ,0, NULL},
70
{ "hostname", ST_OFF(track), CONF_TYPE_INT, M_OPT_MIN, 1, 0, NULL },
71
{ "filename", ST_OFF(device), CONF_TYPE_STRING, 0, 0 ,0, NULL},
72
{ NULL, NULL, 0, 0, 0, 0, NULL }
74
static const struct m_struct_st stream_opts = {
76
sizeof(struct stream_priv_s),
81
static int fill_buffer(stream_t *s, char* buffer, int max_len){
82
if(s->pos > s->end_pos) /// don't past end of current track
84
return vcd_read(s->priv,buffer);
87
static int seek(stream_t *s,off_t newpos) {
89
vcd_set_msf(s->priv,s->pos/VCD_SECTOR_DATA);
93
static int control(stream_t *stream, int cmd, void *arg) {
94
struct stream_priv_s *p = stream->priv;
96
case STREAM_CTRL_GET_NUM_CHAPTERS:
98
mp_vcd_priv_t *vcd = vcd_read_toc(stream->fd);
101
*(unsigned int *)arg = vcd_end_track(vcd);
104
case STREAM_CTRL_SEEK_TO_CHAPTER:
107
unsigned int track = *(unsigned int *)arg + 1;
108
mp_vcd_priv_t *vcd = vcd_read_toc(stream->fd);
111
r = vcd_seek_to_track(vcd, track);
118
case STREAM_CTRL_GET_CURRENT_CHAPTER:
120
*(unsigned int *)arg = p->track - 1;
124
return STREAM_UNSUPPORTED;
127
static void close_s(stream_t *stream) {
131
static int open_s(stream_t *stream,int mode, void* opts, int* file_format) {
132
struct stream_priv_s* p = opts;
133
int ret,ret2,f,sect,tmp;
135
#if defined(__FreeBSD__) || defined(__FreeBSD_kernel__)
136
int bsize = VCD_SECTOR_SIZE;
138
#if defined(__MINGW32__) || defined(__CYGWIN__)
140
char device[] = "\\\\.\\?:";
143
char device[] = "X:";
149
if(mode != STREAM_READ
150
#if defined(__MINGW32__) || defined(__CYGWIN__)
151
|| GetVersion() > 0x80000000 // Win9x
154
m_struct_free(&stream_opts,opts);
155
return STREAM_UNSUPPORTED;
160
p->device = strdup(cdrom_device);
162
p->device = strdup(DEFAULT_CDROM_DEVICE);
165
#if defined(__MINGW32__) || defined(__CYGWIN__)
166
device[4] = p->device[0];
167
/* open() can't be used for devices so do it the complicated way */
168
hd = CreateFile(device, GENERIC_READ, FILE_SHARE_READ, NULL,
169
OPEN_EXISTING, FILE_FLAG_SEQUENTIAL_SCAN, NULL);
170
f = _open_osfhandle((long)hd, _O_RDONLY);
171
#elif defined(__OS2__)
172
device[0] = p->device[0];
173
rc = DosOpen(device, &hcd, &ulAction, 0, FILE_NORMAL,
174
OPEN_ACTION_OPEN_IF_EXISTS | OPEN_ACTION_FAIL_IF_NEW,
175
OPEN_ACCESS_READONLY | OPEN_SHARE_DENYNONE | OPEN_FLAGS_DASD,
179
f=open(p->device,O_RDONLY);
182
mp_tmsg(MSGT_OPEN,MSGL_ERR,"CD-ROM Device '%s' not found.\n",p->device);
183
m_struct_free(&stream_opts,opts);
187
vcd = vcd_read_toc(f);
189
mp_msg(MSGT_OPEN,MSGL_ERR,"Failed to get cd toc\n");
191
m_struct_free(&stream_opts,opts);
194
ret2=vcd_get_track_end(vcd,p->track);
196
mp_msg(MSGT_OPEN, MSGL_ERR, "%s (get)\n",
197
mp_gtext("Error selecting VCD track."));
200
m_struct_free(&stream_opts,opts);
203
ret=vcd_seek_to_track(vcd,p->track);
205
mp_msg(MSGT_OPEN, MSGL_ERR, "%s (seek)\n",
206
mp_gtext("Error selecting VCD track."));
209
m_struct_free(&stream_opts,opts);
212
/* search forward up to at most 3 seconds to skip leading margin */
213
sect = ret / VCD_SECTOR_DATA;
214
for (tmp = sect; tmp < sect + 3 * 75; tmp++) {
215
char mem[VCD_SECTOR_DATA];
216
//since MPEG packs are block-aligned we stop discarding sectors if they are non-null
217
if (vcd_read(vcd, mem) != VCD_SECTOR_DATA || mem[2] || mem[3])
220
mp_msg(MSGT_OPEN, MSGL_DBG2, "%d leading sectors skipped\n", tmp - sect);
221
vcd_set_msf(vcd, tmp);
222
ret = tmp * VCD_SECTOR_DATA;
224
mp_msg(MSGT_OPEN,MSGL_V,"VCD start byte position: 0x%X end: 0x%X\n",ret,ret2);
226
#if defined(__FreeBSD__) || defined(__FreeBSD_kernel__)
227
if (ioctl (f, CDRIOCSETBLOCKSIZE, &bsize) == -1) {
228
mp_msg(MSGT_OPEN,MSGL_WARN,"Error in CDRIOCSETBLOCKSIZE");
233
stream->type = STREAMTYPE_VCD;
234
stream->sector_size = VCD_SECTOR_DATA;
235
stream->start_pos=ret;
236
stream->end_pos=ret2;
239
stream->fill_buffer = fill_buffer;
241
stream->control = control;
242
stream->close = close_s;
243
*file_format = DEMUXER_TYPE_MPEG_PS;
245
m_struct_free(&stream_opts,opts);
249
const stream_info_t stream_info_vcd = {
253
"based on the code from ???",
257
1 // Urls are an option string