Версия для печати темы
Нажмите сюда для просмотра этой темы в оригинальном формате
Форум программистов > C/C++: Системное программирование и WinAPI > DbgPrint и va_start не могу адаптировать


Автор: 12usver12 27.6.2010, 23:49
собсно есть такой код, вроде делал все как написано для этих макросов...
в итоге так и не получилось адаптировать
Код


// прототип из хидера oscalls.h    int DbgPrint(__in_z __format_string char *_Format, ...);

FARPROC adrDbgPrint = 0;
int DbgPrint_(char *_Format, ...)
{
    if (!adrDbgPrint) adrDbgPrint = GetProcAddress(GetModuleHandleW(L"ntdll.dll"),"DbgPrint");
    va_list args;
    va_start(args, _Format);
    int res = ((int (__stdcall*)(char*, ...)) adrDbgPrint)(_Format, args);
    /*
    __asm
    {
        push args
        push _Format
        call adrDbgPrint
    }
    */
    va_end(args);
    return res;
}


выводит в консоль неправильное значение , где же косяк ? 

Автор: 586 28.6.2010, 00:41
Судя по прототипу, DbgPrint требует передачи самих параметров, а не указателя va_list. А количество параметров не известно, и поэтому их не передать (либо вычислить количество параметров, пропарсив строку форматирования).
Если нужно замаскировать вызов GetProcAddress, то можно так написать:
Код
ULONG (__cdecl *DbgPrint_)(LPCSTR, ...) = (ULONG (__cdecl*)(LPCSTR, ...))
    GetProcAddress(GetModuleHandleW(L"ntdll.dll"),"DbgPrint");

int main()
{
    DbgPrint_("%u\n", 5555);
}

Автор: 12usver12 28.6.2010, 09:05
работает, получается каждый раз при вызове этой функи выполняется GetProcAddress(GetModuleHandleW(L"ntdll.dll"),"DbgPrint")  или оно только раз при инициализации выполняется ?

Автор: 586 28.6.2010, 10:31
Цитата(12usver12 @  28.6.2010,  10:05 Найти цитируемый пост)
работает, получается каждый раз при вызове этой функи выполняется GetProcAddress(GetModuleHandleW(L"ntdll.dll"),"DbgPrint")  или оно только раз при инициализации выполняется ? 

Один раз выполнится при инициализации. Только в конструкторах классов надо использовать этот указатель с осторожностью - указатель DbgPrint_ может быть ещё не инициализирован.

Кстати! Есть же ещё функция OutputDebugString:
Код
#include <windows.h>
#include <tchar.h>

#ifndef _ARRAYSIZE
# define _ARRAYSIZE(x) (sizeof(x)/sizeof(x[0]))
#endif

void vDebugPrintf(LPCTSTR lpFmt, va_list v)
{
    TCHAR buf[2048];
    _vsntprintf(buf, _ARRAYSIZE(buf), lpFmt, v);
    buf[_ARRAYSIZE(buf)-1] = _T('\0');
    OutputDebugString(buf);
}

void DebugPrintf(LPCTSTR lpFmt, ...)
{
    va_list v;
    va_start(v, lpFmt);
    vDebugPrintf(lpFmt, v);
    va_end(v);
}

int main()
{
    DebugPrintf(_T("%u\n"), 5555);
}

Автор: 12usver12 28.6.2010, 11:06
ну да я тоже использовал подобную конструкцию, но проблема в том что нужно было выделять фиксированный буфер TCHAR buf[2048] , иногда это мешало, так как данные иногда были больше размера буфера, впринципе можно было динамически стек выделять, но не хотелось лепить велики, так как есть уже DbgPrint  

Powered by Invision Power Board (http://www.invisionboard.com)
© Invision Power Services (http://www.invisionpower.com)