Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(476)

Side by Side Diff: ipc/ipc_message_utils.h

Issue 3170020: Completely revert all my IPC work to see if this was what regressed the page cycler. (Closed) Base URL: http://src.chromium.org/git/chromium.git
Patch Set: Rebase Created 10 years, 4 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« no previous file with comments | « ipc/ipc_message_macros.h ('k') | ipc/ipc_message_utils.cc » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. 1 // Copyright (c) 2010 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be 2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file. 3 // found in the LICENSE file.
4 4
5 #ifndef IPC_IPC_MESSAGE_UTILS_H_ 5 #ifndef IPC_IPC_MESSAGE_UTILS_H_
6 #define IPC_IPC_MESSAGE_UTILS_H_ 6 #define IPC_IPC_MESSAGE_UTILS_H_
7 #pragma once 7 #pragma once
8 8
9 #include <algorithm> 9 #include <algorithm>
10 #include <string> 10 #include <string>
11 #include <vector> 11 #include <vector>
12 #include <map> 12 #include <map>
13 #include <set> 13 #include <set>
14 14
15 #include "base/file_path.h"
15 #include "base/format_macros.h" 16 #include "base/format_macros.h"
17 #include "base/nullable_string16.h"
16 #include "base/string16.h" 18 #include "base/string16.h"
17 #include "base/string_number_conversions.h" 19 #include "base/string_number_conversions.h"
18 #include "base/string_util.h" 20 #include "base/string_util.h"
21 #include "base/time.h"
19 #include "base/tuple.h" 22 #include "base/tuple.h"
20 #include "base/utf_string_conversions.h" 23 #include "base/utf_string_conversions.h"
24 #include "base/values.h"
25 #if defined(OS_POSIX)
26 #include "ipc/file_descriptor_set_posix.h"
27 #endif
28 #include "ipc/ipc_channel_handle.h"
21 #include "ipc/ipc_sync_message.h" 29 #include "ipc/ipc_sync_message.h"
22 30
23 #if defined(COMPILER_GCC)
24 // GCC "helpfully" tries to inline template methods in release mode. Except we
25 // want the majority of the template junk being expanded once in the
26 // implementation file (and only provide the definitions in
27 // ipc_message_utils_impl.h in those files) and exported, instead of expanded
28 // at every call site. Special note: GCC happily accepts the attribute before
29 // the method declaration, but only acts on it if it is after.
30 #define IPC_MSG_NOINLINE __attribute__((noinline));
31 #elif defined(COMPILER_MSVC)
32 // MSVC++ doesn't do this.
33 #define IPC_MSG_NOINLINE
34 #else
35 #error "Please add the noinline property for your new compiler here."
36 #endif
37
38 // Used by IPC_BEGIN_MESSAGES so that each message class starts from a unique 31 // Used by IPC_BEGIN_MESSAGES so that each message class starts from a unique
39 // base. Messages have unique IDs across channels in order for the IPC logging 32 // base. Messages have unique IDs across channels in order for the IPC logging
40 // code to figure out the message class from its ID. 33 // code to figure out the message class from its ID.
41 enum IPCMessageStart { 34 enum IPCMessageStart {
42 // By using a start value of 0 for automation messages, we keep backward 35 // By using a start value of 0 for automation messages, we keep backward
43 // compatibility with old builds. 36 // compatibility with old builds.
44 AutomationMsgStart = 0, 37 AutomationMsgStart = 0,
45 ViewMsgStart, 38 ViewMsgStart,
46 ViewHostMsgStart, 39 ViewHostMsgStart,
47 PluginProcessMsgStart, 40 PluginProcessMsgStart,
(...skipping 19 matching lines...) Expand all
67 GpuChannelMsgStart, 60 GpuChannelMsgStart,
68 GpuVideoDecoderHostMsgStart, 61 GpuVideoDecoderHostMsgStart,
69 GpuVideoDecoderMsgStart, 62 GpuVideoDecoderMsgStart,
70 ServiceMsgStart, 63 ServiceMsgStart,
71 ServiceHostMsgStart, 64 ServiceHostMsgStart,
72 // NOTE: When you add a new message class, also update 65 // NOTE: When you add a new message class, also update
73 // IPCStatusView::IPCStatusView to ensure logging works. 66 // IPCStatusView::IPCStatusView to ensure logging works.
74 LastMsgIndex 67 LastMsgIndex
75 }; 68 };
76 69
77 class DictionaryValue;
78 class FilePath;
79 class ListValue;
80 class NullableString16;
81
82 namespace base {
83 class Time;
84 struct FileDescriptor;
85 }
86
87 namespace IPC { 70 namespace IPC {
88 71
89 struct ChannelHandle;
90
91 //----------------------------------------------------------------------------- 72 //-----------------------------------------------------------------------------
92 // An iterator class for reading the fields contained within a Message. 73 // An iterator class for reading the fields contained within a Message.
93 74
94 class MessageIterator { 75 class MessageIterator {
95 public: 76 public:
96 explicit MessageIterator(const Message& m) : msg_(m), iter_(NULL) { 77 explicit MessageIterator(const Message& m) : msg_(m), iter_(NULL) {
97 } 78 }
98 int NextInt() const { 79 int NextInt() const {
99 int val = -1; 80 int val = -1;
100 if (!msg_.ReadInt(&iter_, &val)) 81 if (!msg_.ReadInt(&iter_, &val))
(...skipping 217 matching lines...) Expand 10 before | Expand all | Expand 10 after
318 return result; 299 return result;
319 } 300 }
320 static void Log(const param_type& p, std::wstring* l) { 301 static void Log(const param_type& p, std::wstring* l) {
321 l->append(StringPrintf(L"%lc", p)); 302 l->append(StringPrintf(L"%lc", p));
322 } 303 }
323 }; 304 };
324 305
325 template <> 306 template <>
326 struct ParamTraits<base::Time> { 307 struct ParamTraits<base::Time> {
327 typedef base::Time param_type; 308 typedef base::Time param_type;
328 static void Write(Message* m, const param_type& p); 309 static void Write(Message* m, const param_type& p) {
329 static bool Read(const Message* m, void** iter, param_type* r); 310 ParamTraits<int64>::Write(m, p.ToInternalValue());
330 static void Log(const param_type& p, std::wstring* l); 311 }
312 static bool Read(const Message* m, void** iter, param_type* r) {
313 int64 value;
314 if (!ParamTraits<int64>::Read(m, iter, &value))
315 return false;
316 *r = base::Time::FromInternalValue(value);
317 return true;
318 }
319 static void Log(const param_type& p, std::wstring* l) {
320 ParamTraits<int64>::Log(p.ToInternalValue(), l);
321 }
331 }; 322 };
332 323
333 #if defined(OS_WIN) 324 #if defined(OS_WIN)
334 template <> 325 template <>
335 struct ParamTraits<LOGFONT> { 326 struct ParamTraits<LOGFONT> {
336 typedef LOGFONT param_type; 327 typedef LOGFONT param_type;
337 static void Write(Message* m, const param_type& p) { 328 static void Write(Message* m, const param_type& p) {
338 m->WriteData(reinterpret_cast<const char*>(&p), sizeof(LOGFONT)); 329 m->WriteData(reinterpret_cast<const char*>(&p), sizeof(LOGFONT));
339 } 330 }
340 static bool Read(const Message* m, void** iter, param_type* r) { 331 static bool Read(const Message* m, void** iter, param_type* r) {
(...skipping 26 matching lines...) Expand all
367 bool result = m->ReadData(iter, &data, &data_size); 358 bool result = m->ReadData(iter, &data, &data_size);
368 if (result && data_size == sizeof(MSG)) { 359 if (result && data_size == sizeof(MSG)) {
369 memcpy(r, data, sizeof(MSG)); 360 memcpy(r, data, sizeof(MSG));
370 } else { 361 } else {
371 result = false; 362 result = false;
372 NOTREACHED(); 363 NOTREACHED();
373 } 364 }
374 365
375 return result; 366 return result;
376 } 367 }
377 static void Log(const param_type& p, std::wstring* l) {
378 l->append(L"<MSG>");
379 }
380 }; 368 };
381 #endif // defined(OS_WIN) 369 #endif // defined(OS_WIN)
382 370
383 template <> 371 template <>
384 struct ParamTraits<DictionaryValue> { 372 struct ParamTraits<DictionaryValue> {
385 typedef DictionaryValue param_type; 373 typedef DictionaryValue param_type;
386 static void Write(Message* m, const param_type& p); 374 static void Write(Message* m, const param_type& p);
387 static bool Read(const Message* m, void** iter, param_type* r); 375 static bool Read(const Message* m, void** iter, param_type* r);
388 static void Log(const param_type& p, std::wstring* l); 376 static void Log(const param_type& p, std::wstring* l);
389 }; 377 };
(...skipping 213 matching lines...) Expand 10 before | Expand all | Expand 10 after
603 LogParam(p.first, l); 591 LogParam(p.first, l);
604 l->append(L", "); 592 l->append(L", ");
605 LogParam(p.second, l); 593 LogParam(p.second, l);
606 l->append(L")"); 594 l->append(L")");
607 } 595 }
608 }; 596 };
609 597
610 template <> 598 template <>
611 struct ParamTraits<NullableString16> { 599 struct ParamTraits<NullableString16> {
612 typedef NullableString16 param_type; 600 typedef NullableString16 param_type;
613 static void Write(Message* m, const param_type& p); 601 static void Write(Message* m, const param_type& p) {
614 static bool Read(const Message* m, void** iter, param_type* r); 602 WriteParam(m, p.string());
615 static void Log(const param_type& p, std::wstring* l); 603 WriteParam(m, p.is_null());
604 }
605 static bool Read(const Message* m, void** iter, param_type* r) {
606 string16 string;
607 if (!ReadParam(m, iter, &string))
608 return false;
609 bool is_null;
610 if (!ReadParam(m, iter, &is_null))
611 return false;
612 *r = NullableString16(string, is_null);
613 return true;
614 }
615 static void Log(const param_type& p, std::wstring* l) {
616 l->append(L"(");
617 LogParam(p.string(), l);
618 l->append(L", ");
619 LogParam(p.is_null(), l);
620 l->append(L")");
621 }
616 }; 622 };
617 623
618 // If WCHAR_T_IS_UTF16 is defined, then string16 is a std::wstring so we don't 624 // If WCHAR_T_IS_UTF16 is defined, then string16 is a std::wstring so we don't
619 // need this trait. 625 // need this trait.
620 #if !defined(WCHAR_T_IS_UTF16) 626 #if !defined(WCHAR_T_IS_UTF16)
621 template <> 627 template <>
622 struct ParamTraits<string16> { 628 struct ParamTraits<string16> {
623 typedef string16 param_type; 629 typedef string16 param_type;
624 static void Write(Message* m, const param_type& p) { 630 static void Write(Message* m, const param_type& p) {
625 m->WriteString16(p); 631 m->WriteString16(p);
(...skipping 70 matching lines...) Expand 10 before | Expand all | Expand 10 after
696 } 702 }
697 static void Log(const param_type& p, std::wstring* l) { 703 static void Log(const param_type& p, std::wstring* l) {
698 l->append(StringPrintf(L"(%d, %d)", p.x, p.y)); 704 l->append(StringPrintf(L"(%d, %d)", p.x, p.y));
699 } 705 }
700 }; 706 };
701 #endif // defined(OS_WIN) 707 #endif // defined(OS_WIN)
702 708
703 template <> 709 template <>
704 struct ParamTraits<FilePath> { 710 struct ParamTraits<FilePath> {
705 typedef FilePath param_type; 711 typedef FilePath param_type;
706 static void Write(Message* m, const param_type& p); 712 static void Write(Message* m, const param_type& p) {
707 static bool Read(const Message* m, void** iter, param_type* r); 713 ParamTraits<FilePath::StringType>::Write(m, p.value());
708 static void Log(const param_type& p, std::wstring* l); 714 }
715 static bool Read(const Message* m, void** iter, param_type* r) {
716 FilePath::StringType value;
717 if (!ParamTraits<FilePath::StringType>::Read(m, iter, &value))
718 return false;
719 *r = FilePath(value);
720 return true;
721 }
722 static void Log(const param_type& p, std::wstring* l) {
723 ParamTraits<FilePath::StringType>::Log(p.value(), l);
724 }
709 }; 725 };
710 726
711 #if defined(OS_POSIX) 727 #if defined(OS_POSIX)
712 // FileDescriptors may be serialised over IPC channels on POSIX. On the 728 // FileDescriptors may be serialised over IPC channels on POSIX. On the
713 // receiving side, the FileDescriptor is a valid duplicate of the file 729 // receiving side, the FileDescriptor is a valid duplicate of the file
714 // descriptor which was transmitted: *it is not just a copy of the integer like 730 // descriptor which was transmitted: *it is not just a copy of the integer like
715 // HANDLEs on Windows*. The only exception is if the file descriptor is < 0. In 731 // HANDLEs on Windows*. The only exception is if the file descriptor is < 0. In
716 // this case, the receiving end will see a value of -1. *Zero is a valid file 732 // this case, the receiving end will see a value of -1. *Zero is a valid file
717 // descriptor*. 733 // descriptor*.
718 // 734 //
719 // The received file descriptor will have the |auto_close| flag set to true. The 735 // The received file descriptor will have the |auto_close| flag set to true. The
720 // code which handles the message is responsible for taking ownership of it. 736 // code which handles the message is responsible for taking ownership of it.
721 // File descriptors are OS resources and must be closed when no longer needed. 737 // File descriptors are OS resources and must be closed when no longer needed.
722 // 738 //
723 // When sending a file descriptor, the file descriptor must be valid at the time 739 // When sending a file descriptor, the file descriptor must be valid at the time
724 // of transmission. Since transmission is not synchronous, one should consider 740 // of transmission. Since transmission is not synchronous, one should consider
725 // dup()ing any file descriptors to be transmitted and setting the |auto_close| 741 // dup()ing any file descriptors to be transmitted and setting the |auto_close|
726 // flag, which causes the file descriptor to be closed after writing. 742 // flag, which causes the file descriptor to be closed after writing.
727 template<> 743 template<>
728 struct ParamTraits<base::FileDescriptor> { 744 struct ParamTraits<base::FileDescriptor> {
729 typedef base::FileDescriptor param_type; 745 typedef base::FileDescriptor param_type;
730 static void Write(Message* m, const param_type& p); 746 static void Write(Message* m, const param_type& p) {
731 static bool Read(const Message* m, void** iter, param_type* r); 747 const bool valid = p.fd >= 0;
732 static void Log(const param_type& p, std::wstring* l); 748 WriteParam(m, valid);
749
750 if (valid) {
751 if (!m->WriteFileDescriptor(p))
752 NOTREACHED();
753 }
754 }
755 static bool Read(const Message* m, void** iter, param_type* r) {
756 bool valid;
757 if (!ReadParam(m, iter, &valid))
758 return false;
759
760 if (!valid) {
761 r->fd = -1;
762 r->auto_close = false;
763 return true;
764 }
765
766 return m->ReadFileDescriptor(iter, r);
767 }
768 static void Log(const param_type& p, std::wstring* l) {
769 if (p.auto_close) {
770 l->append(StringPrintf(L"FD(%d auto-close)", p.fd));
771 } else {
772 l->append(StringPrintf(L"FD(%d)", p.fd));
773 }
774 }
733 }; 775 };
734 #endif // defined(OS_POSIX) 776 #endif // defined(OS_POSIX)
735 777
736 // A ChannelHandle is basically a platform-inspecific wrapper around the 778 // A ChannelHandle is basically a platform-inspecific wrapper around the
737 // fact that IPC endpoints are handled specially on POSIX. See above comments 779 // fact that IPC endpoints are handled specially on POSIX. See above comments
738 // on FileDescriptor for more background. 780 // on FileDescriptor for more background.
739 template<> 781 template<>
740 struct ParamTraits<IPC::ChannelHandle> { 782 struct ParamTraits<IPC::ChannelHandle> {
741 typedef ChannelHandle param_type; 783 typedef ChannelHandle param_type;
742 static void Write(Message* m, const param_type& p); 784 static void Write(Message* m, const param_type& p) {
743 static bool Read(const Message* m, void** iter, param_type* r); 785 WriteParam(m, p.name);
744 static void Log(const param_type& p, std::wstring* l); 786 #if defined(OS_POSIX)
787 WriteParam(m, p.socket);
788 #endif
789 }
790 static bool Read(const Message* m, void** iter, param_type* r) {
791 return ReadParam(m, iter, &r->name)
792 #if defined(OS_POSIX)
793 && ReadParam(m, iter, &r->socket)
794 #endif
795 ;
796 }
797 static void Log(const param_type& p, std::wstring* l) {
798 l->append(ASCIIToWide(StringPrintf("ChannelHandle(%s", p.name.c_str())));
799 #if defined(OS_POSIX)
800 ParamTraits<base::FileDescriptor>::Log(p.socket, l);
801 #endif
802 l->append(L")");
803 }
745 }; 804 };
746 805
747 #if defined(OS_WIN) 806 #if defined(OS_WIN)
748 template <> 807 template <>
749 struct ParamTraits<XFORM> { 808 struct ParamTraits<XFORM> {
750 typedef XFORM param_type; 809 typedef XFORM param_type;
751 static void Write(Message* m, const param_type& p) { 810 static void Write(Message* m, const param_type& p) {
752 m->WriteData(reinterpret_cast<const char*>(&p), sizeof(XFORM)); 811 m->WriteData(reinterpret_cast<const char*>(&p), sizeof(XFORM));
753 } 812 }
754 static bool Read(const Message* m, void** iter, param_type* r) { 813 static bool Read(const Message* m, void** iter, param_type* r) {
(...skipping 205 matching lines...) Expand 10 before | Expand all | Expand 10 after
960 }; 1019 };
961 1020
962 //----------------------------------------------------------------------------- 1021 //-----------------------------------------------------------------------------
963 // Generic message subclasses 1022 // Generic message subclasses
964 1023
965 // Used for asynchronous messages. 1024 // Used for asynchronous messages.
966 template <class ParamType> 1025 template <class ParamType>
967 class MessageWithTuple : public Message { 1026 class MessageWithTuple : public Message {
968 public: 1027 public:
969 typedef ParamType Param; 1028 typedef ParamType Param;
970 typedef typename TupleTypes<ParamType>::ParamTuple RefParam; 1029 typedef typename ParamType::ParamTuple RefParam;
971 1030
972 // The constructor and the Read() method's templated implementations are in 1031 MessageWithTuple(int32 routing_id, uint32 type, const RefParam& p)
973 // ipc_message_utils_impl.h. The subclass constructor and Log() methods call 1032 : Message(routing_id, type, PRIORITY_NORMAL) {
974 // the templated versions of these and make sure there are instantiations in 1033 WriteParam(this, p);
975 // those translation units. 1034 }
976 MessageWithTuple(int32 routing_id, uint32 type, const RefParam& p);
977 1035
978 static bool Read(const Message* msg, Param* p) IPC_MSG_NOINLINE; 1036 static bool Read(const Message* msg, Param* p) {
1037 void* iter = NULL;
1038 if (ReadParam(msg, &iter, p))
1039 return true;
1040 NOTREACHED() << "Error deserializing message " << msg->type();
1041 return false;
1042 }
979 1043
980 // Generic dispatcher. Should cover most cases. 1044 // Generic dispatcher. Should cover most cases.
981 template<class T, class Method> 1045 template<class T, class Method>
982 static bool Dispatch(const Message* msg, T* obj, Method func) { 1046 static bool Dispatch(const Message* msg, T* obj, Method func) {
983 Param p; 1047 Param p;
984 if (Read(msg, &p)) { 1048 if (Read(msg, &p)) {
985 DispatchToMethod(obj, func, p); 1049 DispatchToMethod(obj, func, p);
986 return true; 1050 return true;
987 } 1051 }
988 return false; 1052 return false;
(...skipping 51 matching lines...) Expand 10 before | Expand all | Expand 10 after
1040 static bool Dispatch(const Message* msg, T* obj, 1104 static bool Dispatch(const Message* msg, T* obj,
1041 void (T::*func)(const Message&, TA, TB, TC, TD, TE)) { 1105 void (T::*func)(const Message&, TA, TB, TC, TD, TE)) {
1042 Param p; 1106 Param p;
1043 if (Read(msg, &p)) { 1107 if (Read(msg, &p)) {
1044 (obj->*func)(*msg, p.a, p.b, p.c, p.d, p.e); 1108 (obj->*func)(*msg, p.a, p.b, p.c, p.d, p.e);
1045 return true; 1109 return true;
1046 } 1110 }
1047 return false; 1111 return false;
1048 } 1112 }
1049 1113
1114 static void Log(const Message* msg, std::wstring* l) {
1115 Param p;
1116 if (Read(msg, &p))
1117 LogParam(p, l);
1118 }
1119
1050 // Functions used to do manual unpacking. Only used by the automation code, 1120 // Functions used to do manual unpacking. Only used by the automation code,
1051 // these should go away once that code uses SyncChannel. 1121 // these should go away once that code uses SyncChannel.
1052 template<typename TA, typename TB> 1122 template<typename TA, typename TB>
1053 static bool Read(const IPC::Message* msg, TA* a, TB* b) { 1123 static bool Read(const IPC::Message* msg, TA* a, TB* b) {
1054 ParamType params; 1124 ParamType params;
1055 if (!Read(msg, &params)) 1125 if (!Read(msg, &params))
1056 return false; 1126 return false;
1057 *a = params.a; 1127 *a = params.a;
1058 *b = params.b; 1128 *b = params.b;
1059 return true; 1129 return true;
(...skipping 29 matching lines...) Expand all
1089 return false; 1159 return false;
1090 *a = params.a; 1160 *a = params.a;
1091 *b = params.b; 1161 *b = params.b;
1092 *c = params.c; 1162 *c = params.c;
1093 *d = params.d; 1163 *d = params.d;
1094 *e = params.e; 1164 *e = params.e;
1095 return true; 1165 return true;
1096 } 1166 }
1097 }; 1167 };
1098 1168
1099 // defined in ipc_logging.cc
1100 void GenerateLogData(const std::string& channel, const Message& message,
1101 LogData* data);
1102
1103
1104 #if defined(IPC_MESSAGE_LOG_ENABLED)
1105 inline void AddOutputParamsToLog(const Message* msg, std::wstring* l) {
1106 const std::wstring& output_params = msg->output_params();
1107 if (!l->empty() && !output_params.empty())
1108 l->append(L", ");
1109
1110 l->append(output_params);
1111 }
1112
1113 template <class ReplyParamType>
1114 inline void LogReplyParamsToMessage(const ReplyParamType& reply_params,
1115 const Message* msg) {
1116 if (msg->received_time() != 0) {
1117 std::wstring output_params;
1118 LogParam(reply_params, &output_params);
1119 msg->set_output_params(output_params);
1120 }
1121 }
1122
1123 inline void ConnectMessageAndReply(const Message* msg, Message* reply) {
1124 if (msg->sent_time()) {
1125 // Don't log the sync message after dispatch, as we don't have the
1126 // output parameters at that point. Instead, save its data and log it
1127 // with the outgoing reply message when it's sent.
1128 LogData* data = new LogData;
1129 GenerateLogData("", *msg, data);
1130 msg->set_dont_log();
1131 reply->set_sync_log_data(data);
1132 }
1133 }
1134 #else
1135 inline void AddOutputParamsToLog(const Message* msg, std::wstring* l) {}
1136
1137 template <class ReplyParamType>
1138 inline void LogReplyParamsToMessage(const ReplyParamType& reply_params,
1139 const Message* msg) {}
1140
1141 inline void ConnectMessageAndReply(const Message* msg, Message* reply) {}
1142 #endif
1143
1144 // This class assumes that its template argument is a RefTuple (a Tuple with 1169 // This class assumes that its template argument is a RefTuple (a Tuple with
1145 // reference elements). This would go into ipc_message_utils_impl.h, but it is 1170 // reference elements).
1146 // also used by chrome_frame.
1147 template <class RefTuple> 1171 template <class RefTuple>
1148 class ParamDeserializer : public MessageReplyDeserializer { 1172 class ParamDeserializer : public MessageReplyDeserializer {
1149 public: 1173 public:
1150 explicit ParamDeserializer(const RefTuple& out) : out_(out) { } 1174 explicit ParamDeserializer(const RefTuple& out) : out_(out) { }
1151 1175
1152 bool SerializeOutputParameters(const IPC::Message& msg, void* iter) { 1176 bool SerializeOutputParameters(const IPC::Message& msg, void* iter) {
1153 return ReadParam(&msg, &iter, &out_); 1177 return ReadParam(&msg, &iter, &out_);
1154 } 1178 }
1155 1179
1156 RefTuple out_; 1180 RefTuple out_;
1157 }; 1181 };
1158 1182
1183 // defined in ipc_logging.cc
1184 void GenerateLogData(const std::string& channel, const Message& message,
1185 LogData* data);
1186
1159 // Used for synchronous messages. 1187 // Used for synchronous messages.
1160 template <class SendParamType, class ReplyParamType> 1188 template <class SendParamType, class ReplyParamType>
1161 class MessageWithReply : public SyncMessage { 1189 class MessageWithReply : public SyncMessage {
1162 public: 1190 public:
1163 typedef SendParamType SendParam; 1191 typedef SendParamType SendParam;
1164 typedef typename TupleTypes<SendParam>::ParamTuple RefSendParam; 1192 typedef typename SendParam::ParamTuple RefSendParam;
1165 typedef ReplyParamType ReplyParam; 1193 typedef ReplyParamType ReplyParam;
1166 1194
1167 MessageWithReply(int32 routing_id, uint32 type, 1195 MessageWithReply(int32 routing_id, uint32 type,
1168 const RefSendParam& send, const ReplyParam& reply); 1196 const RefSendParam& send, const ReplyParam& reply)
1169 static bool ReadSendParam(const Message* msg, SendParam* p) IPC_MSG_NOINLINE; 1197 : SyncMessage(routing_id, type, PRIORITY_NORMAL,
1170 static bool ReadReplyParam( 1198 new ParamDeserializer<ReplyParam>(reply)) {
1171 const Message* msg, 1199 WriteParam(this, send);
1172 typename TupleTypes<ReplyParam>::ValueTuple* p) IPC_MSG_NOINLINE; 1200 }
1201
1202 static void Log(const Message* msg, std::wstring* l) {
1203 if (msg->is_sync()) {
1204 SendParam p;
1205 void* iter = SyncMessage::GetDataIterator(msg);
1206 if (ReadParam(msg, &iter, &p))
1207 LogParam(p, l);
1208
1209 #if defined(IPC_MESSAGE_LOG_ENABLED)
1210 const std::wstring& output_params = msg->output_params();
1211 if (!l->empty() && !output_params.empty())
1212 l->append(L", ");
1213
1214 l->append(output_params);
1215 #endif
1216 } else {
1217 // This is an outgoing reply. Now that we have the output parameters, we
1218 // can finally log the message.
1219 typename ReplyParam::ValueTuple p;
1220 void* iter = SyncMessage::GetDataIterator(msg);
1221 if (ReadParam(msg, &iter, &p))
1222 LogParam(p, l);
1223 }
1224 }
1173 1225
1174 template<class T, class Method> 1226 template<class T, class Method>
1175 static bool Dispatch(const Message* msg, T* obj, Method func) { 1227 static bool Dispatch(const Message* msg, T* obj, Method func) {
1176 SendParam send_params; 1228 SendParam send_params;
1229 void* iter = GetDataIterator(msg);
1177 Message* reply = GenerateReply(msg); 1230 Message* reply = GenerateReply(msg);
1178 bool error; 1231 bool error;
1179 if (ReadSendParam(msg, &send_params)) { 1232 if (ReadParam(msg, &iter, &send_params)) {
1180 typename TupleTypes<ReplyParam>::ValueTuple reply_params; 1233 typename ReplyParam::ValueTuple reply_params;
1181 DispatchToMethod(obj, func, send_params, &reply_params); 1234 DispatchToMethod(obj, func, send_params, &reply_params);
1182 WriteParam(reply, reply_params); 1235 WriteParam(reply, reply_params);
1183 error = false; 1236 error = false;
1184 LogReplyParamsToMessage(reply_params, msg); 1237 #ifdef IPC_MESSAGE_LOG_ENABLED
1238 if (msg->received_time() != 0) {
1239 std::wstring output_params;
1240 LogParam(reply_params, &output_params);
1241 msg->set_output_params(output_params);
1242 }
1243 #endif
1185 } else { 1244 } else {
1186 NOTREACHED() << "Error deserializing message " << msg->type(); 1245 NOTREACHED() << "Error deserializing message " << msg->type();
1187 reply->set_reply_error(); 1246 reply->set_reply_error();
1188 error = true; 1247 error = true;
1189 } 1248 }
1190 1249
1191 obj->Send(reply); 1250 obj->Send(reply);
1192 return !error; 1251 return !error;
1193 } 1252 }
1194 1253
1195 template<class T, class Method> 1254 template<class T, class Method>
1196 static bool DispatchDelayReply(const Message* msg, T* obj, Method func) { 1255 static bool DispatchDelayReply(const Message* msg, T* obj, Method func) {
1197 SendParam send_params; 1256 SendParam send_params;
1257 void* iter = GetDataIterator(msg);
1198 Message* reply = GenerateReply(msg); 1258 Message* reply = GenerateReply(msg);
1199 bool error; 1259 bool error;
1200 if (ReadSendParam(msg, &send_params)) { 1260 if (ReadParam(msg, &iter, &send_params)) {
1201 Tuple1<Message&> t = MakeRefTuple(*reply); 1261 Tuple1<Message&> t = MakeRefTuple(*reply);
1202 ConnectMessageAndReply(msg, reply); 1262
1263 #ifdef IPC_MESSAGE_LOG_ENABLED
1264 if (msg->sent_time()) {
1265 // Don't log the sync message after dispatch, as we don't have the
1266 // output parameters at that point. Instead, save its data and log it
1267 // with the outgoing reply message when it's sent.
1268 LogData* data = new LogData;
1269 GenerateLogData("", *msg, data);
1270 msg->set_dont_log();
1271 reply->set_sync_log_data(data);
1272 }
1273 #endif
1203 DispatchToMethod(obj, func, send_params, &t); 1274 DispatchToMethod(obj, func, send_params, &t);
1204 error = false; 1275 error = false;
1205 } else { 1276 } else {
1206 NOTREACHED() << "Error deserializing message " << msg->type(); 1277 NOTREACHED() << "Error deserializing message " << msg->type();
1207 reply->set_reply_error(); 1278 reply->set_reply_error();
1208 obj->Send(reply); 1279 obj->Send(reply);
1209 error = true; 1280 error = true;
1210 } 1281 }
1211 return !error; 1282 return !error;
1212 } 1283 }
(...skipping 27 matching lines...) Expand all
1240 ReplyParam p(a, b, c, d, e); 1311 ReplyParam p(a, b, c, d, e);
1241 WriteParam(reply, p); 1312 WriteParam(reply, p);
1242 } 1313 }
1243 }; 1314 };
1244 1315
1245 //----------------------------------------------------------------------------- 1316 //-----------------------------------------------------------------------------
1246 1317
1247 } // namespace IPC 1318 } // namespace IPC
1248 1319
1249 #endif // IPC_IPC_MESSAGE_UTILS_H_ 1320 #endif // IPC_IPC_MESSAGE_UTILS_H_
OLDNEW
« no previous file with comments | « ipc/ipc_message_macros.h ('k') | ipc/ipc_message_utils.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698