Форум программистов, компьютерный форум, киберфорум
C++ Builder
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 5.00/8: Рейтинг темы: голосов - 8, средняя оценка - 5.00
0 / 0 / 1
Регистрация: 12.08.2013
Сообщений: 23

Не работает Synchronize() в DLL при сборке без рантайм библиотек

31.08.2017, 13:41. Показов 1704. Ответов 11
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Добрый день. Есть библиотека, в ней есть поток TThread. Библиотека должна использоваться на компьютере без IDE. Когда библиотека собирается с выставленными галочками Link with runtime packages и Link with dynamic RTL, всё работает хорошо. Убираю галочки - программа виснет на методе Synchronize, не вызывая полученную функцию. Код:

NetThread.h
Кликните здесь для просмотра всего текста
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
//---------------------------------------------------------------------------
 
#ifndef NetworkH
#define NetworkH
#include <System.Classes.hpp>
#include <IdBaseComponent.hpp>
#include <IdComponent.hpp>
#include <IdUDPClient.hpp>
#include "OMSDefs.h"
typedef void TCallback(OMSDeviceT Dev, unsigned char * Data, unsigned int Size);
class NetThread : public TThread
{
    public:
        __fastcall NetThread(OMSDeviceT Dev, unsigned int Port);
        TIdUDPClient * udp;
        void WriteReg(unsigned int Address, unsigned char * Value, unsigned int Size);
        void ReadReg(unsigned int Address, unsigned int Size, TCallback * Callback);
        void StartFlow(TCallback * Callback);
        void StopFlow();
    protected:
        void __fastcall Execute();
    private:
        //synchronize methods
        void __fastcall FlowResponse();
        void __fastcall SingleResponse();
 
        OMSDeviceT _dev;
        unsigned int _port;
 
 
        void (*_callbackFlow)(OMSDeviceT, unsigned char *, unsigned int);
        void (*_callbackSingle)(OMSDeviceT, unsigned char *, unsigned int);
 
};
//---------------------------------------------------------------------------
#endif


Network.cpp
Кликните здесь для просмотра всего текста
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
//---------------------------------------------------------------------------
 
#pragma hdrstop
 
#include "Network.h"
#include "OMSSensorSimple.h"
//---------------------------------------------------------------------------
#pragma package(smart_init)
__fastcall NetThread::NetThread(OMSDeviceT Dev, unsigned int Port) :
    TThread(true), _dev(Dev), _port(Port)
{
 
}
void NetThread::StartFlow(TCallback * Callback)
{
    _callbackFlow = Callback;
    Priority = tpNormal;
    Start();
}
void __fastcall NetThread::Execute()
{
 
    while(!Terminated)
    {
        Sleep(500);
        Synchronize(FlowResponse);
    }
}
void NetThread::WriteReg(unsigned int Address, unsigned char * Value, unsigned int Size)
{
 
}
void NetThread::ReadReg(unsigned int Address, unsigned int Size, TCallback * Callback)
{
    _callbackSingle = Callback;
}
void __fastcall NetThread::FlowResponse()
{
    byte a[2];
    a[0] = 1;
    a[1] = 2;
    _callbackFlow(_dev,a,2);
}
void NetThread::StopFlow()
{
 
}



Создание потока:
C++
1
2
3
4
5
6
7
8
std::vector<HANDLE> vDevs;
HANDLE __stdcall OMSConnect(OMSDeviceT & Dev,unsigned int Port)
{
    NetThread * nt = new NetThread(Dev,Port);
    HANDLE h = (void*)nt;
    vDevs.push_back(h);
    return vDevs.back();
}
Запуск потока:

C++
1
2
3
4
5
void __stdcall OMSStartFlow(HANDLE Dev,  CallbackFlowT * CallbackFlow)
{
    NetThread * dev = (NetThread*)Dev;
    dev->StartFlow(CallbackFlow);
}
Сам callback:
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
//---------------------------------------------------------------------------
void CallbackFlow(OMSDeviceT Dev, unsigned char * Data, unsigned int Size)
{
    switch(Dev.devIP)
    {
        case 0xC0A80002:                            //ip = 192.168.0.2
            Form6->Memo2->Lines->Add("flow 1");
            break;
        case 0xC0A80003:
            Form6->Memo3->Lines->Add("flow 2");     //ip = 192.168.0.3
            break;
        default:
            break;
    }
 
}
Что я делаю не так?
0
Лучшие ответы (1)
IT_Exp
Эксперт
34794 / 4073 / 2104
Регистрация: 17.06.2006
Сообщений: 32,602
Блог
31.08.2017, 13:41
Ответы с готовыми решениями:

