blob: 64a191a1c20a196d07d81f18913955da67b7be2d [file] [log] [blame]
// Copyright 2010 Google Inc. All Rights Reserved.
#ifndef TALK_BASE_WIN32TOOLHELP_H_
#define TALK_BASE_WIN32TOOLHELP_H_
#ifndef WIN32
#error WIN32 Only
#endif
#include "talk/base/win32.h"
// Should be included first, but that causes redefinitions.
#include <tlhelp32.h>
#include "talk/base/constructormagic.h"
namespace talk_base {
// The toolhelp api used to enumerate processes and their modules
// on Windows is very repetetive and clunky to use. This little
// template wraps it to make it a little more programmer friendly.
//
// Traits: Traits type that adapts the enumerator to the corresponding
// win32 toolhelp api. Each traits class need to:
// - define the type of the enumerated data as a public symbol Type
//
// - implement bool First(HANDLE, T*) normally calls a
// Xxxx32First method in the toolhelp API. Ex Process32First(...)
//
// - implement bool Next(HANDLE, T*) normally calls a
// Xxxx32Next method in the toolhelp API. Ex Process32Next(...)
//
// - implement bool CloseHandle(HANDLE)
//
template<typename Traits>
class ToolhelpEnumeratorBase {
public:
ToolhelpEnumeratorBase(HANDLE snapshot)
: snapshot_(snapshot), broken_(false), first_(true) {
// Clear out the Traits::Type structure instance.
Zero(&current_);
}
virtual ~ToolhelpEnumeratorBase() {
Close();
}
// Moves forward to the next object using the First and Next
// pointers. If either First or Next ever indicates an failure
// all subsequent calls to this method will fail; the enumerator
// object is considered broken.
bool Next() {
if (!Valid()) {
return false;
}
// Move the iteration forward.
current_.dwSize = sizeof(typename Traits::Type);
bool incr_ok = false;
if (first_) {
incr_ok = Traits::First(snapshot_, &current_);
first_ = false;
} else {
incr_ok = Traits::Next(snapshot_, &current_);
}
if (!incr_ok) {
Zero(&current_);
broken_ = true;
}
return incr_ok;
}
const typename Traits::Type& current() const {
return current_;
}
void Close() {
if (snapshot_ != INVALID_HANDLE_VALUE) {
Traits::CloseHandle(snapshot_);
snapshot_ = INVALID_HANDLE_VALUE;
}
}
private:
// Checks the state of the snapshot handle.
bool Valid() {
return snapshot_ != INVALID_HANDLE_VALUE && !broken_;
}
static void Zero(typename Traits::Type* buff) {
ZeroMemory(buff, sizeof(typename Traits::Type));
}
HANDLE snapshot_;
typename Traits::Type current_;
bool broken_;
bool first_;
};
class ToolhelpTraits {
public:
static HANDLE CreateSnapshot(uint32 flags, uint32 process_id) {
return CreateToolhelp32Snapshot(flags, process_id);
}
static bool CloseHandle(HANDLE handle) {
return ::CloseHandle(handle) == TRUE;
}
};
class ToolhelpProcessTraits : public ToolhelpTraits {
public:
typedef PROCESSENTRY32 Type;
static bool First(HANDLE handle, Type* t) {
return ::Process32First(handle, t) == TRUE;
}
static bool Next(HANDLE handle, Type* t) {
return ::Process32Next(handle, t) == TRUE;
}
};
class ProcessEnumerator : public ToolhelpEnumeratorBase<ToolhelpProcessTraits> {
public:
ProcessEnumerator()
: ToolhelpEnumeratorBase(
ToolhelpProcessTraits::CreateSnapshot(TH32CS_SNAPPROCESS, 0)) {
}
private:
DISALLOW_EVIL_CONSTRUCTORS(ProcessEnumerator);
};
class ToolhelpModuleTraits : public ToolhelpTraits {
public:
typedef MODULEENTRY32 Type;
static bool First(HANDLE handle, Type* t) {
return ::Module32First(handle, t) == TRUE;
}
static bool Next(HANDLE handle, Type* t) {
return ::Module32Next(handle, t) == TRUE;
}
};
class ModuleEnumerator : public ToolhelpEnumeratorBase<ToolhelpModuleTraits> {
public:
explicit ModuleEnumerator(uint32 process_id)
: ToolhelpEnumeratorBase(
ToolhelpModuleTraits::CreateSnapshot(TH32CS_SNAPMODULE,
process_id)) {
}
private:
DISALLOW_EVIL_CONSTRUCTORS(ModuleEnumerator);
};
} // namespace talk_base
#endif // TALK_BASE_WIN32TOOLHELP_H_