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