15821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Copyright (c) 2012 The Chromium Authors. All rights reserved.
25821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Use of this source code is governed by a BSD-style license that can be
35821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// found in the LICENSE file.
45821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
55821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#ifndef PPAPI_TESTS_TEST_UTILS_H_
65821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#define PPAPI_TESTS_TEST_UTILS_H_
75821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
85821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include <string>
95821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/c/dev/ppb_testing_dev.h"
115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/c/pp_instance.h"
125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/c/pp_stdint.h"
135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/cpp/completion_callback.h"
142a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "ppapi/cpp/message_loop.h"
155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/utility/completion_callback_factory.h"
165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
177d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)namespace pp {
18eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdochclass NetAddress;
197d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)}
207d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)
215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Timeout to wait for some action to complete.
225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)extern const int kActionTimeoutMs;
235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)const PPB_Testing_Dev* GetTestingInterface();
255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)std::string ReportError(const char* method, int32_t error);
265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void PlatformSleep(int duration_ms);
275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool GetLocalHostPort(PP_Instance instance, std::string* host, uint16_t* port);
285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
29868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)uint16_t ConvertFromNetEndian16(uint16_t x);
30868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)uint16_t ConvertToNetEndian16(uint16_t x);
31eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdochbool EqualNetAddress(const pp::NetAddress& addr1, const pp::NetAddress& addr2);
327d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)// Only returns the first address if there are more than one available.
337d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)bool ResolveHost(PP_Instance instance,
347d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)                 const std::string& host,
357d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)                 uint16_t port,
36eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch                 pp::NetAddress* addr);
37868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)
385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// NestedEvent allows you to run a nested MessageLoop and wait for a particular
395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// event to complete. For example, you can use it to wait for a callback on a
405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// PPP interface, which will "Signal" the event and make the loop quit.
415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// "Wait()" will return immediately if it has already been signalled. Otherwise,
425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// it will run a nested message loop (using PPB_Testing.RunMessageLoop) and will
435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// return only after it has been signalled.
445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Example:
455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)//  std::string TestFullscreen::TestNormalToFullscreen() {
465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)//    pp::Fullscreen screen_mode(instance);
475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)//    screen_mode.SetFullscreen(true);
485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)//    SimulateUserGesture();
495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)//    // Let DidChangeView run in a nested message loop.
505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)//    nested_event_.Wait();
515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)//    Pass();
525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)//  }
535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)//
545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)//  void TestFullscreen::DidChangeView(const pp::View& view) {
555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)//    nested_event_.Signal();
565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)//  }
57c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)//
58c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)// All methods except Signal and PostSignal must be invoked on the main thread.
59c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)// It's OK to signal from a background thread, so you can (for example) Signal()
60c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)// from the Audio thread.
615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)class NestedEvent {
625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) public:
635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  explicit NestedEvent(PP_Instance instance)
645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      : instance_(instance), waiting_(false), signalled_(false) {
655821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Run a nested message loop and wait until Signal() is called. If Signal()
675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // has already been called, return immediately without running a nested loop.
685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void Wait();
695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Signal the NestedEvent. If Wait() has been called, quit the message loop.
70c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  // This can be called from any thread.
715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void Signal();
72c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  // Signal the NestedEvent in |wait_ms| milliseconds. This can be called from
73c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  // any thread.
74c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  void PostSignal(int32_t wait_ms);
75c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)
765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Reset the NestedEvent so it can be used again.
775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void Reset();
785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) private:
79c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  void SignalOnMainThread();
80c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  static void SignalThunk(void* async_event, int32_t result);
81c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)
825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  PP_Instance instance_;
835821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool waiting_;
845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool signalled_;
855821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Disable copy and assign.
865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NestedEvent(const NestedEvent&);
875821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  NestedEvent& operator=(const NestedEvent&);
885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)};
895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)enum CallbackType { PP_REQUIRED, PP_OPTIONAL, PP_BLOCKING };
915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)class TestCompletionCallback {
925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) public:
935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  class Delegate {
945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)   public:
955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    virtual ~Delegate() {}
965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    virtual void OnCallback(void* user_data, int32_t result) = 0;
975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  };
985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  explicit TestCompletionCallback(PP_Instance instance);
995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // TODO(dmichael): Remove this constructor.
1005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  TestCompletionCallback(PP_Instance instance, bool force_async);
1015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  TestCompletionCallback(PP_Instance instance, CallbackType callback_type);
1035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Sets a Delegate instance. OnCallback() of this instance will be invoked
1055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // when the completion callback is invoked.
1065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // The delegate will be reset when Reset() or GetCallback() is called.
1075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void SetDelegate(Delegate* delegate) { delegate_ = delegate; }
1085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Wait for a result, given the return from the call which took this callback
1105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // as a parameter. If |result| is PP_OK_COMPLETIONPENDING, WaitForResult will
1115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // block until its callback has been invoked (in some cases, this will already
1125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // have happened, and WaitForCallback can return immediately).
1135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // For any other values, WaitForResult will simply set its internal "result_"
1145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // field. To retrieve the final result of the operation (i.e., the result
1155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // the callback has run, if necessary), call result(). You can call result()
1165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // as many times as necessary until a new pp::CompletionCallback is retrieved.
1175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //
1185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // In some cases, you may want to check that the callback was invoked in the
1195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // expected way (i.e., if the callback was "Required", then it should be
1205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // invoked asynchronously). Within the body of a test (where returning a non-
1215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // empty string indicates test failure), you can use the
1225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // CHECK_CALLBACK_BEHAVIOR(callback) macro. From within a helper function,
1235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // you can use failed() and errors().
1245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //
1255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Example usage within a test:
1265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //  callback.WaitForResult(foo.DoSomething(callback));
1275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //  CHECK_CALLBACK_BEHAVIOR(callback);
1285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //  ASSERT_EQ(PP_OK, callback.result());
1295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //
1305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Example usage within a helper function:
1315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //  void HelperFunction(std::string* error_message) {
1325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //    callback.WaitForResult(foo.DoSomething(callback));
1335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //    if (callback.failed())
1345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //      error_message->assign(callback.errors());
1355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //  }
1365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void WaitForResult(int32_t result);
1375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Used when you expect to receive either synchronous completion with PP_OK
1395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // or a PP_ERROR_ABORTED asynchronously.
1405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //  Example usage:
1415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //  int32_t result = 0;
1425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //  {
1435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //    pp::URLLoader temp(instance_);
1445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //    result = temp.Open(request, callback);
1455821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //  }
1465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //  callback.WaitForAbortResult(result);
1475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  //  CHECK_CALLBACK_BEHAVIOR(callback);
1485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void WaitForAbortResult(int32_t result);
1495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Retrieve a pp::CompletionCallback for use in testing. This Reset()s the
1515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // TestCompletionCallback.
1525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  pp::CompletionCallback GetCallback();
1535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool failed() { return !errors_.empty(); }
1555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  const std::string& errors() { return errors_; }
1565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  int32_t result() const { return result_; }
1585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Reset so that this callback can be used again.
1605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void Reset();
1615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
162c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  CallbackType callback_type() { return callback_type_; }
163c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  void set_target_loop(const pp::MessageLoop& loop) { target_loop_ = loop; }
1645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  static void Handler(void* user_data, int32_t result);
165c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)
166c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) protected:
1675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void RunMessageLoop();
1685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  void QuitMessageLoop();
1695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Used to check that WaitForResult is only called once for each usage of the
1715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // callback.
1725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool wait_for_result_called_;
1735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Indicates whether we have already been invoked.
1745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool have_result_;
1755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // The last result received (or PP_OK_COMPLETIONCALLBACK if none).
1765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  int32_t result_;
1775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  CallbackType callback_type_;
1785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  bool post_quit_task_;
1795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  std::string errors_;
1805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  PP_Instance instance_;
1815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  Delegate* delegate_;
1822a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  pp::MessageLoop target_loop_;
1832a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)};
1842a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
185c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)namespace internal {
186c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)
187c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)template <typename OutputT, typename CallbackT>
188c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)class TestCompletionCallbackWithOutputBase {
1892a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) public:
190c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  explicit TestCompletionCallbackWithOutputBase(PP_Instance instance)
1917dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      : callback_(instance),
1927dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch        output_storage_() {
1937dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch    CallbackT::TraitsType::Initialize(&output_storage_);
1942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
1952a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
196c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  TestCompletionCallbackWithOutputBase(PP_Instance instance, bool force_async)
1977dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      : callback_(instance, force_async),
1987dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch        output_storage_() {
1997dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch    CallbackT::TraitsType::Initialize(&output_storage_);
2002a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
2012a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
202c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  TestCompletionCallbackWithOutputBase(PP_Instance instance,
203c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                                       CallbackType callback_type)
2047dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      : callback_(instance, callback_type),
2057dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch        output_storage_() {
2067dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch    CallbackT::TraitsType::Initialize(&output_storage_);
2072a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
2082a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
209c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  CallbackT GetCallback();
210c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  OutputT output() {
211c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    return CallbackT::TraitsType::StorageToPluginArg(
212c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)        output_storage_);
2132a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
2142a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
215c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  // Delegate functions to TestCompletionCallback
216c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  void SetDelegate(TestCompletionCallback::Delegate* delegate) {
217c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    callback_.SetDelegate(delegate);
218c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  }
219c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  void WaitForResult(int32_t result) { callback_.WaitForResult(result); }
220c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  void WaitForAbortResult(int32_t result) {
221c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    callback_.WaitForAbortResult(result);
222c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  }
223c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  bool failed() { return callback_.failed(); }
224c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  const std::string& errors() { return callback_.errors(); }
225c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  int32_t result() const { return callback_.result(); }
2267dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  void Reset() {
2277dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch    CallbackT::TraitsType::Initialize(&output_storage_);
2287dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch    return callback_.Reset();
2297dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  }
2302a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
231c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) private:
232c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  TestCompletionCallback callback_;
233c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  typename CallbackT::OutputStorageType output_storage_;
2345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)};
2355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
236c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)template <typename OutputT, typename CallbackT>
237c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)CallbackT
238c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)TestCompletionCallbackWithOutputBase<OutputT, CallbackT>::GetCallback() {
2397dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  this->Reset();
240c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  if (callback_.callback_type() == PP_BLOCKING) {
241c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    CallbackT cc(&output_storage_);
2422a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    return cc;
2432a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
2442a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
245c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  callback_.set_target_loop(pp::MessageLoop::GetCurrent());
246c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  CallbackT cc(&TestCompletionCallback::Handler, this, &output_storage_);
247c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  if (callback_.callback_type() == PP_OPTIONAL)
2482a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    cc.set_flags(PP_COMPLETIONCALLBACK_FLAG_OPTIONAL);
2492a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  return cc;
2502a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
2512a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
252c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)}  // namespace internal
253c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)
254c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)template <typename OutputT>
255c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)class TestCompletionCallbackWithOutput
256c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    : public internal::TestCompletionCallbackWithOutputBase<
257c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)        OutputT, pp::CompletionCallbackWithOutput<OutputT> > {
258c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) public:
259c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  explicit TestCompletionCallbackWithOutput(PP_Instance instance)
260c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      : BaseType(instance) {
261c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  }
262c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)
263c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  TestCompletionCallbackWithOutput(PP_Instance instance, bool force_async)
264c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      : BaseType(instance, force_async) {
265c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  }
266c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)
267c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  TestCompletionCallbackWithOutput(PP_Instance instance,
268c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                                   CallbackType callback_type)
269c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      : BaseType(instance, callback_type) {
270c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  }
271c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)
272c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) private:
273c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  typedef internal::TestCompletionCallbackWithOutputBase<
274c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      OutputT, pp::CompletionCallbackWithOutput<OutputT> > BaseType;
275c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)};
276c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)
277c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)template <typename OutputT>
278c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)class TestExtCompletionCallbackWithOutput
279c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    : public internal::TestCompletionCallbackWithOutputBase<
280c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)        OutputT, pp::ext::ExtCompletionCallbackWithOutput<OutputT> > {
281c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) public:
282c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  explicit TestExtCompletionCallbackWithOutput(PP_Instance instance)
283c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      : BaseType(instance) {
284c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  }
285c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)
286c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  TestExtCompletionCallbackWithOutput(PP_Instance instance, bool force_async)
287c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      : BaseType(instance, force_async) {
288c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  }
289c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)
290c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  TestExtCompletionCallbackWithOutput(PP_Instance instance,
291c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                                      CallbackType callback_type)
292c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      : BaseType(instance, callback_type) {
293c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  }
294c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)
295c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles) private:
296c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  typedef internal::TestCompletionCallbackWithOutputBase<
297c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)      OutputT, pp::ext::ExtCompletionCallbackWithOutput<OutputT> > BaseType;
298c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)};
2992a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
3005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Verifies that the callback didn't record any errors. If the callback is run
3015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// in an unexpected way (e.g., if it's invoked asynchronously when the call
3025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// should have blocked), this returns an appropriate error string.
3035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#define CHECK_CALLBACK_BEHAVIOR(callback) \
3045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)do { \
3055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if ((callback).failed()) \
306c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)    return MakeFailureMessage(__FILE__, __LINE__, \
307c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)                              (callback).errors().c_str()); \
3085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)} while (false)
3095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)/*
3115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) * A set of macros to use for platform detection. These were largely copied
3125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) * from chromium's build_config.h.
3135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles) */
3145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if defined(__APPLE__)
3155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#define PPAPI_OS_MACOSX 1
3165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#elif defined(ANDROID)
3175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#define PPAPI_OS_ANDROID 1
3185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#elif defined(__native_client__)
3195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#define PPAPI_OS_NACL 1
3205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#elif defined(__linux__)
3215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#define PPAPI_OS_LINUX 1
3225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#elif defined(_WIN32)
3235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#define PPAPI_OS_WIN 1
3245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#elif defined(__FreeBSD__)
3255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#define PPAPI_OS_FREEBSD 1
3265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#elif defined(__OpenBSD__)
3275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#define PPAPI_OS_OPENBSD 1
3285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#elif defined(__sun)
3295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#define PPAPI_OS_SOLARIS 1
3305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#else
3315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#error Please add support for your platform in ppapi/tests/test_utils.h
3325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif
3335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)/* These are used to determine POSIX-like implementations vs Windows. */
3355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#if defined(__linux__) || defined(__APPLE__) || defined(__FreeBSD__) || \
3365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    defined(__OpenBSD__) || defined(__sun) || defined(__native_client__)
3375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#define PPAPI_POSIX 1
3385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif
3395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
3405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#endif  // PPAPI_TESTS_TEST_UTILS_H_
341