#!/usr/bin/perl # Shared code for glibc conformance tests. # Copyright (C) 2014-2018 Free Software Foundation, Inc. # This file is part of the GNU C Library. # The GNU C Library is free software; you can redistribute it and/or # modify it under the terms of the GNU Lesser General Public # License as published by the Free Software Foundation; either # version 2.1 of the License, or (at your option) any later version. # The GNU C Library is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU # Lesser General Public License for more details. # You should have received a copy of the GNU Lesser General Public # License along with the GNU C Library; if not, see # . package GlibcConform; require Exporter; @ISA = qw(Exporter); @EXPORT = qw(%CFLAGS list_exported_functions); # Compiler options for each standard. $CFLAGS{"ISO"} = "-ansi"; $CFLAGS{"ISO99"} = "-std=c99"; $CFLAGS{"ISO11"} = "-std=c11"; $CFLAGS{"POSIX"} = "-D_POSIX_C_SOURCE=199506L -ansi"; $CFLAGS{"XPG4"} = "-ansi -D_XOPEN_SOURCE"; $CFLAGS{"XPG42"} = "-ansi -D_XOPEN_SOURCE -D_XOPEN_SOURCE_EXTENDED"; $CFLAGS{"UNIX98"} = "-ansi -D_XOPEN_SOURCE=500"; $CFLAGS{"XOPEN2K"} = "-std=c99 -D_XOPEN_SOURCE=600"; $CFLAGS{"XOPEN2K8"} = "-std=c99 -D_XOPEN_SOURCE=700"; $CFLAGS{"POSIX2008"} = "-std=c99 -D_POSIX_C_SOURCE=200809L"; # Return a list of functions exported by a header, empty if an include # of the header does not compile. sub list_exported_functions { my ($cc, $standard, $header, $tmpdir, $withclang) = @_; my ($cc_all) = "$cc -D_ISOMAC $CFLAGS{$standard}"; my ($tmpfile) = "$tmpdir/list-$$.c"; my ($auxfile) = "$tmpdir/list-$$.c.aux"; my ($astfile) = "$tmpdir/list-$$.c.ast"; my ($ret); my (%res) = (); open (TMPFILE, ">$tmpfile") || die ("open $tmpfile: $!\n"); print TMPFILE "#include <$header>\n"; close (TMPFILE) || die ("close $tmpfile: $!\n"); if ($withclang ne "yes") { $ret = system "$cc_all -c $tmpfile -o /dev/null -aux-info $auxfile > /dev/null"; } else { $ret = system "$cc_all -c $tmpfile -o /dev/null -Xclang -ast-dump |grep FunctionDecl > $astfile"; } unlink ($tmpfile) || die ("unlink $tmpfile: $!\n"); if ($ret != 0) { return; } if ($withclang ne "yes") { open (AUXFILE, "<$auxfile") || die ("open $auxfile: $!\n"); while () { s|/\*.*?\*/||g; if (/^\s*$/) { next; } # The word before a '(' that isn't '(*' is the function name # before the argument list (not fully general, but sufficient for # -aux-info output on standard headers). if (/(\w+)\s*\([^*]/) { $res{$1} = 1; } else { die ("couldn't parse -aux-info output: $_\n"); } } close (AUXFILE) || die ("close $auxfile: $!\n"); unlink ($auxfile) || die ("unlink $auxfile: $!\n"); } else { open (ASTFILE, "<$astfile") || die ("open $astfile: $!\n"); while () { s/^.*:[0-9][0-9]*:[0-9][0-9]* //g; s/^.*:[0-9][0-9]* implicit //g; s/^.*:[0-9][0-9]* //g; if (/(\w+)\s* /) { $res{$1} = 1; } else { die ("couldn't parse -ast-dump output: $_\n"); } } close (ASTFILE) || die ("close $astfile: $!\n"); unlink ($astfile) || die ("unlink $astfile: $!\n"); } return sort keys %res; }