惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

Exploit-DB.com RSS Feed
Exploit-DB.com RSS Feed
阮一峰的网络日志
阮一峰的网络日志
V
Visual Studio Blog
人人都是产品经理
人人都是产品经理
GbyAI
GbyAI
WordPress大学
WordPress大学
博客园 - 聂微东
大猫的无限游戏
大猫的无限游戏
博客园_首页
博客园 - 三生石上(FineUI控件)
Recorded Future
Recorded Future
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
P
Palo Alto Networks Blog
T
Threat Research - Cisco Blogs
L
LINUX DO - 热门话题
Latest news
Latest news
C
Cybersecurity and Infrastructure Security Agency CISA
Jina AI
Jina AI
Recent Announcements
Recent Announcements
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
N
Netflix TechBlog - Medium
The Cloudflare Blog
月光博客
月光博客
I
Intezer
S
Schneier on Security
雷峰网
雷峰网
NISL@THU
NISL@THU
P
Privacy & Cybersecurity Law Blog
L
Lohrmann on Cybersecurity
The GitHub Blog
The GitHub Blog
V
Vulnerabilities – Threatpost
Cyber Security Advisories - MS-ISAC
Cyber Security Advisories - MS-ISAC
Google DeepMind News
Google DeepMind News
P
Privacy International News Feed
V
V2EX
T
Tor Project blog
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
宝玉的分享
宝玉的分享
cs.CL updates on arXiv.org
cs.CL updates on arXiv.org
Microsoft Azure Blog
Microsoft Azure Blog
C
Cyber Attacks, Cyber Crime and Cyber Security
Engineering at Meta
Engineering at Meta
Hugging Face - Blog
Hugging Face - Blog
博客园 - Franky
Project Zero
Project Zero
G
GRAHAM CLULEY
有赞技术团队
有赞技术团队
Scott Helme
Scott Helme
Cisco Talos Blog
Cisco Talos Blog
P
Proofpoint News Feed

Все публикации подряд на Хабре

