/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /*************************************************************************** * decoder.cc * * Mon Mar 6 20:14:30 CET 2006 * Copyright 2006 Bent Bisballe Nyeng * deva@aasimon.org ****************************************************************************/ /* * This file is part of MIaV. * * MIaV 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. * * MIaV 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 MIaV; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA. */ #include "decoder.h" #include "info.h" #define READ_DV_FROM_FILE #include "dv.h" #ifdef READ_DV_FROM_FILE #include "dvfile.h" #else/* READ_DV_FROM_FILE*/ #include "dv1394.h" #endif/* READ_DV_FROM_FILE*/ #include #include #include "control.h" #include "libdv_wrapper.h" Decoder::Decoder() { running = true; memset(pframe, 0, sizeof(pframe)); // Init an empty frame qApp->installEventFilter(this); } Decoder::~Decoder() { } void Decoder::run() { bool newconnection = true; // closesem.acquire(); // Lock the shutdown process #ifdef READ_DV_FROM_FILE dvfile reader; #else/* READ_DV_FROM_FILE*/ dv1394 reader; reader.connect(); #endif/* READ_DV_FROM_FILE*/ while(running) { char *frame = (char*)reader.readFrame(); if(!frame) continue; // An empty frame if(MIaV::control.isFrozen() == false) { pmutex.lock(); memcpy(pframe, frame, DVPACKAGE_SIZE); pmutex.unlock(); } if(MIaV::control.isRecording()) { if(newconnection) { NetworkSender *sender = new NetworkSender(MIaV::control.getCpr()); senders.push_back(sender); sender->start(); newconnection = false; } senders.back()->pushFrame(frame, false, false); } else { // Remove idle senders QLinkedList::iterator i; for (i = senders.begin(); i != senders.end(); i++) { NetworkSender *ns = *i; if(ns->queueSize() == 0) { i = senders.erase(i); ns->stop(); delete ns; } } free(frame); newconnection = true; } } // closesem.release(); // Unlock the shutdown process } char *Decoder::pframeAcquire() { pmutex.lock(); return pframe; } void Decoder::pframeRelease() { pmutex.unlock(); } void Decoder::snapshot(char *rgb) { LibDVWrapper dv; dv.setOutputBuffer(rgb, DV::BGR0); pmutex.lock(); dv.decode(pframe); pmutex.unlock(); } bool Decoder::eventFilter(QObject *o, QEvent *e) { if (e->type() == QEvent::Close) { // printf("QUIT from: %p, this: %p, testing: %p\n", o, this, qApp->activeWindow()); if(qApp->activeWindow() == (QWidget*)o) { // Ignore close events from non top level widgets running = false; // Tell the thread to stop. sleep(1);// Wait for the thread to stop. (The ugly way!) // closesem.acquire(); // Wait for the thread to stop. } } // standard event processing return false; } Status Decoder::status() { Status s; QLinkedList::iterator i; for(i = senders.begin(); i != senders.end(); i++) { NetworkSender *ns = *i; s.queue_sizes.push_back(ns->queueSize()); } return s; }