Files
external_libcamera/utils/codegen/gen-gst-controls.py
Barnabás Pőcze b9fd53fbef treewide: Use argparse.FileType in more places
Convert some scripts to use `argparse.FileType` where the change is relatively
easily doable. This allows better error messages as e.g. missing input files
will be detected during argument parsing. And it also makes writing to stdout
in absence of an explicit argument simpler.

Signed-off-by: Barnabás Pőcze <barnabas.pocze@ideasonboard.com>
Reviewed-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com>
2025-11-17 09:19:36 +01:00

182 lines
5.3 KiB
Python
Executable File

#!/usr/bin/env python3
# SPDX-License-Identifier: GPL-2.0-or-later
# Copyright (C) 2019, Google Inc.
# Copyright (C) 2024, Jaslo Ziska
#
# Authors:
# Laurent Pinchart <laurent.pinchart@ideasonboard.com>
# Jaslo Ziska <jaslo@ziska.de>
#
# Generate gstreamer control properties from YAML
import argparse
import jinja2
import re
import sys
import yaml
from controls import Control
exposed_controls = [
'AeEnable', 'AeMeteringMode', 'AeConstraintMode', 'AeExposureMode',
'ExposureValue', 'ExposureTime', 'ExposureTimeMode',
'AnalogueGain', 'AnalogueGainMode', 'AeFlickerPeriod',
'Brightness', 'Contrast', 'AwbEnable', 'AwbMode', 'ColourGains',
'Saturation', 'Sharpness', 'ColourCorrectionMatrix', 'ScalerCrop',
'DigitalGain', 'AfMode', 'AfRange', 'AfSpeed', 'AfMetering', 'AfWindows',
'LensPosition', 'Gamma',
]
def find_common_prefix(strings):
prefix = strings[0]
for string in strings[1:]:
while string[:len(prefix)] != prefix and prefix:
prefix = prefix[:len(prefix) - 1]
if not prefix:
break
return prefix
def format_description(description):
# Substitute doxygen keywords \sa (see also) and \todo
description = re.sub(r'\\sa((?: \w+)+)',
lambda match: 'See also: ' + ', '.join(
map(kebab_case, match.group(1).strip().split(' '))
) + '.', description)
description = re.sub(r'\\todo', 'Todo:', description)
description = description.strip().split('\n')
return '\n'.join([
'"' + line.replace('\\', r'\\').replace('"', r'\"') + ' "' for line in description if line
]).rstrip()
# Custom filter to allow indenting by a string prior to Jinja version 3.0
#
# This function can be removed and the calls to indent_str() replaced by the
# built-in indent() filter when dropping Jinja versions older than 3.0
def indent_str(s, indention):
s += '\n'
lines = s.splitlines()
rv = lines.pop(0)
if lines:
rv += '\n' + '\n'.join(
indention + line if line else line for line in lines
)
return rv
def snake_case(s):
return ''.join([
c.isupper() and ('_' + c.lower()) or c for c in s
]).strip('_')
def kebab_case(s):
return snake_case(s).replace('_', '-')
def extend_control(ctrl):
if ctrl.vendor != 'libcamera':
ctrl.namespace = f'{ctrl.vendor}::'
ctrl.vendor_prefix = f'{ctrl.vendor}-'
else:
ctrl.namespace = ''
ctrl.vendor_prefix = ''
ctrl.is_array = ctrl.size is not None
if ctrl.is_enum:
# Remove common prefix from enum variant names
prefix = find_common_prefix([enum.name for enum in ctrl.enum_values])
for enum in ctrl.enum_values:
enum.gst_name = kebab_case(enum.name.removeprefix(prefix))
ctrl.gtype = 'enum'
ctrl.default = '0'
elif ctrl.element_type == 'bool':
ctrl.gtype = 'boolean'
ctrl.default = 'false'
elif ctrl.element_type == 'float':
ctrl.gtype = 'float'
ctrl.default = '0'
ctrl.min = '-G_MAXFLOAT'
ctrl.max = 'G_MAXFLOAT'
elif ctrl.element_type == 'int32_t':
ctrl.gtype = 'int'
ctrl.default = '0'
ctrl.min = 'G_MININT'
ctrl.max = 'G_MAXINT'
elif ctrl.element_type == 'int64_t':
ctrl.gtype = 'int64'
ctrl.default = '0'
ctrl.min = 'G_MININT64'
ctrl.max = 'G_MAXINT64'
elif ctrl.element_type == 'uint8_t':
ctrl.gtype = 'uchar'
ctrl.default = '0'
ctrl.min = '0'
ctrl.max = 'G_MAXUINT8'
elif ctrl.element_type == 'Rectangle':
ctrl.is_rectangle = True
ctrl.default = '0'
ctrl.min = '0'
ctrl.max = 'G_MAXINT'
else:
raise RuntimeError(f'The type `{ctrl.element_type}` is unknown')
return ctrl
def main(argv):
# Parse command line arguments
parser = argparse.ArgumentParser()
parser.add_argument('--output', '-o', metavar='file', default=sys.stdout,
type=argparse.FileType('w', encoding='utf-8'),
help='Output file name. Defaults to standard output if not specified.')
parser.add_argument('--template', '-t', dest='template', required=True,
type=argparse.FileType('r', encoding='utf-8'),
help='Template file name.')
parser.add_argument('input', nargs='+', type=argparse.FileType('rb'),
help='Input file name.')
args = parser.parse_args(argv[1:])
controls = {}
for input in args.input:
data = yaml.safe_load(input)
vendor = data['vendor']
ctrls = controls.setdefault(vendor, [])
for ctrl in data['controls']:
ctrl = Control(*ctrl.popitem(), vendor, mode='controls')
if ctrl.name in exposed_controls:
ctrls.append(extend_control(ctrl))
data = {'controls': list(controls.items())}
env = jinja2.Environment()
env.filters['format_description'] = format_description
env.filters['indent_str'] = indent_str
env.filters['snake_case'] = snake_case
env.filters['kebab_case'] = kebab_case
template = env.from_string(args.template.read())
string = template.render(data)
args.output.write(string)
return 0
if __name__ == '__main__':
sys.exit(main(sys.argv))