stress_cache.cc revision a1401311d1ab56c4ed0a474bd38c108f75cb0cd9
1// Copyright (c) 2012 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5// This is a simple application that stress-tests the crash recovery of the disk
6// cache. The main application starts a copy of itself on a loop, checking the
7// exit code of the child process. When the child dies in an unexpected way,
8// the main application quits.
9
10// The child application has two threads: one to exercise the cache in an
11// infinite loop, and another one to asynchronously kill the process.
12
13// A regular build should never crash.
14// To test that the disk cache doesn't generate critical errors with regular
15// application level crashes, edit stress_support.h.
16
17#include <string>
18#include <vector>
19
20#include "base/at_exit.h"
21#include "base/bind.h"
22#include "base/command_line.h"
23#include "base/debug/debugger.h"
24#include "base/files/file_path.h"
25#include "base/logging.h"
26#include "base/message_loop/message_loop.h"
27#include "base/path_service.h"
28#include "base/process/kill.h"
29#include "base/process/launch.h"
30#include "base/process/process_handle.h"
31#include "base/strings/string_number_conversions.h"
32#include "base/strings/string_util.h"
33#include "base/strings/utf_string_conversions.h"
34#include "base/threading/platform_thread.h"
35#include "base/threading/thread.h"
36#include "net/base/io_buffer.h"
37#include "net/base/net_errors.h"
38#include "net/base/test_completion_callback.h"
39#include "net/disk_cache/blockfile/backend_impl.h"
40#include "net/disk_cache/blockfile/stress_support.h"
41#include "net/disk_cache/blockfile/trace.h"
42#include "net/disk_cache/disk_cache.h"
43#include "net/disk_cache/disk_cache_test_util.h"
44
45#if defined(OS_WIN)
46#include "base/logging_win.h"
47#endif
48
49using base::Time;
50
51const int kError = -1;
52const int kExpectedCrash = 100;
53
54// Starts a new process.
55int RunSlave(int iteration) {
56  base::FilePath exe;
57  PathService::Get(base::FILE_EXE, &exe);
58
59  CommandLine cmdline(exe);
60  cmdline.AppendArg(base::IntToString(iteration));
61
62  base::ProcessHandle handle;
63  if (!base::LaunchProcess(cmdline, base::LaunchOptions(), &handle)) {
64    printf("Unable to run test\n");
65    return kError;
66  }
67
68  int exit_code;
69  if (!base::WaitForExitCode(handle, &exit_code)) {
70    printf("Unable to get return code\n");
71    return kError;
72  }
73  return exit_code;
74}
75
76// Main loop for the master process.
77int MasterCode() {
78  for (int i = 0; i < 100000; i++) {
79    int ret = RunSlave(i);
80    if (kExpectedCrash != ret)
81      return ret;
82  }
83
84  printf("More than enough...\n");
85
86  return 0;
87}
88
89// -----------------------------------------------------------------------
90
91std::string GenerateStressKey() {
92  char key[20 * 1024];
93  size_t size = 50 + rand() % 20000;
94  CacheTestFillBuffer(key, size, true);
95
96  key[size - 1] = '\0';
97  return std::string(key);
98}
99
100// This thread will loop forever, adding and removing entries from the cache.
101// iteration is the current crash cycle, so the entries on the cache are marked
102// to know which instance of the application wrote them.
103void StressTheCache(int iteration) {
104  int cache_size = 0x2000000;  // 32MB.
105  uint32 mask = 0xfff;  // 4096 entries.
106
107  base::FilePath path;
108  PathService::Get(base::DIR_TEMP, &path);
109  path = path.AppendASCII("cache_test_stress");
110
111  base::Thread cache_thread("CacheThread");
112  if (!cache_thread.StartWithOptions(
113          base::Thread::Options(base::MessageLoop::TYPE_IO, 0)))
114    return;
115
116  disk_cache::BackendImpl* cache =
117      new disk_cache::BackendImpl(path, mask,
118                                  cache_thread.message_loop_proxy().get(),
119                                  NULL);
120  cache->SetMaxSize(cache_size);
121  cache->SetFlags(disk_cache::kNoLoadProtection);
122
123  net::TestCompletionCallback cb;
124  int rv = cache->Init(cb.callback());
125
126  if (cb.GetResult(rv) != net::OK) {
127    printf("Unable to initialize cache.\n");
128    return;
129  }
130  printf("Iteration %d, initial entries: %d\n", iteration,
131         cache->GetEntryCount());
132
133  int seed = static_cast<int>(Time::Now().ToInternalValue());
134  srand(seed);
135
136  // kNumKeys is meant to be enough to have about 3x or 4x iterations before
137  // the process crashes.
138#ifdef NDEBUG
139  const int kNumKeys = 4000;
140#else
141  const int kNumKeys = 1200;
142#endif
143  const int kNumEntries = 30;
144  std::string keys[kNumKeys];
145  disk_cache::Entry* entries[kNumEntries] = {0};
146
147  for (int i = 0; i < kNumKeys; i++) {
148    keys[i] = GenerateStressKey();
149  }
150
151  const int kSize = 20000;
152  scoped_refptr<net::IOBuffer> buffer(new net::IOBuffer(kSize));
153  memset(buffer->data(), 'k', kSize);
154
155  for (int i = 0;; i++) {
156    int slot = rand() % kNumEntries;
157    int key = rand() % kNumKeys;
158    bool truncate = (rand() % 2 == 0);
159    int size = kSize - (rand() % 20) * kSize / 20;
160
161    if (entries[slot])
162      entries[slot]->Close();
163
164    net::TestCompletionCallback cb;
165    rv = cache->OpenEntry(keys[key], &entries[slot], cb.callback());
166    if (cb.GetResult(rv) != net::OK) {
167      rv = cache->CreateEntry(keys[key], &entries[slot], cb.callback());
168      CHECK_EQ(net::OK, cb.GetResult(rv));
169    }
170
171    base::snprintf(buffer->data(), kSize,
172                   "i: %d iter: %d, size: %d, truncate: %d     ", i, iteration,
173                   size, truncate ? 1 : 0);
174    rv = entries[slot]->WriteData(0, 0, buffer.get(), size, cb.callback(),
175                                  truncate);
176    CHECK_EQ(size, cb.GetResult(rv));
177
178    if (rand() % 100 > 80) {
179      key = rand() % kNumKeys;
180      net::TestCompletionCallback cb2;
181      rv = cache->DoomEntry(keys[key], cb2.callback());
182      cb2.GetResult(rv);
183    }
184
185    if (!(i % 100))
186      printf("Entries: %d    \r", i);
187  }
188}
189
190// We want to prevent the timer thread from killing the process while we are
191// waiting for the debugger to attach.
192bool g_crashing = false;
193
194// RunSoon() and CrashCallback() reference each other, unfortunately.
195void RunSoon(base::MessageLoop* target_loop);
196
197void CrashCallback() {
198  // Keep trying to run.
199  RunSoon(base::MessageLoop::current());
200
201  if (g_crashing)
202    return;
203
204  if (rand() % 100 > 30) {
205    printf("sweet death...\n");
206#if defined(OS_WIN)
207    // Windows does more work on _exit() that we would like, so we use Kill.
208    base::KillProcessById(base::GetCurrentProcId(), kExpectedCrash, false);
209#elif defined(OS_POSIX)
210    // On POSIX, _exit() will terminate the process with minimal cleanup,
211    // and it is cleaner than killing.
212    _exit(kExpectedCrash);
213#endif
214  }
215}
216
217void RunSoon(base::MessageLoop* target_loop) {
218  const base::TimeDelta kTaskDelay = base::TimeDelta::FromSeconds(10);
219  target_loop->PostDelayedTask(
220      FROM_HERE, base::Bind(&CrashCallback), kTaskDelay);
221}
222
223// We leak everything here :)
224bool StartCrashThread() {
225  base::Thread* thread = new base::Thread("party_crasher");
226  if (!thread->Start())
227    return false;
228
229  RunSoon(thread->message_loop());
230  return true;
231}
232
233void CrashHandler(const std::string& str) {
234  g_crashing = true;
235  base::debug::BreakDebugger();
236}
237
238bool MessageHandler(int severity, const char* file, int line,
239                    size_t message_start, const std::string& str) {
240  const size_t kMaxMessageLen = 48;
241  char message[kMaxMessageLen];
242  size_t len = std::min(str.length() - message_start, kMaxMessageLen - 1);
243
244  memcpy(message, str.c_str() + message_start, len);
245  message[len] = '\0';
246#if !defined(DISK_CACHE_TRACE_TO_LOG)
247  disk_cache::Trace("%s", message);
248#endif
249  return false;
250}
251
252// -----------------------------------------------------------------------
253
254#if defined(OS_WIN)
255// {B9A153D4-31C3-48e4-9ABF-D54383F14A0D}
256const GUID kStressCacheTraceProviderName = {
257    0xb9a153d4, 0x31c3, 0x48e4,
258        { 0x9a, 0xbf, 0xd5, 0x43, 0x83, 0xf1, 0x4a, 0xd } };
259#endif
260
261int main(int argc, const char* argv[]) {
262  // Setup an AtExitManager so Singleton objects will be destructed.
263  base::AtExitManager at_exit_manager;
264
265  if (argc < 2)
266    return MasterCode();
267
268  logging::SetLogAssertHandler(CrashHandler);
269  logging::SetLogMessageHandler(MessageHandler);
270
271#if defined(OS_WIN)
272  logging::LogEventProvider::Initialize(kStressCacheTraceProviderName);
273#else
274  CommandLine::Init(argc, argv);
275  logging::LoggingSettings settings;
276  settings.logging_dest = logging::LOG_TO_SYSTEM_DEBUG_LOG;
277  logging::InitLogging(settings);
278#endif
279
280  // Some time for the memory manager to flush stuff.
281  base::PlatformThread::Sleep(base::TimeDelta::FromSeconds(3));
282  base::MessageLoopForIO message_loop;
283
284  char* end;
285  long int iteration = strtol(argv[1], &end, 0);
286
287  if (!StartCrashThread()) {
288    printf("failed to start thread\n");
289    return kError;
290  }
291
292  StressTheCache(iteration);
293  return 0;
294}
295