Ловим музу за клавиатуру: как айтишнику стать автором Что умеет Midjourney в 2026? Мой немного грустный разбор этого шикарного инструмента Никто не любит писать тесты, но ИИ может исправить это IPv8 выглядит как мечта. Поэтому почти наверняка не взлетит Производители вернули в продажу материнки с DDR3. Что происходит? Управление агентом с телефона через Telegram теперь в KodaCode От координации к лидерству: как меняется роль руководителя разработки Я сделала родителям бизнес вместо пенсии: зарабатываем 70 тысяч, мама не даёт продать В три раза быстрее приемка товара и оптимизация трудозатрат на 73%: как «РСТ-Инвент» помог Gulliver Group ИИ-шечный мир победил? О влиянии искусственного интеллекта на игропром Кремль снижает давление на Телеграмм пока Европа строит интернет по паспорту Как CEO, CTO и CIO за 8 часов собрали ИИ-директора, который умеет держать позицию под давлением Как (не) потерять домен за выходные Вместо 8 разных VPS: как я организовал практику студентам на одном сервере Почему твой Open Source проект не замечают? R&D: искусство управления неопределенностью в разработке AI-дефляция: вакансий для разработчиков больше, а рост зарплат — худший за 15 лет Мы отдали управление роботами OpenClaw. Что из этого вышло Галактический ID: система идентификации для всех форм разумной жизни Шесть основ бизнес-анализа: начинаем с вопроса «Кто в игре?» Код-ревью, в котором дело не в коде Данные переехали. Команда — нет Системной подход к сдаче OSWE в 2025 Почему комната управления реактором покрашена в цвет морской пены 4 YAML-файла вместо PySpark: как аналитикам строить пайплайны без разработчиков LLM-агент для поиска свободных доменов: автоматизируем подбор Когда, зачем и как правильно начинать новую сессию в Claude Code? Как я заставил нейросеть писать макросы для FreeCAD Анатомия ИИ‑агента для подбора персонала. От тысячи резюме к топ‑10 за минуты Опыт разработчика как экономика внимания Автономность как точка невозврата: кто будет субъектом в цифровом будущем Обучение ИИ в «диких» условиях: как рутинные действия превращаются в датасеты Как измерить LLM для задач кибербеза: обзор открытых бенчмарков Где хранить код? Сравнение GitHub, GitLab и Bitbucket Математика объясняет, почему нормальное распределение встречается повсюду Почему ваш FinOps не работает: 12 тезисов от практиков Как подписать проектную документацию УКЭП с использованием бесплатных лицензий Pilot Адаптивное администрирование Sigla Vision Я грузил уран в бочки, а потом 20 лет строил ИТ в атомной отрасли Чем позвонить с Эвереста? История и обзор спутниковой связи. Часть 2 Как языковая модель помогает контролировать качество инструктажей по охране труда в металлургии Как не передать на desktop свой IP в РКН Анатомия SAP Privileges: как устроено управление правами в macOS MoneyDev: Сказка про три главных слова Обновлённый токенизатор видео K-VAE 2.0 от Сбера Как сделать диспетчеризацию дома на 1284 квартиры почти бесплатно Как мы разогнали железную дорогу Мы дали агентам рутину. Теперь надо решить — что делать с освободившимся временем Токсичный контент, промпт-хакинг и защита ИИ — всё о Guardrails для LLM Умный город начинается с точного взгляда: как «Фалькон Тех» меняет пространство к лучшему Навайбкодил приложение для анализа графов Почему Дюну так интересно читать? Упрощаем работу с рутиной или как стать Гендальфом Белым Деконструкция Go: CPU, RAM и что там происходит. Go Assembler база. Часть 1.1 Какие профессии исчезнут из-за ИИ, а какие появятся? И что с этим делать Как мы построили IT-отдел, где хочется расти: архитектурные встречи, прозрачные метрики и книжные подарки Rufler: Делаем из Claude Code автономный рой через один YAML-конфиг Sing-box и белый список приложений Как построить надёжный обмен сообщениями в микросервисах: лучшие практики для enterprise OpenAI строит MLM-пирамиду, а McKinsey и Accenture помогают ей в этом Дом, который не построил Фишер (Часть 2) «Сверхзвуковой математик» против «Вдумчивого логиста»: битва алгоритмов 3D-упаковки Мультимодальные модели – грубый и дорогой инструмент Разговоры ничего не стоят. Код тоже Проверки физических лиц: с кого начнет ФНС Топ-10 бесплатных нейросетей для создания видео в 2026 году Первые слои кода: как наши решения сегодня определяют архитектуру ИИ на десятилетия Разработка нового статического анализатора: PVS-Studio JavaScript Поиск уязвимостей ПО: базовый минимум или роскошный максимум Почему оценка персонала не работает как инструмент управления Как мы разработали ИИ-ассистента и сократили рутину продуктовой команды на 50% Как я ушел из найма, нажарил косточек и продал на маркетплейсах на 168 млн в год Когда 1С:ERP уже внедрена, а нормального производственного плана всё ещё нет Как я сделал Claude мультимодальным, подключив к нему Qwen Omni Как приглашение на вакансию мечты превращается в атаку Infrastructure as Code: философия и лучшие практики IaC Тестируем Yandex Code Assistant на задаче, в которой нужно хранить секреты nxs-universal-chart v3.0: новое поколение универсального Helm-чарта Callback Injection: Техника, которая отправила Microsoft Defender в глухой нокаут «Все идеи на стол»: митап как способ вывести проект из тупика Сегодня я узнал нечто новое о GPU благодаря багу в своей игре Как заставить LLM ̶ ̶г̶а̶л̶л̶ю̶ ̶ эволюционировать Карта событий как фундамент аналитики: практический кейс для E-commerce Что выбрать для AI: x86, ARM или RISC-V? Дайджест железа за март Роль соматических мутаций в развитии аутоиммунных заболеваний: путь к избирательной терапии Mythos от Anthropic — тревожный сигнал для всех, а не только для банков Guardrails для LLM на Java: как приручить промпт‑инъекции и токсичные ответы Green-VLA: как мы собрали VLA-модель для реального антропоморфного робота и не потеряли обобщение Финансовая гонка вооружений: почему умные люди добровольно в ней участвуют Эра ИИ-агентов наступила: выбираем лучшего цифрового сотрудника # Практический опыт внедрения WinCC Redundancy на производственном предприятии Сделал MVP за 3 дня, а потом неделю прикручивал оплату. Оно того стоило? Физика против Маска: почему Starship V3 может оказаться ещё одной катастрофой Нефть Венесуэлы: крупнейшие запасы в мире, но не крупнейшая нефтяная держава JPA 4. Переосмысление Hibernate Почему зеркальная фотокамера Nikon D5 десятилетней давности идеально подошла для миссии «Артемида-2» Проект «Уровень-Спутник» или как мы сделали платформу для гидрологов «Замедлиться, чтобы ускориться»: почему ИИ повышает цену ошибок в требованиях и архитектуре Как с нуля поднять трафик IT-компании на 1657% при бюджете 55 тыс. и выжить Pixel-perfect Downsampling — идеальная отрисовка 50 миллионов точек без потерь
Компиляция yolov8n в формат HEF для Hailo-8L на Raspberry Pi 5
Oleg L · 2026-06-18 · via Все публикации подряд на Хабре

