diff options
Diffstat (limited to 'src/xvplayer.cpp')
-rw-r--r-- | src/xvplayer.cpp | 704 |
1 files changed, 704 insertions, 0 deletions
diff --git a/src/xvplayer.cpp b/src/xvplayer.cpp new file mode 100644 index 0000000..7c773c3 --- /dev/null +++ b/src/xvplayer.cpp @@ -0,0 +1,704 @@ +/* This file is part of the KMPlayer application + Copyright (C) 2004 Koos Vriezen <[email protected]> + + This program 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. + + This program 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; see the file COPYING. If not, write to + the Free Software Foundation, Inc., 51 Franklin Steet, Fifth Floor, + Boston, MA 02110-1301, USA. +*/ + +#include <stdio.h> +#include <string.h> +#include <math.h> +#include <config.h> +#include <dcopclient.h> +#include <qcstring.h> +#include <qtimer.h> +#include <qfile.h> +#include <qurl.h> +#include <qthread.h> +#include <qmutex.h> +#include <qdom.h> +#include "kmplayer_backend.h" +#include "kmplayer_callback_stub.h" +#include "kmplayer_callback.h" +#include "xvplayer.h" +#include <X11/X.h> +#include <X11/Xlib.h> +#include <X11/Xutil.h> +#include <X11/keysym.h> +#include <X11/Xatom.h> +#include <X11/Xutil.h> +#include <X11/extensions/XShm.h> +#include <X11/extensions/Xvlib.h> + + +static char configfile[2048]; + +static Display *display; +static KXVideoPlayer *xvapp; +static KMPlayer::Callback_stub * callback; +static Window wid; +static GC gc; +static bool window_created = true; +static bool wants_config; +static bool verbose; +static bool running; +static bool have_freq; +static bool xv_success; +static bool reset_xv_autopaint_colorkey; +static bool reset_xv_mute; +static int xvport; +static int xv_encoding = -1; +static QString xv_norm; +static int xv_frequency; +static int screen; +static int movie_width; +static int movie_height; +static int current_volume; +static int tmp_volume; +static const int start_vol_timeout = 100; +static const int inc_vol_timeout = 20; +Atom xv_enc_atom; +Atom xv_hue_atom; +Atom xv_saturation_atom; +Atom xv_brightness_atom; +Atom xv_contrast_atom; +Atom xv_freq_atom; +Atom xv_volume_atom; +Atom xv_mute_atom; +Atom xv_autopaint_colorkey_atom; +enum { + limit_volume = 0, + limit_hue, limit_contrast, limit_brightness, limit_saturation, + limit_last +}; +static struct Limit { int min; int max; } xv_limits [limit_last]; +static QString elmentry ("entry"); +static QString elmitem ("item"); +static QString attname ("name"); +static QString atttype ("type"); +static QString attdefault ("DEFAULT"); +static QString attvalue ("value"); +//static QString attstart ("START"); +//static QString attend ("END"); +//static QString valrange ("range"); +//static QString valnum ("num"); +//static QString valbool ("bool"); +//static QString valenum ("enum"); +//static QString valstring ("string"); +static QString valtree ("tree"); +static QByteArray config_buf; + +extern "C" { + +} // extern "C" + +static void putVideo () { + XWindowAttributes attr; + XGetWindowAttributes (display, wid, &attr); + XvPutVideo (display, xvport, wid, gc, 0, 0, attr.width, attr.height, 0, 0, attr.width, attr.height); +} + +using namespace KMPlayer; + +Backend::Backend () + : DCOPObject (QCString ("Backend")) { +} + +Backend::~Backend () {} + +void Backend::setURL (QString) { +} + +void Backend::setSubTitleURL (QString) { +} + +void Backend::play (int) { + xvapp->play (); +} + +void Backend::stop () { + QTimer::singleShot (0, xvapp, SLOT (stop ())); +} + +void Backend::pause () { +} + +void Backend::seek (int, bool /*absolute*/) { +} + +void Backend::hue (int h, bool) { + if (xv_limits[limit_hue].max > xv_limits[limit_hue].min) + xvapp->hue ((h + 100) * (xv_limits[limit_hue].max - xv_limits[limit_hue].min)/200 + xv_limits[limit_hue].min); +} + +void Backend::saturation (int s, bool) { + if (xv_limits[limit_saturation].max > xv_limits[limit_saturation].min) + xvapp->saturation ((s + 100) * (xv_limits[limit_saturation].max - xv_limits[limit_saturation].min)/200 + xv_limits[limit_saturation].min); +} + +void Backend::contrast (int c, bool) { + if (xv_limits[limit_contrast].max > xv_limits[limit_contrast].min) + xvapp->contrast ((c + 100)*(xv_limits[limit_contrast].max - xv_limits[limit_contrast].min)/200 + xv_limits[limit_contrast].min); +} + +void Backend::brightness (int b, bool) { + if (xv_limits[limit_brightness].max > xv_limits[limit_brightness].min) + xvapp->brightness ((b + 100)*(xv_limits[limit_brightness].max - xv_limits[limit_brightness].min)/200 + xv_limits[limit_brightness].min); +} + +void Backend::volume (int v, bool) { + if (xv_limits[limit_volume].max > xv_limits[limit_volume].min) + xvapp->volume (v*(xv_limits[limit_volume].max - xv_limits[limit_volume].min)/100 + xv_limits[limit_volume].min); +} + +void Backend::frequency (int f) { + xvapp->frequency (f); +} + +void Backend::setAudioLang (int, QString) { +} + +void Backend::setSubtitle (int, QString) { +} + +void Backend::quit () { + delete callback; + callback = 0L; + if (running) + stop (); + else + QTimer::singleShot (0, qApp, SLOT (quit ())); +} + +bool updateConfigEntry (const QString & name, const QString & value) { + fprintf (stderr, "%s=%s\n", name.ascii (), (const char *) value.local8Bit ()); + return true; +} + +void Backend::setConfig (QByteArray data) { + QString err; + int line, column; + QDomDocument dom; + if (dom.setContent (data, false, &err, &line, &column)) { + if (dom.childNodes().length() == 1) { + for (QDomNode node = dom.firstChild().firstChild(); + !node.isNull (); + node = node.nextSibling ()) { + QDomNamedNodeMap attr = node.attributes (); + updateConfigEntry (attr.namedItem (attname).nodeValue (), + attr.namedItem (attvalue).nodeValue ()); + } + } else + err = QString ("invalid data"); + } + if (callback) + callback->errorMessage (0, err); +} + +bool Backend::isPlaying () { + return running; +} + +KXVideoPlayer::KXVideoPlayer (int _argc, char ** _argv) + : QApplication (_argc, _argv, false), mute_timer (0) { +} + +void KXVideoPlayer::init () { + int xpos = 0; + int ypos = 0; + int width = 320; + int height = 200; + + XLockDisplay(display); + if (window_created) + wid = XCreateSimpleWindow(display, XDefaultRootWindow(display), + xpos, ypos, width, height, 1, 0, 0); + if (!callback) + QTimer::singleShot (10, this, SLOT (play ())); + XSelectInput (display, wid, + (PointerMotionMask | ExposureMask | KeyPressMask | ButtonPressMask | StructureNotifyMask)); // | SubstructureNotifyMask)); + XvAdaptorInfo * ai; + unsigned int adaptors; + xv_success = true; + QDomDocument doc; + QDomElement root = doc.createElement (QString ("document")); + if (XvQueryAdaptors (display, XDefaultRootWindow (display), &adaptors, &ai) == Success) { + QDomElement elm = doc.createElement (elmentry); + elm.setAttribute (attname, QString ("XVideo")); + elm.setAttribute (atttype, valtree); + for (unsigned i = 0; i < adaptors; i++) { + if ((ai[i].type & XvInputMask) && + (ai[i].type & XvVideoMask) && + ai[i].base_id > 0) { + int port = ai[i].base_id; + fprintf (stderr, "xvport %d\n", port); + bool freq_found = false; + XvAttribute *attributes = 0L; + int nr_attr, cur_val; + attributes = XvQueryPortAttributes (display, port, &nr_attr); + if (attributes) { + for (int i = 0; i < nr_attr; i++) { + if (!strcmp (attributes[i].name, "XV_FREQ")) + freq_found = true; + Atom atom = XInternAtom (display, attributes[i].name, false); + fprintf (stderr, "%s[%d] (%d .. %d)", attributes[i].name, ( int ) atom, attributes[i].min_value, attributes[i].max_value); + if ((attributes[i].flags & XvGettable) && XvGetPortAttribute (display, port, atom, &cur_val) == Success) + fprintf (stderr, " current: %d", cur_val); + fprintf (stderr, "\n"); + } + XFree(attributes); + } + if (!xvport && ((xv_frequency > 0) == freq_found)) { + fprintf (stderr, "using xvport %d\n", port); + xvport = port; + } + if (xvport == port) + have_freq = freq_found; + XvEncodingInfo * encodings = 0L; + unsigned nr_encode; + XvQueryEncodings (display, port, &nr_encode, &encodings); + if (encodings) { + QDomElement port_item = doc.createElement (QString("Port")); + port_item.setAttribute (attvalue, QString::number (port)); + if (freq_found) + port_item.setAttribute (QString("FREQ"), QString("1")); + for (unsigned i = 0; i < nr_encode; i++) { + if (strcmp (encodings[i].name, "XV_IMAGE")) { + if (xvport == port && xv_encoding < 0 && !xv_norm.isEmpty () && QString (encodings[i].name).lower ().startsWith(xv_norm.lower ())) + xv_encoding = encodings[i].encoding_id; + if (port == xvport && encodings[i].encoding_id == xv_encoding) { + movie_width = encodings[i].width; + movie_height = encodings[i].height; + } + QDomElement item = doc.createElement (QString ("Input")); + item.setAttribute (attvalue, QString::number (encodings[i].encoding_id)); + item.setAttribute (attname, QString (encodings[i].name)); + port_item.appendChild (item); + fprintf (stderr, " encoding: %d %s\n", ( int ) encodings[i].encoding_id, encodings[i].name); + } + } + elm.appendChild (port_item); + XvFreeEncodingInfo (encodings); + } + } + } + root.appendChild (elm); + XvFreeAdaptorInfo(ai); + } + doc.appendChild (root); + QCString exp = doc.toCString (); + config_buf = exp; + //fprintf (stderr, "%s\n", (const char *)exp); + config_buf.resize (exp.length ()); // strip terminating \0 + + if (xvport <= 0) { + fprintf (stderr, "no valid xvport found\n"); + xv_success = false; + return; + } + if (window_created) { + fprintf (stderr, "map %lu\n", wid); + if (movie_width > 0 && movie_height > 0) + XResizeWindow (display, wid, movie_width, movie_height); + XMapRaised(display, wid); + XSync(display, False); + } + XUnlockDisplay(display); + if (!xv_success) + fprintf (stderr, "Failed to init %d port\n", xvport); +} + +KXVideoPlayer::~KXVideoPlayer () { + if (window_created) { + XLockDisplay (display); + fprintf (stderr, "unmap %lu\n", wid); + XUnmapWindow (display, wid); + XDestroyWindow(display, wid); + XSync (display, False); + XUnlockDisplay (display); + } + xvapp = 0L; +} + +void getConfigEntries (QByteArray & buf) { + QDomDocument doc; + QDomElement root = doc.createElement (QString ("document")); + doc.appendChild (root); + QCString exp = doc.toCString (); + buf = exp; + buf.resize (exp.length ()); // strip terminating \0 +} + +void KXVideoPlayer::play () { + fprintf (stderr, "play xv://%d:%d/%d\n", xvport, xv_encoding, xv_frequency); + if (!xv_success) + return; + if (callback && movie_width > 0 && movie_height > 0) + callback->movieParams (0, movie_width, movie_height, 1.0*movie_width/movie_height, QStringList (), QStringList ()); + XLockDisplay (display); + if (!running && XvGrabPort (display, xvport, CurrentTime) == Success) { + gc = XCreateGC (display, wid, 0, NULL); + XvSelectPortNotify (display, xvport, 1); + XvSelectVideoNotify (display, wid, 1); + int nr, cur_val; + if (XvGetPortAttribute (display, xvport, xv_autopaint_colorkey_atom, &cur_val) == Success && cur_val == 0) { + fprintf (stderr, "XV_AUTOPAINT_COLORKEY is 0\n"); + XvSetPortAttribute (display, xvport, xv_autopaint_colorkey_atom, 1); + reset_xv_autopaint_colorkey = true; + } + XvAttribute *attributes = XvQueryPortAttributes (display, xvport, &nr); + if (attributes) { + for (int i = 0; i < nr; i++) { + Limit * limit = 0; + Atom atom = XInternAtom (display, attributes[i].name, false); + if (atom == xv_volume_atom) { + limit = xv_limits + limit_volume; + XvGetPortAttribute (display, xvport, + xv_volume_atom, ¤t_volume); + } else if (atom == xv_hue_atom) { + limit = xv_limits + limit_hue; + } else if (atom == xv_saturation_atom) { + limit = xv_limits + limit_saturation; + } else if (atom == xv_brightness_atom) { + limit = xv_limits + limit_brightness; + } else if (atom == xv_contrast_atom) { + limit = xv_limits + limit_contrast; + } else + continue; + limit->min = attributes[i].min_value; + limit->max = attributes[i].max_value; + } + XFree (attributes); + } + if (xv_frequency > 0) + XvSetPortAttribute (display, xvport, xv_freq_atom, int (1.0*xv_frequency/62.5)); + if (xv_encoding >= 0) + XvSetPortAttribute (display, xvport, xv_enc_atom, xv_encoding); + if (XvGetPortAttribute (display, xvport, xv_mute_atom, &cur_val) == + Success && cur_val == 1) { + fprintf (stderr, "XV_MUTE is 1\n"); + if (xv_limits[limit_volume].min != xv_limits[limit_volume].max) { + tmp_volume = xv_limits[limit_volume].min; + XvSetPortAttribute(display, xvport, xv_volume_atom, tmp_volume); + mute_timer = startTimer (start_vol_timeout); + } + XvSetPortAttribute (display, xvport, xv_mute_atom, 0); + reset_xv_mute = true; + } + //XvGetVideo (.. + running = true; + } + if (running) { + putVideo (); + if (callback) { + callback->playing (); + callback->statusMessage ((int) KMPlayer::Callback::stat_hasvideo, QString ()); + } + } + XUnlockDisplay (display); +} + +void KXVideoPlayer::stop () { + if (mute_timer) { + killTimer (mute_timer); + mute_timer = 0; + } + if (running) { + running = false; + XLockDisplay (display); + XvStopVideo (display, xvport, wid); + if (reset_xv_autopaint_colorkey) { + XvSetPortAttribute (display, xvport, xv_autopaint_colorkey_atom, 0); + reset_xv_autopaint_colorkey = false; + } + if (reset_xv_mute) { + XvSetPortAttribute (display, xvport, xv_mute_atom, 1); + reset_xv_mute = false; + } + XvUngrabPort (display, xvport, CurrentTime); + XFreeGC (display, gc); + XClearArea (display, wid, 0, 0, 0, 0, true); + XUnlockDisplay (display); + } + if (callback) + callback->finished (); + else + QTimer::singleShot (0, qApp, SLOT (quit ())); +} + +void KXVideoPlayer::finished () { + QTimer::singleShot (10, this, SLOT (stop ())); +} + +void KXVideoPlayer::saturation (int val) { + XLockDisplay(display); + XvSetPortAttribute (display, xvport, xv_saturation_atom, val); + XFlush (display); + XUnlockDisplay(display); +} + +void KXVideoPlayer::hue (int val) { + XLockDisplay(display); + XvSetPortAttribute (display, xvport, xv_hue_atom, val); + XFlush (display); + XUnlockDisplay(display); +} + +void KXVideoPlayer::contrast (int val) { + XLockDisplay(display); + XvSetPortAttribute (display, xvport, xv_contrast_atom, val); + XFlush (display); + XUnlockDisplay(display); +} + +void KXVideoPlayer::brightness (int val) { + XLockDisplay(display); + XvSetPortAttribute (display, xvport, xv_brightness_atom, val); + XFlush (display); + XUnlockDisplay(display); +} + +void KXVideoPlayer::volume (int val) { + current_volume = val; + if (mute_timer) + return; + XLockDisplay(display); + XvSetPortAttribute (display, xvport, xv_volume_atom, val); + XFlush (display); + XUnlockDisplay(display); +} + +void KXVideoPlayer::frequency (int val) { + // this doesn't work, changing frequency kills audio for me + if (mute_timer) { + killTimer (mute_timer); + mute_timer = 0; + } + xv_frequency = val; + if (running && have_freq) { + XLockDisplay(display); + if (xv_limits[limit_volume].min != xv_limits[limit_volume].max) { + tmp_volume = xv_limits[limit_volume].min; + XvSetPortAttribute (display, xvport, xv_volume_atom, tmp_volume); + mute_timer = startTimer (start_vol_timeout); + XFlush (display); + XvSetPortAttribute (display, xvport, xv_mute_atom, 0); + } + XvSetPortAttribute (display, xvport, xv_freq_atom, int (1.0*val/6.25)); + XFlush (display); + XUnlockDisplay(display); + } +} + +void KXVideoPlayer::saveState (QSessionManager & sm) { + if (callback) + sm.setRestartHint (QSessionManager::RestartNever); +} + +void KXVideoPlayer::timerEvent (QTimerEvent * e) { + if (e->timerId () == mute_timer) { + int step = (current_volume - xv_limits[limit_volume].min) / 20; + if (step > 0 && tmp_volume == xv_limits[limit_volume].min) { + killTimer (mute_timer); + mute_timer = startTimer (inc_vol_timeout); + } + tmp_volume += step; + if (tmp_volume >= current_volume || step <= 0) { + tmp_volume = current_volume; + killTimer (mute_timer); + mute_timer = 0; + } + XLockDisplay(display); + XvSetPortAttribute (display, xvport, xv_volume_atom, tmp_volume); + XFlush (display); + XUnlockDisplay(display); + } else + killTimer (e->timerId ()); +} + +class XEventThread : public QThread { +protected: + void run () { + Time prev_click_time = 0; + int prev_click_x = 0; + int prev_click_y = 0; + while (true) { + XEvent xevent; + XNextEvent(display, &xevent); + switch(xevent.type) { + case ClientMessage: + if (xevent.xclient.format == 8 && + !strncmp(xevent.xclient.data.b, "quit_now", 8)) { + fprintf(stderr, "request quit\n"); + return; + } + break; + case KeyPress: { + XKeyEvent kevent; + KeySym ksym; + char kbuf[256]; + int len; + kevent = xevent.xkey; + XLockDisplay(display); + len = XLookupString(&kevent, kbuf, sizeof(kbuf), &ksym, NULL); + XUnlockDisplay(display); + fprintf(stderr, "keypressed 0x%x 0x%x\n", ( int ) kevent.keycode, ( int ) ksym); + switch (ksym) { + case XK_q: + case XK_Q: + xvapp->lock (); + xvapp->stop (); + xvapp->unlock (); + break; + } + break; + } + case Expose: + if(xevent.xexpose.count != 0 || xevent.xexpose.window != wid) + break; + break; + + case ConfigureNotify: + if (::running) + putVideo (); + break; + case XvVideoNotify: + fprintf (stderr, "xvevent %lu\n", ((XvEvent*)&xevent)->xvvideo.reason); + break; + case ButtonPress: { + XButtonEvent *bev = (XButtonEvent *) &xevent; + int dx = prev_click_x - bev->x; + int dy = prev_click_y - bev->y; + if (bev->time - prev_click_time < 400 && + (dx * dx + dy * dy) < 25) { + xvapp->lock (); + if (callback) + callback->toggleFullScreen (); + xvapp->unlock (); + } + prev_click_time = bev->time; + prev_click_x = bev->x; + prev_click_y = bev->y; + break; + } + default: + if (xevent.type < LASTEvent) { + //fprintf (stderr, "event %d\n", xevent.type); + } + } + } + } +}; + +int main(int argc, char **argv) { + if (!XInitThreads ()) { + fprintf (stderr, "XInitThreads () failed\n"); + return 1; + } + display = XOpenDisplay(NULL); + screen = XDefaultScreen(display); + + unsigned int ver, rel, req, evb, err; + if (XvQueryExtension (display, &ver, &rel, &req, &evb, &err) != Success) { + fprintf (stderr, "XVideo not supported on display\n"); + XCloseDisplay (display); + return 1; + } + xv_enc_atom = XInternAtom (display, "XV_ENCODING", false); + xv_hue_atom = XInternAtom (display, "XV_HUE", false); + xv_saturation_atom = XInternAtom (display, "XV_SATURATION", false); + xv_brightness_atom = XInternAtom (display, "XV_BRIGHTNESS", false); + xv_contrast_atom = XInternAtom (display, "XV_CONTRAST", false); + xv_freq_atom = XInternAtom (display, "XV_FREQ", false); + xv_volume_atom = XInternAtom (display, "XV_VOLUME", false); + xv_mute_atom = XInternAtom (display, "XV_MUTE", false); + xv_autopaint_colorkey_atom = XInternAtom (display, "XV_AUTOPAINT_COLORKEY", false); + + xvapp = new KXVideoPlayer (argc, argv); + + for(int i = 1; i < argc; i++) { + if (!strcmp (argv [i], "-port")) { + xvport = strtol (argv [++i], 0L, 10); + } else if (!strcmp (argv [i], "-wid") || !strcmp (argv [i], "-window-id")) { + wid = atol (argv [++i]); + window_created = false; + } else if (!strcmp (argv [i], "-root")) { + wid = XDefaultRootWindow (display); + window_created = false; + } else if (!strcmp (argv [i], "-window")) { + ; + } else if (!strcmp (argv [i], "-v")) { + verbose = true; + } else if (!strcmp (argv [i], "-c")) { + wants_config = true; + } else if (!strcmp (argv [i], "-f") && i < argc - 1) { + strncpy (configfile, argv [++i], sizeof (configfile)); + configfile[sizeof (configfile) - 1] = 0; + } else if (!strcmp (argv [i], "-cb")) { + QString str = argv [++i]; + int pos = str.find ('/'); + if (pos > -1) { + fprintf (stderr, "callback is %s %s\n", str.left (pos).ascii (), str.mid (pos + 1).ascii ()); + callback = new KMPlayer::Callback_stub + (str.left (pos).ascii (), str.mid (pos + 1).ascii ()); + } + } else if (!strcmp (argv [i], "-enc")) { + xv_encoding = strtol (argv [++i], 0L, 10); + } else if (!strcmp (argv [i], "-norm")) { + xv_norm = argv [++i]; + } else if (!strcmp (argv [i], "-freq")) { + xv_frequency = strtol (argv [++i], 0L, 10); + } else { + fprintf (stderr, "usage: %s [-port <xv port>] [-enc <encoding>] [-freq <frequency>] [-f <config file>] [-v] [(-wid|-window-id) <window>] [(-root|-window)] [-cb <DCOP callback name> [-c]]\n", argv[0]); + delete xvapp; + return 1; + } + } + + DCOPClient dcopclient; + dcopclient.registerAs ("kxvideoplayer"); + Backend player; + + XEventThread * eventThread = new XEventThread; + eventThread->start (); + + xvapp->init (); + + if (callback) + callback->started (dcopclient.appId (), config_buf); + + xvapp->exec (); + + XLockDisplay(display); + XClientMessageEvent ev = { + ClientMessage, 0, true, display, wid, + XInternAtom (display, "XVIDEO", false), 8, {"quit_now"} + }; + XSendEvent (display, wid, false, StructureNotifyMask, (XEvent *) & ev); + XFlush (display); + XUnlockDisplay(display); + eventThread->wait (500); + delete eventThread; + + xvapp->stop (); + delete xvapp; + + fprintf (stderr, "closing display\n"); + XCloseDisplay (display); + fprintf (stderr, "done\n"); + return 0; +} + +#include "xvplayer.moc" |