Форум программистов, компьютерный форум CyberForum.ru

С++ для начинающих

Войти
Регистрация
Восстановить пароль
 
 
Рейтинг: Рейтинг темы: голосов - 24, средняя оценка - 4.96
demogor
0 / 0 / 0
Регистрация: 11.02.2013
Сообщений: 10
#1

Использование строк в макросах С++ - C++

11.02.2013, 13:19. Просмотров 3127. Ответов 23
Метки нет (Все метки)

Доброго времени суток всем!
Возникла небольшая проблема - не могу понять, возможно ли сделать нечто подобное, и, если возможно, то как.
Есть макрос, объявляющий функцию:

C++
1
2
3
4
#define DECLARE_FUNC(name1, body, type_cl, ...)\
    __declspec(dllexport) void type_cl name1##_func(__VA_ARGS__){\
    body;\
}\
Используется следующим образом:
C++
1
DECLARE_FUNC(testDeclare, "debug_msg(\"TestDeclare\",\"hello!\")", void, void);
где строка - вызов функции debug_msg(запись в файл форматированной строки).

Собственно, вопрос - как это сделать правильно? В текущем варианте нет никакой реакции на вызов функции testDeclare_func().

P.S. В следующем варианте
C++
1
2
3
4
5
#define DECLARE_FUNC(name1, body, type_cl, ...)\
    __declspec(dllexport) void type_cl name1##_func(__VA_ARGS__){\
    debug_msg("TestDeclare","hello!");\
    body;\
}\
вывод из debug_msg("TestDeclare","hello!"); вполне адекватно отрабатывает.
После регистрации реклама в сообщениях будет скрыта и будут доступны все возможности форума.
Evg
Эксперт CАвтор FAQ
17463 / 5701 / 361
Регистрация: 30.03.2009
Сообщений: 15,654
Записей в блоге: 26
13.02.2013, 17:35     Использование строк в макросах С++ #21
Напомню ещё раз. Правильно поставленный вопрос - это половина ответа. Если ты хочешь, чтобы тебе ответили - потрать 10 минут и нормально опиши проблему. Мне, если честно, уже становится впадлу читать по три раза твои посты, чтобы догадаться, что же ты имеешь ввиду
demogor
0 / 0 / 0
Регистрация: 11.02.2013
Сообщений: 10
14.02.2013, 22:00  [ТС]     Использование строк в макросах С++ #22
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
#define RF_IMP(ret, type_cl, name1, format, adres, ...)\
    typedef ret (type_cl * t##name1 ) (va_list args);\
    t##name1 name1##_Detour = ( t##name1 ) ( adres );\
    void type_cl name1##_hook(char *txt, ...);\
    BYTE name1##_var[6];\
    void type_cl name1##_hook (char *txt, ...) {\
    txt=format;\
    va_list arg;\
    va_start(arg, txt);\
    name1##_Detour(arg);\
    debug_msg("ParseArgs",txt, arg);\
    va_end(arg);\
}\
Написал вышеприведенный код.
Вроде бы все хорошо, без возврата в оригинальную функцию выводит полученные аргументы, бла-бла-бла, всемирное добро и счастье.
Но вот как перейти к оригинальной функции - вот тут-то и затык. Возникает ошибка http://*************/s/2PHjK, предположительно связанная с тем, что переменное число аргументов не может работать с calling conventions, очищающими стек(__stdcall, к примеру). Впрочем, и с __cdecl'ом та же проблема возникла.
На самом деле, va_list здесь не нужен, достаточно бы передать аргументы из макроса(__VA_ARGS__) - но тут возникает проблема: вызов функции не получается провести с __VA_ARGS__(насколько понимаю, внутри содержится список аргументов с типами, в результате чего компилятор ругается на то, что нельзя вызвать функцию name_Detour(int a1).

Попробую четко сформулировать вопрос - как, собственно, вернуть управление оригинальной функции, передав при этом ей список аргументов, объявленных в макросе?

Пример использования:
C++
1
2
3
RF_IMP(int, WINAPI, send2, "%d, %d, 0x%X, %d, %d, %d, %d", 0x17E4D18, SOCKET s, LPWSABUF lpBuffers, DWORD dwBufferCount, LPDWORD lpNumberOfBytesSent, DWORD dwFlags, 
       LPWSAOVERLAPPED lpOverlapped, 
       LPWSAOVERLAPPED_COMPLETION_ROUTINE lpCompletionRoutine);
Благодарю за внимание!
Evg
Эксперт CАвтор FAQ
17463 / 5701 / 361
Регистрация: 30.03.2009
Сообщений: 15,654
Записей в блоге: 26
15.02.2013, 18:25     Использование строк в макросах С++ #23
Сначала был пример:

Цитата Сообщение от Evg Посмотреть сообщение
Код

C++
1
2
3
4
5
6
#define DECLARE_FUNC(name1, body, type_cl, ...) \
  __declspec(dllexport) type_cl name1##_func(__VA_ARGS__) {\
    body; \
  }
 
