Files
2026-06-25 08:41:49 -05:00

102 lines
4.7 KiB
Python

import os
import glob
Import('env', 'envCython', 'arch', 'cereal', 'messaging', 'common', 'visionipc')
lenv = env.Clone()
lenvCython = envCython.Clone()
libs = [cereal, messaging, visionipc, common, 'capnp', 'kj', 'pthread']
frameworks = []
common_src = [
"models/commonmodel.cc",
"transforms/loadyuv.cc",
"transforms/transform.cc",
]
# OpenCL is a framework on Mac
if arch == "Darwin":
frameworks += ['OpenCL']
else:
libs += ['OpenCL']
# Set path definitions
for pathdef, fn in {'TRANSFORM': 'transforms/transform.cl', 'LOADYUV': 'transforms/loadyuv.cl'}.items():
for xenv in (lenv, lenvCython):
xenv['CXXFLAGS'].append(f'-D{pathdef}_PATH=\\"{File(fn).abspath}\\"')
# Compile cython
cython_libs = envCython["LIBS"] + libs
commonmodel_lib = lenv.Library('commonmodel', common_src)
lenvCython.Program('models/commonmodel_pyx.so', 'models/commonmodel_pyx.pyx', LIBS=[commonmodel_lib, *cython_libs], FRAMEWORKS=frameworks)
tinygrad_files = ["#"+x for x in glob.glob(env.Dir("#tinygrad_repo").relpath + "/**", recursive=True, root_dir=env.Dir("#").abspath) if 'pycache' not in x]
# Get model metadata
PC = not os.path.isfile('/TICI')
if PC:
inputs = tinygrad_files + [File(Dir("#iqpilot/modeld_v2").File("install_models_pc.py").abspath)]
outputs = []
model_dir = Dir("models").abspath
cmd = f'python3 {Dir("#iqpilot/modeld_v2").abspath}/install_models_pc.py {model_dir}'
for model_name in ['supercombo', 'driving_vision', 'driving_off_policy', 'driving_on_policy', 'driving_policy']:
if File(f"models/{model_name}.onnx").exists():
inputs.append(File(f"models/{model_name}.onnx"))
inputs.append(File(f"models/{model_name}_tinygrad.pkl"))
outputs.append(File(f"models/{model_name}_metadata.pkl"))
if outputs:
lenv.Command(outputs, inputs, cmd)
def tg_compile(flags, model_name):
pythonpath_string = 'PYTHONPATH="${PYTHONPATH}:' + env.Dir("#tinygrad_repo").abspath + '"'
fn = File(f"models/{model_name}").abspath
return lenv.Command(
fn + "_tinygrad.pkl",
[fn + ".onnx"] + tinygrad_files,
f'{pythonpath_string} {flags} python3 {Dir("#tinygrad_repo").abspath}/examples/openpilot/compile3.py {fn}.onnx {fn}_tinygrad.pkl'
)
# Compile small models
for model_name in ['supercombo', 'driving_vision', 'driving_off_policy', 'driving_on_policy', 'driving_policy']:
if File(f"models/{model_name}.onnx").exists():
flags = {
'larch64': 'DEV=QCOM',
'Darwin': f'DEV=CPU HOME={os.path.expanduser("~")} IMAGE=0', # tinygrad calls brew which needs a $HOME in the env
}.get(arch, 'DEV=CPU CPU_LLVM=1 IMAGE=0')
tg_compile(flags, model_name)
# Compile fused models (op_model16_deep style: warp+vision+on/off policy in one pkl).
# IMAGE=2 (not upstream's IMAGE=1) for the Adreno the fleet runs. Opt-in: only builds
# when the fused triplet onnx are dropped into models/.
from openpilot.common.transformations.model import MEDMODEL_INPUT_SIZE
from openpilot.common.transformations.camera import _ar_ox_fisheye, _os_fisheye
FUSED_CAMERA_CONFIGS = [(_ar_ox_fisheye.width, _ar_ox_fisheye.height), (_os_fisheye.width, _os_fisheye.height)]
FUSED_FRAME_SKIP = 4 # MODEL_RUN_FREQ // MODEL_CONTEXT_FREQ (20 // 5)
FUSED_TRIPLET = ['driving_vision', 'driving_off_policy', 'driving_on_policy']
def tg_compile_fused(file_prefix, flags):
pythonpath_string = 'PYTHONPATH="${PYTHONPATH}:' + env.Dir("#tinygrad_repo").abspath + '"'
mdir = Dir("models").abspath
model_w, model_h = MEDMODEL_INPUT_SIZE
cam_args = ' '.join(f'{cw}x{ch}' for cw, ch in FUSED_CAMERA_CONFIGS)
out_pkl = File(f"models/{file_prefix}driving_fused_tinygrad.pkl").abspath
onnx_deps = [File(f"models/{file_prefix}{m}.onnx") for m in FUSED_TRIPLET]
cmd = (f'{pythonpath_string} {flags} python3 {Dir("#iqpilot/modeld_v2").abspath}/compile_modeld.py '
f'--model-size {model_w}x{model_h} --camera-resolutions {cam_args} '
f'--vision-onnx {mdir}/{file_prefix}driving_vision.onnx '
f'--off-policy-onnx {mdir}/{file_prefix}driving_off_policy.onnx '
f'--on-policy-onnx {mdir}/{file_prefix}driving_on_policy.onnx '
f'--output {out_pkl} --frame-skip {FUSED_FRAME_SKIP}')
return lenv.Command(out_pkl, onnx_deps + tinygrad_files, cmd)
if all(File(f"models/{m}.onnx").exists() for m in FUSED_TRIPLET):
fused_flags = {
'larch64': 'DEV=QCOM IMAGE=2 FLOAT16=1 NOLOCALS=1 JIT_BATCH_SIZE=0',
'Darwin': f'DEV=CPU HOME={os.path.expanduser("~")} IMAGE=0 FLOAT16=1',
}.get(arch, 'DEV=CPU CPU_LLVM=1 IMAGE=0 FLOAT16=1')
tg_compile_fused('', fused_flags)
# usbgpu (eGPU): big vision on an external AMD GPU over USB, warp stays on QCOM
if all(File(f"models/big_{m}.onnx").exists() for m in FUSED_TRIPLET):
tg_compile_fused('big_', 'DEV=USB+AMD:LLVM WARP_DEV=QCOM FLOAT16=1 JIT_BATCH_SIZE=0 GMMU=0')