diff --git a/3rdparty/qextserialport/ChangeLog b/3rdparty/qextserialport/ChangeLog deleted file mode 100644 index 49361d01..00000000 --- a/3rdparty/qextserialport/ChangeLog +++ /dev/null @@ -1,267 +0,0 @@ -Change history for QextSerialPort (formerly QwSerialPort): -(Lines beginning with + represent new functionality, * represent changed or -fixed functionality, - represent removed or deprecated functionality) - -Version 1.2 rc (2012 Debao Zhang) - * Build-system refactor - * Issue 145 : Custom baud support for MacOS - * Issue 36 : Fix Devices Notification for Vista - * Issue 54 and Issue 108 : Try detect more ports for windows - * Issue 139 : Adjust the name of generated library - - QextWinEventNotifier has been removed - -Version 1.2 beta2 (2012 Debao Zhang) - * Issue 124 : implement canReadLine - * Issue 122 : make Dtr default to TRUE under Windows. - * Issue 127 : fix QObject::MoveToThread brokes SerialPort on Windows - * Issue 129 : Add custom baud support for Windows. - * Issue 131 : Make sure portName returned by QextSerialEnumerator can be used by QextSerialPort - * Issue 134 : Make "make install" really works - * Issue 2: device discovery / removal notification on linux (read config_example.pri to figure out how to enable it.) - -Version 1.2 beta1 (2012 Debao Zhang) - * D-pointer and Q_PRIVATE_SLOT are used to moving private members from QextSerialPort to QextSerialPortPrivate - * qdoc3 instead of doxygen is used for generating documents - * MIT license header add to all sources files - + add a helper class QextWinEventNotifier for windows user, when user's SDK doesnot contain Qt's private files, this class will be auto selected. - -Version 1.2win-alpha (2007 Michal Policht) - + Added QextSerialEnumerator pre-alpha. Works under W2k and later versions of Windows. - + Event driven mechanism (alternative to polling) is now available on Windows. - - Removed default (=0) parameter from open() functions. - * Fixed bug #1714917 in Win_QextSerialPort::close() method (by Kurt). - * Fixed problem with lack of proper blocking in readData() on win32 (by Brandon Fosdick). - * Removed QT_THREAD_SUPPORT option. Now QextSerialPort must be always compiled with threads support. - * Mutexes are not static. - * setTimeout() now accepts only one parameter. - * bytesAvailable() on POSIX now shows 0 bytes instead of -1 when no bytes are available. - * bytesAvailable() is const. - * native POSIX file descriptors instead of QFile->handle() calls - + POSIX: Save and restore original termios when opening and closing the device - * POSIX: Only disable special characters on systems that support it - * POSIX: Use cfmakeraw(3) to get a non-canonical termios - + POSIX: Call close(2) in close() to actually close the device - -Version 1.1 (official release) - -Version 1.0.1 - * Minor changes (mostly in test application) - -Version 1.0.0e (by Micha? Policht) - * Fixed bytesAvailable(). Includes buffered bytes to the result. - + Added isSequential() method. - + Provided test application - -Version 1.0.0d ( changes by Micha? Policht ) - - Removed isOpen() overriden declaration/implementation from qextserialport's classes. isOpen() relies on QIODevice now. - - Removed bool portOpen variable. Replaced by internal QIODevice.openMode. - - Removed getChar(), putChar() overriden declaration/implementation. QIODevice can handle this. - * Calling open() with specified OpenMode invokes QIODevice::open() which result in proper openMode setting. - * readData(), writeData() are protected as in QIODevice declaration. - * QIODevice:: read() and write() function are working now (use them instead of readData() writeData()). - * readData(), writeData() don't check if port is open any more (read() and write() assures that). The same behaviour can be found in QFile for example. - * Fixed readLine(). - - * Fixed randomly crash on deletion bug on Windows ( by Stuart Nixon ) - http://lists.trolltech.com/qt-interest/2007-02/thread00340-0.html#msg00351 - -Version 0.9 (March 3, 2005) Stefan Sander : - + Added a new precompiler constant, _TTY_FREEBSD_ - to support FreeBSD port names. - + Added _TTY_WIN_ constant in qextserialport.pro win32:DEFINES - to have Windows port names as default when compiling on it. - - Removed construct() call from QextSerialBase constructors, - it is called indirectly through Win_QextSerialPort::construct() - and Posix_QextSerialPort::construct(). - + Added construct() call to Win_QextSerialPort constructors. - + Added setTimeout(0, 500) call to Win_QextSerialPort::construct(). - - Removed setTimeout(0, 500) call from Win_QextSerialPort(const char* name). - * Fixed Posix_QextSerialPort::open(int) control flow, now the port settings - are only applied if the associated file could be opened. - * Fixed masking CR to NL, in Posix_CommConfig.c_iflag - -Version 0.8 (, 2003) (Alpha release): - * Added code to set the port timeouts in Win_QextSerialPort's default - constructor. - * Fixed Posix_QextSerialPort::construct() to set up the port correctly. - * Fixed syntax errors in 2 ioctl() calls in posix_QextSerialPort. - * lastError is now initialized to E_NO_ERROR in the QextSerialBase - constructor. - * The select() call in posix_QextSerialPort::bytesWaiting() is now - properly coded. Previously it would always time out. - * Fixed runtime errors in the ioctl() calls for - Posix_QextSerialPort::setDtr() and Posix_QextSerialPort::setRts(). - Thanks to Marc Pignat. - -Version 0.7 (June 15, 2002) : - (0.61 - unofficial release) - * Fixed a small bug in the initializations of the static members when - QT_THREAD_SUPPORT was defined. - * Fixed a bug that caused Borland's compiler to choke on Windows platforms - (which perversely actually stemmed from a shortcoming of Visual C++ that - Borland doesn't have). - - (0.62 - unofficial release) - * Fixed a bug that gave Q_LONG the wrong typedef for QT versions prior to - 3.0. - - (0.63 - unofficial release) - * Fixed 2 incorrect references to Posix_Comm_Config. - * Fixed scoping of Posix_QextSerialPort::operator=(). - * Posix_QextSerialPort::construct should now be coded correctly. - * Fixed return type for Posix_QextSerialPort::size(). - - (0.64 - unofficial release) - * Fixed all the port settings functions to work properly when opening the - port for the first time - previously none of the settings were being - applied when the port was opened. - * Fixed an oversight in Win_QextSerialPort::open() that caused the setting - of port parameters to fail on NT and 2000 systems. - - (0.7 - official release) - * Fixed some calls to QextSerialBase constructors that no longer exist on - the POSIX side. - * Fixed the bad memcpy()'s in the POSIX copy constructor. - * Fixed the Offset scoping problem under gcc 2.95. - * The CBAUD flag has been deprecated on some POSIX systems. Fixed - Posix_QextSerialPort::setBaudRate() to reflect this. - * Added construct() calls to all of the Posix_QextSerialPort constructors. - * Fixed double (and conflicting) typedefs of Offset when using QT versions - prior to 3.0 - * Changed the call to CreateFile() to CreateFileA() in - Win_QextSerialPort.cpp. This should get rid of problems for those using - Unicode or other multibyte character sets for their string literals. - * A few tweaks to the documentation. - - - Removed the protected Posix_Handle variable from Posix_QextSerialPort. - -Version 0.6 (March 11, 2002) : - + Added a new precompiler constant, QTVER_PRE_30. QT3 changed the return - types of some QIODevice functions. Therefore, if compiling on versions - of QT prior to 3.0, you should always define QTVER_PRE_30 in your project. - Also had to add some preprocessor blocks to support both 3.0 and earlier - versions of QT. - + Added implementations of 2 of the new constructors added in 0.5 to both - Win_QextSerialPort and Posix_QextSerialPort. - - * The scoping of the enums used in the PortSettings struct has been fixed. - * QObject inheritance has been removed. This should not affect the - functionality of the classes. - * Replaced a few stray references to mutex->unlock() with UNLOCK_MUTEX() in - the Windows code. - * Fixed several runtime errors caused by calling nonexistent members of - QextSerialBase. - * Fixed a whole bunch of little things that were causing MSVC to choke when - compiling for Windows. - -Version 0.5 (February 15, 2002): - + There are 4 new macros (LOCK_MUTEX, UNLOCK_MUTEX, TTY_WARNING, and - TTY_PORTABILITY_WARNING) that replace most of those ugly #ifdef blocks in - the code. - + In place of the old namingConvention stuff, there is a new function, - setName(). It is used to set the name of the device to be associated with - the object. The new name() function can be used to retrieve the device - name, which is stored in the new member variable portName. - + There is a new version of open() that takes a const char* as a parameter. - It can be used to specify the name of the device when it is opened rather - than at construction time. - - * 3 constructors have been removed and 3 more added. There is now a copy - constructor (and operator=()) as well as a constructor that takes a - PortSettings structure as a parameter, and another that takes both a - device name and a PortSettings structure. As a result of these changes - the PortSettings structure declaration is no longer local to the - QextSerialBase class. All of the removed constructors had to do with - the setNamingConvention() system. - * The static mutex member should now be reference-counted and only deleted - when it is no longer referenced. - * Most of the object construction duties have been pushed back into - QextSerialBase - * Fixed a couple resource leaks, mostly to do with unlocking the mutex - properly - - - Removed the setNamingConvention() nonsense. - - Removed all QStrings and calls to sprintf() for thread compatibility. - - Removed setNumber() functions as well as the portNumber member variable, - as they were only necessary under the setNamingConvention() system. - - I am grateful to Jorg Preiss (Preisz? Sorry, American keyboards don't have - an ess-tset character ;)) for his invaluable input on most of the changes - that went into this version. - -Version 0.4 (March 20, 2001): - + All of the classes now derive from QObject as well as QIODevice. This - is pretty much useless at the moment - signals and slots may be used - to implement asynchronous communications in a future version - + Added configurable timeouts via the setTimeout() function. The default - timeout for read and write operations is now 500 milliseconds - + There is now a functional .pro file for the library (thanks to - Gunnstein Lye) - + The prefixes for all of the classes have changed from Qw to Qext, in - compliance with the qt-addons project standard - - * Fixed a bug that caused port settings to be restored incorrectly when - switching ports with setNumber() - * Minor changes to QextSerialBase::setNumber(). Functionality should now - reflect the documentation, which has also been updated to reflect the - changes that went in on version 0.3. - * Some fixes to the documentation. The Posix_QextSerialPort and - Win_QextSerialPort classes should no longer have any unnecessary - references to inapplicable platforms, and the documentation for open() has - been updated. - * Should now compile without QT_THREAD_SUPPORT defined (ie, in single- - threaded environments), although it will require slight changes to the - makefile (tmake "CONFIG-=thread" should work) - * Fixed a few compilation issues, especially on the POSIX side (should - compile under Linux now :)) - * POSIX code is a little cleaner and more efficient - * Various small fixes to the documentation - * Constants now follow a consistent naming convention, with underscores at - the beginning and end of each. For example TTY_POSIX has become - _TTY_POSIX_ - -Version 0.3 (Feb. 14, 2001): - + Added a warning that appears when QwSerialPort is compiled on a POSIX - platform that does not implement 76800 baud operation. In this situation - QwSerialPort will also switch to 57600 baud. - + Major code reorganization - there are now 4 classes instead of 1. This - should remove a lot of the #ifdef...#else...#endif constructs and - hopefully make the code easier to read. Including the class in your - project is still done by including QwSerialPort.h and instantiating a - QwSerialPort object. - - * The serial port associated with a QwSerialPort object is no longer - opened on construction, or upon calling the setNumber() function. You - must now explicitly call open() to open the port. - -Version 0.2 (Jan. 3, 2001): - + Added lastError() function with rudimentary error codes - + Better documentation - + Added ability to examine the empty/not empty state of a port's input - buffer with atEnd() - + Added ability to retrieve the number of bytes in a port's input buffer - with size() (thanks to Olivier Tubach) - + Added ability to turn off portability warnings by defining - TTY_NOWARN_PORT in your project - + Added ability to turn off all warning messages by defining TTY_NOWARN - in your project - + Added ability to select POSIX serial functions in Windows NT/2000 by - defining TTY_POSIX in your project (untested) - + Added control over RTS and DTR lines with setRts() and setDtr() - respectively - + Added ability to query line status using lineStatus(). - + Added readLine() functionality (thanks to Olivier Tubach) - + Added bytesWaiting(), a non-const/thread-safe version of size() - + The class should now be thread-safe through the use of a recursive - QMutex (untested) - - * Fixed a bug that could cause hardware flow control not to work on some - POSIX systems - * Put in a few missing fileno() calls in the POSIX code - * Fixed a few syntax errors that caused compilation to fail on POSIX systems - - - BAUD0 is no longer a valid baud rate setting - to drop the DTR line, - call setDtr(FALSE) - -Version 0.1 (Dec. 11, 2000): - Initial public release. diff --git a/3rdparty/qextserialport/LICENSE b/3rdparty/qextserialport/LICENSE deleted file mode 100644 index bfa683e2..00000000 --- a/3rdparty/qextserialport/LICENSE +++ /dev/null @@ -1,94 +0,0 @@ -From QextSerialPort 1.2-beta on, we use MIT license for QextSerialPort project. - -== License == - - Copyright (c) 2000-2003 Wayne Roth - Copyright (c) 2004-2007 Stefan Sander - Copyright (c) 2007 Michal Policht - Copyright (c) 2008 Brandon Fosdick - Copyright (c) 2009-2010 Liam Staskawicz - Copyright (c) 2011 Debao Zhang - - Web: http://code.google.com/p/qextserialport/ - - Permission is hereby granted, free of charge, to any person obtaining - a copy of this software and associated documentation files (the - "Software"), to deal in the Software without restriction, including - without limitation the rights to use, copy, modify, merge, publish, - distribute, sublicense, and/or sell copies of the Software, and to - permit persons to whom the Software is furnished to do so, subject to - the following conditions: - - The above copyright notice and this permission notice shall be - included in all copies or substantial portions of the Software. - - THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, - EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF - MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND - NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE - LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION - OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION - WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. - -== Why license needed? == - - Many users complains that, without a proper licence they can not use this library. - - * http://groups.google.com/group/qextserialport/browse_thread/thread/0e8756920b01da82 - - Hi, - we are considering using a modified version of QExtSerialPort in one of our - projects (Qt Creator, http://qt.gitorious.org/qt-creator). - Would it be possible to add license header information or a license file to the - QExtSerialPort code base? - This would make re-use of the code base easier. - If that is not possible, could we redistribute the source code with BSD- - license headers manually added? - -And - - I am also considering packaging the software for Debian, but I - couldn't do it yet just because of the license. - - * http://code.google.com/p/qextserialport/issues/detail?id=8 - - Questions: - Can I use qextserialport in a commercial product? - If yes, how? - Compile it in? I guess no. - If I can use it as a library, how should the README be formulated? - Is the "MIT license" from 2008 appropriate? - -== Why can we use MIT? == - -Form the history of [http://lists.trolltech.com/qt-interest/2004-12/msg01022.html qt-interest mail list] - - * Wayne Roth, the original author of the project, had said that: - - the code is in the public domain. Do whatever you like with it. Right - now I have too many other things to do to put any serious time into - fixing it. Trolltech should be aware of this already; they asked - about a license when they offered to host the tarball. - - * Stefan Sander, the maintainer of qextserialport on sourceforge, said that - - Hello, - My project registration at !SourceForge have been approved. - http://www.sf.net/projects/qextserialport - I thought an initial licence of Public Domain would be best solution. - Someone wrote: - Because its public domain, some could fork it under different licenses - - -And from [http://groups.google.com/group/qextserialport/browse_thread/thread/fbcddbfb4a0b5a51?pli=1 this thread] on qesp mail list, we can see that, current maintainers and users agree with a MIT licence. - - * Brandon Fosdick, - - I would vote for BSD or MIT :) - - * Liam Staskawicz, - - That works for me - let's call it MIT and go for it :) - -And from [[https://groups.google.com/forum/?fromgroups#!topic/qextserialport/P_5TrNHBICE this other thread]] on the same mailing list: - - * Michal Policht, - - I agree to license. diff --git a/3rdparty/qextserialport/README b/3rdparty/qextserialport/README deleted file mode 100644 index f7dc1440..00000000 --- a/3rdparty/qextserialport/README +++ /dev/null @@ -1,64 +0,0 @@ -= About QextSerialPort = - -QextSerialPort provides an interface to old fashioned serial ports for Qt-based applications. It currently supports Mac OS X, Windows, Linux, FreeBSD. - - http://code.google.com/p/qextserialport/ - -== How to use (1) == - - * Download the source code. - - * Put the source code in any directory you like. For example, 3rdparty: - - |-- project.pro - |-- .... - |-- 3rdparty\ - | |-- qextserialport\ - | | - - * Add following line to your qmake project file: - - include(3rdparty/qextserialport/src/qextserialport.pri) - - * Using QextSerialPort in your code. Enjoy it! - - #include "qextserialport.h" - .... - QextSerialPort * port = new QextSerialPort(); - .... - -== How to use (2) == -It's very easy to compile QextSerialPort directly into your application -(see above section), however, we would prefer to use it as a shared library. - - * Download the source code, and put it in any location you like. - - * Run following command to generate library. - - qmake - make (or nmake) - sudo make install (or nmake install) - - * Add following line to your project's file - - CONFIG += extserialport - - * Using QextSerialPort in your code. Enjoy it! - - #include "qextserialport.h" - .... - QextSerialPort * port = new QextSerialPort(); - .... - -== Build Documents == - * Run - qmake - make docs - - * Note: More information can be found in doc/readme.txt - -== Build examples == - * Goto examples directory, then run - qmake (or qmake -r) - make (or nmake) - diff --git a/3rdparty/qextserialport/doc/doc.pri b/3rdparty/qextserialport/doc/doc.pri deleted file mode 100644 index 5d1babb2..00000000 --- a/3rdparty/qextserialport/doc/doc.pri +++ /dev/null @@ -1,12 +0,0 @@ -OTHER_FILES += $$PWD/qextserialport.qdocconf - -#name of qdoc3 has been changed to qdoc under Qt5 -QESP_QDOC = qdoc -lessThan(QT_MAJOR_VERSION, 5):QESP_QDOC = qdoc3 - -docs_target.target = docs -docs_target.commands = $$QESP_QDOC $$PWD/qextserialport.qdocconf - -QMAKE_EXTRA_TARGETS = docs_target -QMAKE_CLEAN += "-r $$PWD/html" - diff --git a/3rdparty/qextserialport/doc/examples/enumerator.qdoc b/3rdparty/qextserialport/doc/examples/enumerator.qdoc deleted file mode 100644 index eab0ff8f..00000000 --- a/3rdparty/qextserialport/doc/examples/enumerator.qdoc +++ /dev/null @@ -1,16 +0,0 @@ -/*! - \example examples/enumerator - \title enumerator Demo - - The example demonstrates how to use QextSerialEnumerator. - - Include the proper header file - \snippet examples/enumerator/main.cpp 0 - - Get available ports in the system. - \snippet examples/enumerator/main.cpp 1 - - Output - \snippet examples/enumerator/main.cpp 2 -*/ - diff --git a/3rdparty/qextserialport/doc/examples/images/uartassistant.png b/3rdparty/qextserialport/doc/examples/images/uartassistant.png deleted file mode 100644 index 1b500b86..00000000 Binary files a/3rdparty/qextserialport/doc/examples/images/uartassistant.png and /dev/null differ diff --git a/3rdparty/qextserialport/doc/examples/qespta.qdoc b/3rdparty/qextserialport/doc/examples/qespta.qdoc deleted file mode 100644 index 035ab294..00000000 --- a/3rdparty/qextserialport/doc/examples/qespta.qdoc +++ /dev/null @@ -1,7 +0,0 @@ -/*! - \example examples/qespta - \title qespta Demo - - The example demonstrates how to use QextSerialPort. -*/ - diff --git a/3rdparty/qextserialport/doc/examples/uartassistant.qdoc b/3rdparty/qextserialport/doc/examples/uartassistant.qdoc deleted file mode 100644 index 7ad6b8ef..00000000 --- a/3rdparty/qextserialport/doc/examples/uartassistant.qdoc +++ /dev/null @@ -1,24 +0,0 @@ -/*! - \example examples/uartassistant - \title UartAssistant Demo - - The example demonstrates how to use QextSerialPort. - - Initialze UI element. - \snippet examples/uartassistant/dialog.cpp 0 - - Initialize serial port - \snippet examples/uartassistant/dialog.cpp 1 - - port Settings - \snippet examples/uartassistant/dialog.cpp 2 - - Open or Close the port. - \snippet examples/uartassistant/dialog.cpp 3 - - Read from or Write to the port - \snippet examples/uartassistant/dialog.cpp 4 - - \image uartassistant.png -*/ - diff --git a/3rdparty/qextserialport/doc/index.qdoc b/3rdparty/qextserialport/doc/index.qdoc deleted file mode 100644 index fad749bc..00000000 --- a/3rdparty/qextserialport/doc/index.qdoc +++ /dev/null @@ -1,199 +0,0 @@ -/*! - \page index.html - \title QextSerialPort Manual - - \section1 Overview - QextSerialPort provides an interface to old fashioned serial ports for - Qt-based applications. It currently supports Mac OS X, Windows, Linux, FreeBSD. - - From QextSerialPort 1.2-beta on, license of the project has been changed to MIT. - - \list - \o Revision 0.9.x is Qt 2 & 3 compatible. - \o Revision 1.x.x is Qt 4 compatible. - \o From revision 1.2beta1 on, Qt 5 support is added. - \endlist - - - \section1 Classes - \list - \o \l QextSerialPort encapsulates a serial port on both POSIX and Windows systems. - \o \l QextSerialEnumerator enumerates ports currently available in the system. - \endlist - - \section1 Getting Started - - - \section2 Usage(1): Source Code Only - - The package contains a qextserialport.pri file that allows you to integrate the component into programs that use qmake for the build step. - - Download the source code. - Put the source code in any directory you like. For example, 3rdparty: - - \code - |-- project.pro - |-- .... - |-- 3rdparty\ - | |-- qextserialport\ - | | - \endcode - - Add following line to your qmake project file: - \code - include(pathToPri/qextserialport.pri) - \endcode - - Then, using QextSerialPort in your code - \code - #include "qextserialport.h" - ... - MyClass::MyClass() - { - port = new QextSerialPort("COM1"); - connect(port, SIGNAL(readyRead()), this, SLOT(onDataAvailable())); - port->open(); - } - - void MyClass::onDataAvailable() - { - QByteArray data = port->readAll(); - processNewData(usbdata); - } - \endcode - - \section2 Usage(2): shared library - Although QextSerialPort can be directly compiled into your application, You may prefer - to use QextSerailPort as an library, which is very easy too. - - 1. Download the source code, and put it in any location you like. - - 2. Goto the top level directory ,run following command to generate library. - - \code - qmake - sudo make install (or nmake install) - \endcode - - 3. Add following line to your project's file - - \code - CONFIG += extserialport - \endcode - - 4. Using QextSerialPort in your code. Enjoy it! - - \code - #include "qextserialport.h" - .... - QextSerialPort * port = new QextSerialPort(); - .... - \endcode - - \section2 Usage(3): Static library - - Someone prefer to use QextSerailPort as static library. - - Open the project file: qextserialport.pro, add uncomment follow line - - \code - # CONFIG += qesp_static - \endcode - - Then follow the same steps as shared library - - \code - qmake - sudo make install - \endcode - - The static library, the header files, and the feature file will be installed to your system. - - Add following line to your qmake's project file: - - \code - CONFIG += extserialport - \endcode - - \section1 Platform Special - - \section2 For MacX: Build as framework - - Open the project file: *qextserialport.pro*, and uncomment follow line - - \code - # CONFIG += qesp_mac_framework - \endcode - - Then follow the same steps as shared library, Goto the top level directory , and run - - \code - qmake - sudo make install - \endcode - - The framework which includes libraries and the header files, and the feature file will be installed to your system. - - Add following line to your qmake's project file: - - \code - CONFIG += extserialport - \endcode - - \section2 For Linux: Enable udev - - Open the project file: *qextserialport.pro*, uncomment follow line - - \code - #linux*:CONFIG += qesp_linux_udev - \endcode - - Note, If you are using the usage(1), Add following line before include the qextserialport.pri file. - \code - CONFIG += qesp_linux_udev - \endcode - - - \section2 Build documents - \code - make docs - \endcode - - \section1 Examples - \list - \o \l examples/enumerator - \o \l examples/qespta - \o \l examples/uartassistant - \endlist - - \section1 Resources - \section2 Nokia(Trolltech) - \list - \o \l {http://doc.trolltech.com/qq/qq12-iodevice.html} {Writing a Custom I/O Device} - \o \l {http://doc.trolltech.com/3.3/qiodevice.html} {Qt 3.3: QIODevice Class Reference} - \o \l {http://doc.trolltech.com/4.7/qiodevice.html} {Qt 4.7: QIODevice Class Reference} - \endlist - \section2 MSDN - \list - \o \l {http://msdn.microsoft.com/library/default.asp?url=/library/en-us/devio/base/communications_resources.asp} {Communications Resources} - \o \l {http://msdn.microsoft.com/library/default.asp?url=/library/en-us/devio/base/about_communications_resources.asp} {About Communications Resources} - \o \l {http://msdn.microsoft.com/library/default.asp?url=/library/en-us/devio/base/using_communications_resources.asp}{Using Communications Resources} - \o \l {http://msdn.microsoft.com/library/default.asp?url=/library/en-us/devio/base/communications_functions.asp} {Communications Functions} - \o \l {http://msdn.microsoft.com/library/default.asp?url=/library/en-us/devio/base/communications_structures.asp} {Communications Structures} - \endlist - \section2 TLDP - \list - \o \l {http://www.tldp.org/HOWTO/Serial-HOWTO.html}{Serial HOWTO} - \o \l {http://www.tldp.org/HOWTO/Serial-Programming-HOWTO/}{Serial Programming HOWTO} - \endlist - \section2 Other - \list - \o \l {http://www.easysw.com/~mike/serial/serial.html} {Serial Programming Guide for POSIX Operating Systems} - \endlist - -*/ - -/*! - \page classes.html - \generatelist annotatedclasses -*/ diff --git a/3rdparty/qextserialport/doc/qextserialport.qdocconf b/3rdparty/qextserialport/doc/qextserialport.qdocconf deleted file mode 100644 index 5e7abfd9..00000000 --- a/3rdparty/qextserialport/doc/qextserialport.qdocconf +++ /dev/null @@ -1,53 +0,0 @@ -# Run qdoc3 from the directory that contains this file. -project = qesp -description = QextSerialPort Reference Documentation -url = http://code.google.com/p/qextserialport - -outputencoding = UTF-8 -language = Cpp - -#Paths are relative to the location of this file -headerdirs = . ../src -sourcedirs = . ../src -exampledirs = ../examples .. -imagedirs = ./examples/images images - -Cpp.ignoretokens = QEXTSERIALPORT_EXPORT - -indexes = $QTDIR/doc/html/qt.index - -qhp.projects = qesp -qhp.qesp.file = qesp.qhp -qhp.qesp.namespace = com.google.code.qextserialport.120 -qhp.qesp.virtualFolder = qdoc -qhp.qesp.indexTitle = QextSerialPort Reference Documentation -qhp.qesp.indexRoot = -qhp.qesp.extraFiles = style/style.css - - -#------------------------------------------------------------------ -outputdir = html -outputformats = HTML - -headers.fileextensions = "*.h" -sources.fileextensions = "*.cpp *.qdoc" - -HTML.templatedir = . -HTML.stylesheets = style/style.css - -HTML.headerstyles = " \n" -HTML.endheader = "\n" - -HTML.postheader = "\n" \ - "\n" \ - "
" \ - "Home ·" \ - " All Classes ·" \ - "
" - -HTML.footer = "


