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

Unified Diff: tools/ipc_fuzzer/ipc_fuzzer_main.cc

Issue 18254010: IPC fuzzer child process component (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: addressed comments Created 7 years, 3 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 side-by-side diff with in-line comments
Download patch
Index: tools/ipc_fuzzer/ipc_fuzzer_main.cc
diff --git a/tools/ipc_fuzzer/ipc_fuzzer_main.cc b/tools/ipc_fuzzer/ipc_fuzzer_main.cc
new file mode 100644
index 0000000000000000000000000000000000000000..6911e857c0ad7637848208160ffeba0e846c4bf7
--- /dev/null
+++ b/tools/ipc_fuzzer/ipc_fuzzer_main.cc
@@ -0,0 +1,204 @@
+// Copyright 2013 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.
+
+#include <limits.h>
+#include <stdio.h>
+#include <vector>
+
+#include "base/bind.h"
+#include "base/command_line.h"
+#include "base/files/memory_mapped_file.h"
+#include "base/logging.h"
+#include "base/memory/scoped_ptr.h"
+#include "base/message_loop/message_loop.h"
+#include "base/posix/global_descriptors.h"
+#include "base/synchronization/waitable_event.h"
+#include "base/threading/thread.h"
+#include "base/timer/timer.h"
+#include "chrome/common/chrome_switches.h"
+#include "ipc/ipc_channel_proxy.h"
+#include "ipc/ipc_descriptors.h"
+#include "ipc/ipc_listener.h"
+#include "ipc/ipc_message.h"
+#include "ipc/ipc_platform_file.h"
+#include "ipc/ipc_switches.h"
+
+namespace {
+
+class IpcFuzzerProcess : public IPC::Listener {
+ public:
+ IpcFuzzerProcess();
+ virtual ~IpcFuzzerProcess();
+ void Initialize(int argc, const char **argv);
jochen (gone - plz use gerrit) 2013/10/31 10:47:00 please add comments explaining what the methods do
aedla 2013/11/26 17:09:50 Done.
+ void OpenChannel();
+ bool OpenTestcase();
+ void StartSendingMessages();
+
+ virtual bool OnMessageReceived(const IPC::Message& message) OVERRIDE;
jochen (gone - plz use gerrit) 2013/10/31 10:47:00 nit. add // IPC::Listener implementation. comment
aedla 2013/11/26 17:09:50 Done.
+ virtual void OnChannelError() OVERRIDE;
+
+ private:
+ bool ExtractMessages(const char *data, size_t len);
+ IPC::Message* GetNextMessage();
+ void SendNextMessage();
+
+ scoped_ptr<IPC::ChannelProxy> channel_;
+ base::MessageLoop main_loop_;
+ base::Thread io_thread_;
jochen (gone - plz use gerrit) 2013/10/31 10:47:00 during shutdown, the io thread will die before tha
aedla 2013/11/26 17:09:50 Nice catch. I added channel_.reset() to the destru
+ base::WaitableEvent shutdown_event_;
+ scoped_ptr<base::Timer> timer_;
+ scoped_ptr<base::MemoryMappedFile> testcase_map_;
+ std::vector<scoped_ptr<IPC::Message>> messages_;
+ size_t current_message_;
+
+ DISALLOW_COPY_AND_ASSIGN(IpcFuzzerProcess);
+};
+
+IpcFuzzerProcess::IpcFuzzerProcess()
+ : main_loop_(base::MessageLoop::TYPE_DEFAULT),
+ io_thread_("Chrome_ChildIOThread"),
+ shutdown_event_(true, false),
+ current_message_(0) {
+}
+
+IpcFuzzerProcess::~IpcFuzzerProcess() {
+}
+
+void IpcFuzzerProcess::Initialize(int argc, const char **argv) {
+ CommandLine::Init(argc, argv);
+
+ // Log to default destination.
+ logging::SetMinLogLevel(logging::LOG_ERROR);
+ logging::InitLogging(logging::LoggingSettings());
+
+ io_thread_.StartWithOptions(
+ base::Thread::Options(base::MessageLoop::TYPE_IO, 0));
+
+#if defined(OS_POSIX)
+ base::GlobalDescriptors* g_fds = base::GlobalDescriptors::GetInstance();
+ g_fds->Set(kPrimaryIPCChannel,
+ kPrimaryIPCChannel + base::GlobalDescriptors::kBaseDescriptor);
+#endif
+}
+
+void IpcFuzzerProcess::OpenChannel() {
+ std::string channel_name =
+ CommandLine::ForCurrentProcess()->GetSwitchValueASCII(
+ switches::kProcessChannelID);
+
+ channel_.reset(
+ new IPC::ChannelProxy(channel_name,
+ IPC::Channel::MODE_CLIENT,
+ this,
+ io_thread_.message_loop_proxy()));
+}
+
+bool IpcFuzzerProcess::ExtractMessages(const char *data, size_t len) {
+ const char* end = data + len;
+
+ while (data < end) {
+ const char* message_tail = IPC::Message::FindNext(data, end);
+ if (!message_tail)
+ break;
+
+ size_t len = message_tail - data;
+ if (len > INT_MAX)
+ break;
+
+ IPC::Message* message = new IPC::Message(data, len);
+ data = message_tail;
+
+ messages_.resize(messages_.size() + 1);
+ messages_.back().reset(message);
+ }
+
+ if (data < end) {
+ unsigned long left = end - data;
+ LOG(ERROR) << left << " bytes left while extracting messages";
+ return false;
+ }
+
+ return true;
+}
+
+bool IpcFuzzerProcess::OpenTestcase() {
+ const CommandLine& command_line = *CommandLine::ForCurrentProcess();
+
+ if (!command_line.HasSwitch(switches::kIpcFuzzerTestcase)) {
+ LOG(ERROR) << "No IPC fuzzer testcase specified";
+ return false;
+ }
+
+ base::FilePath path =
+ command_line.GetSwitchValuePath(switches::kIpcFuzzerTestcase);
+ testcase_map_.reset(new base::MemoryMappedFile());
+ if (!testcase_map_->Initialize(path)) {
+ LOG(ERROR) << "Failed to map testcase: " << path.value();
+ return false;
+ }
+
+ const char* data = reinterpret_cast<const char *>(testcase_map_->data());
+ size_t len = testcase_map_->length();
+
+ return ExtractMessages(data, len);
+}
+
+IPC::Message* IpcFuzzerProcess::GetNextMessage() {
+ if (current_message_ == messages_.size())
+ return NULL;
+
+ LOG(INFO) << "Sending message "
+ << current_message_ << "/"
+ << messages_.size();
+
+ IPC::Message* const_message = messages_[current_message_].get();
+ IPC::Message* message = new IPC::Message(*const_message);
+ static_cast<uint16*>(const_cast<void*>(message->data()))[8] = 0;
jochen (gone - plz use gerrit) 2013/10/31 10:47:00 what if data is too small?
aedla 2013/11/26 17:09:50 It will actually be large enough, IPC::Message::Fi
+ ++current_message_;
+
+ return message;
+}
+
+void IpcFuzzerProcess::SendNextMessage() {
+ IPC::Message* message = GetNextMessage();
+ if (!message) {
+ base::MessageLoop::current()->Quit();
+ return;
+ }
+
+ channel_->Send(message);
+}
+
+void IpcFuzzerProcess::StartSendingMessages() {
+ timer_.reset(new base::Timer(false, true));
+ timer_->Start(FROM_HERE,
+ base::TimeDelta::FromMilliseconds(1),
jochen (gone - plz use gerrit) 2013/10/31 10:47:00 nit. indentation
aedla 2013/11/26 17:09:50 Done.
+ base::Bind(&IpcFuzzerProcess::SendNextMessage,
+ base::Unretained(this)));
+}
+
+bool IpcFuzzerProcess::OnMessageReceived(const IPC::Message& msg) {
+ return true;
+}
+
+void IpcFuzzerProcess::OnChannelError() {
+ LOG(ERROR) << "Channel error, quitting";
+ base::MessageLoop::current()->Quit();
+}
+
+} // namespace
+
+int main(int argc, const char **argv) {
+ IpcFuzzerProcess fuzzer;
+ fuzzer.Initialize(argc, argv);
+ fuzzer.OpenChannel();
+
+ if (!fuzzer.OpenTestcase())
+ return 0;
+
+ fuzzer.StartSendingMessages();
+
+ base::MessageLoop::current()->Run();
+ return 0;
+}

Powered by Google App Engine
This is Rietveld 408576698