Re: Runtime определение апартмента из кода.
От: Алекс Россия http://wise-orm.com
Дата: 16.10.02 05:33
Оценка: 84 (5)
Здравствуйте George Seryakov, Вы писали:

GS>Привет!


GS>Тут мы (не я, кстати) пытаемя получить поддержку из MS, и получили в ответ "<..> components are marked as “Both”. If they are created by DCOM clients they will be bound to the MTA." Речь идет про MTS/COM+ компонент. Чушь, по моему. Но как это проверить? Как можно в коде компонента найти в каком апартменте он создан, причем для STA — в каком именно STA — host, main или ином, и как-то идентифицировать апартмент (по хендлу окна?) нельзя ли?


По смещению 0F80h в TEB находится адрес структуры, которая определяет параметры апартамента. Если указатель равен нулю, СОМ не инициализированна. Если инициализированна, по смещению 12 в этой структуре находится переменная, по значению которой можно определить тип аппартамента.

Пример кода:

        int k;
    __asm{
        mov eax,fs:[18h];
        mov eax,[eax+0F80h];
        mov ebx,[eax+0Ch];
        mov k,ebx
    }
if (k == 0x81){
   //STA
}
else if (k == 0x141){
   //MTA
}


А пользоваться документированными функциями любой сможет! Шутка!
Re: Runtime определение апартмента из кода.
От: Ivan Россия www.rsdn.ru
Дата: 26.10.02 10:02
Оценка: 37 (5)
Здравствуйте George Seryakov, Вы писали:

GS>Привет!


GS>Тут мы (не я, кстати) пытаемя получить поддержку из MS, и получили в ответ "<..> components are marked as “Both”. If they are created by DCOM clients they will be bound to the MTA." Речь идет про MTS/COM+ компонент. Чушь, по моему. Но как это проверить? Как можно в коде компонента найти в каком апартменте он создан, причем для STA — в каком именно STA — host, main или ином, и как-то идентифицировать апартмент (по хендлу окна?) нельзя ли?


Можно использовать HRESULT CoGetApartmentID(DWORD dwFlag, DWORD* pID);
dwFlag надо передавать -1
Функция недокументированная.
Находится в ole32.dll
С ее помощью всегда можно определить в каком апартменте мы сейчас находимся
Re: Runtime определение апартмента из кода.
От: VladD2 Российская Империя www.nemerle.org
Дата: 15.10.02 23:25
Оценка: 24 (1)
Здравствуйте George Seryakov, Вы писали:

GS>Тут мы (не я, кстати) пытаемя получить поддержку из MS, и получили в ответ "<..> components are marked as “Both”. If they are created by DCOM clients they will be bound to the MTA." Речь идет про MTS/COM+ компонент. Чушь, по моему. Но как это проверить? Как можно в коде компонента найти в каком апартменте он создан, причем для STA — в каком именно STA — host, main или ином, и как-то идентифицировать апартмент (по хендлу окна?) нельзя ли?


Попробуй прогнать CoInitializeEx и смотри сообщение об ошибке:

S_FALSE — The COM library is already initialized on the calling thread.

RPC_E_CHANGED_MODE — A previous call to CoInitializeEx specified a different concurrency model for the calling thread. If running Windows 2000, this could also mean that a change from neutral threaded apartment to single threaded apartment occurred.
... << RSDN@Home 1.0 alpha VladD2.1.0.alpha 11.1.0.1019.1018 >>
Есть логика намерений и логика обстоятельств, последняя всегда сильнее.
Re[2]: Runtime определение апартмента из кода.
От: George Seryakov Россия  
Дата: 16.10.02 16:13
Оценка: 13 (1)
Здравствуйте Алекс, Вы писали:

А>По смещению 0F80h в TEB находится адрес структуры, которая определяет параметры апартамента. Если указатель равен нулю, СОМ не инициализированна. Если инициализированна, по смещению 12 в этой структуре находится переменная, по значению которой можно определить тип аппартамента.


А>
А>        int k;
А>    __asm{
А>        mov eax,fs:[18h];
А>        mov eax,[eax+0F80h];
А>        mov ebx,[eax+0Ch];
А>        mov k,ebx
А>    }
А>if (k == 0x81){
А>   //STA
А>}
А>else if (k == 0x141){
А>   //MTA
А>}
А>


Сильно. А дровишки откуда? Типа, где-то это написано?

А>А пользоваться документированными функциями любой сможет! Шутка!


Попробовал я. В MTA там действительно 0x141. А вот при вызове из-под STA, сгенеренного VC клиентом — имеем 0x89, a VB клиентом — 0x8b.
GS
Re[3]: Runtime определение апартмента из кода.
От: George Seryakov Россия  
Дата: 16.10.02 16:28
Оценка: 13 (1)
Здравствуйте George Seryakov, Вы писали:


