1#!/usr/bin/perl -w
2# SPDX-License-Identifier: GPL-2.0-only OR BSD-3-Clause
3# Copyright (C) 2019--2020 Intel Corporation
4
5use Getopt::Long qw(:config no_ignore_case);
6use File::Basename;
7
8my $ccsregs = "ccs-regs.asc";
9my $header;
10my $regarray;
11my $limitc;
12my $limith;
13my $kernel;
14my $help;
15
16GetOptions("ccsregs|c=s" => \$ccsregs,
17	   "header|e=s" => \$header,
18	   "regarray|r=s" => \$regarray,
19	   "limitc|l=s" => \$limitc,
20	   "limith|L=s" => \$limith,
21	   "kernel|k" => \$kernel,
22	   "help|h" => \$help) or die "can't parse options";
23
24$help = 1 if ! defined $header || ! defined $limitc || ! defined $limith;
25
26if (defined $help) {
27	print <<EOH
28$0 - Create CCS register definitions for C
29
30usage: $0 -c ccs-regs.asc -e header -r regarray -l limit-c -L limit-header [-k]
31
32	-c ccs register file
33	-e header file name
34	-r register description array file name
35	-l limit and capability array file name
36	-L limit and capability header file name
37	-k generate files for kernel space consumption
38EOH
39	  ;
40	exit 0;
41}
42
43my $lh_hdr = ! defined $kernel
44	? '#include "ccs-os.h"' . "\n"
45	: "#include <linux/bits.h>\n#include <linux/types.h>\n";
46my $uint32_t = ! defined $kernel ? 'uint32_t' : 'u32';
47my $uint16_t = ! defined $kernel ? 'uint16_t' : 'u16';
48
49open(my $R, "< $ccsregs") or die "can't open $ccsregs";
50
51open(my $H, "> $header") or die "can't open $header";
52my $A;
53if (defined $regarray) {
54	open($A, "> $regarray") or die "can't open $regarray";
55}
56open(my $LC, "> $limitc") or die "can't open $limitc";
57open(my $LH, "> $limith") or die "can't open $limith";
58
59my %this;
60
61sub is_limit_reg($) {
62	my $addr = hex $_[0];
63
64	return 0 if $addr < 0x40; # weed out status registers
65	return 0 if $addr >= 0x100 && $addr < 0xfff; # weed out configuration registers
66
67	return 1;
68}
69
70my $uc_header = basename uc $header;
71$uc_header =~ s/[^A-Z0-9]/_/g;
72
73my $copyright = "/* Copyright (C) 2019--2020 Intel Corporation */\n";
74my $license = "SPDX-License-Identifier: GPL-2.0-only OR BSD-3-Clause";
75my $note = "/*\n * Generated by $0;\n * do not modify.\n */\n";
76
77for my $fh ($A, $LC) {
78	print $fh "// $license\n$copyright$note\n" if defined $fh;
79}
80
81for my $fh ($H, $LH) {
82	print $fh "/* $license */\n$copyright$note\n";
83}
84
85print $H <<EOF
86#ifndef __${uc_header}__
87#define __${uc_header}__
88
89EOF
90  ;
91
92print $H <<EOF
93#include <linux/bits.h>
94
95#include <media/v4l2-cci.h>
96
97EOF
98	if defined $kernel;
99
100print $H "#define CCS_FL_BASE		" .
101    (defined $kernel ? "CCI_REG_PRIVATE_SHIFT" : 16) . "\n";
102
103my $flag = -1;
104my $all_flags;
105
106sub bit_def($) {
107	my $bit = shift @_;
108
109	if (defined $kernel) {
110		return "BIT$bit" if $bit =~ /^\(.*\)$/;
111		return "BIT($bit)";
112	}
113	return "(1U << $bit)";
114}
115
116sub flag_str($$) {
117	my ($flag, $check) = @_;
118
119	$$flag++;
120
121	my $flag_str = !$$flag ? "CCS_FL_BASE" : "(CCS_FL_BASE + $$flag)";
122
123	$flag_str = bit_def($flag_str);
124
125	$$check .= " | " if defined $$check;
126
127	$$check .= $flag_str;
128
129	return $flag_str;
130}
131
132if (! defined $kernel) {
133	print $H "#define CCS_FL_16BIT		" . flag_str(\$flag, \$all_flags) . "\n";
134	print $H "#define CCS_FL_32BIT		" . flag_str(\$flag, \$all_flags) . "\n";
135}
136
137print $H "#define CCS_FL_FLOAT_IREAL	" . flag_str(\$flag, \$all_flags) . "\n";
138print $H "#define CCS_FL_IREAL		" . flag_str(\$flag, \$all_flags) . "\n";
139print $H "#define CCS_BUILD_BUG \\
140	BUILD_BUG_ON(~CCI_REG_PRIVATE_MASK & ($all_flags))\n"
141    if defined $kernel;
142
143print $H <<EOF
144
145#define CCS_R_ADDR(r)		((r) & 0xffff)
146
147EOF
148    if ! defined $kernel;
149
150print $A <<EOF
151#include <stdint.h>
152#include <stdio.h>
153#include "ccs-extra.h"
154#include "ccs-regs.h"
155
156EOF
157	if defined $A;
158
159my $uc_limith = basename uc $limith;
160$uc_limith =~ s/[^A-Z0-9]/_/g;
161
162print $LH <<EOF
163#ifndef __${uc_limith}__
164#define __${uc_limith}__
165
166$lh_hdr
167struct ccs_limit {
168	$uint32_t reg;
169	$uint16_t size;
170	$uint16_t flags;
171	const char *name;
172};
173
174EOF
175  ;
176print $LH "#define CCS_L_FL_SAME_REG	" . bit_def(0) . "\n\n";
177
178print $LH <<EOF
179extern const struct ccs_limit ccs_limits[];
180
181EOF
182  ;
183
184print $LC <<EOF
185#include "ccs-limits.h"
186#include "ccs-regs.h"
187
188const struct ccs_limit ccs_limits[] = {
189EOF
190  ;
191
192my $limitcount = 0;
193my $argdescs;
194my $reglist = "const struct ccs_reg_desc ccs_reg_desc[] = {\n";
195
196sub name_split($$) {
197	my ($name, $addr) = @_;
198	my $args;
199
200	$name =~ /([^\(]+?)(\(.*)/;
201	($name, $args) = ($1, $2);
202	$args = [split /,\s*/, $args];
203	foreach my $t (@$args) {
204		$t =~ s/[\(\)]//g;
205		$t =~ s/\//\\\//g;
206	}
207
208	return ($name, $addr, $args);
209}
210
211sub tabconv($) {
212	$_ = shift;
213
214	my @l = split "\n", $_;
215
216	map {
217		s/ {8,8}/\t/g;
218		s/\t\K +//;
219	} @l;
220
221	return (join "\n", @l) . "\n";
222}
223
224sub elem_bits(@) {
225	my @flags = @_;
226
227	return 16 if grep /^16$/, @flags;
228	return 32 if grep /^32$/, @flags;
229	return 8;
230}
231
232sub arr_size($) {
233	my $this = $_[0];
234	my $size = $this->{elsize};
235	my $h = $this->{argparams};
236
237	foreach my $arg (@{$this->{args}}) {
238		my $apref = $h->{$arg};
239
240		$size *= $apref->{max} - $apref->{min} + 1;
241	}
242
243	return $size;
244}
245
246sub print_args($$$) {
247	my ($this, $postfix, $is_same_reg) = @_;
248	my ($args, $argparams, $name) =
249	  ($this->{args}, $this->{argparams}, $this->{name});
250	my $varname = "ccs_reg_arg_" . (lc $name) . $postfix;
251	my @mins;
252	my @sorted_args = @{$this->{sorted_args}};
253	my $lim_arg;
254	my $size = arr_size($this);
255
256	$argdescs .= "static const struct ccs_reg_arg " . $varname . "[] = {\n";
257
258	foreach my $sorted_arg (@sorted_args) {
259		push @mins, $argparams->{$sorted_arg}->{min};
260	}
261
262	foreach my $sorted_arg (@sorted_args) {
263		my $h = $argparams->{$sorted_arg};
264
265		$argdescs .= "\t{ \"$sorted_arg\", $h->{min}, $h->{max}, $h->{elsize} },\n";
266
267		$lim_arg .= defined $lim_arg ? ", $h->{min}" : "$h->{min}";
268	}
269
270	$argdescs .= "};\n\n";
271
272	$reglist .= "\t{ CCS_R_" . (uc $name) . "(" . (join ",", (@mins)) .
273	  "), $size, sizeof($varname) / sizeof(*$varname)," .
274	    " \"" . (lc $name) . "\", $varname },\n";
275
276	print $LC tabconv sprintf "\t{ CCS_R_" . (uc $name) . "($lim_arg), " .
277	  $size . ", " . ($is_same_reg ? "CCS_L_FL_SAME_REG" : "0") .
278	    ", \"$name" . (defined $this->{discontig} ? " $lim_arg" : "") . "\" },\n"
279	      if is_limit_reg $this->{base_addr};
280}
281
282my $hdr_data;
283
284while (<$R>) {
285	chop;
286	s/^\s*//;
287	next if /^[#;]/ || /^$/;
288	if (s/^-\s*//) {
289		if (s/^b\s*//) {
290			my ($bit, $addr) = split /\t+/;
291			$bit = uc $bit;
292			$hdr_data .= sprintf "#define %-62s %s", "CCS_" . (uc ${this{name}}) ."_$bit", bit_def($addr) . "\n";
293		} elsif (s/^f\s*//) {
294			s/[,\.-]/_/g;
295			my @a = split /\s+/;
296			my ($msb, $lsb, $this_field) = reverse @a;
297		        @a = ( { "name" => "SHIFT", "addr" => $lsb, "fmt" => "%uU", },
298			       { "name" => "MASK", "addr" => (1 << ($msb + 1)) - 1 - ((1 << $lsb) - 1), "fmt" => "0x%" . join(".", ($this{"elsize"} >> 2) x 2) . "x" } );
299			$this{"field"} = $this_field;
300			foreach my $ar (@a) {
301				#print $ar->{fmt}."\n";
302				$hdr_data .= sprintf "#define %-62s " . $ar->{"fmt"} . "\n", "CCS_" . (uc $this{"name"}) . (defined $this_field ? "_" . uc $this_field : "") . "_" . $ar->{"name"}, $ar->{"addr"} . "\n";
303			}
304		} elsif (s/^e\s*//) {
305			s/[,\.-]/_/g;
306			my ($enum, $addr) = split /\s+/;
307			$enum = uc $enum;
308			$hdr_data .= sprintf "#define %-62s %s", "CCS_" . (uc ${this{name}}) . (defined $this{"field"} ? "_" . uc $this{"field"} : "") ."_$enum", $addr . ($addr =~ /0x/i ? "" : "U") . "\n";
309		} elsif (s/^l\s*//) {
310			my ($arg, $min, $max, $elsize, @discontig) = split /\s+/;
311			my $size;
312
313			foreach my $num ($min, $max) {
314				$num = hex $num if $num =~ /0x/i;
315			}
316
317			$hdr_data .= sprintf "#define %-62s %s", "CCS_LIM_" . (uc ${this{name}} . "_MIN_$arg"), $min . ($min =~ /0x/i ? "" : "U") . "\n";
318			$hdr_data .= sprintf "#define %-62s %s", "CCS_LIM_" . (uc ${this{name}} . "_MAX_$arg"), $max . ($max =~ /0x/i ? "" : "U") . "\n";
319
320			my $h = $this{argparams};
321
322			$h->{$arg} = { "min" => $min,
323				       "max" => $max,
324				       "elsize" => $elsize =~ /^0x/ ? hex $elsize : $elsize,
325				       "discontig" => \@discontig };
326
327			$this{discontig} = $arg if @discontig;
328
329			next if $#{$this{args}} + 1 != scalar keys %{$this{argparams}};
330
331			my $reg_formula = "$this{addr}";
332			my $lim_formula;
333
334			chop $reg_formula;
335
336			$reg_formula = "(" . $reg_formula if $this{flagstring} ne "";
337
338			foreach my $arg (@{$this{args}}) {
339				my $d = $h->{$arg}->{discontig};
340				my $times = $h->{$arg}->{elsize} != 1 ?
341				  " * " . $h->{$arg}->{elsize} : "";
342
343				if (@$d) {
344					my ($lim, $offset) = split /,/, $d->[0];
345
346					$reg_formula .= " + (($arg) < $lim ? ($arg)$times : $offset + (($arg) - $lim)$times)";
347				} else {
348					$reg_formula .= " + ($arg)$times";
349				}
350
351				$lim_formula .= (defined $lim_formula ? " + " : "") . "($arg)$times";
352			}
353
354			$reg_formula .= ")";
355			$lim_formula =~ s/^\(([a-z0-9]+)\)$/$1/i;
356
357			print $H tabconv sprintf("#define %-62s %s", "CCS_R_" . (uc $this{name}) .
358						 $this{arglist}, $reg_formula .
359						 (($this{flagstring} eq "") ? "" :
360						  " | " . $this{flagstring} . ")") . "\n");
361
362			print $H tabconv $hdr_data;
363			undef $hdr_data;
364
365			# Sort arguments in descending order by size
366			@{$this{sorted_args}} = sort {
367				$h->{$a}->{elsize} <= $h->{$b}->{elsize}
368			} @{$this{args}};
369
370			if (defined $this{discontig}) {
371				my $da = $this{argparams}->{$this{discontig}};
372				my ($first_discontig) = split /,/, $da->{discontig}->[0];
373				my $max = $da->{max};
374
375				$da->{max} = $first_discontig - 1;
376				print_args(\%this, "", 0);
377
378				$da->{min} = $da->{max} + 1;
379				$da->{max} = $max;
380				print_args(\%this, $first_discontig, 1);
381			} else {
382				print_args(\%this, "", 0);
383			}
384
385			next unless is_limit_reg $this{base_addr};
386
387			print $LH tabconv sprintf "#define %-63s%s\n",
388			  "CCS_L_" . (uc $this{name}) . "_OFFSET(" .
389			    (join ", ", @{$this{args}}) . ")", "($lim_formula)";
390		}
391
392		if (! @{$this{args}}) {
393			print $H tabconv($hdr_data);
394			undef $hdr_data;
395		}
396
397		next;
398	}
399
400	my ($name, $addr, @flags) = split /\t+/, $_;
401	my $args = [];
402
403	my $sp;
404
405	($name, $addr, $args) = name_split($name, $addr) if /\(.*\)/;
406
407	$name =~ s/[,\.-]/_/g;
408
409	my $flagstring = "";
410	my $bits = elem_bits(@flags);
411	if (! defined $kernel) {
412		$flagstring .= "| CCS_FL_16BIT " if $bits == 16;
413		$flagstring .= "| CCS_FL_32BIT " if $bits == 32;
414	}
415	$flagstring .= "| CCS_FL_FLOAT_IREAL " if grep /^float_ireal$/, @flags;
416	$flagstring .= "| CCS_FL_IREAL " if grep /^ireal$/, @flags;
417	$flagstring =~ s/^\| //;
418	$flagstring =~ s/ $//;
419	$flagstring = "($flagstring)" if $flagstring =~ /\|/;
420	my $base_addr = $addr;
421	$addr = "CCI_REG$bits($addr)" if defined $kernel;
422
423	if ($flagstring ne "" && !@$args) {
424		$addr = "($addr | $flagstring)";
425		$flagstring = "";
426	}
427
428	my $arglist = @$args ? "(" . (join ", ", @$args) . ")" : "";
429	$hdr_data .= sprintf "#define %-62s %s\n", "CCS_R_" . (uc $name), $addr
430	  if !@$args;
431
432	$name =~ s/\(.*//;
433
434	%this = ( name => $name,
435		  addr => $addr,
436		  flagstring => $flagstring,
437		  base_addr => $base_addr,
438		  argparams => {},
439		  args => $args,
440		  arglist => $arglist,
441		  elsize => $bits / 8,
442		);
443
444	if (!@$args) {
445		$reglist .= "\t{ CCS_R_" . (uc $name) . ", 1,  0, \"" . (lc $name) . "\", NULL },\n";
446		print $H tabconv $hdr_data;
447		undef $hdr_data;
448
449		print $LC tabconv sprintf "\t{ CCS_R_" . (uc $name) . ", " .
450		  $this{elsize} . ", 0, \"$name\" },\n"
451		    if is_limit_reg $this{base_addr};
452	}
453
454	print $LH tabconv sprintf "#define %-63s%s\n",
455	  "CCS_L_" . (uc $this{name}), $limitcount++
456	    if is_limit_reg $this{base_addr};
457}
458
459if (defined $A) {
460	print $A $argdescs, $reglist;
461
462	print $A "\t{ 0 }\n";
463
464	print $A "};\n";
465}
466
467print $H "\n#endif /* __${uc_header}__ */\n";
468
469print $LH tabconv sprintf "#define %-63s%s\n", "CCS_L_LAST", $limitcount;
470
471print $LH "\n#endif /* __${uc_limith}__ */\n";
472
473print $LC "\t{ 0 } /* Guardian */\n";
474print $LC "};\n";
475
476close($R);
477close($H);
478close($A) if defined $A;
479close($LC);
480close($LH);
481