Простой

7 мин

0

Введение

Уже некоторое время увлекаюсь машинным обучением и нейросетями. В какой‑то момент стало интересно снабдить имеющуюся Raspberry Pi 5 нейрочипом, который берет на себя нагрузку по работе с нейронными моделями. В качестве первого экземпляра для тренировки и опытов был приобретен модуль Pi AI Hat+ с чипом Hailo-8L на борту. Данная версия является младшей в линейке и характеризуется производтельностью 13 TOPS. На следующем по рангу модуле стоит уже чип Hailo-8 с 26 TOPS. Подробнее про разновидности, характеристики и первичные настройки можно прочитать тут.

Но дело в том, что просто так модельки запустить на этом чипе не получится, так как он не принимает стандартные экспорты типа onnx. Чипу нужен свой бинарник hef, который можно скомпилировать специальным ПО. Называется оно Hailo DataFlow Compiler (DFC). Скачать его можно с фирменного сайта Hailo, предварительно там зарегистрировавшись. К слову, DFC работает только на Linux, версии для Win нету. Для компиляции моделей само железо (Raspberry Pi) не обязательно, а скорее будет не очень подходяще, так как медленное. Процесс ресурсоемкий, поэтому лучше делать это на ПК с Ubuntu, в моем случае 24.04.

Установки DFC оказалось мало, при попытке двумя кликами скомпилировать нужный файл столкнулся с проблемами, чтобы решить которые пришлось покопаться. Собственно, чтобы самому не забыть в будущем (и можно было подсмотреть) и заодно чтобы следующим путникам было проще, решил написать статью. Пишу от руки, без ИИ, и если что, то нужно «понять и простить».

Итак, в этой статье мы разберем, как взять готовую модель (например, YOLO), оптимизировать ее и скомпилировать в проприетарный формат .hef для запуска на NPU Hailo. На сайте Hailo подробно описан процесс установки DFC, на нем останавливаться не будем. Тормознем только на процессе компиляции.

1. Экспорт модели в onnx.

Сначала нужно принять во внимание, что hef не понимает динамические входы, поэтому при экспорте сразу это учитываем.

from ultralytics import YOLO  
  
print("Подгрузка модели")  
model = YOLO("yolov8n.pt")  
  
print("Экспорт модели")  
model_onnx = model.export(  
    format="onnx",  
    imgsz=(480, 640),  
    dynamic=False,  
    simplify=True  
)  
  
print("Экспорт выполнен")

Для «своего» кода у меня в привычку вошло вместо закомментенных пояснений писать принты, так проще для отладки, посему их будет много.
В результате вышенаписанного кода у нас рождается файл yolov8n.onnx

2. Парсинг.

На следующем этапе нужно провести так называемый парсинг модели и создать специальный промежуточных архивный har‑ файл (Hailo Archive). Это некий предварительный контейнер для нейросети. И вот тут начинается интересное. Чип Hailo не может съесть всю модель целиком за один присест. Причины следующие:

  • Чип изначально рассчитан на статические матричные вычисления (свертки), а постобработка гибкая и быстрее выполнится в CPU.

  • Конечные слои модели содержат динамические слои типа Reshape/Transpose и алгоритм Non Maximum Supression (NMS), а чип Hailo не умеет работать с такими узлами.

  • Настройки уверенности (Confidence Threshold) иногда приходится корректировать, как и параметры NMS. И для любого изменения, если бы они жестко зашивались в скомпилированный бинарник hef, пришлось бы заново его компилировать.

Перед созданием har‑файла на модель нужно посмотреть через приложение Netron, которое доступно в виде web‑приложения. Открываем нашу onnx модель в этом приложении и почти в конце графа находим узлы свертки, на которых мы обрежем сеть для har‑файла.


Эти узлы нужно обозначить как конечные. Выглядит это так:

from hailo_sdk_client import ClientRunner  
  
TARGET_CHIP = "hailo8l"  
END_NODES = [  
    "/model.22/cv2.0/cv2.0.2/Conv",  
    "/model.22/cv3.0/cv3.0.2/Conv",  
    "/model.22/cv2.1/cv2.1.2/Conv",  
    "/model.22/cv3.1/cv3.1.2/Conv",  
    "/model.22/cv2.2/cv2.2.2/Conv",  
    "/model.22/cv3.2/cv3.2.2/Conv",  
]  
  
print("Формирование har")  
runner = ClientRunner(hw_arch=TARGET_CHIP)  
runner.translate_onnx_model(  
    "yolov8n.onnx",   
    end_node_names=END_NODES  
)

Далее сохраняем har‑файл, он нам понадобится.

print("Сохранение har файла модели")  
runner.save_har("yolov8n.har")

3. Подготовка конфигурации

Теперь нужно подготовить конфигурационные файлы для компиляции. Они включают в себя alls‑файл (Allocator Script) и json.

В alls‑файле нужно указать параметры нормализации и NMS. Но помимо этого нужно еще указать наши ветки классификации (их у нас три из разных слоев нейросети), для которых нужно применить сигмоидную функцию, чтобы в результате значения на выходе у них были в диапазоне [0, 1].
И чтобы получить правильные названия этих узлов классификации, какими они обзываются в подготовленном контейнере har, нам нужно этот har так же, как и перед этим onnx, открыть через Netron.
В приложении слева сразу выбираем пункт с характеристиками (Properties):

И справа видим нужную нам информацию:

Из этого сейчас нам потребуются только узлы с классификацией, у которых shape заканчивается на 80 (последняя размерность, указывающая на количество детектируемых классов модели).
Т.е. нам нужны conv42, conv53 и conv63

Создаем alls со следующим содержимым:

nomalization1 = normalization([0.0, 0.0, 0.0], [255.0, 255.0, 255.0])  
change_output_activation(conv42, sigmoid)  
change_output_activation(conv53, sigmoid)  
change_output_activation(conv63, sigmoid)  
nms_postprocess("yolov8n.json", meta_arch=yolov8, engine=cpu)  
allocator_param(width_splitter_defuse=disabled)

Также создаем соответствующий json, в котором прописываем «динамические» параметры будущей модели:

{  
    "nms_scores_th": 0.2,  
    "nms_iou_th": 0.6,  
    "image_dims": [  
        480,  
        640  
    ],  
    "max_proposals_per_class": 100,  
    "classes": 80,  
    "regression_length": 16,  
    "background_removal": false,  
    "bbox_decoders": [  
        {  
            "name": "bbox_decoder02",  
            "stride": 8,  
            "reg_layer": "conv41",  
            "cls_layer": "conv42"  
        },  
        {  
            "name": "bbox_decoder12",  
            "stride": 16,  
            "reg_layer": "conv52",  
            "cls_layer": "conv53"  
        },  
        {  
            "name": "bbox_decoder22",  
            "stride": 32,  
            "reg_layer": "conv62",  
            "cls_layer": "conv63"  
        }  
    ]  
}

В json также указываем названия наших нод (уже всех: и для регрессии рамок, и для классификации), соответствующие им страйды, количество классов и размерность входа с параметрами порогов.

Подгружаем эти данные в заготовку для компиляции:

print("Загружаем alls")  
runner.load_model_script("yolov8n.alls")

4. Квантование модели

Для формирования целочесленных весов нам нужно прогнать приближенные к целевым изображения через модель, чтобы откалибровать правильные коэффициенты нейронов. Для этого можно использовать имеющийся в ultralytics датасет, предварительно сформировав из него нужный нам набор (обращаем внимание на размерности):

import numpy as np  
from PIL import Image  
from ultralytics.utils import DATASETS_DIR  
from ultralytics.data.utils import check_det_dataset

