162306a36Sopenharmony_ci#!/usr/bin/env perl
262306a36Sopenharmony_ci# SPDX-License-Identifier: GPL-2.0
362306a36Sopenharmony_ci#
462306a36Sopenharmony_ciuse warnings;
562306a36Sopenharmony_ciuse strict;
662306a36Sopenharmony_ciuse Math::BigInt;
762306a36Sopenharmony_ciuse Fcntl "SEEK_SET";
862306a36Sopenharmony_ci
962306a36Sopenharmony_cidie "Format: $0 [-s <systemmap-file>] <vmlinux-file> <keyring-file>\n"
1062306a36Sopenharmony_ci    if ($#ARGV != 1 && $#ARGV != 3 ||
1162306a36Sopenharmony_ci	$#ARGV == 3 && $ARGV[0] ne "-s");
1262306a36Sopenharmony_ci
1362306a36Sopenharmony_cimy $sysmap = "";
1462306a36Sopenharmony_ciif ($#ARGV == 3) {
1562306a36Sopenharmony_ci    shift;
1662306a36Sopenharmony_ci    $sysmap = $ARGV[0];
1762306a36Sopenharmony_ci    shift;
1862306a36Sopenharmony_ci}
1962306a36Sopenharmony_ci
2062306a36Sopenharmony_cimy $vmlinux = $ARGV[0];
2162306a36Sopenharmony_cimy $keyring = $ARGV[1];
2262306a36Sopenharmony_ci
2362306a36Sopenharmony_ci#
2462306a36Sopenharmony_ci# Parse the vmlinux section table
2562306a36Sopenharmony_ci#
2662306a36Sopenharmony_ciopen FD, "objdump -h $vmlinux |" || die $vmlinux;
2762306a36Sopenharmony_cimy @lines = <FD>;
2862306a36Sopenharmony_ciclose(FD) || die $vmlinux;
2962306a36Sopenharmony_ci
3062306a36Sopenharmony_cimy @sections = ();
3162306a36Sopenharmony_ci
3262306a36Sopenharmony_ciforeach my $line (@lines) {
3362306a36Sopenharmony_ci    chomp($line);
3462306a36Sopenharmony_ci    if ($line =~ /\s*([0-9]+)\s+(\S+)\s+([0-9a-f]+)\s+([0-9a-f]+)\s+([0-9a-f]+)\s+([0-9a-f]+)\s+2[*][*]([0-9]+)/
3562306a36Sopenharmony_ci	) {
3662306a36Sopenharmony_ci	my $seg  = $1;
3762306a36Sopenharmony_ci	my $name = $2;
3862306a36Sopenharmony_ci	my $len  = Math::BigInt->new("0x" . $3);
3962306a36Sopenharmony_ci	my $vma  = Math::BigInt->new("0x" . $4);
4062306a36Sopenharmony_ci	my $lma  = Math::BigInt->new("0x" . $5);
4162306a36Sopenharmony_ci	my $foff = Math::BigInt->new("0x" . $6);
4262306a36Sopenharmony_ci	my $align = 2 ** $7;
4362306a36Sopenharmony_ci
4462306a36Sopenharmony_ci	push @sections, { name => $name,
4562306a36Sopenharmony_ci			  vma => $vma,
4662306a36Sopenharmony_ci			  len => $len,
4762306a36Sopenharmony_ci			  foff => $foff };
4862306a36Sopenharmony_ci    }
4962306a36Sopenharmony_ci}
5062306a36Sopenharmony_ci
5162306a36Sopenharmony_ciprint "Have $#sections sections\n";
5262306a36Sopenharmony_ci
5362306a36Sopenharmony_ci#
5462306a36Sopenharmony_ci# Try and parse the vmlinux symbol table.  If the vmlinux file has been created
5562306a36Sopenharmony_ci# from a vmlinuz file with extract-vmlinux then the symbol table will be empty.
5662306a36Sopenharmony_ci#
5762306a36Sopenharmony_ciopen FD, "nm $vmlinux 2>/dev/null |" || die $vmlinux;
5862306a36Sopenharmony_ci@lines = <FD>;
5962306a36Sopenharmony_ciclose(FD) || die $vmlinux;
6062306a36Sopenharmony_ci
6162306a36Sopenharmony_cimy %symbols = ();
6262306a36Sopenharmony_cimy $nr_symbols = 0;
6362306a36Sopenharmony_ci
6462306a36Sopenharmony_cisub parse_symbols(@) {
6562306a36Sopenharmony_ci    foreach my $line (@_) {
6662306a36Sopenharmony_ci	chomp($line);
6762306a36Sopenharmony_ci	if ($line =~ /([0-9a-f]+)\s([a-zA-Z])\s(\S+)/
6862306a36Sopenharmony_ci	    ) {
6962306a36Sopenharmony_ci	    my $addr = "0x" . $1;
7062306a36Sopenharmony_ci	    my $type = $2;
7162306a36Sopenharmony_ci	    my $name = $3;
7262306a36Sopenharmony_ci
7362306a36Sopenharmony_ci	    $symbols{$name} = $addr;
7462306a36Sopenharmony_ci	    $nr_symbols++;
7562306a36Sopenharmony_ci	}
7662306a36Sopenharmony_ci    }
7762306a36Sopenharmony_ci}
7862306a36Sopenharmony_ciparse_symbols(@lines);
7962306a36Sopenharmony_ci
8062306a36Sopenharmony_ciif ($nr_symbols == 0 && $sysmap ne "") {
8162306a36Sopenharmony_ci    print "No symbols in vmlinux, trying $sysmap\n";
8262306a36Sopenharmony_ci
8362306a36Sopenharmony_ci    open FD, "<$sysmap" || die $sysmap;
8462306a36Sopenharmony_ci    @lines = <FD>;
8562306a36Sopenharmony_ci    close(FD) || die $sysmap;
8662306a36Sopenharmony_ci    parse_symbols(@lines);
8762306a36Sopenharmony_ci}
8862306a36Sopenharmony_ci
8962306a36Sopenharmony_cidie "No symbols available\n"
9062306a36Sopenharmony_ci    if ($nr_symbols == 0);
9162306a36Sopenharmony_ci
9262306a36Sopenharmony_ciprint "Have $nr_symbols symbols\n";
9362306a36Sopenharmony_ci
9462306a36Sopenharmony_cidie "Can't find system certificate list"
9562306a36Sopenharmony_ci    unless (exists($symbols{"__cert_list_start"}) &&
9662306a36Sopenharmony_ci	    exists($symbols{"system_certificate_list_size"}));
9762306a36Sopenharmony_ci
9862306a36Sopenharmony_cimy $start = Math::BigInt->new($symbols{"__cert_list_start"});
9962306a36Sopenharmony_cimy $end;
10062306a36Sopenharmony_cimy $size;
10162306a36Sopenharmony_cimy $size_sym = Math::BigInt->new($symbols{"system_certificate_list_size"});
10262306a36Sopenharmony_ci
10362306a36Sopenharmony_ciopen FD, "<$vmlinux" || die $vmlinux;
10462306a36Sopenharmony_cibinmode(FD);
10562306a36Sopenharmony_ci
10662306a36Sopenharmony_cimy $s = undef;
10762306a36Sopenharmony_ciforeach my $sec (@sections) {
10862306a36Sopenharmony_ci    my $s_name = $sec->{name};
10962306a36Sopenharmony_ci    my $s_vma = $sec->{vma};
11062306a36Sopenharmony_ci    my $s_len = $sec->{len};
11162306a36Sopenharmony_ci    my $s_foff = $sec->{foff};
11262306a36Sopenharmony_ci    my $s_vend = $s_vma + $s_len;
11362306a36Sopenharmony_ci
11462306a36Sopenharmony_ci    next unless ($start >= $s_vma);
11562306a36Sopenharmony_ci    next if ($start >= $s_vend);
11662306a36Sopenharmony_ci
11762306a36Sopenharmony_ci    die "Certificate list size was not found on the same section\n"
11862306a36Sopenharmony_ci	if ($size_sym < $s_vma || $size_sym > $s_vend);
11962306a36Sopenharmony_ci
12062306a36Sopenharmony_ci    die "Cert object in multiple sections: ", $s_name, " and ", $s->{name}, "\n"
12162306a36Sopenharmony_ci	if ($s);
12262306a36Sopenharmony_ci
12362306a36Sopenharmony_ci    my $size_off = $size_sym -$s_vma + $s_foff;
12462306a36Sopenharmony_ci    my $packed;
12562306a36Sopenharmony_ci    die $vmlinux if (!defined(sysseek(FD, $size_off, SEEK_SET)));
12662306a36Sopenharmony_ci    sysread(FD, $packed, 8);
12762306a36Sopenharmony_ci    $size = unpack 'L!', $packed;
12862306a36Sopenharmony_ci    $end = $start + $size;
12962306a36Sopenharmony_ci
13062306a36Sopenharmony_ci    printf "Have %u bytes of certs at VMA 0x%x\n", $size, $start;
13162306a36Sopenharmony_ci
13262306a36Sopenharmony_ci    die "Cert object partially overflows section $s_name\n"
13362306a36Sopenharmony_ci	if ($end > $s_vend);
13462306a36Sopenharmony_ci
13562306a36Sopenharmony_ci    $s = $sec;
13662306a36Sopenharmony_ci}
13762306a36Sopenharmony_ci
13862306a36Sopenharmony_cidie "Cert object not inside a section\n"
13962306a36Sopenharmony_ci    unless ($s);
14062306a36Sopenharmony_ci
14162306a36Sopenharmony_ciprint "Certificate list in section ", $s->{name}, "\n";
14262306a36Sopenharmony_ci
14362306a36Sopenharmony_cimy $foff = $start - $s->{vma} + $s->{foff};
14462306a36Sopenharmony_ci
14562306a36Sopenharmony_ciprintf "Certificate list at file offset 0x%x\n", $foff;
14662306a36Sopenharmony_ci
14762306a36Sopenharmony_cidie $vmlinux if (!defined(sysseek(FD, $foff, SEEK_SET)));
14862306a36Sopenharmony_cimy $buf = "";
14962306a36Sopenharmony_cimy $len = sysread(FD, $buf, $size);
15062306a36Sopenharmony_cidie "$vmlinux" if (!defined($len));
15162306a36Sopenharmony_cidie "Short read on $vmlinux\n" if ($len != $size);
15262306a36Sopenharmony_ciclose(FD) || die $vmlinux;
15362306a36Sopenharmony_ci
15462306a36Sopenharmony_ciopen FD, ">$keyring" || die $keyring;
15562306a36Sopenharmony_cibinmode(FD);
15662306a36Sopenharmony_ci$len = syswrite(FD, $buf, $size);
15762306a36Sopenharmony_cidie "$keyring" if (!defined($len));
15862306a36Sopenharmony_cidie "Short write on $keyring\n" if ($len != $size);
15962306a36Sopenharmony_ciclose(FD) || die $keyring;
160