Мне сложно судить о практической ценности данной статьи, поскольку я уже весьма далек от .NET-разработки прикладных приложений, статья написана больше в исследовательских целях разгребания окаменелых кхм.. мамонта, и, возможно, кто-то из практикующих программистов найдет ее забавной.
На просторах мне не удалось обнаружить каких-либо упоминаний об этой проблеме, вскользь лишь исследователь Joe Duffy писал об этом в своем блоге. Еще косвенно было упоминание вот на этой google-борде. Отсутствие упоминаний об этом явлении должно говорить о околонулевой ценности исследования, но статья сама себя не напишет.
Речь пойдет об очень необычной причине дедлока GUI-потока в оконных приложениях .NET – WinForms и WPF. Когда такое приложение виснет намертво, разработчик ищет причину там, где привык: в собственных локах, в гонке между своими потоками, в неудачном lock или забытом .Wait(). Он перетряхивает свой код - и нередко не находит ничего криминального. Потому что причина лежит не этажом ниже, чем он смотрит, и даже не в подвале: в механизме, который он не писал, не видит и с которым, как ему кажется, вообще не работает.
Если вы работаете с WinForms/WPF, то вы наверняка знаете, что весь GUI-стек построен на однопоточной модели COM - Single-threaded apartments: буфер обмена, drag-drop, общие диалоги, shell-интеграция, OLE - всё это STA-компоненты.
У WinForms сгенерированная точка входа Main имеет явный атрибут [STAThread] (видно прямо в шаблоне), у WPF точка входа генерируется автоматически и тоже имеет атрибут [STAThread].
Так вот, именно эта модель, точнее, тот самый однопоточный апартамент (STA) с обработкой оконных сообщений, на котором незаметно для всех стоит GUI-поток любого WinForms- и WPF-приложения. Дедлок, выглядящий как баг async/await, на деле - тридцатилетняя машина COM, заглохшая под капотом.
Что скрывает COM за реализацией однопоточной модели STA? Собака, как оказалась, порылась весьма в интересном месте.
Абсолютно любой вызов CoInitializeEx() влечет за собой создание невидимого окна с классом «OleMainThreadWndClass». Казалось бы, зачем COM создавать окна и тем более работать с ними? Исторически дело обстояло так. Сначала появилась технология OLE 1.0 (начало 90-х) - компаундные документы, и его механизм межпроцессного взаимодействия IPC был буквально построен на DDE – Dynamic Data Exchange, то есть на оконных сообщениях (WM_DDE_*), что позволяло одной программе запрашивать данные у другой. Потом под OLE 2.0 в 1993 году Microsoft переписала всё на новый объектный слой - COM, заменив DDE нормальным (интересно, в каком месте он нормальный – прим. моё) маршалингом. И для обратной совместимости с OLE-компонентами оставили в COM оставили работу с окнами. И да, забегая вперед скажу, что где-то очень глубоко в недрах COM сидит обработка DDE-сообщений из начала 90-х.
Но обо всем по порядку.
Вызов CoInitializeEx() активирует цепочку вызовов API ThreadFirstInitialize() -> RegisterOleWndClass(), которая, как я писал, создает скрытое окно с классом «OleMainThreadWndClass». Внутри OleMainThreadWndProc нашлось неожиданно мало. Я ожидал увидеть диспетчер вызовов. Но его там не оказалось - сама процедура есть, а вот диспетчеризации в ней нет: внутри есть обработка только трех типов сообщений - на WM_CLOSE и WM_DESTROY процедура лишь пишет трассировку и возвращает ноль - окно даже не сносится, всё сводится к логированию и защите времени жизни. На приватном WM_USER+5 она через GetSingleThreadedHost отдаёт указатель на host-объект апартамента. Остальное уходит в DefWindowProcW. И все. Тупик.
Ладно, пойдем с другой стороны. В Ghidra задизасмил вызов того, что STA-поток реально вызывает, когда находится в режиме ожидания: функции CoWaitForMultipleHandles. И все сложилось: CoWaitForMultipleHandles() дергает ClassicSTAThreadWaitForHandles(), а та в свою очередь создает «клиентский» цикл CCliModalLoop::BlockFn(), где видны вызовы:
- InitChannelIfNecessary - API поднимает RPC-канал: входящий вызов едет по LPC, а не «сообщением в окно»;
- API MsgWaitForMultipleObjectsEx - спим до срабатывания хэндла или прихода сообщения в очередь (!);
- PeekRPCAndDDEMessage выгребает из очереди именно RPC- и DDE- сообщения.
Итого имеем: вызов приходит по LPC, но его доставка вплетена в очередь сообщений, и модальный цикл достаёт её, пока поток ждёт.
Я думал это дно, но тут снизу постучали.
Вызов PeekRPCAndDDEMessage через обертку CCliModalLoop::MyPeekMessage дергает PeekMessageW (PM_REMOVE) + DispatchMessageW, ровно те же два вызова, как в любом учебнике while (GetMessage()) DispatchMessage() любого GUI.
И где здесь засада, спросишь ты?

