wcsmbs: Improve fortify with clang
[glibc.git] / scripts / versions.awk
blobef1600bc03b3418222190f7907f866731c8035d6
1 # Combine version map fragments into version scripts for our shared objects.
2 # Copyright (C) 1998-2024 Free Software Foundation, Inc.
4 # This script expects the following variables to be defined:
5 # defsfile name of Versions.def file
6 # buildroot name of build directory with trailing slash
7 # move_if_change move-if-change command
9 # Read definitions for the versions.
10 BEGIN {
11 lossage = 0;
13 nlibs=0;
14 while (getline < defsfile) {
15 if (/^[a-zA-Z0-9_.]+ \{/) {
16 libs[$1] = 1;
17 curlib = $1;
18 while (getline < defsfile && ! /^}/) {
19 if ($2 == "=") {
20 renamed[curlib "::" $1] = $3;
22 else
23 versions[curlib "::" $1] = 1;
27 close(defsfile);
29 tmpfile = buildroot "Versions.tmp";
30 # POSIX sort needed.
31 sort = "sort -t. -k 1,1 -k 2n,2n -k 3 > " tmpfile;
34 # GNU awk does not implement the ord and chr functions.
35 # <https://www.gnu.org/software/gawk/manual/html_node/Ordinal-Functions.html>
36 # says that they are "written very nicely", using code similar to what
37 # is included here.
38 function chr(c) {
39 return sprintf("%c", c)
42 BEGIN {
43 for (c = 1; c < 127; c++) {
44 ord_table[chr(c)] = c;
48 function ord(c) {
49 if (ord_table[c]) {
50 return ord_table[c];
51 } else {
52 printf("Invalid character reference: '%c'\n", c) > "/dev/stderr";
53 ++lossage;
57 # Remove comment lines.
58 /^ *#/ {
59 next;
62 # This matches the beginning of the version information for a new library.
63 /^[a-zA-Z0-9_.]+/ {
64 actlib = $1;
65 if (!libs[$1]) {
66 printf("no versions defined for %s\n", $1) > "/dev/stderr";
67 ++lossage;
69 next;
72 # This matches the beginning of a new version for the current library.
73 /^ [A-Za-z_]/ {
74 if (renamed[actlib "::" $1])
75 actver = renamed[actlib "::" $1];
76 else if (!versions[actlib "::" $1] && $1 != "GLIBC_PRIVATE") {
77 printf("version %s not defined for %s\n", $1, actlib) > "/dev/stderr";
78 ++lossage;
80 else
81 actver = $1;
82 next;
85 # This matches lines with names to be added to the current version in the
86 # current library. This is the only place where we print something to
87 # the intermediate file.
88 /^ / {
89 sortver=actver
90 # Ensure GLIBC_ versions come always first
91 sub(/^GLIBC_/," GLIBC_",sortver)
92 printf("%s %s %s\n", actlib, sortver, $0) | sort;
95 # Some targets do not set the ABI baseline for libdl. As a result,
96 # symbols originally in libdl need to be moved under historic symbol
97 # versions, without altering the baseline version for libc itself.
98 /^ *!libc_pre_versions/ {
99 libc_pre_versions_active = 1;
102 function libc_pre_versions() {
103 # No local: * here, so that we do not have to update this script
104 # if symbols are moved into libc. The abilist files and the other
105 # targets (with a real GLIBC_2.0 baseline) provide testing
106 # coverage.
107 printf("\
108 GLIBC_2.0 {\n\
109 };\n\
110 GLIBC_2.1 {\n\
111 } GLIBC_2.0;\n\
112 ") > outfile;
113 return "GLIBC_2.1";
116 function closeversion(name, oldname) {
117 printf(" local:\n *;\n") > outfile;
118 # This version inherits from the last one only if they
119 # have the same nonnumeric prefix, i.e. GLIBC_x.y and GLIBC_x.z
120 # or FOO_x and FOO_y but not GLIBC_x and FOO_y.
121 pfx = oldname;
122 sub(/[0-9.]+/,".+",pfx);
123 if (oldname == "" || name !~ pfx) print "};" > outfile;
124 else printf("} %s;\n", oldname) > outfile;
127 function close_and_move(name, real_name) {
128 close(name);
129 system(move_if_change " " name " " real_name " >&2");
132 # ELF hash, for use with symbol versions.
133 function elf_hash(s, i, acc) {
134 acc = 0;
135 for (i = 1; i <= length(s); ++i) {
136 acc = and(lshift(acc, 4) + ord(substr(s, i, 1)), 0xffffffff);
137 top = and(acc, 0xf0000000);
138 acc = and(xor(acc, rshift(top, 24)), compl(top));
140 return acc;
143 # Now print the accumulated information.
144 END {
145 close(sort);
147 if (lossage) {
148 system("rm -f " tmpfile);
149 exit 1;
152 oldlib = "";
153 oldver = "";
154 real_first_ver_header = buildroot "first-versions.h"
155 first_ver_header = real_first_ver_header "T"
156 printf("#ifndef _FIRST_VERSIONS_H\n") > first_ver_header;
157 printf("#define _FIRST_VERSIONS_H\n") > first_ver_header;
158 real_ldbl_compat_header = buildroot "ldbl-compat-choose.h"
159 ldbl_compat_header = real_ldbl_compat_header "T"
160 printf("#ifndef _LDBL_COMPAT_CHOOSE_H\n") > ldbl_compat_header;
161 printf("#define _LDBL_COMPAT_CHOOSE_H\n") > ldbl_compat_header;
162 printf("#ifndef LONG_DOUBLE_COMPAT\n") > ldbl_compat_header;
163 printf("# error LONG_DOUBLE_COMPAT not defined\n") > ldbl_compat_header;
164 printf("#endif\n") > ldbl_compat_header;
165 printf("version-maps =");
166 while (getline < tmpfile) {
167 if ($1 != oldlib) {
168 if (oldlib != "") {
169 closeversion(oldver, veryoldver);
170 oldver = "";
171 close_and_move(outfile, real_outfile);
173 oldlib = $1;
174 real_outfile = buildroot oldlib ".map";
175 outfile = real_outfile "T";
176 if ($1 == "libc" && libc_pre_versions_active) {
177 veryoldver = libc_pre_versions();
178 } else {
179 veryoldver = "";
181 printf(" %s.map", oldlib);
183 if ($2 != oldver) {
184 if (oldver != "") {
185 closeversion(oldver, veryoldver);
186 veryoldver = oldver;
188 oldver = $2;
189 # Skip the placeholder symbol used only for empty version map.
190 if ($3 == "__placeholder_only_for_empty_version_map;") {
191 printf("%s {\n", $2) > outfile;
192 continue;
194 printf("%s {\n global:\n", $2) > outfile;
196 printf(" ") > outfile;
197 for (n = 3; n <= NF; ++n) {
198 printf(" %s", $n) > outfile;
199 sym = $n;
200 sub(";", "", sym);
201 first_ver_macro = "FIRST_VERSION_" oldlib "_" sym;
202 if (!(first_ver_macro in first_ver_seen) \
203 && oldver ~ "^GLIBC_[0-9]" \
204 && sym ~ "^[A-Za-z0-9_]*$") {
205 ver_val = oldver;
206 printf("#define %s_STRING \"%s\"\n", first_ver_macro, ver_val) > first_ver_header;
207 printf("#define %s_HASH 0x%x\n", first_ver_macro, elf_hash(ver_val)) > first_ver_header;
208 gsub("\\.", "_", ver_val);
209 printf("#define %s %s\n", first_ver_macro, ver_val) > first_ver_header;
210 first_ver_seen[first_ver_macro] = 1;
211 if (oldlib == "libc" || oldlib == "libm") {
212 printf("#if LONG_DOUBLE_COMPAT (%s, %s)\n",
213 oldlib, ver_val) > ldbl_compat_header;
214 printf("# define LONG_DOUBLE_COMPAT_CHOOSE_%s_%s(a, b) a\n",
215 oldlib, sym) > ldbl_compat_header;
216 printf("#else\n") > ldbl_compat_header;
217 printf("# define LONG_DOUBLE_COMPAT_CHOOSE_%s_%s(a, b) b\n",
218 oldlib, sym) > ldbl_compat_header;
219 printf("#endif\n") > ldbl_compat_header;
223 printf("\n") > outfile;
225 printf("\n");
226 printf("#endif /* first-versions.h */\n") > first_ver_header;
227 printf("#endif /* ldbl-compat-choose.h */\n") > ldbl_compat_header;
228 closeversion(oldver, veryoldver);
229 close_and_move(outfile, real_outfile);
230 close_and_move(first_ver_header, real_first_ver_header);
231 close_and_move(ldbl_compat_header, real_ldbl_compat_header);
232 #system("rm -f " tmpfile);