Дата публикации:Wed, 27 Aug 2025 18:02:04 +0300
После шести месяцев разработки представлен релиз проекта LLVM 21.1.0, развивающего инструментарий (компиляторы, оптимизаторы и генераторы кода), компилирующий программы в промежуточный биткод RISC-подобных виртуальных инструкций (низкоуровневая виртуальная машина с многоуровневой системой оптимизаций). Сгенерированный псевдокод может быть преобразован в машинный код для заданной целевой платформы или использован JIT-компилятором для формирования машинных инструкций непосредственно во время выполнения программы. На базе технологий LLVM проектом развивается компилятор Clang, поддерживающий языки программирования C, C++ и Objective-C. Начиная с ветки 18.x проект перешёл на новую схему формирования номеров версий, в соответствии с которой нулевой выпуск ("N.0") используется в процессе разработки, а первая стабильная версия снабжается номером "N.1".
Среди улучшений в Clang 21:
- Возможности, связанные с С++: По аналогии с GCC реализована возможность использования константных выражений в ассемблерных вставках, определяемых директивой "asm": int foo() { asm((std::string_view("nop")) ::: (std::string_view("memory"))); } Добавлены расширенные варианты выражений "new" и "delete", поддерживающие аргумент "std::type_identity<T>", через который можно указать информацию о типе объекта, для которого выделяется или освобождается память. Добавлена возможность вычисления на этапе компиляции (в контексте константного выражения) лямбда-функций, захватывающих структурированные привязки (structured binding) .
- В структурированные привязки добавлена возможность использования синтаксиса "..." для указания пакетов (pack), захватывающих оставшееся число элементов из присваиваемой последовательности. auto [x,y,z] = f(); // в переменные x, y, z будут записаны три элемента, возвращённые f(). auto [...xs] = f(); // в пакет xs будут записаны все элементы, возвращённые f(). auto [x, ...rest] = f(); // В x будет записан первый элемент, а в rest - остальные. auto [x, y, ...rest] = f(); // В x будет записан первый элемент, в y - второй, а в rest - третий. auto [x, ...rest, z] = f(); // в x - первый, в rest - второй, в z - третий. Добавлена поддержка "тривиальной перемещаемости" типов (Trivial Relocatability), позволяющей оптимизировать перемещения объектов заданного типа через их клонирование в памяти без вызова конструкторов или деструкторов. Для классов реализованы свойства memberwise_trivially_relocatable и memberwise_replaceable, а для низкоуровневого перемещения одного или нескольких объектов добавлены функции trivially_relocate_at и trivially_relocate. Появилась возможность применения структурированного связывания (structured binding) в качестве условия в операторах "if" и "switch". Реализована поддержка прикрепления функции main() к глобальному модулю и определения функции main() в именованных модулях.
- Устранено неопределённое поведение при использовании выражений с типом void в некоторых контекстах, например, "(void)(void)1;". Разрешено не завершать файл с исходным кодом символом новой строки. Добавлены новые префиксы для восьмеричных литералов - "0o" и "0O", а также восьмеричные и шестнадцатеричные escape-последовательности "\o{...}" и "\x{...}". Поддержка восьмеричных литералов 0xxx объявлена устаревшей. Например, "0o123" и "\o{123}" вместо "0123". Добавлен оператор "_Countof" для определения количества элементов в массиве. Также добавлен заголовочный файл stdcountof.h, определяющий вариант макроса "countof", реализованный через "_Countof".
- Разрешено переопределять tag‑типы (struct, union, enum) в пределах одного блока трансляции, если повторные определения структурно эквивалентны (то же число членов, одинаковые типы и имена тегов). Упрощено использование списков с переменным числом аргументов (variadic). Разрешено использовать одиночный вариативный параметр в имени типа. Добавлена совместимая с GCC встроенная функция "__builtin_c23_va_start()", улучшающая поведение диагностики для макроса va_start() в режиме C23.
- "-Wdefault-const-init-var" и "-Wdefault-const-init-field" - выявление помеченных признаком const переменных и полей, определённых без явной инициализации. "-Wimplicit-void-ptr-cast" - диагностика неявного преобразования из типа "void*" в другой тип указателя. "-Wc++-keyword" - выявление использования ключевых слов "C++" в качестве идентификаторов в "C". "-Wc++-hidden-decl" - выявление использования типов тегов, видимых в "C", но не видимых в "C++" из-за ограничения области видимости. struct S { struct T { int x; } t; }; struct T t; // предупреждение, так как корректно в Си, но некорректно в C++ "-Wimplicit-int-enum-cast" - выявление неявных преобразований в С-коде из целочисленных типов в тип перечислений, несовместимых с "C++". "-Wtentative-definition-compat" - диагностика повторяющихся определений в "C", несовместимых с "C++" (например, "int i;int i;"). "-Wunterminated-string-initialization" и "-Wc++-unterminated-string-initialization" - выявление операций инициализации строковых литералов, в которых не вмещается разделитель с нулевым кодом. Для пометки полей и переменных в коде на Си, не требующих финального нулевого символа, добавлен атрибут "nonstring". char buf1[3] = "foo"; // предупреждение char buf2[3] = "fo\0"; // всё Ok. "-Wjump-misses-init" - диагностика перехода через goto или switch/case, пропускающего инициализацию локальной переменной. "-Wundef-true" - предупреждает об использовании значения "true" в препроцессоре C без определения. "-Wnrvo" - диагностика пропущенных NRVO (Named Return Value Optimization).
- "-fprofile-continuous" - включение непрерывной синхронизации профиля в файл. "-ftime-report-json" - вывод сведений о времени компиляции в формате JSON. "-ignore-pch" - отключение предкомпилированных заголовков. "-fthinlto-distributor" и "-Xthinlto-distributor" - для применения DTLTO (Integrated Distributed ThinLTO). "-static-libclosure" - для статического связывания runtime расширения Blocks на платформе Windows.
Новость позаимствована с opennet.ru
Ссылка на оригинал: https://www.opennet.ru/opennews/art.shtml?num=63783