wincommevtbreaker.cpp 3.01 KB
Newer Older
James Goppert's avatar
James Goppert committed
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120
/*
 * Unofficial Qt Serial Port Library
 *
 * Copyright (c) 2010 Inbiza Systems Inc. All rights reserved.
 *
 * This program is free software: you can redistribute it and/or modify it
 * under the terms of the GNU Lesser General Public License as published by the
 * Free Software Foundation, either version 3 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 Lesser General Public License
 * along with this program.  If not, see <http://www.gnu.org/licenses/>
 *
 * author labs@inbiza.com
 */

#include <QMetaType>
#include <QDebug>
#include <QMutexLocker>
#include <QTimerEvent>
#include "commdcbhelper.h"
#include "wincommevtbreaker.h"

namespace TNX {

TimeKeeper::TimeKeeper(WinCommEvtBreaker *owner, QMutex &mutex, QObject *parent)
  : QObject(parent), mutex_(mutex)
{
  if ( !owner )
    return;

  qRegisterMetaType<HANDLE>("HANDLE");

  connect(owner, SIGNAL(startTimer(HANDLE, int)), SLOT(startWaitTimer(HANDLE, int)),
          Qt::BlockingQueuedConnection);
  connect(owner, SIGNAL(stopTimer(HANDLE)), SLOT(stopWaitTimer(HANDLE)),
          Qt::BlockingQueuedConnection);
}

/**
 *
 */
void TimeKeeper::timerEvent(QTimerEvent *event)
{
  QMutexLocker locker(&mutex_);

  killTimer(event->timerId());

  if ( !handleList_.contains(event->timerId()) )
    return;

  HANDLE commHandle = handleList_.take(event->timerId());

  // force WaitCommEvent() to exit from blocked-wait by setting different event mask

  DWORD allFlags = EV_RXCHAR | EV_TXEMPTY | EV_CTS | EV_DSR | EV_RLSD | EV_BREAK | EV_ERR | EV_RING |
                   EV_PERR | EV_RX80FULL | EV_EVENT1 | EV_EVENT2;

   if ( !SetCommMask(commHandle, allFlags) ) {
    qDebug() << QString("TimeKeeper::timerEvent(hnd: %1) failed when resetting comm event mask: %2(Err #%3)")
                .arg((int)commHandle)
                .arg(CommDCBHelper::errorText(GetLastError()))
                .arg(GetLastError());
  }
}

void TimeKeeper::startWaitTimer(HANDLE commHnd, int timeout)
{
  QMutexLocker locker(&mutex_);

  int timerId = startTimer(timeout);
  handleList_.insert(timerId, commHnd);
}

void TimeKeeper::stopWaitTimer(HANDLE commHnd)
{
  QMutexLocker locker(&mutex_);

  QList<int> timerIdList = handleList_.keys(commHnd);
  foreach (int timerId, timerIdList) {
    killTimer(timerId);
    handleList_.remove(timerId);
  }
}

/**
 *
 */
void WinCommEvtBreaker::run()
{
  // store this thread's id
  threadId_ = QThread::currentThreadId();
  timeKeeper_ = new TimeKeeper(this, mutex_); // as owner, not as parent

  Q_CHECK_PTR(timeKeeper_);

  exec();

  delete timeKeeper_;
}

void WinCommEvtBreaker::startWaitTimer(HANDLE commHnd, int timeout)
{
  emit startTimer(commHnd, timeout);
}

void WinCommEvtBreaker::stopWaitTimer(HANDLE commHnd)
{
  emit stopTimer(commHnd);
}


} // namespace