GS>Попробовал я. В MTA там действительно 0x141. А вот при вызове из-под STA, сгенеренного VC клиентом — имеем 0x89, a VB клиентом — 0x8b.


Поправочка: когда из-под дебаггера — 0x8b, из-под компилированного кода (и скрипта) — 0x89. Дровишки я, кстати, нашел в MSDN. Вот только бы найти еще описание COM TLS structure.

0x80 — это маска апартментной модели (0x140 — MTA), а вот за что отвечает 9-ка и 11-ка?
GS
Runtime определение апартмента из кода.
От: George Seryakov Россия  
Дата: 15.10.02 21:29
Оценка:
Привет!

Тут мы (не я, кстати) пытаемя получить поддержку из MS, и получили в ответ "<..> components are marked as “Both”. If they are created by DCOM clients they will be bound to the MTA." Речь идет про MTS/COM+ компонент. Чушь, по моему. Но как это проверить? Как можно в коде компонента найти в каком апартменте он создан, причем для STA — в каком именно STA — host, main или ином, и как-то идентифицировать апартмент (по хендлу окна?) нельзя ли?
GS
Re[2]: Runtime определение апартмента из кода.
От: TK Лес кывт.рф
Дата: 16.10.02 04:54
Оценка:
Здравствуйте VladD2, Вы писали:

VD>Здравствуйте George Seryakov, Вы писали:


GS>>Тут мы (не я, кстати) пытаемя получить поддержку из MS, и получили в ответ "<..> components are marked as “Both”. If they are created by DCOM clients they will be bound to the MTA." Речь идет про MTS/COM+ компонент. Чушь, по моему. Но как это проверить? Как можно в коде компонента найти в каком апартменте он создан, причем для STA — в каком именно STA — host, main или ином, и как-то идентифицировать апартмент (по хендлу окна?) нельзя ли?


VD>Попробуй прогнать CoInitializeEx и смотри сообщение об ошибке:


VD>S_FALSE — The COM library is already initialized on the calling thread.


VD>RPC_E_CHANGED_MODE — A previous call to CoInitializeEx specified a different concurrency model for the calling thread. If running Windows 2000, this could also mean that a change from neutral threaded apartment to single threaded apartment occurred.


VD>


When an object that is configured to run in the neutral threaded apartment (NTA) is called by a thread that is in either an STA or the MTA, that thread transfers to the NTA. If this thread subsequently calls CoInitializeEx, the call fails and returns RPC_E_CHANGED_MODE.
Если у Вас нет паранойи, то это еще не значит, что они за Вами не следят.
Re: Runtime определение апартмента из кода.
От: Tom Россия http://www.RSDN.ru
Дата: 16.10.02 07:11
Оценка:
Здравствуйте George Seryakov, Вы писали:

GS>Привет!


GS>Тут мы (не я, кстати) пытаемя получить поддержку из MS, и получили в ответ "<..> components are marked as “Both”. If they are created by DCOM clients they will be bound to the MTA." Речь идет про MTS/COM+ компонент. Чушь, по моему. Но как это проверить? Как можно в коде компонента найти в каком апартменте он создан, причем для STA — в каком именно STA — host, main или ином, и как-то идентифицировать апартмент (по хендлу окна?) нельзя ли?


Конечно можно. Используй TLS, так как аппартаменты всё равно привязаны к TLS, то этот способ самый надёжный и простой (его по моему и ком использует)
Народная мудрось
всем все никому ничего(с).
Re: Runtime определение апартмента из кода.
От: Lexey Россия  
Дата: 16.10.02 08:29
Оценка:
Здравствуйте George Seryakov, Вы писали:

GS>Тут мы (не я, кстати) пытаемя получить поддержку из MS, и получили в ответ "<..> components are marked as “Both”. If they are created by DCOM clients they will be bound to the MTA." Речь идет про MTS/COM+ компонент. Чушь, по моему. Но как это проверить? Как можно в коде компонента найти в каком апартменте он создан, причем для STA — в каком именно STA — host, main или ином, и как-то идентифицировать апартмент (по хендлу окна?) нельзя ли?


Да нет, видимо не чушь. Мы тут вчера тоже наступили на грабли, когда случайно установили пулинг для COM+ — приложения, объекты которого держат внутренние ссылки на другие объекты. На втором вызове компонента стабильно получаем "Interface was marshaled from a different thread". В STA такого бы не было.
"Будь достоин победы" (c) 8th Wizard's rule.
Re[2]: Runtime определение апартмента из кода.
От: George Seryakov Россия  
Дата: 16.10.02 17:49
Оценка:
Здравствуйте Lexey, Вы писали:

