Мне сложно судить о практической ценности данной статьи, поскольку я уже весьма далек от .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.

И где здесь засада, спросишь ты?

combase!PeekRPCAndDDEMessage - привет из 90-х
combase!PeekRPCAndDDEMessage - привет из 90-х

Этот цикл 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)


  1. MonkAlex
    15.06.2026 09:30

    Так где дедлок? Буквально первая рекомендация времён async-await - делать ConfigureAwait(false) и не лочить UI поток, она же получается решает вопросы.


    1. steelfactor Автор
      15.06.2026 09:30

      Ну, раз вы настаиваете...
      Совет «не лочить 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-вызов вызывают удивление на лице разраба


      1. MonkAlex
        15.06.2026 09:30

        - .Result / .Wait на коде, который вы не контролируете;

        Засовываем его в Task.Run и STA-треду должно быть хорошо. Не идеальное решение, но рабочее.

        - lock, Sleep, блокирующий I/O, тяжелое вычисление прямо на UI-потоке повесят цикл сообщений независимо ни от какого ConfigureAwait;

        и их тоже в Task.Run, да.

        ну самое и главное, COM-сторона...

        вот тут я уже не уверен, возможно действительно там можно устроить дедлок легким движением. Работал с офис-интеропом и с 1с старым, но там вся апишка выглядела как "синхронная", но при этом вроде не завязывалась на UI-тред и могла работать вне его. Не уверен, допускаю что эта проблема актуальна и её стоит опасаться.

        Не блокировать UI-поток\очередь сообщений в целом хорошо, тут соглашусь полностью.


        1. steelfactor Автор
          15.06.2026 09:30

          Спасибо за честный, конструктивный и въедливый ответ, который приятно читать, редкое качество, честно.
          По первым двум пунктам - отдать все на съедение Task.Run, абсолютно согласен, не идеально, но рабочее решение.
          Насчет Office-interop и 1С - вы правы 100%, они умеют работать вне UI-треда, но причина не в Task.Run, а в привязке самого объекта:он или создавался и обрабатывался в отдельном потоке, либо он был threadingModel=Both, то есть не был привязан занятому UI-треду. То есть всё упиралось в апартамент и threading model объекта, а не в то, откуда вы его дергаете.
          По поводу COM-стороны. Я в статье пытался показать не то, как решить проблему непонятного дедлока, а почему он возникает.
          На уровне COM Task.Run не работает так, как вы от него ждете. STA-объект привязан к апартаменту, то есть к потоку, который его создал и вызовы к нему обязаны обслуживаться на этом потоке. Вызовом Task.Run() с пул-потока маршалится обратно в апартамент объекта, то есть в ваш UI-тред, и если тот заблокирован и не обрабатывает очередь, то получите дедлок.
          Что я хотел сказать в статье - не давать вам рабочее решение, там нет таких советов. Я про то, что понимание механизма на уровне "я думал это дно, но тут снизу постучали" должно предостеречь от неочевидных ошибок


  1. corcheaginaksu
    15.06.2026 09:30

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


    1. steelfactor Автор
      15.06.2026 09:30

      Плюс от меня, согласен абсолютно. Могу только отметить, что это похоже на костыль, вплетать LPC-вызов в очередь сообщений PeekMessageWDispatchMessageW