sigar/src/os/win32/peb.c

165 lines
4.1 KiB
C
Raw Normal View History

2006-07-16 01:46:36 +08:00
/*
* Copyright (C) [2004, 2005, 2006], Hyperic, Inc.
* This file is part of SIGAR.
*
* SIGAR is free software; you can redistribute it and/or modify
* it under the terms version 2 of the GNU General Public License as
* published by the Free Software Foundation. This program is distributed
* in the hope that it will be useful, but WITHOUT ANY WARRANTY; without
* even the implied warranty of MERCHANTABILITY or FITNESS FOR A
* PARTICULAR PURPOSE. See the GNU General Public License for more
* details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
* USA.
*/
2004-06-22 06:37:04 +08:00
/*
* functions for getting info from the Process Environment Block
*/
#define UNICODE
#define _UNICODE
#include "sigar.h"
#include "sigar_private.h"
#include "sigar_os.h"
#include <shellapi.h>
2004-06-22 06:37:04 +08:00
2004-08-02 04:31:42 +08:00
#define PAGE_START 0x00020000
#define CWD_OFFSET PAGE_START + 0x0290
#define PATH_OFFSET PAGE_START + 0x0498
#define START_ADDRESS PAGE_START + 0x0498
2004-06-22 06:37:04 +08:00
static int sigar_peb_get(sigar_t *sigar, HANDLE proc, DWORD *base)
{
MEMORY_BASIC_INFORMATION mbi;
DWORD bytes;
SIZE_T size = sigar->pagesize;
2004-06-22 06:37:04 +08:00
if (!sigar->peb) {
sigar->peb = malloc(sigar->pagesize*2);
2004-06-22 06:37:04 +08:00
}
2004-08-02 04:31:42 +08:00
if (!VirtualQueryEx(proc, (char*)START_ADDRESS, &mbi, sizeof(mbi))) {
2004-06-22 06:37:04 +08:00
return GetLastError();
}
if (mbi.RegionSize > sigar->pagesize) {
/* in the event args crosses the first page boundry.
* seen with WebSphere.
*/
size *= 2;
}
2004-06-22 06:37:04 +08:00
if (!ReadProcessMemory(proc, mbi.BaseAddress, sigar->peb,
size, &bytes))
2004-06-22 06:37:04 +08:00
{
return GetLastError();
}
*base = (DWORD)mbi.BaseAddress;
return SIGAR_OK;
}
2004-08-02 04:31:42 +08:00
#define SKIP_NULL(scratch) \
if (*scratch == '\0') scratch += sizeof(WCHAR)
#define PEB_START(scratch, base, offset) \
scratch = sigar->peb + ((DWORD)offset - base)
2004-06-22 06:37:04 +08:00
2004-07-29 05:47:14 +08:00
//point scratch to next string (assumes PEB_FIRST)
#define PEB_NEXT(scratch) \
2004-08-02 04:31:42 +08:00
scratch = scratch + (wcslen((LPWSTR)scratch) + 1) * sizeof(WCHAR); \
SKIP_NULL(scratch)
2004-06-22 06:37:04 +08:00
2004-08-02 04:31:42 +08:00
int sigar_proc_exe_peb_get(sigar_t *sigar, HANDLE proc,
sigar_proc_exe_t *procexe)
2004-06-22 06:37:04 +08:00
{
int status;
LPBYTE scratch;
DWORD base;
WCHAR buf[MAX_PATH];
if ((status = sigar_peb_get(sigar, proc, &base)) != SIGAR_OK) {
return status;
}
2004-08-02 04:31:42 +08:00
PEB_START(scratch, base, CWD_OFFSET);
2004-06-22 06:37:04 +08:00
2004-08-02 04:31:42 +08:00
wcsncpy(buf, (LPWSTR)scratch, MAX_PATH);
buf[MAX_PATH-1] = L'\0';
2004-06-22 06:37:04 +08:00
2004-08-02 04:31:42 +08:00
SIGAR_W2A(buf, procexe->cwd, sizeof(procexe->cwd));
PEB_START(scratch, base, PATH_OFFSET);
PEB_NEXT(scratch); //skip PATH
2004-06-22 06:37:04 +08:00
/* XXX seen on non-english windows, random leading char */
if (*(scratch + sizeof(WCHAR)) != L':') {
scratch += sizeof(WCHAR);
}
2004-06-22 06:37:04 +08:00
wcsncpy(buf, (LPWSTR)scratch, MAX_PATH);
buf[MAX_PATH-1] = L'\0';
2004-08-02 04:31:42 +08:00
SIGAR_W2A(buf, procexe->name, sizeof(procexe->name));
2004-06-22 06:37:04 +08:00
return SIGAR_OK;
}
2004-07-29 05:47:14 +08:00
int sigar_parse_proc_args(sigar_t *sigar, WCHAR *buf,
sigar_proc_args_t *procargs)
{
char arg[SIGAR_CMDLINE_MAX];
LPWSTR *args;
int num, i;
if (!buf) {
buf = GetCommandLine();
}
args = CommandLineToArgvW(buf, &num);
if (args == NULL) {
return SIGAR_OK;
}
for (i=0; i<num; i++) {
SIGAR_W2A(args[i], arg, SIGAR_CMDLINE_MAX);
SIGAR_PROC_ARGS_GROW(procargs);
2007-06-27 09:10:12 +08:00
procargs->data[procargs->number++] = sigar_strdup(arg);
}
GlobalFree(args);
return SIGAR_OK;
}
int sigar_proc_args_peb_get(sigar_t *sigar, HANDLE proc,
sigar_proc_args_t *procargs)
2004-07-29 05:47:14 +08:00
{
int status;
LPBYTE scratch;
DWORD base;
2004-08-03 10:45:26 +08:00
WCHAR buf[SIGAR_CMDLINE_MAX];
2004-07-29 05:47:14 +08:00
if ((status = sigar_peb_get(sigar, proc, &base)) != SIGAR_OK) {
return status;
}
2004-08-02 04:31:42 +08:00
PEB_START(scratch, base, PATH_OFFSET);
2004-07-29 05:47:14 +08:00
2004-08-02 04:31:42 +08:00
PEB_NEXT(scratch); //skip PATH
2004-07-29 05:47:14 +08:00
2004-08-02 04:31:42 +08:00
PEB_NEXT(scratch); //skip exe name
2004-07-29 05:47:14 +08:00
2004-08-03 10:45:26 +08:00
wcsncpy(buf, (LPWSTR)scratch, SIGAR_CMDLINE_MAX);
buf[SIGAR_CMDLINE_MAX-1] = L'\0';
2004-07-29 05:47:14 +08:00
return sigar_parse_proc_args(sigar, buf, procargs);
2004-07-29 05:47:14 +08:00
}