включение библиотек при сборке
Прошу помочь. Написал прогу в Builder6, собрал, перенес на другую машину - не работает . Просит файлы. В настройках редактора неактивно...

При статической сборке Qt требует 3 dll
Я собрал Qt статически и он всё равно требует 3 dll libgcc_s_dw2-1.dll libstdc++-6.dll libwinpthread-1.dll

Возникают ошибки при сборке dll qwt
значит хочу себе поставить qwt 6.0.1, на копме стоит WIN XP SP3, MS VS 2010, Qt 4.8.1. Все делал так, ка написано здесь...

11
Эксперт С++
 Аватар для Avazart
8489 / 6156 / 615
Регистрация: 10.12.2010
Сообщений: 28,683
Записей в блоге: 30
31.08.2017, 13:51
Цитата Сообщение от romasv Посмотреть сообщение
Что я делаю не так?
Всё.

Сам идея дохлая.
Начиная с использование экспорта классов из DLL.
И заканчивая Synchronize в dll в котором нет основного потока gui с которым нужно синхронизироваться.
Я уже не говорю - нафига оно вообще сдалось пихать в dll
0
0 / 0 / 1
Регистрация: 12.08.2013
Сообщений: 23
31.08.2017, 14:03  [ТС]
Класс не экспортируется из DLL. Экспортируются OMSConnect и StartFlow. Коннект возвращает хэндл на поток, СтартФлоу принимает хэндл и запускает по этому адресу поток. Может это неправильно...

GUI есть, оно вызывает экспортные функции, и передает в DLL коллбэки на свои функции. Получается, GUI и DLL работают в одном потоке, но в DLL есть второй поток, который и должен синхронизоваться с потоком GUI.

Как лучше сделать?
0
Эксперт С++
 Аватар для Avazart
8489 / 6156 / 615
Регистрация: 10.12.2010
Сообщений: 28,683
Записей в блоге: 30
31.08.2017, 14:08
Цитата Сообщение от romasv Посмотреть сообщение
Класс не экспортируется из DLL.
Что тогда такое OMSDeviceT?
Цитата Сообщение от romasv Посмотреть сообщение
GUI есть, оно вызывает экспортные функции, и передает в DLL коллбэки на свои функции.
Я не вижу кода GUI в dll...
Цитата Сообщение от romasv Посмотреть сообщение
Получается, GUI и DLL работают в одном потоке,
Не канает, поток в dll не знает ничего о GUI где-то во вне.
0
0 / 0 / 1
Регистрация: 12.08.2013
Сообщений: 23
31.08.2017, 14:10  [ТС]
Код GUI:
Кликните здесь для просмотра всего текста
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
//---------------------------------------------------------------------------
 
#include <vcl.h>
#pragma hdrstop
 
#include "tester.h"
//---------------------------------------------------------------------------
#pragma package(smart_init)
#pragma resource "*.dfm"
TForm6 *Form6;
//---------------------------------------------------------------------------
__fastcall TForm6::TForm6(TComponent* Owner)
    : TForm(Owner)
{
}
//---------------------------------------------------------------------------
void __fastcall TForm6::FormCreate(TObject *Sender)
{
    OMSDeviceT * omsDevice;
    OMSDeviceT dev;
    HANDLE hDev1,hDev2;
    unsigned int n;
    unsigned int type;
    unsigned int ip;
    OMSHello(&omsDevice,&n);
    for(int i = 0; i < n; i++)
    {
        type = omsDevice->devType;
        ip = omsDevice->devIP;
        dev.devType = type;
        dev.devIP = ip;
        vDevs.push_back(dev);
        String s = "Device"+IntToStr(i)+": Type = " + UIntToStr(type) + " IP = " + UIntToStr(ip);
        Memo1->Lines->Add(s);
        omsDevice++;
 
    }
 
    Memo1->Lines->Add("Connect device 1");
    hDev1 = OMSConnect(vDevs[0],5005);
    Memo1->Lines->Add("Start flow for device 1");
    OMSStartFlow(hDev1,*CallbackFlow);
 
//  Memo1->Lines->Add("Connect device 2");
//  hDev2 = OMSConnect(vDevs[1],5023);
//  Memo1->Lines->Add("Start flow for device 2");
//  OMSStartFlow(hDev2,*CallbackFlow);
}
//---------------------------------------------------------------------------
void CallbackFlow(OMSDeviceT Dev, unsigned char * Data, unsigned int Size)
{
    switch(Dev.devIP)
    {
        case 0xC0A80002:                            //ip = 192.168.0.2
            Form6->Memo2->Lines->Add("flow 1");
            break;
        case 0xC0A80003:
            Form6->Memo3->Lines->Add("flow 2");     //ip = 192.168.0.3
            break;
        default:
            break;
    }
 
}
//---------------------------------------------------------------------------
void CallbackSingle(OMSDeviceT Dev, unsigned char * Data, unsigned int Size)
{
 
}
0
Эксперт С++
 Аватар для Avazart
