summaryrefslogtreecommitdiffstats
path: root/chrome_frame/sync_msg_reply_dispatcher.h
blob: 724c0074fcbecedf27a92220fe9f00d80f9eaae2 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
// Copyright (c) 2009 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#ifndef CHROME_FRAME_SYNC_MSG_REPLY_DISPATCHER_H_
#define CHROME_FRAME_SYNC_MSG_REPLY_DISPATCHER_H_

#include <deque>

#include "base/callback.h"
#include "base/lock.h"
#include "ipc/ipc_channel_proxy.h"

// Base class used to allow synchronous IPC messages to be sent and
// received in an asynchronous manner. To use this class add it as a filter to
// your IPC channel using ChannelProxy::AddFilter(). From then on, before
// sending a synchronous message, call SyncMessageReplyDispatcher::Push() with
// a callback and a key. This class will then handle the message response and
// will call the callback when it is received.
//
// This class is intended to be extended by classes implementing
// HandleMessageType with delegation for the messages they expect to receive in
// cases where you care about the return values of synchronous messages.
//
// Sample usage pattern:
//
// // Add handling for desired message types.
// class SyncMessageReplyDispatcherImpl : public SyncMessageReplyDispatcher {
//   virtual bool HandleMessageType(const IPC::Message& msg,
//                                  const MessageSent& origin) {
//    switch (origin.type) {
//      case AutomationMsg_CreateExternalTab::ID:
//        InvokeCallback<Tuple3<HWND, HWND, int> >(msg, origin);
//        break;
//     [HANDLING FOR OTHER EXPECTED MESSAGE TYPES]
//   }
// }
//
// // Add the filter
// IPC::SyncChannel channel_;
// channel_.AddFilter(new SyncMessageReplyDispatcherImpl());
//
// sync_->Push(msg, NewCallback(this, CallbackMethod, this);
// channel_->ChannelProxy::Send(msg);
//
class SyncMessageReplyDispatcher : public IPC::ChannelProxy::MessageFilter {
 public:
  SyncMessageReplyDispatcher() {}
  void Push(IPC::SyncMessage* msg, void* callback, void* key);
  void Cancel(void* key);

 protected:
  struct MessageSent {
    MessageSent() {}
    MessageSent(int id, uint32 type, void* callback, void* key)
      : id(id), callback(callback), type(type), key(key) {}
    int id;
    void* callback;
    void* key;
    uint32 type;
  };

  typedef std::deque<MessageSent> PendingSyncMessageQueue;

  bool Pop(const IPC::Message& msg, MessageSent* t);
  virtual bool OnMessageReceived(const IPC::Message& msg);

  // Child classes must implement a handler for the message types they are
  // interested in handling responses for. If you don't care about the replies
  // to any of the sync messages you are handling, then you don't have to
  // implement this.
  virtual bool HandleMessageType(const IPC::Message& msg,
                                 const MessageSent& origin);

  template <typename T>
  void InvokeCallback(const IPC::Message& msg, const MessageSent& origin) {
    // Ensure we delete the callback
    scoped_ptr<CallbackRunner<T> > c(
        reinterpret_cast<CallbackRunner<T>*>(origin.callback));
    if (origin.key == NULL) {  // Canceled
      return;
    }

    T tmp;  // Acts as "initializer" for output parameters.
    IPC::ParamDeserializer<T> deserializer(tmp);
    if (deserializer.MessageReplyDeserializer::SerializeOutputParameters(msg)) {
      c->RunWithParams(deserializer.out_);
    } else {
      // TODO(stoyan): How to handle errors?
    }
  }

  PendingSyncMessageQueue message_queue_;
  Lock message_queue_lock_;
};

#endif  // CHROME_FRAME_SYNC_MSG_REPLY_DISPATCHER_H_