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

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

11.02.2013, 13:19. Показов 10271. Ответов 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
Ответ Создать тему
Новые блоги и статьи
SDL3 для Desktop (MinGW): Создаём пустое окно с нуля для 2D-графики на SDL3, Си и C++
8Observer8 10.03.2026
Содержание блога Финальные проекты на Си и на C++: hello-sdl3-c. zip hello-sdl3-cpp. zip Результат:
Установка CMake и MinGW 13.1 для сборки С и C++ приложений из консоли и из Qt Creator в EXE
8Observer8 10.03.2026
Содержание блога MinGW - это коллекция инструментов для сборки приложений в EXE. CMake - это система сборки приложений. Здесь описаны базовые шаги для старта программирования с помощью CMake и. . .
Как дизайн сайта влияет на конверсию: 7 решений, которые реально повышают заявки
Neotwalker 08.03.2026
Многие до сих пор воспринимают дизайн сайта как “красивую оболочку”. На практике всё иначе: дизайн напрямую влияет на то, оставит человек заявку или уйдёт через несколько секунд. Даже если у вас. . .
Модульная разработка через nuget packages
DevAlt 07.03.2026
Сложившийся в . Net-среде способ разработки чаще всего предполагает монорепозиторий в котором находятся все исходники. При создании нового решения, мы просто добавляем нужные проекты и имеем. . .
Модульный подход на примере F#
DevAlt 06.03.2026
В блоге дяди Боба наткнулся на такое определение: В этой книге («Подход, основанный на вариантах использования») Ивар утверждает, что архитектура программного обеспечения — это структуры,. . .
Управление камерой с помощью скрипта OrbitControls.js на Three.js: Вращение, зум и панорамирование
8Observer8 05.03.2026
Содержание блога Финальная демка в браузере работает на Desktop и мобильных браузерах. Итоговый код: orbit-controls-threejs-js. zip. Сканируйте QR-код на мобильном. Вращайте камеру одним пальцем,. . .
SDL3 для Web (WebAssembly): Синхронизация спрайтов SDL3 и тел Box2D
8Observer8 04.03.2026
Содержание блога Финальная демка в браузере. Итоговый код: finish-sync-physics-sprites-sdl3-c. zip На первой гифке отладочные линии отключены, а на второй включены:. . .
SDL3 для Web (WebAssembly): Идентификация объектов на Box2D v3 - использование userData и событий коллизий
8Observer8 02.03.2026
Содержание блога Финальная демка в браузере. Итоговый код: finish-collision-events-sdl3-c. zip Сканируйте QR-код на мобильном и вы увидите, что появится джойстик для управления главным героем. . . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru