Skip to main content

BoxDecode Decode Types

nodes::SimaBoxDecode converts raw detection-head tensors into detection results. It runs after model inference, applies the decode math for the selected model family, filters low-confidence boxes, runs NMS, and emits a tensor payload that starts with decoded boxes. Detection models can parse that payload as boxes; pose and segmentation models can also parse the keypoints or masks that follow the boxes.

For normal model-pack usage, prefer the Model-aware constructor. The model archive supplies the tensor order, layout, quantization, class count, resize metadata, and score-domain hints needed by the decoder. Your application usually only chooses the decode family and filtering thresholds.

Quick start

using namespace simaai::neat;

Model model("/path/to/yolov8_model.tar.gz");

auto boxdecode = nodes::SimaBoxDecode(
model,
BoxDecodeType::YoloV8,
/* detection_threshold */ 0.25,
/* nms_iou_threshold */ 0.45,
/* top_k */ 100);

For standalone stage usage:

simaai::neat::stages::BoxDecodeOptions opt(simaai::neat::BoxDecodeType::YoloV8);
opt.detection_threshold = 0.25;
opt.nms_iou_threshold = 0.45;
opt.top_k = 100;

Arguments

ArgumentMeaning
decode_typeModel family/head format, such as BoxDecodeType::YoloV8 or BoxDecodeType::YoloX. Required.
detection_thresholdMinimum score required to keep a detection. Use a model-appropriate value such as 0.25.
nms_iou_thresholdIoU threshold used by non-maximum suppression.
top_kMaximum number of detections to keep. 0 uses the backend/model default.
original_width, original_heightSource image size for coordinate mapping when using the raw-geometry constructor.
model_width, model_heightModel input size override. With the Model constructor this changes spatial decode knobs, not the packaged tensor contract.
resize_mode_overrideUse only when no upstream Preproc stage writes resize metadata and you need to specify stretch/letterbox/crop behavior explicitly.
decode_type_optionAdvanced sub-layout selector. Leave as Auto for model-pack usage unless you know the exported head layout.

Inputs and outputs

Input: raw detection tensors from the model. The expected tensor shapes depend on the model family. With an MPK/model archive, Neat reads those details from the packaged contract.

Output: one BoxDecode tensor containing decoded detections. Detection models use the standard BBOX payload. Pose and segmentation models keep the same leading boxes and append their task-specific payload:

Model taskC++ helperPython helperDecoded tensors
Detectiondecode_bbox(...)pyneat.decode_bbox(...)[N, 6] float32 boxes: x1, y1, x2, y2, score, class_id
Posedecode_pose(...)pyneat.decode_pose(...)boxes [N, 6] and keypoints [N, 17, 3] float32: x, y, visibility
Segmentationdecode_segmentation(...)pyneat.decode_segmentation(...)boxes [N, 6] float32 and masks [N, 160, 160] uint8

Detection-display graphs can feed the result to SimaRender. Application code that only needs boxes can continue to use decode_bbox(...) on BoxDecode outputs.

Decode type mapping

API enumBackend tokenTypical model family
BoxDecodeType::YoloyoloGeneric YOLO-style heads
BoxDecodeType::YoloV5yolov5YOLOv5 detection
BoxDecodeType::YoloV5Segyolov5-segYOLOv5 segmentation
BoxDecodeType::YoloV7yolov7YOLOv7 detection
BoxDecodeType::YoloV7Segyolov7-segYOLOv7 segmentation
BoxDecodeType::YoloV8yolov8YOLOv8 detection
BoxDecodeType::YoloV8Segyolov8-segYOLOv8 segmentation
BoxDecodeType::YoloV8Poseyolov8-poseYOLOv8 pose
BoxDecodeType::YoloV9yolov9YOLOv9 detection
BoxDecodeType::YoloV9Segyolov9-segYOLOv9 segmentation
BoxDecodeType::YoloV10yolov10YOLOv10 detection
BoxDecodeType::YoloV10Segyolov10-segYOLOv10 segmentation
BoxDecodeType::YoloV26yolo26YOLO26 detection
BoxDecodeType::YoloV26Poseyolo26-poseYOLO26 pose
BoxDecodeType::YoloV26Segyolo26-segYOLO26 segmentation
BoxDecodeType::YoloV6yolov6YOLOv6 detection
BoxDecodeType::YoloXyoloxYOLOX detection
BoxDecodeType::DetrdetrDETR-style transformer detection
BoxDecodeType::EffDeteffdetEfficientDet detection
BoxDecodeType::RcnnStage1rcnn-stage1R-CNN proposal stage
BoxDecodeType::CenternetcenternetCenterNet detection

BoxDecodeType::Unspecified is an unset sentinel and fails before runtime. Always choose a concrete type.

Choosing the right type

  • If you are using a SiMa-provided or SiMa-compiled model pack, choose the BoxDecodeType that matches the model family and leave decode_type_option as Auto.
  • If your detections are missing or all scores are unexpectedly low, first verify that the decode family matches the exported model head. YOLOX, YOLOv6, and YOLO26 use raw/logit-style heads and should not be treated like probability-only YOLO heads.
  • If boxes are shifted or scaled incorrectly, check the image resize policy. Use resize_mode_override only when your graph does not have an upstream Preproc stage writing resize metadata.
  • If you are authoring a custom model pack, ensure the archive describes the detection heads accurately: tensor order, logical shape, physical storage, dtype/quantization, score domain, class count, and any sliced outputs. Application code should not need to compensate for these details.

Shape and layout guidance

Different detection models expose different head layouts. Some use one tensor per feature-map level; others split boxes, objectness, classes, keypoints, or masks into separate tensors. Some model outputs are dense HWC tensors; others are packed or sliced by the compiler/runtime.

For model-pack flows this is handled by the packaged contract. For manually wired tensors, the key rule is: match the exported head format exactly. Do not choose a decode type based only on rank or channel count.

Python note

When configuring model options from Python, use the typed enum rather than a string when available:

opt = pyneat.ModelOptions()
opt.decode_type = pyneat.BoxDecodeType.YoloV8

Parse outputs with the helper that matches the model task:

outputs = model.run([image])

boxes = pyneat.decode_bbox(outputs)[0].to_numpy()

pose = pyneat.decode_pose(outputs)[0]
pose_boxes = pose.boxes.to_numpy()
keypoints = pose.keypoints.to_numpy()

seg = pyneat.decode_segmentation(outputs)[0]
seg_boxes = seg.boxes.to_numpy()
masks = seg.masks.to_numpy()