8489 / 6156 / 615
Регистрация: 10.12.2010
Сообщений: 28,683
Записей в блоге: 30
31.08.2017, 14:20
Лучший ответ Сообщение было отмечено romasv как решение

Решение

Цитата Сообщение от romasv Посмотреть сообщение
Как лучше сделать?
1. Не пихать все в dll (А нафига?)
2. Не использовать Synchronize.
3. Не использовать стат линковку.

итд...

Цитата Сообщение от romasv Посмотреть сообщение
Код GUI:
Это код приложения? Или dll ?
Если приложения - читайте что я написал выше, поток в dll ничего не знает о вашем приложении.

В динамике оно работает лишь по тому что грузятся одни и те же библиотеки и используются совместно
(глобальные объекты, менеджер памяти идр вещи скрытые от глаз.)

Добавлено через 6 минут
В обще это мутная тема где все не очевидно и непонятно.

Можете почитать:
Создать DLL, которая создаст кнопку на форме
1
0 / 0 / 1
Регистрация: 12.08.2013
Сообщений: 23
31.08.2017, 15:42  [ТС]
Хм, довольно интересно всё расписали, спасибо!

Цитата Сообщение от Avazart Посмотреть сообщение
Это код приложения? Или dll ?
Если приложения - читайте что я написал выше, поток в dll ничего не знает о вашем приложении.
Код приложения. Он же и передает указатель на свой метод в библиотеку.

В DLL заказчик просил запихать... Dll должна принимать запросы с gui-программки, и принимать udp-поток данных. То, что она приняла, она должна пихнуть обратно в gui. Заказчику видимо лень писать свой поток.

Добавлено через 1 час 0 минут
И спасибо за советы! В общем, убрал Synchronize, теперь вызываю метод напрямую. Всё работает, и никак не глючит... Не понимаю, почему, ведь какой-то совершенно левый поток что-то меняет в другой программе, да ещё и на форме, и ему это прекрасно удается! Магия...
0
Эксперт С++
 Аватар для Avazart
8489 / 6156 / 615
Регистрация: 10.12.2010
Сообщений: 28,683
Записей в блоге: 30
31.08.2017, 16:04
Цитата Сообщение от romasv Посмотреть сообщение
теперь вызываю метод напрямую.
Это как ? Без синхронизации? Если да - то это нифига не работает.
0
0 / 0 / 1
Регистрация: 12.08.2013
Сообщений: 23
31.08.2017, 16:08  [ТС]
Да, без синхронизации...

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
void __fastcall NetThread::Execute()
{
    while(!Terminated)
    {
        Sleep(500);
//      Synchronize(FlowResponse);      //don't use synchronize in static linked dll (won't be working)
        FlowResponse();
    }
}
void __fastcall NetThread::FlowResponse()
{
    byte a[2];
    a[0] = 1;
    a[1] = 2;
    _callbackFlow(_dev,a,2);
}
И колбэк-функции спокойно пишут в Memo на форму, при этом я могу сам что-то в тот же Memo писать. Пытался заглючить прогу тем, что кинул на форму таймер и заставил его каждую миллисекунду писать текст в тот же Memo. Никаких сбоев. Очень странно...
0
Эксперт С++
 Аватар для Avazart
8489 / 6156 / 615
Регистрация: 10.12.2010
Сообщений: 28,683
Записей в блоге: 30
31.08.2017, 17:22
Цитата Сообщение от romasv Посмотреть сообщение
Да, без синхронизации...
Тогда в топку.