GS>>Тут мы (не я, кстати) пытаемя получить поддержку из MS, и получили в ответ "<..> components are marked as “Both”. If they are created by DCOM clients they will be bound to the MTA." Речь идет про MTS/COM+ компонент. Чушь, по моему. Но как это проверить? Как можно в коде компонента найти в каком апартменте он создан, причем для STA — в каком именно STA — host, main или ином, и как-то идентифицировать апартмент (по хендлу окна?) нельзя ли?


L>Да нет, видимо не чушь. Мы тут вчера тоже наступили на грабли, когда случайно установили пулинг для COM+ — приложения, объекты которого держат внутренние ссылки на другие объекты. На втором вызове компонента стабильно получаем "Interface was marshaled from a different thread". В STA такого бы не было.


Да, отнюдь не чушь. Вот что я напроверял. Компонент — под COM+/ конфигурированный. Строки — вызовы из, соответственно, STA и MTA (CoInitializeEx(NULL, COINIT_APARTMENTTHREADED или COINIT_MULTITHREADED)). Стоблцы — параметр ThreadingModel в реджистри. На пересечении — результат: STA, если CoInitializeEx(NULL, COINIT_APARTMENTTHREADED) отозвался S_FALSE и MTA, соответвенно. NTA — оба вызова отозвались как RPC_E_CHANGED_MODE. Цифры — значение двух байтов по смещению 0xC в COM TLS structure. Для Both компонент ведет себя как Free.

       Apartment    Free            no            Neutral
STA    STA, 0x8a    MTA, 0x1002    STA, 0x208a    NTA, 0x1802
MTA    STA, 0x8a    MTA, 0x1002    STA, 0x208a    NTA, 0x1802


Пулинг и прочее я не трогал. Этот МасЛеод знал заранее, что под компласом все куда хитрее...
GS
Re[3]: Runtime определение апартмента из кода.
От: VladD2 Российская Империя www.nemerle.org
Дата: 16.10.02 19:46
Оценка:
Здравствуйте TK, Вы писали:

TK>When an object that is configured to run in the neutral threaded apartment (NTA) is called by a thread that is in either an STA or the MTA, that thread transfers to the NTA. If this thread subsequently calls CoInitializeEx, the call fails and returns RPC_E_CHANGED_MODE.


Ну, и?
... << RSDN@Home 1.0 alpha VladD2.1.0.alpha 12.1.0.1019.39707 >>
Есть логика намерений и логика обстоятельств, последняя всегда сильнее.
Re[4]: Runtime определение апартмента из кода.
От: George Seryakov Россия  
Дата: 16.10.02 19:59
Оценка:
Здравствуйте VladD2, Вы писали:

TK>>When an object that is configured to run in the neutral threaded apartment (NTA) is called by a thread that is in either an STA or the MTA, that thread transfers to the NTA. If this thread subsequently calls CoInitializeEx, the call fails and returns RPC_E_CHANGED_MODE.


VD>Ну, и?


Признаком NTA будет отлуп RPC_E_CHANGED_MODE из STA и MTA.
GS
Re[5]: Runtime определение апартмента из кода.
От: VladD2 Российская Империя www.nemerle.org
Дата: 16.10.02 20:13
Оценка:
Здравствуйте George Seryakov, Вы писали:
TK>>>When an object that is configured to run in the neutral threaded apartment (NTA) is called by a thread that is in either an STA or the MTA, that thread transfers to the NTA. If this thread subsequently calls CoInitializeEx, the call fails and returns RPC_E_CHANGED_MODE.

VD>>Ну, и?


GS>Признаком NTA будет отлуп RPC_E_CHANGED_MODE из STA и MTA.


А!
... << RSDN@Home 1.0 alpha VladD2.1.0.alpha 12.1.0.1019.39707 >>
Есть логика намерений и логика обстоятельств, последняя всегда сильнее.
Re[3]: Runtime определение апартмента из кода.
От: George Seryakov Россия  
Дата: 17.10.02 20:03
Оценка:
Здравствуйте, мы писали:

GS>Да, отнюдь не чушь. Вот что я напроверял. Компонент — под COM+/ конфигурированный. Строки — вызовы из, соответственно, STA и MTA (CoInitializeEx(NULL, COINIT_APARTMENTTHREADED или COINIT_MULTITHREADED)). Стоблцы — параметр ThreadingModel в реджистри. На пересечении — результат: STA, если CoInitializeEx(NULL, COINIT_APARTMENTTHREADED) отозвался S_FALSE и MTA, соответвенно. NTA — оба вызова отозвались как RPC_E_CHANGED_MODE. Цифры — значение двух байтов по смещению 0xC в COM TLS structure. Для Both компонент ведет себя как Free.


GS>
GS>       Apartment    Free            no            Neutral
GS>STA    STA, 0x8a    MTA, 0x1002    STA, 0x208a    NTA, 0x1802
GS>MTA    STA, 0x8a    MTA, 0x1002    STA, 0x208a    NTA, 0x1802
GS>