DECLARE_FUNC (testDeclare, debug_msg ("TestDeclare", "hello!"), void, int x, int y)
после препроцессора превратится в

C++
1
__declspec(dllexport) void testDeclare_func(int x, int y) { debug_msg ("TestDeclare", "hello!"); }
Затем был вопрос:

Цитата Сообщение от vxg Посмотреть сообщение
я думал что препроцессор споткнется об запятую перед hello, но пример работает. как это возможно?
Вот что на это говорит стандарт:

ISO/IEC 9899:1999 (E)
...
6.10.3 Macro replacement
...
11. The sequence of preprocessing tokens bounded by the outside-most matching parentheses forms the list of arguments or the function-like macro. The individual arguments within the list are separated by comma preprocessing tokens, but comma preprocessing tokens between matching inner parentheses do not separate arguments. If there are sequences of preprocessing tokens within the list of arguments that would otherwise act as preprocessing directives, the behavior is undefined.
Вот как надо поступать, если мы хотим вставить круглые скобки в качестве параметра макроса:

C
#define LB (
#define RB )
#define A(x1, x2, x3) x1 qqq x2 qqq x3
A (LB , x , RB)
после препроцессора превратится

C
( qqq x qqq )
MoreAnswers
Эксперт
37091 / 29110 / 5898
Регистрация: 17.06.2006
Сообщений: 43,301
15.02.2013, 19:46     Использование строк в макросах С++
Еще ссылки по теме:
Использование строк, как одномерные массивы символов C++
C++ функция возвращает int, возвращаемые значения определены в макросах, как сразу выводить определённые макросы?
C++ Л/р "Использование строк"
C++ По некоторому количеству строк (не более N) найти пары строк с общим словом
C++ Вводится массив строк текста с неизвестным заранее количеством строк

Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
demogor
0 / 0 / 0
Регистрация: 11.02.2013
Сообщений: 10
15.02.2013, 19:46  [ТС]     Использование строк в макросах С++ #24
Любопытно. Спасибо за информацию.
А что насчет передачи аргументов в функцию(Использование строк в макросах С++
И есть ли какой-либо способ перехвата функциидля различных соглашений о вызовах(__stdcall, к примеру) без указания аргументов функции(к примеру, объявление общего прототипа с va_list в качестве аргумента)?

Тобишь, к примеру, есть 3 функции:
C++
1
2
3
int WINAPI WSASend(Socket s, blablabla...);
void __stdcall WhatDidThisFunctionDoAnyImagine(int a1, char *a2, bool a3); 
void * __cdecl BlaBlaReallyBla(void *arg);
и общий прототип для перехвата в стиле
C++
1
2
3
4
5
6
void name##_hook(char *format, va_list va)
{
   va_start(va,format);
   name##_original(va);
   va_end(va);
}
?
Yandex
Объявления
15.02.2013, 19:46     Использование строк в макросах С++
Ответ Создать тему
Опции темы

КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin® Version 3.8.9
Copyright ©2000 - 2017, vBulletin Solutions, Inc.
Рейтинг@Mail.ru