libcamera/utils/codegen/gen-gst-controls.py
Paul Elder 39fe4ad968 utils: codegen: controls.py: Parse direction information
In preparation for adding support for querying direction information
from controls, parse the direction information from control ID
definitions. This can later be plugged in directly to the IPA code
generators simply by using ctrl.direction.

Signed-off-by: Paul Elder <paul.elder@ideasonboard.com>
Reviewed-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com>
Reviewed-by: Kieran Bingham <kieran.bingham@ideasonboard.com>
Reviewed-by: Stefan Klug <stefan.klug@ideasonboard.com>
2024-12-18 17:26:52 +09: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', 'AnalogueGain', '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', type=str,
help='Output file name. Defaults to standard output if not specified.')
parser.add_argument('--template', '-t', dest='template', type=str, required=True,
help='Template file name.')
parser.add_argument('input', type=str, nargs='+',
help='Input file name.')
args = parser.parse_args(argv[1:])
controls = {}
for input in args.input:
data = yaml.safe_load(open(input, 'rb').read())
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(open(args.template, 'r', encoding='utf-8').read())
string = template.render(data)
if args.output:
with open(args.output, 'w', encoding='utf-8') as output:
output.write(string)
else:
sys.stdout.write(string)
return 0
if __name__ == '__main__':
sys.exit(main(sys.argv))