\n" \ - "\n" \ - "\n" \ - "\n" \ - "\n" \ - "
Copyright © 2000-2012QextSerialPort Project
QextSerialPort Manual
" diff --git a/3rdparty/qextserialport/doc/readme.txt b/3rdparty/qextserialport/doc/readme.txt deleted file mode 100644 index fe7ee326..00000000 --- a/3rdparty/qextserialport/doc/readme.txt +++ /dev/null @@ -1,35 +0,0 @@ -Note: - - If you are using qextserialport-XXX.tar.gz, the qesp.qch and - html files have been provided. - - Open the file "html/index.html" using your web browser. - Or integrated the "html/qesp.qch" into your QtCreator. - - -== How to generate help files? == - -Simply run following commands at toplevel directory. - qmake - make docs - -Or run the following command at this directory - qdoc3 qextserialport.qdocconf - -Then a folder called "html" will be generated. -Open the file "html/index.html" using your web browser. - -== How to integrated into Qt Creator or Qt Assistant? == - -Once the html files are generated. run following commands - cd doc/html - qhelpgenerator qesp.qhp - -A file called "qesp.qch" will be generated. - -For Qt Assistant: - Edit ==> Preferences ==> Documentations ==> Add... - -For Qt Creator - Tools ==> Options ==> Help ==> Documentations ==> Add... - diff --git a/3rdparty/qextserialport/doc/style/style.css b/3rdparty/qextserialport/doc/style/style.css deleted file mode 100644 index 0a5959f9..00000000 --- a/3rdparty/qextserialport/doc/style/style.css +++ /dev/null @@ -1,137 +0,0 @@ -a:link, a:visited { - color: #00732F; - text-decoration: none; - font-weight: bold; -} - -body { - font: normal 400 14px/1.2 Arial; - margin-top: 85px; -} - -h1 { - margin: 0; -} - -h2 { - font: 500 20px/1.2 Arial; -} - -h3.fn, span.fn { - -moz-border-radius: 7px 7px 7px 7px; - -webkit-border-radius: 7px 7px 7px 7px; - border-radius: 7px 7px 7px 7px; - background-color: #F6F6F6; - border-width: 1px; - border-style: solid; - border-color: #E6E6E6; - word-spacing: 3px; - padding: 3px 5px; -} - -table, pre { - -moz-border-radius: 7px 7px 7px 7px; - -webkit-border-radius: 7px 7px 7px 7px; - border-radius: 7px 7px 7px 7px; - background-color: #F6F6F6; - border: 1px solid #E6E6E6; - border-collapse: separate; - font-size: 12px; - line-height: 1.2; - margin-bottom: 25px; - margin-left: 15px; -} - -table td { - padding: 3px 15px 3px 20px; -} - -table tr.even { - background-color: white; - color: #66666E; -} - -table tr.odd { - background-color: #F6F6F6; - color: #66666E; -} - -li { - margin-bottom: 10px; - padding-left: 12px; -} - -.cpp { - display: block; - margin: 10; - overflow: hidden; - overflow-x: hidden; - overflow-y: hidden; - padding: 20px 0 20px 0; -} - -.footer { - margin-top: 50px; -} - -.memItemLeft { - padding-right: 3px; -} - -.memItemRight { - padding: 3px 15px 3px 0; -} - -.qml { - display: block; - margin: 10; - overflow: hidden; - overflow-x: hidden; - overflow-y: hidden; - padding: 20px 0 20px 0; -} - -.qmldefault { - padding-left: 5px; - float: right; - color: red; -} - -.qmlreadonly { - padding-left: 5px; - float: right; - color: #254117; -} - -.rightAlign { - padding: 3px 5px 3px 10px; - text-align: right; -} - -.title { - background-color: white; - color: #44A51C; - font-family: Verdana; - font-size: 35px; - font-weight: normal; - left: 0; - padding-bottom: 5px; - padding-left: 16px; - padding-top: 20px; - position: absolute; - right: 0; - top: 0; -} - -.toc { - float: right; - -moz-border-radius: 7px 7px 7px 7px; - -webkit-border-radius: 7px 7px 7px 7px; - border-radius: 7px 7px 7px 7px; - background-color: #F6F6F6; - border: 1px solid #DDD; - margin: 0 20px 10px 10px; - padding: 20px 15px 20px 20px; - height: auto; - width: 200px; -} diff --git a/3rdparty/qextserialport/examples/enumerator/enumerator.pro b/3rdparty/qextserialport/examples/enumerator/enumerator.pro deleted file mode 100644 index 7eed9257..00000000 --- a/3rdparty/qextserialport/examples/enumerator/enumerator.pro +++ /dev/null @@ -1,6 +0,0 @@ -TEMPLATE = app -DEPENDPATH += . -CONFIG += console -include(../../src/qextserialport.pri) -SOURCES += main.cpp - diff --git a/3rdparty/qextserialport/examples/enumerator/main.cpp b/3rdparty/qextserialport/examples/enumerator/main.cpp deleted file mode 100644 index e3557040..00000000 --- a/3rdparty/qextserialport/examples/enumerator/main.cpp +++ /dev/null @@ -1,31 +0,0 @@ -/** - * @file main.cpp - * @brief Main file. - * @author Micha? Policht - */ -//! [0] -#include "qextserialenumerator.h" -//! [0] -#include -#include -int main() -{ - //! [1] - QList ports = QextSerialEnumerator::getPorts(); - //! [1] - qDebug() << "List of ports:"; - //! [2] - foreach (QextPortInfo info, ports) { - qDebug() << "port name:" << info.portName; - qDebug() << "friendly name:" << info.friendName; - qDebug() << "physical name:" << info.physName; - qDebug() << "enumerator name:" << info.enumName; - qDebug() << "vendor ID:" << info.vendorID; - qDebug() << "product ID:" << info.productID; - - qDebug() << "==================================="; - } - //! [2] - return 0; -} - diff --git a/3rdparty/qextserialport/examples/event/PortListener.cpp b/3rdparty/qextserialport/examples/event/PortListener.cpp deleted file mode 100644 index 57467d04..00000000 --- a/3rdparty/qextserialport/examples/event/PortListener.cpp +++ /dev/null @@ -1,43 +0,0 @@ - -#include "PortListener.h" -#include - -PortListener::PortListener(const QString &portName) -{ - qDebug() << "hi there"; - this->port = new QextSerialPort(portName, QextSerialPort::EventDriven); - port->setBaudRate(BAUD9600); - port->setFlowControl(FLOW_OFF); - port->setParity(PAR_NONE); - port->setDataBits(DATA_8); - port->setStopBits(STOP_2); - - if (port->open(QIODevice::ReadWrite) == true) { - connect(port, SIGNAL(readyRead()), this, SLOT(onReadyRead())); - connect(port, SIGNAL(dsrChanged(bool)), this, SLOT(onDsrChanged(bool))); - if (!(port->lineStatus() & LS_DSR)) - qDebug() << "warning: device is not turned on"; - qDebug() << "listening for data on" << port->portName(); - } - else { - qDebug() << "device failed to open:" << port->errorString(); - } -} - -void PortListener::onReadyRead() -{ - QByteArray bytes; - int a = port->bytesAvailable(); - bytes.resize(a); - port->read(bytes.data(), bytes.size()); - qDebug() << "bytes read:" << bytes.size(); - qDebug() << "bytes:" << bytes; -} - -void PortListener::onDsrChanged(bool status) -{ - if (status) - qDebug() << "device was turned on"; - else - qDebug() << "device was turned off"; -} diff --git a/3rdparty/qextserialport/examples/event/PortListener.h b/3rdparty/qextserialport/examples/event/PortListener.h deleted file mode 100644 index 8b298b0a..00000000 --- a/3rdparty/qextserialport/examples/event/PortListener.h +++ /dev/null @@ -1,26 +0,0 @@ - - - -#ifndef PORTLISTENER_H_ -#define PORTLISTENER_H_ - -#include -#include "qextserialport.h" - -class PortListener : public QObject -{ -Q_OBJECT -public: - PortListener(const QString &portName); - -private: - QextSerialPort *port; - -private slots: - void onReadyRead(); - void onDsrChanged(bool status); - -}; - - -#endif /*PORTLISTENER_H_*/ diff --git a/3rdparty/qextserialport/examples/event/event.pro b/3rdparty/qextserialport/examples/event/event.pro deleted file mode 100644 index 585bbb9a..00000000 --- a/3rdparty/qextserialport/examples/event/event.pro +++ /dev/null @@ -1,7 +0,0 @@ -TEMPLATE = app -DEPENDPATH += . -CONFIG += console -include(../../src/qextserialport.pri) - -SOURCES += main.cpp PortListener.cpp -HEADERS += PortListener.h diff --git a/3rdparty/qextserialport/examples/event/main.cpp b/3rdparty/qextserialport/examples/event/main.cpp deleted file mode 100644 index 976040b5..00000000 --- a/3rdparty/qextserialport/examples/event/main.cpp +++ /dev/null @@ -1,19 +0,0 @@ -/** - * @file main.cpp - * @brief Main file. - * @author Michal Policht - */ - -#include -#include "PortListener.h" - -int main(int argc, char *argv[]) -{ - QCoreApplication app(argc, argv); - - QString portName = QLatin1String("COM1"); // update this to use your port of choice - PortListener listener(portName); // signals get hooked up internally - - // start the event loop and wait for signals - return app.exec(); -} diff --git a/3rdparty/qextserialport/examples/examples.pro b/3rdparty/qextserialport/examples/examples.pro deleted file mode 100644 index ec8df641..00000000 --- a/3rdparty/qextserialport/examples/examples.pro +++ /dev/null @@ -1,5 +0,0 @@ -TEMPLATE = subdirs -SUBDIRS = qespta enumerator \ - uartassistant -win32:SUBDIRS += event - diff --git a/3rdparty/qextserialport/examples/qespta/MainWindow.cpp b/3rdparty/qextserialport/examples/qespta/MainWindow.cpp deleted file mode 100644 index 159eb7aa..00000000 --- a/3rdparty/qextserialport/examples/qespta/MainWindow.cpp +++ /dev/null @@ -1,61 +0,0 @@ -/** - * @file MainWindow.cpp - * @brief MainWindow Implementation. - * @see MainWindow.h - * @author Micha? Policht - */ - - -#include -#include -#include "MainWindow.h" -#include "MessageWindow.h" -#include "QespTest.h" - -MainWindow::MainWindow() -{ - //central widget - QespTest *qespTest = new QespTest(); - setCentralWidget(qespTest); - //bottom dock widget - MessageWindow *msgWindow = new MessageWindow(); - addDockWidget(Qt::BottomDockWidgetArea, msgWindow); - - createActions(); - createMenus(); - - setWindowTitle(tr("QextSerialPort Test Application")); -} - -void MainWindow::about() -{ - QMessageBox::about(this, tr("About "), - tr("""
" - "author: Michal Policht
" - "xpolik@users.sourceforge.net")); -} - -void MainWindow::createActions() -{ - //File actions - exitAct = new QAction(tr("E&xit"), this); - exitAct->setShortcut(tr("CTRL+D")); - exitAct->setStatusTip(tr("Exit the application")); - connect(exitAct, SIGNAL(triggered()), this, SLOT(close())); - - //Help actions - aboutAct = new QAction(tr("&About"), this); - aboutAct->setShortcut(tr("CTRL+A")); - aboutAct->setStatusTip(tr("About application")); - connect(aboutAct, SIGNAL(triggered()), this, SLOT(about())); -} - -void MainWindow::createMenus() -{ - fileMenu = menuBar()->addMenu(tr("&File")); - fileMenu->addAction(exitAct); - - helpMenu = menuBar()->addMenu(tr("&Help")); - helpMenu->addAction(aboutAct); -} - diff --git a/3rdparty/qextserialport/examples/qespta/MainWindow.h b/3rdparty/qextserialport/examples/qespta/MainWindow.h deleted file mode 100644 index 1fb28a35..00000000 --- a/3rdparty/qextserialport/examples/qespta/MainWindow.h +++ /dev/null @@ -1,38 +0,0 @@ -/** - * @file MainWindow.h - * @brief Application's Main Window. - * @see MainWindow - * @author Micha? Policht - */ - -#ifndef MAINWINDOW_H_ -#define MAINWINDOW_H_ - -#include - -class QMenu; -class QAction; - -class MainWindow : public QMainWindow -{ - Q_OBJECT - - QMenu *fileMenu; - QAction *exitAct; - QMenu *helpMenu; - QAction *aboutAct; - -private: - void createMenus(); - void createActions(); - -private slots: - void about(); - -public: - MainWindow(); - -}; - -#endif /*MAINWINDOW_H_*/ - diff --git a/3rdparty/qextserialport/examples/qespta/MessageWindow.cpp b/3rdparty/qextserialport/examples/qespta/MessageWindow.cpp deleted file mode 100644 index 5d06aa45..00000000 --- a/3rdparty/qextserialport/examples/qespta/MessageWindow.cpp +++ /dev/null @@ -1,102 +0,0 @@ -/** - * @file MessageWindow.cpp - * @brief MessageWindow Implementation. - * @see MessageWindow.h - * @author Micha? Policht - */ - -#include -#include "MessageWindow.h" -#include -#include -#include - -const char *MessageWindow::WINDOW_TITLE = "Message Window"; -MessageWindow *MessageWindow::MsgHandler = NULL; - -MessageWindow::MessageWindow(QWidget *parent, Qt::WindowFlags flags) - : QDockWidget(parent, flags), - msgTextEdit(this) -{ - setWindowTitle(tr(WINDOW_TITLE)); - msgTextEdit.setReadOnly(true); - setWidget(&msgTextEdit); - - MessageWindow::MsgHandler = this; -} - -//static -QString MessageWindow::QtMsgToQString(QtMsgType type, const char *msg) -{ - switch (type) { - case QtDebugMsg: - return QLatin1String("Debug: ")+QLatin1String(msg); - case QtWarningMsg: - return QLatin1String("Warning: ")+QLatin1String(msg); - case QtCriticalMsg: - return QLatin1String("Critical: ")+QLatin1String(msg); - case QtFatalMsg: - return QLatin1String("Fatal: ")+QLatin1String(msg); - default: - return QLatin1String("Unrecognized message type: ")+QLatin1String(msg); - } -} - -//static -void MessageWindow::AppendMsgWrapper(QtMsgType type, const char *msg) -{ - static QMutex mutex; - QMutexLocker locker(&mutex); - - if (MessageWindow::MsgHandler != NULL) - return MessageWindow::MsgHandler->postMsgEvent(type, msg); - else - fprintf(stderr, "%s", MessageWindow::QtMsgToQString(type, msg).toLatin1().data()); -} - -#if QT_VERSION >= QT_VERSION_CHECK(5,0,0) -void MessageWindow::AppendMsgWrapper(QtMsgType type, const QMessageLogContext & /*context*/, const QString &msg) -{ - AppendMsgWrapper(type, msg.toLatin1().data()); -} -#endif - -void MessageWindow::customEvent(QEvent *event) -{ - if (static_cast(event->type()) == MessageWindow::MessageEventType) - msgTextEdit.append(dynamic_cast(event)->msg); -} - -void MessageWindow::postMsgEvent(QtMsgType type, const char *msg) -{ - QString qmsg = MessageWindow::QtMsgToQString(type, msg); - switch (type) { - case QtDebugMsg: - break; - case QtWarningMsg: - qmsg.prepend(QLatin1String("")); - qmsg.append(QLatin1String("")); - break; - case QtCriticalMsg: - if (QMessageBox::critical(this, QLatin1String("Critical Error"), qmsg, - QMessageBox::Ignore, - QMessageBox::Abort, - QMessageBox::NoButton) == QMessageBox::Abort) - abort(); // core dump - qmsg.prepend(QLatin1String("")); - qmsg.append(QLatin1String("")); - break; - case QtFatalMsg: - QMessageBox::critical(this, QLatin1String("Fatal Error"), qmsg, QMessageBox::Ok, QMessageBox::NoButton, QMessageBox::NoButton); - abort(); // deliberately core dump - } - //it's impossible to change GUI directly from thread other than the main thread - //so post message encapsulated by MessageEvent to the main thread's event queue - QCoreApplication::postEvent(this, new MessageEvent(qmsg)); -} - -MessageEvent::MessageEvent(QString &msg): - QEvent(static_cast(MessageWindow::MessageEventType)) -{ - this->msg = msg; -} diff --git a/3rdparty/qextserialport/examples/qespta/MessageWindow.h b/3rdparty/qextserialport/examples/qespta/MessageWindow.h deleted file mode 100644 index 60cc164d..00000000 --- a/3rdparty/qextserialport/examples/qespta/MessageWindow.h +++ /dev/null @@ -1,84 +0,0 @@ -/** - * @file MessageWindow.h - * @brief Message Window. - * @see MessageWindow - * @author Micha? Policht - */ - -#ifndef MESSAGEWINDOW_H_ -#define MESSAGEWINDOW_H_ - -#include -#include -#include - -/** - * Message Window. Handling errors and other messages. - */ -class MessageWindow: public QDockWidget -{ - Q_OBJECT - - QTextEdit msgTextEdit; ///< Main widget. - static MessageWindow *MsgHandler; ///< Set in constructor. - static const char *WINDOW_TITLE; ///< Window title. - -private: - static QString QtMsgToQString(QtMsgType type, const char *msg); - -protected: - /** - * Handle custom events. MessageWindow hadles custom events listed in - * EventType enum. - */ - virtual void customEvent(QEvent* event); - -public: - enum EventType {MessageEventType = QEvent::User}; ///< Custom event types. - - /** - * Default constructor. - * @param parent parent widget. - * @param flags widget flags. - */ - MessageWindow(QWidget* parent = 0, Qt::WindowFlags flags = 0); - - /** - * Append message wrapper. Since ISO forbids casting member functions - * to C functions, wrapper is needed to use this class as QtMsgHandler. - * This method is thread-safe but not reentrant. - * @param type message type. - * @param msg message string. - */ - static void AppendMsgWrapper(QtMsgType type, const char *msg); -#if QT_VERSION >= QT_VERSION_CHECK(5,0,0) - static void AppendMsgWrapper(QtMsgType type, const QMessageLogContext &context, const QString &msg); -#endif - /** - * Post message event to the main event loop. This function encapsulates - * message into MessageEvent object and passes it to the main event loop. - * @param type message type. - * @param msg message string. - */ - void postMsgEvent(QtMsgType type, const char *msg); - -}; - - -/** - * Message Event. Custom event used by @ref MessageWindow to provide multi-threaded - * access. Encapsulates message inside @a msg variable. - */ -class MessageEvent: public QEvent -{ -public: - QString msg; ///< Message string. - - /** - * Contructor. - * @param msg message to post. - */ - MessageEvent(QString &msg); -}; - -#endif /*MESSAGEWINDOW_H_*/ diff --git a/3rdparty/qextserialport/examples/qespta/QespTest.cpp b/3rdparty/qextserialport/examples/qespta/QespTest.cpp deleted file mode 100644 index d338e846..00000000 --- a/3rdparty/qextserialport/examples/qespta/QespTest.cpp +++ /dev/null @@ -1,128 +0,0 @@ -/* QespTest.cpp -**************************************/ -#include "QespTest.h" -#include "qextserialport.h" -#include -#include -#include -#include -#include - - -QespTest::QespTest(QWidget *parent) - : QWidget(parent) - -{ - //modify the port settings on your own -#ifdef Q_OS_UNIX - port = new QextSerialPort(QLatin1String("/dev/ttyS0"), QextSerialPort::Polling); -#else - port = new QextSerialPort(QLatin1String("COM1"), QextSerialPort::Polling); -#endif /*Q_OS_UNIX*/ - port->setBaudRate(BAUD19200); - port->setFlowControl(FLOW_OFF); - port->setParity(PAR_NONE); - port->setDataBits(DATA_8); - port->setStopBits(STOP_2); - //set timeouts to 500 ms - port->setTimeout(500); - - message = new QLineEdit(this); - - // transmit receive - QPushButton *transmitButton = new QPushButton(tr("Transmit")); - connect(transmitButton, SIGNAL(clicked()), SLOT(transmitMsg())); - QPushButton *receiveButton = new QPushButton(tr("Receive")); - connect(receiveButton, SIGNAL(clicked()), SLOT(receiveMsg())); - QHBoxLayout *trLayout = new QHBoxLayout; - trLayout->addWidget(transmitButton); - trLayout->addWidget(receiveButton); - - //CR LF - QPushButton *CRButton = new QPushButton(tr("CR")); - connect(CRButton, SIGNAL(clicked()), SLOT(appendCR())); - QPushButton *LFButton = new QPushButton(tr("LF")); - connect(LFButton, SIGNAL(clicked()), SLOT(appendLF())); - QHBoxLayout *crlfLayout = new QHBoxLayout; - crlfLayout->addWidget(CRButton); - crlfLayout->addWidget(LFButton); - - //open close - QPushButton *openButton = new QPushButton(tr("Open")); - connect(openButton, SIGNAL(clicked()), SLOT(openPort())); - QPushButton *closeButton = new QPushButton(tr("Close")); - connect(closeButton, SIGNAL(clicked()), SLOT(closePort())); - QHBoxLayout *ocLayout = new QHBoxLayout; - ocLayout->addWidget(openButton); - ocLayout->addWidget(closeButton); - - received_msg = new QTextEdit(); - - QVBoxLayout *myVBox = new QVBoxLayout; - myVBox->addWidget(message); - myVBox->addLayout(crlfLayout); - myVBox->addLayout(trLayout); - myVBox->addLayout(ocLayout); - myVBox->addWidget(received_msg); - setLayout(myVBox); - - qDebug("isOpen : %d", port->isOpen()); -} - -QespTest::~QespTest() -{ - delete port; - port = NULL; -} - -void QespTest::transmitMsg() -{ - int i = port->write(message->text().toLatin1()); - qDebug("trasmitted : %d", i); -} - -void QespTest::receiveMsg() -{ - char buff[1024]; - int numBytes; - - numBytes = port->bytesAvailable(); - if(numBytes > 1024) - numBytes = 1024; - - int i = port->read(buff, numBytes); - if (i != -1) - buff[i] = '\0'; - else - buff[0] = '\0'; - QString msg = QLatin1String(buff); - - received_msg->append(msg); - received_msg->ensureCursorVisible(); - qDebug("bytes available: %d", numBytes); - qDebug("received: %d", i); -} - - -void QespTest::appendCR() -{ - message->insert(QLatin1String("\x0D")); -} - -void QespTest::appendLF() -{ - message->insert(QLatin1String("\x0A")); -} - -void QespTest::closePort() -{ - port->close(); - qDebug("is open: %d", port->isOpen()); -} - -void QespTest::openPort() -{ - port->open(QIODevice::ReadWrite | QIODevice::Unbuffered); - qDebug("is open: %d", port->isOpen()); -} - diff --git a/3rdparty/qextserialport/examples/qespta/QespTest.h b/3rdparty/qextserialport/examples/qespta/QespTest.h deleted file mode 100644 index 58c6815a..00000000 --- a/3rdparty/qextserialport/examples/qespta/QespTest.h +++ /dev/null @@ -1,36 +0,0 @@ -/* qesptest.h -**************************************/ -#ifndef _QESPTEST_H_ -#define _QESPTEST_H_ - -#include - -class QLineEdit; -class QTextEdit; -class QextSerialPort; -class QSpinBox; - -class QespTest : public QWidget -{ - Q_OBJECT -public: - QespTest(QWidget *parent=0); - - virtual ~QespTest(); - -private: - QLineEdit *message; - QSpinBox *delaySpinBox; - QTextEdit *received_msg; - QextSerialPort *port; - -private slots: - void transmitMsg(); - void receiveMsg(); - void appendCR(); - void appendLF(); - void closePort(); - void openPort(); -}; - -#endif diff --git a/3rdparty/qextserialport/examples/qespta/README b/3rdparty/qextserialport/examples/qespta/README deleted file mode 100644 index 643f6cd2..00000000 --- a/3rdparty/qextserialport/examples/qespta/README +++ /dev/null @@ -1,4 +0,0 @@ -This is simple application using QextSerialPort library. - -Port settings are in QespTest constructor (QespTest.cpp) - diff --git a/3rdparty/qextserialport/examples/qespta/main.cpp b/3rdparty/qextserialport/examples/qespta/main.cpp deleted file mode 100644 index 2b2c187f..00000000 --- a/3rdparty/qextserialport/examples/qespta/main.cpp +++ /dev/null @@ -1,30 +0,0 @@ -/** - * @file main.cpp - * @brief Main file. - * @author Micha? Policht - */ - -#include -#include "MainWindow.h" -#include "MessageWindow.h" - - -int main(int argc, char *argv[]) -{ - QApplication app(argc, argv); - //! [0] -#if QT_VERSION < QT_VERSION_CHECK(5,0,0) - //redirect debug messages to the MessageWindow dialog - qInstallMsgHandler(MessageWindow::AppendMsgWrapper); -#else - qInstallMessageHandler(MessageWindow::AppendMsgWrapper); -#endif - //! [0] - - MainWindow mainWindow; - mainWindow.show(); - - return app.exec(); -} - - diff --git a/3rdparty/qextserialport/examples/qespta/qespta.pro b/3rdparty/qextserialport/examples/qespta/qespta.pro deleted file mode 100644 index c18c1c71..00000000 --- a/3rdparty/qextserialport/examples/qespta/qespta.pro +++ /dev/null @@ -1,14 +0,0 @@ -TEMPLATE = app -DEPENDPATH += . -QT += core gui -contains(QT_VERSION, ^5\\..*\\..*): QT += widgets -HEADERS += MainWindow.h \ - MessageWindow.h \ - QespTest.h - -SOURCES += main.cpp \ - MainWindow.cpp \ - MessageWindow.cpp \ - QespTest.cpp - -include(../../src/qextserialport.pri) diff --git a/3rdparty/qextserialport/examples/uartassistant/dialog.cpp b/3rdparty/qextserialport/examples/uartassistant/dialog.cpp deleted file mode 100644 index 3795f0a3..00000000 --- a/3rdparty/qextserialport/examples/uartassistant/dialog.cpp +++ /dev/null @@ -1,179 +0,0 @@ -#include "qextserialport.h" -#include "qextserialenumerator.h" -#include "dialog.h" -#include "ui_dialog.h" -#include - -Dialog::Dialog(QWidget *parent) : - QDialog(parent), - ui(new Ui::Dialog) -{ - ui->setupUi(this); - - //! [0] - foreach (QextPortInfo info, QextSerialEnumerator::getPorts()) - ui->portBox->addItem(info.portName); - //make sure user can input their own port name! - ui->portBox->setEditable(true); - - ui->baudRateBox->addItem("1200", BAUD1200); - ui->baudRateBox->addItem("2400", BAUD2400); - ui->baudRateBox->addItem("4800", BAUD4800); - ui->baudRateBox->addItem("9600", BAUD9600); - ui->baudRateBox->addItem("19200", BAUD19200); - ui->baudRateBox->setCurrentIndex(3); - - ui->parityBox->addItem("NONE", PAR_NONE); - ui->parityBox->addItem("ODD", PAR_ODD); - ui->parityBox->addItem("EVEN", PAR_EVEN); - - ui->dataBitsBox->addItem("5", DATA_5); - ui->dataBitsBox->addItem("6", DATA_6); - ui->dataBitsBox->addItem("7", DATA_7); - ui->dataBitsBox->addItem("8", DATA_8); - ui->dataBitsBox->setCurrentIndex(3); - - ui->stopBitsBox->addItem("1", STOP_1); - ui->stopBitsBox->addItem("2", STOP_2); - - ui->queryModeBox->addItem("Polling", QextSerialPort::Polling); - ui->queryModeBox->addItem("EventDriven", QextSerialPort::EventDriven); - //! [0] - - ui->led->turnOff(); - - timer = new QTimer(this); - timer->setInterval(40); - //! [1] - PortSettings settings = {BAUD9600, DATA_8, PAR_NONE, STOP_1, FLOW_OFF, 10}; - port = new QextSerialPort(ui->portBox->currentText(), settings, QextSerialPort::Polling); - //! [1] - - enumerator = new QextSerialEnumerator(this); - enumerator->setUpNotifications(); - - connect(ui->baudRateBox, SIGNAL(currentIndexChanged(int)), SLOT(onBaudRateChanged(int))); - connect(ui->parityBox, SIGNAL(currentIndexChanged(int)), SLOT(onParityChanged(int))); - connect(ui->dataBitsBox, SIGNAL(currentIndexChanged(int)), SLOT(onDataBitsChanged(int))); - connect(ui->stopBitsBox, SIGNAL(currentIndexChanged(int)), SLOT(onStopBitsChanged(int))); - connect(ui->queryModeBox, SIGNAL(currentIndexChanged(int)), SLOT(onQueryModeChanged(int))); - connect(ui->timeoutBox, SIGNAL(valueChanged(int)), SLOT(onTimeoutChanged(int))); - connect(ui->portBox, SIGNAL(editTextChanged(QString)), SLOT(onPortNameChanged(QString))); - connect(ui->openCloseButton, SIGNAL(clicked()), SLOT(onOpenCloseButtonClicked())); - connect(ui->sendButton, SIGNAL(clicked()), SLOT(onSendButtonClicked())); - connect(timer, SIGNAL(timeout()), SLOT(onReadyRead())); - connect(port, SIGNAL(readyRead()), SLOT(onReadyRead())); - - connect(enumerator, SIGNAL(deviceDiscovered(QextPortInfo)), SLOT(onPortAddedOrRemoved())); - connect(enumerator, SIGNAL(deviceRemoved(QextPortInfo)), SLOT(onPortAddedOrRemoved())); - - setWindowTitle(tr("QextSerialPort Demo")); -} - -Dialog::~Dialog() -{ - delete ui; - delete port; -} - -void Dialog::changeEvent(QEvent *e) -{ - QDialog::changeEvent(e); - switch (e->type()) { - case QEvent::LanguageChange: - ui->retranslateUi(this); - break; - default: - break; - } -} - -void Dialog::onPortNameChanged(const QString & /*name*/) -{ - if (port->isOpen()) { - port->close(); - ui->led->turnOff(); - } -} -//! [2] -void Dialog::onBaudRateChanged(int idx) -{ - port->setBaudRate((BaudRateType)ui->baudRateBox->itemData(idx).toInt()); -} - -void Dialog::onParityChanged(int idx) -{ - port->setParity((ParityType)ui->parityBox->itemData(idx).toInt()); -} - -void Dialog::onDataBitsChanged(int idx) -{ - port->setDataBits((DataBitsType)ui->dataBitsBox->itemData(idx).toInt()); -} - -void Dialog::onStopBitsChanged(int idx) -{ - port->setStopBits((StopBitsType)ui->stopBitsBox->itemData(idx).toInt()); -} - -void Dialog::onQueryModeChanged(int idx) -{ - port->setQueryMode((QextSerialPort::QueryMode)ui->queryModeBox->itemData(idx).toInt()); -} - -void Dialog::onTimeoutChanged(int val) -{ - port->setTimeout(val); -} -//! [2] -//! [3] -void Dialog::onOpenCloseButtonClicked() -{ - if (!port->isOpen()) { - port->setPortName(ui->portBox->currentText()); - port->open(QIODevice::ReadWrite); - } - else { - port->close(); - } - - //If using polling mode, we need a QTimer - if (port->isOpen() && port->queryMode() == QextSerialPort::Polling) - timer->start(); - else - timer->stop(); - - //update led's status - ui->led->turnOn(port->isOpen()); -} -//! [3] -//! [4] -void Dialog::onSendButtonClicked() -{ - if (port->isOpen() && !ui->sendEdit->toPlainText().isEmpty()) - port->write(ui->sendEdit->toPlainText().toLatin1()); -} - -void Dialog::onReadyRead() -{ - if (port->bytesAvailable()) { - ui->recvEdit->moveCursor(QTextCursor::End); - ui->recvEdit->insertPlainText(QString::fromLatin1(port->readAll())); - } -} - -void Dialog::onPortAddedOrRemoved() -{ - QString current = ui->portBox->currentText(); - - ui->portBox->blockSignals(true); - ui->portBox->clear(); - foreach (QextPortInfo info, QextSerialEnumerator::getPorts()) - ui->portBox->addItem(info.portName); - - ui->portBox->setCurrentIndex(ui->portBox->findText(current)); - - ui->portBox->blockSignals(false); -} - -//! [4] diff --git a/3rdparty/qextserialport/examples/uartassistant/dialog.h b/3rdparty/qextserialport/examples/uartassistant/dialog.h deleted file mode 100644 index 63c0b811..00000000 --- a/3rdparty/qextserialport/examples/uartassistant/dialog.h +++ /dev/null @@ -1,45 +0,0 @@ -#ifndef DIALOG_H -#define DIALOG_H - -#include - -namespace Ui { - class Dialog; -} -class QTimer; -class QextSerialPort; -class QextSerialEnumerator; - -class Dialog : public QDialog -{ - Q_OBJECT - -public: - explicit Dialog(QWidget *parent = 0); - ~Dialog(); - -protected: - void changeEvent(QEvent *e); - -private Q_SLOTS: - void onPortNameChanged(const QString &name); - void onBaudRateChanged(int idx); - void onParityChanged(int idx); - void onDataBitsChanged(int idx); - void onStopBitsChanged(int idx); - void onQueryModeChanged(int idx); - void onTimeoutChanged(int val); - void onOpenCloseButtonClicked(); - void onSendButtonClicked(); - void onReadyRead(); - - void onPortAddedOrRemoved(); - -private: - Ui::Dialog *ui; - QTimer *timer; - QextSerialPort *port; - QextSerialEnumerator *enumerator; -}; - -#endif // DIALOG_H diff --git a/3rdparty/qextserialport/examples/uartassistant/dialog.ui b/3rdparty/qextserialport/examples/uartassistant/dialog.ui deleted file mode 100644 index 0e68d34d..00000000 --- a/3rdparty/qextserialport/examples/uartassistant/dialog.ui +++ /dev/null @@ -1,191 +0,0 @@ - - - Dialog - - - - 0 - 0 - 604 - 485 - - - - Dialog - - - - - - - - 800 - - - - - - - - - - - - - - - - Port: - - - - - - - - - - BaudRate: - - - - - - - - - - DataBits: - - - - - - - - - - Parity: - - - - - - - - - - StopBits: - - - - - - - - - - QueryMode: - - - - - - - - - - Timeout: - - - - - - - ms - - - -1 - - - 10000 - - - 10 - - - 10 - - - - - - - - - - - - 0 - 0 - - - - - 20 - 20 - - - - - 25 - 25 - - - - - - - - Open/Close - - - - - - - - - Qt::Vertical - - - - 20 - 40 - - - - - - - - Send - - - - - - - - - - - HLed - QWidget -
hled.h
- 1 -
-
- - -
diff --git a/3rdparty/qextserialport/examples/uartassistant/hled.cpp b/3rdparty/qextserialport/examples/uartassistant/hled.cpp deleted file mode 100644 index d85f90c2..00000000 --- a/3rdparty/qextserialport/examples/uartassistant/hled.cpp +++ /dev/null @@ -1,133 +0,0 @@ -#include -#include "hled.h" - -struct HLed::Private -{ -public: - Private() - : darkerFactor(300), color(Qt::green), isOn(true) - { } - - int darkerFactor; - QColor color; - bool isOn; -}; - -HLed::HLed(QWidget *parent) - :QWidget(parent), m_d(new Private) -{ -} - -HLed::~HLed() -{ - delete m_d; -} - -QColor HLed::color() const -{ - return m_d->color; -} - -void HLed::setColor(const QColor &color) -{ - if (m_d->color == color) - return; - update(); -} - -QSize HLed::sizeHint() const -{ - return QSize(20, 20); -} - -QSize HLed::minimumSizeHint() const -{ - return QSize(16, 16); -} - -void HLed::toggle() -{ - m_d->isOn = !m_d->isOn; - update(); -} - -void HLed::turnOn(bool on) -{ - m_d->isOn = on; - update(); -} - -void HLed::turnOff(bool off) -{ - turnOn(!off); -} - -void HLed::paintEvent(QPaintEvent * /*event*/) -{ - int width = ledWidth(); - - QPainter painter(this); - painter.setRenderHint(QPainter::Antialiasing); - - QColor color = m_d->isOn ? m_d->color - : m_d->color.darker(m_d->darkerFactor); - - QBrush brush; - brush.setStyle(Qt::SolidPattern); - brush.setColor(color); - painter.setBrush(brush); - // draw plain - painter.drawEllipse(1, 1, width-1, width-1); - - QPen pen; - pen.setWidth(2); - - int pos = width / 5 + 1; - int lightWidth = width * 2 / 3; - int lightQuote = 130 * 2 / (lightWidth ? lightWidth : 1) + 100; - - // draw bright spot - while (lightWidth) { - color = color.lighter(lightQuote); - pen.setColor(color); - painter.setPen(pen); - painter.drawEllipse(pos, pos, lightWidth, lightWidth); - lightWidth--; - - if (!lightWidth) - break; - - painter.drawEllipse(pos, pos, lightWidth, lightWidth); - lightWidth--; - - if (!lightWidth) - break; - - painter.drawEllipse(pos, pos, lightWidth, lightWidth); - pos++; - lightWidth--; - } - - //draw border - painter.setBrush(Qt::NoBrush); - - int angle = -720; - color = palette().color(QPalette::Light); - - for (int arc=120; arc<2880; arc+=240) { - pen.setColor(color); - painter.setPen(pen); - int w = width - pen.width()/2; - painter.drawArc(pen.width()/2, pen.width()/2, w, w, angle+arc, 240); - painter.drawArc(pen.width()/2, pen.width()/2, w, w, angle-arc, 240); - color = color.darker(110); - } -} - -int HLed::ledWidth() const -{ - int width = qMin(this->width(), this->height()); - width -= 2; - return width > 0 ? width : 0; -} - diff --git a/3rdparty/qextserialport/examples/uartassistant/hled.h b/3rdparty/qextserialport/examples/uartassistant/hled.h deleted file mode 100644 index 77e1f52c..00000000 --- a/3rdparty/qextserialport/examples/uartassistant/hled.h +++ /dev/null @@ -1,34 +0,0 @@ -#ifndef HLED_H -#define HLED_H - -#include - -class QColor; - -class HLed : public QWidget -{ - Q_OBJECT -public: - HLed(QWidget *parent = 0); - ~HLed(); - - QColor color() const; - QSize sizeHint() const; - QSize minimumSizeHint() const; - -public slots: - void setColor(const QColor &color); - void toggle(); - void turnOn(bool on=true); - void turnOff(bool off=true); - -protected: - void paintEvent(QPaintEvent *); - int ledWidth() const; - -private: - struct Private; - Private * const m_d; -}; - -#endif // HLED_H diff --git a/3rdparty/qextserialport/examples/uartassistant/main.cpp b/3rdparty/qextserialport/examples/uartassistant/main.cpp deleted file mode 100644 index 6bf4fa00..00000000 --- a/3rdparty/qextserialport/examples/uartassistant/main.cpp +++ /dev/null @@ -1,11 +0,0 @@ -#include -#include "dialog.h" - -int main(int argc, char *argv[]) -{ - QApplication a(argc, argv); - Dialog w; - w.show(); - - return a.exec(); -} diff --git a/3rdparty/qextserialport/examples/uartassistant/uartassistant.pro b/3rdparty/qextserialport/examples/uartassistant/uartassistant.pro deleted file mode 100644 index ddca2f8f..00000000 --- a/3rdparty/qextserialport/examples/uartassistant/uartassistant.pro +++ /dev/null @@ -1,22 +0,0 @@ -#------------------------------------------------- -# -# Project created by QtCreator 2011-11-06T21:37:41 -# -#------------------------------------------------- - -QT += core gui -contains(QT_VERSION, ^5\\..*\\..*): QT += widgets - -TARGET = uartassistant -TEMPLATE = app - -include(../../src/qextserialport.pri) - -SOURCES += main.cpp\ - dialog.cpp\ - hled.cpp - -HEADERS += dialog.h \ - hled.h - -FORMS += dialog.ui diff --git a/3rdparty/qextserialport/extserialport.prf b/3rdparty/qextserialport/extserialport.prf deleted file mode 100644 index 610518d3..00000000 --- a/3rdparty/qextserialport/extserialport.prf +++ /dev/null @@ -1,17 +0,0 @@ -defineReplace(qextLibraryName) { - unset(LIBRARY_NAME) - LIBRARY_NAME = $$1 - greaterThan(QT_MAJOR_VERSION, 4):LIBRARY_NAME ~= s,^Qt,Qt$$QT_MAJOR_VERSION, - CONFIG(debug, debug|release) { - !debug_and_release|build_pass { - mac:LIBRARY_NAME = $${LIBRARY_NAME}_debug - else:win32:LIBRARY_NAME = $${LIBRARY_NAME}d - } - } - return($$LIBRARY_NAME) -} - -LIBS += -framework qextserialport -INCLUDEPATH += /Library/Frameworks/qextserialport.framework/Headers - -DEFINES += QEXTSERIALPORT_USING_SHARED diff --git a/3rdparty/qextserialport/extserialport.prf.in b/3rdparty/qextserialport/extserialport.prf.in deleted file mode 100644 index 684d17f0..00000000 --- a/3rdparty/qextserialport/extserialport.prf.in +++ /dev/null @@ -1,24 +0,0 @@ -defineReplace(qextLibraryName) { - unset(LIBRARY_NAME) - LIBRARY_NAME = \$\$1 - greaterThan(QT_MAJOR_VERSION, 4):LIBRARY_NAME ~= s,^Qt,Qt\$\$QT_MAJOR_VERSION, - CONFIG(debug, debug|release) { - !debug_and_release|build_pass { - mac:LIBRARY_NAME = \$\${LIBRARY_NAME}_debug - else:win32:LIBRARY_NAME = \$\${LIBRARY_NAME}d - } - } - return(\$\$LIBRARY_NAME) -} - -!!IF qesp_mac_framework -LIBS += -framework $$QESP_LIB_BASENAME -INCLUDEPATH += $$[QT_INSTALL_LIBS]/$${QESP_LIB_BASENAME}.framework/Headers -!!ELSE -LIBS += -l\$\$qextLibraryName($$QESP_LIB_BASENAME) -INCLUDEPATH += $$[QT_INSTALL_HEADERS]/QtExtSerialPort -!!ENDIF - -!!IF !qesp_static -DEFINES += QEXTSERIALPORT_USING_SHARED -!!ENDIF diff --git a/3rdparty/qextserialport/qextserialport.pro b/3rdparty/qextserialport/qextserialport.pro deleted file mode 100644 index 5fcb46fd..00000000 --- a/3rdparty/qextserialport/qextserialport.pro +++ /dev/null @@ -1,96 +0,0 @@ -############################### *User Config* ############################### - -# Uncomment following line if you want to build a static library -# CONFIG += qesp_static - -# Uncomment following line if you want to build framework for mac -macx:CONFIG += qesp_mac_framework - -# Uncomment following line if you want to enable udev for linux -# linux*:CONFIG += qesp_linux_udev - -# Note: you can create a ".qmake.cache" file, then copy these lines to it. -# If so, you can avoid to change this project file. -############################### *User Config* ############################### - -defineReplace(qextLibraryName) { - unset(LIBRARY_NAME) - LIBRARY_NAME = $$1 - macx:qesp_mac_framework { - QMAKE_FRAMEWORK_BUNDLE_NAME = $$LIBRARY_NAME - export(QMAKE_FRAMEWORK_BUNDLE_NAME) - } else { - greaterThan(QT_MAJOR_VERSION, 4):LIBRARY_NAME ~= s,^Qt,Qt$$QT_MAJOR_VERSION, - } - CONFIG(debug, debug|release) { - !debug_and_release|build_pass { - mac:LIBRARY_NAME = $${LIBRARY_NAME}_debug - else:win32:LIBRARY_NAME = $${LIBRARY_NAME}d - } - } - return($$LIBRARY_NAME) -} - -TEMPLATE=lib -include(src/qextserialport.pri) - -#create_prl is needed, otherwise, MinGW can't found libqextserialport1.a -CONFIG += create_prl - -#mac framework is designed for shared library -macx:qesp_mac_framework:qesp_static: CONFIG -= qesp_static -!macx:qesp_mac_framework:CONFIG -= qesp_mac_framework - -qesp_static { - CONFIG += static -} else { - CONFIG += shared - macx:!qesp_mac_framework:CONFIG += absolute_library_soname - DEFINES += QEXTSERIALPORT_BUILD_SHARED -} - -#Creare lib bundle for mac -macx:qesp_mac_framework { - CONFIG += lib_bundle - FRAMEWORK_HEADERS.files = $$PUBLIC_HEADERS - FRAMEWORK_HEADERS.path = Headers - QMAKE_BUNDLE_DATA += FRAMEWORK_HEADERS -} - -win32|mac:!wince*:!win32-msvc:!macx-xcode:CONFIG += debug_and_release build_all - -#For non-windows system, only depends on QtCore module -unix:QT = core -else:QT = core gui - -#generate proper library name -greaterThan(QT_MAJOR_VERSION, 4) { - QESP_LIB_BASENAME = QtExtSerialPort -} else { - QESP_LIB_BASENAME = qextserialport -} -TARGET = $$qextLibraryName($$QESP_LIB_BASENAME) -VERSION = 1.2.0 - -# generate feature file by qmake based on this *.in file. -QMAKE_SUBSTITUTES += extserialport.prf.in -OTHER_FILES += extserialport.prf.in - -# for make docs -include(doc/doc.pri) - -# for make install -win32:!qesp_static { - dlltarget.path = $$[QT_INSTALL_BINS] - INSTALLS += dlltarget -} -!macx|!qesp_mac_framework { - headers.files = $$PUBLIC_HEADERS - headers.path = $$[QT_INSTALL_HEADERS]/QtExtSerialPort - INSTALLS += headers -} -target.path = $$[QT_INSTALL_LIBS] - -features.files = extserialport.prf -features.path = $$[QMAKE_MKSPECS]/features -INSTALLS += target features diff --git a/3rdparty/qextserialport/src/qextserialenumerator.cpp b/3rdparty/qextserialport/src/qextserialenumerator.cpp deleted file mode 100644 index 62c531ea..00000000 --- a/3rdparty/qextserialport/src/qextserialenumerator.cpp +++ /dev/null @@ -1,163 +0,0 @@ -/**************************************************************************** -** Copyright (c) 2000-2003 Wayne Roth -** Copyright (c) 2004-2007 Stefan Sander -** Copyright (c) 2007 Michal Policht -** Copyright (c) 2008 Brandon Fosdick -** Copyright (c) 2009-2010 Liam Staskawicz -** Copyright (c) 2011 Debao Zhang -** All right reserved. -** Web: http://code.google.com/p/qextserialport/ -** -** Permission is hereby granted, free of charge, to any person obtaining -** a copy of this software and associated documentation files (the -** "Software"), to deal in the Software without restriction, including -** without limitation the rights to use, copy, modify, merge, publish, -** distribute, sublicense, and/or sell copies of the Software, and to -** permit persons to whom the Software is furnished to do so, subject to -** the following conditions: -** -** The above copyright notice and this permission notice shall be -** included in all copies or substantial portions of the Software. -** -** THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, -** EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND -** NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE -** LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION -** OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION -** WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. -** -****************************************************************************/ - -#include "qextserialenumerator.h" -#include "qextserialenumerator_p.h" - -#include -#include -#include - -QextSerialEnumeratorPrivate::QextSerialEnumeratorPrivate(QextSerialEnumerator *enumrator) - :q_ptr(enumrator) -{ - platformSpecificInit(); -} - -QextSerialEnumeratorPrivate::~QextSerialEnumeratorPrivate() -{ - platformSpecificDestruct(); -} - -/*! - \class QextPortInfo - - \brief The QextPortInfo class containing port information. - - Structure containing port information. - - \code - QString portName; ///< Port name. - QString physName; ///< Physical name. - QString friendName; ///< Friendly name. - QString enumName; ///< Enumerator name. - int vendorID; ///< Vendor ID. - int productID; ///< Product ID - \endcode - */ - -/*! \class QextSerialEnumerator - - \brief The QextSerialEnumerator class provides list of ports available in the system. - - \section1 Usage - To poll the system for a list of connected devices, simply use getPorts(). Each - QextPortInfo structure will populated with information about the corresponding device. - - \bold Example - \code - QList ports = QextSerialEnumerator::getPorts(); - foreach (QextPortInfo port, ports) { - // inspect port... - } - \endcode - - To enable event-driven notification of device connection events, first call - setUpNotifications() and then connect to the deviceDiscovered() and deviceRemoved() - signals. Event-driven behavior is currently available only on Windows and OS X. - - \bold Example - \code - QextSerialEnumerator *enumerator = new QextSerialEnumerator(); - connect(enumerator, SIGNAL(deviceDiscovered(const QextPortInfo &)), - myClass, SLOT(onDeviceDiscovered(const QextPortInfo &))); - connect(enumerator, SIGNAL(deviceRemoved(const QextPortInfo &)), - myClass, SLOT(onDeviceRemoved(const QextPortInfo &))); - \endcode - - \section1 Credits - Windows implementation is based on Zach Gorman's work from - \l {http://www.codeproject.com}{The Code Project} (\l http://www.codeproject.com/system/setupdi.asp). - - OS X implementation, see \l http://developer.apple.com/documentation/DeviceDrivers/Conceptual/AccessingHardware/AH_Finding_Devices/chapter_4_section_2.html - - \bold author Michal Policht, Liam Staskawicz -*/ - -/*! - \fn void QextSerialEnumerator::deviceDiscovered(const QextPortInfo &info) - A new device has been connected to the system. - - setUpNotifications() must be called first to enable event-driven device notifications. - Currently only implemented on Windows and OS X. - - \a info The device that has been discovered. -*/ - -/*! - \fn void QextSerialEnumerator::deviceRemoved(const QextPortInfo &info); - A device has been disconnected from the system. - - setUpNotifications() must be called first to enable event-driven device notifications. - Currently only implemented on Windows and OS X. - - \a info The device that was disconnected. -*/ - -/*! - Constructs a QextSerialEnumerator object with the given \a parent. -*/ -QextSerialEnumerator::QextSerialEnumerator(QObject *parent) - :QObject(parent), d_ptr(new QextSerialEnumeratorPrivate(this)) -{ - if (!QMetaType::isRegistered(QMetaType::type("QextPortInfo"))) - qRegisterMetaType("QextPortInfo"); -} - -/*! - Destructs the QextSerialEnumerator object. -*/ -QextSerialEnumerator::~QextSerialEnumerator() -{ - delete d_ptr; -} - -/*! - Get list of ports. - - return list of ports currently available in the system. -*/ -QList QextSerialEnumerator::getPorts() -{ - return QextSerialEnumeratorPrivate::getPorts_sys(); -} - -/*! - Enable event-driven notifications of board discovery/removal. -*/ -void QextSerialEnumerator::setUpNotifications() -{ - Q_D(QextSerialEnumerator); - if (!d->setUpNotifications_sys(true)) - QESP_WARNING("Setup Notification Failed..."); -} - -#include "moc_qextserialenumerator.cpp" diff --git a/3rdparty/qextserialport/src/qextserialenumerator.h b/3rdparty/qextserialport/src/qextserialenumerator.h deleted file mode 100644 index f207d0d3..00000000 --- a/3rdparty/qextserialport/src/qextserialenumerator.h +++ /dev/null @@ -1,72 +0,0 @@ -/**************************************************************************** -** Copyright (c) 2000-2003 Wayne Roth -** Copyright (c) 2004-2007 Stefan Sander -** Copyright (c) 2007 Michal Policht -** Copyright (c) 2008 Brandon Fosdick -** Copyright (c) 2009-2010 Liam Staskawicz -** Copyright (c) 2011 Debao Zhang -** All right reserved. -** Web: http://code.google.com/p/qextserialport/ -** -** Permission is hereby granted, free of charge, to any person obtaining -** a copy of this software and associated documentation files (the -** "Software"), to deal in the Software without restriction, including -** without limitation the rights to use, copy, modify, merge, publish, -** distribute, sublicense, and/or sell copies of the Software, and to -** permit persons to whom the Software is furnished to do so, subject to -** the following conditions: -** -** The above copyright notice and this permission notice shall be -** included in all copies or substantial portions of the Software. -** -** THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, -** EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND -** NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE -** LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION -** OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION -** WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. -** -****************************************************************************/ - -#ifndef _QEXTSERIALENUMERATOR_H_ -#define _QEXTSERIALENUMERATOR_H_ - -#include -#include -#include "qextserialport_global.h" - -struct QextPortInfo { - QString portName; ///< Port name. - QString physName; ///< Physical name. - QString friendName; ///< Friendly name. - QString enumName; ///< Enumerator name. - int vendorID; ///< Vendor ID. - int productID; ///< Product ID -}; - -class QextSerialEnumeratorPrivate; -class QEXTSERIALPORT_EXPORT QextSerialEnumerator : public QObject -{ - Q_OBJECT - Q_DECLARE_PRIVATE(QextSerialEnumerator) -public: - QextSerialEnumerator(QObject *parent=0); - ~QextSerialEnumerator(); - - static QList getPorts(); - void setUpNotifications(); - -Q_SIGNALS: - void deviceDiscovered(const QextPortInfo &info); - void deviceRemoved(const QextPortInfo &info); - -private: - Q_DISABLE_COPY(QextSerialEnumerator) -#if defined(Q_OS_LINUX) && !defined(QESP_NO_UDEV) - Q_PRIVATE_SLOT(d_func(), void _q_deviceEvent()) -#endif - QextSerialEnumeratorPrivate *d_ptr; -}; - -#endif /*_QEXTSERIALENUMERATOR_H_*/ diff --git a/3rdparty/qextserialport/src/qextserialenumerator_linux.cpp b/3rdparty/qextserialport/src/qextserialenumerator_linux.cpp deleted file mode 100644 index a37174c8..00000000 --- a/3rdparty/qextserialport/src/qextserialenumerator_linux.cpp +++ /dev/null @@ -1,210 +0,0 @@ -/**************************************************************************** -** Copyright (c) 2000-2003 Wayne Roth -** Copyright (c) 2004-2007 Stefan Sander -** Copyright (c) 2007 Michal Policht -** Copyright (c) 2008 Brandon Fosdick -** Copyright (c) 2009-2010 Liam Staskawicz -** Copyright (c) 2011 Debao Zhang -** Copyright (c) 2012 Doug Brown -** All right reserved. -** Web: http://code.google.com/p/qextserialport/ -** -** Permission is hereby granted, free of charge, to any person obtaining -** a copy of this software and associated documentation files (the -** "Software"), to deal in the Software without restriction, including -** without limitation the rights to use, copy, modify, merge, publish, -** distribute, sublicense, and/or sell copies of the Software, and to -** permit persons to whom the Software is furnished to do so, subject to -** the following conditions: -** -** The above copyright notice and this permission notice shall be -** included in all copies or substantial portions of the Software. -** -** THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, -** EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND -** NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE -** LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION -** OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION -** WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. -** -****************************************************************************/ - -#include "qextserialenumerator.h" -#include "qextserialenumerator_p.h" -#include -#include -#include - -void QextSerialEnumeratorPrivate::platformSpecificInit() -{ -#ifndef QESP_NO_UDEV - monitor = NULL; - notifierFd = -1; - notifier = NULL; - - udev = udev_new(); - if (!udev) - qCritical() << "Unable to initialize udev notifications"; -#endif -} - -void QextSerialEnumeratorPrivate::platformSpecificDestruct() -{ -#ifndef QESP_NO_UDEV - if (notifier) { - notifier->setEnabled(false); - delete notifier; - } - - if (monitor) - udev_monitor_unref(monitor); - - if (udev) - udev_unref(udev); -#endif -} - -#ifndef QESP_NO_UDEV -static QextPortInfo portInfoFromDevice(struct udev_device *dev) -{ - QString vendor = QString::fromLatin1(udev_device_get_property_value(dev, "ID_VENDOR_ID")); - QString product = QString::fromLatin1(udev_device_get_property_value(dev, "ID_MODEL_ID")); - - QextPortInfo pi; - pi.vendorID = vendor.toInt(0, 16); - pi.productID = product.toInt(0, 16); - pi.portName = QString::fromLatin1(udev_device_get_devnode(dev)); - pi.physName = pi.portName; - - return pi; -} -#endif - -QList QextSerialEnumeratorPrivate::getPorts_sys() -{ - QList infoList; -#ifndef QESP_NO_UDEV - struct udev *ud = udev_new(); - if (!ud) { - qCritical() << "Unable to enumerate ports because udev is not initialized."; - return infoList; - } - - struct udev_enumerate *enumerate = udev_enumerate_new(ud); - udev_enumerate_add_match_subsystem(enumerate, "tty"); - udev_enumerate_scan_devices(enumerate); - struct udev_list_entry *list = udev_enumerate_get_list_entry(enumerate); - struct udev_list_entry *entry; - udev_list_entry_foreach(entry, list) { - const char *path; - struct udev_device *dev; - - // Have to grab the actual udev device here... - path = udev_list_entry_get_name(entry); - dev = udev_device_new_from_syspath(ud, path); - - infoList.append(portInfoFromDevice(dev)); - - // Done with this device - udev_device_unref(dev); - } - // Done with the list and this udev - udev_enumerate_unref(enumerate); - udev_unref(ud); -#else - QStringList portNamePrefixes, portNameList; - portNamePrefixes << QLatin1String("ttyS*"); // list normal serial ports first - - QDir dir(QLatin1String("/dev")); - portNameList = dir.entryList(portNamePrefixes, (QDir::System | QDir::Files), QDir::Name); - - // remove the values which are not serial ports for e.g. /dev/ttysa - for (int i = 0; i < portNameList.size(); i++) { - bool ok; - QString current = portNameList.at(i); - // remove the ttyS part, and check, if the other part is a number - current.remove(0,4).toInt(&ok, 10); - if (!ok) { - portNameList.removeAt(i); - i--; - } - } - - // get the non standard serial ports names - // (USB-serial, bluetooth-serial, 18F PICs, and so on) - // if you know an other name prefix for serial ports please let us know - portNamePrefixes.clear(); - portNamePrefixes << QLatin1String("ttyACM*") << QLatin1String("ttyUSB*") << QLatin1String("rfcomm*"); - portNameList += dir.entryList(portNamePrefixes, (QDir::System | QDir::Files), QDir::Name); - - foreach (QString str , portNameList) { - QextPortInfo inf; - inf.physName = QLatin1String("/dev/")+str; - inf.portName = str; - - if (str.contains(QLatin1String("ttyS"))) { - inf.friendName = QLatin1String("Serial port ")+str.remove(0, 4); - } - else if (str.contains(QLatin1String("ttyUSB"))) { - inf.friendName = QLatin1String("USB-serial adapter ")+str.remove(0, 6); - } - else if (str.contains(QLatin1String("rfcomm"))) { - inf.friendName = QLatin1String("Bluetooth-serial adapter ")+str.remove(0, 6); - } - inf.enumName = QLatin1String("/dev"); // is there a more helpful name for this? - infoList.append(inf); - } -#endif - - return infoList; -} - -bool QextSerialEnumeratorPrivate::setUpNotifications_sys(bool setup) -{ - Q_UNUSED(setup); -#ifndef QESP_NO_UDEV - Q_Q(QextSerialEnumerator); - if (!udev) { - qCritical() << "Unable to initialize notifications because udev is not initialized."; - return false; - } - - // Emit signals immediately for devices already connected (Windows version seems to behave - // this way) - foreach (QextPortInfo i, getPorts_sys()) - Q_EMIT q->deviceDiscovered(i); - - // Look for tty devices from udev. - monitor = udev_monitor_new_from_netlink(udev, "udev"); - udev_monitor_filter_add_match_subsystem_devtype(monitor, "tty", NULL); - udev_monitor_enable_receiving(monitor); - notifierFd = udev_monitor_get_fd(monitor); - notifier = new QSocketNotifier(notifierFd, QSocketNotifier::Read); - q->connect(notifier, SIGNAL(activated(int)), q, SLOT(_q_deviceEvent())); - notifier->setEnabled(true); - - return true; -#else - return false; -#endif -} - -#ifndef QESP_NO_UDEV -void QextSerialEnumeratorPrivate::_q_deviceEvent() -{ - Q_Q(QextSerialEnumerator); - struct udev_device *dev = udev_monitor_receive_device(monitor); - if (dev) { - QextPortInfo pi = portInfoFromDevice(dev); - QLatin1String action(udev_device_get_action(dev)); - - if (action == QLatin1String("add")) - Q_EMIT q->deviceDiscovered(pi); - else if (action == QLatin1String("remove")) - Q_EMIT q->deviceRemoved(pi); - - udev_device_unref(dev); - } -} -#endif diff --git a/3rdparty/qextserialport/src/qextserialenumerator_osx.cpp b/3rdparty/qextserialport/src/qextserialenumerator_osx.cpp deleted file mode 100644 index 89a6d884..00000000 --- a/3rdparty/qextserialport/src/qextserialenumerator_osx.cpp +++ /dev/null @@ -1,307 +0,0 @@ -/**************************************************************************** -** Copyright (c) 2000-2003 Wayne Roth -** Copyright (c) 2004-2007 Stefan Sander -** Copyright (c) 2007 Michal Policht -** Copyright (c) 2008 Brandon Fosdick -** Copyright (c) 2009-2010 Liam Staskawicz -** Copyright (c) 2011 Debao Zhang -** All right reserved. -** Web: http://code.google.com/p/qextserialport/ -** -** Permission is hereby granted, free of charge, to any person obtaining -** a copy of this software and associated documentation files (the -** "Software"), to deal in the Software without restriction, including -** without limitation the rights to use, copy, modify, merge, publish, -** distribute, sublicense, and/or sell copies of the Software, and to -** permit persons to whom the Software is furnished to do so, subject to -** the following conditions: -** -** The above copyright notice and this permission notice shall be -** included in all copies or substantial portions of the Software. -** -** THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, -** EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND -** NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE -** LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION -** OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION -** WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. -** -****************************************************************************/ - -#include "qextserialenumerator.h" -#include "qextserialenumerator_p.h" -#include -#include -#include -#include - -void QextSerialEnumeratorPrivate::platformSpecificInit() -{ -} - -void QextSerialEnumeratorPrivate::platformSpecificDestruct() -{ - IONotificationPortDestroy(notificationPortRef); -} - -// static -QList QextSerialEnumeratorPrivate::getPorts_sys() -{ - QList infoList; - io_iterator_t serialPortIterator = 0; - kern_return_t kernResult = KERN_FAILURE; - CFMutableDictionaryRef matchingDictionary; - - // first try to get any serialbsd devices, then try any USBCDC devices - if (!(matchingDictionary = IOServiceMatching(kIOSerialBSDServiceValue))) { - QESP_WARNING("IOServiceMatching returned a NULL dictionary."); - return infoList; - } - CFDictionaryAddValue(matchingDictionary, CFSTR(kIOSerialBSDTypeKey), CFSTR(kIOSerialBSDAllTypes)); - - // then create the iterator with all the matching devices - if (IOServiceGetMatchingServices(kIOMasterPortDefault, matchingDictionary, &serialPortIterator) != KERN_SUCCESS) { - qCritical() << "IOServiceGetMatchingServices failed, returned" << kernResult; - return infoList; - } - iterateServicesOSX(serialPortIterator, infoList); - IOObjectRelease(serialPortIterator); - serialPortIterator = 0; - - if (!(matchingDictionary = IOServiceNameMatching("AppleUSBCDC"))) { - QESP_WARNING("IOServiceNameMatching returned a NULL dictionary."); - return infoList; - } - - if (IOServiceGetMatchingServices(kIOMasterPortDefault, matchingDictionary, &serialPortIterator) != KERN_SUCCESS) { - qCritical() << "IOServiceGetMatchingServices failed, returned" << kernResult; - return infoList; - } - iterateServicesOSX(serialPortIterator, infoList); - IOObjectRelease(serialPortIterator); - - return infoList; -} - -void QextSerialEnumeratorPrivate::iterateServicesOSX(io_object_t service, QList &infoList) -{ - // Iterate through all modems found. - io_object_t usbService; - while ((usbService = IOIteratorNext(service))) { - QextPortInfo info; - info.vendorID = 0; - info.productID = 0; - getServiceDetailsOSX(usbService, &info); - infoList.append(info); - } -} - -bool QextSerialEnumeratorPrivate::getServiceDetailsOSX(io_object_t service, QextPortInfo *portInfo) -{ - bool retval = true; - CFTypeRef bsdPathAsCFString = NULL; - CFTypeRef productNameAsCFString = NULL; - CFTypeRef vendorIdAsCFNumber = NULL; - CFTypeRef productIdAsCFNumber = NULL; - // check the name of the modem's callout device - bsdPathAsCFString = IORegistryEntryCreateCFProperty(service, CFSTR(kIOCalloutDeviceKey), - kCFAllocatorDefault, 0); - - // wander up the hierarchy until we find the level that can give us the - // vendor/product IDs and the product name, if available - io_registry_entry_t parent; - kern_return_t kernResult = IORegistryEntryGetParentEntry(service, kIOServicePlane, &parent); - while (kernResult == KERN_SUCCESS && !vendorIdAsCFNumber && !productIdAsCFNumber) { - if (!productNameAsCFString) - productNameAsCFString = IORegistryEntrySearchCFProperty(parent, - kIOServicePlane, - CFSTR("Product Name"), - kCFAllocatorDefault, 0); - vendorIdAsCFNumber = IORegistryEntrySearchCFProperty(parent, - kIOServicePlane, - CFSTR(kUSBVendorID), - kCFAllocatorDefault, 0); - productIdAsCFNumber = IORegistryEntrySearchCFProperty(parent, - kIOServicePlane, - CFSTR(kUSBProductID), - kCFAllocatorDefault, 0); - io_registry_entry_t oldparent = parent; - kernResult = IORegistryEntryGetParentEntry(parent, kIOServicePlane, &parent); - IOObjectRelease(oldparent); - } - - io_string_t ioPathName; - IORegistryEntryGetPath(service, kIOServicePlane, ioPathName); - portInfo->physName = ioPathName; - - if (bsdPathAsCFString) { - char path[MAXPATHLEN]; - if (CFStringGetCString((CFStringRef)bsdPathAsCFString, path, - PATH_MAX, kCFStringEncodingUTF8)) - portInfo->portName = path; - CFRelease(bsdPathAsCFString); - } - - if (productNameAsCFString) { - char productName[MAXPATHLEN]; - if (CFStringGetCString((CFStringRef)productNameAsCFString, productName, - PATH_MAX, kCFStringEncodingUTF8)) - portInfo->friendName = productName; - CFRelease(productNameAsCFString); - } - - if (vendorIdAsCFNumber) { - SInt32 vID; - if (CFNumberGetValue((CFNumberRef)vendorIdAsCFNumber, kCFNumberSInt32Type, &vID)) - portInfo->vendorID = vID; - CFRelease(vendorIdAsCFNumber); - } - - if (productIdAsCFNumber) { - SInt32 pID; - if (CFNumberGetValue((CFNumberRef)productIdAsCFNumber, kCFNumberSInt32Type, &pID)) - portInfo->productID = pID; - CFRelease(productIdAsCFNumber); - } - IOObjectRelease(service); - return retval; -} - -// IOKit callbacks registered via setupNotifications() -void deviceDiscoveredCallbackOSX(void *ctxt, io_iterator_t serialPortIterator) -{ - QextSerialEnumeratorPrivate *d = (QextSerialEnumeratorPrivate *)ctxt; - io_object_t serialService; - while ((serialService = IOIteratorNext(serialPortIterator))) - d->onDeviceDiscoveredOSX(serialService); -} - -void deviceTerminatedCallbackOSX(void *ctxt, io_iterator_t serialPortIterator) -{ - QextSerialEnumeratorPrivate *d = (QextSerialEnumeratorPrivate *)ctxt; - io_object_t serialService; - while ((serialService = IOIteratorNext(serialPortIterator))) - d->onDeviceTerminatedOSX(serialService); -} - -/* - A device has been discovered via IOKit. - Create a QextPortInfo if possible, and emit the signal indicating that we've found it. -*/ -void QextSerialEnumeratorPrivate::onDeviceDiscoveredOSX(io_object_t service) -{ - Q_Q(QextSerialEnumerator); - QextPortInfo info; - info.vendorID = 0; - info.productID = 0; - if (getServiceDetailsOSX(service, &info)) - Q_EMIT q->deviceDiscovered(info); -} - -/* - Notification via IOKit that a device has been removed. - Create a QextPortInfo if possible, and emit the signal indicating that it's gone. -*/ -void QextSerialEnumeratorPrivate::onDeviceTerminatedOSX(io_object_t service) -{ - Q_Q(QextSerialEnumerator); - QextPortInfo info; - info.vendorID = 0; - info.productID = 0; - if (getServiceDetailsOSX(service, &info)) - Q_EMIT q->deviceRemoved(info); -} - -/* - Create matching dictionaries for the devices we want to get notifications for, - and add them to the current run loop. Invoke the callbacks that will be responding - to these notifications once to arm them, and discover any devices that - are currently connected at the time notifications are setup. -*/ -bool QextSerialEnumeratorPrivate::setUpNotifications_sys(bool /*setup*/) -{ - kern_return_t kernResult; - mach_port_t masterPort; - CFRunLoopSourceRef notificationRunLoopSource; - CFMutableDictionaryRef classesToMatch; - CFMutableDictionaryRef cdcClassesToMatch; - io_iterator_t portIterator; - - kernResult = IOMasterPort(MACH_PORT_NULL, &masterPort); - if (KERN_SUCCESS != kernResult) { - qDebug() << "IOMasterPort returned:" << kernResult; - return false; - } - - classesToMatch = IOServiceMatching(kIOSerialBSDServiceValue); - if (classesToMatch == NULL) - qDebug("IOServiceMatching returned a NULL dictionary."); - else - CFDictionarySetValue(classesToMatch, CFSTR(kIOSerialBSDTypeKey), CFSTR(kIOSerialBSDAllTypes)); - - if (!(cdcClassesToMatch = IOServiceNameMatching("AppleUSBCDC"))) { - QESP_WARNING("couldn't create cdc matching dict"); - return false; - } - - // Retain an additional reference since each call to IOServiceAddMatchingNotification consumes one. - classesToMatch = (CFMutableDictionaryRef) CFRetain(classesToMatch); - cdcClassesToMatch = (CFMutableDictionaryRef) CFRetain(cdcClassesToMatch); - - notificationPortRef = IONotificationPortCreate(masterPort); - if (notificationPortRef == NULL) { - qDebug("IONotificationPortCreate return a NULL IONotificationPortRef."); - return false; - } - - notificationRunLoopSource = IONotificationPortGetRunLoopSource(notificationPortRef); - if (notificationRunLoopSource == NULL) { - qDebug("IONotificationPortGetRunLoopSource returned NULL CFRunLoopSourceRef."); - return false; - } - - CFRunLoopAddSource(CFRunLoopGetCurrent(), notificationRunLoopSource, kCFRunLoopDefaultMode); - - kernResult = IOServiceAddMatchingNotification(notificationPortRef, kIOMatchedNotification, classesToMatch, - deviceDiscoveredCallbackOSX, this, &portIterator); - if (kernResult != KERN_SUCCESS) { - qDebug() << "IOServiceAddMatchingNotification return:" << kernResult; - return false; - } - - // arm the callback, and grab any devices that are already connected - deviceDiscoveredCallbackOSX(this, portIterator); - - kernResult = IOServiceAddMatchingNotification(notificationPortRef, kIOMatchedNotification, cdcClassesToMatch, - deviceDiscoveredCallbackOSX, this, &portIterator); - if (kernResult != KERN_SUCCESS) { - qDebug() << "IOServiceAddMatchingNotification return:" << kernResult; - return false; - } - - // arm the callback, and grab any devices that are already connected - deviceDiscoveredCallbackOSX(this, portIterator); - - kernResult = IOServiceAddMatchingNotification(notificationPortRef, kIOTerminatedNotification, classesToMatch, - deviceTerminatedCallbackOSX, this, &portIterator); - if (kernResult != KERN_SUCCESS) { - qDebug() << "IOServiceAddMatchingNotification return:" << kernResult; - return false; - } - - // arm the callback, and clear any devices that are terminated - deviceTerminatedCallbackOSX(this, portIterator); - - kernResult = IOServiceAddMatchingNotification(notificationPortRef, kIOTerminatedNotification, cdcClassesToMatch, - deviceTerminatedCallbackOSX, this, &portIterator); - if (kernResult != KERN_SUCCESS) { - qDebug() << "IOServiceAddMatchingNotification return:" << kernResult; - return false; - } - - // arm the callback, and clear any devices that are terminated - deviceTerminatedCallbackOSX(this, portIterator); - return true; -} - diff --git a/3rdparty/qextserialport/src/qextserialenumerator_p.h b/3rdparty/qextserialport/src/qextserialenumerator_p.h deleted file mode 100644 index 3412a8b7..00000000 --- a/3rdparty/qextserialport/src/qextserialenumerator_p.h +++ /dev/null @@ -1,125 +0,0 @@ -/**************************************************************************** -** Copyright (c) 2000-2003 Wayne Roth -** Copyright (c) 2004-2007 Stefan Sander -** Copyright (c) 2007 Michal Policht -** Copyright (c) 2008 Brandon Fosdick -** Copyright (c) 2009-2010 Liam Staskawicz -** Copyright (c) 2011 Debao Zhang -** Copyright (c) 2012 Doug Brown -** All right reserved. -** Web: http://code.google.com/p/qextserialport/ -** -** Permission is hereby granted, free of charge, to any person obtaining -** a copy of this software and associated documentation files (the -** "Software"), to deal in the Software without restriction, including -** without limitation the rights to use, copy, modify, merge, publish, -** distribute, sublicense, and/or sell copies of the Software, and to -** permit persons to whom the Software is furnished to do so, subject to -** the following conditions: -** -** The above copyright notice and this permission notice shall be -** included in all copies or substantial portions of the Software. -** -** THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, -** EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND -** NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE -** LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION -** OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION -** WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. -** -****************************************************************************/ -#ifndef _QEXTSERIALENUMERATOR_P_H_ -#define _QEXTSERIALENUMERATOR_P_H_ - -// -// W A R N I N G -// ------------- -// -// This file is not part of the QESP API. It exists for the convenience -// of other QESP classes. This header file may change from version to -// version without notice, or even be removed. -// -// We mean it. -// - -#include "qextserialenumerator.h" - -#ifdef Q_CC_MINGW -// needed for mingw to pull in appropriate dbt business... -// probably a better way to do this -// http://mingw-users.1079350.n2.nabble.com/DEV-BROADCAST-DEVICEINTERFACE-was-not-declared-in-this-scope-td3552762.html -// http://msdn.microsoft.com/en-us/library/6sehtctf.aspx -# ifndef WINVER -# define WINVER 0x0501 -# endif -# ifndef _WIN32_WINNT -# define _WIN32_WINNT WINVER -# endif -#endif - -#ifdef Q_OS_WIN -# include -#endif /*Q_OS_WIN*/ - -#ifdef Q_OS_MAC -# include -#endif /*Q_OS_MAC*/ - -#if defined(Q_OS_LINUX) && !defined(QESP_NO_UDEV) -# include -extern "C" { -# include -} -#endif - -class QextSerialRegistrationWidget; -class QextSerialEnumeratorPrivate -{ - Q_DECLARE_PUBLIC(QextSerialEnumerator) -public: - QextSerialEnumeratorPrivate(QextSerialEnumerator *enumrator); - ~QextSerialEnumeratorPrivate(); - void platformSpecificInit(); - void platformSpecificDestruct(); - - static QList getPorts_sys(); - bool setUpNotifications_sys(bool setup); - -#ifdef Q_OS_WIN - LRESULT onDeviceChanged(WPARAM wParam, LPARAM lParam); - bool matchAndDispatchChangedDevice(const QString &deviceID, const GUID &guid, WPARAM wParam); -# ifdef QT_GUI_LIB - QextSerialRegistrationWidget *notificationWidget; -# endif -#endif /*Q_OS_WIN*/ - -#ifdef Q_OS_MAC - /*! - * Search for serial ports using IOKit. - * \param infoList list with result. - */ - static void iterateServicesOSX(io_object_t service, QList &infoList); - static bool getServiceDetailsOSX(io_object_t service, QextPortInfo *portInfo); - void onDeviceDiscoveredOSX(io_object_t service); - void onDeviceTerminatedOSX(io_object_t service); - friend void deviceDiscoveredCallbackOSX(void *ctxt, io_iterator_t serialPortIterator); - friend void deviceTerminatedCallbackOSX(void *ctxt, io_iterator_t serialPortIterator); - - IONotificationPortRef notificationPortRef; -#endif // Q_OS_MAC - -#if defined(Q_OS_LINUX) && !defined(QESP_NO_UDEV) - QSocketNotifier *notifier; - int notifierFd; - struct udev *udev; - struct udev_monitor *monitor; - - void _q_deviceEvent(); -#endif - -private: - QextSerialEnumerator *q_ptr; -}; - -#endif //_QEXTSERIALENUMERATOR_P_H_ diff --git a/3rdparty/qextserialport/src/qextserialenumerator_unix.cpp b/3rdparty/qextserialport/src/qextserialenumerator_unix.cpp deleted file mode 100644 index 1e6615f8..00000000 --- a/3rdparty/qextserialport/src/qextserialenumerator_unix.cpp +++ /dev/null @@ -1,56 +0,0 @@ -/**************************************************************************** -** Copyright (c) 2000-2003 Wayne Roth -** Copyright (c) 2004-2007 Stefan Sander -** Copyright (c) 2007 Michal Policht -** Copyright (c) 2008 Brandon Fosdick -** Copyright (c) 2009-2010 Liam Staskawicz -** Copyright (c) 2011 Debao Zhang -** All right reserved. -** Web: http://code.google.com/p/qextserialport/ -** -** Permission is hereby granted, free of charge, to any person obtaining -** a copy of this software and associated documentation files (the -** "Software"), to deal in the Software without restriction, including -** without limitation the rights to use, copy, modify, merge, publish, -** distribute, sublicense, and/or sell copies of the Software, and to -** permit persons to whom the Software is furnished to do so, subject to -** the following conditions: -** -** The above copyright notice and this permission notice shall be -** included in all copies or substantial portions of the Software. -** -** THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, -** EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND -** NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE -** LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION -** OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION -** WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. -** -****************************************************************************/ - -#include "qextserialenumerator.h" -#include "qextserialenumerator_p.h" -#include - -void QextSerialEnumeratorPrivate::platformSpecificInit() -{ -} - -void QextSerialEnumeratorPrivate::platformSpecificDestruct() -{ -} - -QList QextSerialEnumeratorPrivate::getPorts_sys() -{ - QList infoList; - QESP_WARNING("Enumeration for POSIX systems (except Linux) is not implemented yet."); - return infoList; -} - -bool QextSerialEnumeratorPrivate::setUpNotifications_sys(bool setup) -{ - Q_UNUSED(setup) - QESP_WARNING("Notifications for *Nix/FreeBSD are not implemented yet"); - return false; -} diff --git a/3rdparty/qextserialport/src/qextserialenumerator_win.cpp b/3rdparty/qextserialport/src/qextserialenumerator_win.cpp deleted file mode 100644 index 852209f8..00000000 --- a/3rdparty/qextserialport/src/qextserialenumerator_win.cpp +++ /dev/null @@ -1,321 +0,0 @@ -/**************************************************************************** -** Copyright (c) 2000-2003 Wayne Roth -** Copyright (c) 2004-2007 Stefan Sander -** Copyright (c) 2007 Michal Policht -** Copyright (c) 2008 Brandon Fosdick -** Copyright (c) 2009-2010 Liam Staskawicz -** Copyright (c) 2011 Debao Zhang -** All right reserved. -** Web: http://code.google.com/p/qextserialport/ -** -** Permission is hereby granted, free of charge, to any person obtaining -** a copy of this software and associated documentation files (the -** "Software"), to deal in the Software without restriction, including -** without limitation the rights to use, copy, modify, merge, publish, -** distribute, sublicense, and/or sell copies of the Software, and to -** permit persons to whom the Software is furnished to do so, subject to -** the following conditions: -** -** The above copyright notice and this permission notice shall be -** included in all copies or substantial portions of the Software. -** -** THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, -** EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND -** NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE -** LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION -** OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION -** WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. -** -****************************************************************************/ - -#include "qextserialenumerator.h" -#include "qextserialenumerator_p.h" -#include -#include -#include -#include -#include -#include -#include -#include "qextserialport.h" - -#ifdef QT_GUI_LIB -#if QT_VERSION < QT_VERSION_CHECK(5, 0, 0) -#include -class QextSerialRegistrationWidget : public QWidget -#else -#include -class QextSerialRegistrationWidget : public QWindow -#endif -{ -public: - QextSerialRegistrationWidget(QextSerialEnumeratorPrivate *qese) { - this->qese = qese; - } - ~QextSerialRegistrationWidget() {} - -protected: - -#if QT_VERSION < QT_VERSION_CHECK(5, 0, 0) - bool winEvent(MSG *message, long *result) { -#else - bool nativeEvent(const QByteArray & /*eventType*/, void *msg, long *result) { - MSG *message = static_cast(msg); -#endif - if (message->message == WM_DEVICECHANGE) { - qese->onDeviceChanged(message->wParam, message->lParam); - *result = 1; - return true; - } - return false; - } -private: - QextSerialEnumeratorPrivate *qese; -}; - -#endif // QT_GUI_LIB - -void QextSerialEnumeratorPrivate::platformSpecificInit() -{ -#ifdef QT_GUI_LIB - notificationWidget = 0; -#endif // QT_GUI_LIB -} - -/*! - default -*/ -void QextSerialEnumeratorPrivate::platformSpecificDestruct() -{ -#ifdef QT_GUI_LIB - if (notificationWidget) - delete notificationWidget; -#endif -} - -// see http://msdn.microsoft.com/en-us/library/windows/hardware/ff553426(v=vs.85).aspx -// for list of GUID classes -const GUID deviceClassGuids[] = -{ - // Ports (COM & LPT ports), Class = Ports - {0x4D36E978, 0xE325, 0x11CE, {0xBF, 0xC1, 0x08, 0x00, 0x2B, 0xE1, 0x03, 0x18}}, - // Modem, Class = Modem - {0x4D36E96D, 0xE325, 0x11CE, {0xBF, 0xC1, 0x08, 0x00, 0x2B, 0xE1, 0x03, 0x18}}, - // Bluetooth Devices, Class = Bluetooth - {0xE0CBF06C, 0xCD8B, 0x4647, {0xBB, 0x8A, 0x26, 0x3B, 0x43, 0xF0, 0xF9, 0x74}}, - // Added by Arne Kristian Jansen, for use with com0com virtual ports (See Issue 54) - {0xDF799E12, 0x3C56, 0x421B, {0xB2, 0x98, 0xB6, 0xD3, 0x64, 0x2B, 0xC8, 0x78}} -}; - -/* Gordon Schumacher's macros for TCHAR -> QString conversions and vice versa */ -#ifdef UNICODE - #define QStringToTCHAR(x) (wchar_t *) x.utf16() - #define PQStringToTCHAR(x) (wchar_t *) x->utf16() - #define TCHARToQString(x) QString::fromUtf16((ushort *)(x)) - #define TCHARToQStringN(x,y) QString::fromUtf16((ushort *)(x),(y)) -#else - #define QStringToTCHAR(x) x.local8Bit().constData() - #define PQStringToTCHAR(x) x->local8Bit().constData() - #define TCHARToQString(x) QString::fromLocal8Bit((char *)(x)) - #define TCHARToQStringN(x,y) QString::fromLocal8Bit((char *)(x),(y)) -#endif /*UNICODE*/ - -/*! - \internal - Get value of specified property from the registry. - \a key handle to an open key. - \a property property name. - - return property value. -*/ -static QString getRegKeyValue(HKEY key, LPCTSTR property) -{ - DWORD size = 0; - DWORD type; - if (::RegQueryValueEx(key, property, NULL, NULL, NULL, &size) != ERROR_SUCCESS) - return QString(); - BYTE *buff = new BYTE[size]; - QString result; - if (::RegQueryValueEx(key, property, NULL, &type, buff, &size) == ERROR_SUCCESS) - result = TCHARToQString(buff); - ::RegCloseKey(key); - delete [] buff; - return result; -} - -/*! - \internal - Get specific property from registry. - \a devInfo pointer to the device information set that contains the interface - and its underlying device. Returned by SetupDiGetClassDevs() function. - \a devData pointer to an SP_DEVINFO_DATA structure that defines the device instance. - this is returned by SetupDiGetDeviceInterfaceDetail() function. - \a property registry property. One of defined SPDRP_* constants. - - return property string. - */ -static QString getDeviceProperty(HDEVINFO devInfo, PSP_DEVINFO_DATA devData, DWORD property) -{ - DWORD buffSize = 0; - ::SetupDiGetDeviceRegistryProperty(devInfo, devData, property, NULL, NULL, 0, &buffSize); - if (GetLastError() != ERROR_INSUFFICIENT_BUFFER) - return QString(); - BYTE *buff = new BYTE[buffSize]; - ::SetupDiGetDeviceRegistryProperty(devInfo, devData, property, NULL, buff, buffSize, NULL); - QString result = TCHARToQString(buff); - delete [] buff; - return result; -} - -/*! - \internal -*/ -static bool getDeviceDetailsWin(QextPortInfo *portInfo, HDEVINFO devInfo, PSP_DEVINFO_DATA devData - , WPARAM wParam = DBT_DEVICEARRIVAL) -{ - portInfo->friendName = getDeviceProperty(devInfo, devData, SPDRP_FRIENDLYNAME); - if (wParam == DBT_DEVICEARRIVAL) - portInfo->physName = getDeviceProperty(devInfo, devData, SPDRP_PHYSICAL_DEVICE_OBJECT_NAME); - portInfo->enumName = getDeviceProperty(devInfo, devData, SPDRP_ENUMERATOR_NAME); - QString hardwareIDs = getDeviceProperty(devInfo, devData, SPDRP_HARDWAREID); - HKEY devKey = ::SetupDiOpenDevRegKey(devInfo, devData, DICS_FLAG_GLOBAL, 0, DIREG_DEV, KEY_QUERY_VALUE); - portInfo->portName = getRegKeyValue(devKey, TEXT("PortName")); - QRegExp idRx(QLatin1String("VID_(\\w+)&PID_(\\w+)")); - if (hardwareIDs.toUpper().contains(idRx)) { - bool dummy; - portInfo->vendorID = idRx.cap(1).toInt(&dummy, 16); - portInfo->productID = idRx.cap(2).toInt(&dummy, 16); - //qDebug() << "got vid:" << vid << "pid:" << pid; - } - return true; -} - -/*! - \internal -*/ -static void enumerateDevicesWin(const GUID &guid, QList *infoList) -{ - HDEVINFO devInfo; - if ((devInfo = ::SetupDiGetClassDevs(&guid, NULL, NULL, DIGCF_PRESENT)) != INVALID_HANDLE_VALUE) { - SP_DEVINFO_DATA devInfoData; - devInfoData.cbSize = sizeof(SP_DEVINFO_DATA); - for(int i = 0; ::SetupDiEnumDeviceInfo(devInfo, i, &devInfoData); i++) { - QextPortInfo info; - info.productID = info.vendorID = 0; - getDeviceDetailsWin(&info, devInfo, &devInfoData); - if (!info.portName.startsWith(QLatin1String("LPT"), Qt::CaseInsensitive)) - infoList->append(info); - } - ::SetupDiDestroyDeviceInfoList(devInfo); - } -} - - -static bool lessThan(const QextPortInfo &s1, const QextPortInfo &s2) -{ - if (s1.portName.startsWith(QLatin1String("COM")) - && s2.portName.startsWith(QLatin1String("COM"))) { - return s1.portName.mid(3).toInt() QextSerialEnumeratorPrivate::getPorts_sys() -{ - QList ports; - const int count = sizeof(deviceClassGuids)/sizeof(deviceClassGuids[0]); - for (int i=0; iwinId(), &dbh, flags) == NULL) { - QESP_WARNING() << "RegisterDeviceNotification failed:" << GetLastError(); - return false; - } - // setting up notifications doesn't tell us about devices already connected - // so get those manually - foreach (QextPortInfo port, getPorts_sys()) - Q_EMIT q->deviceDiscovered(port); - return true; -#endif // QT_GUI_LIB -} - -LRESULT QextSerialEnumeratorPrivate::onDeviceChanged(WPARAM wParam, LPARAM lParam) -{ - if (DBT_DEVICEARRIVAL == wParam || DBT_DEVICEREMOVECOMPLETE == wParam) { - PDEV_BROADCAST_HDR pHdr = (PDEV_BROADCAST_HDR)lParam; - if (pHdr->dbch_devicetype == DBT_DEVTYP_DEVICEINTERFACE) { - PDEV_BROADCAST_DEVICEINTERFACE pDevInf = (PDEV_BROADCAST_DEVICEINTERFACE)pHdr; - // delimiters are different across APIs...change to backslash. ugh. - QString deviceID = TCHARToQString(pDevInf->dbcc_name).toUpper().replace(QLatin1String("#"), QLatin1String("\\")); - - const int count = sizeof(deviceClassGuids)/sizeof(deviceClassGuids[0]); - for (int i=0; ideviceDiscovered(info); - else if (wParam == DBT_DEVICEREMOVECOMPLETE) - Q_EMIT q->deviceRemoved(info); - break; - } - } - SetupDiDestroyDeviceInfoList(devInfo); - } - return rv; -} diff --git a/3rdparty/qextserialport/src/qextserialport.cpp b/3rdparty/qextserialport/src/qextserialport.cpp deleted file mode 100644 index d1089e23..00000000 --- a/3rdparty/qextserialport/src/qextserialport.cpp +++ /dev/null @@ -1,1005 +0,0 @@ -/**************************************************************************** -** Copyright (c) 2000-2003 Wayne Roth -** Copyright (c) 2004-2007 Stefan Sander -** Copyright (c) 2007 Michal Policht -** Copyright (c) 2008 Brandon Fosdick -** Copyright (c) 2009-2010 Liam Staskawicz -** Copyright (c) 2011 Debao Zhang -** All right reserved. -** Web: http://code.google.com/p/qextserialport/ -** -** Permission is hereby granted, free of charge, to any person obtaining -** a copy of this software and associated documentation files (the -** "Software"), to deal in the Software without restriction, including -** without limitation the rights to use, copy, modify, merge, publish, -** distribute, sublicense, and/or sell copies of the Software, and to -** permit persons to whom the Software is furnished to do so, subject to -** the following conditions: -** -** The above copyright notice and this permission notice shall be -** included in all copies or substantial portions of the Software. -** -** THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, -** EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND -** NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE -** LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION -** OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION -** WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. -** -****************************************************************************/ - -#include "qextserialport.h" -#include "qextserialport_p.h" -#include -#include -#include -#include - -/*! - \class PortSettings - - \brief The PortSettings class contain port settings - - Structure to contain port settings. - - \code - BaudRateType BaudRate; - DataBitsType DataBits; - ParityType Parity; - StopBitsType StopBits; - FlowType FlowControl; - long Timeout_Millisec; - \endcode -*/ - -QextSerialPortPrivate::QextSerialPortPrivate(QextSerialPort *q) - :lock(QReadWriteLock::Recursive), q_ptr(q) -{ - lastErr = E_NO_ERROR; - settings.BaudRate = BAUD9600; - settings.Parity = PAR_NONE; - settings.FlowControl = FLOW_OFF; - settings.DataBits = DATA_8; - settings.StopBits = STOP_1; - settings.Timeout_Millisec = 10; - settingsDirtyFlags = DFE_ALL; - - platformSpecificInit(); -} - -QextSerialPortPrivate::~QextSerialPortPrivate() -{ - platformSpecificDestruct(); -} - -void QextSerialPortPrivate::setBaudRate(BaudRateType baudRate, bool update) -{ - switch (baudRate) { -#ifdef Q_OS_WIN - //Windows Special - case BAUD14400: - case BAUD56000: - case BAUD128000: - case BAUD256000: - QESP_PORTABILITY_WARNING()<<"QextSerialPort Portability Warning: POSIX does not support baudRate:"<isOpen()) - updatePortSettings(); - break; -#if !(defined(Q_OS_WIN) || defined(Q_OS_MAC)) - default: - QESP_WARNING()<<"QextSerialPort does not support baudRate:"<isOpen()) - updatePortSettings(); -} - -void QextSerialPortPrivate::setDataBits(DataBitsType dataBits, bool update) -{ - switch(dataBits) { - - case DATA_5: - if (settings.StopBits == STOP_2) { - QESP_WARNING("QextSerialPort: 5 Data bits cannot be used with 2 stop bits."); - } else { - settings.DataBits = dataBits; - settingsDirtyFlags |= DFE_DataBits; - } - break; - - case DATA_6: -#ifdef Q_OS_WIN - if (settings.StopBits == STOP_1_5) { - QESP_WARNING("QextSerialPort: 6 Data bits cannot be used with 1.5 stop bits."); - } - else -#endif - { - settings.DataBits = dataBits; - settingsDirtyFlags |= DFE_DataBits; - } - break; - - case DATA_7: -#ifdef Q_OS_WIN - if (settings.StopBits == STOP_1_5) { - QESP_WARNING("QextSerialPort: 7 Data bits cannot be used with 1.5 stop bits."); - } - else -#endif - { - settings.DataBits = dataBits; - settingsDirtyFlags |= DFE_DataBits; - } - break; - - case DATA_8: -#ifdef Q_OS_WIN - if (settings.StopBits == STOP_1_5) { - QESP_WARNING("QextSerialPort: 8 Data bits cannot be used with 1.5 stop bits."); - } - else -#endif - { - settings.DataBits = dataBits; - settingsDirtyFlags |= DFE_DataBits; - } - break; - default: - QESP_WARNING()<<"QextSerialPort does not support Data bits:"<isOpen()) - updatePortSettings(); -} - -void QextSerialPortPrivate::setStopBits(StopBitsType stopBits, bool update) -{ - switch (stopBits) { - - /*one stop bit*/ - case STOP_1: - settings.StopBits = stopBits; - settingsDirtyFlags |= DFE_StopBits; - break; - -#ifdef Q_OS_WIN - /*1.5 stop bits*/ - case STOP_1_5: - QESP_PORTABILITY_WARNING("QextSerialPort Portability Warning: 1.5 stop bit operation is not supported by POSIX."); - if (settings.DataBits != DATA_5) { - QESP_WARNING("QextSerialPort: 1.5 stop bits can only be used with 5 data bits"); - } else { - settings.StopBits = stopBits; - settingsDirtyFlags |= DFE_StopBits; - } - break; -#endif - - /*two stop bits*/ - case STOP_2: - if (settings.DataBits == DATA_5) { - QESP_WARNING("QextSerialPort: 2 stop bits cannot be used with 5 data bits"); - } else { - settings.StopBits = stopBits; - settingsDirtyFlags |= DFE_StopBits; - } - break; - default: - QESP_WARNING()<<"QextSerialPort does not support stop bits: "<isOpen()) - updatePortSettings(); -} - -void QextSerialPortPrivate::setFlowControl(FlowType flow, bool update) -{ - settings.FlowControl = flow; - settingsDirtyFlags |= DFE_Flow; - if (update && q_func()->isOpen()) - updatePortSettings(); -} - -void QextSerialPortPrivate::setTimeout(long millisec, bool update) -{ - settings.Timeout_Millisec = millisec; - settingsDirtyFlags |= DFE_TimeOut; - if (update && q_func()->isOpen()) - updatePortSettings(); -} - -void QextSerialPortPrivate::setPortSettings(const PortSettings &settings, bool update) -{ - setBaudRate(settings.BaudRate, false); - setDataBits(settings.DataBits, false); - setStopBits(settings.StopBits, false); - setParity(settings.Parity, false); - setFlowControl(settings.FlowControl, false); - setTimeout(settings.Timeout_Millisec, false); - settingsDirtyFlags = DFE_ALL; - if (update && q_func()->isOpen()) - updatePortSettings(); -} - - -void QextSerialPortPrivate::_q_canRead() -{ - qint64 maxSize = bytesAvailable_sys(); - if (maxSize > 0) { - char *writePtr = readBuffer.reserve(size_t(maxSize)); - qint64 bytesRead = readData_sys(writePtr, maxSize); - if (bytesRead < maxSize) - readBuffer.chop(maxSize - bytesRead); - Q_Q(QextSerialPort); - Q_EMIT q->readyRead(); - } -} - -/*! \class QextSerialPort - - \brief The QextSerialPort class encapsulates a serial port on both POSIX and Windows systems. - - \section1 Usage - QextSerialPort offers both a polling and event driven API. Event driven - is typically easier to use, since you never have to worry about checking - for new data. - - \bold Example - \code - QextSerialPort *port = new QextSerialPort("COM1"); - connect(port, SIGNAL(readyRead()), myClass, SLOT(onDataAvailable())); - port->open(); - - void MyClass::onDataAvailable() - { - QByteArray data = port->readAll(); - processNewData(usbdata); - } - \endcode - - \section1 Compatibility - The user will be notified of errors and possible portability conflicts at run-time - by default. - - For example, if a application has used BAUD1800, when it is runing under unix, you - will get following message. - - \code - QextSerialPort Portability Warning: Windows does not support baudRate:1800 - \endcode - - This behavior can be turned off by defining macro QESP_NO_WARN (to turn off all warnings) - or QESP_NO_PORTABILITY_WARN (to turn off portability warnings) in the project. - - - \bold Author: Stefan Sander, Michal Policht, Brandon Fosdick, Liam Staskawicz, Debao Zhang -*/ - -/*! - \enum QextSerialPort::QueryMode - - This enum type specifies query mode used in a serial port: - - \value Polling - asynchronously read and write - \value EventDriven - synchronously read and write -*/ - -/*! - \fn void QextSerialPort::dsrChanged(bool status) - This signal is emitted whenever dsr line has changed its state. You may - use this signal to check if device is connected. - - \a status true when DSR signal is on, false otherwise. - */ - - -/*! - \fn QueryMode QextSerialPort::queryMode() const - Get query mode. - */ - -/*! - Default constructor. Note that the name of the device used by a QextSerialPort is dependent on - your OS. Possible naming conventions and their associated OS are: - - \code - - OS Constant Used By Naming Convention - ------------- ------------- ------------------------ - Q_OS_WIN Windows COM1, COM2 - Q_OS_IRIX SGI/IRIX /dev/ttyf1, /dev/ttyf2 - Q_OS_HPUX HP-UX /dev/tty1p0, /dev/tty2p0 - Q_OS_SOLARIS SunOS/Slaris /dev/ttya, /dev/ttyb - Q_OS_OSF Digital UNIX /dev/tty01, /dev/tty02 - Q_OS_FREEBSD FreeBSD /dev/ttyd0, /dev/ttyd1 - Q_OS_OPENBSD OpenBSD /dev/tty00, /dev/tty01 - Q_OS_LINUX Linux /dev/ttyS0, /dev/ttyS1 - /dev/ttyS0, /dev/ttyS1 - \endcode - - This constructor assigns the device name to the name of the first port on the specified system. - See the other constructors if you need to open a different port. Default \a mode is EventDriven. - As a subclass of QObject, \a parent can be specified. -*/ - -QextSerialPort::QextSerialPort(QextSerialPort::QueryMode mode, QObject *parent) - : QIODevice(parent), d_ptr(new QextSerialPortPrivate(this)) -{ -#ifdef Q_OS_WIN - setPortName(QLatin1String("COM1")); - -#elif defined(Q_OS_IRIX) - setPortName(QLatin1String("/dev/ttyf1")); - -#elif defined(Q_OS_HPUX) - setPortName(QLatin1String("/dev/tty1p0")); - -#elif defined(Q_OS_SOLARIS) - setPortName(QLatin1String("/dev/ttya")); - -#elif defined(Q_OS_OSF) //formally DIGITAL UNIX - setPortName(QLatin1String("/dev/tty01")); - -#elif defined(Q_OS_FREEBSD) - setPortName(QLatin1String("/dev/ttyd1")); - -#elif defined(Q_OS_OPENBSD) - setPortName(QLatin1String("/dev/tty00")); - -#else - setPortName(QLatin1String("/dev/ttyS0")); -#endif - setQueryMode(mode); -} - -/*! - Constructs a serial port attached to the port specified by name. - \a name is the name of the device, which is windowsystem-specific, - e.g."COM1" or "/dev/ttyS0". \a mode -*/ -QextSerialPort::QextSerialPort(const QString &name, QextSerialPort::QueryMode mode, QObject *parent) - : QIODevice(parent), d_ptr(new QextSerialPortPrivate(this)) -{ - setQueryMode(mode); - setPortName(name); -} - -/*! - Constructs a port with default name and specified \a settings. -*/ -QextSerialPort::QextSerialPort(const PortSettings &settings, QextSerialPort::QueryMode mode, QObject *parent) - : QIODevice(parent), d_ptr(new QextSerialPortPrivate(this)) -{ - Q_D(QextSerialPort); - setQueryMode(mode); - d->setPortSettings(settings); -} - -/*! - Constructs a port with specified \a name , \a mode and \a settings. -*/ -QextSerialPort::QextSerialPort(const QString &name, const PortSettings &settings, QextSerialPort::QueryMode mode, QObject *parent) - : QIODevice(parent), d_ptr(new QextSerialPortPrivate(this)) -{ - Q_D(QextSerialPort); - setPortName(name); - setQueryMode(mode); - d->setPortSettings(settings); -} - -/*! - Opens a serial port and sets its OpenMode to \a mode. - Note that this function does not specify which device to open. - Returns true if successful; otherwise returns false.This function has no effect - if the port associated with the class is already open. The port is also - configured to the current settings, as stored in the settings structure. -*/ -bool QextSerialPort::open(OpenMode mode) -{ - Q_D(QextSerialPort); - QWriteLocker locker(&d->lock); - if (mode != QIODevice::NotOpen && !isOpen()) - d->open_sys(mode); - - return isOpen(); -} - - -/*! \reimp - Closes a serial port. This function has no effect if the serial port associated with the class - is not currently open. -*/ -void QextSerialPort::close() -{ - Q_D(QextSerialPort); - QWriteLocker locker(&d->lock); - if (isOpen()) { - // Be a good QIODevice and call QIODevice::close() before really close() - // so the aboutToClose() signal is emitted at the proper time - QIODevice::close(); // mark ourselves as closed - d->close_sys(); - d->readBuffer.clear(); - } -} - -/*! - Flushes all pending I/O to the serial port. This function has no effect if the serial port - associated with the class is not currently open. -*/ -void QextSerialPort::flush() -{ - Q_D(QextSerialPort); - QWriteLocker locker(&d->lock); - if (isOpen()) - d->flush_sys(); -} - -/*! \reimp - Returns the number of bytes waiting in the port's receive queue. This function will return 0 if - the port is not currently open, or -1 on error. -*/ -qint64 QextSerialPort::bytesAvailable() const -{ - QWriteLocker locker(&d_func()->lock); - if (isOpen()) { - qint64 bytes = d_func()->bytesAvailable_sys(); - if (bytes != -1) { - return bytes + d_func()->readBuffer.size() - + QIODevice::bytesAvailable(); - } - return -1; - } - return 0; -} - -/*! \reimp - -*/ -bool QextSerialPort::canReadLine() const -{ - QReadLocker locker(&d_func()->lock); - return QIODevice::canReadLine() || d_func()->readBuffer.canReadLine(); -} - -/*! - * Set desired serial communication handling style. You may choose from polling - * or event driven approach. This function does nothing when port is open; to - * apply changes port must be reopened. - * - * In event driven approach read() and write() functions are acting - * asynchronously. They return immediately and the operation is performed in - * the background, so they doesn't freeze the calling thread. - * To determine when operation is finished, QextSerialPort runs separate thread - * and monitors serial port events. Whenever the event occurs, adequate signal - * is emitted. - * - * When polling is set, read() and write() are acting synchronously. Signals are - * not working in this mode and some functions may not be available. The advantage - * of polling is that it generates less overhead due to lack of signals emissions - * and it doesn't start separate thread to monitor events. - * - * Generally event driven approach is more capable and friendly, although some - * applications may need as low overhead as possible and then polling comes. - * - * \a mode query mode. - */ -void QextSerialPort::setQueryMode(QueryMode mode) -{ - Q_D(QextSerialPort); - QWriteLocker locker(&d->lock); - if (mode != d->queryMode) - d->queryMode = mode; -} - -/*! - Sets the \a name of the device associated with the object, e.g. "COM1", or "/dev/ttyS0". -*/ -void QextSerialPort::setPortName(const QString &name) -{ - Q_D(QextSerialPort); - QWriteLocker locker(&d->lock); - d->port = name; -} - -/*! - Returns the name set by setPortName(). -*/ -QString QextSerialPort::portName() const -{ - QReadLocker locker(&d_func()->lock); - return d_func()->port; -} - -QextSerialPort::QueryMode QextSerialPort::queryMode() const -{ - QReadLocker locker(&d_func()->lock); - return d_func()->queryMode; -} - -/*! - Reads all available data from the device, and returns it as a QByteArray. - This function has no way of reporting errors; returning an empty QByteArray() - can mean either that no data was currently available for reading, or that an error occurred. -*/ -QByteArray QextSerialPort::readAll() -{ - int avail = this->bytesAvailable(); - return (avail > 0) ? this->read(avail) : QByteArray(); -} - -/*! - Returns the baud rate of the serial port. For a list of possible return values see - the definition of the enum BaudRateType. -*/ -BaudRateType QextSerialPort::baudRate() const -{ - QReadLocker locker(&d_func()->lock); - return d_func()->settings.BaudRate; -} - -/*! - Returns the number of data bits used by the port. For a list of possible values returned by - this function, see the definition of the enum DataBitsType. -*/ -DataBitsType QextSerialPort::dataBits() const -{ - QReadLocker locker(&d_func()->lock); - return d_func()->settings.DataBits; -} - -/*! - Returns the type of parity used by the port. For a list of possible values returned by - this function, see the definition of the enum ParityType. -*/ -ParityType QextSerialPort::parity() const -{ - QReadLocker locker(&d_func()->lock); - return d_func()->settings.Parity; -} - -/*! - Returns the number of stop bits used by the port. For a list of possible return values, see - the definition of the enum StopBitsType. -*/ -StopBitsType QextSerialPort::stopBits() const -{ - QReadLocker locker(&d_func()->lock); - return d_func()->settings.StopBits; -} - -/*! - Returns the type of flow control used by the port. For a list of possible values returned - by this function, see the definition of the enum FlowType. -*/ -FlowType QextSerialPort::flowControl() const -{ - QReadLocker locker(&d_func()->lock); - return d_func()->settings.FlowControl; -} - -/*! - \reimp - Returns true if device is sequential, otherwise returns false. Serial port is sequential device - so this function always returns true. Check QIODevice::isSequential() documentation for more - information. -*/ -bool QextSerialPort::isSequential() const -{ - return true; -} - -/*! - Return the error number, or 0 if no error occurred. -*/ -ulong QextSerialPort::lastError() const -{ - QReadLocker locker(&d_func()->lock); - return d_func()->lastErr; -} - -/*! - Returns the line status as stored by the port function. This function will retrieve the states - of the following lines: DCD, CTS, DSR, and RI. On POSIX systems, the following additional lines - can be monitored: DTR, RTS, Secondary TXD, and Secondary RXD. The value returned is an unsigned - long with specific bits indicating which lines are high. The following constants should be used - to examine the states of individual lines: - - \code - Mask Line - ------ ---- - LS_CTS CTS - LS_DSR DSR - LS_DCD DCD - LS_RI RI - LS_RTS RTS (POSIX only) - LS_DTR DTR (POSIX only) - LS_ST Secondary TXD (POSIX only) - LS_SR Secondary RXD (POSIX only) - \endcode - - This function will return 0 if the port associated with the class is not currently open. -*/ -unsigned long QextSerialPort::lineStatus() -{ - Q_D(QextSerialPort); - QWriteLocker locker(&d->lock); - if (isOpen()) - return d->lineStatus_sys(); - return 0; -} - -/*! - Returns a human-readable description of the last device error that occurred. -*/ -QString QextSerialPort::errorString() -{ - Q_D(QextSerialPort); - QReadLocker locker(&d->lock); - switch(d->lastErr) { - case E_NO_ERROR: - return tr("No Error has occurred"); - case E_INVALID_FD: - return tr("Invalid file descriptor (port was not opened correctly)"); - case E_NO_MEMORY: - return tr("Unable to allocate memory tables (POSIX)"); - case E_CAUGHT_NON_BLOCKED_SIGNAL: - return tr("Caught a non-blocked signal (POSIX)"); - case E_PORT_TIMEOUT: - return tr("Operation timed out (POSIX)"); - case E_INVALID_DEVICE: - return tr("The file opened by the port is not a valid device"); - case E_BREAK_CONDITION: - return tr("The port detected a break condition"); - case E_FRAMING_ERROR: - return tr("The port detected a framing error (usually caused by incorrect baud rate settings)"); - case E_IO_ERROR: - return tr("There was an I/O error while communicating with the port"); - case E_BUFFER_OVERRUN: - return tr("Character buffer overrun"); - case E_RECEIVE_OVERFLOW: - return tr("Receive buffer overflow"); - case E_RECEIVE_PARITY_ERROR: - return tr("The port detected a parity error in the received data"); - case E_TRANSMIT_OVERFLOW: - return tr("Transmit buffer overflow"); - case E_READ_FAILED: - return tr("General read operation failure"); - case E_WRITE_FAILED: - return tr("General write operation failure"); - case E_FILE_NOT_FOUND: - return tr("The %1 file doesn't exists").arg(this->portName()); - case E_PERMISSION_DENIED: - return tr("Permission denied"); - case E_AGAIN: - return tr("Device is already locked"); - default: - return tr("Unknown error: %1").arg(d->lastErr); - } -} - -/*! - Destructs the QextSerialPort object. -*/ -QextSerialPort::~QextSerialPort() -{ - if (isOpen()) - close(); - - delete d_ptr; -} - -/*! - Sets the flow control used by the port to \a flow. Possible values of flow are: - \code - FLOW_OFF No flow control - FLOW_HARDWARE Hardware (RTS/CTS) flow control - FLOW_XONXOFF Software (XON/XOFF) flow control - \endcode -*/ -void QextSerialPort::setFlowControl(FlowType flow) -{ - Q_D(QextSerialPort); - QWriteLocker locker(&d->lock); - if (d->settings.FlowControl != flow) - d->setFlowControl(flow, true); -} - -/*! - Sets the parity associated with the serial port to \a parity. The possible values of parity are: - \code - PAR_SPACE Space Parity - PAR_MARK Mark Parity - PAR_NONE No Parity - PAR_EVEN Even Parity - PAR_ODD Odd Parity - \endcode -*/ -void QextSerialPort::setParity(ParityType parity) -{ - Q_D(QextSerialPort); - QWriteLocker locker(&d->lock); - if (d->settings.Parity != parity) - d->setParity(parity, true); -} - -/*! - Sets the number of data bits used by the serial port to \a dataBits. Possible values of dataBits are: - \code - DATA_5 5 data bits - DATA_6 6 data bits - DATA_7 7 data bits - DATA_8 8 data bits - \endcode - - \bold note: - This function is subject to the following restrictions: - \list - \o 5 data bits cannot be used with 2 stop bits. - \o 1.5 stop bits can only be used with 5 data bits. - \o 8 data bits cannot be used with space parity on POSIX systems. - \endlist - */ -void QextSerialPort::setDataBits(DataBitsType dataBits) -{ - Q_D(QextSerialPort); - QWriteLocker locker(&d->lock); - if (d->settings.DataBits != dataBits) - d->setDataBits(dataBits, true); -} - -/*! - Sets the number of stop bits used by the serial port to \a stopBits. Possible values of stopBits are: - \code - STOP_1 1 stop bit - STOP_1_5 1.5 stop bits - STOP_2 2 stop bits - \endcode - - \bold note: - This function is subject to the following restrictions: - \list - \o 2 stop bits cannot be used with 5 data bits. - \o 1.5 stop bits cannot be used with 6 or more data bits. - \o POSIX does not support 1.5 stop bits. - \endlist -*/ -void QextSerialPort::setStopBits(StopBitsType stopBits) -{ - Q_D(QextSerialPort); - QWriteLocker locker(&d->lock); - if (d->settings.StopBits != stopBits) - d->setStopBits(stopBits, true); -} - -/*! - Sets the baud rate of the serial port to \a baudRate. Note that not all rates are applicable on - all platforms. The following table shows translations of the various baud rate - constants on Windows(including NT/2000) and POSIX platforms. Speeds marked with an * - are speeds that are usable on both Windows and POSIX. - \code - - RATE Windows Speed POSIX Speed - ----------- ------------- ----------- - BAUD50 X 50 - BAUD75 X 75 - *BAUD110 110 110 - BAUD134 X 134.5 - BAUD150 X 150 - BAUD200 X 200 - *BAUD300 300 300 - *BAUD600 600 600 - *BAUD1200 1200 1200 - BAUD1800 X 1800 - *BAUD2400 2400 2400 - *BAUD4800 4800 4800 - *BAUD9600 9600 9600 - BAUD14400 14400 X - *BAUD19200 19200 19200 - *BAUD38400 38400 38400 - BAUD56000 56000 X - *BAUD57600 57600 57600 - BAUD76800 X 76800 - *BAUD115200 115200 115200 - BAUD128000 128000 X - BAUD230400 X 230400 - BAUD256000 256000 X - BAUD460800 X 460800 - BAUD500000 X 500000 - BAUD576000 X 576000 - BAUD921600 X 921600 - BAUD1000000 X 1000000 - BAUD1152000 X 1152000 - BAUD1500000 X 1500000 - BAUD2000000 X 2000000 - BAUD2500000 X 2500000 - BAUD3000000 X 3000000 - BAUD3500000 X 3500000 - BAUD4000000 X 4000000 - \endcode -*/ - -void QextSerialPort::setBaudRate(BaudRateType baudRate) -{ - Q_D(QextSerialPort); - QWriteLocker locker(&d->lock); - if (d->settings.BaudRate != baudRate) - d->setBaudRate(baudRate, true); -} - -/*! - For Unix: - - Sets the read and write timeouts for the port to \a millisec milliseconds. - Note that this is a per-character timeout, i.e. the port will wait this long for each - individual character, not for the whole read operation. This timeout also applies to the - bytesWaiting() function. - - \bold note: - POSIX does not support millisecond-level control for I/O timeout values. Any - timeout set using this function will be set to the next lowest tenth of a second for - the purposes of detecting read or write timeouts. For example a timeout of 550 milliseconds - will be seen by the class as a timeout of 500 milliseconds for the purposes of reading and - writing the port. However millisecond-level control is allowed by the select() system call, - so for example a 550-millisecond timeout will be seen as 550 milliseconds on POSIX systems for - the purpose of detecting available bytes in the read buffer. - - For Windows: - - Sets the read and write timeouts for the port to \a millisec milliseconds. - Setting 0 indicates that timeouts are not used for read nor write operations; - however read() and write() functions will still block. Set -1 to provide - non-blocking behaviour (read() and write() will return immediately). - - \bold note: this function does nothing in event driven mode. -*/ -void QextSerialPort::setTimeout(long millisec) -{ - Q_D(QextSerialPort); - QWriteLocker locker(&d->lock); - if (d->settings.Timeout_Millisec != millisec) - d->setTimeout(millisec, true); -} - -/*! - Sets DTR line to the requested state (\a set default to high). This function will have no effect if - the port associated with the class is not currently open. -*/ -void QextSerialPort::setDtr(bool set) -{ - Q_D(QextSerialPort); - QWriteLocker locker(&d->lock); - if (isOpen()) - d->setDtr_sys(set); -} - -/*! - Sets RTS line to the requested state \a set (high by default). - This function will have no effect if - the port associated with the class is not currently open. -*/ -void QextSerialPort::setRts(bool set) -{ - Q_D(QextSerialPort); - QWriteLocker locker(&d->lock); - if (isOpen()) - d->setRts_sys(set); -} - -/*! \reimp - Reads a block of data from the serial port. This function will read at most maxlen bytes from - the serial port and place them in the buffer pointed to by data. Return value is the number of - bytes actually read, or -1 on error. - - \warning before calling this function ensure that serial port associated with this class - is currently open (use isOpen() function to check if port is open). -*/ -qint64 QextSerialPort::readData(char *data, qint64 maxSize) -{ - Q_D(QextSerialPort); - QWriteLocker locker(&d->lock); - qint64 bytesFromBuffer = 0; - if (!d->readBuffer.isEmpty()) { - bytesFromBuffer = d->readBuffer.read(data, maxSize); - if (bytesFromBuffer == maxSize) - return bytesFromBuffer; - } - qint64 bytesFromDevice = d->readData_sys(data+bytesFromBuffer, maxSize-bytesFromBuffer); - if (bytesFromDevice < 0) - return -1; - return bytesFromBuffer + bytesFromDevice; -} - -/*! \reimp - Writes a block of data to the serial port. This function will write len bytes - from the buffer pointed to by data to the serial port. Return value is the number - of bytes actually written, or -1 on error. - - \warning before calling this function ensure that serial port associated with this class - is currently open (use isOpen() function to check if port is open). -*/ -qint64 QextSerialPort::writeData(const char *data, qint64 maxSize) -{ - Q_D(QextSerialPort); - QWriteLocker locker(&d->lock); - return d->writeData_sys(data, maxSize); -} - -#include "moc_qextserialport.cpp" diff --git a/3rdparty/qextserialport/src/qextserialport.h b/3rdparty/qextserialport/src/qextserialport.h deleted file mode 100644 index a0bc8220..00000000 --- a/3rdparty/qextserialport/src/qextserialport.h +++ /dev/null @@ -1,240 +0,0 @@ -/**************************************************************************** -** Copyright (c) 2000-2003 Wayne Roth -** Copyright (c) 2004-2007 Stefan Sander -** Copyright (c) 2007 Michal Policht -** Copyright (c) 2008 Brandon Fosdick -** Copyright (c) 2009-2010 Liam Staskawicz -** Copyright (c) 2011 Debao Zhang -** All right reserved. -** Web: http://code.google.com/p/qextserialport/ -** -** Permission is hereby granted, free of charge, to any person obtaining -** a copy of this software and associated documentation files (the -** "Software"), to deal in the Software without restriction, including -** without limitation the rights to use, copy, modify, merge, publish, -** distribute, sublicense, and/or sell copies of the Software, and to -** permit persons to whom the Software is furnished to do so, subject to -** the following conditions: -** -** The above copyright notice and this permission notice shall be -** included in all copies or substantial portions of the Software. -** -** THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, -** EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND -** NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE -** LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION -** OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION -** WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. -** -****************************************************************************/ - -#ifndef _QEXTSERIALPORT_H_ -#define _QEXTSERIALPORT_H_ - -#include -#include "qextserialport_global.h" -#ifdef Q_OS_UNIX -#include -#endif -/*line status constants*/ -// ### QESP2.0 move to enum -#define LS_CTS 0x01 -#define LS_DSR 0x02 -#define LS_DCD 0x04 -#define LS_RI 0x08 -#define LS_RTS 0x10 -#define LS_DTR 0x20 -#define LS_ST 0x40 -#define LS_SR 0x80 - -/*error constants*/ -// ### QESP2.0 move to enum -#define E_NO_ERROR 0 -#define E_INVALID_FD 1 -#define E_NO_MEMORY 2 -#define E_CAUGHT_NON_BLOCKED_SIGNAL 3 -#define E_PORT_TIMEOUT 4 -#define E_INVALID_DEVICE 5 -#define E_BREAK_CONDITION 6 -#define E_FRAMING_ERROR 7 -#define E_IO_ERROR 8 -#define E_BUFFER_OVERRUN 9 -#define E_RECEIVE_OVERFLOW 10 -#define E_RECEIVE_PARITY_ERROR 11 -#define E_TRANSMIT_OVERFLOW 12 -#define E_READ_FAILED 13 -#define E_WRITE_FAILED 14 -#define E_FILE_NOT_FOUND 15 -#define E_PERMISSION_DENIED 16 -#define E_AGAIN 17 - -enum BaudRateType -{ -#if defined(Q_OS_UNIX) || defined(qdoc) - BAUD50 = 50, //POSIX ONLY - BAUD75 = 75, //POSIX ONLY - BAUD134 = 134, //POSIX ONLY - BAUD150 = 150, //POSIX ONLY - BAUD200 = 200, //POSIX ONLY - BAUD1800 = 1800, //POSIX ONLY -# if defined(B76800) || defined(qdoc) - BAUD76800 = 76800, //POSIX ONLY -# endif -# if (defined(B230400) && defined(B4000000)) || defined(qdoc) - BAUD230400 = 230400, //POSIX ONLY - BAUD460800 = 460800, //POSIX ONLY - BAUD500000 = 500000, //POSIX ONLY - BAUD576000 = 576000, //POSIX ONLY - BAUD921600 = 921600, //POSIX ONLY - BAUD1000000 = 1000000, //POSIX ONLY - BAUD1152000 = 1152000, //POSIX ONLY - BAUD1500000 = 1500000, //POSIX ONLY - BAUD2000000 = 2000000, //POSIX ONLY - BAUD2500000 = 2500000, //POSIX ONLY - BAUD3000000 = 3000000, //POSIX ONLY - BAUD3500000 = 3500000, //POSIX ONLY - BAUD4000000 = 4000000, //POSIX ONLY -# endif -#endif //Q_OS_UNIX -#if defined(Q_OS_WIN) || defined(qdoc) - BAUD14400 = 14400, //WINDOWS ONLY - BAUD56000 = 56000, //WINDOWS ONLY - BAUD128000 = 128000, //WINDOWS ONLY - BAUD256000 = 256000, //WINDOWS ONLY -#endif //Q_OS_WIN - BAUD110 = 110, - BAUD300 = 300, - BAUD600 = 600, - BAUD1200 = 1200, - BAUD2400 = 2400, - BAUD4800 = 4800, - BAUD9600 = 9600, - BAUD19200 = 19200, - BAUD38400 = 38400, - BAUD57600 = 57600, - BAUD115200 = 115200 -}; - -enum DataBitsType -{ - DATA_5 = 5, - DATA_6 = 6, - DATA_7 = 7, - DATA_8 = 8 -}; - -enum ParityType -{ - PAR_NONE, - PAR_ODD, - PAR_EVEN, -#if defined(Q_OS_WIN) || defined(qdoc) - PAR_MARK, //WINDOWS ONLY -#endif - PAR_SPACE -}; - -enum StopBitsType -{ - STOP_1, -#if defined(Q_OS_WIN) || defined(qdoc) - STOP_1_5, //WINDOWS ONLY -#endif - STOP_2 -}; - -enum FlowType -{ - FLOW_OFF, - FLOW_HARDWARE, - FLOW_XONXOFF -}; - -/** - * structure to contain port settings - */ -struct PortSettings -{ - BaudRateType BaudRate; - DataBitsType DataBits; - ParityType Parity; - StopBitsType StopBits; - FlowType FlowControl; - long Timeout_Millisec; -}; - -class QextSerialPortPrivate; -class QEXTSERIALPORT_EXPORT QextSerialPort: public QIODevice -{ - Q_OBJECT - Q_DECLARE_PRIVATE(QextSerialPort) - Q_ENUMS(QueryMode) - Q_PROPERTY(QString portName READ portName WRITE setPortName) - Q_PROPERTY(QueryMode queryMode READ queryMode WRITE setQueryMode) -public: - enum QueryMode { - Polling, - EventDriven - }; - - explicit QextSerialPort(QueryMode mode = EventDriven, QObject *parent = 0); - explicit QextSerialPort(const QString &name, QueryMode mode = EventDriven, QObject *parent = 0); - explicit QextSerialPort(const PortSettings &s, QueryMode mode = EventDriven, QObject *parent = 0); - QextSerialPort(const QString &name, const PortSettings &s, QueryMode mode = EventDriven, QObject *parent=0); - - ~QextSerialPort(); - - QString portName() const; - QueryMode queryMode() const; - BaudRateType baudRate() const; - DataBitsType dataBits() const; - ParityType parity() const; - StopBitsType stopBits() const; - FlowType flowControl() const; - - bool open(OpenMode mode); - bool isSequential() const; - void close(); - void flush(); - qint64 bytesAvailable() const; - bool canReadLine() const; - QByteArray readAll(); - - ulong lastError() const; - - ulong lineStatus(); - QString errorString(); - -public Q_SLOTS: - void setPortName(const QString &name); - void setQueryMode(QueryMode mode); - void setBaudRate(BaudRateType); - void setDataBits(DataBitsType); - void setParity(ParityType); - void setStopBits(StopBitsType); - void setFlowControl(FlowType); - void setTimeout(long); - - void setDtr(bool set=true); - void setRts(bool set=true); - -Q_SIGNALS: - void dsrChanged(bool status); - -protected: - qint64 readData(char *data, qint64 maxSize); - qint64 writeData(const char *data, qint64 maxSize); - -private: - Q_DISABLE_COPY(QextSerialPort) - -#ifdef Q_OS_WIN - Q_PRIVATE_SLOT(d_func(), void _q_onWinEvent(HANDLE)) -#endif - Q_PRIVATE_SLOT(d_func(), void _q_canRead()) - - QextSerialPortPrivate * const d_ptr; -}; - -#endif diff --git a/3rdparty/qextserialport/src/qextserialport.pri b/3rdparty/qextserialport/src/qextserialport.pri deleted file mode 100644 index 461d56f6..00000000 --- a/3rdparty/qextserialport/src/qextserialport.pri +++ /dev/null @@ -1,36 +0,0 @@ -INCLUDEPATH += $$PWD -DEPENDPATH += $$PWD - -PUBLIC_HEADERS += $$PWD/qextserialport.h \ - $$PWD/qextserialenumerator.h \ - $$PWD/qextserialport_global.h - -HEADERS += $$PUBLIC_HEADERS \ - $$PWD/qextserialport_p.h \ - $$PWD/qextserialenumerator_p.h \ - -SOURCES += $$PWD/qextserialport.cpp \ - $$PWD/qextserialenumerator.cpp -unix { - SOURCES += $$PWD/qextserialport_unix.cpp - linux* { - SOURCES += $$PWD/qextserialenumerator_linux.cpp - } else:macx { - SOURCES += $$PWD/qextserialenumerator_osx.cpp - } else { - SOURCES += $$PWD/qextserialenumerator_unix.cpp - } -} -win32:SOURCES += $$PWD/qextserialport_win.cpp \ - $$PWD/qextserialenumerator_win.cpp - -linux*{ - !qesp_linux_udev:DEFINES += QESP_NO_UDEV - qesp_linux_udev: LIBS += -ludev -} - -macx:LIBS += -framework IOKit -framework CoreFoundation -win32:LIBS += -lsetupapi -ladvapi32 -luser32 - -# moc doesn't detect Q_OS_LINUX correctly, so add this to make it work -linux*:DEFINES += __linux__ diff --git a/3rdparty/qextserialport/src/qextserialport_global.h b/3rdparty/qextserialport/src/qextserialport_global.h deleted file mode 100644 index 824d4554..00000000 --- a/3rdparty/qextserialport/src/qextserialport_global.h +++ /dev/null @@ -1,72 +0,0 @@ -/**************************************************************************** -** Copyright (c) 2000-2003 Wayne Roth -** Copyright (c) 2004-2007 Stefan Sander -** Copyright (c) 2007 Michal Policht -** Copyright (c) 2008 Brandon Fosdick -** Copyright (c) 2009-2010 Liam Staskawicz -** Copyright (c) 2011 Debao Zhang -** All right reserved. -** Web: http://code.google.com/p/qextserialport/ -** -** Permission is hereby granted, free of charge, to any person obtaining -** a copy of this software and associated documentation files (the -** "Software"), to deal in the Software without restriction, including -** without limitation the rights to use, copy, modify, merge, publish, -** distribute, sublicense, and/or sell copies of the Software, and to -** permit persons to whom the Software is furnished to do so, subject to -** the following conditions: -** -** The above copyright notice and this permission notice shall be -** included in all copies or substantial portions of the Software. -** -** THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, -** EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND -** NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE -** LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION -** OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION -** WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. -** -****************************************************************************/ - -#ifndef QEXTSERIALPORT_GLOBAL_H -#define QEXTSERIALPORT_GLOBAL_H - -#include - -#ifdef QEXTSERIALPORT_BUILD_SHARED -# define QEXTSERIALPORT_EXPORT Q_DECL_EXPORT -#elif defined(QEXTSERIALPORT_USING_SHARED) -# define QEXTSERIALPORT_EXPORT Q_DECL_IMPORT -#else -# define QEXTSERIALPORT_EXPORT -#endif - -// ### for compatible with old version. should be removed in QESP 2.0 -#ifdef _TTY_NOWARN_ -# define QESP_NO_WARN -#endif -#ifdef _TTY_NOWARN_PORT_ -# define QESP_NO_PORTABILITY_WARN -#endif - -/*if all warning messages are turned off, flag portability warnings to be turned off as well*/ -#ifdef QESP_NO_WARN -# define QESP_NO_PORTABILITY_WARN -#endif - -/*macros for warning and debug messages*/ -#ifdef QESP_NO_PORTABILITY_WARN -# define QESP_PORTABILITY_WARNING while (false)qWarning -#else -# define QESP_PORTABILITY_WARNING qWarning -#endif /*QESP_NOWARN_PORT*/ - -#ifdef QESP_NO_WARN -# define QESP_WARNING while (false)qWarning -#else -# define QESP_WARNING qWarning -#endif /*QESP_NOWARN*/ - -#endif // QEXTSERIALPORT_GLOBAL_H - diff --git a/3rdparty/qextserialport/src/qextserialport_p.h b/3rdparty/qextserialport/src/qextserialport_p.h deleted file mode 100644 index 84f71c99..00000000 --- a/3rdparty/qextserialport/src/qextserialport_p.h +++ /dev/null @@ -1,250 +0,0 @@ -/**************************************************************************** -** Copyright (c) 2000-2003 Wayne Roth -** Copyright (c) 2004-2007 Stefan Sander -** Copyright (c) 2007 Michal Policht -** Copyright (c) 2008 Brandon Fosdick -** Copyright (c) 2009-2010 Liam Staskawicz -** Copyright (c) 2011 Debao Zhang -** All right reserved. -** Web: http://code.google.com/p/qextserialport/ -** -** Permission is hereby granted, free of charge, to any person obtaining -** a copy of this software and associated documentation files (the -** "Software"), to deal in the Software without restriction, including -** without limitation the rights to use, copy, modify, merge, publish, -** distribute, sublicense, and/or sell copies of the Software, and to -** permit persons to whom the Software is furnished to do so, subject to -** the following conditions: -** -** The above copyright notice and this permission notice shall be -** included in all copies or substantial portions of the Software. -** -** THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, -** EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND -** NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE -** LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION -** OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION -** WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. -** -****************************************************************************/ - -#ifndef _QEXTSERIALPORT_P_H_ -#define _QEXTSERIALPORT_P_H_ - -// -// W A R N I N G -// ------------- -// -// This file is not part of the QESP API. It exists for the convenience -// of other QESP classes. This header file may change from version to -// version without notice, or even be removed. -// -// We mean it. -// - -#include "qextserialport.h" -#include -#ifdef Q_OS_UNIX -# include -#elif (defined Q_OS_WIN) -# include -#endif -#include - -// This is QextSerialPort's read buffer, needed by posix system. -// ref: QRingBuffer & QIODevicePrivateLinearBuffer -class QextReadBuffer -{ -public: - inline QextReadBuffer(size_t growth=4096) - : len(0), first(0), buf(0), capacity(0), basicBlockSize(growth) { - } - - ~QextReadBuffer() { - delete [] buf; - } - - inline void clear() { - first = buf; - len = 0; - } - - inline int size() const { - return len; - } - - inline bool isEmpty() const { - return len == 0; - } - - inline int read(char *target, int size) { - int r = qMin(size, len); - if (r == 1) { - *target = *first; - --len; - ++first; - } else { - memcpy(target, first, r); - len -= r; - first += r; - } - return r; - } - - inline char *reserve(size_t size) { - if ((first - buf) + len + size > capacity) { - size_t newCapacity = qMax(capacity, basicBlockSize); - while (newCapacity < len + size) - newCapacity *= 2; - if (newCapacity > capacity) { - // allocate more space - char *newBuf = new char[newCapacity]; - memmove(newBuf, first, len); - delete [] buf; - buf = newBuf; - capacity = newCapacity; - } else { - // shift any existing data to make space - memmove(buf, first, len); - } - first = buf; - } - char *writePtr = first + len; - len += (int)size; - return writePtr; - } - - inline void chop(int size) { - if (size >= len) - clear(); - else - len -= size; - } - - inline void squeeze() { - if (first != buf) { - memmove(buf, first, len); - first = buf; - } - size_t newCapacity = basicBlockSize; - while (newCapacity < size_t(len)) - newCapacity *= 2; - if (newCapacity < capacity) { - char *tmp = static_cast(realloc(buf, newCapacity)); - if (tmp) { - buf = tmp; - capacity = newCapacity; - } - } - } - - inline QByteArray readAll() { - char *f = first; - int l = len; - clear(); - return QByteArray(f, l); - } - - inline int readLine(char *target, int size) { - int r = qMin(size, len); - char *eol = static_cast(memchr(first, '\n', r)); - if (eol) - r = 1+(eol-first); - memcpy(target, first, r); - len -= r; - first += r; - return int(r); - } - - inline bool canReadLine() const { - return memchr(first, '\n', len); - } - -private: - int len; - char *first; - char *buf; - size_t capacity; - size_t basicBlockSize; -}; - -class QWinEventNotifier; -class QReadWriteLock; -class QSocketNotifier; - -class QextSerialPortPrivate -{ - Q_DECLARE_PUBLIC(QextSerialPort) -public: - QextSerialPortPrivate(QextSerialPort *q); - ~QextSerialPortPrivate(); - enum DirtyFlagEnum - { - DFE_BaudRate = 0x0001, - DFE_Parity = 0x0002, - DFE_StopBits = 0x0004, - DFE_DataBits = 0x0008, - DFE_Flow = 0x0010, - DFE_TimeOut = 0x0100, - DFE_ALL = 0x0fff, - DFE_Settings_Mask = 0x00ff //without TimeOut - }; - mutable QReadWriteLock lock; - QString port; - PortSettings settings; - QextReadBuffer readBuffer; - int settingsDirtyFlags; - ulong lastErr; - QextSerialPort::QueryMode queryMode; - - // platform specific members -#ifdef Q_OS_UNIX - int fd; - QSocketNotifier *readNotifier; - struct termios currentTermios; - struct termios oldTermios; -#elif (defined Q_OS_WIN) - HANDLE handle; - OVERLAPPED overlap; - COMMCONFIG commConfig; - COMMTIMEOUTS commTimeouts; - QWinEventNotifier *winEventNotifier; - DWORD eventMask; - QList pendingWrites; - QReadWriteLock *bytesToWriteLock; -#endif - - /*fill PortSettings*/ - void setBaudRate(BaudRateType baudRate, bool update=true); - void setDataBits(DataBitsType dataBits, bool update=true); - void setParity(ParityType parity, bool update=true); - void setStopBits(StopBitsType stopbits, bool update=true); - void setFlowControl(FlowType flow, bool update=true); - void setTimeout(long millisec, bool update=true); - void setPortSettings(const PortSettings &settings, bool update=true); - - void platformSpecificDestruct(); - void platformSpecificInit(); - void translateError(ulong error); - void updatePortSettings(); - - qint64 readData_sys(char *data, qint64 maxSize); - qint64 writeData_sys(const char *data, qint64 maxSize); - void setDtr_sys(bool set=true); - void setRts_sys(bool set=true); - bool open_sys(QIODevice::OpenMode mode); - bool close_sys(); - bool flush_sys(); - ulong lineStatus_sys(); - qint64 bytesAvailable_sys() const; - -#ifdef Q_OS_WIN - void _q_onWinEvent(HANDLE h); -#endif - void _q_canRead(); - - QextSerialPort *q_ptr; -}; - -#endif //_QEXTSERIALPORT_P_H_ diff --git a/3rdparty/qextserialport/src/qextserialport_unix.cpp b/3rdparty/qextserialport/src/qextserialport_unix.cpp deleted file mode 100644 index 09c65685..00000000 --- a/3rdparty/qextserialport/src/qextserialport_unix.cpp +++ /dev/null @@ -1,458 +0,0 @@ -/**************************************************************************** -** Copyright (c) 2000-2003 Wayne Roth -** Copyright (c) 2004-2007 Stefan Sander -** Copyright (c) 2007 Michal Policht -** Copyright (c) 2008 Brandon Fosdick -** Copyright (c) 2009-2010 Liam Staskawicz -** Copyright (c) 2011 Debao Zhang -** All right reserved. -** Web: http://code.google.com/p/qextserialport/ -** -** Permission is hereby granted, free of charge, to any person obtaining -** a copy of this software and associated documentation files (the -** "Software"), to deal in the Software without restriction, including -** without limitation the rights to use, copy, modify, merge, publish, -** distribute, sublicense, and/or sell copies of the Software, and to -** permit persons to whom the Software is furnished to do so, subject to -** the following conditions: -** -** The above copyright notice and this permission notice shall be -** included in all copies or substantial portions of the Software. -** -** THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, -** EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND -** NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE -** LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION -** OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION -** WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. -** -****************************************************************************/ - -#include "qextserialport.h" -#include "qextserialport_p.h" -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -void QextSerialPortPrivate::platformSpecificInit() -{ - fd = 0; - readNotifier = 0; -} - -/*! - Standard destructor. -*/ -void QextSerialPortPrivate::platformSpecificDestruct() -{ -} - -static QString fullPortName(const QString &name) -{ - if (name.startsWith(QLatin1Char('/'))) - return name; - return QLatin1String("/dev/")+name; -} - -bool QextSerialPortPrivate::open_sys(QIODevice::OpenMode mode) -{ - Q_Q(QextSerialPort); - //note: linux 2.6.21 seems to ignore O_NDELAY flag - if ((fd = ::open(fullPortName(port).toLatin1() ,O_RDWR | O_NOCTTY | O_NDELAY)) != -1) { - - /*In the Private class, We can not call QIODevice::open()*/ - q->setOpenMode(mode); // Flag the port as opened - ::tcgetattr(fd, &oldTermios); // Save the old termios - currentTermios = oldTermios; // Make a working copy - ::cfmakeraw(¤tTermios); // Enable raw access - - /*set up other port settings*/ - currentTermios.c_cflag |= CREAD|CLOCAL; - currentTermios.c_lflag &= (~(ICANON|ECHO|ECHOE|ECHOK|ECHONL|ISIG)); - currentTermios.c_iflag &= (~(INPCK|IGNPAR|PARMRK|ISTRIP|ICRNL|IXANY)); - currentTermios.c_oflag &= (~OPOST); - currentTermios.c_cc[VMIN] = 0; -#ifdef _POSIX_VDISABLE // Is a disable character available on this system? - // Some systems allow for per-device disable-characters, so get the - // proper value for the configured device - const long vdisable = ::fpathconf(fd, _PC_VDISABLE); - currentTermios.c_cc[VINTR] = vdisable; - currentTermios.c_cc[VQUIT] = vdisable; - currentTermios.c_cc[VSTART] = vdisable; - currentTermios.c_cc[VSTOP] = vdisable; - currentTermios.c_cc[VSUSP] = vdisable; -#endif //_POSIX_VDISABLE - settingsDirtyFlags = DFE_ALL; - updatePortSettings(); - - if (queryMode == QextSerialPort::EventDriven) { - readNotifier = new QSocketNotifier(fd, QSocketNotifier::Read, q); - q->connect(readNotifier, SIGNAL(activated(int)), q, SLOT(_q_canRead())); - } - return true; - } else { - translateError(errno); - return false; - } -} - -bool QextSerialPortPrivate::close_sys() -{ - // Force a flush and then restore the original termios - flush_sys(); - // Using both TCSAFLUSH and TCSANOW here discards any pending input - ::tcsetattr(fd, TCSAFLUSH | TCSANOW, &oldTermios); // Restore termios - ::close(fd); - if (readNotifier) { - delete readNotifier; - readNotifier = 0; - } - return true; -} - -bool QextSerialPortPrivate::flush_sys() -{ - ::tcdrain(fd); - return true; -} - -qint64 QextSerialPortPrivate::bytesAvailable_sys() const -{ - int bytesQueued; - if (::ioctl(fd, FIONREAD, &bytesQueued) == -1) - return (qint64)-1; - return bytesQueued; -} - -/*! - Translates a system-specific error code to a QextSerialPort error code. Used internally. -*/ -void QextSerialPortPrivate::translateError(ulong error) -{ - switch (error) { - case EBADF: - case ENOTTY: - lastErr = E_INVALID_FD; - break; - case EINTR: - lastErr = E_CAUGHT_NON_BLOCKED_SIGNAL; - break; - case ENOMEM: - lastErr = E_NO_MEMORY; - break; - case EACCES: - lastErr = E_PERMISSION_DENIED; - break; - case EAGAIN: - lastErr = E_AGAIN; - break; - } -} - -void QextSerialPortPrivate::setDtr_sys(bool set) -{ - int status; - ::ioctl(fd, TIOCMGET, &status); - if (set) - status |= TIOCM_DTR; - else - status &= ~TIOCM_DTR; - ::ioctl(fd, TIOCMSET, &status); -} - -void QextSerialPortPrivate::setRts_sys(bool set) -{ - int status; - ::ioctl(fd, TIOCMGET, &status); - if (set) - status |= TIOCM_RTS; - else - status &= ~TIOCM_RTS; - ::ioctl(fd, TIOCMSET, &status); -} - -unsigned long QextSerialPortPrivate::lineStatus_sys() -{ - unsigned long Status=0, Temp=0; - ::ioctl(fd, TIOCMGET, &Temp); - if (Temp & TIOCM_CTS) Status |= LS_CTS; - if (Temp & TIOCM_DSR) Status |= LS_DSR; - if (Temp & TIOCM_RI) Status |= LS_RI; - if (Temp & TIOCM_CD) Status |= LS_DCD; - if (Temp & TIOCM_DTR) Status |= LS_DTR; - if (Temp & TIOCM_RTS) Status |= LS_RTS; - if (Temp & TIOCM_ST) Status |= LS_ST; - if (Temp & TIOCM_SR) Status |= LS_SR; - return Status; -} - -/*! - Reads a block of data from the serial port. This function will read at most maxSize bytes from - the serial port and place them in the buffer pointed to by data. Return value is the number of - bytes actually read, or -1 on error. - - \warning before calling this function ensure that serial port associated with this class - is currently open (use isOpen() function to check if port is open). -*/ -qint64 QextSerialPortPrivate::readData_sys(char *data, qint64 maxSize) -{ - int retVal = ::read(fd, data, maxSize); - if (retVal == -1) - lastErr = E_READ_FAILED; - - return retVal; -} - -/*! - Writes a block of data to the serial port. This function will write maxSize bytes - from the buffer pointed to by data to the serial port. Return value is the number - of bytes actually written, or -1 on error. - - \warning before calling this function ensure that serial port associated with this class - is currently open (use isOpen() function to check if port is open). -*/ -qint64 QextSerialPortPrivate::writeData_sys(const char *data, qint64 maxSize) -{ - int retVal = ::write(fd, data, maxSize); - if (retVal == -1) - lastErr = E_WRITE_FAILED; - - return (qint64)retVal; -} - -static void setBaudRate2Termios(termios *config, int baudRate) -{ -#ifdef CBAUD - config->c_cflag &= (~CBAUD); - config->c_cflag |= baudRate; -#else - ::cfsetispeed(config, baudRate); - ::cfsetospeed(config, baudRate); -#endif -} - -/* - All the platform settings was performed in this function. -*/ -void QextSerialPortPrivate::updatePortSettings() -{ - if (!q_func()->isOpen() || !settingsDirtyFlags) - return; - - if (settingsDirtyFlags & DFE_BaudRate) { - switch (settings.BaudRate) { - case BAUD50: - setBaudRate2Termios(¤tTermios, B50); - break; - case BAUD75: - setBaudRate2Termios(¤tTermios, B75); - break; - case BAUD110: - setBaudRate2Termios(¤tTermios, B110); - break; - case BAUD134: - setBaudRate2Termios(¤tTermios, B134); - break; - case BAUD150: - setBaudRate2Termios(¤tTermios, B150); - break; - case BAUD200: - setBaudRate2Termios(¤tTermios, B200); - break; - case BAUD300: - setBaudRate2Termios(¤tTermios, B300); - break; - case BAUD600: - setBaudRate2Termios(¤tTermios, B600); - break; - case BAUD1200: - setBaudRate2Termios(¤tTermios, B1200); - break; - case BAUD1800: - setBaudRate2Termios(¤tTermios, B1800); - break; - case BAUD2400: - setBaudRate2Termios(¤tTermios, B2400); - break; - case BAUD4800: - setBaudRate2Termios(¤tTermios, B4800); - break; - case BAUD9600: - setBaudRate2Termios(¤tTermios, B9600); - break; - case BAUD19200: - setBaudRate2Termios(¤tTermios, B19200); - break; - case BAUD38400: - setBaudRate2Termios(¤tTermios, B38400); - break; - case BAUD57600: - setBaudRate2Termios(¤tTermios, B57600); - break; -#ifdef B76800 - case BAUD76800: - setBaudRate2Termios(¤tTermios, B76800); - break; -#endif - case BAUD115200: - setBaudRate2Termios(¤tTermios, B115200); - break; -#if defined(B230400) && defined(B4000000) - case BAUD230400: - setBaudRate2Termios(¤tTermios, B230400); - break; - case BAUD460800: - setBaudRate2Termios(¤tTermios, B460800); - break; - case BAUD500000: - setBaudRate2Termios(¤tTermios, B500000); - break; - case BAUD576000: - setBaudRate2Termios(¤tTermios, B576000); - break; - case BAUD921600: - setBaudRate2Termios(¤tTermios, B921600); - break; - case BAUD1000000: - setBaudRate2Termios(¤tTermios, B1000000); - break; - case BAUD1152000: - setBaudRate2Termios(¤tTermios, B1152000); - break; - case BAUD1500000: - setBaudRate2Termios(¤tTermios, B1500000); - break; - case BAUD2000000: - setBaudRate2Termios(¤tTermios, B2000000); - break; - case BAUD2500000: - setBaudRate2Termios(¤tTermios, B2500000); - break; - case BAUD3000000: - setBaudRate2Termios(¤tTermios, B3000000); - break; - case BAUD3500000: - setBaudRate2Termios(¤tTermios, B3500000); - break; - case BAUD4000000: - setBaudRate2Termios(¤tTermios, B4000000); - break; -#endif -#ifdef Q_OS_MAC - default: - setBaudRate2Termios(¤tTermios, settings.BaudRate); - break; -#endif - } - } - if (settingsDirtyFlags & DFE_Parity) { - switch (settings.Parity) { - case PAR_SPACE: - /*space parity not directly supported - add an extra data bit to simulate it*/ - settingsDirtyFlags |= DFE_DataBits; - break; - case PAR_NONE: - currentTermios.c_cflag &= (~PARENB); - break; - case PAR_EVEN: - currentTermios.c_cflag &= (~PARODD); - currentTermios.c_cflag |= PARENB; - break; - case PAR_ODD: - currentTermios.c_cflag |= (PARENB|PARODD); - break; - } - } - /*must after Parity settings*/ - if (settingsDirtyFlags & DFE_DataBits) { - if (settings.Parity != PAR_SPACE) { - currentTermios.c_cflag &= (~CSIZE); - switch(settings.DataBits) { - case DATA_5: - currentTermios.c_cflag |= CS5; - break; - case DATA_6: - currentTermios.c_cflag |= CS6; - break; - case DATA_7: - currentTermios.c_cflag |= CS7; - break; - case DATA_8: - currentTermios.c_cflag |= CS8; - break; - } - } else { - /*space parity not directly supported - add an extra data bit to simulate it*/ - currentTermios.c_cflag &= ~(PARENB|CSIZE); - switch(settings.DataBits) { - case DATA_5: - currentTermios.c_cflag |= CS6; - break; - case DATA_6: - currentTermios.c_cflag |= CS7; - break; - case DATA_7: - currentTermios.c_cflag |= CS8; - break; - case DATA_8: - /*this will never happen, put here to Suppress an warning*/ - break; - } - } - } - if (settingsDirtyFlags & DFE_StopBits) { - switch (settings.StopBits) { - case STOP_1: - currentTermios.c_cflag &= (~CSTOPB); - break; - case STOP_2: - currentTermios.c_cflag |= CSTOPB; - break; - } - } - if (settingsDirtyFlags & DFE_Flow) { - switch(settings.FlowControl) { - case FLOW_OFF: - currentTermios.c_cflag &= (~CRTSCTS); - currentTermios.c_iflag &= (~(IXON|IXOFF|IXANY)); - break; - case FLOW_XONXOFF: - /*software (XON/XOFF) flow control*/ - currentTermios.c_cflag &= (~CRTSCTS); - currentTermios.c_iflag |= (IXON|IXOFF|IXANY); - break; - case FLOW_HARDWARE: - currentTermios.c_cflag |= CRTSCTS; - currentTermios.c_iflag &= (~(IXON|IXOFF|IXANY)); - break; - } - } - - /*if any thing in currentTermios changed, flush*/ - if (settingsDirtyFlags & DFE_Settings_Mask) - ::tcsetattr(fd, TCSAFLUSH, ¤tTermios); - - if (settingsDirtyFlags & DFE_TimeOut) { - int millisec = settings.Timeout_Millisec; - if (millisec == -1) { - ::fcntl(fd, F_SETFL, O_NDELAY); - } else { - //O_SYNC should enable blocking ::write() - //however this seems not working on Linux 2.6.21 (works on OpenBSD 4.2) - ::fcntl(fd, F_SETFL, O_SYNC); - } - ::tcgetattr(fd, ¤tTermios); - currentTermios.c_cc[VTIME] = millisec/100; - ::tcsetattr(fd, TCSAFLUSH, ¤tTermios); - } - - settingsDirtyFlags = 0; -} diff --git a/3rdparty/qextserialport/src/qextserialport_win.cpp b/3rdparty/qextserialport/src/qextserialport_win.cpp deleted file mode 100644 index 1d25e0e6..00000000 --- a/3rdparty/qextserialport/src/qextserialport_win.cpp +++ /dev/null @@ -1,405 +0,0 @@ -/**************************************************************************** -** Copyright (c) 2000-2003 Wayne Roth -** Copyright (c) 2004-2007 Stefan Sander -** Copyright (c) 2007 Michal Policht -** Copyright (c) 2008 Brandon Fosdick -** Copyright (c) 2009-2010 Liam Staskawicz -** Copyright (c) 2011 Debao Zhang -** All right reserved. -** Web: http://code.google.com/p/qextserialport/ -** -** Permission is hereby granted, free of charge, to any person obtaining -** a copy of this software and associated documentation files (the -** "Software"), to deal in the Software without restriction, including -** without limitation the rights to use, copy, modify, merge, publish, -** distribute, sublicense, and/or sell copies of the Software, and to -** permit persons to whom the Software is furnished to do so, subject to -** the following conditions: -** -** The above copyright notice and this permission notice shall be -** included in all copies or substantial portions of the Software. -** -** THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, -** EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF -** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND -** NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE -** LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION -** OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION -** WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. -** -****************************************************************************/ - -#include "qextserialport.h" -#include "qextserialport_p.h" -#include -#include -#include -#include -#include -#include -#if QT_VERSION >= QT_VERSION_CHECK(5, 0, 0) -# include -#else -# include -#endif -void QextSerialPortPrivate::platformSpecificInit() -{ - handle = INVALID_HANDLE_VALUE; - ZeroMemory(&overlap, sizeof(OVERLAPPED)); - overlap.hEvent = CreateEvent(NULL, true, false, NULL); - winEventNotifier = 0; - bytesToWriteLock = new QReadWriteLock; -} - -void QextSerialPortPrivate::platformSpecificDestruct() { - CloseHandle(overlap.hEvent); - delete bytesToWriteLock; -} - - -/*! - \internal - COM ports greater than 9 need \\.\ prepended - - This is only need when open the port. -*/ -static QString fullPortNameWin(const QString &name) -{ - QRegExp rx(QLatin1String("^COM(\\d+)")); - QString fullName(name); - if (fullName.contains(rx)) - fullName.prepend(QLatin1String("\\\\.\\")); - return fullName; -} - -bool QextSerialPortPrivate::open_sys(QIODevice::OpenMode mode) -{ - Q_Q(QextSerialPort); - DWORD confSize = sizeof(COMMCONFIG); - commConfig.dwSize = confSize; - DWORD dwFlagsAndAttributes = 0; - if (queryMode == QextSerialPort::EventDriven) - dwFlagsAndAttributes += FILE_FLAG_OVERLAPPED; - - /*open the port*/ - handle = CreateFileW((wchar_t *)fullPortNameWin(port).utf16(), GENERIC_READ|GENERIC_WRITE, - 0, NULL, OPEN_EXISTING, dwFlagsAndAttributes, NULL); - if (handle != INVALID_HANDLE_VALUE) { - q->setOpenMode(mode); - /*configure port settings*/ - GetCommConfig(handle, &commConfig, &confSize); - GetCommState(handle, &(commConfig.dcb)); - - /*set up parameters*/ - commConfig.dcb.fBinary = TRUE; - commConfig.dcb.fInX = FALSE; - commConfig.dcb.fOutX = FALSE; - commConfig.dcb.fAbortOnError = FALSE; - commConfig.dcb.fNull = FALSE; - /* Dtr default to true. See Issue 122*/ - commConfig.dcb.fDtrControl = TRUE; - /*flush all settings*/ - settingsDirtyFlags = DFE_ALL; - updatePortSettings(); - - //init event driven approach - if (queryMode == QextSerialPort::EventDriven) { - if (!SetCommMask(handle, EV_TXEMPTY | EV_RXCHAR | EV_DSR)) { - QESP_WARNING()<<"failed to set Comm Mask. Error code:"<("HANDLE"); - q->connect(winEventNotifier, SIGNAL(activated(HANDLE)), q, SLOT(_q_onWinEvent(HANDLE)), Qt::DirectConnection); - WaitCommEvent(handle, &eventMask, &overlap); - } - return true; - } - return false; -} - -bool QextSerialPortPrivate::close_sys() -{ - flush_sys(); - CancelIo(handle); - if (CloseHandle(handle)) - handle = INVALID_HANDLE_VALUE; - if (winEventNotifier) { - winEventNotifier->setEnabled(false); - winEventNotifier->deleteLater(); - winEventNotifier = 0; - } - - foreach (OVERLAPPED *o, pendingWrites) { - CloseHandle(o->hEvent); - delete o; - } - pendingWrites.clear(); - return true; -} - -bool QextSerialPortPrivate::flush_sys() -{ - FlushFileBuffers(handle); - return true; -} - -qint64 QextSerialPortPrivate::bytesAvailable_sys() const -{ - DWORD Errors; - COMSTAT Status; - if (ClearCommError(handle, &Errors, &Status)) - return Status.cbInQue; - - return (qint64)-1; -} - -/* - Translates a system-specific error code to a QextSerialPort error code. Used internally. -*/ -void QextSerialPortPrivate::translateError(ulong error) -{ - if (error & CE_BREAK) { - lastErr = E_BREAK_CONDITION; - } else if (error & CE_FRAME) { - lastErr = E_FRAMING_ERROR; - } else if (error & CE_IOE) { - lastErr = E_IO_ERROR; - } else if (error & CE_MODE) { - lastErr = E_INVALID_FD; - } else if (error & CE_OVERRUN) { - lastErr = E_BUFFER_OVERRUN; - } else if (error & CE_RXPARITY) { - lastErr = E_RECEIVE_PARITY_ERROR; - } else if (error & CE_RXOVER) { - lastErr = E_RECEIVE_OVERFLOW; - } else if (error & CE_TXFULL) { - lastErr = E_TRANSMIT_OVERFLOW; - } -} - -/* - Reads a block of data from the serial port. This function will read at most maxlen bytes from - the serial port and place them in the buffer pointed to by data. Return value is the number of - bytes actually read, or -1 on error. - - \warning before calling this function ensure that serial port associated with this class - is currently open (use isOpen() function to check if port is open). -*/ -qint64 QextSerialPortPrivate::readData_sys(char *data, qint64 maxSize) -{ - DWORD bytesRead = 0; - bool failed = false; - if (queryMode == QextSerialPort::EventDriven) { - OVERLAPPED overlapRead; - ZeroMemory(&overlapRead, sizeof(OVERLAPPED)); - if (!ReadFile(handle, (void *)data, (DWORD)maxSize, &bytesRead, &overlapRead)) { - if (GetLastError() == ERROR_IO_PENDING) - GetOverlappedResult(handle, &overlapRead, &bytesRead, true); - else - failed = true; - } - } else if (!ReadFile(handle, (void *)data, (DWORD)maxSize, &bytesRead, NULL)) { - failed = true; - } - if (!failed) - return (qint64)bytesRead; - - lastErr = E_READ_FAILED; - return -1; -} - -/* - Writes a block of data to the serial port. This function will write len bytes - from the buffer pointed to by data to the serial port. Return value is the number - of bytes actually written, or -1 on error. - - \warning before calling this function ensure that serial port associated with this class - is currently open (use isOpen() function to check if port is open). -*/ -qint64 QextSerialPortPrivate::writeData_sys(const char *data, qint64 maxSize) -{ - DWORD bytesWritten = 0; - bool failed = false; - if (queryMode == QextSerialPort::EventDriven) { - OVERLAPPED *newOverlapWrite = new OVERLAPPED; - ZeroMemory(newOverlapWrite, sizeof(OVERLAPPED)); - newOverlapWrite->hEvent = CreateEvent(NULL, true, false, NULL); - if (WriteFile(handle, (void *)data, (DWORD)maxSize, &bytesWritten, newOverlapWrite)) { - CloseHandle(newOverlapWrite->hEvent); - delete newOverlapWrite; - } else if (GetLastError() == ERROR_IO_PENDING) { - // writing asynchronously...not an error - QWriteLocker writelocker(bytesToWriteLock); - pendingWrites.append(newOverlapWrite); - } else { - QESP_WARNING()<<"QextSerialPort write error:"<hEvent)) - QESP_WARNING("QextSerialPort: couldn't cancel IO"); - if (!CloseHandle(newOverlapWrite->hEvent)) - QESP_WARNING("QextSerialPort: couldn't close OVERLAPPED handle"); - delete newOverlapWrite; - } - } else if (!WriteFile(handle, (void *)data, (DWORD)maxSize, &bytesWritten, NULL)) { - failed = true; - } - - if (!failed) - return (qint64)bytesWritten; - - lastErr = E_WRITE_FAILED; - return -1; -} - -void QextSerialPortPrivate::setDtr_sys(bool set) { - EscapeCommFunction(handle, set ? SETDTR : CLRDTR); -} - -void QextSerialPortPrivate::setRts_sys(bool set) { - EscapeCommFunction(handle, set ? SETRTS : CLRRTS); -} - -ulong QextSerialPortPrivate::lineStatus_sys(void) { - unsigned long Status = 0, Temp = 0; - GetCommModemStatus(handle, &Temp); - if (Temp & MS_CTS_ON) Status |= LS_CTS; - if (Temp & MS_DSR_ON) Status |= LS_DSR; - if (Temp & MS_RING_ON) Status |= LS_RI; - if (Temp & MS_RLSD_ON) Status |= LS_DCD; - return Status; -} - -/* - Triggered when there's activity on our HANDLE. -*/ -void QextSerialPortPrivate::_q_onWinEvent(HANDLE h) -{ - Q_Q(QextSerialPort); - if (h == overlap.hEvent) { - if (eventMask & EV_RXCHAR) { - if (q->sender() != q && bytesAvailable_sys() > 0) - _q_canRead(); - } - if (eventMask & EV_TXEMPTY) { - /* - A write completed. Run through the list of OVERLAPPED writes, and if - they completed successfully, take them off the list and delete them. - Otherwise, leave them on there so they can finish. - */ - qint64 totalBytesWritten = 0; - QList overlapsToDelete; - QWriteLocker writelocker(bytesToWriteLock); - foreach (OVERLAPPED *o, pendingWrites) { - DWORD numBytes = 0; - if (GetOverlappedResult(handle, o, &numBytes, false)) { - overlapsToDelete.append(o); - totalBytesWritten += numBytes; - } else if (GetLastError() != ERROR_IO_INCOMPLETE) { - overlapsToDelete.append(o); - QESP_WARNING()<<"CommEvent overlapped write error:" << GetLastError(); - } - } - - if (q->sender() != q && totalBytesWritten > 0) - Q_EMIT q->bytesWritten(totalBytesWritten); - - foreach (OVERLAPPED *o, overlapsToDelete) { - OVERLAPPED *toDelete = pendingWrites.takeAt(pendingWrites.indexOf(o)); - CloseHandle(toDelete->hEvent); - delete toDelete; - } - } - if (eventMask & EV_DSR) { - if (lineStatus_sys() & LS_DSR) - Q_EMIT q->dsrChanged(true); - else - Q_EMIT q->dsrChanged(false); - } - } - WaitCommEvent(handle, &eventMask, &overlap); -} - -void QextSerialPortPrivate::updatePortSettings() -{ - if (!q_ptr->isOpen() || !settingsDirtyFlags) - return; - - //fill struct : COMMCONFIG - if (settingsDirtyFlags & DFE_BaudRate) - commConfig.dcb.BaudRate = settings.BaudRate; - if (settingsDirtyFlags & DFE_Parity) { - commConfig.dcb.Parity = (BYTE)settings.Parity; - commConfig.dcb.fParity = (settings.Parity == PAR_NONE) ? FALSE : TRUE; - } - if (settingsDirtyFlags & DFE_DataBits) - commConfig.dcb.ByteSize = (BYTE)settings.DataBits; - if (settingsDirtyFlags & DFE_StopBits) { - switch (settings.StopBits) { - case STOP_1: - commConfig.dcb.StopBits = ONESTOPBIT; - break; - case STOP_1_5: - commConfig.dcb.StopBits = ONE5STOPBITS; - break; - case STOP_2: - commConfig.dcb.StopBits = TWOSTOPBITS; - break; - } - } - if (settingsDirtyFlags & DFE_Flow) { - switch(settings.FlowControl) { - /*no flow control*/ - case FLOW_OFF: - commConfig.dcb.fOutxCtsFlow = FALSE; - commConfig.dcb.fRtsControl = RTS_CONTROL_DISABLE; - commConfig.dcb.fInX = FALSE; - commConfig.dcb.fOutX = FALSE; - break; - /*software (XON/XOFF) flow control*/ - case FLOW_XONXOFF: - commConfig.dcb.fOutxCtsFlow = FALSE; - commConfig.dcb.fRtsControl = RTS_CONTROL_DISABLE; - commConfig.dcb.fInX = TRUE; - commConfig.dcb.fOutX = TRUE; - break; - /*hardware flow control*/ - case FLOW_HARDWARE: - commConfig.dcb.fOutxCtsFlow = TRUE; - commConfig.dcb.fRtsControl = RTS_CONTROL_HANDSHAKE; - commConfig.dcb.fInX = FALSE; - commConfig.dcb.fOutX = FALSE; - break; - } - } - - //fill struct : COMMTIMEOUTS - if (settingsDirtyFlags & DFE_TimeOut) { - if (queryMode != QextSerialPort::EventDriven) { - int millisec = settings.Timeout_Millisec; - if (millisec == -1) { - commTimeouts.ReadIntervalTimeout = MAXDWORD; - commTimeouts.ReadTotalTimeoutConstant = 0; - } else { - commTimeouts.ReadIntervalTimeout = millisec; - commTimeouts.ReadTotalTimeoutConstant = millisec; - } - commTimeouts.ReadTotalTimeoutMultiplier = 0; - commTimeouts.WriteTotalTimeoutMultiplier = millisec; - commTimeouts.WriteTotalTimeoutConstant = 0; - } else { - commTimeouts.ReadIntervalTimeout = MAXDWORD; - commTimeouts.ReadTotalTimeoutMultiplier = 0; - commTimeouts.ReadTotalTimeoutConstant = 0; - commTimeouts.WriteTotalTimeoutMultiplier = 0; - commTimeouts.WriteTotalTimeoutConstant = 0; - } - } - - - if (settingsDirtyFlags & DFE_Settings_Mask) - SetCommConfig(handle, &commConfig, sizeof(COMMCONFIG)); - if ((settingsDirtyFlags & DFE_TimeOut)) - SetCommTimeouts(handle, &commTimeouts); - settingsDirtyFlags = 0; -} diff --git a/sleepyhead/daily.cpp b/sleepyhead/daily.cpp index 87643dd6..2f65d0a6 100644 --- a/sleepyhead/daily.cpp +++ b/sleepyhead/daily.cpp @@ -504,10 +504,6 @@ void Daily::Link_clicked(const QUrl &url) } else if (code=="oxi") { day=PROFILE.GetDay(previous_date,MT_OXIMETER); Session *sess=day->machine->sessionlist[sid]; - if (mainwin->getOximetry()) { - mainwin->getOximetry()->openSession(sess); - mainwin->selectOximetryTab(); - } return; } else if (code=="event") { QList list=ui->treeWidget->findItems(schema::channel[sid].fullname(),Qt::MatchContains); diff --git a/sleepyhead/docs/release_notes.html b/sleepyhead/docs/release_notes.html index 493c7407..e7caddde 100644 --- a/sleepyhead/docs/release_notes.html +++ b/sleepyhead/docs/release_notes.html @@ -9,11 +9,12 @@