GS>Пулинг и прочее я не трогал. Этот МасЛеод знал заранее, что под компласом все куда хитрее...


Да, под обычным MTS-ом всегда STA, 0x8a. Т.е. COM+ это далеко не MTS.

Тенденция, однако. Интефейс все проще, реализации все изощреннее. Во что это упрется, интересно...
GS
Re[2]: Runtime определение апартмента из кода.
От: Алекс Россия http://wise-orm.com
Дата: 26.10.02 11:09
Оценка:
Здравствуйте Ivan, Вы писали:

I>Здравствуйте George Seryakov, Вы писали:


хъ

I>Можно использовать HRESULT CoGetApartmentID(DWORD dwFlag, DWORD* pID);

I>dwFlag надо передавать -1
I>Функция недокументированная.
I>Находится в ole32.dll
I>С ее помощью всегда можно определить в каком апартменте мы сейчас находимся

Интересная функция! По ней можно всю COM TLS изучить!
Re[2]: Runtime определение апартмента из кода.
От: Tom Россия http://www.RSDN.ru
Дата: 26.10.02 11:27
Оценка:
[skip]
Может ещё какие дровишки есть ?
Народная мудрось
всем все никому ничего(с).
Re[3]: Runtime определение апартмента из кода.
От: Ivan Россия www.rsdn.ru
Дата: 26.10.02 12:33
Оценка:
Здравствуйте Tom, Вы писали:

Tom>[skip]

Tom>Может ещё какие дровишки есть ?

Смотря о чем речь
Если из недокументированного , могу предложить CoGetCurrentLogicalThreadID — возвращает ID текущего логического потока (учитывает переключения потоков при пересечении границы апартмента)
Re[4]: Runtime определение апартмента из кода.
От: Tom Россия http://www.RSDN.ru
Дата: 26.10.02 12:43
Оценка:
Здравствуйте Ivan, Вы писали:

I>Здравствуйте Tom, Вы писали:


Tom>>[skip]

Tom>>Может ещё какие дровишки есть ?

I>Смотря о чем речь

I>Если из недокументированного , могу предложить CoGetCurrentLogicalThreadID — возвращает ID текущего логического потока (учитывает переключения потоков при пересечении границы апартмента)
Ну эту хрень я у себя в MSDN нашёл...
Народная мудрось
всем все никому ничего(с).
Re[5]: Runtime определение апартмента из кода.
От: Ivan Россия www.rsdn.ru
Дата: 26.10.02 12:45
Оценка:
Здравствуйте Tom, Вы писали:

Tom>Ну эту хрень я у себя в MSDN нашёл...

Это что за MSDN такой ? У меня в July 2002 описания нет
Re[6]: Runtime определение апартмента из кода.
От: Tom Россия http://www.RSDN.ru
Дата: 26.10.02 12:46
Оценка:
Здравствуйте Ivan, Вы писали:

I>Здравствуйте Tom, Вы писали:


Tom>>Ну эту хрень я у себя в MSDN нашёл...

I>Это что за MSDN такой ? У меня в July 2002 описания нет
July 2001. и то описание в Win CE и написано всего: "This function is not yet supported."
Народная мудрось
всем все никому ничего(с).
Re[7]: Runtime определение апартмента из кода.
От: Ivan Россия www.rsdn.ru
Дата: 26.10.02 12:51
Оценка:
Tom> is not yet supported.
Ага.. прошел год,а она все еще not supported
Re[2]: Runtime определение апартмента из кода.
От: George Seryakov Россия  
Дата: 28.10.02 17:43
Оценка:
Здравствуйте Ivan, Вы писали:

I>Можно использовать HRESULT CoGetApartmentID(DWORD dwFlag, DWORD* pID);

I>dwFlag надо передавать -1
I>Функция недокументированная.
I>Находится в ole32.dll
I>С ее помощью всегда можно определить в каком апартменте мы сейчас находимся

Работает, позволяет увидеть смену апартмента в тех случаях, когда он заведомо должен меняться. Под dllhost показывает MTA как 0xcccccccc. Вот только насчет "каком апартменте" — непонятно. Не видно как по номеру апартмента получить его тип. Кроме апартмента 0xcccccccc для конфигурированных компонент.
GS
Re[3]: Runtime определение апартмента из кода.
От: George Seryakov Россия  
Дата: 28.10.02 17:44
Оценка:
Здравствуйте Алекс, Вы писали:

I>>Можно использовать HRESULT CoGetApartmentID(DWORD dwFlag, DWORD* pID);


А>Интересная функция! По ней можно всю COM TLS изучить!


В смысле рассматривая ее винарный код? Поделись, что нарыл. А исходники wine ты посмотрел — там есть нечто, выдаваемое за структуру COM TSL.
GS
 
Подождите ...
Wait...
Пока на собственное сообщение не было ответов, его можно удалить.