CALIB_IMAGES = 128

print("Создаем датасет")  
check_det_dataset("coco128.yaml")  
calib_dir = DATASETS_DIR/"coco128"/"images"/"train2017"  
image_files = list(calib_dir.glob("*.jpg")) + list(calib_dir.glob("*.png"))  
if not image_files:  
    raise FileNotFoundError(f"No calibration images found in {calib_dir}")  
  
calibset = np.zeros((CALIB_IMAGES, 480, 640, 3), dtype=np.float32)  
for i in range(CALIB_IMAGES):  
    img = Image.open(np.random.choice(image_files)).convert("RGB").resize((640, 480))  
    calibset[i] = np.array(img, dtype=np.float32)

Этот датасет прогоняем через заготовку модели:

print("Оптимизируем веса к датасету")  
runner.optimize(calibset)

После чего можно для себя сохранить уже оптимизированную модель:

print("Сохранение оптимизированного har-файла")  
runner.save_har(f"{MODEL}.o.har")

5. Компиляция

Мы подошли к завершающему этапу. Теперь можно скомпилировать преследуемый бинарник соответствующим методом, компилятор распределяет веса модели по вычислительным элементам чипа Hailo.

hef = runner.compile()

После чего сохраняем наш скомпилированный hef.

with open(f"{MODEL}.hef", "wb") as f:  
    f.write(hef)

Также можно сделать командой:

# Компиляция через CLI (указываем архитектуру чипа, для RPi это hailo8l)
hailo compiler yolov8n_quantized.har --hw-arch hailo8l --output-edf yolov8n.hef

Но я сделал первым методом. Второй для справки.

Заключение

Файл yolov8n.hef готов к переносу на Raspberry Pi 5. Его можно запускать через hailortcli или использовать в пайплайнах GStreamer.
Если моя инструкция вам помогла, плюсаните.
Ниже полный листинг:

from hailo_sdk_client import ClientRunner  
import numpy as np  
from PIL import Image  
from ultralytics.utils import DATASETS_DIR  
from ultralytics.data.utils import check_det_dataset  
  
# НАСТРОЙКИ  
MODEL = "yolov8n"  
model = "yolov8n.onnx"  
TARGET_CHIP = "hailo8l"  
END_NODES = [  
    "/model.22/cv2.0/cv2.0.2/Conv",  
    "/model.22/cv3.0/cv3.0.2/Conv",  
    "/model.22/cv2.1/cv2.1.2/Conv",  
    "/model.22/cv3.1/cv3.1.2/Conv",  
    "/model.22/cv2.2/cv2.2.2/Conv",  
    "/model.22/cv3.2/cv3.2.2/Conv",  
]  
CALIB_IMAGES = 128  
  
print("Формирование har")  
runner = ClientRunner(hw_arch=TARGET_CHIP)  
runner.translate_onnx_model("yolov8n.onnx", end_node_names=END_NODES)  
  
print("Сохранение har файла модели")  
runner.save_har("yolov8n.har")  
  
print("Выходные слои модели")  
hn_model = runner.get_hn_model()  
for layer in hn_model.get_output_layers():  
    print(layer.name)  
  
print("Загружаем alls")  
runner.load_model_script("yolov8n.alls")  
  
print("Создаем датасет")  
check_det_dataset("coco128.yaml")  
calib_dir = DATASETS_DIR/"coco128"/"images"/"train2017"  
image_files = list(calib_dir.glob("*.jpg")) + list(calib_dir.glob("*.png"))  
if not image_files:  
    raise FileNotFoundError(f"No calibration images found in {calib_dir}")  
  
calibset = np.zeros((CALIB_IMAGES, 480, 640, 3), dtype=np.float32)  
for i in range(CALIB_IMAGES):  
    img = Image.open(np.random.choice(image_files)).convert("RGB").resize((640, 480))  
    calibset[i] = np.array(img, dtype=np.float32)  
  
print("Оптимизируем веса к датасету")  
runner.optimize(calibset)  
  
print("Сохранение оптимизированного har-файла")  
runner.save_har(f"{MODEL}.o.har")  
  
hef = runner.compile()  
with open(f"{MODEL}.hef", "wb") as f:  
    f.write(hef)