Добавлено через 32 секунды
Цитата Сообщение от romasv Посмотреть сообщение
Пытался заглючить прогу тем, что кинул на форму таймер и заставил его каждую миллисекунду писать текст в тот же Memo. Никаких сбоев.
Хреново пытались.
0
0 / 0 / 1
Регистрация: 12.08.2013
Сообщений: 23
01.09.2017, 08:30  [ТС]
Тогда какой выход? Использовать WINAPI-поток? Насколько безопасно будет в нем использовать TIdUDPClient? И как быть с синхронизацией?
0
Эксперт С++
 Аватар для Avazart
8489 / 6156 / 615
Регистрация: 10.12.2010
Сообщений: 28,683
Записей в блоге: 30
01.09.2017, 10:15
Цитата Сообщение от romasv Посмотреть сообщение
И как быть с синхронизацией?
Можете попробовать использовать системную очередь сообщений, для синхронизации.
Но как я сказал лучше отказаться от такой затеи вовсе или же ликоваться динамически.
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
BasicMan
Эксперт
29316 / 5623 / 2384
Регистрация: 17.02.2009
Сообщений: 30,364
Блог
01.09.2017, 10:15
Помогаю со студенческими работами здесь

Как изменить место создания dll-библиотек в своём проекте при компиляции
Здравствуйте, возник вопрос. В моём решении используются dll-библиотеки. Все они как обычно подключены через референсы. При компиляции...

Почему при сборке Releas'а прога без VC++ не идёт?
Прога использует библиотеки ДАО. Собираю как Release, на машине Win 98 она выдаёт ошибку MS VC++ Runtime Error.

Не работает Synchronize в TThread
Доброго времени суток! .h class TMyThread : public TThread { private: void __fastcall send_mess(); protected: void...

Как лучше копировать память, обмениваться указателями и т.д. при работе с кучей динамических библиотек (DLL) ?
Привет! Вот есть много DLL-к, каждая работает со своей внутренней динамической памятью, а так же есть необходимость работать с объектами...

При сборке работает только полпроекта
при сборке работает только полпроекта. Что делать? в свойствах все так же как и для отладки, а именно подсистема - винда и точка входа -...


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

Или воспользуйтесь поиском по форуму:
12
Ответ Создать тему
Новые блоги и статьи
Уведомление о неверно выбранном значении справочника
Maks 06.04.2026
Алгоритм из решения ниже реализован на примере нетипового документа "НарядПутевка", разработанного в конфигурации КА2. Задача: уведомлять пользователя, если в документе выбран неверный склад. . .
Установка Qt Creator для C и C++: ставим среду, CMake и MinGW без фреймворка Qt
8Observer8 05.04.2026
Среду разработки Qt Creator можно установить без фреймворка Qt. Есть отдельный репозиторий для этой среды: https:/ / github. com/ qt-creator/ qt-creator, где можно скачать установщик, на вкладке Releases:. . .
AkelPad-скрипты, структуры, и немного лирики..
testuser2 05.04.2026
Такая программа, как AkelPad существует уже давно, и также давно существуют скрипты под нее. Тем не менее, прога живет, периодически что-то не спеша дополняется, улучшается. Что меня в первую очередь. . .
Отображение реквизитов в документе по условию и контроль их заполнения
Maks 04.04.2026
Алгоритм из решения ниже реализован на примере нетипового документа "ПланированиеСпецтехники", разработанного в конфигурации КА2. Данный документ берёт данные из другого нетипового документа. . .
Фото всей Земли с борта корабля Orion миссии Artemis II
kumehtar 04.04.2026
Это первое подобное фото сделанное человеком за 50 лет. Снимок называют новым вариантом легендарной фотографии «The Blue Marble» 1972 года, сделанной с борта корабля «Аполлон-17». Новое фото. . .
Вывод диалогового окна перед закрытием, если документ не проведён
Maks 04.04.2026
Алгоритм из решения ниже реализован на примере нетипового документа "СписаниеМатериалов", разработанного в конфигурации КА2. Задача: реализовать программный контроль на предмет проведения документа. . .
Программный контроль заполнения реквизитов табличной части документа
Maks 02.04.2026
Алгоритм из решения ниже реализован на примере нетипового документа "СписаниеМатериалов", разработанного в конфигурации КА2. Задача: 1. Реализовать контроль заполнения реквизита. . .
wmic не является внутренней или внешней командой
Maks 02.04.2026
Решение: DISM / Online / Add-Capability / CapabilityName:WMIC~~~~ Отсюда: https:/ / winitpro. ru/ index. php/ 2025/ 02/ 14/ komanda-wmic-ne-naydena/
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru