Форум программистов, компьютерный форум, киберфорум
С++ для начинающих
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
 
Рейтинг 4.60/48: Рейтинг темы: голосов - 48, средняя оценка - 4.60
0 / 0 / 0
Регистрация: 11.02.2013
Сообщений: 10

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

11.02.2013, 13:19. Показов 10317. Ответов 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!"); вполне адекватно отрабатывает.
0
IT_Exp
Эксперт
34794 / 4073 / 2104
Регистрация: 17.06.2006
Сообщений: 32,602
Блог
11.02.2013, 13:19
Ответы с готовыми решениями:

Использование строк.Использование структур
Задачка: Дана строка,состоящая из групп нулей и едениц. Найти и вывести на экран группы с нечетным количеством символов. И еще одна: ...

Метки в макросах в ассемблерной вставке
В C++ написан макрос с ассемблерной вставкой, который содержит в себе метку "doit" и команду перехода на неё: #define MYMACRO(x, y,...

Использование строк
Помогите с алгоритмом программы: вводим текст происходит проверка на наличие цифр прописью, заменяет их и выводит исправленый...

23
Evg
Эксперт CАвтор FAQ
 Аватар для Evg
21281 / 8305 / 637
Регистрация: 30.03.2009
Сообщений: 22,660
Записей в блоге: 30
13.02.2013, 17:35
Студворк — интернет-сервис помощи студентам
Напомню ещё раз. Правильно поставленный вопрос - это половина ответа. Если ты хочешь, чтобы тебе ответили - потрать 10 минут и нормально опиши проблему. Мне, если честно, уже становится впадлу читать по три раза твои посты, чтобы догадаться, что же ты имеешь ввиду
0
0 / 0 / 0
Регистрация: 11.02.2013
Сообщений: 10
14.02.2013, 22:00  [ТС]
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);\
}\
Написал вышеприведенный код.
Вроде бы все хорошо, без возврата в оригинальную функцию выводит полученные аргументы, бла-бла-бла, всемирное добро и счастье.
Но вот как перейти к оригинальной функции - вот тут-то и затык. Возникает ошибка , предположительно связанная с тем, что переменное число аргументов не может работать с 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);
Благодарю за внимание!
0
Evg
Эксперт CАвтор FAQ
 Аватар для Evg
21281 / 8305 / 637
Регистрация: 30.03.2009
Сообщений: 22,660
Записей в блоге: 30
15.02.2013, 18:25
Сначала был пример:

Цитата Сообщение от 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 )
0
0 / 0 / 0
Регистрация: 11.02.2013
Сообщений: 10
15.02.2013, 19:46  [ТС]
Любопытно. Спасибо за информацию.
А что насчет передачи аргументов в функцию(Использование строк в макросах С++
И есть ли какой-либо способ перехвата функциидля различных соглашений о вызовах(__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);
}
?
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
BasicMan
Эксперт
29316 / 5623 / 2384
Регистрация: 17.02.2009
Сообщений: 30,364
Блог
15.02.2013, 19:46
Помогаю со студенческими работами здесь

Использование файлов и строк
Уважаемые программисты! Нужна ваша помощь, заканчиваю написание диплома и возникла следующая проблемка: Есть программа, которая все...

Использование строк и файлов
С помощью текстового редактора создать файл, содержащий текст, длина которого не превышает 1000 символов (длина строки текста не должна...

Использование одномерного массива строк
Здравствуйте! Задание: Имеется файл 1.txt в котором содержатся строки неизвестной длины. Так же имеется файл 2.txt в котором так же...

Использование строк(как правильно?)
Здравствуйте! Была где-то реализация (условно) const char buffer; int i=0; sprintf(buffer,"message d%",i); Вопрос: можно...

Использование собственного класса строк
Здравствуйте, уважаемые господа. Есть реализация собственного класса строк: class MyString { private: size_t length; ...


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

Или воспользуйтесь поиском по форуму:
24
Ответ Создать тему
Новые блоги и статьи
Модель здравосоХранения 6. ESG-повестка и устойчивое развитие; углублённый анализ кадрового бренда
anaschu 31.03.2026
В прикрепленном документе раздумья о том, как можно поменять модель в будущем
10 пpимет, которые всегда сбываются
Maks 31.03.2026
1. Чтобы, наконец, пришла маршрутка, надо закурить. Если сигарета последняя, маршрутка придет еще до второй затяжки даже вопреки расписанию. 2. Нaдоели зима и снег? Не надо переезжать. Достаточно. . .
Перемещение выделенных строк ТЧ из одного документа в другой
Maks 31.03.2026
Реализация из решения ниже выполнена на примере нетипового документа "ВыдачаОборудованияНаСпецтехнику" с единственной табличной частью "ОборудованиеИКомплектующие" разработанного в конфигурации КА2. . . .
Functional First Web Framework Suave
DevAlt 30.03.2026
Sauve. IO Апнулись до NET10. Из зависимостей один пакет, работает одинаково хорошо как в режиме проекта так и в интерактивном режиме. из сложностей - чисто функциональный подход. Решил. . .
Автоматическое создание документа при проведении другого документа
Maks 29.03.2026
Реализация из решения ниже выполнена на нетиповых документах, разработанных в конфигурации КА2. Есть нетиповой документ "ЗаявкаНаРемонтСпецтехники" и нетиповой документ "ПланированиеСпецтехники". В. . .
Настройка движения справочника по регистру сведений
Maks 29.03.2026
Решение ниже реализовано на примере нетипового справочника "ТарифыМобильнойСвязи" разработанного в конфигурации КА2, с целью учета корпоративной мобильной связи в коммерческом предприятии. . . .
Автозаполнение реквизита при выборе элемента справочника
Maks 27.03.2026
Программный код из решения ниже на примере нетипового документа "ЗаявкаНаРемонтСпецтехники" разработанного в конфигурации КА2. При выборе "Спецтехники" (Тип Справочник. Спецтехника), заполняется. . .
Сумматор с применением элементов трёх состояний.
Hrethgir 26.03.2026
Тут. https:/ / fips. ru/ EGD/ ab3c85c8-836d-4866-871b-c2f0c5d77fbc Первый документ красиво выглядит, но без схемы. Это конечно не даёт никаких плюсов автору, но тем не менее. . . всё может быть. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru