source: trunk/src/corelib/io/qwindowspipewriter.cpp@ 25

Last change on this file since 25 was 2, checked in by Dmitry A. Kuminov, 16 years ago

Initially imported qt-all-opensource-src-4.5.1 from Trolltech.

File size: 5.2 KB
Line 
1/****************************************************************************
2**
3** Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies).
4** Contact: Qt Software Information ([email protected])
5**
6** This file is part of the QtCore module of the Qt Toolkit.
7**
8** $QT_BEGIN_LICENSE:LGPL$
9** Commercial Usage
10** Licensees holding valid Qt Commercial licenses may use this file in
11** accordance with the Qt Commercial License Agreement provided with the
12** Software or, alternatively, in accordance with the terms contained in
13** a written agreement between you and Nokia.
14**
15** GNU Lesser General Public License Usage
16** Alternatively, this file may be used under the terms of the GNU Lesser
17** General Public License version 2.1 as published by the Free Software
18** Foundation and appearing in the file LICENSE.LGPL included in the
19** packaging of this file. Please review the following information to
20** ensure the GNU Lesser General Public License version 2.1 requirements
21** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
22**
23** In addition, as a special exception, Nokia gives you certain
24** additional rights. These rights are described in the Nokia Qt LGPL
25** Exception version 1.0, included in the file LGPL_EXCEPTION.txt in this
26** package.
27**
28** GNU General Public License Usage
29** Alternatively, this file may be used under the terms of the GNU
30** General Public License version 3.0 as published by the Free Software
31** Foundation and appearing in the file LICENSE.GPL included in the
32** packaging of this file. Please review the following information to
33** ensure the GNU General Public License version 3.0 requirements will be
34** met: http://www.gnu.org/copyleft/gpl.html.
35**
36** If you are unsure which license is appropriate for your use, please
37** contact the sales department at [email protected].
38** $QT_END_LICENSE$
39**
40****************************************************************************/
41
42#include "qwindowspipewriter_p.h"
43
44QT_BEGIN_NAMESPACE
45
46#ifndef QT_NO_THREAD
47
48QWindowsPipeWriter::QWindowsPipeWriter(HANDLE pipe, QObject * parent)
49 : QThread(parent),
50 writePipe(INVALID_HANDLE_VALUE),
51 quitNow(false),
52 hasWritten(false)
53{
54#if !defined(Q_OS_WINCE) || (_WIN32_WCE >= 0x600)
55 DuplicateHandle(GetCurrentProcess(), pipe, GetCurrentProcess(),
56 &writePipe, 0, FALSE, DUPLICATE_SAME_ACCESS);
57#else
58 Q_UNUSED(pipe);
59 writePipe = GetCurrentProcess();
60#endif
61}
62
63QWindowsPipeWriter::~QWindowsPipeWriter()
64{
65 lock.lock();
66 quitNow = true;
67 waitCondition.wakeOne();
68 lock.unlock();
69 if (!wait(100))
70 terminate();
71#if !defined(Q_OS_WINCE) || (_WIN32_WCE >= 0x600)
72 CloseHandle(writePipe);
73#endif
74}
75
76bool QWindowsPipeWriter::waitForWrite(int msecs)
77{
78 QMutexLocker locker(&lock);
79 bool hadWritten = hasWritten;
80 hasWritten = false;
81 if (hadWritten)
82 return true;
83 if (!waitCondition.wait(&lock, msecs))
84 return false;
85 hadWritten = hasWritten;
86 hasWritten = false;
87 return hadWritten;
88}
89
90qint64 QWindowsPipeWriter::write(const char *ptr, qint64 maxlen)
91{
92 if (!isRunning())
93 return -1;
94
95 QMutexLocker locker(&lock);
96 data.append(QByteArray(ptr, maxlen));
97 waitCondition.wakeOne();
98 return maxlen;
99}
100
101void QWindowsPipeWriter::run()
102{
103 OVERLAPPED overl = {0, 0, 0, 0, NULL};
104 overl.hEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
105 forever {
106 lock.lock();
107 while(data.isEmpty() && (!quitNow)) {
108 waitCondition.wakeOne();
109 waitCondition.wait(&lock);
110 }
111
112 if (quitNow) {
113 lock.unlock();
114 quitNow = false;
115 break;
116 }
117
118 QByteArray copy = data;
119
120 lock.unlock();
121
122 const char *ptrData = copy.data();
123 qint64 maxlen = copy.size();
124 qint64 totalWritten = 0;
125 overl.Offset = 0;
126 overl.OffsetHigh = 0;
127 while ((!quitNow) && totalWritten < maxlen) {
128 DWORD written = 0;
129 // Write 2k at a time to prevent flooding the pipe. If you
130 // write too much (4k-8k), the pipe can close
131 // unexpectedly.
132 if (!WriteFile(writePipe, ptrData + totalWritten,
133 qMin<int>(2048, maxlen - totalWritten), &written, &overl)) {
134 if (GetLastError() == 0xE8/*NT_STATUS_INVALID_USER_BUFFER*/) {
135 // give the os a rest
136 msleep(100);
137 continue;
138 }
139#ifndef Q_OS_WINCE
140 if (GetLastError() == ERROR_IO_PENDING) {
141 if (!GetOverlappedResult(writePipe, &overl, &written, TRUE)) {
142 CloseHandle(overl.hEvent);
143 return;
144 }
145 } else {
146 CloseHandle(overl.hEvent);
147 return;
148 }
149#else
150 return;
151#endif
152 }
153 totalWritten += written;
154#if defined QPIPEWRITER_DEBUG
155 qDebug("QWindowsPipeWriter::run() wrote %d %d/%d bytes",
156 written, int(totalWritten), int(maxlen));
157#endif
158 lock.lock();
159 data.remove(0, written);
160 hasWritten = true;
161 lock.unlock();
162 }
163 emit canWrite();
164 }
165 CloseHandle(overl.hEvent);
166}
167
168#endif //QT_NO_THREAD
169
170QT_END_NAMESPACE
Note: See TracBrowser for help on using the repository browser.