blob: 86e946fc0d2d6c52959c6ea26ba107ce0d50fe5c [file] [log] [blame]
henrike@webrtc.org0e118e72013-07-10 00:45:36 +00001/*
2 * libjingle
3 * Copyright 2004--2011, Google Inc.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions are met:
7 *
8 * 1. Redistributions of source code must retain the above copyright notice,
9 * this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright notice,
11 * this list of conditions and the following disclaimer in the documentation
12 * and/or other materials provided with the distribution.
13 * 3. The name of the author may not be used to endorse or promote products
14 * derived from this software without specific prior written permission.
15 *
16 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
17 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
18 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
19 * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
20 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
21 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
22 * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
23 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
24 * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
25 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26 */
27
28#ifndef TALK_BASE_TESTUTILS_H__
29#define TALK_BASE_TESTUTILS_H__
30
31// Utilities for testing talk_base infrastructure in unittests
32
mallinath@webrtc.org55280702013-09-27 23:04:10 +000033#ifdef LINUX
34#include <X11/Xlib.h>
mallinath@webrtc.orgb881d272014-02-03 16:57:16 +000035#include <X11/extensions/Xrandr.h>
36
mallinath@webrtc.org55280702013-09-27 23:04:10 +000037// X defines a few macros that stomp on types that gunit.h uses.
38#undef None
39#undef Bool
40#endif
41
henrike@webrtc.org0e118e72013-07-10 00:45:36 +000042#include <map>
43#include <vector>
44#include "talk/base/asyncsocket.h"
45#include "talk/base/common.h"
46#include "talk/base/gunit.h"
47#include "talk/base/nethelpers.h"
buildbot@webrtc.orgaf42ac52014-05-14 21:35:19 +000048#include "talk/base/pathutils.h"
henrike@webrtc.org0e118e72013-07-10 00:45:36 +000049#include "talk/base/stream.h"
50#include "talk/base/stringencode.h"
51#include "talk/base/stringutils.h"
52#include "talk/base/thread.h"
53
54namespace testing {
55
56using namespace talk_base;
57
58///////////////////////////////////////////////////////////////////////////////
59// StreamSink - Monitor asynchronously signalled events from StreamInterface
60// or AsyncSocket (which should probably be a StreamInterface.
61///////////////////////////////////////////////////////////////////////////////
62
63// Note: Any event that is an error is treaded as SSE_ERROR instead of that
64// event.
65
66enum StreamSinkEvent {
67 SSE_OPEN = SE_OPEN,
68 SSE_READ = SE_READ,
69 SSE_WRITE = SE_WRITE,
70 SSE_CLOSE = SE_CLOSE,
71 SSE_ERROR = 16
72};
73
74class StreamSink : public sigslot::has_slots<> {
75 public:
76 void Monitor(StreamInterface* stream) {
77 stream->SignalEvent.connect(this, &StreamSink::OnEvent);
78 events_.erase(stream);
79 }
80 void Unmonitor(StreamInterface* stream) {
81 stream->SignalEvent.disconnect(this);
82 // In case you forgot to unmonitor a previous object with this address
83 events_.erase(stream);
84 }
85 bool Check(StreamInterface* stream, StreamSinkEvent event, bool reset = true) {
86 return DoCheck(stream, event, reset);
87 }
88 int Events(StreamInterface* stream, bool reset = true) {
89 return DoEvents(stream, reset);
90 }
91
92 void Monitor(AsyncSocket* socket) {
93 socket->SignalConnectEvent.connect(this, &StreamSink::OnConnectEvent);
94 socket->SignalReadEvent.connect(this, &StreamSink::OnReadEvent);
95 socket->SignalWriteEvent.connect(this, &StreamSink::OnWriteEvent);
96 socket->SignalCloseEvent.connect(this, &StreamSink::OnCloseEvent);
97 // In case you forgot to unmonitor a previous object with this address
98 events_.erase(socket);
99 }
100 void Unmonitor(AsyncSocket* socket) {
101 socket->SignalConnectEvent.disconnect(this);
102 socket->SignalReadEvent.disconnect(this);
103 socket->SignalWriteEvent.disconnect(this);
104 socket->SignalCloseEvent.disconnect(this);
105 events_.erase(socket);
106 }
107 bool Check(AsyncSocket* socket, StreamSinkEvent event, bool reset = true) {
108 return DoCheck(socket, event, reset);
109 }
110 int Events(AsyncSocket* socket, bool reset = true) {
111 return DoEvents(socket, reset);
112 }
113
114 private:
115 typedef std::map<void*,int> EventMap;
116
117 void OnEvent(StreamInterface* stream, int events, int error) {
118 if (error) {
119 events = SSE_ERROR;
120 }
121 AddEvents(stream, events);
122 }
123 void OnConnectEvent(AsyncSocket* socket) {
124 AddEvents(socket, SSE_OPEN);
125 }
126 void OnReadEvent(AsyncSocket* socket) {
127 AddEvents(socket, SSE_READ);
128 }
129 void OnWriteEvent(AsyncSocket* socket) {
130 AddEvents(socket, SSE_WRITE);
131 }
132 void OnCloseEvent(AsyncSocket* socket, int error) {
133 AddEvents(socket, (0 == error) ? SSE_CLOSE : SSE_ERROR);
134 }
135
136 void AddEvents(void* obj, int events) {
137 EventMap::iterator it = events_.find(obj);
138 if (events_.end() == it) {
139 events_.insert(EventMap::value_type(obj, events));
140 } else {
141 it->second |= events;
142 }
143 }
144 bool DoCheck(void* obj, StreamSinkEvent event, bool reset) {
145 EventMap::iterator it = events_.find(obj);
146 if ((events_.end() == it) || (0 == (it->second & event))) {
147 return false;
148 }
149 if (reset) {
150 it->second &= ~event;
151 }
152 return true;
153 }
154 int DoEvents(void* obj, bool reset) {
155 EventMap::iterator it = events_.find(obj);
156 if (events_.end() == it)
157 return 0;
158 int events = it->second;
159 if (reset) {
160 it->second = 0;
161 }
162 return events;
163 }
164
165 EventMap events_;
166};
167
168///////////////////////////////////////////////////////////////////////////////
169// StreamSource - Implements stream interface and simulates asynchronous
170// events on the stream, without a network. Also buffers written data.
171///////////////////////////////////////////////////////////////////////////////
172
173class StreamSource : public StreamInterface {
174public:
175 StreamSource() {
176 Clear();
177 }
178
179 void Clear() {
180 readable_data_.clear();
181 written_data_.clear();
182 state_ = SS_CLOSED;
183 read_block_ = 0;
184 write_block_ = SIZE_UNKNOWN;
185 }
186 void QueueString(const char* data) {
187 QueueData(data, strlen(data));
188 }
189 void QueueStringF(const char* format, ...) {
190 va_list args;
191 va_start(args, format);
192 char buffer[1024];
193 size_t len = vsprintfn(buffer, sizeof(buffer), format, args);
194 ASSERT(len < sizeof(buffer) - 1);
195 va_end(args);
196 QueueData(buffer, len);
197 }
198 void QueueData(const char* data, size_t len) {
199 readable_data_.insert(readable_data_.end(), data, data + len);
200 if ((SS_OPEN == state_) && (readable_data_.size() == len)) {
201 SignalEvent(this, SE_READ, 0);
202 }
203 }
204 std::string ReadData() {
205 std::string data;
206 // avoid accessing written_data_[0] if it is undefined
207 if (written_data_.size() > 0) {
208 data.insert(0, &written_data_[0], written_data_.size());
209 }
210 written_data_.clear();
211 return data;
212 }
213 void SetState(StreamState state) {
214 int events = 0;
215 if ((SS_OPENING == state_) && (SS_OPEN == state)) {
216 events |= SE_OPEN;
217 if (!readable_data_.empty()) {
218 events |= SE_READ;
219 }
220 } else if ((SS_CLOSED != state_) && (SS_CLOSED == state)) {
221 events |= SE_CLOSE;
222 }
223 state_ = state;
224 if (events) {
225 SignalEvent(this, events, 0);
226 }
227 }
228 // Will cause Read to block when there are pos bytes in the read queue.
229 void SetReadBlock(size_t pos) { read_block_ = pos; }
230 // Will cause Write to block when there are pos bytes in the write queue.
231 void SetWriteBlock(size_t pos) { write_block_ = pos; }
232
233 virtual StreamState GetState() const { return state_; }
234 virtual StreamResult Read(void* buffer, size_t buffer_len,
235 size_t* read, int* error) {
236 if (SS_CLOSED == state_) {
237 if (error) *error = -1;
238 return SR_ERROR;
239 }
240 if ((SS_OPENING == state_) || (readable_data_.size() <= read_block_)) {
241 return SR_BLOCK;
242 }
243 size_t count = _min(buffer_len, readable_data_.size() - read_block_);
244 memcpy(buffer, &readable_data_[0], count);
245 size_t new_size = readable_data_.size() - count;
246 // Avoid undefined access beyond the last element of the vector.
247 // This only happens when new_size is 0.
248 if (count < readable_data_.size()) {
249 memmove(&readable_data_[0], &readable_data_[count], new_size);
250 }
251 readable_data_.resize(new_size);
252 if (read) *read = count;
253 return SR_SUCCESS;
254 }
255 virtual StreamResult Write(const void* data, size_t data_len,
256 size_t* written, int* error) {
257 if (SS_CLOSED == state_) {
258 if (error) *error = -1;
259 return SR_ERROR;
260 }
261 if (SS_OPENING == state_) {
262 return SR_BLOCK;
263 }
264 if (SIZE_UNKNOWN != write_block_) {
265 if (written_data_.size() >= write_block_) {
266 return SR_BLOCK;
267 }
268 if (data_len > (write_block_ - written_data_.size())) {
269 data_len = write_block_ - written_data_.size();
270 }
271 }
272 if (written) *written = data_len;
273 const char* cdata = static_cast<const char*>(data);
274 written_data_.insert(written_data_.end(), cdata, cdata + data_len);
275 return SR_SUCCESS;
276 }
277 virtual void Close() { state_ = SS_CLOSED; }
278
279private:
280 typedef std::vector<char> Buffer;
281 Buffer readable_data_, written_data_;
282 StreamState state_;
283 size_t read_block_, write_block_;
284};
285
286///////////////////////////////////////////////////////////////////////////////
287// SocketTestClient
288// Creates a simulated client for testing. Works on real and virtual networks.
289///////////////////////////////////////////////////////////////////////////////
290
291class SocketTestClient : public sigslot::has_slots<> {
292public:
293 SocketTestClient() {
294 Init(NULL, AF_INET);
295 }
296 SocketTestClient(AsyncSocket* socket) {
297 Init(socket, socket->GetLocalAddress().family());
298 }
299 SocketTestClient(const SocketAddress& address) {
300 Init(NULL, address.family());
301 socket_->Connect(address);
302 }
303
304 AsyncSocket* socket() { return socket_.get(); }
305
306 void QueueString(const char* data) {
307 QueueData(data, strlen(data));
308 }
309 void QueueStringF(const char* format, ...) {
310 va_list args;
311 va_start(args, format);
312 char buffer[1024];
313 size_t len = vsprintfn(buffer, sizeof(buffer), format, args);
314 ASSERT(len < sizeof(buffer) - 1);
315 va_end(args);
316 QueueData(buffer, len);
317 }
318 void QueueData(const char* data, size_t len) {
319 send_buffer_.insert(send_buffer_.end(), data, data + len);
320 if (Socket::CS_CONNECTED == socket_->GetState()) {
321 Flush();
322 }
323 }
324 std::string ReadData() {
325 std::string data(&recv_buffer_[0], recv_buffer_.size());
326 recv_buffer_.clear();
327 return data;
328 }
329
330 bool IsConnected() const {
331 return (Socket::CS_CONNECTED == socket_->GetState());
332 }
333 bool IsClosed() const {
334 return (Socket::CS_CLOSED == socket_->GetState());
335 }
336
337private:
338 typedef std::vector<char> Buffer;
339
340 void Init(AsyncSocket* socket, int family) {
341 if (!socket) {
342 socket = Thread::Current()->socketserver()
343 ->CreateAsyncSocket(family, SOCK_STREAM);
344 }
345 socket_.reset(socket);
346 socket_->SignalConnectEvent.connect(this,
347 &SocketTestClient::OnConnectEvent);
348 socket_->SignalReadEvent.connect(this, &SocketTestClient::OnReadEvent);
349 socket_->SignalWriteEvent.connect(this, &SocketTestClient::OnWriteEvent);
350 socket_->SignalCloseEvent.connect(this, &SocketTestClient::OnCloseEvent);
351 }
352
353 void Flush() {
354 size_t sent = 0;
355 while (sent < send_buffer_.size()) {
356 int result = socket_->Send(&send_buffer_[sent],
357 send_buffer_.size() - sent);
358 if (result > 0) {
359 sent += result;
360 } else {
361 break;
362 }
363 }
364 size_t new_size = send_buffer_.size() - sent;
365 memmove(&send_buffer_[0], &send_buffer_[sent], new_size);
366 send_buffer_.resize(new_size);
367 }
368
369 void OnConnectEvent(AsyncSocket* socket) {
370 if (!send_buffer_.empty()) {
371 Flush();
372 }
373 }
374 void OnReadEvent(AsyncSocket* socket) {
375 char data[64 * 1024];
376 int result = socket_->Recv(data, ARRAY_SIZE(data));
377 if (result > 0) {
378 recv_buffer_.insert(recv_buffer_.end(), data, data + result);
379 }
380 }
381 void OnWriteEvent(AsyncSocket* socket) {
382 if (!send_buffer_.empty()) {
383 Flush();
384 }
385 }
386 void OnCloseEvent(AsyncSocket* socket, int error) {
387 }
388
389 scoped_ptr<AsyncSocket> socket_;
390 Buffer send_buffer_, recv_buffer_;
391};
392
393///////////////////////////////////////////////////////////////////////////////
394// SocketTestServer
395// Creates a simulated server for testing. Works on real and virtual networks.
396///////////////////////////////////////////////////////////////////////////////
397
398class SocketTestServer : public sigslot::has_slots<> {
399 public:
400 SocketTestServer(const SocketAddress& address)
401 : socket_(Thread::Current()->socketserver()
402 ->CreateAsyncSocket(address.family(), SOCK_STREAM))
403 {
404 socket_->SignalReadEvent.connect(this, &SocketTestServer::OnReadEvent);
405 socket_->Bind(address);
406 socket_->Listen(5);
407 }
408 virtual ~SocketTestServer() {
409 clear();
410 }
411
412 size_t size() const { return clients_.size(); }
413 SocketTestClient* client(size_t index) const { return clients_[index]; }
414 SocketTestClient* operator[](size_t index) const { return client(index); }
415
416 void clear() {
417 for (size_t i=0; i<clients_.size(); ++i) {
418 delete clients_[i];
419 }
420 clients_.clear();
421 }
422
423 private:
424 void OnReadEvent(AsyncSocket* socket) {
425 AsyncSocket* accepted =
426 static_cast<AsyncSocket*>(socket_->Accept(NULL));
427 if (!accepted)
428 return;
429 clients_.push_back(new SocketTestClient(accepted));
430 }
431
432 scoped_ptr<AsyncSocket> socket_;
433 std::vector<SocketTestClient*> clients_;
434};
435
436///////////////////////////////////////////////////////////////////////////////
437// Generic Utilities
438///////////////////////////////////////////////////////////////////////////////
439
440inline bool ReadFile(const char* filename, std::string* contents) {
441 FILE* fp = fopen(filename, "rb");
442 if (!fp)
443 return false;
444 char buffer[1024*64];
445 size_t read;
446 contents->clear();
447 while ((read = fread(buffer, 1, sizeof(buffer), fp))) {
448 contents->append(buffer, read);
449 }
450 bool success = (0 != feof(fp));
451 fclose(fp);
452 return success;
453}
454
buildbot@webrtc.orgaf42ac52014-05-14 21:35:19 +0000455// Look in parent dir for parallel directory.
456inline talk_base::Pathname GetSiblingDirectory(
457 const std::string& parallel_dir) {
458 talk_base::Pathname path = talk_base::Filesystem::GetCurrentDirectory();
459 while (!path.empty()) {
460 talk_base::Pathname potential_parallel_dir = path;
461 potential_parallel_dir.AppendFolder(parallel_dir);
462 if (talk_base::Filesystem::IsFolder(potential_parallel_dir)) {
463 return potential_parallel_dir;
464 }
465
466 path.SetFolder(path.parent_folder());
467 }
468 return path;
469}
470
471inline talk_base::Pathname GetGoogle3Directory() {
472 return GetSiblingDirectory("google3");
473}
474
475inline talk_base::Pathname GetTalkDirectory() {
476 return GetSiblingDirectory("talk");
477}
478
henrike@webrtc.org0e118e72013-07-10 00:45:36 +0000479///////////////////////////////////////////////////////////////////////////////
480// Unittest predicates which are similar to STREQ, but for raw memory
481///////////////////////////////////////////////////////////////////////////////
482
483inline AssertionResult CmpHelperMemEq(const char* expected_expression,
484 const char* expected_length_expression,
485 const char* actual_expression,
486 const char* actual_length_expression,
487 const void* expected,
488 size_t expected_length,
489 const void* actual,
490 size_t actual_length)
491{
492 if ((expected_length == actual_length)
493 && (0 == memcmp(expected, actual, expected_length))) {
494 return AssertionSuccess();
495 }
496
497 Message msg;
498 msg << "Value of: " << actual_expression
499 << " [" << actual_length_expression << "]";
500 if (true) { //!actual_value.Equals(actual_expression)) {
501 size_t buffer_size = actual_length * 2 + 1;
502 char* buffer = STACK_ARRAY(char, buffer_size);
503 hex_encode(buffer, buffer_size,
504 reinterpret_cast<const char*>(actual), actual_length);
505 msg << "\n Actual: " << buffer << " [" << actual_length << "]";
506 }
507
508 msg << "\nExpected: " << expected_expression
509 << " [" << expected_length_expression << "]";
510 if (true) { //!expected_value.Equals(expected_expression)) {
511 size_t buffer_size = expected_length * 2 + 1;
512 char* buffer = STACK_ARRAY(char, buffer_size);
513 hex_encode(buffer, buffer_size,
514 reinterpret_cast<const char*>(expected), expected_length);
515 msg << "\nWhich is: " << buffer << " [" << expected_length << "]";
516 }
517
518 return AssertionFailure(msg);
519}
520
521inline AssertionResult CmpHelperFileEq(const char* expected_expression,
522 const char* expected_length_expression,
523 const char* actual_filename,
524 const void* expected,
525 size_t expected_length,
526 const char* filename)
527{
528 std::string contents;
529 if (!ReadFile(filename, &contents)) {
530 Message msg;
531 msg << "File '" << filename << "' could not be read.";
532 return AssertionFailure(msg);
533 }
534 return CmpHelperMemEq(expected_expression, expected_length_expression,
535 actual_filename, "",
536 expected, expected_length,
537 contents.c_str(), contents.size());
538}
539
540#define EXPECT_MEMEQ(expected, expected_length, actual, actual_length) \
541 EXPECT_PRED_FORMAT4(::testing::CmpHelperMemEq, expected, expected_length, \
542 actual, actual_length)
543
544#define ASSERT_MEMEQ(expected, expected_length, actual, actual_length) \
545 ASSERT_PRED_FORMAT4(::testing::CmpHelperMemEq, expected, expected_length, \
546 actual, actual_length)
547
548#define EXPECT_FILEEQ(expected, expected_length, filename) \
549 EXPECT_PRED_FORMAT3(::testing::CmpHelperFileEq, expected, expected_length, \
550 filename)
551
552#define ASSERT_FILEEQ(expected, expected_length, filename) \
553 ASSERT_PRED_FORMAT3(::testing::CmpHelperFileEq, expected, expected_length, \
554 filename)
555
556///////////////////////////////////////////////////////////////////////////////
557// Helpers for initializing constant memory with integers in a particular byte
558// order
559///////////////////////////////////////////////////////////////////////////////
560
561#define BYTE_CAST(x) static_cast<uint8>((x) & 0xFF)
562
563// Declare a N-bit integer as a little-endian sequence of bytes
564#define LE16(x) BYTE_CAST(((uint16)x) >> 0), BYTE_CAST(((uint16)x) >> 8)
565
566#define LE32(x) BYTE_CAST(((uint32)x) >> 0), BYTE_CAST(((uint32)x) >> 8), \
567 BYTE_CAST(((uint32)x) >> 16), BYTE_CAST(((uint32)x) >> 24)
568
569#define LE64(x) BYTE_CAST(((uint64)x) >> 0), BYTE_CAST(((uint64)x) >> 8), \
570 BYTE_CAST(((uint64)x) >> 16), BYTE_CAST(((uint64)x) >> 24), \
571 BYTE_CAST(((uint64)x) >> 32), BYTE_CAST(((uint64)x) >> 40), \
572 BYTE_CAST(((uint64)x) >> 48), BYTE_CAST(((uint64)x) >> 56)
573
574// Declare a N-bit integer as a big-endian (Internet) sequence of bytes
575#define BE16(x) BYTE_CAST(((uint16)x) >> 8), BYTE_CAST(((uint16)x) >> 0)
576
577#define BE32(x) BYTE_CAST(((uint32)x) >> 24), BYTE_CAST(((uint32)x) >> 16), \
578 BYTE_CAST(((uint32)x) >> 8), BYTE_CAST(((uint32)x) >> 0)
579
580#define BE64(x) BYTE_CAST(((uint64)x) >> 56), BYTE_CAST(((uint64)x) >> 48), \
581 BYTE_CAST(((uint64)x) >> 40), BYTE_CAST(((uint64)x) >> 32), \
582 BYTE_CAST(((uint64)x) >> 24), BYTE_CAST(((uint64)x) >> 16), \
583 BYTE_CAST(((uint64)x) >> 8), BYTE_CAST(((uint64)x) >> 0)
584
585// Declare a N-bit integer as a this-endian (local machine) sequence of bytes
586#ifndef BIG_ENDIAN
587#define BIG_ENDIAN 1
588#endif // BIG_ENDIAN
589
590#if BIG_ENDIAN
591#define TE16 BE16
592#define TE32 BE32
593#define TE64 BE64
594#else // !BIG_ENDIAN
595#define TE16 LE16
596#define TE32 LE32
597#define TE64 LE64
598#endif // !BIG_ENDIAN
599
600///////////////////////////////////////////////////////////////////////////////
601
mallinath@webrtc.org55280702013-09-27 23:04:10 +0000602// Helpers for determining if X/screencasting is available (on linux).
603
604#define MAYBE_SKIP_SCREENCAST_TEST() \
605 if (!testing::IsScreencastingAvailable()) { \
606 LOG(LS_WARNING) << "Skipping test, since it doesn't have the requisite " \
607 << "X environment for screen capture."; \
608 return; \
609 } \
610
611#ifdef LINUX
612struct XDisplay {
613 XDisplay() : display_(XOpenDisplay(NULL)) { }
614 ~XDisplay() { if (display_) XCloseDisplay(display_); }
615 bool IsValid() const { return display_ != NULL; }
616 operator Display*() { return display_; }
617 private:
618 Display* display_;
619};
620#endif
621
622// Returns true if screencasting is available. When false, anything that uses
623// screencasting features may fail.
624inline bool IsScreencastingAvailable() {
625#ifdef LINUX
626 XDisplay display;
627 if (!display.IsValid()) {
628 LOG(LS_WARNING) << "No X Display available.";
629 return false;
630 }
mallinath@webrtc.orgb881d272014-02-03 16:57:16 +0000631 int ignored_int, major_version, minor_version;
632 if (!XRRQueryExtension(display, &ignored_int, &ignored_int) ||
633 !XRRQueryVersion(display, &major_version, &minor_version) ||
634 major_version < 1 ||
635 (major_version < 2 && minor_version < 3)) {
636 LOG(LS_WARNING) << "XRandr version: " << major_version << "."
637 << minor_version;
638 LOG(LS_WARNING) << "XRandr is not supported or is too old (pre 1.3).";
639 return false;
640 }
mallinath@webrtc.org55280702013-09-27 23:04:10 +0000641#endif
642 return true;
643}
henrike@webrtc.org0e118e72013-07-10 00:45:36 +0000644} // namespace testing
645
646#endif // TALK_BASE_TESTUTILS_H__