0.9.7 already? Yup.. otherwise I can't tell them apart, and you probably won't be able to either. Before anyone asks, yes, version numbers will go past 0.9.10 before 1.0.0 is reached.

Not a gigantic set of changes this time, but I've been busily working my way through a lot of bugs mentioned on the forums, trying to knock them down one by one. Please don't be offended if yours isn't in here yet.. I have a memory like a goldfish, plus I tried to focus on stability issues first.

A gentle reminder for anyone submitting bug reports (either via message or through sourceforges bug reporting system), please (please please) remember to state the SleepyHead version number, your computers platform and operating system type version (and service packs), and your CPAP machines type and model number. Without this information I pretty much don't have a clue what you are talking about.

-

Known Issues with this build

-There is still a problem with time splits for summary only.. +

Known Issues with this build
+There is still an issue involving time splits for late sleepers who use ResMed machines.. Sessions sorting capabilities were recently locked down, because currently this is the only way to access summary-only STR.edf data, and this


New features & bug fixes in v0.9.7
+
  • Added PRS1 SD Card automatic backup functionality
  • Stopped faulty statistics calculations where only summary data was present (For now it should show zero where calculations are impossible)
  • Added .spo2 file import support for firmware 3.7 CMS50F oximeters.
  • Some pre-v3.7 firmware CMS50 oximeter serial importer improvements
  • @@ -22,7 +23,7 @@ There is still a problem with time splits for summary only..
  • Restore display of MaskPressureHi plots when available
  • Fix wiki URL's thanks to sourceForge forcing move of SleepyHead's wiki
  • Fix ResMed importer crashes, finished rewriting stage2 multithreading code
  • -
  • Made database upgrade/purge process a bit more friendly (ResMed users Keep your backups switched on!)
  • +
  • Made database upgrade/purge process a bit more friendly (Keep your backups switched on!)
  • Fixed a SleepyHead summary error that prevented count indexes from being stored properly
  • Some other minor stuff
  • diff --git a/sleepyhead/mainwindow.cpp b/sleepyhead/mainwindow.cpp index 449e0cd8..4f994a7c 100644 --- a/sleepyhead/mainwindow.cpp +++ b/sleepyhead/mainwindow.cpp @@ -180,7 +180,6 @@ MainWindow::MainWindow(QWidget *parent) : overview = nullptr; daily = nullptr; - oximetry = nullptr; prefdialog = nullptr; m_inRecalculation = false; @@ -316,11 +315,6 @@ void MainWindow::closeEvent(QCloseEvent * event) overview->deleteLater(); } - if (oximetry) { - oximetry->close(); - oximetry->deleteLater(); - } - // Shutdown and Save the current User profile Profiles::Done(); @@ -417,11 +411,6 @@ void MainWindow::Startup() overview = new Overview(ui->tabWidget, daily->graphView()); ui->tabWidget->insertTab(2, overview, STR_TR_Overview); - if (PROFILE.oxi->oximetryEnabled()) { - oximetry = new Oximetry(ui->tabWidget, daily->graphView()); - ui->tabWidget->insertTab(3, oximetry, STR_TR_Oximetry); - } - GenerateStatistics(); ui->tabWidget->setCurrentWidget(ui->statisticsTab); @@ -1258,10 +1247,6 @@ void MainWindow::on_action_Preferences_triggered() prefdialog = nullptr; } -void MainWindow::selectOximetryTab() -{ - on_oximetryButton_clicked(); -} #include "oximeterimport.h" QDateTime datetimeDialog(QDateTime datetime, QString message); @@ -1270,61 +1255,6 @@ void MainWindow::on_oximetryButton_clicked() { OximeterImport oxiimp(this); oxiimp.exec(); - - return; - -// QDateTime current=QDateTime::currentDateTime(); -// DateTimeDialog datedlg("Oximetry Session Start Time", this); -// current = datedlg.execute(current); -// return; -// bool first = false; - -// if (!oximetry) { -// if (!PROFILE.oxi->oximetryEnabled()) { -// if (QMessageBox::question(this, STR_MessageBox_Question, -// tr("Do you have a CMS50[x] Oximeter?\nOne is required to use this section."), QMessageBox::Yes, -// QMessageBox::No) == QMessageBox::No) { return; } - -// PROFILE.oxi->setOximetryEnabled(true); -// } - -// // oximetry = new Oximetry(ui->tabWidget, daily->graphView()); -// // ui->tabWidget->insertTab(3, oximetry, STR_TR_Oximetry); -// first = true; -// } - - -// QDialog dlg(this, Qt::CustomizeWindowHint | Qt::WindowTitleHint | Qt::Dialog); - -// dlg.setWindowModality(Qt::ApplicationModal); -// dlg.setModal(true); - -// QVBoxLayout layout(&dlg); -// QLabel msg("

    "+tr("Please connect your %1 oximeter.").arg(active)+"

    "); -// QPushButton cancel(STR_MessageBox_Cancel); - -// layout.addWidget(&msg,1); -// layout.addWidget(qprogress,1); -// layout.addWidget(&cancel); - -// qprogress->setMaximum(PORTSCAN_TIMEOUT); -// qprogress->setVisible(true); - -// dlg.connect(&cancel, SIGNAL(clicked()), &dlg, SLOT(hide())); -// dlg.show(); - -// QApplication::processEvents(); - - - - // MW: Instead, how about starting a direct import? -// oximetry->serialImport(); - -// ui->tabWidget->setCurrentWidget(oximetry); - -// if (!first) { oximetry->RedrawGraphs(); } - - qstatus2->setText(STR_TR_Oximetry); } void MainWindow::CheckForUpdates() @@ -1399,10 +1329,6 @@ void MainWindow::on_actionPrint_Report_triggered() Report::PrintReport(overview->graphView(), STR_TR_Overview); } else if (ui->tabWidget->currentWidget() == daily) { Report::PrintReport(daily->graphView(), STR_TR_Daily, daily->getDate()); - } else if (ui->tabWidget->currentWidget() == oximetry) { - if (oximetry) { - Report::PrintReport(oximetry->graphView(), STR_TR_Oximetry); - } } else { QPrinter printer(QPrinter::HighResolution); #ifdef Q_WS_X11 @@ -1979,9 +1905,6 @@ void MainWindow::on_tabWidget_currentChanged(int index) } else if (widget == overview) { qstatus2->setVisible(true); overview->graphView()->selectionTime(); - } else if (widget == oximetry) { - qstatus2->setVisible(true); - oximetry->graphView()->selectionTime(); } } diff --git a/sleepyhead/mainwindow.h b/sleepyhead/mainwindow.h index af57fd2e..257bbe28 100644 --- a/sleepyhead/mainwindow.h +++ b/sleepyhead/mainwindow.h @@ -21,7 +21,6 @@ #include "version.h" #include "daily.h" #include "overview.h" -#include "oximetry.h" #include "preferencesdialog.h" extern Profile *profile; @@ -115,9 +114,6 @@ class MainWindow : public QMainWindow //! \brief Returns the Overview Tab object Overview *getOverview() { return overview; } - //! \brief Returns the Oximetry Tab object - Oximetry *getOximetry() { return oximetry; } - /*! \fn void RestartApplication(bool force_login=false); \brief Closes down SleepyHead and restarts it \param bool force_login @@ -126,9 +122,6 @@ class MainWindow : public QMainWindow */ static void RestartApplication(bool force_login = false, bool change_datafolder = false); - //! \brief Self explainitory, selects the Oximetry Tab - void selectOximetryTab(); - void JumpDaily(); void sendStatsUrl(QString msg) { on_recordsBox_linkClicked(QUrl(msg)); } @@ -329,7 +322,6 @@ private: Ui::MainWindow *ui; Daily *daily; Overview *overview; - Oximetry *oximetry; bool first_load; PreferencesDialog *prefdialog; QTime logtime; diff --git a/sleepyhead/mainwindow.ui b/sleepyhead/mainwindow.ui index 393bf88d..c20cf0ad 100644 --- a/sleepyhead/mainwindow.ui +++ b/sleepyhead/mainwindow.ui @@ -3108,7 +3108,6 @@ border-radius: 10px; - @@ -3159,6 +3158,8 @@ border-radius: 10px; + + @@ -3280,7 +3281,7 @@ border-radius: 10px; - View O&ximetry + O&ximetry Wizard F7 diff --git a/sleepyhead/oximetry.cpp b/sleepyhead/oximetry.cpp deleted file mode 100644 index 97f160cc..00000000 --- a/sleepyhead/oximetry.cpp +++ /dev/null @@ -1,2164 +0,0 @@ -/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- - * vim: set ts=8 sts=4 et sw=4 tw=99: - * - * Oximetry - * - * Copyright (c) 2011-2014 Mark Watkins - * - * This file is subject to the terms and conditions of the GNU General Public - * License. See the file COPYING in the main directory of the Linux - * distribution for more details. */ - -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#include - -#include "oximetry.h" -#include "ui_oximetry.h" -#include "common_gui.h" -#include "SleepLib/loader_plugins/cms50_loader.h" -#include "SleepLib/event.h" -#include "SleepLib/calcs.h" -#include "Graphs/gFooBar.h" -#include "Graphs/gXAxis.h" -#include "Graphs/gSummaryChart.h" -#include "Graphs/gLineChart.h" -#include "Graphs/gYAxis.h" -#include "Graphs/gLineOverlay.h" - -#define SERIAL_DEBUG 1 - - -extern QLabel *qstatus2; -#include "mainwindow.h" -extern MainWindow *mainwin; - -int lastpulse; -ZSerialOximeter::ZSerialOximeter(QObject *parent, QString oxiname, QString portname, - BaudRateType baud, FlowType flow, ParityType parity, DataBitsType databits, - StopBitsType stopbits) : - QObject(parent), - session(nullptr), pulse(nullptr), spo2(nullptr), plethy(nullptr), m_port(nullptr), - m_opened(false), - m_oxiname(oxiname), - m_portname(portname), - m_baud(baud), - m_flow(flow), - m_parity(parity), - m_databits(databits), - m_stopbits(stopbits) -{ - machine = PROFILE.GetMachine(MT_OXIMETER); - - if (!machine) { - // Create generic Serial Oximeter object.. - CMS50Loader *l = dynamic_cast(GetLoader("CMS50")); - - if (l) { - machine = l->CreateMachine(p_profile); - } - - qDebug() << "Create Oximeter device"; - } - - timer = new QTimer(this); - connect(timer, SIGNAL(timeout()), this, SLOT(Timeout())); - import_mode = false; - m_mode = SO_WAIT; -} - -ZSerialOximeter::~ZSerialOximeter() -{ - if (m_opened) { - if (m_port) { m_port->close(); } - } - - disconnect(timer, SIGNAL(timeout()), this, SLOT(Timeout())); - delete timer; -} - -void ZSerialOximeter::Timeout() -{ - qDebug() << "Timeout!"; - - if (!import_mode) { emit(liveStopped(session)); } -} - -bool ZSerialOximeter::Open(QextSerialPort::QueryMode mode) -{ - if (m_portname.isEmpty()) { - qDebug() << "Tried to open with empty portname"; - return false; - } - - qDebug() << "Opening serial port" << m_portname << "in mode" << mode; - - if (m_opened) { // Open already? - // Just close it - if (m_port) { m_port->close(); } - } - - m_portmode = mode; - m_callbacks = 0; - - m_port = new QextSerialPort(m_portname, m_portmode); - m_port->setBaudRate(m_baud); - m_port->setFlowControl(m_flow); - m_port->setParity(m_parity); - m_port->setDataBits(m_databits); - m_port->setStopBits(m_stopbits); - - if (m_port->open(QIODevice::ReadWrite) == true) { - // if (m_mode==QextSerialPort::EventDriven) - connect(m_port, SIGNAL(readyRead()), this, SLOT(ReadyRead())); - - //connect(port, SIGNAL(dsrChanged(bool)), this, SLOT(DsrChanged(bool))); - if (!(m_port->lineStatus() & LS_DSR)) { - qDebug() << "check device is turned on"; - } - - qDebug() << "listening for data on" << m_port->portName(); - return m_opened = true; - } else { - qDebug() << "device failed to open:" << m_port->errorString(); - return m_opened = false; - } -} - -void ZSerialOximeter::Close() -{ - qDebug() << "Closing serial port" << m_portname; - - if (!m_opened) { - return; - } - - m_port->flush(); - disconnect(m_port, 0, 0, 0); // SIGNAL(readyRead()), this, SLOT(ReadyRead())); - - if (m_port) { - m_port->close(); - } - - //if (m_portmode==QextSerialPort::EventDriven) - m_mode = SO_OFF; - m_opened = false; -} - -void ZSerialOximeter::setPortName(QString portname) -{ - if (m_opened) { - qDebug() << "Can't change serial PortName settings while port is open!"; - return; - } - - m_portname = portname; -} - -void ZSerialOximeter::setBaudRate(BaudRateType baud) -{ - if (m_opened) { - qDebug() << "Can't change serial BaudRate settings while port is open!"; - return; - } - - m_baud = baud; -} - -void ZSerialOximeter::setFlowControl(FlowType flow) -{ - if (m_opened) { - qDebug() << "Can't change serial FlowControl settings while port is open!"; - return; - } - - m_flow = flow; -} - -void ZSerialOximeter::setParity(ParityType parity) -{ - if (m_opened) { - qDebug() << "Can't change serial Parity settings while port is open!"; - return; - } - - m_parity = parity; -} - -void ZSerialOximeter::setDataBits(DataBitsType databits) -{ - if (m_opened) { - qDebug() << "Can't change serial DataBit settings while port is open!"; - return; - } - - m_databits = databits; -} - -void ZSerialOximeter::setStopBits(StopBitsType stopbits) -{ - if (m_opened) { - qDebug() << "Can't change serial StopBit settings while port is open!"; - return; - } - - m_stopbits = stopbits; -} - -void ZSerialOximeter::addPulse(qint64 time, EventDataType pr) -{ - //EventDataType min=0,max=0; - if (pr > 0) { - if (lastpr == 0) { - if (pulse->count() == 0) { - pulse->setFirst(time); - - if (session->eventlist[OXI_Pulse].size() <= 1) { - session->setFirst(OXI_Pulse, time); - - if (session->first() == 0) { - session->set_first(time); - } - } - - } else { - qDebug() << "Shouldn't happen in addPulse()"; - } - } - - pulse->AddEvent(time, pr); - session->setCount(OXI_Pulse, session->count(OXI_Pulse) + 1); - session->setLast(OXI_Pulse, time); - session->set_last(time); - } else { - if (lastpr != 0) { - if (pulse->count() > 0) { - pulse->AddEvent(time, lastpr); - this->compactToEvent(pulse); - session->setLast(OXI_Pulse, time); - pulse = session->AddEventList(OXI_Pulse, EVL_Event); - } - } - } - - lastpr = pr; - emit(updatePulse(pr)); -} - -void ZSerialOximeter::addSpO2(qint64 time, EventDataType o2) -{ - //EventDataType min=0,max=0; - if (o2 > 0) { - if (lasto2 == 0) { - if (spo2->count() == 0) { - spo2->setFirst(time); - - if (session->eventlist[OXI_SPO2].size() <= 1) { - session->setFirst(OXI_SPO2, time); - - if (session->first() == 0) { - session->set_first(time); - } - } - } else { - qDebug() << "Shouldn't happen in addSpO2()"; - } - } - - spo2->AddEvent(time, o2); - session->setCount(OXI_SPO2, session->count(OXI_SPO2) + 1); - session->setLast(OXI_SPO2, time); - session->set_last(time); - } else { - if (lasto2 != 0) { - if (spo2->count() > 0) { - spo2->AddEvent(time, lasto2); - this->compactToEvent(spo2); - session->setLast(OXI_SPO2, time); - spo2 = session->AddEventList(OXI_SPO2, EVL_Event); - } - } - } - - lasto2 = o2; - emit(updateSpO2(o2)); -} - -void ZSerialOximeter::addPlethy(qint64 time, EventDataType pleth) -{ - if (!plethy) { - plethy = new EventList(EVL_Event); - session->eventlist[OXI_Plethy].push_back(plethy); - session->setFirst(OXI_Plethy, lasttime); - plethy->setFirst(lasttime); - } - - plethy->AddEvent(time, pleth); - session->setCount(OXI_Plethy, plethy->count()); // update the cache - session->setMin(OXI_Plethy, plethy->Min()); - session->setMax(OXI_Plethy, plethy->Max()); - session->setLast(OXI_Plethy, time); - session->set_last(time); - plethy->setLast(time); -} -void ZSerialOximeter::compactToWaveform(EventList *el) -{ - double rate = double(el->duration()) / double(el->count()); - el->setType(EVL_Waveform); - el->setRate(rate); - el->getTime().clear(); -} -void ZSerialOximeter::compactToEvent(EventList *el) -{ - if (el->count() < 2) { return; } - - EventList nel(EVL_Waveform); - EventDataType t = 0, lastt = 0; //el->data(0); - qint64 ti = 0; //=el->time(0); - //nel.AddEvent(ti,lastt); - bool f = false; - qint64 lasttime = 0; - EventDataType min = 999, max = 0; - - for (quint32 i = 0; i < el->count(); i++) { - t = el->data(i); - ti = el->time(i); - f = false; - - if (t != 0) { - if (t != lastt) { - if (!lasttime) { - nel.setFirst(ti); - } - - nel.AddEvent(ti, t); - - if (t < min) { min = t; } - - if (t > max) { max = t; } - - lasttime = ti; - f = true; - } - } else { - if (lastt != 0) { - nel.AddEvent(ti, lastt); - lasttime = ti; - f = true; - } - } - - lastt = t; - } - - if (!f) { - if (t != 0) { - nel.AddEvent(ti, t); - lasttime = ti; - } - } - - el->setFirst(nel.first()); - el->setLast(nel.last()); - el->setMin(min); - el->setMax(max); - - el->getData().clear(); - el->getTime().clear(); - el->setCount(nel.count()); - - el->getData() = nel.getData(); - el->getTime() = nel.getTime(); -} - -void ZSerialOximeter::compactAll() -{ - if (!session) { return; } - - QHash >::iterator i; - - qint64 tminx = 0, tmaxx = 0, minx, maxx; - EventDataType min, max; - - for (i = session->eventlist.begin(); i != session->eventlist.end(); i++) { - const ChannelID &code = i.key(); - min = 999, max = 0; - minx = maxx = 0; - - for (int j = 0; j < i.value().size(); j++) { - EventList *e = i.value()[j]; - - if ((code == OXI_SPO2) || (code == OXI_Pulse)) { - compactToEvent(e); - } else if (code == OXI_Plethy) { - compactToWaveform(e); - } - - if (min > e->Min()) { - min = e->Min(); - } - - if (max < e->Max()) { - max = e->Max(); - } - - if (!minx || (minx > e->first())) { - minx = e->first(); - } - - if (!maxx || (maxx < e->last())) { - maxx = e->last(); - } - } - - if ((code == OXI_SPO2) || (code == OXI_Pulse) || (code == OXI_Plethy)) { - session->setMin(code, min); - session->setMax(code, max); - - if (minx != 0) { - session->setFirst(code, minx); - - if (!tminx || tminx > minx) { tminx = minx; } - } - - if (maxx != 0) { - session->setLast(code, maxx); - - if (!tmaxx || tmaxx < max) { tmaxx = maxx; } - } - } - } - - if (tminx > 0) { session->really_set_first(tminx); } - - if (tmaxx > 0) { session->really_set_last(tmaxx); } -} - -Session *ZSerialOximeter::createSession(QDateTime date) -{ - if (session) { - delete session; - } - - int sid = date.toTime_t(); - lasttime = qint64(sid) * 1000L; - lasto2 = lastpr = 0; - - session = new Session(machine, sid); - session->SetChanged(true); - - session->set_first(lasttime); - pulse = new EventList(EVL_Event); - spo2 = new EventList(EVL_Event); - plethy = nullptr; - session->eventlist[OXI_Pulse].push_back(pulse); - session->eventlist[OXI_SPO2].push_back(spo2); - - session->setFirst(OXI_Pulse, lasttime); - session->setFirst(OXI_SPO2, lasttime); - - pulse->setFirst(lasttime); - spo2->setFirst(lasttime); - - m_callbacks = 0; - - emit(sessionCreated(session)); - return session; -} - -bool ZSerialOximeter::startLive() -{ - import_mode = false; - killTimers(); - m_mode = SO_LIVE; - - lastpr = lasto2 = 0; - buffer.clear(); - - if (!m_opened && !Open(QextSerialPort::EventDriven)) { return false; } - - createSession(); - - return true; -} - -void ZSerialOximeter::stopLive() -{ - if (timer->isActive()) { timer->stop(); } - - m_mode = SO_WAIT; - - if (session) { - compactAll(); - calcSPO2Drop(session); - calcPulseChange(session); - } -} - -CMS50Serial::CMS50Serial(QObject *parent, QString portname = "") : - ZSerialOximeter(parent, "CMS50", portname, BAUD19200, FLOW_OFF, PAR_ODD, DATA_8, STOP_1) -{ - cms50timer = new QTimer(this); - cms50timer2 = new QTimer(this); -} - -CMS50Serial::~CMS50Serial() -{ - delete cms50timer2; - delete cms50timer; -} -void CMS50Serial::killTimers() -{ - if (cms50timer->isActive()) { - cms50timer->stop(); - } - - if (cms50timer2->isActive()) { - cms50timer2->stop(); - } -} - -void CMS50Serial::import_process() -{ - if (!session) { - qDebug() << "User pushing import too many times in a row?"; - return; - } - - mainwin->getOximetry()->graphView()->setEmptyText(tr("Processing...")); - mainwin->getOximetry()->graphView()->redraw(); - - qDebug() << "CMS50 import complete. Processing" << data.size() << "bytes"; - unsigned short a, pl, o2, lastpl = 0, lasto2 = 0; - int i = 0; - int size = data.size(); - - - EventList *pulse = (session->eventlist[OXI_Pulse][0]); - EventList *spo2 = (session->eventlist[OXI_SPO2][0]); - - // int d=abs(oxitime.secsTo(cpaptime)); - - QDateTime seltime = oxitime; - - if (!cpaptime.isNull()) { - if (QMessageBox::question(mainwin, STR_MessageBox_Question, - tr("Did you remember to start your oximeter recording at exactly the same time you started your CPAP machine?"), - QMessageBox::Yes, QMessageBox::No) == QMessageBox::No) { - if (!cms50dplus) { - // Oximeter has a clock.. Hopefully the user remembered to set their clock on the device.. - QMessageBox::information(mainwin, STR_MessageBox_Information, - tr("That's ok, I will use the time provided by your oximeter, however it will sync better next time if you start your oximeter recording at the same time your CPAP machine starts up.")+ - "\n\n"+ - STR_MessageBox_PleaseNote+": "+tr("If you haven't set your oximeter clock you will have to manually edit this time before saving this oximetry session."), - QMessageBox::Ok); - } else { - //CMS50D+, and the user didn't start at the same time.. Kludge it because they likely turned it on around about the same time anyway. - QMessageBox::information(mainwin, STR_MessageBox_Information, - tr("It looks like your oximeter doesn't provide a valid start time, I'm going to set this oximetry session starting time to the CPAP starting time anyway.")+"\n\n"+ - tr("You may have to adjust it manually if you remember the real start time before saving this session.")+"\n\n"+ - tr("(Also, did you remember to import todays CPAP data first?)"), - QMessageBox::Ok); - seltime = cpaptime; - } - } else { - // The best solution.. the user (hopefully) started both devices at the same time, so we pick the cpap sessions start time for optimal sync. - QMessageBox::information(mainwin, STR_MessageBox_Information, - tr("The most recent CPAP Session time has been selected as the start of your oximetry session.")+"\n\n"+ - tr("If you forgot to import todays CPAP data first, go and do that now, then import again from your oximeter."), - QMessageBox::Ok); - seltime = cpaptime; - } - } else { - if (cms50dplus) { - // Worst case, CMS50D+ and no CPAP data.. the time is basically set to midnight the current day. - QMessageBox::information(mainwin, STR_MessageBox_Information, - tr("No valid start time was provided for this oximeter session.")+"\n\n"+ - tr("You will likely have to adjust your oximeter sessions start time before saving."), - QMessageBox::Ok); - } else { - // No point nagging the user at all in this case.. they don't have any CPAP data loaded, so they are just using SleepyHead with the oximeter - } - } - - lasttime = seltime.toTime_t(); - session->SetSessionID(lasttime); - lasttime *= 1000; - - //spo2->setFirst(lasttime); - - EventDataType plmin = 999, plmax = 0; - EventDataType o2min = 100, o2max = 0; - int plcnt = 0, o2cnt = 0; - qint64 lastpltime = 0, lasto2time = 0; - bool first = true; - - while (i < (size - 3)) { - a = data.at(i++); // low bits are supposedly the high bits of the heart rate? not here - pl = ((data.at(i++) & 0x7f) | ((a & 1) << 7)) & 0xff; - o2 = data.at(i++) & 0x7f; - - // Faulty data..? - if (o2 < 50) { - o2 = 0; - } - - if (pl != 0) { - if (lastpl != pl) { - if (lastpl == 0 || !pulse) { - if (first) { - session->set_first(lasttime); - first = false; - } - - if (plcnt == 0) { - session->setFirst(OXI_Pulse, lasttime); - } - - if (pulse && pulse->count() == 0) { - - } else { - pulse = new EventList(EVL_Event); - session->eventlist[OXI_Pulse].push_back(pulse); - } - } - - lastpltime = lasttime; - pulse->AddEvent(lasttime, pl); - - if (pl < plmin) { plmin = pl; } - - if (pl > plmax) { plmax = pl; } - - plcnt++; - } - } else { - if (lastpl != 0) { - pulse->AddEvent(lasttime, pl); - lastpltime = lasttime; - plcnt++; - } - } - - if (o2 != 0) { - if (lasto2 != o2) { - if (lasto2 == 0 || !spo2) { - if (first) { - session->set_first(lasttime); - first = false; - } - - if (o2cnt == 0) { - session->setFirst(OXI_SPO2, lasttime); - } - - if (spo2 && spo2->count() == 0) { - } else { - spo2 = new EventList(EVL_Event); - session->eventlist[OXI_SPO2].push_back(spo2); - } - } - - lasto2time = lasttime; - spo2->AddEvent(lasttime, o2); - - if (o2 < o2min) { o2min = o2; } - - if (o2 > o2max) { o2max = o2; } - - o2cnt++; - } - } else { - if (lasto2 != 0) { - spo2->AddEvent(lasttime, o2); - lasto2time = lasttime; - o2cnt++; - } - } - - lasttime += 1000; - //emit(updateProgress(float(i)/float(size))); - - lastpl = pl; - lasto2 = o2; - } - - /*if (pulse && (lastpltime!=lasttime) && (pl!=0)) { - // lastpl==pl - pulse->AddEvent(lasttime,pl); - lastpltime=lastpltime; - plcnt++; - } - if (spo2 && (lasto2time!=lasttime) && (o2!=0)) { - spo2->AddEvent(lasttime,o2); - lasto2time=lasttime; - o2cnt++; - }*/ - qint64 rlasttime = qMax(lastpltime, lasto2time); - session->set_last(rlasttime); - session->setLast(OXI_Pulse, lastpltime); - session->setLast(OXI_SPO2, lasto2time); - session->setMin(OXI_Pulse, plmin); - session->setMax(OXI_Pulse, plmax); - session->setMin(OXI_SPO2, o2min); - session->setMax(OXI_SPO2, o2max); - session->setCount(OXI_Pulse, plcnt); - session->setCount(OXI_SPO2, o2cnt); - session->UpdateSummaries(); - emit(importComplete(session)); - disconnect(this, SIGNAL(importProcess()), this, SLOT(import_process())); -} - -void CMS50Serial::ReadyRead() -{ - if (m_mode == SO_OFF) { - return; - } - - static int lastbytesize = 0; - - QByteArray bytes; - int available = m_port->bytesAvailable(); - bytes.resize(available); - m_port->read(bytes.data(), bytes.size()); - - if (m_mode == SO_WAIT) { - killTimers(); - // Close(); - return; - } - - m_callbacks++; - - if (!import_mode) { - if (bytes.size() == 1) { // CMS50D+ transmits a single 0 when switching off from finger out.. - if (lastbytesize != 1) { - if (timer->isActive()) { - timer->stop(); - } - - // Set the Shutdown timer - timer->setSingleShot(true); - timer->setInterval(10000); - timer->start(); - } - - qDebug() << "Oximeter switched off.. wait for timeout?" << hex << bytes.at(0); - return; - } else { - - // Cancel any shutdown timer if running - if (timer->isActive()) { - timer->stop(); - } - } - } - - lastbytesize = available; - - unsigned char c, bc = import_mode ? 0xf0 : 0x80; - int i = 0; - - - while ((i < available) && (((c = (unsigned char)bytes.at(i)) & bc) != bc)) { - if (buffer.length() > 0) { - // If buffer is the start of a valid but short frame, add to it.. - buffer.append(c); - }// otherwise dump these bytes, as they are corrupt. - - ++i; - } - - // Copy the rest to the buffer. - for (; i < available; ++i) { - buffer.append(bytes.at(i)); - } - - unsigned char pulse, spo2, pwave, pbeat; - - available = buffer.length(); - - bool pkt_short = false; - i = 0; - short hour, minute; - bool updated = false; - - QString zdata = ""; - - int precbytes = received_bytes; - - while (i < available) { - c = (unsigned char)buffer.at(i); - - if ((c & 0xf0) == 0xf0) { // Record transmit trios all start with 0xf# - if ((i + 3) >= available) { - pkt_short = true; - break; - } - - if (!import_mode) { // Skip if live mode and the user bumped the upload button by mistake - i += 3; - continue; - } - - imp_callbacks++; - - if (!started_import) { - for (int j = 0; j < 3; j++) { zdata += QString().sprintf("%02X ", (unsigned char)buffer.at(i + j)); } - - qDebug() << "Recieved Rec Header:" << zdata; - zdata = ""; - } - - if (c == 0xf2) { // Is this a 3 byte header trio? (there are 3 sets of these at start of record data) - if (!started_import) { // Is this a record block header? Only need the first one - - hour = (unsigned char)buffer.at(i + 1) & 0x7f; - minute = (unsigned char)buffer.at(i + 2) & 0x7f; - - qDebug() << QString("Receiving Oximeter transmission %1:%2").arg(hour).arg(minute); - // set importing to true or whatever.. - - finished_import = false; - started_import = true; - started_reading = false; - - killTimers(); - qDebug() << "Getting ready for import"; - - // Hide the connect message box - if (mainwin->getOximetry()->connectDeviceMsgBox) { - //mainwin->getOximetry()->connectDeviceMsgBox->setText("Transfering Oximetry data from device"); - mainwin->getOximetry()->disconnect(mainwin->getOximetry()->connectDeviceMsgBox, - SIGNAL(buttonClicked(QAbstractButton *)), mainwin->getOximetry(), SLOT(cancel_CheckPorts())); - mainwin->getOximetry()->close(); - delete mainwin->getOximetry()->connectDeviceMsgBox; - mainwin->getOximetry()->connectDeviceMsgBox = nullptr; - } - - mainwin->getOximetry()->graphView()->setEmptyText(tr("Please Wait, Importing...")); - mainwin->getOximetry()->graphView()->timedRedraw(50); - - QDate oda = QDate::currentDate(); - QTime oti = QTime(hour, - minute); // Only CMS50E/F's have a realtime clock. CMS50D+ will set this to midnight - - cms50dplus = (hour == 0) - && (minute == 0); // Either a CMS50D+ or it's really midnight, set a flag anyway for later to help choose the right sync time - - // If the oximeter record time is more than the current time, then assume it was from the day before - // Or should I use split time preference instead??? Foggy Confusements.. - if (oti > QTime::currentTime()) { - oda = oda.addDays(-1); - } - - oxitime = QDateTime(oda, oti); - - // Convert it to UTC - oxitime = oxitime.toTimeSpec(Qt::UTC); - - qDebug() << "Session start (according to CMS50)" << oxitime << hex << buffer.at( - i + 1) << buffer.at(i + 2) << ":" << dec << hour << minute ; - - // As an alternative, pick the first session of the last days data.. - Day *day = PROFILE.GetDay(PROFILE.LastDay(), MT_CPAP); - - if (day) { - int ti = day->first() / 1000L; - - cpaptime = QDateTime::fromTime_t(ti); - - qDebug() << "Guessing session starting from CPAP data" << cpaptime; - } else { cpaptime = QDateTime(); } // null - - qDebug() << "Record Import:" << oxitime << cpaptime; - - cb_reset = 1; - - // CMS50D+ needs an end timer because it just stops dead after uploading - cms50timer2->singleShot(2000, this, SLOT(resetImportTimeout())); - } - - i += 3; - } else { - if (!started_import) { - // Crap.. Missed the 0xf2 headers.. - m_mode = SO_WAIT; - killTimers(); - emit(importAborted()); - mainwin->getOximetry()->graphView()->setEmptyText( - tr("Import Failed. Wait for oximeter and try again.")); - mainwin->Notify(tr("Something went wrong with reading from the Oximeter.")+"\n"+tr("Please wait for oximeter to finish tranmitting than try restarting import again."), - tr("Import Failed")); - mainwin->getOximetry()->graphView()->timedRedraw(50); - break; - } - - started_reading = true; // Sometimes errornous crap is sent after data rec header - - // Recording import - if ((i + 2) >= available) { - pkt_short = true; - break; - } - - pulse = (unsigned char)buffer.at(i + 1) & 0x7f; - spo2 = (unsigned char)buffer.at(i + 2) & 0x7f; - data.push_back(buffer.at(i)); - data.push_back(buffer.at(i + 1)); - data.push_back(buffer.at(i + 2)); - received_bytes += 3; - - i += 3; - } - - } else if ((c & 0x80) == 0x80) { - if (import_mode && started_reading) { // (Sometimes errornous bytes get transfered after header) - qDebug() << "Stopped importing due to live data"; - // We were importing, but now are done - - killTimers(); - - started_import = false; - started_reading = false; - finished_import = true; - m_mode = SO_WAIT; // Temporarily pause until complete shutdown. - - emit(importProcess()); - break; - } - - // Standard frame.. make sure theres the full 5 byte sequence - if ((i + 4) > available) { - pkt_short = true; - break; - } - - if (!import_mode) { - pwave = (unsigned char)buffer.at(i + 1); - pbeat = (unsigned char)buffer.at(i + 2); - pulse = ((unsigned char)buffer.at(i + 3) & 0x7f) | ((pbeat & 0x40) << 1); - spo2 = (unsigned char)buffer.at(i + 4) & 0x7f; - - addPlethy(lasttime, pwave); - addPulse(lasttime, pulse); - addSpO2(lasttime, spo2); - lasttime += 20; - updated = true; - } - -#if SERIAL_DEBUG - //qDebug() << "Live: " << pulse << spo2 << pwave << pbeat; -#endif - // whatever depending on mode.. - i += 5; - } else { break; } - } - - -#if SERIAL_DEBUG - - if ((import_mode) && (received_bytes > precbytes)) { - qDebug() << "Received Bytes" << received_bytes - precbytes; // << ":" << zdata; - } - -#endif - - if (i > 0) { - // Trim any processed bytes from the buffer. - buffer = buffer.mid(i); - i = 0; - // available should be < 5 or 3 here.. - } - - available = buffer.length(); - - if (available > 0) { - if ((buffer.at(0) & 0x80) != 0x80) { - buffer.clear(); - } - } - - if (updated) { - emit(dataChanged()); - } -} -void CMS50Serial::resetDevice() -{ - qDebug() << "Sending reset code to CMS50 device"; - //m_port->flush(); - static unsigned char b1[3] = {0xf6, 0xf6, 0xf6}; - - if (m_port->write((char *)b1, 3) == -1) { - qDebug() << "Couldn't write data reset bytes to CMS50"; - } -} - -void CMS50Serial::requestData() -{ - static unsigned char b1[2] = {0xf5, 0xf5}; - - qDebug() << "Sending request code to CMS50 device"; - - if (m_port->write((char *)b1, 2) == -1) { - qDebug() << "Couldn't write data request bytes to CMS50"; - } -} - -bool CMS50Serial::startImport() -{ - buffer.clear(); - data.clear(); - - killTimers(); - - m_mode = SO_IMPORT; - import_mode = true; - started_import = false; - started_reading = false; - finished_import = false; - - imptime.start(); - - imp_callbacks = 0; - m_callbacks = 0; - - cb_reset = 0; - import_fails = 0; - failcnt = 0; - - connect(this, SIGNAL(importProcess()), this, SLOT(import_process())); - - if (!m_opened && !Open(QextSerialPort::EventDriven)) { - return false; - } - - // doesn't need to happen until process.. - createSession(); - -#ifdef SERIAL_DEBUG - qDebug() << "Starting startImportTimer"; -#endif - startImportTimeout(); - - return true; -} - -void CMS50Serial::stopImport() // Silently stops import dead -{ - disconnect(this, SIGNAL(importProcess()), this, SLOT(import_process())); - - m_mode = SO_OFF; - // killTimers(); - started_import = false; - import_mode = false; - finished_import = false; -} - -void CMS50Serial::startImportTimeout() -{ - if ((m_mode == SO_WAIT) || (m_mode == SO_OFF) - || finished_import - || started_import) { - - return; - } - - // Wait until events really are jammed on the CMS50D+ before re-requesting data. - if (imp_callbacks == 0) { // Frozen, but still hasn't started? - int elapsed = imptime.elapsed() / 1000; - - if (elapsed > 30) { // Give up after ~20 seconds - m_mode = SO_WAIT; - killTimers(); - emit(importAborted()); - mainwin->getOximetry()->graphView()->setEmptyText(tr("Import Failed")); - mainwin->getOximetry()->graphView()->timedRedraw(50); - return; - } else { - QString a = tr("Set Oximeter to Upload"); - - for (int j = 0; j < elapsed; j++) { a += "."; } - - mainwin->getOximetry()->graphView()->setEmptyText(a); - mainwin->getOximetry()->graphView()->timedRedraw(50); - // Note: Newer CMS50 devices transmit from user input - requestData(); - } - - // Schedule another callback to make sure it's started - cms50timer->singleShot(1000, this, SLOT(startImportTimeout())); - } -} -void CMS50Serial::resetImportTimeout() -{ - if ((m_mode == SO_WAIT) || (finished_import)) { - return; - } - - if (imp_callbacks != cb_reset) { - // Still receiving data.. reset timer - qDebug() << "Still receiving data in resetImportTimeout()"; - - if (cms50timer2->isActive()) { - cms50timer2->stop(); - } - - cms50timer2->singleShot(2000, this, SLOT(resetImportTimeout())); - } else { - qDebug() << "Oximeter device stopped transmitting.. Transfer complete"; - - // We were importing, but now are done - if (!finished_import && (started_import && started_reading)) { - qDebug() << "Switching CMS50 back to live mode and finalizing import"; - // Turn back on live streaming so the end of capture can be dealt with - cms50timer2->stop(); - - m_port->flush(); - resetDevice(); // Send Reset to CMS50D+ - //started_import=false; - finished_import = true; - m_mode = SO_WAIT; // Temporarily pause until complete shutdown. - - emit(importProcess()); - return; - } - - qDebug() << "Should CMS50 resetImportTimeout reach here?"; - // else what??? - } - - cb_reset = imp_callbacks; -} - - -Oximetry::Oximetry(QWidget *parent, gGraphView *shared) : - QWidget(parent), - ui(new Ui::Oximetry) -{ - m_shared = shared; - ui->setupUi(this); - - port = nullptr; - portname = ""; - - oximeter = new CMS50Serial(this); - - day = new Day(oximeter->getMachine()); - - QVBoxLayout *framelayout = new QVBoxLayout; - ui->graphArea->setLayout(framelayout); - - QFrame *border = new QFrame(ui->graphArea); - - framelayout->setMargin(1); - border->setFrameShape(QFrame::StyledPanel); - framelayout->addWidget(border,1); - - - layout = new QHBoxLayout; - layout->setSpacing(0); - layout->setMargin(0); - layout->setContentsMargins(0, 0, 0, 0); - border->setLayout(layout); - border->setAutoFillBackground(false); - - GraphView = new gGraphView(ui->graphArea, m_shared); - GraphView->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); - - scrollbar = new MyScrollBar(ui->graphArea); - scrollbar->setOrientation(Qt::Vertical); - scrollbar->setSizePolicy(QSizePolicy::Maximum, QSizePolicy::Expanding); - scrollbar->setMaximumWidth(20); - - GraphView->setScrollBar(scrollbar); - layout->addWidget(GraphView, 1); - layout->addWidget(scrollbar, 0); - - layout->layout(); - - PLETHY = new gGraph(GraphView, schema::channel[OXI_Plethy].label(), - schema::channel[OXI_Plethy].units(), 120); - CONTROL = new gGraph(GraphView, tr("Control"), "", 75); - PULSE = new gGraph(GraphView, schema::channel[OXI_Pulse].label(), - schema::channel[OXI_Pulse].units(), 120); - SPO2 = new gGraph(GraphView, schema::channel[OXI_SPO2].label(), schema::channel[OXI_SPO2].units(), - 120); - foobar = new gShadowArea(); - CONTROL->AddLayer(foobar); - Layer *cl = new gLineChart(OXI_Plethy); - CONTROL->AddLayer(cl); - cl->SetDay(day); - CONTROL->setBlockZoom(true); - - for (int i = 0; i < GraphView->size(); i++) { - gGraph *g = (*GraphView)[i]; - g->AddLayer(new gXAxis(), LayerBottom, 0, gXAxis::Margin); - g->AddLayer(new gYAxis(), LayerLeft, gYAxis::Margin); - g->AddLayer(new gXGrid()); - } - - plethy = new gLineChart(OXI_Plethy, COLOR_Plethy, false, true); - plethy->SetDay(day); - - CONTROL->AddLayer(plethy); //new gLineChart(OXI_Plethysomogram)); - - - pulse = new gLineChart(OXI_Pulse, COLOR_Pulse, true); - //pulse->SetDay(day); - - spo2 = new gLineChart(OXI_SPO2, COLOR_SPO2, true); - //spo2->SetDay(day); - - PLETHY->AddLayer(plethy); - - PULSE->AddLayer(lo1 = new gLineOverlayBar(OXI_PulseChange, COLOR_PulseChange, STR_TR_PC, FT_Span)); - SPO2->AddLayer(lo2 = new gLineOverlayBar(OXI_SPO2Drop, COLOR_SPO2Drop, STR_TR_O2, FT_Span)); - PULSE->AddLayer(pulse); - SPO2->AddLayer(spo2); - PULSE->setDay(day); - SPO2->setDay(day); - - lo1->SetDay(day); - lo2->SetDay(day); - //go->SetDay(day); - - GraphView->setEmptyText(tr("No Oximetry Data")); - GraphView->redraw(); - - on_RefreshPortsButton_clicked(); - ui->RunButton->setChecked(false); - - ui->saveButton->setEnabled(false); - GraphView->LoadSettings("Oximetry"); - - QLocale locale = QLocale::system(); - QString shortformat = locale.dateFormat(QLocale::ShortFormat); - - if (!shortformat.toLower().contains("yyyy")) { - shortformat.replace("yy", "yyyy"); - } - - ui->dateEdit->setDisplayFormat(shortformat + " HH:mm:ss"); - //Qt::DayOfWeek dow=firstDayOfWeekFromLocale(); - - //ui->dateEdit->calendarWidget()->setFirstDayOfWeek(dow); - - - // Stop both calendar drop downs highlighting weekends in red - //QTextCharFormat format = ui->dateEdit->calendarWidget()->weekdayTextFormat(Qt::Saturday); - //format.setForeground(QBrush(COLOR_Text, Qt::SolidPattern)); - //ui->dateEdit->calendarWidget()->setWeekdayTextFormat(Qt::Saturday, format); - //ui->dateEdit->calendarWidget()->setWeekdayTextFormat(Qt::Sunday, format); - dont_update_date = false; -} - -Oximetry::~Oximetry() -{ - delete ui; -} - -void Oximetry::closeEvent(QCloseEvent *event) -{ - delete oximeter; - GraphView->SaveSettings("Oximetry"); - - QWidget::closeEvent(event); -} - -void Oximetry::on_RefreshPortsButton_clicked() -{ - QList ports = QextSerialEnumerator::getPorts(); - - ui->SerialPortsCombo->clear(); - int z = 0; - QString firstport; - bool current_found = false; - const QString STR_USB = "USB"; - - // Windows build mixes these up -#if defined(Q_OS_WIN32) || defined(Q_OS_MAC) -#define qesPORTNAME portName -#else -#define qesPORTNAME physName -#endif - - for (int i = 0; i < ports.size(); i++) { - if (!ports.at(i).friendName.isEmpty()) { - if (ports.at(i).friendName.toUpper().contains(STR_USB)) { - if (firstport.isEmpty()) { firstport = ports.at(i). qesPORTNAME; } - - if (!portname.isEmpty() && ports.at(i).qesPORTNAME == portname) { current_found = true; } - - ui->SerialPortsCombo->addItem(ports.at(i).qesPORTNAME); - z++; - } - } else { // Mac stuff. - if (ports.at(i).portName.toUpper().contains(STR_USB) - || ports.at(i).portName.toUpper().contains("SPO2")) { - if (firstport.isEmpty()) { firstport = ports.at(i).portName; } - - if (!portname.isEmpty() && ports.at(i).portName == portname) { current_found = true; } - - ui->SerialPortsCombo->addItem(ports.at(i).portName); - z++; - } - } - - qDebug() << "Serial Port:" << ports.at(i).qesPORTNAME << ports.at(i).friendName; - qDebug() << "port name:" << ports.at(i).portName; - qDebug() << "phys name:" << ports.at(i).physName; - qDebug() << "friendly name:" << ports.at(i).friendName; - qDebug() << "enumerator name:" << ports.at(i).enumName; - } - - if (z > 0) { - ui->RunButton->setEnabled(true); - ui->ImportButton->setEnabled(true); - - if (!current_found) { portname = firstport; } - } else { - ui->RunButton->setEnabled(false); - ui->ImportButton->setEnabled(false); - portname = ""; - } - - oximeter->setPortName(portname); -} -void Oximetry::serialImport() -{ - connect(oximeter, SIGNAL(importComplete(Session *)), this, SLOT(import_complete(Session *))); - connect(oximeter, SIGNAL(importAborted()), this, SLOT(import_aborted())); - connect(oximeter, SIGNAL(updateProgress(float)), this, SLOT(update_progress(float))); - PLETHY->setVisible(false); - day->getSessions().clear(); - GraphView->setDay(day); - //GraphView->setEmptyText(tr("Make Sure Oximeter is Ready")); - //GraphView->redraw(); - - cancel_Import = false; - QTimer::singleShot(100, this, SLOT(timeout_CheckPorts())); - connectDeviceMsgBox = new QMessageBox(QMessageBox::Information, tr("Connect Oximeter"), - tr("Please connect oximeter device"), - QMessageBox::Cancel); - connect(connectDeviceMsgBox, SIGNAL(buttonClicked(QAbstractButton *)), this, - SLOT(cancel_CheckPorts(QAbstractButton *))); - connectDeviceMsgBox->exec(); - // QApplication::processEvents(); -} - -void Oximetry::cancel_CheckPorts(QAbstractButton *) -{ - qDebug() << "cancel_CheckPorts()"; - cancel_Import = true; - - disconnect(connectDeviceMsgBox, 0, 0, - 0); //SIGNAL(buttonClicked(QAbstractButton*)),this,SLOT(cancel_CheckPorts())); - disconnect(oximeter, SIGNAL(importComplete(Session *)), this, SLOT(import_complete(Session *))); - disconnect(oximeter, SIGNAL(importAborted()), this, SLOT(import_aborted())); - disconnect(oximeter, SIGNAL(updateProgress(float)), this, SLOT(update_progress(float))); - - connectDeviceMsgBox->close(); - delete connectDeviceMsgBox; - connectDeviceMsgBox = nullptr; - - - if (oximeter->isImporting()) { - oximeter->stopImport(); - } -} - -void Oximetry::timeout_CheckPorts() -{ - if (cancel_Import) { - return; - } - - if (portname == "") { - qDebug() << "restarting timeout_CheckPorts()"; - on_RefreshPortsButton_clicked(); - - if (portname == "") { - QTimer::singleShot(1000, this, SLOT(timeout_CheckPorts())); - return; - } - } - - qDebug() << "Calling oximeter->startImport()"; - - if (!oximeter->startImport()) { - QTimer::singleShot(1000, this, SLOT(timeout_CheckPorts())); - return; - } - - qDebug() << "Success at oximeter->startImport()"; - - disconnect(connectDeviceMsgBox, SIGNAL(buttonClicked(QAbstractButton *)), this, - SLOT(cancel_CheckPorts(QAbstractButton *))); - connectDeviceMsgBox->close(); - delete connectDeviceMsgBox; - - //connectDeviceMsgBox=nullptr; - connectDeviceMsgBox = new QMessageBox(QMessageBox::Information, tr("Device Connected"), - tr("Please make sure Oximeter device is in upload mode."), - QMessageBox::Cancel); - connect(connectDeviceMsgBox, SIGNAL(buttonClicked(QAbstractButton *)), this, - SLOT(cancel_CheckPorts(QAbstractButton *))); - connectDeviceMsgBox->exec(); - //mainwin->Notify(tr("Please make sure Oximeter device is in upload mode."),tr("Device Connected")); -} - -void Oximetry::RedrawGraphs() -{ - GraphView->redraw(); -} -void Oximetry::on_SerialPortsCombo_activated(const QString &arg1) -{ - portname = arg1; - oximeter->setPortName(arg1); -} - -void Oximetry::live_stopped(Session *session) -{ - Q_UNUSED(session); - mainwin->Notify(tr("Oximetry live recording has been terminated due to timeout.")); - //qDebug () << "Live Stopped"; - on_RunButton_toggled(false); -} - -void Oximetry::on_RunButton_toggled(bool checked) -{ - if (!checked) { - oximeter->stopLive(); - ui->RunButton->setText(tr("&Start")); - ui->SerialPortsCombo->setEnabled(true); - disconnect(oximeter, SIGNAL(dataChanged()), this, SLOT(data_changed())); - disconnect(oximeter, SIGNAL(updatePulse(float)), this, SLOT(pulse_changed(float))); - disconnect(oximeter, SIGNAL(updateSpO2(float)), this, SLOT(spo2_changed(float))); - disconnect(oximeter, SIGNAL(liveStopped(Session *)), this, SLOT(live_stopped(Session *))); - ui->saveButton->setEnabled(true); - ui->ImportButton->setEnabled(true); - lo2->SetDay(day); - lo1->SetDay(day); - - if (oximeter->getSession()) { - saved_starttime = oximeter->getSession()->first(); - } - - - - //CONTROL->setVisible(true); - } else { - if (oximeter->getSession() && oximeter->getSession()->IsChanged()) { - int res=askSaveSession(); - if (res == 0) { - on_saveButton_clicked(); - return; - } else if (res == 2) { - return; - } - } // else it's already saved. - - GraphView->setEmptyText(tr("Please Wait")); - GraphView->redraw(); - - PLETHY->setVisible(true); - SPO2->setVisible(true); - PULSE->setVisible(true); - - PLETHY->setRecMinY(0); - PLETHY->setRecMaxY(128); - PULSE->setRecMinY(60); - PULSE->setRecMaxY(100); - SPO2->setRecMinY(90); - SPO2->setRecMaxY(100); - - day->getSessions().clear(); - - //QTimer::singleShot(10000,this,SLOT(oximeter_running_check())); - if (!oximeter->startLive()) { - mainwin->Notify(tr("Oximetry Error!\n\nSomething is wrong with the device connection.")); - return; - } - - ui->saveButton->setEnabled(false); - day->AddSession(oximeter->getSession()); - - firstPulseUpdate = true; - firstSPO2Update = true; - secondPulseUpdate = true; - secondSPO2Update = true; - - qint64 f = oximeter->getSession()->first(); - //day->setFirst(f); - plethy->setMinX(f); - pulse->setMinX(f); - spo2->setMinX(f); - PLETHY->SetMinX(f); - CONTROL->SetMinX(f); - PULSE->SetMinX(f); - SPO2->SetMinX(f); - - //graphView()->updateScale(); - /*PLETHY->setForceMinY(0); - PLETHY->setForceMaxY(128); - PULSE->setForceMinY(30); - PULSE->setForceMaxY(180); - SPO2->setForceMinY(50); - SPO2->setForceMaxY(100); */ - - connect(oximeter, SIGNAL(dataChanged()), this, SLOT(data_changed())); - connect(oximeter, SIGNAL(updatePulse(float)), this, SLOT(pulse_changed(float))); - connect(oximeter, SIGNAL(updateSpO2(float)), this, SLOT(spo2_changed(float))); - connect(oximeter, SIGNAL(liveStopped(Session *)), this, SLOT(live_stopped(Session *))); - - CONTROL->setVisible(false); - // connect. - ui->RunButton->setText(tr("&Stop")); - ui->SerialPortsCombo->setEnabled(false); - ui->ImportButton->setEnabled(false); - } - -} -void Oximetry::data_changed() -{ - - qint64 last = oximeter->lastTime(); - qint64 first = last - 30000L; - //day->setLast(last); - - plethy->setMinX(first); - plethy->setMaxX(last); - pulse->setMinX(first); - pulse->setMaxX(last); - spo2->setMinX(first); - spo2->setMaxX(last); - - plethy->setMinY((oximeter->Plethy())->Min()); - plethy->setMaxY((oximeter->Plethy())->Max()); - pulse->setMinY((oximeter->Pulse())->Min()); - pulse->setMaxY((oximeter->Pulse())->Max()); - spo2->setMinY((oximeter->Spo2())->Min()); - spo2->setMaxY((oximeter->Spo2())->Max()); - - PLETHY->SetMinY((oximeter->Plethy())->Min()); - PLETHY->SetMaxY((oximeter->Plethy())->Max()); - PULSE->SetMinY((oximeter->Pulse())->Min()); - PULSE->SetMaxY((oximeter->Pulse())->Max()); - SPO2->SetMinY((oximeter->Spo2())->Min()); - SPO2->SetMaxY((oximeter->Spo2())->Max()); - - /*PLETHY->MinY(); - PLETHY->MaxY(); - PULSE->MinY(); - PULSE->MaxY(); - SPO2->MinY(); - SPO2->MaxY(); */ - - PLETHY->SetMaxX(last); - PULSE->SetMaxX(last); - CONTROL->SetMaxX(last); - SPO2->SetMaxX(last); - - for (int i = 0; i < GraphView->size(); i++) { - (*GraphView)[i]->SetXBounds(first, last); - } - - { - int len = (last - first) / 1000L; - int h = len / 3600; - int m = (len / 60) % 60; - int s = (len % 60); - - if (qstatus2) { qstatus2->setText(QString().sprintf("Rec %02i:%02i:%02i", h, m, s)); } // translation fix? - } - - GraphView->updateScale(); - GraphView->timedRedraw(25); -} - - -extern QProgressBar *qprogress; -extern QLabel *qstatus; - - -void DumpBytes(int blocks, unsigned char *b, int len) -{ - QString a = QString::number(blocks, 16) + ": Bytes " + QString::number(len, 16) + ": "; - - for (int i = 0; i < len; i++) { - a.append(QString::number(b[i], 16) + " "); - } - - qDebug() << a; -} -void Oximetry::oximeter_running_check() -{ - if (!oximeter->isOpen()) { - if (oximeter->callbacks() == 0) { - qDebug() << "Not sure how oximeter_running_check gets called with a closed oximeter.. Restarting import process"; - //mainwin->Notify(tr("Oximeter Error\n\nThe device has not responded.. Make sure it's switched on2")); - on_ImportButton_clicked(); - return; - } - } - - if (oximeter->callbacks() == 0) { - mainwin->Notify(tr("Oximeter Error\n\nThe device has not responded.. Make sure it's switched on.")); - - if (oximeter->mode() == SO_IMPORT) { oximeter->stopImport(); } - - if (oximeter->mode() == SO_LIVE) { oximeter->stopLive(); } - - oximeter->destroySession(); - day->getSessions().clear(); - ui->SerialPortsCombo->setEnabled(true); - qstatus->setText(STR_TR_Ready); - ui->ImportButton->setEnabled(true); - ui->RunButton->setChecked(false); - ui->saveButton->setEnabled(false); - GraphView->setEmptyText(tr("Check Oximeter is Ready")); - GraphView->redraw(); - - } -} - -// Move this code to CMS50 Importer?? -void Oximetry::on_ImportButton_clicked() -{ - connect(oximeter, SIGNAL(importComplete(Session *)), this, SLOT(import_complete(Session *))); - connect(oximeter, SIGNAL(importAborted()), this, SLOT(import_aborted())); - connect(oximeter, SIGNAL(updateProgress(float)), this, SLOT(update_progress(float))); - - PLETHY->setVisible(false); - day->getSessions().clear(); - GraphView->setDay(day); - //GraphView->setEmptyText(tr("Make Sure Oximeter is Ready")); - //GraphView->redraw(); - - if (!oximeter->startImport()) { - mainwin->Notify(tr("Oximeter Error\n\nThe device did not respond.. Make sure it's switched on.")); - import_finished(); - // disconnect(oximeter,SIGNAL(importComplete(Session*)),this,SLOT(import_complete(Session*))); - // disconnect(oximeter,SIGNAL(importAborted()),this,SLOT(import_aborted())); - // disconnect(oximeter,SIGNAL(updateProgress(float)),this,SLOT(update_progress(float))); - //qDebug() << "Error starting oximetry serial import process"; - return; - } - - //QTimer::singleShot(1000,this,SLOT(oximeter_running_check())); - - if (qprogress) { - qprogress->setValue(0); - qprogress->setMaximum(100); - qprogress->show(); - } - - ui->ImportButton->setDisabled(true); - ui->SerialPortsCombo->setEnabled(false); - ui->RunButton->setText(tr("&Start")); - ui->RunButton->setChecked(false); -} - -void Oximetry::import_finished() -{ - if (connectDeviceMsgBox) { - connectDeviceMsgBox->close(); - disconnect(connectDeviceMsgBox, SIGNAL(buttonClicked(QAbstractButton *)), this, - SLOT(cancel_CheckPorts())); - delete connectDeviceMsgBox; - connectDeviceMsgBox = nullptr; - } - - disconnect(oximeter, SIGNAL(importComplete(Session *)), this, SLOT(import_complete(Session *))); - disconnect(oximeter, SIGNAL(importAborted()), this, SLOT(import_aborted())); - disconnect(oximeter, SIGNAL(updateProgress(float)), this, SLOT(update_progress(float))); - oximeter->disconnect(oximeter, SIGNAL(importProcess()), 0, 0); - - // oximeter->Close(); - // Hanging here.. :( - - ui->SerialPortsCombo->setEnabled(true); - qstatus->setText(STR_TR_Ready); - ui->ImportButton->setDisabled(false); - ui->saveButton->setEnabled(true); - - if (qprogress) { - qprogress->setValue(100); - qprogress->hide(); - } -} - -void Oximetry::import_aborted() -{ - day->getSessions().clear(); - //QMessageBox::warning(mainwin,tr("Oximeter Error"),tr("Please make sure your oximeter is switched on, and able to transmit data.\n(You may need to enter the oximeters Settings screen for it to be able to transmit.)"),QMessageBox::Ok); - mainwin->Notify( - tr("Please make sure your oximeter is switched on, and in the right mode to transmit data."), - tr("Oximeter Error!"), 5000); - //qDebug() << "Oximetry import failed"; - - import_finished(); -} -void Oximetry::import_complete(Session *session) -{ - qDebug() << "Oximetry import complete"; - import_finished(); - day->AddSession(oximeter->getSession()); - - if (!session) { - qDebug() << "Shouldn't happen"; - return; - } - - //calcSPO2Drop(session); - //calcPulseChange(session); - //PLETHY->setVisible(false); - - CONTROL->setVisible(false); - - saved_starttime = session->first(); - dont_update_date = true; - ui->dateEdit->setDateTime(QDateTime::fromTime_t(saved_starttime / 1000L)); - dont_update_date = false; - - - updateGraphs(); -} - -void Oximetry::pulse_changed(float p) -{ - ui->pulseLCD->display(p); - return; - - if (firstPulseUpdate) { - if (secondPulseUpdate) { - secondPulseUpdate = false; - } else { - firstPulseUpdate = false; - GraphView->updateScale(); - } - } -} - -// Only really need to do this once.. -void Oximetry::spo2_changed(float o2) -{ - ui->spo2LCD->display(o2); - return; - - if (firstSPO2Update) { - if (secondSPO2Update) { - secondSPO2Update = false; - } else { - firstSPO2Update = false; - GraphView->updateScale(); - } - } -} - -void Oximetry::on_saveButton_clicked() -{ - if (QMessageBox::question(this, tr("Keep This Recording?"), - tr("Would you like to save this oximetery session?"), - QMessageBox::Yes | QMessageBox::No) == QMessageBox::Yes) { - Session *session = oximeter->getSession(); - - // Process??? - //session->UpdateSummaries(); - PROFILE.RemoveSession(session); - Machine *m = oximeter->getMachine(); - - if (m->SessionExists(session->session())) { - m->sessionlist.erase(m->sessionlist.find(session->session())); - } - - QString path = PROFILE.Get(m->properties[STR_PROP_Path]) + QString().sprintf("%08lx", - session->session()); - QString f1 = path + ".000"; - QString f2 = path + ".001"; - QFile::remove(f1); - QFile::remove(f2); - // Forgetting to reset the session ID sucks, as it will delete sessions you don't want to delete.. - session->SetSessionID(qint64(session->first()) / 1000L); - - m->AddSession(session, p_profile); - - oximeter->getMachine()->Save(); - day->getSessions().clear(); - - mainwin->getDaily()->LoadDate(mainwin->getDaily()->getDate()); - mainwin->getOverview()->ReloadGraphs(); - GraphView->setEmptyText(tr("No Oximetry Data")); - GraphView->redraw(); - } -} -void Oximetry::update_progress(float f) -{ - if (qprogress) { - qprogress->setValue(f * 100.0); - QApplication::processEvents(); - } -} - -bool Oximetry::openSPOFile(QString filename) -{ - QFile f(filename); - - if (!f.open(QFile::ReadOnly)) { return false; } - - QByteArray data; - - data = f.readAll(); - long size = data.size(); - - int pos = ((unsigned char)data.at(1) << 8) | (unsigned char)data.at(0); - char dchr[20]; - int j = 0; - - for (int i = 0; i < 18 * 2; i += 2) { - dchr[j++] = data.at(8 + i); - } - - dchr[j] = 0; - QString dstr(dchr); - QDateTime date = QDateTime::fromString(dstr, "MM/dd/yy HH:mm:ss"); - - if (date.date().year() < 2000) { date = date.addYears(100); } - - //ui->dateEdit->setDateTime(date); - - day->getSessions().clear(); - oximeter->createSession(date); - Session *session = oximeter->getSession(); - day->AddSession(session); - session->set_first(0); - - firstPulseUpdate = true; - firstSPO2Update = true; - secondPulseUpdate = true; - secondSPO2Update = true; - - unsigned char o2, pr; - //quint16 pl; - qint64 tt = qint64(date.toTime_t()) * 1000L; - - for (int i = pos; i < size - 5;) { - o2 = (unsigned char)(data.at(i + 4)); - pr = (unsigned char)(data.at(i + 3)); - //oximeter->setLastTime(tt); - oximeter->addPulse(tt, pr); - oximeter->addSpO2(tt, o2); - //pl=(unsigned char)(data.at(i+1)); - - - //oximeter->addPlethy(tt,pl); - //pl=(unsigned char)(data.at(i+1)); - //oximeter->addPlethy(tt,pl); - //pl=(unsigned char)(data.at(i+2)); - //oximeter->addPlethy(tt,pl); - i += 5; - //data_changed(); - tt += 1000; - } - - qint64 t1 = session->first(OXI_Pulse); - qint64 t2 = session->first(OXI_SPO2); - qint64 t3 = qMin(t1, t2); - session->set_first(t3); - //day->setFirst(t3); - int zi = t3 / 1000L; - session->SetSessionID(zi); - date.fromTime_t(zi); - dont_update_date = true; - ui->dateEdit->setDateTime(date); - dont_update_date = false; - - t1 = session->last(OXI_Pulse); - t2 = session->last(OXI_SPO2); - t3 = qMax(t1, t2); - session->set_last(t3); - //day->setLast(t3); - CONTROL->setVisible(false); - - updateGraphs(); - - f.close(); - ui->saveButton->setEnabled(true); - return true; -} - -bool Oximetry::openSPORFile(QString filename) -{ - //GraphView->setEmptyText(tr("Please Wait")); - //GraphView->redraw(); - QFile f(filename); - - if (!f.open(QFile::ReadOnly)) { return false; } - - QByteArray data; - - data = f.readAll(); - long size = data.size(); - - int pos = ((unsigned char)data.at(1) << 8) | (unsigned char)data.at(0); - char dchr[20]; - int j = 0; - - for (int i = 0; i < 18 * 2; i += 2) { - dchr[j++] = data.at(8 + i); - } - - dchr[j] = 0; - QString dstr(dchr); - QDateTime date = QDateTime::fromString(dstr, "MM/dd/yy HH:mm:ss"); - - if (date.date().year() < 2000) { date = date.addYears(100); } - - day->getSessions().clear(); - oximeter->createSession(date); - Session *session = oximeter->getSession(); - day->AddSession(session); - session->set_first(0); - - firstPulseUpdate = true; - firstSPO2Update = true; - secondPulseUpdate = true; - secondSPO2Update = true; - - unsigned char o2, pr; - //quint16 pl; - qint64 tt = qint64(date.toTime_t()) * 1000L; - - for (int i = pos; i < size - 2;) { - o2 = (unsigned char)(data.at(i + 1)); - pr = (unsigned char)(data.at(i + 0)); - oximeter->addPulse(tt, pr); - oximeter->addSpO2(tt, o2); - //pl=(unsigned char)(data.at(i+1)); - i += 2; - tt += 1000; - } - - qint64 t1 = session->first(OXI_Pulse); - qint64 t2 = session->first(OXI_SPO2); - qint64 t3 = qMin(t1, t2); - session->set_first(t3); - //day->setFirst(t3); - int zi = t3 / 1000L; - session->SetSessionID(zi); - date.fromTime_t(zi); - dont_update_date = true; - ui->dateEdit->setDateTime(date); - dont_update_date = false; - - - t1 = session->last(OXI_Pulse); - t2 = session->last(OXI_SPO2); - t3 = qMax(t1, t2); - session->set_last(t3); - //day->setLast(t3); - - - //PLETHY->setVisible(false); - CONTROL->setVisible(false); - - updateGraphs(); - f.close(); - ui->saveButton->setEnabled(true); - return true; -} - -void Oximetry::on_openButton_clicked() -{ - if (oximeter->getSession() && oximeter->getSession()->IsChanged()) { - int res=askSaveSession(); - if (res == 0) { - on_saveButton_clicked(); - return; - } else if (res == 2) { - return; - } - } // else it's already saved. - - QString dir = ""; - QFileDialog fd(this, tr("Select an oximetry file"), dir, tr("Oximetry Files (*.spo *.spoR)")); - fd.setAcceptMode(QFileDialog::AcceptOpen); - fd.setFileMode(QFileDialog::ExistingFile); - - if (fd.exec() != QFileDialog::Accepted) { return; } - - QStringList filenames = fd.selectedFiles(); - - if (filenames.size() > 1) { - qDebug() << "Can only open one oximetry file at a time"; - } - - QString filename = filenames[0]; - bool r = false; - - if (filename.toLower().endsWith(".spo")) { r = openSPOFile(filename); } - else if (filename.toLower().endsWith(".spor")) { r = openSPORFile(filename); } - - if (!r) { - mainwin->Notify(tr("Couldn't open oximetry file \"") + filename + "\"."); - } - - qDebug() << "opening" << filename; -} - -void Oximetry::on_dateEdit_dateTimeChanged(const QDateTime &date) -{ - Session *session = oximeter->getSession(); - - if (!session) { - return; - } - - if (dont_update_date) { - return; - } - - qint64 first = session->first(); - //qint64 last=session->last(); - qint64 tt = qint64(date.toTime_t()) * 1000L; - qint64 offset = tt - first; - - if (offset != 0) { - session->SetChanged(true); - ui->saveButton->setEnabled(true); - } - - session->offsetSession(offset); - - updateGraphs(); -} - -int Oximetry::askSaveSession() -{ - return QMessageBox::question(this, STR_MessageBox_Question, - tr("Current oximetry session still has unsaved data in it.")+"\n\n"+ - tr("Would you like to save it first?"), - STR_MessageBox_Save, STR_MessageBox_Destroy, STR_MessageBox_Cancel, 0, 2); - -} - -void Oximetry::openSession(Session *session) -{ - if (oximeter->getSession() && oximeter->getSession()->IsChanged()) { - int res=askSaveSession(); - if (res == 0) { - on_saveButton_clicked(); - return; - } else if (res == 2) { - return; - } - } // else it's already saved. - - day->getSessions().clear(); - day->AddSession(session); - - oximeter->setSession(session); - - saved_starttime = session->first(); - oximeter->compactAll(); - - QDateTime date = QDateTime::fromTime_t(saved_starttime / 1000L); - dont_update_date = true; - ui->dateEdit->setDateTime(date); - dont_update_date = false; - updateGraphs(); - -} -void Oximetry::updateGraphs() -{ - Session *session = oximeter->getSession(); - - if (!session) { return; } - - qint64 first = session->first(); - qint64 last = session->last(); - - ui->pulseLCD->display(session->Min(OXI_Pulse)); - ui->spo2LCD->display(session->Min(OXI_SPO2)); - - pulse->setMinY(session->Min(OXI_Pulse)); - pulse->setMaxY(session->Max(OXI_Pulse)); - spo2->setMinY(session->Min(OXI_SPO2)); - spo2->setMaxY(session->Max(OXI_SPO2)); - - PULSE->setRecMinY(60); - PULSE->setRecMaxY(100); - SPO2->setRecMinY(90); - SPO2->setRecMaxY(100); - - //day->setFirst(first); - //day->setLast(last); - pulse->setMinY(session->Min(OXI_Pulse)); - pulse->setMaxY(session->Max(OXI_Pulse)); - spo2->setMinY(session->Min(OXI_SPO2)); - spo2->setMaxY(session->Max(OXI_SPO2)); - - PULSE->setRecMinY(60); - PULSE->setRecMaxY(100); - SPO2->setRecMinY(90); - SPO2->setRecMaxY(100); - - plethy->setMinX(first); - pulse->setMinX(first); - spo2->setMinX(first); - PLETHY->SetMinX(first); - CONTROL->SetMinX(first); - PULSE->SetMinX(first); - SPO2->SetMinX(first); - - plethy->setMaxX(last); - pulse->setMaxX(last); - spo2->setMaxX(last); - PLETHY->SetMaxX(last); - CONTROL->SetMaxX(last); - PULSE->SetMaxX(last); - SPO2->SetMaxX(last); - - PULSE->setDay(day); - SPO2->setDay(day); - - for (int i = 0; i < GraphView->size(); i++) { - (*GraphView)[i]->SetXBounds(first, last); - } - - { - int len = (last - first) / 1000L; - int h = len / 3600; - int m = (len / 60) % 60; - int s = (len % 60); - - if (qstatus2) { qstatus2->setText(QString().sprintf("%02i:%02i:%02i", h, m, s)); } - } - - GraphView->updateScale(); - GraphView->redraw(); -} - -void Oximetry::on_resetTimeButton_clicked() //revert to original session time -{ - if (oximeter->getSession()) { - - //qint64 tt=ui->dateEdit->dateTime().toTime_t()*1000; - //qint64 offset=saved_starttime-tt; - //oximeter->getSession()->offsetSession(offset); - dont_update_date = false; - //ui->dateEdit->setDateTime(QDateTime::fromTime_t(saved_starttime/1000L)); - ui->dateEdit->setDateTime(QDateTime::fromTime_t(saved_starttime / 1000L)); - dont_update_date = false; - } -} diff --git a/sleepyhead/oximetry.h b/sleepyhead/oximetry.h deleted file mode 100644 index 61d2be4a..00000000 --- a/sleepyhead/oximetry.h +++ /dev/null @@ -1,427 +0,0 @@ -/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- - * vim: set ts=8 sts=4 et sw=4 tw=99: - * - * Oximetry GUI Headers - * - * Copyright (c) 2011-2014 Mark Watkins - * - * This file is subject to the terms and conditions of the GNU General Public - * License. See the file COPYING in the main directory of the Linux - * distribution for more details. */ - -#ifndef OXIMETRY_H -#define OXIMETRY_H - -#include -#include -#include -#include -#include -#include - -#include - -#include "SleepLib/profiles.h" -#include "SleepLib/day.h" -#include "SleepLib/session.h" - -#include "Graphs/gLineChart.h" -#include "Graphs/gGraphView.h" - -//! \brief Oximeters current mode -enum SerialOxMode { SO_OFF, SO_IMPORT, SO_LIVE, SO_WAIT }; - -/*! \class SerialOximeter - \author Mark Watkins - \brief Base class for Serial Oximeters - */ -class ZSerialOximeter: public QObject -{ - Q_OBJECT - public: - explicit ZSerialOximeter(QObject *parent, QString oxiname, QString portname = "", - BaudRateType baud = BAUD19200, FlowType flow = FLOW_OFF, ParityType parity = PAR_ODD, - DataBitsType databits = DATA_8, StopBitsType stopbits = STOP_1); - virtual ~ZSerialOximeter(); - - virtual void setSession(Session *sess) { session = sess; } - - //! \brief Open the serial port in either EventDriven or Polling mode - virtual bool Open(QextSerialPort::QueryMode mode = QextSerialPort::EventDriven); - - //! \brief Close the serial port - virtual void Close(); - - //! \brief Virtual method for Importing the Oximeters internal recording. - virtual bool startImport() = 0; - //! \brief Virtual method to Abort importing the Oximeters internal recording. - virtual void stopImport() {} // abort, default do nothing. - - //! \brief Start Serial "Live" Recording - virtual bool startLive(); - //! \brief Stop Serial "Live" Recording - virtual void stopLive(); - - //! \brief Put the device in standard transmit mode - virtual void resetDevice() = 0; - - //! \brief Put the device in record request mode - virtual void requestData() = 0; - - //! \brief Return the current SerialOxMode, either SO_OFF, SO_IMPORT, SO_LIVE, SO_WAIT - SerialOxMode mode() { return m_mode; } - - //! \brief Trash the session object - void destroySession() { delete session; session = nullptr; } - - //! \brief Returns true if the serial port is currently open - bool isOpen() { return m_opened; } - - //! \brief Returns a count of callbacks, so a Timer can see the ports alive or dead. - int callbacks() { return m_callbacks; } - - //! \brief Returns the time of the last callback in milliseconds since epoch - qint64 lastTime() { return lasttime; } - //! \brief Sets the time of the last callback in milliseconds since epoch - void setLastTime(qint64 t) { lasttime = t; } - - //! \brief Return the current machine object - Machine *getMachine() { return machine; } - - //! \brief Create a new Session object for the specified date - Session *createSession(QDateTime date = QDateTime::currentDateTime()); - - //! \brief Returns the current session - Session *getSession() { return session; } - - //! \brief Removes the TimeCodes, converting the EventList to Waveform type - void compactToWaveform(EventList *el); - - //! \brief Packs EventList to time delta format, also pruning zeros. - static void compactToEvent(EventList *el); - - //! \brief Packs SPO2 & Pulse to Events, and Plethy to Waveform EventList types. - void compactAll(); - - //! \brief Sets the serial port device name - void setPortName(QString portname); - - //! \brief Sets the serial ports Baud Rate (eg. BAUD19200, BAUD115200) - void setBaudRate(BaudRateType baud); - - //! \brief Sets the serial ports Flow control to one of FLOW_OFF, FLOW_HARDWARE, or FLOW_XONXOFF - void setFlowControl(FlowType flow); - - //! \brief Sets the serial ports Parity to one of PAR_NONE, PAR_ODD, PAR_EVEN, PAR_MARK (WINDOWS ONLY), PAR_SPACE - void setParity(ParityType parity); - - //! \brief Sets the serial ports Data Bits to either DATA_5, DATA_6, DATA_7, or DATA_8 - void setDataBits(DataBitsType databits); - - //! \brief Sets the serial ports Stop Bits to either STOP_1, STOP_1_5 (WINDOWS ONLY) or STOP_2 - void setStopBits(StopBitsType stopbits); - - //! \brief Returns the serial port device name - QString portName() { return m_portname; } - - //! \brief Returns the serial ports baud rate - BaudRateType baudRate() { return m_baud; } - //! \brief Returns the serial ports flow control setting - FlowType flowControl() { return m_flow; } - - //! \brief Returns the serial ports parity setting - ParityType parity() { return m_parity; } - - //! \brief Returns the serial ports data bits setting - DataBitsType dataBits() { return m_databits; } - - //! \brief Returns the serial ports stop bits setting - StopBitsType stopBits() { return m_stopbits; } - - bool isImporting() { return import_mode; } - - EventList *Pulse() { return pulse; } - EventList *Spo2() { return spo2; } - EventList *Plethy() { return plethy; } - virtual void addPulse(qint64 time, EventDataType pr); - virtual void addSpO2(qint64 time, EventDataType o2); - virtual void addPlethy(qint64 time, EventDataType pleth); - virtual void killTimers() = 0; - - signals: - void sessionCreated(Session *); - void dataChanged(); - - //! \brief This signal is called after import completion, to parse the event data. - void importProcess(); - - //! \brief importProcess emits this signal after completion. - void importComplete(Session *); - - //! \brief emitted when something goes wrong during import - void importAborted(); - - //! \brief emitted to allow for UI updates to the progress bar - void updateProgress(float f); // between 0 and 1. - - //! \brief emitted when live mode stops recording, passing the current Session - void liveStopped(Session *); - - - void updatePulse(float p); - void updateSpO2(float p); - - protected slots: - //! \brief Override this to process the serial import as it's received - virtual void ReadyRead() = 0; - - //! \brief Override this to parse the read import data - virtual void import_process() = 0; - - //! \brief This slot gets called when the serial port Times out - virtual void Timeout(); - - //! \brief Override this to start the Import Timeout - virtual void startImportTimeout() = 0; - virtual void resetImportTimeout() = 0; - - protected: - - //virtual void addEvents(EventDataType pr, EventDataType o2, EventDataType pleth=-1000000); - - //! \brief Pointer to current session object - Session *session; - - EventList *pulse; - EventList *spo2; - EventList *plethy; - - //! \brief Holds the serial port object - QextSerialPort *m_port; - - SerialOxMode m_mode; - bool m_opened; - QString m_oxiname; - QString m_portname; - BaudRateType m_baud; - FlowType m_flow; - ParityType m_parity; - DataBitsType m_databits; - StopBitsType m_stopbits; - QextSerialPort::QueryMode m_portmode; - Machine *machine; - - qint64 lasttime; - bool import_mode; - - int m_callbacks, cb_start, cb_reset; - bool done_import; - bool started_import, started_reading, finished_import; - QTimer *timer; - EventDataType lasto2, lastpr; - - QByteArray buffer; -}; - -/*! \class CMS50Serial - \author Mark Watkins - \brief Serial Import & Live module - */ -class CMS50Serial: public ZSerialOximeter -{ - Q_OBJECT - public: - explicit CMS50Serial(QObject *parent, QString portname); - virtual ~CMS50Serial(); - - //! \brief Start the serial parts of Import mode. - virtual bool startImport(); - - //! \brief Stop the serial parts of Import mode. - virtual void stopImport(); - - //! \brief Sends the 0xf6,0xf6,0xf6 data string to the serial port to start live mode again - virtual void resetDevice(); - - //! \brief Sends the 0xf5, 0xf5 data string to request devices serial recording - virtual void requestData(); - - //! \brief Kill any CMS50 specific timers (used internally) - virtual void killTimers(); - - protected: - //! \brief CMS50 Time-out detection - virtual void startImportTimeout(); - virtual void resetImportTimeout(); - - - //! \brief Called on completion of data import, to convert bytearray into event data - virtual void import_process(); - - //! \brief Serial callback to process live view & store import data - virtual void ReadyRead(); - bool waitf6; - short cntf6; - short failcnt; - - QByteArray data; - QByteArray buffer; - - QDateTime oxitime, cpaptime; - - bool cms50dplus; - int datasize; - - int received_bytes; - int import_fails; - - int imp_callbacks; - - QTime imptime; - - EventDataType plmin, plmax; - EventDataType o2min, o2max; - int plcnt, o2cnt; - qint64 lastpltime, lasto2time; - short lastpl, lasto2; - bool first; - QTimer *cms50timer, *cms50timer2; -}; - -namespace Ui { -class Oximetry; -} - -enum PORTMODE { PM_LIVE, PM_RECORDING }; -const int max_data_points = 1000000; - -/*! \class Oximetry - \author Mark Watkins - \brief Oximetry view for working with Pulse Oximetry data and devices - */ -class Oximetry : public QWidget -{ - Q_OBJECT - - public: - explicit Oximetry(QWidget *parent, gGraphView *shared = nullptr); - ~Oximetry(); - void closeEvent(QCloseEvent *); - - //! \brief Calls updateGL to redraw the graphs - void RedrawGraphs(); - - //! \brief Returns the gGraphView object containing Oximetry graphs - gGraphView *graphView() { return GraphView; } - - //! \brief Loads and displays a session containing oximetry data into into the Oximetry module - void openSession(Session *session); - - //! \brief Initiate an automated serial import - void serialImport(); - QMessageBox *connectDeviceMsgBox; - - private slots: - //! \brief Scans the list of serial ports and detects any oximetry devices - void on_RefreshPortsButton_clicked(); - - //! \brief Start or Stop live view mode - void on_RunButton_toggled(bool checked); // Live mode button - - //! \brief This slot gets called when a new serial port is selected from the drop down - void on_SerialPortsCombo_activated(const QString &arg1); - - //! \brief Start the Serial import process from the devices internal recordings - void on_ImportButton_clicked(); - - //! \brief Asks to save oximetry session into SleepLib database - void on_saveButton_clicked(); - - //! \brief Data has been changed, so it sets all the bits for live graph display - void data_changed(); - - //! \brief Updates the Pulse Rate LCD widget when the live pulse changes - void pulse_changed(float p); - - //! \brief Updates the SpO2 LCD widget when the live spO2 changes - void spo2_changed(float o2); - - //! \brief Updates the progress bar during import - void update_progress(float f); - - //! \brief Import failed, so cleanup. - void import_aborted(); - - //! \brief Import completed, so get ready to display graphs - void import_complete(Session *session); - - //! \brief Callback to make sure the oximeter is running - void oximeter_running_check(); - - //! \brief Callback after liveView mode is stopped - void live_stopped(Session *session); - - //! \brief Open button was clicked, so select and load .spo/.spoR data files - void on_openButton_clicked(); - - //! \brief The datetime editor changed, so move the session data accordingly. - void on_dateEdit_dateTimeChanged(const QDateTime &date); - - //! \brief Reset the datetime to what was set when first loaded - void on_resetTimeButton_clicked(); - - void timeout_CheckPorts(); - void cancel_CheckPorts(QAbstractButton *); - - private: - //! \brief Imports a .spo file - bool openSPOFile(QString filename); - //! \brief Imports a .spoR file (from SPO2Review software in windows) - bool openSPORFile(QString filename); - - //! \brief Clean up after import process, whether successful or not - void import_finished(); - - //! \brief update the graphs to show the session information - void updateGraphs(); - Ui::Oximetry *ui; - - bool cancel_Import; - gGraphView *GraphView; - MyScrollBar *scrollbar; - QHBoxLayout *layout; - - gLineChart *pulse, *spo2, *plethy; - Layer *lo1, *lo2; - gGraph *PULSE, *SPO2, *PLETHY, *CONTROL; - - //! \brief Contains a list of gLineCharts that display Pulse, Plethy & SPO2 data - QVector Data; - - QextSerialPort *port; - QString portname; - PORTMODE portmode; - double lasttime, starttime; - int lastpulse, lastspo2; - - Day *day; - //Session * session; - //EventList * ev_pulse; - //EventList * ev_spo2; - //EventList * ev_plethy; - Layer *foobar; - gGraphView *m_shared; - - ZSerialOximeter *oximeter; - qint64 saved_starttime; - bool firstSPO2Update; - bool firstPulseUpdate; - bool secondPulseUpdate; - bool secondSPO2Update; - bool dont_update_date; - - int askSaveSession(); - -}; - -#endif // OXIMETRY_H diff --git a/sleepyhead/preferencesdialog.cpp b/sleepyhead/preferencesdialog.cpp index fe095c95..4802e5e1 100644 --- a/sleepyhead/preferencesdialog.cpp +++ b/sleepyhead/preferencesdialog.cpp @@ -228,7 +228,6 @@ PreferencesDialog::PreferencesDialog(QWidget *parent, Profile *_profile) : ui->overlayFlagsCombo->setCurrentIndex(profile->appearance->overlayType()); ui->overviewLinecharts->setCurrentIndex(profile->appearance->overviewLinechartMode()); - ui->oximetryGroupBox->setChecked(profile->oxi->oximetryEnabled()); ui->oximetrySync->setChecked(profile->oxi->syncOximetry()); int ot = ui->oximetryType->findText(profile->oxi->oximeterType(), Qt::MatchExactly); @@ -442,7 +441,6 @@ bool PreferencesDialog::Save() profile->cpap->setLeakMode(ui->leakModeCombo->currentIndex()); profile->cpap->setMaskType((MaskType)ui->maskTypeCombo->currentIndex()); - profile->oxi->setOximetryEnabled(ui->oximetryGroupBox->isChecked()); profile->oxi->setSyncOximetry(ui->oximetrySync->isChecked()); int oxigrp = ui->oximetrySync->isChecked() ? 0 : 1; gGraphView *gv = mainwin->getDaily()->graphView(); @@ -642,10 +640,6 @@ void PreferencesDialog::graphModel_changed(QStandardItem *item) gv = mainwin->getOverview()->graphView(); break; - case 2: - gv = mainwin->getOximetry()->graphView(); - break; - default: ; } @@ -781,34 +775,6 @@ void PreferencesDialog::resetGraphModel() overview->insertRow(i, items); } - if (mainwin->getOximetry()) { - QStandardItem *oximetry = new QStandardItem(tr("Oximetry Graphs")); - graphModel->appendRow(oximetry); - oximetry->setEditable(false); - gv = mainwin->getOximetry()->graphView(); - - for (int i = 0; i < gv->size(); i++) { - QList items; - QStandardItem *it = new QStandardItem((*gv)[i]->title()); - it->setCheckable(true); - it->setCheckState((*gv)[i]->visible() ? Qt::Checked : Qt::Unchecked); - it->setEditable(false); - it->setData(2, Qt::UserRole + 1); - it->setData(i, Qt::UserRole + 2); - items.push_back(it); - - it = new QStandardItem(QString::number((*gv)[i]->rec_miny, 'f', 1)); - it->setEditable(true); - items.push_back(it); - - it = new QStandardItem(QString::number((*gv)[i]->rec_maxy, 'f', 1)); - it->setEditable(true); - items.push_back(it); - - oximetry->insertRow(i, items); - } - } - connect(graphModel, SIGNAL(itemChanged(QStandardItem *)), this, SLOT(graphModel_changed(QStandardItem *))); @@ -832,8 +798,6 @@ void PreferencesDialog::on_resetGraphButton_clicked() gGraphView *views[3] = {0}; views[0] = mainwin->getDaily()->graphView(); views[1] = mainwin->getOverview()->graphView(); - if (mainwin->getOximetry()) - views[2] = mainwin->getOximetry()->graphView(); // Iterate over all graph containers. for (unsigned j = 0; j < 3; j++) { diff --git a/sleepyhead/preferencesdialog.ui b/sleepyhead/preferencesdialog.ui index a6d84f8d..2f29891c 100644 --- a/sleepyhead/preferencesdialog.ui +++ b/sleepyhead/preferencesdialog.ui @@ -51,7 +51,7 @@ - 0 + 4 @@ -1266,10 +1266,10 @@ Defaults to 60 minutes.. Highly recommend it's left at this value. - Use Oximetry + Oximetery Settings - true + false @@ -1310,7 +1310,17 @@ Defaults to 60 minutes.. Highly recommend it's left at this value. - Contec CMS50 + Contec CMS50D+ + + + + + Contec CMS50E/F + + + + + Contec CMS50F v3.7 diff --git a/sleepyhead/sleepyhead.pro b/sleepyhead/sleepyhead.pro index d7ca2889..c8870090 100644 --- a/sleepyhead/sleepyhead.pro +++ b/sleepyhead/sleepyhead.pro @@ -102,7 +102,6 @@ SOURCES += \ main.cpp \ mainwindow.cpp \ newprofile.cpp \ - oximetry.cpp \ overview.cpp \ preferencesdialog.cpp \ profileselect.cpp \ @@ -156,7 +155,6 @@ HEADERS += \ exportcsv.h \ mainwindow.h \ newprofile.h \ - oximetry.h \ overview.h \ preferencesdialog.h \ profileselect.h \ @@ -269,11 +267,6 @@ mac { bundlelibs = $$cat($$PWD/../Bundle3rdParty) -#QExtSerialPort will be replaced soon with Qt5's QSerialPort -include($$PWD/../3rdparty/qextserialport/src/qextserialport.pri) -INCLUDEPATH += $$PWD/../3rdparty/qextserialport/src -DEPENDPATH += $$PWD/../3rdparty/qextserialport/src - contains(bundlelibs, true) { include(../3rdparty/quazip/quazip/quazip.pri) INCLUDEPATH += $$PWD/../3rdparty/quazip