Этот цикл PeekMessage/DispatchMessage - ты привык считать «своим», родным: он организует обработку кликов, перерисовок, ввода. И это тот же цикл, которым COM доставляет входящие вызовы, а async - свои продолжения.
Одна очередь, один поток, одна пара вызовов на всех. И - две ловушки, обе из того, что цикл обработки сообщений общий.
Заблокируешь его вызовами .Result, .Wait(), lock, тяжёлым расчётом - и ты не просто «подвесил UI на секунду», ты остановил единственный механизм, которым к тебе придёт тот самый результат, которого ты ждёшь, ждёшь ответа по каналу, который сам же и заглушил.
Запустишь цикл обработки сообщений не там, где надо, и Application.DoEvents(), модальный диалог, MessageBox, да хоть обычный await - цикл посреди твоего кода вытащит из очереди следующее сообщение и продиспетчерит его. Твой обработчик войдёт повторно, пока первый ещё на стеке: клик поверх клика, таймер поверх обработчика, COM-callback поверх метода. И пошло, поехало…
Те же два вызова, которым учат на первом уроке по Win32, под капотом оказываются транспортом и COM, и async, там где-то окаменелом 30-летнем наследстве от OLE - в простой обработке оконных сообщений, которая на первый взгляд выглядит безобидной.
Что это значит на практике?
Вам не нужно «работать с COM», чтобы влететь в COM STA-дедлок. Если вы в UI-потоке .NET - вы уже на COM STA. А STA-поток обязан непрерывно обрабатывать очередь сообщений; стоит ему заблокироваться - обработка прекращается, и входящий вызов не доставляется.
Что делать?
Не парковать UI-поток на асинхронной операции или тяжелых вычислениях. Его единственная работа, которая не должна прерываться, крутить цикл сообщений. А как это сделать, решать уже вам.
Комментарии (6)

corcheaginaksu
15.06.2026 09:30Этот дедлок является системным ограничением среды исполнения, где любой блокирующий вызов в UI-потоке не просто стопорит поток, а перекрывает единственный доступный транспорт для маршалинга ответов и обработки событий

steelfactor Автор
15.06.2026 09:30Плюс от меня, согласен абсолютно. Могу только отметить, что это похоже на костыль, вплетать LPC-вызов в очередь сообщений
PeekMessageW+DispatchMessageW
MonkAlex
Так где дедлок? Буквально первая рекомендация времён async-await - делать ConfigureAwait(false) и не лочить UI поток, она же получается решает вопросы.
steelfactor Автор
Ну, раз вы настаиваете...
Совет «не лочить UI-поток и ставить ConfigureAwait(false)» абсолютно верный и справедливый и я в статье с ним не спорю. Я вообще про то, не про то, как избежать дедлока, а про то, почему он возникает.
Вы пишите: "ConfigureAwait(false) же решает". Он решает один частный случай, когда продолжение после await хочет вернуться на UI-поток. Ок, с этим все гуд.
Но:
ConfigureAwait(false) вообще мимо когда:
- .Result / .Wait на коде, который вы не контролируете;
- lock, Sleep, блокирующий I/O, тяжелое вычисление прямо на UI-потоке повесят цикл сообщений независимо ни от какого ConfigureAwait;
- ну самое и главное, COM-сторона: если реальный STA-COM-объект (Office interop, shell-расширение, out-of-proc сервер) попытается вызвать вас обратно на UI-поток, а вы его заблокировали - это не async-продолжение, а межапартаментный COM-вызов по той же очереди, и ConfigureAwait тут ни при чём.
Так что дедлок вполне реальный и до сих пор массовый.
В статье хотел показать единый механизм под всеми этими случаями: заблокировал STA-поток → встал цикл сообщений → доставка, будь то async-продолжение или COM-вызов вызывают удивление на лице разраба
MonkAlex
Засовываем его в Task.Run и STA-треду должно быть хорошо. Не идеальное решение, но рабочее.
и их тоже в Task.Run, да.
вот тут я уже не уверен, возможно действительно там можно устроить дедлок легким движением. Работал с офис-интеропом и с 1с старым, но там вся апишка выглядела как "синхронная", но при этом вроде не завязывалась на UI-тред и могла работать вне его. Не уверен, допускаю что эта проблема актуальна и её стоит опасаться.
Не блокировать UI-поток\очередь сообщений в целом хорошо, тут соглашусь полностью.
steelfactor Автор
Спасибо за честный, конструктивный и въедливый ответ, который приятно читать, редкое качество, честно.
По первым двум пунктам - отдать все на съедение Task.Run, абсолютно согласен, не идеально, но рабочее решение.
Насчет Office-interop и 1С - вы правы 100%, они умеют работать вне UI-треда, но причина не в
Task.Run, а в привязке самого объекта:он или создавался и обрабатывался в отдельном потоке, либо он был threadingModel=Both, то есть не был привязан занятому UI-треду. То есть всё упиралось в апартамент и threading model объекта, а не в то, откуда вы его дергаете.По поводу COM-стороны. Я в статье пытался показать не то, как решить проблему непонятного дедлока, а почему он возникает.
На уровне COM Task.Run не работает так, как вы от него ждете. STA-объект привязан к апартаменту, то есть к потоку, который его создал и вызовы к нему обязаны обслуживаться на этом потоке. Вызовом Task.Run() с пул-потока маршалится обратно в апартамент объекта, то есть в ваш UI-тред, и если тот заблокирован и не обрабатывает очередь, то получите дедлок.
Что я хотел сказать в статье - не давать вам рабочее решение, там нет таких советов. Я про то, что понимание механизма на уровне "я думал это дно, но тут снизу постучали" должно предостеречь от неочевидных ошибок