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

Unified Diff: net/disk_cache/blockfile/stress_cache.cc

Issue 963163004: Move stress_cache to the tools directory (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: Created 5 years, 10 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
« no previous file with comments | « net/BUILD.gn ('k') | net/net.gyp » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: net/disk_cache/blockfile/stress_cache.cc
diff --git a/net/disk_cache/blockfile/stress_cache.cc b/net/disk_cache/blockfile/stress_cache.cc
deleted file mode 100644
index d43cb74b3cc844a44b0d531b4dc1207a1d8577d3..0000000000000000000000000000000000000000
--- a/net/disk_cache/blockfile/stress_cache.cc
+++ /dev/null
@@ -1,293 +0,0 @@
-// Copyright (c) 2012 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.
-
-// This is a simple application that stress-tests the crash recovery of the disk
-// cache. The main application starts a copy of itself on a loop, checking the
-// exit code of the child process. When the child dies in an unexpected way,
-// the main application quits.
-
-// The child application has two threads: one to exercise the cache in an
-// infinite loop, and another one to asynchronously kill the process.
-
-// A regular build should never crash.
-// To test that the disk cache doesn't generate critical errors with regular
-// application level crashes, edit stress_support.h.
-
-#include <string>
-#include <vector>
-
-#include "base/at_exit.h"
-#include "base/bind.h"
-#include "base/command_line.h"
-#include "base/debug/debugger.h"
-#include "base/files/file_path.h"
-#include "base/logging.h"
-#include "base/message_loop/message_loop.h"
-#include "base/path_service.h"
-#include "base/process/kill.h"
-#include "base/process/launch.h"
-#include "base/strings/string_number_conversions.h"
-#include "base/strings/string_util.h"
-#include "base/strings/utf_string_conversions.h"
-#include "base/threading/platform_thread.h"
-#include "base/threading/thread.h"
-#include "net/base/io_buffer.h"
-#include "net/base/net_errors.h"
-#include "net/base/test_completion_callback.h"
-#include "net/disk_cache/blockfile/backend_impl.h"
-#include "net/disk_cache/blockfile/stress_support.h"
-#include "net/disk_cache/blockfile/trace.h"
-#include "net/disk_cache/disk_cache.h"
-#include "net/disk_cache/disk_cache_test_util.h"
-
-#if defined(OS_WIN)
-#include "base/logging_win.h"
-#endif
-
-using base::Time;
-
-const int kError = -1;
-const int kExpectedCrash = 100;
-
-// Starts a new process.
-int RunSlave(int iteration) {
- base::FilePath exe;
- PathService::Get(base::FILE_EXE, &exe);
-
- base::CommandLine cmdline(exe);
- cmdline.AppendArg(base::IntToString(iteration));
-
- base::Process process = base::LaunchProcess(cmdline, base::LaunchOptions());
- if (!process.IsValid()) {
- printf("Unable to run test\n");
- return kError;
- }
-
- int exit_code;
- if (!process.WaitForExit(&exit_code)) {
- printf("Unable to get return code\n");
- return kError;
- }
- return exit_code;
-}
-
-// Main loop for the master process.
-int MasterCode() {
- for (int i = 0; i < 100000; i++) {
- int ret = RunSlave(i);
- if (kExpectedCrash != ret)
- return ret;
- }
-
- printf("More than enough...\n");
-
- return 0;
-}
-
-// -----------------------------------------------------------------------
-
-std::string GenerateStressKey() {
- char key[20 * 1024];
- size_t size = 50 + rand() % 20000;
- CacheTestFillBuffer(key, size, true);
-
- key[size - 1] = '\0';
- return std::string(key);
-}
-
-// This thread will loop forever, adding and removing entries from the cache.
-// iteration is the current crash cycle, so the entries on the cache are marked
-// to know which instance of the application wrote them.
-void StressTheCache(int iteration) {
- int cache_size = 0x2000000; // 32MB.
- uint32 mask = 0xfff; // 4096 entries.
-
- base::FilePath path;
- PathService::Get(base::DIR_TEMP, &path);
- path = path.AppendASCII("cache_test_stress");
-
- base::Thread cache_thread("CacheThread");
- if (!cache_thread.StartWithOptions(
- base::Thread::Options(base::MessageLoop::TYPE_IO, 0)))
- return;
-
- disk_cache::BackendImpl* cache =
- new disk_cache::BackendImpl(path, mask,
- cache_thread.message_loop_proxy().get(),
- NULL);
- cache->SetMaxSize(cache_size);
- cache->SetFlags(disk_cache::kNoLoadProtection);
-
- net::TestCompletionCallback cb;
- int rv = cache->Init(cb.callback());
-
- if (cb.GetResult(rv) != net::OK) {
- printf("Unable to initialize cache.\n");
- return;
- }
- printf("Iteration %d, initial entries: %d\n", iteration,
- cache->GetEntryCount());
-
- int seed = static_cast<int>(Time::Now().ToInternalValue());
- srand(seed);
-
- // kNumKeys is meant to be enough to have about 3x or 4x iterations before
- // the process crashes.
-#ifdef NDEBUG
- const int kNumKeys = 4000;
-#else
- const int kNumKeys = 1200;
-#endif
- const int kNumEntries = 30;
- std::string keys[kNumKeys];
- disk_cache::Entry* entries[kNumEntries] = {0};
-
- for (int i = 0; i < kNumKeys; i++) {
- keys[i] = GenerateStressKey();
- }
-
- const int kSize = 20000;
- scoped_refptr<net::IOBuffer> buffer(new net::IOBuffer(kSize));
- memset(buffer->data(), 'k', kSize);
-
- for (int i = 0;; i++) {
- int slot = rand() % kNumEntries;
- int key = rand() % kNumKeys;
- bool truncate = (rand() % 2 == 0);
- int size = kSize - (rand() % 20) * kSize / 20;
-
- if (entries[slot])
- entries[slot]->Close();
-
- net::TestCompletionCallback cb;
- rv = cache->OpenEntry(keys[key], &entries[slot], cb.callback());
- if (cb.GetResult(rv) != net::OK) {
- rv = cache->CreateEntry(keys[key], &entries[slot], cb.callback());
- CHECK_EQ(net::OK, cb.GetResult(rv));
- }
-
- base::snprintf(buffer->data(), kSize,
- "i: %d iter: %d, size: %d, truncate: %d ", i, iteration,
- size, truncate ? 1 : 0);
- rv = entries[slot]->WriteData(0, 0, buffer.get(), size, cb.callback(),
- truncate);
- CHECK_EQ(size, cb.GetResult(rv));
-
- if (rand() % 100 > 80) {
- key = rand() % kNumKeys;
- net::TestCompletionCallback cb2;
- rv = cache->DoomEntry(keys[key], cb2.callback());
- cb2.GetResult(rv);
- }
-
- if (!(i % 100))
- printf("Entries: %d \r", i);
- }
-}
-
-// We want to prevent the timer thread from killing the process while we are
-// waiting for the debugger to attach.
-bool g_crashing = false;
-
-// RunSoon() and CrashCallback() reference each other, unfortunately.
-void RunSoon(base::MessageLoop* target_loop);
-
-void CrashCallback() {
- // Keep trying to run.
- RunSoon(base::MessageLoop::current());
-
- if (g_crashing)
- return;
-
- if (rand() % 100 > 30) {
- printf("sweet death...\n");
-#if defined(OS_WIN)
- // Windows does more work on _exit() that we would like, so we use Kill.
- base::KillProcessById(base::GetCurrentProcId(), kExpectedCrash, false);
-#elif defined(OS_POSIX)
- // On POSIX, _exit() will terminate the process with minimal cleanup,
- // and it is cleaner than killing.
- _exit(kExpectedCrash);
-#endif
- }
-}
-
-void RunSoon(base::MessageLoop* target_loop) {
- const base::TimeDelta kTaskDelay = base::TimeDelta::FromSeconds(10);
- target_loop->PostDelayedTask(
- FROM_HERE, base::Bind(&CrashCallback), kTaskDelay);
-}
-
-// We leak everything here :)
-bool StartCrashThread() {
- base::Thread* thread = new base::Thread("party_crasher");
- if (!thread->Start())
- return false;
-
- RunSoon(thread->message_loop());
- return true;
-}
-
-void CrashHandler(const std::string& str) {
- g_crashing = true;
- base::debug::BreakDebugger();
-}
-
-bool MessageHandler(int severity, const char* file, int line,
- size_t message_start, const std::string& str) {
- const size_t kMaxMessageLen = 48;
- char message[kMaxMessageLen];
- size_t len = std::min(str.length() - message_start, kMaxMessageLen - 1);
-
- memcpy(message, str.c_str() + message_start, len);
- message[len] = '\0';
-#if !defined(DISK_CACHE_TRACE_TO_LOG)
- disk_cache::Trace("%s", message);
-#endif
- return false;
-}
-
-// -----------------------------------------------------------------------
-
-#if defined(OS_WIN)
-// {B9A153D4-31C3-48e4-9ABF-D54383F14A0D}
-const GUID kStressCacheTraceProviderName = {
- 0xb9a153d4, 0x31c3, 0x48e4,
- { 0x9a, 0xbf, 0xd5, 0x43, 0x83, 0xf1, 0x4a, 0xd } };
-#endif
-
-int main(int argc, const char* argv[]) {
- // Setup an AtExitManager so Singleton objects will be destructed.
- base::AtExitManager at_exit_manager;
-
- if (argc < 2)
- return MasterCode();
-
- logging::SetLogAssertHandler(CrashHandler);
- logging::SetLogMessageHandler(MessageHandler);
-
-#if defined(OS_WIN)
- logging::LogEventProvider::Initialize(kStressCacheTraceProviderName);
-#else
- base::CommandLine::Init(argc, argv);
- logging::LoggingSettings settings;
- settings.logging_dest = logging::LOG_TO_SYSTEM_DEBUG_LOG;
- logging::InitLogging(settings);
-#endif
-
- // Some time for the memory manager to flush stuff.
- base::PlatformThread::Sleep(base::TimeDelta::FromSeconds(3));
- base::MessageLoopForIO message_loop;
-
- char* end;
- long int iteration = strtol(argv[1], &end, 0);
-
- if (!StartCrashThread()) {
- printf("failed to start thread\n");
- return kError;
- }
-
- StressTheCache(iteration);
- return 0;
-}
« no previous file with comments | « net/BUILD.gn ('k') | net/net.gyp » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698