diff options
Diffstat (limited to 'src/dxr3/dxr3_decoder.c')
-rw-r--r-- | src/dxr3/dxr3_decoder.c | 292 |
1 files changed, 292 insertions, 0 deletions
diff --git a/src/dxr3/dxr3_decoder.c b/src/dxr3/dxr3_decoder.c new file mode 100644 index 000000000..833fa8b4f --- /dev/null +++ b/src/dxr3/dxr3_decoder.c @@ -0,0 +1,292 @@ +/* + * Copyright (C) 2000-2001 the xine project + * + * This file is part of xine, a unix video player. + * + * xine 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. + * + * xine 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 this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA + * + * $Id: dxr3_decoder.c,v 1.1 2001/07/20 17:28:21 ehasenle Exp $ + * + * dxr3 video and spu decoder plugin. Accepts the video and spu data + * from XINE and sends it directly to the corresponding dxr3 devices. + * Takes precedence over the libmpeg2 and libspudec due to a higher + * priority. + */ + + +#include <stdlib.h> +#include <string.h> + +#include <buffer.h> +#include <xine_internal.h> +#include <linux/soundcard.h> +#include <linux/em8300.h> + +#include <stdio.h> +#include <errno.h> +#include <stdint.h> +#include <unistd.h> +#include <fcntl.h> +#include <sys/types.h> +#include <sys/ioctl.h> +#include <inttypes.h> + +char devname[]="/dev/em8300"; + +typedef struct dxr3_decoder_s { + video_decoder_t video_decoder; + vo_instance_t *video_out; + + int fd_video; +#if 0 + int fd_control; +#endif + int last_pts; +} dxr3_decoder_t; + +static int dxr3_tested = 0; +static int dxr3_ok; + +static void dxr3_presence_test() +{ + int fd, val; + dxr3_tested = 1; + dxr3_ok = 0; + + if ((fd = open(devname, O_WRONLY))<0) { + fprintf(stderr, "dxr3: not detected. (%s: %s)\n", + devname, strerror(errno)); + return; + } + if (ioctl(fd, EM8300_IOCTL_GET_AUDIOMODE, &val)<0) { + fprintf(stderr, "dxr3: ioctl failed. (%s)\n", strerror(errno)); + return; + } + close(fd); + dxr3_ok = 1; +} + +static int dxr3_can_handle (video_decoder_t *this_gen, int buf_type) +{ + if (!dxr3_tested) + dxr3_presence_test(); + return (dxr3_ok && (buf_type & 0xFFFF0000) == BUF_VIDEO_MPEG) ; +} + +static void dxr3_init (video_decoder_t *this_gen, vo_instance_t *video_out) +{ + dxr3_decoder_t *this = (dxr3_decoder_t *) this_gen; + char tmpstr[100]; + + printf("dxr3: Entering video init, devname=%s.\n",devname); + +#if 0 + /* open control device */ + if ((this->fd_control = open(devname, O_WRONLY)) < 0) { + fprintf(stderr, "dxr3: Failed to open control device %s (%s)\n", + devname, strerror(errno)); + this->fd_video = -1; + return; + } +#endif + + /* open video device */ + snprintf (tmpstr, sizeof(tmpstr), "%s_mv", devname); + if ((this->fd_video = open (tmpstr, O_WRONLY)) < 0) { + fprintf(stderr, "dxr3: Failed to open video device %s (%s)\n", + tmpstr, strerror(errno)); + return; + } + + video_out->open(video_out); + this->video_out = video_out; + + this->last_pts = 0; +} + +static void dxr3_decode_data (video_decoder_t *this_gen, buf_element_t *buf) +{ + dxr3_decoder_t *this = (dxr3_decoder_t *) this_gen; + ssize_t written; + + /* The dxr3 does not need the preview-data */ + if (buf->decoder_info[0] == 0) return; + + if (buf->PTS) { + int vpts,pts; + pts = this->video_decoder.metronom->get_current_time + (this->video_decoder.metronom); + vpts = this->video_decoder.metronom->got_video_frame + (this->video_decoder.metronom, buf->PTS); + +// fprintf(stderr, "pts %d %d %d\n",vpts,pts,buf->PTS); + + if (vpts > pts && this->last_pts < vpts) + { + this->last_pts = vpts; + if (ioctl(this->fd_video,EM8300_IOCTL_VIDEO_SETSCR,&pts)) + fprintf(stderr, "dxr3: set master pts failed.\n"); + + if (ioctl(this->fd_video, EM8300_IOCTL_VIDEO_SETPTS, &vpts)) + fprintf(stderr, "dxr3: set video pts failed.\n"); + } + } + + written = write(this->fd_video, buf->content, buf->size); + if (written < 0) { + fprintf(stderr, "dxr3: video device write failed (%s)\n", + strerror(errno)); + return; + } + if (written != buf->size) + fprintf(stderr, "dxr3: Could only write %d of %d video bytes.\n", + written, buf->size); +} + +static void dxr3_close (video_decoder_t *this_gen) +{ + dxr3_decoder_t *this = (dxr3_decoder_t *) this_gen; + +#if 0 + ioctl(this->fd_control, EM8300_IOCTL_SET_PLAYMODE, EM8300_PLAYMODE_STOPPED); + close(this->fd_control); +#endif + close(this->fd_video); + + this->video_out->close(this->video_out); +} + +static char *dxr3_get_id(void) { + return "dxr3-mpeg2"; +} + +video_decoder_t *init_video_decoder_plugin (int iface_version, + config_values_t *cfg) +{ + dxr3_decoder_t *this ; + + if (iface_version != 1) + return NULL; + + this = (dxr3_decoder_t *) malloc (sizeof (dxr3_decoder_t)); + + this->video_decoder.interface_version = 2; + this->video_decoder.can_handle = dxr3_can_handle; + this->video_decoder.init = dxr3_init; + this->video_decoder.decode_data = dxr3_decode_data; + this->video_decoder.close = dxr3_close; + this->video_decoder.get_identifier = dxr3_get_id; + this->video_decoder.priority = 10; + + return (video_decoder_t *) this; +} + +/* + * Second part of the dxr3 plugin: subpicture decoder + */ + +typedef struct spudec_decoder_s { + spu_decoder_t spu_decoder; + + vo_instance_t *vo_out; + int fd_spu; +} spudec_decoder_t; + +static int spudec_can_handle (spu_decoder_t *this_gen, int buf_type) +{ + if (!dxr3_tested) + dxr3_presence_test(); + return (dxr3_ok && (buf_type & 0xFFFF0000) == BUF_SPU_PACKAGE); +} + +static void spudec_init (spu_decoder_t *this_gen, vo_instance_t *vo_out) +{ + spudec_decoder_t *this = (spudec_decoder_t *) this_gen; + char tmpstr[100]; + + this->vo_out = vo_out; + + /* open spu device */ + snprintf (tmpstr, sizeof(tmpstr), "%s_sp", devname); + if ((this->fd_spu = open (tmpstr, O_WRONLY)) < 0) { + fprintf(stderr, "dxr3: Failed to open spu device %s (%s)\n", + tmpstr, strerror(errno)); + return; + } + +} + +static void spudec_decode_data (spu_decoder_t *this_gen, buf_element_t *buf) +{ + spudec_decoder_t *this = (spudec_decoder_t *) this_gen; + ssize_t written; + + /* Is this also needed for subpicture? */ + if (buf->decoder_info[0] == 0) return; + + if (buf->PTS) { + int vpts; + vpts = this->spu_decoder.metronom->got_spu_packet + (this->spu_decoder.metronom, buf->PTS, 0); + +// fprintf(stderr, "spu pts %d %d\n",vpts,buf->PTS); + if (ioctl(this->fd_spu, EM8300_IOCTL_SPU_SETPTS, &vpts)) + fprintf(stderr, "spu write failed\n"); + } + + written = write(this->fd_spu, buf->content, buf->size); + if (written < 0) { + fprintf(stderr, "dxr3: spu device write failed (%s)\n", + strerror(errno)); + return; + } + if (written != buf->size) + fprintf(stderr, "dxr3: Could only write %d of %d spu bytes.\n", + written, buf->size); +} + +static void spudec_close (spu_decoder_t *this_gen) +{ + spudec_decoder_t *this = (spudec_decoder_t *) this_gen; + + close(this->fd_spu); +} + +static char *spudec_get_id(void) +{ + return "dxr3-spudec"; +} + +spu_decoder_t *init_spu_decoder_plugin (int iface_version, + config_values_t *cfg) +{ + spudec_decoder_t *this; + + if (iface_version != 1) + return NULL; + + this = (spudec_decoder_t *) malloc (sizeof (spudec_decoder_t)); + + this->spu_decoder.interface_version = 2; + this->spu_decoder.can_handle = spudec_can_handle; + this->spu_decoder.init = spudec_init; + this->spu_decoder.decode_data = spudec_decode_data; + this->spu_decoder.close = spudec_close; + this->spu_decoder.get_identifier = spudec_get_id; + this->spu_decoder.priority = 10; + + return (spu_decoder_t *) this; +} + |