mirror of git://sourceware.org/git/glibc.git
				
				
				
			
		
			
				
	
	
		
			234 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			Awk
		
	
	
	
			
		
		
	
	
			234 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			Awk
		
	
	
	
| # Combine version map fragments into version scripts for our shared objects.
 | |
| # Copyright (C) 1998-2025 Free Software Foundation, Inc.
 | |
| 
 | |
| # This script expects the following variables to be defined:
 | |
| # defsfile		name of Versions.def file
 | |
| # buildroot		name of build directory with trailing slash
 | |
| # move_if_change	move-if-change command
 | |
| 
 | |
| # Read definitions for the versions.
 | |
| BEGIN {
 | |
|   lossage = 0;
 | |
| 
 | |
|   nlibs=0;
 | |
|   while (getline < defsfile) {
 | |
|     if (/^[a-zA-Z0-9_.]+ \{/) {
 | |
|       libs[$1] = 1;
 | |
|       curlib = $1;
 | |
|       while (getline < defsfile && ! /^}/) {
 | |
| 	if ($2 == "=") {
 | |
| 	  renamed[curlib "::" $1] = $3;
 | |
| 	}
 | |
| 	else
 | |
| 	  versions[curlib "::" $1] = 1;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   close(defsfile);
 | |
| 
 | |
|   tmpfile = buildroot "Versions.tmp";
 | |
|   # POSIX sort needed.
 | |
|   sort = "sort -t. -k 1,1 -k 2n,2n -k 3 > " tmpfile;
 | |
| }
 | |
| 
 | |
| # GNU awk does not implement the ord and chr functions.
 | |
| # <https://www.gnu.org/software/gawk/manual/html_node/Ordinal-Functions.html>
 | |
| # says that they are "written very nicely", using code similar to what
 | |
| # is included here.
 | |
| function chr(c) {
 | |
|     return sprintf("%c", c)
 | |
| }
 | |
| 
 | |
| BEGIN {
 | |
|     for (c = 1; c < 127; c++) {
 | |
| 	ord_table[chr(c)] = c;
 | |
|     }
 | |
| }
 | |
| 
 | |
| function ord(c) {
 | |
|     if (ord_table[c]) {
 | |
| 	return ord_table[c];
 | |
|     } else {
 | |
| 	printf("Invalid character reference: '%c'\n", c) > "/dev/stderr";
 | |
| 	++lossage;
 | |
|     }
 | |
| }
 | |
| 
 | |
| # Remove comment lines.
 | |
| /^ *#/ {
 | |
|   next;
 | |
| }
 | |
| 
 | |
| # This matches the beginning of the version information for a new library.
 | |
| /^[a-zA-Z0-9_.]+/ {
 | |
|   actlib = $1;
 | |
|   if (!libs[$1]) {
 | |
|     printf("no versions defined for %s\n", $1) > "/dev/stderr";
 | |
|     ++lossage;
 | |
|   }
 | |
|   next;
 | |
| }
 | |
| 
 | |
| # This matches the beginning of a new version for the current library.
 | |
| /^  [A-Za-z_]/ {
 | |
|   if (renamed[actlib "::" $1])
 | |
|     actver = renamed[actlib "::" $1];
 | |
|   else if (!versions[actlib "::" $1] && $1 != "GLIBC_PRIVATE") {
 | |
|     printf("version %s not defined for %s\n", $1, actlib) > "/dev/stderr";
 | |
|     ++lossage;
 | |
|   }
 | |
|   else
 | |
|     actver = $1;
 | |
|   next;
 | |
| }
 | |
| 
 | |
| # This matches lines with names to be added to the current version in the
 | |
| # current library.  This is the only place where we print something to
 | |
| # the intermediate file.
 | |
| /^   / {
 | |
|   sortver=actver
 | |
|   # Ensure GLIBC_ versions come always first
 | |
|   sub(/^GLIBC_/," GLIBC_",sortver)
 | |
|   printf("%s %s %s\n", actlib, sortver, $0) | sort;
 | |
| }
 | |
| 
 | |
| # Some targets do not set the ABI baseline for libdl.  As a result,
 | |
| # symbols originally in libdl need to be moved under historic symbol
 | |
| # versions, without altering the baseline version for libc itself.
 | |
| /^ *!libc_pre_versions/ {
 | |
|     libc_pre_versions_active = 1;
 | |
| }
 | |
| 
 | |
| function libc_pre_versions() {
 | |
|     # No local: * here, so that we do not have to update this script
 | |
|     # if symbols are moved into libc.  The abilist files and the other
 | |
|     # targets (with a real GLIBC_2.0 baseline) provide testing
 | |
|     # coverage.
 | |
|     printf("\
 | |
| GLIBC_2.0 {\n\
 | |
| };\n\
 | |
| GLIBC_2.1 {\n\
 | |
| } GLIBC_2.0;\n\
 | |
| ") > outfile;
 | |
|     return "GLIBC_2.1";
 | |
| }
 | |
| 
 | |
| function closeversion(name, oldname) {
 | |
|   printf("  local:\n    *;\n") > outfile;
 | |
|   # This version inherits from the last one only if they
 | |
|   # have the same nonnumeric prefix, i.e. GLIBC_x.y and GLIBC_x.z
 | |
|   # or FOO_x and FOO_y but not GLIBC_x and FOO_y.
 | |
|   pfx = oldname;
 | |
|   sub(/[0-9.]+/,".+",pfx);
 | |
|   if (oldname == "" || name !~ pfx) print "};" > outfile;
 | |
|   else printf("} %s;\n", oldname) > outfile;
 | |
| }
 | |
| 
 | |
| function close_and_move(name, real_name) {
 | |
|   close(name);
 | |
|   system(move_if_change " " name " " real_name " >&2");
 | |
| }
 | |
| 
 | |
| # ELF hash, for use with symbol versions.
 | |
| function elf_hash(s, i, acc) {
 | |
|   acc = 0;
 | |
|   for (i = 1; i <= length(s); ++i) {
 | |
|       acc = and(lshift(acc, 4) + ord(substr(s, i, 1)), 0xffffffff);
 | |
|       top = and(acc, 0xf0000000);
 | |
|       acc = and(xor(acc, rshift(top, 24)), compl(top));
 | |
|   }
 | |
|   return acc;
 | |
| }
 | |
| 
 | |
| # Now print the accumulated information.
 | |
| END {
 | |
|   close(sort);
 | |
| 
 | |
|   if (lossage) {
 | |
|     system("rm -f " tmpfile);
 | |
|     exit 1;
 | |
|   }
 | |
| 
 | |
|   oldlib = "";
 | |
|   oldver = "";
 | |
|   real_first_ver_header = buildroot "first-versions.h"
 | |
|   first_ver_header = real_first_ver_header "T"
 | |
|   printf("#ifndef _FIRST_VERSIONS_H\n") > first_ver_header;
 | |
|   printf("#define _FIRST_VERSIONS_H\n") > first_ver_header;
 | |
|   real_ldbl_compat_header = buildroot "ldbl-compat-choose.h"
 | |
|   ldbl_compat_header = real_ldbl_compat_header "T"
 | |
|   printf("#ifndef _LDBL_COMPAT_CHOOSE_H\n") > ldbl_compat_header;
 | |
|   printf("#define _LDBL_COMPAT_CHOOSE_H\n") > ldbl_compat_header;
 | |
|   printf("#ifndef LONG_DOUBLE_COMPAT\n") > ldbl_compat_header;
 | |
|   printf("# error LONG_DOUBLE_COMPAT not defined\n") > ldbl_compat_header;
 | |
|   printf("#endif\n") > ldbl_compat_header;
 | |
|   printf("version-maps =");
 | |
|   while (getline < tmpfile) {
 | |
|     if ($1 != oldlib) {
 | |
|       if (oldlib != "") {
 | |
| 	closeversion(oldver, veryoldver);
 | |
| 	oldver = "";
 | |
| 	close_and_move(outfile, real_outfile);
 | |
|       }
 | |
|       oldlib = $1;
 | |
|       real_outfile = buildroot oldlib ".map";
 | |
|       outfile = real_outfile "T";
 | |
|       if ($1 == "libc" && libc_pre_versions_active) {
 | |
| 	  veryoldver = libc_pre_versions();
 | |
|       } else {
 | |
| 	  veryoldver = "";
 | |
|       }
 | |
|       printf(" %s.map", oldlib);
 | |
|     }
 | |
|     if ($2 != oldver) {
 | |
|       if (oldver != "") {
 | |
| 	closeversion(oldver, veryoldver);
 | |
| 	veryoldver = oldver;
 | |
|       }
 | |
|       oldver = $2;
 | |
|       # Skip the placeholder symbol used only for empty version map.
 | |
|       if ($3 == "__placeholder_only_for_empty_version_map;") {
 | |
| 	printf("%s {\n", $2) > outfile;
 | |
| 	continue;
 | |
|       }
 | |
|       printf("%s {\n  global:\n", $2) > outfile;
 | |
|     }
 | |
|     printf("   ") > outfile;
 | |
|     for (n = 3; n <= NF; ++n) {
 | |
|       printf(" %s", $n) > outfile;
 | |
|       sym = $n;
 | |
|       sub(";", "", sym);
 | |
|       first_ver_macro = "FIRST_VERSION_" oldlib "_" sym;
 | |
|       if (!(first_ver_macro in first_ver_seen) \
 | |
| 	  && oldver ~ "^GLIBC_[0-9]" \
 | |
| 	  && sym ~ "^[A-Za-z0-9_]*$") {
 | |
| 	ver_val = oldver;
 | |
| 	printf("#define %s_STRING \"%s\"\n", first_ver_macro, ver_val) > first_ver_header;
 | |
| 	printf("#define %s_HASH 0x%x\n", first_ver_macro, elf_hash(ver_val)) > first_ver_header;
 | |
| 	gsub("\\.", "_", ver_val);
 | |
| 	printf("#define %s %s\n", first_ver_macro, ver_val) > first_ver_header;
 | |
| 	first_ver_seen[first_ver_macro] = 1;
 | |
| 	if (oldlib == "libc" || oldlib == "libm") {
 | |
| 	  printf("#if LONG_DOUBLE_COMPAT (%s, %s)\n",
 | |
| 		 oldlib, ver_val) > ldbl_compat_header;
 | |
| 	  printf("# define LONG_DOUBLE_COMPAT_CHOOSE_%s_%s(a, b) a\n",
 | |
| 		 oldlib, sym) > ldbl_compat_header;
 | |
| 	  printf("#else\n") > ldbl_compat_header;
 | |
| 	  printf("# define LONG_DOUBLE_COMPAT_CHOOSE_%s_%s(a, b) b\n",
 | |
| 		 oldlib, sym) > ldbl_compat_header;
 | |
| 	  printf("#endif\n") > ldbl_compat_header;
 | |
| 	}
 | |
|       }
 | |
|     }
 | |
|     printf("\n") > outfile;
 | |
|   }
 | |
|   printf("\n");
 | |
|   printf("#endif /* first-versions.h */\n") > first_ver_header;
 | |
|   printf("#endif /* ldbl-compat-choose.h */\n") > ldbl_compat_header;
 | |
|   closeversion(oldver, veryoldver);
 | |
|   close_and_move(outfile, real_outfile);
 | |
|   close_and_move(first_ver_header, real_first_ver_header);
 | |
|   close_and_move(ldbl_compat_header, real_ldbl_compat_header);
 | |
|   #system("rm -f " tmpfile);
 | |
| }
 |